Linux-native multi-agent terminal with a programmable local socket API, first-class git worktrees, and prompt-aware notifications.
ForkTTY runs coding agents in isolated workspaces, exposes a user-local Unix socket for automation, and can place long-running tasks in dedicated git worktrees without tying the UI to one agent vendor.
Status: Early alpha (v0.2.0-alpha.5). ForkTTY is Linux-only and the GTK/VTE runtime is now the primary implementation. The AppImage is the default download for this alpha and remains experimental; the Debian package is still available for Debian/Ubuntu users.
- Agent-agnostic automation: the same socket API and CLI flow work for Codex, Claude Code, Gemini CLI, shell scripts, and custom tools.
- First-class worktree workflows: create, attach, remove, and merge isolated worktree workspaces through native
git2operations and optional.forktty/setup/.forktty/teardownhooks. - Native Linux terminal stack: GTK4/libadwaita shell with embedded VTE terminals, split panes, session restore, notifications, command palette, settings, and quake mode, plus experimental WebKitGTK6 browser panes behind the
browserfeature. - Local-first posture: no telemetry, no update checks, no external service dependency, owner-only Unix socket permissions, bounded request/session/config files, and argv-based command execution.
The fastest paths are the prebuilt artifacts from the v0.2.0-alpha.5 release. Each release ships:
forktty-0.2.0-alpha.5-x86_64.AppImage— experimental, recommended for quick trials.forktty_0.2.0~alpha.5_amd64.deb— Debian/Ubuntu package.SHA256SUMS— checksums for both artifacts.
After downloading, verify checksums:
sha256sum -c SHA256SUMSchmod +x forktty-0.2.0-alpha.5-x86_64.AppImage
./forktty-0.2.0-alpha.5-x86_64.AppImageThe AppImage bundles the GTK4, libadwaita, VTE, and WebKitGTK shared libraries it links against, but it still depends on the host system for glibc, the GSettings/GIO data tree, Wayland/X11 session services, fontconfig, the OpenGL/Vulkan/Mesa driver stack, and desktop notification services. Treat the AppImage as a smoke-test artifact: it works on most modern distros that ship VTE 0.76+ and a recent glibc, but it is not as portable as a typical Linux binary and should be tested on the target distro before being relied on.
If the AppImage launches but the GTK interface renders incorrectly, try an explicit GTK renderer from a terminal:
GSK_RENDERER=ngl ./forktty-0.2.0-alpha.5-x86_64.AppImagesudo apt install ./forktty_0.2.0~alpha.5_amd64.deb
# or, if apt cannot read the file path directly:
sudo dpkg -i forktty_0.2.0~alpha.5_amd64.deb
sudo apt -f installThe package installs the forktty binary, the desktop entry, and the
icon. Removing it (sudo apt remove forktty) cleans up
/usr/bin/forktty and the desktop integration.
Requirements:
- Linux
- Rust 1.88+
- GTK4, libadwaita, VTE GTK4 development libraries
- WebKitGTK 6 development files for the full GTK/browser build
Debian / Ubuntu:
sudo apt install build-essential libssl-dev libgtk-4-dev libadwaita-1-dev libvte-2.91-gtk4-dev libwebkitgtk-6.0-dev desktop-file-utilsFedora:
sudo dnf install gcc gcc-c++ openssl-devel gtk4-devel libadwaita-devel vte291-gtk4-devel webkitgtk6.0-devel desktop-file-utilsArch / CachyOS:
sudo pacman -S base-devel openssl gtk4 libadwaita vte4 webkitgtk-6.0 desktop-file-utilsForkTTY requires libadwaita 1.4+ and VTE 0.76 or newer, matching Ubuntu 24.04 LTS and newer distro packages. For compositor-anchored quake/dropdown placement on Wayland, install gtk4-layer-shell as an optional runtime dependency.
Clone and run:
git clone https://github.com/Lucenx9/forktty.git
cd forktty
cargo run -p forktty-ui-gtk --features browserThe full source and packaged builds use --features browser, which
includes the GTK/VTE terminal runtime and WebKitGTK browser panes. For a
terminal-only development build on systems without WebKitGTK, use:
cargo run -p forktty-ui-gtk --features gtk-vteBuild the Debian package locally:
bash scripts/build-deb.sh
sudo dpkg -i target/packaging/deb/forktty_*.debBuild the experimental AppImage locally (requires appimagetool on
PATH, or APPIMAGETOOL=/path/to/appimagetool):
bash scripts/build-appimage.sh
./target/packaging/appimage/forktty-*-x86_64.AppImageAfter install, confirm the runtime looks healthy:
forktty --version
forktty doctorforktty doctor is a local-only inspector. It reports the resolved
config, session, socket, hook config paths, and known recovery
behaviors, and exits 0 on a clean environment or 2 with explicit
warnings.
ForkTTY opens the current directory as the main workspace. Use the
command palette for most navigation and pane actions:
Ctrl+Shift+P: command paletteCtrl+Shift+N: new workspaceCtrl+Shift+O: open workspaceCtrl+Shift+H: split pane rightCtrl+Shift+E: split pane downCtrl+Alt+Left/Ctrl+Alt+Right: focus previous/next paneCtrl+Shift+W: close paneCtrl+BorF9: toggle workspace sidebarCtrl+Shift+M: notificationsF1: keyboard shortcutsCtrl+,: settings
The same forktty binary speaks the local socket API. Diagnostic
commands work even when the GTK app is not running:
forktty --help
forktty --version
forktty doctor # local diagnostics, no socket required
forktty ping # check the running daemonWorkspace, surface, worktree, notification, and metadata commands talk to the live socket:
forktty list
forktty focus "Workspace 2"
forktty surfaces --workspace-name main
forktty split-surface --axis vertical
forktty send-text "cargo test\n"
forktty worktree-status
forktty notify --title "Input needed" --kind prompt "Blocked on test fixture"
forktty set-status --key agent:codex --label Codex --value Running --color blue
forktty set-progress --key build --label Build --value 42 --total 100
forktty log --level warn "Waiting for reviewer input"
forktty notifications
forktty capabilities
forktty eventsSource builds with the browser feature also expose browser-pane
automation over the same socket:
forktty browser open --profile Default https://example.com
forktty browser navigate <surface-id> https://www.rust-lang.org
forktty browser snapshot <surface-id>
forktty browser click <surface-id> <ref>
forktty browser fill <surface-id> <ref> "value"
forktty browser eval <surface-id> "document.title"
forktty browser profile list
forktty browser profile create WorkThe socket CLI and agent hook bridge are native Rust code in the
forktty binary, so source checkouts and packaged builds do not
require Node.js.
Spawned shells receive:
TERM=xterm-256colorCOLORTERM=truecolorTERM_PROGRAM=ForkTTYTERM_PROGRAM_VERSIONFORKTTY_WORKSPACE_IDFORKTTY_SURFACE_IDFORKTTY_SOCKET_PATH
That lets hooks and scripts target the current workspace without extra flags.
Manual socket overrides must be absolute paths; blank or relative
FORKTTY_SOCKET_PATH values are ignored so the app and CLI fall back to the
default socket location.
Install hook templates for Codex, Claude Code, and Gemini CLI:
forktty hooks setup # install all three
forktty hooks setup codex # install just one
forktty hooks setup codex claude --dry-run--dry-run prints the would-be diff without touching disk.
The installer merges commands into the agent's own config file:
| Agent | Destination |
|---|---|
| Codex | $CODEX_HOME/hooks.json or ~/.codex/hooks.json |
| Claude Code | $CLAUDE_CONFIG_DIR/settings.json or ~/.claude/settings.json |
| Gemini CLI | ~/.gemini/settings.json |
When HOME is overridden, the ~ defaults are resolved under that home
directory. Existing configs are written atomically (tmp + rename) and a
timestamped .bak-* backup is created when content changes.
Diagnose and exercise installed hooks:
forktty hooks doctor codex # inspect socket, launcher, env, hook config
forktty hooks test codex # round-trip a status update through the socketEach agent's hook commands honor a per-agent disable variable:
FORKTTY_CODEX_HOOKS_DISABLED=1FORKTTY_CLAUDE_HOOKS_DISABLED=1FORKTTY_GEMINI_HOOKS_DISABLED=1
Hooks report status, progress, logs, and prompt notifications through
the same local socket pipeline. Manual hook-event commands can pass
--socket <path> when they run outside a ForkTTY-spawned shell.
- Native GTK4/libadwaita desktop shell with embedded VTE terminals.
- Recursive split panes, pane focus/close, command palette, settings dialog, notification panel, and workspace sidebar.
- Quake/dropdown mode through config and F12 where global shortcuts are supported.
- Direct Unix socket JSON-RPC server for workspace, surface, notification, worktree, metadata, event-stream, capabilities, and browser automation.
- Git worktree create/attach/remove/merge/status with dirty-state protection and hook execution inside verified worktrees. Setup hooks are advisory; teardown hook failures or teardown-created dirty state block removal.
- Session restore for workspace order, active workspace, pane tree, focused surface, cwd, branch, and worktree metadata.
- Prompt-aware notifications from VTE shell integration signals, bounded visible prompt fallback, VTE bell, and hook/socket events.
- Experimental WebKitGTK6 browser panes (behind the
browserfeature) with scriptable snapshot/click/fill/eval verbs, per-profile persistent WebKit sessions, profile CRUD, and history/bookmark socket plus CLI access. - Bounded config/session/socket handling and local-only privacy defaults.
Config file: ~/.config/forktty/config.toml. All fields are optional.
[general]
theme_source = "auto"
shell = "/bin/bash"
worktree_layout = "nested" # "nested", "sibling", or "outer-nested"
enable_pr_lookup = false
notification_command = ""
[appearance]
font_family = ""
font_size = 14
scrollback_lines = 20000
terminal_audible_bell = true
sidebar_position = "left" # "left" or "right"
sidebar_visible = true
terminal_renderer = "auto" # legacy compatibility; native GTK uses VTE
terminal_theme = "system" # "system", "catppuccin-mocha", "rose-pine", "tokyo-night", "dracula", or "gruvbox-dark"
window_mode = "normal" # "normal" or "quake"
[notifications]
desktop = true
sound = truenotification_command is split with shell_words; ForkTTY does not use sh -c. The first token must be an absolute executable path, and notification title/body are passed through FORKTTY_NOTIFICATION_TITLE and FORKTTY_NOTIFICATION_BODY.
scrollback_lines controls VTE scrollback per pane; set it to 0 to disable scrollback. terminal_theme = "system" follows the app light/dark scheme; named terminal themes use fixed dark palettes. terminal_renderer is kept for config compatibility; the native GTK app uses VTE.
See SPEC.md for the full list of validated fields and their bounds.
GTK/VTE sessions are stored as:
~/.local/share/forktty/session-v2.json
ForkTTY imports legacy session.json when present, but saves the native runtime as v2. Restore does not preserve running PTY processes or scrollback; new VTE terminals are spawned for restored panes. Corrupt or structurally invalid session files are quarantined.
- Local Linux desktop threat model; same-user processes remain part of the local trust boundary.
- Unix socket defaults to
$XDG_RUNTIME_DIR/forktty.sockwith/tmp/forktty-<uid>/forktty.sockfallback and owner-only permissions. - Socket request lines, config files, and session files are size bounded.
- Shell paths, hooks, and custom notification commands use validated argv execution, not shell pipelines.
- Worktree names, socket-provided repo paths, and hook locations are validated before mutation or execution.
- ForkTTY makes no telemetry, update-check, or product network calls. Optional browser panes only load URLs the user or local socket automation opens.
See SECURITY.md and PRIVACY.md.
- Linux only. There are no supported macOS or Windows builds.
- VTE 0.76+ and libadwaita 1.4+ are required by the native terminal integration.
- The AppImage bundles GTK4/libadwaita/VTE but still relies on the host's glibc, GSettings/GIO data, fontconfig, OpenGL/Vulkan/Mesa driver stack, and desktop session services. Treat it as experimental and smoke-test it on the target distro; prefer the
.debon Debian/Ubuntu when possible. - PTYs and scrollback are not persisted across restart; restored sessions spawn fresh shells.
- Byte-level OSC 9/99 parsing from the old PTY-owner path is not fully ported because VTE owns the child PTY.
- Quake global shortcuts and layer-shell placement depend on desktop/compositor support.
- Full theme customization, multi-window, persistent scrollback, browser history/bookmark GTK UI integration, and browser import from external browsers are backlog items.
- Browser panes are included in full source and packaged builds; use
--features gtk-vteonly for a terminal-only development build.
forktty doctoris the first stop: it explains config, session, socket, and hook config problems before they trigger a launch failure.- If the GTK app refuses to start, run it from a terminal to see GLib/GTK error output, then re-run
forktty doctor. - The local socket lives at
$XDG_RUNTIME_DIR/forktty.sock(or/tmp/forktty-<uid>/forktty.sock). Stale or foreign sockets are refused on startup; remove them by hand only after confirming no other ForkTTY instance owns them. - A corrupt
~/.config/forktty/config.tomlor~/.local/share/forktty/session-v2.jsonis renamed aside as*.bad-<timestamp>so the app can start with defaults; the rename reason is logged to stderr. - Local logs live under
~/.local/share/forktty/logs/.
Useful commands:
cargo fmt --all --check
cargo run -p xtask -- check
cargo test --workspace
cargo clippy --workspace --features browser -- -D warnings
cargo build -p forktty-ui-gtk --features browser
desktop-file-validate packaging/linux/forktty.desktop
bash scripts/build-deb.sh
bash scripts/build-appimage.shSee SPEC.md, ROADMAP.md, and docs/native-gtk-vte.md. Use docs/release-qa.md before tagging alpha releases.
Built from scratch for Linux, inspired by cmux and other multi-agent terminal workflows.
GNU Affero General Public License v3.0 (AGPL-3.0-only)
