Skip to content

noizbuster/ground-control

Repository files navigation

Ground Control banner

Ground Control

gctrl is a terminal TUI for monitoring OpenCode, Codex, and Claude Code sessions in real time. It presents session status, active agents, subagent activity, richer source metadata, and recent updates in one card-based interface, while also supporting source-native session deletion flows where the upstream CLI exposes them and conservative local cleanup for Claude Code sessions.

Quick Start

Make sure these are available before running gctrl:

  • Node.js 22.13.0 or later
  • OpenCode, Codex, and/or Claude Code installed on the same machine
  • ~/.local/share/opencode/opencode.db exists for OpenCode monitoring
  • ~/.codex/state_*.sqlite and ~/.codex/sessions/ exist for Codex monitoring
  • ~/.claude/projects/ and/or ~/.claude/sessions/ exist for Claude Code monitoring

Run with bunx

bunx gctrl

bunx works by relaunching the CLI with Node under the hood.

Run with npx

npx gctrl

demo image of gctrl

demo image of gctrl

Overview

  • Displays OpenCode, Codex, and Claude Code sessions in a live terminal list
  • Refreshes automatically every 2 seconds
  • Shows status, source, project label, session ID, update time, and richer source metadata
  • Supports source-aware actions, copy ID, refresh, and keyboard navigation

Usage

After launch, use these shortcuts to navigate and control the monitor:

Key Action
h / j / k / l / / / / Move selection in the session grid
j / k / / (detail focus) Scroll session detail
Tab Switch focus between grid and detail pane (when sideview is enabled)
Enter Open the selected session detail view
e / p Toggle sideview layout
f Cycle session filter mode (active → recent → busy → all)
s Cycle sort mode (status → update → create)
c Open hierarchy view
t Open hierarchy directly in timeline view
a Attach to the selected session
i Copy the selected session ID
K Stop active child sessions when supported (detail/sideview mode only)
d Request delete for the selected session
y / n Confirm or cancel delete prompt
r Refresh immediately
Esc / q Cancel prompt, close current view, or quit from the main view
Ctrl+C Quit immediately

Stop Child Sessions (K)

Available in detail or sideview mode. Press K (Shift+K) to gracefully stop all active (non-completed, non-failed) child sessions of the selected session.

Codex sessions support attach/inspect/copy/hierarchy/delete flows, Claude Code sessions support attach/inspect/copy/hierarchy/delete flows, and child-abort remains OpenCode-only. Codex attach uses codex resume <session-id>, Claude Code attach uses claude --resume <session-id>, and both fall back to the current monitor directory if the original session directory no longer exists. Claude Code subagent attach resolves back to the root session because the upstream CLI resumes root conversations by ID.

The stop flow works in two stages:

  1. Graceful stop: sends a "stop" message to each child session via opencode run --session <id>, which triggers normal completion (finish: "stop").
  2. Delete fallback: if graceful stop fails for any child, a per-item confirmation dialog appears where you can choose to delete (y), skip (n), or cancel all (Esc/q).

Session Filter Modes (f)

  • active: non-completed sessions, plus externally attached completed sessions. Directory-count fallback is applied as non-complete first, and only remaining slots can surface latest completed sessions.
  • recent: everything visible in active, plus (1) the globally latest completed session and (2) each project's latest updated session.
  • busy: non-completed sessions only.
  • all: all sessions.

Hierarchy View (c)

Press c on a selected session to open the agent hierarchy view, or press t to open it directly in timeline mode. This shows the session's subagent tree with status, timing, and metadata.

Key Action
Tab Cycle view mode (tree / timeline)
x Cycle info mode (standard / detailed)
f Cycle filter mode (latest / busy / all)
/ / h / l Pan timeline (timeline mode only)
j / k / Up / Down Scroll
Esc / q Close hierarchy view

