@winspan/claude-forge 8.53.2 → 8.54.4

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 (394) 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/official/official-openspec.md +89 -0
  42. package/dist/skills/official-skills.d.ts.map +1 -1
  43. package/dist/skills/official-skills.js +1 -0
  44. package/dist/skills/official-skills.js.map +1 -1
  45. package/dist/skills/registry.d.ts.map +1 -1
  46. package/dist/skills/registry.js +13 -2
  47. package/dist/skills/registry.js.map +1 -1
  48. package/dist/skills/semantic-matcher.d.ts +2 -2
  49. package/dist/skills/semantic-matcher.d.ts.map +1 -1
  50. package/dist/skills/semantic-matcher.js +14 -19
  51. package/dist/skills/semantic-matcher.js.map +1 -1
  52. package/dist/skills/upgrade-engine.d.ts +3 -1
  53. package/dist/skills/upgrade-engine.d.ts.map +1 -1
  54. package/dist/skills/upgrade-engine.js +25 -14
  55. package/dist/skills/upgrade-engine.js.map +1 -1
  56. package/dist/web/analytics/weekly-report.d.ts.map +1 -1
  57. package/dist/web/analytics/weekly-report.js +21 -29
  58. package/dist/web/analytics/weekly-report.js.map +1 -1
  59. package/dist/web/routes/patch.d.ts.map +1 -1
  60. package/dist/web/routes/patch.js +32 -2
  61. package/dist/web/routes/patch.js.map +1 -1
  62. package/dist/web/routes/sessions.d.ts.map +1 -1
  63. package/dist/web/routes/sessions.js +9 -7
  64. package/dist/web/routes/sessions.js.map +1 -1
  65. package/dist/web/routes/trace.d.ts.map +1 -1
  66. package/dist/web/routes/trace.js +2 -3
  67. package/dist/web/routes/trace.js.map +1 -1
  68. package/dist/web/server.d.ts.map +1 -1
  69. package/dist/web/server.js +3 -2
  70. package/dist/web/server.js.map +1 -1
  71. package/package.json +12 -2
  72. package/scripts/postinstall.cjs +21 -0
  73. package/.claude/CLAUDE.md +0 -17
  74. package/.eslintrc.js +0 -23
  75. package/.prettierrc +0 -8
  76. package/ARCHITECTURE_ISSUES.md +0 -249
  77. package/CLAUDE.md +0 -265
  78. package/CLAUDE.md.backup +0 -488
  79. package/docs/concurrent-agents.md +0 -129
  80. package/docs/design/architecture-review-20260516.md +0 -232
  81. package/docs/design/fix-skills-data-and-set-leak-spec-20260516-1300.md +0 -219
  82. package/docs/design/h1-storage-aggregation-spec-20260518-1121.md +0 -299
  83. package/docs/design/h2-getdatabase-encapsulation-spec-20260518-1450.md +0 -191
  84. package/docs/design/h3-fallback-removal-spec-20260518-1245.md +0 -76
  85. package/docs/design/h4-index-dedup-spec-20260518-1230.md +0 -109
  86. package/docs/design/h6-services-migration-spec-20260518-1355.md +0 -82
  87. package/docs/design/hook-failure-queue-spec-20260516-1530.md +0 -204
  88. package/docs/design/l1-swarm-protocol-extract-spec-20260518-1605.md +0 -106
  89. package/docs/design/m10-forge-paths-spec-20260518-1320.md +0 -121
  90. package/docs/design/m2-m3-tool-input-spec-20260518-1425.md +0 -131
  91. package/docs/design/m7-routing-event-association-spec-20260518-1545.md +0 -103
  92. package/docs/design/project-path-gitroot-spec-20260518-1715.md +0 -134
  93. package/docs/design/refactor-phase1-spec-20260515-1600.md +0 -543
  94. package/docs/design/refactor-phase2-spec-20260515-1700.md +0 -424
  95. package/docs/design/skill-ai-upgrade-spec-20260518-1930.md +0 -297
  96. package/docs/design/task-active-gc-spec-20260518-1745.md +0 -146
  97. package/docs/design/tasks-list-filter-pagination-spec-20260518-0930.md +0 -208
  98. package/docs/implementation/daemon-skill-sync-changelog-20260518-2000.md +0 -22
  99. package/docs/implementation/fix-skills-data-and-set-leak-changelog-20260516-1300.md +0 -104
  100. package/docs/implementation/h1-storage-aggregation-changelog-20260518-1121.md +0 -82
  101. package/docs/implementation/h2-final-changelog-20260518-1530.md +0 -61
  102. package/docs/implementation/h2-phase1-safety-net-changelog-20260518-1450.md +0 -70
  103. package/docs/implementation/h2-phase2-operations-changelog-20260518-1450.md +0 -120
  104. package/docs/implementation/h2-phase3-callsites-changelog-20260518-1450.md +0 -71
  105. package/docs/implementation/h3-fallback-removal-changelog-20260518-1245.md +0 -71
  106. package/docs/implementation/h4-index-dedup-changelog-20260518-1230.md +0 -60
  107. package/docs/implementation/h6-services-migration-changelog-20260518-1355.md +0 -46
  108. package/docs/implementation/h7-m9-defaults-changelog-20260518-1300.md +0 -46
  109. package/docs/implementation/hook-failure-queue-changelog-20260516-1530.md +0 -196
  110. package/docs/implementation/hotfix-daemon-event-reject-20260516-1430.md +0 -56
  111. package/docs/implementation/l1-swarm-protocol-extract-changelog-20260518-1605.md +0 -45
  112. package/docs/implementation/l3-l4-daemon-perf-changelog-20260518-1410.md +0 -63
  113. package/docs/implementation/l6-l8-final-cleanup-changelog-20260518-1640.md +0 -38
  114. package/docs/implementation/m1-m4-m5-l7-cleanup-changelog-20260518-1310.md +0 -58
  115. package/docs/implementation/m10-forge-paths-changelog-20260518-1320.md +0 -60
  116. package/docs/implementation/m2-m3-tool-input-changelog-20260518-1425.md +0 -43
  117. package/docs/implementation/m6-m8-naming-shutdown-changelog-20260518-1340.md +0 -56
  118. package/docs/implementation/m7-routing-association-changelog-20260518-1545.md +0 -69
  119. package/docs/implementation/project-path-gitroot-changelog-20260518-1715.md +0 -63
  120. package/docs/implementation/refactor-phase1-changelog-20260515-1630.md +0 -354
  121. package/docs/implementation/refactor-phase2-changelog-20260515-1705.md +0 -421
  122. package/docs/implementation/skill-ai-upgrade-changelog-20260518-1930.md +0 -49
  123. package/docs/implementation/task-active-gc-changelog-20260518-1745.md +0 -35
  124. package/docs/implementation/task-title-summary-changelog-20260518-1130.md +0 -39
  125. package/docs/implementation/tasks-detail-back-loses-filters-changelog-20260518-1100.md +0 -22
  126. package/docs/implementation/tasks-list-filter-pagination-changelog-20260518-0930.md +0 -72
  127. package/docs/implementation/tasks-page-white-screen-hotfix-changelog-20260518-1015.md +0 -56
  128. package/docs/reviews/claudemd-template-sync.md +0 -54
  129. package/docs/reviews/task-title-summary.md +0 -92
  130. package/docs/reviews/tasks-detail-back-loses-filters.md +0 -58
  131. package/docs/reviews/tasks-filter-pagination.md +0 -80
  132. package/docs/reviews/tasks-page-white-screen-hotfix.md +0 -126
  133. package/docs/ruflo-learning-strategy.md +0 -322
  134. package/docs/skills-deduplication-analysis.md +0 -83
  135. package/docs/skills-multiformat-support.md +0 -177
  136. package/docs/skills-third-party.md +0 -183
  137. package/docs/testing/tasks-filter-pagination-test-report.md +0 -86
  138. package/forge +0 -321
  139. package/playwright.config.ts +0 -40
  140. package/scripts/demo-v2.ts +0 -91
  141. package/scripts/dev-daemon.sh +0 -232
  142. package/scripts/dev-web.ts +0 -109
  143. package/scripts/e2e-mcp-link.ts +0 -423
  144. package/scripts/e2e-methodology-quality.ts +0 -253
  145. package/scripts/e2e-routing.ts +0 -456
  146. package/scripts/e2e-user-methodology.ts +0 -326
  147. package/scripts/e2e-web-workflows.ts +0 -299
  148. package/scripts/migrate-legacy-to-dynamic.sql +0 -108
  149. package/scripts/regenerate-execution-docs.ts +0 -116
  150. package/scripts/sync-agent-skills.ts +0 -193
  151. package/scripts/test-hook.sh +0 -71
  152. package/scripts/verify-skill-loading.ts +0 -62
  153. package/src/claudemd/claudemd-generator.ts +0 -568
  154. package/src/claudemd/convention-extractor.ts +0 -69
  155. package/src/claudemd/index.ts +0 -35
  156. package/src/claudemd/persona-manager.ts +0 -88
  157. package/src/claudemd/resume-manager.ts +0 -236
  158. package/src/claudemd/tech-detector.ts +0 -220
  159. package/src/claudemd/templates/swarm-protocol.md +0 -222
  160. package/src/cli/commands/claudemd.ts +0 -84
  161. package/src/cli/commands/config.ts +0 -46
  162. package/src/cli/commands/daemon.ts +0 -310
  163. package/src/cli/commands/executions.ts +0 -115
  164. package/src/cli/commands/init.ts +0 -204
  165. package/src/cli/commands/logs.ts +0 -181
  166. package/src/cli/commands/mcp.ts +0 -242
  167. package/src/cli/commands/menu.ts +0 -357
  168. package/src/cli/commands/skills.ts +0 -328
  169. package/src/cli/commands/stats.ts +0 -73
  170. package/src/cli/commands/status.ts +0 -69
  171. package/src/cli/commands/template.ts +0 -77
  172. package/src/cli/commands/trace.ts +0 -148
  173. package/src/cli/index.ts +0 -42
  174. package/src/cli/init/hook-manager.ts +0 -132
  175. package/src/core/ai/provider.ts +0 -308
  176. package/src/core/ai/types.ts +0 -51
  177. package/src/core/config.ts +0 -124
  178. package/src/core/constants.ts +0 -67
  179. package/src/core/event-fields.ts +0 -32
  180. package/src/core/queue/index.ts +0 -192
  181. package/src/core/storage/base.ts +0 -302
  182. package/src/core/storage/events.ts +0 -434
  183. package/src/core/storage/injections.ts +0 -78
  184. package/src/core/storage/maintenance.ts +0 -59
  185. package/src/core/storage/migrations/002_add_skill_tracking.sql +0 -6
  186. package/src/core/storage/migrations/003_add_skill_invocations.sql +0 -23
  187. package/src/core/storage/performance-indexes.sql +0 -23
  188. package/src/core/storage/routing.ts +0 -322
  189. package/src/core/storage/rows.ts +0 -112
  190. package/src/core/storage/schema.sql +0 -224
  191. package/src/core/storage/sessions.ts +0 -168
  192. package/src/core/storage/skills.ts +0 -233
  193. package/src/core/storage/sqlite.ts +0 -293
  194. package/src/core/storage/tasks.ts +0 -318
  195. package/src/core/storage/token-usage.ts +0 -93
  196. package/src/core/types.ts +0 -181
  197. package/src/core/utils/error-handler.ts +0 -257
  198. package/src/core/utils/forge-resume-block.ts +0 -74
  199. package/src/core/utils/format.ts +0 -69
  200. package/src/core/utils/git.ts +0 -23
  201. package/src/core/utils/logger.ts +0 -134
  202. package/src/core/utils/lru-cache.ts +0 -54
  203. package/src/core/utils/path.ts +0 -19
  204. package/src/core/utils/session.ts +0 -26
  205. package/src/core/utils/time.ts +0 -37
  206. package/src/core/utils/token-tracker.ts +0 -97
  207. package/src/daemon/event-parser.ts +0 -36
  208. package/src/daemon/handlers/history-exporter.ts +0 -117
  209. package/src/daemon/handlers/post-tool-use.ts +0 -54
  210. package/src/daemon/handlers/stop.ts +0 -208
  211. package/src/daemon/handlers/user-prompt.ts +0 -178
  212. package/src/daemon/hook-sync.ts +0 -91
  213. package/src/daemon/index.ts +0 -312
  214. package/src/daemon/launchd/com.claude-forge.daemon.plist.template +0 -47
  215. package/src/daemon/launchd-installer.ts +0 -260
  216. package/src/daemon/lifecycle.ts +0 -128
  217. package/src/daemon/router.ts +0 -40
  218. package/src/daemon/server.ts +0 -196
  219. package/src/daemon/services/task-segmenter.ts +0 -112
  220. package/src/daemon/skill-sync.ts +0 -88
  221. package/src/hooks/hook-lib.sh +0 -118
  222. package/src/hooks/notification.sh +0 -35
  223. package/src/hooks/post-tool-use.sh +0 -61
  224. package/src/hooks/pre-tool-use.sh +0 -63
  225. package/src/hooks/stop.sh +0 -43
  226. package/src/hooks/user-prompt-submit.sh +0 -69
  227. package/src/mcp/server.ts +0 -322
  228. package/src/skills/index.ts +0 -2
  229. package/src/skills/invocation-guard.ts +0 -177
  230. package/src/skills/matcher.ts +0 -148
  231. package/src/skills/official/code-simplifier.md +0 -52
  232. package/src/skills/official/find-skills.md +0 -142
  233. package/src/skills/official/official-api-design.md +0 -30
  234. package/src/skills/official/official-architecture-decision.md +0 -41
  235. package/src/skills/official/official-bmad.md +0 -118
  236. package/src/skills/official/official-db-schema-design.md +0 -34
  237. package/src/skills/official/official-debug.md +0 -25
  238. package/src/skills/official/official-doc-driven.md +0 -31
  239. package/src/skills/official/official-harness-engineering.md +0 -108
  240. package/src/skills/official/official-performance-optimization.md +0 -30
  241. package/src/skills/official/official-pr-review.md +0 -35
  242. package/src/skills/official/official-release-checklist.md +0 -30
  243. package/src/skills/official/official-security-hardening.md +0 -32
  244. package/src/skills/official/official-spec-driven-design.md +0 -31
  245. package/src/skills/official/planning-with-files.md +0 -241
  246. package/src/skills/official/ui-ux-pro-max.md +0 -105
  247. package/src/skills/official/webapp-testing.md +0 -96
  248. package/src/skills/official-skills.ts +0 -89
  249. package/src/skills/registry.ts +0 -355
  250. package/src/skills/semantic-matcher.ts +0 -234
  251. package/src/skills/tools/pipeline-suggest.ts +0 -226
  252. package/src/skills/tools/skill-invoke.ts +0 -168
  253. package/src/skills/tools/skill-list.ts +0 -59
  254. package/src/skills/upgrade-engine.ts +0 -541
  255. package/src/skills/upgrade-prompt.ts +0 -84
  256. package/src/templates/go.yaml +0 -53
  257. package/src/templates/python.yaml +0 -59
  258. package/src/templates/react.yaml +0 -55
  259. package/src/templates/template-manager.ts +0 -170
  260. package/src/web/analytics/anti-pattern-detector.ts +0 -367
  261. package/src/web/analytics/drift-detector.ts +0 -219
  262. package/src/web/analytics/weekly-report.ts +0 -431
  263. package/src/web/auth-middleware.ts +0 -54
  264. package/src/web/routes/_helpers.ts +0 -34
  265. package/src/web/routes/ai.ts +0 -204
  266. package/src/web/routes/auth.ts +0 -22
  267. package/src/web/routes/drift.ts +0 -25
  268. package/src/web/routes/error-handler.ts +0 -120
  269. package/src/web/routes/events.ts +0 -47
  270. package/src/web/routes/insights.ts +0 -43
  271. package/src/web/routes/patch.ts +0 -117
  272. package/src/web/routes/reports.ts +0 -34
  273. package/src/web/routes/rules.ts +0 -76
  274. package/src/web/routes/sessions.ts +0 -250
  275. package/src/web/routes/skill-stats.ts +0 -92
  276. package/src/web/routes/skills.ts +0 -350
  277. package/src/web/routes/static.ts +0 -67
  278. package/src/web/routes/stats.ts +0 -50
  279. package/src/web/routes/status.ts +0 -30
  280. package/src/web/routes/tasks.ts +0 -193
  281. package/src/web/routes/token-usage.ts +0 -20
  282. package/src/web/routes/trace.ts +0 -126
  283. package/src/web/routes/types.ts +0 -57
  284. package/src/web/server.ts +0 -134
  285. package/src/web/ssrf-guard.ts +0 -112
  286. package/src/web/static/index.html +0 -3251
  287. package/src/web/static/vendor/chart.umd.min.js +0 -20
  288. package/tests/e2e/dashboard.spec.ts +0 -205
  289. package/tests/e2e/routing-skill-e2e.test.ts +0 -39
  290. package/tests/helpers/mock-ai.ts +0 -92
  291. package/tests/helpers/mock-storage.ts +0 -159
  292. package/tests/integration/claudemd-generator.test.ts +0 -90
  293. package/tests/integration/queue-replay.integration.test.ts +0 -193
  294. package/tests/integration/tasks-filter.integration.test.ts +0 -154
  295. package/tests/integration/web-analytics.integration.test.ts +0 -133
  296. package/tests/integration/web-stats.integration.test.ts +0 -135
  297. package/tests/integration/web-trace.integration.test.ts +0 -175
  298. package/tests/performance/database.benchmark.ts +0 -161
  299. package/tests/semantic-matcher.test.ts +0 -99
  300. package/tests/skill-matcher.test.ts +0 -110
  301. package/tests/unit/ai-provider-retry.test.ts +0 -194
  302. package/tests/unit/ai-provider-vision.test.ts +0 -224
  303. package/tests/unit/claudemd-generator.test.ts +0 -68
  304. package/tests/unit/cli-mcp.test.ts +0 -141
  305. package/tests/unit/core/forge-paths.test.ts +0 -99
  306. package/tests/unit/daemon/hook-sync.test.ts +0 -71
  307. package/tests/unit/daemon/post-tool-use.test.ts +0 -121
  308. package/tests/unit/daemon/skill-sync.test.ts +0 -75
  309. package/tests/unit/daemon/stop-handler-behavior-summary.test.ts +0 -202
  310. package/tests/unit/daemon/task-segmenter-recover.test.ts +0 -84
  311. package/tests/unit/event-fields.test.ts +0 -88
  312. package/tests/unit/event-parser.test.ts +0 -55
  313. package/tests/unit/handlers.test.ts +0 -171
  314. package/tests/unit/hooks/resolve-project-path.test.ts +0 -122
  315. package/tests/unit/invocation-guard.test.ts +0 -125
  316. package/tests/unit/queue.test.ts +0 -272
  317. package/tests/unit/router.test.ts +0 -138
  318. package/tests/unit/security.test.ts +0 -128
  319. package/tests/unit/skill-invocations-workflow.test.ts +0 -495
  320. package/tests/unit/skill-registry.test.ts +0 -94
  321. package/tests/unit/skills/invocation-guard-ttl.test.ts +0 -211
  322. package/tests/unit/skills/official-skills-loader.test.ts +0 -126
  323. package/tests/unit/skills/registry-multiformat.test.ts +0 -92
  324. package/tests/unit/skills/upgrade-engine-parse.test.ts +0 -138
  325. package/tests/unit/skills/upgrade-engine.test.ts +0 -401
  326. package/tests/unit/skills/upgrade-prompt.test.ts +0 -89
  327. package/tests/unit/socket-server.test.ts +0 -183
  328. package/tests/unit/storage/event-operations-aggregates.test.ts +0 -342
  329. package/tests/unit/storage/migration-idempotent.test.ts +0 -304
  330. package/tests/unit/storage/routing-aggregates.test.ts +0 -276
  331. package/tests/unit/storage/routing.test.ts +0 -117
  332. package/tests/unit/storage/schema-missing.test.ts +0 -81
  333. package/tests/unit/storage/session-operations-aggregates.test.ts +0 -120
  334. package/tests/unit/storage/sessions-aggregate.test.ts +0 -435
  335. package/tests/unit/storage/skill-operations-counts.test.ts +0 -106
  336. package/tests/unit/storage/skills-aggregates.test.ts +0 -104
  337. package/tests/unit/storage/sqlite-refactor-harness.test.ts +0 -314
  338. package/tests/unit/storage/task-operations-counts.test.ts +0 -46
  339. package/tests/unit/storage/tasks-getById.test.ts +0 -343
  340. package/tests/unit/storage/tasks-stale-gc.test.ts +0 -86
  341. package/tests/unit/storage.test.ts +0 -172
  342. package/tests/unit/token-usage.test.ts +0 -144
  343. package/tests/unit/type-guards.test.ts +0 -201
  344. package/tests/unit/utils/format.test.ts +0 -189
  345. package/tests/unit/utils/session.test.ts +0 -89
  346. package/tests/unit/utils/time.test.ts +0 -112
  347. package/tests/unit/web/navigation-back-contract.test.ts +0 -134
  348. package/tests/unit/web/routes-auth.test.ts +0 -93
  349. package/tests/unit/web/routes-events.test.ts +0 -101
  350. package/tests/unit/web/routes-rules.test.ts +0 -182
  351. package/tests/unit/web/routes-sessions.test.ts +0 -181
  352. package/tests/unit/web/routes-skill-stats.test.ts +0 -179
  353. package/tests/unit/web/routes-stats.test.ts +0 -92
  354. package/tests/unit/web/routes-tasks.test.ts +0 -385
  355. package/tests/unit/web/task-title-contract.test.ts +0 -210
  356. package/tests/unit/web/tasks-component-contract.test.ts +0 -179
  357. package/tsconfig.json +0 -22
  358. package/vitest.config.ts +0 -21
  359. package/vitest.integration.config.ts +0 -16
  360. package/web/CLAUDE.md +0 -20
  361. package/web/index.html +0 -13
  362. package/web/package-lock.json +0 -4854
  363. package/web/package.json +0 -35
  364. package/web/postcss.config.js +0 -6
  365. package/web/src/App.tsx +0 -110
  366. package/web/src/components/CodeBlock.tsx +0 -31
  367. package/web/src/components/Confirm.tsx +0 -96
  368. package/web/src/components/Drawer.tsx +0 -60
  369. package/web/src/components/Layout.tsx +0 -145
  370. package/web/src/components/MarkdownRenderer.tsx +0 -77
  371. package/web/src/components/SearchInput.tsx +0 -31
  372. package/web/src/components/SessionDetailContent.tsx +0 -157
  373. package/web/src/components/Toast.tsx +0 -92
  374. package/web/src/index.css +0 -19
  375. package/web/src/main.tsx +0 -31
  376. package/web/src/pages/AIConfig.tsx +0 -233
  377. package/web/src/pages/Dashboard.tsx +0 -572
  378. package/web/src/pages/Events.tsx +0 -271
  379. package/web/src/pages/Reports.tsx +0 -428
  380. package/web/src/pages/SessionDetail.tsx +0 -162
  381. package/web/src/pages/Sessions.tsx +0 -205
  382. package/web/src/pages/Skills.tsx +0 -180
  383. package/web/src/pages/TaskDetail.tsx +0 -515
  384. package/web/src/pages/Tasks.tsx +0 -415
  385. package/web/src/utils/auth.ts +0 -59
  386. package/web/src/utils/export.ts +0 -54
  387. package/web/src/utils/navigation.ts +0 -25
  388. package/web/src/utils/task-title.ts +0 -49
  389. package/web/src/utils/time.ts +0 -13
  390. package/web/tailwind.config.js +0 -11
  391. package/web/tsconfig.json +0 -21
  392. package/web/tsconfig.node.json +0 -10
  393. package/web/vite.config.ts +0 -76
  394. package/winspan-claude-forge-8.43.0.tgz +0 -0
