@pugi/cli 0.1.0-beta.8 → 0.1.0-beta.87
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +96 -0
- package/THIRD_PARTY_NOTICES.md +40 -0
- package/assets/pugi-prozr2-mascot.ansi +9 -0
- package/bin/run.js +33 -1
- package/dist/commands/deploy.js +40 -40
- package/dist/commands/flatten.js +191 -0
- package/dist/commands/jobs-watch.js +201 -0
- package/dist/commands/jobs.js +42 -27
- package/dist/commands/smoke.js +133 -0
- package/dist/core/agent-progress/cleanup.js +134 -0
- package/dist/core/agent-progress/schema.js +144 -0
- package/dist/core/agent-progress/writer.js +101 -0
- package/dist/core/agents/adaptive-router.js +330 -0
- package/dist/core/agents/query-decomposer.js +297 -0
- package/dist/core/agents/registry.js +2 -2
- package/dist/core/approvals/shortcut-resolver.js +98 -0
- package/dist/core/artifact-chain/dispatcher.js +148 -0
- package/dist/core/artifact-chain/exporter.js +164 -0
- package/dist/core/artifact-chain/state.js +243 -0
- package/dist/core/artifact-chain/steps.js +169 -0
- package/dist/core/ask-user/question.js +92 -0
- package/dist/core/audit/audit-trail.js +275 -0
- package/dist/core/auth/ensure-authenticated.js +129 -0
- package/dist/core/auth/env-provider.js +238 -0
- package/dist/core/auto-open-browser.js +4 -4
- package/dist/core/auto-update/channels.js +122 -0
- package/dist/core/auto-update/checker.js +241 -0
- package/dist/core/auto-update/state.js +235 -0
- package/dist/core/bare-mode/index.js +107 -0
- package/dist/core/bash/redirect.js +281 -0
- package/dist/core/bash-classifier.js +436 -40
- package/dist/core/checkpoint/resumer.js +149 -0
- package/dist/core/checkpoint/rewinder.js +291 -0
- package/dist/core/checkpoints/shadow-git.js +670 -0
- package/dist/core/citations/parser.js +109 -0
- package/dist/core/classifier/yolo-classifier.js +88 -0
- package/dist/core/codegraph/decision-store.js +248 -0
- package/dist/core/codegraph/detect-repo.js +459 -0
- package/dist/core/codegraph/install.js +134 -0
- package/dist/core/codegraph/offer-hook.js +220 -0
- package/dist/core/compact/auto-trigger.js +96 -0
- package/dist/core/compact/buffer-rewriter.js +115 -0
- package/dist/core/compact/summarizer.js +208 -0
- package/dist/core/compact/token-counter.js +108 -0
- package/dist/core/consensus/anvil-fanout.js +25 -25
- package/dist/core/consensus/diff-capture.js +121 -12
- package/dist/core/consensus/rubric.js +21 -21
- package/dist/core/context/builder.js +6 -6
- package/dist/core/context/compaction-events.js +8 -8
- package/dist/core/context/compaction.js +31 -31
- package/dist/core/context/index.js +15 -8
- package/dist/core/context/invariants.js +51 -51
- package/dist/core/context/markdown-loader.js +28 -10
- package/dist/core/context/markdown-traverse.js +255 -0
- package/dist/core/context/pugiignore.js +41 -41
- package/dist/core/context/repo-skeleton.js +37 -37
- package/dist/core/context/tool-eviction.js +55 -0
- package/dist/core/context/watcher.js +32 -32
- package/dist/core/context/working-set.js +23 -23
- package/dist/core/coordinator/agent-tools.js +77 -0
- package/dist/core/coordinator/agent-toolset.js +65 -0
- package/dist/core/coordinator/fsm.js +73 -0
- package/dist/core/coordinator/mode-fsm.js +70 -0
- package/dist/core/cost/rate-card.js +129 -0
- package/dist/core/cost/tracker.js +221 -0
- package/dist/core/credentials.js +12 -12
- package/dist/core/cron/scheduler.js +138 -0
- package/dist/core/denial-tracking/index.js +8 -0
- package/dist/core/denial-tracking/state.js +264 -0
- package/dist/core/diagnostics/probe-runner.js +93 -0
- package/dist/core/diagnostics/probes/api.js +46 -0
- package/dist/core/diagnostics/probes/auth.js +93 -0
- package/dist/core/diagnostics/probes/bare-mode.js +42 -0
- package/dist/core/diagnostics/probes/cli-version.js +127 -0
- package/dist/core/diagnostics/probes/config.js +72 -0
- package/dist/core/diagnostics/probes/denial-tracking.js +57 -0
- package/dist/core/diagnostics/probes/disk.js +81 -0
- package/dist/core/diagnostics/probes/engine-live.js +46 -0
- package/dist/core/diagnostics/probes/git.js +65 -0
- package/dist/core/diagnostics/probes/hooks.js +118 -0
- package/dist/core/diagnostics/probes/mcp.js +75 -0
- package/dist/core/diagnostics/probes/node.js +59 -0
- package/dist/core/diagnostics/probes/pnpm.js +36 -0
- package/dist/core/diagnostics/probes/pugi-md.js +89 -0
- package/dist/core/diagnostics/probes/sandbox.js +40 -0
- package/dist/core/diagnostics/probes/session.js +74 -0
- package/dist/core/diagnostics/probes/status-snapshot.js +488 -0
- package/dist/core/diagnostics/probes/workspace.js +63 -0
- package/dist/core/diagnostics/types.js +70 -0
- package/dist/core/dispatch/cache-cleanup.js +197 -0
- package/dist/core/dispatch/cache-handoff.js +295 -0
- package/dist/core/edits/apply-patch-layer-e.js +189 -0
- package/dist/core/edits/dispatch.js +293 -7
- package/dist/core/edits/format-matrix.js +26 -0
- package/dist/core/edits/fuzzy-ladder.js +650 -0
- package/dist/core/edits/index.js +3 -1
- package/dist/core/edits/journal.js +199 -0
- package/dist/core/edits/layer-a-apply.js +15 -15
- package/dist/core/edits/layer-a-fuzzy-apply.js +198 -0
- package/dist/core/edits/layer-b-apply.js +9 -9
- package/dist/core/edits/layer-c-apply.js +6 -6
- package/dist/core/edits/layer-d-ast.js +557 -14
- package/dist/core/edits/marker-parser.js +12 -12
- package/dist/core/edits/security-gate.js +27 -27
- package/dist/core/edits/verify-hook.js +273 -0
- package/dist/core/edits/worktree.js +322 -0
- package/dist/core/engine/anvil-client.js +140 -26
- package/dist/core/engine/auto-compact.js +179 -0
- package/dist/core/engine/budgets.js +186 -0
- package/dist/core/engine/context-prefix.js +155 -0
- package/dist/core/engine/index.js +1 -1
- package/dist/core/engine/intensity.js +158 -0
- package/dist/core/engine/intent.js +260 -0
- package/dist/core/engine/native-pugi.js +1295 -227
- package/dist/core/engine/prompts.js +134 -16
- package/dist/core/engine/strip-internal-fields.js +124 -0
- package/dist/core/engine/tool-bridge.js +1295 -59
- package/dist/core/evaluation/golden-dataset.js +293 -0
- package/dist/core/feedback/queue.js +177 -0
- package/dist/core/feedback/submitter.js +145 -0
- package/dist/core/file-cache.js +113 -1
- package/dist/core/flatten/flatten-repo.js +439 -0
- package/dist/core/format/osc8-link.js +28 -0
- package/dist/core/hook-chains.js +392 -0
- package/dist/core/hooks/citation-verify-hook.js +138 -0
- package/dist/core/hooks/citation-verify.js +112 -0
- package/dist/core/hooks/events.js +44 -0
- package/dist/core/hooks/index.js +15 -0
- package/dist/core/hooks/registry.js +213 -0
- package/dist/core/hooks/runner.js +236 -0
- package/dist/core/hooks/v2/event-emitter.js +115 -0
- package/dist/core/hooks/v2/executor.js +282 -0
- package/dist/core/hooks/v2/index.js +25 -0
- package/dist/core/hooks/v2/lifecycle.js +104 -0
- package/dist/core/hooks/v2/loader.js +216 -0
- package/dist/core/hooks/v2/matcher.js +125 -0
- package/dist/core/hooks/v2/trust.js +143 -0
- package/dist/core/hooks/v2/types.js +86 -0
- package/dist/core/image/renderer.js +71 -0
- package/dist/core/init/detector.js +582 -0
- package/dist/core/init/template-renderer.js +242 -0
- package/dist/core/jobs/registry.js +18 -18
- package/dist/core/ledger/results-tsv.js +142 -0
- package/dist/core/log-discipline/stdout-redirect.js +51 -0
- package/dist/core/lsp/cache.js +105 -0
- package/dist/core/lsp/client.js +776 -0
- package/dist/core/lsp/language-detect.js +66 -0
- package/dist/core/lsp/post-edit-diagnostics.js +171 -0
- package/dist/core/lsp/symbol-tools.js +372 -0
- package/dist/core/mcp/client.js +97 -28
- package/dist/core/mcp/http-server.js +553 -0
- package/dist/core/mcp/orchestrator-tools.js +662 -0
- package/dist/core/mcp/permission.js +190 -0
- package/dist/core/mcp/registry.js +39 -17
- package/dist/core/mcp/server-tools.js +219 -0
- package/dist/core/mcp/server.js +397 -0
- package/dist/core/mcp/trust.js +10 -10
- package/dist/core/memory/dual-write.js +416 -0
- package/dist/core/memory/passive-extract.js +130 -0
- package/dist/core/memory/phase1-kinds.js +20 -0
- package/dist/core/memory/secret-scanner.js +304 -0
- package/dist/core/memory-sync/queue.js +170 -0
- package/dist/core/metrics/extract.js +113 -0
- package/dist/core/modes/roo-modes.js +68 -0
- package/dist/core/onboarding/ensure-initialized.js +133 -0
- package/dist/core/onboarding/marker.js +111 -0
- package/dist/core/onboarding/telemetry-state.js +108 -0
- package/dist/core/output-style/presets.js +176 -0
- package/dist/core/output-style/state.js +185 -0
- package/dist/core/path-security.js +287 -5
- package/dist/core/permission.js +82 -22
- package/dist/core/permissions/auto-classifier.js +124 -0
- package/dist/core/permissions/bash-parser.js +371 -0
- package/dist/core/permissions/circuit-breaker.js +83 -0
- package/dist/core/permissions/constrained-edit.js +91 -0
- package/dist/core/permissions/gate.js +278 -0
- package/dist/core/permissions/index.js +20 -0
- package/dist/core/permissions/mode.js +174 -0
- package/dist/core/permissions/network-egress.js +137 -0
- package/dist/core/permissions/state.js +241 -0
- package/dist/core/permissions/tool-class.js +93 -0
- package/dist/core/plan-mode/ui-state.js +51 -0
- package/dist/core/plans/plan-artifact.js +721 -0
- package/dist/core/policy-limits/etag-store.js +122 -0
- package/dist/core/prd-check/parser.js +215 -0
- package/dist/core/prd-check/reporter.js +127 -0
- package/dist/core/prd-check/session-review.js +557 -0
- package/dist/core/prd-check/verifiers.js +223 -0
- package/dist/core/prompt-cache/client-cache.js +99 -0
- package/dist/core/prompts/assembly.js +29 -0
- package/dist/core/prompts/registry.js +364 -0
- package/dist/core/pugi-md/cc-compat-rules.js +735 -0
- package/dist/core/pugi-md/context-injector.js +76 -0
- package/dist/core/pugi-md/walk-up.js +207 -0
- package/dist/core/python/uv-installer.js +270 -0
- package/dist/core/python/uv-resolver.js +83 -0
- package/dist/core/rate-limit/narrator.js +146 -0
- package/dist/core/recipes/cli-types.js +20 -0
- package/dist/core/recipes/loader.js +103 -0
- package/dist/core/recipes/runner.js +345 -0
- package/dist/core/recipes/schema.js +587 -0
- package/dist/core/release-notes/parser.js +241 -0
- package/dist/core/release-notes/state.js +116 -0
- package/dist/core/repl/ask.js +37 -37
- package/dist/core/repl/cancellation.js +26 -26
- package/dist/core/repl/cap-warning.js +4 -4
- package/dist/core/repl/clipboard-read.js +11 -11
- package/dist/core/repl/dispatch-fsm.js +12 -12
- package/dist/core/repl/history-search.js +15 -15
- package/dist/core/repl/history.js +28 -18
- package/dist/core/repl/kill-ring.js +5 -5
- package/dist/core/repl/model-pricing.js +135 -0
- package/dist/core/repl/privacy-banner.js +22 -22
- package/dist/core/repl/session.js +2157 -214
- package/dist/core/repl/slash-commands.js +533 -40
- package/dist/core/repl/store/index.js +1 -1
- package/dist/core/repl/store/jsonl-log.js +22 -22
- package/dist/core/repl/store/lockfile.js +10 -10
- package/dist/core/repl/store/session-store.js +136 -107
- package/dist/core/repl/store/types.js +15 -15
- package/dist/core/repl/store/uuid-v7.js +12 -12
- package/dist/core/repl/workspace-context.js +43 -21
- package/dist/core/repo-map/build.js +125 -0
- package/dist/core/repo-map/cache.js +185 -0
- package/dist/core/repo-map/extractor.js +254 -0
- package/dist/core/repo-map/formatter.js +145 -0
- package/dist/core/repo-map/page-rank.js +105 -0
- package/dist/core/repo-map/scanner.js +211 -0
- package/dist/core/retry-budget/budget.js +284 -0
- package/dist/core/retry-budget/index.js +5 -0
- package/dist/core/retry-budget/retry-cap.js +74 -0
- package/dist/core/routing/lead-worker.js +43 -0
- package/dist/core/routing/pre-flight-estimator.js +108 -0
- package/dist/core/runs/run-tree.js +103 -0
- package/dist/core/security/injection-scanner.js +367 -0
- package/dist/core/security/output-filter.js +418 -0
- package/dist/core/session/env-file.js +105 -0
- package/dist/core/session/section-budgets.js +140 -0
- package/dist/core/session.js +92 -0
- package/dist/core/settings.js +286 -5
- package/dist/core/share/formatter.js +271 -0
- package/dist/core/share/redactor.js +221 -0
- package/dist/core/share/uploader.js +267 -0
- package/dist/core/skills/defaults.js +457 -0
- package/dist/core/skills/loader.js +22 -22
- package/dist/core/skills/sources.js +27 -27
- package/dist/core/smoke/headless-driver.js +174 -0
- package/dist/core/smoke/orchestrator.js +194 -0
- package/dist/core/smoke/runner.js +238 -0
- package/dist/core/smoke/scenario-parser.js +316 -0
- package/dist/core/statusline.js +99 -0
- package/dist/core/subagents/dispatcher-real.js +600 -0
- package/dist/core/subagents/dispatcher.js +132 -43
- package/dist/core/subagents/index.js +19 -6
- package/dist/core/subagents/isolation-matrix.js +213 -0
- package/dist/core/subagents/spawn.js +19 -4
- package/dist/core/telemetry/emitter.js +229 -0
- package/dist/core/telemetry/queue.js +251 -0
- package/dist/core/theme/context.js +91 -0
- package/dist/core/theme/presets.js +228 -0
- package/dist/core/theme/state.js +181 -0
- package/dist/core/todos/invariant.js +10 -0
- package/dist/core/todos/state.js +177 -0
- package/dist/core/tool-schema/compressor.js +89 -0
- package/dist/core/transport/version-interceptor.js +166 -0
- package/dist/core/trust.js +2 -2
- package/dist/core/tui/thinking-block.js +64 -0
- package/dist/core/vim/keymap.js +288 -0
- package/dist/core/vim/state.js +92 -0
- package/dist/core/watch-markers/marker-watcher.js +133 -0
- package/dist/core/worktree-manager/cleanup.js +123 -0
- package/dist/core/worktree-manager/manager.js +303 -0
- package/dist/index.js +28 -0
- package/dist/runtime/bootstrap.js +190 -0
- package/dist/runtime/cli.js +4151 -489
- package/dist/runtime/commands/agents.js +30 -30
- package/dist/runtime/commands/budget.js +5 -5
- package/dist/runtime/commands/cancel.js +231 -0
- package/dist/runtime/commands/chain.js +489 -0
- package/dist/runtime/commands/codegraph-status.js +227 -0
- package/dist/runtime/commands/compact.js +297 -0
- package/dist/runtime/commands/config.js +32 -32
- package/dist/runtime/commands/cost.js +199 -0
- package/dist/runtime/commands/delegate.js +244 -13
- package/dist/runtime/commands/dispatch.js +126 -0
- package/dist/runtime/commands/doctor.js +579 -0
- package/dist/runtime/commands/feedback.js +184 -0
- package/dist/runtime/commands/hooks.js +184 -0
- package/dist/runtime/commands/init.js +254 -0
- package/dist/runtime/commands/lsp.js +368 -0
- package/dist/runtime/commands/mcp.js +879 -0
- package/dist/runtime/commands/memory.js +582 -0
- package/dist/runtime/commands/model.js +237 -0
- package/dist/runtime/commands/onboarding.js +275 -0
- package/dist/runtime/commands/patch.js +128 -0
- package/dist/runtime/commands/permissions.js +112 -0
- package/dist/runtime/commands/plan.js +143 -0
- package/dist/runtime/commands/prd-check.js +285 -0
- package/dist/runtime/commands/privacy.js +17 -17
- package/dist/runtime/commands/recipe.js +325 -0
- package/dist/runtime/commands/redo-blob-store.js +92 -0
- package/dist/runtime/commands/redo.js +361 -0
- package/dist/runtime/commands/release-notes.js +229 -0
- package/dist/runtime/commands/repo-map.js +95 -0
- package/dist/runtime/commands/report.js +299 -0
- package/dist/runtime/commands/resume.js +118 -0
- package/dist/runtime/commands/review-consensus.js +68 -53
- package/dist/runtime/commands/rewind.js +333 -0
- package/dist/runtime/commands/roster.js +14 -14
- package/dist/runtime/commands/sessions.js +163 -0
- package/dist/runtime/commands/share.js +316 -0
- package/dist/runtime/commands/skills.js +31 -31
- package/dist/runtime/commands/status.js +186 -0
- package/dist/runtime/commands/stickers.js +82 -0
- package/dist/runtime/commands/style.js +194 -0
- package/dist/runtime/commands/theme.js +196 -0
- package/dist/runtime/commands/undo.js +54 -22
- package/dist/runtime/commands/update.js +289 -0
- package/dist/runtime/commands/vim.js +140 -0
- package/dist/runtime/commands/worktree.js +177 -0
- package/dist/runtime/commands/worktrees.js +155 -0
- package/dist/runtime/headless-repl.js +195 -0
- package/dist/runtime/headless.js +543 -0
- package/dist/runtime/load-hooks-or-exit.js +71 -0
- package/dist/runtime/plan-decompose.js +531 -0
- package/dist/runtime/update-check.js +28 -28
- package/dist/runtime/version.js +65 -0
- package/dist/skills/bundled/batch.js +617 -0
- package/dist/skills/bundled/index.js +45 -0
- package/dist/skills/bundled/loop.js +358 -0
- package/dist/skills/bundled/remember.js +383 -0
- package/dist/skills/bundled/simplify.js +289 -0
- package/dist/skills/bundled/skillify.js +373 -0
- package/dist/skills/bundled/stuck.js +558 -0
- package/dist/skills/bundled/verify.js +439 -0
- package/dist/testing/vcr.js +486 -0
- package/dist/tools/agent-tool.js +229 -0
- package/dist/tools/apply-patch.js +556 -0
- package/dist/tools/ask-user-question.js +222 -0
- package/dist/tools/ask-user.js +115 -0
- package/dist/tools/bash.js +623 -45
- package/dist/tools/brief.js +224 -0
- package/dist/tools/enter-worktree.js +250 -0
- package/dist/tools/exit-worktree.js +147 -0
- package/dist/tools/file-tools.js +161 -44
- package/dist/tools/lsp-tools.js +189 -0
- package/dist/tools/mcp-tool.js +260 -0
- package/dist/tools/multi-edit.js +361 -0
- package/dist/tools/powershell.js +268 -0
- package/dist/tools/registry.js +85 -0
- package/dist/tools/skill-tool.js +96 -0
- package/dist/tools/sleep.js +99 -0
- package/dist/tools/synthetic-output.js +133 -0
- package/dist/tools/tasks.js +208 -0
- package/dist/tools/todo-write.js +184 -0
- package/dist/tools/verify-plan-execution.js +295 -0
- package/dist/tools/web-fetch-injection-scanner.js +207 -0
- package/dist/tools/web-fetch.js +195 -10
- package/dist/tools/web-search.js +458 -0
- package/dist/tui/agent-progress-card.js +111 -0
- package/dist/tui/agent-tree.js +11 -1
- package/dist/tui/ask-modal.js +14 -14
- package/dist/tui/ask-user-question-prompt.js +203 -0
- package/dist/tui/compact-banner.js +81 -0
- package/dist/tui/conversation-pane.js +85 -11
- package/dist/tui/cost-table.js +111 -0
- package/dist/tui/device-flow.js +2 -2
- package/dist/tui/doctor-table.js +46 -0
- package/dist/tui/feedback-prompt.js +156 -0
- package/dist/tui/input-box.js +247 -32
- package/dist/tui/login-picker.js +3 -3
- package/dist/tui/markdown-render.js +6 -6
- package/dist/tui/onboarding-wizard.js +240 -0
- package/dist/tui/permissions-picker.js +86 -0
- package/dist/tui/render.js +35 -0
- package/dist/tui/repl-render.js +332 -54
- package/dist/tui/repl-splash-art.js +16 -16
- package/dist/tui/repl-splash-mascot.js +48 -24
- package/dist/tui/repl-splash.js +22 -22
- package/dist/tui/repl.js +124 -44
- package/dist/tui/slash-palette.js +6 -6
- package/dist/tui/splash.js +2 -2
- package/dist/tui/status-bar.js +109 -31
- package/dist/tui/status-table.js +7 -0
- package/dist/tui/stickers-art.js +136 -0
- package/dist/tui/style-table.js +28 -0
- package/dist/tui/theme-table.js +29 -0
- package/dist/tui/thinking-spinner.js +123 -0
- package/dist/tui/tool-stream-pane.js +53 -4
- package/dist/tui/update-banner.js +27 -2
- package/dist/tui/vim-input.js +267 -0
- package/dist/tui/welcome-banner.js +107 -0
- package/dist/tui/welcome-data.js +293 -0
- package/dist/tui/workspace-context.js +2 -2
- package/docs/examples/codegraph.mcp.json +10 -0
- package/package.json +23 -6
- package/test/scenarios/codegen-create-file.scenario.txt +13 -0
- package/test/scenarios/compact-force.scenario.txt +11 -0
- package/test/scenarios/identity.scenario.txt +11 -0
- package/test/scenarios/persona-handoff.scenario.txt +11 -0
- package/test/scenarios/walkback.scenario.txt +12 -0
- package/dist/core/engine/compaction-hook.js +0 -154
|
@@ -0,0 +1,227 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* `/codegraph-status` runner — .
|
|
3
|
+
*
|
|
4
|
+
* Single source of truth for the codegraph adoption + index-freshness
|
|
5
|
+
* surface. Used by:
|
|
6
|
+
*
|
|
7
|
+
* - REPL slash `/codegraph-status` (the discoverable surface)
|
|
8
|
+
* - REPL slash `/codegraph` (short alias)
|
|
9
|
+
*
|
|
10
|
+
* The runner is intentionally narrow — every information surface
|
|
11
|
+
* (installed? / index age / symbol count / last reindex / refresh CTA)
|
|
12
|
+
* is computed from the workspace-local `.pugi/codegraph-decision.json`
|
|
13
|
+
* + the live `.pugi/mcp.json` + the bounded scanner at
|
|
14
|
+
* `core/codegraph/detect-repo.ts`. NO network round-trip; everything
|
|
15
|
+
* runs offline so an air-gapped operator still gets the status table.
|
|
16
|
+
*
|
|
17
|
+
* Flags:
|
|
18
|
+
* --install — merge codegraph into .pugi/mcp.json (accept decision)
|
|
19
|
+
* --reindex — stamp lastIndexedAt now + hint operator к run
|
|
20
|
+
* `codegraph index` from a fresh shell. Pugi does NOT
|
|
21
|
+
* spawn codegraph itself (no upstream dep guarantee).
|
|
22
|
+
* --offer — surface the install prompt even after a decline,
|
|
23
|
+
* useful when the operator declined three weeks ago and
|
|
24
|
+
* wants to revisit без waiting for the 30-day cadence.
|
|
25
|
+
*
|
|
26
|
+
* Brand voice gate: ASCII output, no emoji, no decorative dividers.
|
|
27
|
+
*/
|
|
28
|
+
import { resolve } from 'node:path';
|
|
29
|
+
import { detectRepo } from '../../core/codegraph/detect-repo.js';
|
|
30
|
+
import { detectCodegraphInstalled, CODEGRAPH_DOCS_URL, } from '../../core/codegraph/install.js';
|
|
31
|
+
import { readDecision, markIndexed, indexAgeDays, STALE_INDEX_DAYS, } from '../../core/codegraph/decision-store.js';
|
|
32
|
+
import { evaluateOffer, applyOfferDecision, emitOfferShown, } from '../../core/codegraph/offer-hook.js';
|
|
33
|
+
export function parseCodegraphStatusArgs(args) {
|
|
34
|
+
const out = {
|
|
35
|
+
install: false,
|
|
36
|
+
reindex: false,
|
|
37
|
+
offer: false,
|
|
38
|
+
unknown: null,
|
|
39
|
+
};
|
|
40
|
+
for (const arg of args) {
|
|
41
|
+
switch (arg) {
|
|
42
|
+
case '--install':
|
|
43
|
+
case '-i':
|
|
44
|
+
out.install = true;
|
|
45
|
+
break;
|
|
46
|
+
case '--reindex':
|
|
47
|
+
case '-r':
|
|
48
|
+
out.reindex = true;
|
|
49
|
+
break;
|
|
50
|
+
case '--offer':
|
|
51
|
+
case '-o':
|
|
52
|
+
out.offer = true;
|
|
53
|
+
break;
|
|
54
|
+
default:
|
|
55
|
+
out.unknown = arg;
|
|
56
|
+
return out;
|
|
57
|
+
}
|
|
58
|
+
}
|
|
59
|
+
return out;
|
|
60
|
+
}
|
|
61
|
+
/**
|
|
62
|
+
* Render the status table. Pure copy assembly — separates rendering
|
|
63
|
+
* from side effects so spec callers can pin the exact layout.
|
|
64
|
+
*/
|
|
65
|
+
export function renderStatusLines(input) {
|
|
66
|
+
const lines = ['Codegraph status:', ''];
|
|
67
|
+
lines.push(` Installed: ${input.installed ? 'yes' : 'no'}`);
|
|
68
|
+
if (input.installed) {
|
|
69
|
+
lines.push(` Trust state: ${input.trust ?? 'pending'}`);
|
|
70
|
+
lines.push(` Config path: ${input.configPath}`);
|
|
71
|
+
}
|
|
72
|
+
if (input.primarySymbolCount !== null) {
|
|
73
|
+
lines.push(` Symbol count: ~${input.primarySymbolCount} source files`);
|
|
74
|
+
}
|
|
75
|
+
if (input.languages && input.languages.length > 0) {
|
|
76
|
+
lines.push(` Languages: ${input.languages.join(', ')}`);
|
|
77
|
+
}
|
|
78
|
+
if (input.sizeCategory) {
|
|
79
|
+
lines.push(` Size category: ${input.sizeCategory}`);
|
|
80
|
+
}
|
|
81
|
+
if (input.installed) {
|
|
82
|
+
if (input.lastIndexedAt) {
|
|
83
|
+
lines.push(` Last indexed: ${input.lastIndexedAt}${input.ageDays !== null ? ` (${input.ageDays} day${input.ageDays === 1 ? '' : 's'} ago)` : ''}`);
|
|
84
|
+
}
|
|
85
|
+
else {
|
|
86
|
+
lines.push(' Last indexed: never recorded — run `codegraph index` and `/codegraph-status --reindex`');
|
|
87
|
+
}
|
|
88
|
+
if (input.staleNudge) {
|
|
89
|
+
lines.push('');
|
|
90
|
+
lines.push(` Index is ${input.ageDays} day${input.ageDays === 1 ? '' : 's'} old (threshold ${STALE_INDEX_DAYS}). ` +
|
|
91
|
+
'Run `codegraph index` then `/codegraph-status --reindex` to refresh.');
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
else {
|
|
95
|
+
lines.push('');
|
|
96
|
+
lines.push(` Docs: ${CODEGRAPH_DOCS_URL}`);
|
|
97
|
+
lines.push(' Install via `/codegraph-status --install` OR `pugi mcp install codegraph codegraph serve --mcp`.');
|
|
98
|
+
}
|
|
99
|
+
return lines;
|
|
100
|
+
}
|
|
101
|
+
/**
|
|
102
|
+
* Single entry-point used by the slash dispatch + (future) standalone
|
|
103
|
+
* `pugi codegraph status` shell command. Pure orchestration — every
|
|
104
|
+
* side effect lives behind one of the flag branches.
|
|
105
|
+
*/
|
|
106
|
+
export async function runCodegraphStatusCommand(args, ctx) {
|
|
107
|
+
const flags = parseCodegraphStatusArgs(args);
|
|
108
|
+
if (flags.unknown) {
|
|
109
|
+
ctx.writeOutput({ command: 'codegraph-status', error: 'unknown-flag', flag: flags.unknown }, `/codegraph-status: unknown flag "${flags.unknown}". Allowed: --install, --reindex, --offer.`);
|
|
110
|
+
return;
|
|
111
|
+
}
|
|
112
|
+
const workspaceRoot = resolve(ctx.workspaceRoot);
|
|
113
|
+
// --reindex — stamp lastIndexedAt + remind the operator how to actually
|
|
114
|
+
// refresh the index. We do NOT spawn `codegraph index` — that is the
|
|
115
|
+
// operator's call (upstream codegraph may or may not be installed; we
|
|
116
|
+
// refuse to silently spawn a binary we did not vet).
|
|
117
|
+
if (flags.reindex) {
|
|
118
|
+
const decision = markIndexed(workspaceRoot);
|
|
119
|
+
if (!decision) {
|
|
120
|
+
ctx.writeOutput({ command: 'codegraph-status', error: 'no-decision' }, '/codegraph-status --reindex: no decision recorded yet. Accept the install first via `--install`.');
|
|
121
|
+
return;
|
|
122
|
+
}
|
|
123
|
+
ctx.writeOutput({ command: 'codegraph-status', reindexed: true, lastIndexedAt: decision.lastIndexedAt }, [
|
|
124
|
+
`Codegraph index timestamp updated to ${decision.lastIndexedAt}.`,
|
|
125
|
+
'Run `codegraph index` from a fresh shell to actually rebuild the SQLite cache.',
|
|
126
|
+
].join('\n'));
|
|
127
|
+
return;
|
|
128
|
+
}
|
|
129
|
+
// --install / --offer — surface the install prompt OR accept it
|
|
130
|
+
// directly. `--install` skips the prompt + writes the entry; `--offer`
|
|
131
|
+
// renders the prompt copy even after a decline (operator-initiated
|
|
132
|
+
// revisit). The two are mutually exclusive at runtime — `--install`
|
|
133
|
+
// wins when both are set.
|
|
134
|
+
if (flags.install || flags.offer) {
|
|
135
|
+
const evaluation = evaluateOffer({
|
|
136
|
+
workspaceRoot,
|
|
137
|
+
...(ctx.nowIso ? { nowIso: ctx.nowIso } : {}),
|
|
138
|
+
ignorePriorDecision: flags.offer,
|
|
139
|
+
});
|
|
140
|
+
if (!evaluation.shouldPrompt) {
|
|
141
|
+
const detection = detectRepo(workspaceRoot);
|
|
142
|
+
// Already-installed is the success path for `--install`; treat
|
|
143
|
+
// it as PASS rather than ERROR so a re-run is idempotent.
|
|
144
|
+
if (evaluation.reason === 'already-installed') {
|
|
145
|
+
ctx.writeOutput({ command: 'codegraph-status', alreadyInstalled: true }, 'Codegraph is already declared in .pugi/mcp.json. Run `pugi mcp trust codegraph` if not yet trusted.');
|
|
146
|
+
return;
|
|
147
|
+
}
|
|
148
|
+
ctx.writeOutput({ command: 'codegraph-status', skipped: true, reason: evaluation.reason }, `/codegraph-status --${flags.install ? 'install' : 'offer'}: nothing to do (${evaluation.reason}).` +
|
|
149
|
+
(detection.isRepo
|
|
150
|
+
? ` Repo: ${detection.sizeCategory}, ${detection.primarySymbolCount} src files.`
|
|
151
|
+
: ''));
|
|
152
|
+
return;
|
|
153
|
+
}
|
|
154
|
+
if (flags.install) {
|
|
155
|
+
const result = applyOfferDecision({
|
|
156
|
+
workspaceRoot,
|
|
157
|
+
accepted: true,
|
|
158
|
+
detection: evaluation.detection,
|
|
159
|
+
...(ctx.nowIso ? { nowIso: ctx.nowIso } : {}),
|
|
160
|
+
});
|
|
161
|
+
if (result.kind === 'declined') {
|
|
162
|
+
// Defensive — applyOfferDecision with accepted=true cannot
|
|
163
|
+
// surface a `declined` verdict by construction. We still
|
|
164
|
+
// narrow the union so TS understands the safe path below.
|
|
165
|
+
ctx.writeOutput({ command: 'codegraph-status', installFailed: true, reason: 'unexpected-decline' }, '/codegraph-status --install: install path returned a decline verdict (internal bug).');
|
|
166
|
+
return;
|
|
167
|
+
}
|
|
168
|
+
if (result.kind === 'accepted-install-failed') {
|
|
169
|
+
ctx.writeOutput({ command: 'codegraph-status', installFailed: true, reason: result.install.reason }, `/codegraph-status --install failed: ${result.install.reason}`);
|
|
170
|
+
return;
|
|
171
|
+
}
|
|
172
|
+
ctx.writeOutput({
|
|
173
|
+
command: 'codegraph-status',
|
|
174
|
+
installed: true,
|
|
175
|
+
configPath: result.install.status === 'installed' ? result.install.configPath : null,
|
|
176
|
+
docsUrl: result.docsUrl,
|
|
177
|
+
trustCommand: result.trustCommand,
|
|
178
|
+
}, [
|
|
179
|
+
result.install.status === 'installed'
|
|
180
|
+
? `Codegraph added к .pugi/mcp.json. Trust gate: ${result.trustCommand}`
|
|
181
|
+
: 'Codegraph already declared. Trust gate: ' + result.trustCommand,
|
|
182
|
+
`Docs: ${result.docsUrl}`,
|
|
183
|
+
].join('\n'));
|
|
184
|
+
return;
|
|
185
|
+
}
|
|
186
|
+
// --offer path — surface the copy but do not install yet
|
|
187
|
+
emitOfferShown(evaluation.detection);
|
|
188
|
+
ctx.writeOutput({
|
|
189
|
+
command: 'codegraph-status',
|
|
190
|
+
offered: true,
|
|
191
|
+
copy: evaluation.promptCopy,
|
|
192
|
+
docsUrl: evaluation.docsUrl,
|
|
193
|
+
}, [
|
|
194
|
+
evaluation.promptCopy,
|
|
195
|
+
`Docs: ${evaluation.docsUrl}`,
|
|
196
|
+
'Accept: /codegraph-status --install',
|
|
197
|
+
'Decline: do nothing (the prompt re-appears in 30 days).',
|
|
198
|
+
].join('\n'));
|
|
199
|
+
return;
|
|
200
|
+
}
|
|
201
|
+
// Bare invocation — render the status table.
|
|
202
|
+
const installed = detectCodegraphInstalled(workspaceRoot);
|
|
203
|
+
const detection = detectRepo(workspaceRoot);
|
|
204
|
+
const decision = readDecision(workspaceRoot);
|
|
205
|
+
const ageDays = decision ? indexAgeDays(decision, ctx.nowIso) : null;
|
|
206
|
+
const lines = renderStatusLines({
|
|
207
|
+
installed: installed.installed,
|
|
208
|
+
trust: installed.trust,
|
|
209
|
+
configPath: installed.configPath,
|
|
210
|
+
ageDays,
|
|
211
|
+
primarySymbolCount: detection.isRepo ? detection.primarySymbolCount : null,
|
|
212
|
+
languages: detection.isRepo ? detection.languages : null,
|
|
213
|
+
sizeCategory: detection.isRepo ? detection.sizeCategory : null,
|
|
214
|
+
lastIndexedAt: decision?.lastIndexedAt ?? null,
|
|
215
|
+
staleNudge: ageDays !== null && ageDays >= STALE_INDEX_DAYS,
|
|
216
|
+
});
|
|
217
|
+
ctx.writeOutput({
|
|
218
|
+
command: 'codegraph-status',
|
|
219
|
+
installed: installed.installed,
|
|
220
|
+
trust: installed.trust,
|
|
221
|
+
configPath: installed.configPath,
|
|
222
|
+
detection,
|
|
223
|
+
decision,
|
|
224
|
+
ageDays,
|
|
225
|
+
}, lines.join('\n'));
|
|
226
|
+
}
|
|
227
|
+
//# sourceMappingURL=codegraph-status.js.map
|
|
@@ -0,0 +1,297 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* `pugi compact` — top-level command + companion to the REPL slash.
|
|
3
|
+
*
|
|
4
|
+
* Summarises older events from the most recent SessionStore session
|
|
5
|
+
* into a single boundary marker, freeing context budget for the next
|
|
6
|
+
* REPL session that resumes the same id.
|
|
7
|
+
*
|
|
8
|
+
* Wire: writeOutput contract matches every other dispatch in cli.ts
|
|
9
|
+
* (one structured payload + one human line). Exit code:
|
|
10
|
+
* 0 — boundary marker appended
|
|
11
|
+
* 1 — summarizer transport failed / store unavailable
|
|
12
|
+
* 2 — no events to compact / nothing to do
|
|
13
|
+
*
|
|
14
|
+
* The companion REPL slash dispatches through this same runner so the
|
|
15
|
+
* surface stays single-sourced.
|
|
16
|
+
*/
|
|
17
|
+
import { homedir } from 'node:os';
|
|
18
|
+
import { loadRuntimeConfig } from '@pugi/sdk';
|
|
19
|
+
import { AnvilEngineLoopClient } from '../../core/engine/anvil-client.js';
|
|
20
|
+
import { SqliteSessionStore, resolveProjectStoreDir, } from '../../core/repl/store/session-store.js';
|
|
21
|
+
import { appendCompactBoundary, isCompactBoundary, } from '../../core/compact/buffer-rewriter.js';
|
|
22
|
+
import { summarizeEvents, SummarizerError } from '../../core/compact/summarizer.js';
|
|
23
|
+
/** Number of tail turns preserved verbatim on every compaction. */
|
|
24
|
+
export const KEPT_TAIL_TURNS = 5;
|
|
25
|
+
/** Minimum source events before we accept a compaction. */
|
|
26
|
+
const MIN_EVENTS_TO_COMPACT = 6;
|
|
27
|
+
/**
|
|
28
|
+
* Entry point reused by the slash command + the top-level dispatcher.
|
|
29
|
+
*
|
|
30
|
+
* The function is exhaustive: every early-exit branch produces a
|
|
31
|
+
* structured `CompactCommandResult` for the JSON path AND a one-line
|
|
32
|
+
* human message for the TTY path. Tests assert on the returned shape;
|
|
33
|
+
* the top-level dispatcher forwards it to `writeOutput`.
|
|
34
|
+
*/
|
|
35
|
+
export async function runCompactCommand(_args, ctx) {
|
|
36
|
+
const trigger = ctx.trigger ?? 'manual';
|
|
37
|
+
// Resolve the target session id. When the caller supplied a store
|
|
38
|
+
// (REPL slash path, tests) we trust it has the session bound; the
|
|
39
|
+
// standalone CLI path needs to discover the most-recent id via the
|
|
40
|
+
// read-only view + then open the live store on that id.
|
|
41
|
+
let store = ctx.store ?? null;
|
|
42
|
+
let sessionId = ctx.sessionId ?? null;
|
|
43
|
+
if (store === null) {
|
|
44
|
+
sessionId = sessionId ?? (await pickMostRecentSessionIdReadOnly(ctx.workspaceRoot));
|
|
45
|
+
if (!sessionId) {
|
|
46
|
+
return emit(ctx, {
|
|
47
|
+
command: 'compact',
|
|
48
|
+
status: 'failed_no_session',
|
|
49
|
+
trigger,
|
|
50
|
+
reason: 'No active session to compact. Start a REPL with `pugi`.',
|
|
51
|
+
});
|
|
52
|
+
}
|
|
53
|
+
store = await openStoreForCwd(ctx.workspaceRoot, sessionId);
|
|
54
|
+
if (!store) {
|
|
55
|
+
return emit(ctx, {
|
|
56
|
+
command: 'compact',
|
|
57
|
+
status: 'failed_no_session',
|
|
58
|
+
trigger,
|
|
59
|
+
reason: 'Local session store unavailable (lock held by REPL?).',
|
|
60
|
+
});
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
else if (sessionId === null) {
|
|
64
|
+
// Caller supplied a store but no session id — try to discover one.
|
|
65
|
+
sessionId = await pickMostRecentSessionId(store, ctx.workspaceRoot);
|
|
66
|
+
if (!sessionId) {
|
|
67
|
+
return emit(ctx, {
|
|
68
|
+
command: 'compact',
|
|
69
|
+
status: 'failed_no_session',
|
|
70
|
+
trigger,
|
|
71
|
+
reason: 'No active session to compact.',
|
|
72
|
+
});
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
// Snapshot the events. We pass `coversUntilOffset` as the count of
|
|
76
|
+
// events at the moment of the snapshot so the marker's anchor is
|
|
77
|
+
// accurate even if other writers race afterwards.
|
|
78
|
+
let events;
|
|
79
|
+
try {
|
|
80
|
+
events = await store.loadEvents(sessionId);
|
|
81
|
+
}
|
|
82
|
+
catch (error) {
|
|
83
|
+
return emit(ctx, {
|
|
84
|
+
command: 'compact',
|
|
85
|
+
status: 'failed_store',
|
|
86
|
+
sessionId,
|
|
87
|
+
trigger,
|
|
88
|
+
reason: `Could not load events: ${errMsg(error)}`,
|
|
89
|
+
});
|
|
90
|
+
}
|
|
91
|
+
if (events.length < MIN_EVENTS_TO_COMPACT && ctx.force !== true) {
|
|
92
|
+
return emit(ctx, {
|
|
93
|
+
command: 'compact',
|
|
94
|
+
status: 'noop_empty',
|
|
95
|
+
sessionId,
|
|
96
|
+
trigger,
|
|
97
|
+
reason: `Only ${events.length} events on disk; need at least ${MIN_EVENTS_TO_COMPACT}. `
|
|
98
|
+
+ 'Pass --force to compact anyway.',
|
|
99
|
+
});
|
|
100
|
+
}
|
|
101
|
+
// Locate the source slice: everything strictly after the latest
|
|
102
|
+
// compact-boundary, EXCLUDING the kept-tail window. If no prior
|
|
103
|
+
// boundary exists, walk from offset 0.
|
|
104
|
+
const lastBoundaryIdx = findLastCompactBoundaryIndex(events);
|
|
105
|
+
const sliceStart = lastBoundaryIdx === -1 ? 0 : lastBoundaryIdx + 1;
|
|
106
|
+
const sliceEnd = Math.max(sliceStart, events.length - KEPT_TAIL_TURNS);
|
|
107
|
+
const sourceSlice = events.slice(sliceStart, sliceEnd);
|
|
108
|
+
if (sourceSlice.length === 0) {
|
|
109
|
+
return emit(ctx, {
|
|
110
|
+
command: 'compact',
|
|
111
|
+
status: 'noop_recent_marker',
|
|
112
|
+
sessionId,
|
|
113
|
+
trigger,
|
|
114
|
+
reason: 'Conversation already compact — nothing new to fold.',
|
|
115
|
+
});
|
|
116
|
+
}
|
|
117
|
+
// Build the engine client. Production uses Anvil; tests inject a
|
|
118
|
+
// fixture client to avoid the network round-trip.
|
|
119
|
+
const engineClient = ctx.engineClient ?? buildAnvilClient();
|
|
120
|
+
if (!engineClient) {
|
|
121
|
+
return emit(ctx, {
|
|
122
|
+
command: 'compact',
|
|
123
|
+
status: 'failed_transport',
|
|
124
|
+
sessionId,
|
|
125
|
+
trigger,
|
|
126
|
+
reason: 'Could not build engine client (run `pugi login`).',
|
|
127
|
+
});
|
|
128
|
+
}
|
|
129
|
+
// Run the summariser.
|
|
130
|
+
let summary;
|
|
131
|
+
try {
|
|
132
|
+
summary = await summarizeEvents({
|
|
133
|
+
events: sourceSlice,
|
|
134
|
+
client: engineClient,
|
|
135
|
+
personaSlug: 'mira',
|
|
136
|
+
});
|
|
137
|
+
}
|
|
138
|
+
catch (error) {
|
|
139
|
+
const code = error instanceof SummarizerError ? error.code : 'unknown';
|
|
140
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
141
|
+
return emit(ctx, {
|
|
142
|
+
command: 'compact',
|
|
143
|
+
status: 'failed_transport',
|
|
144
|
+
sessionId,
|
|
145
|
+
trigger,
|
|
146
|
+
reason: `Summarizer failed (${code}): ${message}`,
|
|
147
|
+
});
|
|
148
|
+
}
|
|
149
|
+
// Append the boundary marker. Recompute `coversUntilOffset` AFTER
|
|
150
|
+
// the summariser round-trip so any events that landed mid-flight
|
|
151
|
+
// fall into the kept-tail window — the operator does not lose work
|
|
152
|
+
// that arrived during compaction. The next event we append (the
|
|
153
|
+
// marker itself) lands at index `coversUntilOffset` in the on-disk
|
|
154
|
+
// log.
|
|
155
|
+
let postRoundTripEvents;
|
|
156
|
+
try {
|
|
157
|
+
postRoundTripEvents = await store.loadEvents(sessionId);
|
|
158
|
+
}
|
|
159
|
+
catch (error) {
|
|
160
|
+
return emit(ctx, {
|
|
161
|
+
command: 'compact',
|
|
162
|
+
status: 'failed_store',
|
|
163
|
+
sessionId,
|
|
164
|
+
trigger,
|
|
165
|
+
reason: `Could not refresh events: ${errMsg(error)}`,
|
|
166
|
+
});
|
|
167
|
+
}
|
|
168
|
+
const coversUntilOffset = postRoundTripEvents.length;
|
|
169
|
+
try {
|
|
170
|
+
await appendCompactBoundary({
|
|
171
|
+
store,
|
|
172
|
+
trigger,
|
|
173
|
+
summary: summary.summary,
|
|
174
|
+
summaryTokenCount: summary.tokensSummarised,
|
|
175
|
+
summaryTurnsBefore: summary.eventsSummarised,
|
|
176
|
+
keptTailTurns: KEPT_TAIL_TURNS,
|
|
177
|
+
coversUntilOffset,
|
|
178
|
+
...(ctx.now !== undefined ? { now: ctx.now } : {}),
|
|
179
|
+
});
|
|
180
|
+
}
|
|
181
|
+
catch (error) {
|
|
182
|
+
return emit(ctx, {
|
|
183
|
+
command: 'compact',
|
|
184
|
+
status: 'failed_store',
|
|
185
|
+
sessionId,
|
|
186
|
+
trigger,
|
|
187
|
+
reason: `Could not append marker: ${errMsg(error)}`,
|
|
188
|
+
});
|
|
189
|
+
}
|
|
190
|
+
return emit(ctx, {
|
|
191
|
+
command: 'compact',
|
|
192
|
+
status: 'compacted',
|
|
193
|
+
sessionId,
|
|
194
|
+
trigger,
|
|
195
|
+
turnsBefore: summary.eventsSummarised,
|
|
196
|
+
tailKept: KEPT_TAIL_TURNS,
|
|
197
|
+
tokensSummarised: summary.tokensSummarised,
|
|
198
|
+
}, `Compacted ${summary.eventsSummarised} events (~${summary.tokensSummarised} tokens) into 1 summary. ` +
|
|
199
|
+
`Last ${KEPT_TAIL_TURNS} turns kept verbatim.`);
|
|
200
|
+
}
|
|
201
|
+
/* ------------------------------------------------------------------ */
|
|
202
|
+
/* Helpers */
|
|
203
|
+
/* ------------------------------------------------------------------ */
|
|
204
|
+
/**
|
|
205
|
+
* Emit a payload through the context and return it for the caller. We
|
|
206
|
+
* keep the function exhaustive so every branch lands on the same
|
|
207
|
+
* writeOutput contract (JSON-mode + text-mode).
|
|
208
|
+
*/
|
|
209
|
+
function emit(ctx, payload, text) {
|
|
210
|
+
const human = text ?? payload.reason ?? `compact: ${payload.status}`;
|
|
211
|
+
ctx.writeOutput(payload, human);
|
|
212
|
+
return payload;
|
|
213
|
+
}
|
|
214
|
+
function findLastCompactBoundaryIndex(events) {
|
|
215
|
+
for (let i = events.length - 1; i >= 0; i -= 1) {
|
|
216
|
+
if (isCompactBoundary(events[i]))
|
|
217
|
+
return i;
|
|
218
|
+
}
|
|
219
|
+
return -1;
|
|
220
|
+
}
|
|
221
|
+
function errMsg(error) {
|
|
222
|
+
return error instanceof Error ? error.message : String(error);
|
|
223
|
+
}
|
|
224
|
+
/**
|
|
225
|
+
* Open the SqliteSessionStore for the workspace's project slug and
|
|
226
|
+
* bind the active log to `sessionId`. Returns null when the open
|
|
227
|
+
* fails (lock contention, IO error, missing project dir).
|
|
228
|
+
*
|
|
229
|
+
* The PID lockfile contention is the expected failure mode here — a
|
|
230
|
+
* running REPL holds the lock, and `pugi compact` from a second
|
|
231
|
+
* terminal cannot grab it. The caller surfaces a clear message rather
|
|
232
|
+
* than crashing the dispatch.
|
|
233
|
+
*/
|
|
234
|
+
async function openStoreForCwd(workspaceRoot, sessionId) {
|
|
235
|
+
try {
|
|
236
|
+
const slug = projectSlugForCwd(workspaceRoot);
|
|
237
|
+
const store = new SqliteSessionStore({ projectSlug: slug, home: homedir() });
|
|
238
|
+
await store.open({
|
|
239
|
+
id: sessionId,
|
|
240
|
+
workspaceRoot,
|
|
241
|
+
projectSlug: slug,
|
|
242
|
+
});
|
|
243
|
+
return store;
|
|
244
|
+
}
|
|
245
|
+
catch {
|
|
246
|
+
return null;
|
|
247
|
+
}
|
|
248
|
+
}
|
|
249
|
+
/**
|
|
250
|
+
* Read-only-view variant of `pickMostRecentSessionId` for the
|
|
251
|
+
* pre-open phase. Walks `~/.pugi/projects/<slug>/session.db` without
|
|
252
|
+
* taking the lockfile so a live REPL in another terminal does not
|
|
253
|
+
* make discovery fail.
|
|
254
|
+
*/
|
|
255
|
+
async function pickMostRecentSessionIdReadOnly(workspaceRoot) {
|
|
256
|
+
try {
|
|
257
|
+
const slug = projectSlugForCwd(workspaceRoot);
|
|
258
|
+
const dir = resolveProjectStoreDir(slug, homedir());
|
|
259
|
+
const view = await SqliteSessionStore.openReadOnly(dir);
|
|
260
|
+
try {
|
|
261
|
+
const rows = await view.list({ project: slug, limit: 1, status: 'active' });
|
|
262
|
+
return rows.length > 0 ? rows[0].id : null;
|
|
263
|
+
}
|
|
264
|
+
finally {
|
|
265
|
+
await view.close();
|
|
266
|
+
}
|
|
267
|
+
}
|
|
268
|
+
catch {
|
|
269
|
+
return null;
|
|
270
|
+
}
|
|
271
|
+
}
|
|
272
|
+
/**
|
|
273
|
+
* Locate the most recent active session for the workspace. Returns
|
|
274
|
+
* null when no session row exists.
|
|
275
|
+
*/
|
|
276
|
+
async function pickMostRecentSessionId(store, workspaceRoot) {
|
|
277
|
+
const slug = projectSlugForCwd(workspaceRoot);
|
|
278
|
+
const rows = await store.listSessions({ project: slug, limit: 1, status: 'active' });
|
|
279
|
+
if (rows.length === 0)
|
|
280
|
+
return null;
|
|
281
|
+
return rows[0].id;
|
|
282
|
+
}
|
|
283
|
+
function projectSlugForCwd(workspaceRoot) {
|
|
284
|
+
const base = workspaceRoot.split('/').filter((s) => s.length > 0).pop() ?? 'workspace';
|
|
285
|
+
return base.toLowerCase().replace(/[^a-z0-9-]/g, '-').slice(0, 64);
|
|
286
|
+
}
|
|
287
|
+
/**
|
|
288
|
+
* Construct an Anvil-backed engine client from env-resolved runtime
|
|
289
|
+
* config. Returns null when the operator is not authenticated.
|
|
290
|
+
*/
|
|
291
|
+
function buildAnvilClient() {
|
|
292
|
+
const config = loadRuntimeConfig();
|
|
293
|
+
if (!config)
|
|
294
|
+
return null;
|
|
295
|
+
return new AnvilEngineLoopClient(config);
|
|
296
|
+
}
|
|
297
|
+
//# sourceMappingURL=compact.js.map
|
|
@@ -11,21 +11,21 @@ import { resolveActiveCredential } from '../../core/credentials.js';
|
|
|
11
11
|
* `pugi config` — operator-level configuration surface.
|
|
12
12
|
*
|
|
13
13
|
* Subcommands:
|
|
14
|
-
*
|
|
15
|
-
*
|
|
16
|
-
*
|
|
17
|
-
*
|
|
18
|
-
*
|
|
19
|
-
*
|
|
20
|
-
*
|
|
14
|
+
* - `pugi config get <key>` read a value from `~/.pugi/config.json`
|
|
15
|
+
* - `pugi config set <key> <value>` write a value
|
|
16
|
+
* - `pugi config list` dump all values
|
|
17
|
+
* - `pugi config trust .` trust the current workspace (delegates to core/trust.ts)
|
|
18
|
+
* - `pugi config mcp trust <name>` flip MCP server to trusted
|
|
19
|
+
* - `pugi config mcp deny <name>` flip MCP server to denied
|
|
20
|
+
* - `pugi config mcp list` show declared servers + their trust state
|
|
21
21
|
*
|
|
22
22
|
* Schema (pugi-config-v1):
|
|
23
|
-
*
|
|
24
|
-
*
|
|
25
|
-
*
|
|
26
|
-
*
|
|
27
|
-
*
|
|
28
|
-
*
|
|
23
|
+
* {
|
|
24
|
+
* "permissionMode": "ask" | "acceptEdits" | "auto" | "plan" | "dontAsk" | "bypassPermissions",
|
|
25
|
+
* "privacy": "local-only" | "metadata" | "full",
|
|
26
|
+
* "model": "<id>" | null,
|
|
27
|
+
* "preferredEndpoint": "https://api.pugi.io"
|
|
28
|
+
* }
|
|
29
29
|
*
|
|
30
30
|
* The config file lives at `~/.pugi/config.json` (PUGI_HOME-aware) and uses
|
|
31
31
|
* mode 0o600. Unknown keys are rejected by `set` so a typo never silently
|
|
@@ -63,18 +63,18 @@ export async function runConfigCommand(args, ctx) {
|
|
|
63
63
|
],
|
|
64
64
|
}, [
|
|
65
65
|
'Usage:',
|
|
66
|
-
'
|
|
67
|
-
'
|
|
68
|
-
'
|
|
69
|
-
'
|
|
70
|
-
'
|
|
71
|
-
'
|
|
72
|
-
'
|
|
73
|
-
'
|
|
74
|
-
'
|
|
75
|
-
'
|
|
76
|
-
'
|
|
77
|
-
'
|
|
66
|
+
' pugi config get <key> Read a config value.',
|
|
67
|
+
' pugi config set <key> <value> Write a config value.',
|
|
68
|
+
' pugi config list Show all config values.',
|
|
69
|
+
' pugi config trust . Trust the current workspace for hooks + MCP.',
|
|
70
|
+
' pugi config mcp trust <name> Mark an MCP server as trusted.',
|
|
71
|
+
' pugi config mcp deny <name> Block an MCP server.',
|
|
72
|
+
' pugi config mcp list Show declared MCP servers + trust state.',
|
|
73
|
+
' pugi config get routing Show effective routing table (defaults + tenant overrides).',
|
|
74
|
+
' pugi config set routing.<tag>.<budget>=<model> Override the model for one (tag, budget) lane.',
|
|
75
|
+
' pugi config unset routing.<tag>.<budget> Remove a routing override (revert to default).',
|
|
76
|
+
' pugi config get privacy Show current tenant privacy mode + last-flip metadata.',
|
|
77
|
+
' pugi config set privacy=<mode> Flip privacy mode (strict | balanced | permissive).',
|
|
78
78
|
].join('\n'));
|
|
79
79
|
return;
|
|
80
80
|
}
|
|
@@ -108,7 +108,7 @@ export async function runConfigCommand(args, ctx) {
|
|
|
108
108
|
// names. The unit spec for config.ts has a regression test for
|
|
109
109
|
// both code paths.
|
|
110
110
|
//
|
|
111
|
-
// Triple-review P2 fix
|
|
111
|
+
// Triple-review P2 fix : the prior disambiguation
|
|
112
112
|
// only excluded the bare form (`privacy local-only`) - the `=`
|
|
113
113
|
// form (`privacy=local-only`) still routed to runPrivacySet and
|
|
114
114
|
// 4xx'd. We now check the value AFTER `=` and route legacy local
|
|
@@ -279,7 +279,7 @@ async function runConfigMcpList(ctx) {
|
|
|
279
279
|
}
|
|
280
280
|
ctx.writeOutput({ command: 'config.mcp.list', servers: declared, ledger }, [
|
|
281
281
|
'MCP servers:',
|
|
282
|
-
...declared.map((server) => `
|
|
282
|
+
...declared.map((server) => ` ${server.name.padEnd(20)} ${server.trust.padEnd(8)} ${server.command} ${server.args.join(' ')}`),
|
|
283
283
|
].join('\n'));
|
|
284
284
|
}
|
|
285
285
|
async function runConfigMcpFlip(args, ctx, state) {
|
|
@@ -297,7 +297,7 @@ async function runConfigMcpFlip(args, ctx, state) {
|
|
|
297
297
|
: `MCP server "${name}" is now denied.`);
|
|
298
298
|
}
|
|
299
299
|
/* ------------------------------------------------------------------ */
|
|
300
|
-
/*
|
|
300
|
+
/* multi-model routing — config.routing.* subcommands */
|
|
301
301
|
/* ------------------------------------------------------------------ */
|
|
302
302
|
/**
|
|
303
303
|
* Closed sets — match
|
|
@@ -360,7 +360,7 @@ async function runRoutingGet(ctx) {
|
|
|
360
360
|
});
|
|
361
361
|
const text = [
|
|
362
362
|
'Routing table (effective = override | default):',
|
|
363
|
-
...cells.map((cell) => `
|
|
363
|
+
...cells.map((cell) => ` routing.${cell.tag.padEnd(10)}.${cell.budgetHint.padEnd(3)} = ${cell.model.padEnd(28)} (${cell.source})`),
|
|
364
364
|
].join('\n');
|
|
365
365
|
ctx.writeOutput({
|
|
366
366
|
command: 'config.routing.get',
|
|
@@ -439,7 +439,7 @@ async function runRoutingUnset(args, ctx) {
|
|
|
439
439
|
: `routing.${tag}.${budget} had no override (nothing to remove).`);
|
|
440
440
|
}
|
|
441
441
|
/* ------------------------------------------------------------------ */
|
|
442
|
-
/* alpha 6.13 privacy 3-mode - config.privacy.* subcommands
|
|
442
|
+
/* alpha 6.13 privacy 3-mode - config.privacy.* subcommands */
|
|
443
443
|
/* ------------------------------------------------------------------ */
|
|
444
444
|
/**
|
|
445
445
|
* Closed mirror of the server-side PRIVACY_MODES enum
|
|
@@ -459,7 +459,7 @@ function isPrivacyMode(value) {
|
|
|
459
459
|
* Legacy local-config privacy values from before alpha 6.13. Kept so
|
|
460
460
|
* `pugi config set privacy=local-only` continues to write to the local
|
|
461
461
|
* config file (matching the bare-form behaviour). Triple-review P2 fix
|
|
462
|
-
*
|
|
462
|
+
* : the prior disambiguation only excluded the bare form;
|
|
463
463
|
* the `=` form routed to runPrivacySet and 4xx'd on the unknown mode.
|
|
464
464
|
*/
|
|
465
465
|
const LEGACY_LOCAL_PRIVACY_VALUES = [
|
|
@@ -536,7 +536,7 @@ async function runPrivacySet(args, ctx) {
|
|
|
536
536
|
* — undici's `request` honours the global dispatcher reliably across the
|
|
537
537
|
* pinned undici version. Kept local (not shared with `pugi whoami`) so
|
|
538
538
|
* the routing surface is self-contained — extracting a common helper is
|
|
539
|
-
*
|
|
539
|
+
* cleanup once we see two callers.
|
|
540
540
|
*/
|
|
541
541
|
async function fetchJson(url, apiKey, options = {}) {
|
|
542
542
|
const method = options.method ?? 'GET';
|