@winspan/claude-forge 8.41.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 +23 -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 +60 -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/web/routes/ai.d.ts.map +1 -1
  223. package/dist/web/routes/ai.js +16 -22
  224. package/dist/web/routes/ai.js.map +1 -1
  225. package/dist/web/routes/drift.d.ts +10 -0
  226. package/dist/web/routes/drift.d.ts.map +1 -0
  227. package/dist/web/routes/drift.js +21 -0
  228. package/dist/web/routes/drift.js.map +1 -0
  229. package/dist/web/routes/error-handler.d.ts +43 -0
  230. package/dist/web/routes/error-handler.d.ts.map +1 -0
  231. package/dist/web/routes/error-handler.js +99 -0
  232. package/dist/web/routes/error-handler.js.map +1 -0
  233. package/dist/web/routes/insights.d.ts +9 -0
  234. package/dist/web/routes/insights.d.ts.map +1 -0
  235. package/dist/web/routes/insights.js +34 -0
  236. package/dist/web/routes/insights.js.map +1 -0
  237. package/dist/web/routes/patch.js +2 -2
  238. package/dist/web/routes/patch.js.map +1 -1
  239. package/dist/web/routes/reports.d.ts +10 -0
  240. package/dist/web/routes/reports.d.ts.map +1 -0
  241. package/dist/web/routes/reports.js +27 -0
  242. package/dist/web/routes/reports.js.map +1 -0
  243. package/dist/web/routes/rules.d.ts +10 -3
  244. package/dist/web/routes/rules.d.ts.map +1 -1
  245. package/dist/web/routes/rules.js +80 -95
  246. package/dist/web/routes/rules.js.map +1 -1
  247. package/dist/web/routes/sessions.d.ts +1 -2
  248. package/dist/web/routes/sessions.d.ts.map +1 -1
  249. package/dist/web/routes/sessions.js +27 -39
  250. package/dist/web/routes/sessions.js.map +1 -1
  251. package/dist/web/routes/skill-stats.d.ts.map +1 -1
  252. package/dist/web/routes/skill-stats.js +38 -0
  253. package/dist/web/routes/skill-stats.js.map +1 -1
  254. package/dist/web/routes/skills.d.ts.map +1 -1
  255. package/dist/web/routes/skills.js +34 -0
  256. package/dist/web/routes/skills.js.map +1 -1
  257. package/dist/web/routes/stats.d.ts +7 -0
  258. package/dist/web/routes/stats.d.ts.map +1 -0
  259. package/dist/web/routes/stats.js +44 -0
  260. package/dist/web/routes/stats.js.map +1 -0
  261. package/dist/web/routes/status.js +1 -1
  262. package/dist/web/routes/status.js.map +1 -1
  263. package/dist/web/routes/tasks.d.ts +4 -0
  264. package/dist/web/routes/tasks.d.ts.map +1 -0
  265. package/dist/web/routes/tasks.js +181 -0
  266. package/dist/web/routes/tasks.js.map +1 -0
  267. package/dist/web/routes/trace.d.ts +10 -0
  268. package/dist/web/routes/trace.d.ts.map +1 -0
  269. package/dist/web/routes/trace.js +123 -0
  270. package/dist/web/routes/trace.js.map +1 -0
  271. package/dist/web/routes/types.d.ts +1 -14
  272. package/dist/web/routes/types.d.ts.map +1 -1
  273. package/dist/web/routes/types.js +8 -17
  274. package/dist/web/routes/types.js.map +1 -1
  275. package/dist/web/server.d.ts +1 -9
  276. package/dist/web/server.d.ts.map +1 -1
  277. package/dist/web/server.js +28 -28
  278. package/dist/web/server.js.map +1 -1
  279. package/dist/web/static/assets/AIConfig-BQCAQE9D.js +2 -0
  280. package/dist/web/static/assets/AIConfig-BQCAQE9D.js.map +1 -0
  281. package/dist/web/static/assets/Dashboard-D7Bo6Kan.js +2 -0
  282. package/dist/web/static/assets/Dashboard-D7Bo6Kan.js.map +1 -0
  283. package/dist/web/static/assets/{Drawer-DcU3ln98.js → Drawer-BeHRQxUS.js} +2 -2
  284. package/dist/web/static/assets/{Drawer-DcU3ln98.js.map → Drawer-BeHRQxUS.js.map} +1 -1
  285. package/dist/web/static/assets/Events-K_tCY2ti.js +2 -0
  286. package/dist/web/static/assets/Events-K_tCY2ti.js.map +1 -0
  287. package/dist/web/static/assets/Reports-BJCmBnc_.js +2 -0
  288. package/dist/web/static/assets/Reports-BJCmBnc_.js.map +1 -0
  289. package/dist/web/static/assets/SearchInput-BX2KhMkw.js +2 -0
  290. package/dist/web/static/assets/SearchInput-BX2KhMkw.js.map +1 -0
  291. package/dist/web/static/assets/SessionDetail-Bkr-kC7V.js +2 -0
  292. package/dist/web/static/assets/SessionDetail-Bkr-kC7V.js.map +1 -0
  293. package/dist/web/static/assets/Sessions-Chx9OCLH.js +2 -0
  294. package/dist/web/static/assets/Sessions-Chx9OCLH.js.map +1 -0
  295. package/dist/web/static/assets/Skills-O0GT1i7m.js +2 -0
  296. package/dist/web/static/assets/Skills-O0GT1i7m.js.map +1 -0
  297. package/dist/web/static/assets/TaskDetail-5SR8zGzv.js +2 -0
  298. package/dist/web/static/assets/TaskDetail-5SR8zGzv.js.map +1 -0
  299. package/dist/web/static/assets/Tasks-DCgDqvOZ.js +2 -0
  300. package/dist/web/static/assets/Tasks-DCgDqvOZ.js.map +1 -0
  301. package/dist/web/static/assets/export-L_VBD2p1.js +4 -0
  302. package/dist/web/static/assets/export-L_VBD2p1.js.map +1 -0
  303. package/dist/web/static/assets/index-D8AKj26b.css +1 -0
  304. package/dist/web/static/assets/index-DxIbmNmr.js +3 -0
  305. package/dist/web/static/assets/index-DxIbmNmr.js.map +1 -0
  306. package/dist/web/static/assets/{lucide-53bR2rki.js → lucide-fJlPI3H7.js} +68 -38
  307. package/dist/web/static/assets/lucide-fJlPI3H7.js.map +1 -0
  308. package/dist/web/static/assets/time-Bxuk0M-C.js +2 -0
  309. package/dist/web/static/assets/time-Bxuk0M-C.js.map +1 -0
  310. package/dist/web/static/index.html +3 -3
  311. package/docs/concurrent-agents.md +129 -0
  312. package/docs/design/architecture-review-20260516.md +232 -0
  313. package/docs/design/fix-skills-data-and-set-leak-spec-20260516-1300.md +219 -0
  314. package/docs/design/hook-failure-queue-spec-20260516-1530.md +204 -0
  315. package/docs/design/refactor-phase1-spec-20260515-1600.md +543 -0
  316. package/docs/design/refactor-phase2-spec-20260515-1700.md +424 -0
  317. package/docs/design/tasks-list-filter-pagination-spec-20260518-0930.md +208 -0
  318. package/docs/implementation/fix-skills-data-and-set-leak-changelog-20260516-1300.md +104 -0
  319. package/docs/implementation/hook-failure-queue-changelog-20260516-1530.md +196 -0
  320. package/docs/implementation/hotfix-daemon-event-reject-20260516-1430.md +56 -0
  321. package/docs/implementation/refactor-phase1-changelog-20260515-1630.md +354 -0
  322. package/docs/implementation/refactor-phase2-changelog-20260515-1705.md +421 -0
  323. package/docs/implementation/tasks-list-filter-pagination-changelog-20260518-0930.md +72 -0
  324. package/docs/reviews/claudemd-template-sync.md +54 -0
  325. package/docs/reviews/tasks-filter-pagination.md +80 -0
  326. package/docs/ruflo-learning-strategy.md +322 -0
  327. package/docs/skills-deduplication-analysis.md +83 -0
  328. package/docs/skills-multiformat-support.md +177 -0
  329. package/docs/skills-third-party.md +183 -0
  330. package/docs/testing/tasks-filter-pagination-test-report.md +86 -0
  331. package/forge +321 -0
  332. package/package.json +28 -62
  333. package/playwright.config.ts +40 -0
  334. package/scripts/demo-v2.ts +91 -0
  335. package/scripts/dev-daemon.sh +232 -0
  336. package/scripts/dev-web.ts +109 -0
  337. package/scripts/e2e-mcp-link.ts +423 -0
  338. package/scripts/e2e-methodology-quality.ts +253 -0
  339. package/scripts/e2e-routing.ts +456 -0
  340. package/scripts/e2e-user-methodology.ts +326 -0
  341. package/scripts/e2e-web-workflows.ts +299 -0
  342. package/scripts/migrate-legacy-to-dynamic.sql +108 -0
  343. package/scripts/regenerate-execution-docs.ts +116 -0
  344. package/scripts/sync-agent-skills.ts +193 -0
  345. package/scripts/test-hook.sh +71 -0
  346. package/scripts/verify-skill-loading.ts +62 -0
  347. package/src/claudemd/claudemd-generator.ts +777 -0
  348. package/src/claudemd/convention-extractor.ts +69 -0
  349. package/src/claudemd/index.ts +35 -0
  350. package/src/claudemd/persona-manager.ts +88 -0
  351. package/src/claudemd/resume-manager.ts +236 -0
  352. package/src/claudemd/tech-detector.ts +220 -0
  353. package/src/cli/commands/claudemd.ts +84 -0
  354. package/src/cli/commands/config.ts +46 -0
  355. package/src/cli/commands/daemon.ts +310 -0
  356. package/src/cli/commands/executions.ts +114 -0
  357. package/src/cli/commands/init.ts +204 -0
  358. package/src/cli/commands/logs.ts +181 -0
  359. package/src/cli/commands/mcp.ts +244 -0
  360. package/src/cli/commands/menu.ts +356 -0
  361. package/src/cli/commands/skills.ts +185 -0
  362. package/src/cli/commands/stats.ts +74 -0
  363. package/src/cli/commands/status.ts +69 -0
  364. package/src/cli/commands/template.ts +77 -0
  365. package/src/cli/commands/trace.ts +164 -0
  366. package/src/cli/index.ts +42 -0
  367. package/src/cli/init/hook-manager.ts +132 -0
  368. package/src/core/ai/provider.ts +308 -0
  369. package/src/core/ai/types.ts +51 -0
  370. package/src/core/config.ts +124 -0
  371. package/src/core/constants.ts +45 -0
  372. package/src/core/queue/index.ts +193 -0
  373. package/src/core/storage/base.ts +226 -0
  374. package/src/core/storage/events.ts +255 -0
  375. package/src/core/storage/injections.ts +78 -0
  376. package/src/core/storage/maintenance.ts +59 -0
  377. package/src/core/storage/migrations/002_add_skill_tracking.sql +6 -0
  378. package/src/core/storage/migrations/003_add_skill_invocations.sql +23 -0
  379. package/src/core/storage/performance-indexes.sql +23 -0
  380. package/src/core/storage/routing.ts +194 -0
  381. package/src/core/storage/rows.ts +112 -0
  382. package/src/core/storage/schema.sql +214 -0
  383. package/src/core/storage/sessions.ts +104 -0
  384. package/src/core/storage/skills.ts +164 -0
  385. package/src/core/storage/sqlite.ts +194 -0
  386. package/src/core/storage/tasks.ts +170 -0
  387. package/src/core/storage/token-usage.ts +93 -0
  388. package/src/core/types.ts +154 -0
  389. package/src/core/utils/error-handler.ts +256 -0
  390. package/src/core/utils/forge-resume-block.ts +74 -0
  391. package/src/core/utils/format.ts +69 -0
  392. package/src/core/utils/logger.ts +119 -0
  393. package/src/core/utils/lru-cache.ts +50 -0
  394. package/src/core/utils/path.ts +19 -0
  395. package/src/core/utils/session.ts +26 -0
  396. package/src/core/utils/time.ts +37 -0
  397. package/src/core/utils/token-tracker.ts +97 -0
  398. package/src/daemon/event-parser.ts +35 -0
  399. package/src/daemon/handlers/history-exporter.ts +117 -0
  400. package/src/daemon/handlers/post-tool-use.ts +50 -0
  401. package/src/daemon/handlers/stop.ts +215 -0
  402. package/src/daemon/handlers/user-prompt.ts +188 -0
  403. package/src/daemon/index.ts +278 -0
  404. package/src/daemon/launchd/com.claude-forge.daemon.plist.template +47 -0
  405. package/src/daemon/launchd-installer.ts +260 -0
  406. package/src/daemon/lifecycle.ts +128 -0
  407. package/src/daemon/router.ts +40 -0
  408. package/src/daemon/server.ts +209 -0
  409. package/src/daemon/services/anti-pattern-detector.ts +412 -0
  410. package/src/daemon/services/drift-detector.ts +232 -0
  411. package/src/daemon/services/task-segmenter.ts +112 -0
  412. package/src/daemon/services/weekly-report.ts +454 -0
  413. package/src/hooks/hook-lib.sh +81 -0
  414. package/src/hooks/notification.sh +35 -0
  415. package/src/hooks/post-tool-use.sh +61 -0
  416. package/src/hooks/pre-tool-use.sh +63 -0
  417. package/src/hooks/stop.sh +40 -0
  418. package/src/hooks/user-prompt-submit.sh +69 -0
  419. package/src/mcp/server.ts +322 -0
  420. package/src/skills/index.ts +2 -0
  421. package/src/skills/invocation-guard.ts +177 -0
  422. package/src/skills/matcher.ts +148 -0
  423. package/src/skills/official/code-simplifier.md +16 -0
  424. package/src/skills/official/find-skills.md +23 -0
  425. package/src/skills/official/official-api-design.md +17 -0
  426. package/src/skills/official/official-architecture-decision.md +20 -0
  427. package/src/skills/official/official-bmad.md +118 -0
  428. package/src/skills/official/official-db-schema-design.md +16 -0
  429. package/src/skills/official/official-debug.md +17 -0
  430. package/src/skills/official/official-doc-driven.md +31 -0
  431. package/src/skills/official/official-harness-engineering.md +108 -0
  432. package/src/skills/official/official-performance-optimization.md +30 -0
  433. package/src/skills/official/official-pr-review.md +35 -0
  434. package/src/skills/official/official-release-checklist.md +30 -0
  435. package/src/skills/official/official-security-hardening.md +26 -0
  436. package/src/skills/official/official-spec-driven-design.md +31 -0
  437. package/src/skills/official/planning-with-files.md +37 -0
  438. package/src/skills/official/ui-ux-pro-max.md +18 -0
  439. package/src/skills/official/webapp-testing.md +12 -0
  440. package/src/skills/official-skills.ts +89 -0
  441. package/src/skills/registry.ts +355 -0
  442. package/src/skills/semantic-matcher.ts +231 -0
  443. package/src/skills/tools/pipeline-suggest.ts +226 -0
  444. package/src/skills/tools/skill-invoke.ts +168 -0
  445. package/src/skills/tools/skill-list.ts +59 -0
  446. package/src/templates/go.yaml +53 -0
  447. package/src/templates/python.yaml +59 -0
  448. package/src/templates/react.yaml +55 -0
  449. package/src/templates/template-manager.ts +170 -0
  450. package/src/web/auth-middleware.ts +55 -0
  451. package/src/web/routes/ai.ts +204 -0
  452. package/src/web/routes/auth.ts +22 -0
  453. package/src/web/routes/drift.ts +25 -0
  454. package/src/web/routes/error-handler.ts +120 -0
  455. package/src/web/routes/events.ts +47 -0
  456. package/src/web/routes/insights.ts +43 -0
  457. package/src/web/routes/patch.ts +117 -0
  458. package/src/web/routes/reports.ts +34 -0
  459. package/src/web/routes/rules.ts +101 -0
  460. package/src/web/routes/sessions.ts +262 -0
  461. package/src/web/routes/skill-stats.ts +132 -0
  462. package/src/web/routes/skills.ts +349 -0
  463. package/src/web/routes/static.ts +67 -0
  464. package/src/web/routes/stats.ts +60 -0
  465. package/src/web/routes/status.ts +30 -0
  466. package/src/web/routes/tasks.ts +218 -0
  467. package/src/web/routes/token-usage.ts +20 -0
  468. package/src/web/routes/trace.ts +138 -0
  469. package/src/web/routes/types.ts +56 -0
  470. package/src/web/server.ts +134 -0
  471. package/src/web/ssrf-guard.ts +112 -0
  472. package/src/web/static/index.html +3251 -0
  473. package/src/web/static/vendor/chart.umd.min.js +20 -0
  474. package/tests/e2e/dashboard.spec.ts +205 -0
  475. package/tests/e2e/routing-skill-e2e.test.ts +39 -0
  476. package/tests/helpers/mock-ai.ts +92 -0
  477. package/tests/helpers/mock-storage.ts +159 -0
  478. package/tests/integration/queue-replay.integration.test.ts +193 -0
  479. package/tests/integration/tasks-filter.integration.test.ts +154 -0
  480. package/tests/performance/database.benchmark.ts +161 -0
  481. package/tests/semantic-matcher.test.ts +99 -0
  482. package/tests/skill-matcher.test.ts +110 -0
  483. package/tests/unit/ai-provider-retry.test.ts +194 -0
  484. package/tests/unit/ai-provider-vision.test.ts +224 -0
  485. package/tests/unit/claudemd-generator.test.ts +68 -0
  486. package/tests/unit/cli-mcp.test.ts +141 -0
  487. package/tests/unit/handlers.test.ts +171 -0
  488. package/tests/unit/invocation-guard.test.ts +125 -0
  489. package/tests/unit/queue.test.ts +272 -0
  490. package/tests/unit/router.test.ts +138 -0
  491. package/tests/unit/security.test.ts +128 -0
  492. package/tests/unit/skill-invocations-workflow.test.ts +495 -0
  493. package/tests/unit/skill-registry.test.ts +94 -0
  494. package/tests/unit/skills/invocation-guard-ttl.test.ts +211 -0
  495. package/tests/unit/skills/official-skills-loader.test.ts +126 -0
  496. package/tests/unit/skills/registry-multiformat.test.ts +92 -0
  497. package/tests/unit/storage/sessions-aggregate.test.ts +435 -0
  498. package/tests/unit/storage/sqlite-refactor-harness.test.ts +314 -0
  499. package/tests/unit/storage.test.ts +172 -0
  500. package/tests/unit/token-usage.test.ts +144 -0
  501. package/tests/unit/type-guards.test.ts +201 -0
  502. package/tests/unit/utils/format.test.ts +189 -0
  503. package/tests/unit/utils/session.test.ts +89 -0
  504. package/tests/unit/utils/time.test.ts +112 -0
  505. package/tests/unit/web/routes-auth.test.ts +93 -0
  506. package/tests/unit/web/routes-events.test.ts +101 -0
  507. package/tests/unit/web/routes-sessions.test.ts +181 -0
  508. package/tests/unit/web/routes-skill-stats.test.ts +179 -0
  509. package/tests/unit/web/routes-stats.test.ts +92 -0
  510. package/tests/unit/web/routes-tasks.test.ts +351 -0
  511. package/tsconfig.json +22 -0
  512. package/vitest.config.ts +21 -0
  513. package/vitest.integration.config.ts +16 -0
  514. package/web/CLAUDE.md +20 -0
  515. package/web/index.html +13 -0
  516. package/web/package-lock.json +4854 -0
  517. package/web/package.json +35 -0
  518. package/web/postcss.config.js +6 -0
  519. package/web/src/App.tsx +110 -0
  520. package/web/src/components/CodeBlock.tsx +31 -0
  521. package/web/src/components/Confirm.tsx +96 -0
  522. package/web/src/components/Drawer.tsx +60 -0
  523. package/web/src/components/Layout.tsx +145 -0
  524. package/web/src/components/MarkdownRenderer.tsx +77 -0
  525. package/web/src/components/SearchInput.tsx +31 -0
  526. package/web/src/components/SessionDetailContent.tsx +157 -0
  527. package/web/src/components/Toast.tsx +92 -0
  528. package/web/src/index.css +19 -0
  529. package/web/src/main.tsx +31 -0
  530. package/web/src/pages/AIConfig.tsx +233 -0
  531. package/web/src/pages/Dashboard.tsx +572 -0
  532. package/web/src/pages/Events.tsx +271 -0
  533. package/web/src/pages/Reports.tsx +428 -0
  534. package/web/src/pages/SessionDetail.tsx +162 -0
  535. package/web/src/pages/Sessions.tsx +205 -0
  536. package/web/src/pages/Skills.tsx +180 -0
  537. package/web/src/pages/TaskDetail.tsx +511 -0
  538. package/web/src/pages/Tasks.tsx +150 -0
  539. package/web/src/utils/auth.ts +59 -0
  540. package/web/src/utils/export.ts +54 -0
  541. package/web/src/utils/time.ts +13 -0
  542. package/web/tailwind.config.js +11 -0
  543. package/web/tsconfig.json +21 -0
  544. package/web/tsconfig.node.json +10 -0
  545. package/web/vite.config.ts +76 -0
  546. package/winspan-claude-forge-8.43.0.tgz +0 -0
  547. package/dist/agents/definition.d.ts +0 -62
  548. package/dist/agents/definition.d.ts.map +0 -1
  549. package/dist/agents/definition.js +0 -27
  550. package/dist/agents/definition.js.map +0 -1
  551. package/dist/agents/distributor.d.ts +0 -23
  552. package/dist/agents/distributor.d.ts.map +0 -1
  553. package/dist/agents/distributor.js +0 -85
  554. package/dist/agents/distributor.js.map +0 -1
  555. package/dist/agents/index.d.ts +0 -5
  556. package/dist/agents/index.d.ts.map +0 -1
  557. package/dist/agents/index.js +0 -5
  558. package/dist/agents/index.js.map +0 -1
  559. package/dist/agents/methodologies/agent-builder.d.ts +0 -21
  560. package/dist/agents/methodologies/agent-builder.d.ts.map +0 -1
  561. package/dist/agents/methodologies/agent-builder.js +0 -149
  562. package/dist/agents/methodologies/agent-builder.js.map +0 -1
  563. package/dist/agents/methodologies/phases/bmad/analyze.d.ts +0 -3
  564. package/dist/agents/methodologies/phases/bmad/analyze.d.ts.map +0 -1
  565. package/dist/agents/methodologies/phases/bmad/analyze.js +0 -19
  566. package/dist/agents/methodologies/phases/bmad/analyze.js.map +0 -1
  567. package/dist/agents/methodologies/phases/bmad/design.d.ts +0 -3
  568. package/dist/agents/methodologies/phases/bmad/design.d.ts.map +0 -1
  569. package/dist/agents/methodologies/phases/bmad/design.js +0 -18
  570. package/dist/agents/methodologies/phases/bmad/design.js.map +0 -1
  571. package/dist/agents/methodologies/phases/bmad/implement.d.ts +0 -3
  572. package/dist/agents/methodologies/phases/bmad/implement.d.ts.map +0 -1
  573. package/dist/agents/methodologies/phases/bmad/implement.js +0 -17
  574. package/dist/agents/methodologies/phases/bmad/implement.js.map +0 -1
  575. package/dist/agents/methodologies/phases/bmad/index.d.ts +0 -6
  576. package/dist/agents/methodologies/phases/bmad/index.d.ts.map +0 -1
  577. package/dist/agents/methodologies/phases/bmad/index.js +0 -6
  578. package/dist/agents/methodologies/phases/bmad/index.js.map +0 -1
  579. package/dist/agents/methodologies/phases/bmad/review.d.ts +0 -3
  580. package/dist/agents/methodologies/phases/bmad/review.d.ts.map +0 -1
  581. package/dist/agents/methodologies/phases/bmad/review.js +0 -17
  582. package/dist/agents/methodologies/phases/bmad/review.js.map +0 -1
  583. package/dist/agents/methodologies/phases/bmad/test.d.ts +0 -3
  584. package/dist/agents/methodologies/phases/bmad/test.d.ts.map +0 -1
  585. package/dist/agents/methodologies/phases/bmad/test.js +0 -21
  586. package/dist/agents/methodologies/phases/bmad/test.js.map +0 -1
  587. package/dist/agents/methodologies/phases/harness/fix.d.ts +0 -3
  588. package/dist/agents/methodologies/phases/harness/fix.d.ts.map +0 -1
  589. package/dist/agents/methodologies/phases/harness/fix.js +0 -17
  590. package/dist/agents/methodologies/phases/harness/fix.js.map +0 -1
  591. package/dist/agents/methodologies/phases/harness/index.d.ts +0 -6
  592. package/dist/agents/methodologies/phases/harness/index.d.ts.map +0 -1
  593. package/dist/agents/methodologies/phases/harness/index.js +0 -6
  594. package/dist/agents/methodologies/phases/harness/index.js.map +0 -1
  595. package/dist/agents/methodologies/phases/harness/reproduce.d.ts +0 -3
  596. package/dist/agents/methodologies/phases/harness/reproduce.d.ts.map +0 -1
  597. package/dist/agents/methodologies/phases/harness/reproduce.js +0 -20
  598. package/dist/agents/methodologies/phases/harness/reproduce.js.map +0 -1
  599. package/dist/agents/methodologies/phases/harness/root-cause.d.ts +0 -3
  600. package/dist/agents/methodologies/phases/harness/root-cause.d.ts.map +0 -1
  601. package/dist/agents/methodologies/phases/harness/root-cause.js +0 -21
  602. package/dist/agents/methodologies/phases/harness/root-cause.js.map +0 -1
  603. package/dist/agents/methodologies/phases/harness/safety-net.d.ts +0 -3
  604. package/dist/agents/methodologies/phases/harness/safety-net.d.ts.map +0 -1
  605. package/dist/agents/methodologies/phases/harness/safety-net.js +0 -17
  606. package/dist/agents/methodologies/phases/harness/safety-net.js.map +0 -1
  607. package/dist/agents/methodologies/phases/harness/verify.d.ts +0 -3
  608. package/dist/agents/methodologies/phases/harness/verify.d.ts.map +0 -1
  609. package/dist/agents/methodologies/phases/harness/verify.js +0 -22
  610. package/dist/agents/methodologies/phases/harness/verify.js.map +0 -1
  611. package/dist/agents/methodologies/presets.d.ts +0 -10
  612. package/dist/agents/methodologies/presets.d.ts.map +0 -1
  613. package/dist/agents/methodologies/presets.js +0 -79
  614. package/dist/agents/methodologies/presets.js.map +0 -1
  615. package/dist/agents/methodologies/types.d.ts +0 -45
  616. package/dist/agents/methodologies/types.d.ts.map +0 -1
  617. package/dist/agents/methodologies/types.js +0 -10
  618. package/dist/agents/methodologies/types.js.map +0 -1
  619. package/dist/agents/methodologies/user-config-loader.d.ts +0 -30
  620. package/dist/agents/methodologies/user-config-loader.d.ts.map +0 -1
  621. package/dist/agents/methodologies/user-config-loader.js +0 -159
  622. package/dist/agents/methodologies/user-config-loader.js.map +0 -1
  623. package/dist/agents/official-agents.d.ts +0 -4
  624. package/dist/agents/official-agents.d.ts.map +0 -1
  625. package/dist/agents/official-agents.js +0 -559
  626. package/dist/agents/official-agents.js.map +0 -1
  627. package/dist/agents/registry.d.ts +0 -57
  628. package/dist/agents/registry.d.ts.map +0 -1
  629. package/dist/agents/registry.js +0 -271
  630. package/dist/agents/registry.js.map +0 -1
  631. package/dist/capability/index.d.ts +0 -10
  632. package/dist/capability/index.d.ts.map +0 -1
  633. package/dist/capability/index.js +0 -10
  634. package/dist/capability/index.js.map +0 -1
  635. package/dist/capability/types.d.ts +0 -10
  636. package/dist/capability/types.d.ts.map +0 -1
  637. package/dist/capability/types.js +0 -10
  638. package/dist/capability/types.js.map +0 -1
  639. package/dist/cli/commands/agents.d.ts +0 -3
  640. package/dist/cli/commands/agents.d.ts.map +0 -1
  641. package/dist/cli/commands/agents.js +0 -62
  642. package/dist/cli/commands/agents.js.map +0 -1
  643. package/dist/cli/commands/rules.d.ts +0 -8
  644. package/dist/cli/commands/rules.d.ts.map +0 -1
  645. package/dist/cli/commands/rules.js +0 -89
  646. package/dist/cli/commands/rules.js.map +0 -1
  647. package/dist/daemon/auto-disable-scheduler.d.ts +0 -53
  648. package/dist/daemon/auto-disable-scheduler.d.ts.map +0 -1
  649. package/dist/daemon/auto-disable-scheduler.js +0 -114
  650. package/dist/daemon/auto-disable-scheduler.js.map +0 -1
  651. package/dist/daemon/handlers/pre-tool-use.d.ts +0 -39
  652. package/dist/daemon/handlers/pre-tool-use.d.ts.map +0 -1
  653. package/dist/daemon/handlers/pre-tool-use.js +0 -166
  654. package/dist/daemon/handlers/pre-tool-use.js.map +0 -1
  655. package/dist/daemon/routing-observer.d.ts +0 -42
  656. package/dist/daemon/routing-observer.d.ts.map +0 -1
  657. package/dist/daemon/routing-observer.js +0 -264
  658. package/dist/daemon/routing-observer.js.map +0 -1
  659. package/dist/daemon/routing-state.d.ts +0 -64
  660. package/dist/daemon/routing-state.d.ts.map +0 -1
  661. package/dist/daemon/routing-state.js +0 -240
  662. package/dist/daemon/routing-state.js.map +0 -1
  663. package/dist/engine/agent-router.d.ts +0 -142
  664. package/dist/engine/agent-router.d.ts.map +0 -1
  665. package/dist/engine/agent-router.js +0 -276
  666. package/dist/engine/agent-router.js.map +0 -1
  667. package/dist/engine/context-builder.d.ts +0 -23
  668. package/dist/engine/context-builder.d.ts.map +0 -1
  669. package/dist/engine/context-builder.js +0 -63
  670. package/dist/engine/context-builder.js.map +0 -1
  671. package/dist/engine/conventions/basic-security.yaml +0 -109
  672. package/dist/engine/conventions/code-quality.yaml +0 -123
  673. package/dist/engine/conventions/database-safety.yaml +0 -74
  674. package/dist/engine/conventions/dependency-safety.yaml +0 -132
  675. package/dist/engine/conventions/docker-safety.yaml +0 -69
  676. package/dist/engine/conventions/git-safety.yaml +0 -118
  677. package/dist/engine/conventions/go-best-practices.yaml +0 -84
  678. package/dist/engine/conventions/python-best-practices.yaml +0 -96
  679. package/dist/engine/conventions/react-best-practices.yaml +0 -96
  680. package/dist/engine/conventions/routing.yaml +0 -378
  681. package/dist/engine/conventions/strict-security.yaml +0 -30
  682. package/dist/engine/conventions/ts-quality.yaml +0 -49
  683. package/dist/engine/dsl/compiler.d.ts +0 -34
  684. package/dist/engine/dsl/compiler.d.ts.map +0 -1
  685. package/dist/engine/dsl/compiler.js +0 -702
  686. package/dist/engine/dsl/compiler.js.map +0 -1
  687. package/dist/engine/dsl/parser.d.ts +0 -25
  688. package/dist/engine/dsl/parser.d.ts.map +0 -1
  689. package/dist/engine/dsl/parser.js +0 -208
  690. package/dist/engine/dsl/parser.js.map +0 -1
  691. package/dist/engine/dsl/runtime.d.ts +0 -46
  692. package/dist/engine/dsl/runtime.d.ts.map +0 -1
  693. package/dist/engine/dsl/runtime.js +0 -173
  694. package/dist/engine/dsl/runtime.js.map +0 -1
  695. package/dist/engine/dsl/types.d.ts +0 -139
  696. package/dist/engine/dsl/types.d.ts.map +0 -1
  697. package/dist/engine/dsl/types.js +0 -11
  698. package/dist/engine/dsl/types.js.map +0 -1
  699. package/dist/engine/evidence-store.d.ts +0 -44
  700. package/dist/engine/evidence-store.d.ts.map +0 -1
  701. package/dist/engine/evidence-store.js +0 -109
  702. package/dist/engine/evidence-store.js.map +0 -1
  703. package/dist/engine/experiment-router.d.ts +0 -102
  704. package/dist/engine/experiment-router.d.ts.map +0 -1
  705. package/dist/engine/experiment-router.js +0 -289
  706. package/dist/engine/experiment-router.js.map +0 -1
  707. package/dist/engine/recommender.d.ts +0 -52
  708. package/dist/engine/recommender.d.ts.map +0 -1
  709. package/dist/engine/recommender.js +0 -162
  710. package/dist/engine/recommender.js.map +0 -1
  711. package/dist/engine/rule-engine.d.ts +0 -33
  712. package/dist/engine/rule-engine.d.ts.map +0 -1
  713. package/dist/engine/rule-engine.js +0 -250
  714. package/dist/engine/rule-engine.js.map +0 -1
  715. package/dist/engine/security-gates.d.ts +0 -42
  716. package/dist/engine/security-gates.d.ts.map +0 -1
  717. package/dist/engine/security-gates.js +0 -210
  718. package/dist/engine/security-gates.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,412 @@
