@winspan/claude-forge 8.39.0 → 8.50.6

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 (804) hide show
  1. package/.claude/CLAUDE.md +17 -0
  2. package/.eslintrc.js +23 -0
  3. package/.prettierrc +8 -0
  4. package/ARCHITECTURE_ISSUES.md +249 -0
  5. package/CLAUDE.md +265 -0
  6. package/CLAUDE.md.backup +488 -0
  7. package/DEVELOPMENT.md +310 -0
  8. package/dist/claudemd/claudemd-generator.d.ts +38 -3
  9. package/dist/claudemd/claudemd-generator.d.ts.map +1 -1
  10. package/dist/claudemd/claudemd-generator.js +629 -11
  11. package/dist/claudemd/claudemd-generator.js.map +1 -1
  12. package/dist/claudemd/index.d.ts +2 -2
  13. package/dist/claudemd/index.d.ts.map +1 -1
  14. package/dist/claudemd/index.js.map +1 -1
  15. package/dist/claudemd/resume-manager.d.ts.map +1 -1
  16. package/dist/claudemd/resume-manager.js +5 -2
  17. package/dist/claudemd/resume-manager.js.map +1 -1
  18. package/dist/claudemd/tech-detector.d.ts +1 -0
  19. package/dist/claudemd/tech-detector.d.ts.map +1 -1
  20. package/dist/claudemd/tech-detector.js +53 -0
  21. package/dist/claudemd/tech-detector.js.map +1 -1
  22. package/dist/cli/commands/claudemd.js +2 -2
  23. package/dist/cli/commands/claudemd.js.map +1 -1
  24. package/dist/cli/commands/daemon.d.ts +28 -0
  25. package/dist/cli/commands/daemon.d.ts.map +1 -1
  26. package/dist/cli/commands/daemon.js +200 -8
  27. package/dist/cli/commands/daemon.js.map +1 -1
  28. package/dist/cli/commands/init.d.ts.map +1 -1
  29. package/dist/cli/commands/init.js +3 -35
  30. package/dist/cli/commands/init.js.map +1 -1
  31. package/dist/cli/commands/menu.js +10 -10
  32. package/dist/cli/commands/menu.js.map +1 -1
  33. package/dist/cli/commands/skills.d.ts.map +1 -1
  34. package/dist/cli/commands/skills.js +8 -2
  35. package/dist/cli/commands/skills.js.map +1 -1
  36. package/dist/cli/commands/stats.d.ts.map +1 -1
  37. package/dist/cli/commands/stats.js +0 -17
  38. package/dist/cli/commands/stats.js.map +1 -1
  39. package/dist/cli/commands/trace.d.ts +9 -0
  40. package/dist/cli/commands/trace.d.ts.map +1 -0
  41. package/dist/cli/commands/trace.js +137 -0
  42. package/dist/cli/commands/trace.js.map +1 -0
  43. package/dist/cli/index.js +2 -4
  44. package/dist/cli/index.js.map +1 -1
  45. package/dist/core/ai/provider.d.ts +10 -2
  46. package/dist/core/ai/provider.d.ts.map +1 -1
  47. package/dist/core/ai/provider.js.map +1 -1
  48. package/dist/core/ai/types.d.ts +1 -19
  49. package/dist/core/ai/types.d.ts.map +1 -1
  50. package/dist/core/ai/types.js +1 -1
  51. package/dist/core/config.d.ts +2 -1
  52. package/dist/core/config.d.ts.map +1 -1
  53. package/dist/core/config.js +30 -6
  54. package/dist/core/config.js.map +1 -1
  55. package/dist/core/constants.d.ts +2 -2
  56. package/dist/core/constants.js +2 -2
  57. package/dist/core/constants.js.map +1 -1
  58. package/dist/core/queue/index.d.ts +52 -0
  59. package/dist/core/queue/index.d.ts.map +1 -0
  60. package/dist/core/queue/index.js +176 -0
  61. package/dist/core/queue/index.js.map +1 -0
  62. package/dist/core/storage/base.d.ts +33 -0
  63. package/dist/core/storage/base.d.ts.map +1 -0
  64. package/dist/core/storage/base.js +211 -0
  65. package/dist/core/storage/base.js.map +1 -0
  66. package/dist/core/storage/events.d.ts +52 -0
  67. package/dist/core/storage/events.d.ts.map +1 -0
  68. package/dist/core/storage/events.js +201 -0
  69. package/dist/core/storage/events.js.map +1 -0
  70. package/dist/core/storage/injections.d.ts +27 -0
  71. package/dist/core/storage/injections.d.ts.map +1 -0
  72. package/dist/core/storage/injections.js +51 -0
  73. package/dist/core/storage/injections.js.map +1 -0
  74. package/dist/core/storage/maintenance.d.ts +21 -0
  75. package/dist/core/storage/maintenance.d.ts.map +1 -0
  76. package/dist/core/storage/maintenance.js +52 -0
  77. package/dist/core/storage/maintenance.js.map +1 -0
  78. package/dist/core/storage/routing.d.ts +71 -0
  79. package/dist/core/storage/routing.d.ts.map +1 -0
  80. package/dist/core/storage/routing.js +141 -0
  81. package/dist/core/storage/routing.js.map +1 -0
  82. package/dist/core/storage/rows.d.ts +0 -47
  83. package/dist/core/storage/rows.d.ts.map +1 -1
  84. package/dist/core/storage/schema.sql +74 -136
  85. package/dist/core/storage/sessions.d.ts +34 -0
  86. package/dist/core/storage/sessions.d.ts.map +1 -0
  87. package/dist/core/storage/sessions.js +78 -0
  88. package/dist/core/storage/sessions.js.map +1 -0
  89. package/dist/core/storage/skills.d.ts +40 -0
  90. package/dist/core/storage/skills.d.ts.map +1 -0
  91. package/dist/core/storage/skills.js +107 -0
  92. package/dist/core/storage/skills.js.map +1 -0
  93. package/dist/core/storage/sqlite.d.ts +63 -265
  94. package/dist/core/storage/sqlite.d.ts.map +1 -1
  95. package/dist/core/storage/sqlite.js +102 -759
  96. package/dist/core/storage/sqlite.js.map +1 -1
  97. package/dist/core/storage/tasks.d.ts +64 -0
  98. package/dist/core/storage/tasks.d.ts.map +1 -0
  99. package/dist/core/storage/tasks.js +134 -0
  100. package/dist/core/storage/tasks.js.map +1 -0
  101. package/dist/core/storage/token-usage.d.ts +36 -0
  102. package/dist/core/storage/token-usage.d.ts.map +1 -0
  103. package/dist/core/storage/token-usage.js +59 -0
  104. package/dist/core/storage/token-usage.js.map +1 -0
  105. package/dist/core/types.d.ts +70 -4
  106. package/dist/core/types.d.ts.map +1 -1
  107. package/dist/core/types.js +24 -1
  108. package/dist/core/types.js.map +1 -1
  109. package/dist/core/utils/format.d.ts +28 -0
  110. package/dist/core/utils/format.d.ts.map +1 -0
  111. package/dist/core/utils/format.js +68 -0
  112. package/dist/core/utils/format.js.map +1 -0
  113. package/dist/core/utils/logger.d.ts +6 -1
  114. package/dist/core/utils/logger.d.ts.map +1 -1
  115. package/dist/core/utils/logger.js +72 -2
  116. package/dist/core/utils/logger.js.map +1 -1
  117. package/dist/core/utils/session.d.ts +16 -0
  118. package/dist/core/utils/session.d.ts.map +1 -0
  119. package/dist/core/utils/session.js +25 -0
  120. package/dist/core/utils/session.js.map +1 -0
  121. package/dist/core/utils/time.d.ts +22 -0
  122. package/dist/core/utils/time.d.ts.map +1 -0
  123. package/dist/core/utils/time.js +38 -0
  124. package/dist/core/utils/time.js.map +1 -0
  125. package/dist/daemon/handlers/history-exporter.d.ts.map +1 -1
  126. package/dist/daemon/handlers/history-exporter.js +6 -4
  127. package/dist/daemon/handlers/history-exporter.js.map +1 -1
  128. package/dist/daemon/handlers/post-tool-use.d.ts +5 -12
  129. package/dist/daemon/handlers/post-tool-use.d.ts.map +1 -1
  130. package/dist/daemon/handlers/post-tool-use.js +21 -79
  131. package/dist/daemon/handlers/post-tool-use.js.map +1 -1
  132. package/dist/daemon/handlers/stop.d.ts +24 -12
  133. package/dist/daemon/handlers/stop.d.ts.map +1 -1
  134. package/dist/daemon/handlers/stop.js +141 -42
  135. package/dist/daemon/handlers/stop.js.map +1 -1
  136. package/dist/daemon/handlers/user-prompt.d.ts +18 -19
  137. package/dist/daemon/handlers/user-prompt.d.ts.map +1 -1
  138. package/dist/daemon/handlers/user-prompt.js +103 -227
  139. package/dist/daemon/handlers/user-prompt.js.map +1 -1
  140. package/dist/daemon/index.d.ts +6 -2
  141. package/dist/daemon/index.d.ts.map +1 -1
  142. package/dist/daemon/index.js +76 -120
  143. package/dist/daemon/index.js.map +1 -1
  144. package/dist/daemon/launchd/com.claude-forge.daemon.plist.template +47 -0
  145. package/dist/daemon/launchd-installer.d.ts +61 -0
  146. package/dist/daemon/launchd-installer.d.ts.map +1 -0
  147. package/dist/daemon/launchd-installer.js +182 -0
  148. package/dist/daemon/launchd-installer.js.map +1 -0
  149. package/dist/daemon/lifecycle.d.ts +11 -0
  150. package/dist/daemon/lifecycle.d.ts.map +1 -1
  151. package/dist/daemon/lifecycle.js +44 -0
  152. package/dist/daemon/lifecycle.js.map +1 -1
  153. package/dist/daemon/router.d.ts +9 -2
  154. package/dist/daemon/router.d.ts.map +1 -1
  155. package/dist/daemon/router.js +27 -3
  156. package/dist/daemon/router.js.map +1 -1
  157. package/dist/daemon/server.d.ts.map +1 -1
  158. package/dist/daemon/server.js +6 -5
  159. package/dist/daemon/server.js.map +1 -1
  160. package/dist/daemon/services/anti-pattern-detector.d.ts +50 -0
  161. package/dist/daemon/services/anti-pattern-detector.d.ts.map +1 -0
  162. package/dist/daemon/services/anti-pattern-detector.js +357 -0
  163. package/dist/daemon/services/anti-pattern-detector.js.map +1 -0
  164. package/dist/daemon/services/drift-detector.d.ts +64 -0
  165. package/dist/daemon/services/drift-detector.d.ts.map +1 -0
  166. package/dist/daemon/services/drift-detector.js +201 -0
  167. package/dist/daemon/services/drift-detector.js.map +1 -0
  168. package/dist/{intelligence → daemon/services}/task-segmenter.d.ts +7 -1
  169. package/dist/daemon/services/task-segmenter.d.ts.map +1 -0
  170. package/dist/{intelligence → daemon/services}/task-segmenter.js +29 -6
  171. package/dist/daemon/services/task-segmenter.js.map +1 -0
  172. package/dist/daemon/services/weekly-report.d.ts +91 -0
  173. package/dist/daemon/services/weekly-report.d.ts.map +1 -0
  174. package/dist/daemon/services/weekly-report.js +327 -0
  175. package/dist/daemon/services/weekly-report.js.map +1 -0
  176. package/dist/hooks/hook-lib.sh +81 -0
  177. package/dist/hooks/notification.sh +7 -3
  178. package/dist/hooks/post-tool-use.sh +8 -4
  179. package/dist/hooks/pre-tool-use.sh +7 -4
  180. package/dist/hooks/stop.sh +1 -1
  181. package/dist/hooks/user-prompt-submit.sh +8 -9
  182. package/dist/mcp/server.d.ts +2 -2
  183. package/dist/mcp/server.d.ts.map +1 -1
  184. package/dist/mcp/server.js +71 -11
  185. package/dist/mcp/server.js.map +1 -1
  186. package/dist/skills/invocation-guard.d.ts +20 -0
  187. package/dist/skills/invocation-guard.d.ts.map +1 -1
  188. package/dist/skills/invocation-guard.js +63 -0
  189. package/dist/skills/invocation-guard.js.map +1 -1
  190. package/dist/skills/matcher.d.ts.map +1 -1
  191. package/dist/skills/matcher.js +12 -3
  192. package/dist/skills/matcher.js.map +1 -1
  193. package/dist/skills/official/code-simplifier.md +16 -0
  194. package/dist/skills/official/find-skills.md +23 -0
  195. package/dist/skills/official/official-api-design.md +17 -0
  196. package/dist/skills/official/official-architecture-decision.md +20 -0
  197. package/dist/skills/official/official-bmad.md +118 -0
  198. package/dist/skills/official/official-db-schema-design.md +16 -0
  199. package/dist/skills/official/official-debug.md +17 -0
  200. package/dist/skills/official/official-doc-driven.md +31 -0
  201. package/dist/skills/official/official-harness-engineering.md +108 -0
  202. package/dist/skills/official/official-performance-optimization.md +30 -0
  203. package/dist/skills/official/official-pr-review.md +35 -0
  204. package/dist/skills/official/official-release-checklist.md +30 -0
  205. package/dist/skills/official/official-security-hardening.md +26 -0
  206. package/dist/skills/official/official-spec-driven-design.md +31 -0
  207. package/dist/skills/official/planning-with-files.md +37 -0
  208. package/dist/skills/official/ui-ux-pro-max.md +18 -0
  209. package/dist/skills/official/webapp-testing.md +12 -0
  210. package/dist/skills/official-skills.d.ts +8 -4
  211. package/dist/skills/official-skills.d.ts.map +1 -1
  212. package/dist/skills/official-skills.js +48 -704
  213. package/dist/skills/official-skills.js.map +1 -1
  214. package/dist/skills/registry.d.ts +5 -0
  215. package/dist/skills/registry.d.ts.map +1 -1
  216. package/dist/skills/registry.js +48 -15
  217. package/dist/skills/registry.js.map +1 -1
  218. package/dist/skills/tools/pipeline-suggest.d.ts +30 -0
  219. package/dist/skills/tools/pipeline-suggest.d.ts.map +1 -0
  220. package/dist/skills/tools/pipeline-suggest.js +178 -0
  221. package/dist/skills/tools/pipeline-suggest.js.map +1 -0
  222. package/dist/skills/tools/skill-invoke.d.ts +2 -0
  223. package/dist/skills/tools/skill-invoke.d.ts.map +1 -1
  224. package/dist/skills/tools/skill-invoke.js +5 -2
  225. package/dist/skills/tools/skill-invoke.js.map +1 -1
  226. package/dist/web/routes/ai.d.ts.map +1 -1
  227. package/dist/web/routes/ai.js +16 -22
  228. package/dist/web/routes/ai.js.map +1 -1
  229. package/dist/web/routes/drift.d.ts +10 -0
  230. package/dist/web/routes/drift.d.ts.map +1 -0
  231. package/dist/web/routes/drift.js +21 -0
  232. package/dist/web/routes/drift.js.map +1 -0
  233. package/dist/web/routes/error-handler.d.ts +43 -0
  234. package/dist/web/routes/error-handler.d.ts.map +1 -0
  235. package/dist/web/routes/error-handler.js +99 -0
  236. package/dist/web/routes/error-handler.js.map +1 -0
  237. package/dist/web/routes/insights.d.ts +9 -0
  238. package/dist/web/routes/insights.d.ts.map +1 -0
  239. package/dist/web/routes/insights.js +34 -0
  240. package/dist/web/routes/insights.js.map +1 -0
  241. package/dist/web/routes/patch.js +2 -2
  242. package/dist/web/routes/patch.js.map +1 -1
  243. package/dist/web/routes/reports.d.ts +10 -0
  244. package/dist/web/routes/reports.d.ts.map +1 -0
  245. package/dist/web/routes/reports.js +27 -0
  246. package/dist/web/routes/reports.js.map +1 -0
  247. package/dist/web/routes/rules.d.ts +10 -3
  248. package/dist/web/routes/rules.d.ts.map +1 -1
  249. package/dist/web/routes/rules.js +80 -95
  250. package/dist/web/routes/rules.js.map +1 -1
  251. package/dist/web/routes/sessions.d.ts +1 -2
  252. package/dist/web/routes/sessions.d.ts.map +1 -1
  253. package/dist/web/routes/sessions.js +27 -39
  254. package/dist/web/routes/sessions.js.map +1 -1
  255. package/dist/web/routes/skill-stats.d.ts.map +1 -1
  256. package/dist/web/routes/skill-stats.js +38 -0
  257. package/dist/web/routes/skill-stats.js.map +1 -1
  258. package/dist/web/routes/skills.d.ts.map +1 -1
  259. package/dist/web/routes/skills.js +34 -0
  260. package/dist/web/routes/skills.js.map +1 -1
  261. package/dist/web/routes/stats.d.ts +7 -0
  262. package/dist/web/routes/stats.d.ts.map +1 -0
  263. package/dist/web/routes/stats.js +44 -0
  264. package/dist/web/routes/stats.js.map +1 -0
  265. package/dist/web/routes/status.js +1 -1
  266. package/dist/web/routes/status.js.map +1 -1
  267. package/dist/web/routes/tasks.d.ts +4 -0
  268. package/dist/web/routes/tasks.d.ts.map +1 -0
  269. package/dist/web/routes/tasks.js +181 -0
  270. package/dist/web/routes/tasks.js.map +1 -0
  271. package/dist/web/routes/trace.d.ts +10 -0
  272. package/dist/web/routes/trace.d.ts.map +1 -0
  273. package/dist/web/routes/trace.js +123 -0
  274. package/dist/web/routes/trace.js.map +1 -0
  275. package/dist/web/routes/types.d.ts +1 -14
  276. package/dist/web/routes/types.d.ts.map +1 -1
  277. package/dist/web/routes/types.js +8 -17
  278. package/dist/web/routes/types.js.map +1 -1
  279. package/dist/web/server.d.ts +1 -9
  280. package/dist/web/server.d.ts.map +1 -1
  281. package/dist/web/server.js +28 -28
  282. package/dist/web/server.js.map +1 -1
  283. package/dist/web/static/assets/AIConfig-BQCAQE9D.js +2 -0
  284. package/dist/web/static/assets/AIConfig-BQCAQE9D.js.map +1 -0
  285. package/dist/web/static/assets/Dashboard-D7Bo6Kan.js +2 -0
  286. package/dist/web/static/assets/Dashboard-D7Bo6Kan.js.map +1 -0
  287. package/dist/web/static/assets/{Drawer-DcU3ln98.js → Drawer-BeHRQxUS.js} +2 -2
  288. package/dist/web/static/assets/{Drawer-DcU3ln98.js.map → Drawer-BeHRQxUS.js.map} +1 -1
  289. package/dist/web/static/assets/Events-K_tCY2ti.js +2 -0
  290. package/dist/web/static/assets/Events-K_tCY2ti.js.map +1 -0
  291. package/dist/web/static/assets/Reports-BJCmBnc_.js +2 -0
  292. package/dist/web/static/assets/Reports-BJCmBnc_.js.map +1 -0
  293. package/dist/web/static/assets/SearchInput-BX2KhMkw.js +2 -0
  294. package/dist/web/static/assets/SearchInput-BX2KhMkw.js.map +1 -0
  295. package/dist/web/static/assets/SessionDetail-Bkr-kC7V.js +2 -0
  296. package/dist/web/static/assets/SessionDetail-Bkr-kC7V.js.map +1 -0
  297. package/dist/web/static/assets/Sessions-Chx9OCLH.js +2 -0
  298. package/dist/web/static/assets/Sessions-Chx9OCLH.js.map +1 -0
  299. package/dist/web/static/assets/Skills-O0GT1i7m.js +2 -0
  300. package/dist/web/static/assets/Skills-O0GT1i7m.js.map +1 -0
  301. package/dist/web/static/assets/TaskDetail-5SR8zGzv.js +2 -0
  302. package/dist/web/static/assets/TaskDetail-5SR8zGzv.js.map +1 -0
  303. package/dist/web/static/assets/Tasks-DCgDqvOZ.js +2 -0
  304. package/dist/web/static/assets/Tasks-DCgDqvOZ.js.map +1 -0
  305. package/dist/web/static/assets/export-L_VBD2p1.js +4 -0
  306. package/dist/web/static/assets/export-L_VBD2p1.js.map +1 -0
  307. package/dist/web/static/assets/index-D8AKj26b.css +1 -0
  308. package/dist/web/static/assets/index-DxIbmNmr.js +3 -0
  309. package/dist/web/static/assets/index-DxIbmNmr.js.map +1 -0
  310. package/dist/web/static/assets/{lucide-53bR2rki.js → lucide-fJlPI3H7.js} +68 -38
  311. package/dist/web/static/assets/lucide-fJlPI3H7.js.map +1 -0
  312. package/dist/web/static/assets/time-Bxuk0M-C.js +2 -0
  313. package/dist/web/static/assets/time-Bxuk0M-C.js.map +1 -0
  314. package/dist/web/static/index.html +3 -3
  315. package/docs/concurrent-agents.md +129 -0
  316. package/docs/design/architecture-review-20260516.md +232 -0
  317. package/docs/design/fix-skills-data-and-set-leak-spec-20260516-1300.md +219 -0
  318. package/docs/design/hook-failure-queue-spec-20260516-1530.md +204 -0
  319. package/docs/design/refactor-phase1-spec-20260515-1600.md +543 -0
  320. package/docs/design/refactor-phase2-spec-20260515-1700.md +424 -0
  321. package/docs/design/tasks-list-filter-pagination-spec-20260518-0930.md +208 -0
  322. package/docs/implementation/fix-skills-data-and-set-leak-changelog-20260516-1300.md +104 -0
  323. package/docs/implementation/hook-failure-queue-changelog-20260516-1530.md +196 -0
  324. package/docs/implementation/hotfix-daemon-event-reject-20260516-1430.md +56 -0
  325. package/docs/implementation/refactor-phase1-changelog-20260515-1630.md +354 -0
  326. package/docs/implementation/refactor-phase2-changelog-20260515-1705.md +421 -0
  327. package/docs/implementation/tasks-list-filter-pagination-changelog-20260518-0930.md +72 -0
  328. package/docs/reviews/claudemd-template-sync.md +54 -0
  329. package/docs/reviews/tasks-filter-pagination.md +80 -0
  330. package/docs/ruflo-learning-strategy.md +322 -0
  331. package/docs/skills-deduplication-analysis.md +83 -0
  332. package/docs/skills-multiformat-support.md +177 -0
  333. package/docs/skills-third-party.md +183 -0
  334. package/docs/testing/tasks-filter-pagination-test-report.md +86 -0
  335. package/forge +321 -0
  336. package/package.json +28 -62
  337. package/playwright.config.ts +40 -0
  338. package/scripts/demo-v2.ts +91 -0
  339. package/scripts/dev-daemon.sh +232 -0
  340. package/scripts/dev-web.ts +109 -0
  341. package/scripts/e2e-mcp-link.ts +423 -0
  342. package/scripts/e2e-methodology-quality.ts +253 -0
  343. package/scripts/e2e-routing.ts +456 -0
  344. package/scripts/e2e-user-methodology.ts +326 -0
  345. package/scripts/e2e-web-workflows.ts +299 -0
  346. package/scripts/migrate-legacy-to-dynamic.sql +108 -0
  347. package/scripts/regenerate-execution-docs.ts +116 -0
  348. package/scripts/sync-agent-skills.ts +193 -0
  349. package/scripts/test-hook.sh +71 -0
  350. package/scripts/verify-skill-loading.ts +62 -0
  351. package/src/claudemd/claudemd-generator.ts +777 -0
  352. package/src/claudemd/convention-extractor.ts +69 -0
  353. package/src/claudemd/index.ts +35 -0
  354. package/src/claudemd/persona-manager.ts +88 -0
  355. package/src/claudemd/resume-manager.ts +236 -0
  356. package/src/claudemd/tech-detector.ts +220 -0
  357. package/src/cli/commands/claudemd.ts +84 -0
  358. package/src/cli/commands/config.ts +46 -0
  359. package/src/cli/commands/daemon.ts +310 -0
  360. package/src/cli/commands/executions.ts +114 -0
  361. package/src/cli/commands/init.ts +204 -0
  362. package/src/cli/commands/logs.ts +181 -0
  363. package/src/cli/commands/mcp.ts +244 -0
  364. package/src/cli/commands/menu.ts +356 -0
  365. package/src/cli/commands/skills.ts +185 -0
  366. package/src/cli/commands/stats.ts +74 -0
  367. package/src/cli/commands/status.ts +69 -0
  368. package/src/cli/commands/template.ts +77 -0
  369. package/src/cli/commands/trace.ts +164 -0
  370. package/src/cli/index.ts +42 -0
  371. package/src/cli/init/hook-manager.ts +132 -0
  372. package/src/core/ai/provider.ts +308 -0
  373. package/src/core/ai/types.ts +51 -0
  374. package/src/core/config.ts +124 -0
  375. package/src/core/constants.ts +45 -0
  376. package/src/core/queue/index.ts +193 -0
  377. package/src/core/storage/base.ts +226 -0
  378. package/src/core/storage/events.ts +255 -0
  379. package/src/core/storage/injections.ts +78 -0
  380. package/src/core/storage/maintenance.ts +59 -0
  381. package/src/core/storage/migrations/002_add_skill_tracking.sql +6 -0
  382. package/src/core/storage/migrations/003_add_skill_invocations.sql +23 -0
  383. package/src/core/storage/performance-indexes.sql +23 -0
  384. package/src/core/storage/routing.ts +194 -0
  385. package/src/core/storage/rows.ts +112 -0
  386. package/src/core/storage/schema.sql +214 -0
  387. package/src/core/storage/sessions.ts +104 -0
  388. package/src/core/storage/skills.ts +164 -0
  389. package/src/core/storage/sqlite.ts +194 -0
  390. package/src/core/storage/tasks.ts +170 -0
  391. package/src/core/storage/token-usage.ts +93 -0
  392. package/src/core/types.ts +154 -0
  393. package/src/core/utils/error-handler.ts +256 -0
  394. package/src/core/utils/forge-resume-block.ts +74 -0
  395. package/src/core/utils/format.ts +69 -0
  396. package/src/core/utils/logger.ts +119 -0
  397. package/src/core/utils/lru-cache.ts +50 -0
  398. package/src/core/utils/path.ts +19 -0
  399. package/src/core/utils/session.ts +26 -0
  400. package/src/core/utils/time.ts +37 -0
  401. package/src/core/utils/token-tracker.ts +97 -0
  402. package/src/daemon/event-parser.ts +35 -0
  403. package/src/daemon/handlers/history-exporter.ts +117 -0
  404. package/src/daemon/handlers/post-tool-use.ts +50 -0
  405. package/src/daemon/handlers/stop.ts +215 -0
  406. package/src/daemon/handlers/user-prompt.ts +188 -0
  407. package/src/daemon/index.ts +278 -0
  408. package/src/daemon/launchd/com.claude-forge.daemon.plist.template +47 -0
  409. package/src/daemon/launchd-installer.ts +260 -0
  410. package/src/daemon/lifecycle.ts +128 -0
  411. package/src/daemon/router.ts +40 -0
  412. package/src/daemon/server.ts +209 -0
  413. package/src/daemon/services/anti-pattern-detector.ts +412 -0
  414. package/src/daemon/services/drift-detector.ts +232 -0
  415. package/src/daemon/services/task-segmenter.ts +112 -0
  416. package/src/daemon/services/weekly-report.ts +454 -0
  417. package/src/hooks/hook-lib.sh +81 -0
  418. package/src/hooks/notification.sh +35 -0
  419. package/src/hooks/post-tool-use.sh +61 -0
  420. package/src/hooks/pre-tool-use.sh +63 -0
  421. package/src/hooks/stop.sh +40 -0
  422. package/src/hooks/user-prompt-submit.sh +69 -0
  423. package/src/mcp/server.ts +322 -0
  424. package/src/skills/index.ts +2 -0
  425. package/src/skills/invocation-guard.ts +177 -0
  426. package/src/skills/matcher.ts +148 -0
  427. package/src/skills/official/code-simplifier.md +16 -0
  428. package/src/skills/official/find-skills.md +23 -0
  429. package/src/skills/official/official-api-design.md +17 -0
  430. package/src/skills/official/official-architecture-decision.md +20 -0
  431. package/src/skills/official/official-bmad.md +118 -0
  432. package/src/skills/official/official-db-schema-design.md +16 -0
  433. package/src/skills/official/official-debug.md +17 -0
  434. package/src/skills/official/official-doc-driven.md +31 -0
  435. package/src/skills/official/official-harness-engineering.md +108 -0
  436. package/src/skills/official/official-performance-optimization.md +30 -0
  437. package/src/skills/official/official-pr-review.md +35 -0
  438. package/src/skills/official/official-release-checklist.md +30 -0
  439. package/src/skills/official/official-security-hardening.md +26 -0
  440. package/src/skills/official/official-spec-driven-design.md +31 -0
  441. package/src/skills/official/planning-with-files.md +37 -0
  442. package/src/skills/official/ui-ux-pro-max.md +18 -0
  443. package/src/skills/official/webapp-testing.md +12 -0
  444. package/src/skills/official-skills.ts +89 -0
  445. package/src/skills/registry.ts +355 -0
  446. package/src/skills/semantic-matcher.ts +231 -0
  447. package/src/skills/tools/pipeline-suggest.ts +226 -0
  448. package/src/skills/tools/skill-invoke.ts +168 -0
  449. package/src/skills/tools/skill-list.ts +59 -0
  450. package/src/templates/go.yaml +53 -0
  451. package/src/templates/python.yaml +59 -0
  452. package/src/templates/react.yaml +55 -0
  453. package/src/templates/template-manager.ts +170 -0
  454. package/src/web/auth-middleware.ts +55 -0
  455. package/src/web/routes/ai.ts +204 -0
  456. package/src/web/routes/auth.ts +22 -0
  457. package/src/web/routes/drift.ts +25 -0
  458. package/src/web/routes/error-handler.ts +120 -0
  459. package/src/web/routes/events.ts +47 -0
  460. package/src/web/routes/insights.ts +43 -0
  461. package/src/web/routes/patch.ts +117 -0
  462. package/src/web/routes/reports.ts +34 -0
  463. package/src/web/routes/rules.ts +101 -0
  464. package/src/web/routes/sessions.ts +262 -0
  465. package/src/web/routes/skill-stats.ts +132 -0
  466. package/src/web/routes/skills.ts +349 -0
  467. package/src/web/routes/static.ts +67 -0
  468. package/src/web/routes/stats.ts +60 -0
  469. package/src/web/routes/status.ts +30 -0
  470. package/src/web/routes/tasks.ts +218 -0
  471. package/src/web/routes/token-usage.ts +20 -0
  472. package/src/web/routes/trace.ts +138 -0
  473. package/src/web/routes/types.ts +56 -0
  474. package/src/web/server.ts +134 -0
  475. package/src/web/ssrf-guard.ts +112 -0
  476. package/src/web/static/index.html +3251 -0
  477. package/src/web/static/vendor/chart.umd.min.js +20 -0
  478. package/tests/e2e/dashboard.spec.ts +205 -0
  479. package/tests/e2e/routing-skill-e2e.test.ts +39 -0
  480. package/tests/helpers/mock-ai.ts +92 -0
  481. package/tests/helpers/mock-storage.ts +159 -0
  482. package/tests/integration/queue-replay.integration.test.ts +193 -0
  483. package/tests/integration/tasks-filter.integration.test.ts +154 -0
  484. package/tests/performance/database.benchmark.ts +161 -0
  485. package/tests/semantic-matcher.test.ts +99 -0
  486. package/tests/skill-matcher.test.ts +110 -0
  487. package/tests/unit/ai-provider-retry.test.ts +194 -0
  488. package/tests/unit/ai-provider-vision.test.ts +224 -0
  489. package/tests/unit/claudemd-generator.test.ts +68 -0
  490. package/tests/unit/cli-mcp.test.ts +141 -0
  491. package/tests/unit/handlers.test.ts +171 -0
  492. package/tests/unit/invocation-guard.test.ts +125 -0
  493. package/tests/unit/queue.test.ts +272 -0
  494. package/tests/unit/router.test.ts +138 -0
  495. package/tests/unit/security.test.ts +128 -0
  496. package/tests/unit/skill-invocations-workflow.test.ts +495 -0
  497. package/tests/unit/skill-registry.test.ts +94 -0
  498. package/tests/unit/skills/invocation-guard-ttl.test.ts +211 -0
  499. package/tests/unit/skills/official-skills-loader.test.ts +126 -0
  500. package/tests/unit/skills/registry-multiformat.test.ts +92 -0
  501. package/tests/unit/storage/sessions-aggregate.test.ts +435 -0
  502. package/tests/unit/storage/sqlite-refactor-harness.test.ts +314 -0
  503. package/tests/unit/storage.test.ts +172 -0
  504. package/tests/unit/token-usage.test.ts +144 -0
  505. package/tests/unit/type-guards.test.ts +201 -0
  506. package/tests/unit/utils/format.test.ts +189 -0
  507. package/tests/unit/utils/session.test.ts +89 -0
  508. package/tests/unit/utils/time.test.ts +112 -0
  509. package/tests/unit/web/routes-auth.test.ts +93 -0
  510. package/tests/unit/web/routes-events.test.ts +101 -0
  511. package/tests/unit/web/routes-sessions.test.ts +181 -0
  512. package/tests/unit/web/routes-skill-stats.test.ts +179 -0
  513. package/tests/unit/web/routes-stats.test.ts +92 -0
  514. package/tests/unit/web/routes-tasks.test.ts +351 -0
  515. package/tsconfig.json +22 -0
  516. package/vitest.config.ts +21 -0
  517. package/vitest.integration.config.ts +16 -0
  518. package/web/CLAUDE.md +20 -0
  519. package/web/index.html +13 -0
  520. package/web/package-lock.json +4854 -0
  521. package/web/package.json +35 -0
  522. package/web/postcss.config.js +6 -0
  523. package/web/src/App.tsx +110 -0
  524. package/web/src/components/CodeBlock.tsx +31 -0
  525. package/web/src/components/Confirm.tsx +96 -0
  526. package/web/src/components/Drawer.tsx +60 -0
  527. package/web/src/components/Layout.tsx +145 -0
  528. package/web/src/components/MarkdownRenderer.tsx +77 -0
  529. package/web/src/components/SearchInput.tsx +31 -0
  530. package/web/src/components/SessionDetailContent.tsx +157 -0
  531. package/web/src/components/Toast.tsx +92 -0
  532. package/web/src/index.css +19 -0
  533. package/web/src/main.tsx +31 -0
  534. package/web/src/pages/AIConfig.tsx +233 -0
  535. package/web/src/pages/Dashboard.tsx +572 -0
  536. package/web/src/pages/Events.tsx +271 -0
  537. package/web/src/pages/Reports.tsx +428 -0
  538. package/web/src/pages/SessionDetail.tsx +162 -0
  539. package/web/src/pages/Sessions.tsx +205 -0
  540. package/web/src/pages/Skills.tsx +180 -0
  541. package/web/src/pages/TaskDetail.tsx +511 -0
  542. package/web/src/pages/Tasks.tsx +150 -0
  543. package/web/src/utils/auth.ts +59 -0
  544. package/web/src/utils/export.ts +54 -0
  545. package/web/src/utils/time.ts +13 -0
  546. package/web/tailwind.config.js +11 -0
  547. package/web/tsconfig.json +21 -0
  548. package/web/tsconfig.node.json +10 -0
  549. package/web/vite.config.ts +76 -0
  550. package/winspan-claude-forge-8.43.0.tgz +0 -0
  551. package/dist/agents/definition.d.ts +0 -62
  552. package/dist/agents/definition.d.ts.map +0 -1
  553. package/dist/agents/definition.js +0 -27
  554. package/dist/agents/definition.js.map +0 -1
  555. package/dist/agents/distributor.d.ts +0 -23
  556. package/dist/agents/distributor.d.ts.map +0 -1
  557. package/dist/agents/distributor.js +0 -85
  558. package/dist/agents/distributor.js.map +0 -1
  559. package/dist/agents/index.d.ts +0 -5
  560. package/dist/agents/index.d.ts.map +0 -1
  561. package/dist/agents/index.js +0 -5
  562. package/dist/agents/index.js.map +0 -1
  563. package/dist/agents/methodologies/agent-builder.d.ts +0 -21
  564. package/dist/agents/methodologies/agent-builder.d.ts.map +0 -1
  565. package/dist/agents/methodologies/agent-builder.js +0 -124
  566. package/dist/agents/methodologies/agent-builder.js.map +0 -1
  567. package/dist/agents/methodologies/phases/bmad/analyze.d.ts +0 -3
  568. package/dist/agents/methodologies/phases/bmad/analyze.d.ts.map +0 -1
  569. package/dist/agents/methodologies/phases/bmad/analyze.js +0 -18
  570. package/dist/agents/methodologies/phases/bmad/analyze.js.map +0 -1
  571. package/dist/agents/methodologies/phases/bmad/design.d.ts +0 -3
  572. package/dist/agents/methodologies/phases/bmad/design.d.ts.map +0 -1
  573. package/dist/agents/methodologies/phases/bmad/design.js +0 -17
  574. package/dist/agents/methodologies/phases/bmad/design.js.map +0 -1
  575. package/dist/agents/methodologies/phases/bmad/implement.d.ts +0 -3
  576. package/dist/agents/methodologies/phases/bmad/implement.d.ts.map +0 -1
  577. package/dist/agents/methodologies/phases/bmad/implement.js +0 -16
  578. package/dist/agents/methodologies/phases/bmad/implement.js.map +0 -1
  579. package/dist/agents/methodologies/phases/bmad/index.d.ts +0 -6
  580. package/dist/agents/methodologies/phases/bmad/index.d.ts.map +0 -1
  581. package/dist/agents/methodologies/phases/bmad/index.js +0 -6
  582. package/dist/agents/methodologies/phases/bmad/index.js.map +0 -1
  583. package/dist/agents/methodologies/phases/bmad/review.d.ts +0 -3
  584. package/dist/agents/methodologies/phases/bmad/review.d.ts.map +0 -1
  585. package/dist/agents/methodologies/phases/bmad/review.js +0 -16
  586. package/dist/agents/methodologies/phases/bmad/review.js.map +0 -1
  587. package/dist/agents/methodologies/phases/bmad/test.d.ts +0 -3
  588. package/dist/agents/methodologies/phases/bmad/test.d.ts.map +0 -1
  589. package/dist/agents/methodologies/phases/bmad/test.js +0 -20
  590. package/dist/agents/methodologies/phases/bmad/test.js.map +0 -1
  591. package/dist/agents/methodologies/phases/harness/fix.d.ts +0 -3
  592. package/dist/agents/methodologies/phases/harness/fix.d.ts.map +0 -1
  593. package/dist/agents/methodologies/phases/harness/fix.js +0 -16
  594. package/dist/agents/methodologies/phases/harness/fix.js.map +0 -1
  595. package/dist/agents/methodologies/phases/harness/index.d.ts +0 -6
  596. package/dist/agents/methodologies/phases/harness/index.d.ts.map +0 -1
  597. package/dist/agents/methodologies/phases/harness/index.js +0 -6
  598. package/dist/agents/methodologies/phases/harness/index.js.map +0 -1
  599. package/dist/agents/methodologies/phases/harness/reproduce.d.ts +0 -3
  600. package/dist/agents/methodologies/phases/harness/reproduce.d.ts.map +0 -1
  601. package/dist/agents/methodologies/phases/harness/reproduce.js +0 -19
  602. package/dist/agents/methodologies/phases/harness/reproduce.js.map +0 -1
  603. package/dist/agents/methodologies/phases/harness/root-cause.d.ts +0 -3
  604. package/dist/agents/methodologies/phases/harness/root-cause.d.ts.map +0 -1
  605. package/dist/agents/methodologies/phases/harness/root-cause.js +0 -20
  606. package/dist/agents/methodologies/phases/harness/root-cause.js.map +0 -1
  607. package/dist/agents/methodologies/phases/harness/safety-net.d.ts +0 -3
  608. package/dist/agents/methodologies/phases/harness/safety-net.d.ts.map +0 -1
  609. package/dist/agents/methodologies/phases/harness/safety-net.js +0 -16
  610. package/dist/agents/methodologies/phases/harness/safety-net.js.map +0 -1
  611. package/dist/agents/methodologies/phases/harness/verify.d.ts +0 -3
  612. package/dist/agents/methodologies/phases/harness/verify.d.ts.map +0 -1
  613. package/dist/agents/methodologies/phases/harness/verify.js +0 -21
  614. package/dist/agents/methodologies/phases/harness/verify.js.map +0 -1
  615. package/dist/agents/methodologies/presets.d.ts +0 -10
  616. package/dist/agents/methodologies/presets.d.ts.map +0 -1
  617. package/dist/agents/methodologies/presets.js +0 -79
  618. package/dist/agents/methodologies/presets.js.map +0 -1
  619. package/dist/agents/methodologies/types.d.ts +0 -43
  620. package/dist/agents/methodologies/types.d.ts.map +0 -1
  621. package/dist/agents/methodologies/types.js +0 -10
  622. package/dist/agents/methodologies/types.js.map +0 -1
  623. package/dist/agents/methodologies/user-config-loader.d.ts +0 -30
  624. package/dist/agents/methodologies/user-config-loader.d.ts.map +0 -1
  625. package/dist/agents/methodologies/user-config-loader.js +0 -159
  626. package/dist/agents/methodologies/user-config-loader.js.map +0 -1
  627. package/dist/agents/official-agents.d.ts +0 -4
  628. package/dist/agents/official-agents.d.ts.map +0 -1
  629. package/dist/agents/official-agents.js +0 -559
  630. package/dist/agents/official-agents.js.map +0 -1
  631. package/dist/agents/registry.d.ts +0 -57
  632. package/dist/agents/registry.d.ts.map +0 -1
  633. package/dist/agents/registry.js +0 -271
  634. package/dist/agents/registry.js.map +0 -1
  635. package/dist/capability/index.d.ts +0 -10
  636. package/dist/capability/index.d.ts.map +0 -1
  637. package/dist/capability/index.js +0 -10
  638. package/dist/capability/index.js.map +0 -1
  639. package/dist/capability/types.d.ts +0 -10
  640. package/dist/capability/types.d.ts.map +0 -1
  641. package/dist/capability/types.js +0 -10
  642. package/dist/capability/types.js.map +0 -1
  643. package/dist/cli/commands/agents.d.ts +0 -3
  644. package/dist/cli/commands/agents.d.ts.map +0 -1
  645. package/dist/cli/commands/agents.js +0 -62
  646. package/dist/cli/commands/agents.js.map +0 -1
  647. package/dist/cli/commands/rules.d.ts +0 -8
  648. package/dist/cli/commands/rules.d.ts.map +0 -1
  649. package/dist/cli/commands/rules.js +0 -89
  650. package/dist/cli/commands/rules.js.map +0 -1
  651. package/dist/daemon/auto-disable-scheduler.d.ts +0 -53
  652. package/dist/daemon/auto-disable-scheduler.d.ts.map +0 -1
  653. package/dist/daemon/auto-disable-scheduler.js +0 -114
  654. package/dist/daemon/auto-disable-scheduler.js.map +0 -1
  655. package/dist/daemon/handlers/pre-tool-use.d.ts +0 -30
  656. package/dist/daemon/handlers/pre-tool-use.d.ts.map +0 -1
  657. package/dist/daemon/handlers/pre-tool-use.js +0 -173
  658. package/dist/daemon/handlers/pre-tool-use.js.map +0 -1
  659. package/dist/daemon/routing-observer.d.ts +0 -42
  660. package/dist/daemon/routing-observer.d.ts.map +0 -1
  661. package/dist/daemon/routing-observer.js +0 -264
  662. package/dist/daemon/routing-observer.js.map +0 -1
  663. package/dist/daemon/routing-state.d.ts +0 -63
  664. package/dist/daemon/routing-state.d.ts.map +0 -1
  665. package/dist/daemon/routing-state.js +0 -223
  666. package/dist/daemon/routing-state.js.map +0 -1
  667. package/dist/engine/agent-router.d.ts +0 -142
  668. package/dist/engine/agent-router.d.ts.map +0 -1
  669. package/dist/engine/agent-router.js +0 -276
  670. package/dist/engine/agent-router.js.map +0 -1
  671. package/dist/engine/context-builder.d.ts +0 -23
  672. package/dist/engine/context-builder.d.ts.map +0 -1
  673. package/dist/engine/context-builder.js +0 -63
  674. package/dist/engine/context-builder.js.map +0 -1
  675. package/dist/engine/conventions/basic-security.yaml +0 -109
  676. package/dist/engine/conventions/code-quality.yaml +0 -123
  677. package/dist/engine/conventions/database-safety.yaml +0 -74
  678. package/dist/engine/conventions/dependency-safety.yaml +0 -132
  679. package/dist/engine/conventions/docker-safety.yaml +0 -69
  680. package/dist/engine/conventions/git-safety.yaml +0 -118
  681. package/dist/engine/conventions/go-best-practices.yaml +0 -84
  682. package/dist/engine/conventions/python-best-practices.yaml +0 -96
  683. package/dist/engine/conventions/react-best-practices.yaml +0 -96
  684. package/dist/engine/conventions/routing.yaml +0 -378
  685. package/dist/engine/conventions/strict-security.yaml +0 -30
  686. package/dist/engine/conventions/ts-quality.yaml +0 -49
  687. package/dist/engine/dsl/compiler.d.ts +0 -34
  688. package/dist/engine/dsl/compiler.d.ts.map +0 -1
  689. package/dist/engine/dsl/compiler.js +0 -702
  690. package/dist/engine/dsl/compiler.js.map +0 -1
  691. package/dist/engine/dsl/parser.d.ts +0 -25
  692. package/dist/engine/dsl/parser.d.ts.map +0 -1
  693. package/dist/engine/dsl/parser.js +0 -208
  694. package/dist/engine/dsl/parser.js.map +0 -1
  695. package/dist/engine/dsl/runtime.d.ts +0 -46
  696. package/dist/engine/dsl/runtime.d.ts.map +0 -1
  697. package/dist/engine/dsl/runtime.js +0 -173
  698. package/dist/engine/dsl/runtime.js.map +0 -1
  699. package/dist/engine/dsl/types.d.ts +0 -139
  700. package/dist/engine/dsl/types.d.ts.map +0 -1
  701. package/dist/engine/dsl/types.js +0 -11
  702. package/dist/engine/dsl/types.js.map +0 -1
  703. package/dist/engine/evidence-store.d.ts +0 -44
  704. package/dist/engine/evidence-store.d.ts.map +0 -1
  705. package/dist/engine/evidence-store.js +0 -109
  706. package/dist/engine/evidence-store.js.map +0 -1
  707. package/dist/engine/experiment-router.d.ts +0 -102
  708. package/dist/engine/experiment-router.d.ts.map +0 -1
  709. package/dist/engine/experiment-router.js +0 -289
  710. package/dist/engine/experiment-router.js.map +0 -1
  711. package/dist/engine/recommender.d.ts +0 -52
  712. package/dist/engine/recommender.d.ts.map +0 -1
  713. package/dist/engine/recommender.js +0 -162
  714. package/dist/engine/recommender.js.map +0 -1
  715. package/dist/engine/rule-engine.d.ts +0 -33
  716. package/dist/engine/rule-engine.d.ts.map +0 -1
  717. package/dist/engine/rule-engine.js +0 -250
  718. package/dist/engine/rule-engine.js.map +0 -1
  719. package/dist/intelligence/classifier.d.ts +0 -75
  720. package/dist/intelligence/classifier.d.ts.map +0 -1
  721. package/dist/intelligence/classifier.js +0 -352
  722. package/dist/intelligence/classifier.js.map +0 -1
  723. package/dist/intelligence/context-gatherer.d.ts +0 -101
  724. package/dist/intelligence/context-gatherer.d.ts.map +0 -1
  725. package/dist/intelligence/context-gatherer.js +0 -417
  726. package/dist/intelligence/context-gatherer.js.map +0 -1
  727. package/dist/intelligence/cot-classifier.d.ts +0 -95
  728. package/dist/intelligence/cot-classifier.d.ts.map +0 -1
  729. package/dist/intelligence/cot-classifier.js +0 -391
  730. package/dist/intelligence/cot-classifier.js.map +0 -1
  731. package/dist/intelligence/distiller.d.ts +0 -22
  732. package/dist/intelligence/distiller.d.ts.map +0 -1
  733. package/dist/intelligence/distiller.js +0 -108
  734. package/dist/intelligence/distiller.js.map +0 -1
  735. package/dist/intelligence/execution-doc-builder.d.ts +0 -151
  736. package/dist/intelligence/execution-doc-builder.d.ts.map +0 -1
  737. package/dist/intelligence/execution-doc-builder.js +0 -1018
  738. package/dist/intelligence/execution-doc-builder.js.map +0 -1
  739. package/dist/intelligence/intent-types.d.ts +0 -13
  740. package/dist/intelligence/intent-types.d.ts.map +0 -1
  741. package/dist/intelligence/intent-types.js +0 -19
  742. package/dist/intelligence/intent-types.js.map +0 -1
  743. package/dist/intelligence/multimodal-parser.d.ts +0 -105
  744. package/dist/intelligence/multimodal-parser.d.ts.map +0 -1
  745. package/dist/intelligence/multimodal-parser.js +0 -425
  746. package/dist/intelligence/multimodal-parser.js.map +0 -1
  747. package/dist/intelligence/quality-gate.d.ts +0 -45
  748. package/dist/intelligence/quality-gate.d.ts.map +0 -1
  749. package/dist/intelligence/quality-gate.js +0 -193
  750. package/dist/intelligence/quality-gate.js.map +0 -1
  751. package/dist/intelligence/task-segmenter.d.ts.map +0 -1
  752. package/dist/intelligence/task-segmenter.js.map +0 -1
  753. package/dist/web/routes/agents.d.ts +0 -7
  754. package/dist/web/routes/agents.d.ts.map +0 -1
  755. package/dist/web/routes/agents.js +0 -209
  756. package/dist/web/routes/agents.js.map +0 -1
  757. package/dist/web/routes/execution-trace.d.ts +0 -21
  758. package/dist/web/routes/execution-trace.d.ts.map +0 -1
  759. package/dist/web/routes/execution-trace.js +0 -353
  760. package/dist/web/routes/execution-trace.js.map +0 -1
  761. package/dist/web/routes/experiments.d.ts +0 -15
  762. package/dist/web/routes/experiments.d.ts.map +0 -1
  763. package/dist/web/routes/experiments.js +0 -187
  764. package/dist/web/routes/experiments.js.map +0 -1
  765. package/dist/web/routes/routing.d.ts +0 -17
  766. package/dist/web/routes/routing.d.ts.map +0 -1
  767. package/dist/web/routes/routing.js +0 -592
  768. package/dist/web/routes/routing.js.map +0 -1
  769. package/dist/web/routes/workflows.d.ts +0 -19
  770. package/dist/web/routes/workflows.d.ts.map +0 -1
  771. package/dist/web/routes/workflows.js +0 -86
  772. package/dist/web/routes/workflows.js.map +0 -1
  773. package/dist/web/static/assets/AIConfig-R5wZ3ZKT.js +0 -2
  774. package/dist/web/static/assets/AIConfig-R5wZ3ZKT.js.map +0 -1
  775. package/dist/web/static/assets/Agents-Beg34V1g.js +0 -2
  776. package/dist/web/static/assets/Agents-Beg34V1g.js.map +0 -1
  777. package/dist/web/static/assets/CodeBlock--H53gk46.js +0 -2
  778. package/dist/web/static/assets/CodeBlock--H53gk46.js.map +0 -1
  779. package/dist/web/static/assets/Dashboard-Cy1xsj1J.js +0 -2
  780. package/dist/web/static/assets/Dashboard-Cy1xsj1J.js.map +0 -1
  781. package/dist/web/static/assets/Events-mFhXl4zI.js +0 -2
  782. package/dist/web/static/assets/Events-mFhXl4zI.js.map +0 -1
  783. package/dist/web/static/assets/ExecutionTrace-DG901hLR.js +0 -3
  784. package/dist/web/static/assets/ExecutionTrace-DG901hLR.js.map +0 -1
  785. package/dist/web/static/assets/MarkdownRenderer-CCIz1MOz.js +0 -2
  786. package/dist/web/static/assets/MarkdownRenderer-CCIz1MOz.js.map +0 -1
  787. package/dist/web/static/assets/Routing-B7BFLfjh.js +0 -2
  788. package/dist/web/static/assets/Routing-B7BFLfjh.js.map +0 -1
  789. package/dist/web/static/assets/SessionDetail-BT0l4RrK.js +0 -2
  790. package/dist/web/static/assets/SessionDetail-BT0l4RrK.js.map +0 -1
  791. package/dist/web/static/assets/Sessions-C6J_HQ_u.js +0 -2
  792. package/dist/web/static/assets/Sessions-C6J_HQ_u.js.map +0 -1
  793. package/dist/web/static/assets/Skills-4DQWLaTv.js +0 -2
  794. package/dist/web/static/assets/Skills-4DQWLaTv.js.map +0 -1
  795. package/dist/web/static/assets/WorkflowDetail-zhNqUkBE.js +0 -2
  796. package/dist/web/static/assets/WorkflowDetail-zhNqUkBE.js.map +0 -1
  797. package/dist/web/static/assets/Workflows-Btvi-lGw.js +0 -2
  798. package/dist/web/static/assets/Workflows-Btvi-lGw.js.map +0 -1
  799. package/dist/web/static/assets/export-BQQZLaHV.js +0 -4
  800. package/dist/web/static/assets/export-BQQZLaHV.js.map +0 -1
  801. package/dist/web/static/assets/index-Cgr9qMtq.js +0 -3
  802. package/dist/web/static/assets/index-Cgr9qMtq.js.map +0 -1
  803. package/dist/web/static/assets/index-CngWb5gC.css +0 -1
  804. package/dist/web/static/assets/lucide-53bR2rki.js.map +0 -1
