@winspan/claude-forge 8.53.2 → 8.54.3

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 (390) hide show
  1. package/DEVELOPMENT.md +290 -221
  2. package/README.md +50 -8
  3. package/dist/cli/commands/skills.d.ts.map +1 -1
  4. package/dist/cli/commands/skills.js +7 -3
  5. package/dist/cli/commands/skills.js.map +1 -1
  6. package/dist/cli/init/hook-manager.d.ts +1 -1
  7. package/dist/cli/init/hook-manager.d.ts.map +1 -1
  8. package/dist/cli/init/hook-manager.js +1 -0
  9. package/dist/cli/init/hook-manager.js.map +1 -1
  10. package/dist/core/storage/events.d.ts.map +1 -1
  11. package/dist/core/storage/events.js +0 -1
  12. package/dist/core/storage/events.js.map +1 -1
  13. package/dist/core/storage/maintenance.d.ts +25 -3
  14. package/dist/core/storage/maintenance.d.ts.map +1 -1
  15. package/dist/core/storage/maintenance.js +33 -4
  16. package/dist/core/storage/maintenance.js.map +1 -1
  17. package/dist/core/storage/routing.d.ts +4 -0
  18. package/dist/core/storage/routing.d.ts.map +1 -1
  19. package/dist/core/storage/routing.js +10 -4
  20. package/dist/core/storage/routing.js.map +1 -1
  21. package/dist/core/storage/sessions.d.ts +17 -0
  22. package/dist/core/storage/sessions.d.ts.map +1 -1
  23. package/dist/core/storage/sessions.js +64 -0
  24. package/dist/core/storage/sessions.js.map +1 -1
  25. package/dist/core/storage/skills.d.ts +4 -0
  26. package/dist/core/storage/skills.d.ts.map +1 -1
  27. package/dist/core/storage/skills.js +10 -2
  28. package/dist/core/storage/skills.js.map +1 -1
  29. package/dist/core/storage/sqlite.d.ts +5 -0
  30. package/dist/core/storage/sqlite.d.ts.map +1 -1
  31. package/dist/core/storage/sqlite.js +6 -0
  32. package/dist/core/storage/sqlite.js.map +1 -1
  33. package/dist/core/storage/tasks.d.ts.map +1 -1
  34. package/dist/core/storage/tasks.js +2 -0
  35. package/dist/core/storage/tasks.js.map +1 -1
  36. package/dist/core/types.d.ts +7 -0
  37. package/dist/core/types.d.ts.map +1 -1
  38. package/dist/daemon/index.d.ts.map +1 -1
  39. package/dist/daemon/index.js +19 -4
  40. package/dist/daemon/index.js.map +1 -1
  41. package/dist/skills/registry.d.ts.map +1 -1
  42. package/dist/skills/registry.js +13 -2
  43. package/dist/skills/registry.js.map +1 -1
  44. package/dist/skills/semantic-matcher.d.ts +2 -2
  45. package/dist/skills/semantic-matcher.d.ts.map +1 -1
  46. package/dist/skills/semantic-matcher.js +14 -19
  47. package/dist/skills/semantic-matcher.js.map +1 -1
  48. package/dist/skills/upgrade-engine.d.ts +3 -1
  49. package/dist/skills/upgrade-engine.d.ts.map +1 -1
  50. package/dist/skills/upgrade-engine.js +25 -14
  51. package/dist/skills/upgrade-engine.js.map +1 -1
  52. package/dist/web/analytics/weekly-report.d.ts.map +1 -1
  53. package/dist/web/analytics/weekly-report.js +21 -29
  54. package/dist/web/analytics/weekly-report.js.map +1 -1
  55. package/dist/web/routes/patch.d.ts.map +1 -1
  56. package/dist/web/routes/patch.js +32 -2
  57. package/dist/web/routes/patch.js.map +1 -1
  58. package/dist/web/routes/sessions.d.ts.map +1 -1
  59. package/dist/web/routes/sessions.js +9 -7
  60. package/dist/web/routes/sessions.js.map +1 -1
  61. package/dist/web/routes/trace.d.ts.map +1 -1
  62. package/dist/web/routes/trace.js +2 -3
  63. package/dist/web/routes/trace.js.map +1 -1
  64. package/dist/web/server.d.ts.map +1 -1
  65. package/dist/web/server.js +3 -2
  66. package/dist/web/server.js.map +1 -1
  67. package/package.json +12 -2
  68. package/scripts/postinstall.cjs +21 -0
  69. package/.claude/CLAUDE.md +0 -17
  70. package/.eslintrc.js +0 -23
  71. package/.prettierrc +0 -8
  72. package/ARCHITECTURE_ISSUES.md +0 -249
  73. package/CLAUDE.md +0 -265
  74. package/CLAUDE.md.backup +0 -488
  75. package/docs/concurrent-agents.md +0 -129
  76. package/docs/design/architecture-review-20260516.md +0 -232
  77. package/docs/design/fix-skills-data-and-set-leak-spec-20260516-1300.md +0 -219
  78. package/docs/design/h1-storage-aggregation-spec-20260518-1121.md +0 -299
  79. package/docs/design/h2-getdatabase-encapsulation-spec-20260518-1450.md +0 -191
  80. package/docs/design/h3-fallback-removal-spec-20260518-1245.md +0 -76
  81. package/docs/design/h4-index-dedup-spec-20260518-1230.md +0 -109
  82. package/docs/design/h6-services-migration-spec-20260518-1355.md +0 -82
  83. package/docs/design/hook-failure-queue-spec-20260516-1530.md +0 -204
  84. package/docs/design/l1-swarm-protocol-extract-spec-20260518-1605.md +0 -106
  85. package/docs/design/m10-forge-paths-spec-20260518-1320.md +0 -121
  86. package/docs/design/m2-m3-tool-input-spec-20260518-1425.md +0 -131
  87. package/docs/design/m7-routing-event-association-spec-20260518-1545.md +0 -103
  88. package/docs/design/project-path-gitroot-spec-20260518-1715.md +0 -134
  89. package/docs/design/refactor-phase1-spec-20260515-1600.md +0 -543
  90. package/docs/design/refactor-phase2-spec-20260515-1700.md +0 -424
  91. package/docs/design/skill-ai-upgrade-spec-20260518-1930.md +0 -297
  92. package/docs/design/task-active-gc-spec-20260518-1745.md +0 -146
  93. package/docs/design/tasks-list-filter-pagination-spec-20260518-0930.md +0 -208
  94. package/docs/implementation/daemon-skill-sync-changelog-20260518-2000.md +0 -22
  95. package/docs/implementation/fix-skills-data-and-set-leak-changelog-20260516-1300.md +0 -104
  96. package/docs/implementation/h1-storage-aggregation-changelog-20260518-1121.md +0 -82
  97. package/docs/implementation/h2-final-changelog-20260518-1530.md +0 -61
  98. package/docs/implementation/h2-phase1-safety-net-changelog-20260518-1450.md +0 -70
  99. package/docs/implementation/h2-phase2-operations-changelog-20260518-1450.md +0 -120
  100. package/docs/implementation/h2-phase3-callsites-changelog-20260518-1450.md +0 -71
  101. package/docs/implementation/h3-fallback-removal-changelog-20260518-1245.md +0 -71
  102. package/docs/implementation/h4-index-dedup-changelog-20260518-1230.md +0 -60
  103. package/docs/implementation/h6-services-migration-changelog-20260518-1355.md +0 -46
  104. package/docs/implementation/h7-m9-defaults-changelog-20260518-1300.md +0 -46
  105. package/docs/implementation/hook-failure-queue-changelog-20260516-1530.md +0 -196
  106. package/docs/implementation/hotfix-daemon-event-reject-20260516-1430.md +0 -56
  107. package/docs/implementation/l1-swarm-protocol-extract-changelog-20260518-1605.md +0 -45
  108. package/docs/implementation/l3-l4-daemon-perf-changelog-20260518-1410.md +0 -63
  109. package/docs/implementation/l6-l8-final-cleanup-changelog-20260518-1640.md +0 -38
  110. package/docs/implementation/m1-m4-m5-l7-cleanup-changelog-20260518-1310.md +0 -58
  111. package/docs/implementation/m10-forge-paths-changelog-20260518-1320.md +0 -60
  112. package/docs/implementation/m2-m3-tool-input-changelog-20260518-1425.md +0 -43
  113. package/docs/implementation/m6-m8-naming-shutdown-changelog-20260518-1340.md +0 -56
  114. package/docs/implementation/m7-routing-association-changelog-20260518-1545.md +0 -69
  115. package/docs/implementation/project-path-gitroot-changelog-20260518-1715.md +0 -63
  116. package/docs/implementation/refactor-phase1-changelog-20260515-1630.md +0 -354
  117. package/docs/implementation/refactor-phase2-changelog-20260515-1705.md +0 -421
  118. package/docs/implementation/skill-ai-upgrade-changelog-20260518-1930.md +0 -49
  119. package/docs/implementation/task-active-gc-changelog-20260518-1745.md +0 -35
  120. package/docs/implementation/task-title-summary-changelog-20260518-1130.md +0 -39
  121. package/docs/implementation/tasks-detail-back-loses-filters-changelog-20260518-1100.md +0 -22
  122. package/docs/implementation/tasks-list-filter-pagination-changelog-20260518-0930.md +0 -72
  123. package/docs/implementation/tasks-page-white-screen-hotfix-changelog-20260518-1015.md +0 -56
  124. package/docs/reviews/claudemd-template-sync.md +0 -54
  125. package/docs/reviews/task-title-summary.md +0 -92
  126. package/docs/reviews/tasks-detail-back-loses-filters.md +0 -58
  127. package/docs/reviews/tasks-filter-pagination.md +0 -80
  128. package/docs/reviews/tasks-page-white-screen-hotfix.md +0 -126
  129. package/docs/ruflo-learning-strategy.md +0 -322
  130. package/docs/skills-deduplication-analysis.md +0 -83
  131. package/docs/skills-multiformat-support.md +0 -177
  132. package/docs/skills-third-party.md +0 -183
  133. package/docs/testing/tasks-filter-pagination-test-report.md +0 -86
  134. package/forge +0 -321
  135. package/playwright.config.ts +0 -40
  136. package/scripts/demo-v2.ts +0 -91
  137. package/scripts/dev-daemon.sh +0 -232
  138. package/scripts/dev-web.ts +0 -109
  139. package/scripts/e2e-mcp-link.ts +0 -423
  140. package/scripts/e2e-methodology-quality.ts +0 -253
  141. package/scripts/e2e-routing.ts +0 -456
  142. package/scripts/e2e-user-methodology.ts +0 -326
  143. package/scripts/e2e-web-workflows.ts +0 -299
  144. package/scripts/migrate-legacy-to-dynamic.sql +0 -108
  145. package/scripts/regenerate-execution-docs.ts +0 -116
  146. package/scripts/sync-agent-skills.ts +0 -193
  147. package/scripts/test-hook.sh +0 -71
  148. package/scripts/verify-skill-loading.ts +0 -62
  149. package/src/claudemd/claudemd-generator.ts +0 -568
  150. package/src/claudemd/convention-extractor.ts +0 -69
  151. package/src/claudemd/index.ts +0 -35
  152. package/src/claudemd/persona-manager.ts +0 -88
  153. package/src/claudemd/resume-manager.ts +0 -236
  154. package/src/claudemd/tech-detector.ts +0 -220
  155. package/src/claudemd/templates/swarm-protocol.md +0 -222
  156. package/src/cli/commands/claudemd.ts +0 -84
  157. package/src/cli/commands/config.ts +0 -46
  158. package/src/cli/commands/daemon.ts +0 -310
  159. package/src/cli/commands/executions.ts +0 -115
  160. package/src/cli/commands/init.ts +0 -204
  161. package/src/cli/commands/logs.ts +0 -181
  162. package/src/cli/commands/mcp.ts +0 -242
  163. package/src/cli/commands/menu.ts +0 -357
  164. package/src/cli/commands/skills.ts +0 -328
  165. package/src/cli/commands/stats.ts +0 -73
  166. package/src/cli/commands/status.ts +0 -69
  167. package/src/cli/commands/template.ts +0 -77
  168. package/src/cli/commands/trace.ts +0 -148
  169. package/src/cli/index.ts +0 -42
  170. package/src/cli/init/hook-manager.ts +0 -132
  171. package/src/core/ai/provider.ts +0 -308
  172. package/src/core/ai/types.ts +0 -51
  173. package/src/core/config.ts +0 -124
  174. package/src/core/constants.ts +0 -67
  175. package/src/core/event-fields.ts +0 -32
  176. package/src/core/queue/index.ts +0 -192
  177. package/src/core/storage/base.ts +0 -302
  178. package/src/core/storage/events.ts +0 -434
  179. package/src/core/storage/injections.ts +0 -78
  180. package/src/core/storage/maintenance.ts +0 -59
  181. package/src/core/storage/migrations/002_add_skill_tracking.sql +0 -6
  182. package/src/core/storage/migrations/003_add_skill_invocations.sql +0 -23
  183. package/src/core/storage/performance-indexes.sql +0 -23
  184. package/src/core/storage/routing.ts +0 -322
  185. package/src/core/storage/rows.ts +0 -112
  186. package/src/core/storage/schema.sql +0 -224
  187. package/src/core/storage/sessions.ts +0 -168
  188. package/src/core/storage/skills.ts +0 -233
  189. package/src/core/storage/sqlite.ts +0 -293
  190. package/src/core/storage/tasks.ts +0 -318
  191. package/src/core/storage/token-usage.ts +0 -93
  192. package/src/core/types.ts +0 -181
  193. package/src/core/utils/error-handler.ts +0 -257
  194. package/src/core/utils/forge-resume-block.ts +0 -74
  195. package/src/core/utils/format.ts +0 -69
  196. package/src/core/utils/git.ts +0 -23
  197. package/src/core/utils/logger.ts +0 -134
  198. package/src/core/utils/lru-cache.ts +0 -54
  199. package/src/core/utils/path.ts +0 -19
  200. package/src/core/utils/session.ts +0 -26
  201. package/src/core/utils/time.ts +0 -37
  202. package/src/core/utils/token-tracker.ts +0 -97
  203. package/src/daemon/event-parser.ts +0 -36
  204. package/src/daemon/handlers/history-exporter.ts +0 -117
  205. package/src/daemon/handlers/post-tool-use.ts +0 -54
  206. package/src/daemon/handlers/stop.ts +0 -208
  207. package/src/daemon/handlers/user-prompt.ts +0 -178
  208. package/src/daemon/hook-sync.ts +0 -91
  209. package/src/daemon/index.ts +0 -312
  210. package/src/daemon/launchd/com.claude-forge.daemon.plist.template +0 -47
  211. package/src/daemon/launchd-installer.ts +0 -260
  212. package/src/daemon/lifecycle.ts +0 -128
  213. package/src/daemon/router.ts +0 -40
  214. package/src/daemon/server.ts +0 -196
  215. package/src/daemon/services/task-segmenter.ts +0 -112
  216. package/src/daemon/skill-sync.ts +0 -88
  217. package/src/hooks/hook-lib.sh +0 -118
  218. package/src/hooks/notification.sh +0 -35
  219. package/src/hooks/post-tool-use.sh +0 -61
  220. package/src/hooks/pre-tool-use.sh +0 -63
  221. package/src/hooks/stop.sh +0 -43
  222. package/src/hooks/user-prompt-submit.sh +0 -69
  223. package/src/mcp/server.ts +0 -322
  224. package/src/skills/index.ts +0 -2
  225. package/src/skills/invocation-guard.ts +0 -177
  226. package/src/skills/matcher.ts +0 -148
  227. package/src/skills/official/code-simplifier.md +0 -52
  228. package/src/skills/official/find-skills.md +0 -142
  229. package/src/skills/official/official-api-design.md +0 -30
  230. package/src/skills/official/official-architecture-decision.md +0 -41
  231. package/src/skills/official/official-bmad.md +0 -118
  232. package/src/skills/official/official-db-schema-design.md +0 -34
  233. package/src/skills/official/official-debug.md +0 -25
  234. package/src/skills/official/official-doc-driven.md +0 -31
  235. package/src/skills/official/official-harness-engineering.md +0 -108
  236. package/src/skills/official/official-performance-optimization.md +0 -30
  237. package/src/skills/official/official-pr-review.md +0 -35
  238. package/src/skills/official/official-release-checklist.md +0 -30
  239. package/src/skills/official/official-security-hardening.md +0 -32
  240. package/src/skills/official/official-spec-driven-design.md +0 -31
  241. package/src/skills/official/planning-with-files.md +0 -241
  242. package/src/skills/official/ui-ux-pro-max.md +0 -105
  243. package/src/skills/official/webapp-testing.md +0 -96
  244. package/src/skills/official-skills.ts +0 -89
  245. package/src/skills/registry.ts +0 -355
  246. package/src/skills/semantic-matcher.ts +0 -234
  247. package/src/skills/tools/pipeline-suggest.ts +0 -226
  248. package/src/skills/tools/skill-invoke.ts +0 -168
  249. package/src/skills/tools/skill-list.ts +0 -59
  250. package/src/skills/upgrade-engine.ts +0 -541
  251. package/src/skills/upgrade-prompt.ts +0 -84
  252. package/src/templates/go.yaml +0 -53
  253. package/src/templates/python.yaml +0 -59
  254. package/src/templates/react.yaml +0 -55
  255. package/src/templates/template-manager.ts +0 -170
  256. package/src/web/analytics/anti-pattern-detector.ts +0 -367
  257. package/src/web/analytics/drift-detector.ts +0 -219
  258. package/src/web/analytics/weekly-report.ts +0 -431
  259. package/src/web/auth-middleware.ts +0 -54
  260. package/src/web/routes/_helpers.ts +0 -34
  261. package/src/web/routes/ai.ts +0 -204
  262. package/src/web/routes/auth.ts +0 -22
  263. package/src/web/routes/drift.ts +0 -25
  264. package/src/web/routes/error-handler.ts +0 -120
  265. package/src/web/routes/events.ts +0 -47
  266. package/src/web/routes/insights.ts +0 -43
  267. package/src/web/routes/patch.ts +0 -117
  268. package/src/web/routes/reports.ts +0 -34
  269. package/src/web/routes/rules.ts +0 -76
  270. package/src/web/routes/sessions.ts +0 -250
  271. package/src/web/routes/skill-stats.ts +0 -92
  272. package/src/web/routes/skills.ts +0 -350
  273. package/src/web/routes/static.ts +0 -67
  274. package/src/web/routes/stats.ts +0 -50
  275. package/src/web/routes/status.ts +0 -30
  276. package/src/web/routes/tasks.ts +0 -193
  277. package/src/web/routes/token-usage.ts +0 -20
  278. package/src/web/routes/trace.ts +0 -126
  279. package/src/web/routes/types.ts +0 -57
  280. package/src/web/server.ts +0 -134
  281. package/src/web/ssrf-guard.ts +0 -112
  282. package/src/web/static/index.html +0 -3251
  283. package/src/web/static/vendor/chart.umd.min.js +0 -20
  284. package/tests/e2e/dashboard.spec.ts +0 -205
  285. package/tests/e2e/routing-skill-e2e.test.ts +0 -39
  286. package/tests/helpers/mock-ai.ts +0 -92
  287. package/tests/helpers/mock-storage.ts +0 -159
  288. package/tests/integration/claudemd-generator.test.ts +0 -90
  289. package/tests/integration/queue-replay.integration.test.ts +0 -193
  290. package/tests/integration/tasks-filter.integration.test.ts +0 -154
  291. package/tests/integration/web-analytics.integration.test.ts +0 -133
  292. package/tests/integration/web-stats.integration.test.ts +0 -135
  293. package/tests/integration/web-trace.integration.test.ts +0 -175
  294. package/tests/performance/database.benchmark.ts +0 -161
  295. package/tests/semantic-matcher.test.ts +0 -99
  296. package/tests/skill-matcher.test.ts +0 -110
  297. package/tests/unit/ai-provider-retry.test.ts +0 -194
  298. package/tests/unit/ai-provider-vision.test.ts +0 -224
  299. package/tests/unit/claudemd-generator.test.ts +0 -68
  300. package/tests/unit/cli-mcp.test.ts +0 -141
  301. package/tests/unit/core/forge-paths.test.ts +0 -99
  302. package/tests/unit/daemon/hook-sync.test.ts +0 -71
  303. package/tests/unit/daemon/post-tool-use.test.ts +0 -121
  304. package/tests/unit/daemon/skill-sync.test.ts +0 -75
  305. package/tests/unit/daemon/stop-handler-behavior-summary.test.ts +0 -202
  306. package/tests/unit/daemon/task-segmenter-recover.test.ts +0 -84
  307. package/tests/unit/event-fields.test.ts +0 -88
  308. package/tests/unit/event-parser.test.ts +0 -55
  309. package/tests/unit/handlers.test.ts +0 -171
  310. package/tests/unit/hooks/resolve-project-path.test.ts +0 -122
  311. package/tests/unit/invocation-guard.test.ts +0 -125
  312. package/tests/unit/queue.test.ts +0 -272
  313. package/tests/unit/router.test.ts +0 -138
  314. package/tests/unit/security.test.ts +0 -128
  315. package/tests/unit/skill-invocations-workflow.test.ts +0 -495
  316. package/tests/unit/skill-registry.test.ts +0 -94
  317. package/tests/unit/skills/invocation-guard-ttl.test.ts +0 -211
  318. package/tests/unit/skills/official-skills-loader.test.ts +0 -126
  319. package/tests/unit/skills/registry-multiformat.test.ts +0 -92
  320. package/tests/unit/skills/upgrade-engine-parse.test.ts +0 -138
  321. package/tests/unit/skills/upgrade-engine.test.ts +0 -401
  322. package/tests/unit/skills/upgrade-prompt.test.ts +0 -89
  323. package/tests/unit/socket-server.test.ts +0 -183
  324. package/tests/unit/storage/event-operations-aggregates.test.ts +0 -342
  325. package/tests/unit/storage/migration-idempotent.test.ts +0 -304
  326. package/tests/unit/storage/routing-aggregates.test.ts +0 -276
  327. package/tests/unit/storage/routing.test.ts +0 -117
  328. package/tests/unit/storage/schema-missing.test.ts +0 -81
  329. package/tests/unit/storage/session-operations-aggregates.test.ts +0 -120
  330. package/tests/unit/storage/sessions-aggregate.test.ts +0 -435
  331. package/tests/unit/storage/skill-operations-counts.test.ts +0 -106
  332. package/tests/unit/storage/skills-aggregates.test.ts +0 -104
  333. package/tests/unit/storage/sqlite-refactor-harness.test.ts +0 -314
  334. package/tests/unit/storage/task-operations-counts.test.ts +0 -46
  335. package/tests/unit/storage/tasks-getById.test.ts +0 -343
  336. package/tests/unit/storage/tasks-stale-gc.test.ts +0 -86
  337. package/tests/unit/storage.test.ts +0 -172
  338. package/tests/unit/token-usage.test.ts +0 -144
  339. package/tests/unit/type-guards.test.ts +0 -201
  340. package/tests/unit/utils/format.test.ts +0 -189
  341. package/tests/unit/utils/session.test.ts +0 -89
  342. package/tests/unit/utils/time.test.ts +0 -112
  343. package/tests/unit/web/navigation-back-contract.test.ts +0 -134
  344. package/tests/unit/web/routes-auth.test.ts +0 -93
  345. package/tests/unit/web/routes-events.test.ts +0 -101
  346. package/tests/unit/web/routes-rules.test.ts +0 -182
  347. package/tests/unit/web/routes-sessions.test.ts +0 -181
  348. package/tests/unit/web/routes-skill-stats.test.ts +0 -179
  349. package/tests/unit/web/routes-stats.test.ts +0 -92
  350. package/tests/unit/web/routes-tasks.test.ts +0 -385
  351. package/tests/unit/web/task-title-contract.test.ts +0 -210
  352. package/tests/unit/web/tasks-component-contract.test.ts +0 -179
  353. package/tsconfig.json +0 -22
  354. package/vitest.config.ts +0 -21
  355. package/vitest.integration.config.ts +0 -16
  356. package/web/CLAUDE.md +0 -20
  357. package/web/index.html +0 -13
  358. package/web/package-lock.json +0 -4854
  359. package/web/package.json +0 -35
  360. package/web/postcss.config.js +0 -6
  361. package/web/src/App.tsx +0 -110
  362. package/web/src/components/CodeBlock.tsx +0 -31
  363. package/web/src/components/Confirm.tsx +0 -96
  364. package/web/src/components/Drawer.tsx +0 -60
  365. package/web/src/components/Layout.tsx +0 -145
  366. package/web/src/components/MarkdownRenderer.tsx +0 -77
  367. package/web/src/components/SearchInput.tsx +0 -31
  368. package/web/src/components/SessionDetailContent.tsx +0 -157
  369. package/web/src/components/Toast.tsx +0 -92
  370. package/web/src/index.css +0 -19
  371. package/web/src/main.tsx +0 -31
  372. package/web/src/pages/AIConfig.tsx +0 -233
  373. package/web/src/pages/Dashboard.tsx +0 -572
  374. package/web/src/pages/Events.tsx +0 -271
  375. package/web/src/pages/Reports.tsx +0 -428
  376. package/web/src/pages/SessionDetail.tsx +0 -162
  377. package/web/src/pages/Sessions.tsx +0 -205
  378. package/web/src/pages/Skills.tsx +0 -180
  379. package/web/src/pages/TaskDetail.tsx +0 -515
  380. package/web/src/pages/Tasks.tsx +0 -415
  381. package/web/src/utils/auth.ts +0 -59
  382. package/web/src/utils/export.ts +0 -54
  383. package/web/src/utils/navigation.ts +0 -25
  384. package/web/src/utils/task-title.ts +0 -49
  385. package/web/src/utils/time.ts +0 -13
  386. package/web/tailwind.config.js +0 -11
  387. package/web/tsconfig.json +0 -21
  388. package/web/tsconfig.node.json +0 -10
  389. package/web/vite.config.ts +0 -76
  390. package/winspan-claude-forge-8.43.0.tgz +0 -0