@@ -1,415 +0,0 @@
1
- import { useQuery } from '@tanstack/react-query'
2
- import { useState, useMemo, useCallback } from 'react'
3
- import { useNavigate, useSearchParams, useLocation } from 'react-router-dom'
4
- import { formatDistanceToNow } from 'date-fns'
5
- import { parseUTC } from '../utils/time'
6
- import { zhCN } from 'date-fns/locale'
7
- import {
8
- Clock,
9
- Activity,
10
- CheckCircle2,
11
- Loader2,
12
- ChevronLeft,
13
- ChevronRight,
14
- FolderOpen,
15
- } from 'lucide-react'
16
- import SearchInput from '../components/SearchInput'
17
- import clsx from 'clsx'
18
- import { normalizeTaskTitle } from '../utils/task-title'
19
-
20
- // ── Types ────────────────────────────────────────────────────────────────────
21
-
22
- interface Task {
23
- id: string
24
- title: string
25
- status: 'active' | 'completed' | 'abandoned'
26
- start_time: string
27
- end_time?: string
28
- event_count: number
29
- session_id: string
30
- project_path?: string
31
- first_prompt?: string | null
32
- }
33
-
34
- interface TaskPage {
35
- items: Task[]
36
- total: number
37
- has_more: boolean
38
- }
39
-
40
- // ── Constants ─────────────────────────────────────────────────────────────────
41
-
42
- const PAGE_SIZES = [20, 50, 100] as const
43
- const DEFAULT_PAGE_SIZE = 50
44
-
45
- type TimePreset = '1h' | '24h' | '7d' | 'custom' | ''
46
-
47
- function presetToFrom(preset: TimePreset): string {
48
- const now = Date.now()
49
- if (preset === '1h') return new Date(now - 3600_000).toISOString()
50
- if (preset === '24h') return new Date(now - 86_400_000).toISOString()
51
- if (preset === '7d') return new Date(now - 7 * 86_400_000).toISOString()
52
- return ''
53
- }
54
-
55
- // ── Fetch ─────────────────────────────────────────────────────────────────────
56
-
57
- async function fetchTasks(params: URLSearchParams): Promise<TaskPage> {
58
- const res = await fetch(`/api/tasks?${params.toString()}`)
59
- if (!res.ok) throw new Error('Failed to fetch tasks')
60
- return res.json() as Promise<TaskPage>
61
- }
62
-
63
- async function fetchProjects(): Promise<string[]> {
64
- const res = await fetch('/api/tasks/projects')
65
- if (!res.ok) throw new Error('Failed to fetch projects')
66
- return res.json() as Promise<string[]>
67
- }
68
-
69
- // ── Helpers ───────────────────────────────────────────────────────────────────
70
-
71
- function shortPath(p: string): string {
72
- const parts = p.replace(/\\/g, '/').split('/').filter(Boolean)
73
- if (parts.length <= 2) return p
74
- return '…/' + parts.slice(-2).join('/')
75
- }
76
-
77
- export default function Tasks() {
78
- const navigate = useNavigate()
79
- const location = useLocation()
80
- const [searchParams, setSearchParams] = useSearchParams()
81
-
82
- // ── URL-synced state ──────────────────────────────────────────────────────
83
- const page = Math.max(1, Number(searchParams.get('page') ?? '1'))
84
- const pageSize = (PAGE_SIZES as readonly number[]).includes(Number(searchParams.get('size') ?? ''))
85
- ? Number(searchParams.get('size'))
86
- : DEFAULT_PAGE_SIZE
87
- const selectedProjects = searchParams.getAll('project')
88
- const timePreset = (searchParams.get('preset') ?? '') as TimePreset
89
- const fromParam = searchParams.get('from') ?? ''
90
- const toParam = searchParams.get('to') ?? ''
91
- const searchVal = searchParams.get('search') ?? ''
92
-
93
- // ── Local state for project dropdown ─────────────────────────────────────
94
- const [projectDropdownOpen, setProjectDropdownOpen] = useState(false)
95
-
96
- // ── Patch URL helper ──────────────────────────────────────────────────────
97
- const patchSearch = useCallback((patch: Record<string, string | string[] | null>) => {
98
- setSearchParams(prev => {
99
- const next = new URLSearchParams(prev)
100
- for (const [k, v] of Object.entries(patch)) {
101
- if (v === null || v === '') {
102
- next.delete(k)
103
- } else if (Array.isArray(v)) {
104
- next.delete(k)
105
- v.forEach(val => next.append(k, val))
106
- } else {
107
- next.set(k, v)
108
- }
109
- }
110
- return next
111
- }, { replace: true })
112
- }, [setSearchParams])
113
-
114
- // ── Compute effective from/to ─────────────────────────────────────────────
115
- const effectiveFrom = timePreset && timePreset !== 'custom' ? presetToFrom(timePreset) : fromParam
116
- const effectiveTo = timePreset && timePreset !== 'custom' ? '' : toParam
117
-
118
- // ── Build query params for /api/tasks ────────────────────────────────────
119
- const queryParams = useMemo(() => {
120
- const p = new URLSearchParams()
121
- p.set('limit', String(pageSize))
122
- p.set('offset', String((page - 1) * pageSize))
123
- selectedProjects.forEach(pr => p.append('project', pr))
124
- if (effectiveFrom) p.set('from', effectiveFrom)
125
- if (effectiveTo) p.set('to', effectiveTo)
126
- if (searchVal) p.set('search', searchVal)
127
- return p
128
- }, [page, pageSize, selectedProjects, effectiveFrom, effectiveTo, searchVal])
129
-
130
- // ── Queries ───────────────────────────────────────────────────────────────
131
- const { data, isLoading, error } = useQuery<TaskPage>({
132
- queryKey: ['tasks', queryParams.toString()],
133
- queryFn: () => fetchTasks(queryParams),
134
- refetchInterval: 15_000,
135
- placeholderData: prev => prev,
136
- })
137
-
138
- const { data: projects = [] } = useQuery<string[]>({
139
- queryKey: ['task-projects'],
140
- queryFn: fetchProjects,
141
- staleTime: 60_000,
142
- })
143
-
144
- const tasks = data?.items ?? []
145
- const total = data?.total ?? 0
146
- const hasMore = data?.has_more ?? false
147
- const totalPages = Math.ceil(total / pageSize) || 1
148
-
149
- // ── Event handlers ────────────────────────────────────────────────────────
150
- const handleSearchChange = (val: string) => {
151
- patchSearch({ search: val || null, page: '1' })
152
- }
153
-
154
- const toggleProject = (proj: string) => {
155
- const next = selectedProjects.includes(proj)
156
- ? selectedProjects.filter(p => p !== proj)
157
- : [...selectedProjects, proj]
158
- patchSearch({ project: next.length ? next : null, page: '1' })
159
- }
160
-
161
- const clearProjects = () => patchSearch({ project: null, page: '1' })
162
-
163
- const handlePreset = (preset: TimePreset) => {
164
- patchSearch({ preset: preset || null, from: null, to: null, page: '1' })
165
- }
166
-
167
- const handlePageSize = (size: number) => {
168
- patchSearch({ size: String(size), page: '1' })
169
- }
170
-
171
- const goPage = (p: number) => {
172
- patchSearch({ page: String(Math.max(1, Math.min(p, totalPages))) })
173
- }
174
-
175
- // ── Status icon ───────────────────────────────────────────────────────────
176
- const getStatusIcon = (status: string) => {
177
- switch (status) {
178
- case 'completed':
179
- return <CheckCircle2 className="h-4 w-4 text-green-600" />
180
- case 'active':
181
- return <Loader2 className="h-4 w-4 text-blue-600 animate-spin" />
182
- default:
183
- return <Activity className="h-4 w-4 text-gray-400" />
184
- }
185
- }
186
-
187
- // ── Loading / error states ────────────────────────────────────────────────
188
- if (isLoading && !data) {
189
- return (
190
- <div>
191
- <h1 className="text-2xl font-bold text-gray-900 mb-6">任务</h1>
192
- <div className="bg-white rounded-lg shadow p-6">
193
- <p className="text-gray-600">加载中...</p>
194
- </div>
195
- </div>
196
- )
197
- }
198
-
199
- if (error) {
200
- return (
201
- <div>
202
- <h1 className="text-2xl font-bold text-gray-900 mb-6">任务</h1>
203
- <div className="bg-white rounded-lg shadow p-6">
204
- <p className="text-red-600">加载失败: {(error as Error).message}</p>
205
- </div>
206
- </div>
207
- )
208
- }
209
-
210
- // ── Render ────────────────────────────────────────────────────────────────
211
- return (
212
- <div>
213
- {/* Header */}
214
- <div className="flex items-center justify-between mb-4 flex-wrap gap-3">
215
- <h1 className="text-2xl font-bold text-gray-900">任务</h1>
216
- <SearchInput
217
- value={searchVal}
218
- onChange={handleSearchChange}
219
- placeholder="搜索任务标题..."
220
- className="w-64"
221
- />
222
- </div>
223
-
224
- {/* Filters bar */}
225
- <div className="flex flex-wrap items-center gap-2 mb-4">
226
- {/* Time presets */}
227
- {(['1h', '24h', '7d'] as TimePreset[]).map(preset => (
228
- <button
229
- key={preset}
230
- onClick={() => handlePreset(timePreset === preset ? '' : preset)}
231
- className={clsx(
232
- 'px-3 py-1.5 text-xs font-medium rounded-md border transition-colors',
233
- timePreset === preset
234
- ? 'bg-indigo-600 text-white border-indigo-600'
235
- : 'bg-white text-gray-700 border-gray-300 hover:bg-gray-50'
236
- )}
237
- >
238
- {preset === '1h' ? '近 1 小时' : preset === '24h' ? '近 24 小时' : '近 7 天'}
239
- </button>
240
- ))}
241
-
242
- {/* Project multi-select */}
243
- {projects.length > 0 && (
244
- <div className="relative">
245
- <button
246
- onClick={() => setProjectDropdownOpen(v => !v)}
247
- className={clsx(
248
- 'inline-flex items-center gap-1.5 px-3 py-1.5 text-xs font-medium rounded-md border transition-colors',
249
- selectedProjects.length > 0
250
- ? 'bg-indigo-50 text-indigo-700 border-indigo-300'
251
- : 'bg-white text-gray-700 border-gray-300 hover:bg-gray-50'
252
- )}
253
- >
254
- <FolderOpen className="h-3.5 w-3.5" />
255
- {selectedProjects.length > 0
256
- ? `项目 (${selectedProjects.length})`
257
- : '所有项目'}
258
- </button>
259
- {projectDropdownOpen && (
260
- <>
261
- <div className="fixed inset-0 z-10" onClick={() => setProjectDropdownOpen(false)} />
262
- <div className="absolute left-0 mt-1 w-72 bg-white border border-gray-200 rounded-md shadow-lg z-20 max-h-64 overflow-y-auto">
263
- {selectedProjects.length > 0 && (
264
- <button
265
- onClick={() => { clearProjects(); setProjectDropdownOpen(false) }}
266
- className="w-full text-left px-3 py-2 text-xs text-indigo-600 hover:bg-indigo-50 border-b border-gray-100"
267
- >
268
- 清除筛选
269
- </button>
270
- )}
271
- {projects.map(proj => (
272
- <label
273
- key={proj}
274
- className="flex items-center gap-2 px-3 py-2 text-sm text-gray-700 hover:bg-gray-50 cursor-pointer"
275
- >
276
- <input
277
- type="checkbox"
278
- className="rounded text-indigo-600"
279
- checked={selectedProjects.includes(proj)}
280
- onChange={() => toggleProject(proj)}
281
- />
282
- <span className="truncate font-mono text-xs" title={proj}>
283
- {shortPath(proj)}
284
- </span>
285
- </label>
286
- ))}
287
- </div>
288
- </>
289
- )}
290
- </div>
291
- )}
292
-
293
- {/* Active filter summary */}
294
- {(selectedProjects.length > 0 || timePreset || searchVal) && (
295
- <button
296
- onClick={() => setSearchParams({}, { replace: true })}
297
- className="px-2 py-1.5 text-xs text-gray-500 hover:text-gray-700 underline"
298
- >
299
- 清除所有筛选
300
- </button>
301
- )}
302
- </div>
303
-
304
- {/* Result count + page-size selector */}
305
- <div className="flex items-center justify-between mb-3 text-sm text-gray-500">
306
- <span>
307
- 共 <span className="font-semibold text-gray-900">{total}</span> 条
308
- {isLoading && <span className="ml-2 text-xs text-indigo-500">刷新中...</span>}
309
- </span>
310
- <div className="flex items-center gap-2">
311
- <span className="text-xs">每页</span>
312
- {PAGE_SIZES.map(size => (
313
- <button
314
- key={size}
315
- onClick={() => handlePageSize(size)}
316
- className={clsx(
317
- 'px-2 py-0.5 text-xs rounded border',
318
- pageSize === size
319
- ? 'bg-indigo-600 text-white border-indigo-600'
320
- : 'bg-white text-gray-600 border-gray-300 hover:bg-gray-50'
321
- )}
322
- >
323
- {size}
324
- </button>
325
- ))}
326
- </div>
327
- </div>
328
-
329
- {/* Task list */}
330
- <div className="grid gap-4">
331
- {tasks.map((task) => (
332
- <div
333
- key={task.id}
334
- onClick={() => navigate(`/tasks/${task.id}`, { state: { from: location.pathname + location.search } })}
335
- className="bg-white rounded-lg shadow hover:shadow-md transition-shadow cursor-pointer p-5"
336
- >
337
- <div className="flex items-start justify-between mb-3">
338
- <div className="flex items-start gap-3 flex-1 min-w-0">
339
- <div className="mt-0.5 flex-shrink-0">{getStatusIcon(task.status)}</div>
340
- <div className="flex-1 min-w-0">
341
- <h3 className="text-base font-semibold text-gray-900 mb-1 truncate">
342
- {normalizeTaskTitle(task.title, task.first_prompt)}
343
- </h3>
344
- {task.project_path && (
345
- <p className="text-xs text-gray-400 font-mono truncate" title={task.project_path}>
346
- {shortPath(task.project_path)}
347
- </p>
348
- )}
349
- </div>
350
- </div>
351
- <span className={clsx(
352
- 'ml-3 flex-shrink-0 px-2 py-1 text-xs font-medium rounded-full whitespace-nowrap',
353
- task.status === 'completed' && 'bg-green-100 text-green-700',
354
- task.status === 'active' && 'bg-blue-100 text-blue-700',
355
- task.status === 'abandoned' && 'bg-gray-100 text-gray-600',
356
- )}>
357
- {task.status}
358
- </span>
359
- </div>
360
-
361
- <div className="flex items-center gap-6 text-sm text-gray-600">
362
- <span className="inline-flex items-center gap-1">
363
- <Activity className="h-4 w-4" />
364
- {task.event_count} 个事件
365
- </span>
366
- <span className="inline-flex items-center gap-1">
367
- <Clock className="h-4 w-4" />
368
- {task.start_time
369
- ? formatDistanceToNow(parseUTC(task.start_time), {
370
- addSuffix: true,
371
- locale: zhCN,
372
- })
373
- : '-'}
374
- </span>
375
- </div>
376
- </div>
377
- ))}
378
-
379
- {tasks.length === 0 && !isLoading && (
380
- <div className="bg-white rounded-lg shadow p-12 text-center text-gray-500">
381
- {searchVal || selectedProjects.length > 0 || timePreset
382
- ? '无匹配结果'
383
- : '暂无任务记录'}
384
- </div>
385
- )}
386
- </div>
387
-
388
- {/* Pagination */}
389
- {total > pageSize && (
390
- <div className="flex items-center justify-between mt-6">
391
- <button
392
- onClick={() => goPage(page - 1)}
393
- disabled={page <= 1}
394
- className="inline-flex items-center gap-1 px-3 py-1.5 text-sm border rounded-md bg-white hover:bg-gray-50 disabled:opacity-40 disabled:cursor-not-allowed"
395
- >
396
- <ChevronLeft className="h-4 w-4" />
397
- 上一页
398
- </button>
399
- <span className="text-sm text-gray-500">
400
- 第 {page} / {totalPages} 页
401
- {hasMore ? '' : '(已到末页)'}
402
- </span>
403
- <button
404
- onClick={() => goPage(page + 1)}
405
- disabled={!hasMore}
406
- className="inline-flex items-center gap-1 px-3 py-1.5 text-sm border rounded-md bg-white hover:bg-gray-50 disabled:opacity-40 disabled:cursor-not-allowed"
407
- >
408
- 下一页
409
- <ChevronRight className="h-4 w-4" />
410
- </button>
411
- </div>
412
- )}
413
- </div>
414
- )
415
- }
@@ -1,59 +0,0 @@
1
- /**
2
- * Frontend auth helper — fetches the daemon's bearer token from the
3
- * local-only /api/auth/token endpoint and wraps fetch() to attach
4
- * the Authorization header on write calls.
5
- *
6
- * Token fetch is cached for the lifetime of the page load. On 401, the
7
- * cache is invalidated and refetched once before giving up.
8
- */
9
-
10
- let cachedToken: string | null = null
11
- let inflight: Promise<string | null> | null = null
12
-
13
- async function fetchToken(): Promise<string | null> {
14
- if (cachedToken) return cachedToken
15
- if (inflight) return inflight
16
- inflight = (async () => {
17
- try {
18
- const res = await fetch('/api/auth/token')
19
- if (!res.ok) return null
20
- const data = (await res.json()) as { token?: string }
21
- cachedToken = data.token ?? null
22
- return cachedToken
23
- } catch {
24
- return null
25
- } finally {
26
- inflight = null
27
- }
28
- })()
29
- return inflight
30
- }
31
-
32
- /** 清除 token 缓存(收到 401 时调用) */
33
- export function invalidateToken(): void {
34
- cachedToken = null
35
- }
36
-
37
- /**
38
- * 带鉴权的 fetch:自动附加 Authorization: Bearer <token>
39
- * 写操作(POST/PUT/DELETE)应统一用这个函数
40
- */
41
- export async function authFetch(input: RequestInfo | URL, init: RequestInit = {}): Promise<Response> {
42
- const token = await fetchToken()
43
- const headers = new Headers(init.headers ?? {})
44
- if (token && !headers.has('Authorization')) {
45
- headers.set('Authorization', `Bearer ${token}`)
46
- }
47
- const res = await fetch(input, { ...init, headers })
48
- if (res.status === 401) {
49
- // 失效重试一次(token 可能被 daemon 轮换)
50
- invalidateToken()
51
- const retryToken = await fetchToken()
52
- if (retryToken) {
53
- const retryHeaders = new Headers(init.headers ?? {})
54
- retryHeaders.set('Authorization', `Bearer ${retryToken}`)
55
- return fetch(input, { ...init, headers: retryHeaders })
56
- }
57
- }
58
- return res
59
- }
@@ -1,54 +0,0 @@
1
- /**
2
- * 导出工具函数:JSON / CSV
3
- */
4
-
5
- export function downloadJSON(filename: string, data: unknown): void {
6
- const content = JSON.stringify(data, null, 2)
7
- const blob = new Blob([content], { type: 'application/json' })
8
- triggerDownload(filename, blob)
9
- }
10
-
11
- export function downloadCSV<T extends object>(filename: string, rows: T[]): void {
12
- if (!rows || rows.length === 0) {
13
- downloadJSON(filename.replace(/\.csv$/, '.json'), rows)
14
- return
15
- }
16
-
17
- const headers = Object.keys(rows[0]) as (keyof T & string)[]
18
- const csv = [
19
- headers.join(','),
20
- ...rows.map(row =>
21
- headers.map(h => escapeCSV(row[h])).join(',')
22
- ),
23
- ].join('\n')
24
-
25
- // 添加 BOM 让 Excel 识别 UTF-8
26
- const blob = new Blob(['' + csv], { type: 'text/csv;charset=utf-8' })
27
- triggerDownload(filename, blob)
28
- }
29
-
30
- function escapeCSV(value: unknown): string {
31
- if (value === null || value === undefined) return ''
32
- const str = typeof value === 'object' ? JSON.stringify(value) : String(value)
33
- if (str.includes(',') || str.includes('"') || str.includes('\n')) {
34
- return `"${str.replace(/"/g, '""')}"`
35
- }
36
- return str
37
- }
38
-
39
- function triggerDownload(filename: string, blob: Blob): void {
40
- const url = URL.createObjectURL(blob)
41
- const a = document.createElement('a')
42
- a.href = url
43
- a.download = filename
44
- document.body.appendChild(a)
45
- a.click()
46
- document.body.removeChild(a)
47
- URL.revokeObjectURL(url)
48
- }
49
-
50
- export function getTimestamp(): string {
51
- const now = new Date()
52
- const pad = (n: number) => String(n).padStart(2, '0')
53
- return `${now.getFullYear()}${pad(now.getMonth() + 1)}${pad(now.getDate())}-${pad(now.getHours())}${pad(now.getMinutes())}${pad(now.getSeconds())}`
54
- }
@@ -1,25 +0,0 @@
1
- /**
2
- * Navigation helpers for react-router-based back-navigation with state preservation.
3
- */
4
-
5
- /**
6
- * Resolves the "back" URL from react-router navigation state.
7
- *
8
- * Usage in a detail page:
9
- * const location = useLocation()
10
- * const backTo = resolveBackTo(location.state)
11
- * // <Link to={backTo}>Back</Link>
12
- *
13
- * The navigation state is set by the list page when navigating forward:
14
- * navigate(`/tasks/${id}`, { state: { from: location.pathname + location.search } })
15
- *
16
- * @param state - location.state from react-router (typed as unknown for safety)
17
- * @param fallback - URL to use when state carries no valid `from` field (default: '/tasks')
18
- */
19
- export function resolveBackTo(state: unknown, fallback = '/tasks'): string {
20
- if (state !== null && state !== undefined && typeof state === 'object' && 'from' in state) {
21
- const from = (state as { from?: unknown }).from
22
- if (typeof from === 'string' && from.length > 0) return from
23
- }
24
- return fallback
25
- }
@@ -1,49 +0,0 @@
1
- /**
2
- * normalizeTaskTitle — pure helper for rendering task titles.
3
- *
4
- * Some tasks are stored with their title set to the raw <task-notification>
5
- * XML blob (sub-agent completion callbacks). When the blob is truncated to
6
- * 19 characters (just the opening tag), the title is unreadable.
7
- *
8
- * This function detects that case and falls back to `firstPrompt` — the first
9
- * UserPromptSubmit user_prompt for the task, supplied from the backend via the
10
- * `first_prompt` field in /api/tasks responses.
11
- *
12
- * No React dependency. Safe to import in unit tests.
13
- */
14
-
15
- const MAX_LEN = 80
16
-
17
- export function normalizeTaskTitle(
18
- title: string | null | undefined,
19
- firstPrompt?: string | null,
20
- ): string {
21
- if (!title?.trim()) return '(无标题)'
22
- const trimmed = title.trim()
23
-
24
- // Detect <task-notification> XML callback blobs (including bare 19-char token)
25
- if (trimmed.startsWith('<task-notification>')) {
26
- // First, try to extract <summary> from the title itself (full XML blob)
27
- const matchInTitle = trimmed.match(/<summary>([\s\S]*?)<\/summary>/)
28
- if (matchInTitle && matchInTitle[1].trim()) {
29
- const summary = matchInTitle[1].trim()
30
- return summary.length > MAX_LEN ? summary.slice(0, MAX_LEN) + '…' : summary
31
- }
32
-
33
- // Title has no summary (e.g. bare 19-char token) — try firstPrompt
34
- if (firstPrompt) {
35
- const matchInPrompt = firstPrompt.match(/<summary>([\s\S]*?)<\/summary>/)
36
- if (matchInPrompt && matchInPrompt[1].trim()) {
37
- const summary = matchInPrompt[1].trim()
38
- return summary.length > MAX_LEN ? summary.slice(0, MAX_LEN) + '…' : summary
39
- }
40
- // Has firstPrompt but no <summary> tag — show head of prompt
41
- const head = firstPrompt.trim().slice(0, 60).replace(/\s+/g, ' ')
42
- return `(子任务回调) — ${head}${firstPrompt.trim().length > 60 ? '…' : ''}`
43
- }
44
-
45
- return '(子任务回调)'
46
- }
47
-
48
- return trimmed.length > MAX_LEN ? trimmed.slice(0, MAX_LEN) + '…' : trimmed
49
- }
@@ -1,13 +0,0 @@
1
- /**
2
- * SQLite stores ISO timestamps without 'Z' suffix.
3
- * JS `new Date()` treats such strings as local time, causing an offset.
4
- * This helper ensures the timestamp is always parsed as UTC.
5
- */
6
- export function parseUTC(timestamp: string | null | undefined): Date {
7
- if (!timestamp) return new Date(0);
8
- const s = timestamp.trim();
9
- if (s.endsWith('Z') || /[+-]\d{2}:\d{2}$/.test(s)) {
10
- return new Date(s);
11
- }
12
- return new Date(s + 'Z');
13
- }
@@ -1,11 +0,0 @@
1
- /** @type {import('tailwindcss').Config} */
2
- export default {
3
- content: [
4
- "./index.html",
5
- "./src/**/*.{js,ts,jsx,tsx}",
6
- ],
7
- theme: {
8
- extend: {},
9
- },
10
- plugins: [],
11
- }
package/web/tsconfig.json DELETED
@@ -1,21 +0,0 @@
1
- {
2
- "compilerOptions": {
3
- "target": "ES2020",
4
- "useDefineForClassFields": true,
5
- "lib": ["ES2020", "DOM", "DOM.Iterable"],
6
- "module": "ESNext",
7
- "skipLibCheck": true,
8
- "moduleResolution": "bundler",
9
- "allowImportingTsExtensions": true,
10
- "resolveJsonModule": true,
11
- "isolatedModules": true,
12
- "noEmit": true,
13
- "jsx": "react-jsx",
14
- "strict": true,
15
- "noUnusedLocals": true,
16
- "noUnusedParameters": true,
17
- "noFallthroughCasesInSwitch": true
18
- },
19
- "include": ["src"],
20
- "references": [{ "path": "./tsconfig.node.json" }]
21
- }
@@ -1,10 +0,0 @@
1
- {
2
- "compilerOptions": {
3
- "composite": true,
4
- "skipLibCheck": true,
5
- "module": "ESNext",
6
- "moduleResolution": "bundler",
7
- "allowSyntheticDefaultImports": true
8
- },
9
- "include": ["vite.config.ts"]
10
- }