@@ -0,0 +1,423 @@
1
+ #!/usr/bin/env tsx
2
+ /**
3
+ * E2E Verification: MCP call chain → skill_invoke → DB persistence
4
+ *
5
+ * Black-box verification of the full pipeline:
6
+ *
7
+ * MCP Client (this script)
8
+ * → HTTP POST /mcp (Bearer auth)
9
+ * → StreamableHTTPServerTransport
10
+ * → McpServer.callTool('skill_invoke' / 'skill_list')
11
+ * → skillInvoke() / skillList()
12
+ * → SQLiteStorage.writeSkillInvocation()
13
+ *
14
+ * Steps:
15
+ * 1. Start a standalone WebServer (independent of system daemon)
16
+ * 2. Use MCP SDK Client to connect via HTTP
17
+ * 3. Call skill_invoke / skill_list
18
+ * 4. Query SQLite directly to verify skill_invocations table
19
+ * 5. Print verification report
20
+ *
21
+ * Isolation: temp dir for DB, swaps in test daemon.token (restored on exit).
22
+ */
23
+
24
+ import { mkdtempSync, rmSync, writeFileSync, readFileSync, existsSync, mkdirSync, unlinkSync } from 'node:fs';
25
+ import { tmpdir } from 'node:os';
26
+ import { join } from 'node:path';
27
+ import type { AddressInfo } from 'node:net';
28
+
29
+ import { SQLiteStorage } from '../src/core/storage/sqlite.js';
30
+ import { RuleEngine } from '../src/engine/rule-engine.js';
31
+ import { SkillRegistry } from '../src/skills/registry.js';
32
+ import { InvocationGuard } from '../src/skills/invocation-guard.js';
33
+ import { WebServer } from '../src/web/server.js';
34
+ import { FORGE_PATHS } from '../src/core/constants.js';
35
+ import { routingState } from '../src/daemon/routing-state.js';
36
+
37
+ import { Client } from '@modelcontextprotocol/sdk/client/index.js';
38
+ import { StreamableHTTPClientTransport } from '@modelcontextprotocol/sdk/client/streamableHttp.js';
39
+
40
+ // ─── Test infra ────────────────────────────────────────────────────────────
41
+
42
+ const TEST_TOKEN = `e2e-mcp-token-${Date.now()}`;
43
+ const TOKEN_FILE = join(FORGE_PATHS.home(), 'daemon.token');
44
+
45
+ const tmp = mkdtempSync(join(tmpdir(), 'forge-e2e-mcp-'));
46
+ const dbPath = join(tmp, 'data.db');
47
+
48
+ interface TestResult {
49
+ step: string;
50
+ pass: boolean;
51
+ details: string;
52
+ }
53
+ const results: TestResult[] = [];
54
+ function record(step: string, pass: boolean, details: string): void {
55
+ results.push({ step, pass, details });
56
+ const mark = pass ? 'PASS' : 'FAIL';
57
+ console.log(`[${mark}] ${step}: ${details}`);
58
+ }
59
+
60
+ // Backup user's daemon.token, install test token (mirrors mcp-server.test.ts pattern).
61
+ let backupToken: string | null = null;
62
+ function installTestToken(): void {
63
+ if (existsSync(TOKEN_FILE)) {
64
+ backupToken = readFileSync(TOKEN_FILE, 'utf-8');
65
+ } else {
66
+ backupToken = null;
67
+ mkdirSync(FORGE_PATHS.home(), { recursive: true });
68
+ }
69
+ writeFileSync(TOKEN_FILE, TEST_TOKEN, { mode: 0o600 });
70
+ }
71
+ function restoreTestToken(): void {
72
+ if (backupToken !== null) {
73
+ writeFileSync(TOKEN_FILE, backupToken, { mode: 0o600 });
74
+ } else if (existsSync(TOKEN_FILE)) {
75
+ unlinkSync(TOKEN_FILE);
76
+ }
77
+ }
78
+
79
+ function authHeaders(extra: Record<string, string> = {}): Record<string, string> {
80
+ return { Authorization: `Bearer ${TEST_TOKEN}`, ...extra };
81
+ }
82
+
83
+ async function newClient(port: number, headers: Record<string, string>): Promise<Client> {
84
+ const transport = new StreamableHTTPClientTransport(new URL(`http://127.0.0.1:${port}/mcp`), {
85
+ requestInit: { headers },
86
+ });
87
+ const client = new Client({ name: 'forge-e2e-client', version: '1.0.0' });
88
+ await client.connect(transport);
89
+ return client;
90
+ }
91
+
92
+ // ─── Main ──────────────────────────────────────────────────────────────────
93
+
94
+ async function main(): Promise<void> {
95
+ console.log('=== E2E MCP Link Verification ===\n');
96
+ console.log(`tmp dir: ${tmp}`);
97
+ console.log(`db: ${dbPath}`);
98
+ console.log(`token: ${TEST_TOKEN}\n`);
99
+
100
+ installTestToken();
101
+
102
+ const storage = new SQLiteStorage(dbPath);
103
+ const ruleEngine = new RuleEngine();
104
+ const skillRegistry = new SkillRegistry();
105
+ const guard = new InvocationGuard();
106
+
107
+ const allSkills = skillRegistry.getAll();
108
+ console.log(`Registry loaded ${allSkills.length} skills`);
109
+
110
+ // ── Start WebServer ──────────────────────────────────────────────────────
111
+ const server = new WebServer({
112
+ port: 0,
113
+ storage,
114
+ ruleEngine,
115
+ skillRegistry,
116
+ invocationGuard: guard,
117
+ });
118
+
119
+ // Pull the express app out and listen on a random port (matches mcp-server.test.ts).
120
+ const app = (server as unknown as { app: import('express').Application }).app;
121
+ const port: number = await new Promise<number>((resolve) => {
122
+ const handle = app.listen(0, () => {
123
+ const addr = handle.address() as AddressInfo;
124
+ // Also give server.server back so afterEach-style cleanup keeps a handle.
125
+ (server as unknown as { server: ReturnType<typeof app.listen> | null }).server = handle;
126
+ resolve(addr.port);
127
+ });
128
+ });
129
+ console.log(`Server listening on http://127.0.0.1:${port}/mcp\n`);
130
+
131
+ const activeClients: Client[] = [];
132
+
133
+ let exitCode = 0;
134
+ try {
135
+ // ── Test 1: tools/list ────────────────────────────────────────────────
136
+ {
137
+ const client = await newClient(port, authHeaders());
138
+ activeClients.push(client);
139
+ const r = await client.listTools();
140
+ const names = r.tools.map((t) => t.name);
141
+ record(
142
+ 'list tools',
143
+ names.includes('skill_invoke') && names.includes('skill_list'),
144
+ `tools: ${names.join(', ')}`,
145
+ );
146
+ }
147
+
148
+ // ── Test 2: skill_list returns skills ─────────────────────────────────
149
+ {
150
+ const client = await newClient(port, authHeaders());
151
+ activeClients.push(client);
152
+ const r = await client.callTool({ name: 'skill_list', arguments: {} });
153
+ const content = r.content as Array<{ type: string; text: string }>;
154
+ const parsed = JSON.parse(content[0].text) as { skills: unknown[]; total: number };
155
+ record(
156
+ 'skill_list returns skills',
157
+ parsed.total > 0 && parsed.skills.length === parsed.total,
158
+ `${parsed.total} skills returned`,
159
+ );
160
+ }
161
+
162
+ // ── Test 3: skill_invoke base call (with explicit X-Forge-* headers) ──
163
+ const baseSession = 'e2e-session-1';
164
+ const baseRouteId = 'e2e-route-1';
165
+ const baseAgent = 'e2e-agent-bmad';
166
+ let invokeContentLen = 0;
167
+ {
168
+ const client = await newClient(
169
+ port,
170
+ authHeaders({
171
+ 'X-Forge-Session-Id': baseSession,
172
+ 'X-Forge-Route-Request-Id': baseRouteId,
173
+ 'X-Forge-Agent-Id': baseAgent,
174
+ }),
175
+ );
176
+ activeClients.push(client);
177
+ const r = await client.callTool({
178
+ name: 'skill_invoke',
179
+ arguments: { skill_id: 'official-tdd', reason: 'bmad/implement:e2e-test' },
180
+ });
181
+ const content = r.content as Array<{ type: string; text: string }>;
182
+ invokeContentLen = content[0]?.text?.length ?? 0;
183
+ record(
184
+ 'skill_invoke base',
185
+ !r.isError && invokeContentLen > 100 && content[0].text.startsWith('# '),
186
+ `content length ${invokeContentLen}`,
187
+ );
188
+ }
189
+
190
+ // ── Test 4: DB row inserted with all linkage fields ───────────────────
191
+ {
192
+ const rows = storage.querySkillInvocations({ session_id: baseSession });
193
+ const row = rows[0];
194
+ record('DB row inserted', rows.length === 1, `rows count = ${rows.length}`);
195
+ record(
196
+ 'workflow parsed',
197
+ row?.workflow === 'bmad' && row?.phase === 'implement',
198
+ `workflow=${row?.workflow}, phase=${row?.phase}`,
199
+ );
200
+ record(
201
+ 'feature_slug parsed',
202
+ row?.feature_slug === 'e2e-test',
203
+ `feature_slug=${row?.feature_slug}`,
204
+ );
205
+ record(
206
+ 'session/route/agent linkage',
207
+ row?.session_id === baseSession &&
208
+ row?.route_request_id === baseRouteId &&
209
+ row?.agent_id === baseAgent,
210
+ `session=${row?.session_id}, route=${row?.route_request_id}, agent=${row?.agent_id}`,
211
+ );
212
+ record(
213
+ 'invocation_type=dynamic + success=1',
214
+ row?.invocation_type === 'dynamic' && row?.success === 1,
215
+ `invocation_type=${row?.invocation_type}, success=${row?.success}`,
216
+ );
217
+ }
218
+
219
+ // ── Test 5: Multi-phase aggregation via queryWorkflowProgress ─────────
220
+ // Use a fresh session to avoid the per-session idempotency guard, and
221
+ // a single shared route_request_id so all phases group together.
222
+ const multiSession = 'e2e-session-multi';
223
+ const multiRouteId = 'e2e-route-multi';
224
+ const multiAgent = 'e2e-agent-multi';
225
+ {
226
+ const phaseSkills: Array<[string, string]> = [
227
+ ['analyze', 'official-spec-driven-design'],
228
+ ['design', 'official-architecture-decision'],
229
+ ['implement', 'official-tdd'],
230
+ ];
231
+ for (const [phase, skill] of phaseSkills) {
232
+ const c = await newClient(
233
+ port,
234
+ authHeaders({
235
+ 'X-Forge-Session-Id': multiSession,
236
+ 'X-Forge-Route-Request-Id': multiRouteId,
237
+ 'X-Forge-Agent-Id': multiAgent,
238
+ }),
239
+ );
240
+ activeClients.push(c);
241
+ await c.callTool({
242
+ name: 'skill_invoke',
243
+ arguments: { skill_id: skill, reason: `bmad/${phase}:multi-phase-test` },
244
+ });
245
+ }
246
+ const progress = storage.queryWorkflowProgress({ session_id: multiSession });
247
+ const ok =
248
+ progress.length === 1 &&
249
+ progress[0].workflow === 'bmad' &&
250
+ progress[0].route_request_id === multiRouteId &&
251
+ progress[0].phases_completed.length === 3 &&
252
+ progress[0].phases_completed.includes('analyze') &&
253
+ progress[0].phases_completed.includes('design') &&
254
+ progress[0].phases_completed.includes('implement') &&
255
+ progress[0].feature_slug === 'multi-phase-test';
256
+ record(
257
+ 'multi-phase aggregation',
258
+ ok,
259
+ `routes=${progress.length}, phases=[${progress[0]?.phases_completed.join(', ')}], feature=${progress[0]?.feature_slug}`,
260
+ );
261
+ }
262
+
263
+ // ── Test 6: Error path — non-existent skill records success=0 ────────
264
+ const errSession = 'e2e-session-error';
265
+ {
266
+ const client = await newClient(
267
+ port,
268
+ authHeaders({ 'X-Forge-Session-Id': errSession }),
269
+ );
270
+ activeClients.push(client);
271
+ const r = await client.callTool({
272
+ name: 'skill_invoke',
273
+ arguments: { skill_id: 'nonexistent-skill-xyz', reason: 'bmad/analyze' },
274
+ });
275
+ const isErrorFlag = r.isError === true;
276
+ const rows = storage.querySkillInvocations({ session_id: errSession });
277
+ const row = rows[0];
278
+ record(
279
+ 'error skill returns isError + records success=0',
280
+ isErrorFlag &&
281
+ rows.length === 1 &&
282
+ row.success === 0 &&
283
+ (row.error?.includes('Skill not found') ?? false),
284
+ `isError=${isErrorFlag}, rows=${rows.length}, success=${row?.success}, error="${row?.error?.slice(0, 60) ?? ''}"`,
285
+ );
286
+ }
287
+
288
+ // ── Test 7: Guard blocks repeated calls (idempotency) ─────────────────
289
+ const dupSession = 'e2e-session-dup';
290
+ {
291
+ // First invocation succeeds
292
+ const c1 = await newClient(
293
+ port,
294
+ authHeaders({ 'X-Forge-Session-Id': dupSession }),
295
+ );
296
+ activeClients.push(c1);
297
+ await c1.callTool({
298
+ name: 'skill_invoke',
299
+ arguments: { skill_id: 'official-tdd', reason: 'first call' },
300
+ });
301
+
302
+ // Second invocation of same skill in same session should be blocked
303
+ const c2 = await newClient(
304
+ port,
305
+ authHeaders({ 'X-Forge-Session-Id': dupSession }),
306
+ );
307
+ activeClients.push(c2);
308
+ const r2 = await c2.callTool({
309
+ name: 'skill_invoke',
310
+ arguments: { skill_id: 'official-tdd', reason: 'duplicate' },
311
+ });
312
+
313
+ const rows = storage.querySkillInvocations({ session_id: dupSession });
314
+ // Rows ordered by timestamp DESC: rows[0] is the duplicate (Blocked), rows[1] is the first
315
+ const blockedRow = rows.find((r) => r.success === 0 && r.error?.includes('Blocked'));
316
+ const successRow = rows.find((r) => r.success === 1);
317
+ record(
318
+ 'guard blocks repeated calls',
319
+ r2.isError === true &&
320
+ rows.length === 2 &&
321
+ blockedRow !== undefined &&
322
+ successRow !== undefined,
323
+ `2 rows; isError=${r2.isError}; success=${successRow?.success}, blocked.error="${blockedRow?.error?.slice(0, 60) ?? ''}"`,
324
+ );
325
+ }
326
+
327
+ // ── Test 8: Fallback to routingState when X-Forge-Session-Id missing ──
328
+ const fallbackSession = 'e2e-session-fallback';
329
+ const fallbackRouteId = 'e2e-route-fallback';
330
+ {
331
+ routingState.setRouting(fallbackSession, {
332
+ agentName: 'fallback-agent',
333
+ agentDescription: 'desc',
334
+ timestamp: Date.now(),
335
+ routeRequestId: fallbackRouteId,
336
+ });
337
+ // No X-Forge-* headers — server should fall back to routingState.getMostRecent().
338
+ const client = await newClient(port, authHeaders());
339
+ activeClients.push(client);
340
+ await client.callTool({
341
+ name: 'skill_invoke',
342
+ arguments: { skill_id: 'official-refactor', reason: 'fallback test' },
343
+ });
344
+
345
+ const rows = storage.querySkillInvocations({ session_id: fallbackSession });
346
+ const row = rows[0];
347
+ record(
348
+ 'routingState fallback (no headers)',
349
+ rows.length === 1 &&
350
+ row?.route_request_id === fallbackRouteId &&
351
+ row?.agent_id === 'fallback-agent',
352
+ `rows=${rows.length}, route=${row?.route_request_id}, agent=${row?.agent_id}`,
353
+ );
354
+ routingState.clearRouting(fallbackSession);
355
+ }
356
+
357
+ // ── Test 9: mcp-anon-* fallback when neither header nor routingState ──
358
+ {
359
+ // routingState should be empty by now (we cleared the fallback above and
360
+ // no daemon is feeding it). Issue a call and look for an mcp-anon-* row.
361
+ const before = Date.now();
362
+ const client = await newClient(port, authHeaders());
363
+ activeClients.push(client);
364
+ await client.callTool({
365
+ name: 'skill_invoke',
366
+ arguments: { skill_id: 'official-debug', reason: 'anon test' },
367
+ });
368
+
369
+ // Fetch all rows and filter for mcp-anon-* with timestamp >= before.
370
+ // querySkillInvocations is keyed by session_id so we use a wide query.
371
+ const allRows = storage.querySkillInvocations({});
372
+ const anonRow = allRows.find(
373
+ (r) => r.session_id.startsWith('mcp-anon-') && r.timestamp >= before,
374
+ );
375
+ record(
376
+ 'mcp-anon-* fallback (no header, no routingState)',
377
+ anonRow !== undefined && anonRow.success === 1,
378
+ `session_id=${anonRow?.session_id ?? '<none>'}`,
379
+ );
380
+ }
381
+ } catch (err) {
382
+ console.error('\n[FATAL] Unhandled error during tests:', err);
383
+ record('e2e-script', false, err instanceof Error ? err.message : String(err));
384
+ exitCode = 1;
385
+ } finally {
386
+ // Cleanup: clients first, then HTTP server, then storage, then tmp dir.
387
+ for (const c of activeClients) {
388
+ try {
389
+ await c.close();
390
+ } catch {
391
+ /* ignore */
392
+ }
393
+ }
394
+ const httpServer = (server as unknown as { server: import('node:net').Server | null }).server;
395
+ if (httpServer) {
396
+ (httpServer as unknown as { closeAllConnections?: () => void }).closeAllConnections?.();
397
+ await new Promise<void>((res) => httpServer.close(() => res()));
398
+ }
399
+ storage.close();
400
+ rmSync(tmp, { recursive: true, force: true });
401
+ restoreTestToken();
402
+ }
403
+
404
+ // ── Final report ─────────────────────────────────────────────────────────
405
+ const passed = results.filter((r) => r.pass).length;
406
+ const total = results.length;
407
+ console.log(`\n=== Result: ${passed}/${total} passed ===`);
408
+ if (passed !== total) {
409
+ console.log('\nFailures:');
410
+ for (const r of results.filter((x) => !x.pass)) {
411
+ console.log(` - ${r.step}: ${r.details}`);
412
+ }
413
+ }
414
+
415
+ process.exit(passed === total && exitCode === 0 ? 0 : 1);
416
+ }
417
+
418
+ main().catch((err) => {
419
+ console.error('E2E failed:', err);
420
+ restoreTestToken();
421
+ rmSync(tmp, { recursive: true, force: true });
422
+ process.exit(1);
423
+ });
@@ -0,0 +1,253 @@
1
+ #!/usr/bin/env tsx
2
+ /**
3
+ * E2E 验证:6 个方法论 Agent 的 prompt 质量
4
+ */
5
+
6
+ import { OFFICIAL_AGENTS, findOfficialAgent } from '../src/agents/index.js';
7
+ import { METHODOLOGY_PRESETS } from '../src/agents/methodologies/presets.js';
8
+ import { getPhase, getAllPhases, buildMethodologyAgent } from '../src/agents/methodologies/agent-builder.js';
9
+ import { OFFICIAL_SKILL_KEYWORDS } from '../src/skills/official-skills.js';
10
+ import { parseWorkflowMeta } from '../src/skills/tools/skill-invoke.js';
11
+
12
+ interface Issue {
13
+ agent: string;
14
+ severity: 'error' | 'warning' | 'info';
15
+ message: string;
16
+ }
17
+
18
+ const issues: Issue[] = [];
19
+
20
+ function record(agent: string, severity: Issue['severity'], message: string) {
21
+ issues.push({ agent, severity, message });
22
+ }
23
+
24
+ const PRESET_IDS = [
25
+ 'bmad-feature-full',
26
+ 'bmad-feature-quick',
27
+ 'bmad-spec-only',
28
+ 'harness-debug-full',
29
+ 'harness-hotfix',
30
+ 'harness-investigate',
31
+ ];
32
+
33
+ const VALID_SKILLS = new Set(Object.keys(OFFICIAL_SKILL_KEYWORDS));
34
+
35
+ console.log('=== 方法论 Agent Prompt 质量 E2E 验证 ===\n');
36
+
37
+ // ── Check 1: 6 个 preset 都注册成功 ──────────────────────
38
+ console.log('-- Check 1: Preset 注册检查 --');
39
+ for (const id of PRESET_IDS) {
40
+ const agent = findOfficialAgent(id);
41
+ if (!agent) {
42
+ record(id, 'error', '未注册到 OFFICIAL_AGENTS');
43
+ } else {
44
+ console.log(`✓ ${id}: 已注册 (version=${agent.version}, skills=${agent.skills?.length ?? 0})`);
45
+ }
46
+ }
47
+
48
+ // ── Check 2: 每个 preset 的 phase 列表合法 ────────────────
49
+ console.log('\n-- Check 2: Phase 列表合法性 --');
50
+ for (const preset of METHODOLOGY_PRESETS) {
51
+ for (const phaseId of preset.phases) {
52
+ const phase = getPhase(phaseId);
53
+ if (!phase) {
54
+ record(preset.id, 'error', `phase ${phaseId} 不存在`);
55
+ }
56
+ }
57
+ }
58
+ console.log(`(已检查 ${METHODOLOGY_PRESETS.length} 个 preset 的 phase 列表)`);
59
+
60
+ // ── Check 3: prompt 内容质量 ─────────────────────────────
61
+ console.log('\n-- Check 3: prompt 内容质量 --');
62
+ for (const id of PRESET_IDS) {
63
+ const agent = findOfficialAgent(id);
64
+ if (!agent) continue;
65
+ const content = agent.content;
66
+
67
+ // 必须包含
68
+ if (!content.includes('软性建议') && !content.includes('软性')) {
69
+ record(id, 'warning', 'prompt 未明确"软性建议"语义');
70
+ }
71
+ if (content.length < 500) {
72
+ record(id, 'warning', `prompt 过短(${content.length} 字符),可能内容缺失`);
73
+ }
74
+
75
+ // 每个 phase 都该出现在 prompt 里
76
+ const preset = METHODOLOGY_PRESETS.find(p => p.id === id);
77
+ if (preset) {
78
+ for (const phaseId of preset.phases) {
79
+ const phase = getPhase(phaseId);
80
+ if (phase && !content.includes(`Phase: ${phase.id}`)) {
81
+ record(id, 'error', `prompt 缺少 "Phase: ${phase.id}" 标识`);
82
+ }
83
+ }
84
+ }
85
+ console.log(` ${id}: ${content.length} 字符`);
86
+ }
87
+
88
+ // ── Check 4: reason 格式约定一致 ─────────────────────────
89
+ // Prompt 里的 reason 用 <feature-slug> / <bug-slug> 当模板占位符。
90
+ // 运行时 LLM 会替换成实际 slug,再传给 skill_invoke。
91
+ // 因此校验时把占位符替换成示例 slug,再过 parseWorkflowMeta。
92
+ console.log('\n-- Check 4: reason 格式 --');
93
+ const TEMPLATE_PLACEHOLDERS = /<(feature-slug|bug-slug)>/g;
94
+ const PLACEHOLDER_SUBSTITUTE = 'example-slug';
95
+
96
+ for (const id of PRESET_IDS) {
97
+ const agent = findOfficialAgent(id);
98
+ if (!agent) continue;
99
+
100
+ // 提取所有 reason: "..." 出现
101
+ const reasonMatches = [...agent.content.matchAll(/reason:\s*"([^"]+)"/g)];
102
+ for (const match of reasonMatches) {
103
+ const reasonTemplate = match[1];
104
+ const reasonResolved = reasonTemplate.replace(TEMPLATE_PLACEHOLDERS, PLACEHOLDER_SUBSTITUTE);
105
+
106
+ // 子检查 4a:原始模板必须含 <feature-slug> 或 <bug-slug>(强制带 slug 元数据)
107
+ const hasSlugPlaceholder = reasonTemplate.includes('<feature-slug>') || reasonTemplate.includes('<bug-slug>');
108
+ if (!hasSlugPlaceholder) {
109
+ record(id, 'info', `reason "${reasonTemplate}" 缺少 <feature-slug>/<bug-slug> 占位符(仍可工作,但 slug 元数据会缺失)`);
110
+ }
111
+
112
+ // 子检查 4b:占位符替换后必须能被解析
113
+ const meta = parseWorkflowMeta(reasonResolved);
114
+ if (!meta.workflow) {
115
+ record(id, 'error', `reason "${reasonTemplate}" 替换占位符后 ("${reasonResolved}") 仍无法被 parseWorkflowMeta 解析`);
116
+ } else if (meta.workflow !== 'bmad' && meta.workflow !== 'harness') {
117
+ record(id, 'error', `reason "${reasonTemplate}" workflow=${meta.workflow}(应为 bmad/harness)`);
118
+ } else {
119
+ // 子检查 4c:解析出来的 workflow 应该跟 preset 主线一致
120
+ const preset = METHODOLOGY_PRESETS.find(p => p.id === id);
121
+ const presetWorkflow = id.startsWith('bmad-') ? 'bmad' : id.startsWith('harness-') ? 'harness' : null;
122
+ if (presetWorkflow && meta.workflow !== presetWorkflow) {
123
+ record(id, 'warning', `reason "${reasonTemplate}" workflow=${meta.workflow}(preset 主线=${presetWorkflow})`);
124
+ }
125
+ }
126
+ }
127
+
128
+ if (reasonMatches.length === 0) {
129
+ const preset = METHODOLOGY_PRESETS.find(p => p.id === id);
130
+ const needsSkill = preset?.phases.some(p => getPhase(p)?.skillId);
131
+ if (needsSkill) {
132
+ record(id, 'warning', '至少有 phase 需要调用 skill 但 prompt 里找不到 reason: "..." 字面量');
133
+ }
134
+ }
135
+ console.log(` ${id}: 发现 ${reasonMatches.length} 个 reason 引用`);
136
+ }
137
+
138
+ // ── Check 5: 所有引用的 skill 都真实存在 ─────────────────
139
+ console.log('\n-- Check 5: Skill 引用有效性 --');
140
+ let phaseSkillCount = 0;
141
+ for (const phase of getAllPhases()) {
142
+ if (phase.skillId) {
143
+ phaseSkillCount++;
144
+ if (!VALID_SKILLS.has(phase.skillId)) {
145
+ record(`phase:${phase.workflow}/${phase.id}`, 'error',
146
+ `skillId "${phase.skillId}" 不在 OFFICIAL_SKILL_KEYWORDS`);
147
+ }
148
+ }
149
+ }
150
+ console.log(` 共 ${getAllPhases().length} 个 phase,其中 ${phaseSkillCount} 个引用 skill`);
151
+
152
+ // ── Check 6: artifact 路径模板合理 ───────────────────────
153
+ console.log('\n-- Check 6: artifact 路径模板 --');
154
+ for (const phase of getAllPhases()) {
155
+ if (phase.artifactHint) {
156
+ // 检查路径是否在 docs/ 下(按用户决策)
157
+ if (!phase.artifactHint.startsWith('docs/') && !phase.artifactHint.startsWith('src/') && !phase.artifactHint.startsWith('tests/')) {
158
+ record(`phase:${phase.workflow}/${phase.id}`, 'warning',
159
+ `artifact 路径 "${phase.artifactHint}" 不在 docs/src/tests 下`);
160
+ }
161
+ // 检查是否含 <feature-slug> 或 <bug-slug> 占位符
162
+ if (!phase.artifactHint.includes('<')) {
163
+ record(`phase:${phase.workflow}/${phase.id}`, 'info',
164
+ `artifact 路径 "${phase.artifactHint}" 没有占位符(不可变路径?)`);
165
+ }
166
+ }
167
+ }
168
+ console.log(` (已检查所有 phase 的 artifactHint)`);
169
+
170
+ // ── Check 7: 跨方法论 Phase 没混淆 ───────────────────────
171
+ console.log('\n-- Check 7: 跨方法论组合 --');
172
+ for (const preset of METHODOLOGY_PRESETS) {
173
+ const workflows = new Set(preset.phases.map(p => getPhase(p)?.workflow));
174
+ // official preset 应该是单方法论的
175
+ if (workflows.size > 1) {
176
+ record(preset.id, 'info',
177
+ `跨方法论组合:${[...workflows].join(', ')}(official 预设建议单方法论)`);
178
+ }
179
+ }
180
+ console.log(` (已检查 ${METHODOLOGY_PRESETS.length} 个 preset 的 workflow 一致性)`);
181
+
182
+ // ── Check 8: Phase 的 skillId 跟 workflow 一致性 ──────────
183
+ console.log('\n-- Check 8: Skill 与 workflow 一致性 --');
184
+ for (const phase of getAllPhases()) {
185
+ if (phase.skillId) {
186
+ // 简单检查:bmad phase 的 skill 不该是纯 harness 类的
187
+ // (这是 heuristic,不是硬规则)
188
+ if (phase.workflow === 'bmad' && phase.skillId.includes('harness')) {
189
+ record(`phase:${phase.workflow}/${phase.id}`, 'warning',
190
+ `bmad phase 引用 harness skill "${phase.skillId}"`);
191
+ }
192
+ }
193
+ }
194
+ console.log(` (已检查所有 phase 的 skill 归属)`);
195
+
196
+ // ── Check 9: buildMethodologyAgent 是确定性的 ────────────
197
+ console.log('\n-- Check 9: buildMethodologyAgent 确定性 --');
198
+ const preset = METHODOLOGY_PRESETS[0];
199
+ const built1 = buildMethodologyAgent(preset);
200
+ const built2 = buildMethodologyAgent(preset);
201
+ if (built1.content !== built2.content) {
202
+ record(preset.id, 'error', 'buildMethodologyAgent 输出不确定');
203
+ } else {
204
+ console.log(` ✓ ${preset.id} 多次构建结果一致`);
205
+ }
206
+
207
+ // ── Check 10: harness 系列的 slug 占位符在整份 prompt 里保持一致 ──
208
+ // agent-builder 的"软性建议"行硬编码 `<feature-slug>`,但 harness phase body
209
+ // 全部使用 `<bug-slug>`。同一份 prompt 混用两种 slug 名会误导 LLM。
210
+ console.log('\n-- Check 10: slug 占位符在 prompt 内部一致 --');
211
+ for (const id of PRESET_IDS) {
212
+ const agent = findOfficialAgent(id);
213
+ if (!agent) continue;
214
+ const content = agent.content;
215
+ const hasFeatureSlug = content.includes('<feature-slug>');
216
+ const hasBugSlug = content.includes('<bug-slug>');
217
+ const isHarness = id.startsWith('harness-');
218
+ const isBmad = id.startsWith('bmad-');
219
+
220
+ if (isHarness && hasFeatureSlug) {
221
+ record(id, 'warning',
222
+ `harness preset 里出现 <feature-slug>(应为 <bug-slug>)— 可能来自 agent-builder 模板硬编码`);
223
+ }
224
+ if (isBmad && hasBugSlug) {
225
+ record(id, 'warning',
226
+ `bmad preset 里出现 <bug-slug>(应为 <feature-slug>)`);
227
+ }
228
+ if (hasFeatureSlug && hasBugSlug) {
229
+ record(id, 'warning',
230
+ `prompt 同时出现 <feature-slug> 和 <bug-slug>,会让 LLM 困惑`);
231
+ }
232
+ }
233
+
234
+ // ── 输出报告 ─────────────────────────────────────────────
235
+ const errors = issues.filter(i => i.severity === 'error');
236
+ const warnings = issues.filter(i => i.severity === 'warning');
237
+ const infos = issues.filter(i => i.severity === 'info');
238
+
239
+ console.log('\n=== 问题汇总 ===');
240
+ console.log(`Errors: ${errors.length}`);
241
+ console.log(`Warnings: ${warnings.length}`);
242
+ console.log(`Infos: ${infos.length}`);
243
+
244
+ if (issues.length > 0) {
245
+ console.log('\n=== 问题详情 ===');
246
+ for (const issue of issues) {
247
+ const icon = issue.severity === 'error' ? '✗' : issue.severity === 'warning' ? '⚠' : 'ℹ';
248
+ console.log(`${icon} [${issue.agent}] ${issue.message}`);
249
+ }
250
+ }
251
+
252
+ console.log(`\n=== Result: ${errors.length === 0 ? 'PASS' : 'FAIL'} ===`);
253
+ process.exit(errors.length === 0 ? 0 : 1);