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

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (448) hide show
  1. package/CHANGELOG.md +132 -0
  2. package/LICENSE +1 -1
  3. package/README.md +53 -11
  4. package/THIRD_PARTY_NOTICES.md +40 -0
  5. package/assets/pugi-mascot.ansi +15 -40
  6. package/assets/pugi-prozr2-mascot.ansi +9 -0
  7. package/bin/run.js +33 -1
  8. package/dist/commands/deploy.js +40 -40
  9. package/dist/commands/flatten.js +191 -0
  10. package/dist/commands/jobs-watch.js +201 -0
  11. package/dist/commands/jobs.js +42 -27
  12. package/dist/commands/retro.js +210 -0
  13. package/dist/commands/smoke.js +133 -0
  14. package/dist/core/agent-progress/cleanup.js +134 -0
  15. package/dist/core/agent-progress/schema.js +144 -0
  16. package/dist/core/agent-progress/writer.js +101 -0
  17. package/dist/core/agents/adaptive-router.js +330 -0
  18. package/dist/core/agents/query-decomposer.js +297 -0
  19. package/dist/core/agents/registry.js +3 -3
  20. package/dist/core/approvals/shortcut-resolver.js +98 -0
  21. package/dist/core/artifact-chain/dispatcher.js +148 -0
  22. package/dist/core/artifact-chain/exporter.js +164 -0
  23. package/dist/core/artifact-chain/state.js +243 -0
  24. package/dist/core/artifact-chain/steps.js +169 -0
  25. package/dist/core/ask-user/question.js +92 -0
  26. package/dist/core/audit/audit-trail.js +275 -0
  27. package/dist/core/auth/ensure-authenticated.js +129 -0
  28. package/dist/core/auth/env-provider.js +238 -0
  29. package/dist/core/auto-open-browser.js +4 -4
  30. package/dist/core/auto-update/channels.js +122 -0
  31. package/dist/core/auto-update/checker.js +241 -0
  32. package/dist/core/auto-update/state.js +235 -0
  33. package/dist/core/bare-mode/index.js +107 -0
  34. package/dist/core/bash/redirect.js +281 -0
  35. package/dist/core/bash-classifier.js +436 -40
  36. package/dist/core/checkpoint/resumer.js +149 -0
  37. package/dist/core/checkpoint/rewinder.js +291 -0
  38. package/dist/core/checkpoints/shadow-git.js +670 -0
  39. package/dist/core/citations/parser.js +109 -0
  40. package/dist/core/classifier/yolo-classifier.js +88 -0
  41. package/dist/core/codegraph/db.js +506 -0
  42. package/dist/core/codegraph/decision-store.js +248 -0
  43. package/dist/core/codegraph/detect-repo.js +459 -0
  44. package/dist/core/codegraph/install.js +134 -0
  45. package/dist/core/codegraph/offer-hook.js +220 -0
  46. package/dist/core/codegraph/parser.js +71 -0
  47. package/dist/core/codegraph/types.js +34 -0
  48. package/dist/core/compact/auto-trigger.js +96 -0
  49. package/dist/core/compact/buffer-rewriter.js +115 -0
  50. package/dist/core/compact/summarizer.js +208 -0
  51. package/dist/core/compact/token-counter.js +108 -0
  52. package/dist/core/consensus/anvil-fanout.js +25 -25
  53. package/dist/core/consensus/diff-capture.js +121 -12
  54. package/dist/core/consensus/rubric.js +21 -21
  55. package/dist/core/context/builder.js +6 -6
  56. package/dist/core/context/compaction-events.js +8 -8
  57. package/dist/core/context/compaction.js +31 -31
  58. package/dist/core/context/index.js +15 -8
  59. package/dist/core/context/invariants.js +51 -51
  60. package/dist/core/context/markdown-loader.js +28 -10
  61. package/dist/core/context/markdown-traverse.js +255 -0
  62. package/dist/core/context/pugiignore.js +41 -41
  63. package/dist/core/context/repo-skeleton.js +37 -37
  64. package/dist/core/context/tool-eviction.js +55 -0
  65. package/dist/core/context/watcher.js +32 -32
  66. package/dist/core/context/working-set.js +23 -23
  67. package/dist/core/coordinator/agent-tools.js +77 -0
  68. package/dist/core/coordinator/agent-toolset.js +65 -0
  69. package/dist/core/coordinator/fsm.js +73 -0
  70. package/dist/core/coordinator/mode-fsm.js +70 -0
  71. package/dist/core/cost/rate-card.js +129 -0
  72. package/dist/core/cost/tracker.js +221 -0
  73. package/dist/core/credentials.js +13 -13
  74. package/dist/core/cron/scheduler.js +138 -0
  75. package/dist/core/denial-tracking/index.js +8 -0
  76. package/dist/core/denial-tracking/state.js +264 -0
  77. package/dist/core/diagnostics/probe-runner.js +93 -0
  78. package/dist/core/diagnostics/probes/api.js +46 -0
  79. package/dist/core/diagnostics/probes/auth.js +93 -0
  80. package/dist/core/diagnostics/probes/bare-mode.js +42 -0
  81. package/dist/core/diagnostics/probes/cli-version.js +127 -0
  82. package/dist/core/diagnostics/probes/config.js +72 -0
  83. package/dist/core/diagnostics/probes/denial-tracking.js +57 -0
  84. package/dist/core/diagnostics/probes/disk.js +81 -0
  85. package/dist/core/diagnostics/probes/engine-live.js +46 -0
  86. package/dist/core/diagnostics/probes/git.js +65 -0
  87. package/dist/core/diagnostics/probes/hooks.js +118 -0
  88. package/dist/core/diagnostics/probes/mcp.js +75 -0
  89. package/dist/core/diagnostics/probes/node.js +59 -0
  90. package/dist/core/diagnostics/probes/pnpm.js +36 -0
  91. package/dist/core/diagnostics/probes/pugi-md.js +89 -0
  92. package/dist/core/diagnostics/probes/sandbox.js +72 -0
  93. package/dist/core/diagnostics/probes/session.js +74 -0
  94. package/dist/core/diagnostics/probes/status-snapshot.js +488 -0
  95. package/dist/core/diagnostics/probes/workspace.js +63 -0
  96. package/dist/core/diagnostics/types.js +70 -0
  97. package/dist/core/dispatch/cache-cleanup.js +197 -0
  98. package/dist/core/dispatch/cache-handoff.js +295 -0
  99. package/dist/core/edits/apply-patch-layer-e.js +189 -0
  100. package/dist/core/edits/dispatch.js +333 -7
  101. package/dist/core/edits/format-detector.js +260 -0
  102. package/dist/core/edits/format-matrix.js +26 -0
  103. package/dist/core/edits/fuzzy-ladder.js +650 -0
  104. package/dist/core/edits/index.js +5 -1
  105. package/dist/core/edits/journal.js +199 -0
  106. package/dist/core/edits/layer-a-apply.js +15 -15
  107. package/dist/core/edits/layer-a-fuzzy-apply.js +198 -0
  108. package/dist/core/edits/layer-b-apply.js +9 -9
  109. package/dist/core/edits/layer-c-apply.js +6 -6
  110. package/dist/core/edits/layer-d-ast.js +557 -14
  111. package/dist/core/edits/marker-parser.js +12 -12
  112. package/dist/core/edits/security-gate.js +27 -27
  113. package/dist/core/edits/verify-hook.js +273 -0
  114. package/dist/core/edits/worktree.js +322 -0
  115. package/dist/core/engine/anvil-client.js +214 -26
  116. package/dist/core/engine/auto-compact.js +247 -0
  117. package/dist/core/engine/budgets.js +220 -0
  118. package/dist/core/engine/compact-llm-summarizer.js +124 -0
  119. package/dist/core/engine/context-prefix.js +155 -0
  120. package/dist/core/engine/index.js +1 -1
  121. package/dist/core/engine/intensity.js +163 -0
  122. package/dist/core/engine/intent.js +260 -0
  123. package/dist/core/engine/native-pugi.js +1559 -227
  124. package/dist/core/engine/prompts.js +192 -16
  125. package/dist/core/engine/strip-internal-fields.js +124 -0
  126. package/dist/core/engine/tool-bridge.js +1887 -59
  127. package/dist/core/engine/verification-patterns.js +195 -0
  128. package/dist/core/evaluation/golden-dataset.js +293 -0
  129. package/dist/core/feedback/queue.js +177 -0
  130. package/dist/core/feedback/submitter.js +145 -0
  131. package/dist/core/file-cache.js +113 -1
  132. package/dist/core/flatten/flatten-repo.js +439 -0
  133. package/dist/core/format/osc8-link.js +28 -0
  134. package/dist/core/hook-chains.js +392 -0
  135. package/dist/core/hooks/citation-verify-hook.js +138 -0
  136. package/dist/core/hooks/citation-verify.js +112 -0
  137. package/dist/core/hooks/events.js +46 -0
  138. package/dist/core/hooks/index.js +15 -0
  139. package/dist/core/hooks/registry.js +216 -0
  140. package/dist/core/hooks/runner.js +236 -0
  141. package/dist/core/hooks/v2/event-emitter.js +115 -0
  142. package/dist/core/hooks/v2/executor.js +282 -0
  143. package/dist/core/hooks/v2/index.js +25 -0
  144. package/dist/core/hooks/v2/lifecycle.js +104 -0
  145. package/dist/core/hooks/v2/loader.js +216 -0
  146. package/dist/core/hooks/v2/matcher.js +125 -0
  147. package/dist/core/hooks/v2/trust.js +143 -0
  148. package/dist/core/hooks/v2/types.js +86 -0
  149. package/dist/core/hooks/worktree-events.js +158 -0
  150. package/dist/core/image/renderer.js +71 -0
  151. package/dist/core/init/detector.js +582 -0
  152. package/dist/core/init/template-renderer.js +242 -0
  153. package/dist/core/jobs/registry.js +18 -18
  154. package/dist/core/ledger/results-tsv.js +142 -0
  155. package/dist/core/log-discipline/stdout-redirect.js +51 -0
  156. package/dist/core/lsp/cache.js +105 -0
  157. package/dist/core/lsp/client.js +1229 -0
  158. package/dist/core/lsp/language-detect.js +66 -0
  159. package/dist/core/lsp/post-edit-diagnostics.js +171 -0
  160. package/dist/core/lsp/server-detect.js +173 -0
  161. package/dist/core/lsp/symbol-cache.js +162 -0
  162. package/dist/core/lsp/symbol-tools.js +664 -0
  163. package/dist/core/mcp/client.js +97 -28
  164. package/dist/core/mcp/http-server.js +553 -0
  165. package/dist/core/mcp/orchestrator-config.js +192 -0
  166. package/dist/core/mcp/orchestrator-tools.js +806 -0
  167. package/dist/core/mcp/permission.js +190 -0
  168. package/dist/core/mcp/registry.js +39 -17
  169. package/dist/core/mcp/server-tools.js +219 -0
  170. package/dist/core/mcp/server.js +397 -0
  171. package/dist/core/mcp/trust.js +10 -10
  172. package/dist/core/memory/dual-write.js +416 -0
  173. package/dist/core/memory/passive-extract.js +130 -0
  174. package/dist/core/memory/phase1-kinds.js +20 -0
  175. package/dist/core/memory/secret-scanner.js +304 -0
  176. package/dist/core/memory-sync/queue.js +170 -0
  177. package/dist/core/metrics/extract.js +113 -0
  178. package/dist/core/modes/roo-modes.js +68 -0
  179. package/dist/core/notes/notes-paths.js +113 -0
  180. package/dist/core/notes/notes-recorder.js +140 -0
  181. package/dist/core/notes/notes-writer.js +53 -0
  182. package/dist/core/notes/renderers.js +0 -0
  183. package/dist/core/notes/slug.js +105 -0
  184. package/dist/core/onboarding/ensure-initialized.js +133 -0
  185. package/dist/core/onboarding/marker.js +111 -0
  186. package/dist/core/onboarding/telemetry-state.js +108 -0
  187. package/dist/core/output-style/presets.js +176 -0
  188. package/dist/core/output-style/state.js +185 -0
  189. package/dist/core/path-security.js +287 -5
  190. package/dist/core/permission.js +82 -22
  191. package/dist/core/permissions/auto-classifier.js +124 -0
  192. package/dist/core/permissions/bash-parser.js +371 -0
  193. package/dist/core/permissions/circuit-breaker.js +83 -0
  194. package/dist/core/permissions/constrained-edit.js +91 -0
  195. package/dist/core/permissions/gate.js +278 -0
  196. package/dist/core/permissions/index.js +20 -0
  197. package/dist/core/permissions/mode.js +174 -0
  198. package/dist/core/permissions/network-egress.js +137 -0
  199. package/dist/core/permissions/state.js +241 -0
  200. package/dist/core/permissions/tool-class.js +107 -0
  201. package/dist/core/plan-mode/ui-state.js +51 -0
  202. package/dist/core/plans/plan-artifact.js +721 -0
  203. package/dist/core/policy-limits/etag-store.js +122 -0
  204. package/dist/core/prd-check/parser.js +215 -0
  205. package/dist/core/prd-check/reporter.js +127 -0
  206. package/dist/core/prd-check/session-review.js +557 -0
  207. package/dist/core/prd-check/verifiers.js +223 -0
  208. package/dist/core/prompt-cache/client-cache.js +99 -0
  209. package/dist/core/prompts/assembly.js +29 -0
  210. package/dist/core/prompts/registry.js +364 -0
  211. package/dist/core/pugi-gitignore.js +52 -0
  212. package/dist/core/pugi-md/cc-compat-rules.js +735 -0
  213. package/dist/core/pugi-md/context-injector.js +76 -0
  214. package/dist/core/pugi-md/walk-up.js +207 -0
  215. package/dist/core/python/uv-installer.js +270 -0
  216. package/dist/core/python/uv-resolver.js +83 -0
  217. package/dist/core/rate-limit/narrator.js +146 -0
  218. package/dist/core/recipes/cli-types.js +20 -0
  219. package/dist/core/recipes/loader.js +103 -0
  220. package/dist/core/recipes/runner.js +345 -0
  221. package/dist/core/recipes/schema.js +587 -0
  222. package/dist/core/release-notes/parser.js +241 -0
  223. package/dist/core/release-notes/state.js +116 -0
  224. package/dist/core/repl/ask.js +37 -37
  225. package/dist/core/repl/cancellation.js +26 -26
  226. package/dist/core/repl/cap-warning.js +4 -4
  227. package/dist/core/repl/clipboard-read.js +11 -11
  228. package/dist/core/repl/dispatch-fsm.js +12 -12
  229. package/dist/core/repl/engine-bridge.js +303 -0
  230. package/dist/core/repl/history-search.js +15 -15
  231. package/dist/core/repl/history.js +28 -18
  232. package/dist/core/repl/kill-ring.js +5 -5
  233. package/dist/core/repl/model-pricing.js +135 -0
  234. package/dist/core/repl/privacy-banner.js +22 -22
  235. package/dist/core/repl/session.js +2714 -228
  236. package/dist/core/repl/slash-commands.js +572 -40
  237. package/dist/core/repl/store/index.js +1 -1
  238. package/dist/core/repl/store/jsonl-log.js +22 -22
  239. package/dist/core/repl/store/lockfile.js +10 -10
  240. package/dist/core/repl/store/session-store.js +136 -107
  241. package/dist/core/repl/store/types.js +15 -15
  242. package/dist/core/repl/store/uuid-v7.js +12 -12
  243. package/dist/core/repl/tool-route.js +382 -0
  244. package/dist/core/repl/workspace-context.js +43 -21
  245. package/dist/core/repo-map/build.js +125 -0
  246. package/dist/core/repo-map/cache.js +185 -0
  247. package/dist/core/repo-map/extractor.js +254 -0
  248. package/dist/core/repo-map/formatter.js +145 -0
  249. package/dist/core/repo-map/page-rank.js +105 -0
  250. package/dist/core/repo-map/scanner.js +211 -0
  251. package/dist/core/retro/git-collector.js +251 -0
  252. package/dist/core/retro/health-card.js +25 -0
  253. package/dist/core/retro/metrics.js +342 -0
  254. package/dist/core/retro/narrative.js +249 -0
  255. package/dist/core/retro/plane-collector.js +274 -0
  256. package/dist/core/retro/pr-issue-link.js +65 -0
  257. package/dist/core/retro/types.js +16 -0
  258. package/dist/core/retry-budget/budget.js +284 -0
  259. package/dist/core/retry-budget/index.js +5 -0
  260. package/dist/core/retry-budget/retry-cap.js +74 -0
  261. package/dist/core/routing/lead-worker.js +43 -0
  262. package/dist/core/routing/pre-flight-estimator.js +108 -0
  263. package/dist/core/runs/run-tree.js +103 -0
  264. package/dist/core/sandboxing/adapter.js +29 -0
  265. package/dist/core/sandboxing/index.js +49 -0
  266. package/dist/core/sandboxing/none.js +19 -0
  267. package/dist/core/sandboxing/seatbelt.js +183 -0
  268. package/dist/core/security/injection-scanner.js +367 -0
  269. package/dist/core/security/output-filter.js +418 -0
  270. package/dist/core/session/env-file.js +105 -0
  271. package/dist/core/session/section-budgets.js +140 -0
  272. package/dist/core/session.js +119 -0
  273. package/dist/core/settings.js +378 -5
  274. package/dist/core/share/formatter.js +271 -0
  275. package/dist/core/share/redactor.js +221 -0
  276. package/dist/core/share/uploader.js +267 -0
  277. package/dist/core/skills/defaults.js +457 -0
  278. package/dist/core/skills/loader.js +22 -22
  279. package/dist/core/skills/sources.js +27 -27
  280. package/dist/core/smoke/headless-driver.js +174 -0
  281. package/dist/core/smoke/orchestrator.js +194 -0
  282. package/dist/core/smoke/runner.js +238 -0
  283. package/dist/core/smoke/scenario-parser.js +316 -0
  284. package/dist/core/statusline.js +99 -0
  285. package/dist/core/subagents/dispatcher-real.js +600 -0
  286. package/dist/core/subagents/dispatcher.js +146 -52
  287. package/dist/core/subagents/index.js +19 -6
  288. package/dist/core/subagents/isolation-matrix.js +213 -0
  289. package/dist/core/subagents/spawn.js +19 -4
  290. package/dist/core/telemetry/emitter.js +229 -0
  291. package/dist/core/telemetry/queue.js +251 -0
  292. package/dist/core/theme/context.js +91 -0
  293. package/dist/core/theme/presets.js +228 -0
  294. package/dist/core/theme/state.js +181 -0
  295. package/dist/core/todos/invariant.js +10 -0
  296. package/dist/core/todos/state.js +177 -0
  297. package/dist/core/tool-schema/compressor.js +89 -0
  298. package/dist/core/transport/version-interceptor.js +166 -0
  299. package/dist/core/trust.js +2 -2
  300. package/dist/core/tui/thinking-block.js +64 -0
  301. package/dist/core/vim/keymap.js +288 -0
  302. package/dist/core/vim/state.js +92 -0
  303. package/dist/core/watch-markers/marker-watcher.js +133 -0
  304. package/dist/core/worktree/include-parser.js +249 -0
  305. package/dist/core/worktree-manager/cleanup.js +123 -0
  306. package/dist/core/worktree-manager/manager.js +303 -0
  307. package/dist/index.js +36 -0
  308. package/dist/runtime/bootstrap.js +190 -0
  309. package/dist/runtime/cli.js +4536 -477
  310. package/dist/runtime/commands/agents.js +31 -31
  311. package/dist/runtime/commands/budget.js +5 -5
  312. package/dist/runtime/commands/cancel.js +231 -0
  313. package/dist/runtime/commands/chain.js +489 -0
  314. package/dist/runtime/commands/codegraph-status.js +227 -0
  315. package/dist/runtime/commands/compact.js +297 -0
  316. package/dist/runtime/commands/config.js +74 -40
  317. package/dist/runtime/commands/cost.js +199 -0
  318. package/dist/runtime/commands/delegate.js +312 -0
  319. package/dist/runtime/commands/dispatch.js +126 -0
  320. package/dist/runtime/commands/doctor.js +579 -0
  321. package/dist/runtime/commands/feedback.js +184 -0
  322. package/dist/runtime/commands/hooks.js +187 -0
  323. package/dist/runtime/commands/index-cmd.js +353 -0
  324. package/dist/runtime/commands/init.js +254 -0
  325. package/dist/runtime/commands/lsp.js +368 -0
  326. package/dist/runtime/commands/mcp.js +935 -0
  327. package/dist/runtime/commands/memory.js +582 -0
  328. package/dist/runtime/commands/model.js +237 -0
  329. package/dist/runtime/commands/onboarding.js +275 -0
  330. package/dist/runtime/commands/patch.js +128 -0
  331. package/dist/runtime/commands/permissions.js +112 -0
  332. package/dist/runtime/commands/plan.js +143 -0
  333. package/dist/runtime/commands/prd-check.js +285 -0
  334. package/dist/runtime/commands/privacy.js +17 -17
  335. package/dist/runtime/commands/recipe.js +325 -0
  336. package/dist/runtime/commands/redo-blob-store.js +92 -0
  337. package/dist/runtime/commands/redo.js +361 -0
  338. package/dist/runtime/commands/release-notes.js +229 -0
  339. package/dist/runtime/commands/repo-map.js +95 -0
  340. package/dist/runtime/commands/report.js +299 -0
  341. package/dist/runtime/commands/resume.js +118 -0
  342. package/dist/runtime/commands/review-consensus.js +68 -53
  343. package/dist/runtime/commands/rewind.js +333 -0
  344. package/dist/runtime/commands/roster.js +117 -0
  345. package/dist/runtime/commands/servers.js +236 -0
  346. package/dist/runtime/commands/sessions.js +163 -0
  347. package/dist/runtime/commands/share.js +316 -0
  348. package/dist/runtime/commands/skills.js +31 -31
  349. package/dist/runtime/commands/status.js +186 -0
  350. package/dist/runtime/commands/stickers.js +82 -0
  351. package/dist/runtime/commands/style.js +194 -0
  352. package/dist/runtime/commands/theme.js +196 -0
  353. package/dist/runtime/commands/undo.js +54 -22
  354. package/dist/runtime/commands/update.js +289 -0
  355. package/dist/runtime/commands/vim.js +140 -0
  356. package/dist/runtime/commands/worktree.js +177 -0
  357. package/dist/runtime/commands/worktrees.js +155 -0
  358. package/dist/runtime/deprecation-warning.js +69 -0
  359. package/dist/runtime/engine-exit-code.js +50 -0
  360. package/dist/runtime/headless-repl.js +195 -0
  361. package/dist/runtime/headless.js +548 -0
  362. package/dist/runtime/load-hooks-or-exit.js +71 -0
  363. package/dist/runtime/plan-decompose.js +531 -0
  364. package/dist/runtime/sigint-guard.js +272 -0
  365. package/dist/runtime/stream-renderer.js +195 -0
  366. package/dist/runtime/update-check.js +28 -28
  367. package/dist/runtime/version.js +65 -0
  368. package/dist/runtime/worktree-bootstrap.js +579 -0
  369. package/dist/skills/bundled/batch.js +617 -0
  370. package/dist/skills/bundled/index.js +45 -0
  371. package/dist/skills/bundled/loop.js +358 -0
  372. package/dist/skills/bundled/remember.js +383 -0
  373. package/dist/skills/bundled/simplify.js +289 -0
  374. package/dist/skills/bundled/skillify.js +373 -0
  375. package/dist/skills/bundled/stuck.js +558 -0
  376. package/dist/skills/bundled/verify.js +439 -0
  377. package/dist/testing/vcr.js +486 -0
  378. package/dist/tools/agent-tool.js +229 -0
  379. package/dist/tools/apply-patch.js +556 -0
  380. package/dist/tools/ask-user-question.js +337 -0
  381. package/dist/tools/ask-user.js +115 -0
  382. package/dist/tools/bash.js +624 -46
  383. package/dist/tools/brief.js +224 -0
  384. package/dist/tools/cron.js +433 -0
  385. package/dist/tools/enter-worktree.js +250 -0
  386. package/dist/tools/exit-worktree.js +147 -0
  387. package/dist/tools/file-tools.js +161 -44
  388. package/dist/tools/http-request.js +336 -0
  389. package/dist/tools/lsp-tools.js +565 -0
  390. package/dist/tools/mcp-tool.js +260 -0
  391. package/dist/tools/multi-edit.js +361 -0
  392. package/dist/tools/powershell.js +268 -0
  393. package/dist/tools/registry.js +142 -1
  394. package/dist/tools/server-tools.js +892 -0
  395. package/dist/tools/skill-tool.js +96 -0
  396. package/dist/tools/sleep.js +99 -0
  397. package/dist/tools/synthetic-output.js +133 -0
  398. package/dist/tools/tasks.js +208 -0
  399. package/dist/tools/todo-write.js +184 -0
  400. package/dist/tools/verify-plan-execution.js +295 -0
  401. package/dist/tools/web-fetch-injection-scanner.js +207 -0
  402. package/dist/tools/web-fetch.js +195 -10
  403. package/dist/tools/web-search.js +458 -0
  404. package/dist/tui/agent-progress-card.js +111 -0
  405. package/dist/tui/agent-tree.js +22 -1
  406. package/dist/tui/ask-modal.js +14 -14
  407. package/dist/tui/ask-user-question-chips.js +315 -0
  408. package/dist/tui/ask-user-question-prompt.js +203 -0
  409. package/dist/tui/compact-banner.js +81 -0
  410. package/dist/tui/conversation-pane.js +85 -11
  411. package/dist/tui/cost-table.js +111 -0
  412. package/dist/tui/device-flow.js +2 -2
  413. package/dist/tui/doctor-table.js +46 -0
  414. package/dist/tui/feedback-prompt.js +156 -0
  415. package/dist/tui/input-box.js +247 -32
  416. package/dist/tui/login-picker.js +3 -3
  417. package/dist/tui/markdown-render.js +6 -6
  418. package/dist/tui/multi-file-diff-approval.js +375 -0
  419. package/dist/tui/onboarding-wizard.js +240 -0
  420. package/dist/tui/permissions-picker.js +86 -0
  421. package/dist/tui/render.js +36 -1
  422. package/dist/tui/repl-render.js +405 -32
  423. package/dist/tui/repl-splash-art.js +16 -16
  424. package/dist/tui/repl-splash-mascot.js +48 -24
  425. package/dist/tui/repl-splash.js +22 -22
  426. package/dist/tui/repl.js +136 -43
  427. package/dist/tui/slash-palette.js +6 -6
  428. package/dist/tui/splash.js +2 -2
  429. package/dist/tui/status-bar.js +109 -31
  430. package/dist/tui/status-table.js +7 -0
  431. package/dist/tui/stickers-art.js +136 -0
  432. package/dist/tui/style-table.js +28 -0
  433. package/dist/tui/theme-table.js +29 -0
  434. package/dist/tui/thinking-spinner.js +123 -0
  435. package/dist/tui/tool-stream-pane.js +53 -4
  436. package/dist/tui/update-banner.js +27 -2
  437. package/dist/tui/vim-input.js +267 -0
  438. package/dist/tui/welcome-banner.js +107 -0
  439. package/dist/tui/welcome-data.js +293 -0
  440. package/dist/tui/workspace-context.js +2 -2
  441. package/docs/examples/codegraph.mcp.json +10 -0
  442. package/package.json +25 -7
  443. package/test/scenarios/codegen-create-file.scenario.txt +13 -0
  444. package/test/scenarios/compact-force.scenario.txt +12 -0
  445. package/test/scenarios/identity.scenario.txt +11 -0
  446. package/test/scenarios/persona-handoff.scenario.txt +12 -0
  447. package/test/scenarios/walkback.scenario.txt +12 -0
  448. package/dist/core/engine/compaction-hook.js +0 -154
