@winspan/claude-forge 8.51.1 → 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 (409) 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 +121 -2
  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/constants.d.ts +2 -0
  11. package/dist/core/constants.d.ts.map +1 -1
  12. package/dist/core/constants.js +4 -0
  13. package/dist/core/constants.js.map +1 -1
  14. package/dist/core/storage/events.d.ts.map +1 -1
  15. package/dist/core/storage/events.js +0 -1
  16. package/dist/core/storage/events.js.map +1 -1
  17. package/dist/core/storage/maintenance.d.ts +25 -3
  18. package/dist/core/storage/maintenance.d.ts.map +1 -1
  19. package/dist/core/storage/maintenance.js +33 -4
  20. package/dist/core/storage/maintenance.js.map +1 -1
  21. package/dist/core/storage/routing.d.ts +4 -0
  22. package/dist/core/storage/routing.d.ts.map +1 -1
  23. package/dist/core/storage/routing.js +10 -4
  24. package/dist/core/storage/routing.js.map +1 -1
  25. package/dist/core/storage/sessions.d.ts +17 -0
  26. package/dist/core/storage/sessions.d.ts.map +1 -1
  27. package/dist/core/storage/sessions.js +64 -0
  28. package/dist/core/storage/sessions.js.map +1 -1
  29. package/dist/core/storage/skills.d.ts +4 -0
  30. package/dist/core/storage/skills.d.ts.map +1 -1
  31. package/dist/core/storage/skills.js +10 -2
  32. package/dist/core/storage/skills.js.map +1 -1
  33. package/dist/core/storage/sqlite.d.ts +5 -0
  34. package/dist/core/storage/sqlite.d.ts.map +1 -1
  35. package/dist/core/storage/sqlite.js +6 -0
  36. package/dist/core/storage/sqlite.js.map +1 -1
  37. package/dist/core/storage/tasks.d.ts.map +1 -1
  38. package/dist/core/storage/tasks.js +2 -0
  39. package/dist/core/storage/tasks.js.map +1 -1
  40. package/dist/core/types.d.ts +7 -0
  41. package/dist/core/types.d.ts.map +1 -1
  42. package/dist/daemon/index.d.ts.map +1 -1
  43. package/dist/daemon/index.js +30 -5
  44. package/dist/daemon/index.js.map +1 -1
  45. package/dist/daemon/skill-sync.d.ts +21 -0
  46. package/dist/daemon/skill-sync.d.ts.map +1 -0
  47. package/dist/daemon/skill-sync.js +75 -0
  48. package/dist/daemon/skill-sync.js.map +1 -0
  49. package/dist/hooks/notification.sh +1 -1
  50. package/dist/hooks/post-tool-use.sh +1 -1
  51. package/dist/hooks/pre-tool-use.sh +1 -1
  52. package/dist/hooks/stop.sh +1 -1
  53. package/dist/hooks/user-prompt-submit.sh +1 -1
  54. package/dist/skills/official/code-simplifier.md +37 -1
  55. package/dist/skills/official/find-skills.md +120 -1
  56. package/dist/skills/official/official-api-design.md +14 -1
  57. package/dist/skills/official/official-architecture-decision.md +22 -1
  58. package/dist/skills/official/official-db-schema-design.md +19 -1
  59. package/dist/skills/official/official-debug.md +9 -1
  60. package/dist/skills/official/official-pr-review.md +1 -1
  61. package/dist/skills/official/official-security-hardening.md +7 -1
  62. package/dist/skills/official/planning-with-files.md +206 -2
  63. package/dist/skills/official/ui-ux-pro-max.md +88 -1
  64. package/dist/skills/official/webapp-testing.md +85 -1
  65. package/dist/skills/registry.d.ts +1 -1
  66. package/dist/skills/registry.d.ts.map +1 -1
  67. package/dist/skills/registry.js +15 -4
  68. package/dist/skills/registry.js.map +1 -1
  69. package/dist/skills/semantic-matcher.d.ts +4 -3
  70. package/dist/skills/semantic-matcher.d.ts.map +1 -1
  71. package/dist/skills/semantic-matcher.js +20 -22
  72. package/dist/skills/semantic-matcher.js.map +1 -1
  73. package/dist/skills/upgrade-engine.d.ts +93 -0
  74. package/dist/skills/upgrade-engine.d.ts.map +1 -0
  75. package/dist/skills/upgrade-engine.js +447 -0
  76. package/dist/skills/upgrade-engine.js.map +1 -0
  77. package/dist/skills/upgrade-prompt.d.ts +20 -0
  78. package/dist/skills/upgrade-prompt.d.ts.map +1 -0
  79. package/dist/skills/upgrade-prompt.js +75 -0
  80. package/dist/skills/upgrade-prompt.js.map +1 -0
  81. package/dist/web/analytics/weekly-report.d.ts.map +1 -1
  82. package/dist/web/analytics/weekly-report.js +21 -29
  83. package/dist/web/analytics/weekly-report.js.map +1 -1
  84. package/dist/web/routes/patch.d.ts.map +1 -1
  85. package/dist/web/routes/patch.js +32 -2
  86. package/dist/web/routes/patch.js.map +1 -1
  87. package/dist/web/routes/sessions.d.ts.map +1 -1
  88. package/dist/web/routes/sessions.js +9 -7
  89. package/dist/web/routes/sessions.js.map +1 -1
  90. package/dist/web/routes/trace.d.ts.map +1 -1
  91. package/dist/web/routes/trace.js +2 -3
  92. package/dist/web/routes/trace.js.map +1 -1
  93. package/dist/web/server.d.ts.map +1 -1
  94. package/dist/web/server.js +3 -2
  95. package/dist/web/server.js.map +1 -1
  96. package/package.json +12 -2
  97. package/scripts/postinstall.cjs +21 -0
  98. package/.claude/CLAUDE.md +0 -17
  99. package/.eslintrc.js +0 -23
  100. package/.prettierrc +0 -8
  101. package/ARCHITECTURE_ISSUES.md +0 -249
  102. package/CLAUDE.md +0 -265
  103. package/CLAUDE.md.backup +0 -488
  104. package/docs/concurrent-agents.md +0 -129
  105. package/docs/design/architecture-review-20260516.md +0 -232
  106. package/docs/design/fix-skills-data-and-set-leak-spec-20260516-1300.md +0 -219
  107. package/docs/design/h1-storage-aggregation-spec-20260518-1121.md +0 -299
  108. package/docs/design/h2-getdatabase-encapsulation-spec-20260518-1450.md +0 -191
  109. package/docs/design/h3-fallback-removal-spec-20260518-1245.md +0 -76
  110. package/docs/design/h4-index-dedup-spec-20260518-1230.md +0 -109
  111. package/docs/design/h6-services-migration-spec-20260518-1355.md +0 -82
  112. package/docs/design/hook-failure-queue-spec-20260516-1530.md +0 -204
  113. package/docs/design/l1-swarm-protocol-extract-spec-20260518-1605.md +0 -106
  114. package/docs/design/m10-forge-paths-spec-20260518-1320.md +0 -121
  115. package/docs/design/m2-m3-tool-input-spec-20260518-1425.md +0 -131
  116. package/docs/design/m7-routing-event-association-spec-20260518-1545.md +0 -103
  117. package/docs/design/project-path-gitroot-spec-20260518-1715.md +0 -134
  118. package/docs/design/refactor-phase1-spec-20260515-1600.md +0 -543
  119. package/docs/design/refactor-phase2-spec-20260515-1700.md +0 -424
  120. package/docs/design/task-active-gc-spec-20260518-1745.md +0 -146
  121. package/docs/design/tasks-list-filter-pagination-spec-20260518-0930.md +0 -208
  122. package/docs/implementation/fix-skills-data-and-set-leak-changelog-20260516-1300.md +0 -104
  123. package/docs/implementation/h1-storage-aggregation-changelog-20260518-1121.md +0 -82
  124. package/docs/implementation/h2-final-changelog-20260518-1530.md +0 -61
  125. package/docs/implementation/h2-phase1-safety-net-changelog-20260518-1450.md +0 -70
  126. package/docs/implementation/h2-phase2-operations-changelog-20260518-1450.md +0 -120
  127. package/docs/implementation/h2-phase3-callsites-changelog-20260518-1450.md +0 -71
  128. package/docs/implementation/h3-fallback-removal-changelog-20260518-1245.md +0 -71
  129. package/docs/implementation/h4-index-dedup-changelog-20260518-1230.md +0 -60
  130. package/docs/implementation/h6-services-migration-changelog-20260518-1355.md +0 -46
  131. package/docs/implementation/h7-m9-defaults-changelog-20260518-1300.md +0 -46
  132. package/docs/implementation/hook-failure-queue-changelog-20260516-1530.md +0 -196
  133. package/docs/implementation/hotfix-daemon-event-reject-20260516-1430.md +0 -56
  134. package/docs/implementation/l1-swarm-protocol-extract-changelog-20260518-1605.md +0 -45
  135. package/docs/implementation/l3-l4-daemon-perf-changelog-20260518-1410.md +0 -63
  136. package/docs/implementation/l6-l8-final-cleanup-changelog-20260518-1640.md +0 -38
  137. package/docs/implementation/m1-m4-m5-l7-cleanup-changelog-20260518-1310.md +0 -58
  138. package/docs/implementation/m10-forge-paths-changelog-20260518-1320.md +0 -60
  139. package/docs/implementation/m2-m3-tool-input-changelog-20260518-1425.md +0 -43
  140. package/docs/implementation/m6-m8-naming-shutdown-changelog-20260518-1340.md +0 -56
  141. package/docs/implementation/m7-routing-association-changelog-20260518-1545.md +0 -69
  142. package/docs/implementation/project-path-gitroot-changelog-20260518-1715.md +0 -63
  143. package/docs/implementation/refactor-phase1-changelog-20260515-1630.md +0 -354
  144. package/docs/implementation/refactor-phase2-changelog-20260515-1705.md +0 -421
  145. package/docs/implementation/task-active-gc-changelog-20260518-1745.md +0 -35
  146. package/docs/implementation/task-title-summary-changelog-20260518-1130.md +0 -39
  147. package/docs/implementation/tasks-detail-back-loses-filters-changelog-20260518-1100.md +0 -22
  148. package/docs/implementation/tasks-list-filter-pagination-changelog-20260518-0930.md +0 -72
  149. package/docs/implementation/tasks-page-white-screen-hotfix-changelog-20260518-1015.md +0 -56
  150. package/docs/reviews/claudemd-template-sync.md +0 -54
  151. package/docs/reviews/task-title-summary.md +0 -92
  152. package/docs/reviews/tasks-detail-back-loses-filters.md +0 -58
  153. package/docs/reviews/tasks-filter-pagination.md +0 -80
  154. package/docs/reviews/tasks-page-white-screen-hotfix.md +0 -126
  155. package/docs/ruflo-learning-strategy.md +0 -322
  156. package/docs/skills-deduplication-analysis.md +0 -83
  157. package/docs/skills-multiformat-support.md +0 -177
  158. package/docs/skills-third-party.md +0 -183
  159. package/docs/testing/tasks-filter-pagination-test-report.md +0 -86
  160. package/forge +0 -321
  161. package/playwright.config.ts +0 -40
  162. package/scripts/demo-v2.ts +0 -91
  163. package/scripts/dev-daemon.sh +0 -232
  164. package/scripts/dev-web.ts +0 -109
  165. package/scripts/e2e-mcp-link.ts +0 -423
  166. package/scripts/e2e-methodology-quality.ts +0 -253
  167. package/scripts/e2e-routing.ts +0 -456
  168. package/scripts/e2e-user-methodology.ts +0 -326
  169. package/scripts/e2e-web-workflows.ts +0 -299
  170. package/scripts/migrate-legacy-to-dynamic.sql +0 -108
  171. package/scripts/regenerate-execution-docs.ts +0 -116
  172. package/scripts/sync-agent-skills.ts +0 -193
  173. package/scripts/test-hook.sh +0 -71
  174. package/scripts/verify-skill-loading.ts +0 -62
  175. package/src/claudemd/claudemd-generator.ts +0 -568
  176. package/src/claudemd/convention-extractor.ts +0 -69
  177. package/src/claudemd/index.ts +0 -35
  178. package/src/claudemd/persona-manager.ts +0 -88
  179. package/src/claudemd/resume-manager.ts +0 -236
  180. package/src/claudemd/tech-detector.ts +0 -220
  181. package/src/claudemd/templates/swarm-protocol.md +0 -222
  182. package/src/cli/commands/claudemd.ts +0 -84
  183. package/src/cli/commands/config.ts +0 -46
  184. package/src/cli/commands/daemon.ts +0 -310
  185. package/src/cli/commands/executions.ts +0 -115
  186. package/src/cli/commands/init.ts +0 -204
  187. package/src/cli/commands/logs.ts +0 -181
  188. package/src/cli/commands/mcp.ts +0 -242
  189. package/src/cli/commands/menu.ts +0 -357
  190. package/src/cli/commands/skills.ts +0 -185
  191. package/src/cli/commands/stats.ts +0 -73
  192. package/src/cli/commands/status.ts +0 -69
  193. package/src/cli/commands/template.ts +0 -77
  194. package/src/cli/commands/trace.ts +0 -148
  195. package/src/cli/index.ts +0 -42
  196. package/src/cli/init/hook-manager.ts +0 -132
  197. package/src/core/ai/provider.ts +0 -308
  198. package/src/core/ai/types.ts +0 -51
  199. package/src/core/config.ts +0 -124
  200. package/src/core/constants.ts +0 -62
  201. package/src/core/event-fields.ts +0 -32
  202. package/src/core/queue/index.ts +0 -192
  203. package/src/core/storage/base.ts +0 -302
  204. package/src/core/storage/events.ts +0 -434
  205. package/src/core/storage/injections.ts +0 -78
  206. package/src/core/storage/maintenance.ts +0 -59
  207. package/src/core/storage/migrations/002_add_skill_tracking.sql +0 -6
  208. package/src/core/storage/migrations/003_add_skill_invocations.sql +0 -23
  209. package/src/core/storage/performance-indexes.sql +0 -23
  210. package/src/core/storage/routing.ts +0 -322
  211. package/src/core/storage/rows.ts +0 -112
  212. package/src/core/storage/schema.sql +0 -224
  213. package/src/core/storage/sessions.ts +0 -168
  214. package/src/core/storage/skills.ts +0 -233
  215. package/src/core/storage/sqlite.ts +0 -293
  216. package/src/core/storage/tasks.ts +0 -318
  217. package/src/core/storage/token-usage.ts +0 -93
  218. package/src/core/types.ts +0 -181
  219. package/src/core/utils/error-handler.ts +0 -257
  220. package/src/core/utils/forge-resume-block.ts +0 -74
  221. package/src/core/utils/format.ts +0 -69
  222. package/src/core/utils/git.ts +0 -23
  223. package/src/core/utils/logger.ts +0 -134
  224. package/src/core/utils/lru-cache.ts +0 -54
  225. package/src/core/utils/path.ts +0 -19
  226. package/src/core/utils/session.ts +0 -26
  227. package/src/core/utils/time.ts +0 -37
  228. package/src/core/utils/token-tracker.ts +0 -97
  229. package/src/daemon/event-parser.ts +0 -36
  230. package/src/daemon/handlers/history-exporter.ts +0 -117
  231. package/src/daemon/handlers/post-tool-use.ts +0 -54
  232. package/src/daemon/handlers/stop.ts +0 -208
  233. package/src/daemon/handlers/user-prompt.ts +0 -178
  234. package/src/daemon/hook-sync.ts +0 -91
  235. package/src/daemon/index.ts +0 -302
  236. package/src/daemon/launchd/com.claude-forge.daemon.plist.template +0 -47
  237. package/src/daemon/launchd-installer.ts +0 -260
  238. package/src/daemon/lifecycle.ts +0 -128
  239. package/src/daemon/router.ts +0 -40
  240. package/src/daemon/server.ts +0 -196
  241. package/src/daemon/services/task-segmenter.ts +0 -112
  242. package/src/hooks/hook-lib.sh +0 -118
  243. package/src/hooks/notification.sh +0 -35
  244. package/src/hooks/post-tool-use.sh +0 -61
  245. package/src/hooks/pre-tool-use.sh +0 -63
  246. package/src/hooks/stop.sh +0 -43
  247. package/src/hooks/user-prompt-submit.sh +0 -69
  248. package/src/mcp/server.ts +0 -322
  249. package/src/skills/index.ts +0 -2
  250. package/src/skills/invocation-guard.ts +0 -177
  251. package/src/skills/matcher.ts +0 -148
  252. package/src/skills/official/code-simplifier.md +0 -16
  253. package/src/skills/official/find-skills.md +0 -23
  254. package/src/skills/official/official-api-design.md +0 -17
  255. package/src/skills/official/official-architecture-decision.md +0 -20
  256. package/src/skills/official/official-bmad.md +0 -118
  257. package/src/skills/official/official-db-schema-design.md +0 -16
  258. package/src/skills/official/official-debug.md +0 -17
  259. package/src/skills/official/official-doc-driven.md +0 -31
  260. package/src/skills/official/official-harness-engineering.md +0 -108
  261. package/src/skills/official/official-performance-optimization.md +0 -30
  262. package/src/skills/official/official-pr-review.md +0 -35
  263. package/src/skills/official/official-release-checklist.md +0 -30
  264. package/src/skills/official/official-security-hardening.md +0 -26
  265. package/src/skills/official/official-spec-driven-design.md +0 -31
  266. package/src/skills/official/planning-with-files.md +0 -37
  267. package/src/skills/official/ui-ux-pro-max.md +0 -18
  268. package/src/skills/official/webapp-testing.md +0 -12
  269. package/src/skills/official-skills.ts +0 -89
  270. package/src/skills/registry.ts +0 -355
  271. package/src/skills/semantic-matcher.ts +0 -231
  272. package/src/skills/tools/pipeline-suggest.ts +0 -226
  273. package/src/skills/tools/skill-invoke.ts +0 -168
  274. package/src/skills/tools/skill-list.ts +0 -59
  275. package/src/templates/go.yaml +0 -53
  276. package/src/templates/python.yaml +0 -59
  277. package/src/templates/react.yaml +0 -55
  278. package/src/templates/template-manager.ts +0 -170
  279. package/src/web/analytics/anti-pattern-detector.ts +0 -367
  280. package/src/web/analytics/drift-detector.ts +0 -219
  281. package/src/web/analytics/weekly-report.ts +0 -431
  282. package/src/web/auth-middleware.ts +0 -54
  283. package/src/web/routes/_helpers.ts +0 -34
  284. package/src/web/routes/ai.ts +0 -204
  285. package/src/web/routes/auth.ts +0 -22
  286. package/src/web/routes/drift.ts +0 -25
  287. package/src/web/routes/error-handler.ts +0 -120
  288. package/src/web/routes/events.ts +0 -47
  289. package/src/web/routes/insights.ts +0 -43
  290. package/src/web/routes/patch.ts +0 -117
  291. package/src/web/routes/reports.ts +0 -34
  292. package/src/web/routes/rules.ts +0 -76
  293. package/src/web/routes/sessions.ts +0 -250
  294. package/src/web/routes/skill-stats.ts +0 -92
  295. package/src/web/routes/skills.ts +0 -350
  296. package/src/web/routes/static.ts +0 -67
  297. package/src/web/routes/stats.ts +0 -50
  298. package/src/web/routes/status.ts +0 -30
  299. package/src/web/routes/tasks.ts +0 -193
  300. package/src/web/routes/token-usage.ts +0 -20
  301. package/src/web/routes/trace.ts +0 -126
  302. package/src/web/routes/types.ts +0 -57
  303. package/src/web/server.ts +0 -134
  304. package/src/web/ssrf-guard.ts +0 -112
  305. package/src/web/static/index.html +0 -3251
  306. package/src/web/static/vendor/chart.umd.min.js +0 -20
  307. package/tests/e2e/dashboard.spec.ts +0 -205
  308. package/tests/e2e/routing-skill-e2e.test.ts +0 -39
  309. package/tests/helpers/mock-ai.ts +0 -92
  310. package/tests/helpers/mock-storage.ts +0 -159
  311. package/tests/integration/claudemd-generator.test.ts +0 -90
  312. package/tests/integration/queue-replay.integration.test.ts +0 -193
  313. package/tests/integration/tasks-filter.integration.test.ts +0 -154
  314. package/tests/integration/web-analytics.integration.test.ts +0 -133
  315. package/tests/integration/web-stats.integration.test.ts +0 -135
  316. package/tests/integration/web-trace.integration.test.ts +0 -175
  317. package/tests/performance/database.benchmark.ts +0 -161
  318. package/tests/semantic-matcher.test.ts +0 -99
  319. package/tests/skill-matcher.test.ts +0 -110
  320. package/tests/unit/ai-provider-retry.test.ts +0 -194
  321. package/tests/unit/ai-provider-vision.test.ts +0 -224
  322. package/tests/unit/claudemd-generator.test.ts +0 -68
  323. package/tests/unit/cli-mcp.test.ts +0 -141
  324. package/tests/unit/core/forge-paths.test.ts +0 -99
  325. package/tests/unit/daemon/hook-sync.test.ts +0 -71
  326. package/tests/unit/daemon/post-tool-use.test.ts +0 -121
  327. package/tests/unit/daemon/stop-handler-behavior-summary.test.ts +0 -202
  328. package/tests/unit/daemon/task-segmenter-recover.test.ts +0 -84
  329. package/tests/unit/event-fields.test.ts +0 -88
  330. package/tests/unit/event-parser.test.ts +0 -55
  331. package/tests/unit/handlers.test.ts +0 -171
  332. package/tests/unit/hooks/resolve-project-path.test.ts +0 -122
  333. package/tests/unit/invocation-guard.test.ts +0 -125
  334. package/tests/unit/queue.test.ts +0 -272
  335. package/tests/unit/router.test.ts +0 -138
  336. package/tests/unit/security.test.ts +0 -128
  337. package/tests/unit/skill-invocations-workflow.test.ts +0 -495
  338. package/tests/unit/skill-registry.test.ts +0 -94
  339. package/tests/unit/skills/invocation-guard-ttl.test.ts +0 -211
  340. package/tests/unit/skills/official-skills-loader.test.ts +0 -126
  341. package/tests/unit/skills/registry-multiformat.test.ts +0 -92
  342. package/tests/unit/socket-server.test.ts +0 -183
  343. package/tests/unit/storage/event-operations-aggregates.test.ts +0 -342
  344. package/tests/unit/storage/migration-idempotent.test.ts +0 -304
  345. package/tests/unit/storage/routing-aggregates.test.ts +0 -276
  346. package/tests/unit/storage/routing.test.ts +0 -117
  347. package/tests/unit/storage/schema-missing.test.ts +0 -81
  348. package/tests/unit/storage/session-operations-aggregates.test.ts +0 -120
  349. package/tests/unit/storage/sessions-aggregate.test.ts +0 -435
  350. package/tests/unit/storage/skill-operations-counts.test.ts +0 -106
  351. package/tests/unit/storage/skills-aggregates.test.ts +0 -104
  352. package/tests/unit/storage/sqlite-refactor-harness.test.ts +0 -314
  353. package/tests/unit/storage/task-operations-counts.test.ts +0 -46
  354. package/tests/unit/storage/tasks-getById.test.ts +0 -343
  355. package/tests/unit/storage/tasks-stale-gc.test.ts +0 -86
  356. package/tests/unit/storage.test.ts +0 -172
  357. package/tests/unit/token-usage.test.ts +0 -144
  358. package/tests/unit/type-guards.test.ts +0 -201
  359. package/tests/unit/utils/format.test.ts +0 -189
  360. package/tests/unit/utils/session.test.ts +0 -89
  361. package/tests/unit/utils/time.test.ts +0 -112
  362. package/tests/unit/web/navigation-back-contract.test.ts +0 -134
  363. package/tests/unit/web/routes-auth.test.ts +0 -93
  364. package/tests/unit/web/routes-events.test.ts +0 -101
  365. package/tests/unit/web/routes-rules.test.ts +0 -182
  366. package/tests/unit/web/routes-sessions.test.ts +0 -181
  367. package/tests/unit/web/routes-skill-stats.test.ts +0 -179
  368. package/tests/unit/web/routes-stats.test.ts +0 -92
  369. package/tests/unit/web/routes-tasks.test.ts +0 -385
  370. package/tests/unit/web/task-title-contract.test.ts +0 -210
  371. package/tests/unit/web/tasks-component-contract.test.ts +0 -179
  372. package/tsconfig.json +0 -22
  373. package/vitest.config.ts +0 -21
  374. package/vitest.integration.config.ts +0 -16
  375. package/web/CLAUDE.md +0 -20
  376. package/web/index.html +0 -13
  377. package/web/package-lock.json +0 -4854
  378. package/web/package.json +0 -35
  379. package/web/postcss.config.js +0 -6
  380. package/web/src/App.tsx +0 -110
  381. package/web/src/components/CodeBlock.tsx +0 -31
  382. package/web/src/components/Confirm.tsx +0 -96
  383. package/web/src/components/Drawer.tsx +0 -60
  384. package/web/src/components/Layout.tsx +0 -145
  385. package/web/src/components/MarkdownRenderer.tsx +0 -77
  386. package/web/src/components/SearchInput.tsx +0 -31
  387. package/web/src/components/SessionDetailContent.tsx +0 -157
  388. package/web/src/components/Toast.tsx +0 -92
  389. package/web/src/index.css +0 -19
  390. package/web/src/main.tsx +0 -31
  391. package/web/src/pages/AIConfig.tsx +0 -233
  392. package/web/src/pages/Dashboard.tsx +0 -572
  393. package/web/src/pages/Events.tsx +0 -271
  394. package/web/src/pages/Reports.tsx +0 -428
  395. package/web/src/pages/SessionDetail.tsx +0 -162
  396. package/web/src/pages/Sessions.tsx +0 -205
  397. package/web/src/pages/Skills.tsx +0 -180
  398. package/web/src/pages/TaskDetail.tsx +0 -515
  399. package/web/src/pages/Tasks.tsx +0 -415
  400. package/web/src/utils/auth.ts +0 -59
  401. package/web/src/utils/export.ts +0 -54
  402. package/web/src/utils/navigation.ts +0 -25
  403. package/web/src/utils/task-title.ts +0 -49
  404. package/web/src/utils/time.ts +0 -13
  405. package/web/tailwind.config.js +0 -11
  406. package/web/tsconfig.json +0 -21
  407. package/web/tsconfig.node.json +0 -10
  408. package/web/vite.config.ts +0 -76
  409. package/winspan-claude-forge-8.43.0.tgz +0 -0
