aiwcli 0.15.5 → 0.17.0

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 (435) hide show
  1. package/README.md +108 -1124
  2. package/bin/run.js +0 -4
  3. package/dist/capabilities/branch/adapters.d.ts +2 -0
  4. package/dist/capabilities/branch/adapters.js +21 -0
  5. package/dist/capabilities/branch/contracts.d.ts +57 -0
  6. package/dist/capabilities/branch/contracts.js +1 -0
  7. package/dist/capabilities/branch/control-plane.d.ts +2 -0
  8. package/dist/capabilities/branch/control-plane.js +343 -0
  9. package/dist/capabilities/branch/runtime-core.d.ts +5 -0
  10. package/dist/capabilities/branch/runtime-core.js +36 -0
  11. package/dist/capabilities/installation/control-plane/clean-command.d.ts +41 -0
  12. package/dist/capabilities/installation/control-plane/clean-command.js +196 -0
  13. package/dist/capabilities/installation/control-plane/clear-command.d.ts +162 -0
  14. package/dist/capabilities/installation/control-plane/clear-command.js +1249 -0
  15. package/dist/capabilities/installation/control-plane/init-command.d.ts +81 -0
  16. package/dist/capabilities/installation/control-plane/init-command.js +449 -0
  17. package/dist/capabilities/launch/contracts.d.ts +86 -0
  18. package/dist/capabilities/launch/contracts.js +1 -0
  19. package/dist/capabilities/launch/control-plane/execute-launch.d.ts +2 -0
  20. package/dist/capabilities/launch/control-plane/execute-launch.js +261 -0
  21. package/dist/capabilities/launch/runtime-core/launch-decisions.d.ts +82 -0
  22. package/dist/capabilities/launch/runtime-core/launch-decisions.js +202 -0
  23. package/dist/capabilities/launch/runtime-core/launch-options.d.ts +14 -0
  24. package/dist/capabilities/launch/runtime-core/launch-options.js +69 -0
  25. package/dist/cli/base-command.d.ts +18 -0
  26. package/dist/cli/base-command.js +55 -0
  27. package/dist/commands/branch.d.ts +1 -21
  28. package/dist/commands/branch.js +25 -417
  29. package/dist/commands/clean.d.ts +1 -41
  30. package/dist/commands/clean.js +1 -196
  31. package/dist/commands/clear.d.ts +1 -161
  32. package/dist/commands/clear.js +1 -1121
  33. package/dist/commands/init/index.d.ts +1 -98
  34. package/dist/commands/init/index.js +4 -478
  35. package/dist/commands/launch.d.ts +32 -12
  36. package/dist/commands/launch.js +107 -166
  37. package/dist/lib/claude-settings-types.d.ts +31 -19
  38. package/dist/lib/config.js +1 -2
  39. package/dist/lib/context/context-formatter.d.ts +74 -0
  40. package/dist/lib/context/context-formatter.js +493 -0
  41. package/dist/lib/context/context-selector.d.ts +42 -0
  42. package/dist/lib/context/context-selector.js +451 -0
  43. package/dist/lib/context/context-store.d.ts +100 -0
  44. package/dist/lib/context/context-store.js +644 -0
  45. package/dist/lib/context/plan-manager.d.ts +54 -0
  46. package/dist/lib/context/plan-manager.js +282 -0
  47. package/dist/lib/context/task-tracker.d.ts +44 -0
  48. package/dist/lib/context/task-tracker.js +146 -0
  49. package/dist/lib/core-ide-base.d.ts +4 -0
  50. package/dist/lib/core-ide-base.js +77 -0
  51. package/dist/lib/core-installer.d.ts +5 -0
  52. package/dist/lib/core-installer.js +33 -0
  53. package/dist/lib/debug.d.ts +0 -10
  54. package/dist/lib/debug.js +0 -10
  55. package/dist/lib/env-sanitizer.d.ts +25 -0
  56. package/dist/lib/env-sanitizer.js +46 -0
  57. package/dist/lib/errors.d.ts +0 -13
  58. package/dist/lib/errors.js +0 -15
  59. package/dist/lib/git-exclude-manager.d.ts +2 -2
  60. package/dist/lib/git-exclude-manager.js +3 -3
  61. package/dist/lib/hooks/context-monitor-logic.d.ts +6 -0
  62. package/dist/lib/hooks/context-monitor-logic.js +25 -0
  63. package/dist/lib/hooks/hook-utils.d.ts +143 -0
  64. package/dist/lib/hooks/hook-utils.js +620 -0
  65. package/dist/lib/hooks/prompt-binding-logic.d.ts +7 -0
  66. package/dist/lib/hooks/prompt-binding-logic.js +50 -0
  67. package/dist/lib/hooks/session-end-logic.d.ts +5 -0
  68. package/dist/lib/hooks/session-end-logic.js +51 -0
  69. package/dist/lib/hooks-merger.js +25 -19
  70. package/dist/lib/ide-path-resolver.d.ts +19 -7
  71. package/dist/lib/ide-path-resolver.js +25 -9
  72. package/dist/lib/install-state.d.ts +34 -0
  73. package/dist/lib/install-state.js +154 -0
  74. package/dist/lib/json-io.d.ts +12 -0
  75. package/dist/lib/json-io.js +30 -0
  76. package/dist/lib/lsp-patch.d.ts +12 -0
  77. package/dist/lib/lsp-patch.js +156 -0
  78. package/dist/lib/multiplexer.d.ts +65 -0
  79. package/dist/lib/multiplexer.js +38 -0
  80. package/dist/lib/multiplexers/psmux.d.ts +55 -0
  81. package/dist/lib/multiplexers/psmux.js +324 -0
  82. package/dist/lib/multiplexers/tmux.d.ts +36 -0
  83. package/dist/lib/multiplexers/tmux.js +221 -0
  84. package/dist/lib/multiplexers/wezterm.d.ts +38 -0
  85. package/dist/lib/multiplexers/wezterm.js +225 -0
  86. package/dist/lib/mux-utils.d.ts +6 -0
  87. package/dist/lib/mux-utils.js +36 -0
  88. package/dist/lib/paths.d.ts +2 -2
  89. package/dist/lib/paths.js +2 -2
  90. package/dist/lib/platform-commands.d.ts +27 -0
  91. package/dist/lib/platform-commands.js +49 -0
  92. package/dist/lib/prompt-file-manager.d.ts +23 -0
  93. package/dist/lib/prompt-file-manager.js +41 -0
  94. package/dist/lib/runtime/agent-launcher.d.ts +67 -0
  95. package/dist/lib/runtime/agent-launcher.js +262 -0
  96. package/dist/lib/runtime/aiw-cli.d.ts +39 -0
  97. package/dist/lib/runtime/aiw-cli.js +76 -0
  98. package/dist/lib/runtime/atomic-write.d.ts +19 -0
  99. package/dist/lib/runtime/atomic-write.js +121 -0
  100. package/dist/lib/runtime/cli-args.d.ts +58 -0
  101. package/dist/lib/runtime/cli-args.js +200 -0
  102. package/dist/lib/runtime/constants.d.ts +56 -0
  103. package/dist/lib/runtime/constants.js +230 -0
  104. package/dist/lib/runtime/executable-policy.d.ts +16 -0
  105. package/dist/lib/runtime/executable-policy.js +57 -0
  106. package/dist/lib/runtime/git-state.d.ts +9 -0
  107. package/dist/lib/runtime/git-state.js +59 -0
  108. package/dist/lib/runtime/inference.d.ts +37 -0
  109. package/dist/lib/runtime/inference.js +251 -0
  110. package/dist/lib/runtime/lint-dispatch.d.ts +40 -0
  111. package/dist/lib/runtime/lint-dispatch.js +285 -0
  112. package/dist/lib/runtime/logger.d.ts +66 -0
  113. package/dist/lib/runtime/logger.js +201 -0
  114. package/dist/lib/runtime/models.d.ts +20 -0
  115. package/dist/lib/runtime/models.js +20 -0
  116. package/dist/lib/runtime/platform-adapter.d.ts +7 -0
  117. package/dist/lib/runtime/platform-adapter.js +21 -0
  118. package/dist/lib/runtime/preflight.d.ts +24 -0
  119. package/dist/lib/runtime/preflight.js +65 -0
  120. package/dist/lib/runtime/sentinel-ipc.d.ts +14 -0
  121. package/dist/lib/runtime/sentinel-ipc.js +67 -0
  122. package/dist/lib/runtime/state-io.d.ts +31 -0
  123. package/dist/lib/runtime/state-io.js +179 -0
  124. package/dist/lib/runtime/stop-words.d.ts +20 -0
  125. package/dist/lib/runtime/stop-words.js +150 -0
  126. package/dist/lib/runtime/subprocess-utils.d.ts +29 -0
  127. package/dist/lib/runtime/subprocess-utils.js +96 -0
  128. package/dist/lib/runtime/tmux-preflight.d.ts +13 -0
  129. package/dist/lib/runtime/tmux-preflight.js +78 -0
  130. package/dist/lib/runtime/utils.d.ts +62 -0
  131. package/dist/lib/runtime/utils.js +192 -0
  132. package/dist/lib/schemas.d.ts +250 -0
  133. package/dist/lib/schemas.js +216 -0
  134. package/dist/lib/sentinel-manager.d.ts +32 -0
  135. package/dist/lib/sentinel-manager.js +62 -0
  136. package/dist/lib/sentinel-wrapper.d.ts +10 -0
  137. package/dist/lib/sentinel-wrapper.js +29 -0
  138. package/dist/lib/settings-hierarchy.js +3 -20
  139. package/dist/lib/shell-adapters/bash-adapter.d.ts +18 -0
  140. package/dist/lib/shell-adapters/bash-adapter.js +69 -0
  141. package/dist/lib/shell-adapters/index.d.ts +5 -0
  142. package/dist/lib/shell-adapters/index.js +7 -0
  143. package/dist/lib/shell-adapters/powershell-adapter.d.ts +18 -0
  144. package/dist/lib/shell-adapters/powershell-adapter.js +62 -0
  145. package/dist/lib/shell-adapters/shell-adapter.d.ts +45 -0
  146. package/dist/lib/shell-adapters/shell-adapter.js +5 -0
  147. package/dist/lib/shell-quoting.d.ts +5 -0
  148. package/dist/lib/shell-quoting.js +17 -0
  149. package/dist/lib/spawn-errors.d.ts +9 -0
  150. package/dist/lib/spawn-errors.js +29 -0
  151. package/dist/lib/spawn.js +5 -11
  152. package/dist/lib/spinner.d.ts +0 -5
  153. package/dist/lib/spinner.js +0 -16
  154. package/dist/lib/template-installer.d.ts +14 -5
  155. package/dist/lib/template-installer.js +40 -38
  156. package/dist/lib/template-resolver.d.ts +6 -7
  157. package/dist/lib/template-resolver.js +26 -21
  158. package/dist/lib/template-settings-reconstructor.d.ts +7 -2
  159. package/dist/lib/template-settings-reconstructor.js +76 -45
  160. package/dist/lib/terminal-strategy.d.ts +12 -0
  161. package/dist/lib/terminal-strategy.js +55 -0
  162. package/dist/lib/terminal.d.ts +34 -4
  163. package/dist/lib/terminal.js +192 -119
  164. package/dist/lib/tmux-pane-placement.d.ts +17 -0
  165. package/dist/lib/tmux-pane-placement.js +58 -0
  166. package/dist/lib/tmux-primitives.d.ts +3 -0
  167. package/dist/lib/tmux-primitives.js +11 -0
  168. package/dist/lib/tmux-session.d.ts +32 -0
  169. package/dist/lib/tmux-session.js +87 -0
  170. package/dist/lib/tty-detection.js +1 -1
  171. package/dist/lib/types.d.ts +168 -0
  172. package/dist/lib/types.js +6 -0
  173. package/dist/lib/version.d.ts +1 -1
  174. package/dist/lib/version.js +1 -1
  175. package/dist/lib/windsurf-hooks-hierarchy.js +6 -23
  176. package/dist/platform/launch.d.ts +11 -0
  177. package/dist/platform/launch.js +11 -0
  178. package/dist/templates/CLAUDE.md +30 -40
  179. package/dist/templates/cc-native/.claude/settings.json +26 -36
  180. package/dist/templates/cc-native/CC-NATIVE-README.md +1 -1
  181. package/dist/templates/cc-native/TEMPLATE-SCHEMA.md +20 -12
  182. package/dist/templates/cc-native/_cc-native/cc-native.config.json +2 -6
  183. package/dist/templates/cc-native/_cc-native/hooks/CLAUDE.md +39 -59
  184. package/dist/templates/cc-native/_cc-native/hooks/cc-native-plan-review.ts +9 -11
  185. package/dist/templates/cc-native/_cc-native/hooks/enhance_plan_post_subagent.ts +2 -2
  186. package/dist/templates/cc-native/_cc-native/hooks/enhance_plan_post_write.ts +4 -5
  187. package/dist/templates/cc-native/_cc-native/hooks/mark_questions_asked.ts +4 -4
  188. package/dist/templates/cc-native/_cc-native/hooks/plan_questions_early.ts +2 -27
  189. package/dist/templates/cc-native/_cc-native/hooks/validate_task_prompt.ts +7 -7
  190. package/dist/templates/cc-native/_cc-native/lib-ts/.mocharc.json +9 -0
  191. package/dist/templates/cc-native/_cc-native/lib-ts/__tests__/aggregate-agents.test.ts +118 -0
  192. package/dist/templates/cc-native/_cc-native/lib-ts/__tests__/artifacts.test.ts +234 -0
  193. package/dist/templates/cc-native/_cc-native/lib-ts/__tests__/cc-native-state.test.ts +170 -0
  194. package/dist/templates/cc-native/_cc-native/lib-ts/__tests__/cli-output-parser.test.ts +73 -0
  195. package/dist/templates/cc-native/_cc-native/lib-ts/__tests__/config.test.ts +64 -0
  196. package/dist/templates/cc-native/_cc-native/lib-ts/__tests__/constants.test.ts +40 -0
  197. package/dist/templates/cc-native/_cc-native/lib-ts/__tests__/debug.test.ts +42 -0
  198. package/dist/templates/cc-native/_cc-native/lib-ts/__tests__/exports.test.ts +58 -0
  199. package/dist/templates/cc-native/_cc-native/lib-ts/__tests__/helpers.ts +107 -0
  200. package/dist/templates/cc-native/_cc-native/lib-ts/__tests__/hooks/add-plan-context.hook.test.ts +97 -0
  201. package/dist/templates/cc-native/_cc-native/lib-ts/__tests__/hooks/plan-questions.hook.test.ts +81 -0
  202. package/dist/templates/cc-native/_cc-native/lib-ts/__tests__/hooks/plan-review.hook.test.ts +71 -0
  203. package/dist/templates/cc-native/_cc-native/lib-ts/__tests__/json-parser.test.ts +99 -0
  204. package/dist/templates/cc-native/_cc-native/lib-ts/__tests__/orchestrator-agent.test.ts +288 -0
  205. package/dist/templates/cc-native/_cc-native/lib-ts/__tests__/orchestrator.test.ts +48 -0
  206. package/dist/templates/cc-native/_cc-native/lib-ts/__tests__/reviewers.test.ts +32 -0
  207. package/dist/templates/cc-native/_cc-native/lib-ts/__tests__/state.test.ts +124 -0
  208. package/dist/templates/cc-native/_cc-native/lib-ts/__tests__/verdict.test.ts +93 -0
  209. package/dist/templates/cc-native/_cc-native/lib-ts/agent-selection.ts +163 -0
  210. package/dist/templates/cc-native/_cc-native/lib-ts/aggregate-agents.ts +6 -14
  211. package/dist/templates/cc-native/_cc-native/{artifacts/lib → lib-ts/artifacts}/format.ts +597 -599
  212. package/dist/templates/cc-native/_cc-native/{artifacts/lib → lib-ts/artifacts}/index.ts +26 -26
  213. package/dist/templates/cc-native/_cc-native/{artifacts/lib → lib-ts/artifacts}/tracker.ts +106 -107
  214. package/dist/templates/cc-native/_cc-native/{artifacts/lib → lib-ts/artifacts}/write.ts +118 -119
  215. package/dist/templates/cc-native/_cc-native/lib-ts/artifacts.ts +21 -0
  216. package/dist/templates/cc-native/_cc-native/lib-ts/cc-native-state.ts +17 -16
  217. package/dist/templates/cc-native/_cc-native/lib-ts/cli-output-parser.ts +132 -10
  218. package/dist/templates/cc-native/_cc-native/lib-ts/config.ts +1 -1
  219. package/dist/templates/cc-native/_cc-native/lib-ts/constants.ts +6 -6
  220. package/dist/templates/cc-native/_cc-native/lib-ts/corroboration.ts +119 -0
  221. package/dist/templates/cc-native/_cc-native/lib-ts/debug.ts +2 -3
  222. package/dist/templates/cc-native/_cc-native/{plan-review/lib → lib-ts}/graduation.ts +132 -132
  223. package/dist/templates/cc-native/_cc-native/lib-ts/index.ts +88 -86
  224. package/dist/templates/cc-native/_cc-native/lib-ts/json-parser.ts +5 -6
  225. package/dist/templates/cc-native/_cc-native/{plan-review/lib → lib-ts}/orchestrator.ts +70 -70
  226. package/dist/templates/cc-native/_cc-native/{plan-review/lib → lib-ts}/output-builder.ts +130 -121
  227. package/dist/templates/cc-native/_cc-native/lib-ts/package-lock.json +1679 -0
  228. package/dist/templates/cc-native/_cc-native/lib-ts/package.json +24 -0
  229. package/dist/templates/cc-native/_cc-native/lib-ts/plan-discovery.ts +5 -5
  230. package/dist/templates/cc-native/_cc-native/lib-ts/plan-enhancement.ts +1 -6
  231. package/dist/templates/cc-native/_cc-native/{plan-review/lib → lib-ts}/plan-questions.ts +101 -101
  232. package/dist/templates/cc-native/_cc-native/{plan-review/lib → lib-ts}/review-pipeline.ts +511 -543
  233. package/dist/templates/cc-native/_cc-native/lib-ts/reviewers/__tests__/agent-providers.test.ts +262 -0
  234. package/dist/templates/cc-native/_cc-native/{plan-review/lib → lib-ts}/reviewers/agent.ts +71 -85
  235. package/dist/templates/{_shared/lib-ts/agent-exec → cc-native/_cc-native/lib-ts/reviewers/base}/base-agent.ts +138 -150
  236. package/dist/templates/cc-native/_cc-native/{plan-review/lib → lib-ts}/reviewers/index.ts +12 -12
  237. package/dist/templates/cc-native/_cc-native/{plan-review/lib → lib-ts}/reviewers/providers/claude-agent.ts +66 -57
  238. package/dist/templates/cc-native/_cc-native/{plan-review/lib → lib-ts}/reviewers/providers/codex-agent.ts +185 -200
  239. package/dist/templates/cc-native/_cc-native/{plan-review/lib → lib-ts}/reviewers/providers/gemini-agent.ts +39 -40
  240. package/dist/templates/cc-native/_cc-native/{plan-review/lib → lib-ts}/reviewers/providers/orchestrator-claude-agent.ts +196 -225
  241. package/dist/templates/cc-native/_cc-native/{plan-review/lib → lib-ts}/reviewers/schemas.ts +201 -201
  242. package/dist/templates/cc-native/_cc-native/{plan-review/lib → lib-ts}/reviewers/types.ts +21 -23
  243. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/__tests__/hyde.test.ts +365 -0
  244. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/__tests__/ollama-client.test.ts +223 -0
  245. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/embedding-indexer.ts +12 -16
  246. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/hyde.ts +3 -2
  247. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/index.ts +31 -31
  248. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/logger.ts +7 -8
  249. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/ollama-client.ts +7 -9
  250. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/retrieval-pipeline.ts +16 -19
  251. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/transcript-indexer.ts +37 -41
  252. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/transcript-loader.ts +33 -43
  253. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/transcript-searcher.ts +20 -20
  254. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/types.ts +9 -10
  255. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/vector-store.ts +3 -4
  256. package/dist/templates/cc-native/_cc-native/lib-ts/settings.ts +50 -126
  257. package/dist/templates/cc-native/_cc-native/lib-ts/state.ts +20 -22
  258. package/dist/templates/cc-native/_cc-native/lib-ts/tsconfig.json +2 -2
  259. package/dist/templates/cc-native/_cc-native/lib-ts/types.ts +14 -89
  260. package/dist/templates/cc-native/_cc-native/{plan-review/lib → lib-ts}/verdict.ts +72 -72
  261. package/dist/templates/cc-native/_cc-native/plan-review/CLAUDE.md +38 -1
  262. package/dist/templates/cc-native/_cc-native/plan-review/lib/__tests__/agent-selection.test.ts +345 -0
  263. package/dist/templates/cc-native/_cc-native/plan-review/lib/__tests__/preflight.test.ts +344 -0
  264. package/dist/templates/cc-native/_cc-native/plan-review/lib/agent-selection.ts +38 -16
  265. package/dist/templates/cc-native/_cc-native/plan-review/lib/preflight.ts +56 -26
  266. package/dist/templates/cc-native/_cc-native/scripts/council_debate.ts +242 -0
  267. package/dist/templates/cc-native/_cc-native/scripts/council_debate_simple.ts +294 -0
  268. package/dist/templates/cc-native/_cc-native/{plan-review/workflows → workflows}/specdev.md +9 -9
  269. package/dist/templates/core/.claude/skills/codex/SKILL.md +25 -0
  270. package/dist/templates/core/.claude/skills/devin/SKILL.md +25 -0
  271. package/dist/templates/core/.claude/skills/handoff/SKILL.md +11 -0
  272. package/dist/templates/core/.claude/skills/handoff-resume/SKILL.md +11 -0
  273. package/dist/templates/core/.claude/skills/meta-plan/SKILL.md +13 -0
  274. package/dist/templates/core/.codex/skills/codex/SKILL.md +13 -0
  275. package/dist/templates/core/.codex/skills/devin/SKILL.md +19 -0
  276. package/dist/templates/core/.codex/skills/handoff/SKILL.md +11 -0
  277. package/dist/templates/core/.codex/skills/handoff-resume/SKILL.md +11 -0
  278. package/dist/templates/core/.codex/skills/meta-plan/SKILL.md +13 -0
  279. package/dist/templates/core/.devin/AGENTS.md +5 -0
  280. package/dist/templates/core/.devin/config.json +12 -0
  281. package/dist/templates/core/.devin/skills/codex/SKILL.md +19 -0
  282. package/dist/templates/core/.devin/skills/devin/SKILL.md +13 -0
  283. package/dist/templates/core/.devin/skills/handoff/SKILL.md +11 -0
  284. package/dist/templates/core/.devin/skills/handoff-resume/SKILL.md +11 -0
  285. package/dist/templates/core/.devin/skills/meta-plan/SKILL.md +13 -0
  286. package/dist/templates/core/.windsurf/workflows/handoff-resume.md +9 -0
  287. package/dist/templates/{_shared → core}/.windsurf/workflows/handoff.md +1 -1
  288. package/dist/templates/{_shared → core}/.windsurf/workflows/meta-plan.md +1 -1
  289. package/dist/templates/core/hooks-ts/_utils/git-state.ts +2 -0
  290. package/dist/templates/{_shared → core}/hooks-ts/archive_plan.ts +15 -44
  291. package/dist/templates/core/hooks-ts/codex_explorer.ts +160 -0
  292. package/dist/templates/{_shared → core}/hooks-ts/context_monitor.ts +23 -55
  293. package/dist/templates/{_shared → core}/hooks-ts/file-suggestion.ts +5 -22
  294. package/dist/templates/{_shared → core}/hooks-ts/lint_after_edit.ts +7 -9
  295. package/dist/templates/core/hooks-ts/pre_compact.ts +36 -0
  296. package/dist/templates/{_shared → core}/hooks-ts/session_end.ts +38 -78
  297. package/dist/templates/{_shared → core}/hooks-ts/session_start.ts +5 -5
  298. package/dist/templates/core/hooks-ts/task_create_capture.ts +32 -0
  299. package/dist/templates/{_shared → core}/hooks-ts/task_update_capture.ts +9 -24
  300. package/dist/templates/core/hooks-ts/user_prompt_submit.ts +46 -0
  301. package/dist/templates/{_shared → core}/lib-ts/CLAUDE.md +27 -16
  302. package/dist/templates/{_shared → core}/lib-ts/context/CLAUDE.md +9 -6
  303. package/dist/templates/{_shared → core}/lib-ts/context/context-formatter.ts +16 -21
  304. package/dist/templates/{_shared → core}/lib-ts/context/context-selector.ts +8 -6
  305. package/dist/templates/{_shared → core}/lib-ts/context/context-store.ts +59 -20
  306. package/dist/templates/{_shared → core}/lib-ts/context/plan-manager.ts +19 -15
  307. package/dist/templates/{_shared → core}/lib-ts/context/task-tracker.ts +3 -3
  308. package/dist/templates/core/lib-ts/hooks/context-monitor-logic.ts +32 -0
  309. package/dist/templates/{_shared/lib-ts/base → core/lib-ts/hooks}/hook-utils.ts +179 -41
  310. package/dist/templates/core/lib-ts/hooks/prompt-binding-logic.ts +80 -0
  311. package/dist/templates/core/lib-ts/hooks/session-end-logic.ts +82 -0
  312. package/dist/templates/core/lib-ts/package.json +19 -0
  313. package/dist/templates/core/lib-ts/runtime/agent-launcher.ts +369 -0
  314. package/dist/templates/core/lib-ts/runtime/aiw-cli.ts +108 -0
  315. package/dist/templates/{_shared/lib-ts/base → core/lib-ts/runtime}/atomic-write.ts +12 -7
  316. package/dist/templates/{_shared/lib-ts/base → core/lib-ts/runtime}/cli-args.ts +24 -8
  317. package/dist/templates/{_shared/lib-ts/base → core/lib-ts/runtime}/constants.ts +326 -324
  318. package/dist/templates/core/lib-ts/runtime/executable-policy.ts +89 -0
  319. package/dist/templates/{_shared/lib-ts/base → core/lib-ts/runtime}/git-state.ts +6 -4
  320. package/dist/templates/{_shared/lib-ts/base → core/lib-ts/runtime}/inference.ts +60 -23
  321. package/dist/templates/{_shared/lib-ts/base → core/lib-ts/runtime}/lint-dispatch.ts +25 -23
  322. package/dist/templates/{_shared/lib-ts/base → core/lib-ts/runtime}/logger.ts +32 -29
  323. package/dist/templates/{_shared/lib-ts/base → core/lib-ts/runtime}/models.ts +9 -2
  324. package/dist/templates/core/lib-ts/runtime/platform-adapter.ts +33 -0
  325. package/dist/templates/{_shared/lib-ts/base → core/lib-ts/runtime}/preflight.ts +4 -3
  326. package/dist/templates/core/lib-ts/runtime/sentinel-ipc.ts +91 -0
  327. package/dist/templates/{_shared/lib-ts/base → core/lib-ts/runtime}/state-io.ts +20 -11
  328. package/dist/templates/core/lib-ts/runtime/stop-words.ts +185 -0
  329. package/dist/templates/core/lib-ts/runtime/subprocess-utils.ts +147 -0
  330. package/dist/templates/core/lib-ts/runtime/tmux-preflight.ts +93 -0
  331. package/dist/templates/{_shared/lib-ts/base → core/lib-ts/runtime}/utils.ts +34 -4
  332. package/dist/templates/core/lib-ts/schemas.ts +233 -0
  333. package/dist/templates/{_shared → core}/lib-ts/templates/formatters.ts +7 -5
  334. package/dist/templates/{_shared → core}/lib-ts/templates/plan-context.ts +2 -1
  335. package/dist/templates/{_shared → core}/lib-ts/tsconfig.json +3 -1
  336. package/dist/templates/{_shared → core}/lib-ts/types.ts +78 -77
  337. package/dist/templates/core/scripts/resolve-run.ts +93 -0
  338. package/dist/templates/{_shared → core}/scripts/resolve_context.ts +3 -3
  339. package/dist/templates/{_shared → core}/scripts/status_line.ts +26 -21
  340. package/dist/templates/core/skills/codex/CLAUDE.md +83 -0
  341. package/dist/templates/{_shared → core}/skills/codex/SKILL.md +27 -18
  342. package/dist/templates/{_shared → core}/skills/codex/lib/codex-watcher.ts +79 -113
  343. package/dist/templates/{_shared → core}/skills/codex/scripts/launch-codex.ts +134 -148
  344. package/dist/templates/{_shared → core}/skills/codex/scripts/watch-codex.ts +6 -4
  345. package/dist/templates/core/skills/devin/CLAUDE.md +122 -0
  346. package/dist/templates/core/skills/devin/SKILL.md +73 -0
  347. package/dist/templates/core/skills/devin/lib/devin-watcher.ts +300 -0
  348. package/dist/templates/core/skills/devin/scripts/launch-devin.ts +258 -0
  349. package/dist/templates/{_shared → core}/skills/handoff-system/CLAUDE.md +436 -433
  350. package/dist/templates/{_shared → core}/skills/handoff-system/lib/document-generator.ts +9 -7
  351. package/dist/templates/{_shared → core}/skills/handoff-system/lib/handoff-reader.ts +6 -4
  352. package/dist/templates/{_shared → core}/skills/handoff-system/scripts/resume_handoff.ts +10 -8
  353. package/dist/templates/{_shared → core}/skills/handoff-system/scripts/save_handoff.ts +12 -10
  354. package/dist/templates/{_shared → core}/skills/handoff-system/workflows/handoff-resume.md +2 -2
  355. package/dist/templates/{_shared → core}/skills/handoff-system/workflows/handoff.md +6 -5
  356. package/dist/templates/{_shared → core}/skills/meta-plan/CLAUDE.md +2 -1
  357. package/dist/templates/{_shared → core}/skills/meta-plan/workflows/meta-plan.md +8 -7
  358. package/oclif.manifest.json +89 -13
  359. package/package.json +13 -12
  360. package/dist/lib/base-command.d.ts +0 -114
  361. package/dist/lib/base-command.js +0 -153
  362. package/dist/lib/env-compat.d.ts +0 -18
  363. package/dist/lib/env-compat.js +0 -23
  364. package/dist/lib/stdin.d.ts +0 -48
  365. package/dist/lib/stdin.js +0 -60
  366. package/dist/templates/_shared/.claude/settings.json +0 -120
  367. package/dist/templates/_shared/.claude/skills/codex/SKILL.md +0 -35
  368. package/dist/templates/_shared/.claude/skills/handoff/SKILL.md +0 -13
  369. package/dist/templates/_shared/.claude/skills/handoff-resume/SKILL.md +0 -13
  370. package/dist/templates/_shared/.claude/skills/meta-plan/SKILL.md +0 -43
  371. package/dist/templates/_shared/.codex/workflows/codex.md +0 -11
  372. package/dist/templates/_shared/.codex/workflows/handoff.md +0 -226
  373. package/dist/templates/_shared/.codex/workflows/meta-plan.md +0 -347
  374. package/dist/templates/_shared/hooks-ts/_utils/git-state.ts +0 -2
  375. package/dist/templates/_shared/hooks-ts/pre_compact.ts +0 -49
  376. package/dist/templates/_shared/hooks-ts/task_create_capture.ts +0 -48
  377. package/dist/templates/_shared/hooks-ts/user_prompt_submit.ts +0 -93
  378. package/dist/templates/_shared/lib-ts/agent-exec/backends/headless.ts +0 -33
  379. package/dist/templates/_shared/lib-ts/agent-exec/backends/index.ts +0 -6
  380. package/dist/templates/_shared/lib-ts/agent-exec/backends/tmux.ts +0 -119
  381. package/dist/templates/_shared/lib-ts/agent-exec/execution-backend.ts +0 -50
  382. package/dist/templates/_shared/lib-ts/agent-exec/index.ts +0 -6
  383. package/dist/templates/_shared/lib-ts/agent-exec/structured-output.ts +0 -166
  384. package/dist/templates/_shared/lib-ts/base/launchers/tmux-launcher.ts +0 -173
  385. package/dist/templates/_shared/lib-ts/base/launchers/window-launcher.ts +0 -93
  386. package/dist/templates/_shared/lib-ts/base/launchers/wt-launcher.ts +0 -64
  387. package/dist/templates/_shared/lib-ts/base/pane-launcher.ts +0 -55
  388. package/dist/templates/_shared/lib-ts/base/sentinel-ipc.ts +0 -87
  389. package/dist/templates/_shared/lib-ts/base/stop-words.ts +0 -184
  390. package/dist/templates/_shared/lib-ts/base/subprocess-utils.ts +0 -249
  391. package/dist/templates/_shared/lib-ts/base/tmux-driver.ts +0 -341
  392. package/dist/templates/_shared/lib-ts/base/tmux-pane-placement.ts +0 -78
  393. package/dist/templates/_shared/lib-ts/package.json +0 -20
  394. package/dist/templates/_shared/scripts/resolve-run.ts +0 -62
  395. package/dist/templates/_shared/skills/codex/CLAUDE.md +0 -70
  396. package/dist/templates/cc-native/_cc-native/CLAUDE.md +0 -73
  397. package/dist/templates/cc-native/_cc-native/artifacts/CLAUDE.md +0 -64
  398. package/dist/templates/cc-native/_cc-native/lib-ts/CLAUDE.md +0 -70
  399. package/dist/templates/cc-native/_cc-native/plan-review/CODING-STANDARDS-CHECKLIST.md +0 -75
  400. package/dist/templates/cc-native/_cc-native/plan-review/agents/CLAUDE.md +0 -143
  401. package/dist/templates/cc-native/_cc-native/plan-review/agents/PLAN-ORCHESTRATOR.md +0 -213
  402. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-questions/PLAN-QUESTIONER.md +0 -70
  403. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/ARCH-EVOLUTION.md +0 -62
  404. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/ARCH-PATTERNS.md +0 -61
  405. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/ARCH-STRUCTURE.md +0 -62
  406. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/ASSUMPTION-TRACER.md +0 -56
  407. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/CLARITY-AUDITOR.md +0 -53
  408. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/COMPLETENESS-FEASIBILITY.md +0 -66
  409. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/COMPLETENESS-GAPS.md +0 -70
  410. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/COMPLETENESS-ORDERING.md +0 -62
  411. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/CONSTRAINT-VALIDATOR.md +0 -72
  412. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/DESIGN-ADR-VALIDATOR.md +0 -61
  413. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/DESIGN-SCALE-MATCHER.md +0 -64
  414. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/DEVILS-ADVOCATE.md +0 -56
  415. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/DOCUMENTATION-PHILOSOPHY.md +0 -86
  416. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/HANDOFF-READINESS.md +0 -59
  417. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/HIDDEN-COMPLEXITY.md +0 -58
  418. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/INCREMENTAL-DELIVERY.md +0 -66
  419. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/RISK-DEPENDENCY.md +0 -62
  420. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/RISK-FMEA.md +0 -66
  421. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/RISK-PREMORTEM.md +0 -71
  422. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/RISK-REVERSIBILITY.md +0 -74
  423. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/SCOPE-BOUNDARY.md +0 -77
  424. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/SIMPLICITY-GUARDIAN.md +0 -62
  425. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/SKEPTIC.md +0 -68
  426. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/TESTDRIVEN-BEHAVIOR-AUDITOR.md +0 -61
  427. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/TESTDRIVEN-CHARACTERIZATION.md +0 -71
  428. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/TESTDRIVEN-FIRST-VALIDATOR.md +0 -61
  429. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/TESTDRIVEN-PYRAMID-ANALYZER.md +0 -61
  430. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/TRADEOFF-COSTS.md +0 -67
  431. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/TRADEOFF-STAKEHOLDERS.md +0 -65
  432. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/VERIFY-COVERAGE.md +0 -74
  433. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/VERIFY-STRENGTH.md +0 -69
  434. package/dist/templates/cc-native/_cc-native/plan-review/lib/corroboration.ts +0 -172
  435. package/dist/templates/cc-native/_cc-native/plan-review/lib/reviewers/base/base-agent.ts +0 -7
