@pugi/cli 0.1.0-beta.1 → 0.1.0-beta.100

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (448) hide show
  1. package/CHANGELOG.md +132 -0
  2. package/LICENSE +1 -1
  3. package/README.md +53 -11
  4. package/THIRD_PARTY_NOTICES.md +40 -0
  5. package/assets/pugi-mascot.ansi +15 -40
  6. package/assets/pugi-prozr2-mascot.ansi +9 -0
  7. package/bin/run.js +33 -1
  8. package/dist/commands/deploy.js +40 -40
  9. package/dist/commands/flatten.js +191 -0
  10. package/dist/commands/jobs-watch.js +201 -0
  11. package/dist/commands/jobs.js +42 -27
  12. package/dist/commands/retro.js +210 -0
  13. package/dist/commands/smoke.js +133 -0
  14. package/dist/core/agent-progress/cleanup.js +134 -0
  15. package/dist/core/agent-progress/schema.js +144 -0
  16. package/dist/core/agent-progress/writer.js +101 -0
  17. package/dist/core/agents/adaptive-router.js +330 -0
  18. package/dist/core/agents/query-decomposer.js +297 -0
  19. package/dist/core/agents/registry.js +3 -3
  20. package/dist/core/approvals/shortcut-resolver.js +98 -0
  21. package/dist/core/artifact-chain/dispatcher.js +148 -0
  22. package/dist/core/artifact-chain/exporter.js +164 -0
  23. package/dist/core/artifact-chain/state.js +243 -0
  24. package/dist/core/artifact-chain/steps.js +169 -0
  25. package/dist/core/ask-user/question.js +92 -0
  26. package/dist/core/audit/audit-trail.js +275 -0
  27. package/dist/core/auth/ensure-authenticated.js +129 -0
  28. package/dist/core/auth/env-provider.js +238 -0
  29. package/dist/core/auto-open-browser.js +4 -4
  30. package/dist/core/auto-update/channels.js +122 -0
  31. package/dist/core/auto-update/checker.js +241 -0
  32. package/dist/core/auto-update/state.js +235 -0
  33. package/dist/core/bare-mode/index.js +107 -0
  34. package/dist/core/bash/redirect.js +281 -0
  35. package/dist/core/bash-classifier.js +436 -40
  36. package/dist/core/checkpoint/resumer.js +149 -0
  37. package/dist/core/checkpoint/rewinder.js +291 -0
  38. package/dist/core/checkpoints/shadow-git.js +670 -0
  39. package/dist/core/citations/parser.js +109 -0
  40. package/dist/core/classifier/yolo-classifier.js +88 -0
  41. package/dist/core/codegraph/db.js +506 -0
  42. package/dist/core/codegraph/decision-store.js +248 -0
  43. package/dist/core/codegraph/detect-repo.js +459 -0
  44. package/dist/core/codegraph/install.js +134 -0
  45. package/dist/core/codegraph/offer-hook.js +220 -0
  46. package/dist/core/codegraph/parser.js +71 -0
  47. package/dist/core/codegraph/types.js +34 -0
  48. package/dist/core/compact/auto-trigger.js +96 -0
  49. package/dist/core/compact/buffer-rewriter.js +115 -0
  50. package/dist/core/compact/summarizer.js +208 -0
  51. package/dist/core/compact/token-counter.js +108 -0
  52. package/dist/core/consensus/anvil-fanout.js +25 -25
  53. package/dist/core/consensus/diff-capture.js +121 -12
  54. package/dist/core/consensus/rubric.js +21 -21
  55. package/dist/core/context/builder.js +6 -6
  56. package/dist/core/context/compaction-events.js +8 -8
  57. package/dist/core/context/compaction.js +31 -31
  58. package/dist/core/context/index.js +15 -8
  59. package/dist/core/context/invariants.js +51 -51
  60. package/dist/core/context/markdown-loader.js +28 -10
  61. package/dist/core/context/markdown-traverse.js +255 -0
  62. package/dist/core/context/pugiignore.js +41 -41
  63. package/dist/core/context/repo-skeleton.js +37 -37
  64. package/dist/core/context/tool-eviction.js +55 -0
  65. package/dist/core/context/watcher.js +32 -32
  66. package/dist/core/context/working-set.js +23 -23
  67. package/dist/core/coordinator/agent-tools.js +77 -0
  68. package/dist/core/coordinator/agent-toolset.js +65 -0
  69. package/dist/core/coordinator/fsm.js +73 -0
  70. package/dist/core/coordinator/mode-fsm.js +70 -0
  71. package/dist/core/cost/rate-card.js +129 -0
  72. package/dist/core/cost/tracker.js +221 -0
  73. package/dist/core/credentials.js +13 -13
  74. package/dist/core/cron/scheduler.js +138 -0
  75. package/dist/core/denial-tracking/index.js +8 -0
  76. package/dist/core/denial-tracking/state.js +264 -0
  77. package/dist/core/diagnostics/probe-runner.js +93 -0
  78. package/dist/core/diagnostics/probes/api.js +46 -0
  79. package/dist/core/diagnostics/probes/auth.js +93 -0
  80. package/dist/core/diagnostics/probes/bare-mode.js +42 -0
  81. package/dist/core/diagnostics/probes/cli-version.js +127 -0
  82. package/dist/core/diagnostics/probes/config.js +72 -0
  83. package/dist/core/diagnostics/probes/denial-tracking.js +57 -0
  84. package/dist/core/diagnostics/probes/disk.js +81 -0
  85. package/dist/core/diagnostics/probes/engine-live.js +46 -0
  86. package/dist/core/diagnostics/probes/git.js +65 -0
  87. package/dist/core/diagnostics/probes/hooks.js +118 -0
  88. package/dist/core/diagnostics/probes/mcp.js +75 -0
  89. package/dist/core/diagnostics/probes/node.js +59 -0
  90. package/dist/core/diagnostics/probes/pnpm.js +36 -0
  91. package/dist/core/diagnostics/probes/pugi-md.js +89 -0
  92. package/dist/core/diagnostics/probes/sandbox.js +72 -0
  93. package/dist/core/diagnostics/probes/session.js +74 -0
  94. package/dist/core/diagnostics/probes/status-snapshot.js +488 -0
  95. package/dist/core/diagnostics/probes/workspace.js +63 -0
  96. package/dist/core/diagnostics/types.js +70 -0
  97. package/dist/core/dispatch/cache-cleanup.js +197 -0
  98. package/dist/core/dispatch/cache-handoff.js +295 -0
  99. package/dist/core/edits/apply-patch-layer-e.js +189 -0
  100. package/dist/core/edits/dispatch.js +333 -7
  101. package/dist/core/edits/format-detector.js +260 -0
  102. package/dist/core/edits/format-matrix.js +26 -0
  103. package/dist/core/edits/fuzzy-ladder.js +650 -0
  104. package/dist/core/edits/index.js +5 -1
  105. package/dist/core/edits/journal.js +199 -0
  106. package/dist/core/edits/layer-a-apply.js +15 -15
  107. package/dist/core/edits/layer-a-fuzzy-apply.js +198 -0
  108. package/dist/core/edits/layer-b-apply.js +9 -9
  109. package/dist/core/edits/layer-c-apply.js +6 -6
  110. package/dist/core/edits/layer-d-ast.js +557 -14
  111. package/dist/core/edits/marker-parser.js +12 -12
  112. package/dist/core/edits/security-gate.js +27 -27
  113. package/dist/core/edits/verify-hook.js +273 -0
  114. package/dist/core/edits/worktree.js +322 -0
  115. package/dist/core/engine/anvil-client.js +214 -26
  116. package/dist/core/engine/auto-compact.js +247 -0
  117. package/dist/core/engine/budgets.js +220 -0
  118. package/dist/core/engine/compact-llm-summarizer.js +124 -0
  119. package/dist/core/engine/context-prefix.js +155 -0
  120. package/dist/core/engine/index.js +1 -1
  121. package/dist/core/engine/intensity.js +163 -0
  122. package/dist/core/engine/intent.js +260 -0
  123. package/dist/core/engine/native-pugi.js +1559 -227
  124. package/dist/core/engine/prompts.js +192 -16
  125. package/dist/core/engine/strip-internal-fields.js +124 -0
  126. package/dist/core/engine/tool-bridge.js +1887 -59
  127. package/dist/core/engine/verification-patterns.js +195 -0
  128. package/dist/core/evaluation/golden-dataset.js +293 -0
  129. package/dist/core/feedback/queue.js +177 -0
  130. package/dist/core/feedback/submitter.js +145 -0
  131. package/dist/core/file-cache.js +113 -1
  132. package/dist/core/flatten/flatten-repo.js +439 -0
  133. package/dist/core/format/osc8-link.js +28 -0
  134. package/dist/core/hook-chains.js +392 -0
  135. package/dist/core/hooks/citation-verify-hook.js +138 -0
  136. package/dist/core/hooks/citation-verify.js +112 -0
  137. package/dist/core/hooks/events.js +46 -0
  138. package/dist/core/hooks/index.js +15 -0
  139. package/dist/core/hooks/registry.js +216 -0
  140. package/dist/core/hooks/runner.js +236 -0
  141. package/dist/core/hooks/v2/event-emitter.js +115 -0
  142. package/dist/core/hooks/v2/executor.js +282 -0
  143. package/dist/core/hooks/v2/index.js +25 -0
  144. package/dist/core/hooks/v2/lifecycle.js +104 -0
  145. package/dist/core/hooks/v2/loader.js +216 -0
  146. package/dist/core/hooks/v2/matcher.js +125 -0
  147. package/dist/core/hooks/v2/trust.js +143 -0
  148. package/dist/core/hooks/v2/types.js +86 -0
  149. package/dist/core/hooks/worktree-events.js +158 -0
  150. package/dist/core/image/renderer.js +71 -0
  151. package/dist/core/init/detector.js +582 -0
  152. package/dist/core/init/template-renderer.js +242 -0
  153. package/dist/core/jobs/registry.js +18 -18
  154. package/dist/core/ledger/results-tsv.js +142 -0
  155. package/dist/core/log-discipline/stdout-redirect.js +51 -0
  156. package/dist/core/lsp/cache.js +105 -0
  157. package/dist/core/lsp/client.js +1229 -0
  158. package/dist/core/lsp/language-detect.js +66 -0
  159. package/dist/core/lsp/post-edit-diagnostics.js +171 -0
  160. package/dist/core/lsp/server-detect.js +173 -0
  161. package/dist/core/lsp/symbol-cache.js +162 -0
  162. package/dist/core/lsp/symbol-tools.js +664 -0
  163. package/dist/core/mcp/client.js +97 -28
  164. package/dist/core/mcp/http-server.js +553 -0
  165. package/dist/core/mcp/orchestrator-config.js +192 -0
  166. package/dist/core/mcp/orchestrator-tools.js +806 -0
  167. package/dist/core/mcp/permission.js +190 -0
  168. package/dist/core/mcp/registry.js +39 -17
  169. package/dist/core/mcp/server-tools.js +219 -0
  170. package/dist/core/mcp/server.js +397 -0
  171. package/dist/core/mcp/trust.js +10 -10
  172. package/dist/core/memory/dual-write.js +416 -0
  173. package/dist/core/memory/passive-extract.js +130 -0
  174. package/dist/core/memory/phase1-kinds.js +20 -0
  175. package/dist/core/memory/secret-scanner.js +304 -0
  176. package/dist/core/memory-sync/queue.js +170 -0
  177. package/dist/core/metrics/extract.js +113 -0
  178. package/dist/core/modes/roo-modes.js +68 -0
  179. package/dist/core/notes/notes-paths.js +113 -0
  180. package/dist/core/notes/notes-recorder.js +140 -0
  181. package/dist/core/notes/notes-writer.js +53 -0
  182. package/dist/core/notes/renderers.js +0 -0
  183. package/dist/core/notes/slug.js +105 -0
  184. package/dist/core/onboarding/ensure-initialized.js +133 -0
  185. package/dist/core/onboarding/marker.js +111 -0
  186. package/dist/core/onboarding/telemetry-state.js +108 -0
  187. package/dist/core/output-style/presets.js +176 -0
  188. package/dist/core/output-style/state.js +185 -0
  189. package/dist/core/path-security.js +287 -5
  190. package/dist/core/permission.js +82 -22
  191. package/dist/core/permissions/auto-classifier.js +124 -0
  192. package/dist/core/permissions/bash-parser.js +371 -0
  193. package/dist/core/permissions/circuit-breaker.js +83 -0
  194. package/dist/core/permissions/constrained-edit.js +91 -0
  195. package/dist/core/permissions/gate.js +278 -0
  196. package/dist/core/permissions/index.js +20 -0
  197. package/dist/core/permissions/mode.js +174 -0
  198. package/dist/core/permissions/network-egress.js +137 -0
  199. package/dist/core/permissions/state.js +241 -0
  200. package/dist/core/permissions/tool-class.js +107 -0
  201. package/dist/core/plan-mode/ui-state.js +51 -0
  202. package/dist/core/plans/plan-artifact.js +721 -0
  203. package/dist/core/policy-limits/etag-store.js +122 -0
  204. package/dist/core/prd-check/parser.js +215 -0
  205. package/dist/core/prd-check/reporter.js +127 -0
  206. package/dist/core/prd-check/session-review.js +557 -0
  207. package/dist/core/prd-check/verifiers.js +223 -0
  208. package/dist/core/prompt-cache/client-cache.js +99 -0
  209. package/dist/core/prompts/assembly.js +29 -0
  210. package/dist/core/prompts/registry.js +364 -0
  211. package/dist/core/pugi-gitignore.js +52 -0
  212. package/dist/core/pugi-md/cc-compat-rules.js +735 -0
  213. package/dist/core/pugi-md/context-injector.js +76 -0
  214. package/dist/core/pugi-md/walk-up.js +207 -0
  215. package/dist/core/python/uv-installer.js +270 -0
  216. package/dist/core/python/uv-resolver.js +83 -0
  217. package/dist/core/rate-limit/narrator.js +146 -0
  218. package/dist/core/recipes/cli-types.js +20 -0
  219. package/dist/core/recipes/loader.js +103 -0
  220. package/dist/core/recipes/runner.js +345 -0
  221. package/dist/core/recipes/schema.js +587 -0
  222. package/dist/core/release-notes/parser.js +241 -0
  223. package/dist/core/release-notes/state.js +116 -0
  224. package/dist/core/repl/ask.js +37 -37
  225. package/dist/core/repl/cancellation.js +26 -26
  226. package/dist/core/repl/cap-warning.js +4 -4
  227. package/dist/core/repl/clipboard-read.js +11 -11
  228. package/dist/core/repl/dispatch-fsm.js +12 -12
  229. package/dist/core/repl/engine-bridge.js +303 -0
  230. package/dist/core/repl/history-search.js +15 -15
  231. package/dist/core/repl/history.js +28 -18
  232. package/dist/core/repl/kill-ring.js +5 -5
  233. package/dist/core/repl/model-pricing.js +135 -0
  234. package/dist/core/repl/privacy-banner.js +22 -22
  235. package/dist/core/repl/session.js +2714 -228
  236. package/dist/core/repl/slash-commands.js +572 -40
  237. package/dist/core/repl/store/index.js +1 -1
  238. package/dist/core/repl/store/jsonl-log.js +22 -22
  239. package/dist/core/repl/store/lockfile.js +10 -10
  240. package/dist/core/repl/store/session-store.js +136 -107
  241. package/dist/core/repl/store/types.js +15 -15
  242. package/dist/core/repl/store/uuid-v7.js +12 -12
  243. package/dist/core/repl/tool-route.js +382 -0
  244. package/dist/core/repl/workspace-context.js +43 -21
  245. package/dist/core/repo-map/build.js +125 -0
  246. package/dist/core/repo-map/cache.js +185 -0
  247. package/dist/core/repo-map/extractor.js +254 -0
  248. package/dist/core/repo-map/formatter.js +145 -0
  249. package/dist/core/repo-map/page-rank.js +105 -0
  250. package/dist/core/repo-map/scanner.js +211 -0
  251. package/dist/core/retro/git-collector.js +251 -0
  252. package/dist/core/retro/health-card.js +25 -0
  253. package/dist/core/retro/metrics.js +342 -0
  254. package/dist/core/retro/narrative.js +249 -0
  255. package/dist/core/retro/plane-collector.js +274 -0
  256. package/dist/core/retro/pr-issue-link.js +65 -0
  257. package/dist/core/retro/types.js +16 -0
  258. package/dist/core/retry-budget/budget.js +284 -0
  259. package/dist/core/retry-budget/index.js +5 -0
  260. package/dist/core/retry-budget/retry-cap.js +74 -0
  261. package/dist/core/routing/lead-worker.js +43 -0
  262. package/dist/core/routing/pre-flight-estimator.js +108 -0
  263. package/dist/core/runs/run-tree.js +103 -0
  264. package/dist/core/sandboxing/adapter.js +29 -0
  265. package/dist/core/sandboxing/index.js +49 -0
  266. package/dist/core/sandboxing/none.js +19 -0
  267. package/dist/core/sandboxing/seatbelt.js +183 -0
  268. package/dist/core/security/injection-scanner.js +367 -0
  269. package/dist/core/security/output-filter.js +418 -0
  270. package/dist/core/session/env-file.js +105 -0
  271. package/dist/core/session/section-budgets.js +140 -0
  272. package/dist/core/session.js +119 -0
  273. package/dist/core/settings.js +378 -5
  274. package/dist/core/share/formatter.js +271 -0
  275. package/dist/core/share/redactor.js +221 -0
  276. package/dist/core/share/uploader.js +267 -0
  277. package/dist/core/skills/defaults.js +457 -0
  278. package/dist/core/skills/loader.js +22 -22
  279. package/dist/core/skills/sources.js +27 -27
  280. package/dist/core/smoke/headless-driver.js +174 -0
  281. package/dist/core/smoke/orchestrator.js +194 -0
  282. package/dist/core/smoke/runner.js +238 -0
  283. package/dist/core/smoke/scenario-parser.js +316 -0
  284. package/dist/core/statusline.js +99 -0
  285. package/dist/core/subagents/dispatcher-real.js +600 -0
  286. package/dist/core/subagents/dispatcher.js +146 -52
  287. package/dist/core/subagents/index.js +19 -6
  288. package/dist/core/subagents/isolation-matrix.js +213 -0
  289. package/dist/core/subagents/spawn.js +19 -4
  290. package/dist/core/telemetry/emitter.js +229 -0
  291. package/dist/core/telemetry/queue.js +251 -0
  292. package/dist/core/theme/context.js +91 -0
  293. package/dist/core/theme/presets.js +228 -0
  294. package/dist/core/theme/state.js +181 -0
  295. package/dist/core/todos/invariant.js +10 -0
  296. package/dist/core/todos/state.js +177 -0
  297. package/dist/core/tool-schema/compressor.js +89 -0
  298. package/dist/core/transport/version-interceptor.js +166 -0
  299. package/dist/core/trust.js +2 -2
  300. package/dist/core/tui/thinking-block.js +64 -0
  301. package/dist/core/vim/keymap.js +288 -0
  302. package/dist/core/vim/state.js +92 -0
  303. package/dist/core/watch-markers/marker-watcher.js +133 -0
  304. package/dist/core/worktree/include-parser.js +249 -0
  305. package/dist/core/worktree-manager/cleanup.js +123 -0
  306. package/dist/core/worktree-manager/manager.js +303 -0
  307. package/dist/index.js +36 -0
  308. package/dist/runtime/bootstrap.js +190 -0
  309. package/dist/runtime/cli.js +4536 -477
  310. package/dist/runtime/commands/agents.js +31 -31
  311. package/dist/runtime/commands/budget.js +5 -5
  312. package/dist/runtime/commands/cancel.js +231 -0
  313. package/dist/runtime/commands/chain.js +489 -0
  314. package/dist/runtime/commands/codegraph-status.js +227 -0
  315. package/dist/runtime/commands/compact.js +297 -0
  316. package/dist/runtime/commands/config.js +74 -40
  317. package/dist/runtime/commands/cost.js +199 -0
  318. package/dist/runtime/commands/delegate.js +312 -0
  319. package/dist/runtime/commands/dispatch.js +126 -0
  320. package/dist/runtime/commands/doctor.js +579 -0
  321. package/dist/runtime/commands/feedback.js +184 -0
  322. package/dist/runtime/commands/hooks.js +187 -0
  323. package/dist/runtime/commands/index-cmd.js +353 -0
  324. package/dist/runtime/commands/init.js +254 -0
  325. package/dist/runtime/commands/lsp.js +368 -0
  326. package/dist/runtime/commands/mcp.js +935 -0
  327. package/dist/runtime/commands/memory.js +582 -0
  328. package/dist/runtime/commands/model.js +237 -0
  329. package/dist/runtime/commands/onboarding.js +275 -0
  330. package/dist/runtime/commands/patch.js +128 -0
  331. package/dist/runtime/commands/permissions.js +112 -0
  332. package/dist/runtime/commands/plan.js +143 -0
  333. package/dist/runtime/commands/prd-check.js +285 -0
  334. package/dist/runtime/commands/privacy.js +17 -17
  335. package/dist/runtime/commands/recipe.js +325 -0
  336. package/dist/runtime/commands/redo-blob-store.js +92 -0
  337. package/dist/runtime/commands/redo.js +361 -0
  338. package/dist/runtime/commands/release-notes.js +229 -0
  339. package/dist/runtime/commands/repo-map.js +95 -0
  340. package/dist/runtime/commands/report.js +299 -0
  341. package/dist/runtime/commands/resume.js +118 -0
  342. package/dist/runtime/commands/review-consensus.js +68 -53
  343. package/dist/runtime/commands/rewind.js +333 -0
  344. package/dist/runtime/commands/roster.js +117 -0
  345. package/dist/runtime/commands/servers.js +236 -0
  346. package/dist/runtime/commands/sessions.js +163 -0
  347. package/dist/runtime/commands/share.js +316 -0
  348. package/dist/runtime/commands/skills.js +31 -31
  349. package/dist/runtime/commands/status.js +186 -0
  350. package/dist/runtime/commands/stickers.js +82 -0
  351. package/dist/runtime/commands/style.js +194 -0
  352. package/dist/runtime/commands/theme.js +196 -0
  353. package/dist/runtime/commands/undo.js +54 -22
  354. package/dist/runtime/commands/update.js +289 -0
  355. package/dist/runtime/commands/vim.js +140 -0
  356. package/dist/runtime/commands/worktree.js +177 -0
  357. package/dist/runtime/commands/worktrees.js +155 -0
  358. package/dist/runtime/deprecation-warning.js +69 -0
  359. package/dist/runtime/engine-exit-code.js +50 -0
  360. package/dist/runtime/headless-repl.js +195 -0
  361. package/dist/runtime/headless.js +548 -0
  362. package/dist/runtime/load-hooks-or-exit.js +71 -0
  363. package/dist/runtime/plan-decompose.js +531 -0
  364. package/dist/runtime/sigint-guard.js +272 -0
  365. package/dist/runtime/stream-renderer.js +195 -0
  366. package/dist/runtime/update-check.js +28 -28
  367. package/dist/runtime/version.js +65 -0
  368. package/dist/runtime/worktree-bootstrap.js +579 -0
  369. package/dist/skills/bundled/batch.js +617 -0
  370. package/dist/skills/bundled/index.js +45 -0
  371. package/dist/skills/bundled/loop.js +358 -0
  372. package/dist/skills/bundled/remember.js +383 -0
  373. package/dist/skills/bundled/simplify.js +289 -0
  374. package/dist/skills/bundled/skillify.js +373 -0
  375. package/dist/skills/bundled/stuck.js +558 -0
  376. package/dist/skills/bundled/verify.js +439 -0
  377. package/dist/testing/vcr.js +486 -0
  378. package/dist/tools/agent-tool.js +229 -0
  379. package/dist/tools/apply-patch.js +556 -0
  380. package/dist/tools/ask-user-question.js +337 -0
  381. package/dist/tools/ask-user.js +115 -0
  382. package/dist/tools/bash.js +624 -46
  383. package/dist/tools/brief.js +224 -0
  384. package/dist/tools/cron.js +433 -0
  385. package/dist/tools/enter-worktree.js +250 -0
  386. package/dist/tools/exit-worktree.js +147 -0
  387. package/dist/tools/file-tools.js +161 -44
  388. package/dist/tools/http-request.js +336 -0
  389. package/dist/tools/lsp-tools.js +565 -0
  390. package/dist/tools/mcp-tool.js +260 -0
  391. package/dist/tools/multi-edit.js +361 -0
  392. package/dist/tools/powershell.js +268 -0
  393. package/dist/tools/registry.js +142 -1
  394. package/dist/tools/server-tools.js +892 -0
  395. package/dist/tools/skill-tool.js +96 -0
  396. package/dist/tools/sleep.js +99 -0
  397. package/dist/tools/synthetic-output.js +133 -0
  398. package/dist/tools/tasks.js +208 -0
  399. package/dist/tools/todo-write.js +184 -0
  400. package/dist/tools/verify-plan-execution.js +295 -0
  401. package/dist/tools/web-fetch-injection-scanner.js +207 -0
  402. package/dist/tools/web-fetch.js +195 -10
  403. package/dist/tools/web-search.js +458 -0
  404. package/dist/tui/agent-progress-card.js +111 -0
  405. package/dist/tui/agent-tree.js +22 -1
  406. package/dist/tui/ask-modal.js +14 -14
  407. package/dist/tui/ask-user-question-chips.js +315 -0
  408. package/dist/tui/ask-user-question-prompt.js +203 -0
  409. package/dist/tui/compact-banner.js +81 -0
  410. package/dist/tui/conversation-pane.js +85 -11
  411. package/dist/tui/cost-table.js +111 -0
  412. package/dist/tui/device-flow.js +2 -2
  413. package/dist/tui/doctor-table.js +46 -0
  414. package/dist/tui/feedback-prompt.js +156 -0
  415. package/dist/tui/input-box.js +247 -32
  416. package/dist/tui/login-picker.js +3 -3
  417. package/dist/tui/markdown-render.js +6 -6
  418. package/dist/tui/multi-file-diff-approval.js +375 -0
  419. package/dist/tui/onboarding-wizard.js +240 -0
  420. package/dist/tui/permissions-picker.js +86 -0
  421. package/dist/tui/render.js +36 -1
  422. package/dist/tui/repl-render.js +405 -32
  423. package/dist/tui/repl-splash-art.js +16 -16
  424. package/dist/tui/repl-splash-mascot.js +48 -24
  425. package/dist/tui/repl-splash.js +22 -22
  426. package/dist/tui/repl.js +136 -43
  427. package/dist/tui/slash-palette.js +6 -6
  428. package/dist/tui/splash.js +2 -2
  429. package/dist/tui/status-bar.js +109 -31
  430. package/dist/tui/status-table.js +7 -0
  431. package/dist/tui/stickers-art.js +136 -0
  432. package/dist/tui/style-table.js +28 -0
  433. package/dist/tui/theme-table.js +29 -0
  434. package/dist/tui/thinking-spinner.js +123 -0
  435. package/dist/tui/tool-stream-pane.js +53 -4
  436. package/dist/tui/update-banner.js +27 -2
  437. package/dist/tui/vim-input.js +267 -0
  438. package/dist/tui/welcome-banner.js +107 -0
  439. package/dist/tui/welcome-data.js +293 -0
  440. package/dist/tui/workspace-context.js +2 -2
  441. package/docs/examples/codegraph.mcp.json +10 -0
  442. package/package.json +25 -7
  443. package/test/scenarios/codegen-create-file.scenario.txt +13 -0
  444. package/test/scenarios/compact-force.scenario.txt +12 -0
  445. package/test/scenarios/identity.scenario.txt +11 -0
  446. package/test/scenarios/persona-handoff.scenario.txt +12 -0
  447. package/test/scenarios/walkback.scenario.txt +12 -0
  448. package/dist/core/engine/compaction-hook.js +0 -154
