@pugi/cli 0.1.0-beta.10 → 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/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 +29 -29
- package/dist/core/engine/anvil-client.js +214 -26
- package/dist/core/engine/auto-compact.js +247 -0
- package/dist/core/engine/budgets.js +220 -0
- package/dist/core/engine/compact-llm-summarizer.js +124 -0
- package/dist/core/engine/context-prefix.js +155 -0
- package/dist/core/engine/index.js +1 -1
- package/dist/core/engine/intensity.js +163 -0
- package/dist/core/engine/intent.js +260 -0
- package/dist/core/engine/native-pugi.js +1559 -227
- package/dist/core/engine/prompts.js +187 -19
- package/dist/core/engine/strip-internal-fields.js +124 -0
- package/dist/core/engine/tool-bridge.js +1887 -59
- package/dist/core/engine/verification-patterns.js +195 -0
- package/dist/core/evaluation/golden-dataset.js +293 -0
- package/dist/core/feedback/queue.js +177 -0
- package/dist/core/feedback/submitter.js +145 -0
- package/dist/core/file-cache.js +113 -1
- package/dist/core/flatten/flatten-repo.js +439 -0
- package/dist/core/format/osc8-link.js +28 -0
- package/dist/core/hook-chains.js +392 -0
- package/dist/core/hooks/citation-verify-hook.js +138 -0
- package/dist/core/hooks/citation-verify.js +112 -0
- package/dist/core/hooks/events.js +46 -0
- package/dist/core/hooks/index.js +15 -0
- package/dist/core/hooks/registry.js +216 -0
- package/dist/core/hooks/runner.js +236 -0
- package/dist/core/hooks/v2/event-emitter.js +115 -0
- package/dist/core/hooks/v2/executor.js +282 -0
- package/dist/core/hooks/v2/index.js +25 -0
- package/dist/core/hooks/v2/lifecycle.js +104 -0
- package/dist/core/hooks/v2/loader.js +216 -0
- package/dist/core/hooks/v2/matcher.js +125 -0
- package/dist/core/hooks/v2/trust.js +143 -0
- package/dist/core/hooks/v2/types.js +86 -0
- package/dist/core/hooks/worktree-events.js +158 -0
- package/dist/core/image/renderer.js +71 -0
- package/dist/core/init/detector.js +582 -0
- package/dist/core/init/template-renderer.js +242 -0
- package/dist/core/jobs/registry.js +18 -18
- package/dist/core/ledger/results-tsv.js +142 -0
- package/dist/core/log-discipline/stdout-redirect.js +51 -0
- package/dist/core/lsp/cache.js +105 -0
- package/dist/core/lsp/client.js +551 -41
- package/dist/core/lsp/language-detect.js +66 -0
- package/dist/core/lsp/post-edit-diagnostics.js +171 -0
- package/dist/core/lsp/server-detect.js +173 -0
- package/dist/core/lsp/symbol-cache.js +162 -0
- package/dist/core/lsp/symbol-tools.js +664 -0
- package/dist/core/mcp/client.js +97 -28
- package/dist/core/mcp/http-server.js +553 -0
- package/dist/core/mcp/orchestrator-config.js +192 -0
- package/dist/core/mcp/orchestrator-tools.js +806 -0
- package/dist/core/mcp/permission.js +190 -0
- package/dist/core/mcp/registry.js +39 -17
- package/dist/core/mcp/server-tools.js +219 -0
- package/dist/core/mcp/server.js +397 -0
- package/dist/core/mcp/trust.js +10 -10
- package/dist/core/memory/dual-write.js +416 -0
- package/dist/core/memory/passive-extract.js +130 -0
- package/dist/core/memory/phase1-kinds.js +20 -0
- package/dist/core/memory/secret-scanner.js +304 -0
- package/dist/core/memory-sync/queue.js +170 -0
- package/dist/core/metrics/extract.js +113 -0
- package/dist/core/modes/roo-modes.js +68 -0
- package/dist/core/notes/notes-paths.js +113 -0
- package/dist/core/notes/notes-recorder.js +140 -0
- package/dist/core/notes/notes-writer.js +53 -0
- package/dist/core/notes/renderers.js +0 -0
- package/dist/core/notes/slug.js +105 -0
- package/dist/core/onboarding/ensure-initialized.js +133 -0
- package/dist/core/onboarding/marker.js +111 -0
- package/dist/core/onboarding/telemetry-state.js +108 -0
- package/dist/core/output-style/presets.js +176 -0
- package/dist/core/output-style/state.js +185 -0
- package/dist/core/path-security.js +287 -5
- package/dist/core/permission.js +82 -22
- package/dist/core/permissions/auto-classifier.js +124 -0
- package/dist/core/permissions/bash-parser.js +371 -0
- package/dist/core/permissions/circuit-breaker.js +83 -0
- package/dist/core/permissions/constrained-edit.js +91 -0
- package/dist/core/permissions/gate.js +278 -0
- package/dist/core/permissions/index.js +20 -0
- package/dist/core/permissions/mode.js +174 -0
- package/dist/core/permissions/network-egress.js +137 -0
- package/dist/core/permissions/state.js +241 -0
- package/dist/core/permissions/tool-class.js +107 -0
- package/dist/core/plan-mode/ui-state.js +51 -0
- package/dist/core/plans/plan-artifact.js +721 -0
- package/dist/core/policy-limits/etag-store.js +122 -0
- package/dist/core/prd-check/parser.js +215 -0
- package/dist/core/prd-check/reporter.js +127 -0
- package/dist/core/prd-check/session-review.js +557 -0
- package/dist/core/prd-check/verifiers.js +223 -0
- package/dist/core/prompt-cache/client-cache.js +99 -0
- package/dist/core/prompts/assembly.js +29 -0
- package/dist/core/prompts/registry.js +364 -0
- package/dist/core/pugi-gitignore.js +52 -0
- package/dist/core/pugi-md/cc-compat-rules.js +735 -0
- package/dist/core/pugi-md/context-injector.js +76 -0
- package/dist/core/pugi-md/walk-up.js +207 -0
- package/dist/core/python/uv-installer.js +270 -0
- package/dist/core/python/uv-resolver.js +83 -0
- package/dist/core/rate-limit/narrator.js +146 -0
- package/dist/core/recipes/cli-types.js +20 -0
- package/dist/core/recipes/loader.js +103 -0
- package/dist/core/recipes/runner.js +345 -0
- package/dist/core/recipes/schema.js +587 -0
- package/dist/core/release-notes/parser.js +241 -0
- package/dist/core/release-notes/state.js +116 -0
- package/dist/core/repl/ask.js +37 -37
- package/dist/core/repl/cancellation.js +26 -26
- package/dist/core/repl/cap-warning.js +4 -4
- package/dist/core/repl/clipboard-read.js +11 -11
- package/dist/core/repl/dispatch-fsm.js +12 -12
- package/dist/core/repl/engine-bridge.js +303 -0
- package/dist/core/repl/history-search.js +15 -15
- package/dist/core/repl/history.js +28 -18
- package/dist/core/repl/kill-ring.js +5 -5
- package/dist/core/repl/model-pricing.js +135 -0
- package/dist/core/repl/privacy-banner.js +22 -22
- package/dist/core/repl/session.js +2690 -229
- package/dist/core/repl/slash-commands.js +540 -41
- package/dist/core/repl/store/index.js +1 -1
- package/dist/core/repl/store/jsonl-log.js +22 -22
- package/dist/core/repl/store/lockfile.js +10 -10
- package/dist/core/repl/store/session-store.js +136 -107
- package/dist/core/repl/store/types.js +15 -15
- package/dist/core/repl/store/uuid-v7.js +12 -12
- package/dist/core/repl/tool-route.js +382 -0
- package/dist/core/repl/workspace-context.js +43 -21
- package/dist/core/repo-map/build.js +125 -0
- package/dist/core/repo-map/cache.js +185 -0
- package/dist/core/repo-map/extractor.js +254 -0
- package/dist/core/repo-map/formatter.js +145 -0
- package/dist/core/repo-map/page-rank.js +105 -0
- package/dist/core/repo-map/scanner.js +211 -0
- package/dist/core/retro/git-collector.js +251 -0
- package/dist/core/retro/health-card.js +25 -0
- package/dist/core/retro/metrics.js +342 -0
- package/dist/core/retro/narrative.js +249 -0
- package/dist/core/retro/plane-collector.js +274 -0
- package/dist/core/retro/pr-issue-link.js +65 -0
- package/dist/core/retro/types.js +16 -0
- package/dist/core/retry-budget/budget.js +284 -0
- package/dist/core/retry-budget/index.js +5 -0
- package/dist/core/retry-budget/retry-cap.js +74 -0
- package/dist/core/routing/lead-worker.js +43 -0
- package/dist/core/routing/pre-flight-estimator.js +108 -0
- package/dist/core/runs/run-tree.js +103 -0
- package/dist/core/sandboxing/adapter.js +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 +30 -30
- package/dist/core/skills/loader.js +22 -22
- package/dist/core/skills/sources.js +27 -27
- package/dist/core/smoke/headless-driver.js +174 -0
- package/dist/core/smoke/orchestrator.js +194 -0
- package/dist/core/smoke/runner.js +238 -0
- package/dist/core/smoke/scenario-parser.js +316 -0
- package/dist/core/statusline.js +99 -0
- package/dist/core/subagents/dispatcher-real.js +600 -0
- package/dist/core/subagents/dispatcher.js +146 -52
- package/dist/core/subagents/index.js +19 -6
- package/dist/core/subagents/isolation-matrix.js +213 -0
- package/dist/core/subagents/spawn.js +19 -4
- package/dist/core/telemetry/emitter.js +229 -0
- package/dist/core/telemetry/queue.js +251 -0
- package/dist/core/theme/context.js +91 -0
- package/dist/core/theme/presets.js +228 -0
- package/dist/core/theme/state.js +181 -0
- package/dist/core/todos/invariant.js +10 -0
- package/dist/core/todos/state.js +177 -0
- package/dist/core/tool-schema/compressor.js +89 -0
- package/dist/core/transport/version-interceptor.js +166 -0
- package/dist/core/trust.js +2 -2
- package/dist/core/tui/thinking-block.js +64 -0
- package/dist/core/vim/keymap.js +288 -0
- package/dist/core/vim/state.js +92 -0
- package/dist/core/watch-markers/marker-watcher.js +133 -0
- package/dist/core/worktree/include-parser.js +249 -0
- package/dist/core/worktree-manager/cleanup.js +123 -0
- package/dist/core/worktree-manager/manager.js +303 -0
- package/dist/index.js +36 -0
- package/dist/runtime/bootstrap.js +190 -0
- package/dist/runtime/cli.js +4345 -561
- package/dist/runtime/commands/agents.js +31 -31
- package/dist/runtime/commands/budget.js +5 -5
- package/dist/runtime/commands/cancel.js +231 -0
- package/dist/runtime/commands/chain.js +489 -0
- package/dist/runtime/commands/codegraph-status.js +227 -0
- package/dist/runtime/commands/compact.js +297 -0
- package/dist/runtime/commands/config.js +74 -40
- package/dist/runtime/commands/cost.js +199 -0
- package/dist/runtime/commands/delegate.js +27 -4
- package/dist/runtime/commands/dispatch.js +126 -0
- package/dist/runtime/commands/doctor.js +579 -0
- package/dist/runtime/commands/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 +200 -38
- package/dist/runtime/commands/mcp.js +935 -0
- package/dist/runtime/commands/memory.js +582 -0
- package/dist/runtime/commands/model.js +237 -0
- package/dist/runtime/commands/onboarding.js +275 -0
- package/dist/runtime/commands/patch.js +12 -12
- package/dist/runtime/commands/permissions.js +112 -0
- package/dist/runtime/commands/plan.js +143 -0
- package/dist/runtime/commands/prd-check.js +285 -0
- package/dist/runtime/commands/privacy.js +17 -17
- package/dist/runtime/commands/recipe.js +325 -0
- package/dist/runtime/commands/redo-blob-store.js +92 -0
- package/dist/runtime/commands/redo.js +361 -0
- package/dist/runtime/commands/release-notes.js +229 -0
- package/dist/runtime/commands/repo-map.js +95 -0
- package/dist/runtime/commands/report.js +299 -0
- package/dist/runtime/commands/resume.js +118 -0
- package/dist/runtime/commands/review-consensus.js +68 -53
- package/dist/runtime/commands/rewind.js +333 -0
- package/dist/runtime/commands/roster.js +14 -14
- package/dist/runtime/commands/servers.js +236 -0
- package/dist/runtime/commands/sessions.js +163 -0
- package/dist/runtime/commands/share.js +316 -0
- package/dist/runtime/commands/skills.js +31 -31
- package/dist/runtime/commands/status.js +186 -0
- package/dist/runtime/commands/stickers.js +82 -0
- package/dist/runtime/commands/style.js +194 -0
- package/dist/runtime/commands/theme.js +196 -0
- package/dist/runtime/commands/undo.js +54 -22
- package/dist/runtime/commands/update.js +289 -0
- package/dist/runtime/commands/vim.js +140 -0
- package/dist/runtime/commands/worktree.js +8 -8
- package/dist/runtime/commands/worktrees.js +155 -0
- package/dist/runtime/deprecation-warning.js +69 -0
- package/dist/runtime/engine-exit-code.js +50 -0
- package/dist/runtime/headless-repl.js +195 -0
- package/dist/runtime/headless.js +548 -0
- package/dist/runtime/load-hooks-or-exit.js +71 -0
- package/dist/runtime/plan-decompose.js +22 -22
- package/dist/runtime/sigint-guard.js +272 -0
- package/dist/runtime/stream-renderer.js +195 -0
- package/dist/runtime/update-check.js +28 -28
- package/dist/runtime/version.js +65 -0
- package/dist/runtime/worktree-bootstrap.js +579 -0
- package/dist/skills/bundled/batch.js +617 -0
- package/dist/skills/bundled/index.js +45 -0
- package/dist/skills/bundled/loop.js +358 -0
- package/dist/skills/bundled/remember.js +383 -0
- package/dist/skills/bundled/simplify.js +289 -0
- package/dist/skills/bundled/skillify.js +373 -0
- package/dist/skills/bundled/stuck.js +558 -0
- package/dist/skills/bundled/verify.js +439 -0
- package/dist/testing/vcr.js +486 -0
- package/dist/tools/agent-tool.js +229 -0
- package/dist/tools/apply-patch.js +89 -28
- package/dist/tools/ask-user-question.js +337 -0
- package/dist/tools/ask-user.js +115 -0
- package/dist/tools/bash.js +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 +377 -1
- package/dist/tools/mcp-tool.js +260 -0
- package/dist/tools/multi-edit.js +361 -0
- package/dist/tools/powershell.js +268 -0
- package/dist/tools/registry.js +120 -5
- package/dist/tools/server-tools.js +892 -0
- package/dist/tools/skill-tool.js +96 -0
- package/dist/tools/sleep.js +99 -0
- package/dist/tools/synthetic-output.js +133 -0
- package/dist/tools/tasks.js +208 -0
- package/dist/tools/todo-write.js +184 -0
- package/dist/tools/verify-plan-execution.js +295 -0
- package/dist/tools/web-fetch-injection-scanner.js +207 -0
- package/dist/tools/web-fetch.js +195 -10
- package/dist/tools/web-search.js +458 -0
- package/dist/tui/agent-progress-card.js +111 -0
- package/dist/tui/agent-tree.js +22 -1
- package/dist/tui/ask-modal.js +14 -14
- package/dist/tui/ask-user-question-chips.js +315 -0
- package/dist/tui/ask-user-question-prompt.js +203 -0
- package/dist/tui/compact-banner.js +81 -0
- package/dist/tui/conversation-pane.js +85 -11
- package/dist/tui/cost-table.js +111 -0
- package/dist/tui/device-flow.js +2 -2
- package/dist/tui/doctor-table.js +46 -0
- package/dist/tui/feedback-prompt.js +156 -0
- package/dist/tui/input-box.js +247 -32
- package/dist/tui/login-picker.js +3 -3
- package/dist/tui/markdown-render.js +6 -6
- package/dist/tui/multi-file-diff-approval.js +375 -0
- package/dist/tui/onboarding-wizard.js +240 -0
- package/dist/tui/permissions-picker.js +86 -0
- package/dist/tui/render.js +36 -1
- package/dist/tui/repl-render.js +239 -25
- package/dist/tui/repl-splash-art.js +16 -16
- package/dist/tui/repl-splash-mascot.js +48 -24
- package/dist/tui/repl-splash.js +22 -22
- package/dist/tui/repl.js +125 -45
- package/dist/tui/slash-palette.js +6 -6
- package/dist/tui/splash.js +2 -2
- package/dist/tui/status-bar.js +109 -31
- package/dist/tui/status-table.js +7 -0
- package/dist/tui/stickers-art.js +136 -0
- package/dist/tui/style-table.js +28 -0
- package/dist/tui/theme-table.js +29 -0
- package/dist/tui/thinking-spinner.js +123 -0
- package/dist/tui/tool-stream-pane.js +53 -4
- package/dist/tui/update-banner.js +27 -2
- package/dist/tui/vim-input.js +267 -0
- package/dist/tui/welcome-banner.js +107 -0
- package/dist/tui/welcome-data.js +293 -0
- package/dist/tui/workspace-context.js +2 -2
- package/package.json +21 -5
- 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,248 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Codegraph install-decision store — .
|
|
3
|
+
*
|
|
4
|
+
* Persists the operator's verdict on the codegraph install prompt so we
|
|
5
|
+
* never spam them after a single decline. The 30-day reminder cadence
|
|
6
|
+
* lets us re-surface the offer on big-enough repos in case the operator
|
|
7
|
+
* said "not now" the first time and then forgot codegraph exists.
|
|
8
|
+
*
|
|
9
|
+
* Schema (workspace-scoped at `.pugi/codegraph-decision.json`):
|
|
10
|
+
*
|
|
11
|
+
* {
|
|
12
|
+
* "schema": 1,
|
|
13
|
+
* "offeredAt": "2026-05-27T00:00:00.000Z",
|
|
14
|
+
* "accepted": false,
|
|
15
|
+
* "decliningCount": 1,
|
|
16
|
+
* "remindAfter": "2026-06-26T00:00:00.000Z", // 30 days from offeredAt
|
|
17
|
+
* "lastIndexedAt": null, // ISO date string OR null
|
|
18
|
+
* "lastReindexCheckAt": null
|
|
19
|
+
* }
|
|
20
|
+
*
|
|
21
|
+
* The store is workspace-local (each repo gets its own decision) so
|
|
22
|
+
* declining codegraph in repo A does not suppress the prompt in repo B.
|
|
23
|
+
* `.pugi/` already exists by the time we land here (pugi init scaffolds
|
|
24
|
+
* it), so the directory creation is best-effort defence-in-depth.
|
|
25
|
+
*
|
|
26
|
+
* Concurrency: every write is `tmp + rename` so a partial write cannot
|
|
27
|
+
* surface a corrupt JSON. Reads tolerate missing files + corrupt JSON
|
|
28
|
+
* by returning `null` — the caller decides whether to fall back to
|
|
29
|
+
* "offer again" (safe default) or "do nothing" (cold-start path).
|
|
30
|
+
*
|
|
31
|
+
* Pure persistence. No telemetry, no logging. The emitter lives in the
|
|
32
|
+
* call sites so the decision store stays unit-testable in isolation.
|
|
33
|
+
*/
|
|
34
|
+
import { existsSync, mkdirSync, readFileSync, renameSync, unlinkSync, writeFileSync, } from 'node:fs';
|
|
35
|
+
import { resolve } from 'node:path';
|
|
36
|
+
/**
|
|
37
|
+
* Reminder cadence — 30 days from the last decline. Operators who said
|
|
38
|
+
* no in a small repo that grew к medium during a sprint deserve a
|
|
39
|
+
* follow-up; operators who said no last week do not. The window is
|
|
40
|
+
* exposed as a const so the spec can pin it.
|
|
41
|
+
*/
|
|
42
|
+
export const REMIND_AFTER_DAYS = 30;
|
|
43
|
+
/**
|
|
44
|
+
* Stale-index threshold for the cold-start "refresh me" reminder.
|
|
45
|
+
* Seven days is the cadence the upstream codegraph docs recommend for
|
|
46
|
+
* monorepos that ship multiple times a day; lower repos can wait
|
|
47
|
+
* longer. The spec pins it.
|
|
48
|
+
*/
|
|
49
|
+
export const STALE_INDEX_DAYS = 7;
|
|
50
|
+
/**
|
|
51
|
+
* Resolve the decision file path for a workspace root. Pure — exposed
|
|
52
|
+
* для spec parity.
|
|
53
|
+
*/
|
|
54
|
+
export function decisionPath(workspaceRoot) {
|
|
55
|
+
return resolve(workspaceRoot, '.pugi/codegraph-decision.json');
|
|
56
|
+
}
|
|
57
|
+
/**
|
|
58
|
+
* Read the persisted decision. Returns null on missing file, malformed
|
|
59
|
+
* JSON, or wrong schema version. The caller MUST treat null as "no
|
|
60
|
+
* decision yet" — not "operator declined".
|
|
61
|
+
*/
|
|
62
|
+
export function readDecision(workspaceRoot) {
|
|
63
|
+
const path = decisionPath(workspaceRoot);
|
|
64
|
+
if (!existsSync(path))
|
|
65
|
+
return null;
|
|
66
|
+
let raw;
|
|
67
|
+
try {
|
|
68
|
+
raw = readFileSync(path, 'utf8');
|
|
69
|
+
}
|
|
70
|
+
catch {
|
|
71
|
+
return null;
|
|
72
|
+
}
|
|
73
|
+
let parsed;
|
|
74
|
+
try {
|
|
75
|
+
parsed = JSON.parse(raw);
|
|
76
|
+
}
|
|
77
|
+
catch {
|
|
78
|
+
return null;
|
|
79
|
+
}
|
|
80
|
+
if (!isDecisionShape(parsed))
|
|
81
|
+
return null;
|
|
82
|
+
return parsed;
|
|
83
|
+
}
|
|
84
|
+
/**
|
|
85
|
+
* Type guard. Defensive — a future schema bump should land a migration
|
|
86
|
+
* here. For now: schema MUST be 1; required string fields MUST be
|
|
87
|
+
* strings; optional fields may be null OR string.
|
|
88
|
+
*/
|
|
89
|
+
function isDecisionShape(value) {
|
|
90
|
+
if (!value || typeof value !== 'object')
|
|
91
|
+
return false;
|
|
92
|
+
const v = value;
|
|
93
|
+
if (v.schema !== 1)
|
|
94
|
+
return false;
|
|
95
|
+
if (typeof v.offeredAt !== 'string')
|
|
96
|
+
return false;
|
|
97
|
+
if (typeof v.accepted !== 'boolean')
|
|
98
|
+
return false;
|
|
99
|
+
if (typeof v.decliningCount !== 'number' || !Number.isFinite(v.decliningCount))
|
|
100
|
+
return false;
|
|
101
|
+
if (typeof v.remindAfter !== 'string')
|
|
102
|
+
return false;
|
|
103
|
+
if (v.lastIndexedAt !== null && typeof v.lastIndexedAt !== 'string')
|
|
104
|
+
return false;
|
|
105
|
+
if (v.lastReindexCheckAt !== null && typeof v.lastReindexCheckAt !== 'string')
|
|
106
|
+
return false;
|
|
107
|
+
return true;
|
|
108
|
+
}
|
|
109
|
+
/**
|
|
110
|
+
* Atomic write. Creates `.pugi/` if it does not exist (pugi init owns
|
|
111
|
+
* that surface ordinarily; we defend the rare cold-start path).
|
|
112
|
+
*/
|
|
113
|
+
export function writeDecision(workspaceRoot, decision) {
|
|
114
|
+
const path = decisionPath(workspaceRoot);
|
|
115
|
+
const dir = resolve(workspaceRoot, '.pugi');
|
|
116
|
+
if (!existsSync(dir)) {
|
|
117
|
+
mkdirSync(dir, { recursive: true });
|
|
118
|
+
}
|
|
119
|
+
const tmp = `${path}.tmp.${process.pid}`;
|
|
120
|
+
writeFileSync(tmp, `${JSON.stringify(decision, null, 2)}\n`, { mode: 0o600 });
|
|
121
|
+
try {
|
|
122
|
+
renameSync(tmp, path);
|
|
123
|
+
}
|
|
124
|
+
catch (error) {
|
|
125
|
+
// Rename can fail if the destination was concurrently swapped on
|
|
126
|
+
// some platforms (Windows). Fall back to unlink + rename so the
|
|
127
|
+
// best-effort write does not throw to the caller.
|
|
128
|
+
try {
|
|
129
|
+
unlinkSync(path);
|
|
130
|
+
}
|
|
131
|
+
catch {
|
|
132
|
+
// ignore
|
|
133
|
+
}
|
|
134
|
+
renameSync(tmp, path);
|
|
135
|
+
void error;
|
|
136
|
+
}
|
|
137
|
+
}
|
|
138
|
+
/**
|
|
139
|
+
* Decide whether to surface the install prompt on init. Returns the
|
|
140
|
+
* full decision shape for callers that want to inspect the cadence;
|
|
141
|
+
* a `true` verdict means "yes, ask the operator now".
|
|
142
|
+
*/
|
|
143
|
+
export function shouldOfferOnInit(workspaceRoot, nowIso = new Date().toISOString()) {
|
|
144
|
+
const prior = readDecision(workspaceRoot);
|
|
145
|
+
if (!prior) {
|
|
146
|
+
return { shouldOffer: true, reason: 'first-run' };
|
|
147
|
+
}
|
|
148
|
+
if (prior.accepted) {
|
|
149
|
+
return { shouldOffer: false, reason: 'accepted-already' };
|
|
150
|
+
}
|
|
151
|
+
if (Date.parse(nowIso) >= Date.parse(prior.remindAfter)) {
|
|
152
|
+
return { shouldOffer: true, reason: 'reminder-due' };
|
|
153
|
+
}
|
|
154
|
+
return { shouldOffer: false, reason: 'recent-decline' };
|
|
155
|
+
}
|
|
156
|
+
/**
|
|
157
|
+
* Record the operator's decision atomically. Mirrors the structure on
|
|
158
|
+
* disk — callers do NOT hand-craft the schema.
|
|
159
|
+
*/
|
|
160
|
+
export function recordDecision(workspaceRoot, input) {
|
|
161
|
+
const nowIso = input.nowIso ?? new Date().toISOString();
|
|
162
|
+
const prior = readDecision(workspaceRoot);
|
|
163
|
+
const decliningCount = input.accepted ? 0 : (prior?.decliningCount ?? 0) + 1;
|
|
164
|
+
const remindAfter = new Date(Date.parse(nowIso) + REMIND_AFTER_DAYS * 24 * 60 * 60 * 1000).toISOString();
|
|
165
|
+
const decision = {
|
|
166
|
+
schema: 1,
|
|
167
|
+
offeredAt: nowIso,
|
|
168
|
+
accepted: input.accepted,
|
|
169
|
+
decliningCount,
|
|
170
|
+
remindAfter,
|
|
171
|
+
lastIndexedAt: prior?.lastIndexedAt ?? null,
|
|
172
|
+
lastReindexCheckAt: prior?.lastReindexCheckAt ?? null,
|
|
173
|
+
};
|
|
174
|
+
writeDecision(workspaceRoot, decision);
|
|
175
|
+
return decision;
|
|
176
|
+
}
|
|
177
|
+
/**
|
|
178
|
+
* Stamp the last-indexed timestamp. Called by /codegraph-status when
|
|
179
|
+
* the operator triggers a reindex from inside Pugi. Updates the
|
|
180
|
+
* `accepted` decision in place — never flips the install state.
|
|
181
|
+
*/
|
|
182
|
+
export function markIndexed(workspaceRoot, nowIso = new Date().toISOString()) {
|
|
183
|
+
const prior = readDecision(workspaceRoot);
|
|
184
|
+
if (!prior)
|
|
185
|
+
return null;
|
|
186
|
+
const next = {
|
|
187
|
+
...prior,
|
|
188
|
+
lastIndexedAt: nowIso,
|
|
189
|
+
lastReindexCheckAt: nowIso,
|
|
190
|
+
};
|
|
191
|
+
writeDecision(workspaceRoot, next);
|
|
192
|
+
return next;
|
|
193
|
+
}
|
|
194
|
+
/**
|
|
195
|
+
* Stamp the last reindex-check timestamp without changing the index
|
|
196
|
+
* itself. Used by the cold-start hook so we do not show the "index is
|
|
197
|
+
* stale" hint on every keystroke once the operator has acknowledged
|
|
198
|
+
* it.
|
|
199
|
+
*/
|
|
200
|
+
export function markReindexChecked(workspaceRoot, nowIso = new Date().toISOString()) {
|
|
201
|
+
const prior = readDecision(workspaceRoot);
|
|
202
|
+
if (!prior)
|
|
203
|
+
return null;
|
|
204
|
+
const next = {
|
|
205
|
+
...prior,
|
|
206
|
+
lastReindexCheckAt: nowIso,
|
|
207
|
+
};
|
|
208
|
+
writeDecision(workspaceRoot, next);
|
|
209
|
+
return next;
|
|
210
|
+
}
|
|
211
|
+
/**
|
|
212
|
+
* Compute the staleness of the codegraph index. Pure — no IO.
|
|
213
|
+
*
|
|
214
|
+
* - returns null when `lastIndexedAt` is null (never indexed)
|
|
215
|
+
* - returns the day-delta (rounded down) otherwise
|
|
216
|
+
*/
|
|
217
|
+
export function indexAgeDays(decision, nowIso = new Date().toISOString()) {
|
|
218
|
+
if (!decision.lastIndexedAt)
|
|
219
|
+
return null;
|
|
220
|
+
const deltaMs = Date.parse(nowIso) - Date.parse(decision.lastIndexedAt);
|
|
221
|
+
if (!Number.isFinite(deltaMs) || deltaMs < 0)
|
|
222
|
+
return 0;
|
|
223
|
+
return Math.floor(deltaMs / (24 * 60 * 60 * 1000));
|
|
224
|
+
}
|
|
225
|
+
/**
|
|
226
|
+
* Convenience predicate — should the cold-start hook show the stale-
|
|
227
|
+
* index reminder? `true` when the index is older than STALE_INDEX_DAYS
|
|
228
|
+
* AND we did NOT already remind the operator today.
|
|
229
|
+
*/
|
|
230
|
+
export function shouldNudgeStaleIndex(decision, nowIso = new Date().toISOString()) {
|
|
231
|
+
if (!decision.accepted)
|
|
232
|
+
return false;
|
|
233
|
+
const age = indexAgeDays(decision, nowIso);
|
|
234
|
+
if (age === null)
|
|
235
|
+
return false;
|
|
236
|
+
if (age < STALE_INDEX_DAYS)
|
|
237
|
+
return false;
|
|
238
|
+
// Throttle the nudge to once per day so the operator does not see it
|
|
239
|
+
// on every REPL keystroke.
|
|
240
|
+
if (decision.lastReindexCheckAt) {
|
|
241
|
+
const lastCheckDelta = Date.parse(nowIso) - Date.parse(decision.lastReindexCheckAt);
|
|
242
|
+
if (Number.isFinite(lastCheckDelta) && lastCheckDelta < 24 * 60 * 60 * 1000) {
|
|
243
|
+
return false;
|
|
244
|
+
}
|
|
245
|
+
}
|
|
246
|
+
return true;
|
|
247
|
+
}
|
|
248
|
+
//# sourceMappingURL=decision-store.js.map
|
|
@@ -0,0 +1,459 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Repo detection helper — (codegraph
|
|
3
|
+
* context-aware auto-install).
|
|
4
|
+
*
|
|
5
|
+
* Walks up from `cwd` looking for a `.git/` directory; if found, runs a
|
|
6
|
+
* bounded scan (≤ MAX_SCAN_FILES files, depth-limited) to classify the
|
|
7
|
+
* repo size + detect primary languages. The result feeds two surfaces:
|
|
8
|
+
*
|
|
9
|
+
* 1. `pugi init` — when the repo is medium+ AND has a supported
|
|
10
|
+
* primary language, the init flow asks the operator whether to
|
|
11
|
+
* install the codegraph MCP server (Phase 1 example config →
|
|
12
|
+
* auto-merged into .pugi/mcp.json).
|
|
13
|
+
* 2. Cold-start hook — every `pugi` invocation looks up the last-asked
|
|
14
|
+
* timestamp; if the operator declined more than 30 days ago AND
|
|
15
|
+
* the repo still triggers, we surface a one-line nudge.
|
|
16
|
+
*
|
|
17
|
+
* Why a stand-alone scanner (vs. reusing core/repo-map/scanner.ts):
|
|
18
|
+
*
|
|
19
|
+
* - repo-map scans up to 5000 files with statSync per file to build a
|
|
20
|
+
* symbol cache. We need a much cheaper classification: ≤ 1000 files,
|
|
21
|
+
* no stat for size (just dirent.name), bail early on the first
|
|
22
|
+
* manifest file we recognise. The 5000-file walker would dominate
|
|
23
|
+
* cold-start latency on a monorepo.
|
|
24
|
+
* - We deliberately do NOT respect .pugiignore here — codegraph cares
|
|
25
|
+
* about the WHOLE repo (vendored libs included), not the operator's
|
|
26
|
+
* curated workspace view. The repo-map scanner does the opposite.
|
|
27
|
+
* - The output is structurally different (size category + language
|
|
28
|
+
* manifest list) so even if we reused the walker the post-processing
|
|
29
|
+
* would be different. Keeping the scans independent prevents one
|
|
30
|
+
* surface's heuristics from leaking into the other.
|
|
31
|
+
*
|
|
32
|
+
* Pure module: no logging, no network, no telemetry. Errors during
|
|
33
|
+
* readdir on a subtree (permission denied, symlink loop) are swallowed
|
|
34
|
+
* and the walker continues — repo detection is best-effort context.
|
|
35
|
+
* The function NEVER throws; a malformed cwd returns `{ isRepo: false }`.
|
|
36
|
+
*/
|
|
37
|
+
import { readdirSync, existsSync, statSync, readFileSync } from 'node:fs';
|
|
38
|
+
import { dirname, join, resolve } from 'node:path';
|
|
39
|
+
/**
|
|
40
|
+
* Maximum directories we descend into. A 1000-file repo classifies as
|
|
41
|
+
* "large" already; deeper walks add latency without changing the
|
|
42
|
+
* verdict. Hit this cap and we cap the file count at MAX_SCAN_FILES
|
|
43
|
+
* and return the verdict — the codegraph decision does not care
|
|
44
|
+
* whether the repo is 1001 or 100_001 files.
|
|
45
|
+
*/
|
|
46
|
+
export const MAX_SCAN_FILES = 1000;
|
|
47
|
+
/**
|
|
48
|
+
* Hard cap on walk depth. Counted from `gitRoot`. Anything beyond is
|
|
49
|
+
* deep tooling output (node_modules, vendored deps, generated). The
|
|
50
|
+
* scanner short-circuits at this depth and the result is reported as
|
|
51
|
+
* `wasCapped: true` so the consumer can hint the operator if needed.
|
|
52
|
+
*/
|
|
53
|
+
export const MAX_SCAN_DEPTH = 6;
|
|
54
|
+
/**
|
|
55
|
+
* Walk-up cap searching for the `.git/` parent. 12 ancestors matches
|
|
56
|
+
* the bootstrap.ts contract for project-marker detection — operators
|
|
57
|
+
* running `pugi` from `node_modules/foo/bar/baz` are an error case, not
|
|
58
|
+
* a feature.
|
|
59
|
+
*/
|
|
60
|
+
export const MAX_GIT_WALK = 12;
|
|
61
|
+
/**
|
|
62
|
+
* Directory basenames we skip outright. These are pure noise for code
|
|
63
|
+
* navigation — vendored deps, build artefacts, version-control plumbing,
|
|
64
|
+
* cached test outputs. Trimming them at the dirent level keeps the scan
|
|
65
|
+
* O(useful-source-files) rather than O(everything-on-disk).
|
|
66
|
+
*/
|
|
67
|
+
const SKIP_DIRS = new Set([
|
|
68
|
+
'.git',
|
|
69
|
+
'node_modules',
|
|
70
|
+
'dist',
|
|
71
|
+
'build',
|
|
72
|
+
'out',
|
|
73
|
+
'.next',
|
|
74
|
+
'.nuxt',
|
|
75
|
+
'.turbo',
|
|
76
|
+
'.cache',
|
|
77
|
+
'coverage',
|
|
78
|
+
'__pycache__',
|
|
79
|
+
'.venv',
|
|
80
|
+
'venv',
|
|
81
|
+
'target',
|
|
82
|
+
'.gradle',
|
|
83
|
+
'.idea',
|
|
84
|
+
'.vscode',
|
|
85
|
+
'.pugi',
|
|
86
|
+
]);
|
|
87
|
+
/**
|
|
88
|
+
* Source-file extensions that count toward the size category. Mirrors
|
|
89
|
+
* the codegraph upstream's own language list (tree-sitter grammars for
|
|
90
|
+
* 19 languages) but pruned to the set that we actually return as
|
|
91
|
+
* `supported` languages — the categorisation must agree with the
|
|
92
|
+
* language-match gate that drives the install prompt.
|
|
93
|
+
*/
|
|
94
|
+
const SOURCE_EXTENSIONS = new Set([
|
|
95
|
+
'.ts', '.tsx', '.js', '.jsx', '.mjs', '.cjs',
|
|
96
|
+
'.py',
|
|
97
|
+
'.rs',
|
|
98
|
+
'.go',
|
|
99
|
+
'.java', '.kt', '.kts',
|
|
100
|
+
'.rb',
|
|
101
|
+
'.php',
|
|
102
|
+
'.c', '.h', '.cpp', '.hpp', '.cc', '.cxx',
|
|
103
|
+
'.cs',
|
|
104
|
+
'.swift',
|
|
105
|
+
'.scala',
|
|
106
|
+
]);
|
|
107
|
+
export const CODEGRAPH_SUPPORTED_LANGUAGES = Object.freeze([
|
|
108
|
+
'typescript',
|
|
109
|
+
'javascript',
|
|
110
|
+
'python',
|
|
111
|
+
'rust',
|
|
112
|
+
'go',
|
|
113
|
+
'java',
|
|
114
|
+
]);
|
|
115
|
+
const MANIFEST_HINTS = Object.freeze([
|
|
116
|
+
{ filename: 'package.json', languages: ['javascript', 'typescript'] },
|
|
117
|
+
{ filename: 'tsconfig.json', languages: ['typescript'] },
|
|
118
|
+
{ filename: 'pyproject.toml', languages: ['python'] },
|
|
119
|
+
{ filename: 'requirements.txt', languages: ['python'] },
|
|
120
|
+
{ filename: 'setup.py', languages: ['python'] },
|
|
121
|
+
{ filename: 'Pipfile', languages: ['python'] },
|
|
122
|
+
{ filename: 'Cargo.toml', languages: ['rust'] },
|
|
123
|
+
{ filename: 'go.mod', languages: ['go'] },
|
|
124
|
+
{ filename: 'pom.xml', languages: ['java'] },
|
|
125
|
+
{ filename: 'build.gradle', languages: ['java'] },
|
|
126
|
+
{ filename: 'build.gradle.kts', languages: ['java'] },
|
|
127
|
+
]);
|
|
128
|
+
/**
|
|
129
|
+
* Walk up from `cwd` looking for `.git/`. Returns the absolute path of
|
|
130
|
+
* the git-root directory OR null if none is found within the walk cap.
|
|
131
|
+
*/
|
|
132
|
+
export function findGitRoot(cwd) {
|
|
133
|
+
let current = resolve(cwd);
|
|
134
|
+
for (let i = 0; i < MAX_GIT_WALK; i += 1) {
|
|
135
|
+
if (existsSync(join(current, '.git')))
|
|
136
|
+
return current;
|
|
137
|
+
const parent = dirname(current);
|
|
138
|
+
if (parent === current)
|
|
139
|
+
return null;
|
|
140
|
+
current = parent;
|
|
141
|
+
}
|
|
142
|
+
return null;
|
|
143
|
+
}
|
|
144
|
+
/**
|
|
145
|
+
* Read manifest files at the git root + classify their languages.
|
|
146
|
+
* Pure file-system stat — we do NOT parse the manifests (a malformed
|
|
147
|
+
* package.json should not break detection). Multiple manifests can
|
|
148
|
+
* hit; e.g. a polyglot repo with package.json + pyproject.toml lights
|
|
149
|
+
* up both `typescript`/`javascript` and `python`.
|
|
150
|
+
*
|
|
151
|
+
* `package.json` is special-cased: if a `tsconfig.json` lives next to
|
|
152
|
+
* it we drop the `javascript` hint so a pure-TS repo doesn't get
|
|
153
|
+
* counted twice. JS-only repos surface as `javascript`.
|
|
154
|
+
*/
|
|
155
|
+
function detectManifestLanguages(gitRoot) {
|
|
156
|
+
const out = new Set();
|
|
157
|
+
const hasTsconfig = existsSync(join(gitRoot, 'tsconfig.json'));
|
|
158
|
+
for (const hint of MANIFEST_HINTS) {
|
|
159
|
+
if (!existsSync(join(gitRoot, hint.filename)))
|
|
160
|
+
continue;
|
|
161
|
+
for (const lang of hint.languages) {
|
|
162
|
+
if (hint.filename === 'package.json' && hasTsconfig && lang === 'javascript') {
|
|
163
|
+
continue;
|
|
164
|
+
}
|
|
165
|
+
out.add(lang);
|
|
166
|
+
}
|
|
167
|
+
}
|
|
168
|
+
return out;
|
|
169
|
+
}
|
|
170
|
+
/**
|
|
171
|
+
* Bounded BFS-by-stack walk from `root`. Caps at MAX_SCAN_FILES +
|
|
172
|
+
* MAX_SCAN_DEPTH. Returns `{ srcCount, languages, wasCapped }`.
|
|
173
|
+
*
|
|
174
|
+
* Why a manual stack rather than `readdirSync({ recursive: true })`:
|
|
175
|
+
* the recursive readdir loads the entire tree into memory before we
|
|
176
|
+
* see the first entry; on a monorepo we would walk node_modules
|
|
177
|
+
* (forbidden) before our SKIP_DIRS filter could fire. A manual stack
|
|
178
|
+
* lets us prune at every level.
|
|
179
|
+
*/
|
|
180
|
+
function scanRepo(root) {
|
|
181
|
+
const languagesFromExt = new Set();
|
|
182
|
+
let srcCount = 0;
|
|
183
|
+
let wasCapped = false;
|
|
184
|
+
// Per-language file counts so we can apply the "at least 3 files"
|
|
185
|
+
// threshold below — a single stray .py in a TS repo should not light
|
|
186
|
+
// up python.
|
|
187
|
+
const perLang = new Map();
|
|
188
|
+
const stack = [{ abs: root, depth: 0 }];
|
|
189
|
+
while (stack.length > 0) {
|
|
190
|
+
const frame = stack.pop();
|
|
191
|
+
if (!frame)
|
|
192
|
+
break;
|
|
193
|
+
if (frame.depth > MAX_SCAN_DEPTH) {
|
|
194
|
+
wasCapped = true;
|
|
195
|
+
continue;
|
|
196
|
+
}
|
|
197
|
+
let entries;
|
|
198
|
+
try {
|
|
199
|
+
entries = readdirSync(frame.abs, { withFileTypes: true });
|
|
200
|
+
}
|
|
201
|
+
catch {
|
|
202
|
+
continue;
|
|
203
|
+
}
|
|
204
|
+
for (const entry of entries) {
|
|
205
|
+
if (srcCount >= MAX_SCAN_FILES) {
|
|
206
|
+
wasCapped = true;
|
|
207
|
+
break;
|
|
208
|
+
}
|
|
209
|
+
if (entry.isDirectory()) {
|
|
210
|
+
if (SKIP_DIRS.has(entry.name))
|
|
211
|
+
continue;
|
|
212
|
+
if (entry.name.startsWith('.') && entry.name !== '.') {
|
|
213
|
+
// Skip hidden dirs that are NOT in our whitelist (e.g. .yarn,
|
|
214
|
+
// .pnpm-store). .pugi/.git are already in SKIP_DIRS.
|
|
215
|
+
continue;
|
|
216
|
+
}
|
|
217
|
+
stack.push({ abs: join(frame.abs, entry.name), depth: frame.depth + 1 });
|
|
218
|
+
continue;
|
|
219
|
+
}
|
|
220
|
+
if (!entry.isFile())
|
|
221
|
+
continue;
|
|
222
|
+
const ext = extensionOf(entry.name);
|
|
223
|
+
if (!ext)
|
|
224
|
+
continue;
|
|
225
|
+
if (!SOURCE_EXTENSIONS.has(ext))
|
|
226
|
+
continue;
|
|
227
|
+
srcCount += 1;
|
|
228
|
+
const lang = languageForExtension(ext);
|
|
229
|
+
if (lang) {
|
|
230
|
+
perLang.set(lang, (perLang.get(lang) ?? 0) + 1);
|
|
231
|
+
}
|
|
232
|
+
}
|
|
233
|
+
if (srcCount >= MAX_SCAN_FILES) {
|
|
234
|
+
wasCapped = true;
|
|
235
|
+
break;
|
|
236
|
+
}
|
|
237
|
+
}
|
|
238
|
+
// Promote per-language counts to detected languages with a noise
|
|
239
|
+
// floor. The threshold is intentionally low (3) so a small repo
|
|
240
|
+
// with a handful of Python utility scripts in an otherwise-TS
|
|
241
|
+
// codebase still surfaces as polyglot.
|
|
242
|
+
for (const [lang, count] of perLang.entries()) {
|
|
243
|
+
if (count >= 3)
|
|
244
|
+
languagesFromExt.add(lang);
|
|
245
|
+
}
|
|
246
|
+
return { srcCount, languagesFromExt, wasCapped };
|
|
247
|
+
}
|
|
248
|
+
/**
|
|
249
|
+
* Map a file extension to a SupportedLanguage. Returns null for
|
|
250
|
+
* extensions we count toward the size category but do NOT surface as
|
|
251
|
+
* supported languages (c, cs, swift, scala, …) — codegraph supports
|
|
252
|
+
* more languages than our nudge gate.
|
|
253
|
+
*/
|
|
254
|
+
export function languageForExtension(ext) {
|
|
255
|
+
switch (ext) {
|
|
256
|
+
case '.ts':
|
|
257
|
+
case '.tsx':
|
|
258
|
+
return 'typescript';
|
|
259
|
+
case '.js':
|
|
260
|
+
case '.jsx':
|
|
261
|
+
case '.mjs':
|
|
262
|
+
case '.cjs':
|
|
263
|
+
return 'javascript';
|
|
264
|
+
case '.py':
|
|
265
|
+
return 'python';
|
|
266
|
+
case '.rs':
|
|
267
|
+
return 'rust';
|
|
268
|
+
case '.go':
|
|
269
|
+
return 'go';
|
|
270
|
+
case '.java':
|
|
271
|
+
case '.kt':
|
|
272
|
+
case '.kts':
|
|
273
|
+
return 'java';
|
|
274
|
+
default:
|
|
275
|
+
return null;
|
|
276
|
+
}
|
|
277
|
+
}
|
|
278
|
+
/**
|
|
279
|
+
* Lowercase extension including the leading dot. Returns null for
|
|
280
|
+
* dotfiles (no extension) and for paths without a dot.
|
|
281
|
+
*/
|
|
282
|
+
function extensionOf(filename) {
|
|
283
|
+
const idx = filename.lastIndexOf('.');
|
|
284
|
+
if (idx <= 0)
|
|
285
|
+
return null;
|
|
286
|
+
// `.env` style dotfiles return `'.env'` from lastIndexOf — they
|
|
287
|
+
// are filtered by the SOURCE_EXTENSIONS membership check below
|
|
288
|
+
// since `.env` is not in the source-language set.
|
|
289
|
+
return filename.slice(idx).toLowerCase();
|
|
290
|
+
}
|
|
291
|
+
/**
|
|
292
|
+
* Classify a file count into the small / medium / large bucket. Pure —
|
|
293
|
+
* exposed so spec callers can pin the exact thresholds independent of
|
|
294
|
+
* the rest of the walker. Operators reading the prompt copy MUST see
|
|
295
|
+
* the same boundaries the implementation enforces.
|
|
296
|
+
*/
|
|
297
|
+
export function categoriseSize(srcCount) {
|
|
298
|
+
if (srcCount <= 50)
|
|
299
|
+
return 'small';
|
|
300
|
+
if (srcCount <= 500)
|
|
301
|
+
return 'medium';
|
|
302
|
+
return 'large';
|
|
303
|
+
}
|
|
304
|
+
/**
|
|
305
|
+
* Single entry-point. Pure. Never throws. Returns the structured
|
|
306
|
+
* verdict so the init prompt + cold-start hook + status command can
|
|
307
|
+
* all branch off one shared computation.
|
|
308
|
+
*/
|
|
309
|
+
export function detectRepo(cwd) {
|
|
310
|
+
let absCwd;
|
|
311
|
+
try {
|
|
312
|
+
absCwd = resolve(cwd);
|
|
313
|
+
const stat = statSync(absCwd);
|
|
314
|
+
if (!stat.isDirectory()) {
|
|
315
|
+
return { isRepo: false, reason: 'unreadable-cwd' };
|
|
316
|
+
}
|
|
317
|
+
}
|
|
318
|
+
catch {
|
|
319
|
+
return { isRepo: false, reason: 'unreadable-cwd' };
|
|
320
|
+
}
|
|
321
|
+
const gitRoot = findGitRoot(absCwd);
|
|
322
|
+
if (!gitRoot) {
|
|
323
|
+
return { isRepo: false, reason: 'no-git' };
|
|
324
|
+
}
|
|
325
|
+
const manifestLangs = detectManifestLanguages(gitRoot);
|
|
326
|
+
const { srcCount, languagesFromExt, wasCapped } = scanRepo(gitRoot);
|
|
327
|
+
const allLangs = new Set();
|
|
328
|
+
for (const lang of manifestLangs)
|
|
329
|
+
allLangs.add(lang);
|
|
330
|
+
for (const lang of languagesFromExt)
|
|
331
|
+
allLangs.add(lang);
|
|
332
|
+
const sortedLangs = [...allLangs].sort();
|
|
333
|
+
const sizeCategory = categoriseSize(srcCount);
|
|
334
|
+
const offerCodegraph = sizeCategory !== 'small' && sortedLangs.length > 0;
|
|
335
|
+
return {
|
|
336
|
+
isRepo: true,
|
|
337
|
+
gitRoot,
|
|
338
|
+
sizeCategory,
|
|
339
|
+
primarySymbolCount: srcCount,
|
|
340
|
+
languages: sortedLangs,
|
|
341
|
+
offerCodegraph,
|
|
342
|
+
wasCapped,
|
|
343
|
+
};
|
|
344
|
+
}
|
|
345
|
+
/**
|
|
346
|
+
* Render the cold-start nudge copy. Pure — exposed for spec parity so
|
|
347
|
+
* the cold-start hook + the spec assert against the same line.
|
|
348
|
+
*
|
|
349
|
+
* "Detected medium TypeScript repo with ~200 src files. Install
|
|
350
|
+
* codegraph MCP for symbol-aware code navigation? (Y/n)"
|
|
351
|
+
*
|
|
352
|
+
* The "~N" formatting rounds to the nearest 10 so the operator does
|
|
353
|
+
* not see a flapping count between scans on the same repo (one stray
|
|
354
|
+
* test fixture would change a precise N by 1).
|
|
355
|
+
*/
|
|
356
|
+
export function buildOfferCopy(detection) {
|
|
357
|
+
const noun = humanLanguageLabel(detection.languages);
|
|
358
|
+
const sizeLabel = detection.sizeCategory === 'large' ? 'large' : 'medium';
|
|
359
|
+
const approx = approxFileCount(detection.primarySymbolCount, detection.wasCapped);
|
|
360
|
+
const strength = detection.sizeCategory === 'large' ? 'strongly recommended' : 'recommended';
|
|
361
|
+
return `Detected ${sizeLabel} ${noun} repo with ${approx} src files (${strength}). Install codegraph MCP for symbol-aware code navigation? (Y/n)`;
|
|
362
|
+
}
|
|
363
|
+
/**
|
|
364
|
+
* Render the "X days old, refresh?" reminder copy. Single sentence,
|
|
365
|
+
* one CTA. The session module surfaces this on the system pane when
|
|
366
|
+
* the codegraph mcp.json entry exists but `lastIndexedAt` is stale.
|
|
367
|
+
*/
|
|
368
|
+
export function buildStaleIndexCopy(daysOld) {
|
|
369
|
+
return `Codegraph index is ${daysOld} day${daysOld === 1 ? '' : 's'} old. Run /codegraph-status to refresh.`;
|
|
370
|
+
}
|
|
371
|
+
/**
|
|
372
|
+
* Render the primary language label for the prompt. We use the
|
|
373
|
+
* first detected language (alphabetic order from `detectRepo`) so
|
|
374
|
+
* the copy is deterministic. Multi-language repos get a generic
|
|
375
|
+
* `polyglot` suffix — saves us from enumerating "TypeScript +
|
|
376
|
+
* Python + Rust + …" in the headline.
|
|
377
|
+
*/
|
|
378
|
+
function humanLanguageLabel(langs) {
|
|
379
|
+
if (langs.length === 0)
|
|
380
|
+
return 'source';
|
|
381
|
+
const primary = labelFor(langs[0]);
|
|
382
|
+
if (langs.length === 1)
|
|
383
|
+
return primary;
|
|
384
|
+
return `${primary} polyglot`;
|
|
385
|
+
}
|
|
386
|
+
function labelFor(lang) {
|
|
387
|
+
switch (lang) {
|
|
388
|
+
case 'typescript': return 'TypeScript';
|
|
389
|
+
case 'javascript': return 'JavaScript';
|
|
390
|
+
case 'python': return 'Python';
|
|
391
|
+
case 'rust': return 'Rust';
|
|
392
|
+
case 'go': return 'Go';
|
|
393
|
+
case 'java': return 'Java';
|
|
394
|
+
}
|
|
395
|
+
}
|
|
396
|
+
function approxFileCount(count, wasCapped) {
|
|
397
|
+
if (wasCapped)
|
|
398
|
+
return `${MAX_SCAN_FILES}+`;
|
|
399
|
+
if (count < 10)
|
|
400
|
+
return String(count);
|
|
401
|
+
const rounded = Math.round(count / 10) * 10;
|
|
402
|
+
return `~${rounded}`;
|
|
403
|
+
}
|
|
404
|
+
/**
|
|
405
|
+
* Parse a manifest hint into its declared languages. Tiny export so
|
|
406
|
+
* spec callers can drive the same map without re-declaring it.
|
|
407
|
+
*/
|
|
408
|
+
export function manifestHintFor(filename) {
|
|
409
|
+
for (const hint of MANIFEST_HINTS) {
|
|
410
|
+
if (hint.filename === filename)
|
|
411
|
+
return hint.languages;
|
|
412
|
+
}
|
|
413
|
+
return null;
|
|
414
|
+
}
|
|
415
|
+
/**
|
|
416
|
+
* Best-effort manifest-only language detection. Useful for callers
|
|
417
|
+
* that already have a `gitRoot` and do NOT want to walk the filesystem
|
|
418
|
+
* (e.g. /codegraph-status, which trusts the existing scan cache).
|
|
419
|
+
*
|
|
420
|
+
* The file-IO is just `existsSync` per manifest — no read/parse. A
|
|
421
|
+
* package.json that mentions Python in its `engines` will NOT light up
|
|
422
|
+
* python here; that is by design. The full scan is the source of truth.
|
|
423
|
+
*/
|
|
424
|
+
export function detectManifestLanguagesPublic(gitRoot) {
|
|
425
|
+
return [...detectManifestLanguages(gitRoot)].sort();
|
|
426
|
+
}
|
|
427
|
+
/**
|
|
428
|
+
* Sniff whether a manifest is structurally JSON. Pure — exposed only
|
|
429
|
+
* so a future surface can avoid double-checking. The manifest probe
|
|
430
|
+
* itself does NOT depend on this; it uses existsSync and trusts the
|
|
431
|
+
* filename heuristic.
|
|
432
|
+
*
|
|
433
|
+
* @internal
|
|
434
|
+
*/
|
|
435
|
+
export function looksLikeJson(text) {
|
|
436
|
+
const trimmed = text.trim();
|
|
437
|
+
if (trimmed.length === 0)
|
|
438
|
+
return false;
|
|
439
|
+
return trimmed.startsWith('{') || trimmed.startsWith('[');
|
|
440
|
+
}
|
|
441
|
+
/**
|
|
442
|
+
* Read the contents of a manifest file. Truncated к 8 KiB so a
|
|
443
|
+
* misnamed multi-MB file (e.g. a vendored lockfile masquerading as
|
|
444
|
+
* package.json) cannot stall the detector. Returns null on any IO
|
|
445
|
+
* error. Reserved for future use by surfaces that want manifest
|
|
446
|
+
* content beyond presence checks.
|
|
447
|
+
*
|
|
448
|
+
* @internal
|
|
449
|
+
*/
|
|
450
|
+
export function readManifestTruncated(absPath, maxBytes = 8 * 1024) {
|
|
451
|
+
try {
|
|
452
|
+
const raw = readFileSync(absPath, 'utf8');
|
|
453
|
+
return raw.length > maxBytes ? raw.slice(0, maxBytes) : raw;
|
|
454
|
+
}
|
|
455
|
+
catch {
|
|
456
|
+
return null;
|
|
457
|
+
}
|
|
458
|
+
}
|
|
459
|
+
//# sourceMappingURL=detect-repo.js.map
|