@pugi/cli 0.1.0-beta.10 → 0.1.0-beta.101

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (464) hide show
  1. package/CHANGELOG.md +132 -0
  2. package/LICENSE +1 -1
  3. package/README.md +55 -11
  4. package/assets/pugi-prozr2-mascot.ansi +9 -0
  5. package/bin/run.js +33 -1
  6. package/dist/commands/deploy.js +40 -40
  7. package/dist/commands/flatten.js +191 -0
  8. package/dist/commands/jobs-watch.js +201 -0
  9. package/dist/commands/jobs.js +42 -27
  10. package/dist/commands/retro.js +210 -0
  11. package/dist/commands/smoke.js +133 -0
  12. package/dist/core/agent-progress/cleanup.js +134 -0
  13. package/dist/core/agent-progress/schema.js +144 -0
  14. package/dist/core/agent-progress/writer.js +101 -0
  15. package/dist/core/agents/adaptive-router.js +330 -0
  16. package/dist/core/agents/query-decomposer.js +297 -0
  17. package/dist/core/agents/registry.js +3 -3
  18. package/dist/core/approvals/shortcut-resolver.js +98 -0
  19. package/dist/core/artifact-chain/dispatcher.js +148 -0
  20. package/dist/core/artifact-chain/exporter.js +164 -0
  21. package/dist/core/artifact-chain/state.js +243 -0
  22. package/dist/core/artifact-chain/steps.js +169 -0
  23. package/dist/core/ask-user/question.js +92 -0
  24. package/dist/core/audit/audit-trail.js +275 -0
  25. package/dist/core/auth/ensure-authenticated.js +129 -0
  26. package/dist/core/auth/env-provider.js +238 -0
  27. package/dist/core/auto-open-browser.js +4 -4
  28. package/dist/core/auto-update/channels.js +122 -0
  29. package/dist/core/auto-update/checker.js +241 -0
  30. package/dist/core/auto-update/state.js +235 -0
  31. package/dist/core/bare-mode/index.js +107 -0
  32. package/dist/core/bash/redirect.js +281 -0
  33. package/dist/core/bash-classifier.js +436 -40
  34. package/dist/core/checkpoint/resumer.js +149 -0
  35. package/dist/core/checkpoint/rewinder.js +291 -0
  36. package/dist/core/checkpoints/shadow-git.js +670 -0
  37. package/dist/core/citations/parser.js +109 -0
  38. package/dist/core/classifier/yolo-classifier.js +88 -0
  39. package/dist/core/codegraph/db.js +506 -0
  40. package/dist/core/codegraph/decision-store.js +248 -0
  41. package/dist/core/codegraph/detect-repo.js +459 -0
  42. package/dist/core/codegraph/install.js +134 -0
  43. package/dist/core/codegraph/offer-hook.js +220 -0
  44. package/dist/core/codegraph/parser.js +598 -0
  45. package/dist/core/codegraph/queries/go.scm +57 -0
  46. package/dist/core/codegraph/queries/javascript.scm +56 -0
  47. package/dist/core/codegraph/queries/python.scm +55 -0
  48. package/dist/core/codegraph/queries/rust.scm +63 -0
  49. package/dist/core/codegraph/queries/typescript.scm +91 -0
  50. package/dist/core/codegraph/reindex.js +218 -0
  51. package/dist/core/codegraph/resolve-edges.js +107 -0
  52. package/dist/core/codegraph/types.js +34 -0
  53. package/dist/core/codegraph/watcher.js +440 -0
  54. package/dist/core/compact/auto-trigger.js +96 -0
  55. package/dist/core/compact/buffer-rewriter.js +115 -0
  56. package/dist/core/compact/summarizer.js +208 -0
  57. package/dist/core/compact/token-counter.js +108 -0
  58. package/dist/core/consensus/anvil-fanout.js +25 -25
  59. package/dist/core/consensus/diff-capture.js +121 -12
  60. package/dist/core/consensus/rubric.js +21 -21
  61. package/dist/core/context/builder.js +6 -6
  62. package/dist/core/context/compaction-events.js +8 -8
  63. package/dist/core/context/compaction.js +31 -31
  64. package/dist/core/context/index.js +15 -8
  65. package/dist/core/context/invariants.js +51 -51
  66. package/dist/core/context/markdown-loader.js +28 -10
  67. package/dist/core/context/markdown-traverse.js +255 -0
  68. package/dist/core/context/pugiignore.js +41 -41
  69. package/dist/core/context/repo-skeleton.js +37 -37
  70. package/dist/core/context/tool-eviction.js +55 -0
  71. package/dist/core/context/watcher.js +32 -32
  72. package/dist/core/context/working-set.js +23 -23
  73. package/dist/core/coordinator/agent-tools.js +77 -0
  74. package/dist/core/coordinator/agent-toolset.js +65 -0
  75. package/dist/core/coordinator/fsm.js +73 -0
  76. package/dist/core/coordinator/mode-fsm.js +70 -0
  77. package/dist/core/cost/rate-card.js +129 -0
  78. package/dist/core/cost/tracker.js +221 -0
  79. package/dist/core/credentials.js +13 -13
  80. package/dist/core/cron/scheduler.js +138 -0
  81. package/dist/core/denial-tracking/index.js +8 -0
  82. package/dist/core/denial-tracking/state.js +264 -0
  83. package/dist/core/diagnostics/probe-runner.js +93 -0
  84. package/dist/core/diagnostics/probes/api.js +46 -0
  85. package/dist/core/diagnostics/probes/auth.js +93 -0
  86. package/dist/core/diagnostics/probes/bare-mode.js +42 -0
  87. package/dist/core/diagnostics/probes/cli-version.js +127 -0
  88. package/dist/core/diagnostics/probes/config.js +72 -0
  89. package/dist/core/diagnostics/probes/denial-tracking.js +57 -0
  90. package/dist/core/diagnostics/probes/disk.js +81 -0
  91. package/dist/core/diagnostics/probes/engine-live.js +46 -0
  92. package/dist/core/diagnostics/probes/git.js +65 -0
  93. package/dist/core/diagnostics/probes/hooks.js +118 -0
  94. package/dist/core/diagnostics/probes/mcp.js +75 -0
  95. package/dist/core/diagnostics/probes/node.js +59 -0
  96. package/dist/core/diagnostics/probes/pnpm.js +36 -0
  97. package/dist/core/diagnostics/probes/pugi-md.js +89 -0
  98. package/dist/core/diagnostics/probes/sandbox.js +67 -0
  99. package/dist/core/diagnostics/probes/session.js +74 -0
  100. package/dist/core/diagnostics/probes/status-snapshot.js +488 -0
  101. package/dist/core/diagnostics/probes/workspace.js +63 -0
  102. package/dist/core/diagnostics/types.js +70 -0
  103. package/dist/core/dispatch/cache-cleanup.js +197 -0
  104. package/dist/core/dispatch/cache-handoff.js +295 -0
  105. package/dist/core/edits/apply-patch-layer-e.js +189 -0
  106. package/dist/core/edits/dispatch.js +333 -7
  107. package/dist/core/edits/format-detector.js +260 -0
  108. package/dist/core/edits/format-matrix.js +26 -0
  109. package/dist/core/edits/fuzzy-ladder.js +650 -0
  110. package/dist/core/edits/index.js +5 -1
  111. package/dist/core/edits/journal.js +199 -0
  112. package/dist/core/edits/layer-a-apply.js +15 -15
  113. package/dist/core/edits/layer-a-fuzzy-apply.js +198 -0
  114. package/dist/core/edits/layer-b-apply.js +9 -9
  115. package/dist/core/edits/layer-c-apply.js +6 -6
  116. package/dist/core/edits/layer-d-ast.js +557 -14
  117. package/dist/core/edits/marker-parser.js +12 -12
  118. package/dist/core/edits/security-gate.js +27 -27
  119. package/dist/core/edits/verify-hook.js +273 -0
  120. package/dist/core/edits/worktree.js +29 -29
  121. package/dist/core/engine/anvil-client.js +214 -26
  122. package/dist/core/engine/auto-compact.js +247 -0
  123. package/dist/core/engine/budgets.js +220 -0
  124. package/dist/core/engine/compact-llm-summarizer.js +124 -0
  125. package/dist/core/engine/context-prefix.js +155 -0
  126. package/dist/core/engine/index.js +1 -1
  127. package/dist/core/engine/intensity.js +163 -0
  128. package/dist/core/engine/intent.js +260 -0
  129. package/dist/core/engine/native-pugi.js +1559 -227
  130. package/dist/core/engine/prompts.js +219 -19
  131. package/dist/core/engine/strip-internal-fields.js +124 -0
  132. package/dist/core/engine/tool-bridge.js +1887 -59
  133. package/dist/core/engine/verification-patterns.js +195 -0
  134. package/dist/core/eval/v1/ledger.js +83 -0
  135. package/dist/core/eval/v1/runner.js +280 -0
  136. package/dist/core/eval/v1/scoring.js +68 -0
  137. package/dist/core/eval/v1/task-loader.js +191 -0
  138. package/dist/core/eval/v1/types.js +14 -0
  139. package/dist/core/eval/v1/verifier.js +176 -0
  140. package/dist/core/eval/v1/yaml-parser.js +250 -0
  141. package/dist/core/evaluation/golden-dataset.js +293 -0
  142. package/dist/core/feedback/queue.js +177 -0
  143. package/dist/core/feedback/submitter.js +145 -0
  144. package/dist/core/file-cache.js +113 -1
  145. package/dist/core/flatten/flatten-repo.js +439 -0
  146. package/dist/core/format/osc8-link.js +28 -0
  147. package/dist/core/hook-chains.js +392 -0
  148. package/dist/core/hooks/citation-verify-hook.js +138 -0
  149. package/dist/core/hooks/citation-verify.js +112 -0
  150. package/dist/core/hooks/events.js +46 -0
  151. package/dist/core/hooks/index.js +15 -0
  152. package/dist/core/hooks/registry.js +216 -0
  153. package/dist/core/hooks/runner.js +236 -0
  154. package/dist/core/hooks/v2/event-emitter.js +115 -0
  155. package/dist/core/hooks/v2/executor.js +282 -0
  156. package/dist/core/hooks/v2/index.js +25 -0
  157. package/dist/core/hooks/v2/lifecycle.js +104 -0
  158. package/dist/core/hooks/v2/loader.js +216 -0
  159. package/dist/core/hooks/v2/matcher.js +125 -0
  160. package/dist/core/hooks/v2/trust.js +143 -0
  161. package/dist/core/hooks/v2/types.js +86 -0
  162. package/dist/core/hooks/worktree-events.js +158 -0
  163. package/dist/core/image/renderer.js +71 -0
  164. package/dist/core/init/detector.js +582 -0
  165. package/dist/core/init/template-renderer.js +242 -0
  166. package/dist/core/jobs/registry.js +18 -18
  167. package/dist/core/ledger/results-tsv.js +142 -0
  168. package/dist/core/log-discipline/stdout-redirect.js +51 -0
  169. package/dist/core/lsp/cache.js +105 -0
  170. package/dist/core/lsp/client.js +551 -41
  171. package/dist/core/lsp/language-detect.js +66 -0
  172. package/dist/core/lsp/post-edit-diagnostics.js +171 -0
  173. package/dist/core/lsp/server-detect.js +173 -0
  174. package/dist/core/lsp/symbol-cache.js +162 -0
  175. package/dist/core/lsp/symbol-tools.js +664 -0
  176. package/dist/core/mcp/client.js +97 -28
  177. package/dist/core/mcp/http-server.js +553 -0
  178. package/dist/core/mcp/orchestrator-config.js +192 -0
  179. package/dist/core/mcp/orchestrator-tools.js +806 -0
  180. package/dist/core/mcp/permission.js +190 -0
  181. package/dist/core/mcp/registry.js +39 -17
  182. package/dist/core/mcp/server-tools.js +219 -0
  183. package/dist/core/mcp/server.js +397 -0
  184. package/dist/core/mcp/trust.js +10 -10
  185. package/dist/core/memory/dual-write.js +416 -0
  186. package/dist/core/memory/passive-extract.js +130 -0
  187. package/dist/core/memory/phase1-kinds.js +20 -0
  188. package/dist/core/memory/secret-scanner.js +304 -0
  189. package/dist/core/memory-sync/queue.js +170 -0
  190. package/dist/core/metrics/extract.js +113 -0
  191. package/dist/core/modes/roo-modes.js +68 -0
  192. package/dist/core/notes/notes-paths.js +113 -0
  193. package/dist/core/notes/notes-recorder.js +140 -0
  194. package/dist/core/notes/notes-writer.js +53 -0
  195. package/dist/core/notes/renderers.js +0 -0
  196. package/dist/core/notes/slug.js +105 -0
  197. package/dist/core/onboarding/ensure-initialized.js +133 -0
  198. package/dist/core/onboarding/marker.js +111 -0
  199. package/dist/core/onboarding/telemetry-state.js +108 -0
  200. package/dist/core/output-style/presets.js +176 -0
  201. package/dist/core/output-style/state.js +185 -0
  202. package/dist/core/path-security.js +287 -5
  203. package/dist/core/permission.js +82 -22
  204. package/dist/core/permissions/auto-classifier.js +124 -0
  205. package/dist/core/permissions/bash-parser.js +371 -0
  206. package/dist/core/permissions/circuit-breaker.js +83 -0
  207. package/dist/core/permissions/constrained-edit.js +91 -0
  208. package/dist/core/permissions/gate.js +278 -0
  209. package/dist/core/permissions/index.js +20 -0
  210. package/dist/core/permissions/mode.js +174 -0
  211. package/dist/core/permissions/network-egress.js +137 -0
  212. package/dist/core/permissions/state.js +241 -0
  213. package/dist/core/permissions/tool-class.js +107 -0
  214. package/dist/core/plan-mode/ui-state.js +51 -0
  215. package/dist/core/plans/plan-artifact.js +721 -0
  216. package/dist/core/policy-limits/etag-store.js +122 -0
  217. package/dist/core/prd-check/parser.js +215 -0
  218. package/dist/core/prd-check/reporter.js +127 -0
  219. package/dist/core/prd-check/session-review.js +557 -0
  220. package/dist/core/prd-check/verifiers.js +223 -0
  221. package/dist/core/prompt-cache/client-cache.js +99 -0
  222. package/dist/core/prompts/assembly.js +29 -0
  223. package/dist/core/prompts/registry.js +364 -0
  224. package/dist/core/pugi-gitignore.js +52 -0
  225. package/dist/core/pugi-md/cc-compat-rules.js +735 -0
  226. package/dist/core/pugi-md/context-injector.js +76 -0
  227. package/dist/core/pugi-md/walk-up.js +207 -0
  228. package/dist/core/python/uv-installer.js +270 -0
  229. package/dist/core/python/uv-resolver.js +83 -0
  230. package/dist/core/rate-limit/narrator.js +146 -0
  231. package/dist/core/recipes/cli-types.js +20 -0
  232. package/dist/core/recipes/loader.js +103 -0
  233. package/dist/core/recipes/runner.js +345 -0
  234. package/dist/core/recipes/schema.js +587 -0
  235. package/dist/core/release-notes/parser.js +241 -0
  236. package/dist/core/release-notes/state.js +116 -0
  237. package/dist/core/repl/ask.js +37 -37
  238. package/dist/core/repl/cancellation.js +26 -26
  239. package/dist/core/repl/cap-warning.js +4 -4
  240. package/dist/core/repl/clipboard-read.js +11 -11
  241. package/dist/core/repl/dispatch-fsm.js +12 -12
  242. package/dist/core/repl/engine-bridge.js +303 -0
  243. package/dist/core/repl/history-search.js +15 -15
  244. package/dist/core/repl/history.js +28 -18
  245. package/dist/core/repl/kill-ring.js +5 -5
  246. package/dist/core/repl/model-pricing.js +135 -0
  247. package/dist/core/repl/privacy-banner.js +22 -22
  248. package/dist/core/repl/session.js +2690 -229
  249. package/dist/core/repl/slash-commands.js +540 -41
  250. package/dist/core/repl/store/index.js +1 -1
  251. package/dist/core/repl/store/jsonl-log.js +22 -22
  252. package/dist/core/repl/store/lockfile.js +10 -10
  253. package/dist/core/repl/store/session-store.js +136 -107
  254. package/dist/core/repl/store/types.js +15 -15
  255. package/dist/core/repl/store/uuid-v7.js +12 -12
  256. package/dist/core/repl/tool-route.js +382 -0
  257. package/dist/core/repl/workspace-context.js +43 -21
  258. package/dist/core/repo-map/build.js +125 -0
  259. package/dist/core/repo-map/cache.js +185 -0
  260. package/dist/core/repo-map/extractor.js +254 -0
  261. package/dist/core/repo-map/formatter.js +145 -0
  262. package/dist/core/repo-map/page-rank.js +105 -0
  263. package/dist/core/repo-map/scanner.js +211 -0
  264. package/dist/core/retro/git-collector.js +251 -0
  265. package/dist/core/retro/health-card.js +25 -0
  266. package/dist/core/retro/metrics.js +342 -0
  267. package/dist/core/retro/narrative.js +249 -0
  268. package/dist/core/retro/plane-collector.js +274 -0
  269. package/dist/core/retro/pr-issue-link.js +65 -0
  270. package/dist/core/retro/types.js +16 -0
  271. package/dist/core/retry-budget/budget.js +284 -0
  272. package/dist/core/retry-budget/index.js +5 -0
  273. package/dist/core/retry-budget/retry-cap.js +74 -0
  274. package/dist/core/routing/lead-worker.js +43 -0
  275. package/dist/core/routing/pre-flight-estimator.js +108 -0
  276. package/dist/core/runs/run-tree.js +103 -0
  277. package/dist/core/sandboxing/adapter.js +43 -0
  278. package/dist/core/sandboxing/bubblewrap.js +209 -0
  279. package/dist/core/sandboxing/index.js +78 -0
  280. package/dist/core/sandboxing/none.js +19 -0
  281. package/dist/core/sandboxing/policy.js +97 -0
  282. package/dist/core/sandboxing/seatbelt.js +231 -0
  283. package/dist/core/security/injection-scanner.js +367 -0
  284. package/dist/core/security/output-filter.js +418 -0
  285. package/dist/core/session/env-file.js +105 -0
  286. package/dist/core/session/section-budgets.js +140 -0
  287. package/dist/core/session.js +119 -0
  288. package/dist/core/settings.js +402 -5
  289. package/dist/core/share/formatter.js +271 -0
  290. package/dist/core/share/redactor.js +221 -0
  291. package/dist/core/share/uploader.js +267 -0
  292. package/dist/core/skills/defaults.js +30 -30
  293. package/dist/core/skills/loader.js +22 -22
  294. package/dist/core/skills/sources.js +27 -27
  295. package/dist/core/smoke/headless-driver.js +174 -0
  296. package/dist/core/smoke/orchestrator.js +194 -0
  297. package/dist/core/smoke/runner.js +238 -0
  298. package/dist/core/smoke/scenario-parser.js +316 -0
  299. package/dist/core/statusline.js +99 -0
  300. package/dist/core/subagents/dispatcher-real.js +600 -0
  301. package/dist/core/subagents/dispatcher.js +146 -52
  302. package/dist/core/subagents/index.js +19 -6
  303. package/dist/core/subagents/isolation-matrix.js +213 -0
  304. package/dist/core/subagents/spawn.js +19 -4
  305. package/dist/core/telemetry/emitter.js +229 -0
  306. package/dist/core/telemetry/queue.js +251 -0
  307. package/dist/core/theme/context.js +91 -0
  308. package/dist/core/theme/presets.js +228 -0
  309. package/dist/core/theme/state.js +181 -0
  310. package/dist/core/todos/invariant.js +10 -0
  311. package/dist/core/todos/state.js +177 -0
  312. package/dist/core/tool-schema/compressor.js +89 -0
  313. package/dist/core/transport/version-interceptor.js +166 -0
  314. package/dist/core/trust.js +2 -2
  315. package/dist/core/tui/thinking-block.js +64 -0
  316. package/dist/core/vim/keymap.js +288 -0
  317. package/dist/core/vim/state.js +92 -0
  318. package/dist/core/watch-markers/marker-watcher.js +133 -0
  319. package/dist/core/worktree/include-parser.js +249 -0
  320. package/dist/core/worktree-manager/cleanup.js +123 -0
  321. package/dist/core/worktree-manager/manager.js +303 -0
  322. package/dist/index.js +36 -0
  323. package/dist/runtime/bootstrap.js +190 -0
  324. package/dist/runtime/cli.js +4403 -561
  325. package/dist/runtime/commands/agents.js +31 -31
  326. package/dist/runtime/commands/budget.js +5 -5
  327. package/dist/runtime/commands/cancel.js +231 -0
  328. package/dist/runtime/commands/chain.js +489 -0
  329. package/dist/runtime/commands/codegraph-status.js +227 -0
  330. package/dist/runtime/commands/compact.js +297 -0
  331. package/dist/runtime/commands/config.js +74 -40
  332. package/dist/runtime/commands/cost.js +199 -0
  333. package/dist/runtime/commands/delegate.js +27 -4
  334. package/dist/runtime/commands/dispatch.js +126 -0
  335. package/dist/runtime/commands/doctor.js +579 -0
  336. package/dist/runtime/commands/eval-v1.js +266 -0
  337. package/dist/runtime/commands/feedback.js +184 -0
  338. package/dist/runtime/commands/hooks.js +187 -0
  339. package/dist/runtime/commands/index-cmd.js +459 -0
  340. package/dist/runtime/commands/init.js +254 -0
  341. package/dist/runtime/commands/lsp.js +200 -38
  342. package/dist/runtime/commands/mcp.js +935 -0
  343. package/dist/runtime/commands/memory.js +582 -0
  344. package/dist/runtime/commands/model.js +237 -0
  345. package/dist/runtime/commands/onboarding.js +275 -0
  346. package/dist/runtime/commands/patch.js +12 -12
  347. package/dist/runtime/commands/permissions.js +112 -0
  348. package/dist/runtime/commands/plan.js +143 -0
  349. package/dist/runtime/commands/prd-check.js +285 -0
  350. package/dist/runtime/commands/privacy.js +17 -17
  351. package/dist/runtime/commands/recipe.js +325 -0
  352. package/dist/runtime/commands/redo-blob-store.js +92 -0
  353. package/dist/runtime/commands/redo.js +361 -0
  354. package/dist/runtime/commands/release-notes.js +229 -0
  355. package/dist/runtime/commands/repo-map.js +95 -0
  356. package/dist/runtime/commands/report.js +299 -0
  357. package/dist/runtime/commands/resume.js +118 -0
  358. package/dist/runtime/commands/review-consensus.js +68 -53
  359. package/dist/runtime/commands/rewind.js +333 -0
  360. package/dist/runtime/commands/roster.js +14 -14
  361. package/dist/runtime/commands/servers-cli.js +182 -0
  362. package/dist/runtime/commands/servers.js +236 -0
  363. package/dist/runtime/commands/sessions.js +163 -0
  364. package/dist/runtime/commands/share.js +316 -0
  365. package/dist/runtime/commands/skills.js +31 -31
  366. package/dist/runtime/commands/status.js +186 -0
  367. package/dist/runtime/commands/stickers.js +82 -0
  368. package/dist/runtime/commands/style.js +194 -0
  369. package/dist/runtime/commands/theme.js +196 -0
  370. package/dist/runtime/commands/undo.js +54 -22
  371. package/dist/runtime/commands/update.js +289 -0
  372. package/dist/runtime/commands/vim.js +140 -0
  373. package/dist/runtime/commands/worktree.js +8 -8
  374. package/dist/runtime/commands/worktrees.js +155 -0
  375. package/dist/runtime/deprecation-warning.js +69 -0
  376. package/dist/runtime/engine-exit-code.js +50 -0
  377. package/dist/runtime/headless-repl.js +195 -0
  378. package/dist/runtime/headless.js +548 -0
  379. package/dist/runtime/load-hooks-or-exit.js +71 -0
  380. package/dist/runtime/plan-decompose.js +22 -22
  381. package/dist/runtime/sigint-guard.js +272 -0
  382. package/dist/runtime/stream-renderer.js +195 -0
  383. package/dist/runtime/update-check.js +28 -28
  384. package/dist/runtime/version.js +65 -0
  385. package/dist/runtime/worktree-bootstrap.js +579 -0
  386. package/dist/skills/bundled/batch.js +617 -0
  387. package/dist/skills/bundled/index.js +45 -0
  388. package/dist/skills/bundled/loop.js +358 -0
  389. package/dist/skills/bundled/remember.js +383 -0
  390. package/dist/skills/bundled/simplify.js +289 -0
  391. package/dist/skills/bundled/skillify.js +373 -0
  392. package/dist/skills/bundled/stuck.js +558 -0
  393. package/dist/skills/bundled/verify.js +439 -0
  394. package/dist/testing/vcr.js +486 -0
  395. package/dist/tools/agent-tool.js +229 -0
  396. package/dist/tools/apply-patch.js +89 -28
  397. package/dist/tools/ask-user-question.js +337 -0
  398. package/dist/tools/ask-user.js +115 -0
  399. package/dist/tools/bash.js +811 -49
  400. package/dist/tools/brief.js +224 -0
  401. package/dist/tools/cron.js +433 -0
  402. package/dist/tools/enter-worktree.js +250 -0
  403. package/dist/tools/exit-worktree.js +147 -0
  404. package/dist/tools/file-tools.js +161 -44
  405. package/dist/tools/http-request.js +336 -0
  406. package/dist/tools/lsp-tools.js +377 -1
  407. package/dist/tools/mcp-tool.js +260 -0
  408. package/dist/tools/multi-edit.js +361 -0
  409. package/dist/tools/powershell.js +268 -0
  410. package/dist/tools/registry.js +120 -5
  411. package/dist/tools/server-tools.js +892 -0
  412. package/dist/tools/skill-tool.js +96 -0
  413. package/dist/tools/sleep.js +99 -0
  414. package/dist/tools/synthetic-output.js +133 -0
  415. package/dist/tools/tasks.js +208 -0
  416. package/dist/tools/todo-write.js +184 -0
  417. package/dist/tools/verify-plan-execution.js +295 -0
  418. package/dist/tools/web-fetch-injection-scanner.js +207 -0
  419. package/dist/tools/web-fetch.js +195 -10
  420. package/dist/tools/web-search.js +458 -0
  421. package/dist/tui/agent-progress-card.js +111 -0
  422. package/dist/tui/agent-tree.js +22 -1
  423. package/dist/tui/ask-modal.js +14 -14
  424. package/dist/tui/ask-user-question-chips.js +315 -0
  425. package/dist/tui/ask-user-question-prompt.js +203 -0
  426. package/dist/tui/compact-banner.js +81 -0
  427. package/dist/tui/conversation-pane.js +85 -11
  428. package/dist/tui/cost-table.js +111 -0
  429. package/dist/tui/device-flow.js +2 -2
  430. package/dist/tui/doctor-table.js +46 -0
  431. package/dist/tui/feedback-prompt.js +156 -0
  432. package/dist/tui/input-box.js +247 -32
  433. package/dist/tui/login-picker.js +3 -3
  434. package/dist/tui/markdown-render.js +6 -6
  435. package/dist/tui/multi-file-diff-approval.js +375 -0
  436. package/dist/tui/onboarding-wizard.js +240 -0
  437. package/dist/tui/permissions-picker.js +86 -0
  438. package/dist/tui/render.js +36 -1
  439. package/dist/tui/repl-render.js +239 -25
  440. package/dist/tui/repl-splash-art.js +16 -16
  441. package/dist/tui/repl-splash-mascot.js +48 -24
  442. package/dist/tui/repl-splash.js +22 -22
  443. package/dist/tui/repl.js +125 -45
  444. package/dist/tui/slash-palette.js +6 -6
  445. package/dist/tui/splash.js +2 -2
  446. package/dist/tui/status-bar.js +109 -31
  447. package/dist/tui/status-table.js +7 -0
  448. package/dist/tui/stickers-art.js +136 -0
  449. package/dist/tui/style-table.js +28 -0
  450. package/dist/tui/theme-table.js +29 -0
  451. package/dist/tui/thinking-spinner.js +123 -0
  452. package/dist/tui/tool-stream-pane.js +53 -4
  453. package/dist/tui/update-banner.js +27 -2
  454. package/dist/tui/vim-input.js +267 -0
  455. package/dist/tui/welcome-banner.js +107 -0
  456. package/dist/tui/welcome-data.js +293 -0
  457. package/dist/tui/workspace-context.js +2 -2
  458. package/package.json +29 -6
  459. package/test/scenarios/codegen-create-file.scenario.txt +13 -0
  460. package/test/scenarios/compact-force.scenario.txt +12 -0
  461. package/test/scenarios/identity.scenario.txt +11 -0
  462. package/test/scenarios/persona-handoff.scenario.txt +12 -0
  463. package/test/scenarios/walkback.scenario.txt +12 -0
  464. 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