Requirements

  • Node.js 22.13.0+ is required for built-in node:sqlite.
  • bun is optional and only used as an alternate launcher (bunx gctrl).
  • The monitor reads OpenCode session data from ~/.local/share/opencode/opencode.db.
  • The monitor reads Codex thread state from the newest ~/.codex/state_*.sqlite file and enriches it with ~/.codex/sessions/**/*.jsonl.
  • The monitor reads Claude Code session state from ~/.claude/sessions/*.json and enriches it with ~/.claude/projects/**/*.jsonl.
  • Override the OpenCode database path with GCTRL_DB_PATH=/custom/path/opencode.db.
  • Override Codex paths with GCTRL_CODEX_STATE_DB_PATH=/custom/path/state.sqlite, GCTRL_CODEX_SESSIONS_DIR=/custom/path/sessions, GCTRL_CODEX_ARCHIVED_SESSIONS_DIR=/custom/path/archived_sessions, and GCTRL_CODEX_SESSION_INDEX_PATH=/custom/path/session_index.jsonl.
  • Override Claude Code paths with GCTRL_CLAUDE_PROJECTS_DIR=/custom/path/projects and GCTRL_CLAUDE_SESSIONS_DIR=/custom/path/sessions.
  • OpenCode attach/delete/child-abort actions use the opencode CLI.
  • Codex attach/delete actions use the local codex CLI.
  • Claude Code attach actions use the local claude CLI.
  • Codex delete uses the local codex app-server archive flow plus cleanup of archived rollout files and local index/state entries.
  • Claude Code delete intentionally refuses live sessions, then removes matching projects/, file-history/, session-env/, tasks/, and stale sessions/*.json artifacts from local .claude/ storage. This follows the official .claude storage guidance: Claude Code does not expose a delete subcommand, but its local session data can be removed directly.
  • Non-interactive mode (missing TTY stdin/stdout) prints a tab-separated snapshot and exits.

Local Development

bun install
bun run dev

Useful scripts:

bun run start
bun run dev
bun run build
bun run typecheck
bun run lint
bun run check

Project Structure

bin/          CLI wrapper
src/db/       OpenCode + Codex + Claude Code data adapters
src/ui/       TUI components
src/config/   color and agent configuration
src/lib/      status detection logic
dist/         compiled output

Session Status Detection

OpenCode status is derived from the latest message in the OpenCode SQLite database. The detection pipeline reads the most recent message row per session, parses its JSON data column, and applies status detectors in priority order. Codex status is mapped conservatively from thread/task events plus thread_spawn_edges, while Claude Code status is mapped from local session registries plus JSONL conversation logs and subagent transcripts. Richer raw detail is surfaced separately in the UI.

Status Priority

Priority Status Condition
1 failed finish === "error"
2 waiting Question tool is running AND no user response yet (see Waiting Detection below)
3 completed finish === "stop" OR time.completed is a finite number
4 running Not failed, not waiting, not completed
5 unknown Message data is null, empty, or failed JSON parsing

Finish Values

The finish field on the latest message indicates how the session step ended:

finish time.completed Detected Status Display Label
stop completed Completed
tool-calls absent running Running
error failed Failed
other present completed Completed (other)
length present completed Completed (length)
unknown present completed Completed (unknown)
undefined absent running Running
no message unknown Unknown

When finish is "other", "length", or "unknown", the status label appends the reason in parentheses — e.g. Completed (other), Completed (length).

Waiting Detection

A session is waiting when the question tool is active and awaiting user input. This uses a secondary signal from the part table:

  1. Find the latest part row with type === "tool" and tool === "question" where state.status === "running"
  2. Compare its timestamp against the latest user message time
  3. If the question tool time is newer → waiting (overrides running)

Waiting never overrides failed or completed.

Display Override: AWAITING SUBAGENT

A completed root session that has running child sessions displays as AWAITING SUBAGENT instead of Completed. This reflects that the parent is technically done but work continues in subagents. The effective status becomes running for filtering and sorting purposes.

Hierarchy Filter: Latest Mode

In the hierarchy view's "latest" filter mode, subagent sessions are filtered as follows:

  • Active subagents (pending/running/waiting) are always shown
  • AWAITING SUBAGENT subagents (completed with active children) are always shown
  • Terminal subagents (completed/failed/unknown with no active children) — only the most recently updated one is shown

License

MIT


Supervised by NoizBuster, Written by OpenCode

About

Real-time OpenCode session monitor TUI

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

 
 
 

Contributors