@@ -0,0 +1,255 @@
1
+ /**
2
+ * Per-directory PUGI.md / AGENTS.md / CLAUDE.md / GEMINI.md traverse-up
3
+ * loader — β5a R4+P5.
4
+ *
5
+ * the upstream tool, peer CLI, and Gemini CLI all support a "walk up from
6
+ * cwd to the workspace root, pick up agent-context markdown at every
7
+ * level" pattern. Without this, a `pugi explain` invoked from
8
+ * `apps/admin-api/` cannot see project-local conventions encoded in
9
+ * `apps/admin-api/PUGI.md` (or the cross-CLI shim files) — the
10
+ * existing `loadMarkdownContext` only reads files at `workspaceRoot`.
11
+ *
12
+ * The β5a quality gate (≥80% win-rate vs the upstream tool per CEO)
13
+ * surfaces this gap repeatedly: monorepo-local conventions (NestJS
14
+ * controller style, Prisma migration name format, cabinet brand voice
15
+ * gates) live in per-app context files, and Pugi was blind to them
16
+ * pre-β5a.
17
+ *
18
+ * Contract:
19
+ *
20
+ * - Walk from `cwd` upward until we reach `workspaceRoot` OR cross
21
+ * the filesystem boundary. The workspace root file itself is
22
+ * loaded by the legacy `loadMarkdownContext` so we do NOT include
23
+ * it here (no double-load, no double-budget-charge).
24
+ *
25
+ * - At each intermediate directory, look for the four canonical
26
+ * filenames: `PUGI.md` (native), `AGENTS.md` (cross-CLI shim),
27
+ * `CLAUDE.md` (the upstream tool compat), `GEMINI.md` (Gemini CLI compat).
28
+ *
29
+ * - HTML comments stripped, identical to the legacy loader.
30
+ *
31
+ * - `@import` expansion is intentionally NOT performed here — the
32
+ * per-dir surface is "drop a small file with the local
33
+ * conventions"; deep @import chains belong at workspace root.
34
+ * Keeping this surface flat means the per-dir budget cannot be
35
+ * blown out by a runaway @import in an unrelated subtree.
36
+ *
37
+ * - Aggregate budget: `MAX_TRAVERSE_BYTES` across ALL files found
38
+ * in the walk. When exhausted, remaining files are skipped with
39
+ * a `budget_exhausted` warning. Default 32 KB — half the
40
+ * workspace-root budget, because per-dir files are meant to be
41
+ * terse delta conventions, not full project briefs.
42
+ *
43
+ * - The walk is bounded: `MAX_TRAVERSE_DEPTH` levels above
44
+ * workspaceRoot are NEVER traversed (defense against being
45
+ * invoked from a malicious cwd outside the workspace; in
46
+ * practice cwd is always inside workspaceRoot but the symlink
47
+ * case demands belt + suspenders).
48
+ *
49
+ * - Order returned: shallowest-first (workspace root would be
50
+ * first if included, then each level closer to cwd). Closest-
51
+ * to-cwd files are the most specific and the context builder
52
+ * emits them LAST so the model treats them as the highest-
53
+ * priority conventions.
54
+ *
55
+ * This module is pure: no logging, no network, no fs writes. Filesystem
56
+ * reads only.
57
+ */
58
+ import { existsSync, readFileSync, realpathSync, statSync } from 'node:fs';
59
+ import { dirname, isAbsolute, relative, resolve, sep } from 'node:path';
60
+ import { stripHtmlComments } from './markdown-loader.js';
61
+ /**
62
+ * Per-traverse total byte cap. Half the workspace-root budget — these
63
+ * files are meant to be terse "in this subdir, do X". 32 KB still fits
64
+ * ~8000 words of guidance.
65
+ */
66
+ export const MAX_TRAVERSE_BYTES = 32 * 1024;
67
+ /**
68
+ * Per-file byte cap. A single per-dir file should never dominate; the
69
+ * 8 KB cap keeps any one level honest. Files larger than this are
70
+ * loaded up to the cap and flagged truncated.
71
+ */
72
+ export const MAX_TRAVERSE_PER_FILE_BYTES = 8 * 1024;
73
+ /**
74
+ * Maximum number of parent directories above workspaceRoot we will
75
+ * traverse. Zero in normal operation — cwd is always inside the
76
+ * workspace; the cap exists so a misconfigured invocation never
77
+ * walks the whole filesystem looking for AGENTS.md.
78
+ */
79
+ export const MAX_TRAVERSE_DEPTH = 0;
80
+ /**
81
+ * Filenames we look for at every level of the walk. The order here
82
+ * also defines the per-directory load order: PUGI.md first (highest
83
+ * trust), then cross-CLI compat shims. When the same directory has
84
+ * multiple files (e.g. both PUGI.md AND CLAUDE.md), all are loaded
85
+ * — operators sometimes keep both during a tool migration.
86
+ */
87
+ export const TRAVERSE_SOURCES = ['PUGI.md', 'AGENTS.md', 'CLAUDE.md', 'GEMINI.md'];
88
+ /**
89
+ * Walk from `opts.cwd` upward toward `opts.workspaceRoot`, loading
90
+ * every PUGI.md / AGENTS.md / CLAUDE.md / GEMINI.md we encounter at
91
+ * intermediate levels. The workspace root itself is NOT loaded here
92
+ * — `loadMarkdownContext(workspaceRoot)` owns that file.
93
+ *
94
+ * Returned `loaded` is sorted shallowest-first (closest-to-root) so
95
+ * the caller can emit per-dir context in increasing specificity.
96
+ *
97
+ * Safety properties (proven by spec):
98
+ *
99
+ * - Never reads outside the workspace tree (symlinks resolved via
100
+ * realpathSync; off-tree symlinks rejected as
101
+ * `import_escapes_workspace`).
102
+ * - Never reads more than `MAX_TRAVERSE_BYTES` total or more than
103
+ * `MAX_TRAVERSE_PER_FILE_BYTES` per file.
104
+ * - Never walks above the workspace root.
105
+ * - Never visits the workspace root itself (single source of truth
106
+ * for workspace-level docs stays with `loadMarkdownContext`).
107
+ */
108
+ export async function loadTraversedMarkdown(opts) {
109
+ const warnings = [];
110
+ const loaded = [];
111
+ let budgetRemaining = MAX_TRAVERSE_BYTES;
112
+ let absRoot;
113
+ let absCwd;
114
+ try {
115
+ absRoot = realpathSync(resolve(opts.workspaceRoot));
116
+ absCwd = realpathSync(resolve(opts.cwd));
117
+ }
118
+ catch (error) {
119
+ warnings.push({
120
+ kind: 'read_error',
121
+ message: `realpath failed for traverse anchor: ${error.message}`,
122
+ });
123
+ return { loaded, warnings, totalBytes: 0 };
124
+ }
125
+ // Containment guard: if cwd is not inside workspaceRoot, refuse
126
+ // to walk. Returning a clean empty result keeps the engine happy
127
+ // and surfaces nothing about the off-tree cwd to the model.
128
+ const relCwd = relative(absRoot, absCwd);
129
+ if (relCwd.startsWith('..') || isAbsolute(relCwd)) {
130
+ warnings.push({
131
+ kind: 'import_escapes_workspace',
132
+ message: `cwd is outside workspaceRoot; per-dir traverse skipped (cwd=${absCwd}, root=${absRoot})`,
133
+ path: absCwd,
134
+ });
135
+ return { loaded, warnings, totalBytes: 0 };
136
+ }
137
+ // Collect the walk: every directory from cwd UP TO (but not
138
+ // including) workspaceRoot.
139
+ const dirsToVisit = [];
140
+ let current = absCwd;
141
+ while (current !== absRoot) {
142
+ dirsToVisit.push(current);
143
+ const parent = dirname(current);
144
+ if (parent === current)
145
+ break; // hit filesystem root before workspaceRoot — defensive
146
+ current = parent;
147
+ if (dirsToVisit.length > 64)
148
+ break; // pathological depth, refuse
149
+ }
150
+ // Walk shallowest-first so we charge the budget in the order
151
+ // that matches what we return.
152
+ dirsToVisit.reverse();
153
+ for (const dir of dirsToVisit) {
154
+ if (budgetRemaining <= 0) {
155
+ warnings.push({
156
+ kind: 'budget_exhausted',
157
+ message: `per-dir traverse budget exhausted before reaching ${dir}`,
158
+ path: dir,
159
+ });
160
+ break;
161
+ }
162
+ for (const source of TRAVERSE_SOURCES) {
163
+ const candidate = resolve(dir, source);
164
+ if (!existsSync(candidate))
165
+ continue;
166
+ // Symlink guard: same realpath check as the workspace-root
167
+ // loader. A symlink inside the workspace that points outside
168
+ // the workspace must NOT be inlined.
169
+ let realCandidate;
170
+ try {
171
+ realCandidate = realpathSync(candidate);
172
+ }
173
+ catch (error) {
174
+ warnings.push({
175
+ kind: 'read_error',
176
+ message: `realpath failed for ${candidate}: ${error.message}`,
177
+ path: candidate,
178
+ });
179
+ continue;
180
+ }
181
+ const realRel = relative(absRoot, realCandidate);
182
+ if (realRel.startsWith('..') || isAbsolute(realRel)) {
183
+ warnings.push({
184
+ kind: 'import_escapes_workspace',
185
+ message: `traverse file escapes workspace via symlink: ${candidate} -> ${realCandidate}`,
186
+ path: candidate,
187
+ });
188
+ continue;
189
+ }
190
+ let raw;
191
+ let rawBytes;
192
+ try {
193
+ rawBytes = statSync(candidate).size;
194
+ raw = readFileSync(candidate, 'utf8');
195
+ }
196
+ catch (error) {
197
+ warnings.push({
198
+ kind: 'read_error',
199
+ message: `could not read ${candidate}: ${error.message}`,
200
+ path: candidate,
201
+ });
202
+ continue;
203
+ }
204
+ const stripped = stripHtmlComments(raw);
205
+ // Per-file cap first, then global budget.
206
+ const perFileCap = Math.min(MAX_TRAVERSE_PER_FILE_BYTES, budgetRemaining);
207
+ let content = stripped;
208
+ let truncated = false;
209
+ let contentBytes = Buffer.byteLength(content, 'utf8');
210
+ if (contentBytes > perFileCap) {
211
+ // Codepoint-safe slice: convert byte cap to char cap by
212
+ // taking min(byte cap, char-length-up-to-cap). We accept
213
+ // mild over-trim for safety.
214
+ content = content.slice(0, perFileCap);
215
+ truncated = true;
216
+ contentBytes = Buffer.byteLength(content, 'utf8');
217
+ }
218
+ const distance = distanceSegments(absCwd, dir);
219
+ loaded.push({
220
+ source,
221
+ resolvedPath: candidate,
222
+ dir,
223
+ distanceFromCwd: distance,
224
+ rawBytes,
225
+ loadedBytes: contentBytes,
226
+ truncated,
227
+ content,
228
+ });
229
+ budgetRemaining -= contentBytes;
230
+ if (budgetRemaining <= 0)
231
+ break;
232
+ }
233
+ }
234
+ return {
235
+ loaded,
236
+ warnings,
237
+ totalBytes: MAX_TRAVERSE_BYTES - budgetRemaining,
238
+ };
239
+ }
240
+ /**
241
+ * How many path segments separate `from` and `to`. Both must be
242
+ * absolute. `to` is assumed to be an ancestor of (or equal to)
243
+ * `from`; if not, returns -1 so callers can ignore the file.
244
+ */
245
+ function distanceSegments(from, to) {
246
+ if (from === to)
247
+ return 0;
248
+ const rel = relative(to, from);
249
+ if (rel.startsWith('..') || isAbsolute(rel))
250
+ return -1;
251
+ if (rel.length === 0)
252
+ return 0;
253
+ return rel.split(sep).length;
254
+ }
255
+ //# sourceMappingURL=markdown-traverse.js.map
@@ -1,5 +1,5 @@
1
1
  /**
2
- * `.pugiignore` parser - α6.5 Phase 1 (three-tier context).
2
+ * `.pugiignore` parser - Phase 1 (three-tier context).
3
3
  *
4
4
  * The CLI walks the workspace to build a repo skeleton and to feed the
5
5
  * chokidar watcher. Both passes need a single source of truth for which
@@ -9,32 +9,32 @@
9
9
  *
10
10
  * Design choices:
11
11
  *
12
- * 1. We layer four sources, last-write-wins per the .gitignore spec:
13
- * (a) Built-in DEFAULT_IGNORE_PATTERNS - secret / binary / build
14
- * paths every workspace should drop.
15
- * (b) `~/.pugi/global.pugiignore` - operator-global overrides.
16
- * (c) Repo `.gitignore` (when present) - so we don't re-walk
17
- * node_modules / dist / etc.
18
- * (d) Repo `.pugiignore` - workspace-local extensions.
12
+ * 1. We layer four sources, last-write-wins per the .gitignore spec:
13
+ * (a) Built-in DEFAULT_IGNORE_PATTERNS - secret / binary / build
14
+ * paths every workspace should drop.
15
+ * (b) `~/.pugi/global.pugiignore` - operator-global overrides.
16
+ * (c) Repo `.gitignore` (when present) - so we don't re-walk
17
+ * node_modules / dist / etc.
18
+ * (d) Repo `.pugiignore` - workspace-local extensions.
19
19
  *
20
- * 2. The `ignore` npm package handles negation (`!path`) and nested
21
- * directories correctly. Rolling a minimal glob matcher is doable
22
- * but the test surface for gitignore semantics (esp. negation +
23
- * ancestor matching) is wide enough that pulling the audited
24
- * library is the safer call. It is already in the workspace via
25
- * transitive deps; we add it as an explicit pugi-cli dep.
20
+ * 2. The `ignore` npm package handles negation (`!path`) and nested
21
+ * directories correctly. Rolling a minimal glob matcher is doable
22
+ * but the test surface for gitignore semantics (esp. negation +
23
+ * ancestor matching) is wide enough that pulling the audited
24
+ * library is the safer call. It is already in the workspace via
25
+ * transitive deps; we add it as an explicit pugi-cli dep.
26
26
  *
27
- * 3. Privacy is defense-in-depth: even if the operator deletes the
28
- * built-in defaults from their `.pugiignore`, we re-add the
29
- * secret patterns at the END of the chain so a typo in user
30
- * config cannot expose `.env` / `*.pem` / `*.key`. The user can
31
- * explicitly negate (`!.env.example`) but cannot wipe the
32
- * defense.
27
+ * 3. Privacy is defense-in-depth: even if the operator deletes the
28
+ * built-in defaults from their `.pugiignore`, we re-add the
29
+ * secret patterns at the END of the chain so a typo in user
30
+ * config cannot expose `.env` / `*.pem` / `*.key`. The user can
31
+ * explicitly negate (`!.env.example`) but cannot wipe the
32
+ * defense.
33
33
  *
34
- * 4. Paths are normalised to forward slashes (the `ignore` package
35
- * requires POSIX-style separators even on Windows) and stripped
36
- * of any leading workspace root so `isIgnored` answers in
37
- * repo-relative terms.
34
+ * 4. Paths are normalised to forward slashes (the `ignore` package
35
+ * requires POSIX-style separators even on Windows) and stripped
36
+ * of any leading workspace root so `isIgnored` answers in
37
+ * repo-relative terms.
38
38
  *
39
39
  * The API surface is minimal: `loadPugiIgnore(cwd)` returns a
40
40
  * `PugiIgnore` matcher with `isIgnored(absPath): boolean`. The matcher
@@ -59,19 +59,19 @@ const ignoreFactory = ignoreModule.default
59
59
  * Built-in patterns shipped with every Pugi workspace. The list is a
60
60
  * conservative union of three categories:
61
61
  *
62
- * - **Secrets** (defense in depth): `.env*`, `*.pem`, `*.key`,
63
- * `secrets/**`, `.netrc`, `id_rsa*`, `*.secret`, `credentials*`.
64
- * These are re-applied at the END of the chain so user config
65
- * cannot accidentally expose them.
62
+ * - **Secrets** (defense in depth): `.env*`, `*.pem`, `*.key`,
63
+ * `secrets/**`, `.netrc`, `id_rsa*`, `*.secret`, `credentials*`.
64
+ * These are re-applied at the END of the chain so user config
65
+ * cannot accidentally expose them.
66
66
  *
67
- * - **Build outputs / caches**: `node_modules/`, `dist/`, `build/`,
68
- * `.next/`, `.nuxt/`, `.svelte-kit/`, `coverage/`, `.nx/`, `.cache/`,
69
- * `.turbo/`, `.parcel-cache/`, `out/`, `.output/`, `*.tsbuildinfo`.
67
+ * - **Build outputs / caches**: `node_modules/`, `dist/`, `build/`,
68
+ * `.next/`, `.nuxt/`, `.svelte-kit/`, `coverage/`, `.nx/`, `.cache/`,
69
+ * `.turbo/`, `.parcel-cache/`, `out/`, `.output/`, `*.tsbuildinfo`.
70
70
  *
71
- * - **Large binaries / data**: lockfiles (`*.lock`, `*-lock.json`,
72
- * `*-lock.yaml`), images (`*.png`, `*.jpg`, `*.jpeg`, `*.gif`,
73
- * `*.ico`), PDFs, videos, archives (`*.zip`, `*.tar`, `*.gz`),
74
- * DB dumps (`*.sql`, `*.dump`), logs.
71
+ * - **Large binaries / data**: lockfiles (`*.lock`, `*-lock.json`,
72
+ * `*-lock.yaml`), images (`*.png`, `*.jpg`, `*.jpeg`, `*.gif`,
73
+ * `*.ico`), PDFs, videos, archives (`*.zip`, `*.tar`, `*.gz`),
74
+ * DB dumps (`*.sql`, `*.dump`), logs.
75
75
  *
76
76
  * The split between baseline patterns (applied first) and secret
77
77
  * patterns (applied last) matters: a user `.pugiignore` can `!`-negate
@@ -173,7 +173,7 @@ export const SECRET_IGNORE_PATTERNS = Object.freeze([
173
173
  // used by Java + .NET tooling. Both can carry private keys when
174
174
  // bundled (PKCS#7 / PKCS#12) and the file extension alone does not
175
175
  // tell us whether the bundle includes a key. Treat as secrets by
176
- // default. triple-review P1 (PR #380).
176
+ // default. triple-review P1 (PR).
177
177
  '*.cer',
178
178
  '*.der',
179
179
  '*.p12',
@@ -216,11 +216,11 @@ export function workspaceGitIgnorePath(cwd) {
216
216
  * is built in this exact order so later layers override earlier ones,
217
217
  * with the trailing secret block as the inviolable backstop:
218
218
  *
219
- * 1. BASELINE_IGNORE_PATTERNS (built-in defaults)
220
- * 2. ~/.pugi/global.pugiignore (operator global, optional)
221
- * 3. <cwd>/.gitignore (workspace, optional)
222
- * 4. <cwd>/.pugiignore (workspace, optional)
223
- * 5. SECRET_IGNORE_PATTERNS (defense-in-depth backstop)
219
+ * 1. BASELINE_IGNORE_PATTERNS (built-in defaults)
220
+ * 2. ~/.pugi/global.pugiignore (operator global, optional)
221
+ * 3. <cwd>/.gitignore (workspace, optional)
222
+ * 4. <cwd>/.pugiignore (workspace, optional)
223
+ * 5. SECRET_IGNORE_PATTERNS (defense-in-depth backstop)
224
224
  *
225
225
  * Any FS error reading optional files is swallowed - workspace
226
226
  * context is best-effort and must never block the REPL bootstrap.
@@ -1,40 +1,40 @@
1
1
  /**
2
- * Tier 0 repo skeleton builder - α6.5 Phase 1 (three-tier context).
2
+ * Tier 0 repo skeleton builder - Phase 1 (three-tier context).
3
3
  *
4
4
  * The skeleton is the ~5KB "always loaded" tier in the three-tier
5
5
  * model. Goal: give the agent enough structural awareness to navigate
6
6
  * an unfamiliar repo on the first turn without uploading the whole
7
7
  * tree. We capture:
8
8
  *
9
- * - cwd + current branch (best-effort, no exec)
10
- * - detected package manager (lockfile heuristic)
11
- * - primary languages (file-extension histogram, top 5)
12
- * - ASCII directory tree (depth <= MAX_TREE_DEPTH, collapses busy
13
- * dirs to a "(N dirs, M files)" line)
14
- * - package.json projection (name/version/scripts/deps)
15
- * - first MAX_README_LINES of README.md
9
+ * - cwd + current branch (best-effort, no exec)
10
+ * - detected package manager (lockfile heuristic)
11
+ * - primary languages (file-extension histogram, top 5)
12
+ * - ASCII directory tree (depth <= MAX_TREE_DEPTH, collapses busy
13
+ * dirs to a "(N dirs, M files)" line)
14
+ * - package.json projection (name/version/scripts/deps)
15
+ * - first MAX_README_LINES of README.md
16
16
  *
17
17
  * Hard constraints:
18
18
  *
19
- * 1. **5KB cap on the rendered string**. We render in priority
20
- * order (headers + meta first, then tree, then package.json, then
21
- * README) and truncate the tail with a "..." marker rather than
22
- * omitting fields. The walker's bound is independent: it stops
23
- * visiting dirs after a hard `MAX_WALK_NODES` even when the
24
- * skeleton would still fit.
19
+ * 1. **5KB cap on the rendered string**. We render in priority
20
+ * order (headers + meta first, then tree, then package.json, then
21
+ * README) and truncate the tail with a "..." marker rather than
22
+ * omitting fields. The walker's bound is independent: it stops
23
+ * visiting dirs after a hard `MAX_WALK_NODES` even when the
24
+ * skeleton would still fit.
25
25
  *
26
- * 2. **Ignore-aware**: every fs read is filtered through `PugiIgnore`
27
- * so `.env`, `*.pem`, `node_modules/`, etc. stay out. The walker
28
- * uses the same matcher so `node_modules/` is never expanded.
26
+ * 2. **Ignore-aware**: every fs read is filtered through `PugiIgnore`
27
+ * so `.env`, `*.pem`, `node_modules/`, etc. stay out. The walker
28
+ * uses the same matcher so `node_modules/` is never expanded.
29
29
  *
30
- * 3. **No exec**: we read `.git/HEAD` for the branch, never spawn
31
- * `git`. The skeleton is built on session bootstrap; spawning
32
- * git on every launch is slow + makes the CLI flaky on hosts
33
- * without git installed.
30
+ * 3. **No exec**: we read `.git/HEAD` for the branch, never spawn
31
+ * `git`. The skeleton is built on session bootstrap; spawning
32
+ * git on every launch is slow + makes the CLI flaky on hosts
33
+ * without git installed.
34
34
  *
35
- * 4. **Best-effort**: every FS read is wrapped in try/catch.
36
- * Missing README / missing package.json / unreadable file -> the
37
- * field is omitted but the skeleton still builds.
35
+ * 4. **Best-effort**: every FS read is wrapped in try/catch.
36
+ * Missing README / missing package.json / unreadable file -> the
37
+ * field is omitted but the skeleton still builds.
38
38
  */
