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

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (402) hide show
  1. package/CHANGELOG.md +96 -0
  2. package/THIRD_PARTY_NOTICES.md +40 -0
  3. package/assets/pugi-prozr2-mascot.ansi +9 -0
  4. package/bin/run.js +33 -1
  5. package/dist/commands/deploy.js +40 -40
  6. package/dist/commands/flatten.js +191 -0
  7. package/dist/commands/jobs-watch.js +201 -0
  8. package/dist/commands/jobs.js +42 -27
  9. package/dist/commands/smoke.js +133 -0
  10. package/dist/core/agent-progress/cleanup.js +134 -0
  11. package/dist/core/agent-progress/schema.js +144 -0
  12. package/dist/core/agent-progress/writer.js +101 -0
  13. package/dist/core/agents/adaptive-router.js +330 -0
  14. package/dist/core/agents/query-decomposer.js +297 -0
  15. package/dist/core/agents/registry.js +2 -2
  16. package/dist/core/approvals/shortcut-resolver.js +98 -0
  17. package/dist/core/artifact-chain/dispatcher.js +148 -0
  18. package/dist/core/artifact-chain/exporter.js +164 -0
  19. package/dist/core/artifact-chain/state.js +243 -0
  20. package/dist/core/artifact-chain/steps.js +169 -0
  21. package/dist/core/ask-user/question.js +92 -0
  22. package/dist/core/audit/audit-trail.js +275 -0
  23. package/dist/core/auth/ensure-authenticated.js +129 -0
  24. package/dist/core/auth/env-provider.js +238 -0
  25. package/dist/core/auto-open-browser.js +4 -4
  26. package/dist/core/auto-update/channels.js +122 -0
  27. package/dist/core/auto-update/checker.js +241 -0
  28. package/dist/core/auto-update/state.js +235 -0
  29. package/dist/core/bare-mode/index.js +107 -0
  30. package/dist/core/bash/redirect.js +281 -0
  31. package/dist/core/bash-classifier.js +436 -40
  32. package/dist/core/checkpoint/resumer.js +149 -0
  33. package/dist/core/checkpoint/rewinder.js +291 -0
  34. package/dist/core/checkpoints/shadow-git.js +670 -0
  35. package/dist/core/citations/parser.js +109 -0
  36. package/dist/core/classifier/yolo-classifier.js +88 -0
  37. package/dist/core/codegraph/decision-store.js +248 -0
  38. package/dist/core/codegraph/detect-repo.js +459 -0
  39. package/dist/core/codegraph/install.js +134 -0
  40. package/dist/core/codegraph/offer-hook.js +220 -0
  41. package/dist/core/compact/auto-trigger.js +96 -0
  42. package/dist/core/compact/buffer-rewriter.js +115 -0
  43. package/dist/core/compact/summarizer.js +208 -0
  44. package/dist/core/compact/token-counter.js +108 -0
  45. package/dist/core/consensus/anvil-fanout.js +25 -25
  46. package/dist/core/consensus/diff-capture.js +121 -12
  47. package/dist/core/consensus/rubric.js +21 -21
  48. package/dist/core/context/builder.js +6 -6
  49. package/dist/core/context/compaction-events.js +8 -8
  50. package/dist/core/context/compaction.js +31 -31
  51. package/dist/core/context/index.js +15 -8
  52. package/dist/core/context/invariants.js +51 -51
  53. package/dist/core/context/markdown-loader.js +28 -10
  54. package/dist/core/context/markdown-traverse.js +255 -0
  55. package/dist/core/context/pugiignore.js +41 -41
  56. package/dist/core/context/repo-skeleton.js +37 -37
  57. package/dist/core/context/tool-eviction.js +55 -0
  58. package/dist/core/context/watcher.js +32 -32
  59. package/dist/core/context/working-set.js +23 -23
  60. package/dist/core/coordinator/agent-tools.js +77 -0
  61. package/dist/core/coordinator/agent-toolset.js +65 -0
  62. package/dist/core/coordinator/fsm.js +73 -0
  63. package/dist/core/coordinator/mode-fsm.js +70 -0
  64. package/dist/core/cost/rate-card.js +129 -0
  65. package/dist/core/cost/tracker.js +221 -0
  66. package/dist/core/credentials.js +12 -12
  67. package/dist/core/cron/scheduler.js +138 -0
  68. package/dist/core/denial-tracking/index.js +8 -0
  69. package/dist/core/denial-tracking/state.js +264 -0
  70. package/dist/core/diagnostics/probe-runner.js +93 -0
  71. package/dist/core/diagnostics/probes/api.js +46 -0
  72. package/dist/core/diagnostics/probes/auth.js +93 -0
  73. package/dist/core/diagnostics/probes/bare-mode.js +42 -0
  74. package/dist/core/diagnostics/probes/cli-version.js +127 -0
  75. package/dist/core/diagnostics/probes/config.js +72 -0
  76. package/dist/core/diagnostics/probes/denial-tracking.js +57 -0
  77. package/dist/core/diagnostics/probes/disk.js +81 -0
  78. package/dist/core/diagnostics/probes/engine-live.js +46 -0
  79. package/dist/core/diagnostics/probes/git.js +65 -0
  80. package/dist/core/diagnostics/probes/hooks.js +118 -0
  81. package/dist/core/diagnostics/probes/mcp.js +75 -0
  82. package/dist/core/diagnostics/probes/node.js +59 -0
  83. package/dist/core/diagnostics/probes/pnpm.js +36 -0
  84. package/dist/core/diagnostics/probes/pugi-md.js +89 -0
  85. package/dist/core/diagnostics/probes/sandbox.js +40 -0
  86. package/dist/core/diagnostics/probes/session.js +74 -0
  87. package/dist/core/diagnostics/probes/status-snapshot.js +488 -0
  88. package/dist/core/diagnostics/probes/workspace.js +63 -0
  89. package/dist/core/diagnostics/types.js +70 -0
  90. package/dist/core/dispatch/cache-cleanup.js +197 -0
  91. package/dist/core/dispatch/cache-handoff.js +295 -0
  92. package/dist/core/edits/apply-patch-layer-e.js +189 -0
  93. package/dist/core/edits/dispatch.js +293 -7
  94. package/dist/core/edits/format-matrix.js +26 -0
  95. package/dist/core/edits/fuzzy-ladder.js +650 -0
  96. package/dist/core/edits/index.js +3 -1
  97. package/dist/core/edits/journal.js +199 -0
  98. package/dist/core/edits/layer-a-apply.js +15 -15
  99. package/dist/core/edits/layer-a-fuzzy-apply.js +198 -0
  100. package/dist/core/edits/layer-b-apply.js +9 -9
  101. package/dist/core/edits/layer-c-apply.js +6 -6
  102. package/dist/core/edits/layer-d-ast.js +557 -14
  103. package/dist/core/edits/marker-parser.js +12 -12
  104. package/dist/core/edits/security-gate.js +27 -27
  105. package/dist/core/edits/verify-hook.js +273 -0
  106. package/dist/core/edits/worktree.js +322 -0
  107. package/dist/core/engine/anvil-client.js +140 -26
  108. package/dist/core/engine/auto-compact.js +179 -0
  109. package/dist/core/engine/budgets.js +186 -0
  110. package/dist/core/engine/context-prefix.js +155 -0
  111. package/dist/core/engine/index.js +1 -1
  112. package/dist/core/engine/intensity.js +158 -0
  113. package/dist/core/engine/intent.js +260 -0
  114. package/dist/core/engine/native-pugi.js +1295 -227
  115. package/dist/core/engine/prompts.js +134 -16
  116. package/dist/core/engine/strip-internal-fields.js +124 -0
  117. package/dist/core/engine/tool-bridge.js +1295 -59
  118. package/dist/core/evaluation/golden-dataset.js +293 -0
  119. package/dist/core/feedback/queue.js +177 -0
  120. package/dist/core/feedback/submitter.js +145 -0
  121. package/dist/core/file-cache.js +113 -1
  122. package/dist/core/flatten/flatten-repo.js +439 -0
  123. package/dist/core/format/osc8-link.js +28 -0
  124. package/dist/core/hook-chains.js +392 -0
  125. package/dist/core/hooks/citation-verify-hook.js +138 -0
  126. package/dist/core/hooks/citation-verify.js +112 -0
  127. package/dist/core/hooks/events.js +44 -0
  128. package/dist/core/hooks/index.js +15 -0
  129. package/dist/core/hooks/registry.js +213 -0
  130. package/dist/core/hooks/runner.js +236 -0
  131. package/dist/core/hooks/v2/event-emitter.js +115 -0
  132. package/dist/core/hooks/v2/executor.js +282 -0
  133. package/dist/core/hooks/v2/index.js +25 -0
  134. package/dist/core/hooks/v2/lifecycle.js +104 -0
  135. package/dist/core/hooks/v2/loader.js +216 -0
  136. package/dist/core/hooks/v2/matcher.js +125 -0
  137. package/dist/core/hooks/v2/trust.js +143 -0
  138. package/dist/core/hooks/v2/types.js +86 -0
  139. package/dist/core/image/renderer.js +71 -0
  140. package/dist/core/init/detector.js +582 -0
  141. package/dist/core/init/template-renderer.js +242 -0
  142. package/dist/core/jobs/registry.js +18 -18
  143. package/dist/core/ledger/results-tsv.js +142 -0
  144. package/dist/core/log-discipline/stdout-redirect.js +51 -0
  145. package/dist/core/lsp/cache.js +105 -0
  146. package/dist/core/lsp/client.js +776 -0
  147. package/dist/core/lsp/language-detect.js +66 -0
  148. package/dist/core/lsp/post-edit-diagnostics.js +171 -0
  149. package/dist/core/lsp/symbol-tools.js +372 -0
  150. package/dist/core/mcp/client.js +97 -28
  151. package/dist/core/mcp/http-server.js +553 -0
  152. package/dist/core/mcp/orchestrator-tools.js +662 -0
  153. package/dist/core/mcp/permission.js +190 -0
  154. package/dist/core/mcp/registry.js +39 -17
  155. package/dist/core/mcp/server-tools.js +219 -0
  156. package/dist/core/mcp/server.js +397 -0
  157. package/dist/core/mcp/trust.js +10 -10
  158. package/dist/core/memory/dual-write.js +416 -0
  159. package/dist/core/memory/passive-extract.js +130 -0
  160. package/dist/core/memory/phase1-kinds.js +20 -0
  161. package/dist/core/memory/secret-scanner.js +304 -0
  162. package/dist/core/memory-sync/queue.js +170 -0
  163. package/dist/core/metrics/extract.js +113 -0
  164. package/dist/core/modes/roo-modes.js +68 -0
  165. package/dist/core/onboarding/ensure-initialized.js +133 -0
  166. package/dist/core/onboarding/marker.js +111 -0
  167. package/dist/core/onboarding/telemetry-state.js +108 -0
  168. package/dist/core/output-style/presets.js +176 -0
  169. package/dist/core/output-style/state.js +185 -0
  170. package/dist/core/path-security.js +287 -5
  171. package/dist/core/permission.js +82 -22
  172. package/dist/core/permissions/auto-classifier.js +124 -0
  173. package/dist/core/permissions/bash-parser.js +371 -0
  174. package/dist/core/permissions/circuit-breaker.js +83 -0
  175. package/dist/core/permissions/constrained-edit.js +91 -0
  176. package/dist/core/permissions/gate.js +278 -0
  177. package/dist/core/permissions/index.js +20 -0
  178. package/dist/core/permissions/mode.js +174 -0
  179. package/dist/core/permissions/network-egress.js +137 -0
  180. package/dist/core/permissions/state.js +241 -0
  181. package/dist/core/permissions/tool-class.js +93 -0
  182. package/dist/core/plan-mode/ui-state.js +51 -0
  183. package/dist/core/plans/plan-artifact.js +721 -0
  184. package/dist/core/policy-limits/etag-store.js +122 -0
  185. package/dist/core/prd-check/parser.js +215 -0
  186. package/dist/core/prd-check/reporter.js +127 -0
  187. package/dist/core/prd-check/session-review.js +557 -0
  188. package/dist/core/prd-check/verifiers.js +223 -0
  189. package/dist/core/prompt-cache/client-cache.js +99 -0
  190. package/dist/core/prompts/assembly.js +29 -0
  191. package/dist/core/prompts/registry.js +364 -0
  192. package/dist/core/pugi-md/cc-compat-rules.js +735 -0
  193. package/dist/core/pugi-md/context-injector.js +76 -0
  194. package/dist/core/pugi-md/walk-up.js +207 -0
  195. package/dist/core/python/uv-installer.js +270 -0
  196. package/dist/core/python/uv-resolver.js +83 -0
  197. package/dist/core/rate-limit/narrator.js +146 -0
  198. package/dist/core/recipes/cli-types.js +20 -0
  199. package/dist/core/recipes/loader.js +103 -0
  200. package/dist/core/recipes/runner.js +345 -0
  201. package/dist/core/recipes/schema.js +587 -0
  202. package/dist/core/release-notes/parser.js +241 -0
  203. package/dist/core/release-notes/state.js +116 -0
  204. package/dist/core/repl/ask.js +37 -37
  205. package/dist/core/repl/cancellation.js +26 -26
  206. package/dist/core/repl/cap-warning.js +4 -4
  207. package/dist/core/repl/clipboard-read.js +11 -11
  208. package/dist/core/repl/dispatch-fsm.js +12 -12
  209. package/dist/core/repl/history-search.js +15 -15
  210. package/dist/core/repl/history.js +28 -18
  211. package/dist/core/repl/kill-ring.js +5 -5
  212. package/dist/core/repl/model-pricing.js +135 -0
  213. package/dist/core/repl/privacy-banner.js +22 -22
  214. package/dist/core/repl/session.js +2157 -214
  215. package/dist/core/repl/slash-commands.js +533 -40
  216. package/dist/core/repl/store/index.js +1 -1
  217. package/dist/core/repl/store/jsonl-log.js +22 -22
  218. package/dist/core/repl/store/lockfile.js +10 -10
  219. package/dist/core/repl/store/session-store.js +136 -107
  220. package/dist/core/repl/store/types.js +15 -15
  221. package/dist/core/repl/store/uuid-v7.js +12 -12
  222. package/dist/core/repl/workspace-context.js +43 -21
  223. package/dist/core/repo-map/build.js +125 -0
  224. package/dist/core/repo-map/cache.js +185 -0
  225. package/dist/core/repo-map/extractor.js +254 -0
  226. package/dist/core/repo-map/formatter.js +145 -0
  227. package/dist/core/repo-map/page-rank.js +105 -0
  228. package/dist/core/repo-map/scanner.js +211 -0
  229. package/dist/core/retry-budget/budget.js +284 -0
  230. package/dist/core/retry-budget/index.js +5 -0
  231. package/dist/core/retry-budget/retry-cap.js +74 -0
  232. package/dist/core/routing/lead-worker.js +43 -0
  233. package/dist/core/routing/pre-flight-estimator.js +108 -0
  234. package/dist/core/runs/run-tree.js +103 -0
  235. package/dist/core/security/injection-scanner.js +367 -0
  236. package/dist/core/security/output-filter.js +418 -0
  237. package/dist/core/session/env-file.js +105 -0
  238. package/dist/core/session/section-budgets.js +140 -0
  239. package/dist/core/session.js +92 -0
  240. package/dist/core/settings.js +286 -5
  241. package/dist/core/share/formatter.js +271 -0
  242. package/dist/core/share/redactor.js +221 -0
  243. package/dist/core/share/uploader.js +267 -0
  244. package/dist/core/skills/defaults.js +457 -0
  245. package/dist/core/skills/loader.js +22 -22
  246. package/dist/core/skills/sources.js +27 -27
  247. package/dist/core/smoke/headless-driver.js +174 -0
  248. package/dist/core/smoke/orchestrator.js +194 -0
  249. package/dist/core/smoke/runner.js +238 -0
  250. package/dist/core/smoke/scenario-parser.js +316 -0
  251. package/dist/core/statusline.js +99 -0
  252. package/dist/core/subagents/dispatcher-real.js +600 -0
  253. package/dist/core/subagents/dispatcher.js +132 -43
  254. package/dist/core/subagents/index.js +19 -6
  255. package/dist/core/subagents/isolation-matrix.js +213 -0
  256. package/dist/core/subagents/spawn.js +19 -4
  257. package/dist/core/telemetry/emitter.js +229 -0
  258. package/dist/core/telemetry/queue.js +251 -0
  259. package/dist/core/theme/context.js +91 -0
  260. package/dist/core/theme/presets.js +228 -0
  261. package/dist/core/theme/state.js +181 -0
  262. package/dist/core/todos/invariant.js +10 -0
  263. package/dist/core/todos/state.js +177 -0
  264. package/dist/core/tool-schema/compressor.js +89 -0
  265. package/dist/core/transport/version-interceptor.js +166 -0
  266. package/dist/core/trust.js +2 -2
  267. package/dist/core/tui/thinking-block.js +64 -0
  268. package/dist/core/vim/keymap.js +288 -0
  269. package/dist/core/vim/state.js +92 -0
  270. package/dist/core/watch-markers/marker-watcher.js +133 -0
  271. package/dist/core/worktree-manager/cleanup.js +123 -0
  272. package/dist/core/worktree-manager/manager.js +303 -0
  273. package/dist/index.js +28 -0
  274. package/dist/runtime/bootstrap.js +190 -0
  275. package/dist/runtime/cli.js +4162 -488
  276. package/dist/runtime/commands/agents.js +30 -30
  277. package/dist/runtime/commands/budget.js +5 -5
  278. package/dist/runtime/commands/cancel.js +231 -0
  279. package/dist/runtime/commands/chain.js +489 -0
  280. package/dist/runtime/commands/codegraph-status.js +227 -0
  281. package/dist/runtime/commands/compact.js +297 -0
  282. package/dist/runtime/commands/config.js +32 -32
  283. package/dist/runtime/commands/cost.js +199 -0
  284. package/dist/runtime/commands/delegate.js +244 -13
  285. package/dist/runtime/commands/dispatch.js +126 -0
  286. package/dist/runtime/commands/doctor.js +579 -0
  287. package/dist/runtime/commands/feedback.js +184 -0
  288. package/dist/runtime/commands/hooks.js +184 -0
  289. package/dist/runtime/commands/init.js +254 -0
  290. package/dist/runtime/commands/lsp.js +368 -0
  291. package/dist/runtime/commands/mcp.js +879 -0
  292. package/dist/runtime/commands/memory.js +582 -0
  293. package/dist/runtime/commands/model.js +237 -0
  294. package/dist/runtime/commands/onboarding.js +275 -0
  295. package/dist/runtime/commands/patch.js +128 -0
  296. package/dist/runtime/commands/permissions.js +112 -0
  297. package/dist/runtime/commands/plan.js +143 -0
  298. package/dist/runtime/commands/prd-check.js +285 -0
  299. package/dist/runtime/commands/privacy.js +17 -17
  300. package/dist/runtime/commands/recipe.js +325 -0
  301. package/dist/runtime/commands/redo-blob-store.js +92 -0
  302. package/dist/runtime/commands/redo.js +361 -0
  303. package/dist/runtime/commands/release-notes.js +229 -0
  304. package/dist/runtime/commands/repo-map.js +95 -0
  305. package/dist/runtime/commands/report.js +299 -0
  306. package/dist/runtime/commands/resume.js +118 -0
  307. package/dist/runtime/commands/review-consensus.js +68 -53
  308. package/dist/runtime/commands/rewind.js +333 -0
  309. package/dist/runtime/commands/roster.js +14 -14
  310. package/dist/runtime/commands/sessions.js +163 -0
  311. package/dist/runtime/commands/share.js +316 -0
  312. package/dist/runtime/commands/skills.js +31 -31
  313. package/dist/runtime/commands/status.js +186 -0
  314. package/dist/runtime/commands/stickers.js +82 -0
  315. package/dist/runtime/commands/style.js +194 -0
  316. package/dist/runtime/commands/theme.js +196 -0
  317. package/dist/runtime/commands/undo.js +54 -22
  318. package/dist/runtime/commands/update.js +289 -0
  319. package/dist/runtime/commands/vim.js +140 -0
  320. package/dist/runtime/commands/worktree.js +177 -0
  321. package/dist/runtime/commands/worktrees.js +155 -0
  322. package/dist/runtime/headless-repl.js +195 -0
  323. package/dist/runtime/headless.js +543 -0
  324. package/dist/runtime/load-hooks-or-exit.js +71 -0
  325. package/dist/runtime/plan-decompose.js +531 -0
  326. package/dist/runtime/update-check.js +28 -28
  327. package/dist/runtime/version.js +65 -0
  328. package/dist/skills/bundled/batch.js +617 -0
  329. package/dist/skills/bundled/index.js +45 -0
  330. package/dist/skills/bundled/loop.js +358 -0
  331. package/dist/skills/bundled/remember.js +383 -0
  332. package/dist/skills/bundled/simplify.js +289 -0
  333. package/dist/skills/bundled/skillify.js +373 -0
  334. package/dist/skills/bundled/stuck.js +558 -0
  335. package/dist/skills/bundled/verify.js +439 -0
  336. package/dist/testing/vcr.js +486 -0
  337. package/dist/tools/agent-tool.js +229 -0
  338. package/dist/tools/apply-patch.js +556 -0
  339. package/dist/tools/ask-user-question.js +222 -0
  340. package/dist/tools/ask-user.js +115 -0
  341. package/dist/tools/bash.js +623 -45
  342. package/dist/tools/brief.js +224 -0
  343. package/dist/tools/enter-worktree.js +250 -0
  344. package/dist/tools/exit-worktree.js +147 -0
  345. package/dist/tools/file-tools.js +161 -44
  346. package/dist/tools/lsp-tools.js +189 -0
  347. package/dist/tools/mcp-tool.js +260 -0
  348. package/dist/tools/multi-edit.js +361 -0
  349. package/dist/tools/powershell.js +268 -0
  350. package/dist/tools/registry.js +85 -0
  351. package/dist/tools/skill-tool.js +96 -0
  352. package/dist/tools/sleep.js +99 -0
  353. package/dist/tools/synthetic-output.js +133 -0
  354. package/dist/tools/tasks.js +208 -0
  355. package/dist/tools/todo-write.js +184 -0
  356. package/dist/tools/verify-plan-execution.js +295 -0
  357. package/dist/tools/web-fetch-injection-scanner.js +207 -0
  358. package/dist/tools/web-fetch.js +195 -10
  359. package/dist/tools/web-search.js +458 -0
  360. package/dist/tui/agent-progress-card.js +111 -0
  361. package/dist/tui/agent-tree.js +11 -1
  362. package/dist/tui/ask-modal.js +14 -14
  363. package/dist/tui/ask-user-question-prompt.js +203 -0
  364. package/dist/tui/compact-banner.js +81 -0
  365. package/dist/tui/conversation-pane.js +85 -11
  366. package/dist/tui/cost-table.js +111 -0
  367. package/dist/tui/device-flow.js +2 -2
  368. package/dist/tui/doctor-table.js +46 -0
  369. package/dist/tui/feedback-prompt.js +156 -0
  370. package/dist/tui/input-box.js +247 -32
  371. package/dist/tui/login-picker.js +3 -3
  372. package/dist/tui/markdown-render.js +6 -6
  373. package/dist/tui/onboarding-wizard.js +240 -0
  374. package/dist/tui/permissions-picker.js +86 -0
  375. package/dist/tui/render.js +35 -0
  376. package/dist/tui/repl-render.js +332 -54
  377. package/dist/tui/repl-splash-art.js +16 -16
  378. package/dist/tui/repl-splash-mascot.js +48 -24
  379. package/dist/tui/repl-splash.js +22 -22
  380. package/dist/tui/repl.js +124 -44
  381. package/dist/tui/slash-palette.js +6 -6
  382. package/dist/tui/splash.js +2 -2
  383. package/dist/tui/status-bar.js +109 -31
  384. package/dist/tui/status-table.js +7 -0
  385. package/dist/tui/stickers-art.js +136 -0
  386. package/dist/tui/style-table.js +28 -0
  387. package/dist/tui/theme-table.js +29 -0
  388. package/dist/tui/thinking-spinner.js +123 -0
  389. package/dist/tui/tool-stream-pane.js +53 -4
  390. package/dist/tui/update-banner.js +27 -2
  391. package/dist/tui/vim-input.js +267 -0
  392. package/dist/tui/welcome-banner.js +107 -0
  393. package/dist/tui/welcome-data.js +293 -0
  394. package/dist/tui/workspace-context.js +2 -2
  395. package/docs/examples/codegraph.mcp.json +10 -0
  396. package/package.json +23 -6
  397. package/test/scenarios/codegen-create-file.scenario.txt +13 -0
  398. package/test/scenarios/compact-force.scenario.txt +11 -0
  399. package/test/scenarios/identity.scenario.txt +11 -0
  400. package/test/scenarios/persona-handoff.scenario.txt +11 -0
  401. package/test/scenarios/walkback.scenario.txt +12 -0
  402. package/dist/core/engine/compaction-hook.js +0 -154
