From ck
Explains Cavekit autonomous execution loop: state machine, stop hook, completion sentinel, lock, budgets, iteration cap, files, lifecycle. Use before /ck:make, stuck session debug, command dev.
npx claudepluginhub juliusbrussee/cavekitThis skill uses the workspace's default tool permissions.
The loop lets a single `/ck:make` invocation run dozens of agent iterations
Searches, retrieves, and installs Agent Skills from prompts.chat registry using MCP tools like search_skills and get_skill. Activates for finding skills, browsing catalogs, or extending Claude.
Searches prompts.chat for AI prompt templates by keyword or category, retrieves by ID with variable handling, and improves prompts via AI. Use for discovering or enhancing prompts.
Guides agent creation for Claude Code plugins with file templates, frontmatter specs (name, description, model), triggering examples, system prompts, and best practices.
The loop lets a single /ck:make invocation run dozens of agent iterations
without user intervention, while still respecting per-task budgets, session
budgets, and explicit user-approval gates.
┌─────────────────┐ Stop event ┌────────────────────┐
│ Claude Code │──────────────▶│ stop-hook.sh │
│ session │ │ (.cavekit/loop │
└─────────────────┘ │ active?) │
▲ └─────────┬──────────┘
│ │ route()
│ next prompt injected ▼
│ ┌────────────────────┐
│ │ cavekit-tools.cjs │
│ │ routeDecision() │
│ │ + status-block │
│ │ + backprop- │
│ │ directive │
│ └─────────┬──────────┘
│ │
└───────── {decision:"block", │
reason:<next prompt>} ◀─┘
All under <project>/.cavekit/:
| File | Writer | Purpose |
|---|---|---|
.loop.json | setup-loop | Sentinel; stop-hook no-ops without it. |
.loop.lock | stop-hook (heartbeat) | Single-writer lock. PID + hostname + ts. |
state.md | cavekit-tools | phase, current_task, iteration. |
token-ledger.json | token-monitor hook | Session + per-task token tallies. |
task-status.json | commands | Authoritative task registry. |
.progress.json | progress-tracker | Zero-context UI snapshot. |
.auto-backprop-pending.json | auto-backprop hook | Flag file from failed tests. |
history/backprop-log.md | backprop skill | Append-only trace log. |
capabilities.json | discover command | MCP + CLI tool detection. |
Setup — /ck:make calls:
node "${CLAUDE_PLUGIN_ROOT}/scripts/cavekit-tools.cjs" setup-loop
This writes .loop.json (activating the stop hook) and resets state.md.
Work — the agent does one wave of task execution per iteration.
Stop fires — Claude Code's Stop event triggers stop-hook.sh. The hook:
session_id and transcript_path<promise>CAVEKIT COMPLETE</promise>routeDecision() for the next prompt{"decision":"block","reason":<next prompt>}Repeat — Claude Code treats decision:block + reason:... as a new
user message, so the session continues.
Teardown — one of:
CAVEKIT_MAX_ITERATIONS)CAVEKIT_BUDGET_EXHAUSTED)CAVEKIT_LOCK_CONFLICT)To end the loop cleanly, emit exactly:
<promise>CAVEKIT COMPLETE</promise>
The hook searches for this literal in the last 20 transcript lines. Put it on its own line at the very end of the final message. Do not wrap it in code fences or paraphrase it — the search is a literal substring match.
When routeDecision() cannot safely continue, it returns one of these
strings instead of a prompt:
| Sentinel | Meaning |
|---|---|
CAVEKIT_LOOP_DONE | All tasks complete. Hook exits silent. |
CAVEKIT_MAX_ITERATIONS | Iteration cap hit. Loop halted. |
CAVEKIT_BUDGET_EXHAUSTED | Session budget exhausted. |
CAVEKIT_LOCK_CONFLICT | Another session owns the lock. |
The hook translates each into a short user-facing message before returning.
The lock is a JSON file (.loop.lock) with {owner, pid, host, heartbeat_at}.
session:<session_id>.heartbeat_at.CAVEKIT_LOCK_CONFLICT.Never delete .loop.lock while a session might be active. Use
cavekit-tools release-lock --owner <tag> instead.
# Inspect current state
node "${CLAUDE_PLUGIN_ROOT}/scripts/cavekit-tools.cjs" status
# Who holds the lock?
cat .cavekit/.loop.lock
# What would the router do right now?
node "${CLAUDE_PLUGIN_ROOT}/scripts/cavekit-tools.cjs" route
# Drop the loop entirely (safe after a crash)
node "${CLAUDE_PLUGIN_ROOT}/scripts/cavekit-tools.cjs" teardown-loop
Turn on debug logging by exporting CAVEKIT_DEBUG=1. The hook will write to
.cavekit/.debug.log.
state.md's phase field while a loop is active. The
hook assumes single-writer semantics and will overwrite you.complete. The hook trusts the sentinel and tears down immediately.