@pugi/cli 0.1.0-beta.7 → 0.1.0-beta.87

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (402) hide show
  1. package/CHANGELOG.md +96 -0
  2. package/THIRD_PARTY_NOTICES.md +40 -0
  3. package/assets/pugi-prozr2-mascot.ansi +9 -0
  4. package/bin/run.js +33 -1
  5. package/dist/commands/deploy.js +40 -40
  6. package/dist/commands/flatten.js +191 -0
  7. package/dist/commands/jobs-watch.js +201 -0
  8. package/dist/commands/jobs.js +42 -27
  9. package/dist/commands/smoke.js +133 -0
  10. package/dist/core/agent-progress/cleanup.js +134 -0
  11. package/dist/core/agent-progress/schema.js +144 -0
  12. package/dist/core/agent-progress/writer.js +101 -0
  13. package/dist/core/agents/adaptive-router.js +330 -0
  14. package/dist/core/agents/query-decomposer.js +297 -0
  15. package/dist/core/agents/registry.js +2 -2
  16. package/dist/core/approvals/shortcut-resolver.js +98 -0
  17. package/dist/core/artifact-chain/dispatcher.js +148 -0
  18. package/dist/core/artifact-chain/exporter.js +164 -0
  19. package/dist/core/artifact-chain/state.js +243 -0
  20. package/dist/core/artifact-chain/steps.js +169 -0
  21. package/dist/core/ask-user/question.js +92 -0
  22. package/dist/core/audit/audit-trail.js +275 -0
  23. package/dist/core/auth/ensure-authenticated.js +129 -0
  24. package/dist/core/auth/env-provider.js +238 -0
  25. package/dist/core/auto-open-browser.js +4 -4
  26. package/dist/core/auto-update/channels.js +122 -0
  27. package/dist/core/auto-update/checker.js +241 -0
  28. package/dist/core/auto-update/state.js +235 -0
  29. package/dist/core/bare-mode/index.js +107 -0
  30. package/dist/core/bash/redirect.js +281 -0
  31. package/dist/core/bash-classifier.js +436 -40
  32. package/dist/core/checkpoint/resumer.js +149 -0
  33. package/dist/core/checkpoint/rewinder.js +291 -0
  34. package/dist/core/checkpoints/shadow-git.js +670 -0
  35. package/dist/core/citations/parser.js +109 -0
  36. package/dist/core/classifier/yolo-classifier.js +88 -0
  37. package/dist/core/codegraph/decision-store.js +248 -0
  38. package/dist/core/codegraph/detect-repo.js +459 -0
  39. package/dist/core/codegraph/install.js +134 -0
  40. package/dist/core/codegraph/offer-hook.js +220 -0
  41. package/dist/core/compact/auto-trigger.js +96 -0
  42. package/dist/core/compact/buffer-rewriter.js +115 -0
  43. package/dist/core/compact/summarizer.js +208 -0
  44. package/dist/core/compact/token-counter.js +108 -0
  45. package/dist/core/consensus/anvil-fanout.js +25 -25
  46. package/dist/core/consensus/diff-capture.js +121 -12
  47. package/dist/core/consensus/rubric.js +21 -21
  48. package/dist/core/context/builder.js +6 -6
  49. package/dist/core/context/compaction-events.js +8 -8
  50. package/dist/core/context/compaction.js +31 -31
  51. package/dist/core/context/index.js +15 -8
  52. package/dist/core/context/invariants.js +51 -51
  53. package/dist/core/context/markdown-loader.js +28 -10
  54. package/dist/core/context/markdown-traverse.js +255 -0
  55. package/dist/core/context/pugiignore.js +41 -41
  56. package/dist/core/context/repo-skeleton.js +37 -37
  57. package/dist/core/context/tool-eviction.js +55 -0
  58. package/dist/core/context/watcher.js +32 -32
  59. package/dist/core/context/working-set.js +23 -23
  60. package/dist/core/coordinator/agent-tools.js +77 -0
  61. package/dist/core/coordinator/agent-toolset.js +65 -0
  62. package/dist/core/coordinator/fsm.js +73 -0
  63. package/dist/core/coordinator/mode-fsm.js +70 -0
  64. package/dist/core/cost/rate-card.js +129 -0
  65. package/dist/core/cost/tracker.js +221 -0
  66. package/dist/core/credentials.js +12 -12
  67. package/dist/core/cron/scheduler.js +138 -0
  68. package/dist/core/denial-tracking/index.js +8 -0
  69. package/dist/core/denial-tracking/state.js +264 -0
  70. package/dist/core/diagnostics/probe-runner.js +93 -0
  71. package/dist/core/diagnostics/probes/api.js +46 -0
  72. package/dist/core/diagnostics/probes/auth.js +93 -0
  73. package/dist/core/diagnostics/probes/bare-mode.js +42 -0
  74. package/dist/core/diagnostics/probes/cli-version.js +127 -0
  75. package/dist/core/diagnostics/probes/config.js +72 -0
  76. package/dist/core/diagnostics/probes/denial-tracking.js +57 -0
  77. package/dist/core/diagnostics/probes/disk.js +81 -0
  78. package/dist/core/diagnostics/probes/engine-live.js +46 -0
  79. package/dist/core/diagnostics/probes/git.js +65 -0
  80. package/dist/core/diagnostics/probes/hooks.js +118 -0
  81. package/dist/core/diagnostics/probes/mcp.js +75 -0
  82. package/dist/core/diagnostics/probes/node.js +59 -0
  83. package/dist/core/diagnostics/probes/pnpm.js +36 -0
  84. package/dist/core/diagnostics/probes/pugi-md.js +89 -0
  85. package/dist/core/diagnostics/probes/sandbox.js +40 -0
  86. package/dist/core/diagnostics/probes/session.js +74 -0
  87. package/dist/core/diagnostics/probes/status-snapshot.js +488 -0
  88. package/dist/core/diagnostics/probes/workspace.js +63 -0
  89. package/dist/core/diagnostics/types.js +70 -0
  90. package/dist/core/dispatch/cache-cleanup.js +197 -0
  91. package/dist/core/dispatch/cache-handoff.js +295 -0
  92. package/dist/core/edits/apply-patch-layer-e.js +189 -0
  93. package/dist/core/edits/dispatch.js +293 -7
  94. package/dist/core/edits/format-matrix.js +26 -0
  95. package/dist/core/edits/fuzzy-ladder.js +650 -0
  96. package/dist/core/edits/index.js +3 -1
  97. package/dist/core/edits/journal.js +199 -0
  98. package/dist/core/edits/layer-a-apply.js +15 -15
  99. package/dist/core/edits/layer-a-fuzzy-apply.js +198 -0
  100. package/dist/core/edits/layer-b-apply.js +9 -9
  101. package/dist/core/edits/layer-c-apply.js +6 -6
  102. package/dist/core/edits/layer-d-ast.js +557 -14
  103. package/dist/core/edits/marker-parser.js +12 -12
  104. package/dist/core/edits/security-gate.js +27 -27
  105. package/dist/core/edits/verify-hook.js +273 -0
  106. package/dist/core/edits/worktree.js +322 -0
  107. package/dist/core/engine/anvil-client.js +140 -26
  108. package/dist/core/engine/auto-compact.js +179 -0
  109. package/dist/core/engine/budgets.js +186 -0
  110. package/dist/core/engine/context-prefix.js +155 -0
  111. package/dist/core/engine/index.js +1 -1
  112. package/dist/core/engine/intensity.js +158 -0
  113. package/dist/core/engine/intent.js +260 -0
  114. package/dist/core/engine/native-pugi.js +1295 -227
  115. package/dist/core/engine/prompts.js +134 -16
  116. package/dist/core/engine/strip-internal-fields.js +124 -0
  117. package/dist/core/engine/tool-bridge.js +1295 -59
  118. package/dist/core/evaluation/golden-dataset.js +293 -0
  119. package/dist/core/feedback/queue.js +177 -0
  120. package/dist/core/feedback/submitter.js +145 -0
  121. package/dist/core/file-cache.js +113 -1
  122. package/dist/core/flatten/flatten-repo.js +439 -0
  123. package/dist/core/format/osc8-link.js +28 -0
  124. package/dist/core/hook-chains.js +392 -0
  125. package/dist/core/hooks/citation-verify-hook.js +138 -0
  126. package/dist/core/hooks/citation-verify.js +112 -0
  127. package/dist/core/hooks/events.js +44 -0
  128. package/dist/core/hooks/index.js +15 -0
  129. package/dist/core/hooks/registry.js +213 -0
  130. package/dist/core/hooks/runner.js +236 -0
  131. package/dist/core/hooks/v2/event-emitter.js +115 -0
  132. package/dist/core/hooks/v2/executor.js +282 -0
  133. package/dist/core/hooks/v2/index.js +25 -0
  134. package/dist/core/hooks/v2/lifecycle.js +104 -0
  135. package/dist/core/hooks/v2/loader.js +216 -0
  136. package/dist/core/hooks/v2/matcher.js +125 -0
  137. package/dist/core/hooks/v2/trust.js +143 -0
  138. package/dist/core/hooks/v2/types.js +86 -0
  139. package/dist/core/image/renderer.js +71 -0
  140. package/dist/core/init/detector.js +582 -0
  141. package/dist/core/init/template-renderer.js +242 -0
  142. package/dist/core/jobs/registry.js +18 -18
  143. package/dist/core/ledger/results-tsv.js +142 -0
  144. package/dist/core/log-discipline/stdout-redirect.js +51 -0
  145. package/dist/core/lsp/cache.js +105 -0
  146. package/dist/core/lsp/client.js +776 -0
  147. package/dist/core/lsp/language-detect.js +66 -0
  148. package/dist/core/lsp/post-edit-diagnostics.js +171 -0
  149. package/dist/core/lsp/symbol-tools.js +372 -0
  150. package/dist/core/mcp/client.js +97 -28
  151. package/dist/core/mcp/http-server.js +553 -0
  152. package/dist/core/mcp/orchestrator-tools.js +662 -0
  153. package/dist/core/mcp/permission.js +190 -0
  154. package/dist/core/mcp/registry.js +39 -17
  155. package/dist/core/mcp/server-tools.js +219 -0
  156. package/dist/core/mcp/server.js +397 -0
  157. package/dist/core/mcp/trust.js +10 -10
  158. package/dist/core/memory/dual-write.js +416 -0
  159. package/dist/core/memory/passive-extract.js +130 -0
  160. package/dist/core/memory/phase1-kinds.js +20 -0
  161. package/dist/core/memory/secret-scanner.js +304 -0
  162. package/dist/core/memory-sync/queue.js +170 -0
  163. package/dist/core/metrics/extract.js +113 -0
  164. package/dist/core/modes/roo-modes.js +68 -0
  165. package/dist/core/onboarding/ensure-initialized.js +133 -0
  166. package/dist/core/onboarding/marker.js +111 -0
  167. package/dist/core/onboarding/telemetry-state.js +108 -0
  168. package/dist/core/output-style/presets.js +176 -0
  169. package/dist/core/output-style/state.js +185 -0
  170. package/dist/core/path-security.js +287 -5
  171. package/dist/core/permission.js +82 -22
  172. package/dist/core/permissions/auto-classifier.js +124 -0
  173. package/dist/core/permissions/bash-parser.js +371 -0
  174. package/dist/core/permissions/circuit-breaker.js +83 -0
  175. package/dist/core/permissions/constrained-edit.js +91 -0
  176. package/dist/core/permissions/gate.js +278 -0
  177. package/dist/core/permissions/index.js +20 -0
  178. package/dist/core/permissions/mode.js +174 -0
  179. package/dist/core/permissions/network-egress.js +137 -0
  180. package/dist/core/permissions/state.js +241 -0
  181. package/dist/core/permissions/tool-class.js +93 -0
  182. package/dist/core/plan-mode/ui-state.js +51 -0
  183. package/dist/core/plans/plan-artifact.js +721 -0
  184. package/dist/core/policy-limits/etag-store.js +122 -0
  185. package/dist/core/prd-check/parser.js +215 -0
  186. package/dist/core/prd-check/reporter.js +127 -0
  187. package/dist/core/prd-check/session-review.js +557 -0
  188. package/dist/core/prd-check/verifiers.js +223 -0
  189. package/dist/core/prompt-cache/client-cache.js +99 -0
  190. package/dist/core/prompts/assembly.js +29 -0
  191. package/dist/core/prompts/registry.js +364 -0
  192. package/dist/core/pugi-md/cc-compat-rules.js +735 -0
  193. package/dist/core/pugi-md/context-injector.js +76 -0
  194. package/dist/core/pugi-md/walk-up.js +207 -0
  195. package/dist/core/python/uv-installer.js +270 -0
  196. package/dist/core/python/uv-resolver.js +83 -0
  197. package/dist/core/rate-limit/narrator.js +146 -0
  198. package/dist/core/recipes/cli-types.js +20 -0
  199. package/dist/core/recipes/loader.js +103 -0
  200. package/dist/core/recipes/runner.js +345 -0
  201. package/dist/core/recipes/schema.js +587 -0
  202. package/dist/core/release-notes/parser.js +241 -0
  203. package/dist/core/release-notes/state.js +116 -0
  204. package/dist/core/repl/ask.js +37 -37
  205. package/dist/core/repl/cancellation.js +26 -26
  206. package/dist/core/repl/cap-warning.js +4 -4
  207. package/dist/core/repl/clipboard-read.js +11 -11
  208. package/dist/core/repl/dispatch-fsm.js +12 -12
  209. package/dist/core/repl/history-search.js +15 -15
  210. package/dist/core/repl/history.js +28 -18
  211. package/dist/core/repl/kill-ring.js +5 -5
  212. package/dist/core/repl/model-pricing.js +135 -0
  213. package/dist/core/repl/privacy-banner.js +22 -22
  214. package/dist/core/repl/session.js +2157 -214
  215. package/dist/core/repl/slash-commands.js +533 -40
  216. package/dist/core/repl/store/index.js +1 -1
  217. package/dist/core/repl/store/jsonl-log.js +22 -22
  218. package/dist/core/repl/store/lockfile.js +10 -10
  219. package/dist/core/repl/store/session-store.js +136 -107
  220. package/dist/core/repl/store/types.js +15 -15
  221. package/dist/core/repl/store/uuid-v7.js +12 -12
  222. package/dist/core/repl/workspace-context.js +43 -21
  223. package/dist/core/repo-map/build.js +125 -0
  224. package/dist/core/repo-map/cache.js +185 -0
  225. package/dist/core/repo-map/extractor.js +254 -0
  226. package/dist/core/repo-map/formatter.js +145 -0
  227. package/dist/core/repo-map/page-rank.js +105 -0
  228. package/dist/core/repo-map/scanner.js +211 -0
  229. package/dist/core/retry-budget/budget.js +284 -0
  230. package/dist/core/retry-budget/index.js +5 -0
  231. package/dist/core/retry-budget/retry-cap.js +74 -0
  232. package/dist/core/routing/lead-worker.js +43 -0
  233. package/dist/core/routing/pre-flight-estimator.js +108 -0
  234. package/dist/core/runs/run-tree.js +103 -0
  235. package/dist/core/security/injection-scanner.js +367 -0
  236. package/dist/core/security/output-filter.js +418 -0
  237. package/dist/core/session/env-file.js +105 -0
  238. package/dist/core/session/section-budgets.js +140 -0
  239. package/dist/core/session.js +92 -0
  240. package/dist/core/settings.js +286 -5
  241. package/dist/core/share/formatter.js +271 -0
  242. package/dist/core/share/redactor.js +221 -0
  243. package/dist/core/share/uploader.js +267 -0
  244. package/dist/core/skills/defaults.js +457 -0
  245. package/dist/core/skills/loader.js +22 -22
  246. package/dist/core/skills/sources.js +27 -27
  247. package/dist/core/smoke/headless-driver.js +174 -0
  248. package/dist/core/smoke/orchestrator.js +194 -0
  249. package/dist/core/smoke/runner.js +238 -0
  250. package/dist/core/smoke/scenario-parser.js +316 -0
  251. package/dist/core/statusline.js +99 -0
  252. package/dist/core/subagents/dispatcher-real.js +600 -0
  253. package/dist/core/subagents/dispatcher.js +132 -43
  254. package/dist/core/subagents/index.js +19 -6
  255. package/dist/core/subagents/isolation-matrix.js +213 -0
  256. package/dist/core/subagents/spawn.js +19 -4
  257. package/dist/core/telemetry/emitter.js +229 -0
  258. package/dist/core/telemetry/queue.js +251 -0
  259. package/dist/core/theme/context.js +91 -0
  260. package/dist/core/theme/presets.js +228 -0
  261. package/dist/core/theme/state.js +181 -0
  262. package/dist/core/todos/invariant.js +10 -0
  263. package/dist/core/todos/state.js +177 -0
  264. package/dist/core/tool-schema/compressor.js +89 -0
  265. package/dist/core/transport/version-interceptor.js +166 -0
  266. package/dist/core/trust.js +2 -2
  267. package/dist/core/tui/thinking-block.js +64 -0
  268. package/dist/core/vim/keymap.js +288 -0
  269. package/dist/core/vim/state.js +92 -0
  270. package/dist/core/watch-markers/marker-watcher.js +133 -0
  271. package/dist/core/worktree-manager/cleanup.js +123 -0
  272. package/dist/core/worktree-manager/manager.js +303 -0
  273. package/dist/index.js +28 -0
  274. package/dist/runtime/bootstrap.js +190 -0
  275. package/dist/runtime/cli.js +4162 -488
  276. package/dist/runtime/commands/agents.js +30 -30
  277. package/dist/runtime/commands/budget.js +5 -5
  278. package/dist/runtime/commands/cancel.js +231 -0
  279. package/dist/runtime/commands/chain.js +489 -0
  280. package/dist/runtime/commands/codegraph-status.js +227 -0
  281. package/dist/runtime/commands/compact.js +297 -0
  282. package/dist/runtime/commands/config.js +32 -32
  283. package/dist/runtime/commands/cost.js +199 -0
  284. package/dist/runtime/commands/delegate.js +244 -13
  285. package/dist/runtime/commands/dispatch.js +126 -0
  286. package/dist/runtime/commands/doctor.js +579 -0
  287. package/dist/runtime/commands/feedback.js +184 -0
  288. package/dist/runtime/commands/hooks.js +184 -0
  289. package/dist/runtime/commands/init.js +254 -0
  290. package/dist/runtime/commands/lsp.js +368 -0
  291. package/dist/runtime/commands/mcp.js +879 -0
  292. package/dist/runtime/commands/memory.js +582 -0
  293. package/dist/runtime/commands/model.js +237 -0
  294. package/dist/runtime/commands/onboarding.js +275 -0
  295. package/dist/runtime/commands/patch.js +128 -0
  296. package/dist/runtime/commands/permissions.js +112 -0
  297. package/dist/runtime/commands/plan.js +143 -0
  298. package/dist/runtime/commands/prd-check.js +285 -0
  299. package/dist/runtime/commands/privacy.js +17 -17
  300. package/dist/runtime/commands/recipe.js +325 -0
  301. package/dist/runtime/commands/redo-blob-store.js +92 -0
  302. package/dist/runtime/commands/redo.js +361 -0
  303. package/dist/runtime/commands/release-notes.js +229 -0
  304. package/dist/runtime/commands/repo-map.js +95 -0
  305. package/dist/runtime/commands/report.js +299 -0
  306. package/dist/runtime/commands/resume.js +118 -0
  307. package/dist/runtime/commands/review-consensus.js +68 -53
  308. package/dist/runtime/commands/rewind.js +333 -0
  309. package/dist/runtime/commands/roster.js +14 -14
  310. package/dist/runtime/commands/sessions.js +163 -0
  311. package/dist/runtime/commands/share.js +316 -0
  312. package/dist/runtime/commands/skills.js +31 -31
  313. package/dist/runtime/commands/status.js +186 -0
  314. package/dist/runtime/commands/stickers.js +82 -0
  315. package/dist/runtime/commands/style.js +194 -0
  316. package/dist/runtime/commands/theme.js +196 -0
  317. package/dist/runtime/commands/undo.js +54 -22
  318. package/dist/runtime/commands/update.js +289 -0
  319. package/dist/runtime/commands/vim.js +140 -0
  320. package/dist/runtime/commands/worktree.js +177 -0
  321. package/dist/runtime/commands/worktrees.js +155 -0
  322. package/dist/runtime/headless-repl.js +195 -0
  323. package/dist/runtime/headless.js +543 -0
  324. package/dist/runtime/load-hooks-or-exit.js +71 -0
  325. package/dist/runtime/plan-decompose.js +531 -0
  326. package/dist/runtime/update-check.js +28 -28
  327. package/dist/runtime/version.js +65 -0
  328. package/dist/skills/bundled/batch.js +617 -0
  329. package/dist/skills/bundled/index.js +45 -0
  330. package/dist/skills/bundled/loop.js +358 -0
  331. package/dist/skills/bundled/remember.js +383 -0
  332. package/dist/skills/bundled/simplify.js +289 -0
  333. package/dist/skills/bundled/skillify.js +373 -0
  334. package/dist/skills/bundled/stuck.js +558 -0
  335. package/dist/skills/bundled/verify.js +439 -0
  336. package/dist/testing/vcr.js +486 -0
  337. package/dist/tools/agent-tool.js +229 -0
  338. package/dist/tools/apply-patch.js +556 -0
  339. package/dist/tools/ask-user-question.js +222 -0
  340. package/dist/tools/ask-user.js +115 -0
  341. package/dist/tools/bash.js +623 -45
  342. package/dist/tools/brief.js +224 -0
  343. package/dist/tools/enter-worktree.js +250 -0
  344. package/dist/tools/exit-worktree.js +147 -0
  345. package/dist/tools/file-tools.js +161 -44
  346. package/dist/tools/lsp-tools.js +189 -0
  347. package/dist/tools/mcp-tool.js +260 -0
  348. package/dist/tools/multi-edit.js +361 -0
  349. package/dist/tools/powershell.js +268 -0
  350. package/dist/tools/registry.js +85 -0
  351. package/dist/tools/skill-tool.js +96 -0
  352. package/dist/tools/sleep.js +99 -0
  353. package/dist/tools/synthetic-output.js +133 -0
  354. package/dist/tools/tasks.js +208 -0
  355. package/dist/tools/todo-write.js +184 -0
  356. package/dist/tools/verify-plan-execution.js +295 -0
  357. package/dist/tools/web-fetch-injection-scanner.js +207 -0
  358. package/dist/tools/web-fetch.js +195 -10
  359. package/dist/tools/web-search.js +458 -0
  360. package/dist/tui/agent-progress-card.js +111 -0
  361. package/dist/tui/agent-tree.js +11 -1
  362. package/dist/tui/ask-modal.js +14 -14
  363. package/dist/tui/ask-user-question-prompt.js +203 -0
  364. package/dist/tui/compact-banner.js +81 -0
  365. package/dist/tui/conversation-pane.js +85 -11
  366. package/dist/tui/cost-table.js +111 -0
  367. package/dist/tui/device-flow.js +2 -2
  368. package/dist/tui/doctor-table.js +46 -0
  369. package/dist/tui/feedback-prompt.js +156 -0
  370. package/dist/tui/input-box.js +247 -32
  371. package/dist/tui/login-picker.js +3 -3
  372. package/dist/tui/markdown-render.js +6 -6
  373. package/dist/tui/onboarding-wizard.js +240 -0
  374. package/dist/tui/permissions-picker.js +86 -0
  375. package/dist/tui/render.js +35 -0
  376. package/dist/tui/repl-render.js +332 -54
  377. package/dist/tui/repl-splash-art.js +16 -16
  378. package/dist/tui/repl-splash-mascot.js +48 -24
  379. package/dist/tui/repl-splash.js +22 -22
  380. package/dist/tui/repl.js +124 -44
  381. package/dist/tui/slash-palette.js +6 -6
  382. package/dist/tui/splash.js +2 -2
  383. package/dist/tui/status-bar.js +109 -31
  384. package/dist/tui/status-table.js +7 -0
  385. package/dist/tui/stickers-art.js +136 -0
  386. package/dist/tui/style-table.js +28 -0
  387. package/dist/tui/theme-table.js +29 -0
  388. package/dist/tui/thinking-spinner.js +123 -0
  389. package/dist/tui/tool-stream-pane.js +53 -4
  390. package/dist/tui/update-banner.js +27 -2
  391. package/dist/tui/vim-input.js +267 -0
  392. package/dist/tui/welcome-banner.js +107 -0
  393. package/dist/tui/welcome-data.js +293 -0
  394. package/dist/tui/workspace-context.js +2 -2
  395. package/docs/examples/codegraph.mcp.json +10 -0
  396. package/package.json +23 -6
  397. package/test/scenarios/codegen-create-file.scenario.txt +13 -0
  398. package/test/scenarios/compact-force.scenario.txt +11 -0
  399. package/test/scenarios/identity.scenario.txt +11 -0
  400. package/test/scenarios/persona-handoff.scenario.txt +11 -0
  401. package/test/scenarios/walkback.scenario.txt +12 -0
  402. package/dist/core/engine/compaction-hook.js +0 -154
