@pugi/cli 0.1.0-beta.1 → 0.1.0-beta.100
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +132 -0
- package/LICENSE +1 -1
- package/README.md +53 -11
- package/THIRD_PARTY_NOTICES.md +40 -0
- package/assets/pugi-mascot.ansi +15 -40
- package/assets/pugi-prozr2-mascot.ansi +9 -0
- package/bin/run.js +33 -1
- package/dist/commands/deploy.js +40 -40
- package/dist/commands/flatten.js +191 -0
- package/dist/commands/jobs-watch.js +201 -0
- package/dist/commands/jobs.js +42 -27
- package/dist/commands/retro.js +210 -0
- package/dist/commands/smoke.js +133 -0
- package/dist/core/agent-progress/cleanup.js +134 -0
- package/dist/core/agent-progress/schema.js +144 -0
- package/dist/core/agent-progress/writer.js +101 -0
- package/dist/core/agents/adaptive-router.js +330 -0
- package/dist/core/agents/query-decomposer.js +297 -0
- package/dist/core/agents/registry.js +3 -3
- package/dist/core/approvals/shortcut-resolver.js +98 -0
- package/dist/core/artifact-chain/dispatcher.js +148 -0
- package/dist/core/artifact-chain/exporter.js +164 -0
- package/dist/core/artifact-chain/state.js +243 -0
- package/dist/core/artifact-chain/steps.js +169 -0
- package/dist/core/ask-user/question.js +92 -0
- package/dist/core/audit/audit-trail.js +275 -0
- package/dist/core/auth/ensure-authenticated.js +129 -0
- package/dist/core/auth/env-provider.js +238 -0
- package/dist/core/auto-open-browser.js +4 -4
- package/dist/core/auto-update/channels.js +122 -0
- package/dist/core/auto-update/checker.js +241 -0
- package/dist/core/auto-update/state.js +235 -0
- package/dist/core/bare-mode/index.js +107 -0
- package/dist/core/bash/redirect.js +281 -0
- package/dist/core/bash-classifier.js +436 -40
- package/dist/core/checkpoint/resumer.js +149 -0
- package/dist/core/checkpoint/rewinder.js +291 -0
- package/dist/core/checkpoints/shadow-git.js +670 -0
- package/dist/core/citations/parser.js +109 -0
- package/dist/core/classifier/yolo-classifier.js +88 -0
- package/dist/core/codegraph/db.js +506 -0
- package/dist/core/codegraph/decision-store.js +248 -0
- package/dist/core/codegraph/detect-repo.js +459 -0
- package/dist/core/codegraph/install.js +134 -0
- package/dist/core/codegraph/offer-hook.js +220 -0
- package/dist/core/codegraph/parser.js +71 -0
- package/dist/core/codegraph/types.js +34 -0
- package/dist/core/compact/auto-trigger.js +96 -0
- package/dist/core/compact/buffer-rewriter.js +115 -0
- package/dist/core/compact/summarizer.js +208 -0
- package/dist/core/compact/token-counter.js +108 -0
- package/dist/core/consensus/anvil-fanout.js +25 -25
- package/dist/core/consensus/diff-capture.js +121 -12
- package/dist/core/consensus/rubric.js +21 -21
- package/dist/core/context/builder.js +6 -6
- package/dist/core/context/compaction-events.js +8 -8
- package/dist/core/context/compaction.js +31 -31
- package/dist/core/context/index.js +15 -8
- package/dist/core/context/invariants.js +51 -51
- package/dist/core/context/markdown-loader.js +28 -10
- package/dist/core/context/markdown-traverse.js +255 -0
- package/dist/core/context/pugiignore.js +41 -41
- package/dist/core/context/repo-skeleton.js +37 -37
- package/dist/core/context/tool-eviction.js +55 -0
- package/dist/core/context/watcher.js +32 -32
- package/dist/core/context/working-set.js +23 -23
- package/dist/core/coordinator/agent-tools.js +77 -0
- package/dist/core/coordinator/agent-toolset.js +65 -0
- package/dist/core/coordinator/fsm.js +73 -0
- package/dist/core/coordinator/mode-fsm.js +70 -0
- package/dist/core/cost/rate-card.js +129 -0
- package/dist/core/cost/tracker.js +221 -0
- package/dist/core/credentials.js +13 -13
- package/dist/core/cron/scheduler.js +138 -0
- package/dist/core/denial-tracking/index.js +8 -0
- package/dist/core/denial-tracking/state.js +264 -0
- package/dist/core/diagnostics/probe-runner.js +93 -0
- package/dist/core/diagnostics/probes/api.js +46 -0
- package/dist/core/diagnostics/probes/auth.js +93 -0
- package/dist/core/diagnostics/probes/bare-mode.js +42 -0
- package/dist/core/diagnostics/probes/cli-version.js +127 -0
- package/dist/core/diagnostics/probes/config.js +72 -0
- package/dist/core/diagnostics/probes/denial-tracking.js +57 -0
- package/dist/core/diagnostics/probes/disk.js +81 -0
- package/dist/core/diagnostics/probes/engine-live.js +46 -0
- package/dist/core/diagnostics/probes/git.js +65 -0
- package/dist/core/diagnostics/probes/hooks.js +118 -0
- package/dist/core/diagnostics/probes/mcp.js +75 -0
- package/dist/core/diagnostics/probes/node.js +59 -0
- package/dist/core/diagnostics/probes/pnpm.js +36 -0
- package/dist/core/diagnostics/probes/pugi-md.js +89 -0
- package/dist/core/diagnostics/probes/sandbox.js +72 -0
- package/dist/core/diagnostics/probes/session.js +74 -0
- package/dist/core/diagnostics/probes/status-snapshot.js +488 -0
- package/dist/core/diagnostics/probes/workspace.js +63 -0
- package/dist/core/diagnostics/types.js +70 -0
- package/dist/core/dispatch/cache-cleanup.js +197 -0
- package/dist/core/dispatch/cache-handoff.js +295 -0
- package/dist/core/edits/apply-patch-layer-e.js +189 -0
- package/dist/core/edits/dispatch.js +333 -7
- package/dist/core/edits/format-detector.js +260 -0
- package/dist/core/edits/format-matrix.js +26 -0
- package/dist/core/edits/fuzzy-ladder.js +650 -0
- package/dist/core/edits/index.js +5 -1
- package/dist/core/edits/journal.js +199 -0
- package/dist/core/edits/layer-a-apply.js +15 -15
- package/dist/core/edits/layer-a-fuzzy-apply.js +198 -0
- package/dist/core/edits/layer-b-apply.js +9 -9
- package/dist/core/edits/layer-c-apply.js +6 -6
- package/dist/core/edits/layer-d-ast.js +557 -14
- package/dist/core/edits/marker-parser.js +12 -12
- package/dist/core/edits/security-gate.js +27 -27
- package/dist/core/edits/verify-hook.js +273 -0
- package/dist/core/edits/worktree.js +322 -0
- package/dist/core/engine/anvil-client.js +214 -26
- package/dist/core/engine/auto-compact.js +247 -0
- package/dist/core/engine/budgets.js +220 -0
- package/dist/core/engine/compact-llm-summarizer.js +124 -0
- package/dist/core/engine/context-prefix.js +155 -0
- package/dist/core/engine/index.js +1 -1
- package/dist/core/engine/intensity.js +163 -0
- package/dist/core/engine/intent.js +260 -0
- package/dist/core/engine/native-pugi.js +1559 -227
- package/dist/core/engine/prompts.js +192 -16
- package/dist/core/engine/strip-internal-fields.js +124 -0
- package/dist/core/engine/tool-bridge.js +1887 -59
- package/dist/core/engine/verification-patterns.js +195 -0
- package/dist/core/evaluation/golden-dataset.js +293 -0
- package/dist/core/feedback/queue.js +177 -0
- package/dist/core/feedback/submitter.js +145 -0
- package/dist/core/file-cache.js +113 -1
- package/dist/core/flatten/flatten-repo.js +439 -0
- package/dist/core/format/osc8-link.js +28 -0
- package/dist/core/hook-chains.js +392 -0
- package/dist/core/hooks/citation-verify-hook.js +138 -0
- package/dist/core/hooks/citation-verify.js +112 -0
- package/dist/core/hooks/events.js +46 -0
- package/dist/core/hooks/index.js +15 -0
- package/dist/core/hooks/registry.js +216 -0
- package/dist/core/hooks/runner.js +236 -0
- package/dist/core/hooks/v2/event-emitter.js +115 -0
- package/dist/core/hooks/v2/executor.js +282 -0
- package/dist/core/hooks/v2/index.js +25 -0
- package/dist/core/hooks/v2/lifecycle.js +104 -0
- package/dist/core/hooks/v2/loader.js +216 -0
- package/dist/core/hooks/v2/matcher.js +125 -0
- package/dist/core/hooks/v2/trust.js +143 -0
- package/dist/core/hooks/v2/types.js +86 -0
- package/dist/core/hooks/worktree-events.js +158 -0
- package/dist/core/image/renderer.js +71 -0
- package/dist/core/init/detector.js +582 -0
- package/dist/core/init/template-renderer.js +242 -0
- package/dist/core/jobs/registry.js +18 -18
- package/dist/core/ledger/results-tsv.js +142 -0
- package/dist/core/log-discipline/stdout-redirect.js +51 -0
- package/dist/core/lsp/cache.js +105 -0
- package/dist/core/lsp/client.js +1229 -0
- package/dist/core/lsp/language-detect.js +66 -0
- package/dist/core/lsp/post-edit-diagnostics.js +171 -0
- package/dist/core/lsp/server-detect.js +173 -0
- package/dist/core/lsp/symbol-cache.js +162 -0
- package/dist/core/lsp/symbol-tools.js +664 -0
- package/dist/core/mcp/client.js +97 -28
- package/dist/core/mcp/http-server.js +553 -0
- package/dist/core/mcp/orchestrator-config.js +192 -0
- package/dist/core/mcp/orchestrator-tools.js +806 -0
- package/dist/core/mcp/permission.js +190 -0
- package/dist/core/mcp/registry.js +39 -17
- package/dist/core/mcp/server-tools.js +219 -0
- package/dist/core/mcp/server.js +397 -0
- package/dist/core/mcp/trust.js +10 -10
- package/dist/core/memory/dual-write.js +416 -0
- package/dist/core/memory/passive-extract.js +130 -0
- package/dist/core/memory/phase1-kinds.js +20 -0
- package/dist/core/memory/secret-scanner.js +304 -0
- package/dist/core/memory-sync/queue.js +170 -0
- package/dist/core/metrics/extract.js +113 -0
- package/dist/core/modes/roo-modes.js +68 -0
- package/dist/core/notes/notes-paths.js +113 -0
- package/dist/core/notes/notes-recorder.js +140 -0
- package/dist/core/notes/notes-writer.js +53 -0
- package/dist/core/notes/renderers.js +0 -0
- package/dist/core/notes/slug.js +105 -0
- package/dist/core/onboarding/ensure-initialized.js +133 -0
- package/dist/core/onboarding/marker.js +111 -0
- package/dist/core/onboarding/telemetry-state.js +108 -0
- package/dist/core/output-style/presets.js +176 -0
- package/dist/core/output-style/state.js +185 -0
- package/dist/core/path-security.js +287 -5
- package/dist/core/permission.js +82 -22
- package/dist/core/permissions/auto-classifier.js +124 -0
- package/dist/core/permissions/bash-parser.js +371 -0
- package/dist/core/permissions/circuit-breaker.js +83 -0
- package/dist/core/permissions/constrained-edit.js +91 -0
- package/dist/core/permissions/gate.js +278 -0
- package/dist/core/permissions/index.js +20 -0
- package/dist/core/permissions/mode.js +174 -0
- package/dist/core/permissions/network-egress.js +137 -0
- package/dist/core/permissions/state.js +241 -0
- package/dist/core/permissions/tool-class.js +107 -0
- package/dist/core/plan-mode/ui-state.js +51 -0
- package/dist/core/plans/plan-artifact.js +721 -0
- package/dist/core/policy-limits/etag-store.js +122 -0
- package/dist/core/prd-check/parser.js +215 -0
- package/dist/core/prd-check/reporter.js +127 -0
- package/dist/core/prd-check/session-review.js +557 -0
- package/dist/core/prd-check/verifiers.js +223 -0
- package/dist/core/prompt-cache/client-cache.js +99 -0
- package/dist/core/prompts/assembly.js +29 -0
- package/dist/core/prompts/registry.js +364 -0
- package/dist/core/pugi-gitignore.js +52 -0
- package/dist/core/pugi-md/cc-compat-rules.js +735 -0
- package/dist/core/pugi-md/context-injector.js +76 -0
- package/dist/core/pugi-md/walk-up.js +207 -0
- package/dist/core/python/uv-installer.js +270 -0
- package/dist/core/python/uv-resolver.js +83 -0
- package/dist/core/rate-limit/narrator.js +146 -0
- package/dist/core/recipes/cli-types.js +20 -0
- package/dist/core/recipes/loader.js +103 -0
- package/dist/core/recipes/runner.js +345 -0
- package/dist/core/recipes/schema.js +587 -0
- package/dist/core/release-notes/parser.js +241 -0
- package/dist/core/release-notes/state.js +116 -0
- package/dist/core/repl/ask.js +37 -37
- package/dist/core/repl/cancellation.js +26 -26
- package/dist/core/repl/cap-warning.js +4 -4
- package/dist/core/repl/clipboard-read.js +11 -11
- package/dist/core/repl/dispatch-fsm.js +12 -12
- package/dist/core/repl/engine-bridge.js +303 -0
- package/dist/core/repl/history-search.js +15 -15
- package/dist/core/repl/history.js +28 -18
- package/dist/core/repl/kill-ring.js +5 -5
- package/dist/core/repl/model-pricing.js +135 -0
- package/dist/core/repl/privacy-banner.js +22 -22
- package/dist/core/repl/session.js +2714 -228
- package/dist/core/repl/slash-commands.js +572 -40
- package/dist/core/repl/store/index.js +1 -1
- package/dist/core/repl/store/jsonl-log.js +22 -22
- package/dist/core/repl/store/lockfile.js +10 -10
- package/dist/core/repl/store/session-store.js +136 -107
- package/dist/core/repl/store/types.js +15 -15
- package/dist/core/repl/store/uuid-v7.js +12 -12
- package/dist/core/repl/tool-route.js +382 -0
- package/dist/core/repl/workspace-context.js +43 -21
- package/dist/core/repo-map/build.js +125 -0
- package/dist/core/repo-map/cache.js +185 -0
- package/dist/core/repo-map/extractor.js +254 -0
- package/dist/core/repo-map/formatter.js +145 -0
- package/dist/core/repo-map/page-rank.js +105 -0
- package/dist/core/repo-map/scanner.js +211 -0
- package/dist/core/retro/git-collector.js +251 -0
- package/dist/core/retro/health-card.js +25 -0
- package/dist/core/retro/metrics.js +342 -0
- package/dist/core/retro/narrative.js +249 -0
- package/dist/core/retro/plane-collector.js +274 -0
- package/dist/core/retro/pr-issue-link.js +65 -0
- package/dist/core/retro/types.js +16 -0
- package/dist/core/retry-budget/budget.js +284 -0
- package/dist/core/retry-budget/index.js +5 -0
- package/dist/core/retry-budget/retry-cap.js +74 -0
- package/dist/core/routing/lead-worker.js +43 -0
- package/dist/core/routing/pre-flight-estimator.js +108 -0
- package/dist/core/runs/run-tree.js +103 -0
- package/dist/core/sandboxing/adapter.js +29 -0
- package/dist/core/sandboxing/index.js +49 -0
- package/dist/core/sandboxing/none.js +19 -0
- package/dist/core/sandboxing/seatbelt.js +183 -0
- package/dist/core/security/injection-scanner.js +367 -0
- package/dist/core/security/output-filter.js +418 -0
- package/dist/core/session/env-file.js +105 -0
- package/dist/core/session/section-budgets.js +140 -0
- package/dist/core/session.js +119 -0
- package/dist/core/settings.js +378 -5
- package/dist/core/share/formatter.js +271 -0
- package/dist/core/share/redactor.js +221 -0
- package/dist/core/share/uploader.js +267 -0
- package/dist/core/skills/defaults.js +457 -0
- package/dist/core/skills/loader.js +22 -22
- package/dist/core/skills/sources.js +27 -27
- package/dist/core/smoke/headless-driver.js +174 -0
- package/dist/core/smoke/orchestrator.js +194 -0
- package/dist/core/smoke/runner.js +238 -0
- package/dist/core/smoke/scenario-parser.js +316 -0
- package/dist/core/statusline.js +99 -0
- package/dist/core/subagents/dispatcher-real.js +600 -0
- package/dist/core/subagents/dispatcher.js +146 -52
- package/dist/core/subagents/index.js +19 -6
- package/dist/core/subagents/isolation-matrix.js +213 -0
- package/dist/core/subagents/spawn.js +19 -4
- package/dist/core/telemetry/emitter.js +229 -0
- package/dist/core/telemetry/queue.js +251 -0
- package/dist/core/theme/context.js +91 -0
- package/dist/core/theme/presets.js +228 -0
- package/dist/core/theme/state.js +181 -0
- package/dist/core/todos/invariant.js +10 -0
- package/dist/core/todos/state.js +177 -0
- package/dist/core/tool-schema/compressor.js +89 -0
- package/dist/core/transport/version-interceptor.js +166 -0
- package/dist/core/trust.js +2 -2
- package/dist/core/tui/thinking-block.js +64 -0
- package/dist/core/vim/keymap.js +288 -0
- package/dist/core/vim/state.js +92 -0
- package/dist/core/watch-markers/marker-watcher.js +133 -0
- package/dist/core/worktree/include-parser.js +249 -0
- package/dist/core/worktree-manager/cleanup.js +123 -0
- package/dist/core/worktree-manager/manager.js +303 -0
- package/dist/index.js +36 -0
- package/dist/runtime/bootstrap.js +190 -0
- package/dist/runtime/cli.js +4536 -477
- package/dist/runtime/commands/agents.js +31 -31
- package/dist/runtime/commands/budget.js +5 -5
- package/dist/runtime/commands/cancel.js +231 -0
- package/dist/runtime/commands/chain.js +489 -0
- package/dist/runtime/commands/codegraph-status.js +227 -0
- package/dist/runtime/commands/compact.js +297 -0
- package/dist/runtime/commands/config.js +74 -40
- package/dist/runtime/commands/cost.js +199 -0
- package/dist/runtime/commands/delegate.js +312 -0
- package/dist/runtime/commands/dispatch.js +126 -0
- package/dist/runtime/commands/doctor.js +579 -0
- package/dist/runtime/commands/feedback.js +184 -0
- package/dist/runtime/commands/hooks.js +187 -0
- package/dist/runtime/commands/index-cmd.js +353 -0
- package/dist/runtime/commands/init.js +254 -0
- package/dist/runtime/commands/lsp.js +368 -0
- package/dist/runtime/commands/mcp.js +935 -0
- package/dist/runtime/commands/memory.js +582 -0
- package/dist/runtime/commands/model.js +237 -0
- package/dist/runtime/commands/onboarding.js +275 -0
- package/dist/runtime/commands/patch.js +128 -0
- package/dist/runtime/commands/permissions.js +112 -0
- package/dist/runtime/commands/plan.js +143 -0
- package/dist/runtime/commands/prd-check.js +285 -0
- package/dist/runtime/commands/privacy.js +17 -17
- package/dist/runtime/commands/recipe.js +325 -0
- package/dist/runtime/commands/redo-blob-store.js +92 -0
- package/dist/runtime/commands/redo.js +361 -0
- package/dist/runtime/commands/release-notes.js +229 -0
- package/dist/runtime/commands/repo-map.js +95 -0
- package/dist/runtime/commands/report.js +299 -0
- package/dist/runtime/commands/resume.js +118 -0
- package/dist/runtime/commands/review-consensus.js +68 -53
- package/dist/runtime/commands/rewind.js +333 -0
- package/dist/runtime/commands/roster.js +117 -0
- package/dist/runtime/commands/servers.js +236 -0
- package/dist/runtime/commands/sessions.js +163 -0
- package/dist/runtime/commands/share.js +316 -0
- package/dist/runtime/commands/skills.js +31 -31
- package/dist/runtime/commands/status.js +186 -0
- package/dist/runtime/commands/stickers.js +82 -0
- package/dist/runtime/commands/style.js +194 -0
- package/dist/runtime/commands/theme.js +196 -0
- package/dist/runtime/commands/undo.js +54 -22
- package/dist/runtime/commands/update.js +289 -0
- package/dist/runtime/commands/vim.js +140 -0
- package/dist/runtime/commands/worktree.js +177 -0
- package/dist/runtime/commands/worktrees.js +155 -0
- package/dist/runtime/deprecation-warning.js +69 -0
- package/dist/runtime/engine-exit-code.js +50 -0
- package/dist/runtime/headless-repl.js +195 -0
- package/dist/runtime/headless.js +548 -0
- package/dist/runtime/load-hooks-or-exit.js +71 -0
- package/dist/runtime/plan-decompose.js +531 -0
- package/dist/runtime/sigint-guard.js +272 -0
- package/dist/runtime/stream-renderer.js +195 -0
- package/dist/runtime/update-check.js +28 -28
- package/dist/runtime/version.js +65 -0
- package/dist/runtime/worktree-bootstrap.js +579 -0
- package/dist/skills/bundled/batch.js +617 -0
- package/dist/skills/bundled/index.js +45 -0
- package/dist/skills/bundled/loop.js +358 -0
- package/dist/skills/bundled/remember.js +383 -0
- package/dist/skills/bundled/simplify.js +289 -0
- package/dist/skills/bundled/skillify.js +373 -0
- package/dist/skills/bundled/stuck.js +558 -0
- package/dist/skills/bundled/verify.js +439 -0
- package/dist/testing/vcr.js +486 -0
- package/dist/tools/agent-tool.js +229 -0
- package/dist/tools/apply-patch.js +556 -0
- package/dist/tools/ask-user-question.js +337 -0
- package/dist/tools/ask-user.js +115 -0
- package/dist/tools/bash.js +624 -46
- package/dist/tools/brief.js +224 -0
- package/dist/tools/cron.js +433 -0
- package/dist/tools/enter-worktree.js +250 -0
- package/dist/tools/exit-worktree.js +147 -0
- package/dist/tools/file-tools.js +161 -44
- package/dist/tools/http-request.js +336 -0
- package/dist/tools/lsp-tools.js +565 -0
- package/dist/tools/mcp-tool.js +260 -0
- package/dist/tools/multi-edit.js +361 -0
- package/dist/tools/powershell.js +268 -0
- package/dist/tools/registry.js +142 -1
- package/dist/tools/server-tools.js +892 -0
- package/dist/tools/skill-tool.js +96 -0
- package/dist/tools/sleep.js +99 -0
- package/dist/tools/synthetic-output.js +133 -0
- package/dist/tools/tasks.js +208 -0
- package/dist/tools/todo-write.js +184 -0
- package/dist/tools/verify-plan-execution.js +295 -0
- package/dist/tools/web-fetch-injection-scanner.js +207 -0
- package/dist/tools/web-fetch.js +195 -10
- package/dist/tools/web-search.js +458 -0
- package/dist/tui/agent-progress-card.js +111 -0
- package/dist/tui/agent-tree.js +22 -1
- package/dist/tui/ask-modal.js +14 -14
- package/dist/tui/ask-user-question-chips.js +315 -0
- package/dist/tui/ask-user-question-prompt.js +203 -0
- package/dist/tui/compact-banner.js +81 -0
- package/dist/tui/conversation-pane.js +85 -11
- package/dist/tui/cost-table.js +111 -0
- package/dist/tui/device-flow.js +2 -2
- package/dist/tui/doctor-table.js +46 -0
- package/dist/tui/feedback-prompt.js +156 -0
- package/dist/tui/input-box.js +247 -32
- package/dist/tui/login-picker.js +3 -3
- package/dist/tui/markdown-render.js +6 -6
- package/dist/tui/multi-file-diff-approval.js +375 -0
- package/dist/tui/onboarding-wizard.js +240 -0
- package/dist/tui/permissions-picker.js +86 -0
- package/dist/tui/render.js +36 -1
- package/dist/tui/repl-render.js +405 -32
- package/dist/tui/repl-splash-art.js +16 -16
- package/dist/tui/repl-splash-mascot.js +48 -24
- package/dist/tui/repl-splash.js +22 -22
- package/dist/tui/repl.js +136 -43
- package/dist/tui/slash-palette.js +6 -6
- package/dist/tui/splash.js +2 -2
- package/dist/tui/status-bar.js +109 -31
- package/dist/tui/status-table.js +7 -0
- package/dist/tui/stickers-art.js +136 -0
- package/dist/tui/style-table.js +28 -0
- package/dist/tui/theme-table.js +29 -0
- package/dist/tui/thinking-spinner.js +123 -0
- package/dist/tui/tool-stream-pane.js +53 -4
- package/dist/tui/update-banner.js +27 -2
- package/dist/tui/vim-input.js +267 -0
- package/dist/tui/welcome-banner.js +107 -0
- package/dist/tui/welcome-data.js +293 -0
- package/dist/tui/workspace-context.js +2 -2
- package/docs/examples/codegraph.mcp.json +10 -0
- package/package.json +25 -7
- package/test/scenarios/codegen-create-file.scenario.txt +13 -0
- package/test/scenarios/compact-force.scenario.txt +12 -0
- package/test/scenarios/identity.scenario.txt +11 -0
- package/test/scenarios/persona-handoff.scenario.txt +12 -0
- package/test/scenarios/walkback.scenario.txt +12 -0
- package/dist/core/engine/compaction-hook.js +0 -154
|
@@ -0,0 +1,210 @@
|
|
|
1
|
+
import { jsxs as _jsxs, jsx as _jsx } from "react/jsx-runtime";
|
|
2
|
+
import { Box, Text, render } from 'ink';
|
|
3
|
+
import { collectGitContext, countCommitsAheadOfBase, } from '../core/retro/git-collector.js';
|
|
4
|
+
import { ensurePugiGitIgnore } from '../core/pugi-gitignore.js';
|
|
5
|
+
import { computeMetrics } from '../core/retro/metrics.js';
|
|
6
|
+
import { persistRetro } from '../core/retro/narrative.js';
|
|
7
|
+
import { collectPlaneSlice, postRetroToPlane, resolvePlaneConfig, } from '../core/retro/plane-collector.js';
|
|
8
|
+
import { enrichLinks } from '../core/retro/pr-issue-link.js';
|
|
9
|
+
import { computeHealthCard } from '../core/retro/health-card.js';
|
|
10
|
+
/** Parse `7d` | `14d` | `30d` | `24h` into a duration in days
|
|
11
|
+
* (fractional for sub-day windows). Defaults to 7 days when omitted.
|
|
12
|
+
*/
|
|
13
|
+
function parseDurationToken(token) {
|
|
14
|
+
if (!token)
|
|
15
|
+
return undefined;
|
|
16
|
+
const match = /^(\d+)(h|d)$/.exec(token);
|
|
17
|
+
if (!match)
|
|
18
|
+
return undefined;
|
|
19
|
+
const value = Number.parseInt(match[1] ?? '0', 10);
|
|
20
|
+
const unit = match[2];
|
|
21
|
+
if (!Number.isFinite(value) || value <= 0)
|
|
22
|
+
return undefined;
|
|
23
|
+
const days = unit === 'h' ? value / 24 : value;
|
|
24
|
+
return { days, label: token };
|
|
25
|
+
}
|
|
26
|
+
function buildWindow(durationDays, label, now) {
|
|
27
|
+
const until = now;
|
|
28
|
+
const sinceMs = until.getTime() - durationDays * 24 * 60 * 60 * 1000;
|
|
29
|
+
const since = new Date(sinceMs);
|
|
30
|
+
// Midnight-align the lower bound to keep `--since` deterministic per day.
|
|
31
|
+
since.setHours(0, 0, 0, 0);
|
|
32
|
+
return { since, until, label, days: Math.max(1, Math.round(durationDays)) };
|
|
33
|
+
}
|
|
34
|
+
function buildPriorWindow(current) {
|
|
35
|
+
const until = new Date(current.since.getTime());
|
|
36
|
+
const sinceMs = until.getTime() - current.days * 24 * 60 * 60 * 1000;
|
|
37
|
+
const since = new Date(sinceMs);
|
|
38
|
+
since.setHours(0, 0, 0, 0);
|
|
39
|
+
return { since, until, label: `prior ${current.label}`, days: current.days };
|
|
40
|
+
}
|
|
41
|
+
function parseRetroArgs(rawArgs, now) {
|
|
42
|
+
const args = [...rawArgs];
|
|
43
|
+
const postPlane = args.includes('--post-plane');
|
|
44
|
+
const enrichPlane = args.includes('--plane') || postPlane;
|
|
45
|
+
const positional = args.filter((a) => !a.startsWith('-'));
|
|
46
|
+
let compare = false;
|
|
47
|
+
let durationToken;
|
|
48
|
+
if (positional[0] === 'compare') {
|
|
49
|
+
compare = true;
|
|
50
|
+
durationToken = positional[1];
|
|
51
|
+
}
|
|
52
|
+
else {
|
|
53
|
+
durationToken = positional[0];
|
|
54
|
+
}
|
|
55
|
+
const parsed = parseDurationToken(durationToken) ?? { days: 7, label: '7d' };
|
|
56
|
+
return {
|
|
57
|
+
window: buildWindow(parsed.days, parsed.label, now),
|
|
58
|
+
compare,
|
|
59
|
+
enrichPlane,
|
|
60
|
+
postPlane,
|
|
61
|
+
};
|
|
62
|
+
}
|
|
63
|
+
function SummaryCard(props) {
|
|
64
|
+
const { persisted, metrics, plane, planePostUrl } = props;
|
|
65
|
+
return (_jsxs(Box, { flexDirection: "column", borderStyle: "single", borderRight: false, borderTop: false, borderBottom: false, paddingLeft: 1, children: [_jsxs(Text, { bold: true, children: ["pugi retro \u00B7 ", metrics.window.label] }), _jsxs(Text, { dimColor: true, children: ["Branch ", metrics.branch.current, " over ", metrics.branch.base] }), _jsxs(Text, { children: [metrics.commits.total, " commits \u00B7 +", metrics.loc.insertions, " / -", metrics.loc.deletions, " LOC \u00B7 ", metrics.activeDays, " active days"] }), _jsxs(Text, { children: ["Focus ", metrics.focus.score, "% on ", metrics.focus.topDir ?? 'n/a', " \u00B7 Streak ", metrics.streak.personalDays, "d personal / ", metrics.streak.teamDays, "d team"] }), metrics.shipOfTheWeek ? (_jsxs(Text, { children: ["Ship of the week: ", metrics.shipOfTheWeek.subject.slice(0, 60)] })) : null, plane ? (_jsxs(Text, { children: ["Plane: closed ", plane.closedIssues.length, " \u00B7 created ", plane.createdIssues.length, " \u00B7 oversized modules ", plane.oversizedModules.length] })) : null, _jsxs(Text, { dimColor: true, children: ["Markdown: ", persisted.markdownPath] }), _jsxs(Text, { dimColor: true, children: ["JSON: ", persisted.jsonPath] }), planePostUrl ? _jsxs(Text, { children: ["Posted to Plane: ", planePostUrl] }) : null] }));
|
|
66
|
+
}
|
|
67
|
+
function renderSummary(props) {
|
|
68
|
+
const app = render(_jsx(SummaryCard, { ...props }));
|
|
69
|
+
app.unmount();
|
|
70
|
+
}
|
|
71
|
+
export async function runRetroCommand(ctx) {
|
|
72
|
+
const now = ctx.now ?? new Date();
|
|
73
|
+
const parsed = parseRetroArgs(ctx.args, now);
|
|
74
|
+
const gitCtx = await collectGitContext({ cwd: ctx.cwd, window: parsed.window });
|
|
75
|
+
if (!gitCtx.hasGit) {
|
|
76
|
+
const msg = 'pugi retro: not a git workspace - initialise git or cd into one.';
|
|
77
|
+
if (ctx.flags.json) {
|
|
78
|
+
ctx.io.write(`${JSON.stringify({ ok: false, error: 'no_git_workspace' })}\n`);
|
|
79
|
+
}
|
|
80
|
+
else {
|
|
81
|
+
ctx.io.writeError(msg);
|
|
82
|
+
}
|
|
83
|
+
return 2;
|
|
84
|
+
}
|
|
85
|
+
// Triple-review P1.2 (): before we write anything
|
|
86
|
+
// under `.pugi/retros/`, guarantee `.gitignore` covers `.pugi/`. Without
|
|
87
|
+
// this, the first customer run of `pugi retro` in a fresh repo would
|
|
88
|
+
// leave retros (and any future `.pugi/settings.json` secret store)
|
|
89
|
+
// tracked by git on the next `git add -A`. Idempotent.
|
|
90
|
+
//
|
|
91
|
+
// Round 2 P1 (2026-06-04): surface failure к stderr — silent catch
|
|
92
|
+
// defeats the gate's purpose. If `.gitignore` is read-only or perms
|
|
93
|
+
// refuse, the operator must know retros may be tracked by git.
|
|
94
|
+
const gitIgnoreCreated = [];
|
|
95
|
+
const gitIgnoreSkipped = [];
|
|
96
|
+
try {
|
|
97
|
+
ensurePugiGitIgnore(ctx.cwd, gitIgnoreCreated, gitIgnoreSkipped);
|
|
98
|
+
}
|
|
99
|
+
catch (err) {
|
|
100
|
+
const reason = err instanceof Error ? err.message : String(err);
|
|
101
|
+
ctx.io.writeError(`pugi retro: could not update .gitignore (${reason}). ` +
|
|
102
|
+
`Manually add ".pugi/" to .gitignore so retros are not tracked.`);
|
|
103
|
+
}
|
|
104
|
+
const toBaseHeadCount = await countCommitsAheadOfBase(ctx.cwd, gitCtx.baseBranch, parsed.window.since);
|
|
105
|
+
const metrics = computeMetrics({
|
|
106
|
+
window: parsed.window,
|
|
107
|
+
currentBranch: gitCtx.currentBranch,
|
|
108
|
+
baseBranch: gitCtx.baseBranch,
|
|
109
|
+
toBaseHeadCount,
|
|
110
|
+
currentUserName: gitCtx.userName,
|
|
111
|
+
currentUserEmail: gitCtx.userEmail,
|
|
112
|
+
commits: gitCtx.commits,
|
|
113
|
+
});
|
|
114
|
+
let compare;
|
|
115
|
+
if (parsed.compare) {
|
|
116
|
+
const priorWindow = buildPriorWindow(parsed.window);
|
|
117
|
+
const priorCtx = await collectGitContext({ cwd: ctx.cwd, window: priorWindow });
|
|
118
|
+
const priorAhead = await countCommitsAheadOfBase(ctx.cwd, gitCtx.baseBranch, priorWindow.since);
|
|
119
|
+
const priorMetrics = computeMetrics({
|
|
120
|
+
window: priorWindow,
|
|
121
|
+
currentBranch: gitCtx.currentBranch,
|
|
122
|
+
baseBranch: gitCtx.baseBranch,
|
|
123
|
+
toBaseHeadCount: priorAhead,
|
|
124
|
+
currentUserName: gitCtx.userName,
|
|
125
|
+
currentUserEmail: gitCtx.userEmail,
|
|
126
|
+
commits: priorCtx.commits,
|
|
127
|
+
});
|
|
128
|
+
compare = { current: metrics, prior: priorMetrics };
|
|
129
|
+
}
|
|
130
|
+
let plane;
|
|
131
|
+
let planeUnavailableReason;
|
|
132
|
+
if (parsed.enrichPlane) {
|
|
133
|
+
const cfgResult = resolvePlaneConfig(ctx.cwd);
|
|
134
|
+
if (!cfgResult.ok) {
|
|
135
|
+
planeUnavailableReason = cfgResult.reason;
|
|
136
|
+
}
|
|
137
|
+
else {
|
|
138
|
+
try {
|
|
139
|
+
const slice = await collectPlaneSlice({
|
|
140
|
+
config: cfgResult.config,
|
|
141
|
+
since: parsed.window.since,
|
|
142
|
+
});
|
|
143
|
+
const links = enrichLinks(gitCtx.commits, slice.closedIssues.concat(slice.createdIssues));
|
|
144
|
+
const health = computeHealthCard(slice.modules);
|
|
145
|
+
plane = {
|
|
146
|
+
...slice,
|
|
147
|
+
prToIssueLinks: links,
|
|
148
|
+
oversizedModules: health.oversized,
|
|
149
|
+
};
|
|
150
|
+
}
|
|
151
|
+
catch (err) {
|
|
152
|
+
planeUnavailableReason = err instanceof Error ? err.message : String(err);
|
|
153
|
+
}
|
|
154
|
+
}
|
|
155
|
+
}
|
|
156
|
+
const persisted = persistRetro({
|
|
157
|
+
root: ctx.cwd,
|
|
158
|
+
metrics,
|
|
159
|
+
plane,
|
|
160
|
+
compare,
|
|
161
|
+
now,
|
|
162
|
+
});
|
|
163
|
+
let planePostUrl;
|
|
164
|
+
if (parsed.postPlane) {
|
|
165
|
+
if (!plane) {
|
|
166
|
+
ctx.io.writeError(`pugi retro --post-plane: Plane unavailable (${planeUnavailableReason ?? 'unknown'}).`);
|
|
167
|
+
}
|
|
168
|
+
else {
|
|
169
|
+
const cfgResult = resolvePlaneConfig(ctx.cwd);
|
|
170
|
+
if (cfgResult.ok) {
|
|
171
|
+
try {
|
|
172
|
+
const result = await postRetroToPlane({
|
|
173
|
+
config: cfgResult.config,
|
|
174
|
+
markdown: persisted.markdown,
|
|
175
|
+
sequence: persisted.sequence,
|
|
176
|
+
dateLabel: persisted.dateLabel,
|
|
177
|
+
});
|
|
178
|
+
planePostUrl = result.url;
|
|
179
|
+
if (result.alreadyExists) {
|
|
180
|
+
ctx.io.write(`pugi retro: already exists at ${result.url}\n`);
|
|
181
|
+
}
|
|
182
|
+
}
|
|
183
|
+
catch (err) {
|
|
184
|
+
const msg = err instanceof Error ? err.message : String(err);
|
|
185
|
+
ctx.io.writeError(`pugi retro --post-plane failed: ${msg}`);
|
|
186
|
+
}
|
|
187
|
+
}
|
|
188
|
+
}
|
|
189
|
+
}
|
|
190
|
+
if (ctx.flags.json) {
|
|
191
|
+
ctx.io.write(`${JSON.stringify({
|
|
192
|
+
ok: true,
|
|
193
|
+
markdownPath: persisted.markdownPath,
|
|
194
|
+
jsonPath: persisted.jsonPath,
|
|
195
|
+
sequence: persisted.sequence,
|
|
196
|
+
metrics,
|
|
197
|
+
plane: plane ?? null,
|
|
198
|
+
planePostUrl: planePostUrl ?? null,
|
|
199
|
+
planeUnavailableReason: planeUnavailableReason ?? null,
|
|
200
|
+
}, null, 2)}\n`);
|
|
201
|
+
}
|
|
202
|
+
else {
|
|
203
|
+
renderSummary({ persisted, metrics, plane, planePostUrl });
|
|
204
|
+
if (planeUnavailableReason && !plane) {
|
|
205
|
+
ctx.io.writeError(`pugi retro: Plane integration unavailable (${planeUnavailableReason}).`);
|
|
206
|
+
}
|
|
207
|
+
}
|
|
208
|
+
return 0;
|
|
209
|
+
}
|
|
210
|
+
//# sourceMappingURL=retro.js.map
|
|
@@ -0,0 +1,133 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* `pugi smoke` — runs the bundled scenario corpus through the headless
|
|
3
|
+
* harness and reports pass/fail per scenario (
|
|
4
|
+
*).
|
|
5
|
+
*
|
|
6
|
+
* The CLI surface lives here rather than in `runtime/cli.ts` so the
|
|
7
|
+
* dispatch surface stays focused on argv routing. This module owns:
|
|
8
|
+
*
|
|
9
|
+
* - Resolving the scenarios directory (default
|
|
10
|
+
* `<cli-root>/test/scenarios/` when bundled, configurable via
|
|
11
|
+
* `--scenarios-dir`).
|
|
12
|
+
* - Selecting the `pugi` binary the headless driver should spawn.
|
|
13
|
+
* Local development: `node <cli-root>/bin/run.js`. CI / published
|
|
14
|
+
* usage: the `pugi` on PATH.
|
|
15
|
+
* - Forwarding the orchestrator output to the unified
|
|
16
|
+
* `writeOutput` writer so `--json` mode works without a second
|
|
17
|
+
* code path.
|
|
18
|
+
*
|
|
19
|
+
* Phase 1 deliberately ships ONE flag (`--filter`) and one option
|
|
20
|
+
* (`--scenarios-dir`); the rest of the surface comes in Phase 2 once
|
|
21
|
+
* the engine path is wired and we know which scenarios actually need
|
|
22
|
+
* per-run plumbing (timeouts, fixture credentials, hermetic stubs).
|
|
23
|
+
*/
|
|
24
|
+
import { existsSync } from 'node:fs';
|
|
25
|
+
import { dirname, resolve } from 'node:path';
|
|
26
|
+
import { fileURLToPath } from 'node:url';
|
|
27
|
+
import { runSmoke, renderReportText, } from '../core/smoke/orchestrator.js';
|
|
28
|
+
import { runHeadlessScenario } from '../core/smoke/headless-driver.js';
|
|
29
|
+
/**
|
|
30
|
+
* Entry point invoked by `runtime/cli.ts::dispatchSmoke`. Returns the
|
|
31
|
+
* desired process exit code so the dispatcher can set
|
|
32
|
+
* `process.exitCode` without a second round-trip.
|
|
33
|
+
*/
|
|
34
|
+
export async function runSmokeCommand(ctx) {
|
|
35
|
+
// Parse the (small) command-local argv. We only honor flags this
|
|
36
|
+
// command actually consumes; unknown args produce a usage error so
|
|
37
|
+
// typos surface immediately.
|
|
38
|
+
let scenariosDirOverride = ctx.scenariosDir ?? undefined;
|
|
39
|
+
let filter = ctx.filter ?? '';
|
|
40
|
+
for (let i = 0; i < ctx.args.length; i += 1) {
|
|
41
|
+
const arg = ctx.args[i] ?? '';
|
|
42
|
+
if (arg === '--filter') {
|
|
43
|
+
const next = ctx.args[i + 1];
|
|
44
|
+
if (!next || next.startsWith('--')) {
|
|
45
|
+
ctx.writeOutput({ ok: false, error: '--filter requires a pattern' }, 'pugi smoke: --filter requires a pattern (substring or *-glob)');
|
|
46
|
+
return 2;
|
|
47
|
+
}
|
|
48
|
+
filter = next;
|
|
49
|
+
i += 1;
|
|
50
|
+
}
|
|
51
|
+
else if (arg.startsWith('--filter=')) {
|
|
52
|
+
filter = arg.slice('--filter='.length);
|
|
53
|
+
}
|
|
54
|
+
else if (arg === '--scenarios-dir') {
|
|
55
|
+
const next = ctx.args[i + 1];
|
|
56
|
+
if (!next || next.startsWith('--')) {
|
|
57
|
+
ctx.writeOutput({ ok: false, error: '--scenarios-dir requires a path' }, 'pugi smoke: --scenarios-dir requires a path');
|
|
58
|
+
return 2;
|
|
59
|
+
}
|
|
60
|
+
scenariosDirOverride = next;
|
|
61
|
+
i += 1;
|
|
62
|
+
}
|
|
63
|
+
else if (arg.startsWith('--scenarios-dir=')) {
|
|
64
|
+
scenariosDirOverride = arg.slice('--scenarios-dir='.length);
|
|
65
|
+
}
|
|
66
|
+
else if (arg === '--help' || arg === '-h') {
|
|
67
|
+
ctx.writeOutput({
|
|
68
|
+
ok: true,
|
|
69
|
+
usage: 'pugi smoke [--filter <pattern>] [--scenarios-dir <path>]',
|
|
70
|
+
}, [
|
|
71
|
+
'pugi smoke — run the bundled scenario corpus headlessly.',
|
|
72
|
+
'',
|
|
73
|
+
'Flags:',
|
|
74
|
+
' --filter <pattern> Run a subset (substring or *-glob match on scenario id).',
|
|
75
|
+
' --scenarios-dir <path> Override the scenarios directory (default: bundled corpus).',
|
|
76
|
+
].join('\n'));
|
|
77
|
+
return 0;
|
|
78
|
+
}
|
|
79
|
+
else {
|
|
80
|
+
ctx.writeOutput({ ok: false, error: `unknown arg: ${arg}` }, `pugi smoke: unknown arg ${arg}`);
|
|
81
|
+
return 2;
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
const scenariosDir = scenariosDirOverride ?? resolveBundledScenariosDir();
|
|
85
|
+
if (!existsSync(scenariosDir)) {
|
|
86
|
+
ctx.writeOutput({ ok: false, error: `scenarios dir not found: ${scenariosDir}` }, `pugi smoke: scenarios dir not found: ${scenariosDir}`);
|
|
87
|
+
return 2;
|
|
88
|
+
}
|
|
89
|
+
const pugiBin = ctx.pugiBin ?? process.env.PUGI_SMOKE_BIN ?? 'pugi';
|
|
90
|
+
const log = ctx.log ?? ((line) => process.stderr.write(`${line}\n`));
|
|
91
|
+
const report = await runSmoke({
|
|
92
|
+
scenariosDir,
|
|
93
|
+
filter,
|
|
94
|
+
executor: (scenario) => runHeadlessScenario(scenario, { pugiBin }),
|
|
95
|
+
log,
|
|
96
|
+
});
|
|
97
|
+
if (ctx.json) {
|
|
98
|
+
ctx.writeOutput(report, renderReportText(report));
|
|
99
|
+
}
|
|
100
|
+
else {
|
|
101
|
+
process.stdout.write(`${renderReportText(report)}\n`);
|
|
102
|
+
}
|
|
103
|
+
return report.exitCode;
|
|
104
|
+
}
|
|
105
|
+
/**
|
|
106
|
+
* Resolve the scenarios directory that ships alongside the CLI.
|
|
107
|
+
*
|
|
108
|
+
* Both the dev source layout (`<cli-root>/src/commands/smoke.ts`) and
|
|
109
|
+
* the built output layout (`<cli-root>/dist/commands/smoke.js`) land
|
|
110
|
+
* on the same `../../test/scenarios` path relative to this file.
|
|
111
|
+
* The bundled `npm i -g @pugi/cli` install replicates that structure
|
|
112
|
+
* by shipping `test/scenarios` (glob `**\/*.scenario.txt`) via the
|
|
113
|
+
* `package.json` `files` field — so the single resolved path works in
|
|
114
|
+
* dev, in `tsx` runs, and in published installs without a config knob.
|
|
115
|
+
*
|
|
116
|
+
* If a future restructure ever bundles scenarios at
|
|
117
|
+
* `<cli-root>/dist/scenarios/` instead, add that to the fallback chain
|
|
118
|
+
* here AND mirror the path in `package.json` `files`. Until then we
|
|
119
|
+
* keep the resolver intentionally one-line.
|
|
120
|
+
*/
|
|
121
|
+
export function resolveBundledScenariosDir() {
|
|
122
|
+
const here = dirname(fileURLToPath(import.meta.url));
|
|
123
|
+
// Works for both src/commands/smoke.ts and dist/commands/smoke.js —
|
|
124
|
+
// both live two directories below the cli root.
|
|
125
|
+
const bundled = resolve(here, '..', '..', 'test', 'scenarios');
|
|
126
|
+
if (existsSync(bundled))
|
|
127
|
+
return bundled;
|
|
128
|
+
// Last resort: return the expected path so the orchestrator surfaces
|
|
129
|
+
// a clean "scenarios dir not found" diagnostic at the call site
|
|
130
|
+
// rather than the resolver swallowing it silently.
|
|
131
|
+
return bundled;
|
|
132
|
+
}
|
|
133
|
+
//# sourceMappingURL=smoke.js.map
|
|
@@ -0,0 +1,134 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Agent-progress auto-cleanup — live progress .
|
|
3
|
+
*
|
|
4
|
+
* Rule: a progress file whose status === 'completed' OR 'failed' AND
|
|
5
|
+
* whose `lastUpdate` is older than COMPLETION_TTL_MS (default 5 min)
|
|
6
|
+
* gets MOVED into `<dir>/archive/<id>-<ts>.json` rather than deleted.
|
|
7
|
+
* Operators sometimes want to inspect a finished agent's last state;
|
|
8
|
+
* the archive keeps that affordance while preventing the live watcher
|
|
9
|
+
* from cluttering up with stale rows.
|
|
10
|
+
*
|
|
11
|
+
* The function is pure-ish:
|
|
12
|
+
* - All clock reads go through the injected `now` (defaults Date.now).
|
|
13
|
+
* - Returns a structured report so the caller (the watcher or a CLI
|
|
14
|
+
* cron) can log what got swept.
|
|
15
|
+
* - File-system errors degrade silently — the cleanup is best-effort
|
|
16
|
+
* housekeeping, never a hot path.
|
|
17
|
+
*
|
|
18
|
+
* Hook-in point: `pugi jobs --watch` calls `runCleanup()` once per
|
|
19
|
+
* 60-second tick (cheap — readdir on a small directory). A separate
|
|
20
|
+
* cron entry can call it standalone via the (future) `pugi jobs
|
|
21
|
+
* gc` subcommand if the operator never runs --watch.
|
|
22
|
+
*/
|
|
23
|
+
import { existsSync, mkdirSync, readFileSync, readdirSync, renameSync, } from 'node:fs';
|
|
24
|
+
import { join } from 'node:path';
|
|
25
|
+
import { validateAgentProgress, } from './schema.js';
|
|
26
|
+
import { resolveProgressDir } from './writer.js';
|
|
27
|
+
/** Default time a completed/failed entry sits before getting swept. */
|
|
28
|
+
export const COMPLETION_TTL_MS = 5 * 60 * 1000;
|
|
29
|
+
/** Archive subdirectory under the resolved progress dir. */
|
|
30
|
+
export const ARCHIVE_SUBDIR = 'archive';
|
|
31
|
+
/**
|
|
32
|
+
* Run a single cleanup pass. Returns the report for telemetry / tests.
|
|
33
|
+
*/
|
|
34
|
+
export function runCleanup(options = {}) {
|
|
35
|
+
const dir = resolveProgressDir(options.dir);
|
|
36
|
+
const ttl = options.ttlMs ?? COMPLETION_TTL_MS;
|
|
37
|
+
const now = options.now ?? Date.now;
|
|
38
|
+
const report = { dir, archived: [], skipped: [] };
|
|
39
|
+
if (!existsSync(dir))
|
|
40
|
+
return report;
|
|
41
|
+
const archiveDir = join(dir, ARCHIVE_SUBDIR);
|
|
42
|
+
let entries;
|
|
43
|
+
try {
|
|
44
|
+
entries = readdirSync(dir);
|
|
45
|
+
}
|
|
46
|
+
catch (err) {
|
|
47
|
+
report.skipped.push({ path: dir, reason: `readdir failed: ${err.message}` });
|
|
48
|
+
return report;
|
|
49
|
+
}
|
|
50
|
+
for (const name of entries) {
|
|
51
|
+
if (!name.endsWith('.json'))
|
|
52
|
+
continue;
|
|
53
|
+
if (/\.tmp-/.test(name))
|
|
54
|
+
continue;
|
|
55
|
+
const path = join(dir, name);
|
|
56
|
+
let body;
|
|
57
|
+
try {
|
|
58
|
+
body = readFileSync(path, 'utf8');
|
|
59
|
+
}
|
|
60
|
+
catch {
|
|
61
|
+
report.skipped.push({ path, reason: 'read failed' });
|
|
62
|
+
continue;
|
|
63
|
+
}
|
|
64
|
+
let parsed;
|
|
65
|
+
try {
|
|
66
|
+
parsed = JSON.parse(body);
|
|
67
|
+
}
|
|
68
|
+
catch {
|
|
69
|
+
report.skipped.push({ path, reason: 'malformed JSON' });
|
|
70
|
+
continue;
|
|
71
|
+
}
|
|
72
|
+
const validation = validateAgentProgress(parsed);
|
|
73
|
+
if (!validation.ok) {
|
|
74
|
+
report.skipped.push({ path, reason: `invalid: ${validation.error}` });
|
|
75
|
+
continue;
|
|
76
|
+
}
|
|
77
|
+
const progress = validation.value;
|
|
78
|
+
if (!isExpired(progress, now(), ttl)) {
|
|
79
|
+
continue;
|
|
80
|
+
}
|
|
81
|
+
if (!existsSync(archiveDir)) {
|
|
82
|
+
try {
|
|
83
|
+
mkdirSync(archiveDir, { recursive: true });
|
|
84
|
+
}
|
|
85
|
+
catch (err) {
|
|
86
|
+
report.skipped.push({
|
|
87
|
+
path,
|
|
88
|
+
reason: `archive mkdir failed: ${err.message}`,
|
|
89
|
+
});
|
|
90
|
+
continue;
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
const target = join(archiveDir, `${progress.agentId}-${safeStamp(progress.lastUpdate)}.json`);
|
|
94
|
+
try {
|
|
95
|
+
renameSync(path, target);
|
|
96
|
+
report.archived.push({ agentId: progress.agentId, from: path, to: target });
|
|
97
|
+
}
|
|
98
|
+
catch (err) {
|
|
99
|
+
report.skipped.push({ path, reason: `rename failed: ${err.message}` });
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
return report;
|
|
103
|
+
}
|
|
104
|
+
/**
|
|
105
|
+
* Decide whether a single progress doc has aged out. Exported for the
|
|
106
|
+
* spec — kept pure so tests can probe edges (running entries never
|
|
107
|
+
* expire, completed entries before TTL stay, etc).
|
|
108
|
+
*/
|
|
109
|
+
export function isExpired(progress, nowEpochMs, ttlMs = COMPLETION_TTL_MS) {
|
|
110
|
+
if (progress.status === 'running')
|
|
111
|
+
return false;
|
|
112
|
+
const lastTs = Date.parse(progress.lastUpdate);
|
|
113
|
+
if (Number.isNaN(lastTs))
|
|
114
|
+
return false;
|
|
115
|
+
return nowEpochMs - lastTs >= ttlMs;
|
|
116
|
+
}
|
|
117
|
+
// Claude review followup: `pruneArchive` was removed.
|
|
118
|
+
// Rationale (P1 in the Claude review batch on PR):
|
|
119
|
+
// - hardcoded `/tmp` path was POSIX-only (Windows would break),
|
|
120
|
+
// - `renameSync` across volumes throws EXDEV,
|
|
121
|
+
// - collision risk between concurrent hosts/sessions sharing `/tmp`,
|
|
122
|
+
// - and crucially, the function had ZERO call-sites in the codebase.
|
|
123
|
+
// Deleting it is strictly safer than leaving a broken-on-Windows
|
|
124
|
+
// dead helper that future refactors might accidentally wire up.
|
|
125
|
+
// If/when an archive GC is needed, the right shape is:
|
|
126
|
+
// - `os.tmpdir()` (not '/tmp'),
|
|
127
|
+
// - `rmSync(path, {force: true})` for cross-volume safe delete,
|
|
128
|
+
// - exposed as `pugi jobs gc` so we have an explicit caller.
|
|
129
|
+
function safeStamp(iso) {
|
|
130
|
+
// Build a filename-safe slug — strip colons/dots which are friendly
|
|
131
|
+
// в ISO timestamps but hostile к some filesystems.
|
|
132
|
+
return iso.replace(/[:.]/g, '-');
|
|
133
|
+
}
|
|
134
|
+
//# sourceMappingURL=cleanup.js.map
|
|
@@ -0,0 +1,144 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Agent progress JSON schema — live-progress sprint .
|
|
3
|
+
*
|
|
4
|
+
* Long-running agents (spawned externally OR via `pugi /agent`) emit a
|
|
5
|
+
* single JSON document per agent to `~/.pugi/agent-progress/<id>.json`.
|
|
6
|
+
* `pugi jobs --watch` tails the directory via chokidar and re-renders
|
|
7
|
+
* an Ink TUI that mirrors the the upstream tool `/compact` visual pattern
|
|
8
|
+
* (header with elapsed + token counter, unicode progress bar, milestone
|
|
9
|
+
* list with done/active/pending status icons).
|
|
10
|
+
*
|
|
11
|
+
* Schema is intentionally optimistic — every numeric field is clamped
|
|
12
|
+
* by the writer/reader so a malformed document degrades to a partial
|
|
13
|
+
* card instead of crashing the watcher. The `pendingCount` /
|
|
14
|
+
* `completedCount` fields are pre-computed by the agent for the
|
|
15
|
+
* "… +N pending, M completed" footer; the renderer never re-counts
|
|
16
|
+
* (the agent may have collapsed milestone history to save bytes).
|
|
17
|
+
*/
|
|
18
|
+
/**
|
|
19
|
+
* Validate an unknown payload as an `AgentProgress` document. Returns
|
|
20
|
+
* the typed value on success and a string error otherwise. We deliberately
|
|
21
|
+
* keep this hand-rolled (no zod) — every field is checked exactly once,
|
|
22
|
+
* the error message is human-readable, and zero runtime deps.
|
|
23
|
+
*/
|
|
24
|
+
export function validateAgentProgress(value) {
|
|
25
|
+
if (typeof value !== 'object' || value === null) {
|
|
26
|
+
return { ok: false, error: 'progress payload must be a JSON object' };
|
|
27
|
+
}
|
|
28
|
+
const raw = value;
|
|
29
|
+
const requiredString = (field) => {
|
|
30
|
+
const v = raw[field];
|
|
31
|
+
return typeof v === 'string' && v.length > 0 ? v : null;
|
|
32
|
+
};
|
|
33
|
+
const agentId = requiredString('agentId');
|
|
34
|
+
if (!agentId)
|
|
35
|
+
return { ok: false, error: 'agentId required (non-empty string)' };
|
|
36
|
+
if (!/^[a-zA-Z0-9_-]+$/.test(agentId)) {
|
|
37
|
+
return { ok: false, error: 'agentId must match [a-zA-Z0-9_-]+ (filename-safe)' };
|
|
38
|
+
}
|
|
39
|
+
const agentType = requiredString('agentType');
|
|
40
|
+
if (!agentType)
|
|
41
|
+
return { ok: false, error: 'agentType required (non-empty string)' };
|
|
42
|
+
const task = requiredString('task');
|
|
43
|
+
if (!task)
|
|
44
|
+
return { ok: false, error: 'task required (non-empty string)' };
|
|
45
|
+
const startedAt = requiredString('startedAt');
|
|
46
|
+
if (!startedAt)
|
|
47
|
+
return { ok: false, error: 'startedAt required (ISO string)' };
|
|
48
|
+
if (Number.isNaN(Date.parse(startedAt))) {
|
|
49
|
+
return { ok: false, error: 'startedAt must be a parseable ISO timestamp' };
|
|
50
|
+
}
|
|
51
|
+
const lastUpdate = requiredString('lastUpdate');
|
|
52
|
+
if (!lastUpdate)
|
|
53
|
+
return { ok: false, error: 'lastUpdate required (ISO string)' };
|
|
54
|
+
if (Number.isNaN(Date.parse(lastUpdate))) {
|
|
55
|
+
return { ok: false, error: 'lastUpdate must be a parseable ISO timestamp' };
|
|
56
|
+
}
|
|
57
|
+
const elapsedMs = raw.elapsedMs;
|
|
58
|
+
if (typeof elapsedMs !== 'number' || !Number.isFinite(elapsedMs) || elapsedMs < 0) {
|
|
59
|
+
return { ok: false, error: 'elapsedMs required (non-negative number)' };
|
|
60
|
+
}
|
|
61
|
+
const percentComplete = raw.percentComplete;
|
|
62
|
+
if (typeof percentComplete !== 'number' ||
|
|
63
|
+
!Number.isFinite(percentComplete)) {
|
|
64
|
+
return { ok: false, error: 'percentComplete required (number 0..100)' };
|
|
65
|
+
}
|
|
66
|
+
const clampedPercent = Math.max(0, Math.min(100, percentComplete));
|
|
67
|
+
const status = raw.status;
|
|
68
|
+
if (status !== 'running' && status !== 'completed' && status !== 'failed') {
|
|
69
|
+
return { ok: false, error: 'status must be running | completed | failed' };
|
|
70
|
+
}
|
|
71
|
+
const currentStep = raw.currentStep;
|
|
72
|
+
if (typeof currentStep !== 'number' || currentStep < 0) {
|
|
73
|
+
return { ok: false, error: 'currentStep required (non-negative number)' };
|
|
74
|
+
}
|
|
75
|
+
const totalSteps = raw.totalSteps;
|
|
76
|
+
if (typeof totalSteps !== 'number' || totalSteps < 0) {
|
|
77
|
+
return { ok: false, error: 'totalSteps required (non-negative number)' };
|
|
78
|
+
}
|
|
79
|
+
const stepDescription = typeof raw.stepDescription === 'string'
|
|
80
|
+
? raw.stepDescription
|
|
81
|
+
: '';
|
|
82
|
+
const milestonesRaw = raw.milestones;
|
|
83
|
+
if (!Array.isArray(milestonesRaw)) {
|
|
84
|
+
return { ok: false, error: 'milestones required (array, may be empty)' };
|
|
85
|
+
}
|
|
86
|
+
const milestones = [];
|
|
87
|
+
for (let i = 0; i < milestonesRaw.length; i += 1) {
|
|
88
|
+
const m = milestonesRaw[i];
|
|
89
|
+
if (typeof m !== 'object' || m === null) {
|
|
90
|
+
return { ok: false, error: `milestones[${i}] must be an object` };
|
|
91
|
+
}
|
|
92
|
+
const mr = m;
|
|
93
|
+
const label = typeof mr.label === 'string' ? mr.label : '';
|
|
94
|
+
if (!label) {
|
|
95
|
+
return { ok: false, error: `milestones[${i}].label required` };
|
|
96
|
+
}
|
|
97
|
+
const mStatus = mr.status;
|
|
98
|
+
if (mStatus !== 'done' && mStatus !== 'active' && mStatus !== 'pending') {
|
|
99
|
+
return {
|
|
100
|
+
ok: false,
|
|
101
|
+
error: `milestones[${i}].status must be done | active | pending`,
|
|
102
|
+
};
|
|
103
|
+
}
|
|
104
|
+
const ts = typeof mr.ts === 'string' ? mr.ts : undefined;
|
|
105
|
+
milestones.push({ label, status: mStatus, ts });
|
|
106
|
+
}
|
|
107
|
+
const tokensUsed = typeof raw.tokensUsed === 'number' && Number.isFinite(raw.tokensUsed)
|
|
108
|
+
? Math.max(0, raw.tokensUsed)
|
|
109
|
+
: undefined;
|
|
110
|
+
const pendingCount = typeof raw.pendingCount === 'number' && Number.isFinite(raw.pendingCount)
|
|
111
|
+
? Math.max(0, Math.floor(raw.pendingCount))
|
|
112
|
+
: undefined;
|
|
113
|
+
const completedCount = typeof raw.completedCount === 'number' && Number.isFinite(raw.completedCount)
|
|
114
|
+
? Math.max(0, Math.floor(raw.completedCount))
|
|
115
|
+
: undefined;
|
|
116
|
+
return {
|
|
117
|
+
ok: true,
|
|
118
|
+
value: {
|
|
119
|
+
agentId,
|
|
120
|
+
agentType,
|
|
121
|
+
task,
|
|
122
|
+
startedAt,
|
|
123
|
+
lastUpdate,
|
|
124
|
+
elapsedMs,
|
|
125
|
+
tokensUsed,
|
|
126
|
+
percentComplete: clampedPercent,
|
|
127
|
+
status,
|
|
128
|
+
currentStep,
|
|
129
|
+
totalSteps,
|
|
130
|
+
stepDescription,
|
|
131
|
+
milestones,
|
|
132
|
+
pendingCount,
|
|
133
|
+
completedCount,
|
|
134
|
+
},
|
|
135
|
+
};
|
|
136
|
+
}
|
|
137
|
+
/**
|
|
138
|
+
* Default directory the writer and watcher share. Lives under the
|
|
139
|
+
* project workspace by convention so worktree-isolated agents can emit
|
|
140
|
+
* progress without crossing tenant boundaries. Operators can override
|
|
141
|
+
* via `PUGI_AGENT_PROGRESS_DIR` env var.
|
|
142
|
+
*/
|
|
143
|
+
export const DEFAULT_AGENT_PROGRESS_DIRNAME = '.pugi/agent-progress';
|
|
144
|
+
//# sourceMappingURL=schema.js.map
|