@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,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
- }