@@ -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
@@ -0,0 +1,134 @@
1
+ /**
2
+ * Codegraph MCP install helper — .
3
+ *
4
+ * Writes the codegraph MCP server config к `.pugi/mcp.json` so the
5
+ * registry loader picks it up on the next dispatch. Mirrors the
6
+ * Phase 1 example config at `apps/pugi-cli/docs/examples/codegraph.mcp.json`
7
+ * — same `codegraph serve --mcp` command + `pending` trust state. The
8
+ * operator still has to run `pugi mcp trust codegraph` before tools
9
+ * actually surface; the install path NEVER auto-trusts, by design.
10
+ *
11
+ * Idempotent: re-running on a workspace that already has a `codegraph`
12
+ * entry is a no-op + returns `{ status: 'already-installed' }`. Other
13
+ * MCP servers in the same file are preserved.
14
+ *
15
+ * The function is intentionally NOT bundled into core/mcp/ to keep
16
+ * the codegraph product (install copy, decision store, language gate)
17
+ * one cohesive module. The MCP registry is a generic surface; the
18
+ * codegraph adoption is a feature on top of it.
19
+ */
20
+ import { existsSync, mkdirSync, readFileSync, writeFileSync } from 'node:fs';
21
+ import { resolve } from 'node:path';
22
+ /**
23
+ * Canonical config shape we write into `.pugi/mcp.json` under the
24
+ * `codegraph` key. Matches `mcpServerConfigSchema` (Zod) in
25
+ * core/mcp/client.ts and the published example at docs/examples/
26
+ * codegraph.mcp.json. Drift here MUST land alongside a registry
27
+ * schema change OR the next dispatch crashes with a validation error.
28
+ */
29
+ export const CODEGRAPH_MCP_ENTRY = Object.freeze({
30
+ command: 'codegraph',
31
+ args: Object.freeze(['serve', '--mcp']),
32
+ env: Object.freeze({}),
33
+ trust: 'pending',
34
+ });
35
+ /**
36
+ * Documentation URL surfaced after a successful install so the operator
37
+ * knows how to actually run codegraph index commands. Single source of
38
+ * truth — both the init flow and the /codegraph-status command pull
39
+ * from this constant.
40
+ */
41
+ export const CODEGRAPH_DOCS_URL = 'https://github.com/colbymchenry/codegraph';
42
+ /**
43
+ * Merge the codegraph entry into `.pugi/mcp.json`. Creates the file
44
+ * (with `{ schema: 1, servers: { codegraph: ... } }`) when it does not
45
+ * exist. Preserves every other server entry on disk.
46
+ *
47
+ * @param workspaceRoot absolute path to the project root that owns the
48
+ * `.pugi/` directory. The caller is responsible for
49
+ * ensuring `.pugi/` exists (the install helper
50
+ * creates it as a defensive fallback).
51
+ */
52
+ export function installCodegraphMcpEntry(workspaceRoot) {
53
+ const pugiDir = resolve(workspaceRoot, '.pugi');
54
+ const configPath = resolve(pugiDir, 'mcp.json');
55
+ try {
56
+ if (!existsSync(pugiDir)) {
57
+ mkdirSync(pugiDir, { recursive: true });
58
+ }
59
+ let existing = {};
60
+ if (existsSync(configPath)) {
61
+ try {
62
+ const raw = readFileSync(configPath, 'utf8');
63
+ if (raw.trim().length > 0) {
64
+ const parsed = JSON.parse(raw);
65
+ if (parsed && typeof parsed === 'object') {
66
+ existing = parsed;
67
+ }
68
+ }
69
+ }
70
+ catch (error) {
71
+ return {
72
+ status: 'failed',
73
+ reason: `cannot parse existing .pugi/mcp.json: ${error.message}`,
74
+ };
75
+ }
76
+ }
77
+ const servers = (existing.servers && typeof existing.servers === 'object')
78
+ ? { ...existing.servers }
79
+ : {};
80
+ if (servers['codegraph']) {
81
+ return { status: 'already-installed', configPath };
82
+ }
83
+ servers['codegraph'] = {
84
+ command: CODEGRAPH_MCP_ENTRY.command,
85
+ args: [...CODEGRAPH_MCP_ENTRY.args],
86
+ env: { ...CODEGRAPH_MCP_ENTRY.env },
87
+ trust: CODEGRAPH_MCP_ENTRY.trust,
88
+ };
89
+ const out = {
90
+ schema: typeof existing.schema === 'number' ? existing.schema : 1,
91
+ servers,
92
+ };
93
+ writeFileSync(configPath, `${JSON.stringify(out, null, 2)}\n`, { mode: 0o600 });
94
+ return { status: 'installed', configPath };
95
+ }
96
+ catch (error) {
97
+ return { status: 'failed', reason: error.message };
98
+ }
99
+ }
100
+ /**
101
+ * Check whether `.pugi/mcp.json` already declares the `codegraph`
102
+ * server. Pure best-effort — a malformed file returns false (we err
103
+ * on the side of "not installed" so the operator can re-trigger the
104
+ * install path instead of being silently locked out).
105
+ *
106
+ * Returns the parsed `trust` state when present so callers can render
107
+ * the right status copy ("declared, awaiting trust" vs "active").
108
+ */
109
+ export function detectCodegraphInstalled(workspaceRoot) {
110
+ const configPath = resolve(workspaceRoot, '.pugi/mcp.json');
111
+ if (!existsSync(configPath)) {
112
+ return { installed: false, trust: null, configPath };
113
+ }
114
+ try {
115
+ const raw = readFileSync(configPath, 'utf8');
116
+ if (raw.trim().length === 0) {
117
+ return { installed: false, trust: null, configPath };
118
+ }
119
+ const parsed = JSON.parse(raw);
120
+ const codegraph = parsed.servers?.['codegraph'];
121
+ if (!codegraph) {
122
+ return { installed: false, trust: null, configPath };
123
+ }
124
+ return {
125
+ installed: true,
126
+ trust: codegraph.trust === 'trusted' || codegraph.trust === 'denied' ? codegraph.trust : 'pending',
127
+ configPath,
128
+ };
129
+ }
130
+ catch {
131
+ return { installed: false, trust: null, configPath };
132
+ }
133
+ }
134
+ //# sourceMappingURL=install.js.map