Skip to main content

Automate opening of new tabs and windows in terminal programs

Project description

automate-terminal

Automate opening of new tabs and windows in terminal programs. Currently supports iTerm2, Terminal.app, Ghostty, tmux, WezTerm, Kitty, Cursor, and Visual Studio Code.

automate-terminal is a best-effort project. Some terminals do not support automation at all! It's also intended to be used as a component in other tools, so it errs on the side of strictness over fallbacks. See the command reference for specifics.

Installation

pip install automate-terminal
mise install pip:automate-terminal

Supported Terminals

Terminal Platform New Tabs/Windows Switch by ID Switch by Working Dir List Sessions Paste Commands Run in Active Session
iTerm2 macOS
Terminal.app macOS
Ghostty macOS
tmux Cross-platform
WezTerm Cross-platform
Kitty Cross-platform ✅* ✅* ✅* ✅* ✅* ✅*
VSCode Cross-platform ⚠️ (no tabs)
Cursor Cross-platform ⚠️ (no tabs)

* Kitty requires allow_remote_control yes in kitty.conf to enable automation features.

Other terminals are not supported; automate-terminal will exit with an error code in unsupported terminals.

Quick Start

Command Line

# Check if your terminal is supported
automate-terminal check

# Create a new tab
automate-terminal new-tab /path/to/project

# Switch to existing session by directory
automate-terminal switch-to --working-directory=/path/to/project

# Create new window with initialization script
automate-terminal new-window /path/to/project \
  --paste-and-run="source .env && npm run dev"

# Run a command in the currently active session
automate-terminal run-in-active-session "git status"

What is possible with terminal automation?

This is the "manage expectations" section of the README.

The scope of automation covered by automate-terminal is "create or navigate to a terminal session in a specific working directory."

What is a terminal emulator?

A terminal emulator, for the purposes of this project, is the thing you type your terminal commands into. For the author, it's iTerm2. For many people, it's the built-in terminal in Visual Studio Code. For hipsters, it's Ghostty. Linux users have more options than I know of, including KConsole. And there's WezTerm, and the built-in terminals for macOS and Windows. Every few years, somebody spins up a new one.

Within terminal emulators, a session is, for the purposes of this project, one specific terminal running a shell. Your sessions might be organized in windows, or tabs, or splits within a window or tab.

How are they controlled?

In many different ways, which is why this problem is complex enough to warrant a library! The ability to open a new tab or window, or switch to a specific existing session based on some criteria, is not standardized at all among terminal emulators. The spread of what is possible for a given terminal emulator is incredibly wide, as shown by the table at the top of this README.

macOS terminals are primarily controlled via AppleScript. iTerm2, for example, has a comprehensive AppleScript API, which is why it has the best support. Ghostty has no AppleScript API at all, so automate-terminal uses the SystemEvents API to "click" its menu items, and we have no way of knowing which Ghostty tab is in a particular working directory. VSCode also has no AppleScript API, but it does have its code <path> command to open or switch to a specific window.

tmux uses its native CLI commands (tmux list-panes, tmux select-window, tmux send-keys, etc.) and works identically across all platforms where tmux is available.

WezTerm uses its native CLI commands (wezterm cli list, wezterm cli activate-pane, wezterm cli send-text, etc.) and works identically across all platforms where WezTerm is available (macOS, Linux, BSD, Windows).

Kitty uses its remote control protocol (kitten @ ls, kitten @ focus-window, kitten @ send-text, etc.) and works across macOS, Linux, and BSD. Note: Kitty requires allow_remote_control yes in your kitty.conf file to enable automation.

All this is to say, if you are unhappy with the level of automation provided by automate-terminal on macOS, switch to another terminal emulator, or lobby the authors of your favorite one to add automation support.

Commands

check

Detect terminal capabilities.

automate-terminal check
automate-terminal check --output=json

Example output:

Terminal: iTerm2
Terminal Program: iTerm.app
Shell: zsh
Current session ID: w0t5p1:24AF055B-8BD2-4C7F-AB1E-B310FDCBCEA1
Current working directory: /Users/steve/dev/libraries/automate-terminal

Capabilities:
  can_create_tabs: True
  can_create_windows: True
  can_list_sessions: True
  can_switch_to_session: True
  can_detect_session_id: True
  can_detect_working_directory: True
  can_paste_commands: True
  can_run_in_active_session: True

new-tab

Create new tab in a specific directory, optionally pasting content into the new tab.

The term "paste" here means it will be as if the user pasted text into the new terminal and hit Enter. It doesn't use your system pasteboard.

automate-terminal new-tab /path/to/dir

automate-terminal new-tab /path/to/dir --paste-and-run="echo 'I am in the new directory!'"

There are options to run additional scripts only in specific shells. This is useful if your wrapper tool needs to support multiple shells for workflows that require nontrivial shell commands.

automate-terminal new-tab /path/to/dir --paste-and-run-fish="echo 'I am a fish shell user'"

new-window

Create new window. Takes the same arguments as automate-terminal new-tab.

automate-terminal new-window /path/to/dir

