@pugi/cli 0.1.0-beta.8 → 0.1.0-beta.87
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 +96 -0
- package/THIRD_PARTY_NOTICES.md +40 -0
- 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/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 +2 -2
- 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/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/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 +12 -12
- 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 +40 -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 +293 -7
- package/dist/core/edits/format-matrix.js +26 -0
- package/dist/core/edits/fuzzy-ladder.js +650 -0
- package/dist/core/edits/index.js +3 -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 +140 -26
- package/dist/core/engine/auto-compact.js +179 -0
- package/dist/core/engine/budgets.js +186 -0
- package/dist/core/engine/context-prefix.js +155 -0
- package/dist/core/engine/index.js +1 -1
- package/dist/core/engine/intensity.js +158 -0
- package/dist/core/engine/intent.js +260 -0
- package/dist/core/engine/native-pugi.js +1295 -227
- package/dist/core/engine/prompts.js +134 -16
- package/dist/core/engine/strip-internal-fields.js +124 -0
- package/dist/core/engine/tool-bridge.js +1295 -59
- 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 +44 -0
- package/dist/core/hooks/index.js +15 -0
- package/dist/core/hooks/registry.js +213 -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/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 +776 -0
- package/dist/core/lsp/language-detect.js +66 -0
- package/dist/core/lsp/post-edit-diagnostics.js +171 -0
- package/dist/core/lsp/symbol-tools.js +372 -0
- package/dist/core/mcp/client.js +97 -28
- package/dist/core/mcp/http-server.js +553 -0
- package/dist/core/mcp/orchestrator-tools.js +662 -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/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 +93 -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-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/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 +2157 -214
- package/dist/core/repl/slash-commands.js +533 -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/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/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/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 +92 -0
- package/dist/core/settings.js +286 -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 +132 -43
- 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-manager/cleanup.js +123 -0
- package/dist/core/worktree-manager/manager.js +303 -0
- package/dist/index.js +28 -0
- package/dist/runtime/bootstrap.js +190 -0
- package/dist/runtime/cli.js +4151 -489
- package/dist/runtime/commands/agents.js +30 -30
- 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 +32 -32
- package/dist/runtime/commands/cost.js +199 -0
- package/dist/runtime/commands/delegate.js +244 -13
- 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 +184 -0
- package/dist/runtime/commands/init.js +254 -0
- package/dist/runtime/commands/lsp.js +368 -0
- package/dist/runtime/commands/mcp.js +879 -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 +14 -14
- 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/headless-repl.js +195 -0
- package/dist/runtime/headless.js +543 -0
- package/dist/runtime/load-hooks-or-exit.js +71 -0
- package/dist/runtime/plan-decompose.js +531 -0
- package/dist/runtime/update-check.js +28 -28
- package/dist/runtime/version.js +65 -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 +222 -0
- package/dist/tools/ask-user.js +115 -0
- package/dist/tools/bash.js +623 -45
- package/dist/tools/brief.js +224 -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/lsp-tools.js +189 -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 +85 -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 +11 -1
- package/dist/tui/ask-modal.js +14 -14
- 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/onboarding-wizard.js +240 -0
- package/dist/tui/permissions-picker.js +86 -0
- package/dist/tui/render.js +35 -0
- package/dist/tui/repl-render.js +332 -54
- 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 +124 -44
- 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 +23 -6
- package/test/scenarios/codegen-create-file.scenario.txt +13 -0
- package/test/scenarios/compact-force.scenario.txt +11 -0
- package/test/scenarios/identity.scenario.txt +11 -0
- package/test/scenarios/persona-handoff.scenario.txt +11 -0
- package/test/scenarios/walkback.scenario.txt +12 -0
- package/dist/core/engine/compaction-hook.js +0 -154
|
@@ -0,0 +1,133 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* synthetic_output tool — operator-readable echo helper (tool gap
|
|
3
|
+
* pack). EXPERIMENTAL / engine-only.
|
|
4
|
+
*
|
|
5
|
+
* Test fixture cousin of `brief`. Whereas `brief` persists a structured
|
|
6
|
+
* record to `.pugi/briefs/<session>.jsonl`, `synthetic_output` writes
|
|
7
|
+
* the supplied text VERBATIM к the requested stream (`stdout` or
|
|
8
|
+
* `stderr`). The use case is the engine-side test harness: a spec wants
|
|
9
|
+
* to assert that "the loop ran this tool with this payload" without
|
|
10
|
+
* spinning up the entire dispatch pipeline, AND wants to observe the
|
|
11
|
+
* write the same way an operator would. Production agent flows should
|
|
12
|
+
* prefer `brief` because the structured record survives crashes — the
|
|
13
|
+
* stream write does not.
|
|
14
|
+
*
|
|
15
|
+
* The tool is NOT exposed as a slash command. The schema is advertised
|
|
16
|
+
* to the engine only when the executor is built with an
|
|
17
|
+
* `allowSyntheticOutput: true` opt-in. Customer CLIs leave it off so
|
|
18
|
+
* the model cannot use it as a side-channel to bypass the normal tool
|
|
19
|
+
* result envelope (which is logged, classified, hooked, etc.).
|
|
20
|
+
*
|
|
21
|
+
* Wire shape:
|
|
22
|
+
* args: { stream: 'stdout'|'stderr', text: string }
|
|
23
|
+
* - text is capped at 16 KiB UTF-8; over-cap rejects with
|
|
24
|
+
* the SYNTHETIC_OUTPUT_TOO_LARGE sentinel.
|
|
25
|
+
* side: raw `process.stdout.write(text)` or `process.stderr.write(text)`.
|
|
26
|
+
* return: short JSON envelope { ok, stream, bytes } so the engine
|
|
27
|
+
* loop can audit the call without re-reading the streamed
|
|
28
|
+
* bytes (which it does not buffer).
|
|
29
|
+
*
|
|
30
|
+
* Atomicity: writes go to the live process stream; partial writes are
|
|
31
|
+
* the stream's responsibility, not the tool's. The tool does NOT add a
|
|
32
|
+
* trailing newline — the model controls the exact bytes. This is the
|
|
33
|
+
* design point that makes it a useful test fixture (specs assert on
|
|
34
|
+
* verbatim bytes).
|
|
35
|
+
*
|
|
36
|
+
* Brand voice: English only, no emoji, no banned words.
|
|
37
|
+
*/
|
|
38
|
+
/** Per-call byte cap. Mirrors the bash tool's stdout/stderr cap so the
|
|
39
|
+
* model cannot blow either pane with a single synthetic flush. */
|
|
40
|
+
export const SYNTHETIC_OUTPUT_MAX_BYTES = 16 * 1_024;
|
|
41
|
+
/** Canonical stream values. */
|
|
42
|
+
export const SYNTHETIC_OUTPUT_STREAMS = ['stdout', 'stderr'];
|
|
43
|
+
/** Sentinel returned when input validation rejects the call. */
|
|
44
|
+
export const SYNTHETIC_OUTPUT_INVALID_ARGS = 'SYNTHETIC_OUTPUT_INVALID_ARGS';
|
|
45
|
+
/** Sentinel returned when the encoded text exceeds the per-call cap.
|
|
46
|
+
* Distinct from the args-schema failure so the model knows to shorten
|
|
47
|
+
* the payload rather than change the shape. */
|
|
48
|
+
export const SYNTHETIC_OUTPUT_TOO_LARGE = 'SYNTHETIC_OUTPUT_TOO_LARGE';
|
|
49
|
+
/**
|
|
50
|
+
* Validate the raw arguments. Returns the typed value on success or a
|
|
51
|
+
* `SYNTHETIC_OUTPUT_INVALID_ARGS: ...` sentinel string.
|
|
52
|
+
*/
|
|
53
|
+
export function parseSyntheticOutputArgs(raw) {
|
|
54
|
+
if (typeof raw !== 'object' || raw === null || Array.isArray(raw)) {
|
|
55
|
+
return `${SYNTHETIC_OUTPUT_INVALID_ARGS}: arguments must be a JSON object`;
|
|
56
|
+
}
|
|
57
|
+
const obj = raw;
|
|
58
|
+
const issues = [];
|
|
59
|
+
const stream = obj['stream'];
|
|
60
|
+
if (typeof stream !== 'string') {
|
|
61
|
+
issues.push('stream: must be a string');
|
|
62
|
+
}
|
|
63
|
+
else if (!SYNTHETIC_OUTPUT_STREAMS.includes(stream)) {
|
|
64
|
+
issues.push(`stream: must be one of ${SYNTHETIC_OUTPUT_STREAMS.join('|')}`);
|
|
65
|
+
}
|
|
66
|
+
const text = obj['text'];
|
|
67
|
+
if (typeof text !== 'string') {
|
|
68
|
+
issues.push('text: must be a string');
|
|
69
|
+
}
|
|
70
|
+
if (issues.length > 0) {
|
|
71
|
+
return `${SYNTHETIC_OUTPUT_INVALID_ARGS}: ${issues.join('; ')}`;
|
|
72
|
+
}
|
|
73
|
+
return {
|
|
74
|
+
stream: stream,
|
|
75
|
+
text: text,
|
|
76
|
+
};
|
|
77
|
+
}
|
|
78
|
+
/**
|
|
79
|
+
* Dispatch entry point. Validates input, writes the text verbatim to
|
|
80
|
+
* the requested stream, and returns the structured envelope as JSON.
|
|
81
|
+
*
|
|
82
|
+
* Returns sentinel strings (no throw) on recoverable failures so the
|
|
83
|
+
* engine adapter surfaces them as tool results.
|
|
84
|
+
*/
|
|
85
|
+
export function dispatchSyntheticOutput(ctx, raw) {
|
|
86
|
+
const parsed = parseSyntheticOutputArgs(raw);
|
|
87
|
+
if (typeof parsed === 'string') {
|
|
88
|
+
return parsed;
|
|
89
|
+
}
|
|
90
|
+
const bytes = Buffer.byteLength(parsed.text, 'utf8');
|
|
91
|
+
if (bytes > SYNTHETIC_OUTPUT_MAX_BYTES) {
|
|
92
|
+
return `${SYNTHETIC_OUTPUT_TOO_LARGE}: encoded text exceeds ${SYNTHETIC_OUTPUT_MAX_BYTES} bytes`;
|
|
93
|
+
}
|
|
94
|
+
// Resolve writers once per call so a spec can override stdout while
|
|
95
|
+
// letting stderr fall through to the real process stream.
|
|
96
|
+
const writer = parsed.stream === 'stdout'
|
|
97
|
+
? (ctx.stdoutWrite ?? defaultStdoutWrite)
|
|
98
|
+
: (ctx.stderrWrite ?? defaultStderrWrite);
|
|
99
|
+
writer(parsed.text);
|
|
100
|
+
const result = {
|
|
101
|
+
ok: true,
|
|
102
|
+
stream: parsed.stream,
|
|
103
|
+
bytes,
|
|
104
|
+
};
|
|
105
|
+
return JSON.stringify(result);
|
|
106
|
+
}
|
|
107
|
+
function defaultStdoutWrite(chunk) {
|
|
108
|
+
process.stdout.write(chunk);
|
|
109
|
+
}
|
|
110
|
+
function defaultStderrWrite(chunk) {
|
|
111
|
+
process.stderr.write(chunk);
|
|
112
|
+
}
|
|
113
|
+
/**
|
|
114
|
+
* JSON-Schema fragment the schema builder advertises to the model.
|
|
115
|
+
* Hand-written for parity with the rest of the tool surface.
|
|
116
|
+
*/
|
|
117
|
+
export const syntheticOutputJsonSchema = {
|
|
118
|
+
type: 'object',
|
|
119
|
+
additionalProperties: false,
|
|
120
|
+
required: ['stream', 'text'],
|
|
121
|
+
properties: {
|
|
122
|
+
stream: {
|
|
123
|
+
type: 'string',
|
|
124
|
+
enum: [...SYNTHETIC_OUTPUT_STREAMS],
|
|
125
|
+
description: 'Destination stream: stdout or stderr.',
|
|
126
|
+
},
|
|
127
|
+
text: {
|
|
128
|
+
type: 'string',
|
|
129
|
+
description: `Verbatim bytes to write. Capped at ${SYNTHETIC_OUTPUT_MAX_BYTES} bytes UTF-8.`,
|
|
130
|
+
},
|
|
131
|
+
},
|
|
132
|
+
};
|
|
133
|
+
//# sourceMappingURL=synthetic-output.js.map
|
|
@@ -0,0 +1,208 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* task_* tool family — β1 T1/T6 (TodoWrite + agent task ledger).
|
|
3
|
+
*
|
|
4
|
+
* Mirrors the standard tool's TodoWrite tool surface so a model trained on
|
|
5
|
+
* the upstream tool grammar speaks Pugi's variant verbatim. Four ops:
|
|
6
|
+
*
|
|
7
|
+
* - `task_create` — append a new task to the session's todo ledger.
|
|
8
|
+
* Returns the assigned id.
|
|
9
|
+
* - `task_get` — fetch a single task by id.
|
|
10
|
+
* - `task_list` — list every task in the current session, ordered
|
|
11
|
+
* by createdAt ascending.
|
|
12
|
+
* - `task_update` — mutate status/title/notes of an existing task.
|
|
13
|
+
* Append-only journal — every mutation lands as a
|
|
14
|
+
* fresh JSONL line and the latest line per id wins
|
|
15
|
+
* on `task_list` / `task_get` reads.
|
|
16
|
+
*
|
|
17
|
+
* Persistence: append-only JSONL at
|
|
18
|
+
* `.pugi/sessions/<sessionId>/tasks.jsonl`. Append-only keeps crash
|
|
19
|
+
* recovery trivial — a partial write at the end of the file is the
|
|
20
|
+
* worst case and the parser drops the malformed tail line.
|
|
21
|
+
*
|
|
22
|
+
* Scope: this is the local-side ledger surface. Anvil-side mirror
|
|
23
|
+
* (cabinet `/projects/[id]/tasks` page) ships in β5 once the session-
|
|
24
|
+
* memory hook lands; until then the ledger is purely local.
|
|
25
|
+
*/
|
|
26
|
+
import { appendFileSync, chmodSync, existsSync, mkdirSync, readFileSync, } from 'node:fs';
|
|
27
|
+
import { dirname, join } from 'node:path';
|
|
28
|
+
import { randomUUID } from 'node:crypto';
|
|
29
|
+
function ledgerPath(ctx) {
|
|
30
|
+
// Defense-in-depth: the sessionId is supposed to be a UUID minted by
|
|
31
|
+
// openSession() but the tool surface is operator-facing. Validate the
|
|
32
|
+
// shape before composing a path — refuse anything that contains
|
|
33
|
+
// separators or shell wildcards.
|
|
34
|
+
if (!/^[a-zA-Z0-9_-]{1,128}$/.test(ctx.sessionId)) {
|
|
35
|
+
throw new Error(`task_*: invalid sessionId shape: "${ctx.sessionId}"`);
|
|
36
|
+
}
|
|
37
|
+
return join(ctx.workspaceRoot, '.pugi', 'sessions', ctx.sessionId, 'tasks.jsonl');
|
|
38
|
+
}
|
|
39
|
+
function nowIso(ctx) {
|
|
40
|
+
return (ctx.now ? ctx.now() : new Date()).toISOString();
|
|
41
|
+
}
|
|
42
|
+
function ensureDir(path) {
|
|
43
|
+
// β1a r1 : switched from POSIX-only
|
|
44
|
+
// `path.slice(0, path.lastIndexOf('/'))` to `path.dirname()` so
|
|
45
|
+
// Windows path separators (`\`) work. Also chmod the per-session
|
|
46
|
+
// directory to 0o700 — the tasks ledger carries operator-confidential
|
|
47
|
+
// brief text, status notes, and timing metadata that should not be
|
|
48
|
+
// world-readable through an inherited umask.
|
|
49
|
+
const dir = dirname(path);
|
|
50
|
+
if (!existsSync(dir)) {
|
|
51
|
+
mkdirSync(dir, { recursive: true });
|
|
52
|
+
try {
|
|
53
|
+
chmodSync(dir, 0o700);
|
|
54
|
+
}
|
|
55
|
+
catch {
|
|
56
|
+
// Best-effort. POSIX permission setting is a no-op on Windows
|
|
57
|
+
// NTFS, and the dir-creation race with another concurrent task
|
|
58
|
+
// tool call is the only realistic failure case. The 0o600 mode
|
|
59
|
+
// on the JSONL file itself remains the primary guard; the dir
|
|
60
|
+
// chmod is defense in depth for tools that walk `.pugi/`.
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
function readJournal(ctx) {
|
|
65
|
+
const path = ledgerPath(ctx);
|
|
66
|
+
if (!existsSync(path))
|
|
67
|
+
return [];
|
|
68
|
+
const raw = readFileSync(path, 'utf8');
|
|
69
|
+
const out = [];
|
|
70
|
+
for (const line of raw.split('\n')) {
|
|
71
|
+
if (!line.trim())
|
|
72
|
+
continue;
|
|
73
|
+
try {
|
|
74
|
+
const parsed = JSON.parse(line);
|
|
75
|
+
if ((parsed.op === 'create' || parsed.op === 'update') &&
|
|
76
|
+
typeof parsed.id === 'string' &&
|
|
77
|
+
typeof parsed.at === 'string') {
|
|
78
|
+
out.push(parsed);
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
catch {
|
|
82
|
+
// Drop malformed line (partial-write tail or external corruption).
|
|
83
|
+
// The append-only design guarantees only the LAST line can be bad
|
|
84
|
+
// — everything before it is whole.
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
return out;
|
|
88
|
+
}
|
|
89
|
+
function fold(journal) {
|
|
90
|
+
const out = new Map();
|
|
91
|
+
for (const entry of journal) {
|
|
92
|
+
if (entry.op === 'create') {
|
|
93
|
+
if (!entry.title)
|
|
94
|
+
continue;
|
|
95
|
+
out.set(entry.id, {
|
|
96
|
+
id: entry.id,
|
|
97
|
+
title: entry.title,
|
|
98
|
+
status: entry.status ?? 'pending',
|
|
99
|
+
...(entry.notes !== undefined ? { notes: entry.notes } : {}),
|
|
100
|
+
createdAt: entry.at,
|
|
101
|
+
updatedAt: entry.at,
|
|
102
|
+
});
|
|
103
|
+
}
|
|
104
|
+
else {
|
|
105
|
+
const prev = out.get(entry.id);
|
|
106
|
+
if (!prev)
|
|
107
|
+
continue; // update before create — drop silently
|
|
108
|
+
out.set(entry.id, {
|
|
109
|
+
...prev,
|
|
110
|
+
...(entry.title !== undefined ? { title: entry.title } : {}),
|
|
111
|
+
...(entry.status !== undefined ? { status: entry.status } : {}),
|
|
112
|
+
...(entry.notes !== undefined ? { notes: entry.notes } : {}),
|
|
113
|
+
updatedAt: entry.at,
|
|
114
|
+
});
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
return out;
|
|
118
|
+
}
|
|
119
|
+
function appendEntry(ctx, entry) {
|
|
120
|
+
const path = ledgerPath(ctx);
|
|
121
|
+
ensureDir(path);
|
|
122
|
+
appendFileSync(path, `${JSON.stringify(entry)}\n`, {
|
|
123
|
+
encoding: 'utf8',
|
|
124
|
+
mode: 0o600,
|
|
125
|
+
});
|
|
126
|
+
}
|
|
127
|
+
export function taskCreate(ctx, input) {
|
|
128
|
+
const title = input.title?.trim();
|
|
129
|
+
if (!title) {
|
|
130
|
+
throw new Error('task_create: title is required');
|
|
131
|
+
}
|
|
132
|
+
if (title.length > 2_000) {
|
|
133
|
+
throw new Error('task_create: title exceeds 2000 char cap');
|
|
134
|
+
}
|
|
135
|
+
const status = input.status ?? 'pending';
|
|
136
|
+
if (!isValidStatus(status)) {
|
|
137
|
+
throw new Error(`task_create: invalid status "${status}"`);
|
|
138
|
+
}
|
|
139
|
+
const id = `task-${randomUUID()}`;
|
|
140
|
+
const at = nowIso(ctx);
|
|
141
|
+
const entry = {
|
|
142
|
+
op: 'create',
|
|
143
|
+
id,
|
|
144
|
+
title,
|
|
145
|
+
status,
|
|
146
|
+
at,
|
|
147
|
+
...(input.notes !== undefined ? { notes: input.notes } : {}),
|
|
148
|
+
};
|
|
149
|
+
appendEntry(ctx, entry);
|
|
150
|
+
return {
|
|
151
|
+
id,
|
|
152
|
+
title,
|
|
153
|
+
status,
|
|
154
|
+
...(input.notes !== undefined ? { notes: input.notes } : {}),
|
|
155
|
+
createdAt: at,
|
|
156
|
+
updatedAt: at,
|
|
157
|
+
};
|
|
158
|
+
}
|
|
159
|
+
export function taskGet(ctx, id) {
|
|
160
|
+
if (typeof id !== 'string' || id.length === 0) {
|
|
161
|
+
throw new Error('task_get: id is required');
|
|
162
|
+
}
|
|
163
|
+
const folded = fold(readJournal(ctx));
|
|
164
|
+
return folded.get(id) ?? null;
|
|
165
|
+
}
|
|
166
|
+
export function taskList(ctx) {
|
|
167
|
+
const folded = fold(readJournal(ctx));
|
|
168
|
+
return Array.from(folded.values()).sort((a, b) => a.createdAt.localeCompare(b.createdAt));
|
|
169
|
+
}
|
|
170
|
+
export function taskUpdate(ctx, input) {
|
|
171
|
+
if (!input.id)
|
|
172
|
+
throw new Error('task_update: id is required');
|
|
173
|
+
const folded = fold(readJournal(ctx));
|
|
174
|
+
const existing = folded.get(input.id);
|
|
175
|
+
if (!existing) {
|
|
176
|
+
throw new Error(`task_update: unknown id "${input.id}"`);
|
|
177
|
+
}
|
|
178
|
+
if (input.status !== undefined && !isValidStatus(input.status)) {
|
|
179
|
+
throw new Error(`task_update: invalid status "${input.status}"`);
|
|
180
|
+
}
|
|
181
|
+
if (input.title !== undefined && input.title.trim().length === 0) {
|
|
182
|
+
throw new Error('task_update: title cannot be empty');
|
|
183
|
+
}
|
|
184
|
+
const at = nowIso(ctx);
|
|
185
|
+
const entry = {
|
|
186
|
+
op: 'update',
|
|
187
|
+
id: input.id,
|
|
188
|
+
at,
|
|
189
|
+
...(input.title !== undefined ? { title: input.title } : {}),
|
|
190
|
+
...(input.status !== undefined ? { status: input.status } : {}),
|
|
191
|
+
...(input.notes !== undefined ? { notes: input.notes } : {}),
|
|
192
|
+
};
|
|
193
|
+
appendEntry(ctx, entry);
|
|
194
|
+
return {
|
|
195
|
+
...existing,
|
|
196
|
+
...(input.title !== undefined ? { title: input.title } : {}),
|
|
197
|
+
...(input.status !== undefined ? { status: input.status } : {}),
|
|
198
|
+
...(input.notes !== undefined ? { notes: input.notes } : {}),
|
|
199
|
+
updatedAt: at,
|
|
200
|
+
};
|
|
201
|
+
}
|
|
202
|
+
function isValidStatus(status) {
|
|
203
|
+
return (status === 'pending' ||
|
|
204
|
+
status === 'in_progress' ||
|
|
205
|
+
status === 'completed' ||
|
|
206
|
+
status === 'cancelled');
|
|
207
|
+
}
|
|
208
|
+
//# sourceMappingURL=tasks.js.map
|
|
@@ -0,0 +1,184 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* todo_write tool — .
|
|
3
|
+
*
|
|
4
|
+
* Mirrors the standard tool's `TodoWrite` tool 1:1 so a model trained on the
|
|
5
|
+
* upstream grammar speaks Pugi's variant verbatim. The tool dispatches
|
|
6
|
+
* a BATCH replace of the workspace todo board (not an incremental
|
|
7
|
+
* mutation — the model emits the FULL list every call). At most ONE
|
|
8
|
+
* todo may carry `status: 'in_progress'` at any time; violations
|
|
9
|
+
* reject with the `TODO_INVARIANT_VIOLATED` sentinel and the board on
|
|
10
|
+
* disk is left unchanged.
|
|
11
|
+
*
|
|
12
|
+
* Relationship to `task_*` (β1 T1/T6, tools/tasks.ts):
|
|
13
|
+
* - `task_*` is GRANULAR (create/get/list/update one task at a
|
|
14
|
+
* time) with an append-only JSONL journal scoped to the SESSION.
|
|
15
|
+
* - `todo_write` is BATCH (snapshot the whole board) with an atomic
|
|
16
|
+
* JSON snapshot scoped to the WORKSPACE.
|
|
17
|
+
* They are complementary surfaces: agents that prefer the upstream
|
|
18
|
+
* TodoWrite grammar use `todo_write`; agents that want a fine-grained
|
|
19
|
+
* audit trail use `task_*`.
|
|
20
|
+
*
|
|
21
|
+
* Hard rules (enforced by Zod + dispatcher):
|
|
22
|
+
* - `todos.length` ≤ 50 (board overload guard).
|
|
23
|
+
* - Every item: id (≥1 char, ≤128), content (≥1 char), status enum.
|
|
24
|
+
* - At most ONE item with `status === 'in_progress'`.
|
|
25
|
+
* - All ids unique within the batch.
|
|
26
|
+
*
|
|
27
|
+
* Dispatch returns the persisted board as JSON; callers can read
|
|
28
|
+
* `todos: [...]` directly. Errors return the sentinel-prefixed message
|
|
29
|
+
* so the engine adapter can pattern-match.
|
|
30
|
+
*/
|
|
31
|
+
import { z } from 'zod';
|
|
32
|
+
import { saveTodoBoard } from '../core/todos/state.js';
|
|
33
|
+
/** Cap matches the `task_*` family's title cap for parity. */
|
|
34
|
+
export const TODO_CONTENT_MAX = 2_000;
|
|
35
|
+
/** id is opaque to us but must be slug-safe so file paths could embed it. */
|
|
36
|
+
export const TODO_ID_MIN = 1;
|
|
37
|
+
export const TODO_ID_MAX = 128;
|
|
38
|
+
/** Hard cap on board size. Beyond this the operator should split work. */
|
|
39
|
+
export const TODO_BATCH_MAX = 50;
|
|
40
|
+
export const todoItemSchema = z
|
|
41
|
+
.strictObject({
|
|
42
|
+
id: z
|
|
43
|
+
.string()
|
|
44
|
+
.min(TODO_ID_MIN)
|
|
45
|
+
.max(TODO_ID_MAX)
|
|
46
|
+
.describe('Stable id for this todo. Opaque, ≤128 chars.'),
|
|
47
|
+
content: z
|
|
48
|
+
.string()
|
|
49
|
+
.min(1)
|
|
50
|
+
.max(TODO_CONTENT_MAX)
|
|
51
|
+
.describe('Imperative task description. E.g. "Add invariant check".'),
|
|
52
|
+
status: z
|
|
53
|
+
.enum(['pending', 'in_progress', 'completed'])
|
|
54
|
+
.describe('Lifecycle status. At most ONE in_progress per board.'),
|
|
55
|
+
activeForm: z
|
|
56
|
+
.string()
|
|
57
|
+
.min(1)
|
|
58
|
+
.max(TODO_CONTENT_MAX)
|
|
59
|
+
.optional()
|
|
60
|
+
.describe('Present-continuous form. E.g. "Adding invariant check".'),
|
|
61
|
+
});
|
|
62
|
+
export const todoWriteArgsSchema = z.strictObject({
|
|
63
|
+
todos: z
|
|
64
|
+
.array(todoItemSchema)
|
|
65
|
+
.max(TODO_BATCH_MAX)
|
|
66
|
+
.describe(`Full todo board (batch replace, not incremental). Max ${TODO_BATCH_MAX} items. ` +
|
|
67
|
+
`At most ONE item may carry status="in_progress".`),
|
|
68
|
+
});
|
|
69
|
+
/**
|
|
70
|
+
* JSON-Schema fragment surfaced to the model via the tool-bridge
|
|
71
|
+
* `parameters` field. Mirrors the Zod schema 1:1 — kept hand-written
|
|
72
|
+
* (same convention as ask_user_question) because the runtime engine
|
|
73
|
+
* wires OpenAI-compatible JSON Schema and we have not greenlit the
|
|
74
|
+
* zod-to-json-schema transitive dep. Keep both in lockstep.
|
|
75
|
+
*/
|
|
76
|
+
export const todoWriteJsonSchema = {
|
|
77
|
+
type: 'object',
|
|
78
|
+
additionalProperties: false,
|
|
79
|
+
required: ['todos'],
|
|
80
|
+
properties: {
|
|
81
|
+
todos: {
|
|
82
|
+
type: 'array',
|
|
83
|
+
maxItems: TODO_BATCH_MAX,
|
|
84
|
+
description: `Full todo board (batch replace, not incremental). Max ${TODO_BATCH_MAX} items. ` +
|
|
85
|
+
`At most ONE item may carry status="in_progress".`,
|
|
86
|
+
items: {
|
|
87
|
+
type: 'object',
|
|
88
|
+
additionalProperties: false,
|
|
89
|
+
required: ['id', 'content', 'status'],
|
|
90
|
+
properties: {
|
|
91
|
+
id: {
|
|
92
|
+
type: 'string',
|
|
93
|
+
minLength: TODO_ID_MIN,
|
|
94
|
+
maxLength: TODO_ID_MAX,
|
|
95
|
+
description: 'Stable id for this todo. Opaque, ≤128 chars.',
|
|
96
|
+
},
|
|
97
|
+
content: {
|
|
98
|
+
type: 'string',
|
|
99
|
+
minLength: 1,
|
|
100
|
+
maxLength: TODO_CONTENT_MAX,
|
|
101
|
+
description: 'Imperative task description.',
|
|
102
|
+
},
|
|
103
|
+
status: {
|
|
104
|
+
type: 'string',
|
|
105
|
+
enum: ['pending', 'in_progress', 'completed'],
|
|
106
|
+
description: 'Lifecycle status. At most ONE in_progress per board.',
|
|
107
|
+
},
|
|
108
|
+
activeForm: {
|
|
109
|
+
type: 'string',
|
|
110
|
+
minLength: 1,
|
|
111
|
+
maxLength: TODO_CONTENT_MAX,
|
|
112
|
+
description: 'Present-continuous form.',
|
|
113
|
+
},
|
|
114
|
+
},
|
|
115
|
+
},
|
|
116
|
+
},
|
|
117
|
+
},
|
|
118
|
+
};
|
|
119
|
+
/**
|
|
120
|
+
* Sentinel prefix the dispatcher returns when Zod schema validation
|
|
121
|
+
* rejects the raw arguments. Distinct from `TODO_INVARIANT_VIOLATED`
|
|
122
|
+
* (>1 in_progress) and `TODO_DUPLICATE_ID` (collision within batch),
|
|
123
|
+
* which are emitted from `saveTodoBoard` AFTER schema parsing.
|
|
124
|
+
*
|
|
125
|
+
* Surfaced as a return string (not a throw) so the engine adapter sees
|
|
126
|
+
* a recoverable tool error and the model can self-correct its args,
|
|
127
|
+
* instead of the engine loop tearing down on an uncaught ZodError.
|
|
128
|
+
*/
|
|
129
|
+
export const TODO_INVALID_ARGS = 'INVALID_ARGS';
|
|
130
|
+
/**
|
|
131
|
+
* Render a ZodError into a deterministic `INVALID_ARGS: ...` sentinel
|
|
132
|
+
* the model can pattern-match. Each issue contributes one
|
|
133
|
+
* `path: message` clause; clauses are joined with `; ` so the model
|
|
134
|
+
* sees every offence in a single line. Path with the root scope is
|
|
135
|
+
* rendered as `<root>` to avoid an empty colon.
|
|
136
|
+
*/
|
|
137
|
+
function renderZodIssues(error) {
|
|
138
|
+
const parts = error.issues.map((issue) => {
|
|
139
|
+
const path = issue.path.length === 0 ? '<root>' : issue.path.join('.');
|
|
140
|
+
return `${path}: ${issue.message}`;
|
|
141
|
+
});
|
|
142
|
+
return `${TODO_INVALID_ARGS}: ${parts.join('; ')}`;
|
|
143
|
+
}
|
|
144
|
+
/**
|
|
145
|
+
* Validate via Zod + persist atomically. Surfaces three sentinel
|
|
146
|
+
* families the dispatcher pattern-matches on:
|
|
147
|
+
* - `INVALID_ARGS: <path>: <issue>; ...` — Zod schema rejected
|
|
148
|
+
* the raw arguments (returned as STRING, not thrown).
|
|
149
|
+
* - `TODO_INVARIANT_VIOLATED: ...` — >1 in_progress
|
|
150
|
+
* (thrown by `saveTodoBoard`).
|
|
151
|
+
* - `TODO_DUPLICATE_ID: ...` — collision within batch
|
|
152
|
+
* (thrown by `saveTodoBoard`).
|
|
153
|
+
*
|
|
154
|
+
* Why the asymmetry: schema rejection means the model emitted malformed
|
|
155
|
+
* structure (missing field, wrong type) and CAN self-correct given a
|
|
156
|
+
* clear breakdown of the offending path. The invariant + duplicate-id
|
|
157
|
+
* paths mean the model emitted structurally-valid but semantically
|
|
158
|
+
* conflicting state — those still throw so the engine loop's tool-error
|
|
159
|
+
* hook can surface them through `PostToolUseFailure` for observability,
|
|
160
|
+
* mirroring how the file-tools layer surfaces `STALE_READ` / `PermissionDenied`.
|
|
161
|
+
*/
|
|
162
|
+
export function dispatchTodoWrite(ctx, rawArgs) {
|
|
163
|
+
// L16 P1 fix : `.parse` throws a `ZodError` on validation
|
|
164
|
+
// failure. The previous implementation let that throw bubble through
|
|
165
|
+
// the engine adapter's catch arm as a free-form `error.message`,
|
|
166
|
+
// which (a) loses the issue-by-issue structure the model needs to
|
|
167
|
+
// self-correct, and (b) tears down the tool-call as a hard failure
|
|
168
|
+
// rather than a recoverable tool result. Switch to `safeParse` and
|
|
169
|
+
// emit a structured `INVALID_ARGS: <path>: <issue>; ...` sentinel
|
|
170
|
+
// string instead — the engine sees a successful tool call, the model
|
|
171
|
+
// sees the offending paths, and the dispatcher's catch arm reserves
|
|
172
|
+
// throws for the genuine semantic conflicts emitted by `saveTodoBoard`.
|
|
173
|
+
const parsed = todoWriteArgsSchema.safeParse(rawArgs);
|
|
174
|
+
if (!parsed.success) {
|
|
175
|
+
return renderZodIssues(parsed.error);
|
|
176
|
+
}
|
|
177
|
+
const stateCtx = {
|
|
178
|
+
workspaceRoot: ctx.workspaceRoot,
|
|
179
|
+
...(ctx.now ? { now: ctx.now } : {}),
|
|
180
|
+
};
|
|
181
|
+
const board = saveTodoBoard(stateCtx, parsed.data.todos);
|
|
182
|
+
return JSON.stringify(board);
|
|
183
|
+
}
|
|
184
|
+
//# sourceMappingURL=todo-write.js.map
|