@pugi/cli 0.1.0-beta.10 → 0.1.0-beta.101
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +132 -0
- package/LICENSE +1 -1
- package/README.md +55 -11
- package/assets/pugi-prozr2-mascot.ansi +9 -0
- package/bin/run.js +33 -1
- package/dist/commands/deploy.js +40 -40
- package/dist/commands/flatten.js +191 -0
- package/dist/commands/jobs-watch.js +201 -0
- package/dist/commands/jobs.js +42 -27
- package/dist/commands/retro.js +210 -0
- package/dist/commands/smoke.js +133 -0
- package/dist/core/agent-progress/cleanup.js +134 -0
- package/dist/core/agent-progress/schema.js +144 -0
- package/dist/core/agent-progress/writer.js +101 -0
- package/dist/core/agents/adaptive-router.js +330 -0
- package/dist/core/agents/query-decomposer.js +297 -0
- package/dist/core/agents/registry.js +3 -3
- package/dist/core/approvals/shortcut-resolver.js +98 -0
- package/dist/core/artifact-chain/dispatcher.js +148 -0
- package/dist/core/artifact-chain/exporter.js +164 -0
- package/dist/core/artifact-chain/state.js +243 -0
- package/dist/core/artifact-chain/steps.js +169 -0
- package/dist/core/ask-user/question.js +92 -0
- package/dist/core/audit/audit-trail.js +275 -0
- package/dist/core/auth/ensure-authenticated.js +129 -0
- package/dist/core/auth/env-provider.js +238 -0
- package/dist/core/auto-open-browser.js +4 -4
- package/dist/core/auto-update/channels.js +122 -0
- package/dist/core/auto-update/checker.js +241 -0
- package/dist/core/auto-update/state.js +235 -0
- package/dist/core/bare-mode/index.js +107 -0
- package/dist/core/bash/redirect.js +281 -0
- package/dist/core/bash-classifier.js +436 -40
- package/dist/core/checkpoint/resumer.js +149 -0
- package/dist/core/checkpoint/rewinder.js +291 -0
- package/dist/core/checkpoints/shadow-git.js +670 -0
- package/dist/core/citations/parser.js +109 -0
- package/dist/core/classifier/yolo-classifier.js +88 -0
- package/dist/core/codegraph/db.js +506 -0
- package/dist/core/codegraph/decision-store.js +248 -0
- package/dist/core/codegraph/detect-repo.js +459 -0
- package/dist/core/codegraph/install.js +134 -0
- package/dist/core/codegraph/offer-hook.js +220 -0
- package/dist/core/codegraph/parser.js +598 -0
- package/dist/core/codegraph/queries/go.scm +57 -0
- package/dist/core/codegraph/queries/javascript.scm +56 -0
- package/dist/core/codegraph/queries/python.scm +55 -0
- package/dist/core/codegraph/queries/rust.scm +63 -0
- package/dist/core/codegraph/queries/typescript.scm +91 -0
- package/dist/core/codegraph/reindex.js +218 -0
- package/dist/core/codegraph/resolve-edges.js +107 -0
- package/dist/core/codegraph/types.js +34 -0
- package/dist/core/codegraph/watcher.js +440 -0
- package/dist/core/compact/auto-trigger.js +96 -0
- package/dist/core/compact/buffer-rewriter.js +115 -0
- package/dist/core/compact/summarizer.js +208 -0
- package/dist/core/compact/token-counter.js +108 -0
- package/dist/core/consensus/anvil-fanout.js +25 -25
- package/dist/core/consensus/diff-capture.js +121 -12
- package/dist/core/consensus/rubric.js +21 -21
- package/dist/core/context/builder.js +6 -6
- package/dist/core/context/compaction-events.js +8 -8
- package/dist/core/context/compaction.js +31 -31
- package/dist/core/context/index.js +15 -8
- package/dist/core/context/invariants.js +51 -51
- package/dist/core/context/markdown-loader.js +28 -10
- package/dist/core/context/markdown-traverse.js +255 -0
- package/dist/core/context/pugiignore.js +41 -41
- package/dist/core/context/repo-skeleton.js +37 -37
- package/dist/core/context/tool-eviction.js +55 -0
- package/dist/core/context/watcher.js +32 -32
- package/dist/core/context/working-set.js +23 -23
- package/dist/core/coordinator/agent-tools.js +77 -0
- package/dist/core/coordinator/agent-toolset.js +65 -0
- package/dist/core/coordinator/fsm.js +73 -0
- package/dist/core/coordinator/mode-fsm.js +70 -0
- package/dist/core/cost/rate-card.js +129 -0
- package/dist/core/cost/tracker.js +221 -0
- package/dist/core/credentials.js +13 -13
- package/dist/core/cron/scheduler.js +138 -0
- package/dist/core/denial-tracking/index.js +8 -0
- package/dist/core/denial-tracking/state.js +264 -0
- package/dist/core/diagnostics/probe-runner.js +93 -0
- package/dist/core/diagnostics/probes/api.js +46 -0
- package/dist/core/diagnostics/probes/auth.js +93 -0
- package/dist/core/diagnostics/probes/bare-mode.js +42 -0
- package/dist/core/diagnostics/probes/cli-version.js +127 -0
- package/dist/core/diagnostics/probes/config.js +72 -0
- package/dist/core/diagnostics/probes/denial-tracking.js +57 -0
- package/dist/core/diagnostics/probes/disk.js +81 -0
- package/dist/core/diagnostics/probes/engine-live.js +46 -0
- package/dist/core/diagnostics/probes/git.js +65 -0
- package/dist/core/diagnostics/probes/hooks.js +118 -0
- package/dist/core/diagnostics/probes/mcp.js +75 -0
- package/dist/core/diagnostics/probes/node.js +59 -0
- package/dist/core/diagnostics/probes/pnpm.js +36 -0
- package/dist/core/diagnostics/probes/pugi-md.js +89 -0
- package/dist/core/diagnostics/probes/sandbox.js +67 -0
- package/dist/core/diagnostics/probes/session.js +74 -0
- package/dist/core/diagnostics/probes/status-snapshot.js +488 -0
- package/dist/core/diagnostics/probes/workspace.js +63 -0
- package/dist/core/diagnostics/types.js +70 -0
- package/dist/core/dispatch/cache-cleanup.js +197 -0
- package/dist/core/dispatch/cache-handoff.js +295 -0
- package/dist/core/edits/apply-patch-layer-e.js +189 -0
- package/dist/core/edits/dispatch.js +333 -7
- package/dist/core/edits/format-detector.js +260 -0
- package/dist/core/edits/format-matrix.js +26 -0
- package/dist/core/edits/fuzzy-ladder.js +650 -0
- package/dist/core/edits/index.js +5 -1
- package/dist/core/edits/journal.js +199 -0
- package/dist/core/edits/layer-a-apply.js +15 -15
- package/dist/core/edits/layer-a-fuzzy-apply.js +198 -0
- package/dist/core/edits/layer-b-apply.js +9 -9
- package/dist/core/edits/layer-c-apply.js +6 -6
- package/dist/core/edits/layer-d-ast.js +557 -14
- package/dist/core/edits/marker-parser.js +12 -12
- package/dist/core/edits/security-gate.js +27 -27
- package/dist/core/edits/verify-hook.js +273 -0
- package/dist/core/edits/worktree.js +29 -29
- package/dist/core/engine/anvil-client.js +214 -26
- package/dist/core/engine/auto-compact.js +247 -0
- package/dist/core/engine/budgets.js +220 -0
- package/dist/core/engine/compact-llm-summarizer.js +124 -0
- package/dist/core/engine/context-prefix.js +155 -0
- package/dist/core/engine/index.js +1 -1
- package/dist/core/engine/intensity.js +163 -0
- package/dist/core/engine/intent.js +260 -0
- package/dist/core/engine/native-pugi.js +1559 -227
- package/dist/core/engine/prompts.js +219 -19
- package/dist/core/engine/strip-internal-fields.js +124 -0
- package/dist/core/engine/tool-bridge.js +1887 -59
- package/dist/core/engine/verification-patterns.js +195 -0
- package/dist/core/eval/v1/ledger.js +83 -0
- package/dist/core/eval/v1/runner.js +280 -0
- package/dist/core/eval/v1/scoring.js +68 -0
- package/dist/core/eval/v1/task-loader.js +191 -0
- package/dist/core/eval/v1/types.js +14 -0
- package/dist/core/eval/v1/verifier.js +176 -0
- package/dist/core/eval/v1/yaml-parser.js +250 -0
- package/dist/core/evaluation/golden-dataset.js +293 -0
- package/dist/core/feedback/queue.js +177 -0
- package/dist/core/feedback/submitter.js +145 -0
- package/dist/core/file-cache.js +113 -1
- package/dist/core/flatten/flatten-repo.js +439 -0
- package/dist/core/format/osc8-link.js +28 -0
- package/dist/core/hook-chains.js +392 -0
- package/dist/core/hooks/citation-verify-hook.js +138 -0
- package/dist/core/hooks/citation-verify.js +112 -0
- package/dist/core/hooks/events.js +46 -0
- package/dist/core/hooks/index.js +15 -0
- package/dist/core/hooks/registry.js +216 -0
- package/dist/core/hooks/runner.js +236 -0
- package/dist/core/hooks/v2/event-emitter.js +115 -0
- package/dist/core/hooks/v2/executor.js +282 -0
- package/dist/core/hooks/v2/index.js +25 -0
- package/dist/core/hooks/v2/lifecycle.js +104 -0
- package/dist/core/hooks/v2/loader.js +216 -0
- package/dist/core/hooks/v2/matcher.js +125 -0
- package/dist/core/hooks/v2/trust.js +143 -0
- package/dist/core/hooks/v2/types.js +86 -0
- package/dist/core/hooks/worktree-events.js +158 -0
- package/dist/core/image/renderer.js +71 -0
- package/dist/core/init/detector.js +582 -0
- package/dist/core/init/template-renderer.js +242 -0
- package/dist/core/jobs/registry.js +18 -18
- package/dist/core/ledger/results-tsv.js +142 -0
- package/dist/core/log-discipline/stdout-redirect.js +51 -0
- package/dist/core/lsp/cache.js +105 -0
- package/dist/core/lsp/client.js +551 -41
- package/dist/core/lsp/language-detect.js +66 -0
- package/dist/core/lsp/post-edit-diagnostics.js +171 -0
- package/dist/core/lsp/server-detect.js +173 -0
- package/dist/core/lsp/symbol-cache.js +162 -0
- package/dist/core/lsp/symbol-tools.js +664 -0
- package/dist/core/mcp/client.js +97 -28
- package/dist/core/mcp/http-server.js +553 -0
- package/dist/core/mcp/orchestrator-config.js +192 -0
- package/dist/core/mcp/orchestrator-tools.js +806 -0
- package/dist/core/mcp/permission.js +190 -0
- package/dist/core/mcp/registry.js +39 -17
- package/dist/core/mcp/server-tools.js +219 -0
- package/dist/core/mcp/server.js +397 -0
- package/dist/core/mcp/trust.js +10 -10
- package/dist/core/memory/dual-write.js +416 -0
- package/dist/core/memory/passive-extract.js +130 -0
- package/dist/core/memory/phase1-kinds.js +20 -0
- package/dist/core/memory/secret-scanner.js +304 -0
- package/dist/core/memory-sync/queue.js +170 -0
- package/dist/core/metrics/extract.js +113 -0
- package/dist/core/modes/roo-modes.js +68 -0
- package/dist/core/notes/notes-paths.js +113 -0
- package/dist/core/notes/notes-recorder.js +140 -0
- package/dist/core/notes/notes-writer.js +53 -0
- package/dist/core/notes/renderers.js +0 -0
- package/dist/core/notes/slug.js +105 -0
- package/dist/core/onboarding/ensure-initialized.js +133 -0
- package/dist/core/onboarding/marker.js +111 -0
- package/dist/core/onboarding/telemetry-state.js +108 -0
- package/dist/core/output-style/presets.js +176 -0
- package/dist/core/output-style/state.js +185 -0
- package/dist/core/path-security.js +287 -5
- package/dist/core/permission.js +82 -22
- package/dist/core/permissions/auto-classifier.js +124 -0
- package/dist/core/permissions/bash-parser.js +371 -0
- package/dist/core/permissions/circuit-breaker.js +83 -0
- package/dist/core/permissions/constrained-edit.js +91 -0
- package/dist/core/permissions/gate.js +278 -0
- package/dist/core/permissions/index.js +20 -0
- package/dist/core/permissions/mode.js +174 -0
- package/dist/core/permissions/network-egress.js +137 -0
- package/dist/core/permissions/state.js +241 -0
- package/dist/core/permissions/tool-class.js +107 -0
- package/dist/core/plan-mode/ui-state.js +51 -0
- package/dist/core/plans/plan-artifact.js +721 -0
- package/dist/core/policy-limits/etag-store.js +122 -0
- package/dist/core/prd-check/parser.js +215 -0
- package/dist/core/prd-check/reporter.js +127 -0
- package/dist/core/prd-check/session-review.js +557 -0
- package/dist/core/prd-check/verifiers.js +223 -0
- package/dist/core/prompt-cache/client-cache.js +99 -0
- package/dist/core/prompts/assembly.js +29 -0
- package/dist/core/prompts/registry.js +364 -0
- package/dist/core/pugi-gitignore.js +52 -0
- package/dist/core/pugi-md/cc-compat-rules.js +735 -0
- package/dist/core/pugi-md/context-injector.js +76 -0
- package/dist/core/pugi-md/walk-up.js +207 -0
- package/dist/core/python/uv-installer.js +270 -0
- package/dist/core/python/uv-resolver.js +83 -0
- package/dist/core/rate-limit/narrator.js +146 -0
- package/dist/core/recipes/cli-types.js +20 -0
- package/dist/core/recipes/loader.js +103 -0
- package/dist/core/recipes/runner.js +345 -0
- package/dist/core/recipes/schema.js +587 -0
- package/dist/core/release-notes/parser.js +241 -0
- package/dist/core/release-notes/state.js +116 -0
- package/dist/core/repl/ask.js +37 -37
- package/dist/core/repl/cancellation.js +26 -26
- package/dist/core/repl/cap-warning.js +4 -4
- package/dist/core/repl/clipboard-read.js +11 -11
- package/dist/core/repl/dispatch-fsm.js +12 -12
- package/dist/core/repl/engine-bridge.js +303 -0
- package/dist/core/repl/history-search.js +15 -15
- package/dist/core/repl/history.js +28 -18
- package/dist/core/repl/kill-ring.js +5 -5
- package/dist/core/repl/model-pricing.js +135 -0
- package/dist/core/repl/privacy-banner.js +22 -22
- package/dist/core/repl/session.js +2690 -229
- package/dist/core/repl/slash-commands.js +540 -41
- package/dist/core/repl/store/index.js +1 -1
- package/dist/core/repl/store/jsonl-log.js +22 -22
- package/dist/core/repl/store/lockfile.js +10 -10
- package/dist/core/repl/store/session-store.js +136 -107
- package/dist/core/repl/store/types.js +15 -15
- package/dist/core/repl/store/uuid-v7.js +12 -12
- package/dist/core/repl/tool-route.js +382 -0
- package/dist/core/repl/workspace-context.js +43 -21
- package/dist/core/repo-map/build.js +125 -0
- package/dist/core/repo-map/cache.js +185 -0
- package/dist/core/repo-map/extractor.js +254 -0
- package/dist/core/repo-map/formatter.js +145 -0
- package/dist/core/repo-map/page-rank.js +105 -0
- package/dist/core/repo-map/scanner.js +211 -0
- package/dist/core/retro/git-collector.js +251 -0
- package/dist/core/retro/health-card.js +25 -0
- package/dist/core/retro/metrics.js +342 -0
- package/dist/core/retro/narrative.js +249 -0
- package/dist/core/retro/plane-collector.js +274 -0
- package/dist/core/retro/pr-issue-link.js +65 -0
- package/dist/core/retro/types.js +16 -0
- package/dist/core/retry-budget/budget.js +284 -0
- package/dist/core/retry-budget/index.js +5 -0
- package/dist/core/retry-budget/retry-cap.js +74 -0
- package/dist/core/routing/lead-worker.js +43 -0
- package/dist/core/routing/pre-flight-estimator.js +108 -0
- package/dist/core/runs/run-tree.js +103 -0
- package/dist/core/sandboxing/adapter.js +43 -0
- package/dist/core/sandboxing/bubblewrap.js +209 -0
- package/dist/core/sandboxing/index.js +78 -0
- package/dist/core/sandboxing/none.js +19 -0
- package/dist/core/sandboxing/policy.js +97 -0
- package/dist/core/sandboxing/seatbelt.js +231 -0
- package/dist/core/security/injection-scanner.js +367 -0
- package/dist/core/security/output-filter.js +418 -0
- package/dist/core/session/env-file.js +105 -0
- package/dist/core/session/section-budgets.js +140 -0
- package/dist/core/session.js +119 -0
- package/dist/core/settings.js +402 -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 +30 -30
- package/dist/core/skills/loader.js +22 -22
- package/dist/core/skills/sources.js +27 -27
- package/dist/core/smoke/headless-driver.js +174 -0
- package/dist/core/smoke/orchestrator.js +194 -0
- package/dist/core/smoke/runner.js +238 -0
- package/dist/core/smoke/scenario-parser.js +316 -0
- package/dist/core/statusline.js +99 -0
- package/dist/core/subagents/dispatcher-real.js +600 -0
- package/dist/core/subagents/dispatcher.js +146 -52
- package/dist/core/subagents/index.js +19 -6
- package/dist/core/subagents/isolation-matrix.js +213 -0
- package/dist/core/subagents/spawn.js +19 -4
- package/dist/core/telemetry/emitter.js +229 -0
- package/dist/core/telemetry/queue.js +251 -0
- package/dist/core/theme/context.js +91 -0
- package/dist/core/theme/presets.js +228 -0
- package/dist/core/theme/state.js +181 -0
- package/dist/core/todos/invariant.js +10 -0
- package/dist/core/todos/state.js +177 -0
- package/dist/core/tool-schema/compressor.js +89 -0
- package/dist/core/transport/version-interceptor.js +166 -0
- package/dist/core/trust.js +2 -2
- package/dist/core/tui/thinking-block.js +64 -0
- package/dist/core/vim/keymap.js +288 -0
- package/dist/core/vim/state.js +92 -0
- package/dist/core/watch-markers/marker-watcher.js +133 -0
- package/dist/core/worktree/include-parser.js +249 -0
- package/dist/core/worktree-manager/cleanup.js +123 -0
- package/dist/core/worktree-manager/manager.js +303 -0
- package/dist/index.js +36 -0
- package/dist/runtime/bootstrap.js +190 -0
- package/dist/runtime/cli.js +4403 -561
- package/dist/runtime/commands/agents.js +31 -31
- package/dist/runtime/commands/budget.js +5 -5
- package/dist/runtime/commands/cancel.js +231 -0
- package/dist/runtime/commands/chain.js +489 -0
- package/dist/runtime/commands/codegraph-status.js +227 -0
- package/dist/runtime/commands/compact.js +297 -0
- package/dist/runtime/commands/config.js +74 -40
- package/dist/runtime/commands/cost.js +199 -0
- package/dist/runtime/commands/delegate.js +27 -4
- package/dist/runtime/commands/dispatch.js +126 -0
- package/dist/runtime/commands/doctor.js +579 -0
- package/dist/runtime/commands/eval-v1.js +266 -0
- package/dist/runtime/commands/feedback.js +184 -0
- package/dist/runtime/commands/hooks.js +187 -0
- package/dist/runtime/commands/index-cmd.js +459 -0
- package/dist/runtime/commands/init.js +254 -0
- package/dist/runtime/commands/lsp.js +200 -38
- package/dist/runtime/commands/mcp.js +935 -0
- package/dist/runtime/commands/memory.js +582 -0
- package/dist/runtime/commands/model.js +237 -0
- package/dist/runtime/commands/onboarding.js +275 -0
- package/dist/runtime/commands/patch.js +12 -12
- 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/servers-cli.js +182 -0
- package/dist/runtime/commands/servers.js +236 -0
- package/dist/runtime/commands/sessions.js +163 -0
- package/dist/runtime/commands/share.js +316 -0
- package/dist/runtime/commands/skills.js +31 -31
- package/dist/runtime/commands/status.js +186 -0
- package/dist/runtime/commands/stickers.js +82 -0
- package/dist/runtime/commands/style.js +194 -0
- package/dist/runtime/commands/theme.js +196 -0
- package/dist/runtime/commands/undo.js +54 -22
- package/dist/runtime/commands/update.js +289 -0
- package/dist/runtime/commands/vim.js +140 -0
- package/dist/runtime/commands/worktree.js +8 -8
- package/dist/runtime/commands/worktrees.js +155 -0
- package/dist/runtime/deprecation-warning.js +69 -0
- package/dist/runtime/engine-exit-code.js +50 -0
- package/dist/runtime/headless-repl.js +195 -0
- package/dist/runtime/headless.js +548 -0
- package/dist/runtime/load-hooks-or-exit.js +71 -0
- package/dist/runtime/plan-decompose.js +22 -22
- package/dist/runtime/sigint-guard.js +272 -0
- package/dist/runtime/stream-renderer.js +195 -0
- package/dist/runtime/update-check.js +28 -28
- package/dist/runtime/version.js +65 -0
- package/dist/runtime/worktree-bootstrap.js +579 -0
- package/dist/skills/bundled/batch.js +617 -0
- package/dist/skills/bundled/index.js +45 -0
- package/dist/skills/bundled/loop.js +358 -0
- package/dist/skills/bundled/remember.js +383 -0
- package/dist/skills/bundled/simplify.js +289 -0
- package/dist/skills/bundled/skillify.js +373 -0
- package/dist/skills/bundled/stuck.js +558 -0
- package/dist/skills/bundled/verify.js +439 -0
- package/dist/testing/vcr.js +486 -0
- package/dist/tools/agent-tool.js +229 -0
- package/dist/tools/apply-patch.js +89 -28
- package/dist/tools/ask-user-question.js +337 -0
- package/dist/tools/ask-user.js +115 -0
- package/dist/tools/bash.js +811 -49
- package/dist/tools/brief.js +224 -0
- package/dist/tools/cron.js +433 -0
- package/dist/tools/enter-worktree.js +250 -0
- package/dist/tools/exit-worktree.js +147 -0
- package/dist/tools/file-tools.js +161 -44
- package/dist/tools/http-request.js +336 -0
- package/dist/tools/lsp-tools.js +377 -1
- 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 +120 -5
- package/dist/tools/server-tools.js +892 -0
- package/dist/tools/skill-tool.js +96 -0
- package/dist/tools/sleep.js +99 -0
- package/dist/tools/synthetic-output.js +133 -0
- package/dist/tools/tasks.js +208 -0
- package/dist/tools/todo-write.js +184 -0
- package/dist/tools/verify-plan-execution.js +295 -0
- package/dist/tools/web-fetch-injection-scanner.js +207 -0
- package/dist/tools/web-fetch.js +195 -10
- package/dist/tools/web-search.js +458 -0
- package/dist/tui/agent-progress-card.js +111 -0
- package/dist/tui/agent-tree.js +22 -1
- package/dist/tui/ask-modal.js +14 -14
- package/dist/tui/ask-user-question-chips.js +315 -0
- package/dist/tui/ask-user-question-prompt.js +203 -0
- package/dist/tui/compact-banner.js +81 -0
- package/dist/tui/conversation-pane.js +85 -11
- package/dist/tui/cost-table.js +111 -0
- package/dist/tui/device-flow.js +2 -2
- package/dist/tui/doctor-table.js +46 -0
- package/dist/tui/feedback-prompt.js +156 -0
- package/dist/tui/input-box.js +247 -32
- package/dist/tui/login-picker.js +3 -3
- package/dist/tui/markdown-render.js +6 -6
- package/dist/tui/multi-file-diff-approval.js +375 -0
- package/dist/tui/onboarding-wizard.js +240 -0
- package/dist/tui/permissions-picker.js +86 -0
- package/dist/tui/render.js +36 -1
- package/dist/tui/repl-render.js +239 -25
- 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 +125 -45
- 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/package.json +29 -6
- package/test/scenarios/codegen-create-file.scenario.txt +13 -0
- package/test/scenarios/compact-force.scenario.txt +12 -0
- package/test/scenarios/identity.scenario.txt +11 -0
- package/test/scenarios/persona-handoff.scenario.txt +12 -0
- package/test/scenarios/walkback.scenario.txt +12 -0
- package/dist/core/engine/compaction-hook.js +0 -154
|
@@ -0,0 +1,266 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* `pugi eval-v1` - run the frozen 20-task benchmark harness (#120).
|
|
3
|
+
*
|
|
4
|
+
* Surface:
|
|
5
|
+
*
|
|
6
|
+
* pugi eval-v1 --list # list task ids + briefs
|
|
7
|
+
* pugi eval-v1 --task 01-snake-html # run one task
|
|
8
|
+
* pugi eval-v1 --task 01,02,03 # run a subset
|
|
9
|
+
* pugi eval-v1 --all # run the full 20
|
|
10
|
+
* pugi eval-v1 --all --model qwen3-coder # pin the engine model
|
|
11
|
+
* pugi eval-v1 --all --json # JSON envelope output
|
|
12
|
+
* pugi eval-v1 --all --no-ledger # skip results.tsv append
|
|
13
|
+
*
|
|
14
|
+
* The handler stays thin: it resolves the tasks directory + pugi
|
|
15
|
+
* binary, calls `runHarness`, appends every result к the ledger
|
|
16
|
+
* (unless `--no-ledger`), and renders either a text table or a JSON
|
|
17
|
+
* envelope.
|
|
18
|
+
*
|
|
19
|
+
* Aggregate scoring (mean across all per-task scores) drives the CI
|
|
20
|
+
* regression gate in `.github/workflows/eval-v1.yml`. The gate is
|
|
21
|
+
* "aggregatePugiScore must not drop more than 10% vs the last week
|
|
22
|
+
* median" - the workflow computes the median itself from the
|
|
23
|
+
* committed ledger; the CLI's only job is к produce a fresh row.
|
|
24
|
+
*/
|
|
25
|
+
import { execFileSync } from 'node:child_process';
|
|
26
|
+
import { existsSync } from 'node:fs';
|
|
27
|
+
import { dirname, resolve } from 'node:path';
|
|
28
|
+
import { fileURLToPath } from 'node:url';
|
|
29
|
+
import { defaultLedgerPath, defaultTasksDir, loadAllTasks, } from '../../core/eval/v1/task-loader.js';
|
|
30
|
+
import { aggregateScore } from '../../core/eval/v1/scoring.js';
|
|
31
|
+
import { appendLedgerRows } from '../../core/eval/v1/ledger.js';
|
|
32
|
+
import { runHarness } from '../../core/eval/v1/runner.js';
|
|
33
|
+
function parseFlags(args) {
|
|
34
|
+
const out = {
|
|
35
|
+
list: false,
|
|
36
|
+
all: false,
|
|
37
|
+
task: [],
|
|
38
|
+
model: undefined,
|
|
39
|
+
ledger: true,
|
|
40
|
+
tasksDir: undefined,
|
|
41
|
+
ledgerPath: undefined,
|
|
42
|
+
help: false,
|
|
43
|
+
};
|
|
44
|
+
for (let i = 0; i < args.length; i += 1) {
|
|
45
|
+
const arg = args[i] ?? '';
|
|
46
|
+
if (arg === '--list')
|
|
47
|
+
out.list = true;
|
|
48
|
+
else if (arg === '--all')
|
|
49
|
+
out.all = true;
|
|
50
|
+
else if (arg === '--no-ledger')
|
|
51
|
+
out.ledger = false;
|
|
52
|
+
else if (arg === '--help' || arg === '-h')
|
|
53
|
+
out.help = true;
|
|
54
|
+
else if (arg === '--task') {
|
|
55
|
+
const next = args[i + 1];
|
|
56
|
+
if (!next || next.startsWith('--')) {
|
|
57
|
+
throw new Error('--task requires a comma-separated list of ids');
|
|
58
|
+
}
|
|
59
|
+
out.task.push(...next.split(',').map((s) => s.trim()).filter(Boolean));
|
|
60
|
+
i += 1;
|
|
61
|
+
}
|
|
62
|
+
else if (arg.startsWith('--task=')) {
|
|
63
|
+
out.task.push(...arg.slice('--task='.length).split(',').map((s) => s.trim()).filter(Boolean));
|
|
64
|
+
}
|
|
65
|
+
else if (arg === '--model') {
|
|
66
|
+
const next = args[i + 1];
|
|
67
|
+
if (!next)
|
|
68
|
+
throw new Error('--model requires a value');
|
|
69
|
+
out.model = next;
|
|
70
|
+
i += 1;
|
|
71
|
+
}
|
|
72
|
+
else if (arg.startsWith('--model=')) {
|
|
73
|
+
out.model = arg.slice('--model='.length);
|
|
74
|
+
}
|
|
75
|
+
else if (arg === '--tasks-dir') {
|
|
76
|
+
const next = args[i + 1];
|
|
77
|
+
if (!next)
|
|
78
|
+
throw new Error('--tasks-dir requires a path');
|
|
79
|
+
out.tasksDir = next;
|
|
80
|
+
i += 1;
|
|
81
|
+
}
|
|
82
|
+
else if (arg.startsWith('--tasks-dir=')) {
|
|
83
|
+
out.tasksDir = arg.slice('--tasks-dir='.length);
|
|
84
|
+
}
|
|
85
|
+
else if (arg === '--ledger') {
|
|
86
|
+
const next = args[i + 1];
|
|
87
|
+
if (!next)
|
|
88
|
+
throw new Error('--ledger requires a path');
|
|
89
|
+
out.ledgerPath = next;
|
|
90
|
+
i += 1;
|
|
91
|
+
}
|
|
92
|
+
else if (arg.startsWith('--ledger=')) {
|
|
93
|
+
out.ledgerPath = arg.slice('--ledger='.length);
|
|
94
|
+
}
|
|
95
|
+
else if (arg === '--json') {
|
|
96
|
+
// accepted at top-level; ignore here
|
|
97
|
+
}
|
|
98
|
+
else {
|
|
99
|
+
throw new Error(`unknown eval-v1 flag: ${arg}`);
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
return out;
|
|
103
|
+
}
|
|
104
|
+
function resolvePackageRoot(override) {
|
|
105
|
+
if (override)
|
|
106
|
+
return resolve(override);
|
|
107
|
+
// The compiled file lives at `dist/runtime/commands/eval-v1.js`;
|
|
108
|
+
// climb three directories to reach the package root in both dev
|
|
109
|
+
// (`src/runtime/commands/eval-v1.ts`) and dist builds.
|
|
110
|
+
const here = dirname(fileURLToPath(import.meta.url));
|
|
111
|
+
return resolve(here, '..', '..', '..');
|
|
112
|
+
}
|
|
113
|
+
function resolvePugiBin(override, packageRoot) {
|
|
114
|
+
if (override)
|
|
115
|
+
return resolve(override);
|
|
116
|
+
const local = resolve(packageRoot, 'bin', 'run.js');
|
|
117
|
+
if (existsSync(local))
|
|
118
|
+
return local;
|
|
119
|
+
// Fall back к PATH lookup; spawn will pick it up.
|
|
120
|
+
return 'pugi';
|
|
121
|
+
}
|
|
122
|
+
function captureGitSha(packageRoot) {
|
|
123
|
+
try {
|
|
124
|
+
const out = execFileSync('git', ['rev-parse', '--short', 'HEAD'], {
|
|
125
|
+
cwd: packageRoot,
|
|
126
|
+
encoding: 'utf8',
|
|
127
|
+
stdio: ['ignore', 'pipe', 'ignore'],
|
|
128
|
+
});
|
|
129
|
+
const trimmed = out.trim();
|
|
130
|
+
return trimmed === '' ? '(unknown)' : trimmed;
|
|
131
|
+
}
|
|
132
|
+
catch {
|
|
133
|
+
return '(unknown)';
|
|
134
|
+
}
|
|
135
|
+
}
|
|
136
|
+
function renderTable(report) {
|
|
137
|
+
const headers = ['id', 'difficulty', 'status', 'score', 'tokens', 'wall_ms', 'verifs'];
|
|
138
|
+
const rows = report.results.map((r) => [
|
|
139
|
+
r.taskId,
|
|
140
|
+
findDifficulty(r.taskId, report) ?? '?',
|
|
141
|
+
r.status,
|
|
142
|
+
r.pugiScore.toFixed(2),
|
|
143
|
+
String(r.tokensUsed),
|
|
144
|
+
String(r.wallClockMs),
|
|
145
|
+
`${r.verifications.filter((v) => v.passed).length}/${r.verifications.length}`,
|
|
146
|
+
]);
|
|
147
|
+
const widths = headers.map((h, i) => Math.max(h.length, ...rows.map((r) => (r[i] ?? '').length)));
|
|
148
|
+
const renderRow = (cells) => cells.map((c, i) => c.padEnd(widths[i] ?? 0)).join(' ');
|
|
149
|
+
const lines = [renderRow(headers), renderRow(widths.map((w) => '-'.repeat(w)))];
|
|
150
|
+
for (const row of rows)
|
|
151
|
+
lines.push(renderRow(row));
|
|
152
|
+
lines.push('');
|
|
153
|
+
lines.push(`aggregate pugi_score: ${report.aggregatePugiScore.toFixed(2)} (mean of ${report.results.length})`);
|
|
154
|
+
lines.push(`model: ${report.model}`);
|
|
155
|
+
lines.push(`git_sha: ${report.gitSha}`);
|
|
156
|
+
return lines.join('\n');
|
|
157
|
+
}
|
|
158
|
+
const difficultyMap = new WeakMap();
|
|
159
|
+
function findDifficulty(taskId, report) {
|
|
160
|
+
const cached = difficultyMap.get(report);
|
|
161
|
+
return cached?.get(taskId);
|
|
162
|
+
}
|
|
163
|
+
function indexDifficulty(report, specs) {
|
|
164
|
+
const map = new Map();
|
|
165
|
+
for (const s of specs)
|
|
166
|
+
map.set(s.id, s.difficulty);
|
|
167
|
+
difficultyMap.set(report, map);
|
|
168
|
+
}
|
|
169
|
+
export async function runEvalV1Command(ctx) {
|
|
170
|
+
let flags;
|
|
171
|
+
try {
|
|
172
|
+
flags = parseFlags(ctx.args);
|
|
173
|
+
}
|
|
174
|
+
catch (err) {
|
|
175
|
+
ctx.writeOutput({ command: 'eval-v1', status: 'invalid_args', reason: err.message }, `pugi eval-v1: ${err.message}`);
|
|
176
|
+
return 2;
|
|
177
|
+
}
|
|
178
|
+
if (flags.help) {
|
|
179
|
+
const help = [
|
|
180
|
+
'pugi eval-v1 - frozen 20-task benchmark harness (#120)',
|
|
181
|
+
'',
|
|
182
|
+
'usage:',
|
|
183
|
+
' pugi eval-v1 --list',
|
|
184
|
+
' pugi eval-v1 --task <id[,id...]>',
|
|
185
|
+
' pugi eval-v1 --all [--model <slug>] [--no-ledger] [--json]',
|
|
186
|
+
].join('\n');
|
|
187
|
+
ctx.writeOutput({ command: 'eval-v1', status: 'help', help }, help);
|
|
188
|
+
return 0;
|
|
189
|
+
}
|
|
190
|
+
const packageRoot = resolvePackageRoot(ctx.packageRoot);
|
|
191
|
+
const tasksDir = flags.tasksDir ?? defaultTasksDir(packageRoot);
|
|
192
|
+
const ledgerPath = flags.ledgerPath ?? defaultLedgerPath(packageRoot);
|
|
193
|
+
const log = ctx.log ?? ((line) => process.stderr.write(`${line}\n`));
|
|
194
|
+
let loaded;
|
|
195
|
+
try {
|
|
196
|
+
loaded = loadAllTasks(tasksDir);
|
|
197
|
+
}
|
|
198
|
+
catch (err) {
|
|
199
|
+
ctx.writeOutput({
|
|
200
|
+
command: 'eval-v1',
|
|
201
|
+
status: 'tasks_load_failed',
|
|
202
|
+
reason: err.message,
|
|
203
|
+
}, `pugi eval-v1: ${err.message}`);
|
|
204
|
+
return 2;
|
|
205
|
+
}
|
|
206
|
+
const specs = loaded.map((l) => l.spec);
|
|
207
|
+
if (flags.list) {
|
|
208
|
+
const indexLines = specs.map((s) => `${s.id} [${s.command}/${s.difficulty}] ${s.brief}`);
|
|
209
|
+
const out = indexLines.join('\n');
|
|
210
|
+
ctx.writeOutput({ command: 'eval-v1', status: 'listed', tasks: specs.map((s) => ({
|
|
211
|
+
id: s.id, command: s.command, difficulty: s.difficulty, brief: s.brief,
|
|
212
|
+
})) }, out);
|
|
213
|
+
return 0;
|
|
214
|
+
}
|
|
215
|
+
if (!flags.all && flags.task.length === 0) {
|
|
216
|
+
const reason = 'pass --all OR --task <id[,id...]> OR --list';
|
|
217
|
+
ctx.writeOutput({ command: 'eval-v1', status: 'invalid_args', reason }, `pugi eval-v1: ${reason}`);
|
|
218
|
+
return 2;
|
|
219
|
+
}
|
|
220
|
+
const only = flags.all ? undefined : flags.task;
|
|
221
|
+
if (only) {
|
|
222
|
+
const known = new Set(specs.map((s) => s.id));
|
|
223
|
+
const unknown = only.filter((id) => !known.has(id));
|
|
224
|
+
if (unknown.length) {
|
|
225
|
+
const reason = `unknown task id(s): ${unknown.join(', ')}`;
|
|
226
|
+
ctx.writeOutput({ command: 'eval-v1', status: 'invalid_args', reason }, `pugi eval-v1: ${reason}`);
|
|
227
|
+
return 2;
|
|
228
|
+
}
|
|
229
|
+
}
|
|
230
|
+
const pugiBin = resolvePugiBin(ctx.pugiBin, packageRoot);
|
|
231
|
+
const startedAt = new Date().toISOString();
|
|
232
|
+
const gitSha = captureGitSha(packageRoot);
|
|
233
|
+
const results = await runHarness({
|
|
234
|
+
specs,
|
|
235
|
+
options: {
|
|
236
|
+
pugiBin,
|
|
237
|
+
...(flags.model !== undefined ? { model: flags.model } : {}),
|
|
238
|
+
...(only !== undefined ? { only } : {}),
|
|
239
|
+
...(ctx.runner ? { runner: ctx.runner } : {}),
|
|
240
|
+
...(ctx.env !== undefined ? { env: ctx.env } : {}),
|
|
241
|
+
onTaskStart: (s) => log(`[eval-v1] ${s.id} starting (${s.command}/${s.difficulty})`),
|
|
242
|
+
onTaskFinish: (r) => log(`[eval-v1] ${r.taskId} ${r.status} score=${r.pugiScore.toFixed(2)} tokens=${r.tokensUsed} wall=${r.wallClockMs}ms`),
|
|
243
|
+
},
|
|
244
|
+
});
|
|
245
|
+
const report = {
|
|
246
|
+
schemaVersion: 1,
|
|
247
|
+
startedAt,
|
|
248
|
+
model: flags.model ?? '(default)',
|
|
249
|
+
results,
|
|
250
|
+
aggregatePugiScore: aggregateScore(results),
|
|
251
|
+
gitSha,
|
|
252
|
+
};
|
|
253
|
+
indexDifficulty(report, specs);
|
|
254
|
+
if (flags.ledger && results.length > 0) {
|
|
255
|
+
appendLedgerRows(ledgerPath, results.map((r) => ({
|
|
256
|
+
timestamp: startedAt,
|
|
257
|
+
gitSha,
|
|
258
|
+
model: report.model,
|
|
259
|
+
result: r,
|
|
260
|
+
})));
|
|
261
|
+
}
|
|
262
|
+
ctx.writeOutput(report, renderTable(report));
|
|
263
|
+
const anyFail = results.some((r) => r.status !== 'pass');
|
|
264
|
+
return anyFail ? 1 : 0;
|
|
265
|
+
}
|
|
266
|
+
//# sourceMappingURL=eval-v1.js.map
|
|
@@ -0,0 +1,184 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* `pugi feedback` + `/feedback` slash — .
|
|
3
|
+
*
|
|
4
|
+
* In-CLI feedback collector. Parity with the upstream tool's `/feedback`
|
|
5
|
+
* built-in. The operator never has to leave the terminal to file a
|
|
6
|
+
* bug / feature / general comment / praise. The wizard collects:
|
|
7
|
+
*
|
|
8
|
+
* 1. category (bug / feature / general / praise)
|
|
9
|
+
* 2. rating (1-5)
|
|
10
|
+
* 3. comment (multi-line free text)
|
|
11
|
+
* 4. optional redacted session context (last 5 turns)
|
|
12
|
+
* 5. confirm (final y/n)
|
|
13
|
+
*
|
|
14
|
+
* # Module contract
|
|
15
|
+
*
|
|
16
|
+
* - This file owns the WIRING from the CLI surface (TTY mount,
|
|
17
|
+
* non-TTY JSON, slash dispatcher) to the queue + submitter
|
|
18
|
+
* modules. The corpus + redactor + queue persistence live in
|
|
19
|
+
* `core/feedback/{queue.ts,submitter.ts}`. The Ink prompt lives
|
|
20
|
+
* in `tui/feedback-prompt.tsx`. Both have zero coupling to the
|
|
21
|
+
* CLI dispatch surface.
|
|
22
|
+
*
|
|
23
|
+
* - `runFeedbackCommand` is the single entry point. Both the top-
|
|
24
|
+
* level `pugi feedback` handler в `runtime/cli.ts` AND the in-REPL
|
|
25
|
+
* `/feedback` slash dispatcher call it. The function returns the
|
|
26
|
+
* resolved `FeedbackRunResult` so the slash dispatcher can route
|
|
27
|
+
* the outcome message to the REPL's system pane without re-prompting.
|
|
28
|
+
*
|
|
29
|
+
* - Exit code is ALWAYS 0. Feedback is a brand surface — never a
|
|
30
|
+
* gate. Failures land as result variants; the wrapper never
|
|
31
|
+
* turns a network blip into a non-zero shell exit.
|
|
32
|
+
*
|
|
33
|
+
* - The random-source-style test seam: the run helper accepts an
|
|
34
|
+
* `interactive` flag that the spec sets to false, plus an injected
|
|
35
|
+
* `draft` so unit tests can drive the submit + queue branches
|
|
36
|
+
* without mounting Ink.
|
|
37
|
+
*/
|
|
38
|
+
import { existsSync, readFileSync } from 'node:fs';
|
|
39
|
+
import { resolve } from 'node:path';
|
|
40
|
+
import { enqueueFeedback, feedbackQueuePath, flushFeedbackQueue, } from '../../core/feedback/queue.js';
|
|
41
|
+
import { feedbackSubmitUrl, redactSessionContext, submitFeedback, } from '../../core/feedback/submitter.js';
|
|
42
|
+
/**
|
|
43
|
+
* Drive one feedback round. The function is async because of the
|
|
44
|
+
* submit round-trip, but everything else (queue write, redaction) is
|
|
45
|
+
* sync — no surprise concurrency.
|
|
46
|
+
*/
|
|
47
|
+
export async function runFeedbackCommand(ctx) {
|
|
48
|
+
if (ctx.draft == null) {
|
|
49
|
+
return { kind: 'cancelled' };
|
|
50
|
+
}
|
|
51
|
+
const envelope = {
|
|
52
|
+
category: ctx.draft.category,
|
|
53
|
+
rating: ctx.draft.rating,
|
|
54
|
+
comment: ctx.draft.comment,
|
|
55
|
+
ts: new Date().toISOString(),
|
|
56
|
+
cliVersion: ctx.cliVersion,
|
|
57
|
+
...(ctx.tier ? { tier: ctx.tier } : {}),
|
|
58
|
+
};
|
|
59
|
+
if (ctx.draft.includeSessionContext && ctx.sessionContext) {
|
|
60
|
+
const sc = ctx.sessionContext();
|
|
61
|
+
if (sc)
|
|
62
|
+
envelope.sessionContext = sc;
|
|
63
|
+
}
|
|
64
|
+
let result;
|
|
65
|
+
try {
|
|
66
|
+
result = await ctx.submit(envelope);
|
|
67
|
+
}
|
|
68
|
+
catch (err) {
|
|
69
|
+
// Defensive: a thrown submitter (should not happen — the live
|
|
70
|
+
// submitter catches everything) is treated as transient so the
|
|
71
|
+
// envelope lands in the queue.
|
|
72
|
+
const reason = err instanceof Error ? err.message : String(err);
|
|
73
|
+
result = { kind: 'transient', reason };
|
|
74
|
+
}
|
|
75
|
+
if (result.kind === 'ok') {
|
|
76
|
+
return { kind: 'submitted', envelope, httpStatus: result.httpStatus };
|
|
77
|
+
}
|
|
78
|
+
if (result.kind === 'transient') {
|
|
79
|
+
const path = enqueueFeedback(envelope, ctx.cwd);
|
|
80
|
+
return { kind: 'queued', envelope, path, reason: result.reason };
|
|
81
|
+
}
|
|
82
|
+
// permanent — log + drop
|
|
83
|
+
return {
|
|
84
|
+
kind: 'dropped',
|
|
85
|
+
envelope,
|
|
86
|
+
reason: result.reason,
|
|
87
|
+
httpStatus: result.httpStatus,
|
|
88
|
+
};
|
|
89
|
+
}
|
|
90
|
+
/**
|
|
91
|
+
* Render one human-readable toast for the operator. Centralised so the
|
|
92
|
+
* top-level `pugi feedback` shell handler + the in-REPL `/feedback`
|
|
93
|
+
* slash dispatcher agree on the copy.
|
|
94
|
+
*/
|
|
95
|
+
export function renderFeedbackToast(result) {
|
|
96
|
+
switch (result.kind) {
|
|
97
|
+
case 'submitted':
|
|
98
|
+
return 'Feedback submitted. Thank you.';
|
|
99
|
+
case 'queued':
|
|
100
|
+
return `Feedback queued locally. Will sync on next online run. (${result.path})`;
|
|
101
|
+
case 'cancelled':
|
|
102
|
+
return 'Feedback cancelled. Nothing was sent.';
|
|
103
|
+
case 'dropped':
|
|
104
|
+
return `Feedback rejected by server (${result.httpStatus}): ${result.reason}. Not queued.`;
|
|
105
|
+
case 'noop':
|
|
106
|
+
return `No feedback collected: ${result.reason}`;
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
/**
|
|
110
|
+
* Background queue flush. Invoked silently on session start so any
|
|
111
|
+
* envelopes that landed during an offline run get drained when the
|
|
112
|
+
* operator next has connectivity. The function never throws — it
|
|
113
|
+
* returns the flush stats so the caller can log them at debug level.
|
|
114
|
+
*/
|
|
115
|
+
export async function flushFeedbackQueueSilently(cwd, config) {
|
|
116
|
+
// Short-circuit when the queue file does not exist. Avoids a
|
|
117
|
+
// pointless `fs.stat` round-trip on every cold session start.
|
|
118
|
+
if (!existsSync(feedbackQueuePath(cwd))) {
|
|
119
|
+
return { attempted: 0, succeeded: 0, failed: 0 };
|
|
120
|
+
}
|
|
121
|
+
const result = await flushFeedbackQueue(cwd, async (env) => {
|
|
122
|
+
const r = await submitFeedback(env, config);
|
|
123
|
+
if (r.kind === 'ok')
|
|
124
|
+
return true;
|
|
125
|
+
if (r.kind === 'permanent') {
|
|
126
|
+
// Permanent failures are "done" from the queue's POV — they
|
|
127
|
+
// would never resolve on retry. Drop them so the queue does
|
|
128
|
+
// not grow without bound.
|
|
129
|
+
return true;
|
|
130
|
+
}
|
|
131
|
+
return false;
|
|
132
|
+
});
|
|
133
|
+
return {
|
|
134
|
+
attempted: result.attempted,
|
|
135
|
+
succeeded: result.succeeded,
|
|
136
|
+
failed: result.failed,
|
|
137
|
+
};
|
|
138
|
+
}
|
|
139
|
+
/**
|
|
140
|
+
* Re-exports — the spec imports these via the command module so the
|
|
141
|
+
* dependency graph in the test stays single-rooted.
|
|
142
|
+
*/
|
|
143
|
+
export { feedbackQueuePath, feedbackSubmitUrl, redactSessionContext, submitFeedback, };
|
|
144
|
+
/**
|
|
145
|
+
* Read the persona conversation log if present. Best-effort: returns
|
|
146
|
+
* an empty list when the file is missing or malformed. The CLI's REPL
|
|
147
|
+
* persists transcripts via the session module at a canonical relative
|
|
148
|
+
* path under `.pugi/sessions/`. The shell-level `pugi feedback` does
|
|
149
|
+
* not have access to a live session, so it tries to pick up the most
|
|
150
|
+
* recent persisted one for the `--with-context` path.
|
|
151
|
+
*
|
|
152
|
+
* Intentionally tolerant — feedback works even with no transcript.
|
|
153
|
+
*/
|
|
154
|
+
export function readMostRecentTranscript(cwd, options = {}) {
|
|
155
|
+
// The CLI may persist sessions in several places depending on the
|
|
156
|
+
// surface. We probe the conventional default; the spec drives the
|
|
157
|
+
// function via a fixture file instead of a live REPL.
|
|
158
|
+
const candidate = resolve(cwd, '.pugi', 'sessions', 'latest.jsonl');
|
|
159
|
+
if (!existsSync(candidate))
|
|
160
|
+
return [];
|
|
161
|
+
try {
|
|
162
|
+
const text = readFileSync(candidate, 'utf8');
|
|
163
|
+
const lines = text.split('\n').filter((l) => l.trim().length > 0);
|
|
164
|
+
const turns = [];
|
|
165
|
+
for (const line of lines) {
|
|
166
|
+
try {
|
|
167
|
+
const obj = JSON.parse(line);
|
|
168
|
+
if ((obj.role === 'user' || obj.role === 'assistant' || obj.role === 'system')
|
|
169
|
+
&& typeof obj.text === 'string') {
|
|
170
|
+
turns.push({ role: obj.role, text: obj.text });
|
|
171
|
+
}
|
|
172
|
+
}
|
|
173
|
+
catch {
|
|
174
|
+
// skip malformed line
|
|
175
|
+
}
|
|
176
|
+
}
|
|
177
|
+
const cap = options.maxTurns ?? 5;
|
|
178
|
+
return turns.slice(-cap);
|
|
179
|
+
}
|
|
180
|
+
catch {
|
|
181
|
+
return [];
|
|
182
|
+
}
|
|
183
|
+
}
|
|
184
|
+
//# sourceMappingURL=feedback.js.map
|
|
@@ -0,0 +1,187 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* `pugi hooks` — operator surface for user-config hooks (MVP).
|
|
3
|
+
*
|
|
4
|
+
* Two subcommands ship in the MVP:
|
|
5
|
+
*
|
|
6
|
+
* pugi hooks list List configured hooks per event.
|
|
7
|
+
* pugi hooks doctor Validate the config and surface any
|
|
8
|
+
* parse / schema errors.
|
|
9
|
+
*
|
|
10
|
+
* Both accept `--json` for scripted callers. Argument grammar is
|
|
11
|
+
* intentionally narrow — no `add` / `remove` / `test` subcommands in
|
|
12
|
+
* the MVP. Operators hand-edit `~/.pugi/hooks-mvp.json` for now.
|
|
13
|
+
*
|
|
14
|
+
* Exit codes:
|
|
15
|
+
* 0 -> happy path (no hooks OR config valid).
|
|
16
|
+
* 1 -> config present but invalid (only `doctor` returns this).
|
|
17
|
+
* 2 -> unknown subcommand / argument error.
|
|
18
|
+
*
|
|
19
|
+
* Brand voice: ASCII only.
|
|
20
|
+
*/
|
|
21
|
+
import { ALL_HOOK_EVENTS_V2, defaultHooksMvpPath, loadHooksConfig, } from '../../core/hooks/index.js';
|
|
22
|
+
function parseFlags(args) {
|
|
23
|
+
const rest = [];
|
|
24
|
+
const flags = { json: false };
|
|
25
|
+
for (const arg of args) {
|
|
26
|
+
if (arg === '--json') {
|
|
27
|
+
flags.json = true;
|
|
28
|
+
continue;
|
|
29
|
+
}
|
|
30
|
+
rest.push(arg);
|
|
31
|
+
}
|
|
32
|
+
return { rest, flags };
|
|
33
|
+
}
|
|
34
|
+
/**
|
|
35
|
+
* Top-level dispatcher for `pugi hooks <subcommand>`. Returns the
|
|
36
|
+
* intended process exit code. `cli.ts` is expected to set
|
|
37
|
+
* `process.exitCode = <return value>` so error states propagate to
|
|
38
|
+
* scripted callers without throwing.
|
|
39
|
+
*/
|
|
40
|
+
export async function runHooksCommand(args, ctx) {
|
|
41
|
+
const { rest, flags } = parseFlags(args);
|
|
42
|
+
const sub = rest[0];
|
|
43
|
+
if (!sub || sub === 'help' || sub === '--help') {
|
|
44
|
+
emitUsage(ctx, flags);
|
|
45
|
+
return sub ? 0 : 2;
|
|
46
|
+
}
|
|
47
|
+
if (sub === 'list') {
|
|
48
|
+
return runList(ctx, flags);
|
|
49
|
+
}
|
|
50
|
+
if (sub === 'doctor') {
|
|
51
|
+
return runDoctor(ctx, flags);
|
|
52
|
+
}
|
|
53
|
+
ctx.writeOutput({ ok: false, error: `unknown subcommand: ${sub}` }, `pugi hooks: unknown subcommand '${sub}'. Try 'pugi hooks --help'.`);
|
|
54
|
+
return 2;
|
|
55
|
+
}
|
|
56
|
+
function emitUsage(ctx, flags) {
|
|
57
|
+
const text = [
|
|
58
|
+
'pugi hooks — user-config lifecycle hooks (MVP).',
|
|
59
|
+
'',
|
|
60
|
+
'Subcommands:',
|
|
61
|
+
' pugi hooks list Show hooks configured per event.',
|
|
62
|
+
' pugi hooks doctor Validate ~/.pugi/hooks-mvp.json.',
|
|
63
|
+
'',
|
|
64
|
+
'Flags:',
|
|
65
|
+
' --json Emit a JSON envelope instead of human text.',
|
|
66
|
+
'',
|
|
67
|
+
'Config file:',
|
|
68
|
+
' ~/.pugi/hooks-mvp.json',
|
|
69
|
+
'',
|
|
70
|
+
'Status:',
|
|
71
|
+
' MVP — 2 events out of 8. Remaining events (PostToolUse,',
|
|
72
|
+
" UserPromptSubmit, Stop, SubagentStop, PreCompact, Notification)",
|
|
73
|
+
' deferred to fast-follow PR.',
|
|
74
|
+
].join('\n');
|
|
75
|
+
ctx.writeOutput({
|
|
76
|
+
ok: true,
|
|
77
|
+
command: 'hooks',
|
|
78
|
+
usage: text,
|
|
79
|
+
}, text);
|
|
80
|
+
if (flags.json) {
|
|
81
|
+
// The structured payload is already emitted by writeOutput when
|
|
82
|
+
// --json is on; nothing extra to do.
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
function runList(ctx, flags) {
|
|
86
|
+
let config;
|
|
87
|
+
try {
|
|
88
|
+
config = loadHooksConfig(ctx.configPath);
|
|
89
|
+
}
|
|
90
|
+
catch (error) {
|
|
91
|
+
const msg = error.message;
|
|
92
|
+
ctx.writeOutput({ ok: false, error: msg }, `pugi hooks list: ${msg}\nFix the config or remove the file. Run 'pugi hooks doctor' for details.`);
|
|
93
|
+
return 1;
|
|
94
|
+
}
|
|
95
|
+
const perEvent = {
|
|
96
|
+
SessionStart: [],
|
|
97
|
+
PreToolUse: [],
|
|
98
|
+
PostToolUse: [],
|
|
99
|
+
UserPromptSubmit: [],
|
|
100
|
+
Stop: [],
|
|
101
|
+
SubagentStop: [],
|
|
102
|
+
PreCompact: [],
|
|
103
|
+
Notification: [],
|
|
104
|
+
// PUGI-487 - worktree lifecycle events.
|
|
105
|
+
WorktreeCreate: [],
|
|
106
|
+
WorktreeRemove: [],
|
|
107
|
+
};
|
|
108
|
+
for (const event of ALL_HOOK_EVENTS_V2) {
|
|
109
|
+
perEvent[event] = config.list(event).map((entry) => ({
|
|
110
|
+
matcher: entry.matcher,
|
|
111
|
+
command: entry.command,
|
|
112
|
+
timeoutMs: entry.timeoutMs,
|
|
113
|
+
blocking: entry.blocking,
|
|
114
|
+
}));
|
|
115
|
+
}
|
|
116
|
+
const total = Object.values(perEvent).reduce((acc, list) => acc + list.length, 0);
|
|
117
|
+
const payload = {
|
|
118
|
+
ok: true,
|
|
119
|
+
configPath: config.configPath(),
|
|
120
|
+
total,
|
|
121
|
+
perEvent,
|
|
122
|
+
};
|
|
123
|
+
if (flags.json) {
|
|
124
|
+
ctx.writeOutput(payload, JSON.stringify(payload, null, 2));
|
|
125
|
+
return 0;
|
|
126
|
+
}
|
|
127
|
+
const lines = [];
|
|
128
|
+
lines.push(`pugi hooks (${total} configured)`);
|
|
129
|
+
lines.push(` config: ${config.configPath()}`);
|
|
130
|
+
if (total === 0) {
|
|
131
|
+
lines.push(' no hooks configured — create the file above to add one.');
|
|
132
|
+
}
|
|
133
|
+
else {
|
|
134
|
+
for (const event of ALL_HOOK_EVENTS_V2) {
|
|
135
|
+
const list = perEvent[event];
|
|
136
|
+
if (list.length === 0)
|
|
137
|
+
continue;
|
|
138
|
+
lines.push(` ${event}:`);
|
|
139
|
+
for (const entry of list) {
|
|
140
|
+
const tags = [];
|
|
141
|
+
if (entry.matcher)
|
|
142
|
+
tags.push(`matcher=${entry.matcher}`);
|
|
143
|
+
if (entry.timeoutMs)
|
|
144
|
+
tags.push(`timeoutMs=${entry.timeoutMs}`);
|
|
145
|
+
if (entry.blocking)
|
|
146
|
+
tags.push('blocking');
|
|
147
|
+
const suffix = tags.length ? ` [${tags.join(', ')}]` : '';
|
|
148
|
+
lines.push(` - ${entry.command}${suffix}`);
|
|
149
|
+
}
|
|
150
|
+
}
|
|
151
|
+
}
|
|
152
|
+
const text = lines.join('\n');
|
|
153
|
+
ctx.writeOutput(payload, text);
|
|
154
|
+
return 0;
|
|
155
|
+
}
|
|
156
|
+
function runDoctor(ctx, flags) {
|
|
157
|
+
const path = ctx.configPath ?? defaultHooksMvpPath();
|
|
158
|
+
try {
|
|
159
|
+
const config = loadHooksConfig(ctx.configPath);
|
|
160
|
+
const total = config.flatten().length;
|
|
161
|
+
const payload = {
|
|
162
|
+
ok: true,
|
|
163
|
+
configPath: config.configPath(),
|
|
164
|
+
total,
|
|
165
|
+
issues: [],
|
|
166
|
+
};
|
|
167
|
+
const text = total
|
|
168
|
+
? `pugi hooks doctor: ${path} OK (${total} hooks).`
|
|
169
|
+
: `pugi hooks doctor: ${path} not present (no hooks configured).`;
|
|
170
|
+
ctx.writeOutput(payload, text);
|
|
171
|
+
return 0;
|
|
172
|
+
}
|
|
173
|
+
catch (error) {
|
|
174
|
+
const msg = error.message;
|
|
175
|
+
const payload = {
|
|
176
|
+
ok: false,
|
|
177
|
+
configPath: path,
|
|
178
|
+
error: msg,
|
|
179
|
+
};
|
|
180
|
+
const text = `pugi hooks doctor: ${msg}`;
|
|
181
|
+
ctx.writeOutput(payload, text);
|
|
182
|
+
return 1;
|
|
183
|
+
}
|
|
184
|
+
// flags.json is consumed by writeOutput in the host shell.
|
|
185
|
+
void flags;
|
|
186
|
+
}
|
|
187
|
+
//# sourceMappingURL=hooks.js.map
|