@pugi/cli 0.1.0-beta.98 → 1.0.0-alpha.1
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/LICENSE +1 -1
- package/README.md +11 -191
- package/bin/pugi +8 -0
- package/package.json +15 -71
- package/postinstall.mjs +31 -0
- package/CHANGELOG.md +0 -132
- package/THIRD_PARTY_NOTICES.md +0 -40
- package/assets/pugi-mascot.ansi +0 -16
- package/assets/pugi-prozr2-mascot.ansi +0 -9
- package/bin/run.js +0 -34
- package/dist/commands/deploy.js +0 -439
- package/dist/commands/flatten.js +0 -191
- package/dist/commands/jobs-watch.js +0 -201
- package/dist/commands/jobs.js +0 -260
- package/dist/commands/retro.js +0 -210
- package/dist/commands/smoke.js +0 -133
- package/dist/core/agent-progress/cleanup.js +0 -134
- package/dist/core/agent-progress/schema.js +0 -144
- package/dist/core/agent-progress/writer.js +0 -101
- package/dist/core/agents/adaptive-router.js +0 -330
- package/dist/core/agents/loader.js +0 -104
- package/dist/core/agents/query-decomposer.js +0 -297
- package/dist/core/agents/registry.js +0 -69
- package/dist/core/approvals/shortcut-resolver.js +0 -98
- package/dist/core/artifact-chain/dispatcher.js +0 -148
- package/dist/core/artifact-chain/exporter.js +0 -164
- package/dist/core/artifact-chain/state.js +0 -243
- package/dist/core/artifact-chain/steps.js +0 -169
- package/dist/core/ask-user/question.js +0 -92
- package/dist/core/audit/audit-trail.js +0 -275
- package/dist/core/auth/ensure-authenticated.js +0 -129
- package/dist/core/auth/env-provider.js +0 -238
- package/dist/core/auto-open-browser.js +0 -128
- package/dist/core/auto-update/channels.js +0 -122
- package/dist/core/auto-update/checker.js +0 -241
- package/dist/core/auto-update/state.js +0 -235
- package/dist/core/bare-mode/index.js +0 -107
- package/dist/core/bash/redirect.js +0 -281
- package/dist/core/bash-classifier.js +0 -1397
- package/dist/core/checkpoint/resumer.js +0 -149
- package/dist/core/checkpoint/rewinder.js +0 -291
- package/dist/core/checkpoints/shadow-git.js +0 -670
- package/dist/core/citations/parser.js +0 -109
- package/dist/core/classifier/yolo-classifier.js +0 -88
- package/dist/core/clipboard.js +0 -70
- package/dist/core/codegraph/decision-store.js +0 -248
- package/dist/core/codegraph/detect-repo.js +0 -459
- package/dist/core/codegraph/install.js +0 -134
- package/dist/core/codegraph/offer-hook.js +0 -220
- package/dist/core/compact/auto-trigger.js +0 -96
- package/dist/core/compact/buffer-rewriter.js +0 -115
- package/dist/core/compact/summarizer.js +0 -208
- package/dist/core/compact/token-counter.js +0 -108
- package/dist/core/consensus/anvil-fanout.js +0 -276
- package/dist/core/consensus/diff-capture.js +0 -491
- package/dist/core/consensus/rubric.js +0 -233
- package/dist/core/context/builder.js +0 -114
- package/dist/core/context/compaction-events.js +0 -99
- package/dist/core/context/compaction.js +0 -602
- package/dist/core/context/index.js +0 -28
- package/dist/core/context/invariants.js +0 -250
- package/dist/core/context/markdown-loader.js +0 -288
- package/dist/core/context/markdown-traverse.js +0 -255
- package/dist/core/context/pugiignore.js +0 -316
- package/dist/core/context/repo-skeleton.js +0 -533
- package/dist/core/context/tool-eviction.js +0 -55
- package/dist/core/context/watcher.js +0 -342
- package/dist/core/context/working-set.js +0 -165
- package/dist/core/coordinator/agent-tools.js +0 -77
- package/dist/core/coordinator/agent-toolset.js +0 -65
- package/dist/core/coordinator/fsm.js +0 -73
- package/dist/core/coordinator/mode-fsm.js +0 -70
- package/dist/core/cost/rate-card.js +0 -129
- package/dist/core/cost/tracker.js +0 -221
- package/dist/core/credentials.js +0 -355
- package/dist/core/cron/scheduler.js +0 -138
- package/dist/core/denial-tracking/index.js +0 -8
- package/dist/core/denial-tracking/state.js +0 -264
- package/dist/core/diagnostics/probe-runner.js +0 -93
- package/dist/core/diagnostics/probes/api.js +0 -46
- package/dist/core/diagnostics/probes/auth.js +0 -93
- package/dist/core/diagnostics/probes/bare-mode.js +0 -42
- package/dist/core/diagnostics/probes/cli-version.js +0 -127
- package/dist/core/diagnostics/probes/config.js +0 -72
- package/dist/core/diagnostics/probes/denial-tracking.js +0 -57
- package/dist/core/diagnostics/probes/disk.js +0 -81
- package/dist/core/diagnostics/probes/engine-live.js +0 -46
- package/dist/core/diagnostics/probes/git.js +0 -65
- package/dist/core/diagnostics/probes/hooks.js +0 -118
- package/dist/core/diagnostics/probes/mcp.js +0 -75
- package/dist/core/diagnostics/probes/node.js +0 -59
- package/dist/core/diagnostics/probes/pnpm.js +0 -36
- package/dist/core/diagnostics/probes/pugi-md.js +0 -89
- package/dist/core/diagnostics/probes/sandbox.js +0 -72
- package/dist/core/diagnostics/probes/session.js +0 -74
- package/dist/core/diagnostics/probes/status-snapshot.js +0 -488
- package/dist/core/diagnostics/probes/workspace.js +0 -63
- package/dist/core/diagnostics/types.js +0 -70
- package/dist/core/dispatch/cache-cleanup.js +0 -197
- package/dist/core/dispatch/cache-handoff.js +0 -295
- package/dist/core/edits/apply-patch-layer-e.js +0 -189
- package/dist/core/edits/dispatch.js +0 -511
- package/dist/core/edits/format-detector.js +0 -260
- package/dist/core/edits/format-matrix.js +0 -26
- package/dist/core/edits/fuzzy-ladder.js +0 -650
- package/dist/core/edits/index.js +0 -19
- package/dist/core/edits/journal.js +0 -199
- package/dist/core/edits/layer-a-apply.js +0 -217
- package/dist/core/edits/layer-a-fuzzy-apply.js +0 -198
- package/dist/core/edits/layer-b-apply.js +0 -211
- package/dist/core/edits/layer-c-apply.js +0 -160
- package/dist/core/edits/layer-d-ast.js +0 -572
- package/dist/core/edits/marker-parser.js +0 -401
- package/dist/core/edits/security-gate.js +0 -223
- package/dist/core/edits/verify-hook.js +0 -273
- package/dist/core/edits/worktree.js +0 -322
- package/dist/core/engine/adapter-runner.js +0 -8
- package/dist/core/engine/anvil-client.js +0 -344
- package/dist/core/engine/auto-compact.js +0 -179
- package/dist/core/engine/budgets.js +0 -192
- package/dist/core/engine/context-prefix.js +0 -155
- package/dist/core/engine/index.js +0 -12
- package/dist/core/engine/intensity.js +0 -163
- package/dist/core/engine/intent.js +0 -260
- package/dist/core/engine/native-pugi.js +0 -1616
- package/dist/core/engine/noop.js +0 -27
- package/dist/core/engine/prompts.js +0 -236
- package/dist/core/engine/strip-internal-fields.js +0 -124
- package/dist/core/engine/tool-bridge.js +0 -2173
- package/dist/core/engine/verification-patterns.js +0 -195
- package/dist/core/evaluation/golden-dataset.js +0 -293
- package/dist/core/feedback/queue.js +0 -177
- package/dist/core/feedback/submitter.js +0 -145
- package/dist/core/file-cache.js +0 -141
- package/dist/core/flatten/flatten-repo.js +0 -439
- package/dist/core/format/osc8-link.js +0 -28
- package/dist/core/hook-chains.js +0 -392
- package/dist/core/hooks/citation-verify-hook.js +0 -138
- package/dist/core/hooks/citation-verify.js +0 -112
- package/dist/core/hooks/events.js +0 -46
- package/dist/core/hooks/index.js +0 -15
- package/dist/core/hooks/registry.js +0 -216
- package/dist/core/hooks/runner.js +0 -236
- package/dist/core/hooks/v2/event-emitter.js +0 -115
- package/dist/core/hooks/v2/executor.js +0 -282
- package/dist/core/hooks/v2/index.js +0 -25
- package/dist/core/hooks/v2/lifecycle.js +0 -104
- package/dist/core/hooks/v2/loader.js +0 -216
- package/dist/core/hooks/v2/matcher.js +0 -125
- package/dist/core/hooks/v2/trust.js +0 -143
- package/dist/core/hooks/v2/types.js +0 -86
- package/dist/core/hooks/worktree-events.js +0 -158
- package/dist/core/hooks.js +0 -415
- package/dist/core/image/renderer.js +0 -71
- package/dist/core/index-store.js +0 -260
- package/dist/core/init/detector.js +0 -582
- package/dist/core/init/template-renderer.js +0 -242
- package/dist/core/jobs/registry.js +0 -462
- package/dist/core/ledger/results-tsv.js +0 -142
- package/dist/core/log-discipline/stdout-redirect.js +0 -51
- package/dist/core/lsp/cache.js +0 -105
- package/dist/core/lsp/client.js +0 -1229
- package/dist/core/lsp/language-detect.js +0 -66
- package/dist/core/lsp/post-edit-diagnostics.js +0 -171
- package/dist/core/lsp/server-detect.js +0 -173
- package/dist/core/lsp/symbol-cache.js +0 -162
- package/dist/core/lsp/symbol-tools.js +0 -664
- package/dist/core/mcp/client.js +0 -385
- package/dist/core/mcp/http-server.js +0 -553
- package/dist/core/mcp/orchestrator-config.js +0 -192
- package/dist/core/mcp/orchestrator-tools.js +0 -806
- package/dist/core/mcp/permission.js +0 -190
- package/dist/core/mcp/registry.js +0 -193
- package/dist/core/mcp/server-tools.js +0 -219
- package/dist/core/mcp/server.js +0 -397
- package/dist/core/mcp/trust.js +0 -91
- package/dist/core/memory/dual-write.js +0 -416
- package/dist/core/memory/passive-extract.js +0 -130
- package/dist/core/memory/phase1-kinds.js +0 -20
- package/dist/core/memory/secret-scanner.js +0 -304
- package/dist/core/memory-sync/queue.js +0 -170
- package/dist/core/metrics/extract.js +0 -113
- package/dist/core/modes/roo-modes.js +0 -68
- package/dist/core/onboarding/ensure-initialized.js +0 -133
- package/dist/core/onboarding/marker.js +0 -111
- package/dist/core/onboarding/telemetry-state.js +0 -108
- package/dist/core/output-style/presets.js +0 -176
- package/dist/core/output-style/state.js +0 -185
- package/dist/core/path-security.js +0 -345
- package/dist/core/permission.js +0 -369
- package/dist/core/permissions/auto-classifier.js +0 -124
- package/dist/core/permissions/bash-parser.js +0 -371
- package/dist/core/permissions/circuit-breaker.js +0 -83
- package/dist/core/permissions/constrained-edit.js +0 -91
- package/dist/core/permissions/gate.js +0 -278
- package/dist/core/permissions/index.js +0 -20
- package/dist/core/permissions/mode.js +0 -174
- package/dist/core/permissions/network-egress.js +0 -137
- package/dist/core/permissions/state.js +0 -241
- package/dist/core/permissions/tool-class.js +0 -107
- package/dist/core/plan-mode/ui-state.js +0 -51
- package/dist/core/plans/plan-artifact.js +0 -721
- package/dist/core/policy-limits/etag-store.js +0 -122
- package/dist/core/prd-check/parser.js +0 -215
- package/dist/core/prd-check/reporter.js +0 -127
- package/dist/core/prd-check/session-review.js +0 -557
- package/dist/core/prd-check/verifiers.js +0 -223
- package/dist/core/prompt-cache/client-cache.js +0 -99
- package/dist/core/prompts/assembly.js +0 -29
- package/dist/core/prompts/registry.js +0 -364
- package/dist/core/pugi-gitignore.js +0 -52
- package/dist/core/pugi-md/cc-compat-rules.js +0 -735
- package/dist/core/pugi-md/context-injector.js +0 -76
- package/dist/core/pugi-md/walk-up.js +0 -207
- package/dist/core/python/uv-installer.js +0 -270
- package/dist/core/python/uv-resolver.js +0 -83
- package/dist/core/rate-limit/narrator.js +0 -146
- package/dist/core/recipes/cli-types.js +0 -20
- package/dist/core/recipes/loader.js +0 -103
- package/dist/core/recipes/runner.js +0 -345
- package/dist/core/recipes/schema.js +0 -587
- package/dist/core/release-notes/parser.js +0 -241
- package/dist/core/release-notes/state.js +0 -116
- package/dist/core/repl/ask.js +0 -512
- package/dist/core/repl/cancellation.js +0 -98
- package/dist/core/repl/cap-warning.js +0 -91
- package/dist/core/repl/clipboard-read.js +0 -174
- package/dist/core/repl/dispatch-fsm.js +0 -220
- package/dist/core/repl/engine-bridge.js +0 -303
- package/dist/core/repl/history-search.js +0 -175
- package/dist/core/repl/history.js +0 -182
- package/dist/core/repl/kill-ring.js +0 -138
- package/dist/core/repl/model-pricing.js +0 -135
- package/dist/core/repl/privacy-banner.js +0 -71
- package/dist/core/repl/session.js +0 -4962
- package/dist/core/repl/slash-commands.js +0 -747
- package/dist/core/repl/store/index.js +0 -12
- package/dist/core/repl/store/jsonl-log.js +0 -321
- package/dist/core/repl/store/lockfile.js +0 -155
- package/dist/core/repl/store/session-store.js +0 -821
- package/dist/core/repl/store/types.js +0 -44
- package/dist/core/repl/store/uuid-v7.js +0 -68
- package/dist/core/repl/tool-route.js +0 -382
- package/dist/core/repl/workspace-context.js +0 -206
- package/dist/core/repo-map/build.js +0 -125
- package/dist/core/repo-map/cache.js +0 -185
- package/dist/core/repo-map/extractor.js +0 -254
- package/dist/core/repo-map/formatter.js +0 -145
- package/dist/core/repo-map/page-rank.js +0 -105
- package/dist/core/repo-map/scanner.js +0 -211
- package/dist/core/retro/git-collector.js +0 -251
- package/dist/core/retro/health-card.js +0 -25
- package/dist/core/retro/metrics.js +0 -342
- package/dist/core/retro/narrative.js +0 -249
- package/dist/core/retro/plane-collector.js +0 -274
- package/dist/core/retro/pr-issue-link.js +0 -65
- package/dist/core/retro/types.js +0 -16
- package/dist/core/retry-budget/budget.js +0 -284
- package/dist/core/retry-budget/index.js +0 -5
- package/dist/core/retry-budget/retry-cap.js +0 -74
- package/dist/core/routing/lead-worker.js +0 -43
- package/dist/core/routing/pre-flight-estimator.js +0 -108
- package/dist/core/runs/run-tree.js +0 -103
- package/dist/core/sandboxing/adapter.js +0 -29
- package/dist/core/sandboxing/index.js +0 -49
- package/dist/core/sandboxing/none.js +0 -19
- package/dist/core/sandboxing/seatbelt.js +0 -183
- package/dist/core/security/injection-scanner.js +0 -367
- package/dist/core/security/output-filter.js +0 -418
- package/dist/core/session/env-file.js +0 -105
- package/dist/core/session/section-budgets.js +0 -140
- package/dist/core/session.js +0 -377
- package/dist/core/settings.js +0 -400
- package/dist/core/share/formatter.js +0 -271
- package/dist/core/share/redactor.js +0 -221
- package/dist/core/share/uploader.js +0 -267
- package/dist/core/skills/defaults.js +0 -457
- package/dist/core/skills/loader.js +0 -454
- package/dist/core/skills/sources.js +0 -480
- package/dist/core/skills/trust.js +0 -172
- package/dist/core/smoke/headless-driver.js +0 -174
- package/dist/core/smoke/orchestrator.js +0 -194
- package/dist/core/smoke/runner.js +0 -238
- package/dist/core/smoke/scenario-parser.js +0 -316
- package/dist/core/statusline.js +0 -99
- package/dist/core/subagents/dispatcher-real.js +0 -600
- package/dist/core/subagents/dispatcher.js +0 -352
- package/dist/core/subagents/index.js +0 -39
- package/dist/core/subagents/isolation-matrix.js +0 -213
- package/dist/core/subagents/spawn.js +0 -101
- package/dist/core/telemetry/emitter.js +0 -229
- package/dist/core/telemetry/queue.js +0 -251
- package/dist/core/theme/context.js +0 -91
- package/dist/core/theme/presets.js +0 -228
- package/dist/core/theme/state.js +0 -181
- package/dist/core/todos/invariant.js +0 -10
- package/dist/core/todos/state.js +0 -177
- package/dist/core/tool-schema/compressor.js +0 -89
- package/dist/core/transport/version-interceptor.js +0 -166
- package/dist/core/trust.js +0 -109
- package/dist/core/tui/thinking-block.js +0 -64
- package/dist/core/vim/keymap.js +0 -288
- package/dist/core/vim/state.js +0 -92
- package/dist/core/watch-markers/marker-watcher.js +0 -133
- package/dist/core/worktree/include-parser.js +0 -249
- package/dist/core/worktree-manager/cleanup.js +0 -123
- package/dist/core/worktree-manager/manager.js +0 -303
- package/dist/index.js +0 -44
- package/dist/runtime/bootstrap.js +0 -190
- package/dist/runtime/cli.js +0 -8121
- package/dist/runtime/commands/agents.js +0 -385
- package/dist/runtime/commands/budget.js +0 -192
- package/dist/runtime/commands/cancel.js +0 -231
- package/dist/runtime/commands/chain.js +0 -489
- package/dist/runtime/commands/codegraph-status.js +0 -227
- package/dist/runtime/commands/compact.js +0 -297
- package/dist/runtime/commands/config.js +0 -595
- package/dist/runtime/commands/cost.js +0 -199
- package/dist/runtime/commands/delegate.js +0 -312
- package/dist/runtime/commands/dispatch.js +0 -126
- package/dist/runtime/commands/doctor.js +0 -579
- package/dist/runtime/commands/feedback.js +0 -184
- package/dist/runtime/commands/hooks.js +0 -187
- package/dist/runtime/commands/init.js +0 -254
- package/dist/runtime/commands/lsp.js +0 -368
- package/dist/runtime/commands/mcp.js +0 -935
- package/dist/runtime/commands/memory.js +0 -582
- package/dist/runtime/commands/model.js +0 -237
- package/dist/runtime/commands/onboarding.js +0 -275
- package/dist/runtime/commands/patch.js +0 -128
- package/dist/runtime/commands/permissions.js +0 -112
- package/dist/runtime/commands/plan.js +0 -143
- package/dist/runtime/commands/prd-check.js +0 -285
- package/dist/runtime/commands/privacy.js +0 -107
- package/dist/runtime/commands/recipe.js +0 -325
- package/dist/runtime/commands/redo-blob-store.js +0 -92
- package/dist/runtime/commands/redo.js +0 -361
- package/dist/runtime/commands/release-notes.js +0 -229
- package/dist/runtime/commands/repo-map.js +0 -95
- package/dist/runtime/commands/report.js +0 -299
- package/dist/runtime/commands/resume.js +0 -118
- package/dist/runtime/commands/review-consensus.js +0 -414
- package/dist/runtime/commands/rewind.js +0 -333
- package/dist/runtime/commands/roster.js +0 -117
- package/dist/runtime/commands/sessions.js +0 -163
- package/dist/runtime/commands/share.js +0 -316
- package/dist/runtime/commands/skills.js +0 -401
- package/dist/runtime/commands/status.js +0 -186
- package/dist/runtime/commands/stickers.js +0 -82
- package/dist/runtime/commands/style.js +0 -194
- package/dist/runtime/commands/theme.js +0 -196
- package/dist/runtime/commands/undo.js +0 -361
- package/dist/runtime/commands/update.js +0 -289
- package/dist/runtime/commands/vim.js +0 -140
- package/dist/runtime/commands/worktree.js +0 -177
- package/dist/runtime/commands/worktrees.js +0 -155
- package/dist/runtime/deprecation-warning.js +0 -69
- package/dist/runtime/engine-exit-code.js +0 -50
- package/dist/runtime/headless-repl.js +0 -195
- package/dist/runtime/headless.js +0 -548
- package/dist/runtime/load-hooks-or-exit.js +0 -71
- package/dist/runtime/plan-decompose.js +0 -531
- package/dist/runtime/sigint-guard.js +0 -272
- package/dist/runtime/stream-renderer.js +0 -195
- package/dist/runtime/update-check.js +0 -294
- package/dist/runtime/version.js +0 -65
- package/dist/runtime/worktree-bootstrap.js +0 -579
- package/dist/skills/bundled/batch.js +0 -617
- package/dist/skills/bundled/index.js +0 -45
- package/dist/skills/bundled/loop.js +0 -358
- package/dist/skills/bundled/remember.js +0 -383
- package/dist/skills/bundled/simplify.js +0 -289
- package/dist/skills/bundled/skillify.js +0 -373
- package/dist/skills/bundled/stuck.js +0 -558
- package/dist/skills/bundled/verify.js +0 -439
- package/dist/testing/vcr.js +0 -486
- package/dist/tools/agent-tool.js +0 -229
- package/dist/tools/apply-patch.js +0 -556
- package/dist/tools/ask-user-question.js +0 -337
- package/dist/tools/ask-user.js +0 -115
- package/dist/tools/bash.js +0 -1238
- package/dist/tools/brief.js +0 -224
- package/dist/tools/cron.js +0 -433
- package/dist/tools/enter-worktree.js +0 -250
- package/dist/tools/exit-worktree.js +0 -147
- package/dist/tools/file-tools.js +0 -553
- package/dist/tools/http-request.js +0 -336
- package/dist/tools/lsp-tools.js +0 -565
- package/dist/tools/mcp-tool.js +0 -260
- package/dist/tools/multi-edit.js +0 -361
- package/dist/tools/powershell.js +0 -268
- package/dist/tools/registry.js +0 -166
- package/dist/tools/server-tools.js +0 -892
- package/dist/tools/skill-tool.js +0 -96
- package/dist/tools/sleep.js +0 -99
- package/dist/tools/synthetic-output.js +0 -133
- package/dist/tools/tasks.js +0 -208
- package/dist/tools/todo-write.js +0 -184
- package/dist/tools/verify-plan-execution.js +0 -295
- package/dist/tools/web-fetch-injection-scanner.js +0 -207
- package/dist/tools/web-fetch.js +0 -720
- package/dist/tools/web-search.js +0 -458
- package/dist/tui/agent-progress-card.js +0 -111
- package/dist/tui/agent-tree-pane.js +0 -9
- package/dist/tui/agent-tree.js +0 -87
- package/dist/tui/ask-cli.js +0 -52
- package/dist/tui/ask-modal.js +0 -211
- package/dist/tui/ask-user-question-chips.js +0 -315
- package/dist/tui/ask-user-question-prompt.js +0 -203
- package/dist/tui/compact-banner.js +0 -81
- package/dist/tui/conversation-pane.js +0 -164
- package/dist/tui/cost-table.js +0 -111
- package/dist/tui/device-flow.js +0 -142
- package/dist/tui/doctor-table.js +0 -46
- package/dist/tui/feedback-prompt.js +0 -156
- package/dist/tui/input-box.js +0 -732
- package/dist/tui/login-picker.js +0 -69
- package/dist/tui/markdown-render.js +0 -266
- package/dist/tui/multi-file-diff-approval.js +0 -375
- package/dist/tui/onboarding-wizard.js +0 -240
- package/dist/tui/permissions-picker.js +0 -86
- package/dist/tui/render.js +0 -160
- package/dist/tui/repl-render.js +0 -770
- package/dist/tui/repl-splash-art.js +0 -64
- package/dist/tui/repl-splash-mascot.js +0 -154
- package/dist/tui/repl-splash.js +0 -117
- package/dist/tui/repl.js +0 -378
- package/dist/tui/slash-palette.js +0 -106
- package/dist/tui/splash-data.js +0 -61
- package/dist/tui/splash.js +0 -31
- package/dist/tui/status-bar.js +0 -209
- package/dist/tui/status-table.js +0 -7
- package/dist/tui/stickers-art.js +0 -136
- package/dist/tui/style-table.js +0 -28
- package/dist/tui/theme-table.js +0 -29
- package/dist/tui/thinking-spinner.js +0 -123
- package/dist/tui/tool-stream-pane.js +0 -140
- package/dist/tui/update-banner.js +0 -33
- package/dist/tui/vim-input.js +0 -267
- package/dist/tui/welcome-banner.js +0 -107
- package/dist/tui/welcome-data.js +0 -293
- package/dist/tui/workspace-context.js +0 -105
- package/docs/examples/codegraph.mcp.json +0 -10
- package/test/scenarios/codegen-create-file.scenario.txt +0 -13
- package/test/scenarios/compact-force.scenario.txt +0 -12
- package/test/scenarios/identity.scenario.txt +0 -11
- package/test/scenarios/persona-handoff.scenario.txt +0 -12
- package/test/scenarios/walkback.scenario.txt +0 -12
package/dist/core/repl/ask.js
DELETED
|
@@ -1,512 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Office-hours forcing questions + plan-review tag parser - Sprint .
|
|
3
|
-
*
|
|
4
|
-
* Pugi's persona prompt teaches Pugi to emit two structured XML envelopes
|
|
5
|
-
* when she would otherwise have to guess. Operator chat then pauses on a
|
|
6
|
-
* modal until the operator answers, eliminating the "fabricate a default
|
|
7
|
-
* silently" failure mode that peer CLI, the upstream tool, and Gemini CLI all
|
|
8
|
-
* trip on with low-confidence intents.
|
|
9
|
-
*
|
|
10
|
-
* <pugi-ask>
|
|
11
|
-
* <question>Which deployment target?</question>
|
|
12
|
-
* <option value="vercel" label="Vercel" desc="Static + edge"/>
|
|
13
|
-
* <option value="cloudflare" label="Cloudflare Pages" desc="Edge runtime"/>
|
|
14
|
-
* </pugi-ask>
|
|
15
|
-
*
|
|
16
|
-
* <pugi-plan-review>
|
|
17
|
-
* <step>1. Edit src/foo.ts: add new export</step>
|
|
18
|
-
* <step>2. Write tests/foo.spec.ts: 8 test cases</step>
|
|
19
|
-
* <risk>Touches public API surface.</risk>
|
|
20
|
-
* </pugi-plan-review>
|
|
21
|
-
*
|
|
22
|
-
* This module is a pure, framework-free parser. The session module imports
|
|
23
|
-
* `extractAskTags()` / `extractPlanReviewTags()` and routes the typed
|
|
24
|
-
* records to the Ink modal layer; the REPL UI never sees raw XML.
|
|
25
|
-
*
|
|
26
|
-
* # Why a hand-rolled parser
|
|
27
|
-
*
|
|
28
|
-
* Generic XML libraries (sax, fast-xml-parser, xmldoc) carry a large
|
|
29
|
-
* attack surface (external entity expansion, recursive blowup on
|
|
30
|
-
* malformed input, sloppy attribute handling) and require ~50 KB of
|
|
31
|
-
* runtime dependency. The persona-side grammar here is two tag families
|
|
32
|
-
* with a closed attribute set, so a bounded tokenizer is both safer and
|
|
33
|
-
* smaller. Defence-in-depth choices:
|
|
34
|
-
*
|
|
35
|
-
* - Reject raw `&` outside `&` / `<` / `>` / `"` /
|
|
36
|
-
* `'` (entities are decoded; everything else is malformed).
|
|
37
|
-
* - Forbid nested ask-within-ask (would crash the modal stack).
|
|
38
|
-
* - Cap option count at 4 (the upstream tool AskUserQuestion baseline).
|
|
39
|
-
* - Cap label / desc / question / step / risk at 80 chars (terminal
|
|
40
|
-
* rendering budget, also discourages prompt injection via huge
|
|
41
|
-
* payloads).
|
|
42
|
-
* - Reject CDATA, comments, processing instructions, DOCTYPE — none
|
|
43
|
-
* of those appear in the legal grammar.
|
|
44
|
-
* - Reject any attribute that is not in the per-tag allowlist.
|
|
45
|
-
*
|
|
46
|
-
* # Buffering across streaming chunks
|
|
47
|
-
*
|
|
48
|
-
* The session calls `extractAskTags(buffer)` on the accumulated
|
|
49
|
-
* `agent.step.detail` body. If the close tag has not arrived yet, the
|
|
50
|
-
* parser returns `{ tags: [], remainder: buffer }` and the session waits
|
|
51
|
-
* for more chunks. Once `</pugi-ask>` lands, the parser slices the tag
|
|
52
|
-
* out, returns the structured record, and the session continues with the
|
|
53
|
-
* post-tag remainder. The same shape applies to `<pugi-plan-review>`.
|
|
54
|
-
*/
|
|
55
|
-
/* ------------------------------------------------------------------ */
|
|
56
|
-
/* Bounded constants */
|
|
57
|
-
/* ------------------------------------------------------------------ */
|
|
58
|
-
/** Hard cap on options per `<pugi-ask>`. the upstream tool AskUserQuestion uses 4. */
|
|
59
|
-
export const ASK_MAX_OPTIONS = 4;
|
|
60
|
-
/** Hard cap on question / label / desc length. Terminal-row budget. */
|
|
61
|
-
export const ASK_MAX_TEXT_LEN = 80;
|
|
62
|
-
/** Hard cap on steps per `<pugi-plan-review>`. */
|
|
63
|
-
export const PLAN_REVIEW_MAX_STEPS = 12;
|
|
64
|
-
/** Hard cap on step text length. */
|
|
65
|
-
export const PLAN_REVIEW_MAX_STEP_LEN = 200;
|
|
66
|
-
/** Hard cap on risk callout length. */
|
|
67
|
-
export const PLAN_REVIEW_MAX_RISK_LEN = 240;
|
|
68
|
-
/** Hard cap on the entire tag span. Long enough for 4 options + risk; defends against runaway payloads. */
|
|
69
|
-
const TAG_MAX_SPAN_BYTES = 8 * 1024;
|
|
70
|
-
/* ------------------------------------------------------------------ */
|
|
71
|
-
/* Public extraction API */
|
|
72
|
-
/* ------------------------------------------------------------------ */
|
|
73
|
-
/**
|
|
74
|
-
* Find every well-formed `<pugi-ask>` in `body`. Malformed tags are
|
|
75
|
-
* dropped and surfaced via `hadMalformedTag` so the session can log a
|
|
76
|
-
* warning. Streaming-incomplete tags (open observed, close not yet
|
|
77
|
-
* arrived) are kept in `cleaned` verbatim and reported via
|
|
78
|
-
* `pendingOpenTag`, so the session keeps buffering until the close tag
|
|
79
|
-
* lands or the turn ends.
|
|
80
|
-
*/
|
|
81
|
-
export function extractAskTags(body) {
|
|
82
|
-
return extractTags(body, {
|
|
83
|
-
openTag: '<pugi-ask>',
|
|
84
|
-
closeTag: '</pugi-ask>',
|
|
85
|
-
parseInner: parseAskInner,
|
|
86
|
-
});
|
|
87
|
-
}
|
|
88
|
-
/**
|
|
89
|
-
* Find every well-formed `<pugi-plan-review>` in `body`. Same contract
|
|
90
|
-
* as `extractAskTags` — malformed → drop + flag, streaming-incomplete →
|
|
91
|
-
* keep in `cleaned` + flag for buffering.
|
|
92
|
-
*/
|
|
93
|
-
export function extractPlanReviewTags(body) {
|
|
94
|
-
return extractTags(body, {
|
|
95
|
-
openTag: '<pugi-plan-review>',
|
|
96
|
-
closeTag: '</pugi-plan-review>',
|
|
97
|
-
parseInner: parsePlanReviewInner,
|
|
98
|
-
});
|
|
99
|
-
}
|
|
100
|
-
function extractTags(body, config) {
|
|
101
|
-
const tags = [];
|
|
102
|
-
const segments = [];
|
|
103
|
-
let cursor = 0;
|
|
104
|
-
let pendingOpenTag = false;
|
|
105
|
-
let hadMalformedTag = false;
|
|
106
|
-
// Hard cap on tags per buffer so a hostile (or accidental) `<pugi-ask>`
|
|
107
|
-
// flood does not pin the parser. 64 is generous — a real session
|
|
108
|
-
// never has more than 1-2 modals queued.
|
|
109
|
-
const MAX_TAGS_PER_BUFFER = 64;
|
|
110
|
-
// Guard counter to prevent a pathological input from looping forever.
|
|
111
|
-
// The body length bounds the iteration count strictly, but we add a
|
|
112
|
-
// belt-and-braces ceiling proportional to body size for clarity.
|
|
113
|
-
let safetyIterations = body.length + 16;
|
|
114
|
-
while (cursor < body.length && tags.length < MAX_TAGS_PER_BUFFER) {
|
|
115
|
-
if (safetyIterations-- <= 0)
|
|
116
|
-
break;
|
|
117
|
-
const openIndex = body.indexOf(config.openTag, cursor);
|
|
118
|
-
if (openIndex === -1) {
|
|
119
|
-
// No more tags. Flush the rest of the body as cleaned prose and exit.
|
|
120
|
-
segments.push(body.slice(cursor));
|
|
121
|
-
cursor = body.length;
|
|
122
|
-
break;
|
|
123
|
-
}
|
|
124
|
-
// Push everything before the opening tag as cleaned prose.
|
|
125
|
-
segments.push(body.slice(cursor, openIndex));
|
|
126
|
-
const closeIndex = body.indexOf(config.closeTag, openIndex + config.openTag.length);
|
|
127
|
-
if (closeIndex === -1) {
|
|
128
|
-
// Open seen, close not yet streamed in. WITHHOLD the partial
|
|
129
|
-
// span from `cleaned` so the raw `<pugi-ask>` envelope cannot
|
|
130
|
-
// leak into the operator-visible transcript if the stream pauses
|
|
131
|
-
// or completes mid-tag. The caller keeps the original buffer for
|
|
132
|
-
// the next chunk merge via pendingOpenTag; if the turn ends with
|
|
133
|
-
// the tag still open, the caller emits a system warning instead
|
|
134
|
-
// of surfacing the raw XML. Codex triple-review P2 (PR).
|
|
135
|
-
pendingOpenTag = true;
|
|
136
|
-
cursor = body.length;
|
|
137
|
-
break;
|
|
138
|
-
}
|
|
139
|
-
const tagEnd = closeIndex + config.closeTag.length;
|
|
140
|
-
const span = body.slice(openIndex, tagEnd);
|
|
141
|
-
if (span.length > TAG_MAX_SPAN_BYTES) {
|
|
142
|
-
hadMalformedTag = true;
|
|
143
|
-
cursor = tagEnd;
|
|
144
|
-
continue;
|
|
145
|
-
}
|
|
146
|
-
// Reject nested open tags inside the span before parsing — the
|
|
147
|
-
// generic "find next close" lookup would otherwise pair an outer
|
|
148
|
-
// open with an inner close, producing a corrupt tag.
|
|
149
|
-
const innerOpen = span.indexOf(config.openTag, config.openTag.length);
|
|
150
|
-
if (innerOpen !== -1) {
|
|
151
|
-
hadMalformedTag = true;
|
|
152
|
-
cursor = tagEnd;
|
|
153
|
-
continue;
|
|
154
|
-
}
|
|
155
|
-
const inner = body.slice(openIndex + config.openTag.length, closeIndex);
|
|
156
|
-
const parsed = config.parseInner(inner, { start: openIndex, end: tagEnd });
|
|
157
|
-
if (parsed === null) {
|
|
158
|
-
hadMalformedTag = true;
|
|
159
|
-
cursor = tagEnd;
|
|
160
|
-
continue;
|
|
161
|
-
}
|
|
162
|
-
tags.push(parsed);
|
|
163
|
-
cursor = tagEnd;
|
|
164
|
-
}
|
|
165
|
-
return {
|
|
166
|
-
tags,
|
|
167
|
-
cleaned: segments.join('').replace(/\s+\n/g, '\n').trimEnd(),
|
|
168
|
-
pendingOpenTag,
|
|
169
|
-
hadMalformedTag,
|
|
170
|
-
};
|
|
171
|
-
}
|
|
172
|
-
/* ------------------------------------------------------------------ */
|
|
173
|
-
/* `<pugi-ask>` inner parser */
|
|
174
|
-
/* ------------------------------------------------------------------ */
|
|
175
|
-
function parseAskInner(inner, span) {
|
|
176
|
-
// Reject CDATA, comments, processing instructions, DOCTYPE.
|
|
177
|
-
if (/<!--|<!\[|<\?|<!DOCTYPE/i.test(inner))
|
|
178
|
-
return null;
|
|
179
|
-
// Reject raw `&` not in a known entity. Decoded entities are allowed
|
|
180
|
-
// below in `decodeEntities`; anything outside the closed allowlist is
|
|
181
|
-
// malformed.
|
|
182
|
-
if (containsRawAmpersand(inner))
|
|
183
|
-
return null;
|
|
184
|
-
const question = extractSingleChildText(inner, 'question');
|
|
185
|
-
if (question === null)
|
|
186
|
-
return null;
|
|
187
|
-
if (question.length === 0 || question.length > ASK_MAX_TEXT_LEN)
|
|
188
|
-
return null;
|
|
189
|
-
const options = extractOptionTags(inner);
|
|
190
|
-
if (options === null)
|
|
191
|
-
return null;
|
|
192
|
-
if (options.length === 0 || options.length > ASK_MAX_OPTIONS)
|
|
193
|
-
return null;
|
|
194
|
-
// Reject duplicate option values — collapses to a single modal entry
|
|
195
|
-
// visually but corrupts the dedupe signature. Easier to refuse than
|
|
196
|
-
// to silently rewrite.
|
|
197
|
-
const seen = new Set();
|
|
198
|
-
for (const opt of options) {
|
|
199
|
-
if (seen.has(opt.value))
|
|
200
|
-
return null;
|
|
201
|
-
seen.add(opt.value);
|
|
202
|
-
}
|
|
203
|
-
const signature = signatureForAsk(question, options);
|
|
204
|
-
return { question, options, signature, start: span.start, end: span.end };
|
|
205
|
-
}
|
|
206
|
-
/**
|
|
207
|
-
* Pull the body of a single `<question>...</question>` child. Returns
|
|
208
|
-
* null when the tag is missing OR when more than one occurrence is
|
|
209
|
-
* found (the grammar mandates exactly one question per ask).
|
|
210
|
-
*/
|
|
211
|
-
function extractSingleChildText(inner, tagName) {
|
|
212
|
-
const open = `<${tagName}>`;
|
|
213
|
-
const close = `</${tagName}>`;
|
|
214
|
-
const first = inner.indexOf(open);
|
|
215
|
-
if (first === -1)
|
|
216
|
-
return null;
|
|
217
|
-
const end = inner.indexOf(close, first + open.length);
|
|
218
|
-
if (end === -1)
|
|
219
|
-
return null;
|
|
220
|
-
// Ensure no second occurrence.
|
|
221
|
-
if (inner.indexOf(open, end + close.length) !== -1)
|
|
222
|
-
return null;
|
|
223
|
-
const body = inner.slice(first + open.length, end);
|
|
224
|
-
// Reject nested tags inside the question body.
|
|
225
|
-
if (/<[^>]/.test(body))
|
|
226
|
-
return null;
|
|
227
|
-
return stripControlChars(decodeEntities(body)).trim();
|
|
228
|
-
}
|
|
229
|
-
/**
|
|
230
|
-
* Pull every self-closing `<option ... />` element from the inner body.
|
|
231
|
-
* Returns null on any malformed option; the caller drops the whole tag
|
|
232
|
-
* rather than partially accept a corrupt option list.
|
|
233
|
-
*/
|
|
234
|
-
function extractOptionTags(inner) {
|
|
235
|
-
const options = [];
|
|
236
|
-
// Match `<option ... />` (self-closing) OR `<option ...></option>`
|
|
237
|
-
// (paired form). The persona prompt teaches the self-closing form
|
|
238
|
-
// for compactness, but a model that emits the paired form should not
|
|
239
|
-
// be punished.
|
|
240
|
-
const re = /<option\b([^>]*?)(\/>|><\/option>)/g;
|
|
241
|
-
let match;
|
|
242
|
-
while ((match = re.exec(inner)) !== null) {
|
|
243
|
-
const attrBlob = match[1] ?? '';
|
|
244
|
-
const parsed = parseOptionAttrs(attrBlob);
|
|
245
|
-
if (parsed === null)
|
|
246
|
-
return null;
|
|
247
|
-
options.push(parsed);
|
|
248
|
-
}
|
|
249
|
-
// Bail if the inner has any `<option` that the regex did NOT match —
|
|
250
|
-
// means an option element was malformed (e.g. unclosed, attribute
|
|
251
|
-
// missing closing quote).
|
|
252
|
-
const stray = inner.match(/<option\b/g);
|
|
253
|
-
if (stray && stray.length !== options.length)
|
|
254
|
-
return null;
|
|
255
|
-
return options;
|
|
256
|
-
}
|
|
257
|
-
function parseOptionAttrs(attrBlob) {
|
|
258
|
-
// Allowed attributes: value, label, desc. Reject any other.
|
|
259
|
-
const attrs = parseAttrBlob(attrBlob);
|
|
260
|
-
if (attrs === null)
|
|
261
|
-
return null;
|
|
262
|
-
for (const key of Object.keys(attrs)) {
|
|
263
|
-
if (key !== 'value' && key !== 'label' && key !== 'desc')
|
|
264
|
-
return null;
|
|
265
|
-
}
|
|
266
|
-
const value = attrs['value'];
|
|
267
|
-
const label = attrs['label'];
|
|
268
|
-
if (typeof value !== 'string' || value.length === 0)
|
|
269
|
-
return null;
|
|
270
|
-
if (typeof label !== 'string' || label.length === 0)
|
|
271
|
-
return null;
|
|
272
|
-
if (value.length > ASK_MAX_TEXT_LEN || label.length > ASK_MAX_TEXT_LEN)
|
|
273
|
-
return null;
|
|
274
|
-
// Forbid quote / angle-bracket characters in value — those would
|
|
275
|
-
// break the operator-side echo "[ASK-RESPONSE:<value>]".
|
|
276
|
-
if (/[<>"'\\]/.test(value))
|
|
277
|
-
return null;
|
|
278
|
-
const desc = attrs['desc'];
|
|
279
|
-
if (desc !== undefined) {
|
|
280
|
-
if (typeof desc !== 'string')
|
|
281
|
-
return null;
|
|
282
|
-
if (desc.length > ASK_MAX_TEXT_LEN)
|
|
283
|
-
return null;
|
|
284
|
-
}
|
|
285
|
-
const opt = { value, label };
|
|
286
|
-
if (desc !== undefined && desc.length > 0)
|
|
287
|
-
opt.desc = desc;
|
|
288
|
-
return opt;
|
|
289
|
-
}
|
|
290
|
-
/**
|
|
291
|
-
* Parse `key="value"` / `key='value'` pairs out of an attribute blob.
|
|
292
|
-
* Returns null on any malformed attribute (unterminated quote, raw
|
|
293
|
-
* entity outside the allowlist, etc).
|
|
294
|
-
*/
|
|
295
|
-
function parseAttrBlob(blob) {
|
|
296
|
-
const trimmed = blob.trim();
|
|
297
|
-
if (trimmed.length === 0)
|
|
298
|
-
return {};
|
|
299
|
-
const result = {};
|
|
300
|
-
// Bound the iteration count: every step must consume at least one
|
|
301
|
-
// attribute and one separator, so the loop terminates in O(blob length).
|
|
302
|
-
let cursor = 0;
|
|
303
|
-
const maxIterations = trimmed.length + 4;
|
|
304
|
-
let iterations = 0;
|
|
305
|
-
while (cursor < trimmed.length && iterations++ < maxIterations) {
|
|
306
|
-
// Skip whitespace.
|
|
307
|
-
while (cursor < trimmed.length && /\s/.test(trimmed[cursor] ?? ''))
|
|
308
|
-
cursor += 1;
|
|
309
|
-
if (cursor >= trimmed.length)
|
|
310
|
-
break;
|
|
311
|
-
// Parse attribute name (lowercase letters only — keeps the grammar tight).
|
|
312
|
-
const nameStart = cursor;
|
|
313
|
-
while (cursor < trimmed.length && /[a-z]/.test(trimmed[cursor] ?? ''))
|
|
314
|
-
cursor += 1;
|
|
315
|
-
if (cursor === nameStart)
|
|
316
|
-
return null;
|
|
317
|
-
const name = trimmed.slice(nameStart, cursor);
|
|
318
|
-
// Expect `=`.
|
|
319
|
-
if (trimmed[cursor] !== '=')
|
|
320
|
-
return null;
|
|
321
|
-
cursor += 1;
|
|
322
|
-
// Expect a quote (single or double).
|
|
323
|
-
const quote = trimmed[cursor];
|
|
324
|
-
if (quote !== '"' && quote !== "'")
|
|
325
|
-
return null;
|
|
326
|
-
cursor += 1;
|
|
327
|
-
const valueStart = cursor;
|
|
328
|
-
const valueEnd = trimmed.indexOf(quote, valueStart);
|
|
329
|
-
if (valueEnd === -1)
|
|
330
|
-
return null;
|
|
331
|
-
const rawValue = trimmed.slice(valueStart, valueEnd);
|
|
332
|
-
if (containsRawAmpersand(rawValue))
|
|
333
|
-
return null;
|
|
334
|
-
if (/[<>]/.test(rawValue))
|
|
335
|
-
return null;
|
|
336
|
-
result[name] = stripControlChars(decodeEntities(rawValue));
|
|
337
|
-
cursor = valueEnd + 1;
|
|
338
|
-
}
|
|
339
|
-
return result;
|
|
340
|
-
}
|
|
341
|
-
/* ------------------------------------------------------------------ */
|
|
342
|
-
/* `<pugi-plan-review>` inner parser */
|
|
343
|
-
/* ------------------------------------------------------------------ */
|
|
344
|
-
function parsePlanReviewInner(inner, span) {
|
|
345
|
-
if (/<!--|<!\[|<\?|<!DOCTYPE/i.test(inner))
|
|
346
|
-
return null;
|
|
347
|
-
if (containsRawAmpersand(inner))
|
|
348
|
-
return null;
|
|
349
|
-
const steps = extractStepTags(inner);
|
|
350
|
-
if (steps === null)
|
|
351
|
-
return null;
|
|
352
|
-
if (steps.length === 0 || steps.length > PLAN_REVIEW_MAX_STEPS)
|
|
353
|
-
return null;
|
|
354
|
-
// Risk is optional and at most one occurrence.
|
|
355
|
-
const risk = extractOptionalChildText(inner, 'risk', PLAN_REVIEW_MAX_RISK_LEN);
|
|
356
|
-
if (risk === undefined)
|
|
357
|
-
return null;
|
|
358
|
-
const sig = signatureForPlanReview(steps, risk);
|
|
359
|
-
const tag = {
|
|
360
|
-
steps,
|
|
361
|
-
signature: sig,
|
|
362
|
-
start: span.start,
|
|
363
|
-
end: span.end,
|
|
364
|
-
};
|
|
365
|
-
if (risk !== null && risk.length > 0) {
|
|
366
|
-
tag.risk = risk;
|
|
367
|
-
}
|
|
368
|
-
return tag;
|
|
369
|
-
}
|
|
370
|
-
function extractStepTags(inner) {
|
|
371
|
-
const re = /<step>([\s\S]*?)<\/step>/g;
|
|
372
|
-
const steps = [];
|
|
373
|
-
let match;
|
|
374
|
-
while ((match = re.exec(inner)) !== null) {
|
|
375
|
-
const raw = match[1] ?? '';
|
|
376
|
-
if (/<[^>]/.test(raw))
|
|
377
|
-
return null;
|
|
378
|
-
const text = stripControlChars(decodeEntities(raw)).trim();
|
|
379
|
-
if (text.length === 0 || text.length > PLAN_REVIEW_MAX_STEP_LEN)
|
|
380
|
-
return null;
|
|
381
|
-
steps.push({ text });
|
|
382
|
-
}
|
|
383
|
-
// Detect unbalanced `<step>` occurrences (open with no close, etc).
|
|
384
|
-
const opens = (inner.match(/<step>/g) ?? []).length;
|
|
385
|
-
const closes = (inner.match(/<\/step>/g) ?? []).length;
|
|
386
|
-
if (opens !== steps.length || closes !== steps.length)
|
|
387
|
-
return null;
|
|
388
|
-
return steps;
|
|
389
|
-
}
|
|
390
|
-
/**
|
|
391
|
-
* Optional single-child text. Returns:
|
|
392
|
-
* - `null` when the tag is absent (legal absence)
|
|
393
|
-
* - the trimmed body when exactly one occurrence is present
|
|
394
|
-
* - `undefined` to signal a malformed (rejected) state to the caller
|
|
395
|
-
*/
|
|
396
|
-
function extractOptionalChildText(inner, tagName, maxLen) {
|
|
397
|
-
const open = `<${tagName}>`;
|
|
398
|
-
const close = `</${tagName}>`;
|
|
399
|
-
const first = inner.indexOf(open);
|
|
400
|
-
if (first === -1)
|
|
401
|
-
return null;
|
|
402
|
-
const end = inner.indexOf(close, first + open.length);
|
|
403
|
-
if (end === -1)
|
|
404
|
-
return undefined;
|
|
405
|
-
if (inner.indexOf(open, end + close.length) !== -1)
|
|
406
|
-
return undefined;
|
|
407
|
-
const body = inner.slice(first + open.length, end);
|
|
408
|
-
if (/<[^>]/.test(body))
|
|
409
|
-
return undefined;
|
|
410
|
-
const decoded = stripControlChars(decodeEntities(body)).trim();
|
|
411
|
-
if (decoded.length > maxLen)
|
|
412
|
-
return undefined;
|
|
413
|
-
return decoded;
|
|
414
|
-
}
|
|
415
|
-
/* ------------------------------------------------------------------ */
|
|
416
|
-
/* Entity decoding + amp safety */
|
|
417
|
-
/* ------------------------------------------------------------------ */
|
|
418
|
-
const ENTITY_MAP = Object.freeze({
|
|
419
|
-
amp: '&',
|
|
420
|
-
lt: '<',
|
|
421
|
-
gt: '>',
|
|
422
|
-
quot: '"',
|
|
423
|
-
apos: "'",
|
|
424
|
-
});
|
|
425
|
-
function decodeEntities(input) {
|
|
426
|
-
return input.replace(/&([a-z]+);/g, (whole, name) => {
|
|
427
|
-
const decoded = ENTITY_MAP[name];
|
|
428
|
-
return decoded === undefined ? whole : decoded;
|
|
429
|
-
});
|
|
430
|
-
}
|
|
431
|
-
/**
|
|
432
|
-
* Strip C0 control characters (except \t \n \r), DEL, and C1 control
|
|
433
|
-
* characters from decoded text. The persona's grammar is plain ASCII
|
|
434
|
-
* prose; raw control codes are either accidental noise or a deliberate
|
|
435
|
-
* ANSI-escape injection attempt that would corrupt the Ink render.
|
|
436
|
-
*
|
|
437
|
-
* Applied AFTER decodeEntities so an `&` escape cannot smuggle an ESC
|
|
438
|
-
* byte through the entity layer.
|
|
439
|
-
*
|
|
440
|
-
* Claude triple-review P2 (PR).
|
|
441
|
-
*/
|
|
442
|
-
function stripControlChars(input) {
|
|
443
|
-
// eslint-disable-next-line no-control-regex -- deliberately matching control range
|
|
444
|
-
return input.replace(/[\x00-\x08\x0b\x0c\x0e-\x1f\x7f\x80-\x9f]/g, '');
|
|
445
|
-
}
|
|
446
|
-
/**
|
|
447
|
-
* Reject any `&` that is not the head of a recognised entity. This is
|
|
448
|
-
* the defence-in-depth check from the module header: a raw `&` is
|
|
449
|
-
* legal in HTML but malformed in XML, and the persona is taught to
|
|
450
|
-
* always escape via `&`. Anything outside the allowlist is a sign
|
|
451
|
-
* of either accidental sloppy output or an attempted injection.
|
|
452
|
-
*/
|
|
453
|
-
function containsRawAmpersand(input) {
|
|
454
|
-
// Scan once; a recognised entity is `&` + name + `;`. Any `&` that is
|
|
455
|
-
// not followed by an allowed name + `;` is malformed.
|
|
456
|
-
for (let i = 0; i < input.length; i += 1) {
|
|
457
|
-
if (input[i] !== '&')
|
|
458
|
-
continue;
|
|
459
|
-
const semi = input.indexOf(';', i + 1);
|
|
460
|
-
if (semi === -1)
|
|
461
|
-
return true;
|
|
462
|
-
const name = input.slice(i + 1, semi);
|
|
463
|
-
if (!Object.prototype.hasOwnProperty.call(ENTITY_MAP, name))
|
|
464
|
-
return true;
|
|
465
|
-
i = semi;
|
|
466
|
-
}
|
|
467
|
-
return false;
|
|
468
|
-
}
|
|
469
|
-
/* ------------------------------------------------------------------ */
|
|
470
|
-
/* Signatures */
|
|
471
|
-
/* ------------------------------------------------------------------ */
|
|
472
|
-
/**
|
|
473
|
-
* Stable dedupe signature for an ask tag. The session keeps the most
|
|
474
|
-
* recent N signatures in a rolling set so a retry-spammed identical
|
|
475
|
-
* tag does not stack two modals.
|
|
476
|
-
*
|
|
477
|
-
* Algorithm: SHA-256-like collision-resistance is overkill for a
|
|
478
|
-
* single-process modal dedupe; a deterministic lower-case string is
|
|
479
|
-
* enough. We use the literal question + sorted-value join, then base64
|
|
480
|
-
* it for a compact identifier.
|
|
481
|
-
*/
|
|
482
|
-
/**
|
|
483
|
-
* Stable dedupe signature for an ask tag. Exported so synthesisers
|
|
484
|
-
* outside the parser (slash-command `/ask`, `pugi ask` shell command)
|
|
485
|
-
* can produce signatures that collision-match parser-produced
|
|
486
|
-
* signatures. Without a single source of truth, an `<pugi-ask>`
|
|
487
|
-
* emitted by the persona with the same question + same option values
|
|
488
|
-
* could share a signature with a local synthesiser even though the
|
|
489
|
-
* computation diverged - dedupe would suppress the persona modal.
|
|
490
|
-
*
|
|
491
|
-
* Claude triple-review P1 (PR).
|
|
492
|
-
*/
|
|
493
|
-
export function signatureForAsk(question, options) {
|
|
494
|
-
const values = options.map((o) => o.value).sort().join('|');
|
|
495
|
-
const raw = `${question.trim().toLowerCase()}::${values}`;
|
|
496
|
-
return Buffer.from(raw, 'utf8').toString('base64');
|
|
497
|
-
}
|
|
498
|
-
/**
|
|
499
|
-
* Stable dedupe signature for a plan-review tag. Exported for the
|
|
500
|
-
* same reason as signatureForAsk: synthesisers like
|
|
501
|
-
* `synthesiseLocalPlanReview` in cli.ts must produce identical
|
|
502
|
-
* signatures to parser-extracted tags for the same logical input.
|
|
503
|
-
*
|
|
504
|
-
* Codex triple-review P2 (PR).
|
|
505
|
-
*/
|
|
506
|
-
export function signatureForPlanReview(steps, risk) {
|
|
507
|
-
const stepsJoined = steps.map((s) => s.text.trim()).join('|');
|
|
508
|
-
const riskJoined = risk ?? '';
|
|
509
|
-
const raw = `${stepsJoined}::${riskJoined}`;
|
|
510
|
-
return Buffer.from(raw, 'utf8').toString('base64');
|
|
511
|
-
}
|
|
512
|
-
//# sourceMappingURL=ask.js.map
|
|
@@ -1,98 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Cancellation token — Sprint Phase 1 (agent loop FSM + cancellation).
|
|
3
|
-
*
|
|
4
|
-
* A pure-JS one-shot signal that fans out to N listeners. One token is
|
|
5
|
-
* minted per dispatch turn (fresh on each operator brief); when the
|
|
6
|
-
* operator hits Ctrl+C, the REPL calls `abort()` which:
|
|
7
|
-
*
|
|
8
|
-
* 1. Latches `aborted = true` so any future `isAborted` check returns
|
|
9
|
-
* true (a tool that observes the flag mid-execution can short-circuit
|
|
10
|
-
* its loop without waiting for an explicit signal-handler callback).
|
|
11
|
-
* 2. Drains the listener set, firing each callback exactly once. The
|
|
12
|
-
* set is cleared after the drain so a late `onAbort` listener
|
|
13
|
-
* attached AFTER abort does NOT fire — that is the documented
|
|
14
|
-
* contract; late listeners are expected to check `isAborted`
|
|
15
|
-
* explicitly at registration time if they need to know whether the
|
|
16
|
-
* token already tripped.
|
|
17
|
-
*
|
|
18
|
-
* Design choices:
|
|
19
|
-
*
|
|
20
|
-
* - No coupling to `AbortController` / `AbortSignal`. The session +
|
|
21
|
-
* tool path are wrapped around this token; where a Web platform
|
|
22
|
-
* primitive is needed (fetch signal, MCP call), the wrapper bridges
|
|
23
|
-
* `onAbort` → `controller.abort()` at the seam.
|
|
24
|
-
* - Idempotent `abort()`. Calling twice is safe and the second call is
|
|
25
|
-
* a no-op (the listener set is already empty). This matters because
|
|
26
|
-
* two code paths can race to cancel — the Ctrl+C handler and a
|
|
27
|
-
* downstream tool that observed `isAborted` and threw — and both
|
|
28
|
-
* end up calling `dispatch.cancel()` which transitively calls
|
|
29
|
-
* `token.abort()`.
|
|
30
|
-
* - Listener errors do NOT block the drain. A throwing listener
|
|
31
|
-
* stops itself but the next listener still fires. The error is
|
|
32
|
-
* swallowed because the cancellation path is best-effort and
|
|
33
|
-
* surfacing the error mid-drain would leak through the abort
|
|
34
|
-
* pathway into the REPL state (a UI rerender on a half-aborted
|
|
35
|
-
* session is worse than a silent listener crash).
|
|
36
|
-
*
|
|
37
|
-
* Brand voice: no forbidden words. ASCII only. No emoji.
|
|
38
|
-
*/
|
|
39
|
-
export class CancellationToken {
|
|
40
|
-
aborted = false;
|
|
41
|
-
listeners = new Set();
|
|
42
|
-
/**
|
|
43
|
-
* Latch the token to aborted and fire every currently-attached
|
|
44
|
-
* listener exactly once. Subsequent `abort()` calls are no-ops
|
|
45
|
-
* (idempotent — the listener set was already cleared on first abort).
|
|
46
|
-
* Listener callbacks that throw are swallowed; the next listener
|
|
47
|
-
* still fires.
|
|
48
|
-
*/
|
|
49
|
-
abort() {
|
|
50
|
-
if (this.aborted)
|
|
51
|
-
return;
|
|
52
|
-
this.aborted = true;
|
|
53
|
-
// Snapshot the listener set so a listener that mutates the set
|
|
54
|
-
// (e.g. detaches itself via the returned unsubscribe handle while
|
|
55
|
-
// its own callback is running) does not corrupt the iteration.
|
|
56
|
-
const snapshot = Array.from(this.listeners);
|
|
57
|
-
this.listeners.clear();
|
|
58
|
-
for (const listener of snapshot) {
|
|
59
|
-
try {
|
|
60
|
-
listener();
|
|
61
|
-
}
|
|
62
|
-
catch {
|
|
63
|
-
// Swallow listener errors — see header comment for rationale.
|
|
64
|
-
}
|
|
65
|
-
}
|
|
66
|
-
}
|
|
67
|
-
/**
|
|
68
|
-
* True after `abort()` has been called. Mutation observers and tool
|
|
69
|
-
* inner loops should read this BEFORE each potentially-expensive
|
|
70
|
-
* iteration so they can short-circuit on cancel.
|
|
71
|
-
*/
|
|
72
|
-
get isAborted() {
|
|
73
|
-
return this.aborted;
|
|
74
|
-
}
|
|
75
|
-
/**
|
|
76
|
-
* Register a callback that fires on the FIRST `abort()` call. If the
|
|
77
|
-
* token has already aborted at registration time, the callback is
|
|
78
|
-
* NOT auto-fired — the caller is responsible for checking
|
|
79
|
-
* `isAborted` first.
|
|
80
|
-
*
|
|
81
|
-
* Returns an unsubscribe handle. Calling it before abort detaches the
|
|
82
|
-
* listener so it never fires; calling it after abort is a no-op (the
|
|
83
|
-
* set was already drained).
|
|
84
|
-
*/
|
|
85
|
-
onAbort(listener) {
|
|
86
|
-
if (this.aborted) {
|
|
87
|
-
// Document the contract by returning a no-op unsubscribe handle.
|
|
88
|
-
// The listener does NOT fire — late subscribers must check
|
|
89
|
-
// isAborted at registration time.
|
|
90
|
-
return () => undefined;
|
|
91
|
-
}
|
|
92
|
-
this.listeners.add(listener);
|
|
93
|
-
return () => {
|
|
94
|
-
this.listeners.delete(listener);
|
|
95
|
-
};
|
|
96
|
-
}
|
|
97
|
-
}
|
|
98
|
-
//# sourceMappingURL=cancellation.js.map
|