GSD has 7+ commands that check, diagnose, or clean up project state. Several overlap or duplicate each other, and worktree lifecycle management is missing entirely. Users can't answer "what's safe to delete?" without manual git investigation.
| Command | Purpose | Overlap |
|---|---|---|
/gsd doctor |
State integrity, git health, worktrees, runtime, env, prefs | Primary health system |
/gsd doctor fix |
Auto-fix detected issues | |
/gsd doctor heal |
Dispatch unfixable issues to LLM | |
/gsd doctor audit |
Expanded output, no fix | |
/gsd cleanup |
Runs branches + snapshots cleanup | Redundant — doctor already handles branches |
/gsd cleanup branches |
Delete merged gsd/* branches |
Redundant — doctor detects but won't fix legacy branches |
/gsd cleanup snapshots |
Prune old snapshot refs | Gap — doctor has no snapshot check |
/gsd cleanup projects |
Audit orphaned ~/.gsd/projects/ dirs |
Fully redundant — doctor's orphaned_project_state does the same |
/gsd keys doctor |
Per-key health check | Complementary — deeper than doctor's surface provider check |
/gsd skill-health |
Skill usage stats | No overlap — analytics, not health |
/gsd inspect |
SQLite DB diagnostics | No overlap — introspection tool |
/gsd forensics |
Post-failure investigation | No overlap — different lifecycle |
- No worktree lifecycle checks (merged? stale? dirty? unpushed?)
/worktree listshows name/branch/path but no safety status- Doctor checks completed-milestone worktrees but nothing else
Principle: Doctor finds problems. Doctor fix resolves them. One command, not three paths to the same outcome.
Add to doctor-checks.ts → checkGitHealth():
| Check code | Severity | Fixable | Condition | What --fix does |
|---|---|---|---|---|
worktree_branch_merged |
info | yes | Worktree's branch is fully merged into main (merge-base --is-ancestor) | Remove worktree + delete branch |
worktree_stale |
warning | no | No commits in 14+ days AND no open PR on remote | Report only — needs user decision |
worktree_dirty |
warning | no | Stale worktree has uncommitted changes | Report only — data loss risk |
worktree_unpushed |
warning | no | Worktree branch has commits not on any remote | Report only — push first |
Scope: Only GSD-managed worktrees under .gsd/worktrees/. Not .claude/worktrees/, not sibling repos, not /tmp/ worktrees. GSD owns what GSD creates.
Safety rules:
- Never auto-remove a worktree matching
process.cwd()(existing pattern) - Never auto-remove a worktree with uncommitted changes
- Never auto-remove a worktree with unpushed commits
worktree_branch_mergedis the only auto-fixable worktree check — it's the safest (work is already in main)
2a. Make legacy_slice_branches fixable in doctor.
Currently detected as info severity, not fixable. Change to:
- Severity:
info(keep) - Fixable:
true --fixaction:nativeBranchDelete(basePath, branch, true)for each merged legacy branch
This makes cleanup branches redundant — doctor handles both milestone/* and gsd/* branches.
2b. Add snapshot_ref_bloat doctor check.
New check in checkRuntimeHealth():
- Count
refs/gsd/snapshots/refs - If > 50 refs per label, report
snapshot_ref_bloat(warning, fixable) --fixaction: prune to newest 5 per label (same logic as existinghandleCleanupSnapshots)
This makes cleanup snapshots redundant.
2c. /gsd cleanup projects is already redundant.
Doctor's orphaned_project_state check (in checkGlobalHealth) does the same thing. No code change needed — just deprecation.
2d. /gsd cleanup becomes a permanent alias.
/gsd cleanup→ runsdoctor fixscoped to cleanup-class issues (branches, snapshots, projects, worktrees)/gsd cleanup branches→ doctor fix for branch issues/gsd cleanup snapshots→ doctor fix for snapshot issues/gsd cleanup projects→ doctor fix for project state issues/gsd cleanup worktrees→ doctor fix for worktree issues
No deprecation warnings. Same commands, doctor under the hood. Existing muscle memory keeps working.
Enhance handleList() in worktree-command.ts to show safety information inline:
GSD Worktrees
feature-x ● active
branch worktree/feature-x
path .gsd/worktrees/feature-x
status 3 uncommitted files · 2 unpushed commits · last commit 4h ago
old-bugfix
branch worktree/old-bugfix
path .gsd/worktrees/old-bugfix
status ✓ merged into main · safe to remove
stale-experiment
branch worktree/stale-experiment
path .gsd/worktrees/stale-experiment
status ⚠ no commits in 18 days · no open PR
Data to show per worktree:
- Uncommitted file count (if any)
- Unpushed commit count (if any)
- Merge status (merged into main or not)
- Last commit age
- Whether branch has been pushed to remote
For discoverability, add to the cleanup catalog:
/gsd cleanup worktrees — Remove merged/safe-to-delete worktrees
/gsd cleanup worktrees --dry — Preview what would be removed
This is a thin wrapper that runs doctor fix scoped to worktree_branch_merged issues only.
| Command | Why |
|---|---|
/gsd keys doctor |
Deeper per-key analysis; general doctor's provider check is a sufficient surface check |
/gsd inspect |
DB introspection — not a health check |
/gsd skill-health |
Usage analytics — not a health check |
/gsd forensics |
Post-mortem investigation — different purpose and lifecycle |
/gsd logs |
Read-only log viewer |
- Phase 1 — Worktree lifecycle checks in doctor (the core ask)
- Phase 3 — Enhanced
/worktree list(immediate user value, depends on same data as Phase 1) - Phase 2 — Fold cleanup into doctor (reduces surface area)
- Phase 4 — Cleanup worktrees convenience entry (trivial once Phase 1+2 land)
Phase 1 and 3 share git inspection code (merge status, uncommitted changes, unpushed commits). Build that as shared helpers in worktree-manager.ts or a new worktree-health.ts, then both phases consume it.
| File | Changes |
|---|---|
doctor-checks.ts |
New worktree lifecycle checks, make legacy_slice_branches fixable, add snapshot bloat check |
doctor-types.ts |
New issue codes: worktree_branch_merged, worktree_stale, worktree_dirty, worktree_unpushed, snapshot_ref_bloat |
worktree-manager.ts |
New helpers: getWorktreeMergeStatus(), getWorktreeDirtyStatus(), getWorktreeUnpushedCount(), getWorktreeLastCommitAge() |
worktree-command.ts |
Enhanced handleList() with safety status |
commands-maintenance.ts |
Deprecation wrappers for cleanup subcommands |
commands/catalog.ts |
Add worktrees to cleanup subcommands, update doctor subcommand descriptions |
commands/handlers/ops.ts |
Wire up /gsd cleanup worktrees |
- Stale threshold — 14 days default, configurable via preferences.
- Remote PR check — Commit age is the primary signal. PR check is a bonus when
ghis available. Degrade gracefully ifghis missing. - Cleanup as permanent alias —
/gsd cleanupstays as a permanent alias that silently calls doctor fix under the hood. No deprecation noise. Users who learned cleanup keep using it, new users learn doctor.