switch-to

Switch to an existing session, returning an error no matching session can be found or your terminal doesn't provide session information. You can either pass a working directory with --working-directory/--wd, or a session ID if the session ID is known.

If you pass session ID, working directory is ignored. --wd is likely what you want, though.

If you are using an unsupported terminal emulator and want to open a new terminal in a given directory as a fallback, use new-tab or new-window. automate-terminal doesn't do this automatically because it would create unpleasant edge cases for tools that use automate-terminal.

# By working directory (or use --wd alias)
automate-terminal switch-to --working-directory=/path/to/dir

# By session ID (or use --id alias)
automate-terminal switch-to --session-id=w0t0p2:ABC123

list-sessions

List all sessions.

automate-terminal list-sessions
automate-terminal list-sessions --output=json

run-in-active-session

Run a command in the currently active terminal session.

This command sends the specified command to the active terminal session without switching windows or tabs. It's useful for programmatically executing commands in the terminal session you're currently working in.

# Run a simple command
automate-terminal run-in-active-session "echo 'Hello World'"

# Run a git command
automate-terminal run-in-active-session "git status"

# Run multiple commands
automate-terminal run-in-active-session "cd /tmp && ls -la"

Note: This feature requires terminal support. VSCode and Cursor do not support running commands in the active session. Check capabilities with automate-terminal check to see if your terminal supports this feature.

Options

Output Format

  • --output=text - Human-readable (default)
  • --output=json - JSON for programmatic use
  • --output=none - Silent

Paste and Run

Execute commands after creating/switching sessions.

--paste-and-run="echo 'I run unconditionally'"
--paste-and-run-bash="echo 'I only run if the current shell is bash'"
--paste-and-run-zsh="echo 'I only run if the current shell is zsh'"
--paste-and-run-fish="echo 'I only run if the current shell is fish'"

Shell-specific flags override generic --paste-and-run when detected shell matches.

Note: Some terminals (VSCode, Cursor) cannot execute paste scripts programmatically. When using --output=json, check the paste_script_executed field to determine if you need to run the script manually:

  • true: The paste script was executed by the terminal
  • false: The paste script was provided but the terminal cannot execute it (you should run it manually)
  • Field omitted: No paste script was provided

Debug and Dry Run

--debug     # Enable debug logging to stderr
--dry-run   # Log actions instead of executing them

Use --dry-run to see what commands would be executed without actually running them (AppleScript for macOS terminals, tmux CLI commands for tmux). Useful for debugging and understanding what the tool will do.

Python API

from automate_terminal import (
    check,
    new_tab,
    new_window,
    switch_to_session,
    list_sessions,
    get_current_session_id,
    get_shell_name,
    run_in_active_session,
    TerminalNotFoundError,
)

check(dry_run=False, debug=False) -> dict[str, str | Capabilities]

new_tab(working_directory, paste_script=None, dry_run=False, debug=False) -> bool

new_window(
  working_directory,
  paste_script=None,
  dry_run=False,
  debug=False) -> bool

switch_to_session(
  session_id=None,
  working_directory=None,
  paste_script=None,
  subdirectory_ok=False,
  dry_run=False,
  debug=False) -> bool

list_sessions(dry_run=False, debug=False) -> list[dict[str, str]]

get_current_session_id(dry_run=False, debug=False) -> str | None

get_shell_name(dry_run=False, debug=False) -> str | None

run_in_active_session(command, dry_run=False, debug=False) -> bool

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

automate_terminal-0.1.4.tar.gz (52.3 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

automate_terminal-0.1.4-py3-none-any.whl (36.7 kB view details)

Uploaded Python 3

File details

Details for the file automate_terminal-0.1.4.tar.gz.

File metadata

  • Download URL: automate_terminal-0.1.4.tar.gz
  • Upload date:
  • Size: 52.3 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for automate_terminal-0.1.4.tar.gz
Algorithm Hash digest
SHA256 d3489545f39c67ecad7c63cdec4dee9b1d58395875e6af2b26899332b8f32129
MD5 4b17621442728ae26c63c25ea650179d
BLAKE2b-256 f277873645397587a595919402b18feada87dbc5f8ede98b608041722c84713f

See more details on using hashes here.

Provenance

The following attestation bundles were made for automate_terminal-0.1.4.tar.gz:

Publisher: pypi.yml on irskep/automate-terminal

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file automate_terminal-0.1.4-py3-none-any.whl.

File metadata

File hashes

Hashes for automate_terminal-0.1.4-py3-none-any.whl
Algorithm Hash digest
SHA256 9ddeff9dffbb322ce71c4101ce375521c470b681187ce67e6dd547d8ac2f7395
MD5 9aa6aaf8e0a878b65fcd56ab31869b99
BLAKE2b-256 9d1c2b7d0e6f0ce6972cf8f86b1669af9a4d32051e289b508a5a73df79582324

See more details on using hashes here.

Provenance

The following attestation bundles were made for automate_terminal-0.1.4-py3-none-any.whl:

Publisher: pypi.yml on irskep/automate-terminal

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page