@pugi/cli 0.1.0-beta.99 → 1.0.0-alpha.1

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/LICENSE +1 -1
  2. package/README.md +11 -191
  3. package/bin/pugi +8 -0
  4. package/package.json +15 -71
  5. package/postinstall.mjs +31 -0
  6. package/CHANGELOG.md +0 -132
  7. package/THIRD_PARTY_NOTICES.md +0 -40
  8. package/assets/pugi-mascot.ansi +0 -16
  9. package/assets/pugi-prozr2-mascot.ansi +0 -9
  10. package/bin/run.js +0 -34
  11. package/dist/commands/deploy.js +0 -439
  12. package/dist/commands/flatten.js +0 -191
  13. package/dist/commands/jobs-watch.js +0 -201
  14. package/dist/commands/jobs.js +0 -260
  15. package/dist/commands/retro.js +0 -210
  16. package/dist/commands/smoke.js +0 -133
  17. package/dist/core/agent-progress/cleanup.js +0 -134
  18. package/dist/core/agent-progress/schema.js +0 -144
  19. package/dist/core/agent-progress/writer.js +0 -101
  20. package/dist/core/agents/adaptive-router.js +0 -330
  21. package/dist/core/agents/loader.js +0 -104
  22. package/dist/core/agents/query-decomposer.js +0 -297
  23. package/dist/core/agents/registry.js +0 -69
  24. package/dist/core/approvals/shortcut-resolver.js +0 -98
  25. package/dist/core/artifact-chain/dispatcher.js +0 -148
  26. package/dist/core/artifact-chain/exporter.js +0 -164
  27. package/dist/core/artifact-chain/state.js +0 -243
  28. package/dist/core/artifact-chain/steps.js +0 -169
  29. package/dist/core/ask-user/question.js +0 -92
  30. package/dist/core/audit/audit-trail.js +0 -275
  31. package/dist/core/auth/ensure-authenticated.js +0 -129
  32. package/dist/core/auth/env-provider.js +0 -238
  33. package/dist/core/auto-open-browser.js +0 -128
  34. package/dist/core/auto-update/channels.js +0 -122
  35. package/dist/core/auto-update/checker.js +0 -241
  36. package/dist/core/auto-update/state.js +0 -235
  37. package/dist/core/bare-mode/index.js +0 -107
  38. package/dist/core/bash/redirect.js +0 -281
  39. package/dist/core/bash-classifier.js +0 -1397
  40. package/dist/core/checkpoint/resumer.js +0 -149
  41. package/dist/core/checkpoint/rewinder.js +0 -291
  42. package/dist/core/checkpoints/shadow-git.js +0 -670
  43. package/dist/core/citations/parser.js +0 -109
  44. package/dist/core/classifier/yolo-classifier.js +0 -88
  45. package/dist/core/clipboard.js +0 -70
  46. package/dist/core/codegraph/decision-store.js +0 -248
  47. package/dist/core/codegraph/detect-repo.js +0 -459
  48. package/dist/core/codegraph/install.js +0 -134
  49. package/dist/core/codegraph/offer-hook.js +0 -220
  50. package/dist/core/compact/auto-trigger.js +0 -96
  51. package/dist/core/compact/buffer-rewriter.js +0 -115
  52. package/dist/core/compact/summarizer.js +0 -208
  53. package/dist/core/compact/token-counter.js +0 -108
  54. package/dist/core/consensus/anvil-fanout.js +0 -276
  55. package/dist/core/consensus/diff-capture.js +0 -491
  56. package/dist/core/consensus/rubric.js +0 -233
  57. package/dist/core/context/builder.js +0 -114
  58. package/dist/core/context/compaction-events.js +0 -99
  59. package/dist/core/context/compaction.js +0 -602
  60. package/dist/core/context/index.js +0 -28
  61. package/dist/core/context/invariants.js +0 -250
  62. package/dist/core/context/markdown-loader.js +0 -288
  63. package/dist/core/context/markdown-traverse.js +0 -255
  64. package/dist/core/context/pugiignore.js +0 -316
  65. package/dist/core/context/repo-skeleton.js +0 -533
  66. package/dist/core/context/tool-eviction.js +0 -55
  67. package/dist/core/context/watcher.js +0 -342
  68. package/dist/core/context/working-set.js +0 -165
  69. package/dist/core/coordinator/agent-tools.js +0 -77
  70. package/dist/core/coordinator/agent-toolset.js +0 -65
  71. package/dist/core/coordinator/fsm.js +0 -73
  72. package/dist/core/coordinator/mode-fsm.js +0 -70
  73. package/dist/core/cost/rate-card.js +0 -129
  74. package/dist/core/cost/tracker.js +0 -221
  75. package/dist/core/credentials.js +0 -355
  76. package/dist/core/cron/scheduler.js +0 -138
  77. package/dist/core/denial-tracking/index.js +0 -8
  78. package/dist/core/denial-tracking/state.js +0 -264
  79. package/dist/core/diagnostics/probe-runner.js +0 -93
  80. package/dist/core/diagnostics/probes/api.js +0 -46
  81. package/dist/core/diagnostics/probes/auth.js +0 -93
  82. package/dist/core/diagnostics/probes/bare-mode.js +0 -42
  83. package/dist/core/diagnostics/probes/cli-version.js +0 -127
  84. package/dist/core/diagnostics/probes/config.js +0 -72
  85. package/dist/core/diagnostics/probes/denial-tracking.js +0 -57
  86. package/dist/core/diagnostics/probes/disk.js +0 -81
  87. package/dist/core/diagnostics/probes/engine-live.js +0 -46
  88. package/dist/core/diagnostics/probes/git.js +0 -65
  89. package/dist/core/diagnostics/probes/hooks.js +0 -118
  90. package/dist/core/diagnostics/probes/mcp.js +0 -75
  91. package/dist/core/diagnostics/probes/node.js +0 -59
  92. package/dist/core/diagnostics/probes/pnpm.js +0 -36
  93. package/dist/core/diagnostics/probes/pugi-md.js +0 -89
  94. package/dist/core/diagnostics/probes/sandbox.js +0 -72
  95. package/dist/core/diagnostics/probes/session.js +0 -74
  96. package/dist/core/diagnostics/probes/status-snapshot.js +0 -488
  97. package/dist/core/diagnostics/probes/workspace.js +0 -63
  98. package/dist/core/diagnostics/types.js +0 -70
  99. package/dist/core/dispatch/cache-cleanup.js +0 -197
  100. package/dist/core/dispatch/cache-handoff.js +0 -295
  101. package/dist/core/edits/apply-patch-layer-e.js +0 -189
  102. package/dist/core/edits/dispatch.js +0 -511
  103. package/dist/core/edits/format-detector.js +0 -260
  104. package/dist/core/edits/format-matrix.js +0 -26
  105. package/dist/core/edits/fuzzy-ladder.js +0 -650
  106. package/dist/core/edits/index.js +0 -19
  107. package/dist/core/edits/journal.js +0 -199
  108. package/dist/core/edits/layer-a-apply.js +0 -217
  109. package/dist/core/edits/layer-a-fuzzy-apply.js +0 -198
  110. package/dist/core/edits/layer-b-apply.js +0 -211
  111. package/dist/core/edits/layer-c-apply.js +0 -160
  112. package/dist/core/edits/layer-d-ast.js +0 -572
  113. package/dist/core/edits/marker-parser.js +0 -401
  114. package/dist/core/edits/security-gate.js +0 -223
  115. package/dist/core/edits/verify-hook.js +0 -273
  116. package/dist/core/edits/worktree.js +0 -322
  117. package/dist/core/engine/adapter-runner.js +0 -8
  118. package/dist/core/engine/anvil-client.js +0 -344
  119. package/dist/core/engine/auto-compact.js +0 -179
  120. package/dist/core/engine/budgets.js +0 -195
  121. package/dist/core/engine/context-prefix.js +0 -155
  122. package/dist/core/engine/index.js +0 -12
  123. package/dist/core/engine/intensity.js +0 -163
  124. package/dist/core/engine/intent.js +0 -260
  125. package/dist/core/engine/native-pugi.js +0 -1616
  126. package/dist/core/engine/noop.js +0 -27
  127. package/dist/core/engine/prompts.js +0 -236
  128. package/dist/core/engine/strip-internal-fields.js +0 -124
  129. package/dist/core/engine/tool-bridge.js +0 -2173
  130. package/dist/core/engine/verification-patterns.js +0 -195
  131. package/dist/core/evaluation/golden-dataset.js +0 -293
  132. package/dist/core/feedback/queue.js +0 -177
  133. package/dist/core/feedback/submitter.js +0 -145
  134. package/dist/core/file-cache.js +0 -141
  135. package/dist/core/flatten/flatten-repo.js +0 -439
  136. package/dist/core/format/osc8-link.js +0 -28
  137. package/dist/core/hook-chains.js +0 -392
  138. package/dist/core/hooks/citation-verify-hook.js +0 -138
  139. package/dist/core/hooks/citation-verify.js +0 -112
  140. package/dist/core/hooks/events.js +0 -46
  141. package/dist/core/hooks/index.js +0 -15
  142. package/dist/core/hooks/registry.js +0 -216
  143. package/dist/core/hooks/runner.js +0 -236
  144. package/dist/core/hooks/v2/event-emitter.js +0 -115
  145. package/dist/core/hooks/v2/executor.js +0 -282
  146. package/dist/core/hooks/v2/index.js +0 -25
  147. package/dist/core/hooks/v2/lifecycle.js +0 -104
  148. package/dist/core/hooks/v2/loader.js +0 -216
  149. package/dist/core/hooks/v2/matcher.js +0 -125
  150. package/dist/core/hooks/v2/trust.js +0 -143
  151. package/dist/core/hooks/v2/types.js +0 -86
  152. package/dist/core/hooks/worktree-events.js +0 -158
  153. package/dist/core/hooks.js +0 -415
  154. package/dist/core/image/renderer.js +0 -71
  155. package/dist/core/index-store.js +0 -260
  156. package/dist/core/init/detector.js +0 -582
  157. package/dist/core/init/template-renderer.js +0 -242
  158. package/dist/core/jobs/registry.js +0 -462
  159. package/dist/core/ledger/results-tsv.js +0 -142
  160. package/dist/core/log-discipline/stdout-redirect.js +0 -51
  161. package/dist/core/lsp/cache.js +0 -105
  162. package/dist/core/lsp/client.js +0 -1229
  163. package/dist/core/lsp/language-detect.js +0 -66
  164. package/dist/core/lsp/post-edit-diagnostics.js +0 -171
  165. package/dist/core/lsp/server-detect.js +0 -173
  166. package/dist/core/lsp/symbol-cache.js +0 -162
  167. package/dist/core/lsp/symbol-tools.js +0 -664
  168. package/dist/core/mcp/client.js +0 -385
  169. package/dist/core/mcp/http-server.js +0 -553
  170. package/dist/core/mcp/orchestrator-config.js +0 -192
  171. package/dist/core/mcp/orchestrator-tools.js +0 -806
  172. package/dist/core/mcp/permission.js +0 -190
  173. package/dist/core/mcp/registry.js +0 -193
  174. package/dist/core/mcp/server-tools.js +0 -219
  175. package/dist/core/mcp/server.js +0 -397
  176. package/dist/core/mcp/trust.js +0 -91
  177. package/dist/core/memory/dual-write.js +0 -416
  178. package/dist/core/memory/passive-extract.js +0 -130
  179. package/dist/core/memory/phase1-kinds.js +0 -20
  180. package/dist/core/memory/secret-scanner.js +0 -304
  181. package/dist/core/memory-sync/queue.js +0 -170
  182. package/dist/core/metrics/extract.js +0 -113
  183. package/dist/core/modes/roo-modes.js +0 -68
  184. package/dist/core/onboarding/ensure-initialized.js +0 -133
  185. package/dist/core/onboarding/marker.js +0 -111
  186. package/dist/core/onboarding/telemetry-state.js +0 -108
  187. package/dist/core/output-style/presets.js +0 -176
  188. package/dist/core/output-style/state.js +0 -185
  189. package/dist/core/path-security.js +0 -345
  190. package/dist/core/permission.js +0 -369
  191. package/dist/core/permissions/auto-classifier.js +0 -124
  192. package/dist/core/permissions/bash-parser.js +0 -371
  193. package/dist/core/permissions/circuit-breaker.js +0 -83
  194. package/dist/core/permissions/constrained-edit.js +0 -91
  195. package/dist/core/permissions/gate.js +0 -278
  196. package/dist/core/permissions/index.js +0 -20
  197. package/dist/core/permissions/mode.js +0 -174
  198. package/dist/core/permissions/network-egress.js +0 -137
  199. package/dist/core/permissions/state.js +0 -241
  200. package/dist/core/permissions/tool-class.js +0 -107
  201. package/dist/core/plan-mode/ui-state.js +0 -51
  202. package/dist/core/plans/plan-artifact.js +0 -721
  203. package/dist/core/policy-limits/etag-store.js +0 -122
  204. package/dist/core/prd-check/parser.js +0 -215
  205. package/dist/core/prd-check/reporter.js +0 -127
  206. package/dist/core/prd-check/session-review.js +0 -557
  207. package/dist/core/prd-check/verifiers.js +0 -223
  208. package/dist/core/prompt-cache/client-cache.js +0 -99
  209. package/dist/core/prompts/assembly.js +0 -29
  210. package/dist/core/prompts/registry.js +0 -364
  211. package/dist/core/pugi-gitignore.js +0 -52
  212. package/dist/core/pugi-md/cc-compat-rules.js +0 -735
  213. package/dist/core/pugi-md/context-injector.js +0 -76
  214. package/dist/core/pugi-md/walk-up.js +0 -207
  215. package/dist/core/python/uv-installer.js +0 -270
  216. package/dist/core/python/uv-resolver.js +0 -83
  217. package/dist/core/rate-limit/narrator.js +0 -146
  218. package/dist/core/recipes/cli-types.js +0 -20
  219. package/dist/core/recipes/loader.js +0 -103
  220. package/dist/core/recipes/runner.js +0 -345
  221. package/dist/core/recipes/schema.js +0 -587
  222. package/dist/core/release-notes/parser.js +0 -241
  223. package/dist/core/release-notes/state.js +0 -116
  224. package/dist/core/repl/ask.js +0 -512
  225. package/dist/core/repl/cancellation.js +0 -98
  226. package/dist/core/repl/cap-warning.js +0 -91
  227. package/dist/core/repl/clipboard-read.js +0 -174
  228. package/dist/core/repl/dispatch-fsm.js +0 -220
  229. package/dist/core/repl/engine-bridge.js +0 -303
  230. package/dist/core/repl/history-search.js +0 -175
  231. package/dist/core/repl/history.js +0 -182
  232. package/dist/core/repl/kill-ring.js +0 -138
  233. package/dist/core/repl/model-pricing.js +0 -135
  234. package/dist/core/repl/privacy-banner.js +0 -71
  235. package/dist/core/repl/session.js +0 -4962
  236. package/dist/core/repl/slash-commands.js +0 -747
  237. package/dist/core/repl/store/index.js +0 -12
  238. package/dist/core/repl/store/jsonl-log.js +0 -321
  239. package/dist/core/repl/store/lockfile.js +0 -155
  240. package/dist/core/repl/store/session-store.js +0 -821
  241. package/dist/core/repl/store/types.js +0 -44
  242. package/dist/core/repl/store/uuid-v7.js +0 -68
  243. package/dist/core/repl/tool-route.js +0 -382
  244. package/dist/core/repl/workspace-context.js +0 -206
  245. package/dist/core/repo-map/build.js +0 -125
  246. package/dist/core/repo-map/cache.js +0 -185
  247. package/dist/core/repo-map/extractor.js +0 -254
  248. package/dist/core/repo-map/formatter.js +0 -145
  249. package/dist/core/repo-map/page-rank.js +0 -105
  250. package/dist/core/repo-map/scanner.js +0 -211
  251. package/dist/core/retro/git-collector.js +0 -251
  252. package/dist/core/retro/health-card.js +0 -25
  253. package/dist/core/retro/metrics.js +0 -342
  254. package/dist/core/retro/narrative.js +0 -249
  255. package/dist/core/retro/plane-collector.js +0 -274
  256. package/dist/core/retro/pr-issue-link.js +0 -65
  257. package/dist/core/retro/types.js +0 -16
  258. package/dist/core/retry-budget/budget.js +0 -284
  259. package/dist/core/retry-budget/index.js +0 -5
  260. package/dist/core/retry-budget/retry-cap.js +0 -74
  261. package/dist/core/routing/lead-worker.js +0 -43
  262. package/dist/core/routing/pre-flight-estimator.js +0 -108
  263. package/dist/core/runs/run-tree.js +0 -103
  264. package/dist/core/sandboxing/adapter.js +0 -29
  265. package/dist/core/sandboxing/index.js +0 -49
  266. package/dist/core/sandboxing/none.js +0 -19
  267. package/dist/core/sandboxing/seatbelt.js +0 -183
  268. package/dist/core/security/injection-scanner.js +0 -367
  269. package/dist/core/security/output-filter.js +0 -418
  270. package/dist/core/session/env-file.js +0 -105
  271. package/dist/core/session/section-budgets.js +0 -140
  272. package/dist/core/session.js +0 -377
  273. package/dist/core/settings.js +0 -400
  274. package/dist/core/share/formatter.js +0 -271
  275. package/dist/core/share/redactor.js +0 -221
  276. package/dist/core/share/uploader.js +0 -267
  277. package/dist/core/skills/defaults.js +0 -457
  278. package/dist/core/skills/loader.js +0 -454
  279. package/dist/core/skills/sources.js +0 -480
  280. package/dist/core/skills/trust.js +0 -172
  281. package/dist/core/smoke/headless-driver.js +0 -174
  282. package/dist/core/smoke/orchestrator.js +0 -194
  283. package/dist/core/smoke/runner.js +0 -238
  284. package/dist/core/smoke/scenario-parser.js +0 -316
  285. package/dist/core/statusline.js +0 -99
  286. package/dist/core/subagents/dispatcher-real.js +0 -600
  287. package/dist/core/subagents/dispatcher.js +0 -352
  288. package/dist/core/subagents/index.js +0 -39
  289. package/dist/core/subagents/isolation-matrix.js +0 -213
  290. package/dist/core/subagents/spawn.js +0 -101
  291. package/dist/core/telemetry/emitter.js +0 -229
  292. package/dist/core/telemetry/queue.js +0 -251
  293. package/dist/core/theme/context.js +0 -91
  294. package/dist/core/theme/presets.js +0 -228
  295. package/dist/core/theme/state.js +0 -181
  296. package/dist/core/todos/invariant.js +0 -10
  297. package/dist/core/todos/state.js +0 -177
  298. package/dist/core/tool-schema/compressor.js +0 -89
  299. package/dist/core/transport/version-interceptor.js +0 -166
  300. package/dist/core/trust.js +0 -109
  301. package/dist/core/tui/thinking-block.js +0 -64
  302. package/dist/core/vim/keymap.js +0 -288
  303. package/dist/core/vim/state.js +0 -92
  304. package/dist/core/watch-markers/marker-watcher.js +0 -133
  305. package/dist/core/worktree/include-parser.js +0 -249
  306. package/dist/core/worktree-manager/cleanup.js +0 -123
  307. package/dist/core/worktree-manager/manager.js +0 -303
  308. package/dist/index.js +0 -44
  309. package/dist/runtime/bootstrap.js +0 -190
  310. package/dist/runtime/cli.js +0 -8121
  311. package/dist/runtime/commands/agents.js +0 -385
  312. package/dist/runtime/commands/budget.js +0 -192
  313. package/dist/runtime/commands/cancel.js +0 -231
  314. package/dist/runtime/commands/chain.js +0 -489
  315. package/dist/runtime/commands/codegraph-status.js +0 -227
  316. package/dist/runtime/commands/compact.js +0 -297
  317. package/dist/runtime/commands/config.js +0 -595
  318. package/dist/runtime/commands/cost.js +0 -199
  319. package/dist/runtime/commands/delegate.js +0 -312
  320. package/dist/runtime/commands/dispatch.js +0 -126
  321. package/dist/runtime/commands/doctor.js +0 -579
  322. package/dist/runtime/commands/feedback.js +0 -184
  323. package/dist/runtime/commands/hooks.js +0 -187
  324. package/dist/runtime/commands/init.js +0 -254
  325. package/dist/runtime/commands/lsp.js +0 -368
  326. package/dist/runtime/commands/mcp.js +0 -935
  327. package/dist/runtime/commands/memory.js +0 -582
  328. package/dist/runtime/commands/model.js +0 -237
  329. package/dist/runtime/commands/onboarding.js +0 -275
  330. package/dist/runtime/commands/patch.js +0 -128
  331. package/dist/runtime/commands/permissions.js +0 -112
  332. package/dist/runtime/commands/plan.js +0 -143
  333. package/dist/runtime/commands/prd-check.js +0 -285
  334. package/dist/runtime/commands/privacy.js +0 -107
  335. package/dist/runtime/commands/recipe.js +0 -325
  336. package/dist/runtime/commands/redo-blob-store.js +0 -92
  337. package/dist/runtime/commands/redo.js +0 -361
  338. package/dist/runtime/commands/release-notes.js +0 -229
  339. package/dist/runtime/commands/repo-map.js +0 -95
  340. package/dist/runtime/commands/report.js +0 -299
  341. package/dist/runtime/commands/resume.js +0 -118
  342. package/dist/runtime/commands/review-consensus.js +0 -414
  343. package/dist/runtime/commands/rewind.js +0 -333
  344. package/dist/runtime/commands/roster.js +0 -117
  345. package/dist/runtime/commands/sessions.js +0 -163
  346. package/dist/runtime/commands/share.js +0 -316
  347. package/dist/runtime/commands/skills.js +0 -401
  348. package/dist/runtime/commands/status.js +0 -186
  349. package/dist/runtime/commands/stickers.js +0 -82
  350. package/dist/runtime/commands/style.js +0 -194
  351. package/dist/runtime/commands/theme.js +0 -196
  352. package/dist/runtime/commands/undo.js +0 -361
  353. package/dist/runtime/commands/update.js +0 -289
  354. package/dist/runtime/commands/vim.js +0 -140
  355. package/dist/runtime/commands/worktree.js +0 -177
  356. package/dist/runtime/commands/worktrees.js +0 -155
  357. package/dist/runtime/deprecation-warning.js +0 -69
  358. package/dist/runtime/engine-exit-code.js +0 -50
  359. package/dist/runtime/headless-repl.js +0 -195
  360. package/dist/runtime/headless.js +0 -548
  361. package/dist/runtime/load-hooks-or-exit.js +0 -71
  362. package/dist/runtime/plan-decompose.js +0 -531
  363. package/dist/runtime/sigint-guard.js +0 -272
  364. package/dist/runtime/stream-renderer.js +0 -195
  365. package/dist/runtime/update-check.js +0 -294
  366. package/dist/runtime/version.js +0 -65
  367. package/dist/runtime/worktree-bootstrap.js +0 -579
  368. package/dist/skills/bundled/batch.js +0 -617
  369. package/dist/skills/bundled/index.js +0 -45
  370. package/dist/skills/bundled/loop.js +0 -358
  371. package/dist/skills/bundled/remember.js +0 -383
  372. package/dist/skills/bundled/simplify.js +0 -289
  373. package/dist/skills/bundled/skillify.js +0 -373
  374. package/dist/skills/bundled/stuck.js +0 -558
  375. package/dist/skills/bundled/verify.js +0 -439
  376. package/dist/testing/vcr.js +0 -486
  377. package/dist/tools/agent-tool.js +0 -229
  378. package/dist/tools/apply-patch.js +0 -556
  379. package/dist/tools/ask-user-question.js +0 -337
  380. package/dist/tools/ask-user.js +0 -115
  381. package/dist/tools/bash.js +0 -1238
  382. package/dist/tools/brief.js +0 -224
  383. package/dist/tools/cron.js +0 -433
  384. package/dist/tools/enter-worktree.js +0 -250
  385. package/dist/tools/exit-worktree.js +0 -147
  386. package/dist/tools/file-tools.js +0 -553
  387. package/dist/tools/http-request.js +0 -336
  388. package/dist/tools/lsp-tools.js +0 -565
  389. package/dist/tools/mcp-tool.js +0 -260
  390. package/dist/tools/multi-edit.js +0 -361
  391. package/dist/tools/powershell.js +0 -268
  392. package/dist/tools/registry.js +0 -166
  393. package/dist/tools/server-tools.js +0 -892
  394. package/dist/tools/skill-tool.js +0 -96
  395. package/dist/tools/sleep.js +0 -99
  396. package/dist/tools/synthetic-output.js +0 -133
  397. package/dist/tools/tasks.js +0 -208
  398. package/dist/tools/todo-write.js +0 -184
  399. package/dist/tools/verify-plan-execution.js +0 -295
  400. package/dist/tools/web-fetch-injection-scanner.js +0 -207
  401. package/dist/tools/web-fetch.js +0 -720
  402. package/dist/tools/web-search.js +0 -458
  403. package/dist/tui/agent-progress-card.js +0 -111
  404. package/dist/tui/agent-tree-pane.js +0 -9
  405. package/dist/tui/agent-tree.js +0 -87
  406. package/dist/tui/ask-cli.js +0 -52
  407. package/dist/tui/ask-modal.js +0 -211
  408. package/dist/tui/ask-user-question-chips.js +0 -315
  409. package/dist/tui/ask-user-question-prompt.js +0 -203
  410. package/dist/tui/compact-banner.js +0 -81
  411. package/dist/tui/conversation-pane.js +0 -164
  412. package/dist/tui/cost-table.js +0 -111
  413. package/dist/tui/device-flow.js +0 -142
  414. package/dist/tui/doctor-table.js +0 -46
  415. package/dist/tui/feedback-prompt.js +0 -156
  416. package/dist/tui/input-box.js +0 -732
  417. package/dist/tui/login-picker.js +0 -69
  418. package/dist/tui/markdown-render.js +0 -266
  419. package/dist/tui/multi-file-diff-approval.js +0 -375
  420. package/dist/tui/onboarding-wizard.js +0 -240
  421. package/dist/tui/permissions-picker.js +0 -86
  422. package/dist/tui/render.js +0 -160
  423. package/dist/tui/repl-render.js +0 -770
  424. package/dist/tui/repl-splash-art.js +0 -64
  425. package/dist/tui/repl-splash-mascot.js +0 -154
  426. package/dist/tui/repl-splash.js +0 -117
  427. package/dist/tui/repl.js +0 -378
  428. package/dist/tui/slash-palette.js +0 -106
  429. package/dist/tui/splash-data.js +0 -61
  430. package/dist/tui/splash.js +0 -31
  431. package/dist/tui/status-bar.js +0 -209
  432. package/dist/tui/status-table.js +0 -7
  433. package/dist/tui/stickers-art.js +0 -136
  434. package/dist/tui/style-table.js +0 -28
  435. package/dist/tui/theme-table.js +0 -29
  436. package/dist/tui/thinking-spinner.js +0 -123
  437. package/dist/tui/tool-stream-pane.js +0 -140
  438. package/dist/tui/update-banner.js +0 -33
  439. package/dist/tui/vim-input.js +0 -267
  440. package/dist/tui/welcome-banner.js +0 -107
  441. package/dist/tui/welcome-data.js +0 -293
  442. package/dist/tui/workspace-context.js +0 -105
  443. package/docs/examples/codegraph.mcp.json +0 -10
  444. package/test/scenarios/codegen-create-file.scenario.txt +0 -13
  445. package/test/scenarios/compact-force.scenario.txt +0 -12
  446. package/test/scenarios/identity.scenario.txt +0 -11
  447. package/test/scenarios/persona-handoff.scenario.txt +0 -12
  448. package/test/scenarios/walkback.scenario.txt +0 -12
