Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 2 additions & 2 deletions .claude-plugin/marketplace.json
Original file line number Diff line number Diff line change
Expand Up @@ -6,14 +6,14 @@
},
"metadata": {
"description": "Token Pilot \u2014 save 60-90% tokens when AI reads code",
"version": "0.40.0"
"version": "0.41.0"
},
"plugins": [
{
"name": "token-pilot",
"source": "./",
"description": "Reduces token consumption by 60-90% via AST-aware lazy file reading, structural symbol navigation, and cross-session tool-usage analytics. 22 MCP tools + 19 subagents + budget watchdog hooks.",
"version": "0.40.0",
"version": "0.41.0",
"author": {
"name": "Digital-Threads"
},
Expand Down
2 changes: 1 addition & 1 deletion .claude-plugin/plugin.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "token-pilot",
"version": "0.40.0",
"version": "0.41.0",
"description": "Saves 60-90% tokens on AI code reading. AST-aware lazy reads, symbol navigation, find_usages, structural git diff/log, edit-safety guard, Task-routing matcher, cross-session telemetry (errors + diagnostics), 25 tp-* subagents tiered to haiku/sonnet/opus with budget watchdog.",
"author": {
"name": "Digital-Threads",
Expand Down
15 changes: 15 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -232,6 +232,21 @@ lines instead of bouncing the call. The structural summary still
rides along in `additionalContext`. Default OFF because the field
is undocumented and may change.

### Experimental: SubagentStop budget feedback (CC 2.1.163+)

Every subagent completion already lands a task-telemetry row via the
`SubagentStop` hook (that's how `stats --tasks` knows what you
dispatched). With `TOKEN_PILOT_SUBAGENT_FEEDBACK=1` the same hook also
returns `additionalContext` — when a `token-pilot workflow` fan-out is
at ≥90 % of its token ceiling, each completing agent gets a wind-down
note so a hundred-agent `/workflow` run stops before blowing the
budget.

**Requires Claude Code 2.1.163+.** Returning `additionalContext` from
`SubagentStop` is only honoured there; older Claude Code labels it a
hook error. Default OFF for that reason — enable only once
`claude --version` reports 2.1.163 or later.

## What's new for Claude Code 2.1.151+

These notes are about behaviour you'll see automatically once you
Expand Down
2 changes: 1 addition & 1 deletion agents/tp-api-surface-tracker.md
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@ tools:
- mcp__token-pilot__read_symbol
- Bash
model: haiku
token_pilot_version: "0.40.0"
token_pilot_version: "0.41.0"
token_pilot_body_hash: dd184501203fa7f3c73f419c4ffbe33c4be75400cb64a7a51733a3fe23f6e085
requiredMcpServers:
- "token-pilot"
Expand Down
2 changes: 1 addition & 1 deletion agents/tp-audit-scanner.md
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@ tools:
- Grep
- Read
model: sonnet
token_pilot_version: "0.40.0"
token_pilot_version: "0.41.0"
token_pilot_body_hash: d172f600bf32277ea6eb4cbbee4542ddd698a986dcd96997d33930561964569b
requiredMcpServers:
- "token-pilot"
Expand Down
2 changes: 1 addition & 1 deletion agents/tp-commit-writer.md
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@ tools:
- mcp__token-pilot__test_summary
- mcp__token-pilot__outline
- Bash
token_pilot_version: "0.40.0"
token_pilot_version: "0.41.0"
token_pilot_body_hash: de64a406b5176de19f7422619c7de7949b1f28865f225402c9cea9255f377428
requiredMcpServers:
- "token-pilot"
Expand Down
2 changes: 1 addition & 1 deletion agents/tp-context-engineer.md
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,7 @@ tools:
- Edit
- Glob
model: sonnet
token_pilot_version: "0.40.0"
token_pilot_version: "0.41.0"
token_pilot_body_hash: 68b32af2dacd82ebe52c4eec93edb903d452688274c3065218270627c564d8b0
requiredMcpServers:
- "token-pilot"
Expand Down
2 changes: 1 addition & 1 deletion agents/tp-dead-code-finder.md
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@ tools:
- Grep
- Read
model: sonnet
token_pilot_version: "0.40.0"
token_pilot_version: "0.41.0"
token_pilot_body_hash: d9b7f5b7ae6f4ae21305c775361bcab097cc774370a6d976c093571d46d55021
requiredMcpServers:
- "token-pilot"
Expand Down
2 changes: 1 addition & 1 deletion agents/tp-debugger.md
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,7 @@ tools:
- Read
- Bash
model: sonnet
token_pilot_version: "0.40.0"
token_pilot_version: "0.41.0"
token_pilot_body_hash: 052413de8d92377edcde6ae5c823f5378db304baccfa29e8866467f42553a500
requiredMcpServers:
- "token-pilot"
Expand Down
2 changes: 1 addition & 1 deletion agents/tp-dep-health.md
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@ tools:
- Bash
- Read
model: haiku
token_pilot_version: "0.40.0"
token_pilot_version: "0.41.0"
token_pilot_body_hash: e14dc57493d816f8c2e017963e2ef5f66bea50fd0b805a80e8a0d97c968427e7
requiredMcpServers:
- "token-pilot"
Expand Down
2 changes: 1 addition & 1 deletion agents/tp-doc-writer.md
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,7 @@ tools:
- Edit
- Glob
model: haiku
token_pilot_version: "0.40.0"
token_pilot_version: "0.41.0"
token_pilot_body_hash: 57d741794ab40e31a7ac49c68ea39a9088f5827cdef866ce81bfca1b7c9180cf
requiredMcpServers:
- "token-pilot"
Expand Down
2 changes: 1 addition & 1 deletion agents/tp-history-explorer.md
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@ tools:
- Bash
- Read
model: haiku
token_pilot_version: "0.40.0"
token_pilot_version: "0.41.0"
token_pilot_body_hash: 7b70fa76a60e3c58a1de4f56c32c0f166424137e203a0cf1c8654e7c9235d904
requiredMcpServers:
- "token-pilot"
Expand Down
2 changes: 1 addition & 1 deletion agents/tp-impact-analyzer.md
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,7 @@ tools:
- mcp__token-pilot__read_symbols
- Read
model: sonnet
token_pilot_version: "0.40.0"
token_pilot_version: "0.41.0"
token_pilot_body_hash: 351a987e11eba63852f5431a16d8eb53104f4f689f82fdcc5a2bf4db948ba92f
requiredMcpServers:
- "token-pilot"
Expand Down
2 changes: 1 addition & 1 deletion agents/tp-incident-timeline.md
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@ tools:
- mcp__token-pilot__read_symbol
- Bash
model: inherit
token_pilot_version: "0.40.0"
token_pilot_version: "0.41.0"
token_pilot_body_hash: de5722bfea374eaab096c1ae635c37879e7a91370ee3cd0532f4240be03c91eb
requiredMcpServers:
- "token-pilot"
Expand Down
2 changes: 1 addition & 1 deletion agents/tp-incremental-builder.md
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,7 @@ tools:
- Edit
- Bash
model: sonnet
token_pilot_version: "0.40.0"
token_pilot_version: "0.41.0"
token_pilot_body_hash: 375a824d0d847bb5453ec594c7a62ad566ee7e4d92717b0473f771f1a0477c60
requiredMcpServers:
- "token-pilot"
Expand Down
2 changes: 1 addition & 1 deletion agents/tp-migration-scout.md
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@ tools:
- Grep
- Glob
model: sonnet
token_pilot_version: "0.40.0"
token_pilot_version: "0.41.0"
token_pilot_body_hash: 0334de1bf99b431b65359637d125cda7c44c6f780eb92c57cc538715b1939536
requiredMcpServers:
- "token-pilot"
Expand Down
2 changes: 1 addition & 1 deletion agents/tp-onboard.md
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@ tools:
- mcp__token-pilot__smart_read
- mcp__token-pilot__smart_read_many
- mcp__token-pilot__read_section
token_pilot_version: "0.40.0"
token_pilot_version: "0.41.0"
token_pilot_body_hash: 832e95633fbc8e9b0c10f3e540a327d4be062fb4b3f17a6cce6be13f414e2927
requiredMcpServers:
- "token-pilot"
Expand Down
2 changes: 1 addition & 1 deletion agents/tp-performance-profiler.md
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@ tools:
- Bash
- Read
model: sonnet
token_pilot_version: "0.40.0"
token_pilot_version: "0.41.0"
token_pilot_body_hash: b61f06380d80798fa2e49d37bcba0653495bee04dd6bdbc1feff9a75607b0508
requiredMcpServers:
- "token-pilot"
Expand Down
2 changes: 1 addition & 1 deletion agents/tp-pr-reviewer.md
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@ tools:
- mcp__token-pilot__read_for_edit
- Read
model: sonnet
token_pilot_version: "0.40.0"
token_pilot_version: "0.41.0"
token_pilot_body_hash: f83f50d05b4f70285ae7afed2b1a406fc436df56e61a0aedbfb31edc7f2b6e66
requiredMcpServers:
- "token-pilot"
Expand Down
2 changes: 1 addition & 1 deletion agents/tp-refactor-planner.md
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@ tools:
- mcp__token-pilot__outline
- mcp__token-pilot__read_symbol
model: sonnet
token_pilot_version: "0.40.0"
token_pilot_version: "0.41.0"
token_pilot_body_hash: c5f6fc122c89e16e5cf774045f92169ee3468555320b898171ba13eca5323550
requiredMcpServers:
- "token-pilot"
Expand Down
2 changes: 1 addition & 1 deletion agents/tp-review-impact.md
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@ tools:
- mcp__token-pilot__module_info
- Bash
model: sonnet
token_pilot_version: "0.40.0"
token_pilot_version: "0.41.0"
token_pilot_body_hash: 8ef3c3341cbfed4eb8dd130126a9683edc57e378c92ff0ca764d584fd941c55c
requiredMcpServers:
- "token-pilot"
Expand Down
2 changes: 1 addition & 1 deletion agents/tp-run.md
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@ tools:
- Glob
- Bash
model: haiku
token_pilot_version: "0.40.0"
token_pilot_version: "0.41.0"
token_pilot_body_hash: 2b08618d34a61f00aafccbda9fed6d83243296dedb83440edbd2d5c28bb6dbc4
requiredMcpServers:
- "token-pilot"
Expand Down
2 changes: 1 addition & 1 deletion agents/tp-session-restorer.md
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@ tools:
- mcp__token-pilot__session_budget
- Bash
- Read
token_pilot_version: "0.40.0"
token_pilot_version: "0.41.0"
token_pilot_body_hash: 529374ed728f5eed5b758b3be3da65624783c0bf0c1a253d7d661a843eb5f767
requiredMcpServers:
- "token-pilot"
Expand Down
2 changes: 1 addition & 1 deletion agents/tp-ship-coordinator.md
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@ tools:
- Read
- Grep
model: sonnet
token_pilot_version: "0.40.0"
token_pilot_version: "0.41.0"
token_pilot_body_hash: a60f6ae110eb3138064bce074e8ba26fa0ce5f4659df1624a9d9d3646803391b
requiredMcpServers:
- "token-pilot"
Expand Down
2 changes: 1 addition & 1 deletion agents/tp-spec-writer.md
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@ tools:
- Read
- Write
model: sonnet
token_pilot_version: "0.40.0"
token_pilot_version: "0.41.0"
token_pilot_body_hash: c7a4e8b39228fd5158528f389c924c5ff2d98c4b9b05ee0106d54a26c5dc1350
requiredMcpServers:
- "token-pilot"
Expand Down
2 changes: 1 addition & 1 deletion agents/tp-test-coverage-gapper.md
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@ tools:
- mcp__token-pilot__test_summary
- Glob
- Grep
token_pilot_version: "0.40.0"
token_pilot_version: "0.41.0"
token_pilot_body_hash: be81eed53a3720d146cf89e4a14a7a56577633f7c84c234c412ab70d64c05b11
requiredMcpServers:
- "token-pilot"
Expand Down
2 changes: 1 addition & 1 deletion agents/tp-test-triage.md
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@ tools:
- mcp__token-pilot__find_usages
- mcp__token-pilot__read_symbol
model: sonnet
token_pilot_version: "0.40.0"
token_pilot_version: "0.41.0"
token_pilot_body_hash: 362ecf4cb03b059421ea26933473700900073dc38b3a7fe271208dfb1ae14f90
requiredMcpServers:
- "token-pilot"
Expand Down
2 changes: 1 addition & 1 deletion agents/tp-test-writer.md
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,7 @@ tools:
- Edit
- Bash
model: sonnet
token_pilot_version: "0.40.0"
token_pilot_version: "0.41.0"
token_pilot_body_hash: 269f2fe22ff4517c277d3f56ca67d8a5527b93290ab21079a83ba7af22c1b5a9
requiredMcpServers:
- "token-pilot"
Expand Down
2 changes: 1 addition & 1 deletion package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "token-pilot",
"version": "0.40.0",
"version": "0.41.0",
"description": "Save up to 80% tokens when AI reads code — MCP server for token-efficient code navigation, AST-aware structural reading instead of dumping full files into context window",
"type": "module",
"main": "dist/index.js",
Expand Down
61 changes: 61 additions & 0 deletions src/hooks/subagent-stop.ts
Original file line number Diff line number Diff line change
Expand Up @@ -121,3 +121,64 @@ export function buildSubagentTaskEvent(
code: "subagent_stop",
};
}

// ─── v0.41.0 SubagentStop feedback ───────────────────────────────────

export interface SubagentFeedbackContext {
/** Active workflow budget status, when a fleet workflow is running. */
workflow?: {
workflow_id: string;
budget_tokens: number | null;
used_tokens: number;
pct: number | null;
} | null;
}

/**
* Decide the `additionalContext` feedback to hand back from a
* SubagentStop hook. Pure — caller resolves the workflow status.
*
* Returns a short wind-down note when an active fleet workflow is at or
* past 90 % of its token ceiling, so a `/workflow`-style fan-out winds
* down before the budget is blown. Returns null otherwise — we do NOT
* nag on every completion; broad adoption nudges stay in SessionStart.
*
* Emission is the caller's responsibility and is gated behind
* TOKEN_PILOT_SUBAGENT_FEEDBACK=1 + Claude Code 2.1.163+ (older Claude
* Code labels a SubagentStop hookSpecificOutput return as a hook error).
*/
export function decideSubagentFeedback(
_input: SubagentStopInput,
ctx: SubagentFeedbackContext,
): string | null {
const wf = ctx.workflow;
if (
wf &&
wf.budget_tokens != null &&
wf.budget_tokens > 0 &&
wf.used_tokens >= wf.budget_tokens * 0.9
) {
const pct = wf.pct != null ? `${wf.pct}%` : "~90%";
return (
`[token-pilot] workflow ${wf.workflow_id} is at ${pct} of its ` +
`${wf.budget_tokens} token ceiling (${wf.used_tokens} used). ` +
`Wind down the fan-out: finish in-flight branches and report ` +
`rather than dispatching new agents.`
);
}
return null;
}

/**
* Render the SubagentStop hook JSON response carrying feedback. Returns
* null when there is nothing to say (caller writes no stdout).
*/
export function renderSubagentFeedback(message: string | null): string | null {
if (!message) return null;
return JSON.stringify({
hookSpecificOutput: {
hookEventName: "SubagentStop",
additionalContext: message,
},
});
}
37 changes: 34 additions & 3 deletions src/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -407,14 +407,45 @@ export async function main(cliArgs = process.argv.slice(2)): Promise<void> {
await runHookEntryPoint({ hook: "hook-subagent-stop" }, async () => {
const stdin = readFileSync(0, "utf-8");
const input = JSON.parse(stdin);
const { buildSubagentTaskEvent } = await import(
"./hooks/subagent-stop.js"
);
const {
buildSubagentTaskEvent,
decideSubagentFeedback,
renderSubagentFeedback,
} = await import("./hooks/subagent-stop.js");
const ev = buildSubagentTaskEvent(input, Date.now());
if (ev) {
const { appendEvent } = await import("./core/event-log.js");
await appendEvent(process.cwd(), ev);
}

// v0.41.0 — optional SubagentStop feedback. Returning
// hookSpecificOutput.additionalContext from SubagentStop is a
// Claude Code 2.1.163+ feature; on older Claude Code it is
// labelled a hook error (noise). Gate strictly behind
// TOKEN_PILOT_SUBAGENT_FEEDBACK=1 so the default path (telemetry
// only) stays safe on every version.
if (process.env.TOKEN_PILOT_SUBAGENT_FEEDBACK === "1") {
const { activeWorkflowId, workflowStatus } = await import(
"./core/workflow.js"
);
let wf = null;
const wfId = activeWorkflowId();
if (wfId) {
const st = await workflowStatus(process.cwd(), wfId);
if (st) {
wf = {
workflow_id: st.workflow_id,
budget_tokens: st.budget_tokens,
used_tokens: st.used_tokens,
pct: st.pct,
};
}
}
const rendered = renderSubagentFeedback(
decideSubagentFeedback(input, { workflow: wf }),
);
if (rendered) process.stdout.write(rendered);
}
});
return;
}
Expand Down
Loading
Loading