npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2026 – Pkg Stats / Ryan Hefner

gctrl

v0.2.4

Published

Real-time OpenCode, Codex, and Claude Code session monitor CLI

Downloads

112

Readme

Ground Control

gctrl is a terminal TUI for monitoring OpenCode, Codex, Claude Code, Pi, and omp 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 file-backed sources.

Quick Start

Make sure these are available before running gctrl:

  • Node.js 22.13.0 or later
  • OpenCode, Codex, Claude Code, Pi, and/or omp 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
  • ~/.pi/agent/sessions/ exists for Pi monitoring
  • ~/.omp/agent/sessions/ exists for omp monitoring

Run with bunx

bunx gctrl

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

Run with npx

npx gctrl

Overview

  • Displays OpenCode, Codex, Claude Code, Pi, and omp 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, Claude Code, Pi, and omp 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>, Pi attach uses pi --session <session-id> for roots and the exact session JSONL path for child artifacts, and omp attach uses omp --resume <session-jsonl-path> when the JSONL path is known, falling back to the session ID only when no path is available. Attach falls 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.
  • The monitor reads Pi JSONL sessions from ~/.pi/agent/sessions/**/*.jsonl.
  • The monitor reads omp JSONL sessions from ~/.omp/agent/sessions/**/*.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.
  • Override Pi sessions with GCTRL_PI_SESSIONS_DIR=/custom/path/sessions; PI_CODING_AGENT_SESSION_DIR and PI_CODING_AGENT_DIR are also honored.
  • Override omp sessions with GCTRL_OMP_SESSIONS_DIR=/custom/path/sessions; PI_CODING_AGENT_DIR, PI_CONFIG_DIR, and XDG omp candidates are also honored.
  • 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.
  • Pi attach actions use the local pi CLI; Pi delete removes the selected JSONL session and any loaded descendant JSONL sessions.
  • omp attach actions use the local omp CLI; omp delete removes the selected JSONL session, loaded descendant JSONL sessions, and sibling artifact directories, but not shared blob storage.
  • 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 + Pi/omp 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, Claude Code status is mapped from local session registries plus JSONL conversation logs and subagent transcripts, and Pi/omp status is mapped conservatively from JSONL message/model/thinking events. 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