39
39
  import { existsSync, readdirSync, readFileSync, statSync } from 'node:fs';
40
40
  import { basename, join, resolve } from 'node:path';
@@ -64,7 +64,7 @@ export function buildRepoSkeleton(cwd, options) {
64
64
  // contracts, customer-specific data, NDA'd boilerplate) keeps those
65
65
  // files out of the agent context. Without this gate, the direct
66
66
  // skeleton reads would BYPASS the matcher even though the walker
67
- // honours it. triple-review P1 (PR #380).
67
+ // honours it. triple-review P1 (PR).
68
68
  const pkg = readPackageJson(normalised, ignore);
69
69
  const packageManager = detectPackageManager(normalised, ignore);
70
70
  const walk = walkTree(normalised, ignore);
@@ -93,10 +93,10 @@ export function buildRepoSkeleton(cwd, options) {
93
93
  *
94
94
  * Section order (highest priority first):
95
95
  *
96
- * 1. Meta header (cwd / branch / package manager / languages)
97
- * 2. Directory tree
98
- * 3. package.json projection (name / version / scripts / deps count)
99
- * 4. README excerpt
96
+ * 1. Meta header (cwd / branch / package manager / languages)
97
+ * 2. Directory tree
98
+ * 3. package.json projection (name / version / scripts / deps count)
99
+ * 4. README excerpt
100
100
  *
101
101
  * Each lower-priority section is appended only when there is room for
102
102
  * its FULL body. Partial sections would mislead the model (e.g. a
@@ -132,7 +132,7 @@ export function renderSkeleton(skeleton) {
132
132
  return out;
133
133
  }
134
134
  /* ------------------------------------------------------------------ */
135
- /* Section renderers */
135
+ /* Section renderers */
136
136
  /* ------------------------------------------------------------------ */
137
137
  function renderHeader(skeleton) {
138
138
  const lines = [];
@@ -184,7 +184,7 @@ function renderReadmeSection(skeleton) {
184
184
  return `\n## README (first ${MAX_README_LINES} lines)\n${skeleton.readmeExcerpt}`;
185
185
  }
186
186
  /* ------------------------------------------------------------------ */
187
- /* Signal readers */
187
+ /* Signal readers */
188
188
  /* ------------------------------------------------------------------ */
189
189
  /**
190
190
  * Read the current branch from `.git/HEAD` WITHOUT spawning git. The
@@ -264,7 +264,7 @@ export function readPackageJson(cwd, ignore) {
264
264
  // package.json to .pugiignore (private template projects, customer
265
265
  // boilerplate under NDA) must not have it read into the agent
266
266
  // context even though the walker would have skipped it.
267
- // triple-review P1 (PR #380). The matcher is optional so direct callers
267
+ // triple-review P1 (PR). The matcher is optional so direct callers
268
268
  // (e.g. unit tests of readPackageJson in isolation) keep working.
269
269
  if (ignore?.isIgnored(pkgPath))
270
270
  return undefined;
@@ -298,7 +298,7 @@ export function readReadme(cwd, overridePath, ignore) {
298
298
  for (const candidate of candidates) {
299
299
  // Skip candidates the operator has explicitly ignored. A workspace
300
300
  // README that documents private contracts / customer data must
301
- // never land in the rendered skeleton. triple-review P1 (PR #380).
301
+ // never land in the rendered skeleton. triple-review P1 (PR).
302
302
  if (ignore?.isIgnored(candidate))
303
303
  continue;
304
304
  const raw = safeRead(candidate);
@@ -383,7 +383,7 @@ function walkTree(cwd, ignore) {
383
383
  }
384
384
  /**
385
385
  * Render a `DirNode` as a UTF-8 ASCII-tree string. Lines use the
386
- * Claude Code convention: `├──` / `└──` glyphs, two-space indents per
386
+ * the upstream tool convention: `├──` / `└──` glyphs, two-space indents per
387
387
  * level. Collapsed dirs land as `name/ (N dirs, M files)`.
388
388
  */
389
389
  function renderTree(root) {
@@ -408,12 +408,12 @@ function appendChildren(node, prefix, lines) {
408
408
  const isLast = i === children.length - 1;
409
409
  const glyph = isLast ? '└── ' : '├── ';
410
410
  lines.push(`${prefix}${glyph}${child.name}/${counts(child)}`);
411
- const nextPrefix = `${prefix}${isLast ? ' ' : '│ '}`;
411
+ const nextPrefix = `${prefix}${isLast ? ' ' : '│ '}`;
412
412
  appendChildren(child, nextPrefix, lines);
413
413
  }
414
414
  }
415
415
  /* ------------------------------------------------------------------ */
416
- /* Language histogram */
416
+ /* Language histogram */
417
417
  /* ------------------------------------------------------------------ */
418
418
  /**
419
419
  * Map a filename to a human-readable language label. Returns null for
@@ -495,7 +495,7 @@ export function topLanguages(histogram, topN) {
495
495
  return entries.slice(0, topN).map(([lang]) => lang);
496
496
  }
497
497
  /* ------------------------------------------------------------------ */
498
- /* Small helpers */
498
+ /* Small helpers */
499
499
  /* ------------------------------------------------------------------ */
500
500
  function safeRead(path) {
501
501
  // Stat first to keep the regular-file guard (Codex R2 P2): a FIFO or
@@ -0,0 +1,55 @@
1
+ export function estimateTokens(text) {
2
+ return Math.ceil(text.length / 4);
3
+ }
4
+ export function evictByLRU(entries, targetTokens) {
5
+ const sorted = [...entries].sort((a, b) => a.usedAt.localeCompare(b.usedAt));
6
+ const total = sorted.reduce((sum, e) => sum + e.tokensEst, 0);
7
+ if (total <= targetTokens) {
8
+ return {
9
+ evicted: [],
10
+ kept: sorted.map((e) => e.id),
11
+ freedTokens: 0,
12
+ };
13
+ }
14
+ const evicted = [];
15
+ const kept = [];
16
+ let freedTokens = 0;
17
+ let remaining = total;
18
+ for (const entry of sorted) {
19
+ if (remaining > targetTokens) {
20
+ evicted.push(entry.id);
21
+ freedTokens += entry.tokensEst;
22
+ remaining -= entry.tokensEst;
23
+ }
24
+ else {
25
+ kept.push(entry.id);
26
+ }
27
+ }
28
+ return { evicted, kept, freedTokens };
29
+ }
30
+ export function evictByContentClass(entries, rules, now = new Date()) {
31
+ const ruleByTool = new Map(rules.map((r) => [r.tool, r]));
32
+ const nowMs = now.getTime();
33
+ const evicted = [];
34
+ const kept = [];
35
+ let freedTokens = 0;
36
+ for (const entry of entries) {
37
+ const rule = ruleByTool.get(entry.tool);
38
+ if (!rule) {
39
+ kept.push(entry.id);
40
+ continue;
41
+ }
42
+ const ageMs = nowMs - new Date(entry.ts).getTime();
43
+ const tooOld = ageMs > rule.maxAgeMs;
44
+ const tooBig = entry.tokensEst > rule.maxTokens;
45
+ if (tooOld || tooBig) {
46
+ evicted.push(entry.id);
47
+ freedTokens += entry.tokensEst;
48
+ }
49
+ else {
50
+ kept.push(entry.id);
51
+ }
52
+ }
53
+ return { evicted, kept, freedTokens };
54
+ }
55
+ //# sourceMappingURL=tool-eviction.js.map