@@ -1,272 +0,0 @@
1
- /**
2
- * Unit tests for src/core/queue/index.ts
3
- *
4
- * All filesystem operations use a temp directory so tests are hermetic.
5
- * We monkey-patch the QUEUE_DIR / DEAD_DIR constants via module internals
6
- * by pointing the OS homedir to a temp dir through env injection, or by
7
- * calling the functions directly with the temp dir substituted.
8
- *
9
- * Since the module computes QUEUE_DIR at import time from os.homedir(), we
10
- * instead test the exported functions by redirecting them to a temp dir
11
- * via environment manipulation, OR we restructure to accept a dir parameter.
12
- *
13
- * For simplicity and correctness: we use the actual exported constants and
14
- * temporarily redirect by overriding the module. Instead of that complexity,
15
- * we'll test the functions with real temp dirs by using the SQLiteStorage
16
- * integration path. For pure prune/enqueue tests we temporarily substitute
17
- * the real QUEUE_DIR by setting HOME.
18
- */
19
-
20
- import { describe, it, expect, beforeEach, afterEach, vi } from 'vitest';
21
- import { mkdirSync, mkdtempSync, readdirSync, rmSync, writeFileSync, utimesSync } from 'node:fs';
22
- import { join } from 'node:path';
23
- import { tmpdir } from 'node:os';
24
-
25
- // ── We test the queue module with HOME overridden ──────────────────────────
26
- // Reset the module registry for each test block to pick up the new HOME.
27
-
28
- describe('Queue module', () => {
29
- let tempHome: string;
30
- let queueDir: string;
31
- let deadDir: string;
32
- let originalHome: string | undefined;
33
-
34
- beforeEach(async () => {
35
- // Override HOME so that the queue module uses our temp dir
36
- originalHome = process.env['HOME'];
37
- tempHome = mkdtempSync(join(tmpdir(), 'forge-queue-test-'));
38
- process.env['HOME'] = tempHome;
39
-
40
- queueDir = join(tempHome, '.claude-forge', 'queue');
41
- deadDir = join(queueDir, 'dead');
42
- mkdirSync(queueDir, { recursive: true });
43
- mkdirSync(deadDir, { recursive: true });
44
-
45
- // Clear module cache so QUEUE_DIR / DEAD_DIR picks up new HOME
46
- vi.resetModules();
47
- });
48
-
49
- afterEach(() => {
50
- process.env['HOME'] = originalHome;
51
- rmSync(tempHome, { recursive: true, force: true });
52
- vi.resetModules();
53
- });
54
-
55
- // ── enqueueEvent ─────────────────────────────────────────────────────────
56
-
57
- it('enqueueEvent writes a file that can be parsed as JSON', async () => {
58
- const { enqueueEvent } = await import('../../src/core/queue/index.js');
59
-
60
- const event = {
61
- session_id: 'test-001',
62
- project_path: '/tmp/proj',
63
- timestamp: new Date().toISOString(),
64
- hook_type: 'UserPromptSubmit' as const,
65
- user_prompt: 'hello',
66
- };
67
-
68
- enqueueEvent(event);
69
-
70
- const files = readdirSync(queueDir).filter((f) => f.endsWith('.json'));
71
- expect(files).toHaveLength(1);
72
-
73
- const raw = await import('node:fs/promises').then((m) =>
74
- m.readFile(join(queueDir, files[0]!), 'utf8'),
75
- );
76
- const parsed = JSON.parse(raw);
77
- expect(parsed.session_id).toBe('test-001');
78
- expect(parsed.event_id).toBeDefined(); // auto-assigned
79
- });
80
-
81
- it('enqueueEvent preserves existing event_id', async () => {
82
- const { enqueueEvent } = await import('../../src/core/queue/index.js');
83
-
84
- const event = {
85
- session_id: 'test-002',
86
- project_path: '/tmp',
87
- timestamp: new Date().toISOString(),
88
- hook_type: 'PreToolUse' as const,
89
- event_id: '33333333-3333-4333-8333-333333333333',
90
- };
91
-
92
- enqueueEvent(event);
93
-
94
- const files = readdirSync(queueDir).filter((f) => f.endsWith('.json'));
95
- const raw = await import('node:fs/promises').then((m) =>
96
- m.readFile(join(queueDir, files[0]!), 'utf8'),
97
- );
98
- const parsed = JSON.parse(raw);
99
- expect(parsed.event_id).toBe('33333333-3333-4333-8333-333333333333');
100
- });
101
-
102
- // ── pruneQueue ────────────────────────────────────────────────────────────
103
-
104
- it('pruneQueue removes oldest files when >= MAX_FILES', async () => {
105
- const { pruneQueue, MAX_FILES } = await import('../../src/core/queue/index.js');
106
-
107
- // Write MAX_FILES + 10 dummy files
108
- const total = MAX_FILES + 10;
109
- for (let i = 0; i < total; i++) {
110
- const ts = new Date(Date.now() + i * 1000).toISOString().replace(/[:.]/g, '-').replace('Z', 'Z');
111
- writeFileSync(join(queueDir, `${ts}-event-${i}.json`), '{}');
112
- }
113
-
114
- pruneQueue();
115
-
116
- const remaining = readdirSync(queueDir).filter((f) => f.endsWith('.json'));
117
- expect(remaining.length).toBeLessThanOrEqual(MAX_FILES);
118
- });
119
-
120
- it('pruneQueue keeps oldest files below MAX_FILES', async () => {
121
- const { pruneQueue, MAX_FILES } = await import('../../src/core/queue/index.js');
122
-
123
- // Write MAX_FILES - 1 files: prune should NOT remove any
124
- for (let i = 0; i < MAX_FILES - 1; i++) {
125
- const ts = new Date(Date.now() + i * 1000).toISOString().replace(/[:.]/g, '-').replace('Z', 'Z');
126
- writeFileSync(join(queueDir, `${ts}-event-${i}.json`), '{}');
127
- }
128
-
129
- pruneQueue();
130
-
131
- const remaining = readdirSync(queueDir).filter((f) => f.endsWith('.json'));
132
- expect(remaining.length).toBe(MAX_FILES - 1);
133
- });
134
-
135
- // ── replayQueue ───────────────────────────────────────────────────────────
136
-
137
- it('replayQueue replays events into storage and deletes queue files', async () => {
138
- const { replayQueue, enqueueEvent } = await import('../../src/core/queue/index.js');
139
- const { SQLiteStorage } = await import('../../src/core/storage/sqlite.js');
140
-
141
- const dbPath = join(tempHome, 'test.db');
142
- const storage = new SQLiteStorage(dbPath);
143
-
144
- const event = {
145
- session_id: 'replay-session-1',
146
- project_path: '/tmp/proj',
147
- timestamp: new Date().toISOString(),
148
- hook_type: 'UserPromptSubmit' as const,
149
- user_prompt: 'replayed',
150
- event_id: '11111111-1111-4111-8111-111111111111',
151
- };
152
- enqueueEvent(event);
153
-
154
- const result = await replayQueue(storage);
155
-
156
- expect(result.replayed).toBe(1);
157
- expect(result.skipped).toBe(0);
158
- expect(result.dead).toBe(0);
159
-
160
- const files = readdirSync(queueDir).filter((f) => f.endsWith('.json'));
161
- expect(files).toHaveLength(0);
162
-
163
- const events = storage.queryEvents({ session_id: 'replay-session-1' });
164
- expect(events).toHaveLength(1);
165
- expect(events[0]!.user_prompt).toBe('replayed');
166
-
167
- storage.close();
168
- });
169
-
170
- it('replayQueue silently skips duplicate event_id (UNIQUE constraint)', async () => {
171
- const { replayQueue, enqueueEvent } = await import('../../src/core/queue/index.js');
172
- const { SQLiteStorage } = await import('../../src/core/storage/sqlite.js');
173
-
174
- const dbPath = join(tempHome, 'dedup.db');
175
- const storage = new SQLiteStorage(dbPath);
176
-
177
- const event = {
178
- session_id: 'dedup-session',
179
- project_path: '/tmp',
180
- timestamp: new Date().toISOString(),
181
- hook_type: 'PreToolUse' as const,
182
- event_id: '22222222-2222-4222-8222-222222222222',
183
- };
184
-
185
- // Pre-write event into DB directly
186
- storage.writeEvent(event);
187
-
188
- // Enqueue the same event
189
- enqueueEvent(event);
190
-
191
- const result = await replayQueue(storage);
192
-
193
- expect(result.skipped).toBe(1);
194
- expect(result.replayed).toBe(0);
195
- expect(result.dead).toBe(0);
196
-
197
- // Queue file should be removed
198
- const files = readdirSync(queueDir).filter((f) => f.endsWith('.json'));
199
- expect(files).toHaveLength(0);
200
-
201
- // DB still has exactly one copy
202
- const events = storage.queryEvents({ session_id: 'dedup-session' });
203
- expect(events).toHaveLength(1);
204
-
205
- storage.close();
206
- });
207
-
208
- it('replayQueue moves corrupt JSON to dead-letter dir', async () => {
209
- const { replayQueue } = await import('../../src/core/queue/index.js');
210
- const { SQLiteStorage } = await import('../../src/core/storage/sqlite.js');
211
-
212
- const dbPath = join(tempHome, 'corrupt.db');
213
- const storage = new SQLiteStorage(dbPath);
214
-
215
- writeFileSync(join(queueDir, '2020-01-01T00-00-00-000Z-bad.json'), 'NOT_JSON{{{');
216
-
217
- const result = await replayQueue(storage);
218
-
219
- expect(result.dead).toBe(1);
220
- expect(result.replayed).toBe(0);
221
-
222
- const deadFiles = readdirSync(deadDir).filter((f) => f.endsWith('.json'));
223
- expect(deadFiles).toHaveLength(1);
224
-
225
- storage.close();
226
- });
227
-
228
- it('replayQueue moves TTL-expired files to dead-letter dir', async () => {
229
- const { replayQueue, TTL_DAYS } = await import('../../src/core/queue/index.js');
230
- const { SQLiteStorage } = await import('../../src/core/storage/sqlite.js');
231
-
232
- const dbPath = join(tempHome, 'ttl.db');
233
- const storage = new SQLiteStorage(dbPath);
234
-
235
- const filename = '2000-01-01T00-00-00-000Z-old-event.json';
236
- const filePath = join(queueDir, filename);
237
- writeFileSync(filePath, JSON.stringify({
238
- session_id: 'old',
239
- project_path: '/tmp',
240
- timestamp: '2000-01-01T00:00:00.000Z',
241
- hook_type: 'Notification',
242
- event_id: '44444444-4444-4444-8444-444444444444',
243
- }));
244
-
245
- // Set mtime to TTL_DAYS + 1 days ago
246
- const oldTime = new Date(Date.now() - (TTL_DAYS + 1) * 24 * 60 * 60 * 1000);
247
- utimesSync(filePath, oldTime, oldTime);
248
-
249
- const result = await replayQueue(storage);
250
-
251
- expect(result.dead).toBe(1);
252
- expect(result.replayed).toBe(0);
253
-
254
- const deadFiles = readdirSync(deadDir).filter((f) => f.endsWith('.json'));
255
- expect(deadFiles).toHaveLength(1);
256
-
257
- storage.close();
258
- });
259
-
260
- it('replayQueue returns zeros when queue is empty', async () => {
261
- const { replayQueue } = await import('../../src/core/queue/index.js');
262
- const { SQLiteStorage } = await import('../../src/core/storage/sqlite.js');
263
-
264
- const dbPath = join(tempHome, 'empty.db');
265
- const storage = new SQLiteStorage(dbPath);
266
-
267
- const result = await replayQueue(storage);
268
- expect(result).toEqual({ replayed: 0, skipped: 0, dead: 0 });
269
-
270
- storage.close();
271
- });
272
- });
@@ -1,138 +0,0 @@
1
- import { describe, it, expect, beforeEach } from 'vitest';
2
- import { createMockAI, createFailingMockAI } from '../helpers/mock-ai.js';
3
-
4
- describe('Router - Agent Routing', () => {
5
- describe('MockAIProvider', () => {
6
- it('should return mock response', async () => {
7
- const ai = createMockAI('test response');
8
- const result = await ai.complete('test prompt');
9
- expect(result).toBe('test response');
10
- });
11
-
12
- it('should track call history', async () => {
13
- const ai = createMockAI('response');
14
- await ai.complete('prompt 1');
15
- await ai.complete('prompt 2');
16
-
17
- expect(ai.getCallCount()).toBe(2);
18
- expect(ai.getLastPrompt()).toBe('prompt 2');
19
- });
20
-
21
- it('should return multiple responses in sequence', async () => {
22
- const ai = createMockAI();
23
- ai.addResponse('response 1');
24
- ai.addResponse('response 2');
25
-
26
- const result1 = await ai.complete('prompt 1');
27
- const result2 = await ai.complete('prompt 2');
28
-
29
- expect(result1).toBe('mock response');
30
- expect(result2).toBe('response 1');
31
- });
32
-
33
- it('should reset state', async () => {
34
- const ai = createMockAI('response');
35
- await ai.complete('prompt');
36
- expect(ai.getCallCount()).toBe(1);
37
-
38
- ai.reset();
39
- expect(ai.getCallCount()).toBe(0);
40
- });
41
- });
42
-
43
- describe('Error handling', () => {
44
- it('should handle AI provider errors', async () => {
45
- const ai = createFailingMockAI('Test error');
46
- await expect(ai.complete('prompt')).rejects.toThrow('Test error');
47
- });
48
- });
49
-
50
- describe('Intent classification', () => {
51
- it('should classify simple prompts', () => {
52
- const prompts = [
53
- { text: 'fix the bug', expectedIntent: 'bug_fix' },
54
- { text: 'add new feature', expectedIntent: 'feature' },
55
- { text: 'refactor the code', expectedIntent: 'refactor' },
56
- { text: 'optimize performance', expectedIntent: 'performance' },
57
- ];
58
-
59
- prompts.forEach(({ text, expectedIntent }) => {
60
- // Simple keyword-based classification for testing
61
- let intent = 'unknown';
62
- if (text.includes('fix') || text.includes('bug')) intent = 'bug_fix';
63
- else if (text.includes('add') || text.includes('feature')) intent = 'feature';
64
- else if (text.includes('refactor')) intent = 'refactor';
65
- else if (text.includes('optimize') || text.includes('performance')) intent = 'performance';
66
-
67
- expect(intent).toBe(expectedIntent);
68
- });
69
- });
70
- });
71
-
72
- describe('Agent selection', () => {
73
- it('should select appropriate agent for bug fix', () => {
74
- const intent = 'bug_fix';
75
- const agent = selectAgent(intent);
76
- expect(agent).toBe('coder');
77
- });
78
-
79
- it('should select appropriate agent for feature', () => {
80
- const intent = 'feature';
81
- const agent = selectAgent(intent);
82
- expect(agent).toBe('feature-developer');
83
- });
84
-
85
- it('should select appropriate agent for refactor', () => {
86
- const intent = 'refactor';
87
- const agent = selectAgent(intent);
88
- expect(agent).toBe('refactor-specialist');
89
- });
90
-
91
- it('should default to coder for unknown intent', () => {
92
- const intent = 'unknown';
93
- const agent = selectAgent(intent);
94
- expect(agent).toBe('coder');
95
- });
96
- });
97
-
98
- describe('Routing decision', () => {
99
- it('should create routing decision with metadata', () => {
100
- const decision = {
101
- intent: 'bug_fix',
102
- agent: 'coder',
103
- confidence: 0.9,
104
- timestamp: Date.now(),
105
- };
106
-
107
- expect(decision.intent).toBe('bug_fix');
108
- expect(decision.agent).toBe('coder');
109
- expect(decision.confidence).toBeGreaterThan(0.8);
110
- expect(decision.timestamp).toBeGreaterThan(0);
111
- });
112
-
113
- it('should handle low confidence routing', () => {
114
- const decision = {
115
- intent: 'unknown',
116
- agent: 'coder',
117
- confidence: 0.3,
118
- fallback: true,
119
- };
120
-
121
- expect(decision.fallback).toBe(true);
122
- expect(decision.confidence).toBeLessThan(0.5);
123
- });
124
- });
125
- });
126
-
127
- // Helper function for agent selection
128
- function selectAgent(intent: string): string {
129
- const mapping: Record<string, string> = {
130
- bug_fix: 'coder',
131
- feature: 'feature-developer',
132
- refactor: 'refactor-specialist',
133
- performance: 'coder',
134
- investigation: 'researcher',
135
- };
136
-
137
- return mapping[intent] || 'coder';
138
- }
@@ -1,128 +0,0 @@
1
- import { describe, it, expect } from 'vitest';
2
- import { resolvePatchTarget } from '../../src/web/routes/types.js';
3
- import fs from 'node:fs';
4
- import path from 'node:path';
5
- import { tmpdir } from 'node:os';
6
-
7
- describe('Security - Path Traversal Prevention', () => {
8
- describe('resolvePatchTarget', () => {
9
- it('should reject path traversal with ../', () => {
10
- expect(() => resolvePatchTarget('skill', '../../../etc/passwd')).toThrow('Invalid target name');
11
- });
12
-
13
- it('should reject path traversal with ..\\', () => {
14
- expect(() => resolvePatchTarget('skill', '..\\..\\..\\windows\\system32')).toThrow('Invalid target name');
15
- });
16
-
17
- it('should reject absolute paths with /', () => {
18
- expect(() => resolvePatchTarget('skill', '/etc/passwd')).toThrow('Invalid target name');
19
- });
20
-
21
- it('should reject absolute paths with \\', () => {
22
- expect(() => resolvePatchTarget('skill', '\\windows\\system32')).toThrow('Invalid target name');
23
- });
24
-
25
- it('should reject paths with .. in the middle', () => {
26
- expect(() => resolvePatchTarget('skill', 'foo/../bar')).toThrow('Invalid target name');
27
- });
28
-
29
- it('should accept valid skill names', () => {
30
- const result = resolvePatchTarget('skill', 'my-skill');
31
- expect(result.filePath).toContain('my-skill.md');
32
- expect(result.filePath).not.toContain('..');
33
- });
34
-
35
- it('should accept skill names with underscores and hyphens', () => {
36
- const result = resolvePatchTarget('skill', 'my_skill-v2');
37
- expect(result.filePath).toContain('my_skill-v2.md');
38
- });
39
-
40
- it('should reject empty target names', () => {
41
- expect(() => resolvePatchTarget('skill', '')).toThrow('Invalid target name');
42
- });
43
- });
44
- });
45
-
46
- describe('Security - Input Validation', () => {
47
- describe('Skill name validation patterns', () => {
48
- const invalidNames = [
49
- '../../../etc/passwd',
50
- '..\\..\\..\\windows\\system32',
51
- '/etc/passwd',
52
- '\\windows\\system32',
53
- 'foo/../bar',
54
- 'foo\\..\\bar',
55
- '..',
56
- './foo',
57
- '.\\foo',
58
- ];
59
-
60
- invalidNames.forEach(name => {
61
- it(`should reject invalid name: ${name}`, () => {
62
- expect(() => resolvePatchTarget('skill', name)).toThrow();
63
- });
64
- });
65
-
66
- const validNames = [
67
- 'my-skill',
68
- 'my_skill',
69
- 'skill123',
70
- 'SKILL',
71
- 'my-skill-v2',
72
- 'my_skill_v2',
73
- ];
74
-
75
- validNames.forEach(name => {
76
- it(`should accept valid name: ${name}`, () => {
77
- expect(() => resolvePatchTarget('skill', name)).not.toThrow();
78
- });
79
- });
80
- });
81
- });
82
-
83
- describe('Security - Git Command Injection Prevention', () => {
84
- describe('projectPath validation', () => {
85
- it('should reject relative paths', () => {
86
- const relativePath = '../some/project';
87
- expect(path.isAbsolute(relativePath)).toBe(false);
88
- });
89
-
90
- it('should accept absolute paths', () => {
91
- const absolutePath = '/home/user/project';
92
- expect(path.isAbsolute(absolutePath)).toBe(true);
93
- });
94
-
95
- it('should validate directory existence', () => {
96
- const nonExistentPath = '/this/path/does/not/exist/12345';
97
- expect(fs.existsSync(nonExistentPath)).toBe(false);
98
- });
99
-
100
- it('should validate .git directory existence', () => {
101
- const tempDir = tmpdir();
102
- const gitDir = path.join(tempDir, '.git');
103
-
104
- // tmpdir exists but likely doesn't have .git
105
- expect(fs.existsSync(tempDir)).toBe(true);
106
- // Most temp dirs won't have .git
107
- if (!fs.existsSync(gitDir)) {
108
- expect(fs.existsSync(gitDir)).toBe(false);
109
- }
110
- });
111
-
112
- it('should handle path with special characters safely', () => {
113
- // These should be rejected as non-existent or invalid
114
- const dangerousPaths = [
115
- '/tmp; rm -rf /',
116
- '/tmp && echo pwned',
117
- '/tmp | cat /etc/passwd',
118
- '/tmp`whoami`',
119
- '/tmp$(whoami)',
120
- ];
121
-
122
- dangerousPaths.forEach(p => {
123
- // These paths should fail existence check
124
- expect(fs.existsSync(p)).toBe(false);
125
- });
126
- });
127
- });
128
- });