@@ -0,0 +1,210 @@
1
+ import { jsxs as _jsxs, jsx as _jsx } from "react/jsx-runtime";
2
+ import { Box, Text, render } from 'ink';
3
+ import { collectGitContext, countCommitsAheadOfBase, } from '../core/retro/git-collector.js';
4
+ import { ensurePugiGitIgnore } from '../core/pugi-gitignore.js';
5
+ import { computeMetrics } from '../core/retro/metrics.js';
6
+ import { persistRetro } from '../core/retro/narrative.js';
7
+ import { collectPlaneSlice, postRetroToPlane, resolvePlaneConfig, } from '../core/retro/plane-collector.js';
8
+ import { enrichLinks } from '../core/retro/pr-issue-link.js';
9
+ import { computeHealthCard } from '../core/retro/health-card.js';
10
+ /** Parse `7d` | `14d` | `30d` | `24h` into a duration in days
11
+ * (fractional for sub-day windows). Defaults to 7 days when omitted.
12
+ */
13
+ function parseDurationToken(token) {
14
+ if (!token)
15
+ return undefined;
16
+ const match = /^(\d+)(h|d)$/.exec(token);
17
+ if (!match)
18
+ return undefined;
19
+ const value = Number.parseInt(match[1] ?? '0', 10);
20
+ const unit = match[2];
21
+ if (!Number.isFinite(value) || value <= 0)
22
+ return undefined;
23
+ const days = unit === 'h' ? value / 24 : value;
24
+ return { days, label: token };
25
+ }
26
+ function buildWindow(durationDays, label, now) {
27
+ const until = now;
28
+ const sinceMs = until.getTime() - durationDays * 24 * 60 * 60 * 1000;
29
+ const since = new Date(sinceMs);
30
+ // Midnight-align the lower bound to keep `--since` deterministic per day.
31
+ since.setHours(0, 0, 0, 0);
32
+ return { since, until, label, days: Math.max(1, Math.round(durationDays)) };
33
+ }
34
+ function buildPriorWindow(current) {
35
+ const until = new Date(current.since.getTime());
36
+ const sinceMs = until.getTime() - current.days * 24 * 60 * 60 * 1000;
37
+ const since = new Date(sinceMs);
38
+ since.setHours(0, 0, 0, 0);
39
+ return { since, until, label: `prior ${current.label}`, days: current.days };
40
+ }
41
+ function parseRetroArgs(rawArgs, now) {
42
+ const args = [...rawArgs];
43
+ const postPlane = args.includes('--post-plane');
44
+ const enrichPlane = args.includes('--plane') || postPlane;
45
+ const positional = args.filter((a) => !a.startsWith('-'));
46
+ let compare = false;
47
+ let durationToken;
48
+ if (positional[0] === 'compare') {
49
+ compare = true;
50
+ durationToken = positional[1];
51
+ }
52
+ else {
53
+ durationToken = positional[0];
54
+ }
55
+ const parsed = parseDurationToken(durationToken) ?? { days: 7, label: '7d' };
56
+ return {
57
+ window: buildWindow(parsed.days, parsed.label, now),
58
+ compare,
59
+ enrichPlane,
60
+ postPlane,
61
+ };
62
+ }
63
+ function SummaryCard(props) {
64
+ const { persisted, metrics, plane, planePostUrl } = props;
65
+ return (_jsxs(Box, { flexDirection: "column", borderStyle: "single", borderRight: false, borderTop: false, borderBottom: false, paddingLeft: 1, children: [_jsxs(Text, { bold: true, children: ["pugi retro \u00B7 ", metrics.window.label] }), _jsxs(Text, { dimColor: true, children: ["Branch ", metrics.branch.current, " over ", metrics.branch.base] }), _jsxs(Text, { children: [metrics.commits.total, " commits \u00B7 +", metrics.loc.insertions, " / -", metrics.loc.deletions, " LOC \u00B7 ", metrics.activeDays, " active days"] }), _jsxs(Text, { children: ["Focus ", metrics.focus.score, "% on ", metrics.focus.topDir ?? 'n/a', " \u00B7 Streak ", metrics.streak.personalDays, "d personal / ", metrics.streak.teamDays, "d team"] }), metrics.shipOfTheWeek ? (_jsxs(Text, { children: ["Ship of the week: ", metrics.shipOfTheWeek.subject.slice(0, 60)] })) : null, plane ? (_jsxs(Text, { children: ["Plane: closed ", plane.closedIssues.length, " \u00B7 created ", plane.createdIssues.length, " \u00B7 oversized modules ", plane.oversizedModules.length] })) : null, _jsxs(Text, { dimColor: true, children: ["Markdown: ", persisted.markdownPath] }), _jsxs(Text, { dimColor: true, children: ["JSON: ", persisted.jsonPath] }), planePostUrl ? _jsxs(Text, { children: ["Posted to Plane: ", planePostUrl] }) : null] }));
66
+ }
67
+ function renderSummary(props) {
68
+ const app = render(_jsx(SummaryCard, { ...props }));
69
+ app.unmount();
70
+ }
71
+ export async function runRetroCommand(ctx) {
72
+ const now = ctx.now ?? new Date();
73
+ const parsed = parseRetroArgs(ctx.args, now);
74
+ const gitCtx = await collectGitContext({ cwd: ctx.cwd, window: parsed.window });
75
+ if (!gitCtx.hasGit) {
76
+ const msg = 'pugi retro: not a git workspace - initialise git or cd into one.';
77
+ if (ctx.flags.json) {
78
+ ctx.io.write(`${JSON.stringify({ ok: false, error: 'no_git_workspace' })}\n`);
79
+ }
80
+ else {
81
+ ctx.io.writeError(msg);
82
+ }
83
+ return 2;
84
+ }
85
+ // Triple-review P1.2 (): before we write anything
86
+ // under `.pugi/retros/`, guarantee `.gitignore` covers `.pugi/`. Without
87
+ // this, the first customer run of `pugi retro` in a fresh repo would
88
+ // leave retros (and any future `.pugi/settings.json` secret store)
89
+ // tracked by git on the next `git add -A`. Idempotent.
90
+ //
91
+ // Round 2 P1 (2026-06-04): surface failure к stderr — silent catch
92
+ // defeats the gate's purpose. If `.gitignore` is read-only or perms
93
+ // refuse, the operator must know retros may be tracked by git.
94
+ const gitIgnoreCreated = [];
95
+ const gitIgnoreSkipped = [];
96
+ try {
97
+ ensurePugiGitIgnore(ctx.cwd, gitIgnoreCreated, gitIgnoreSkipped);
98
+ }
99
+ catch (err) {
100
+ const reason = err instanceof Error ? err.message : String(err);
101
+ ctx.io.writeError(`pugi retro: could not update .gitignore (${reason}). ` +
102
+ `Manually add ".pugi/" to .gitignore so retros are not tracked.`);
103
+ }
104
+ const toBaseHeadCount = await countCommitsAheadOfBase(ctx.cwd, gitCtx.baseBranch, parsed.window.since);
105
+ const metrics = computeMetrics({
106
+ window: parsed.window,
107
+ currentBranch: gitCtx.currentBranch,
108
+ baseBranch: gitCtx.baseBranch,
109
+ toBaseHeadCount,
110
+ currentUserName: gitCtx.userName,
111
+ currentUserEmail: gitCtx.userEmail,
112
+ commits: gitCtx.commits,
113
+ });
114
+ let compare;
115
+ if (parsed.compare) {
116
+ const priorWindow = buildPriorWindow(parsed.window);
117
+ const priorCtx = await collectGitContext({ cwd: ctx.cwd, window: priorWindow });
118
+ const priorAhead = await countCommitsAheadOfBase(ctx.cwd, gitCtx.baseBranch, priorWindow.since);
119
+ const priorMetrics = computeMetrics({
120
+ window: priorWindow,
121
+ currentBranch: gitCtx.currentBranch,
122
+ baseBranch: gitCtx.baseBranch,
123
+ toBaseHeadCount: priorAhead,
124
+ currentUserName: gitCtx.userName,
125
+ currentUserEmail: gitCtx.userEmail,
126
+ commits: priorCtx.commits,
127
+ });
128
+ compare = { current: metrics, prior: priorMetrics };
129
+ }
130
+ let plane;
131
+ let planeUnavailableReason;
132
+ if (parsed.enrichPlane) {
133
+ const cfgResult = resolvePlaneConfig(ctx.cwd);
134
+ if (!cfgResult.ok) {
135
+ planeUnavailableReason = cfgResult.reason;
136
+ }
137
+ else {
138
+ try {
139
+ const slice = await collectPlaneSlice({
140
+ config: cfgResult.config,
141
+ since: parsed.window.since,
142
+ });
143
+ const links = enrichLinks(gitCtx.commits, slice.closedIssues.concat(slice.createdIssues));
144
+ const health = computeHealthCard(slice.modules);
145
+ plane = {
146
+ ...slice,
147
+ prToIssueLinks: links,
148
+ oversizedModules: health.oversized,
149
+ };
150
+ }
151
+ catch (err) {
152
+ planeUnavailableReason = err instanceof Error ? err.message : String(err);
153
+ }
154
+ }
155
+ }
156
+ const persisted = persistRetro({
157
+ root: ctx.cwd,
158
+ metrics,
159
+ plane,
160
+ compare,
161
+ now,
162
+ });
163
+ let planePostUrl;
164
+ if (parsed.postPlane) {
165
+ if (!plane) {
166
+ ctx.io.writeError(`pugi retro --post-plane: Plane unavailable (${planeUnavailableReason ?? 'unknown'}).`);
167
+ }
168
+ else {
169
+ const cfgResult = resolvePlaneConfig(ctx.cwd);
170
+ if (cfgResult.ok) {
171
+ try {
172
+ const result = await postRetroToPlane({
173
+ config: cfgResult.config,
174
+ markdown: persisted.markdown,
175
+ sequence: persisted.sequence,
176
+ dateLabel: persisted.dateLabel,
177
+ });
178
+ planePostUrl = result.url;
179
+ if (result.alreadyExists) {
180
+ ctx.io.write(`pugi retro: already exists at ${result.url}\n`);
181
+ }
182
+ }
183
+ catch (err) {
184
+ const msg = err instanceof Error ? err.message : String(err);
185
+ ctx.io.writeError(`pugi retro --post-plane failed: ${msg}`);
186
+ }
187
+ }
188
+ }
189
+ }
190
+ if (ctx.flags.json) {
191
+ ctx.io.write(`${JSON.stringify({
192
+ ok: true,
193
+ markdownPath: persisted.markdownPath,
194
+ jsonPath: persisted.jsonPath,
195
+ sequence: persisted.sequence,
196
+ metrics,
197
+ plane: plane ?? null,
198
+ planePostUrl: planePostUrl ?? null,
199
+ planeUnavailableReason: planeUnavailableReason ?? null,
200
+ }, null, 2)}\n`);
201
+ }
202
+ else {
203
+ renderSummary({ persisted, metrics, plane, planePostUrl });
204
+ if (planeUnavailableReason && !plane) {
205
+ ctx.io.writeError(`pugi retro: Plane integration unavailable (${planeUnavailableReason}).`);
206
+ }
207
+ }
208
+ return 0;
209
+ }
210
+ //# sourceMappingURL=retro.js.map
@@ -0,0 +1,133 @@
1
+ /**
2
+ * `pugi smoke` — runs the bundled scenario corpus through the headless
3
+ * harness and reports pass/fail per scenario (
4
+ *).
5
+ *
6
+ * The CLI surface lives here rather than in `runtime/cli.ts` so the
7
+ * dispatch surface stays focused on argv routing. This module owns:
8
+ *
9
+ * - Resolving the scenarios directory (default
10
+ * `<cli-root>/test/scenarios/` when bundled, configurable via
11
+ * `--scenarios-dir`).
12
+ * - Selecting the `pugi` binary the headless driver should spawn.
13
+ * Local development: `node <cli-root>/bin/run.js`. CI / published
14
+ * usage: the `pugi` on PATH.
15
+ * - Forwarding the orchestrator output to the unified
16
+ * `writeOutput` writer so `--json` mode works without a second
17
+ * code path.
18
+ *
19
+ * Phase 1 deliberately ships ONE flag (`--filter`) and one option
20
+ * (`--scenarios-dir`); the rest of the surface comes in Phase 2 once
21
+ * the engine path is wired and we know which scenarios actually need
22
+ * per-run plumbing (timeouts, fixture credentials, hermetic stubs).
23
+ */
24
+ import { existsSync } from 'node:fs';
25
+ import { dirname, resolve } from 'node:path';
26
+ import { fileURLToPath } from 'node:url';
27
+ import { runSmoke, renderReportText, } from '../core/smoke/orchestrator.js';
28
+ import { runHeadlessScenario } from '../core/smoke/headless-driver.js';
29
+ /**
30
+ * Entry point invoked by `runtime/cli.ts::dispatchSmoke`. Returns the
31
+ * desired process exit code so the dispatcher can set
32
+ * `process.exitCode` without a second round-trip.
33
+ */
34
+ export async function runSmokeCommand(ctx) {
35
+ // Parse the (small) command-local argv. We only honor flags this
36
+ // command actually consumes; unknown args produce a usage error so
37
+ // typos surface immediately.
38
+ let scenariosDirOverride = ctx.scenariosDir ?? undefined;
39
+ let filter = ctx.filter ?? '';
40
+ for (let i = 0; i < ctx.args.length; i += 1) {
41
+ const arg = ctx.args[i] ?? '';
42
+ if (arg === '--filter') {
43
+ const next = ctx.args[i + 1];
44
+ if (!next || next.startsWith('--')) {
45
+ ctx.writeOutput({ ok: false, error: '--filter requires a pattern' }, 'pugi smoke: --filter requires a pattern (substring or *-glob)');
46
+ return 2;
47
+ }
48
+ filter = next;
49
+ i += 1;
50
+ }
51
+ else if (arg.startsWith('--filter=')) {
52
+ filter = arg.slice('--filter='.length);
53
+ }
54
+ else if (arg === '--scenarios-dir') {
55
+ const next = ctx.args[i + 1];
56
+ if (!next || next.startsWith('--')) {
57
+ ctx.writeOutput({ ok: false, error: '--scenarios-dir requires a path' }, 'pugi smoke: --scenarios-dir requires a path');
58
+ return 2;
59
+ }
60
+ scenariosDirOverride = next;
61
+ i += 1;
62
+ }
63
+ else if (arg.startsWith('--scenarios-dir=')) {
64
+ scenariosDirOverride = arg.slice('--scenarios-dir='.length);
65
+ }
66
+ else if (arg === '--help' || arg === '-h') {
67
+ ctx.writeOutput({
68
+ ok: true,
69
+ usage: 'pugi smoke [--filter <pattern>] [--scenarios-dir <path>]',
70
+ }, [
71
+ 'pugi smoke — run the bundled scenario corpus headlessly.',
72
+ '',
73
+ 'Flags:',
74
+ ' --filter <pattern> Run a subset (substring or *-glob match on scenario id).',
75
+ ' --scenarios-dir <path> Override the scenarios directory (default: bundled corpus).',
76
+ ].join('\n'));
77
+ return 0;
78
+ }
79
+ else {
80
+ ctx.writeOutput({ ok: false, error: `unknown arg: ${arg}` }, `pugi smoke: unknown arg ${arg}`);
81
+ return 2;
82
+ }
83
+ }
84
+ const scenariosDir = scenariosDirOverride ?? resolveBundledScenariosDir();
85
+ if (!existsSync(scenariosDir)) {
86
+ ctx.writeOutput({ ok: false, error: `scenarios dir not found: ${scenariosDir}` }, `pugi smoke: scenarios dir not found: ${scenariosDir}`);
87
+ return 2;
88
+ }
89
+ const pugiBin = ctx.pugiBin ?? process.env.PUGI_SMOKE_BIN ?? 'pugi';
90
+ const log = ctx.log ?? ((line) => process.stderr.write(`${line}\n`));
91
+ const report = await runSmoke({
92
+ scenariosDir,
93
+ filter,
94
+ executor: (scenario) => runHeadlessScenario(scenario, { pugiBin }),
95
+ log,
96
+ });
97
+ if (ctx.json) {
98
+ ctx.writeOutput(report, renderReportText(report));
99
+ }
100
+ else {
101
+ process.stdout.write(`${renderReportText(report)}\n`);
102
+ }
103
+ return report.exitCode;
104
+ }
105
+ /**
106
+ * Resolve the scenarios directory that ships alongside the CLI.
107
+ *
108
+ * Both the dev source layout (`<cli-root>/src/commands/smoke.ts`) and
109
+ * the built output layout (`<cli-root>/dist/commands/smoke.js`) land
110
+ * on the same `../../test/scenarios` path relative to this file.
111
+ * The bundled `npm i -g @pugi/cli` install replicates that structure
112
+ * by shipping `test/scenarios` (glob `**\/*.scenario.txt`) via the
113
+ * `package.json` `files` field — so the single resolved path works in
114
+ * dev, in `tsx` runs, and in published installs without a config knob.
115
+ *
116
+ * If a future restructure ever bundles scenarios at
117
+ * `<cli-root>/dist/scenarios/` instead, add that to the fallback chain
118
+ * here AND mirror the path in `package.json` `files`. Until then we
119
+ * keep the resolver intentionally one-line.
120
+ */
121
+ export function resolveBundledScenariosDir() {
122
+ const here = dirname(fileURLToPath(import.meta.url));
123
+ // Works for both src/commands/smoke.ts and dist/commands/smoke.js —
124
+ // both live two directories below the cli root.
125
+ const bundled = resolve(here, '..', '..', 'test', 'scenarios');
126
+ if (existsSync(bundled))
127
+ return bundled;
128
+ // Last resort: return the expected path so the orchestrator surfaces
129
+ // a clean "scenarios dir not found" diagnostic at the call site
130
+ // rather than the resolver swallowing it silently.
131
+ return bundled;
132
+ }
133
+ //# sourceMappingURL=smoke.js.map
@@ -0,0 +1,134 @@
1
+ /**
2
+ * Agent-progress auto-cleanup — live progress .
3
+ *
4
+ * Rule: a progress file whose status === 'completed' OR 'failed' AND
5
+ * whose `lastUpdate` is older than COMPLETION_TTL_MS (default 5 min)
6
+ * gets MOVED into `<dir>/archive/<id>-<ts>.json` rather than deleted.
7
+ * Operators sometimes want to inspect a finished agent's last state;
8
+ * the archive keeps that affordance while preventing the live watcher
9
+ * from cluttering up with stale rows.
10
+ *
11
+ * The function is pure-ish:
12
+ * - All clock reads go through the injected `now` (defaults Date.now).
13
+ * - Returns a structured report so the caller (the watcher or a CLI
14
+ * cron) can log what got swept.
15
+ * - File-system errors degrade silently — the cleanup is best-effort
16
+ * housekeeping, never a hot path.
17
+ *
18
+ * Hook-in point: `pugi jobs --watch` calls `runCleanup()` once per
19
+ * 60-second tick (cheap — readdir on a small directory). A separate
20
+ * cron entry can call it standalone via the (future) `pugi jobs
21
+ * gc` subcommand if the operator never runs --watch.
22
+ */
23
+ import { existsSync, mkdirSync, readFileSync, readdirSync, renameSync, } from 'node:fs';
24
+ import { join } from 'node:path';
25
+ import { validateAgentProgress, } from './schema.js';
26
+ import { resolveProgressDir } from './writer.js';
27
+ /** Default time a completed/failed entry sits before getting swept. */
28
+ export const COMPLETION_TTL_MS = 5 * 60 * 1000;
29
+ /** Archive subdirectory under the resolved progress dir. */
30
+ export const ARCHIVE_SUBDIR = 'archive';
31
+ /**
32
+ * Run a single cleanup pass. Returns the report for telemetry / tests.
33
+ */
34
+ export function runCleanup(options = {}) {
35
+ const dir = resolveProgressDir(options.dir);
36
+ const ttl = options.ttlMs ?? COMPLETION_TTL_MS;
37
+ const now = options.now ?? Date.now;
38
+ const report = { dir, archived: [], skipped: [] };
39
+ if (!existsSync(dir))
40
+ return report;
41
+ const archiveDir = join(dir, ARCHIVE_SUBDIR);
42
+ let entries;
43
+ try {
44
+ entries = readdirSync(dir);
45
+ }
46
+ catch (err) {
47
+ report.skipped.push({ path: dir, reason: `readdir failed: ${err.message}` });
48
+ return report;
49
+ }
50
+ for (const name of entries) {
51
+ if (!name.endsWith('.json'))
52
+ continue;
53
+ if (/\.tmp-/.test(name))
54
+ continue;
55
+ const path = join(dir, name);
56
+ let body;
57
+ try {
58
+ body = readFileSync(path, 'utf8');
59
+ }
60
+ catch {
61
+ report.skipped.push({ path, reason: 'read failed' });
62
+ continue;
63
+ }
64
+ let parsed;
65
+ try {
66
+ parsed = JSON.parse(body);
67
+ }
68
+ catch {
69
+ report.skipped.push({ path, reason: 'malformed JSON' });
70
+ continue;
71
+ }
72
+ const validation = validateAgentProgress(parsed);
73
+ if (!validation.ok) {
74
+ report.skipped.push({ path, reason: `invalid: ${validation.error}` });
75
+ continue;
76
+ }
77
+ const progress = validation.value;
78
+ if (!isExpired(progress, now(), ttl)) {
79
+ continue;
80
+ }
81
+ if (!existsSync(archiveDir)) {
82
+ try {
83
+ mkdirSync(archiveDir, { recursive: true });
84
+ }
85
+ catch (err) {
86
+ report.skipped.push({
87
+ path,
88
+ reason: `archive mkdir failed: ${err.message}`,
89
+ });
90
+ continue;
91
+ }
92
+ }
93
+ const target = join(archiveDir, `${progress.agentId}-${safeStamp(progress.lastUpdate)}.json`);
94
+ try {
95
+ renameSync(path, target);
96
+ report.archived.push({ agentId: progress.agentId, from: path, to: target });
97
+ }
98
+ catch (err) {
99
+ report.skipped.push({ path, reason: `rename failed: ${err.message}` });
100
+ }
101
+ }
102
+ return report;
103
+ }
104
+ /**
105
+ * Decide whether a single progress doc has aged out. Exported for the
106
+ * spec — kept pure so tests can probe edges (running entries never
107
+ * expire, completed entries before TTL stay, etc).
108
+ */
109
+ export function isExpired(progress, nowEpochMs, ttlMs = COMPLETION_TTL_MS) {
110
+ if (progress.status === 'running')
111
+ return false;
112
+ const lastTs = Date.parse(progress.lastUpdate);
113
+ if (Number.isNaN(lastTs))
114
+ return false;
115
+ return nowEpochMs - lastTs >= ttlMs;
116
+ }
117
+ // Claude review followup: `pruneArchive` was removed.
118
+ // Rationale (P1 in the Claude review batch on PR):
119
+ // - hardcoded `/tmp` path was POSIX-only (Windows would break),
120
+ // - `renameSync` across volumes throws EXDEV,
121
+ // - collision risk between concurrent hosts/sessions sharing `/tmp`,
122
+ // - and crucially, the function had ZERO call-sites in the codebase.
123
+ // Deleting it is strictly safer than leaving a broken-on-Windows
124
+ // dead helper that future refactors might accidentally wire up.
125
+ // If/when an archive GC is needed, the right shape is:
126
+ // - `os.tmpdir()` (not '/tmp'),
127
+ // - `rmSync(path, {force: true})` for cross-volume safe delete,
128
+ // - exposed as `pugi jobs gc` so we have an explicit caller.
129
+ function safeStamp(iso) {
130
+ // Build a filename-safe slug — strip colons/dots which are friendly
131
+ // в ISO timestamps but hostile к some filesystems.
132
+ return iso.replace(/[:.]/g, '-');
133
+ }
134
+ //# sourceMappingURL=cleanup.js.map
@@ -0,0 +1,144 @@
1
+ /**
2
+ * Agent progress JSON schema — live-progress sprint .
3
+ *
4
+ * Long-running agents (spawned externally OR via `pugi /agent`) emit a
5
+ * single JSON document per agent to `~/.pugi/agent-progress/<id>.json`.
6
+ * `pugi jobs --watch` tails the directory via chokidar and re-renders
7
+ * an Ink TUI that mirrors the the upstream tool `/compact` visual pattern
8
+ * (header with elapsed + token counter, unicode progress bar, milestone
9
+ * list with done/active/pending status icons).
10
+ *
11
+ * Schema is intentionally optimistic — every numeric field is clamped
12
+ * by the writer/reader so a malformed document degrades to a partial
13
+ * card instead of crashing the watcher. The `pendingCount` /
14
+ * `completedCount` fields are pre-computed by the agent for the
15
+ * "… +N pending, M completed" footer; the renderer never re-counts
16
+ * (the agent may have collapsed milestone history to save bytes).
17
+ */
18
+ /**
19
+ * Validate an unknown payload as an `AgentProgress` document. Returns
20
+ * the typed value on success and a string error otherwise. We deliberately
21
+ * keep this hand-rolled (no zod) — every field is checked exactly once,
22
+ * the error message is human-readable, and zero runtime deps.
23
+ */
24
+ export function validateAgentProgress(value) {
25
+ if (typeof value !== 'object' || value === null) {
26
+ return { ok: false, error: 'progress payload must be a JSON object' };
27
+ }
28
+ const raw = value;
29
+ const requiredString = (field) => {
30
+ const v = raw[field];
31
+ return typeof v === 'string' && v.length > 0 ? v : null;
32
+ };
33
+ const agentId = requiredString('agentId');
34
+ if (!agentId)
35
+ return { ok: false, error: 'agentId required (non-empty string)' };
36
+ if (!/^[a-zA-Z0-9_-]+$/.test(agentId)) {
37
+ return { ok: false, error: 'agentId must match [a-zA-Z0-9_-]+ (filename-safe)' };
38
+ }
39
+ const agentType = requiredString('agentType');
40
+ if (!agentType)
41
+ return { ok: false, error: 'agentType required (non-empty string)' };
42
+ const task = requiredString('task');
43
+ if (!task)
44
+ return { ok: false, error: 'task required (non-empty string)' };
45
+ const startedAt = requiredString('startedAt');
46
+ if (!startedAt)
47
+ return { ok: false, error: 'startedAt required (ISO string)' };
48
+ if (Number.isNaN(Date.parse(startedAt))) {
49
+ return { ok: false, error: 'startedAt must be a parseable ISO timestamp' };
50
+ }
51
+ const lastUpdate = requiredString('lastUpdate');
52
+ if (!lastUpdate)
53
+ return { ok: false, error: 'lastUpdate required (ISO string)' };
54
+ if (Number.isNaN(Date.parse(lastUpdate))) {
55
+ return { ok: false, error: 'lastUpdate must be a parseable ISO timestamp' };
56
+ }
57
+ const elapsedMs = raw.elapsedMs;
58
+ if (typeof elapsedMs !== 'number' || !Number.isFinite(elapsedMs) || elapsedMs < 0) {
59
+ return { ok: false, error: 'elapsedMs required (non-negative number)' };
60
+ }
61
+ const percentComplete = raw.percentComplete;
62
+ if (typeof percentComplete !== 'number' ||
63
+ !Number.isFinite(percentComplete)) {
64
+ return { ok: false, error: 'percentComplete required (number 0..100)' };
65
+ }
66
+ const clampedPercent = Math.max(0, Math.min(100, percentComplete));
67
+ const status = raw.status;
68
+ if (status !== 'running' && status !== 'completed' && status !== 'failed') {
69
+ return { ok: false, error: 'status must be running | completed | failed' };
70
+ }
71
+ const currentStep = raw.currentStep;
72
+ if (typeof currentStep !== 'number' || currentStep < 0) {
73
+ return { ok: false, error: 'currentStep required (non-negative number)' };
74
+ }
75
+ const totalSteps = raw.totalSteps;
76
+ if (typeof totalSteps !== 'number' || totalSteps < 0) {
77
+ return { ok: false, error: 'totalSteps required (non-negative number)' };
78
+ }
79
+ const stepDescription = typeof raw.stepDescription === 'string'
80
+ ? raw.stepDescription
81
+ : '';
82
+ const milestonesRaw = raw.milestones;
83
+ if (!Array.isArray(milestonesRaw)) {
84
+ return { ok: false, error: 'milestones required (array, may be empty)' };
85
+ }
86
+ const milestones = [];
87
+ for (let i = 0; i < milestonesRaw.length; i += 1) {
88
+ const m = milestonesRaw[i];
89
+ if (typeof m !== 'object' || m === null) {
90
+ return { ok: false, error: `milestones[${i}] must be an object` };
91
+ }
92
+ const mr = m;
93
+ const label = typeof mr.label === 'string' ? mr.label : '';
94
+ if (!label) {
95
+ return { ok: false, error: `milestones[${i}].label required` };
96
+ }
97
+ const mStatus = mr.status;
98
+ if (mStatus !== 'done' && mStatus !== 'active' && mStatus !== 'pending') {
99
+ return {
100
+ ok: false,
101
+ error: `milestones[${i}].status must be done | active | pending`,
102
+ };
103
+ }
104
+ const ts = typeof mr.ts === 'string' ? mr.ts : undefined;
105
+ milestones.push({ label, status: mStatus, ts });
106
+ }
107
+ const tokensUsed = typeof raw.tokensUsed === 'number' && Number.isFinite(raw.tokensUsed)
108
+ ? Math.max(0, raw.tokensUsed)
109
+ : undefined;
110
+ const pendingCount = typeof raw.pendingCount === 'number' && Number.isFinite(raw.pendingCount)
111
+ ? Math.max(0, Math.floor(raw.pendingCount))
112
+ : undefined;
113
+ const completedCount = typeof raw.completedCount === 'number' && Number.isFinite(raw.completedCount)
114
+ ? Math.max(0, Math.floor(raw.completedCount))
115
+ : undefined;
116
+ return {
117
+ ok: true,
118
+ value: {
119
+ agentId,
120
+ agentType,
121
+ task,
122
+ startedAt,
123
+ lastUpdate,
124
+ elapsedMs,
125
+ tokensUsed,
126
+ percentComplete: clampedPercent,
127
+ status,
128
+ currentStep,
129
+ totalSteps,
130
+ stepDescription,
131
+ milestones,
132
+ pendingCount,
133
+ completedCount,
134
+ },
135
+ };
136
+ }
137
+ /**
138
+ * Default directory the writer and watcher share. Lives under the
139
+ * project workspace by convention so worktree-isolated agents can emit
140
+ * progress without crossing tenant boundaries. Operators can override
141
+ * via `PUGI_AGENT_PROGRESS_DIR` env var.
142
+ */
143
+ export const DEFAULT_AGENT_PROGRESS_DIRNAME = '.pugi/agent-progress';
144
+ //# sourceMappingURL=schema.js.map