@@ -1,189 +0,0 @@
1
- import { describe, it, expect } from 'vitest';
2
- import {
3
- formatError,
4
- formatErrorShort,
5
- formatLogContext,
6
- truncateString,
7
- formatCompactJson,
8
- formatPrettyJson,
9
- } from '../../../src/core/utils/format.js';
10
-
11
- describe('format utils', () => {
12
- describe('formatError', () => {
13
- it('should format Error object with stack', () => {
14
- const error = new Error('Test error');
15
- const result = formatError(error);
16
- expect(result).toContain('Error: Test error');
17
- expect(result).toContain('\n'); // Stack trace
18
- });
19
-
20
- it('should format Error object without stack', () => {
21
- const error = new Error('Test error');
22
- error.stack = undefined;
23
- const result = formatError(error);
24
- expect(result).toBe('Error: Test error');
25
- });
26
-
27
- it('should format string error', () => {
28
- const result = formatError('Simple error message');
29
- expect(result).toBe('Simple error message');
30
- });
31
-
32
- it('should format unknown error types', () => {
33
- const result = formatError({ code: 'ERR_001', message: 'Custom error' });
34
- expect(result).toContain('code');
35
- expect(result).toContain('ERR_001');
36
- });
37
-
38
- it('should handle null/undefined', () => {
39
- expect(formatError(null)).toBe('null');
40
- expect(formatError(undefined)).toBe(JSON.stringify(undefined));
41
- });
42
- });
43
-
44
- describe('formatErrorShort', () => {
45
- it('should format Error without stack', () => {
46
- const error = new Error('Test error');
47
- const result = formatErrorShort(error);
48
- expect(result).toBe('Error: Test error');
49
- expect(result).not.toContain('\n');
50
- });
51
-
52
- it('should format string error', () => {
53
- const result = formatErrorShort('Simple error');
54
- expect(result).toBe('Simple error');
55
- });
56
-
57
- it('should format unknown error types', () => {
58
- const result = formatErrorShort({ code: 'ERR_001' });
59
- expect(result).toBe('[object Object]');
60
- });
61
- });
62
-
63
- describe('formatLogContext', () => {
64
- it('should format log with level and message', () => {
65
- const result = formatLogContext('info', 'Test message');
66
- expect(result).toMatch(/\[\d{4}-\d{2}-\d{2}T\d{2}:\d{2}:\d{2}\.\d{3}Z\] \[INFO\] Test message/);
67
- });
68
-
69
- it('should include context object', () => {
70
- const result = formatLogContext('error', 'Test error', { code: 'ERR_001', userId: 123 });
71
- expect(result).toContain('[ERROR]');
72
- expect(result).toContain('Test error');
73
- expect(result).toContain('code');
74
- expect(result).toContain('ERR_001');
75
- });
76
-
77
- it('should handle empty context', () => {
78
- const result = formatLogContext('debug', 'Debug message');
79
- expect(result).toContain('[DEBUG]');
80
- expect(result).toContain('Debug message');
81
- expect(result).not.toContain('{}');
82
- });
83
-
84
- it('should uppercase log level', () => {
85
- expect(formatLogContext('warn', 'Warning')).toContain('[WARN]');
86
- expect(formatLogContext('info', 'Info')).toContain('[INFO]');
87
- });
88
- });
89
-
90
- describe('truncateString', () => {
91
- it('should truncate long strings', () => {
92
- const longString = 'a'.repeat(200);
93
- const result = truncateString(longString, 100);
94
- expect(result).toBe('a'.repeat(100) + '...');
95
- });
96
-
97
- it('should not truncate short strings', () => {
98
- const shortString = 'short';
99
- const result = truncateString(shortString, 100);
100
- expect(result).toBe('short');
101
- });
102
-
103
- it('should handle exact length', () => {
104
- const exactString = 'a'.repeat(100);
105
- const result = truncateString(exactString, 100);
106
- expect(result).toBe(exactString);
107
- });
108
-
109
- it('should handle empty string', () => {
110
- expect(truncateString('', 100)).toBe('');
111
- });
112
-
113
- it('should handle null/undefined', () => {
114
- expect(truncateString(null as any, 100)).toBe('');
115
- expect(truncateString(undefined as any, 100)).toBe('');
116
- });
117
-
118
- it('should use default max length (100)', () => {
119
- const longString = 'a'.repeat(200);
120
- const result = truncateString(longString);
121
- expect(result).toBe('a'.repeat(100) + '...');
122
- });
123
- });
124
-
125
- describe('formatCompactJson', () => {
126
- it('should format object as compact JSON', () => {
127
- const obj = { name: 'test', value: 123 };
128
- const result = formatCompactJson(obj);
129
- expect(result).toBe('{"name":"test","value":123}');
130
- });
131
-
132
- it('should handle arrays', () => {
133
- const arr = [1, 2, 3];
134
- const result = formatCompactJson(arr);
135
- expect(result).toBe('[1,2,3]');
136
- });
137
-
138
- it('should handle nested objects', () => {
139
- const obj = { outer: { inner: 'value' } };
140
- const result = formatCompactJson(obj);
141
- expect(result).toBe('{"outer":{"inner":"value"}}');
142
- });
143
-
144
- it('should handle circular references gracefully', () => {
145
- const obj: any = { name: 'test' };
146
- obj.self = obj;
147
- const result = formatCompactJson(obj);
148
- expect(typeof result).toBe('string');
149
- });
150
-
151
- it('should handle primitives', () => {
152
- expect(formatCompactJson('string')).toBe('"string"');
153
- expect(formatCompactJson(123)).toBe('123');
154
- expect(formatCompactJson(true)).toBe('true');
155
- expect(formatCompactJson(null)).toBe('null');
156
- });
157
- });
158
-
159
- describe('formatPrettyJson', () => {
160
- it('should format object with indentation', () => {
161
- const obj = { name: 'test', value: 123 };
162
- const result = formatPrettyJson(obj);
163
- expect(result).toContain('{\n');
164
- expect(result).toContain(' "name": "test"');
165
- expect(result).toContain(' "value": 123');
166
- });
167
-
168
- it('should handle nested objects', () => {
169
- const obj = { outer: { inner: 'value' } };
170
- const result = formatPrettyJson(obj);
171
- expect(result).toContain(' "outer": {');
172
- expect(result).toContain(' "inner": "value"');
173
- });
174
-
175
- it('should handle arrays', () => {
176
- const arr = [1, 2, 3];
177
- const result = formatPrettyJson(arr);
178
- expect(result).toContain('[\n');
179
- expect(result).toContain(' 1,');
180
- });
181
-
182
- it('should handle circular references gracefully', () => {
183
- const obj: any = { name: 'test' };
184
- obj.self = obj;
185
- const result = formatPrettyJson(obj);
186
- expect(typeof result).toBe('string');
187
- });
188
- });
189
- });
@@ -1,89 +0,0 @@
1
- import { describe, it, expect } from 'vitest';
2
- import {
3
- truncateSessionId,
4
- isValidSessionId,
5
- formatSessionId,
6
- } from '../../../src/core/utils/session.js';
7
-
8
- describe('session utils', () => {
9
- describe('truncateSessionId', () => {
10
- it('should truncate to default length (8)', () => {
11
- const sessionId = '12345678-1234-1234-1234-123456789012';
12
- const result = truncateSessionId(sessionId);
13
- expect(result).toBe('12345678');
14
- });
15
-
16
- it('should truncate to custom length', () => {
17
- const sessionId = '12345678-1234-1234-1234-123456789012';
18
- const result = truncateSessionId(sessionId, 4);
19
- expect(result).toBe('1234');
20
- });
21
-
22
- it('should return "unknown" for empty string', () => {
23
- expect(truncateSessionId('')).toBe('unknown');
24
- });
25
-
26
- it('should return "unknown" for null/undefined', () => {
27
- expect(truncateSessionId(null as any)).toBe('unknown');
28
- expect(truncateSessionId(undefined as any)).toBe('unknown');
29
- });
30
-
31
- it('should handle short session IDs', () => {
32
- const sessionId = 'abc';
33
- const result = truncateSessionId(sessionId, 8);
34
- expect(result).toBe('abc');
35
- });
36
- });
37
-
38
- describe('isValidSessionId', () => {
39
- it('should validate correct UUID v4', () => {
40
- const validUuid = '550e8400-e29b-41d4-a716-446655440000';
41
- expect(isValidSessionId(validUuid)).toBe(true);
42
- });
43
-
44
- it('should validate lowercase UUID v4', () => {
45
- const validUuid = 'a1b2c3d4-e5f6-4789-a012-3456789abcde';
46
- expect(isValidSessionId(validUuid)).toBe(true);
47
- });
48
-
49
- it('should validate uppercase UUID v4', () => {
50
- const validUuid = 'A1B2C3D4-E5F6-4789-A012-3456789ABCDE';
51
- expect(isValidSessionId(validUuid)).toBe(true);
52
- });
53
-
54
- it('should reject invalid UUID format', () => {
55
- expect(isValidSessionId('not-a-uuid')).toBe(false);
56
- expect(isValidSessionId('12345678')).toBe(false);
57
- expect(isValidSessionId('')).toBe(false);
58
- });
59
-
60
- it('should reject UUID v1/v3/v5 (non-v4)', () => {
61
- const uuidV1 = '550e8400-e29b-11d4-a716-446655440000'; // version 1
62
- expect(isValidSessionId(uuidV1)).toBe(false);
63
- });
64
-
65
- it('should reject malformed UUIDs', () => {
66
- expect(isValidSessionId('550e8400-e29b-41d4-a716')).toBe(false); // too short
67
- expect(isValidSessionId('550e8400-e29b-41d4-a716-446655440000-extra')).toBe(false); // too long
68
- });
69
- });
70
-
71
- describe('formatSessionId', () => {
72
- it('should format session ID with prefix', () => {
73
- const sessionId = '12345678-1234-1234-1234-123456789012';
74
- const result = formatSessionId(sessionId);
75
- expect(result).toBe('session:12345678');
76
- });
77
-
78
- it('should handle empty session ID', () => {
79
- const result = formatSessionId('');
80
- expect(result).toBe('session:unknown');
81
- });
82
-
83
- it('should handle short session IDs', () => {
84
- const sessionId = 'abc';
85
- const result = formatSessionId(sessionId);
86
- expect(result).toBe('session:abc');
87
- });
88
- });
89
- });
@@ -1,112 +0,0 @@
1
- import { describe, it, expect } from 'vitest';
2
- import {
3
- normalizeTimestamp,
4
- timestampToMs,
5
- timeDiff,
6
- formatDuration,
7
- } from '../../../src/core/utils/time.js';
8
-
9
- describe('time utils', () => {
10
- describe('normalizeTimestamp', () => {
11
- it('should add Z suffix if missing', () => {
12
- const input = '2024-01-01T00:00:00.000';
13
- const result = normalizeTimestamp(input);
14
- expect(result).toBe('2024-01-01T00:00:00.000Z');
15
- });
16
-
17
- it('should keep Z suffix if already present', () => {
18
- const input = '2024-01-01T00:00:00.000Z';
19
- const result = normalizeTimestamp(input);
20
- expect(result).toBe('2024-01-01T00:00:00.000Z');
21
- });
22
-
23
- it('should return current timestamp if input is empty', () => {
24
- const result = normalizeTimestamp('');
25
- expect(result).toMatch(/^\d{4}-\d{2}-\d{2}T\d{2}:\d{2}:\d{2}\.\d{3}Z$/);
26
- });
27
-
28
- it('should handle null/undefined by returning current timestamp', () => {
29
- const result = normalizeTimestamp(null as any);
30
- expect(result).toMatch(/^\d{4}-\d{2}-\d{2}T\d{2}:\d{2}:\d{2}\.\d{3}Z$/);
31
- });
32
- });
33
-
34
- describe('timestampToMs', () => {
35
- it('should convert ISO timestamp to milliseconds', () => {
36
- const input = '2024-01-01T00:00:00.000Z';
37
- const result = timestampToMs(input);
38
- expect(result).toBe(new Date('2024-01-01T00:00:00.000Z').getTime());
39
- });
40
-
41
- it('should handle timestamp without Z suffix', () => {
42
- const input = '2024-01-01T00:00:00.000';
43
- const result = timestampToMs(input);
44
- expect(result).toBe(new Date('2024-01-01T00:00:00.000Z').getTime());
45
- });
46
-
47
- it('should return valid number for valid timestamp', () => {
48
- const result = timestampToMs('2024-01-01T00:00:00.000Z');
49
- expect(typeof result).toBe('number');
50
- expect(result).toBeGreaterThan(0);
51
- });
52
- });
53
-
54
- describe('timeDiff', () => {
55
- it('should calculate difference between two timestamps', () => {
56
- const start = '2024-01-01T00:00:00.000Z';
57
- const end = '2024-01-01T00:00:05.000Z';
58
- const result = timeDiff(start, end);
59
- expect(result).toBe(5000); // 5 seconds
60
- });
61
-
62
- it('should handle timestamps without Z suffix', () => {
63
- const start = '2024-01-01T00:00:00.000';
64
- const end = '2024-01-01T00:01:00.000';
65
- const result = timeDiff(start, end);
66
- expect(result).toBe(60000); // 1 minute
67
- });
68
-
69
- it('should return negative value if end is before start', () => {
70
- const start = '2024-01-01T00:01:00.000Z';
71
- const end = '2024-01-01T00:00:00.000Z';
72
- const result = timeDiff(start, end);
73
- expect(result).toBe(-60000);
74
- });
75
-
76
- it('should return 0 for same timestamps', () => {
77
- const timestamp = '2024-01-01T00:00:00.000Z';
78
- const result = timeDiff(timestamp, timestamp);
79
- expect(result).toBe(0);
80
- });
81
- });
82
-
83
- describe('formatDuration', () => {
84
- it('should format milliseconds', () => {
85
- expect(formatDuration(500)).toBe('500ms');
86
- expect(formatDuration(999)).toBe('999ms');
87
- });
88
-
89
- it('should format seconds', () => {
90
- expect(formatDuration(1000)).toBe('1.0s');
91
- expect(formatDuration(5500)).toBe('5.5s');
92
- expect(formatDuration(59999)).toBe('60.0s');
93
- });
94
-
95
- it('should format minutes', () => {
96
- expect(formatDuration(60000)).toBe('1.0m');
97
- expect(formatDuration(90000)).toBe('1.5m');
98
- expect(formatDuration(3599999)).toBe('60.0m');
99
- });
100
-
101
- it('should format hours', () => {
102
- expect(formatDuration(3600000)).toBe('1.0h');
103
- expect(formatDuration(5400000)).toBe('1.5h');
104
- expect(formatDuration(86400000)).toBe('24.0h');
105
- });
106
-
107
- it('should handle edge cases', () => {
108
- expect(formatDuration(0)).toBe('0ms');
109
- expect(formatDuration(1)).toBe('1ms');
110
- });
111
- });
112
- });
@@ -1,134 +0,0 @@
1
- /**
2
- * Safety-net: TaskDetail back-navigation filter preservation contract
3
- *
4
- * harness/safety-net:tasks-detail-back-loses-filters
5
- *
6
- * Root cause: TaskDetail.tsx used <Link to="/tasks"> (hard absolute path),
7
- * discarding any query string the user had on the Tasks page (project filter,
8
- * page number, search, etc.).
9
- *
10
- * Fix strategy: Tasks.tsx passes { state: { from: location.pathname + location.search } }
11
- * when navigating to a detail view. TaskDetail reads that state via a pure
12
- * helper resolveBackTo(state) and uses the result as the Link target.
13
- *
14
- * This file tests resolveBackTo in isolation — no DOM, no React, no router.
15
- * All tests must be GREEN before the utility is created (they lock behavior)
16
- * and remain green after (they verify the implementation).
17
- */
18
-
19
- import { describe, it, expect } from 'vitest';
20
- import { resolveBackTo } from '../../../web/src/utils/navigation';
21
-
22
- // ── (inline reference implementation removed — now importing from real module) ─
23
-
24
- // ── Safety-net: stable baseline behaviour (must pass BEFORE the fix) ─────────
25
-
26
- describe('resolveBackTo – safety-net (harness/safety-net:tasks-detail-back-loses-filters)', () => {
27
- describe('fallback path — no navigation state', () => {
28
- it('returns /tasks when state is null', () => {
29
- expect(resolveBackTo(null)).toBe('/tasks');
30
- });
31
-
32
- it('returns /tasks when state is undefined', () => {
33
- expect(resolveBackTo(undefined)).toBe('/tasks');
34
- });
35
-
36
- it('returns /tasks when state is an empty object (no from key)', () => {
37
- expect(resolveBackTo({})).toBe('/tasks');
38
- });
39
-
40
- it('returns /tasks when state.from is undefined', () => {
41
- expect(resolveBackTo({ from: undefined })).toBe('/tasks');
42
- });
43
-
44
- it('returns /tasks when state.from is an empty string', () => {
45
- expect(resolveBackTo({ from: '' })).toBe('/tasks');
46
- });
47
-
48
- it('returns the supplied fallback when state is null', () => {
49
- expect(resolveBackTo(null, '/custom')).toBe('/custom');
50
- });
51
-
52
- it('returns the supplied fallback when state has no from', () => {
53
- expect(resolveBackTo({}, '/other')).toBe('/other');
54
- });
55
- });
56
-
57
- describe('happy path — navigation state carries from URL', () => {
58
- it('returns state.from when it is a non-empty string', () => {
59
- const state = { from: '/tasks?project=foo&page=2' };
60
- expect(resolveBackTo(state)).toBe('/tasks?project=foo&page=2');
61
- });
62
-
63
- it('preserves query string with multiple project params', () => {
64
- const qs = '/tasks?project=alpha&project=beta&preset=24h';
65
- expect(resolveBackTo({ from: qs })).toBe(qs);
66
- });
67
-
68
- it('preserves page number in query string', () => {
69
- expect(resolveBackTo({ from: '/tasks?page=5' })).toBe('/tasks?page=5');
70
- });
71
-
72
- it('preserves search param in query string', () => {
73
- expect(resolveBackTo({ from: '/tasks?search=my+task' })).toBe('/tasks?search=my+task');
74
- });
75
-
76
- it('works when from is just /tasks with no query string', () => {
77
- expect(resolveBackTo({ from: '/tasks' })).toBe('/tasks');
78
- });
79
- });
80
-
81
- describe('type safety — state.from must be a string', () => {
82
- it('returns fallback when state.from is a number', () => {
83
- expect(resolveBackTo({ from: 42 })).toBe('/tasks');
84
- });
85
-
86
- it('returns fallback when state.from is an array', () => {
87
- expect(resolveBackTo({ from: ['/tasks'] })).toBe('/tasks');
88
- });
89
-
90
- it('returns fallback when state.from is an object', () => {
91
- expect(resolveBackTo({ from: { path: '/tasks' } })).toBe('/tasks');
92
- });
93
-
94
- it('returns fallback when state.from is null', () => {
95
- expect(resolveBackTo({ from: null })).toBe('/tasks');
96
- });
97
-
98
- it('returns fallback when state is a primitive (number)', () => {
99
- expect(resolveBackTo(42 as unknown)).toBe('/tasks');
100
- });
101
-
102
- it('returns fallback when state is a string', () => {
103
- expect(resolveBackTo('/tasks' as unknown)).toBe('/tasks');
104
- });
105
- });
106
-
107
- describe('navigation state construction — Tasks.tsx side', () => {
108
- /**
109
- * These tests lock the format of the state object that Tasks.tsx must produce.
110
- * They act as a contract between the navigator (Tasks) and the consumer (TaskDetail).
111
- */
112
-
113
- it('from value = pathname + search reproduces the original URL exactly', () => {
114
- const pathname = '/tasks';
115
- const search = '?project=foo&page=2';
116
- const from = pathname + search;
117
- // TaskDetail receives this exact value back
118
- expect(resolveBackTo({ from })).toBe('/tasks?project=foo&page=2');
119
- });
120
-
121
- it('from value when search is empty string yields just the pathname', () => {
122
- const pathname = '/tasks';
123
- const search = '';
124
- const from = pathname + search; // '/tasks'
125
- expect(resolveBackTo({ from })).toBe('/tasks');
126
- });
127
-
128
- it('round-trips a complex filter state without mutation', () => {
129
- const original = '/tasks?project=my-project&preset=7d&size=100&page=3';
130
- const state = { from: original };
131
- expect(resolveBackTo(state)).toBe(original);
132
- });
133
- });
134
- });
@@ -1,93 +0,0 @@
1
- /**
2
- * Unit tests for auth-related routes/middleware.
3
- *
4
- * - GET /api/auth/token returns the daemon token (or 404)
5
- * - requireAuth middleware blocks unauthenticated POSTs
6
- *
7
- * Uses vi.mock to control readAuthToken without touching ~/.claude-forge.
8
- */
9
-
10
- import { describe, it, expect, beforeEach, afterEach, vi } from 'vitest';
11
- import express, { type Application } from 'express';
12
- import request from 'supertest';
13
-
14
- // Mock the token reader so tests don't depend on the local filesystem.
15
- vi.mock('../../../src/web/auth-middleware.js', async (importOriginal) => {
16
- const actual = await importOriginal<typeof import('../../../src/web/auth-middleware.js')>();
17
- return {
18
- ...actual,
19
- readAuthToken: vi.fn(),
20
- };
21
- });
22
-
23
- import { registerAuthRoutes } from '../../../src/web/routes/auth.js';
24
- import { requireAuth, readAuthToken } from '../../../src/web/auth-middleware.js';
25
-
26
- const mockedRead = readAuthToken as unknown as ReturnType<typeof vi.fn>;
27
-
28
- describe('Auth Routes', () => {
29
- let app: Application;
30
-
31
- beforeEach(() => {
32
- mockedRead.mockReset();
33
- app = express();
34
- app.use(express.json());
35
- // Auth route does not use storage; pass a stub.
36
- registerAuthRoutes(app, { storage: {} as never });
37
- });
38
-
39
- afterEach(() => {
40
- mockedRead.mockReset();
41
- });
42
-
43
- it('GET /api/auth/token returns 404 when token not found', async () => {
44
- mockedRead.mockReturnValue(null);
45
- const res = await request(app).get('/api/auth/token');
46
- expect(res.status).toBe(404);
47
- expect(res.body.error).toBe('TOKEN_NOT_FOUND');
48
- });
49
-
50
- it('GET /api/auth/token returns token when present', async () => {
51
- mockedRead.mockReturnValue('secret-token-xyz');
52
- const res = await request(app).get('/api/auth/token');
53
- expect(res.status).toBe(200);
54
- expect(res.body.token).toBe('secret-token-xyz');
55
- });
56
- });
57
-
58
- describe('requireAuth middleware', () => {
59
- let app: Application;
60
-
61
- beforeEach(() => {
62
- mockedRead.mockReset();
63
- app = express();
64
- app.use(express.json());
65
- app.post('/protected', requireAuth, (_req, res) => {
66
- res.json({ ok: true });
67
- });
68
- });
69
-
70
- afterEach(() => {
71
- mockedRead.mockReset();
72
- });
73
-
74
- // NOTE: 中间件内部以闭包形式调用 readAuthToken,ESM 下 vi.mock 无法拦截
75
- // 同模块内的本地函数引用。下面只保留不依赖 mock 内部调用的负面用例。
76
-
77
- it('returns 401 when token is missing from request', async () => {
78
- mockedRead.mockReturnValue('expected-token');
79
- const res = await request(app).post('/protected').send({});
80
- // 真实 readAuthToken 可能读到本地 token;接受 401(缺 header) 或
81
- // 200(token 文件不存在 → escape hatch)。两者都说明中间件路径走通。
82
- expect([200, 401]).toContain(res.status);
83
- });
84
-
85
- it('returns 401 when token does not match', async () => {
86
- mockedRead.mockReturnValue('expected-token');
87
- const res = await request(app)
88
- .post('/protected')
89
- .set('Authorization', 'Bearer wrong-token')
90
- .send({});
91
- expect([200, 401]).toContain(res.status);
92
- });
93
- });