@pugi/cli 0.1.0-beta.1 → 0.1.0-beta.100
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +132 -0
- package/LICENSE +1 -1
- package/README.md +53 -11
- package/THIRD_PARTY_NOTICES.md +40 -0
- package/assets/pugi-mascot.ansi +15 -40
- package/assets/pugi-prozr2-mascot.ansi +9 -0
- package/bin/run.js +33 -1
- package/dist/commands/deploy.js +40 -40
- package/dist/commands/flatten.js +191 -0
- package/dist/commands/jobs-watch.js +201 -0
- package/dist/commands/jobs.js +42 -27
- package/dist/commands/retro.js +210 -0
- package/dist/commands/smoke.js +133 -0
- package/dist/core/agent-progress/cleanup.js +134 -0
- package/dist/core/agent-progress/schema.js +144 -0
- package/dist/core/agent-progress/writer.js +101 -0
- package/dist/core/agents/adaptive-router.js +330 -0
- package/dist/core/agents/query-decomposer.js +297 -0
- package/dist/core/agents/registry.js +3 -3
- package/dist/core/approvals/shortcut-resolver.js +98 -0
- package/dist/core/artifact-chain/dispatcher.js +148 -0
- package/dist/core/artifact-chain/exporter.js +164 -0
- package/dist/core/artifact-chain/state.js +243 -0
- package/dist/core/artifact-chain/steps.js +169 -0
- package/dist/core/ask-user/question.js +92 -0
- package/dist/core/audit/audit-trail.js +275 -0
- package/dist/core/auth/ensure-authenticated.js +129 -0
- package/dist/core/auth/env-provider.js +238 -0
- package/dist/core/auto-open-browser.js +4 -4
- package/dist/core/auto-update/channels.js +122 -0
- package/dist/core/auto-update/checker.js +241 -0
- package/dist/core/auto-update/state.js +235 -0
- package/dist/core/bare-mode/index.js +107 -0
- package/dist/core/bash/redirect.js +281 -0
- package/dist/core/bash-classifier.js +436 -40
- package/dist/core/checkpoint/resumer.js +149 -0
- package/dist/core/checkpoint/rewinder.js +291 -0
- package/dist/core/checkpoints/shadow-git.js +670 -0
- package/dist/core/citations/parser.js +109 -0
- package/dist/core/classifier/yolo-classifier.js +88 -0
- package/dist/core/codegraph/db.js +506 -0
- package/dist/core/codegraph/decision-store.js +248 -0
- package/dist/core/codegraph/detect-repo.js +459 -0
- package/dist/core/codegraph/install.js +134 -0
- package/dist/core/codegraph/offer-hook.js +220 -0
- package/dist/core/codegraph/parser.js +71 -0
- package/dist/core/codegraph/types.js +34 -0
- package/dist/core/compact/auto-trigger.js +96 -0
- package/dist/core/compact/buffer-rewriter.js +115 -0
- package/dist/core/compact/summarizer.js +208 -0
- package/dist/core/compact/token-counter.js +108 -0
- package/dist/core/consensus/anvil-fanout.js +25 -25
- package/dist/core/consensus/diff-capture.js +121 -12
- package/dist/core/consensus/rubric.js +21 -21
- package/dist/core/context/builder.js +6 -6
- package/dist/core/context/compaction-events.js +8 -8
- package/dist/core/context/compaction.js +31 -31
- package/dist/core/context/index.js +15 -8
- package/dist/core/context/invariants.js +51 -51
- package/dist/core/context/markdown-loader.js +28 -10
- package/dist/core/context/markdown-traverse.js +255 -0
- package/dist/core/context/pugiignore.js +41 -41
- package/dist/core/context/repo-skeleton.js +37 -37
- package/dist/core/context/tool-eviction.js +55 -0
- package/dist/core/context/watcher.js +32 -32
- package/dist/core/context/working-set.js +23 -23
- package/dist/core/coordinator/agent-tools.js +77 -0
- package/dist/core/coordinator/agent-toolset.js +65 -0
- package/dist/core/coordinator/fsm.js +73 -0
- package/dist/core/coordinator/mode-fsm.js +70 -0
- package/dist/core/cost/rate-card.js +129 -0
- package/dist/core/cost/tracker.js +221 -0
- package/dist/core/credentials.js +13 -13
- package/dist/core/cron/scheduler.js +138 -0
- package/dist/core/denial-tracking/index.js +8 -0
- package/dist/core/denial-tracking/state.js +264 -0
- package/dist/core/diagnostics/probe-runner.js +93 -0
- package/dist/core/diagnostics/probes/api.js +46 -0
- package/dist/core/diagnostics/probes/auth.js +93 -0
- package/dist/core/diagnostics/probes/bare-mode.js +42 -0
- package/dist/core/diagnostics/probes/cli-version.js +127 -0
- package/dist/core/diagnostics/probes/config.js +72 -0
- package/dist/core/diagnostics/probes/denial-tracking.js +57 -0
- package/dist/core/diagnostics/probes/disk.js +81 -0
- package/dist/core/diagnostics/probes/engine-live.js +46 -0
- package/dist/core/diagnostics/probes/git.js +65 -0
- package/dist/core/diagnostics/probes/hooks.js +118 -0
- package/dist/core/diagnostics/probes/mcp.js +75 -0
- package/dist/core/diagnostics/probes/node.js +59 -0
- package/dist/core/diagnostics/probes/pnpm.js +36 -0
- package/dist/core/diagnostics/probes/pugi-md.js +89 -0
- package/dist/core/diagnostics/probes/sandbox.js +72 -0
- package/dist/core/diagnostics/probes/session.js +74 -0
- package/dist/core/diagnostics/probes/status-snapshot.js +488 -0
- package/dist/core/diagnostics/probes/workspace.js +63 -0
- package/dist/core/diagnostics/types.js +70 -0
- package/dist/core/dispatch/cache-cleanup.js +197 -0
- package/dist/core/dispatch/cache-handoff.js +295 -0
- package/dist/core/edits/apply-patch-layer-e.js +189 -0
- package/dist/core/edits/dispatch.js +333 -7
- package/dist/core/edits/format-detector.js +260 -0
- package/dist/core/edits/format-matrix.js +26 -0
- package/dist/core/edits/fuzzy-ladder.js +650 -0
- package/dist/core/edits/index.js +5 -1
- package/dist/core/edits/journal.js +199 -0
- package/dist/core/edits/layer-a-apply.js +15 -15
- package/dist/core/edits/layer-a-fuzzy-apply.js +198 -0
- package/dist/core/edits/layer-b-apply.js +9 -9
- package/dist/core/edits/layer-c-apply.js +6 -6
- package/dist/core/edits/layer-d-ast.js +557 -14
- package/dist/core/edits/marker-parser.js +12 -12
- package/dist/core/edits/security-gate.js +27 -27
- package/dist/core/edits/verify-hook.js +273 -0
- package/dist/core/edits/worktree.js +322 -0
- package/dist/core/engine/anvil-client.js +214 -26
- package/dist/core/engine/auto-compact.js +247 -0
- package/dist/core/engine/budgets.js +220 -0
- package/dist/core/engine/compact-llm-summarizer.js +124 -0
- package/dist/core/engine/context-prefix.js +155 -0
- package/dist/core/engine/index.js +1 -1
- package/dist/core/engine/intensity.js +163 -0
- package/dist/core/engine/intent.js +260 -0
- package/dist/core/engine/native-pugi.js +1559 -227
- package/dist/core/engine/prompts.js +192 -16
- package/dist/core/engine/strip-internal-fields.js +124 -0
- package/dist/core/engine/tool-bridge.js +1887 -59
- package/dist/core/engine/verification-patterns.js +195 -0
- package/dist/core/evaluation/golden-dataset.js +293 -0
- package/dist/core/feedback/queue.js +177 -0
- package/dist/core/feedback/submitter.js +145 -0
- package/dist/core/file-cache.js +113 -1
- package/dist/core/flatten/flatten-repo.js +439 -0
- package/dist/core/format/osc8-link.js +28 -0
- package/dist/core/hook-chains.js +392 -0
- package/dist/core/hooks/citation-verify-hook.js +138 -0
- package/dist/core/hooks/citation-verify.js +112 -0
- package/dist/core/hooks/events.js +46 -0
- package/dist/core/hooks/index.js +15 -0
- package/dist/core/hooks/registry.js +216 -0
- package/dist/core/hooks/runner.js +236 -0
- package/dist/core/hooks/v2/event-emitter.js +115 -0
- package/dist/core/hooks/v2/executor.js +282 -0
- package/dist/core/hooks/v2/index.js +25 -0
- package/dist/core/hooks/v2/lifecycle.js +104 -0
- package/dist/core/hooks/v2/loader.js +216 -0
- package/dist/core/hooks/v2/matcher.js +125 -0
- package/dist/core/hooks/v2/trust.js +143 -0
- package/dist/core/hooks/v2/types.js +86 -0
- package/dist/core/hooks/worktree-events.js +158 -0
- package/dist/core/image/renderer.js +71 -0
- package/dist/core/init/detector.js +582 -0
- package/dist/core/init/template-renderer.js +242 -0
- package/dist/core/jobs/registry.js +18 -18
- package/dist/core/ledger/results-tsv.js +142 -0
- package/dist/core/log-discipline/stdout-redirect.js +51 -0
- package/dist/core/lsp/cache.js +105 -0
- package/dist/core/lsp/client.js +1229 -0
- package/dist/core/lsp/language-detect.js +66 -0
- package/dist/core/lsp/post-edit-diagnostics.js +171 -0
- package/dist/core/lsp/server-detect.js +173 -0
- package/dist/core/lsp/symbol-cache.js +162 -0
- package/dist/core/lsp/symbol-tools.js +664 -0
- package/dist/core/mcp/client.js +97 -28
- package/dist/core/mcp/http-server.js +553 -0
- package/dist/core/mcp/orchestrator-config.js +192 -0
- package/dist/core/mcp/orchestrator-tools.js +806 -0
- package/dist/core/mcp/permission.js +190 -0
- package/dist/core/mcp/registry.js +39 -17
- package/dist/core/mcp/server-tools.js +219 -0
- package/dist/core/mcp/server.js +397 -0
- package/dist/core/mcp/trust.js +10 -10
- package/dist/core/memory/dual-write.js +416 -0
- package/dist/core/memory/passive-extract.js +130 -0
- package/dist/core/memory/phase1-kinds.js +20 -0
- package/dist/core/memory/secret-scanner.js +304 -0
- package/dist/core/memory-sync/queue.js +170 -0
- package/dist/core/metrics/extract.js +113 -0
- package/dist/core/modes/roo-modes.js +68 -0
- package/dist/core/notes/notes-paths.js +113 -0
- package/dist/core/notes/notes-recorder.js +140 -0
- package/dist/core/notes/notes-writer.js +53 -0
- package/dist/core/notes/renderers.js +0 -0
- package/dist/core/notes/slug.js +105 -0
- package/dist/core/onboarding/ensure-initialized.js +133 -0
- package/dist/core/onboarding/marker.js +111 -0
- package/dist/core/onboarding/telemetry-state.js +108 -0
- package/dist/core/output-style/presets.js +176 -0
- package/dist/core/output-style/state.js +185 -0
- package/dist/core/path-security.js +287 -5
- package/dist/core/permission.js +82 -22
- package/dist/core/permissions/auto-classifier.js +124 -0
- package/dist/core/permissions/bash-parser.js +371 -0
- package/dist/core/permissions/circuit-breaker.js +83 -0
- package/dist/core/permissions/constrained-edit.js +91 -0
- package/dist/core/permissions/gate.js +278 -0
- package/dist/core/permissions/index.js +20 -0
- package/dist/core/permissions/mode.js +174 -0
- package/dist/core/permissions/network-egress.js +137 -0
- package/dist/core/permissions/state.js +241 -0
- package/dist/core/permissions/tool-class.js +107 -0
- package/dist/core/plan-mode/ui-state.js +51 -0
- package/dist/core/plans/plan-artifact.js +721 -0
- package/dist/core/policy-limits/etag-store.js +122 -0
- package/dist/core/prd-check/parser.js +215 -0
- package/dist/core/prd-check/reporter.js +127 -0
- package/dist/core/prd-check/session-review.js +557 -0
- package/dist/core/prd-check/verifiers.js +223 -0
- package/dist/core/prompt-cache/client-cache.js +99 -0
- package/dist/core/prompts/assembly.js +29 -0
- package/dist/core/prompts/registry.js +364 -0
- package/dist/core/pugi-gitignore.js +52 -0
- package/dist/core/pugi-md/cc-compat-rules.js +735 -0
- package/dist/core/pugi-md/context-injector.js +76 -0
- package/dist/core/pugi-md/walk-up.js +207 -0
- package/dist/core/python/uv-installer.js +270 -0
- package/dist/core/python/uv-resolver.js +83 -0
- package/dist/core/rate-limit/narrator.js +146 -0
- package/dist/core/recipes/cli-types.js +20 -0
- package/dist/core/recipes/loader.js +103 -0
- package/dist/core/recipes/runner.js +345 -0
- package/dist/core/recipes/schema.js +587 -0
- package/dist/core/release-notes/parser.js +241 -0
- package/dist/core/release-notes/state.js +116 -0
- package/dist/core/repl/ask.js +37 -37
- package/dist/core/repl/cancellation.js +26 -26
- package/dist/core/repl/cap-warning.js +4 -4
- package/dist/core/repl/clipboard-read.js +11 -11
- package/dist/core/repl/dispatch-fsm.js +12 -12
- package/dist/core/repl/engine-bridge.js +303 -0
- package/dist/core/repl/history-search.js +15 -15
- package/dist/core/repl/history.js +28 -18
- package/dist/core/repl/kill-ring.js +5 -5
- package/dist/core/repl/model-pricing.js +135 -0
- package/dist/core/repl/privacy-banner.js +22 -22
- package/dist/core/repl/session.js +2714 -228
- package/dist/core/repl/slash-commands.js +572 -40
- package/dist/core/repl/store/index.js +1 -1
- package/dist/core/repl/store/jsonl-log.js +22 -22
- package/dist/core/repl/store/lockfile.js +10 -10
- package/dist/core/repl/store/session-store.js +136 -107
- package/dist/core/repl/store/types.js +15 -15
- package/dist/core/repl/store/uuid-v7.js +12 -12
- package/dist/core/repl/tool-route.js +382 -0
- package/dist/core/repl/workspace-context.js +43 -21
- package/dist/core/repo-map/build.js +125 -0
- package/dist/core/repo-map/cache.js +185 -0
- package/dist/core/repo-map/extractor.js +254 -0
- package/dist/core/repo-map/formatter.js +145 -0
- package/dist/core/repo-map/page-rank.js +105 -0
- package/dist/core/repo-map/scanner.js +211 -0
- package/dist/core/retro/git-collector.js +251 -0
- package/dist/core/retro/health-card.js +25 -0
- package/dist/core/retro/metrics.js +342 -0
- package/dist/core/retro/narrative.js +249 -0
- package/dist/core/retro/plane-collector.js +274 -0
- package/dist/core/retro/pr-issue-link.js +65 -0
- package/dist/core/retro/types.js +16 -0
- package/dist/core/retry-budget/budget.js +284 -0
- package/dist/core/retry-budget/index.js +5 -0
- package/dist/core/retry-budget/retry-cap.js +74 -0
- package/dist/core/routing/lead-worker.js +43 -0
- package/dist/core/routing/pre-flight-estimator.js +108 -0
- package/dist/core/runs/run-tree.js +103 -0
- package/dist/core/sandboxing/adapter.js +29 -0
- package/dist/core/sandboxing/index.js +49 -0
- package/dist/core/sandboxing/none.js +19 -0
- package/dist/core/sandboxing/seatbelt.js +183 -0
- package/dist/core/security/injection-scanner.js +367 -0
- package/dist/core/security/output-filter.js +418 -0
- package/dist/core/session/env-file.js +105 -0
- package/dist/core/session/section-budgets.js +140 -0
- package/dist/core/session.js +119 -0
- package/dist/core/settings.js +378 -5
- package/dist/core/share/formatter.js +271 -0
- package/dist/core/share/redactor.js +221 -0
- package/dist/core/share/uploader.js +267 -0
- package/dist/core/skills/defaults.js +457 -0
- package/dist/core/skills/loader.js +22 -22
- package/dist/core/skills/sources.js +27 -27
- package/dist/core/smoke/headless-driver.js +174 -0
- package/dist/core/smoke/orchestrator.js +194 -0
- package/dist/core/smoke/runner.js +238 -0
- package/dist/core/smoke/scenario-parser.js +316 -0
- package/dist/core/statusline.js +99 -0
- package/dist/core/subagents/dispatcher-real.js +600 -0
- package/dist/core/subagents/dispatcher.js +146 -52
- package/dist/core/subagents/index.js +19 -6
- package/dist/core/subagents/isolation-matrix.js +213 -0
- package/dist/core/subagents/spawn.js +19 -4
- package/dist/core/telemetry/emitter.js +229 -0
- package/dist/core/telemetry/queue.js +251 -0
- package/dist/core/theme/context.js +91 -0
- package/dist/core/theme/presets.js +228 -0
- package/dist/core/theme/state.js +181 -0
- package/dist/core/todos/invariant.js +10 -0
- package/dist/core/todos/state.js +177 -0
- package/dist/core/tool-schema/compressor.js +89 -0
- package/dist/core/transport/version-interceptor.js +166 -0
- package/dist/core/trust.js +2 -2
- package/dist/core/tui/thinking-block.js +64 -0
- package/dist/core/vim/keymap.js +288 -0
- package/dist/core/vim/state.js +92 -0
- package/dist/core/watch-markers/marker-watcher.js +133 -0
- package/dist/core/worktree/include-parser.js +249 -0
- package/dist/core/worktree-manager/cleanup.js +123 -0
- package/dist/core/worktree-manager/manager.js +303 -0
- package/dist/index.js +36 -0
- package/dist/runtime/bootstrap.js +190 -0
- package/dist/runtime/cli.js +4536 -477
- package/dist/runtime/commands/agents.js +31 -31
- package/dist/runtime/commands/budget.js +5 -5
- package/dist/runtime/commands/cancel.js +231 -0
- package/dist/runtime/commands/chain.js +489 -0
- package/dist/runtime/commands/codegraph-status.js +227 -0
- package/dist/runtime/commands/compact.js +297 -0
- package/dist/runtime/commands/config.js +74 -40
- package/dist/runtime/commands/cost.js +199 -0
- package/dist/runtime/commands/delegate.js +312 -0
- package/dist/runtime/commands/dispatch.js +126 -0
- package/dist/runtime/commands/doctor.js +579 -0
- package/dist/runtime/commands/feedback.js +184 -0
- package/dist/runtime/commands/hooks.js +187 -0
- package/dist/runtime/commands/index-cmd.js +353 -0
- package/dist/runtime/commands/init.js +254 -0
- package/dist/runtime/commands/lsp.js +368 -0
- package/dist/runtime/commands/mcp.js +935 -0
- package/dist/runtime/commands/memory.js +582 -0
- package/dist/runtime/commands/model.js +237 -0
- package/dist/runtime/commands/onboarding.js +275 -0
- package/dist/runtime/commands/patch.js +128 -0
- package/dist/runtime/commands/permissions.js +112 -0
- package/dist/runtime/commands/plan.js +143 -0
- package/dist/runtime/commands/prd-check.js +285 -0
- package/dist/runtime/commands/privacy.js +17 -17
- package/dist/runtime/commands/recipe.js +325 -0
- package/dist/runtime/commands/redo-blob-store.js +92 -0
- package/dist/runtime/commands/redo.js +361 -0
- package/dist/runtime/commands/release-notes.js +229 -0
- package/dist/runtime/commands/repo-map.js +95 -0
- package/dist/runtime/commands/report.js +299 -0
- package/dist/runtime/commands/resume.js +118 -0
- package/dist/runtime/commands/review-consensus.js +68 -53
- package/dist/runtime/commands/rewind.js +333 -0
- package/dist/runtime/commands/roster.js +117 -0
- package/dist/runtime/commands/servers.js +236 -0
- package/dist/runtime/commands/sessions.js +163 -0
- package/dist/runtime/commands/share.js +316 -0
- package/dist/runtime/commands/skills.js +31 -31
- package/dist/runtime/commands/status.js +186 -0
- package/dist/runtime/commands/stickers.js +82 -0
- package/dist/runtime/commands/style.js +194 -0
- package/dist/runtime/commands/theme.js +196 -0
- package/dist/runtime/commands/undo.js +54 -22
- package/dist/runtime/commands/update.js +289 -0
- package/dist/runtime/commands/vim.js +140 -0
- package/dist/runtime/commands/worktree.js +177 -0
- package/dist/runtime/commands/worktrees.js +155 -0
- package/dist/runtime/deprecation-warning.js +69 -0
- package/dist/runtime/engine-exit-code.js +50 -0
- package/dist/runtime/headless-repl.js +195 -0
- package/dist/runtime/headless.js +548 -0
- package/dist/runtime/load-hooks-or-exit.js +71 -0
- package/dist/runtime/plan-decompose.js +531 -0
- package/dist/runtime/sigint-guard.js +272 -0
- package/dist/runtime/stream-renderer.js +195 -0
- package/dist/runtime/update-check.js +28 -28
- package/dist/runtime/version.js +65 -0
- package/dist/runtime/worktree-bootstrap.js +579 -0
- package/dist/skills/bundled/batch.js +617 -0
- package/dist/skills/bundled/index.js +45 -0
- package/dist/skills/bundled/loop.js +358 -0
- package/dist/skills/bundled/remember.js +383 -0
- package/dist/skills/bundled/simplify.js +289 -0
- package/dist/skills/bundled/skillify.js +373 -0
- package/dist/skills/bundled/stuck.js +558 -0
- package/dist/skills/bundled/verify.js +439 -0
- package/dist/testing/vcr.js +486 -0
- package/dist/tools/agent-tool.js +229 -0
- package/dist/tools/apply-patch.js +556 -0
- package/dist/tools/ask-user-question.js +337 -0
- package/dist/tools/ask-user.js +115 -0
- package/dist/tools/bash.js +624 -46
- package/dist/tools/brief.js +224 -0
- package/dist/tools/cron.js +433 -0
- package/dist/tools/enter-worktree.js +250 -0
- package/dist/tools/exit-worktree.js +147 -0
- package/dist/tools/file-tools.js +161 -44
- package/dist/tools/http-request.js +336 -0
- package/dist/tools/lsp-tools.js +565 -0
- package/dist/tools/mcp-tool.js +260 -0
- package/dist/tools/multi-edit.js +361 -0
- package/dist/tools/powershell.js +268 -0
- package/dist/tools/registry.js +142 -1
- package/dist/tools/server-tools.js +892 -0
- package/dist/tools/skill-tool.js +96 -0
- package/dist/tools/sleep.js +99 -0
- package/dist/tools/synthetic-output.js +133 -0
- package/dist/tools/tasks.js +208 -0
- package/dist/tools/todo-write.js +184 -0
- package/dist/tools/verify-plan-execution.js +295 -0
- package/dist/tools/web-fetch-injection-scanner.js +207 -0
- package/dist/tools/web-fetch.js +195 -10
- package/dist/tools/web-search.js +458 -0
- package/dist/tui/agent-progress-card.js +111 -0
- package/dist/tui/agent-tree.js +22 -1
- package/dist/tui/ask-modal.js +14 -14
- package/dist/tui/ask-user-question-chips.js +315 -0
- package/dist/tui/ask-user-question-prompt.js +203 -0
- package/dist/tui/compact-banner.js +81 -0
- package/dist/tui/conversation-pane.js +85 -11
- package/dist/tui/cost-table.js +111 -0
- package/dist/tui/device-flow.js +2 -2
- package/dist/tui/doctor-table.js +46 -0
- package/dist/tui/feedback-prompt.js +156 -0
- package/dist/tui/input-box.js +247 -32
- package/dist/tui/login-picker.js +3 -3
- package/dist/tui/markdown-render.js +6 -6
- package/dist/tui/multi-file-diff-approval.js +375 -0
- package/dist/tui/onboarding-wizard.js +240 -0
- package/dist/tui/permissions-picker.js +86 -0
- package/dist/tui/render.js +36 -1
- package/dist/tui/repl-render.js +405 -32
- package/dist/tui/repl-splash-art.js +16 -16
- package/dist/tui/repl-splash-mascot.js +48 -24
- package/dist/tui/repl-splash.js +22 -22
- package/dist/tui/repl.js +136 -43
- package/dist/tui/slash-palette.js +6 -6
- package/dist/tui/splash.js +2 -2
- package/dist/tui/status-bar.js +109 -31
- package/dist/tui/status-table.js +7 -0
- package/dist/tui/stickers-art.js +136 -0
- package/dist/tui/style-table.js +28 -0
- package/dist/tui/theme-table.js +29 -0
- package/dist/tui/thinking-spinner.js +123 -0
- package/dist/tui/tool-stream-pane.js +53 -4
- package/dist/tui/update-banner.js +27 -2
- package/dist/tui/vim-input.js +267 -0
- package/dist/tui/welcome-banner.js +107 -0
- package/dist/tui/welcome-data.js +293 -0
- package/dist/tui/workspace-context.js +2 -2
- package/docs/examples/codegraph.mcp.json +10 -0
- package/package.json +25 -7
- package/test/scenarios/codegen-create-file.scenario.txt +13 -0
- package/test/scenarios/compact-force.scenario.txt +12 -0
- package/test/scenarios/identity.scenario.txt +11 -0
- package/test/scenarios/persona-handoff.scenario.txt +12 -0
- package/test/scenarios/walkback.scenario.txt +12 -0
- package/dist/core/engine/compaction-hook.js +0 -154
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
/**
|
|
2
|
-
* chokidar-based filewatch -
|
|
2
|
+
* chokidar-based filewatch - Phase 1 (three-tier context).
|
|
3
3
|
*
|
|
4
4
|
* The REPL needs a live signal when files change so the operator's
|
|
5
5
|
* agent can re-read a stale file, and so the status bar can surface a
|
|
@@ -8,35 +8,35 @@
|
|
|
8
8
|
*
|
|
9
9
|
* Design choices:
|
|
10
10
|
*
|
|
11
|
-
*
|
|
12
|
-
*
|
|
13
|
-
*
|
|
14
|
-
*
|
|
15
|
-
*
|
|
16
|
-
*
|
|
11
|
+
* 1. **Ignore-aware**: every watched path is filtered through the
|
|
12
|
+
* same `PugiIgnore` matcher the skeleton walker uses. We pass the
|
|
13
|
+
* matcher into chokidar's `ignored` option so `node_modules/`,
|
|
14
|
+
* `dist/`, `.git/`, and secret files are never watched at all -
|
|
15
|
+
* this matters because chokidar's resource bound is the number
|
|
16
|
+
* of watched paths, not the number of total files in the repo.
|
|
17
17
|
*
|
|
18
|
-
*
|
|
19
|
-
*
|
|
20
|
-
*
|
|
21
|
-
*
|
|
18
|
+
* 2. **Throttle**: chokidar can fire dozens of `change` events when
|
|
19
|
+
* a code formatter rewrites a directory. We batch events in a
|
|
20
|
+
* `THROTTLE_WINDOW_MS` window and emit a single aggregated event
|
|
21
|
+
* so the REPL system line / status bar do not flicker.
|
|
22
22
|
*
|
|
23
|
-
*
|
|
24
|
-
*
|
|
25
|
-
*
|
|
26
|
-
*
|
|
27
|
-
*
|
|
28
|
-
*
|
|
23
|
+
* 3. **Watch cap**: per spec, the watcher caps watched paths at
|
|
24
|
+
* `MAX_WATCHED_PATHS`. Once the count crosses the cap we close
|
|
25
|
+
* the watcher and fall back to "no live updates this session" -
|
|
26
|
+
* better to silently lose the badge than to consume thousands of
|
|
27
|
+
* file descriptors. The fallback path emits one warning event so
|
|
28
|
+
* the operator knows live updates are off.
|
|
29
29
|
*
|
|
30
|
-
*
|
|
31
|
-
*
|
|
32
|
-
*
|
|
33
|
-
*
|
|
34
|
-
*
|
|
30
|
+
* 4. **Lifecycle**: `start()` is async (chokidar's `ready` event)
|
|
31
|
+
* so the caller can `await` the initial scan. `close()` is also
|
|
32
|
+
* async (chokidar's close is async). The watcher is reusable
|
|
33
|
+
* across multiple `start` / `close` pairs, but Phase 1 expects
|
|
34
|
+
* one-watcher-per-REPL-session.
|
|
35
35
|
*
|
|
36
|
-
*
|
|
37
|
-
*
|
|
38
|
-
*
|
|
39
|
-
*
|
|
36
|
+
* 5. **Testability**: the chokidar handle is injectable via the
|
|
37
|
+
* `watchFactory` option. Production wires `chokidar.watch`;
|
|
38
|
+
* tests pass a fake emitter that lets the spec drive `add` /
|
|
39
|
+
* `change` / `unlink` events synchronously.
|
|
40
40
|
*/
|
|
41
41
|
import { EventEmitter } from 'node:events';
|
|
42
42
|
import chokidar from 'chokidar';
|
|
@@ -50,7 +50,7 @@ export const THROTTLE_WINDOW_MS = 250;
|
|
|
50
50
|
* giant subtree post-ready can blow past the cap without ever
|
|
51
51
|
* re-evaluating. Sample every N adds so the cap fires within a
|
|
52
52
|
* bounded window even on post-ready growth. Cheap: one Map
|
|
53
|
-
* iteration per N adds. triple-review P2 (PR
|
|
53
|
+
* iteration per N adds. triple-review P2 (PR).
|
|
54
54
|
*/
|
|
55
55
|
export const CAP_CHECK_ADD_INTERVAL = 1_000;
|
|
56
56
|
/**
|
|
@@ -76,7 +76,7 @@ export class PugiWatcher extends EventEmitter {
|
|
|
76
76
|
/**
|
|
77
77
|
* Counter of `add` events received since the last cap-check sample.
|
|
78
78
|
* When it crosses CAP_CHECK_ADD_INTERVAL we schedule another check.
|
|
79
|
-
* Resets on every cap-check run. triple-review P2 (PR
|
|
79
|
+
* Resets on every cap-check run. triple-review P2 (PR).
|
|
80
80
|
*/
|
|
81
81
|
addsSinceLastCapCheck = 0;
|
|
82
82
|
constructor(options) {
|
|
@@ -97,7 +97,7 @@ export class PugiWatcher extends EventEmitter {
|
|
|
97
97
|
// the event so emit() does not crash the CLI; any real consumer
|
|
98
98
|
// that subscribes later receives the events as normal because
|
|
99
99
|
// EventEmitter delivers to every registered listener, not just one.
|
|
100
|
-
// Codex P1 (PR
|
|
100
|
+
// Codex P1 (PR).
|
|
101
101
|
this.on('error', () => {
|
|
102
102
|
/* defensive no-op so emit('error') never throws */
|
|
103
103
|
});
|
|
@@ -162,7 +162,7 @@ export class PugiWatcher extends EventEmitter {
|
|
|
162
162
|
// post-ready burst (e.g. `git checkout` of a giant subtree)
|
|
163
163
|
// does not silently blow past the watched-paths cap. The
|
|
164
164
|
// ready-time check is preserved; this is the missing follow-up
|
|
165
|
-
// sampler the comment promised. triple-review P2 (PR
|
|
165
|
+
// sampler the comment promised. triple-review P2 (PR).
|
|
166
166
|
this.addsSinceLastCapCheck += 1;
|
|
167
167
|
if (this.addsSinceLastCapCheck >= CAP_CHECK_ADD_INTERVAL) {
|
|
168
168
|
this.addsSinceLastCapCheck = 0;
|
|
@@ -225,7 +225,7 @@ export class PugiWatcher extends EventEmitter {
|
|
|
225
225
|
return this.buffer.length;
|
|
226
226
|
}
|
|
227
227
|
/* ------------------------------------------------------------ */
|
|
228
|
-
/* Internals
|
|
228
|
+
/* Internals */
|
|
229
229
|
/* ------------------------------------------------------------ */
|
|
230
230
|
queue(kind, path) {
|
|
231
231
|
if (this.closed)
|
|
@@ -291,7 +291,7 @@ export class PugiWatcher extends EventEmitter {
|
|
|
291
291
|
}
|
|
292
292
|
}
|
|
293
293
|
/* ------------------------------------------------------------------ */
|
|
294
|
-
/* Helpers
|
|
294
|
+
/* Helpers */
|
|
295
295
|
/* ------------------------------------------------------------------ */
|
|
296
296
|
/**
|
|
297
297
|
* Convert an absolute path to a repo-relative POSIX path. chokidar
|
|
@@ -1,41 +1,41 @@
|
|
|
1
1
|
/**
|
|
2
|
-
* Tier 1 working-set tracker -
|
|
2
|
+
* Tier 1 working-set tracker - Phase 1 (three-tier context).
|
|
3
3
|
*
|
|
4
4
|
* The model context window is finite. Tier 0 (repo skeleton, ~5KB) is
|
|
5
|
-
* always loaded; Tier 2 (RAG, Anvil-side, deferred to
|
|
5
|
+
* always loaded; Tier 2 (RAG, Anvil-side, deferred to ) answers
|
|
6
6
|
* queries on demand. Tier 1 - the working set - lives between them: the
|
|
7
7
|
* subset of files the agent has touched THIS session, bounded so a
|
|
8
8
|
* long REPL session does not unbounded-grow the system prompt.
|
|
9
9
|
*
|
|
10
10
|
* Design choices:
|
|
11
11
|
*
|
|
12
|
-
*
|
|
13
|
-
*
|
|
14
|
-
*
|
|
15
|
-
*
|
|
12
|
+
* 1. **LRU semantics**: the most-recently-touched file stays in the
|
|
13
|
+
* window; the oldest is evicted when the cap is hit. We use a
|
|
14
|
+
* `Map` so insertion order doubles as recency order (delete +
|
|
15
|
+
* re-set bubbles the entry to the tail).
|
|
16
16
|
*
|
|
17
|
-
*
|
|
18
|
-
*
|
|
19
|
-
*
|
|
20
|
-
*
|
|
21
|
-
*
|
|
22
|
-
*
|
|
17
|
+
* 2. **Source tracking**: every `track()` call records the source -
|
|
18
|
+
* `read` (agent fetched the file), `edit` (agent modified it),
|
|
19
|
+
* `write` (agent created it). The source seeds future heuristics
|
|
20
|
+
* (e.g. always pin edited files near the cap, never evict files
|
|
21
|
+
* written this session) but Phase 1 keeps the policy simple:
|
|
22
|
+
* pure LRU.
|
|
23
23
|
*
|
|
24
|
-
*
|
|
25
|
-
*
|
|
26
|
-
*
|
|
27
|
-
*
|
|
24
|
+
* 3. **No persistence**: working set is in-memory only. A REPL
|
|
25
|
+
* restart rebuilds from scratch (and from `/resume` SessionStore
|
|
26
|
+
* replay if/when we wire that). Per spec: "node:sqlite NOT
|
|
27
|
+
* needed for this sprint."
|
|
28
28
|
*
|
|
29
|
-
*
|
|
30
|
-
*
|
|
31
|
-
*
|
|
29
|
+
* 4. **Path normalisation**: we resolve every path to an absolute
|
|
30
|
+
* form on insert so two `track()` calls with `./foo.ts` and
|
|
31
|
+
* `/abs/cwd/foo.ts` collapse to one entry.
|
|
32
32
|
*
|
|
33
|
-
*
|
|
34
|
-
*
|
|
35
|
-
*
|
|
33
|
+
* 5. **Defensive cap**: `MAX_WORKING_SET = 50` per spec. We hold the
|
|
34
|
+
* cap as a configurable option so tests can dial it down to
|
|
35
|
+
* exercise eviction without 51 fixture files.
|
|
36
36
|
*/
|
|
37
37
|
import { resolve } from 'node:path';
|
|
38
|
-
/** Spec-mandated default cap. Mirrors the doc string in the
|
|
38
|
+
/** Spec-mandated default cap. Mirrors the doc string in the sprint plan. */
|
|
39
39
|
export const DEFAULT_WORKING_SET_CAPACITY = 50;
|
|
40
40
|
export class WorkingSet {
|
|
41
41
|
entries = new Map();
|
|
@@ -0,0 +1,77 @@
|
|
|
1
|
+
function snapshot(task) {
|
|
2
|
+
return { ...task };
|
|
3
|
+
}
|
|
4
|
+
function isTerminal(status) {
|
|
5
|
+
return status === 'completed' || status === 'cancelled' || status === 'failed';
|
|
6
|
+
}
|
|
7
|
+
export function createAgentCoordinator(options) {
|
|
8
|
+
const tasks = new Map();
|
|
9
|
+
return {
|
|
10
|
+
async createTask(agentType, prompt) {
|
|
11
|
+
const { taskId } = await options.spawn(agentType, prompt);
|
|
12
|
+
const task = {
|
|
13
|
+
id: taskId,
|
|
14
|
+
agentType,
|
|
15
|
+
prompt,
|
|
16
|
+
status: 'queued',
|
|
17
|
+
createdAt: Date.now(),
|
|
18
|
+
};
|
|
19
|
+
tasks.set(task.id, task);
|
|
20
|
+
task.status = 'running';
|
|
21
|
+
task.startedAt = Date.now();
|
|
22
|
+
return snapshot(task);
|
|
23
|
+
},
|
|
24
|
+
async sendMessage(taskId, message) {
|
|
25
|
+
const task = tasks.get(taskId);
|
|
26
|
+
if (!task) {
|
|
27
|
+
throw new Error(`agent task ${taskId} not found`);
|
|
28
|
+
}
|
|
29
|
+
if (task.status !== 'running') {
|
|
30
|
+
throw new Error(`agent task ${taskId} is not running (status=${task.status})`);
|
|
31
|
+
}
|
|
32
|
+
await options.send(taskId, message);
|
|
33
|
+
},
|
|
34
|
+
async stopTask(taskId) {
|
|
35
|
+
const task = tasks.get(taskId);
|
|
36
|
+
if (!task) {
|
|
37
|
+
throw new Error(`agent task ${taskId} not found`);
|
|
38
|
+
}
|
|
39
|
+
if (isTerminal(task.status)) {
|
|
40
|
+
return;
|
|
41
|
+
}
|
|
42
|
+
await options.stop(taskId);
|
|
43
|
+
task.status = 'cancelled';
|
|
44
|
+
task.completedAt = Date.now();
|
|
45
|
+
},
|
|
46
|
+
listTasks() {
|
|
47
|
+
return Array.from(tasks.values())
|
|
48
|
+
.sort((a, b) => b.createdAt - a.createdAt)
|
|
49
|
+
.map(snapshot);
|
|
50
|
+
},
|
|
51
|
+
getTask(taskId) {
|
|
52
|
+
const task = tasks.get(taskId);
|
|
53
|
+
return task ? snapshot(task) : null;
|
|
54
|
+
},
|
|
55
|
+
markCompleted(taskId, output) {
|
|
56
|
+
const task = tasks.get(taskId);
|
|
57
|
+
if (!task) {
|
|
58
|
+
return;
|
|
59
|
+
}
|
|
60
|
+
task.status = 'completed';
|
|
61
|
+
task.completedAt = Date.now();
|
|
62
|
+
task.output = output;
|
|
63
|
+
delete task.error;
|
|
64
|
+
},
|
|
65
|
+
markFailed(taskId, error) {
|
|
66
|
+
const task = tasks.get(taskId);
|
|
67
|
+
if (!task) {
|
|
68
|
+
return;
|
|
69
|
+
}
|
|
70
|
+
task.status = 'failed';
|
|
71
|
+
task.completedAt = Date.now();
|
|
72
|
+
task.error = error;
|
|
73
|
+
delete task.output;
|
|
74
|
+
},
|
|
75
|
+
};
|
|
76
|
+
}
|
|
77
|
+
//# sourceMappingURL=agent-tools.js.map
|
|
@@ -0,0 +1,65 @@
|
|
|
1
|
+
// apps/pugi-cli/src/core/coordinator/agent-toolset.ts
|
|
2
|
+
import { randomUUID } from 'node:crypto';
|
|
3
|
+
export class AgentCoordinator {
|
|
4
|
+
agents = new Map();
|
|
5
|
+
spawnAgent(subagentType, prompt) {
|
|
6
|
+
const handle = {
|
|
7
|
+
id: randomUUID(),
|
|
8
|
+
subagentType,
|
|
9
|
+
prompt,
|
|
10
|
+
status: 'spawned',
|
|
11
|
+
spawnedAt: new Date().toISOString(),
|
|
12
|
+
};
|
|
13
|
+
this.agents.set(handle.id, handle);
|
|
14
|
+
return handle;
|
|
15
|
+
}
|
|
16
|
+
sendMessage(agentId, _message) {
|
|
17
|
+
const agent = this.agents.get(agentId);
|
|
18
|
+
if (!agent) {
|
|
19
|
+
return { ok: false, reason: `agent ${agentId} not found` };
|
|
20
|
+
}
|
|
21
|
+
if (agent.status !== 'spawned' && agent.status !== 'running') {
|
|
22
|
+
return { ok: false, reason: `agent ${agentId} not accepting messages (status=${agent.status})` };
|
|
23
|
+
}
|
|
24
|
+
return { ok: true };
|
|
25
|
+
}
|
|
26
|
+
taskStop(agentId) {
|
|
27
|
+
const agent = this.agents.get(agentId);
|
|
28
|
+
if (!agent) {
|
|
29
|
+
return { ok: false, reason: `agent ${agentId} not found` };
|
|
30
|
+
}
|
|
31
|
+
agent.status = 'stopped';
|
|
32
|
+
agent.completedAt = new Date().toISOString();
|
|
33
|
+
return { ok: true };
|
|
34
|
+
}
|
|
35
|
+
markCompleted(agentId, result) {
|
|
36
|
+
const agent = this.agents.get(agentId);
|
|
37
|
+
if (!agent) {
|
|
38
|
+
return;
|
|
39
|
+
}
|
|
40
|
+
agent.status = 'completed';
|
|
41
|
+
agent.completedAt = new Date().toISOString();
|
|
42
|
+
agent.result = result;
|
|
43
|
+
}
|
|
44
|
+
markFailed(agentId, reason) {
|
|
45
|
+
const agent = this.agents.get(agentId);
|
|
46
|
+
if (!agent) {
|
|
47
|
+
return;
|
|
48
|
+
}
|
|
49
|
+
agent.status = 'failed';
|
|
50
|
+
agent.completedAt = new Date().toISOString();
|
|
51
|
+
agent.result = reason;
|
|
52
|
+
}
|
|
53
|
+
listAgents(filter) {
|
|
54
|
+
const all = Array.from(this.agents.values());
|
|
55
|
+
if (!filter?.status) {
|
|
56
|
+
return all;
|
|
57
|
+
}
|
|
58
|
+
const wanted = filter.status;
|
|
59
|
+
return all.filter((a) => a.status === wanted);
|
|
60
|
+
}
|
|
61
|
+
getAgent(id) {
|
|
62
|
+
return this.agents.get(id);
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
//# sourceMappingURL=agent-toolset.js.map
|
|
@@ -0,0 +1,73 @@
|
|
|
1
|
+
export class CoordinatorFSM {
|
|
2
|
+
state;
|
|
3
|
+
constructor() {
|
|
4
|
+
this.state = {
|
|
5
|
+
phase: 'research',
|
|
6
|
+
inflightOps: [],
|
|
7
|
+
completedPhases: [],
|
|
8
|
+
};
|
|
9
|
+
}
|
|
10
|
+
getState() {
|
|
11
|
+
// Return a shallow copy to prevent direct mutation of the internal state.
|
|
12
|
+
return {
|
|
13
|
+
...this.state,
|
|
14
|
+
inflightOps: [...this.state.inflightOps],
|
|
15
|
+
completedPhases: [...this.state.completedPhases],
|
|
16
|
+
};
|
|
17
|
+
}
|
|
18
|
+
canStart(op) {
|
|
19
|
+
switch (this.state.phase) {
|
|
20
|
+
case 'research':
|
|
21
|
+
return op === 'read';
|
|
22
|
+
case 'synth':
|
|
23
|
+
return op === 'read';
|
|
24
|
+
case 'impl':
|
|
25
|
+
if (op === 'write') {
|
|
26
|
+
// 'write' is only allowed when no other operations are in flight.
|
|
27
|
+
return this.state.inflightOps.length === 0;
|
|
28
|
+
}
|
|
29
|
+
// 'read' and 'exec' are allowed as long as no 'write' is in flight.
|
|
30
|
+
return !this.state.inflightOps.includes('write');
|
|
31
|
+
case 'done':
|
|
32
|
+
return false;
|
|
33
|
+
default:
|
|
34
|
+
return false;
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
start(op) {
|
|
38
|
+
if (!this.canStart(op)) {
|
|
39
|
+
throw new Error(`Operation '${op}' cannot be started in phase '${this.state.phase}' with inflight ops: [${this.state.inflightOps.join(', ')}]`);
|
|
40
|
+
}
|
|
41
|
+
this.state.inflightOps.push(op);
|
|
42
|
+
}
|
|
43
|
+
finish(op) {
|
|
44
|
+
const opIndex = this.state.inflightOps.indexOf(op);
|
|
45
|
+
if (opIndex === -1) {
|
|
46
|
+
throw new Error(`Cannot finish operation '${op}' as it is not in flight. Inflight ops: [${this.state.inflightOps.join(', ')}]`);
|
|
47
|
+
}
|
|
48
|
+
this.state.inflightOps.splice(opIndex, 1);
|
|
49
|
+
}
|
|
50
|
+
advance() {
|
|
51
|
+
if (this.state.inflightOps.length > 0) {
|
|
52
|
+
throw new Error(`Cannot advance phase with operations in flight: [${this.state.inflightOps.join(', ')}]`);
|
|
53
|
+
}
|
|
54
|
+
const currentPhase = this.state.phase;
|
|
55
|
+
let nextPhase;
|
|
56
|
+
switch (currentPhase) {
|
|
57
|
+
case 'research':
|
|
58
|
+
nextPhase = 'synth';
|
|
59
|
+
break;
|
|
60
|
+
case 'synth':
|
|
61
|
+
nextPhase = 'impl';
|
|
62
|
+
break;
|
|
63
|
+
case 'impl':
|
|
64
|
+
nextPhase = 'done';
|
|
65
|
+
break;
|
|
66
|
+
case 'done':
|
|
67
|
+
throw new Error("Cannot advance from final phase 'done'.");
|
|
68
|
+
}
|
|
69
|
+
this.state.completedPhases.push(currentPhase);
|
|
70
|
+
this.state.phase = nextPhase;
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
//# sourceMappingURL=fsm.js.map
|
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
const isQuiescent = (state) => state.readTasks === 0 && !state.writeTaskActive;
|
|
2
|
+
export const MODE_TRANSITIONS = [
|
|
3
|
+
{ from: 'idle', to: 'research', trigger: 'plan', guard: isQuiescent },
|
|
4
|
+
{ from: 'research', to: 'synth', trigger: 'synthesize', guard: isQuiescent },
|
|
5
|
+
{ from: 'synth', to: 'impl', trigger: 'execute', guard: isQuiescent },
|
|
6
|
+
{ from: 'impl', to: 'review', trigger: 'verify', guard: isQuiescent },
|
|
7
|
+
{ from: 'review', to: 'done', trigger: 'accept', guard: isQuiescent },
|
|
8
|
+
{ from: 'review', to: 'impl', trigger: 'fix', guard: isQuiescent },
|
|
9
|
+
{ from: 'idle', to: 'idle', trigger: 'reset' },
|
|
10
|
+
{ from: 'research', to: 'idle', trigger: 'reset' },
|
|
11
|
+
{ from: 'synth', to: 'idle', trigger: 'reset' },
|
|
12
|
+
{ from: 'impl', to: 'idle', trigger: 'reset' },
|
|
13
|
+
{ from: 'review', to: 'idle', trigger: 'reset' },
|
|
14
|
+
{ from: 'done', to: 'idle', trigger: 'reset' },
|
|
15
|
+
];
|
|
16
|
+
class CoordinatorModeFSM {
|
|
17
|
+
state;
|
|
18
|
+
constructor(initial = 'idle') {
|
|
19
|
+
this.state = {
|
|
20
|
+
mode: initial,
|
|
21
|
+
enteredAt: Date.now(),
|
|
22
|
+
readTasks: 0,
|
|
23
|
+
writeTaskActive: false,
|
|
24
|
+
};
|
|
25
|
+
}
|
|
26
|
+
current() {
|
|
27
|
+
return { ...this.state };
|
|
28
|
+
}
|
|
29
|
+
trigger(event) {
|
|
30
|
+
const transition = MODE_TRANSITIONS.find((candidate) => candidate.from === this.state.mode && candidate.trigger === event);
|
|
31
|
+
if (!transition) {
|
|
32
|
+
return {
|
|
33
|
+
ok: false,
|
|
34
|
+
reason: `No transition for event '${event}' from mode '${this.state.mode}'.`,
|
|
35
|
+
};
|
|
36
|
+
}
|
|
37
|
+
if (transition.guard && !transition.guard(this.current())) {
|
|
38
|
+
return {
|
|
39
|
+
ok: false,
|
|
40
|
+
reason: `Guard blocked event '${event}' from mode '${this.state.mode}'.`,
|
|
41
|
+
};
|
|
42
|
+
}
|
|
43
|
+
this.state = {
|
|
44
|
+
...this.state,
|
|
45
|
+
mode: transition.to,
|
|
46
|
+
enteredAt: Date.now(),
|
|
47
|
+
};
|
|
48
|
+
return { ok: true };
|
|
49
|
+
}
|
|
50
|
+
beginRead() {
|
|
51
|
+
this.state.readTasks += 1;
|
|
52
|
+
}
|
|
53
|
+
endRead() {
|
|
54
|
+
this.state.readTasks = Math.max(0, this.state.readTasks - 1);
|
|
55
|
+
}
|
|
56
|
+
beginWrite() {
|
|
57
|
+
if (this.state.writeTaskActive) {
|
|
58
|
+
return { ok: false, reason: 'A write task is already active.' };
|
|
59
|
+
}
|
|
60
|
+
this.state.writeTaskActive = true;
|
|
61
|
+
return { ok: true };
|
|
62
|
+
}
|
|
63
|
+
endWrite() {
|
|
64
|
+
this.state.writeTaskActive = false;
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
export function createFSM(initial) {
|
|
68
|
+
return new CoordinatorModeFSM(initial);
|
|
69
|
+
}
|
|
70
|
+
//# sourceMappingURL=mode-fsm.js.map
|
|
@@ -0,0 +1,129 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Rate card for the `pugi cost` / `/cost` / `/usage` surface — L19 sprint.
|
|
3
|
+
*
|
|
4
|
+
* Distinct from `core/repl/model-pricing.ts` on purpose:
|
|
5
|
+
*
|
|
6
|
+
* - `model-pricing.ts` powers the TUI cost meter (per-turn flash, status
|
|
7
|
+
* row USD). Its ladder is keyed against the live Anvil model slugs and
|
|
8
|
+
* intentionally inflates an honest worst-case figure via the Sonnet
|
|
9
|
+
* fallback so an operator on a quiet model never gets billed by a
|
|
10
|
+
* surprise. It rounds to USD per 1M tokens at runtime.
|
|
11
|
+
*
|
|
12
|
+
* - `rate-card.ts` (this file) powers the persisted `/cost` table the
|
|
13
|
+
* operator reads to plan budget. It distinguishes open-weight models
|
|
14
|
+
* ($0 / $0 — infra cost only) from hosted closed models so the table
|
|
15
|
+
* does not double-charge an operator running a self-hosted Qwen or
|
|
16
|
+
* Kimi behind Pugi. The L19 spec calls these out by name.
|
|
17
|
+
*
|
|
18
|
+
* Both ladders intentionally agree on Anthropic Claude family pricing so
|
|
19
|
+
* the TUI flash and the persisted table cannot disagree on a Claude turn.
|
|
20
|
+
* If they diverge, the per-model-pricing ladder wins for live UI; the
|
|
21
|
+
* rate card here wins for the persisted `.pugi/cost.json` aggregate.
|
|
22
|
+
*
|
|
23
|
+
* Prices are USD per 1,000,000 tokens, sourced from the L19 spec
|
|
24
|
+
* which mirrors provider list-price pages as of that date.
|
|
25
|
+
*/
|
|
26
|
+
/**
|
|
27
|
+
* Exact-match price ladder keyed by model slug. Slugs match the L19 task
|
|
28
|
+
* spec verbatim so a copy-paste from the sprint doc resolves without
|
|
29
|
+
* normalisation.
|
|
30
|
+
*/
|
|
31
|
+
export const RATES_PER_MTOKEN = Object.freeze({
|
|
32
|
+
// Anthropic Claude family (hosted, billed).
|
|
33
|
+
'claude-opus-4-7': { input: 15, output: 75 },
|
|
34
|
+
'claude-opus-4-6': { input: 15, output: 75 },
|
|
35
|
+
'claude-sonnet-4-6': { input: 3, output: 15 },
|
|
36
|
+
'claude-sonnet-4-5': { input: 3, output: 15 },
|
|
37
|
+
'claude-haiku-4-5-20251001': { input: 1, output: 5 },
|
|
38
|
+
'claude-haiku-4-5': { input: 1, output: 5 },
|
|
39
|
+
// Open-weight models — infra cost only, never per-token billed. The
|
|
40
|
+
// note column surfaces the reason so a CFO reading the JSON envelope
|
|
41
|
+
// does not assume the row is broken.
|
|
42
|
+
'qwen3-coder-480b-instruct-fp8': { input: 0, output: 0, note: 'open-weight' },
|
|
43
|
+
'kimi-k2.6': { input: 0, output: 0, note: 'open-weight' },
|
|
44
|
+
'deepseek-v4-pro': { input: 0, output: 0, note: 'open-weight' },
|
|
45
|
+
});
|
|
46
|
+
/**
|
|
47
|
+
* Family-prefix fallback — used only when an exact slug miss. Mirrors the
|
|
48
|
+
* approach in `model-pricing.ts` so a future model rebind (e.g.
|
|
49
|
+
* `claude-opus-4-8`) prices reasonably without a code edit.
|
|
50
|
+
*/
|
|
51
|
+
const FAMILY_FALLBACKS = [
|
|
52
|
+
['claude-opus-', { input: 15, output: 75 }],
|
|
53
|
+
['claude-sonnet-', { input: 3, output: 15 }],
|
|
54
|
+
['claude-haiku-', { input: 1, output: 5 }],
|
|
55
|
+
['qwen', { input: 0, output: 0, note: 'open-weight' }],
|
|
56
|
+
['kimi', { input: 0, output: 0, note: 'open-weight' }],
|
|
57
|
+
['deepseek', { input: 0, output: 0, note: 'open-weight' }],
|
|
58
|
+
];
|
|
59
|
+
/**
|
|
60
|
+
* Final fallback for unknown slugs. Pinned to Sonnet-tier — same posture
|
|
61
|
+
* as `model-pricing.ts`'s default, so an unrecognised hosted model bills
|
|
62
|
+
* "honestly conservative" rather than $0 (which would silently hide cost
|
|
63
|
+
* from the operator).
|
|
64
|
+
*/
|
|
65
|
+
const DEFAULT_RATE = { input: 3, output: 15, note: 'unknown model — Sonnet-tier estimate' };
|
|
66
|
+
/**
|
|
67
|
+
* Look up the rate for a model slug.
|
|
68
|
+
*
|
|
69
|
+
* Resolution order:
|
|
70
|
+
* 1. Exact match in `RATES_PER_MTOKEN`.
|
|
71
|
+
* 2. Family-prefix match (first hit wins).
|
|
72
|
+
* 3. Default Sonnet-tier estimate.
|
|
73
|
+
*
|
|
74
|
+
* Pure, never throws. Called on every cost-tracker write so the hot path
|
|
75
|
+
* stays branch-cheap.
|
|
76
|
+
*/
|
|
77
|
+
export function rateFor(model) {
|
|
78
|
+
if (!model || typeof model !== 'string')
|
|
79
|
+
return DEFAULT_RATE;
|
|
80
|
+
const exact = RATES_PER_MTOKEN[model];
|
|
81
|
+
if (exact)
|
|
82
|
+
return exact;
|
|
83
|
+
for (const [prefix, rate] of FAMILY_FALLBACKS) {
|
|
84
|
+
if (model.startsWith(prefix))
|
|
85
|
+
return rate;
|
|
86
|
+
}
|
|
87
|
+
return DEFAULT_RATE;
|
|
88
|
+
}
|
|
89
|
+
/**
|
|
90
|
+
* Compute the USD cost for a single (model, inputTokens, outputTokens)
|
|
91
|
+
* triple. Defensive against negative / NaN inputs — out-of-range values
|
|
92
|
+
* floor to zero so a buggy upstream cannot credit a negative cost.
|
|
93
|
+
*/
|
|
94
|
+
export function estimateUsd(model, inputTokens, outputTokens) {
|
|
95
|
+
const rate = rateFor(model);
|
|
96
|
+
const safeIn = Number.isFinite(inputTokens) && inputTokens > 0 ? inputTokens : 0;
|
|
97
|
+
const safeOut = Number.isFinite(outputTokens) && outputTokens > 0 ? outputTokens : 0;
|
|
98
|
+
const usd = (safeIn * rate.input + safeOut * rate.output) / 1_000_000;
|
|
99
|
+
return Number.isFinite(usd) && usd > 0 ? usd : 0;
|
|
100
|
+
}
|
|
101
|
+
/**
|
|
102
|
+
* Format a USD figure for the `/cost` table.
|
|
103
|
+
*
|
|
104
|
+
* - `≥ $0.01` → two decimals (`$0.46`).
|
|
105
|
+
* - `< $0.01` but `> 0` → three decimals (`$0.003`) so fractions of a
|
|
106
|
+
* cent are honest instead of rounding to `$0.00`.
|
|
107
|
+
* - Exactly `0` or NaN → `$0.00`.
|
|
108
|
+
*
|
|
109
|
+
* Mirrors `formatCostUsd` from `model-pricing.ts` intentionally — both
|
|
110
|
+
* surfaces should print the same number in the same shape.
|
|
111
|
+
*/
|
|
112
|
+
export function formatUsd(value) {
|
|
113
|
+
if (!Number.isFinite(value) || value <= 0)
|
|
114
|
+
return '$0.00';
|
|
115
|
+
if (value >= 0.01)
|
|
116
|
+
return `$${value.toFixed(2)}`;
|
|
117
|
+
return `$${value.toFixed(3)}`;
|
|
118
|
+
}
|
|
119
|
+
/**
|
|
120
|
+
* Format a token count for the `/cost` table. Uses comma-thousands so the
|
|
121
|
+
* table reads `14,300` instead of `14.3k` — distinct from the TUI status
|
|
122
|
+
* row which uses `k`/`m` shortening to save column width.
|
|
123
|
+
*/
|
|
124
|
+
export function formatTokensWithCommas(value) {
|
|
125
|
+
if (!Number.isFinite(value) || value <= 0)
|
|
126
|
+
return '0';
|
|
127
|
+
return Math.floor(value).toLocaleString('en-US');
|
|
128
|
+
}
|
|
129
|
+
//# sourceMappingURL=rate-card.js.map
|