1
+ /**
2
+ * AntiPatternDetector — 工作流反模式检测引擎
3
+ *
4
+ * 扫描 events / sessions 表,识别可操作的反模式:
5
+ * - edit_loop: 同一文件反复 Edit/Write(调试鬼打墙)
6
+ * - tool_failure_spike: 最近 24h 工具失败率突增
7
+ * - no_commit: 长会话无 git commit
8
+ * - event_overflow: 单 session 事件数异常
9
+ *
10
+ * 性能策略:
11
+ * - 一次性拉取窗口内事件,4 条规则共享缓存
12
+ * - 仅读 storage(getDatabase + queryEvents + querySessions),不改任何状态
13
+ */
14
+
15
+ import { createHash } from 'node:crypto';
16
+ import type { SQLiteStorage } from '../../core/storage/sqlite.js';
17
+ import type { ForgeEvent } from '../../core/types.js';
18
+ import type { SessionSummary } from '../../core/storage/sessions.js';
19
+
20
+ export type AntiPatternType =
21
+ | 'edit_loop'
22
+ | 'tool_failure_spike'
23
+ | 'no_commit'
24
+ | 'event_overflow';
25
+
26
+ export type AntiPatternSeverity = 'info' | 'warn' | 'critical';
27
+
28
+ export interface AntiPattern {
29
+ id: string;
30
+ type: AntiPatternType;
31
+ severity: AntiPatternSeverity;
32
+ title: string;
33
+ description: string;
34
+ detail: Record<string, unknown>;
35
+ detectedAt: string;
36
+ sessionId?: string;
37
+ filePath?: string;
38
+ }
39
+
40
+ const DAY_MS = 24 * 3600 * 1000;
41
+ const EDIT_TOOLS = new Set(['Edit', 'Write', 'MultiEdit']);
42
+
43
+ export class AntiPatternDetector {
44
+ constructor(private storage: SQLiteStorage) {}
45
+
46
+ /**
47
+ * 运行全部反模式规则。
48
+ *
49
+ * @param windowDays 检测窗口(天),默认 7
50
+ */
51
+ detectAll(windowDays: number = 7): AntiPattern[] {
52
+ const now = Date.now();
53
+ const sinceMs = now - windowDays * DAY_MS;
54
+ const sinceISO = new Date(sinceMs).toISOString();
55
+ const detectedAt = new Date(now).toISOString();
56
+
57
+ // 一次性拉取窗口内事件 + 会话,4 条规则共用
58
+ const events = this.fetchEventsSince(sinceISO);
59
+ const sessions = this.fetchSessionsSince(sinceISO);
60
+
61
+ const patterns: AntiPattern[] = [
62
+ ...this.detectEditLoop(events, detectedAt),
63
+ ...this.detectToolFailureSpike(events, now, windowDays, detectedAt),
64
+ ...this.detectNoCommit(sessions, events, detectedAt),
65
+ ...this.detectEventOverflow(sessions, events, detectedAt),
66
+ ];
67
+
68
+ // 按 severity → critical→warn→info 排序,方便前端取 top N
69
+ const sevOrder: Record<AntiPatternSeverity, number> = { critical: 0, warn: 1, info: 2 };
70
+ patterns.sort((a, b) => sevOrder[a.severity] - sevOrder[b.severity]);
71
+ return patterns;
72
+ }
73
+
74
+ // ── 数据加载 ──────────────────────────────────────────────────────────
75
+
76
+ /**
77
+ * 拉取窗口内全部事件。
78
+ *
79
+ * queryEvents 没有时间过滤参数,因此使用 getDatabase 直接窗口查询。
80
+ * 这是 daemon/services 层的既定做法(见 DriftDetector)。
81
+ */
82
+ private fetchEventsSince(sinceISO: string): ForgeEvent[] {
83
+ const db = this.storage.getDatabase();
84
+ const rows = db.prepare(
85
+ `SELECT * FROM events WHERE timestamp >= ? ORDER BY timestamp ASC`
86
+ ).all(sinceISO) as Array<Record<string, unknown>>;
87
+
88
+ return rows.map((row) => ({
89
+ event_id: row.event_id as string,
90
+ session_id: row.session_id as string,
91
+ project_path: row.project_path as string,
92
+ timestamp: row.timestamp as string,
93
+ hook_type: row.hook_type as ForgeEvent['hook_type'],
94
+ tool_name: (row.tool_name as string) || undefined,
95
+ tool_input: row.tool_input ? safeParse(row.tool_input as string) : undefined,
96
+ tool_output: row.tool_output ? safeParse(row.tool_output as string) : undefined,
97
+ user_prompt: (row.user_prompt as string) || undefined,
98
+ ai_response: (row.ai_response as string) || undefined,
99
+ }));
100
+ }
101
+
102
+ private fetchSessionsSince(sinceISO: string): SessionSummary[] {
103
+ const db = this.storage.getDatabase();
104
+ const rows = db.prepare(
105
+ `SELECT
106
+ session_id,
107
+ first_prompt,
108
+ event_count,
109
+ start_time,
110
+ COALESCE(end_time, last_event_time, start_time) AS end_time
111
+ FROM sessions
112
+ WHERE start_time >= ?
113
+ ORDER BY start_time DESC`
114
+ ).all(sinceISO) as Array<Record<string, unknown>>;
115
+
116
+ return rows.map((r) => ({
117
+ session_id: r.session_id as string,
118
+ first_prompt: (r.first_prompt as string) || '',
119
+ event_count: (r.event_count as number) ?? 0,
120
+ start_time: r.start_time as string,
121
+ end_time: r.end_time as string,
122
+ }));
123
+ }
124
+
125
+ // ── Rule 1: edit_loop ─────────────────────────────────────────────────
126
+
127
+ private detectEditLoop(events: ForgeEvent[], detectedAt: string): AntiPattern[] {
128
+ // 仅 PreToolUse 计一次(避免 Pre + Post 重复计数)
129
+ const buckets = new Map<string, {
130
+ sessionId: string;
131
+ filePath: string;
132
+ count: number;
133
+ firstAt: string;
134
+ lastAt: string;
135
+ }>();
136
+
137
+ for (const ev of events) {
138
+ if (ev.hook_type !== 'PreToolUse') continue;
139
+ if (!ev.tool_name || !EDIT_TOOLS.has(ev.tool_name)) continue;
140
+ const filePath = extractFilePath(ev.tool_input);
141
+ if (!filePath) continue;
142
+
143
+ const key = `${ev.session_id}::${filePath}`;
144
+ const cur = buckets.get(key);
145
+ if (cur) {
146
+ cur.count += 1;
147
+ if (ev.timestamp < cur.firstAt) cur.firstAt = ev.timestamp;
148
+ if (ev.timestamp > cur.lastAt) cur.lastAt = ev.timestamp;
149
+ } else {
150
+ buckets.set(key, {
151
+ sessionId: ev.session_id,
152
+ filePath,
153
+ count: 1,
154
+ firstAt: ev.timestamp,
155
+ lastAt: ev.timestamp,
156
+ });
157
+ }
158
+ }
159
+
160
+ const out: AntiPattern[] = [];
161
+ for (const b of buckets.values()) {
162
+ if (b.count < 5) continue;
163
+ const severity: AntiPatternSeverity = b.count >= 8 ? 'critical' : 'warn';
164
+ out.push({
165
+ id: stableId('edit_loop', b.sessionId, b.filePath),
166
+ type: 'edit_loop',
167
+ severity,
168
+ title: `编辑循环:${shortPath(b.filePath)} 被改动 ${b.count} 次`,
169
+ description: `会话 ${shortId(b.sessionId)} 内对同一文件反复 Edit/Write,疑似调试鬼打墙。`,
170
+ detail: {
171
+ filePath: b.filePath,
172
+ count: b.count,
173
+ sessionId: b.sessionId,
174
+ firstEditAt: b.firstAt,
175
+ lastEditAt: b.lastAt,
176
+ },
177
+ detectedAt,
178
+ sessionId: b.sessionId,
179
+ filePath: b.filePath,
180
+ });
181
+ }
182
+ return out;
183
+ }
184
+
185
+ // ── Rule 2: tool_failure_spike ────────────────────────────────────────
186
+
187
+ private detectToolFailureSpike(
188
+ events: ForgeEvent[],
189
+ nowMs: number,
190
+ windowDays: number,
191
+ detectedAt: string,
192
+ ): AntiPattern[] {
193
+ const recentCutoffMs = nowMs - DAY_MS;
194
+ const recentCutoffISO = new Date(recentCutoffMs).toISOString();
195
+
196
+ let recentTotal = 0;
197
+ let recentFailures = 0;
198
+ let baselineTotal = 0;
199
+ let baselineFailures = 0;
200
+
201
+ for (const ev of events) {
202
+ if (ev.hook_type !== 'PostToolUse') continue;
203
+ const failed = isFailedToolOutput(ev.tool_output);
204
+ if (ev.timestamp >= recentCutoffISO) {
205
+ recentTotal += 1;
206
+ if (failed) recentFailures += 1;
207
+ } else {
208
+ baselineTotal += 1;
209
+ if (failed) baselineFailures += 1;
210
+ }
211
+ }
212
+
213
+ if (recentTotal < 10) {
214
+ // 样本太少,不报警
215
+ return [];
216
+ }
217
+
218
+ const recentRate = recentFailures / recentTotal;
219
+ const baselineRate = baselineTotal > 0 ? baselineFailures / baselineTotal : 0;
220
+
221
+ if (recentRate < 0.2) return [];
222
+
223
+ const severity: AntiPatternSeverity = recentRate >= 0.4 ? 'critical' : 'warn';
224
+ return [{
225
+ id: stableId('tool_failure_spike', detectedAt.slice(0, 13)),
226
+ type: 'tool_failure_spike',
227
+ severity,
228
+ title: `工具失败率 ${(recentRate * 100).toFixed(1)}%(最近 24h)`,
229
+ description: `最近 24 小时 PostToolUse 失败比例显著高于过去 ${windowDays - 1} 天均值(${(baselineRate * 100).toFixed(1)}%)。`,
230
+ detail: {
231
+ recentFailureRate: round(recentRate, 4),
232
+ baselineFailureRate: round(baselineRate, 4),
233
+ recentWindowHours: 24,
234
+ baselineWindowDays: Math.max(windowDays - 1, 1),
235
+ totalToolUses: recentTotal,
236
+ failures: recentFailures,
237
+ baselineTotal,
238
+ baselineFailures,
239
+ },
240
+ detectedAt,
241
+ }];
242
+ }
243
+
244
+ // ── Rule 3: no_commit ─────────────────────────────────────────────────
245
+
246
+ private detectNoCommit(
247
+ sessions: SessionSummary[],
248
+ events: ForgeEvent[],
249
+ detectedAt: string,
250
+ ): AntiPattern[] {
251
+ // 预聚合:每个 session 内的 Bash 命令文本
252
+ const bashBySession = new Map<string, string[]>();
253
+ for (const ev of events) {
254
+ if (ev.tool_name !== 'Bash') continue;
255
+ if (ev.hook_type !== 'PreToolUse') continue;
256
+ const cmd = extractBashCommand(ev.tool_input);
257
+ if (!cmd) continue;
258
+ const list = bashBySession.get(ev.session_id);
259
+ if (list) list.push(cmd);
260
+ else bashBySession.set(ev.session_id, [cmd]);
261
+ }
262
+
263
+ const out: AntiPattern[] = [];
264
+ for (const s of sessions) {
265
+ const durationMs = isoDiffMs(s.start_time, s.end_time);
266
+ const durationMin = durationMs / 60000;
267
+ if (durationMin <= 30) continue;
268
+
269
+ const cmds = bashBySession.get(s.session_id) ?? [];
270
+ const hasCommit = cmds.some((c) => /\bgit\s+commit\b/.test(c));
271
+ if (hasCommit) continue;
272
+
273
+ const severity: AntiPatternSeverity = durationMin > 60 ? 'critical' : 'warn';
274
+ out.push({
275
+ id: stableId('no_commit', s.session_id),
276
+ type: 'no_commit',
277
+ severity,
278
+ title: `${Math.round(durationMin)} 分钟会话无 git commit`,
279
+ description: `会话 ${shortId(s.session_id)} 持续 ${Math.round(durationMin)} 分钟未沉淀提交。`,
280
+ detail: {
281
+ sessionId: s.session_id,
282
+ durationMin: Math.round(durationMin),
283
+ eventCount: s.event_count,
284
+ startTime: s.start_time,
285
+ endTime: s.end_time,
286
+ bashCommands: cmds.length,
287
+ },
288
+ detectedAt,
289
+ sessionId: s.session_id,
290
+ });
291
+ }
292
+ return out;
293
+ }
294
+
295
+ // ── Rule 4: event_overflow ────────────────────────────────────────────
296
+
297
+ private detectEventOverflow(
298
+ sessions: SessionSummary[],
299
+ events: ForgeEvent[],
300
+ detectedAt: string,
301
+ ): AntiPattern[] {
302
+ // 按 session 统计 top tools
303
+ const toolBySession = new Map<string, Map<string, number>>();
304
+ for (const ev of events) {
305
+ if (!ev.tool_name) continue;
306
+ let m = toolBySession.get(ev.session_id);
307
+ if (!m) {
308
+ m = new Map();
309
+ toolBySession.set(ev.session_id, m);
310
+ }
311
+ m.set(ev.tool_name, (m.get(ev.tool_name) ?? 0) + 1);
312
+ }
313
+
314
+ const out: AntiPattern[] = [];
315
+ for (const s of sessions) {
316
+ if (s.event_count <= 500) continue;
317
+ const severity: AntiPatternSeverity = s.event_count > 1000 ? 'critical' : 'warn';
318
+ const durationMin = Math.round(isoDiffMs(s.start_time, s.end_time) / 60000);
319
+ const topTools = topN(toolBySession.get(s.session_id), 5);
320
+
321
+ out.push({
322
+ id: stableId('event_overflow', s.session_id),
323
+ type: 'event_overflow',
324
+ severity,
325
+ title: `事件爆量:${s.event_count} 个事件`,
326
+ description: `会话 ${shortId(s.session_id)} 单次累计 ${s.event_count} 个事件,可能任务规划失败或循环失控。`,
327
+ detail: {
328
+ sessionId: s.session_id,
329
+ eventCount: s.event_count,
330
+ durationMin,
331
+ topTools,
332
+ startTime: s.start_time,
333
+ endTime: s.end_time,
334
+ },
335
+ detectedAt,
336
+ sessionId: s.session_id,
337
+ });
338
+ }
339
+ return out;
340
+ }
341
+ }
342
+
343
+ // ── helpers ─────────────────────────────────────────────────────────────
344
+
345
+ function safeParse(json: string): Record<string, unknown> | undefined {
346
+ try {
347
+ const v = JSON.parse(json);
348
+ return v && typeof v === 'object' ? (v as Record<string, unknown>) : undefined;
349
+ } catch {
350
+ return undefined;
351
+ }
352
+ }
353
+
354
+ function extractFilePath(input: Record<string, unknown> | undefined): string | undefined {
355
+ if (!input) return undefined;
356
+ const fp = input.file_path;
357
+ if (typeof fp === 'string' && fp.length > 0) return fp;
358
+ return undefined;
359
+ }
360
+
361
+ function extractBashCommand(input: Record<string, unknown> | undefined): string | undefined {
362
+ if (!input) return undefined;
363
+ const cmd = input.command;
364
+ return typeof cmd === 'string' ? cmd : undefined;
365
+ }
366
+
367
+ function isFailedToolOutput(out: Record<string, unknown> | undefined): boolean {
368
+ if (!out) return false;
369
+ // Claude Code 的 PostToolUse output 形态:
370
+ // - { error: "..." } → 失败
371
+ // - { success: false, error: "..." } → 失败
372
+ // - { stderr: "...", stdout: "" } → 通常代表失败(仅当 stderr 非空且 stdout 空)
373
+ // - { isError: true } → 失败
374
+ const err = out.error;
375
+ if (typeof err === 'string' && err.length > 0) return true;
376
+ if (out.success === false) return true;
377
+ if (out.isError === true) return true;
378
+ return false;
379
+ }
380
+
381
+ function isoDiffMs(startISO: string, endISO: string): number {
382
+ const s = Date.parse(startISO);
383
+ const e = Date.parse(endISO);
384
+ if (Number.isNaN(s) || Number.isNaN(e)) return 0;
385
+ return Math.max(0, e - s);
386
+ }
387
+
388
+ function topN(map: Map<string, number> | undefined, n: number): Array<{ tool: string; count: number }> {
389
+ if (!map) return [];
390
+ return [...map.entries()]
391
+ .sort((a, b) => b[1] - a[1])
392
+ .slice(0, n)
393
+ .map(([tool, count]) => ({ tool, count }));
394
+ }
395
+
396
+ function round(n: number, digits: number): number {
397
+ const f = 10 ** digits;
398
+ return Math.round(n * f) / f;
399
+ }
400
+
401
+ function shortId(id: string): string {
402
+ return id.length > 8 ? id.slice(0, 8) : id;
403
+ }
404
+
405
+ function shortPath(p: string): string {
406
+ if (p.length <= 60) return p;
407
+ return '…' + p.slice(-58);
408
+ }
409
+
410
+ function stableId(...parts: string[]): string {
411
+ return createHash('sha1').update(parts.join('|')).digest('hex').slice(0, 16);
412
+ }
@@ -0,0 +1,232 @@
1
+ /**
2
+ * DriftDetector — CLAUDE.md 漂移检测引擎
3
+ *
4
+ * 对比 CLAUDE.md 中声明的行为预期与实际事件流数据,
5
+ * 生成合规度报告(DriftReport)。
6
+ *
7
+ * 只读 storage,不写入任何数据。
8
+ */
9
+
10
+ import type { SQLiteStorage } from '../../core/storage/sqlite.js';
11
+
12
+ export interface DriftReport {
13
+ generatedAt: string;
14
+ period: { days: number; since: string };
15
+ checks: DriftCheck[];
16
+ overallScore: number; // 0-100, 100=完全符合
17
+ }
18
+
19
+ export interface DriftCheck {
20
+ rule: string;
21
+ expected: string;
22
+ actual: string;
23
+ status: 'pass' | 'warn' | 'fail';
24
+ detail?: string;
25
+ }
26
+
27
+ export class DriftDetector {
28
+ constructor(private storage: SQLiteStorage) {}
29
+
30
+ /**
31
+ * 执行全部漂移检测规则,返回报告。
32
+ */
33
+ run(days: number = 7): DriftReport {
34
+ const since = Date.now() - days * 24 * 3600 * 1000;
35
+ const sinceISO = new Date(since).toISOString();
36
+
37
+ const checks: DriftCheck[] = [
38
+ this.checkAgentDelegationRate(since),
39
+ this.checkSkillTriggerRate(since),
40
+ this.checkActiveSkillCoverage(since),
41
+ this.checkEventContinuity(),
42
+ this.checkTaskSegmentation(since),
43
+ ];
44
+
45
+ const overallScore = this.computeScore(checks);
46
+
47
+ return {
48
+ generatedAt: new Date().toISOString(),
49
+ period: { days, since: sinceISO },
50
+ checks,
51
+ overallScore,
52
+ };
53
+ }
54
+
55
+ /**
56
+ * 规则 1: Agent 委托率
57
+ * 预期: CLAUDE.md 声明"默认 spawn Agent",obeyed 率应 >30%
58
+ */
59
+ private checkAgentDelegationRate(since: number): DriftCheck {
60
+ const events = this.storage.queryRoutingEvents({ since_ts: since, limit: 10000 });
61
+ const total = events.length;
62
+ const obeyed = events.filter(e => e.obeyed === 1).length;
63
+ const rate = total > 0 ? Math.round((obeyed / total) * 1000) / 10 : 0;
64
+
65
+ let status: DriftCheck['status'];
66
+ if (total === 0) {
67
+ status = 'warn';
68
+ } else if (rate >= 30) {
69
+ status = 'pass';
70
+ } else if (rate >= 10) {
71
+ status = 'warn';
72
+ } else {
73
+ status = 'fail';
74
+ }
75
+
76
+ return {
77
+ rule: 'Agent 委托率',
78
+ expected: '>30% 路由事件被 Agent 执行 (obeyed=1)',
79
+ actual: `${rate}% (${obeyed}/${total})`,
80
+ status,
81
+ detail: total === 0 ? '期间无路由事件' : undefined,
82
+ };
83
+ }
84
+
85
+ /**
86
+ * 规则 2: Skill 触发率
87
+ * 预期: 有 Skill 系统就应该有触发,>10%
88
+ */
89
+ private checkSkillTriggerRate(since: number): DriftCheck {
90
+ const routingEvents = this.storage.queryRoutingEvents({ since_ts: since, limit: 10000 });
91
+ const totalRouting = routingEvents.length;
92
+ const invocations = this.storage.querySkillInvocations({ since, limit: 10000 });
93
+ const totalInvocations = invocations.length;
94
+
95
+ const rate = totalRouting > 0
96
+ ? Math.round((totalInvocations / totalRouting) * 1000) / 10
97
+ : 0;
98
+
99
+ let status: DriftCheck['status'];
100
+ if (totalRouting === 0) {
101
+ status = 'warn';
102
+ } else if (rate >= 10) {
103
+ status = 'pass';
104
+ } else if (rate >= 5) {
105
+ status = 'warn';
106
+ } else {
107
+ status = 'fail';
108
+ }
109
+
110
+ return {
111
+ rule: 'Skill 触发率',
112
+ expected: '>10% (skill_invocations / routing_events)',
113
+ actual: `${rate}% (${totalInvocations}/${totalRouting})`,
114
+ status,
115
+ detail: totalRouting === 0 ? '期间无路由事件' : undefined,
116
+ };
117
+ }
118
+
119
+ /**
120
+ * 规则 3: 活跃 Skill 覆盖率
121
+ * 预期: 至少 3 个不同 Skill 被调用
122
+ */
123
+ private checkActiveSkillCoverage(since: number): DriftCheck {
124
+ const db = this.storage.getDatabase();
125
+ const row = db.prepare(
126
+ `SELECT COUNT(DISTINCT skill_id) as cnt FROM skill_invocations
127
+ WHERE timestamp >= ?`
128
+ ).get(since) as { cnt: number } | undefined;
129
+ const distinctCount = row?.cnt ?? 0;
130
+
131
+ let status: DriftCheck['status'];
132
+ if (distinctCount >= 3) {
133
+ status = 'pass';
134
+ } else if (distinctCount >= 1) {
135
+ status = 'warn';
136
+ } else {
137
+ status = 'fail';
138
+ }
139
+
140
+ return {
141
+ rule: '活跃 Skill 覆盖率',
142
+ expected: '>=3 个不同 Skill 被调用',
143
+ actual: `${distinctCount} 个不同 Skill`,
144
+ status,
145
+ };
146
+ }
147
+
148
+ /**
149
+ * 规则 4: 事件记录连续性
150
+ * 预期: 最近 7 天中至少 5 天有事件(工作日)
151
+ */
152
+ private checkEventContinuity(): DriftCheck {
153
+ const db = this.storage.getDatabase();
154
+ const row = db.prepare(
155
+ `SELECT COUNT(DISTINCT date(timestamp)) as cnt FROM events
156
+ WHERE timestamp >= date('now', '-7 days')`
157
+ ).get() as { cnt: number } | undefined;
158
+ const activeDays = row?.cnt ?? 0;
159
+
160
+ let status: DriftCheck['status'];
161
+ if (activeDays >= 5) {
162
+ status = 'pass';
163
+ } else if (activeDays >= 3) {
164
+ status = 'warn';
165
+ } else {
166
+ status = 'fail';
167
+ }
168
+
169
+ return {
170
+ rule: '事件记录连续性',
171
+ expected: '最近 7 天中 >=5 天有事件',
172
+ actual: `${activeDays} 天有事件`,
173
+ status,
174
+ };
175
+ }
176
+
177
+ /**
178
+ * 规则 5: 任务分段有效性
179
+ * 预期: 每个 session 至少有 1 个 task (tasks/sessions >= 0.8)
180
+ */
181
+ private checkTaskSegmentation(since: number): DriftCheck {
182
+ const db = this.storage.getDatabase();
183
+ const sinceISO = new Date(since).toISOString();
184
+
185
+ const sessionRow = db.prepare(
186
+ `SELECT COUNT(*) as cnt FROM sessions WHERE start_time >= ?`
187
+ ).get(sinceISO) as { cnt: number } | undefined;
188
+ const sessionCount = sessionRow?.cnt ?? 0;
189
+
190
+ const taskRow = db.prepare(
191
+ `SELECT COUNT(*) as cnt FROM tasks WHERE start_time >= ?`
192
+ ).get(sinceISO) as { cnt: number } | undefined;
193
+ const taskCount = taskRow?.cnt ?? 0;
194
+
195
+ const ratio = sessionCount > 0
196
+ ? Math.round((taskCount / sessionCount) * 100) / 100
197
+ : 0;
198
+
199
+ let status: DriftCheck['status'];
200
+ if (sessionCount === 0) {
201
+ status = 'warn';
202
+ } else if (ratio >= 0.8) {
203
+ status = 'pass';
204
+ } else if (ratio >= 0.5) {
205
+ status = 'warn';
206
+ } else {
207
+ status = 'fail';
208
+ }
209
+
210
+ return {
211
+ rule: '任务分段有效性',
212
+ expected: 'tasks/sessions >= 0.8',
213
+ actual: `${ratio} (${taskCount} tasks / ${sessionCount} sessions)`,
214
+ status,
215
+ detail: sessionCount === 0 ? '期间无会话' : undefined,
216
+ };
217
+ }
218
+
219
+ /**
220
+ * 计算总体合规分数 (0-100)
221
+ * pass=20, warn=10, fail=0 (每条规则满分 20)
222
+ */
223
+ private computeScore(checks: DriftCheck[]): number {
224
+ const maxPerCheck = 100 / checks.length;
225
+ let total = 0;
226
+ for (const check of checks) {
227
+ if (check.status === 'pass') total += maxPerCheck;
228
+ else if (check.status === 'warn') total += maxPerCheck * 0.5;
229
+ }
230
+ return Math.round(total);
231
+ }
232
+ }