@@ -1,599 +1,597 @@
1
- /**
2
- * Pure formatting functions for review artifacts.
3
- * Extracted from artifacts.ts — no file I/O.
4
- */
5
-
6
- import { sanitizeFilename } from "../../../_shared/lib-ts/base/constants.js";
7
- import { DEFAULT_DISPLAY } from "../../lib-ts/types.js";
8
- import type {
9
- CombinedReviewResult,
10
- ReviewerResult,
11
- DisplaySettings,
12
- CorroborationResult,
13
- } from "../../lib-ts/types.js";
14
-
15
- // ---------------------------------------------------------------------------
16
- // Markdown Formatting
17
- // ---------------------------------------------------------------------------
18
-
19
- /**
20
- * Format review results as markdown (legacy compat format).
21
- */
22
- export function formatReviewMarkdown(
23
- results: ReviewerResult[],
24
- overall: string,
25
- title = "CC-Native Plan Review",
26
- settings?: Record<string, unknown>,
27
- ): string {
28
- const display = resolveDisplay(settings);
29
-
30
- const lines: string[] = [];
31
- lines.push(`# ${title}\n`);
32
- lines.push(`**Overall verdict:** \`${overall.toUpperCase()}\`\n`);
33
-
34
- for (const r of results) {
35
- const displayName = r.name === r.name.toLowerCase() ? titleCase(r.name) : r.name;
36
- lines.push(`## ${displayName}\n`);
37
- lines.push(`- ok: \`${r.ok}\``);
38
- lines.push(`- verdict: \`${r.verdict}\``);
39
-
40
- if (r.data && Object.keys(r.data).length > 0) {
41
- const summary = String(r.data.summary ?? "").trim();
42
- if (r.data.summary_source === "default") {
43
- lines.push(`- summary: ⚠️ ${summary} *(reviewer did not return summary)*`);
44
- } else {
45
- lines.push(`- summary: ${summary}`);
46
- }
47
- appendReviewDetails(lines, r.data, display);
48
- } else {
49
- lines.push(`- note: ${r.err || "no structured output"}`);
50
- }
51
- lines.push("");
52
- }
53
-
54
- return lines.join("\n").trim() + "\n";
55
- }
56
-
57
- /**
58
- * Format combined review result as a single markdown document.
59
- */
60
- export function formatCombinedMarkdown(
61
- result: CombinedReviewResult,
62
- settings?: Record<string, unknown>,
63
- corroboration?: CorroborationResult,
64
- ): string {
65
- const display = resolveDisplay(settings);
66
-
67
- const lines: string[] = [];
68
- lines.push("# CC-Native Plan Review\n");
69
- lines.push(`**Overall Verdict:** \`${result.overall_verdict.toUpperCase()}\``);
70
- lines.push(`**Plan Hash:** \`${result.plan_hash}\`\n`);
71
-
72
- // Corroboration summary
73
- if (corroboration) {
74
- lines.push("## Corroboration Analysis\n");
75
- if (corroboration.blocking.length > 0) {
76
- lines.push("### Blocking Dimensions\n");
77
- for (const group of corroboration.blocking) {
78
- lines.push(`- **${group.dimension}**: ${group.agentCount} agents agree (threshold: ≥${group.threshold} agents, ${group.issues.length} issues)`);
79
- }
80
- lines.push("");
81
- }
82
- if (corroboration.solo.length > 0) {
83
- lines.push("### Solo Dimensions (informational)\n");
84
- for (const s of corroboration.solo) {
85
- lines.push(`- **${s.dimension}**: ${s.agentCount} agent${s.agentCount !== 1 ? "s" : ""} (threshold: ≥${s.threshold} agents, not met)`);
86
- }
87
- lines.push("");
88
- }
89
- if (corroboration.unclassified.length > 0) {
90
- lines.push(`> ${corroboration.unclassified.length} issue(s) without dimension classification (unclassified, not blocking)\n`);
91
- }
92
- }
93
-
94
- lines.push("---\n");
95
-
96
- // Orchestration section
97
- if (result.orchestration) {
98
- lines.push("---\n");
99
- lines.push("## Orchestration\n");
100
- lines.push(`- **Complexity:** \`${result.orchestration.complexity}\``);
101
- lines.push(`- **Category:** \`${result.orchestration.category}\``);
102
- const agentsStr =
103
- result.orchestration.selected_agents.length > 0
104
- ? result.orchestration.selected_agents.join(", ")
105
- : "None";
106
- lines.push(`- **Agents Selected:** ${agentsStr}`);
107
- lines.push(`- **Reasoning:** ${result.orchestration.reasoning}`);
108
- if (result.orchestration.skip_reason) {
109
- lines.push(`- **Skip Reason:** ${result.orchestration.skip_reason}`);
110
- }
111
- if (result.orchestration.error) {
112
- lines.push(`- **Error:** ${result.orchestration.error}`);
113
- }
114
- lines.push("");
115
- }
116
-
117
- // Agent Reviews section
118
- if (Object.keys(result.agents).length > 0) {
119
- lines.push("---\n");
120
- lines.push("## Agent Reviews\n");
121
- for (const [name, r] of Object.entries(result.agents)) {
122
- lines.push(`### ${name}\n`);
123
- lines.push(`- verdict: \`${r.verdict}\``);
124
- if (r.data && Object.keys(r.data).length > 0) {
125
- appendSummaryLine(lines, r.data);
126
- appendReviewDetails(lines, r.data, display);
127
- } else if (r.err) {
128
- lines.push(`- error: ${r.err}`);
129
- }
130
- lines.push("");
131
- }
132
- }
133
-
134
- return lines.join("\n").trim() + "\n";
135
- }
136
-
137
- // ---------------------------------------------------------------------------
138
- // Inline Summaries
139
- // ---------------------------------------------------------------------------
140
-
141
- /**
142
- * Build compact inline summary of HIGH-severity findings for additionalContext.
143
- * When corroboration data is provided, annotates issues as [CORROBORATED] or [perspective].
144
- */
145
- export function buildInlineReviewSummary(
146
- combined: CombinedReviewResult,
147
- maxIssues = 5,
148
- maxChars = 800,
149
- corroboration?: CorroborationResult,
150
- ): string {
151
- const allReviewers = Object.values(combined.agents);
152
-
153
- const highIssues: Array<Record<string, unknown>> = [];
154
- for (const r of allReviewers) {
155
- if (!r.data) continue;
156
- const issues = r.data.issues as Array<Record<string, unknown>> | undefined;
157
- if (!issues) continue;
158
- for (const issue of issues) {
159
- if (issue.severity === "high") {
160
- highIssues.push({ ...issue, _reviewer: r.name });
161
- }
162
- }
163
- }
164
-
165
- const parts: string[] = [];
166
-
167
- const issueCount = highIssues.length;
168
- const countSuffix =
169
- issueCount > 0
170
- ? ` (${issueCount} high-severity issue${issueCount !== 1 ? "s" : ""})`
171
- : "";
172
- parts.push(`**Plan Review: ${combined.overall_verdict.toUpperCase()}**${countSuffix}`);
173
-
174
- if (corroboration) {
175
- const blockCount = corroboration.blocking.length;
176
- const soloCount = corroboration.solo.length;
177
- if (blockCount > 0) {
178
- parts.push(`**Corroboration:** ${blockCount} dimension${blockCount !== 1 ? "s" : ""} exceeded threshold (blocking), ${soloCount} solo (informational)`);
179
- } else {
180
- parts.push(`**Corroboration:** No dimensions exceeded threshold — all ${soloCount} solo (informational)`);
181
- }
182
- }
183
-
184
- for (const issue of highIssues.slice(0, maxIssues)) {
185
- const cat = (issue.category as string) ?? "general";
186
- const text = (issue.issue as string) ?? "";
187
- const fix = (issue.suggested_fix as string) ?? "";
188
- const reviewer = (issue._reviewer as string) ?? "unknown";
189
- const dim = issue.dimension as string | undefined;
190
-
191
- let annotation = "";
192
- if (corroboration && dim) {
193
- const group = corroboration.blocking.find(g => g.dimension === dim);
194
- if (group) {
195
- annotation = ` [CORROBORATED — ${group.agentCount} agents agree, threshold ≥${group.threshold}]`;
196
- } else {
197
- annotation = " [perspective]";
198
- }
199
- }
200
-
201
- let line = `- [${cat}] ${text}`;
202
- if (fix) line += ` \u2192 ${fix}`;
203
- line += ` (${reviewer})${annotation}`;
204
- parts.push(line);
205
- }
206
-
207
- if (maxIssues > 0) {
208
- const remaining = highIssues.length - maxIssues;
209
- if (remaining > 0) {
210
- parts.push(` ...and ${remaining} more`);
211
- }
212
- }
213
-
214
- let result = parts.join("\n");
215
- if (result.length > maxChars) {
216
- result = result.slice(0, maxChars - 3) + "...";
217
- }
218
- return result;
219
- }
220
-
221
- /**
222
- * Extract top issues as compact text for permissionDecisionReason.
223
- */
224
- export function extractTopIssuesText(
225
- combined: CombinedReviewResult,
226
- maxCount = 3,
227
- severity = "high",
228
- ): string {
229
- const allReviewers = Object.values(combined.agents);
230
-
231
- const issues: string[] = [];
232
- for (const r of allReviewers) {
233
- if (!r.data) continue;
234
- const issueList = r.data.issues as Array<Record<string, unknown>> | undefined;
235
- if (!issueList) continue;
236
- for (const issue of issueList) {
237
- if (issue.severity === severity) {
238
- const text = String(issue.issue ?? "").trim();
239
- if (text) {
240
- issues.push(`[${r.name}] ${text}`);
241
- }
242
- }
243
- if (issues.length >= maxCount) break;
244
- }
245
- if (issues.length >= maxCount) break;
246
- }
247
-
248
- if (issues.length === 0) return "Review found critical issues";
249
- return issues.map(i => `- ${i}`).join("\n");
250
- }
251
-
252
- /**
253
- * Build markdown document containing high-severity issues.
254
- * When corroboration data is provided, only includes corroborated (blocking) issues.
255
- */
256
- export function buildHighIssuesDocument(
257
- combined: CombinedReviewResult,
258
- corroboration?: CorroborationResult,
259
- ): string {
260
- if (corroboration && corroboration.blocking.length > 0) {
261
- const lines = ["# Corroborated High-Severity Issues\n"];
262
- lines.push("> Only issues from dimensions where enough distinct agents independently agreed are shown.\n");
263
-
264
- for (const group of corroboration.blocking) {
265
- lines.push(`## ${group.dimension} (${group.agentCount} agents agree, threshold: ≥${group.threshold} agents, ${group.issues.length} issues)\n`);
266
- for (const { agent, issue } of group.issues) {
267
- const cat = issue.category ?? "general";
268
- const text = String(issue.issue ?? "").trim();
269
- const fix = String(issue.suggested_fix ?? "").trim();
270
- lines.push(`- **[${cat}]** ${text} *(${agent})*`);
271
- if (fix) lines.push(` - Fix: ${fix}`);
272
- }
273
- lines.push("");
274
- }
275
-
276
- if (corroboration.solo.length > 0) {
277
- lines.push("---\n");
278
- lines.push(`> ${corroboration.solo.length} dimension${corroboration.solo.length !== 1 ? "s" : ""} had insufficient agent agreement (not blocking): ${corroboration.solo.map(s => `${s.dimension} (${s.agentCount}/${s.threshold} agents)`).join(", ")}\n`);
279
- }
280
-
281
- return lines.join("\n");
282
- }
283
-
284
- const lines = ["# High-Severity Issues\n"];
285
- const allReviewers = Object.values(combined.agents);
286
-
287
- let foundAny = false;
288
- for (const r of allReviewers) {
289
- if (!r.data) continue;
290
- const issues = r.data.issues as Array<Record<string, unknown>> | undefined;
291
- if (!issues) continue;
292
-
293
- const highIssues = issues.filter((i) => i.severity === "high");
294
- if (highIssues.length === 0) continue;
295
-
296
- foundAny = true;
297
- lines.push(`## ${r.name} (${r.verdict})\n`);
298
- for (const issue of highIssues) {
299
- const cat = (issue.category as string) ?? "general";
300
- const text = String(issue.issue ?? "").trim();
301
- const fix = String(issue.suggested_fix ?? "").trim();
302
- lines.push(`- **[${cat}]** ${text}`);
303
- if (fix) lines.push(` - Fix: ${fix}`);
304
- }
305
- lines.push("");
306
- }
307
-
308
- if (!foundAny) {
309
- lines.push("No high-severity issues found.\n");
310
- }
311
-
312
- return lines.join("\n");
313
- }
314
-
315
- // ---------------------------------------------------------------------------
316
- // Corroboration Report
317
- // ---------------------------------------------------------------------------
318
-
319
- /**
320
- * Build a detailed markdown report of the corroboration analysis.
321
- */
322
- export function buildCorroborationReport(
323
- corroborationResult: CorroborationResult,
324
- ): string {
325
- const lines: string[] = [
326
- "# Corroboration Analysis",
327
- "",
328
- "## Verdict: " + corroborationResult.verdict.toUpperCase(),
329
- "",
330
- ];
331
-
332
- if (corroborationResult.blocking.length > 0) {
333
- lines.push("## Blocking Issues (Corroborated)");
334
- lines.push("");
335
- lines.push("| Dimension | Agents Agreeing | Threshold | Issues | Status |");
336
- lines.push("|-----------|----------------|-----------|--------|--------|");
337
-
338
- for (const group of corroborationResult.blocking) {
339
- lines.push(
340
- `| ${group.dimension} | ${group.agentCount} | ≥${group.threshold} | ${group.issues.length} | ⛔ CORROBORATED |`
341
- );
342
- }
343
- lines.push("");
344
-
345
- for (const group of corroborationResult.blocking) {
346
- lines.push(`### ${group.dimension} (${group.agentCount} agents, ${group.issues.length} issues)`);
347
- lines.push("");
348
- for (const {agent, issue} of group.issues) {
349
- lines.push(`- **[${agent}]** ${issue.issue || "No description"}`);
350
- }
351
- lines.push("");
352
- }
353
- }
354
-
355
- if (corroborationResult.solo.length > 0) {
356
- lines.push("## Solo Findings (Below Threshold)");
357
- lines.push("");
358
- lines.push("| Dimension | Agents Agreeing | Threshold | Issues | Status |");
359
- lines.push("|-----------|----------------|-----------|--------|--------|");
360
-
361
- for (const group of corroborationResult.solo) {
362
- lines.push(
363
- `| ${group.dimension} | ${group.agentCount} | ≥${group.threshold} | ${group.issues.length} | ℹ️ SOLO |`
364
- );
365
- }
366
- lines.push("");
367
-
368
- for (const group of corroborationResult.solo) {
369
- lines.push(`### ${group.dimension} (${group.issues.length} issues)`);
370
- lines.push("");
371
- for (const {agent, issue} of group.issues) {
372
- lines.push(`- **[${agent}]** ${issue.issue || "No description"}`);
373
- }
374
- lines.push("");
375
- }
376
- }
377
-
378
- if (corroborationResult.unclassified.length > 0) {
379
- lines.push("## Unclassified Issues (No Dimension)");
380
- lines.push("");
381
- for (const {agent, issue} of corroborationResult.unclassified) {
382
- lines.push(`- **[${agent}]** ${issue.issue || "No description"}`);
383
- }
384
- lines.push("");
385
- }
386
-
387
- lines.push("## Summary");
388
- lines.push("");
389
- lines.push(`- **Blocking groups**: ${corroborationResult.blocking.length}`);
390
- lines.push(`- **Solo findings**: ${corroborationResult.solo.length}`);
391
- lines.push(`- **Unclassified**: ${corroborationResult.unclassified.length}`);
392
- lines.push(`- **Final verdict**: ${corroborationResult.verdict}`);
393
- lines.push("");
394
- lines.push("**Threshold rule**: Issues in a dimension block when count ≥ 2× distinct agents in that dimension.");
395
-
396
- return lines.join("\n");
397
- }
398
-
399
- // ---------------------------------------------------------------------------
400
- // Index Generation
401
- // ---------------------------------------------------------------------------
402
-
403
- /**
404
- * Generate index.md for a review folder.
405
- */
406
- export function generateReviewIndex(
407
- result: CombinedReviewResult,
408
- iteration?: number,
409
- _settings?: Record<string, unknown>,
410
- ): string {
411
- const now = new Date();
412
-
413
- const lines = [
414
- "---",
415
- `type: review`,
416
- `plan_hash: ${result.plan_hash}`,
417
- `overall_verdict: ${result.overall_verdict}`,
418
- `created_at: ${result.timestamp}`,
419
- ];
420
- if (iteration) lines.push(`iteration: ${iteration}`);
421
- lines.push(
422
- "---",
423
- "",
424
- `# Plan Review - ${formatDate(now)}`,
425
- "",
426
- `**Overall Verdict:** \`${result.overall_verdict.toUpperCase()}\``,
427
- );
428
-
429
- if (iteration) lines.push(`**Iteration:** ${iteration}`);
430
- lines.push(`**Plan Hash:** \`${result.plan_hash}\``, "");
431
-
432
- if (result.orchestration) {
433
- lines.push(
434
- "## Analysis",
435
- `- **Complexity:** \`${result.orchestration.complexity}\``,
436
- `- **Category:** \`${result.orchestration.category}\``,
437
- `- **Reasoning:** ${result.orchestration.reasoning}`,
438
- "",
439
- );
440
- }
441
-
442
- lines.push(
443
- "## Review Files",
444
- "",
445
- "| File | Description |",
446
- "|------|-------------|",
447
- "| [review.md](./review.md) | Full review details |",
448
- "| [review.json](./review.json) | Structured review data |",
449
- "| [plan.md](./plan.md) | Plan snapshot at review time |",
450
- );
451
-
452
- for (const name of Object.keys(result.agents)) {
453
- const safeName = sanitizeFilename(name);
454
- lines.push(
455
- `| [${safeName}.json](./reviewer-output/${safeName}.json) | ${name} agent output |`,
456
- );
457
- }
458
-
459
- lines.push(
460
- "",
461
- "## Verdicts Summary",
462
- "",
463
- "| Reviewer | Verdict |",
464
- "|----------|---------|",
465
- );
466
-
467
- for (const [name, r] of Object.entries(result.agents)) {
468
- lines.push(`| ${name} | \`${r.verdict}\` |`);
469
- }
470
- lines.push("");
471
-
472
- return lines.join("\n");
473
- }
474
-
475
- // ---------------------------------------------------------------------------
476
- // JSON Output
477
- // ---------------------------------------------------------------------------
478
-
479
- /**
480
- * Build combined JSON output structure.
481
- */
482
- export function buildCombinedJson(
483
- result: CombinedReviewResult,
484
- ): Record<string, unknown> {
485
- const output: Record<string, unknown> = {
486
- metadata: {
487
- timestamp: result.timestamp,
488
- plan_hash: result.plan_hash,
489
- },
490
- overall: {
491
- verdict: result.overall_verdict,
492
- },
493
- };
494
-
495
- if (result.orchestration) {
496
- output.orchestration = {
497
- complexity: result.orchestration.complexity,
498
- category: result.orchestration.category,
499
- selectedAgents: result.orchestration.selected_agents,
500
- reasoning: result.orchestration.reasoning,
501
- skipReason: result.orchestration.skip_reason ?? null,
502
- error: result.orchestration.error ?? null,
503
- };
504
- }
505
-
506
- if (Object.keys(result.agents).length > 0) {
507
- const agents: Record<string, unknown> = {};
508
- output.agents = agents;
509
- for (const [name, r] of Object.entries(result.agents)) {
510
- agents[name] = {
511
- verdict: r.verdict,
512
- summary: r.data?.summary ?? null,
513
- summarySource: r.data?.summary_source ?? null,
514
- issues: r.data
515
- ? ((r.data.issues as Array<Record<string, unknown>>) ?? []).filter(
516
- (i) => i.severity !== "low",
517
- )
518
- : [],
519
- missing_sections: r.data?.missing_sections ?? [],
520
- questions: r.data?.questions ?? [],
521
- ok: r.ok,
522
- error: r.err || null,
523
- };
524
- }
525
- }
526
-
527
- return output;
528
- }
529
-
530
- // ---------------------------------------------------------------------------
531
- // Helpers (private)
532
- // ---------------------------------------------------------------------------
533
-
534
- function resolveDisplay(
535
- settings?: Record<string, unknown>,
536
- ): DisplaySettings {
537
- if (!settings) return { ...DEFAULT_DISPLAY };
538
- const display = (settings.display as Partial<DisplaySettings>) ?? {};
539
- return { ...DEFAULT_DISPLAY, ...display };
540
- }
541
-
542
- function appendSummaryLine(lines: string[], data: Record<string, unknown>): void {
543
- const summary = String(data.summary ?? "").trim();
544
- if (data.summary_source === "default") {
545
- lines.push(`- summary: ⚠️ ${summary} *(reviewer did not return summary)*`);
546
- } else {
547
- lines.push(`- summary: ${summary}`);
548
- }
549
- }
550
-
551
- function appendReviewDetails(
552
- lines: string[],
553
- data: Record<string, unknown>,
554
- display: DisplaySettings,
555
- ): void {
556
- const issues = ((data.issues as Array<Record<string, unknown>>) ?? []).filter(
557
- (i) => i.severity !== "low",
558
- );
559
- if (issues.length > 0) {
560
- lines.push("\n**Issues:**");
561
- for (const it of issues.slice(0, display.maxIssues)) {
562
- const sev = (it.severity as string) ?? "medium";
563
- const cat = (it.category as string) ?? "general";
564
- const issue = (it.issue as string) ?? "";
565
- const fix = (it.suggested_fix as string) ?? "";
566
- lines.push(`- **[${sev}] ${cat}**: ${issue}`);
567
- if (fix) lines.push(` - fix: ${fix}`);
568
- }
569
- }
570
-
571
- const missing = (data.missing_sections as string[]) ?? [];
572
- if (missing.length > 0) {
573
- lines.push("\n**Missing Sections:**");
574
- for (const m of missing.slice(0, display.maxMissingSections)) {
575
- lines.push(`- ${m}`);
576
- }
577
- }
578
-
579
- const qs = (data.questions as string[]) ?? [];
580
- if (qs.length > 0) {
581
- lines.push("\n**Questions:**");
582
- for (const q of qs.slice(0, display.maxQuestions)) {
583
- lines.push(`- ${q}`);
584
- }
585
- }
586
- }
587
-
588
- function titleCase(s: string): string {
589
- return s.charAt(0).toUpperCase() + s.slice(1);
590
- }
591
-
592
- function formatDate(d: Date): string {
593
- const year = d.getFullYear();
594
- const month = String(d.getMonth() + 1).padStart(2, "0");
595
- const day = String(d.getDate()).padStart(2, "0");
596
- const hours = String(d.getHours()).padStart(2, "0");
597
- const minutes = String(d.getMinutes()).padStart(2, "0");
598
- return `${year}-${month}-${day} ${hours}:${minutes}`;
599
- }
1
+ /**
2
+ * Pure formatting functions for review artifacts.
3
+ * Extracted from artifacts.ts — no file I/O.
4
+ */
5
+
6
+ import { sanitizeFilename } from "../../../_core/lib-ts/runtime/constants.js";
7
+ import type {
8
+ CombinedReviewResult,
9
+ ReviewerResult,
10
+ DisplaySettings,
11
+ CorroborationResult,
12
+ } from "../types.js";
13
+ import { DEFAULT_DISPLAY } from "../types.js";
14
+
15
+ // ---------------------------------------------------------------------------
16
+ // Markdown Formatting
17
+ // ---------------------------------------------------------------------------
18
+
19
+ /**
20
+ * Format review results as markdown (legacy compat format).
21
+ */
22
+ export function formatReviewMarkdown(
23
+ results: ReviewerResult[],
24
+ overall: string,
25
+ title = "CC-Native Plan Review",
26
+ settings?: Record<string, unknown>,
27
+ ): string {
28
+ const display = resolveDisplay(settings);
29
+
30
+ const lines: string[] = [];
31
+ lines.push(`# ${title}\n`);
32
+ lines.push(`**Overall verdict:** \`${overall.toUpperCase()}\`\n`);
33
+
34
+ for (const r of results) {
35
+ const displayName = r.name === r.name.toLowerCase() ? titleCase(r.name) : r.name;
36
+ lines.push(`## ${displayName}\n`);
37
+ lines.push(`- ok: \`${r.ok}\``);
38
+ lines.push(`- verdict: \`${r.verdict}\``);
39
+
40
+ if (r.data && Object.keys(r.data).length > 0) {
41
+ const summary = String(r.data.summary ?? "").trim();
42
+ if (r.data.summary_source === "default") {
43
+ lines.push(`- summary: ⚠️ ${summary} *(reviewer did not return summary)*`);
44
+ } else {
45
+ lines.push(`- summary: ${summary}`);
46
+ }
47
+ appendReviewDetails(lines, r.data, display);
48
+ } else {
49
+ lines.push(`- note: ${r.err || "no structured output"}`);
50
+ }
51
+ lines.push("");
52
+ }
53
+
54
+ return lines.join("\n").trim() + "\n";
55
+ }
56
+
57
+ /**
58
+ * Format combined review result as a single markdown document.
59
+ */
60
+ export function formatCombinedMarkdown(
61
+ result: CombinedReviewResult,
62
+ settings?: Record<string, unknown>,
63
+ corroboration?: CorroborationResult,
64
+ ): string {
65
+ const display = resolveDisplay(settings);
66
+
67
+ const lines: string[] = [];
68
+ lines.push("# CC-Native Plan Review\n");
69
+ lines.push(`**Overall Verdict:** \`${result.overall_verdict.toUpperCase()}\``);
70
+ lines.push(`**Plan Hash:** \`${result.plan_hash}\`\n`);
71
+
72
+ // Corroboration summary
73
+ if (corroboration) {
74
+ lines.push("## Corroboration Analysis\n");
75
+ if (corroboration.blocking.length > 0) {
76
+ lines.push("### Blocking Dimensions\n");
77
+ for (const group of corroboration.blocking) {
78
+ lines.push(`- **${group.dimension}**: ${group.issues.length} issues from ${group.agentCount} agents (threshold: ≥${group.threshold})`);
79
+ }
80
+ lines.push("");
81
+ }
82
+ if (corroboration.solo.length > 0) {
83
+ lines.push("### Solo Dimensions (informational)\n");
84
+ for (const s of corroboration.solo) {
85
+ lines.push(`- **${s.dimension}**: ${s.issues.length} issues from ${s.agentCount} agents (threshold: >${s.threshold}, not exceeded)`);
86
+ }
87
+ lines.push("");
88
+ }
89
+ if (corroboration.unclassified.length > 0) {
90
+ lines.push(`> ${corroboration.unclassified.length} issue(s) without dimension classification (unclassified, not blocking)\n`);
91
+ }
92
+ }
93
+
94
+ lines.push("---\n");
95
+
96
+ // Orchestration section
97
+ if (result.orchestration) {
98
+ lines.push("---\n");
99
+ lines.push("## Orchestration\n");
100
+ lines.push(`- **Complexity:** \`${result.orchestration.complexity}\``);
101
+ lines.push(`- **Category:** \`${result.orchestration.category}\``);
102
+ const agentsStr =
103
+ result.orchestration.selected_agents.length > 0
104
+ ? result.orchestration.selected_agents.join(", ")
105
+ : "None";
106
+ lines.push(`- **Agents Selected:** ${agentsStr}`);
107
+ lines.push(`- **Reasoning:** ${result.orchestration.reasoning}`);
108
+ if (result.orchestration.skip_reason) {
109
+ lines.push(`- **Skip Reason:** ${result.orchestration.skip_reason}`);
110
+ }
111
+ if (result.orchestration.error) {
112
+ lines.push(`- **Error:** ${result.orchestration.error}`);
113
+ }
114
+ lines.push("");
115
+ }
116
+
117
+ // Agent Reviews section
118
+ if (Object.keys(result.agents).length > 0) {
119
+ lines.push("---\n");
120
+ lines.push("## Agent Reviews\n");
121
+ for (const [name, r] of Object.entries(result.agents)) {
122
+ lines.push(`### ${name}\n`);
123
+ lines.push(`- verdict: \`${r.verdict}\``);
124
+ if (r.data && Object.keys(r.data).length > 0) {
125
+ appendSummaryLine(lines, r.data);
126
+ appendReviewDetails(lines, r.data, display);
127
+ } else if (r.err) {
128
+ lines.push(`- error: ${r.err}`);
129
+ }
130
+ lines.push("");
131
+ }
132
+ }
133
+
134
+ return lines.join("\n").trim() + "\n";
135
+ }
136
+
137
+ // ---------------------------------------------------------------------------
138
+ // Inline Summaries
139
+ // ---------------------------------------------------------------------------
140
+
141
+ /**
142
+ * Build compact inline summary of HIGH-severity findings for additionalContext.
143
+ * When corroboration data is provided, annotates issues as [CORROBORATED] or [perspective].
144
+ */
145
+ export function buildInlineReviewSummary(
146
+ combined: CombinedReviewResult,
147
+ maxIssues = 5,
148
+ maxChars = 800,
149
+ corroboration?: CorroborationResult,
150
+ ): string {
151
+ const allReviewers = Object.values(combined.agents);
152
+
153
+ const highIssues: Array<Record<string, unknown>> = [];
154
+ for (const r of allReviewers) {
155
+ if (!r.data) continue;
156
+ const issues = r.data.issues as Array<Record<string, unknown>> | undefined;
157
+ if (!issues) continue;
158
+ for (const issue of issues) {
159
+ if (issue.severity === "high") {
160
+ highIssues.push({ ...issue, _reviewer: r.name });
161
+ }
162
+ }
163
+ }
164
+
165
+ const parts: string[] = [];
166
+
167
+ const issueCount = highIssues.length;
168
+ const countSuffix =
169
+ issueCount > 0
170
+ ? ` (${issueCount} high-severity issue${issueCount !== 1 ? "s" : ""})`
171
+ : "";
172
+ parts.push(`**Plan Review: ${combined.overall_verdict.toUpperCase()}**${countSuffix}`);
173
+
174
+ if (corroboration) {
175
+ const blockCount = corroboration.blocking.length;
176
+ const soloCount = corroboration.solo.length;
177
+ if (blockCount > 0) {
178
+ parts.push(`**Corroboration:** ${blockCount} dimension${blockCount !== 1 ? "s" : ""} exceeded threshold (blocking), ${soloCount} solo (informational)`);
179
+ } else {
180
+ parts.push(`**Corroboration:** No dimensions exceeded threshold — all ${soloCount} solo (informational)`);
181
+ }
182
+ }
183
+
184
+ for (const issue of highIssues.slice(0, maxIssues)) {
185
+ const cat = (issue.category as string) ?? "general";
186
+ const text = (issue.issue as string) ?? "";
187
+ const fix = (issue.suggested_fix as string) ?? "";
188
+ const reviewer = (issue._reviewer as string) ?? "unknown";
189
+ const dim = issue.dimension as string | undefined;
190
+
191
+ let annotation = "";
192
+ if (corroboration && dim) {
193
+ const group = corroboration.blocking.find(g => g.dimension === dim);
194
+ if (group) {
195
+ annotation = ` [CORROBORATED — ${group.issues.length} issues from ${group.agentCount} agents exceeds threshold ${group.threshold}]`;
196
+ } else {
197
+ annotation = " [perspective]";
198
+ }
199
+ }
200
+
201
+ let line = `- [${cat}] ${text}`;
202
+ if (fix) line += ` \u2192 ${fix}`;
203
+ line += ` (${reviewer})${annotation}`;
204
+ parts.push(line);
205
+ }
206
+
207
+ const remaining = highIssues.length - maxIssues;
208
+ if (remaining > 0) {
209
+ parts.push(` ...and ${remaining} more`);
210
+ }
211
+
212
+ let result = parts.join("\n");
213
+ if (result.length > maxChars) {
214
+ result = result.slice(0, maxChars - 3) + "...";
215
+ }
216
+ return result;
217
+ }
218
+
219
+ /**
220
+ * Extract top issues as compact text for permissionDecisionReason.
221
+ */
222
+ export function extractTopIssuesText(
223
+ combined: CombinedReviewResult,
224
+ maxCount = 3,
225
+ severity = "high",
226
+ ): string {
227
+ const allReviewers = Object.values(combined.agents);
228
+
229
+ const issues: string[] = [];
230
+ for (const r of allReviewers) {
231
+ if (!r.data) continue;
232
+ const issueList = r.data.issues as Array<Record<string, unknown>> | undefined;
233
+ if (!issueList) continue;
234
+ for (const issue of issueList) {
235
+ if (issue.severity === severity) {
236
+ const text = String(issue.issue ?? "").trim();
237
+ if (text) {
238
+ issues.push(`[${r.name}] ${text}`);
239
+ break; // first high issue per reviewer only
240
+ }
241
+ }
242
+ }
243
+ if (issues.length >= maxCount) break;
244
+ }
245
+
246
+ if (issues.length === 0) return "Review found critical issues";
247
+ return issues.join("; ");
248
+ }
249
+
250
+ /**
251
+ * Build markdown document containing high-severity issues.
252
+ * When corroboration data is provided, only includes corroborated (blocking) issues.
253
+ */
254
+ export function buildHighIssuesDocument(
255
+ combined: CombinedReviewResult,
256
+ corroboration?: CorroborationResult,
257
+ ): string {
258
+ if (corroboration && corroboration.blocking.length > 0) {
259
+ const lines = ["# Corroborated High-Severity Issues\n"];
260
+ lines.push("> Only issues from dimensions where the total count exceeded the proportional threshold are shown.\n");
261
+
262
+ for (const group of corroboration.blocking) {
263
+ lines.push(`## ${group.dimension} (${group.issues.length} issues from ${group.agentCount} agents, threshold: ${group.threshold})\n`);
264
+ for (const { agent, issue } of group.issues) {
265
+ const cat = issue.category ?? "general";
266
+ const text = String(issue.issue ?? "").trim();
267
+ const fix = String(issue.suggested_fix ?? "").trim();
268
+ lines.push(`- **[${cat}]** ${text} *(${agent})*`);
269
+ if (fix) lines.push(` - Fix: ${fix}`);
270
+ }
271
+ lines.push("");
272
+ }
273
+
274
+ if (corroboration.solo.length > 0) {
275
+ lines.push("---\n");
276
+ lines.push(`> ${corroboration.solo.length} dimension${corroboration.solo.length !== 1 ? "s" : ""} had issues below threshold (not blocking): ${corroboration.solo.map(s => `${s.dimension} (${s.issues.length}/${s.threshold})`).join(", ")}\n`);
277
+ }
278
+
279
+ return lines.join("\n");
280
+ }
281
+
282
+ const lines = ["# High-Severity Issues\n"];
283
+ const allReviewers = Object.values(combined.agents);
284
+
285
+ let foundAny = false;
286
+ for (const r of allReviewers) {
287
+ if (!r.data) continue;
288
+ const issues = r.data.issues as Array<Record<string, unknown>> | undefined;
289
+ if (!issues) continue;
290
+
291
+ const highIssues = issues.filter((i) => i.severity === "high");
292
+ if (highIssues.length === 0) continue;
293
+
294
+ foundAny = true;
295
+ lines.push(`## ${r.name} (${r.verdict})\n`);
296
+ for (const issue of highIssues) {
297
+ const cat = (issue.category as string) ?? "general";
298
+ const text = String(issue.issue ?? "").trim();
299
+ const fix = String(issue.suggested_fix ?? "").trim();
300
+ lines.push(`- **[${cat}]** ${text}`);
301
+ if (fix) lines.push(` - Fix: ${fix}`);
302
+ }
303
+ lines.push("");
304
+ }
305
+
306
+ if (!foundAny) {
307
+ lines.push("No high-severity issues found.\n");
308
+ }
309
+
310
+ return lines.join("\n");
311
+ }
312
+
313
+ // ---------------------------------------------------------------------------
314
+ // Corroboration Report
315
+ // ---------------------------------------------------------------------------
316
+
317
+ /**
318
+ * Build a detailed markdown report of the corroboration analysis.
319
+ */
320
+ export function buildCorroborationReport(
321
+ corroborationResult: CorroborationResult,
322
+ ): string {
323
+ const lines: string[] = [
324
+ "# Corroboration Analysis",
325
+ "",
326
+ "## Verdict: " + corroborationResult.verdict.toUpperCase(),
327
+ "",
328
+ ];
329
+
330
+ if (corroborationResult.blocking.length > 0) {
331
+ lines.push("## Blocking Issues (Corroborated)");
332
+ lines.push("");
333
+ lines.push("| Dimension | Issues | Agents | Threshold | Status |");
334
+ lines.push("|-----------|--------|--------|-----------|--------|");
335
+
336
+ for (const group of corroborationResult.blocking) {
337
+ lines.push(
338
+ `| ${group.dimension} | ${group.issues.length} | ${group.agentCount} | ${group.threshold} | ⛔ EXCEEDED |`
339
+ );
340
+ }
341
+ lines.push("");
342
+
343
+ for (const group of corroborationResult.blocking) {
344
+ lines.push(`### ${group.dimension} (${group.issues.length} issues)`);
345
+ lines.push("");
346
+ for (const {agent, issue} of group.issues) {
347
+ lines.push(`- **[${agent}]** ${issue.issue || "No description"}`);
348
+ }
349
+ lines.push("");
350
+ }
351
+ }
352
+
353
+ if (corroborationResult.solo.length > 0) {
354
+ lines.push("## Solo Findings (Below Threshold)");
355
+ lines.push("");
356
+ lines.push("| Dimension | Issues | Agents | Threshold | Status |");
357
+ lines.push("|-----------|--------|--------|-----------|--------|");
358
+
359
+ for (const group of corroborationResult.solo) {
360
+ lines.push(
361
+ `| ${group.dimension} | ${group.issues.length} | ${group.agentCount} | ${group.threshold} | ℹ️ SOLO |`
362
+ );
363
+ }
364
+ lines.push("");
365
+
366
+ for (const group of corroborationResult.solo) {
367
+ lines.push(`### ${group.dimension} (${group.issues.length} issues)`);
368
+ lines.push("");
369
+ for (const {agent, issue} of group.issues) {
370
+ lines.push(`- **[${agent}]** ${issue.issue || "No description"}`);
371
+ }
372
+ lines.push("");
373
+ }
374
+ }
375
+
376
+ if (corroborationResult.unclassified.length > 0) {
377
+ lines.push("## Unclassified Issues (No Dimension)");
378
+ lines.push("");
379
+ for (const {agent, issue} of corroborationResult.unclassified) {
380
+ lines.push(`- **[${agent}]** ${issue.issue || "No description"}`);
381
+ }
382
+ lines.push("");
383
+ }
384
+
385
+ lines.push("## Summary");
386
+ lines.push("");
387
+ lines.push(`- **Blocking groups**: ${corroborationResult.blocking.length}`);
388
+ lines.push(`- **Solo findings**: ${corroborationResult.solo.length}`);
389
+ lines.push(`- **Unclassified**: ${corroborationResult.unclassified.length}`);
390
+ lines.push(`- **Final verdict**: ${corroborationResult.verdict}`);
391
+ lines.push("");
392
+ lines.push("**Threshold rule**: Issues in a dimension block when count ≥ 2× distinct agents in that dimension.");
393
+
394
+ return lines.join("\n");
395
+ }
396
+
397
+ // ---------------------------------------------------------------------------
398
+ // Index Generation
399
+ // ---------------------------------------------------------------------------
400
+
401
+ /**
402
+ * Generate index.md for a review folder.
403
+ */
404
+ export function generateReviewIndex(
405
+ result: CombinedReviewResult,
406
+ iteration?: number,
407
+ _settings?: Record<string, unknown>,
408
+ ): string {
409
+ const now = new Date();
410
+
411
+ const lines = [
412
+ "---",
413
+ `type: review`,
414
+ `plan_hash: ${result.plan_hash}`,
415
+ `overall_verdict: ${result.overall_verdict}`,
416
+ `created_at: ${result.timestamp}`,
417
+ ];
418
+ if (iteration) lines.push(`iteration: ${iteration}`);
419
+ lines.push(
420
+ "---",
421
+ "",
422
+ `# Plan Review - ${formatDate(now)}`,
423
+ "",
424
+ `**Overall Verdict:** \`${result.overall_verdict.toUpperCase()}\``,
425
+ );
426
+
427
+ if (iteration) lines.push(`**Iteration:** ${iteration}`);
428
+ lines.push(`**Plan Hash:** \`${result.plan_hash}\``, "");
429
+
430
+ if (result.orchestration) {
431
+ lines.push(
432
+ "## Analysis",
433
+ `- **Complexity:** \`${result.orchestration.complexity}\``,
434
+ `- **Category:** \`${result.orchestration.category}\``,
435
+ `- **Reasoning:** ${result.orchestration.reasoning}`,
436
+ "",
437
+ );
438
+ }
439
+
440
+ lines.push(
441
+ "## Review Files",
442
+ "",
443
+ "| File | Description |",
444
+ "|------|-------------|",
445
+ "| [review.md](./review.md) | Full review details |",
446
+ "| [review.json](./review.json) | Structured review data |",
447
+ "| [plan.md](./plan.md) | Plan snapshot at review time |",
448
+ );
449
+
450
+ for (const name of Object.keys(result.agents)) {
451
+ const safeName = sanitizeFilename(name);
452
+ lines.push(
453
+ `| [${safeName}.json](./reviewer-output/${safeName}.json) | ${name} agent output |`,
454
+ );
455
+ }
456
+
457
+ lines.push(
458
+ "",
459
+ "## Verdicts Summary",
460
+ "",
461
+ "| Reviewer | Verdict |",
462
+ "|----------|---------|",
463
+ );
464
+
465
+ for (const [name, r] of Object.entries(result.agents)) {
466
+ lines.push(`| ${name} | \`${r.verdict}\` |`);
467
+ }
468
+ lines.push("");
469
+
470
+ return lines.join("\n");
471
+ }
472
+
473
+ // ---------------------------------------------------------------------------
474
+ // JSON Output
475
+ // ---------------------------------------------------------------------------
476
+
477
+ /**
478
+ * Build combined JSON output structure.
479
+ */
480
+ export function buildCombinedJson(
481
+ result: CombinedReviewResult,
482
+ ): Record<string, unknown> {
483
+ const output: Record<string, unknown> = {
484
+ metadata: {
485
+ timestamp: result.timestamp,
486
+ plan_hash: result.plan_hash,
487
+ },
488
+ overall: {
489
+ verdict: result.overall_verdict,
490
+ },
491
+ };
492
+
493
+ if (result.orchestration) {
494
+ output.orchestration = {
495
+ complexity: result.orchestration.complexity,
496
+ category: result.orchestration.category,
497
+ selectedAgents: result.orchestration.selected_agents,
498
+ reasoning: result.orchestration.reasoning,
499
+ skipReason: result.orchestration.skip_reason ?? null,
500
+ error: result.orchestration.error ?? null,
501
+ };
502
+ }
503
+
504
+ if (Object.keys(result.agents).length > 0) {
505
+ const agents: Record<string, unknown> = {};
506
+ output.agents = agents;
507
+ for (const [name, r] of Object.entries(result.agents)) {
508
+ agents[name] = {
509
+ verdict: r.verdict,
510
+ summary: r.data?.summary ?? null,
511
+ summarySource: r.data?.summary_source ?? null,
512
+ issues: r.data
513
+ ? ((r.data.issues as Array<Record<string, unknown>>) ?? []).filter(
514
+ (i) => i.severity !== "low",
515
+ )
516
+ : [],
517
+ missing_sections: r.data?.missing_sections ?? [],
518
+ questions: r.data?.questions ?? [],
519
+ ok: r.ok,
520
+ error: r.err || null,
521
+ };
522
+ }
523
+ }
524
+
525
+ return output;
526
+ }
527
+
528
+ // ---------------------------------------------------------------------------
529
+ // Helpers (private)
530
+ // ---------------------------------------------------------------------------
531
+
532
+ function resolveDisplay(
533
+ settings?: Record<string, unknown>,
534
+ ): DisplaySettings {
535
+ if (!settings) return { ...DEFAULT_DISPLAY };
536
+ const display = (settings.display as Partial<DisplaySettings>) ?? {};
537
+ return { ...DEFAULT_DISPLAY, ...display };
538
+ }
539
+
540
+ function appendSummaryLine(lines: string[], data: Record<string, unknown>): void {
541
+ const summary = String(data.summary ?? "").trim();
542
+ if (data.summary_source === "default") {
543
+ lines.push(`- summary: ⚠️ ${summary} *(reviewer did not return summary)*`);
544
+ } else {
545
+ lines.push(`- summary: ${summary}`);
546
+ }
547
+ }
548
+
549
+ function appendReviewDetails(
550
+ lines: string[],
551
+ data: Record<string, unknown>,
552
+ display: DisplaySettings,
553
+ ): void {
554
+ const issues = ((data.issues as Array<Record<string, unknown>>) ?? []).filter(
555
+ (i) => i.severity !== "low",
556
+ );
557
+ if (issues.length > 0) {
558
+ lines.push("\n**Issues:**");
559
+ for (const it of issues.slice(0, display.maxIssues)) {
560
+ const sev = (it.severity as string) ?? "medium";
561
+ const cat = (it.category as string) ?? "general";
562
+ const issue = (it.issue as string) ?? "";
563
+ const fix = (it.suggested_fix as string) ?? "";
564
+ lines.push(`- **[${sev}] ${cat}**: ${issue}`);
565
+ if (fix) lines.push(` - fix: ${fix}`);
566
+ }
567
+ }
568
+
569
+ const missing = (data.missing_sections as string[]) ?? [];
570
+ if (missing.length > 0) {
571
+ lines.push("\n**Missing Sections:**");
572
+ for (const m of missing.slice(0, display.maxMissingSections)) {
573
+ lines.push(`- ${m}`);
574
+ }
575
+ }
576
+
577
+ const qs = (data.questions as string[]) ?? [];
578
+ if (qs.length > 0) {
579
+ lines.push("\n**Questions:**");
580
+ for (const q of qs.slice(0, display.maxQuestions)) {
581
+ lines.push(`- ${q}`);
582
+ }
583
+ }
584
+ }
585
+
586
+ function titleCase(s: string): string {
587
+ return s.charAt(0).toUpperCase() + s.slice(1);
588
+ }
589
+
590
+ function formatDate(d: Date): string {
591
+ const year = d.getFullYear();
592
+ const month = String(d.getMonth() + 1).padStart(2, "0");
593
+ const day = String(d.getDate()).padStart(2, "0");
594
+ const hours = String(d.getHours()).padStart(2, "0");
595
+ const minutes = String(d.getMinutes()).padStart(2, "0");
596
+ return `${year}-${month}-${day} ${hours}:${minutes}`;
597
+ }