@@ -1,548 +0,0 @@
1
- /**
2
- * Headless mode runner for `pugi --print "<brief>"`.
3
- *
4
- * Design intent (CEO directive "нужно тестирование по кругу"):
5
- *
6
- * Pugi must be drivable non-interactively by the upstream tool, Codex, CI
7
- * runners, Docker images, and any agent that can spawn a subprocess.
8
- * The Ink REPL is the human-facing surface; headless mode is the
9
- * machine-facing surface. One round-trip = one engine turn = one
10
- * `--print` invocation.
11
- *
12
- * Output contract (mirrors `claude --print --json`):
13
- *
14
- * Stdout — ONLY NDJSON event lines (one JSON object per line):
15
- * {"type":"session.start", sessionId, cwd, timestamp}
16
- * {"type":"turn.start", turnIndex, timestamp}
17
- * {"type":"tool.call", callId, tool, args, timestamp}
18
- * {"type":"tool.result", callId, status, summary, timestamp}
19
- * {"type":"text.delta", content, timestamp}
20
- * {"type":"thinking.delta",content, blockId, timestamp}
21
- * {"type":"turn.end", turnIndex, usage, timestamp}
22
- * {"type":"session.end", status, totalTokensIn, totalTokensOut,
23
- * durationMs, filesChanged, timestamp}
24
- * {"type":"error", kind, message, exit_code}
25
- *
26
- * Stderr — human-readable status + error details (never NDJSON).
27
- *
28
- * Stdout vs stderr discipline lets a caller pipe cleanly:
29
- *
30
- * pugi --print "..." --json 2>/dev/null | jq .
31
- *
32
- * Exit codes:
33
- *
34
- * 0 session.end emitted with status=done (or "complete")
35
- * 1 hard error (auth, network, server crash, engine_unavailable)
36
- * 2 turn-limit / budget exhausted without completion
37
- *
38
- * Reuses the existing `NativePugiEngineAdapter` (the same path the
39
- * REPL drives) — the only difference is the output sink: instead of
40
- * mounting Ink, we subscribe to the adapter's `streamEmitter` and
41
- * forward each typed event as NDJSON.
42
- */
43
- import { resolve as resolvePath } from 'node:path';
44
- import { existsSync, statSync } from 'node:fs';
45
- import { AnvilEngineLoopClient } from '../core/engine/anvil-client.js';
46
- import { NativePugiEngineAdapter } from '../core/engine/native-pugi.js';
47
- import { defaultNonInteractiveMcpPrompt } from '../tools/mcp-tool.js';
48
- import { loadMcpRegistry } from '../core/mcp/registry.js';
49
- import { openSession, recordCommandStarted, recordCommandCompleted, } from '../core/session.js';
50
- import { buildRuntimeConfig, loadRuntimeConfig, } from '@pugi/sdk';
51
- import { resolveActiveCredential } from '../core/credentials.js';
52
- /**
53
- * NDJSON event emitter. One JSON object per line on stdout. Never emits
54
- * partial lines — every event ends with `\n` so a caller piping into
55
- * `jq -c` can parse incrementally. Stderr is reserved for human prose
56
- * (errors, traces) so the operator's `2>/dev/null` filter keeps stdout
57
- * pure machine-readable.
58
- */
59
- class NdjsonEventSink {
60
- stdoutWrite;
61
- constructor(stdoutWrite) {
62
- this.stdoutWrite = stdoutWrite;
63
- }
64
- emit(event) {
65
- // Stamp every event with `timestamp` if the caller did not supply
66
- // one. ISO-8601 with millisecond precision is the agreed contract
67
- // (per spec); a downstream timeline UI can sort lexicographically.
68
- const stamped = {
69
- ...event,
70
- timestamp: event.timestamp ?? new Date().toISOString(),
71
- };
72
- this.stdoutWrite(`${JSON.stringify(stamped)}\n`);
73
- }
74
- }
75
- /**
76
- * Human-readable text sink. Used when `--print` runs WITHOUT `--json`
77
- * (a human watching the pipe). Tool calls show as `> tool(args)`,
78
- * results as `< ok|fail: summary`, text deltas as raw prose chunks.
79
- * Status frames are demoted to stderr so the conversation flow on
80
- * stdout stays clean.
81
- */
82
- class TextEventSink {
83
- stdoutWrite;
84
- stderrWrite;
85
- constructor(stdoutWrite, stderrWrite) {
86
- this.stdoutWrite = stdoutWrite;
87
- this.stderrWrite = stderrWrite;
88
- }
89
- emit(event) {
90
- switch (event.type) {
91
- case 'session.start':
92
- this.stderrWrite(`pugi headless: session ${event.sessionId} cwd=${event.cwd}\n`);
93
- return;
94
- case 'turn.start':
95
- this.stderrWrite(`> turn ${event.turnIndex}\n`);
96
- return;
97
- case 'tool.call':
98
- this.stderrWrite(`> ${event.tool}(${truncate(JSON.stringify(event.args), 80)})\n`);
99
- return;
100
- case 'tool.result':
101
- this.stderrWrite(`< ${event.status === 'success' ? 'ok' : 'fail'}: ${truncate(String(event.summary ?? ''), 120)}\n`);
102
- return;
103
- case 'text.delta':
104
- // Bare prose on stdout — this IS the answer.
105
- this.stdoutWrite(String(event.content ?? ''));
106
- return;
107
- case 'thinking.delta':
108
- // Thinking is informational; surface on stderr so stdout is the
109
- // visible answer only. Operators who want thinking pipe stderr.
110
- this.stderrWrite(`[think] ${truncate(String(event.content ?? ''), 200)}\n`);
111
- return;
112
- case 'turn.end':
113
- this.stderrWrite(`< turn ${event.turnIndex} end\n`);
114
- return;
115
- case 'session.end':
116
- // Newline separator so the final stdout text and stderr summary
117
- // do not run together when both are tee'd to the same terminal.
118
- this.stdoutWrite('\n');
119
- this.stderrWrite(`pugi headless: ${event.status} · ${event.totalTokensIn ?? 0}+${event.totalTokensOut ?? 0} tokens · ${event.durationMs}ms\n`);
120
- return;
121
- case 'error':
122
- this.stderrWrite(`pugi headless: error (${event.kind}) ${event.message}\n`);
123
- return;
124
- default:
125
- return;
126
- }
127
- }
128
- }
129
- function truncate(s, max) {
130
- return s.length <= max ? s : `${s.slice(0, max - 3)}...`;
131
- }
132
- /**
133
- * Auto-init the workspace if `.pugi/` is missing. Per beta.13 the
134
- * scaffold is silent — headless callers (CI, agents) drop into fresh
135
- * cwds all the time and the first run must not require a manual
136
- * `pugi init` step. We DO surface a stderr breadcrumb so the operator
137
- * sees what changed on disk.
138
- */
139
- async function ensurePugiInitializedWithLog(cwd, stderrWrite) {
140
- const pugiDir = resolvePath(cwd, '.pugi');
141
- if (existsSync(pugiDir) && statSync(pugiDir).isDirectory())
142
- return;
143
- // Late-import so the CLI bootstrap cost stays minimal — scaffold is
144
- // only loaded when the auto-init path actually fires.
145
- const { scaffoldPugiWorkspace } = await import('./cli.js');
146
- stderrWrite(`pugi headless: scaffolding .pugi/ in ${cwd}\n`);
147
- await scaffoldPugiWorkspace({
148
- cwd,
149
- noDefaults: true,
150
- log: (line) => stderrWrite(`[init] ${line}`),
151
- });
152
- }
153
- /**
154
- * Resolve the runtime credential. Headless mode REQUIRES a token — we
155
- * cannot launch a device flow without a TTY to drop a browser into. A
156
- * missing credential is a hard error with exit 1.
157
- */
158
- function resolveHeadlessCredential() {
159
- const credential = resolveActiveCredential();
160
- if (credential) {
161
- return buildRuntimeConfig({
162
- apiUrl: credential.apiUrl,
163
- apiKey: credential.apiKey,
164
- });
165
- }
166
- // Fall through to env-only config (CI sets PUGI_API_KEY directly).
167
- return loadRuntimeConfig();
168
- }
169
- /**
170
- * Headless entry point. Returns the desired process exit code.
171
- *
172
- * 0 — done / completed
173
- * 1 — hard error (auth, engine_unavailable, transport crash)
174
- * 2 — turn-limit / budget exhausted
175
- *
176
- * Caller (cli.ts) sets `process.exitCode`; we do NOT call `process.exit`
177
- * directly so a future REPL-embedded `/print` slash can reuse the same
178
- * function in-process without tearing down the host.
179
- */
180
- export async function runHeadlessPrint(opts) {
181
- const cwd = resolvePath(opts.cwd);
182
- // Default writers route to the real process streams; tests inject
183
- // closures so `node:test`'s binary-IPC hijack of `process.stdout`
184
- // does not collide with the captured NDJSON output.
185
- const stdoutWrite = opts.stdoutWrite ?? ((chunk) => process.stdout.write(chunk));
186
- const stderrWrite = opts.stderrWrite ?? ((chunk) => process.stderr.write(chunk));
187
- const sink = opts.json
188
- ? new NdjsonEventSink(stdoutWrite)
189
- : new TextEventSink(stdoutWrite, stderrWrite);
190
- const startedAt = Date.now();
191
- // 1. Auto-init the workspace if needed.
192
- try {
193
- await ensurePugiInitializedWithLog(cwd, stderrWrite);
194
- }
195
- catch (error) {
196
- const message = error instanceof Error ? error.message : String(error);
197
- sink.emit({
198
- type: 'error',
199
- kind: 'init_failed',
200
- message: `failed to scaffold .pugi/: ${message}`,
201
- exit_code: 1,
202
- });
203
- return 1;
204
- }
205
- // 2. Resolve the credential. Headless cannot interactively log in.
206
- // When a test factory is injected we still build a config (the
207
- // factory ignores it) so the type contract stays narrow.
208
- let config;
209
- if (opts.engineClientFactory) {
210
- config =
211
- loadRuntimeConfig() ??
212
- buildRuntimeConfig({
213
- apiUrl: 'http://127.0.0.1:1/headless-fixture',
214
- apiKey: 'pugi_headless_fixture',
215
- });
216
- }
217
- else {
218
- config = resolveHeadlessCredential();
219
- }
220
- if (!config) {
221
- sink.emit({
222
- type: 'error',
223
- kind: 'auth_missing',
224
- message: 'No Pugi credential. Run `pugi login` or set PUGI_API_KEY before invoking `pugi --print` (headless mode cannot launch device flow without a TTY).',
225
- exit_code: 1,
226
- });
227
- return 1;
228
- }
229
- // 3. Open a session. The session id flows through every event line
230
- // so external tooling can correlate the headless run with its
231
- // server-side trace.
232
- const session = openSession(cwd);
233
- if (opts.sessionIdOverride && opts.sessionIdOverride.length > 0) {
234
- // Override is informational — we cannot reassign `session.id`
235
- // safely (it threads through `recordToolCall`), but we surface
236
- // the requested id alongside the resolved one so the caller can
237
- // join its own logs.
238
- sink.emit({
239
- type: 'session.start',
240
- sessionId: session.id,
241
- requestedSessionId: opts.sessionIdOverride,
242
- cwd,
243
- ...(opts.workspace ? { workspace: opts.workspace } : {}),
244
- });
245
- }
246
- else {
247
- sink.emit({
248
- type: 'session.start',
249
- sessionId: session.id,
250
- cwd,
251
- ...(opts.workspace ? { workspace: opts.workspace } : {}),
252
- });
253
- }
254
- recordCommandStarted(session, 'print');
255
- // 4. Construct the engine client + adapter. The factory seam exists
256
- // purely for tests; production always builds `AnvilEngineLoopClient`.
257
- const client = opts.engineClientFactory
258
- ? opts.engineClientFactory(config)
259
- : new AnvilEngineLoopClient(config);
260
- // MCP registry: best-effort load. Headless mode honors the workspace
261
- // MCP config the same way the REPL does — a failed registry should
262
- // not kill the run, but the operator deserves a stderr breadcrumb.
263
- let mcpRegistry;
264
- try {
265
- mcpRegistry = await loadMcpRegistry(cwd);
266
- }
267
- catch (error) {
268
- stderrWrite(`pugi headless: MCP registry load failed — ${error.message}. Continuing without MCP.\n`);
269
- mcpRegistry = undefined;
270
- }
271
- const adapter = new NativePugiEngineAdapter({
272
- client,
273
- session,
274
- // --no-tools maps to suppressing fetch + search opt-ins. The
275
- // tool-bridge schema is determined per-kind; we cannot dial it to
276
- // zero from the adapter constructor today (that requires a
277
- // tool-bridge plumbing change). For β1 we honor --no-tools by
278
- // refusing the engine path entirely when set AND the kind is one
279
- // that requires tools to be useful — see the gate below.
280
- allowFetch: !opts.noTools,
281
- allowSearch: !opts.noTools,
282
- ...(mcpRegistry ? { mcpRegistry } : {}),
283
- mcpPrompt: defaultNonInteractiveMcpPrompt,
284
- interactive: false,
285
- });
286
- // 5. Subscribe to the adapter's typed stream emitter and forward
287
- // every event as one NDJSON line. The emitter is the canonical
288
- // source for tool / text / thinking deltas — the async-iterable
289
- // `adapter.run()` collapses richer types into lossy `status` for
290
- // legacy SDK consumers, which we deliberately bypass here.
291
- let turnIndex = 0;
292
- let currentTurnTokensIn = 0;
293
- let currentTurnTokensOut = 0;
294
- let totalTokens = 0;
295
- const filesChanged = new Set();
296
- const sawTurnStart = false; // tracked via emitter
297
- const handleStreamEvent = (event) => {
298
- switch (event.type) {
299
- case 'status': {
300
- // Status frames carry adapter lifecycle prose. Most are noise
301
- // for an NDJSON consumer; the structurally meaningful ones we
302
- // synthesize into typed events below. A few worth surfacing:
303
- // `turn N: requesting model` → emit turn.start.
304
- const m = /^turn (\d+): requesting model/.exec(event.message);
305
- if (m) {
306
- turnIndex = Number(m[1]);
307
- sink.emit({ type: 'turn.start', turnIndex, timestamp: event.timestamp });
308
- return;
309
- }
310
- const t = /^turn (\d+): model returned final text/.exec(event.message);
311
- if (t) {
312
- // Closing turn.end is emitted on result yield below so we
313
- // have token usage to attach. Skip the synthetic close here.
314
- return;
315
- }
316
- // All other status frames are diagnostic — surface as a
317
- // structured `status` event so consumers can opt in.
318
- sink.emit({
319
- type: 'status',
320
- message: event.message,
321
- timestamp: event.timestamp,
322
- });
323
- return;
324
- }
325
- case 'tool.start':
326
- sink.emit({
327
- type: 'tool.call',
328
- callId: event.callId,
329
- tool: event.name,
330
- args: safeJsonParse(event.arguments),
331
- timestamp: event.timestamp,
332
- });
333
- return;
334
- case 'tool.end':
335
- sink.emit({
336
- type: 'tool.result',
337
- callId: event.callId,
338
- status: event.ok ? 'success' : 'error',
339
- summary: event.summary,
340
- timestamp: event.timestamp,
341
- });
342
- return;
343
- case 'tool.delta':
344
- // Pass through as a partial chunk so a long-running bash can
345
- // surface progress.
346
- sink.emit({
347
- type: 'tool.delta',
348
- callId: event.callId,
349
- chunk: event.chunk,
350
- timestamp: event.timestamp,
351
- });
352
- return;
353
- case 'text.delta':
354
- sink.emit({
355
- type: 'text.delta',
356
- content: event.chunk,
357
- timestamp: event.timestamp,
358
- });
359
- return;
360
- case 'thinking.start':
361
- sink.emit({
362
- type: 'thinking.start',
363
- blockId: event.blockId,
364
- timestamp: event.timestamp,
365
- });
366
- return;
367
- case 'thinking.delta':
368
- sink.emit({
369
- type: 'thinking.delta',
370
- blockId: event.blockId,
371
- content: event.chunk,
372
- timestamp: event.timestamp,
373
- });
374
- return;
375
- case 'thinking.end':
376
- sink.emit({
377
- type: 'thinking.end',
378
- blockId: event.blockId,
379
- timestamp: event.timestamp,
380
- });
381
- return;
382
- default:
383
- return;
384
- }
385
- };
386
- adapter.streamEmitter.on('event', handleStreamEvent);
387
- // 6. Optional timeout — wire an AbortController so the engine loop
388
- // bails cleanly. The signal also feeds the tool-bridge cancellation
389
- // gate so long-running bash / network tools tear down.
390
- const abort = new AbortController();
391
- let timeoutHandle = null;
392
- if (opts.timeoutSeconds && opts.timeoutSeconds > 0) {
393
- timeoutHandle = setTimeout(() => {
394
- abort.abort();
395
- stderrWrite(`pugi headless: timeout after ${opts.timeoutSeconds}s — aborting engine\n`);
396
- }, opts.timeoutSeconds * 1000);
397
- // Don't keep node alive purely for the timeout — the engine path
398
- // already holds the event loop via the AnvilEngineLoopClient fetch.
399
- timeoutHandle.unref?.();
400
- }
401
- // 7. Drive the adapter to terminal. We iterate the (lossier)
402
- // EngineEvent stream solely to learn when the result frame lands;
403
- // every richer event already flew through `streamEmitter` above.
404
- const kind = opts.kind ?? 'code';
405
- const taskId = `print-${Date.now()}`;
406
- // PUGI-VERIFY-GATE: `needs_verification` is the new fourth
407
- // status the engine adapter surfaces when a `completed` loop ran
408
- // no verification command. Widen the local var to match the
409
- // engine result schema.
410
- let finalStatus = 'failed';
411
- let finalSummary = '';
412
- let resultRisks = [];
413
- let eventRefs = [];
414
- try {
415
- const events = adapter.run({
416
- id: taskId,
417
- kind,
418
- prompt: opts.prompt,
419
- workspaceRoot: cwd,
420
- allowedPaths: [cwd],
421
- deniedPaths: [],
422
- artifacts: [],
423
- permissionMode: 'auto',
424
- ...(opts.maxTurns ? { budget: { tokens: opts.maxTurns * 16384 } } : {}),
425
- }, { sessionId: session.id, signal: abort.signal });
426
- for await (const ev of events) {
427
- if (ev.type === 'result') {
428
- finalStatus = ev.result.status;
429
- finalSummary = ev.result.summary;
430
- resultRisks = ev.result.risks;
431
- eventRefs = ev.result.eventRefs;
432
- for (const f of ev.result.filesChanged)
433
- filesChanged.add(f);
434
- }
435
- // status frames already routed through the emitter — ignore here.
436
- }
437
- }
438
- catch (error) {
439
- const message = error instanceof Error ? error.message : String(error);
440
- sink.emit({
441
- type: 'error',
442
- kind: 'engine_crash',
443
- message,
444
- exit_code: 1,
445
- });
446
- if (timeoutHandle)
447
- clearTimeout(timeoutHandle);
448
- adapter.streamEmitter.off('event', handleStreamEvent);
449
- recordCommandCompleted(session, 'print', 'error');
450
- return 1;
451
- }
452
- if (timeoutHandle)
453
- clearTimeout(timeoutHandle);
454
- adapter.streamEmitter.off('event', handleStreamEvent);
455
- if (mcpRegistry) {
456
- await mcpRegistry.shutdown().catch(() => {
457
- /* swallow — best-effort */
458
- });
459
- }
460
- // 8. Pull headline metrics out of the result's eventRefs (adapter
461
- // echoes `tokens=N` / `turns=N` / `tool_calls=N` / `outcome=...`).
462
- const metrics = parseRefs(eventRefs);
463
- totalTokens = metrics.tokens;
464
- currentTurnTokensIn;
465
- currentTurnTokensOut;
466
- sawTurnStart; // silence TS6133
467
- // Emit one closing turn.end so consumers always see a paired turn
468
- // start/end. We use the final turn index reported by the adapter.
469
- sink.emit({
470
- type: 'turn.end',
471
- turnIndex: metrics.turns || turnIndex || 1,
472
- usage: {
473
- // Engine outcome carries a single cumulative `tokens` figure; we
474
- // split it 50/50 as a best-effort estimate until the SDK splits
475
- // input/output. Documented as approximate so consumers do not
476
- // treat the breakdown as authoritative.
477
- tokensIn: Math.floor(totalTokens / 2),
478
- tokensOut: Math.ceil(totalTokens / 2),
479
- },
480
- });
481
- // 9. Emit session.end. Exit code policy per spec:
482
- // - done → 0
483
- // - needs_verification → 2 (PUGI-VERIFY-GATE)
484
- // - blocked + outcome=budget_exhausted → 2
485
- // - blocked + any other reason → 2 (turn limit, tool refused — both
486
- // count as "did not complete")
487
- // - failed → 1
488
- const exitCode = finalStatus === 'done'
489
- ? 0
490
- : finalStatus === 'blocked' || finalStatus === 'needs_verification'
491
- ? 2
492
- : 1;
493
- const durationMs = Date.now() - startedAt;
494
- sink.emit({
495
- type: 'session.end',
496
- sessionId: session.id,
497
- status: finalStatus,
498
- outcome: metrics.outcome,
499
- totalTokensIn: Math.floor(totalTokens / 2),
500
- totalTokensOut: Math.ceil(totalTokens / 2),
501
- totalToolCalls: metrics.toolCalls,
502
- totalTurns: metrics.turns,
503
- durationMs,
504
- filesChanged: Array.from(filesChanged).sort(),
505
- risks: resultRisks,
506
- finalText: finalSummary,
507
- });
508
- recordCommandCompleted(session, 'print', exitCode === 0 ? 'success' : 'error');
509
- return exitCode;
510
- }
511
- /**
512
- * Best-effort JSON parse for tool arguments. Engine emits the raw
513
- * argument string the model produced; if it is malformed we keep the
514
- * raw string so the consumer can still see what the model sent.
515
- */
516
- function safeJsonParse(s) {
517
- try {
518
- return JSON.parse(s);
519
- }
520
- catch {
521
- return s;
522
- }
523
- }
524
- /**
525
- * Parse the `eventRefs` echo the adapter attaches to its result frame.
526
- * Mirrors `parseEventRefs` in `cli.ts` — kept local so the headless
527
- * module has no circular import on the runtime entry.
528
- */
529
- function parseRefs(refs) {
530
- const out = { toolCalls: 0, turns: 0, tokens: 0, outcome: null };
531
- for (const ref of refs) {
532
- const idx = ref.indexOf('=');
533
- if (idx <= 0)
534
- continue;
535
- const key = ref.slice(0, idx);
536
- const value = ref.slice(idx + 1);
537
- if (key === 'tool_calls')
538
- out.toolCalls = Number(value) || 0;
539
- else if (key === 'turns')
540
- out.turns = Number(value) || 0;
541
- else if (key === 'tokens')
542
- out.tokens = Number(value) || 0;
543
- else if (key === 'outcome')
544
- out.outcome = value || null;
545
- }
546
- return out;
547
- }
548
- //# sourceMappingURL=headless.js.map
@@ -1,71 +0,0 @@
1
- /**
2
- * Fail-closed hook registry loader.
3
- *
4
- * r2 (triple-review P2): the previous fail-open path silently
5
- * disabled BLOCK rules when `.pugi/hooks.json` parsed badly. This module
6
- * is the dedicated load path used by the engine command dispatch — it
7
- * distinguishes three states:
8
- *
9
- * (a) No hooks file present → returns `undefined` (acceptable; the
10
- * operator has no hooks configured).
11
- * (b) Hooks file present + parses + loads → returns the live registry.
12
- * (c) Hooks file present + parse/load FAILS → emits a fatal message
13
- * and signals `refusedExit`. The caller (cli.ts) is expected to
14
- * honor that by calling `process.exit(1)`. `PUGI_HOOKS_BYPASS=1`
15
- * is the explicit operator-typed escape hatch.
16
- *
17
- * Extracted from `cli.ts` purely so the fail-closed contract is unit-
18
- * testable without spawning a subprocess. The shape returned is a
19
- * discriminated union so tests can assert each branch directly.
20
- *
21
- * Brand voice: no forbidden words. ASCII only. No emoji.
22
- */
23
- import { existsSync } from 'node:fs';
24
- import { homedir } from 'node:os';
25
- import { resolve } from 'node:path';
26
- import { HookRegistry } from '../core/hooks.js';
27
- export async function loadHookRegistryOrExit(opts) {
28
- const stderrWrite = opts.deps?.stderrWrite ??
29
- ((msg) => {
30
- process.stderr.write(msg);
31
- });
32
- const env = opts.deps?.env ?? process.env;
33
- const projectHooksPath = resolve(opts.workspaceRoot, '.pugi', 'hooks.json');
34
- const pugiHomeRoot = env.PUGI_HOME ?? resolve(homedir(), '.pugi');
35
- const userHooksPath = resolve(pugiHomeRoot, 'hooks.json');
36
- const presentHookFiles = [projectHooksPath, userHooksPath].filter((p) => existsSync(p));
37
- if (presentHookFiles.length === 0) {
38
- return { kind: 'no-hooks-file', hooks: undefined };
39
- }
40
- try {
41
- const reg = new HookRegistry({
42
- workspaceRoot: opts.workspaceRoot,
43
- session: opts.session,
44
- });
45
- await reg.load();
46
- return { kind: 'loaded', hooks: reg };
47
- }
48
- catch (error) {
49
- const msg = error.message;
50
- stderrWrite(`pugi ${opts.label}: hook configuration present but failed to load — refusing to run.\n` +
51
- ` files checked: ${presentHookFiles.join(', ')}\n` +
52
- ` error: ${msg}\n` +
53
- ` fix: validate the JSON or remove the file; or set PUGI_HOOKS_BYPASS=1 to override (NOT RECOMMENDED — disables block rules).\n`);
54
- if (env.PUGI_HOOKS_BYPASS === '1') {
55
- stderrWrite(`pugi ${opts.label}: PUGI_HOOKS_BYPASS=1 — continuing WITHOUT hooks despite parse failure.\n`);
56
- return {
57
- kind: 'parse-failure-bypassed',
58
- hooks: undefined,
59
- filesChecked: presentHookFiles,
60
- error: msg,
61
- };
62
- }
63
- return {
64
- kind: 'parse-failure-refused',
65
- filesChecked: presentHookFiles,
66
- error: msg,
67
- hooks: undefined,
68
- };
69
- }
70
- }
71
- //# sourceMappingURL=load-hooks-or-exit.js.map