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

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
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 +4151 -489
  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,144 @@
1
+ /**
2
+ * Agent progress JSON schema — live-progress sprint .
3
+ *
4
+ * Long-running agents (spawned externally OR via `pugi /agent`) emit a
5
+ * single JSON document per agent to `~/.pugi/agent-progress/<id>.json`.
6
+ * `pugi jobs --watch` tails the directory via chokidar and re-renders
7
+ * an Ink TUI that mirrors the the upstream tool `/compact` visual pattern
8
+ * (header with elapsed + token counter, unicode progress bar, milestone
9
+ * list with done/active/pending status icons).
10
+ *
11
+ * Schema is intentionally optimistic — every numeric field is clamped
12
+ * by the writer/reader so a malformed document degrades to a partial
13
+ * card instead of crashing the watcher. The `pendingCount` /
14
+ * `completedCount` fields are pre-computed by the agent for the
15
+ * "… +N pending, M completed" footer; the renderer never re-counts
16
+ * (the agent may have collapsed milestone history to save bytes).
17
+ */
18
+ /**
19
+ * Validate an unknown payload as an `AgentProgress` document. Returns
20
+ * the typed value on success and a string error otherwise. We deliberately
21
+ * keep this hand-rolled (no zod) — every field is checked exactly once,
22
+ * the error message is human-readable, and zero runtime deps.
23
+ */
24
+ export function validateAgentProgress(value) {
25
+ if (typeof value !== 'object' || value === null) {
26
+ return { ok: false, error: 'progress payload must be a JSON object' };
27
+ }
28
+ const raw = value;
29
+ const requiredString = (field) => {
30
+ const v = raw[field];
31
+ return typeof v === 'string' && v.length > 0 ? v : null;
32
+ };
33
+ const agentId = requiredString('agentId');
34
+ if (!agentId)
35
+ return { ok: false, error: 'agentId required (non-empty string)' };
36
+ if (!/^[a-zA-Z0-9_-]+$/.test(agentId)) {
37
+ return { ok: false, error: 'agentId must match [a-zA-Z0-9_-]+ (filename-safe)' };
38
+ }
39
+ const agentType = requiredString('agentType');
40
+ if (!agentType)
41
+ return { ok: false, error: 'agentType required (non-empty string)' };
42
+ const task = requiredString('task');
43
+ if (!task)
44
+ return { ok: false, error: 'task required (non-empty string)' };
45
+ const startedAt = requiredString('startedAt');
46
+ if (!startedAt)
47
+ return { ok: false, error: 'startedAt required (ISO string)' };
48
+ if (Number.isNaN(Date.parse(startedAt))) {
49
+ return { ok: false, error: 'startedAt must be a parseable ISO timestamp' };
50
+ }
51
+ const lastUpdate = requiredString('lastUpdate');
52
+ if (!lastUpdate)
53
+ return { ok: false, error: 'lastUpdate required (ISO string)' };
54
+ if (Number.isNaN(Date.parse(lastUpdate))) {
55
+ return { ok: false, error: 'lastUpdate must be a parseable ISO timestamp' };
56
+ }
57
+ const elapsedMs = raw.elapsedMs;
58
+ if (typeof elapsedMs !== 'number' || !Number.isFinite(elapsedMs) || elapsedMs < 0) {
59
+ return { ok: false, error: 'elapsedMs required (non-negative number)' };
60
+ }
61
+ const percentComplete = raw.percentComplete;
62
+ if (typeof percentComplete !== 'number' ||
63
+ !Number.isFinite(percentComplete)) {
64
+ return { ok: false, error: 'percentComplete required (number 0..100)' };
65
+ }
66
+ const clampedPercent = Math.max(0, Math.min(100, percentComplete));
67
+ const status = raw.status;
68
+ if (status !== 'running' && status !== 'completed' && status !== 'failed') {
69
+ return { ok: false, error: 'status must be running | completed | failed' };
70
+ }
71
+ const currentStep = raw.currentStep;
72
+ if (typeof currentStep !== 'number' || currentStep < 0) {
73
+ return { ok: false, error: 'currentStep required (non-negative number)' };
74
+ }
75
+ const totalSteps = raw.totalSteps;
76
+ if (typeof totalSteps !== 'number' || totalSteps < 0) {
77
+ return { ok: false, error: 'totalSteps required (non-negative number)' };
78
+ }
79
+ const stepDescription = typeof raw.stepDescription === 'string'
80
+ ? raw.stepDescription
81
+ : '';
82
+ const milestonesRaw = raw.milestones;
83
+ if (!Array.isArray(milestonesRaw)) {
84
+ return { ok: false, error: 'milestones required (array, may be empty)' };
85
+ }
86
+ const milestones = [];
87
+ for (let i = 0; i < milestonesRaw.length; i += 1) {
88
+ const m = milestonesRaw[i];
89
+ if (typeof m !== 'object' || m === null) {
90
+ return { ok: false, error: `milestones[${i}] must be an object` };
91
+ }
92
+ const mr = m;
93
+ const label = typeof mr.label === 'string' ? mr.label : '';
94
+ if (!label) {
95
+ return { ok: false, error: `milestones[${i}].label required` };
96
+ }
97
+ const mStatus = mr.status;
98
+ if (mStatus !== 'done' && mStatus !== 'active' && mStatus !== 'pending') {
99
+ return {
100
+ ok: false,
101
+ error: `milestones[${i}].status must be done | active | pending`,
102
+ };
103
+ }
104
+ const ts = typeof mr.ts === 'string' ? mr.ts : undefined;
105
+ milestones.push({ label, status: mStatus, ts });
106
+ }
107
+ const tokensUsed = typeof raw.tokensUsed === 'number' && Number.isFinite(raw.tokensUsed)
108
+ ? Math.max(0, raw.tokensUsed)
109
+ : undefined;
110
+ const pendingCount = typeof raw.pendingCount === 'number' && Number.isFinite(raw.pendingCount)
111
+ ? Math.max(0, Math.floor(raw.pendingCount))
112
+ : undefined;
113
+ const completedCount = typeof raw.completedCount === 'number' && Number.isFinite(raw.completedCount)
114
+ ? Math.max(0, Math.floor(raw.completedCount))
115
+ : undefined;
116
+ return {
117
+ ok: true,
118
+ value: {
119
+ agentId,
120
+ agentType,
121
+ task,
122
+ startedAt,
123
+ lastUpdate,
124
+ elapsedMs,
125
+ tokensUsed,
126
+ percentComplete: clampedPercent,
127
+ status,
128
+ currentStep,
129
+ totalSteps,
130
+ stepDescription,
131
+ milestones,
132
+ pendingCount,
133
+ completedCount,
134
+ },
135
+ };
136
+ }
137
+ /**
138
+ * Default directory the writer and watcher share. Lives under the
139
+ * project workspace by convention so worktree-isolated agents can emit
140
+ * progress without crossing tenant boundaries. Operators can override
141
+ * via `PUGI_AGENT_PROGRESS_DIR` env var.
142
+ */
143
+ export const DEFAULT_AGENT_PROGRESS_DIRNAME = '.pugi/agent-progress';
144
+ //# sourceMappingURL=schema.js.map
@@ -0,0 +1,101 @@
1
+ /**
2
+ * Agent progress writer — atomic file ops so the chokidar watcher
3
+ * never reads a half-written JSON document.
4
+ *
5
+ * Pattern (POSIX-portable):
6
+ * 1. Build the canonical JSON body.
7
+ * 2. Write to `<dir>/<agentId>.json.tmp-<pid>-<seq>`.
8
+ * 3. Rename (atomic on the same filesystem) to `<dir>/<agentId>.json`.
9
+ * 4. Cleanup is a no-op on success; on failure the caller bubbles the
10
+ * error and the .tmp file is removed best-effort.
11
+ *
12
+ * The auto-cleanup of completed entries lives in `cleanup.ts` (commit 4);
13
+ * the writer here is the minimal surface the agent prompt boilerplate
14
+ * needs to copy + paste.
15
+ */
16
+ import { existsSync, mkdirSync, renameSync, rmSync, writeFileSync } from 'node:fs';
17
+ import { homedir } from 'node:os';
18
+ import { dirname, join, resolve } from 'node:path';
19
+ import { DEFAULT_AGENT_PROGRESS_DIRNAME, validateAgentProgress, } from './schema.js';
20
+ let writeSequence = 0;
21
+ /**
22
+ * Resolve the effective progress directory. Public so the watcher can
23
+ * share the same resolution rules without duplicating env-var logic.
24
+ */
25
+ export function resolveProgressDir(override) {
26
+ const candidate = override
27
+ ?? process.env.PUGI_AGENT_PROGRESS_DIR
28
+ ?? join(process.cwd(), DEFAULT_AGENT_PROGRESS_DIRNAME);
29
+ if (candidate.startsWith('~/')) {
30
+ return join(homedir(), candidate.slice(2));
31
+ }
32
+ return resolve(candidate);
33
+ }
34
+ /**
35
+ * Write a progress document atomically.
36
+ *
37
+ * Throws on validation failure so the caller is loud about a schema
38
+ * regression. File-system errors propagate untouched.
39
+ */
40
+ export function writeProgress(progress, options = {}) {
41
+ const validation = validateAgentProgress(progress);
42
+ if (!validation.ok) {
43
+ throw new Error(`invalid agent-progress payload: ${validation.error}`);
44
+ }
45
+ const dir = resolveProgressDir(options.dir);
46
+ if (!existsSync(dir)) {
47
+ mkdirSync(dir, { recursive: true });
48
+ }
49
+ const finalPath = join(dir, `${progress.agentId}.json`);
50
+ writeSequence += 1;
51
+ const tmpPath = `${finalPath}.tmp-${process.pid}-${writeSequence}`;
52
+ const body = `${JSON.stringify(progress, null, 2)}\n`;
53
+ try {
54
+ // mkdirSync above may have created `dir`, but a parallel agent could
55
+ // delete it between the existsSync and writeFile — ensure the parent
56
+ // of the tmp file exists right before the write.
57
+ const parent = dirname(tmpPath);
58
+ if (!existsSync(parent)) {
59
+ mkdirSync(parent, { recursive: true });
60
+ }
61
+ writeFileSync(tmpPath, body, 'utf8');
62
+ renameSync(tmpPath, finalPath);
63
+ }
64
+ catch (err) {
65
+ try {
66
+ rmSync(tmpPath, { force: true });
67
+ }
68
+ catch {
69
+ // best-effort cleanup; surface the original error
70
+ }
71
+ throw err;
72
+ }
73
+ return { path: finalPath };
74
+ }
75
+ /**
76
+ * Build a fresh progress document with sensible defaults. Mostly a
77
+ * convenience for tests + the agent prompt boilerplate — the writer
78
+ * does NOT call this implicitly because agents have richer context
79
+ * about their own milestone state.
80
+ */
81
+ export function makeInitialProgress(input) {
82
+ const now = input.now ?? Date.now;
83
+ const iso = new Date(now()).toISOString();
84
+ return {
85
+ agentId: input.agentId,
86
+ agentType: input.agentType,
87
+ task: input.task,
88
+ startedAt: iso,
89
+ lastUpdate: iso,
90
+ elapsedMs: 0,
91
+ percentComplete: 0,
92
+ status: 'running',
93
+ currentStep: 0,
94
+ totalSteps: Math.max(0, Math.floor(input.totalSteps)),
95
+ stepDescription: '',
96
+ milestones: input.milestones ?? [],
97
+ pendingCount: input.milestones?.filter((m) => m.status === 'pending').length,
98
+ completedCount: input.milestones?.filter((m) => m.status === 'done').length,
99
+ };
100
+ }
101
+ //# sourceMappingURL=writer.js.map
@@ -0,0 +1,330 @@
1
+ /**
2
+ * Adaptive router — primitive #4 of the 9-layer RAG architecture gap
3
+ * analysis (docs/research/2026-05-31-9-layer-rag-architecture-.md).
4
+ *
5
+ * Picks the best source (RAG workspace / web fetch / code-intel / cache
6
+ * tier / agent persona / tool) for a given query by asking an LLM to
7
+ * rank candidates from a caller-supplied catalog. The router is pure
8
+ * logic — it never opens a network connection itself. The caller
9
+ * injects a `transport` callback that maps a prompt string to a model
10
+ * response string, which keeps Anvil-engine wiring (Hiroshi persona,
11
+ * auth, retry, telemetry) out of the primitive and makes tests
12
+ * deterministic.
13
+ *
14
+ * Failure model is fail-closed to a heuristic pick. Every malformed
15
+ * response, transport throw, schema violation, empty catalog, or
16
+ * unknown source id collapses to a deterministic ranking based on
17
+ * `kind` priority (cache > agent > everything-else, in catalog order).
18
+ * The caller inspects `usedFallback` plus `reason` and decides whether
19
+ * the degradation is acceptable for the active route. Routing is never
20
+ * load-bearing for correctness — the caller can always fall back to a
21
+ * monolithic default source.
22
+ *
23
+ * Composition with sibling primitives:
24
+ * - #3 query-decomposer feeds an ordered SubQuery list. The engine
25
+ * consumer will call `chooseRoute` once per SubQuery to pick the
26
+ * per-step source. That binding lives in the engine wire-up PR, not
27
+ * here.
28
+ * - #5 pre-flight token estimator may pre-set `tierBudget` to gate
29
+ * out 'expensive' sources before the LLM call is made.
30
+ *
31
+ * Out of scope (separate PRs):
32
+ * - Engine wire-up that binds Hiroshi persona to `transport`.
33
+ * - Source-catalog registry — caller supplies the catalog per call.
34
+ * - Per-source cost telemetry (separate follow-up).
35
+ * - Streaming partial-ranks during LLM emission.
36
+ */
37
+ import { estimateTokens } from '../compact/token-counter.js';
38
+ /** Whitelisted source kinds. New kinds require a corresponding catalog migration. */
39
+ const SOURCE_KIND_VALUES = ['rag', 'web', 'code', 'cache', 'agent', 'tool'];
40
+ const DEFAULT_MAX_FALLBACKS = 3;
41
+ const DEFAULT_MAX_INPUT_TOKENS = 4000;
42
+ /** Hard upper bound on pickIds we accept from the LLM. 1 primary + 3 fallbacks. */
43
+ const HARD_MAX_PICKS = 4;
44
+ /** Trim the user query before substitution. Keeps router prompt within sane bounds. */
45
+ const PROMPT_QUERY_HARD_CAP_CHARS = 2000;
46
+ /**
47
+ * Prompt template embedded as a constant so the primitive has zero
48
+ * filesystem dependencies. The `{contextHint}`, `{query}`, and
49
+ * `{candidates}` markers are substituted at call time. The instruction
50
+ * explicitly forbids markdown fences and prose around the JSON, which
51
+ * trims the most common parse failures observed with mid-tier models.
52
+ */
53
+ export const ADAPTIVE_ROUTER_PROMPT_TEMPLATE = [
54
+ 'You are an adaptive router for a software-engineering AI assistant.',
55
+ "Pick the best source(s) from the candidate list to answer the user's query.",
56
+ '',
57
+ 'Return JSON only matching this schema:',
58
+ '{ "pickIds": ["source-id-1", "source-id-2", ...], "confidence": 0.85, "reasoning": "..." }',
59
+ '',
60
+ 'Rules:',
61
+ '- Order pickIds best to worst (first = primary).',
62
+ '- Max 4 ids total (1 primary + up to 3 fallbacks).',
63
+ '- confidence in [0, 1].',
64
+ '- reasoning is one sentence under 200 chars.',
65
+ '- No prose around JSON. No markdown fences.',
66
+ '',
67
+ 'CONTEXT: {contextHint}',
68
+ 'QUERY: {query}',
69
+ '',
70
+ 'CANDIDATES:',
71
+ '{candidates}',
72
+ ].join('\n');
73
+ /**
74
+ * Pick the best source(s) for a query. Never throws.
75
+ *
76
+ * Every internal failure mode collapses to a heuristic fallback that
77
+ * still produces a usable `RouteDecision`. Caller inspects
78
+ * `usedFallback` and `reason` to decide whether to surface a warning or
79
+ * treat the routing as opaque.
80
+ */
81
+ export async function chooseRoute(opts) {
82
+ const filtered = filterCatalog(opts.catalog, opts.tierBudget);
83
+ const maxFallbacks = clampMaxFallbacks(opts.maxFallbacks);
84
+ const maxInputTokens = clampMaxInputTokens(opts.maxInputTokens);
85
+ if (filtered.length === 0) {
86
+ return emptyCatalogFallback();
87
+ }
88
+ if (filtered.length === 1) {
89
+ // Single available source — no need to round-trip through the LLM.
90
+ const only = filtered[0];
91
+ if (!only)
92
+ return emptyCatalogFallback();
93
+ return {
94
+ primary: only,
95
+ fallbacks: [],
96
+ confidence: 1,
97
+ usedFallback: false,
98
+ };
99
+ }
100
+ const safeQuery = truncateToTokenBudget(opts.query.trim(), maxInputTokens);
101
+ const prompt = buildPrompt(safeQuery, opts.contextHint ?? 'unknown', filtered);
102
+ let raw;
103
+ try {
104
+ raw = await opts.transport(prompt);
105
+ }
106
+ catch {
107
+ return heuristicFallback(filtered, maxFallbacks, 'transport-error');
108
+ }
109
+ const parsed = parseAndValidate(raw, filtered);
110
+ if (!parsed.ok) {
111
+ return heuristicFallback(filtered, maxFallbacks, parsed.reason);
112
+ }
113
+ const ranked = parsed.sources;
114
+ const first = ranked[0];
115
+ if (!first) {
116
+ return heuristicFallback(filtered, maxFallbacks, 'empty-pick-ids');
117
+ }
118
+ const fallbacks = ranked.slice(1, 1 + maxFallbacks);
119
+ return {
120
+ primary: first,
121
+ fallbacks,
122
+ confidence: clampConfidence(parsed.confidence),
123
+ reasoning: parsed.reasoning,
124
+ usedFallback: false,
125
+ };
126
+ }
127
+ // -----------------------------------------------------------------------
128
+ // Internals
129
+ // -----------------------------------------------------------------------
130
+ function filterCatalog(catalog, tierBudget) {
131
+ const out = [];
132
+ for (const source of catalog) {
133
+ if (!source.available)
134
+ continue;
135
+ if (tierBudget === 'cheap' && source.costHint === 'expensive')
136
+ continue;
137
+ out.push(source);
138
+ }
139
+ return out;
140
+ }
141
+ function buildPrompt(query, contextHint, sources) {
142
+ const trimmedQuery = query.length > PROMPT_QUERY_HARD_CAP_CHARS
143
+ ? query.slice(0, PROMPT_QUERY_HARD_CAP_CHARS)
144
+ : query;
145
+ const candidateLines = sources
146
+ .map((s) => `- ${s.id} (${s.kind}, ${s.costHint}, ${s.latencyHint}): ${s.description}`)
147
+ .join('\n');
148
+ return ADAPTIVE_ROUTER_PROMPT_TEMPLATE
149
+ .replace('{contextHint}', contextHint)
150
+ .replace('{query}', trimmedQuery)
151
+ .replace('{candidates}', candidateLines);
152
+ }
153
+ function parseAndValidate(raw, catalog) {
154
+ let parsed;
155
+ try {
156
+ parsed = JSON.parse(stripFences(raw));
157
+ }
158
+ catch {
159
+ return { ok: false, reason: 'parse-failed' };
160
+ }
161
+ if (!isRecord(parsed)) {
162
+ return { ok: false, reason: 'schema-violation' };
163
+ }
164
+ const pickIdsRaw = parsed.pickIds;
165
+ if (!Array.isArray(pickIdsRaw)) {
166
+ return { ok: false, reason: 'schema-violation' };
167
+ }
168
+ if (pickIdsRaw.length === 0) {
169
+ return { ok: false, reason: 'empty-pick-ids' };
170
+ }
171
+ const byId = new Map(catalog.map((s) => [s.id, s]));
172
+ const seen = new Set();
173
+ const resolved = [];
174
+ for (const candidate of pickIdsRaw) {
175
+ if (typeof candidate !== 'string') {
176
+ return { ok: false, reason: 'schema-violation' };
177
+ }
178
+ if (seen.has(candidate)) {
179
+ return { ok: false, reason: 'duplicate-pick-id' };
180
+ }
181
+ const source = byId.get(candidate);
182
+ if (!source) {
183
+ return { ok: false, reason: 'unknown-source-id' };
184
+ }
185
+ seen.add(candidate);
186
+ resolved.push(source);
187
+ // Cap at the hard ceiling so a malicious / runaway LLM cannot make us allocate unbounded fallbacks.
188
+ if (resolved.length >= HARD_MAX_PICKS)
189
+ break;
190
+ }
191
+ const confidenceRaw = parsed.confidence;
192
+ // Confidence is intentionally lenient — we coerce non-numbers to 0.5
193
+ // and clamp the rest. Out-of-range is NOT a fallback trigger per the
194
+ // spec: pickIds is the load-bearing field.
195
+ const confidence = typeof confidenceRaw === 'number' && Number.isFinite(confidenceRaw)
196
+ ? confidenceRaw
197
+ : 0.5;
198
+ const reasoningRaw = parsed.reasoning;
199
+ const reasoning = typeof reasoningRaw === 'string' && reasoningRaw.trim().length > 0
200
+ ? reasoningRaw.trim()
201
+ : undefined;
202
+ return {
203
+ ok: true,
204
+ sources: resolved,
205
+ confidence,
206
+ ...(reasoning !== undefined ? { reasoning } : {}),
207
+ };
208
+ }
209
+ /**
210
+ * Heuristic ranking applied when the LLM call fails. Priority order:
211
+ *
212
+ * 1. `kind === 'cache'` first (always cheapest and safe to try).
213
+ * 2. `kind === 'agent'` next (persona has its own routing intelligence).
214
+ * 3. Everything else in the catalog's natural order.
215
+ *
216
+ * Within each bucket the catalog's natural order acts as a stable
217
+ * tiebreaker — the caller controls the priority by ordering the
218
+ * catalog deliberately.
219
+ */
220
+ function heuristicFallback(catalog, maxFallbacks, reason) {
221
+ const ranked = heuristicRank(catalog);
222
+ const first = ranked[0];
223
+ // Defensive: caller guarantees catalog.length >= 1 by the time we
224
+ // get here (the `filtered.length === 0` short-circuit runs upstream),
225
+ // but the type checker needs the guard.
226
+ if (!first)
227
+ return emptyCatalogFallback();
228
+ const fallbacks = ranked.slice(1, 1 + maxFallbacks);
229
+ return {
230
+ primary: first,
231
+ fallbacks,
232
+ confidence: 0,
233
+ usedFallback: true,
234
+ reason,
235
+ };
236
+ }
237
+ function heuristicRank(catalog) {
238
+ const cache = [];
239
+ const agents = [];
240
+ const rest = [];
241
+ for (const source of catalog) {
242
+ if (source.kind === 'cache')
243
+ cache.push(source);
244
+ else if (source.kind === 'agent')
245
+ agents.push(source);
246
+ else
247
+ rest.push(source);
248
+ }
249
+ return [...cache, ...agents, ...rest];
250
+ }
251
+ /**
252
+ * Synthetic fallback used only when the filtered catalog is empty.
253
+ * The caller MUST treat `usedFallback === true` as a signal to skip
254
+ * the routed step — the `primary` here is a placeholder so the return
255
+ * type stays non-nullable.
256
+ */
257
+ function emptyCatalogFallback() {
258
+ return {
259
+ primary: {
260
+ id: 'unknown',
261
+ kind: 'tool',
262
+ description: 'No source available for this query.',
263
+ costHint: 'cheap',
264
+ latencyHint: 'sub-second',
265
+ available: false,
266
+ },
267
+ fallbacks: [],
268
+ confidence: 0,
269
+ usedFallback: true,
270
+ reason: 'empty-catalog',
271
+ };
272
+ }
273
+ function clampMaxFallbacks(value) {
274
+ if (typeof value !== 'number' || !Number.isFinite(value) || value < 0) {
275
+ return DEFAULT_MAX_FALLBACKS;
276
+ }
277
+ return Math.floor(value);
278
+ }
279
+ function clampMaxInputTokens(value) {
280
+ if (typeof value !== 'number' || !Number.isFinite(value) || value < 1) {
281
+ return DEFAULT_MAX_INPUT_TOKENS;
282
+ }
283
+ return Math.floor(value);
284
+ }
285
+ function clampConfidence(value) {
286
+ if (!Number.isFinite(value))
287
+ return 0.5;
288
+ if (value < 0)
289
+ return 0;
290
+ if (value > 1)
291
+ return 1;
292
+ return value;
293
+ }
294
+ /**
295
+ * Truncate input to the token budget using the existing chars/4
296
+ * heuristic. We slice by characters proportional to the budget rather
297
+ * than binary-searching `estimateTokens`, because the heuristic is
298
+ * monotonic in length and an over-shoot is acceptable — the prompt also
299
+ * carries the static template overhead.
300
+ */
301
+ function truncateToTokenBudget(text, maxTokens) {
302
+ const current = estimateTokens(text);
303
+ if (current <= maxTokens)
304
+ return text;
305
+ const ratio = maxTokens / current;
306
+ const targetLength = Math.max(1, Math.floor(text.length * ratio));
307
+ return text.slice(0, targetLength);
308
+ }
309
+ /**
310
+ * Tolerate the most common mid-tier-model deviation: wrapping JSON in
311
+ * a fenced code block. We strip a single leading and trailing fence
312
+ * marker when both are present; otherwise the raw text is returned
313
+ * unchanged and `JSON.parse` will reject it through the normal path.
314
+ */
315
+ function stripFences(raw) {
316
+ const trimmed = raw.trim();
317
+ if (!trimmed.startsWith('```'))
318
+ return trimmed;
319
+ const firstNewline = trimmed.indexOf('\n');
320
+ if (firstNewline < 0)
321
+ return trimmed;
322
+ const closing = trimmed.lastIndexOf('```');
323
+ if (closing <= firstNewline)
324
+ return trimmed;
325
+ return trimmed.slice(firstNewline + 1, closing).trim();
326
+ }
327
+ function isRecord(value) {
328
+ return typeof value === 'object' && value !== null && !Array.isArray(value);
329
+ }
330
+ //# sourceMappingURL=adaptive-router.js.map