@@ -18,6 +18,98 @@ export function openSession(root) {
18
18
  enabled,
19
19
  };
20
20
  }
21
+ /**
22
+ * MVP — fire the `SessionStart` lifecycle event for all hooks
23
+ * declared in `~/.pugi/hooks-mvp.json`. Single-call surface; the REPL
24
+ * boot path invokes this once after `openSession`. Best-effort: any
25
+ * failure (missing config, hook spawn error) is swallowed so a
26
+ * misconfigured hook can never crash the REPL.
27
+ *
28
+ * Returns the number of hooks that fired (0 when no config / no
29
+ * matching hooks). Tests assert on the return value as the
30
+ * single-call invariant.
31
+ */
32
+ export async function fireSessionStartMvp(session) {
33
+ try {
34
+ const { loadHooksConfig, fireHooks } = await import('./hooks/index.js');
35
+ // Defense-in-depth: `loadHooksConfig` is contractually non-null
36
+ // (returns `HooksConfig.empty(path)` when the file is absent), but
37
+ // the dynamic import boundary above can in principle return an
38
+ // unexpected shape if the module is mis-resolved at runtime. Guard
39
+ // the optional-chained `isEmpty()` call so a malformed loader can
40
+ // never raise `TypeError: Cannot read properties of undefined` and
41
+ // crash the REPL boot path. Belt-and-suspenders with the
42
+ // surrounding try/catch — the catch still swallows everything else.
43
+ const config = loadHooksConfig();
44
+ if (!config || config.isEmpty())
45
+ return 0;
46
+ const outcome = await fireHooks({
47
+ config,
48
+ event: 'SessionStart',
49
+ payload: {
50
+ event: 'SessionStart',
51
+ sessionId: session.id,
52
+ workspaceRoot: session.root,
53
+ startedAt: new Date().toISOString(),
54
+ },
55
+ workspaceRoot: session.root,
56
+ });
57
+ return outcome.results.length;
58
+ }
59
+ catch {
60
+ // SessionStart is never blocking — log nothing, return 0. A
61
+ // broken `hooks-mvp.json` is surfaced via `pugi hooks doctor`.
62
+ return 0;
63
+ }
64
+ }
65
+ /**
66
+ * P1 — fire the v2 `SessionStart` event from `~/.pugi/hooks.json`
67
+ * (global) + `<workspaceRoot>/.pugi/hooks.json` (project). Companion to
68
+ * `fireSessionStartMvp`; both surfaces run because they read different
69
+ * config files.
70
+ *
71
+ * Headless by default (no trust prompt) — the v2 trust ledger gates
72
+ * first-run executions. Operators with no prior trust decision will see
73
+ * the SessionStart hook skipped with a `denied by trust ledger` stderr
74
+ * note; running `pugi hooks trust allow <command>` enrolls it.
75
+ *
76
+ * Returns the number of hooks that ran (excluding trust-denied skips).
77
+ * Never throws.
78
+ */
79
+ export async function fireSessionStartV2(session) {
80
+ try {
81
+ const { fireSessionStart } = await import('./hooks/v2/index.js');
82
+ const outcome = await fireSessionStart({
83
+ sessionId: session.id,
84
+ workspaceRoot: session.root,
85
+ transcriptPath: session.eventsPath,
86
+ permissionMode: 'ask',
87
+ });
88
+ return outcome.results.filter((r) => r.exitCode !== -1).length;
89
+ }
90
+ catch {
91
+ return 0;
92
+ }
93
+ }
94
+ /**
95
+ * P1 — fire the v2 `SessionEnd` event. Called by the REPL
96
+ * teardown path. Companion to `fireSessionStartV2`.
97
+ */
98
+ export async function fireSessionEndV2(session) {
99
+ try {
100
+ const { fireSessionEnd } = await import('./hooks/v2/index.js');
101
+ const outcome = await fireSessionEnd({
102
+ sessionId: session.id,
103
+ workspaceRoot: session.root,
104
+ transcriptPath: session.eventsPath,
105
+ permissionMode: 'ask',
106
+ });
107
+ return outcome.results.filter((r) => r.exitCode !== -1).length;
108
+ }
109
+ catch {
110
+ return 0;
111
+ }
112
+ }
21
113
  export function recordCommandStarted(session, command) {
22
114
  if (!session.enabled)
23
115
  return;
@@ -20,6 +20,22 @@ const pugiSettingsSchema = z.object({
20
20
  allow: z.array(z.string()).default([]),
21
21
  deny: z.array(z.string()).default([]),
22
22
  notAutomatic: z.array(z.string()).default([]),
23
+ // task — operator-declared read-only paths. Every
24
+ // edit / write tool call whose target matches one of these
25
+ // patterns is denied with source `readonly_paths`, regardless
26
+ // of the active permission mode. Match grammar mirrors
27
+ // allow/deny: exact path OR tail-* glob. Targets are
28
+ // workspace-relative, no leading slash.
29
+ //
30
+ // Why a dedicated field вместо leaning on a generic deny rule
31
+ // like `deny: ["edit:vendor/x"]`?
32
+ // - Explicit intent surfaces в `pugi doctor` and the REPL
33
+ // ("3 paths read-only") без parsing rule strings.
34
+ // - Covers BOTH the `edit` tool AND the `write` tool with
35
+ // one entry; a generic deny needs two rules.
36
+ // - Survives a future migration of the `permissions` schema
37
+ // because the field carries its own contract.
38
+ readonlyPaths: z.array(z.string()).default([]),
23
39
  })
24
40
  .default({}),
25
41
  privacy: z
@@ -28,6 +44,17 @@ const pugiSettingsSchema = z.object({
28
44
  telemetry: z.enum(['off', 'anonymous', 'community']).default('off'),
29
45
  })
30
46
  .default({}),
47
+ // beta.13 P1 fix: ui.cyberZoo gates the cyber-zoo splash +
48
+ // ambient art in the REPL. Schema must declare the key explicitly
49
+ // because Zod's strip pass swallows unknown keys, which is how the
50
+ // initial `pugi init` write (which serialises `ui.cyberZoo`) was
51
+ // bypassed by the runtime reader — the value never made it past the
52
+ // schema gate so admin-api always saw the historical 'on' default.
53
+ ui: z
54
+ .object({
55
+ cyberZoo: z.enum(['on', 'off']).default('on'),
56
+ })
57
+ .default({}),
31
58
  artifacts: z
32
59
  .object({
33
60
  defaultPath: z.string().default('.pugi/artifacts'),
@@ -38,6 +65,12 @@ const pugiSettingsSchema = z.object({
38
65
  // fetcher. Default-off matches the spec posture; the schema must
39
66
  // declare it explicitly because Zod's strict-pass strips unknown
40
67
  // keys and would silently swallow the operator's intent.
68
+ //
69
+ // β1b T4 : added `web.search.enabled` to gate the
70
+ // Brave-Search-backed `web_search` tool. Distinct from `web.fetch`
71
+ // because search queries themselves are an egress event that can
72
+ // leak operator intent — an operator may want fetch without
73
+ // implicitly enabling search-as-egress.
41
74
  web: z
42
75
  .object({
43
76
  fetch: z
@@ -45,15 +78,263 @@ const pugiSettingsSchema = z.object({
45
78
  enabled: z.boolean().optional(),
46
79
  })
47
80
  .optional(),
81
+ search: z
82
+ .object({
83
+ enabled: z.boolean().optional(),
84
+ })
85
+ .optional(),
86
+ })
87
+ .optional(),
88
+ // β7 L9 — per-language LSP toggle. When omitted, every supported
89
+ // server is available subject to binary detection on PATH. When
90
+ // present, only languages set to `true` are launched (false silently
91
+ // skips that language even if the binary is installed). Use this in
92
+ // workspaces where a heavyweight server (rust-analyzer indexing a
93
+ // monorepo, pyright on a fresh venv) wastes resources for the
94
+ // current task. The `pugi lsp servers` subcommand surfaces the
95
+ // current toggle state per server.
96
+ //
97
+ // Schema is intentionally permissive (`optional()` on the section AND
98
+ // on every per-language flag) so a partial config keeps the
99
+ // backwards-compatible "every language enabled" default.
100
+ lsp: z
101
+ .object({
102
+ typescript: z.boolean().optional(),
103
+ javascript: z.boolean().optional(),
104
+ python: z.boolean().optional(),
105
+ go: z.boolean().optional(),
106
+ rust: z.boolean().optional(),
107
+ // post-edit auto-diagnostics. When `true`,
108
+ // a successful `edit`/`write`/`multi_edit` triggers a diagnostic
109
+ // pull on the touched file(s) and the result is appended to the
110
+ // tool envelope so the model can self-correct in the same turn.
111
+ // Off by default — the cold-start of `typescript-language-server`
112
+ // is heavy enough that we opt in explicitly until dogfood proves
113
+ // the throughput trade is worth it. Also enabled via env var
114
+ // `PUGI_LSP_POST_EDIT=1` for CI / one-off operator probes.
115
+ postEditDiagnostics: z.boolean().optional(),
48
116
  })
49
117
  .optional(),
118
+ // β1 Pl9 — per-command budget overrides. Optional. Partial
119
+ // overrides merge against the β1 defaults in
120
+ // `core/engine/budgets.ts::beta1DefaultBudgets`. The schema is
121
+ // intentionally loose at the leaf (positive integers) so a typo lands
122
+ // a deterministic `BudgetConfigError` at `resolveBudget()` instead of
123
+ // a Zod parse error two layers up.
124
+ // task — operator-customizable status line.
125
+ // When `command` is set, Pugi spawns it on each turn boundary with
126
+ // a single JSON document on stdin (see `statusline.ts` for the
127
+ // schema). The command's stdout (first non-empty line, trimmed) is
128
+ // displayed in the Ink Footer. Failures are non-fatal — they emit
129
+ // a structured log line and the footer falls back to the default.
130
+ // Mirrors CC's `statusLine` config so cross-tool muscle memory
131
+ // carries over для operators who lived in CC first.
132
+ statusLine: z
133
+ .object({
134
+ command: z.string().min(1),
135
+ timeoutMs: z.number().int().positive().max(5000).default(500),
136
+ })
137
+ .optional(),
138
+ budgets: z
139
+ .object({
140
+ code: z
141
+ .object({ maxTokens: z.number().int().positive().optional(), maxToolCalls: z.number().int().positive().optional() })
142
+ .optional(),
143
+ fix: z
144
+ .object({ maxTokens: z.number().int().positive().optional(), maxToolCalls: z.number().int().positive().optional() })
145
+ .optional(),
146
+ build: z
147
+ .object({ maxTokens: z.number().int().positive().optional(), maxToolCalls: z.number().int().positive().optional() })
148
+ .optional(),
149
+ plan: z
150
+ .object({ maxTokens: z.number().int().positive().optional(), maxToolCalls: z.number().int().positive().optional() })
151
+ .optional(),
152
+ explain: z
153
+ .object({ maxTokens: z.number().int().positive().optional(), maxToolCalls: z.number().int().positive().optional() })
154
+ .optional(),
155
+ review_triple: z
156
+ .object({ maxTokens: z.number().int().positive().optional(), maxToolCalls: z.number().int().positive().optional() })
157
+ .optional(),
158
+ })
159
+ .optional(),
160
+ // #24 (CEO P1) — hook chains. First-class config
161
+ // for `PostToolUseFailure` + `TaskCompleted` event chains. The shape
162
+ // is intentionally loose (`z.any()` at the leaf) because the
163
+ // canonical reader lives in `core/hook-chains.ts` where the
164
+ // matcher/run/timeoutMs grammar is validated. Declaring the key here
165
+ // keeps Zod's strip-pass from swallowing it before the chain reader
166
+ // sees it. See `hook-chains.ts` for the full schema.
167
+ hooks: z.any().optional(),
50
168
  });
51
- export function loadSettings(root) {
169
+ /**
170
+ * #20 — the upstream tool drop-in compat ingest.
171
+ *
172
+ * Operators migrating from the upstream tool typically keep a `.claude/`
173
+ * directory at workspace root with settings.json, slash commands,
174
+ * and ambient guidance files. We honour the existence of that
175
+ * directory and mirror the subset of keys that map cleanly onto
176
+ * Pugi's own settings surface — Pugi values ALWAYS win on conflict
177
+ * (the operator opted into Pugi as their primary), CC fills gaps.
178
+ *
179
+ * Opt-out: `PUGI_CC_COMPAT_DISABLE=1` short-circuits the merger and
180
+ * loads only `.pugi/settings.json` (or the empty default).
181
+ *
182
+ * Key mirror table:
183
+ * - `permissions.defaultMode` → `permissions.mode`
184
+ * (CC values map: `acceptEdits|plan|bypassPermissions|default` →
185
+ * `acceptEdits|plan|bypassPermissions|ask`).
186
+ * - `permissions.allow` → `permissions.allow` (concatenated, deduped).
187
+ * - `permissions.deny` → `permissions.deny` (concatenated, deduped).
188
+ * - `enabledPlugins` → ignored (CC-only concept; Pugi has its own
189
+ * plugin surface and we do not want to silently activate them).
190
+ * - `hooks` → currently ignored. Pugi's hook system is
191
+ * managed via `apps/pugi-cli/src/core/hooks/`; future work can
192
+ * wire CC hook entries through that bridge.
193
+ *
194
+ * Unknown CC keys are dropped on the floor by Zod's strip semantics
195
+ * just like the existing PUGI settings path — we never warn on
196
+ * unrecognised CC keys, because the CC surface is wider and we want
197
+ * fallthrough to be silent (operator does not need a stream of "we
198
+ * skipped this CC concept" warnings on every CLI invocation).
199
+ */
200
+ const ccPermissionsSchema = z
201
+ .object({
202
+ defaultMode: z.string().optional(),
203
+ allow: z.array(z.string()).optional(),
204
+ deny: z.array(z.string()).optional(),
205
+ })
206
+ .passthrough()
207
+ .optional();
208
+ const ccSettingsSchema = z
209
+ .object({
210
+ permissions: ccPermissionsSchema,
211
+ enabledPlugins: z.unknown().optional(),
212
+ hooks: z.unknown().optional(),
213
+ })
214
+ .passthrough();
215
+ /**
216
+ * Env var that disables ingest entirely. Useful for CI
217
+ * sandboxes where a stray `.claude/` from a parent checkout could
218
+ * otherwise leak permissions into Pugi.
219
+ */
220
+ export const CC_COMPAT_DISABLE_ENV = 'PUGI_CC_COMPAT_DISABLE';
221
+ /**
222
+ * Map a CC `permissions.defaultMode` to the closest Pugi permission
223
+ * mode. Unknown / missing values map to `undefined` so the caller
224
+ * keeps Pugi's own default.
225
+ *
226
+ * CC's `default` mode = "ask the user for each tool" which is Pugi's
227
+ * `ask` mode. `acceptEdits` / `plan` / `bypassPermissions` map 1:1.
228
+ */
229
+ export function mapCcPermissionMode(mode) {
230
+ if (typeof mode !== 'string')
231
+ return undefined;
232
+ switch (mode) {
233
+ case 'acceptEdits':
234
+ return 'acceptEdits';
235
+ case 'plan':
236
+ return 'plan';
237
+ case 'bypassPermissions':
238
+ return 'bypassPermissions';
239
+ case 'default':
240
+ return 'ask';
241
+ default:
242
+ return undefined;
243
+ }
244
+ }
245
+ /**
246
+ * Merge a parsed CC settings object into a Pugi settings object.
247
+ * Pugi ALWAYS wins on conflict; CC values fill gaps only.
248
+ */
249
+ export function mergeCcIntoPugi(pugi, cc, opts) {
250
+ const merged = {
251
+ ...pugi,
252
+ permissions: { ...pugi.permissions },
253
+ };
254
+ const pugiWroteMode = pugiPermissionKeyPresent(opts.pugiRawJson, 'mode');
255
+ if (!pugiWroteMode) {
256
+ const ccMode = mapCcPermissionMode(cc.permissions?.defaultMode);
257
+ if (ccMode)
258
+ merged.permissions.mode = ccMode;
259
+ }
260
+ if (Array.isArray(cc.permissions?.allow)) {
261
+ merged.permissions.allow = dedupeKeepFirst([
262
+ ...pugi.permissions.allow,
263
+ ...cc.permissions.allow,
264
+ ]);
265
+ }
266
+ if (Array.isArray(cc.permissions?.deny)) {
267
+ merged.permissions.deny = dedupeKeepFirst([
268
+ ...pugi.permissions.deny,
269
+ ...cc.permissions.deny,
270
+ ]);
271
+ }
272
+ // `enabledPlugins` and `hooks` are intentionally NOT mirrored. See
273
+ // the doc-block above for rationale.
274
+ void opts.pugiSettingsExisted;
275
+ return merged;
276
+ }
277
+ function pugiPermissionKeyPresent(raw, key) {
278
+ if (!raw || typeof raw !== 'object')
279
+ return false;
280
+ const permissions = raw.permissions;
281
+ if (!permissions || typeof permissions !== 'object')
282
+ return false;
283
+ return Object.prototype.hasOwnProperty.call(permissions, key);
284
+ }
285
+ function dedupeKeepFirst(items) {
286
+ const seen = new Set();
287
+ const out = [];
288
+ for (const item of items) {
289
+ if (seen.has(item))
290
+ continue;
291
+ seen.add(item);
292
+ out.push(item);
293
+ }
294
+ return out;
295
+ }
296
+ /**
297
+ * Read + parse `.claude/settings.json` at `root`. Returns `undefined`
298
+ * when the file is absent, malformed, or the operator has opted out
299
+ * via `PUGI_CC_COMPAT_DISABLE=1`. Never throws — a broken CC settings
300
+ * file degrades to "no ingest", not a Pugi boot crash.
301
+ */
302
+ export function loadCcSettings(root, env = process.env) {
303
+ if (env[CC_COMPAT_DISABLE_ENV] === '1')
304
+ return undefined;
305
+ const ccPath = resolve(root, '.claude/settings.json');
306
+ if (!existsSync(ccPath))
307
+ return undefined;
308
+ let parsed;
309
+ try {
310
+ parsed = JSON.parse(readFileSync(ccPath, 'utf8'));
311
+ }
312
+ catch {
313
+ return undefined;
314
+ }
315
+ const result = ccSettingsSchema.safeParse(parsed);
316
+ if (!result.success)
317
+ return undefined;
318
+ return result.data;
319
+ }
320
+ export function loadSettings(root, env = process.env) {
52
321
  const settingsPath = resolve(root, '.pugi/settings.json');
53
- if (!existsSync(settingsPath)) {
54
- return pugiSettingsSchema.parse({});
322
+ const pugiExists = existsSync(settingsPath);
323
+ let pugiRawJson = undefined;
324
+ let pugi;
325
+ if (pugiExists) {
326
+ pugiRawJson = JSON.parse(readFileSync(settingsPath, 'utf8'));
327
+ pugi = pugiSettingsSchema.parse(pugiRawJson);
328
+ }
329
+ else {
330
+ pugi = pugiSettingsSchema.parse({});
55
331
  }
56
- const parsed = JSON.parse(readFileSync(settingsPath, 'utf8'));
57
- return pugiSettingsSchema.parse(parsed);
332
+ const cc = loadCcSettings(root, env);
333
+ if (!cc)
334
+ return pugi;
335
+ return mergeCcIntoPugi(pugi, cc, {
336
+ pugiSettingsExisted: pugiExists,
337
+ pugiRawJson,
338
+ });
58
339
  }
59
340
  //# sourceMappingURL=settings.js.map
@@ -0,0 +1,271 @@
1
+ /**
2
+ * Markdown transcript formatter used by `pugi share` ().
3
+ *
4
+ * Walks the session's `.pugi/events.jsonl` audit log and reconstructs a
5
+ * Markdown document the operator (and downstream Gist / pugi.io readers)
6
+ * can read top-to-bottom. The format is intentionally human-first — turn
7
+ * headers, code-block-wrapped tool I/O, and a small front-matter block
8
+ * with session metadata. Machine-readability is a non-goal here; the
9
+ * JSONL log remains the source of truth for tooling.
10
+ *
11
+ * Why we reconstruct from JSONL instead of from the live REPL state:
12
+ *
13
+ * - The CLI top-level `pugi share` command runs from a fresh shell and
14
+ * has no in-memory REPL state to read; only the event log is
15
+ * persisted.
16
+ * - The in-REPL `/share` slash uses the same handler so behaviour is
17
+ * identical regardless of entry point. Operators sharing a session
18
+ * from inside the REPL get the exact same output they would get from
19
+ * a follow-up shell command.
20
+ * - The JSONL log is append-only and survives REPL crashes, so a
21
+ * `--share` after a crash is the most useful debug surface.
22
+ *
23
+ * Event vocabulary the formatter knows about (see
24
+ * `packages/pugi-sdk/src/audit-trace.ts` for the schema):
25
+ *
26
+ * session.created Session boundary; emits front matter.
27
+ * session.command_started One-line "running pugi <cmd>" header.
28
+ * session.command_completed Status line ("success" / "error").
29
+ * tool_call Markdown turn header + inputSummary
30
+ * rendered as a fenced block.
31
+ * tool_result "Result (status):" + outputSummary
32
+ * rendered as a fenced block.
33
+ * file_mutation Inline `path` + operation summary.
34
+ * subagent.* Indented "[subagent <role>] ..." line.
35
+ * hook.* Quiet "[hook <event>] ..." line.
36
+ * compaction.* "[compaction <tier>] ..." line.
37
+ *
38
+ * Unknown event types are surfaced as a single italic line ("[event
39
+ * type=...]") so a future event added to the SDK does not silently
40
+ * vanish from the transcript.
41
+ *
42
+ * Performance: the formatter is O(n) over the events file, runs entirely
43
+ * in memory, and is bounded by the session log size (currently capped at
44
+ * a few MB per session). No streaming I/O is needed for typical sessions
45
+ * — the operator does not run /share against multi-GB logs.
46
+ */
47
+ /**
48
+ * Format a session's event log as Markdown. Pure — no I/O. The caller
49
+ * reads `.pugi/events.jsonl` and hands the contents in.
50
+ */
51
+ export function formatTranscript(input) {
52
+ const now = input.now ? input.now() : new Date();
53
+ const events = parseEvents(input.eventsJsonl);
54
+ const filteredSessionId = pickSessionId(input.sessionId, events);
55
+ const sessionEvents = events.filter((e) => typeof e.raw.sessionId !== 'string' || e.raw.sessionId === filteredSessionId);
56
+ const lines = [];
57
+ // Front matter — a fenced block at the top so downstream readers can
58
+ // grok the context before any turn content. Not YAML front matter
59
+ // (`---`) because Gist + pugi.io render Markdown directly; the fenced
60
+ // approach renders predictably without a parser.
61
+ lines.push('# Pugi session transcript');
62
+ lines.push('');
63
+ lines.push('```');
64
+ lines.push(`session_id: ${filteredSessionId}`);
65
+ lines.push(`workspace: ${input.workspaceRoot}`);
66
+ lines.push(`cli_version: ${input.cliVersion}`);
67
+ lines.push(`exported_at: ${now.toISOString()}`);
68
+ lines.push(`event_count: ${sessionEvents.length}`);
69
+ lines.push('```');
70
+ lines.push('');
71
+ if (sessionEvents.length === 0) {
72
+ lines.push('_No events recorded for this session._');
73
+ return {
74
+ markdown: `${lines.join('\n')}\n`,
75
+ turnCount: 0,
76
+ eventCount: 0,
77
+ };
78
+ }
79
+ let turnCount = 0;
80
+ for (const event of sessionEvents) {
81
+ const rendered = renderEvent(event);
82
+ if (rendered === null)
83
+ continue;
84
+ lines.push(...rendered.lines);
85
+ lines.push('');
86
+ if (rendered.isTurn)
87
+ turnCount += 1;
88
+ }
89
+ return {
90
+ markdown: `${lines.join('\n').replace(/\n{3,}/g, '\n\n')}\n`,
91
+ turnCount,
92
+ eventCount: sessionEvents.length,
93
+ };
94
+ }
95
+ /**
96
+ * Parse the JSONL log. Malformed lines are skipped silently — the file
97
+ * is append-only and may have partial-write tail rows. Returning the
98
+ * stable typed shape lets the formatter walk without re-checking every
99
+ * field.
100
+ */
101
+ function parseEvents(raw) {
102
+ const out = [];
103
+ for (const line of raw.split('\n')) {
104
+ const trimmed = line.trim();
105
+ if (trimmed.length === 0)
106
+ continue;
107
+ try {
108
+ const parsed = JSON.parse(trimmed);
109
+ const type = typeof parsed.type === 'string' ? parsed.type : '';
110
+ const timestamp = typeof parsed.timestamp === 'string' ? parsed.timestamp : '';
111
+ if (type.length === 0)
112
+ continue;
113
+ out.push({ raw: parsed, type, timestamp });
114
+ }
115
+ catch {
116
+ // partial-write or corrupt row; skip without affecting the rest
117
+ }
118
+ }
119
+ return out;
120
+ }
121
+ /**
122
+ * Resolve the effective session id. When the operator passes a non-empty
123
+ * value we honour it. When they pass an empty string / placeholder
124
+ * (`'no-session'`), we fall back to the newest `session.created` event
125
+ * id in the file. Last-resort fallback is the literal placeholder so the
126
+ * transcript still renders something meaningful in the front matter.
127
+ */
128
+ function pickSessionId(provided, events) {
129
+ if (provided && provided !== 'no-session')
130
+ return provided;
131
+ for (let i = events.length - 1; i >= 0; i -= 1) {
132
+ const e = events[i];
133
+ if (!e)
134
+ continue;
135
+ if (e.type === 'session' && e.raw.name === 'created' && typeof e.raw.sessionId === 'string') {
136
+ return e.raw.sessionId;
137
+ }
138
+ }
139
+ return provided || 'unknown-session';
140
+ }
141
+ /**
142
+ * Format one event as a Markdown block. Returns `null` to suppress the
143
+ * event entirely (e.g. session.created is captured in front matter so we
144
+ * skip it here).
145
+ */
146
+ function renderEvent(event) {
147
+ const ts = event.timestamp || '';
148
+ switch (event.type) {
149
+ case 'session': {
150
+ const name = String(event.raw.name ?? '');
151
+ if (name === 'created')
152
+ return null; // captured by front matter
153
+ if (name === 'command_started') {
154
+ const command = String(event.raw.command ?? '');
155
+ return {
156
+ lines: [`## ${ts} — command \`${escapeInline(command)}\``],
157
+ isTurn: false,
158
+ };
159
+ }
160
+ if (name === 'command_completed') {
161
+ const command = String(event.raw.command ?? '');
162
+ const status = String(event.raw.status ?? 'unknown');
163
+ return {
164
+ lines: [`_command \`${escapeInline(command)}\` finished: ${status}_`],
165
+ isTurn: false,
166
+ };
167
+ }
168
+ return {
169
+ lines: [`_session ${name}_`],
170
+ isTurn: false,
171
+ };
172
+ }
173
+ case 'tool_call': {
174
+ const tool = String(event.raw.tool ?? 'unknown');
175
+ const summary = String(event.raw.inputSummary ?? '');
176
+ const out = [`### ${ts} — tool \`${escapeInline(tool)}\``];
177
+ if (summary.length > 0) {
178
+ out.push('');
179
+ out.push('Input:');
180
+ out.push(fenced(summary));
181
+ }
182
+ return { lines: out, isTurn: true };
183
+ }
184
+ case 'tool_result': {
185
+ const status = String(event.raw.status ?? 'unknown');
186
+ const summary = String(event.raw.outputSummary ?? '');
187
+ const out = [`Result (${status}):`];
188
+ if (summary.length > 0) {
189
+ out.push(fenced(summary));
190
+ }
191
+ return { lines: out, isTurn: false };
192
+ }
193
+ case 'file_mutation': {
194
+ const path = String(event.raw.path ?? '');
195
+ const op = String(event.raw.operation ?? '');
196
+ return {
197
+ lines: [`- file ${op}: \`${escapeInline(path)}\``],
198
+ isTurn: false,
199
+ };
200
+ }
201
+ case 'subagent.spawned':
202
+ case 'subagent.tool_call':
203
+ case 'subagent.completed':
204
+ case 'subagent.blocked':
205
+ case 'subagent.failed': {
206
+ const role = String(event.raw.role ?? '');
207
+ const persona = String(event.raw.personaSlug ?? '');
208
+ const detail = String(event.raw.detail ?? event.raw.error ?? event.raw.toolName ?? '');
209
+ const tail = detail.length > 0 ? ` ${detail}` : '';
210
+ return {
211
+ lines: [`_[subagent ${role} / ${persona}] ${event.type}${tail}_`],
212
+ isTurn: false,
213
+ };
214
+ }
215
+ case 'hook.invoked':
216
+ case 'hook.result':
217
+ case 'hook.skipped': {
218
+ const ev = String(event.raw.event ?? '');
219
+ const reason = String(event.raw.reason ?? event.raw.runSummary ?? event.raw.matchSummary ?? '');
220
+ const tail = reason.length > 0 ? ` ${reason}` : '';
221
+ return {
222
+ lines: [`_[hook ${ev}] ${event.type.replace('hook.', '')}${tail}_`],
223
+ isTurn: false,
224
+ };
225
+ }
226
+ case 'compaction.started':
227
+ case 'compaction.completed':
228
+ case 'compaction.skipped':
229
+ case 'compaction.invariant_violated': {
230
+ const tier = String(event.raw.tier ?? '');
231
+ return {
232
+ lines: [`_[compaction ${tier}] ${event.type.replace('compaction.', '')}_`],
233
+ isTurn: false,
234
+ };
235
+ }
236
+ default: {
237
+ return {
238
+ lines: [`_[event type=${event.type}]_`],
239
+ isTurn: false,
240
+ };
241
+ }
242
+ }
243
+ }
244
+ /**
245
+ * Wrap a string in a fenced code block. Pick a fence length that does
246
+ * not collide with backtick runs inside the content. Markdown 0.30 allows
247
+ * variable-length fences; we pick the shortest that is longer than the
248
+ * longest run inside the content (min 3, max 7).
249
+ */
250
+ function fenced(content) {
251
+ const longestRun = (content.match(/`+/g) ?? [])
252
+ .map((s) => s.length)
253
+ .reduce((max, n) => (n > max ? n : max), 0);
254
+ const fenceLen = Math.min(7, Math.max(3, longestRun + 1));
255
+ const fence = '`'.repeat(fenceLen);
256
+ // Trim trailing whitespace inside content so the closing fence sits
257
+ // tight against the body; preserve leading whitespace (matters for code).
258
+ return `${fence}\n${content.replace(/\s+$/u, '')}\n${fence}`;
259
+ }
260
+ /**
261
+ * Escape inline-Markdown specials (backtick, pipe) inside a span that we
262
+ * are wrapping in inline code. The closing backtick rule says a `<code>`
263
+ * span can contain backticks as long as the fence length differs — for
264
+ * simplicity we replace bare backticks with a Unicode look-alike when
265
+ * they appear in identifier-like positions (e.g. paths or tool names).
266
+ * Backticks in real content go through `fenced()` instead.
267
+ */
268
+ function escapeInline(text) {
269
+ return text.replace(/`/g, 'ˋ');
270
+ }
271
+ //# sourceMappingURL=formatter.js.map