@winspan/claude-forge 8.41.0 → 8.51.0
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.
- package/.claude/CLAUDE.md +17 -0
- package/.eslintrc.js +23 -0
- package/.prettierrc +8 -0
- package/ARCHITECTURE_ISSUES.md +249 -0
- package/CLAUDE.md +265 -0
- package/CLAUDE.md.backup +488 -0
- package/DEVELOPMENT.md +310 -0
- package/dist/claudemd/claudemd-generator.d.ts +38 -3
- package/dist/claudemd/claudemd-generator.d.ts.map +1 -1
- package/dist/claudemd/claudemd-generator.js +420 -12
- package/dist/claudemd/claudemd-generator.js.map +1 -1
- package/dist/claudemd/index.d.ts +2 -2
- package/dist/claudemd/index.d.ts.map +1 -1
- package/dist/claudemd/index.js.map +1 -1
- package/dist/claudemd/resume-manager.d.ts.map +1 -1
- package/dist/claudemd/resume-manager.js +6 -3
- package/dist/claudemd/resume-manager.js.map +1 -1
- package/dist/claudemd/tech-detector.d.ts +1 -0
- package/dist/claudemd/tech-detector.d.ts.map +1 -1
- package/dist/claudemd/tech-detector.js +53 -0
- package/dist/claudemd/tech-detector.js.map +1 -1
- package/dist/claudemd/templates/swarm-protocol.md +222 -0
- package/dist/cli/commands/claudemd.js +2 -2
- package/dist/cli/commands/claudemd.js.map +1 -1
- package/dist/cli/commands/daemon.d.ts +28 -0
- package/dist/cli/commands/daemon.d.ts.map +1 -1
- package/dist/cli/commands/daemon.js +204 -12
- package/dist/cli/commands/daemon.js.map +1 -1
- package/dist/cli/commands/executions.d.ts.map +1 -1
- package/dist/cli/commands/executions.js +4 -3
- package/dist/cli/commands/executions.js.map +1 -1
- package/dist/cli/commands/init.d.ts.map +1 -1
- package/dist/cli/commands/init.js +5 -37
- package/dist/cli/commands/init.js.map +1 -1
- package/dist/cli/commands/logs.js.map +1 -1
- package/dist/cli/commands/mcp.d.ts.map +1 -1
- package/dist/cli/commands/mcp.js +3 -5
- package/dist/cli/commands/mcp.js.map +1 -1
- package/dist/cli/commands/menu.d.ts.map +1 -1
- package/dist/cli/commands/menu.js +14 -13
- package/dist/cli/commands/menu.js.map +1 -1
- package/dist/cli/commands/skills.d.ts.map +1 -1
- package/dist/cli/commands/skills.js +8 -2
- package/dist/cli/commands/skills.js.map +1 -1
- package/dist/cli/commands/stats.d.ts.map +1 -1
- package/dist/cli/commands/stats.js +2 -20
- package/dist/cli/commands/stats.js.map +1 -1
- package/dist/cli/commands/status.js +2 -2
- package/dist/cli/commands/status.js.map +1 -1
- package/dist/cli/commands/trace.d.ts +9 -0
- package/dist/cli/commands/trace.d.ts.map +1 -0
- package/dist/cli/commands/trace.js +125 -0
- package/dist/cli/commands/trace.js.map +1 -0
- package/dist/cli/index.js +2 -4
- package/dist/cli/index.js.map +1 -1
- package/dist/cli/init/hook-manager.d.ts.map +1 -1
- package/dist/cli/init/hook-manager.js +2 -2
- package/dist/cli/init/hook-manager.js.map +1 -1
- package/dist/core/ai/provider.d.ts +10 -2
- package/dist/core/ai/provider.d.ts.map +1 -1
- package/dist/core/ai/provider.js +2 -2
- package/dist/core/ai/provider.js.map +1 -1
- package/dist/core/ai/types.d.ts +1 -19
- package/dist/core/ai/types.d.ts.map +1 -1
- package/dist/core/ai/types.js +1 -1
- package/dist/core/config.d.ts +2 -1
- package/dist/core/config.d.ts.map +1 -1
- package/dist/core/config.js +23 -6
- package/dist/core/config.js.map +1 -1
- package/dist/core/constants.d.ts +14 -3
- package/dist/core/constants.d.ts.map +1 -1
- package/dist/core/constants.js +17 -3
- package/dist/core/constants.js.map +1 -1
- package/dist/core/event-fields.d.ts +16 -0
- package/dist/core/event-fields.d.ts.map +1 -0
- package/dist/core/event-fields.js +19 -0
- package/dist/core/event-fields.js.map +1 -0
- package/dist/core/queue/index.d.ts +52 -0
- package/dist/core/queue/index.d.ts.map +1 -0
- package/dist/core/queue/index.js +175 -0
- package/dist/core/queue/index.js.map +1 -0
- package/dist/core/storage/base.d.ts +66 -0
- package/dist/core/storage/base.d.ts.map +1 -0
- package/dist/core/storage/base.js +254 -0
- package/dist/core/storage/base.js.map +1 -0
- package/dist/core/storage/events.d.ts +141 -0
- package/dist/core/storage/events.d.ts.map +1 -0
- package/dist/core/storage/events.js +348 -0
- package/dist/core/storage/events.js.map +1 -0
- package/dist/core/storage/injections.d.ts +27 -0
- package/dist/core/storage/injections.d.ts.map +1 -0
- package/dist/core/storage/injections.js +51 -0
- package/dist/core/storage/injections.js.map +1 -0
- package/dist/core/storage/maintenance.d.ts +21 -0
- package/dist/core/storage/maintenance.d.ts.map +1 -0
- package/dist/core/storage/maintenance.js +52 -0
- package/dist/core/storage/maintenance.js.map +1 -0
- package/dist/core/storage/routing.d.ts +124 -0
- package/dist/core/storage/routing.d.ts.map +1 -0
- package/dist/core/storage/routing.js +239 -0
- package/dist/core/storage/routing.js.map +1 -0
- package/dist/core/storage/rows.d.ts +0 -47
- package/dist/core/storage/rows.d.ts.map +1 -1
- package/dist/core/storage/schema.sql +85 -137
- package/dist/core/storage/sessions.d.ts +54 -0
- package/dist/core/storage/sessions.d.ts.map +1 -0
- package/dist/core/storage/sessions.js +137 -0
- package/dist/core/storage/sessions.js.map +1 -0
- package/dist/core/storage/skills.d.ts +63 -0
- package/dist/core/storage/skills.d.ts.map +1 -0
- package/dist/core/storage/skills.js +154 -0
- package/dist/core/storage/skills.js.map +1 -0
- package/dist/core/storage/sqlite.d.ts +97 -266
- package/dist/core/storage/sqlite.d.ts.map +1 -1
- package/dist/core/storage/sqlite.js +196 -764
- package/dist/core/storage/sqlite.js.map +1 -1
- package/dist/core/storage/tasks.d.ts +113 -0
- package/dist/core/storage/tasks.d.ts.map +1 -0
- package/dist/core/storage/tasks.js +276 -0
- package/dist/core/storage/tasks.js.map +1 -0
- package/dist/core/storage/token-usage.d.ts +36 -0
- package/dist/core/storage/token-usage.d.ts.map +1 -0
- package/dist/core/storage/token-usage.js +59 -0
- package/dist/core/storage/token-usage.js.map +1 -0
- package/dist/core/types.d.ts +83 -6
- package/dist/core/types.d.ts.map +1 -1
- package/dist/core/types.js +24 -1
- package/dist/core/types.js.map +1 -1
- package/dist/core/utils/error-handler.d.ts.map +1 -1
- package/dist/core/utils/error-handler.js +3 -2
- package/dist/core/utils/error-handler.js.map +1 -1
- package/dist/core/utils/format.d.ts +28 -0
- package/dist/core/utils/format.d.ts.map +1 -0
- package/dist/core/utils/format.js +68 -0
- package/dist/core/utils/format.js.map +1 -0
- package/dist/core/utils/git.d.ts +10 -0
- package/dist/core/utils/git.d.ts.map +1 -0
- package/dist/core/utils/git.js +24 -0
- package/dist/core/utils/git.js.map +1 -0
- package/dist/core/utils/logger.d.ts +6 -1
- package/dist/core/utils/logger.d.ts.map +1 -1
- package/dist/core/utils/logger.js +87 -3
- package/dist/core/utils/logger.js.map +1 -1
- package/dist/core/utils/lru-cache.d.ts +1 -0
- package/dist/core/utils/lru-cache.d.ts.map +1 -1
- package/dist/core/utils/lru-cache.js +3 -0
- package/dist/core/utils/lru-cache.js.map +1 -1
- package/dist/core/utils/session.d.ts +16 -0
- package/dist/core/utils/session.d.ts.map +1 -0
- package/dist/core/utils/session.js +25 -0
- package/dist/core/utils/session.js.map +1 -0
- package/dist/core/utils/time.d.ts +22 -0
- package/dist/core/utils/time.d.ts.map +1 -0
- package/dist/core/utils/time.js +38 -0
- package/dist/core/utils/time.js.map +1 -0
- package/dist/core/utils/token-tracker.js +1 -1
- package/dist/core/utils/token-tracker.js.map +1 -1
- package/dist/daemon/event-parser.d.ts.map +1 -1
- package/dist/daemon/event-parser.js +2 -1
- package/dist/daemon/event-parser.js.map +1 -1
- package/dist/daemon/handlers/history-exporter.d.ts.map +1 -1
- package/dist/daemon/handlers/history-exporter.js +6 -4
- package/dist/daemon/handlers/history-exporter.js.map +1 -1
- package/dist/daemon/handlers/post-tool-use.d.ts +5 -12
- package/dist/daemon/handlers/post-tool-use.d.ts.map +1 -1
- package/dist/daemon/handlers/post-tool-use.js +25 -79
- package/dist/daemon/handlers/post-tool-use.js.map +1 -1
- package/dist/daemon/handlers/stop.d.ts +28 -12
- package/dist/daemon/handlers/stop.d.ts.map +1 -1
- package/dist/daemon/handlers/stop.js +129 -42
- package/dist/daemon/handlers/stop.js.map +1 -1
- package/dist/daemon/handlers/user-prompt.d.ts +18 -19
- package/dist/daemon/handlers/user-prompt.d.ts.map +1 -1
- package/dist/daemon/handlers/user-prompt.js +93 -227
- package/dist/daemon/handlers/user-prompt.js.map +1 -1
- package/dist/daemon/index.d.ts +6 -2
- package/dist/daemon/index.d.ts.map +1 -1
- package/dist/daemon/index.js +97 -127
- package/dist/daemon/index.js.map +1 -1
- package/dist/daemon/launchd/com.claude-forge.daemon.plist.template +47 -0
- package/dist/daemon/launchd-installer.d.ts +61 -0
- package/dist/daemon/launchd-installer.d.ts.map +1 -0
- package/dist/daemon/launchd-installer.js +182 -0
- package/dist/daemon/launchd-installer.js.map +1 -0
- package/dist/daemon/lifecycle.d.ts +11 -0
- package/dist/daemon/lifecycle.d.ts.map +1 -1
- package/dist/daemon/lifecycle.js +47 -4
- package/dist/daemon/lifecycle.js.map +1 -1
- package/dist/daemon/router.d.ts +9 -2
- package/dist/daemon/router.d.ts.map +1 -1
- package/dist/daemon/router.js +27 -3
- package/dist/daemon/router.js.map +1 -1
- package/dist/daemon/server.d.ts +6 -4
- package/dist/daemon/server.d.ts.map +1 -1
- package/dist/daemon/server.js +80 -88
- package/dist/daemon/server.js.map +1 -1
- package/dist/{intelligence → daemon/services}/task-segmenter.d.ts +7 -1
- package/dist/daemon/services/task-segmenter.d.ts.map +1 -0
- package/dist/{intelligence → daemon/services}/task-segmenter.js +30 -7
- package/dist/daemon/services/task-segmenter.js.map +1 -0
- package/dist/hooks/hook-lib.sh +118 -0
- package/dist/hooks/notification.sh +9 -5
- package/dist/hooks/post-tool-use.sh +10 -6
- package/dist/hooks/pre-tool-use.sh +9 -6
- package/dist/hooks/stop.sh +10 -7
- package/dist/hooks/user-prompt-submit.sh +10 -11
- package/dist/mcp/server.d.ts +2 -2
- package/dist/mcp/server.d.ts.map +1 -1
- package/dist/mcp/server.js +71 -11
- package/dist/mcp/server.js.map +1 -1
- package/dist/skills/invocation-guard.d.ts +20 -0
- package/dist/skills/invocation-guard.d.ts.map +1 -1
- package/dist/skills/invocation-guard.js +63 -0
- package/dist/skills/invocation-guard.js.map +1 -1
- package/dist/skills/matcher.d.ts.map +1 -1
- package/dist/skills/matcher.js +12 -3
- package/dist/skills/matcher.js.map +1 -1
- package/dist/skills/official/code-simplifier.md +16 -0
- package/dist/skills/official/find-skills.md +23 -0
- package/dist/skills/official/official-api-design.md +17 -0
- package/dist/skills/official/official-architecture-decision.md +20 -0
- package/dist/skills/official/official-bmad.md +118 -0
- package/dist/skills/official/official-db-schema-design.md +16 -0
- package/dist/skills/official/official-debug.md +17 -0
- package/dist/skills/official/official-doc-driven.md +31 -0
- package/dist/skills/official/official-harness-engineering.md +108 -0
- package/dist/skills/official/official-performance-optimization.md +30 -0
- package/dist/skills/official/official-pr-review.md +35 -0
- package/dist/skills/official/official-release-checklist.md +30 -0
- package/dist/skills/official/official-security-hardening.md +26 -0
- package/dist/skills/official/official-spec-driven-design.md +31 -0
- package/dist/skills/official/planning-with-files.md +37 -0
- package/dist/skills/official/ui-ux-pro-max.md +18 -0
- package/dist/skills/official/webapp-testing.md +12 -0
- package/dist/skills/official-skills.d.ts +8 -4
- package/dist/skills/official-skills.d.ts.map +1 -1
- package/dist/skills/official-skills.js +48 -704
- package/dist/skills/official-skills.js.map +1 -1
- package/dist/skills/registry.d.ts +5 -0
- package/dist/skills/registry.d.ts.map +1 -1
- package/dist/skills/registry.js +48 -15
- package/dist/skills/registry.js.map +1 -1
- package/dist/skills/tools/pipeline-suggest.d.ts +30 -0
- package/dist/skills/tools/pipeline-suggest.d.ts.map +1 -0
- package/dist/skills/tools/pipeline-suggest.js +178 -0
- package/dist/skills/tools/pipeline-suggest.js.map +1 -0
- package/dist/web/analytics/anti-pattern-detector.d.ts +49 -0
- package/dist/web/analytics/anti-pattern-detector.d.ts.map +1 -0
- package/dist/web/analytics/anti-pattern-detector.js +318 -0
- package/dist/web/analytics/anti-pattern-detector.js.map +1 -0
- package/dist/web/analytics/drift-detector.d.ts +64 -0
- package/dist/web/analytics/drift-detector.d.ts.map +1 -0
- package/dist/web/analytics/drift-detector.js +198 -0
- package/dist/web/analytics/drift-detector.js.map +1 -0
- package/dist/web/analytics/weekly-report.d.ts +91 -0
- package/dist/web/analytics/weekly-report.d.ts.map +1 -0
- package/dist/web/analytics/weekly-report.js +328 -0
- package/dist/web/analytics/weekly-report.js.map +1 -0
- package/dist/web/auth-middleware.d.ts.map +1 -1
- package/dist/web/auth-middleware.js +1 -2
- package/dist/web/auth-middleware.js.map +1 -1
- package/dist/web/routes/_helpers.d.ts +16 -0
- package/dist/web/routes/_helpers.d.ts.map +1 -0
- package/dist/web/routes/_helpers.js +32 -0
- package/dist/web/routes/_helpers.js.map +1 -0
- package/dist/web/routes/ai.d.ts.map +1 -1
- package/dist/web/routes/ai.js +16 -22
- package/dist/web/routes/ai.js.map +1 -1
- package/dist/web/routes/drift.d.ts +10 -0
- package/dist/web/routes/drift.d.ts.map +1 -0
- package/dist/web/routes/drift.js +21 -0
- package/dist/web/routes/drift.js.map +1 -0
- package/dist/web/routes/error-handler.d.ts +43 -0
- package/dist/web/routes/error-handler.d.ts.map +1 -0
- package/dist/web/routes/error-handler.js +99 -0
- package/dist/web/routes/error-handler.js.map +1 -0
- package/dist/web/routes/insights.d.ts +9 -0
- package/dist/web/routes/insights.d.ts.map +1 -0
- package/dist/web/routes/insights.js +34 -0
- package/dist/web/routes/insights.js.map +1 -0
- package/dist/web/routes/patch.js +2 -2
- package/dist/web/routes/patch.js.map +1 -1
- package/dist/web/routes/reports.d.ts +10 -0
- package/dist/web/routes/reports.d.ts.map +1 -0
- package/dist/web/routes/reports.js +27 -0
- package/dist/web/routes/reports.js.map +1 -0
- package/dist/web/routes/rules.d.ts +13 -3
- package/dist/web/routes/rules.d.ts.map +1 -1
- package/dist/web/routes/rules.js +58 -97
- package/dist/web/routes/rules.js.map +1 -1
- package/dist/web/routes/sessions.d.ts +1 -2
- package/dist/web/routes/sessions.d.ts.map +1 -1
- package/dist/web/routes/sessions.js +43 -69
- package/dist/web/routes/sessions.js.map +1 -1
- package/dist/web/routes/skill-stats.d.ts +2 -0
- package/dist/web/routes/skill-stats.d.ts.map +1 -1
- package/dist/web/routes/skill-stats.js +41 -39
- package/dist/web/routes/skill-stats.js.map +1 -1
- package/dist/web/routes/skills.d.ts.map +1 -1
- package/dist/web/routes/skills.js +39 -4
- package/dist/web/routes/skills.js.map +1 -1
- package/dist/web/routes/stats.d.ts +11 -0
- package/dist/web/routes/stats.d.ts.map +1 -0
- package/dist/web/routes/stats.js +42 -0
- package/dist/web/routes/stats.js.map +1 -0
- package/dist/web/routes/status.js +1 -1
- package/dist/web/routes/status.js.map +1 -1
- package/dist/web/routes/tasks.d.ts +4 -0
- package/dist/web/routes/tasks.d.ts.map +1 -0
- package/dist/web/routes/tasks.js +156 -0
- package/dist/web/routes/tasks.js.map +1 -0
- package/dist/web/routes/trace.d.ts +10 -0
- package/dist/web/routes/trace.d.ts.map +1 -0
- package/dist/web/routes/trace.js +113 -0
- package/dist/web/routes/trace.js.map +1 -0
- package/dist/web/routes/types.d.ts +1 -14
- package/dist/web/routes/types.d.ts.map +1 -1
- package/dist/web/routes/types.js +12 -20
- package/dist/web/routes/types.js.map +1 -1
- package/dist/web/server.d.ts +1 -9
- package/dist/web/server.d.ts.map +1 -1
- package/dist/web/server.js +28 -28
- package/dist/web/server.js.map +1 -1
- package/dist/web/static/assets/AIConfig-CdDWzJyO.js +2 -0
- package/dist/web/static/assets/AIConfig-CdDWzJyO.js.map +1 -0
- package/dist/web/static/assets/Dashboard-CoEmmIDt.js +2 -0
- package/dist/web/static/assets/Dashboard-CoEmmIDt.js.map +1 -0
- package/dist/web/static/assets/{Drawer-DcU3ln98.js → Drawer-DdRTzlLB.js} +2 -2
- package/dist/web/static/assets/{Drawer-DcU3ln98.js.map → Drawer-DdRTzlLB.js.map} +1 -1
- package/dist/web/static/assets/Events-DrIq1SUS.js +2 -0
- package/dist/web/static/assets/Events-DrIq1SUS.js.map +1 -0
- package/dist/web/static/assets/Reports-DFBM3MDK.js +2 -0
- package/dist/web/static/assets/Reports-DFBM3MDK.js.map +1 -0
- package/dist/web/static/assets/SearchInput-qCj_jAcf.js +2 -0
- package/dist/web/static/assets/SearchInput-qCj_jAcf.js.map +1 -0
- package/dist/web/static/assets/SessionDetail-CCzwdoT7.js +2 -0
- package/dist/web/static/assets/SessionDetail-CCzwdoT7.js.map +1 -0
- package/dist/web/static/assets/Sessions-FfLYkAw9.js +2 -0
- package/dist/web/static/assets/Sessions-FfLYkAw9.js.map +1 -0
- package/dist/web/static/assets/Skills-C8Gvs3Qa.js +2 -0
- package/dist/web/static/assets/Skills-C8Gvs3Qa.js.map +1 -0
- package/dist/web/static/assets/TaskDetail-BS8pYhaR.js +2 -0
- package/dist/web/static/assets/TaskDetail-BS8pYhaR.js.map +1 -0
- package/dist/web/static/assets/Tasks-CyuhizG8.js +2 -0
- package/dist/web/static/assets/Tasks-CyuhizG8.js.map +1 -0
- package/dist/web/static/assets/export-L_VBD2p1.js +4 -0
- package/dist/web/static/assets/export-L_VBD2p1.js.map +1 -0
- package/dist/web/static/assets/index-CBX47X8l.js +3 -0
- package/dist/web/static/assets/index-CBX47X8l.js.map +1 -0
- package/dist/web/static/assets/index-DjIoMdoR.css +1 -0
- package/dist/web/static/assets/{lucide-53bR2rki.js → lucide-Bs_edTLa.js} +73 -38
- package/dist/web/static/assets/lucide-Bs_edTLa.js.map +1 -0
- package/dist/web/static/assets/react-router-r79dBVy4.js +20 -0
- package/dist/web/static/assets/{react-router-I-HqunH7.js.map → react-router-r79dBVy4.js.map} +1 -1
- package/dist/web/static/assets/task-title-BhOcemuR.js +2 -0
- package/dist/web/static/assets/task-title-BhOcemuR.js.map +1 -0
- package/dist/web/static/assets/time-Bxuk0M-C.js +2 -0
- package/dist/web/static/assets/time-Bxuk0M-C.js.map +1 -0
- package/dist/web/static/index.html +4 -4
- package/docs/concurrent-agents.md +129 -0
- package/docs/design/architecture-review-20260516.md +232 -0
- package/docs/design/fix-skills-data-and-set-leak-spec-20260516-1300.md +219 -0
- package/docs/design/h1-storage-aggregation-spec-20260518-1121.md +299 -0
- package/docs/design/h2-getdatabase-encapsulation-spec-20260518-1450.md +191 -0
- package/docs/design/h3-fallback-removal-spec-20260518-1245.md +76 -0
- package/docs/design/h4-index-dedup-spec-20260518-1230.md +109 -0
- package/docs/design/h6-services-migration-spec-20260518-1355.md +82 -0
- package/docs/design/hook-failure-queue-spec-20260516-1530.md +204 -0
- package/docs/design/l1-swarm-protocol-extract-spec-20260518-1605.md +106 -0
- package/docs/design/m10-forge-paths-spec-20260518-1320.md +121 -0
- package/docs/design/m2-m3-tool-input-spec-20260518-1425.md +131 -0
- package/docs/design/m7-routing-event-association-spec-20260518-1545.md +103 -0
- package/docs/design/project-path-gitroot-spec-20260518-1715.md +134 -0
- package/docs/design/refactor-phase1-spec-20260515-1600.md +543 -0
- package/docs/design/refactor-phase2-spec-20260515-1700.md +424 -0
- package/docs/design/task-active-gc-spec-20260518-1745.md +146 -0
- package/docs/design/tasks-list-filter-pagination-spec-20260518-0930.md +208 -0
- package/docs/implementation/fix-skills-data-and-set-leak-changelog-20260516-1300.md +104 -0
- package/docs/implementation/h1-storage-aggregation-changelog-20260518-1121.md +82 -0
- package/docs/implementation/h2-final-changelog-20260518-1530.md +61 -0
- package/docs/implementation/h2-phase1-safety-net-changelog-20260518-1450.md +70 -0
- package/docs/implementation/h2-phase2-operations-changelog-20260518-1450.md +120 -0
- package/docs/implementation/h2-phase3-callsites-changelog-20260518-1450.md +71 -0
- package/docs/implementation/h3-fallback-removal-changelog-20260518-1245.md +71 -0
- package/docs/implementation/h4-index-dedup-changelog-20260518-1230.md +60 -0
- package/docs/implementation/h6-services-migration-changelog-20260518-1355.md +46 -0
- package/docs/implementation/h7-m9-defaults-changelog-20260518-1300.md +46 -0
- package/docs/implementation/hook-failure-queue-changelog-20260516-1530.md +196 -0
- package/docs/implementation/hotfix-daemon-event-reject-20260516-1430.md +56 -0
- package/docs/implementation/l1-swarm-protocol-extract-changelog-20260518-1605.md +45 -0
- package/docs/implementation/l3-l4-daemon-perf-changelog-20260518-1410.md +63 -0
- package/docs/implementation/l6-l8-final-cleanup-changelog-20260518-1640.md +38 -0
- package/docs/implementation/m1-m4-m5-l7-cleanup-changelog-20260518-1310.md +58 -0
- package/docs/implementation/m10-forge-paths-changelog-20260518-1320.md +60 -0
- package/docs/implementation/m2-m3-tool-input-changelog-20260518-1425.md +43 -0
- package/docs/implementation/m6-m8-naming-shutdown-changelog-20260518-1340.md +56 -0
- package/docs/implementation/m7-routing-association-changelog-20260518-1545.md +69 -0
- package/docs/implementation/project-path-gitroot-changelog-20260518-1715.md +63 -0
- package/docs/implementation/refactor-phase1-changelog-20260515-1630.md +354 -0
- package/docs/implementation/refactor-phase2-changelog-20260515-1705.md +421 -0
- package/docs/implementation/task-active-gc-changelog-20260518-1745.md +35 -0
- package/docs/implementation/task-title-summary-changelog-20260518-1130.md +39 -0
- package/docs/implementation/tasks-detail-back-loses-filters-changelog-20260518-1100.md +22 -0
- package/docs/implementation/tasks-list-filter-pagination-changelog-20260518-0930.md +72 -0
- package/docs/implementation/tasks-page-white-screen-hotfix-changelog-20260518-1015.md +56 -0
- package/docs/reviews/claudemd-template-sync.md +54 -0
- package/docs/reviews/task-title-summary.md +92 -0
- package/docs/reviews/tasks-detail-back-loses-filters.md +58 -0
- package/docs/reviews/tasks-filter-pagination.md +80 -0
- package/docs/reviews/tasks-page-white-screen-hotfix.md +126 -0
- package/docs/ruflo-learning-strategy.md +322 -0
- package/docs/skills-deduplication-analysis.md +83 -0
- package/docs/skills-multiformat-support.md +177 -0
- package/docs/skills-third-party.md +183 -0
- package/docs/testing/tasks-filter-pagination-test-report.md +86 -0
- package/forge +321 -0
- package/package.json +28 -62
- package/playwright.config.ts +40 -0
- package/scripts/demo-v2.ts +91 -0
- package/scripts/dev-daemon.sh +232 -0
- package/scripts/dev-web.ts +109 -0
- package/scripts/e2e-mcp-link.ts +423 -0
- package/scripts/e2e-methodology-quality.ts +253 -0
- package/scripts/e2e-routing.ts +456 -0
- package/scripts/e2e-user-methodology.ts +326 -0
- package/scripts/e2e-web-workflows.ts +299 -0
- package/scripts/migrate-legacy-to-dynamic.sql +108 -0
- package/scripts/regenerate-execution-docs.ts +116 -0
- package/scripts/sync-agent-skills.ts +193 -0
- package/scripts/test-hook.sh +71 -0
- package/scripts/verify-skill-loading.ts +62 -0
- package/src/claudemd/claudemd-generator.ts +568 -0
- package/src/claudemd/convention-extractor.ts +69 -0
- package/src/claudemd/index.ts +35 -0
- package/src/claudemd/persona-manager.ts +88 -0
- package/src/claudemd/resume-manager.ts +236 -0
- package/src/claudemd/tech-detector.ts +220 -0
- package/src/claudemd/templates/swarm-protocol.md +222 -0
- package/src/cli/commands/claudemd.ts +84 -0
- package/src/cli/commands/config.ts +46 -0
- package/src/cli/commands/daemon.ts +310 -0
- package/src/cli/commands/executions.ts +115 -0
- package/src/cli/commands/init.ts +204 -0
- package/src/cli/commands/logs.ts +181 -0
- package/src/cli/commands/mcp.ts +242 -0
- package/src/cli/commands/menu.ts +357 -0
- package/src/cli/commands/skills.ts +185 -0
- package/src/cli/commands/stats.ts +73 -0
- package/src/cli/commands/status.ts +69 -0
- package/src/cli/commands/template.ts +77 -0
- package/src/cli/commands/trace.ts +148 -0
- package/src/cli/index.ts +42 -0
- package/src/cli/init/hook-manager.ts +132 -0
- package/src/core/ai/provider.ts +308 -0
- package/src/core/ai/types.ts +51 -0
- package/src/core/config.ts +124 -0
- package/src/core/constants.ts +62 -0
- package/src/core/event-fields.ts +32 -0
- package/src/core/queue/index.ts +192 -0
- package/src/core/storage/base.ts +302 -0
- package/src/core/storage/events.ts +434 -0
- package/src/core/storage/injections.ts +78 -0
- package/src/core/storage/maintenance.ts +59 -0
- package/src/core/storage/migrations/002_add_skill_tracking.sql +6 -0
- package/src/core/storage/migrations/003_add_skill_invocations.sql +23 -0
- package/src/core/storage/performance-indexes.sql +23 -0
- package/src/core/storage/routing.ts +322 -0
- package/src/core/storage/rows.ts +112 -0
- package/src/core/storage/schema.sql +224 -0
- package/src/core/storage/sessions.ts +168 -0
- package/src/core/storage/skills.ts +233 -0
- package/src/core/storage/sqlite.ts +293 -0
- package/src/core/storage/tasks.ts +318 -0
- package/src/core/storage/token-usage.ts +93 -0
- package/src/core/types.ts +181 -0
- package/src/core/utils/error-handler.ts +257 -0
- package/src/core/utils/forge-resume-block.ts +74 -0
- package/src/core/utils/format.ts +69 -0
- package/src/core/utils/git.ts +23 -0
- package/src/core/utils/logger.ts +134 -0
- package/src/core/utils/lru-cache.ts +54 -0
- package/src/core/utils/path.ts +19 -0
- package/src/core/utils/session.ts +26 -0
- package/src/core/utils/time.ts +37 -0
- package/src/core/utils/token-tracker.ts +97 -0
- package/src/daemon/event-parser.ts +36 -0
- package/src/daemon/handlers/history-exporter.ts +117 -0
- package/src/daemon/handlers/post-tool-use.ts +54 -0
- package/src/daemon/handlers/stop.ts +208 -0
- package/src/daemon/handlers/user-prompt.ts +178 -0
- package/src/daemon/index.ts +292 -0
- package/src/daemon/launchd/com.claude-forge.daemon.plist.template +47 -0
- package/src/daemon/launchd-installer.ts +260 -0
- package/src/daemon/lifecycle.ts +128 -0
- package/src/daemon/router.ts +40 -0
- package/src/daemon/server.ts +196 -0
- package/src/daemon/services/task-segmenter.ts +112 -0
- package/src/hooks/hook-lib.sh +118 -0
- package/src/hooks/notification.sh +35 -0
- package/src/hooks/post-tool-use.sh +61 -0
- package/src/hooks/pre-tool-use.sh +63 -0
- package/src/hooks/stop.sh +43 -0
- package/src/hooks/user-prompt-submit.sh +69 -0
- package/src/mcp/server.ts +322 -0
- package/src/skills/index.ts +2 -0
- package/src/skills/invocation-guard.ts +177 -0
- package/src/skills/matcher.ts +148 -0
- package/src/skills/official/code-simplifier.md +16 -0
- package/src/skills/official/find-skills.md +23 -0
- package/src/skills/official/official-api-design.md +17 -0
- package/src/skills/official/official-architecture-decision.md +20 -0
- package/src/skills/official/official-bmad.md +118 -0
- package/src/skills/official/official-db-schema-design.md +16 -0
- package/src/skills/official/official-debug.md +17 -0
- package/src/skills/official/official-doc-driven.md +31 -0
- package/src/skills/official/official-harness-engineering.md +108 -0
- package/src/skills/official/official-performance-optimization.md +30 -0
- package/src/skills/official/official-pr-review.md +35 -0
- package/src/skills/official/official-release-checklist.md +30 -0
- package/src/skills/official/official-security-hardening.md +26 -0
- package/src/skills/official/official-spec-driven-design.md +31 -0
- package/src/skills/official/planning-with-files.md +37 -0
- package/src/skills/official/ui-ux-pro-max.md +18 -0
- package/src/skills/official/webapp-testing.md +12 -0
- package/src/skills/official-skills.ts +89 -0
- package/src/skills/registry.ts +355 -0
- package/src/skills/semantic-matcher.ts +231 -0
- package/src/skills/tools/pipeline-suggest.ts +226 -0
- package/src/skills/tools/skill-invoke.ts +168 -0
- package/src/skills/tools/skill-list.ts +59 -0
- package/src/templates/go.yaml +53 -0
- package/src/templates/python.yaml +59 -0
- package/src/templates/react.yaml +55 -0
- package/src/templates/template-manager.ts +170 -0
- package/src/web/analytics/anti-pattern-detector.ts +367 -0
- package/src/web/analytics/drift-detector.ts +219 -0
- package/src/web/analytics/weekly-report.ts +431 -0
- package/src/web/auth-middleware.ts +54 -0
- package/src/web/routes/_helpers.ts +34 -0
- package/src/web/routes/ai.ts +204 -0
- package/src/web/routes/auth.ts +22 -0
- package/src/web/routes/drift.ts +25 -0
- package/src/web/routes/error-handler.ts +120 -0
- package/src/web/routes/events.ts +47 -0
- package/src/web/routes/insights.ts +43 -0
- package/src/web/routes/patch.ts +117 -0
- package/src/web/routes/reports.ts +34 -0
- package/src/web/routes/rules.ts +76 -0
- package/src/web/routes/sessions.ts +250 -0
- package/src/web/routes/skill-stats.ts +92 -0
- package/src/web/routes/skills.ts +350 -0
- package/src/web/routes/static.ts +67 -0
- package/src/web/routes/stats.ts +50 -0
- package/src/web/routes/status.ts +30 -0
- package/src/web/routes/tasks.ts +193 -0
- package/src/web/routes/token-usage.ts +20 -0
- package/src/web/routes/trace.ts +126 -0
- package/src/web/routes/types.ts +57 -0
- package/src/web/server.ts +134 -0
- package/src/web/ssrf-guard.ts +112 -0
- package/src/web/static/index.html +3251 -0
- package/src/web/static/vendor/chart.umd.min.js +20 -0
- package/tests/e2e/dashboard.spec.ts +205 -0
- package/tests/e2e/routing-skill-e2e.test.ts +39 -0
- package/tests/helpers/mock-ai.ts +92 -0
- package/tests/helpers/mock-storage.ts +159 -0
- package/tests/integration/claudemd-generator.test.ts +90 -0
- package/tests/integration/queue-replay.integration.test.ts +193 -0
- package/tests/integration/tasks-filter.integration.test.ts +154 -0
- package/tests/integration/web-analytics.integration.test.ts +133 -0
- package/tests/integration/web-stats.integration.test.ts +135 -0
- package/tests/integration/web-trace.integration.test.ts +175 -0
- package/tests/performance/database.benchmark.ts +161 -0
- package/tests/semantic-matcher.test.ts +99 -0
- package/tests/skill-matcher.test.ts +110 -0
- package/tests/unit/ai-provider-retry.test.ts +194 -0
- package/tests/unit/ai-provider-vision.test.ts +224 -0
- package/tests/unit/claudemd-generator.test.ts +68 -0
- package/tests/unit/cli-mcp.test.ts +141 -0
- package/tests/unit/core/forge-paths.test.ts +99 -0
- package/tests/unit/daemon/post-tool-use.test.ts +121 -0
- package/tests/unit/daemon/stop-handler-behavior-summary.test.ts +202 -0
- package/tests/unit/daemon/task-segmenter-recover.test.ts +84 -0
- package/tests/unit/event-fields.test.ts +88 -0
- package/tests/unit/event-parser.test.ts +55 -0
- package/tests/unit/handlers.test.ts +171 -0
- package/tests/unit/hooks/resolve-project-path.test.ts +122 -0
- package/tests/unit/invocation-guard.test.ts +125 -0
- package/tests/unit/queue.test.ts +272 -0
- package/tests/unit/router.test.ts +138 -0
- package/tests/unit/security.test.ts +128 -0
- package/tests/unit/skill-invocations-workflow.test.ts +495 -0
- package/tests/unit/skill-registry.test.ts +94 -0
- package/tests/unit/skills/invocation-guard-ttl.test.ts +211 -0
- package/tests/unit/skills/official-skills-loader.test.ts +126 -0
- package/tests/unit/skills/registry-multiformat.test.ts +92 -0
- package/tests/unit/socket-server.test.ts +183 -0
- package/tests/unit/storage/event-operations-aggregates.test.ts +342 -0
- package/tests/unit/storage/migration-idempotent.test.ts +304 -0
- package/tests/unit/storage/routing-aggregates.test.ts +276 -0
- package/tests/unit/storage/routing.test.ts +117 -0
- package/tests/unit/storage/schema-missing.test.ts +81 -0
- package/tests/unit/storage/session-operations-aggregates.test.ts +120 -0
- package/tests/unit/storage/sessions-aggregate.test.ts +435 -0
- package/tests/unit/storage/skill-operations-counts.test.ts +106 -0
- package/tests/unit/storage/skills-aggregates.test.ts +104 -0
- package/tests/unit/storage/sqlite-refactor-harness.test.ts +314 -0
- package/tests/unit/storage/task-operations-counts.test.ts +46 -0
- package/tests/unit/storage/tasks-getById.test.ts +343 -0
- package/tests/unit/storage/tasks-stale-gc.test.ts +86 -0
- package/tests/unit/storage.test.ts +172 -0
- package/tests/unit/token-usage.test.ts +144 -0
- package/tests/unit/type-guards.test.ts +201 -0
- package/tests/unit/utils/format.test.ts +189 -0
- package/tests/unit/utils/session.test.ts +89 -0
- package/tests/unit/utils/time.test.ts +112 -0
- package/tests/unit/web/navigation-back-contract.test.ts +134 -0
- package/tests/unit/web/routes-auth.test.ts +93 -0
- package/tests/unit/web/routes-events.test.ts +101 -0
- package/tests/unit/web/routes-rules.test.ts +182 -0
- package/tests/unit/web/routes-sessions.test.ts +181 -0
- package/tests/unit/web/routes-skill-stats.test.ts +179 -0
- package/tests/unit/web/routes-stats.test.ts +92 -0
- package/tests/unit/web/routes-tasks.test.ts +385 -0
- package/tests/unit/web/task-title-contract.test.ts +210 -0
- package/tests/unit/web/tasks-component-contract.test.ts +179 -0
- package/tsconfig.json +22 -0
- package/vitest.config.ts +21 -0
- package/vitest.integration.config.ts +16 -0
- package/web/CLAUDE.md +20 -0
- package/web/index.html +13 -0
- package/web/package-lock.json +4854 -0
- package/web/package.json +35 -0
- package/web/postcss.config.js +6 -0
- package/web/src/App.tsx +110 -0
- package/web/src/components/CodeBlock.tsx +31 -0
- package/web/src/components/Confirm.tsx +96 -0
- package/web/src/components/Drawer.tsx +60 -0
- package/web/src/components/Layout.tsx +145 -0
- package/web/src/components/MarkdownRenderer.tsx +77 -0
- package/web/src/components/SearchInput.tsx +31 -0
- package/web/src/components/SessionDetailContent.tsx +157 -0
- package/web/src/components/Toast.tsx +92 -0
- package/web/src/index.css +19 -0
- package/web/src/main.tsx +31 -0
- package/web/src/pages/AIConfig.tsx +233 -0
- package/web/src/pages/Dashboard.tsx +572 -0
- package/web/src/pages/Events.tsx +271 -0
- package/web/src/pages/Reports.tsx +428 -0
- package/web/src/pages/SessionDetail.tsx +162 -0
- package/web/src/pages/Sessions.tsx +205 -0
- package/web/src/pages/Skills.tsx +180 -0
- package/web/src/pages/TaskDetail.tsx +515 -0
- package/web/src/pages/Tasks.tsx +415 -0
- package/web/src/utils/auth.ts +59 -0
- package/web/src/utils/export.ts +54 -0
- package/web/src/utils/navigation.ts +25 -0
- package/web/src/utils/task-title.ts +49 -0
- package/web/src/utils/time.ts +13 -0
- package/web/tailwind.config.js +11 -0
- package/web/tsconfig.json +21 -0
- package/web/tsconfig.node.json +10 -0
- package/web/vite.config.ts +76 -0
- package/winspan-claude-forge-8.43.0.tgz +0 -0
- package/dist/agents/definition.d.ts +0 -62
- package/dist/agents/definition.d.ts.map +0 -1
- package/dist/agents/definition.js +0 -27
- package/dist/agents/definition.js.map +0 -1
- package/dist/agents/distributor.d.ts +0 -23
- package/dist/agents/distributor.d.ts.map +0 -1
- package/dist/agents/distributor.js +0 -85
- package/dist/agents/distributor.js.map +0 -1
- package/dist/agents/index.d.ts +0 -5
- package/dist/agents/index.d.ts.map +0 -1
- package/dist/agents/index.js +0 -5
- package/dist/agents/index.js.map +0 -1
- package/dist/agents/methodologies/agent-builder.d.ts +0 -21
- package/dist/agents/methodologies/agent-builder.d.ts.map +0 -1
- package/dist/agents/methodologies/agent-builder.js +0 -149
- package/dist/agents/methodologies/agent-builder.js.map +0 -1
- package/dist/agents/methodologies/phases/bmad/analyze.d.ts +0 -3
- package/dist/agents/methodologies/phases/bmad/analyze.d.ts.map +0 -1
- package/dist/agents/methodologies/phases/bmad/analyze.js +0 -19
- package/dist/agents/methodologies/phases/bmad/analyze.js.map +0 -1
- package/dist/agents/methodologies/phases/bmad/design.d.ts +0 -3
- package/dist/agents/methodologies/phases/bmad/design.d.ts.map +0 -1
- package/dist/agents/methodologies/phases/bmad/design.js +0 -18
- package/dist/agents/methodologies/phases/bmad/design.js.map +0 -1
- package/dist/agents/methodologies/phases/bmad/implement.d.ts +0 -3
- package/dist/agents/methodologies/phases/bmad/implement.d.ts.map +0 -1
- package/dist/agents/methodologies/phases/bmad/implement.js +0 -17
- package/dist/agents/methodologies/phases/bmad/implement.js.map +0 -1
- package/dist/agents/methodologies/phases/bmad/index.d.ts +0 -6
- package/dist/agents/methodologies/phases/bmad/index.d.ts.map +0 -1
- package/dist/agents/methodologies/phases/bmad/index.js +0 -6
- package/dist/agents/methodologies/phases/bmad/index.js.map +0 -1
- package/dist/agents/methodologies/phases/bmad/review.d.ts +0 -3
- package/dist/agents/methodologies/phases/bmad/review.d.ts.map +0 -1
- package/dist/agents/methodologies/phases/bmad/review.js +0 -17
- package/dist/agents/methodologies/phases/bmad/review.js.map +0 -1
- package/dist/agents/methodologies/phases/bmad/test.d.ts +0 -3
- package/dist/agents/methodologies/phases/bmad/test.d.ts.map +0 -1
- package/dist/agents/methodologies/phases/bmad/test.js +0 -21
- package/dist/agents/methodologies/phases/bmad/test.js.map +0 -1
- package/dist/agents/methodologies/phases/harness/fix.d.ts +0 -3
- package/dist/agents/methodologies/phases/harness/fix.d.ts.map +0 -1
- package/dist/agents/methodologies/phases/harness/fix.js +0 -17
- package/dist/agents/methodologies/phases/harness/fix.js.map +0 -1
- package/dist/agents/methodologies/phases/harness/index.d.ts +0 -6
- package/dist/agents/methodologies/phases/harness/index.d.ts.map +0 -1
- package/dist/agents/methodologies/phases/harness/index.js +0 -6
- package/dist/agents/methodologies/phases/harness/index.js.map +0 -1
- package/dist/agents/methodologies/phases/harness/reproduce.d.ts +0 -3
- package/dist/agents/methodologies/phases/harness/reproduce.d.ts.map +0 -1
- package/dist/agents/methodologies/phases/harness/reproduce.js +0 -20
- package/dist/agents/methodologies/phases/harness/reproduce.js.map +0 -1
- package/dist/agents/methodologies/phases/harness/root-cause.d.ts +0 -3
- package/dist/agents/methodologies/phases/harness/root-cause.d.ts.map +0 -1
- package/dist/agents/methodologies/phases/harness/root-cause.js +0 -21
- package/dist/agents/methodologies/phases/harness/root-cause.js.map +0 -1
- package/dist/agents/methodologies/phases/harness/safety-net.d.ts +0 -3
- package/dist/agents/methodologies/phases/harness/safety-net.d.ts.map +0 -1
- package/dist/agents/methodologies/phases/harness/safety-net.js +0 -17
- package/dist/agents/methodologies/phases/harness/safety-net.js.map +0 -1
- package/dist/agents/methodologies/phases/harness/verify.d.ts +0 -3
- package/dist/agents/methodologies/phases/harness/verify.d.ts.map +0 -1
- package/dist/agents/methodologies/phases/harness/verify.js +0 -22
- package/dist/agents/methodologies/phases/harness/verify.js.map +0 -1
- package/dist/agents/methodologies/presets.d.ts +0 -10
- package/dist/agents/methodologies/presets.d.ts.map +0 -1
- package/dist/agents/methodologies/presets.js +0 -79
- package/dist/agents/methodologies/presets.js.map +0 -1
- package/dist/agents/methodologies/types.d.ts +0 -45
- package/dist/agents/methodologies/types.d.ts.map +0 -1
- package/dist/agents/methodologies/types.js +0 -10
- package/dist/agents/methodologies/types.js.map +0 -1
- package/dist/agents/methodologies/user-config-loader.d.ts +0 -30
- package/dist/agents/methodologies/user-config-loader.d.ts.map +0 -1
- package/dist/agents/methodologies/user-config-loader.js +0 -159
- package/dist/agents/methodologies/user-config-loader.js.map +0 -1
- package/dist/agents/official-agents.d.ts +0 -4
- package/dist/agents/official-agents.d.ts.map +0 -1
- package/dist/agents/official-agents.js +0 -559
- package/dist/agents/official-agents.js.map +0 -1
- package/dist/agents/registry.d.ts +0 -57
- package/dist/agents/registry.d.ts.map +0 -1
- package/dist/agents/registry.js +0 -271
- package/dist/agents/registry.js.map +0 -1
- package/dist/capability/index.d.ts +0 -10
- package/dist/capability/index.d.ts.map +0 -1
- package/dist/capability/index.js +0 -10
- package/dist/capability/index.js.map +0 -1
- package/dist/capability/types.d.ts +0 -10
- package/dist/capability/types.d.ts.map +0 -1
- package/dist/capability/types.js +0 -10
- package/dist/capability/types.js.map +0 -1
- package/dist/cli/commands/agents.d.ts +0 -3
- package/dist/cli/commands/agents.d.ts.map +0 -1
- package/dist/cli/commands/agents.js +0 -62
- package/dist/cli/commands/agents.js.map +0 -1
- package/dist/cli/commands/rules.d.ts +0 -8
- package/dist/cli/commands/rules.d.ts.map +0 -1
- package/dist/cli/commands/rules.js +0 -89
- package/dist/cli/commands/rules.js.map +0 -1
- package/dist/daemon/auto-disable-scheduler.d.ts +0 -53
- package/dist/daemon/auto-disable-scheduler.d.ts.map +0 -1
- package/dist/daemon/auto-disable-scheduler.js +0 -114
- package/dist/daemon/auto-disable-scheduler.js.map +0 -1
- package/dist/daemon/handlers/pre-tool-use.d.ts +0 -39
- package/dist/daemon/handlers/pre-tool-use.d.ts.map +0 -1
- package/dist/daemon/handlers/pre-tool-use.js +0 -166
- package/dist/daemon/handlers/pre-tool-use.js.map +0 -1
- package/dist/daemon/routing-observer.d.ts +0 -42
- package/dist/daemon/routing-observer.d.ts.map +0 -1
- package/dist/daemon/routing-observer.js +0 -264
- package/dist/daemon/routing-observer.js.map +0 -1
- package/dist/daemon/routing-state.d.ts +0 -64
- package/dist/daemon/routing-state.d.ts.map +0 -1
- package/dist/daemon/routing-state.js +0 -240
- package/dist/daemon/routing-state.js.map +0 -1
- package/dist/engine/agent-router.d.ts +0 -142
- package/dist/engine/agent-router.d.ts.map +0 -1
- package/dist/engine/agent-router.js +0 -276
- package/dist/engine/agent-router.js.map +0 -1
- package/dist/engine/context-builder.d.ts +0 -23
- package/dist/engine/context-builder.d.ts.map +0 -1
- package/dist/engine/context-builder.js +0 -63
- package/dist/engine/context-builder.js.map +0 -1
- package/dist/engine/conventions/basic-security.yaml +0 -109
- package/dist/engine/conventions/code-quality.yaml +0 -123
- package/dist/engine/conventions/database-safety.yaml +0 -74
- package/dist/engine/conventions/dependency-safety.yaml +0 -132
- package/dist/engine/conventions/docker-safety.yaml +0 -69
- package/dist/engine/conventions/git-safety.yaml +0 -118
- package/dist/engine/conventions/go-best-practices.yaml +0 -84
- package/dist/engine/conventions/python-best-practices.yaml +0 -96
- package/dist/engine/conventions/react-best-practices.yaml +0 -96
- package/dist/engine/conventions/routing.yaml +0 -378
- package/dist/engine/conventions/strict-security.yaml +0 -30
- package/dist/engine/conventions/ts-quality.yaml +0 -49
- package/dist/engine/dsl/compiler.d.ts +0 -34
- package/dist/engine/dsl/compiler.d.ts.map +0 -1
- package/dist/engine/dsl/compiler.js +0 -702
- package/dist/engine/dsl/compiler.js.map +0 -1
- package/dist/engine/dsl/parser.d.ts +0 -25
- package/dist/engine/dsl/parser.d.ts.map +0 -1
- package/dist/engine/dsl/parser.js +0 -208
- package/dist/engine/dsl/parser.js.map +0 -1
- package/dist/engine/dsl/runtime.d.ts +0 -46
- package/dist/engine/dsl/runtime.d.ts.map +0 -1
- package/dist/engine/dsl/runtime.js +0 -173
- package/dist/engine/dsl/runtime.js.map +0 -1
- package/dist/engine/dsl/types.d.ts +0 -139
- package/dist/engine/dsl/types.d.ts.map +0 -1
- package/dist/engine/dsl/types.js +0 -11
- package/dist/engine/dsl/types.js.map +0 -1
- package/dist/engine/evidence-store.d.ts +0 -44
- package/dist/engine/evidence-store.d.ts.map +0 -1
- package/dist/engine/evidence-store.js +0 -109
- package/dist/engine/evidence-store.js.map +0 -1
- package/dist/engine/experiment-router.d.ts +0 -102
- package/dist/engine/experiment-router.d.ts.map +0 -1
- package/dist/engine/experiment-router.js +0 -289
- package/dist/engine/experiment-router.js.map +0 -1
- package/dist/engine/recommender.d.ts +0 -52
- package/dist/engine/recommender.d.ts.map +0 -1
- package/dist/engine/recommender.js +0 -162
- package/dist/engine/recommender.js.map +0 -1
- package/dist/engine/rule-engine.d.ts +0 -33
- package/dist/engine/rule-engine.d.ts.map +0 -1
- package/dist/engine/rule-engine.js +0 -250
- package/dist/engine/rule-engine.js.map +0 -1
- package/dist/engine/security-gates.d.ts +0 -42
- package/dist/engine/security-gates.d.ts.map +0 -1
- package/dist/engine/security-gates.js +0 -210
- package/dist/engine/security-gates.js.map +0 -1
- package/dist/intelligence/classifier.d.ts +0 -75
- package/dist/intelligence/classifier.d.ts.map +0 -1
- package/dist/intelligence/classifier.js +0 -352
- package/dist/intelligence/classifier.js.map +0 -1
- package/dist/intelligence/context-gatherer.d.ts +0 -101
- package/dist/intelligence/context-gatherer.d.ts.map +0 -1
- package/dist/intelligence/context-gatherer.js +0 -417
- package/dist/intelligence/context-gatherer.js.map +0 -1
- package/dist/intelligence/cot-classifier.d.ts +0 -95
- package/dist/intelligence/cot-classifier.d.ts.map +0 -1
- package/dist/intelligence/cot-classifier.js +0 -391
- package/dist/intelligence/cot-classifier.js.map +0 -1
- package/dist/intelligence/distiller.d.ts +0 -22
- package/dist/intelligence/distiller.d.ts.map +0 -1
- package/dist/intelligence/distiller.js +0 -108
- package/dist/intelligence/distiller.js.map +0 -1
- package/dist/intelligence/execution-doc-builder.d.ts +0 -151
- package/dist/intelligence/execution-doc-builder.d.ts.map +0 -1
- package/dist/intelligence/execution-doc-builder.js +0 -1018
- package/dist/intelligence/execution-doc-builder.js.map +0 -1
- package/dist/intelligence/intent-types.d.ts +0 -13
- package/dist/intelligence/intent-types.d.ts.map +0 -1
- package/dist/intelligence/intent-types.js +0 -19
- package/dist/intelligence/intent-types.js.map +0 -1
- package/dist/intelligence/multimodal-parser.d.ts +0 -105
- package/dist/intelligence/multimodal-parser.d.ts.map +0 -1
- package/dist/intelligence/multimodal-parser.js +0 -425
- package/dist/intelligence/multimodal-parser.js.map +0 -1
- package/dist/intelligence/quality-gate.d.ts +0 -45
- package/dist/intelligence/quality-gate.d.ts.map +0 -1
- package/dist/intelligence/quality-gate.js +0 -193
- package/dist/intelligence/quality-gate.js.map +0 -1
- package/dist/intelligence/task-segmenter.d.ts.map +0 -1
- package/dist/intelligence/task-segmenter.js.map +0 -1
- package/dist/web/routes/agents.d.ts +0 -7
- package/dist/web/routes/agents.d.ts.map +0 -1
- package/dist/web/routes/agents.js +0 -209
- package/dist/web/routes/agents.js.map +0 -1
- package/dist/web/routes/execution-trace.d.ts +0 -21
- package/dist/web/routes/execution-trace.d.ts.map +0 -1
- package/dist/web/routes/execution-trace.js +0 -353
- package/dist/web/routes/execution-trace.js.map +0 -1
- package/dist/web/routes/experiments.d.ts +0 -15
- package/dist/web/routes/experiments.d.ts.map +0 -1
- package/dist/web/routes/experiments.js +0 -187
- package/dist/web/routes/experiments.js.map +0 -1
- package/dist/web/routes/routing.d.ts +0 -17
- package/dist/web/routes/routing.d.ts.map +0 -1
- package/dist/web/routes/routing.js +0 -592
- package/dist/web/routes/routing.js.map +0 -1
- package/dist/web/routes/workflows.d.ts +0 -19
- package/dist/web/routes/workflows.d.ts.map +0 -1
- package/dist/web/routes/workflows.js +0 -86
- package/dist/web/routes/workflows.js.map +0 -1
- package/dist/web/static/assets/AIConfig-R5wZ3ZKT.js +0 -2
- package/dist/web/static/assets/AIConfig-R5wZ3ZKT.js.map +0 -1
- package/dist/web/static/assets/Agents-Beg34V1g.js +0 -2
- package/dist/web/static/assets/Agents-Beg34V1g.js.map +0 -1
- package/dist/web/static/assets/CodeBlock--H53gk46.js +0 -2
- package/dist/web/static/assets/CodeBlock--H53gk46.js.map +0 -1
- package/dist/web/static/assets/Dashboard-Cy1xsj1J.js +0 -2
- package/dist/web/static/assets/Dashboard-Cy1xsj1J.js.map +0 -1
- package/dist/web/static/assets/Events-mFhXl4zI.js +0 -2
- package/dist/web/static/assets/Events-mFhXl4zI.js.map +0 -1
- package/dist/web/static/assets/ExecutionTrace-DG901hLR.js +0 -3
- package/dist/web/static/assets/ExecutionTrace-DG901hLR.js.map +0 -1
- package/dist/web/static/assets/MarkdownRenderer-CCIz1MOz.js +0 -2
- package/dist/web/static/assets/MarkdownRenderer-CCIz1MOz.js.map +0 -1
- package/dist/web/static/assets/Routing-B7BFLfjh.js +0 -2
- package/dist/web/static/assets/Routing-B7BFLfjh.js.map +0 -1
- package/dist/web/static/assets/SessionDetail-BT0l4RrK.js +0 -2
- package/dist/web/static/assets/SessionDetail-BT0l4RrK.js.map +0 -1
- package/dist/web/static/assets/Sessions-C6J_HQ_u.js +0 -2
- package/dist/web/static/assets/Sessions-C6J_HQ_u.js.map +0 -1
- package/dist/web/static/assets/Skills-4DQWLaTv.js +0 -2
- package/dist/web/static/assets/Skills-4DQWLaTv.js.map +0 -1
- package/dist/web/static/assets/WorkflowDetail-zhNqUkBE.js +0 -2
- package/dist/web/static/assets/WorkflowDetail-zhNqUkBE.js.map +0 -1
- package/dist/web/static/assets/Workflows-Btvi-lGw.js +0 -2
- package/dist/web/static/assets/Workflows-Btvi-lGw.js.map +0 -1
- package/dist/web/static/assets/export-BQQZLaHV.js +0 -4
- package/dist/web/static/assets/export-BQQZLaHV.js.map +0 -1
- package/dist/web/static/assets/index-Cgr9qMtq.js +0 -3
- package/dist/web/static/assets/index-Cgr9qMtq.js.map +0 -1
- package/dist/web/static/assets/index-CngWb5gC.css +0 -1
- package/dist/web/static/assets/lucide-53bR2rki.js.map +0 -1
- package/dist/web/static/assets/react-router-I-HqunH7.js +0 -20
|
@@ -0,0 +1,423 @@
|
|
|
1
|
+
#!/usr/bin/env tsx
|
|
2
|
+
/**
|
|
3
|
+
* E2E Verification: MCP call chain → skill_invoke → DB persistence
|
|
4
|
+
*
|
|
5
|
+
* Black-box verification of the full pipeline:
|
|
6
|
+
*
|
|
7
|
+
* MCP Client (this script)
|
|
8
|
+
* → HTTP POST /mcp (Bearer auth)
|
|
9
|
+
* → StreamableHTTPServerTransport
|
|
10
|
+
* → McpServer.callTool('skill_invoke' / 'skill_list')
|
|
11
|
+
* → skillInvoke() / skillList()
|
|
12
|
+
* → SQLiteStorage.writeSkillInvocation()
|
|
13
|
+
*
|
|
14
|
+
* Steps:
|
|
15
|
+
* 1. Start a standalone WebServer (independent of system daemon)
|
|
16
|
+
* 2. Use MCP SDK Client to connect via HTTP
|
|
17
|
+
* 3. Call skill_invoke / skill_list
|
|
18
|
+
* 4. Query SQLite directly to verify skill_invocations table
|
|
19
|
+
* 5. Print verification report
|
|
20
|
+
*
|
|
21
|
+
* Isolation: temp dir for DB, swaps in test daemon.token (restored on exit).
|
|
22
|
+
*/
|
|
23
|
+
|
|
24
|
+
import { mkdtempSync, rmSync, writeFileSync, readFileSync, existsSync, mkdirSync, unlinkSync } from 'node:fs';
|
|
25
|
+
import { tmpdir } from 'node:os';
|
|
26
|
+
import { join } from 'node:path';
|
|
27
|
+
import type { AddressInfo } from 'node:net';
|
|
28
|
+
|
|
29
|
+
import { SQLiteStorage } from '../src/core/storage/sqlite.js';
|
|
30
|
+
import { RuleEngine } from '../src/engine/rule-engine.js';
|
|
31
|
+
import { SkillRegistry } from '../src/skills/registry.js';
|
|
32
|
+
import { InvocationGuard } from '../src/skills/invocation-guard.js';
|
|
33
|
+
import { WebServer } from '../src/web/server.js';
|
|
34
|
+
import { FORGE_PATHS } from '../src/core/constants.js';
|
|
35
|
+
import { routingState } from '../src/daemon/routing-state.js';
|
|
36
|
+
|
|
37
|
+
import { Client } from '@modelcontextprotocol/sdk/client/index.js';
|
|
38
|
+
import { StreamableHTTPClientTransport } from '@modelcontextprotocol/sdk/client/streamableHttp.js';
|
|
39
|
+
|
|
40
|
+
// ─── Test infra ────────────────────────────────────────────────────────────
|
|
41
|
+
|
|
42
|
+
const TEST_TOKEN = `e2e-mcp-token-${Date.now()}`;
|
|
43
|
+
const TOKEN_FILE = join(FORGE_PATHS.home(), 'daemon.token');
|
|
44
|
+
|
|
45
|
+
const tmp = mkdtempSync(join(tmpdir(), 'forge-e2e-mcp-'));
|
|
46
|
+
const dbPath = join(tmp, 'data.db');
|
|
47
|
+
|
|
48
|
+
interface TestResult {
|
|
49
|
+
step: string;
|
|
50
|
+
pass: boolean;
|
|
51
|
+
details: string;
|
|
52
|
+
}
|
|
53
|
+
const results: TestResult[] = [];
|
|
54
|
+
function record(step: string, pass: boolean, details: string): void {
|
|
55
|
+
results.push({ step, pass, details });
|
|
56
|
+
const mark = pass ? 'PASS' : 'FAIL';
|
|
57
|
+
console.log(`[${mark}] ${step}: ${details}`);
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
// Backup user's daemon.token, install test token (mirrors mcp-server.test.ts pattern).
|
|
61
|
+
let backupToken: string | null = null;
|
|
62
|
+
function installTestToken(): void {
|
|
63
|
+
if (existsSync(TOKEN_FILE)) {
|
|
64
|
+
backupToken = readFileSync(TOKEN_FILE, 'utf-8');
|
|
65
|
+
} else {
|
|
66
|
+
backupToken = null;
|
|
67
|
+
mkdirSync(FORGE_PATHS.home(), { recursive: true });
|
|
68
|
+
}
|
|
69
|
+
writeFileSync(TOKEN_FILE, TEST_TOKEN, { mode: 0o600 });
|
|
70
|
+
}
|
|
71
|
+
function restoreTestToken(): void {
|
|
72
|
+
if (backupToken !== null) {
|
|
73
|
+
writeFileSync(TOKEN_FILE, backupToken, { mode: 0o600 });
|
|
74
|
+
} else if (existsSync(TOKEN_FILE)) {
|
|
75
|
+
unlinkSync(TOKEN_FILE);
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
function authHeaders(extra: Record<string, string> = {}): Record<string, string> {
|
|
80
|
+
return { Authorization: `Bearer ${TEST_TOKEN}`, ...extra };
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
async function newClient(port: number, headers: Record<string, string>): Promise<Client> {
|
|
84
|
+
const transport = new StreamableHTTPClientTransport(new URL(`http://127.0.0.1:${port}/mcp`), {
|
|
85
|
+
requestInit: { headers },
|
|
86
|
+
});
|
|
87
|
+
const client = new Client({ name: 'forge-e2e-client', version: '1.0.0' });
|
|
88
|
+
await client.connect(transport);
|
|
89
|
+
return client;
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
// ─── Main ──────────────────────────────────────────────────────────────────
|
|
93
|
+
|
|
94
|
+
async function main(): Promise<void> {
|
|
95
|
+
console.log('=== E2E MCP Link Verification ===\n');
|
|
96
|
+
console.log(`tmp dir: ${tmp}`);
|
|
97
|
+
console.log(`db: ${dbPath}`);
|
|
98
|
+
console.log(`token: ${TEST_TOKEN}\n`);
|
|
99
|
+
|
|
100
|
+
installTestToken();
|
|
101
|
+
|
|
102
|
+
const storage = new SQLiteStorage(dbPath);
|
|
103
|
+
const ruleEngine = new RuleEngine();
|
|
104
|
+
const skillRegistry = new SkillRegistry();
|
|
105
|
+
const guard = new InvocationGuard();
|
|
106
|
+
|
|
107
|
+
const allSkills = skillRegistry.getAll();
|
|
108
|
+
console.log(`Registry loaded ${allSkills.length} skills`);
|
|
109
|
+
|
|
110
|
+
// ── Start WebServer ──────────────────────────────────────────────────────
|
|
111
|
+
const server = new WebServer({
|
|
112
|
+
port: 0,
|
|
113
|
+
storage,
|
|
114
|
+
ruleEngine,
|
|
115
|
+
skillRegistry,
|
|
116
|
+
invocationGuard: guard,
|
|
117
|
+
});
|
|
118
|
+
|
|
119
|
+
// Pull the express app out and listen on a random port (matches mcp-server.test.ts).
|
|
120
|
+
const app = (server as unknown as { app: import('express').Application }).app;
|
|
121
|
+
const port: number = await new Promise<number>((resolve) => {
|
|
122
|
+
const handle = app.listen(0, () => {
|
|
123
|
+
const addr = handle.address() as AddressInfo;
|
|
124
|
+
// Also give server.server back so afterEach-style cleanup keeps a handle.
|
|
125
|
+
(server as unknown as { server: ReturnType<typeof app.listen> | null }).server = handle;
|
|
126
|
+
resolve(addr.port);
|
|
127
|
+
});
|
|
128
|
+
});
|
|
129
|
+
console.log(`Server listening on http://127.0.0.1:${port}/mcp\n`);
|
|
130
|
+
|
|
131
|
+
const activeClients: Client[] = [];
|
|
132
|
+
|
|
133
|
+
let exitCode = 0;
|
|
134
|
+
try {
|
|
135
|
+
// ── Test 1: tools/list ────────────────────────────────────────────────
|
|
136
|
+
{
|
|
137
|
+
const client = await newClient(port, authHeaders());
|
|
138
|
+
activeClients.push(client);
|
|
139
|
+
const r = await client.listTools();
|
|
140
|
+
const names = r.tools.map((t) => t.name);
|
|
141
|
+
record(
|
|
142
|
+
'list tools',
|
|
143
|
+
names.includes('skill_invoke') && names.includes('skill_list'),
|
|
144
|
+
`tools: ${names.join(', ')}`,
|
|
145
|
+
);
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
// ── Test 2: skill_list returns skills ─────────────────────────────────
|
|
149
|
+
{
|
|
150
|
+
const client = await newClient(port, authHeaders());
|
|
151
|
+
activeClients.push(client);
|
|
152
|
+
const r = await client.callTool({ name: 'skill_list', arguments: {} });
|
|
153
|
+
const content = r.content as Array<{ type: string; text: string }>;
|
|
154
|
+
const parsed = JSON.parse(content[0].text) as { skills: unknown[]; total: number };
|
|
155
|
+
record(
|
|
156
|
+
'skill_list returns skills',
|
|
157
|
+
parsed.total > 0 && parsed.skills.length === parsed.total,
|
|
158
|
+
`${parsed.total} skills returned`,
|
|
159
|
+
);
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
// ── Test 3: skill_invoke base call (with explicit X-Forge-* headers) ──
|
|
163
|
+
const baseSession = 'e2e-session-1';
|
|
164
|
+
const baseRouteId = 'e2e-route-1';
|
|
165
|
+
const baseAgent = 'e2e-agent-bmad';
|
|
166
|
+
let invokeContentLen = 0;
|
|
167
|
+
{
|
|
168
|
+
const client = await newClient(
|
|
169
|
+
port,
|
|
170
|
+
authHeaders({
|
|
171
|
+
'X-Forge-Session-Id': baseSession,
|
|
172
|
+
'X-Forge-Route-Request-Id': baseRouteId,
|
|
173
|
+
'X-Forge-Agent-Id': baseAgent,
|
|
174
|
+
}),
|
|
175
|
+
);
|
|
176
|
+
activeClients.push(client);
|
|
177
|
+
const r = await client.callTool({
|
|
178
|
+
name: 'skill_invoke',
|
|
179
|
+
arguments: { skill_id: 'official-tdd', reason: 'bmad/implement:e2e-test' },
|
|
180
|
+
});
|
|
181
|
+
const content = r.content as Array<{ type: string; text: string }>;
|
|
182
|
+
invokeContentLen = content[0]?.text?.length ?? 0;
|
|
183
|
+
record(
|
|
184
|
+
'skill_invoke base',
|
|
185
|
+
!r.isError && invokeContentLen > 100 && content[0].text.startsWith('# '),
|
|
186
|
+
`content length ${invokeContentLen}`,
|
|
187
|
+
);
|
|
188
|
+
}
|
|
189
|
+
|
|
190
|
+
// ── Test 4: DB row inserted with all linkage fields ───────────────────
|
|
191
|
+
{
|
|
192
|
+
const rows = storage.querySkillInvocations({ session_id: baseSession });
|
|
193
|
+
const row = rows[0];
|
|
194
|
+
record('DB row inserted', rows.length === 1, `rows count = ${rows.length}`);
|
|
195
|
+
record(
|
|
196
|
+
'workflow parsed',
|
|
197
|
+
row?.workflow === 'bmad' && row?.phase === 'implement',
|
|
198
|
+
`workflow=${row?.workflow}, phase=${row?.phase}`,
|
|
199
|
+
);
|
|
200
|
+
record(
|
|
201
|
+
'feature_slug parsed',
|
|
202
|
+
row?.feature_slug === 'e2e-test',
|
|
203
|
+
`feature_slug=${row?.feature_slug}`,
|
|
204
|
+
);
|
|
205
|
+
record(
|
|
206
|
+
'session/route/agent linkage',
|
|
207
|
+
row?.session_id === baseSession &&
|
|
208
|
+
row?.route_request_id === baseRouteId &&
|
|
209
|
+
row?.agent_id === baseAgent,
|
|
210
|
+
`session=${row?.session_id}, route=${row?.route_request_id}, agent=${row?.agent_id}`,
|
|
211
|
+
);
|
|
212
|
+
record(
|
|
213
|
+
'invocation_type=dynamic + success=1',
|
|
214
|
+
row?.invocation_type === 'dynamic' && row?.success === 1,
|
|
215
|
+
`invocation_type=${row?.invocation_type}, success=${row?.success}`,
|
|
216
|
+
);
|
|
217
|
+
}
|
|
218
|
+
|
|
219
|
+
// ── Test 5: Multi-phase aggregation via queryWorkflowProgress ─────────
|
|
220
|
+
// Use a fresh session to avoid the per-session idempotency guard, and
|
|
221
|
+
// a single shared route_request_id so all phases group together.
|
|
222
|
+
const multiSession = 'e2e-session-multi';
|
|
223
|
+
const multiRouteId = 'e2e-route-multi';
|
|
224
|
+
const multiAgent = 'e2e-agent-multi';
|
|
225
|
+
{
|
|
226
|
+
const phaseSkills: Array<[string, string]> = [
|
|
227
|
+
['analyze', 'official-spec-driven-design'],
|
|
228
|
+
['design', 'official-architecture-decision'],
|
|
229
|
+
['implement', 'official-tdd'],
|
|
230
|
+
];
|
|
231
|
+
for (const [phase, skill] of phaseSkills) {
|
|
232
|
+
const c = await newClient(
|
|
233
|
+
port,
|
|
234
|
+
authHeaders({
|
|
235
|
+
'X-Forge-Session-Id': multiSession,
|
|
236
|
+
'X-Forge-Route-Request-Id': multiRouteId,
|
|
237
|
+
'X-Forge-Agent-Id': multiAgent,
|
|
238
|
+
}),
|
|
239
|
+
);
|
|
240
|
+
activeClients.push(c);
|
|
241
|
+
await c.callTool({
|
|
242
|
+
name: 'skill_invoke',
|
|
243
|
+
arguments: { skill_id: skill, reason: `bmad/${phase}:multi-phase-test` },
|
|
244
|
+
});
|
|
245
|
+
}
|
|
246
|
+
const progress = storage.queryWorkflowProgress({ session_id: multiSession });
|
|
247
|
+
const ok =
|
|
248
|
+
progress.length === 1 &&
|
|
249
|
+
progress[0].workflow === 'bmad' &&
|
|
250
|
+
progress[0].route_request_id === multiRouteId &&
|
|
251
|
+
progress[0].phases_completed.length === 3 &&
|
|
252
|
+
progress[0].phases_completed.includes('analyze') &&
|
|
253
|
+
progress[0].phases_completed.includes('design') &&
|
|
254
|
+
progress[0].phases_completed.includes('implement') &&
|
|
255
|
+
progress[0].feature_slug === 'multi-phase-test';
|
|
256
|
+
record(
|
|
257
|
+
'multi-phase aggregation',
|
|
258
|
+
ok,
|
|
259
|
+
`routes=${progress.length}, phases=[${progress[0]?.phases_completed.join(', ')}], feature=${progress[0]?.feature_slug}`,
|
|
260
|
+
);
|
|
261
|
+
}
|
|
262
|
+
|
|
263
|
+
// ── Test 6: Error path — non-existent skill records success=0 ────────
|
|
264
|
+
const errSession = 'e2e-session-error';
|
|
265
|
+
{
|
|
266
|
+
const client = await newClient(
|
|
267
|
+
port,
|
|
268
|
+
authHeaders({ 'X-Forge-Session-Id': errSession }),
|
|
269
|
+
);
|
|
270
|
+
activeClients.push(client);
|
|
271
|
+
const r = await client.callTool({
|
|
272
|
+
name: 'skill_invoke',
|
|
273
|
+
arguments: { skill_id: 'nonexistent-skill-xyz', reason: 'bmad/analyze' },
|
|
274
|
+
});
|
|
275
|
+
const isErrorFlag = r.isError === true;
|
|
276
|
+
const rows = storage.querySkillInvocations({ session_id: errSession });
|
|
277
|
+
const row = rows[0];
|
|
278
|
+
record(
|
|
279
|
+
'error skill returns isError + records success=0',
|
|
280
|
+
isErrorFlag &&
|
|
281
|
+
rows.length === 1 &&
|
|
282
|
+
row.success === 0 &&
|
|
283
|
+
(row.error?.includes('Skill not found') ?? false),
|
|
284
|
+
`isError=${isErrorFlag}, rows=${rows.length}, success=${row?.success}, error="${row?.error?.slice(0, 60) ?? ''}"`,
|
|
285
|
+
);
|
|
286
|
+
}
|
|
287
|
+
|
|
288
|
+
// ── Test 7: Guard blocks repeated calls (idempotency) ─────────────────
|
|
289
|
+
const dupSession = 'e2e-session-dup';
|
|
290
|
+
{
|
|
291
|
+
// First invocation succeeds
|
|
292
|
+
const c1 = await newClient(
|
|
293
|
+
port,
|
|
294
|
+
authHeaders({ 'X-Forge-Session-Id': dupSession }),
|
|
295
|
+
);
|
|
296
|
+
activeClients.push(c1);
|
|
297
|
+
await c1.callTool({
|
|
298
|
+
name: 'skill_invoke',
|
|
299
|
+
arguments: { skill_id: 'official-tdd', reason: 'first call' },
|
|
300
|
+
});
|
|
301
|
+
|
|
302
|
+
// Second invocation of same skill in same session should be blocked
|
|
303
|
+
const c2 = await newClient(
|
|
304
|
+
port,
|
|
305
|
+
authHeaders({ 'X-Forge-Session-Id': dupSession }),
|
|
306
|
+
);
|
|
307
|
+
activeClients.push(c2);
|
|
308
|
+
const r2 = await c2.callTool({
|
|
309
|
+
name: 'skill_invoke',
|
|
310
|
+
arguments: { skill_id: 'official-tdd', reason: 'duplicate' },
|
|
311
|
+
});
|
|
312
|
+
|
|
313
|
+
const rows = storage.querySkillInvocations({ session_id: dupSession });
|
|
314
|
+
// Rows ordered by timestamp DESC: rows[0] is the duplicate (Blocked), rows[1] is the first
|
|
315
|
+
const blockedRow = rows.find((r) => r.success === 0 && r.error?.includes('Blocked'));
|
|
316
|
+
const successRow = rows.find((r) => r.success === 1);
|
|
317
|
+
record(
|
|
318
|
+
'guard blocks repeated calls',
|
|
319
|
+
r2.isError === true &&
|
|
320
|
+
rows.length === 2 &&
|
|
321
|
+
blockedRow !== undefined &&
|
|
322
|
+
successRow !== undefined,
|
|
323
|
+
`2 rows; isError=${r2.isError}; success=${successRow?.success}, blocked.error="${blockedRow?.error?.slice(0, 60) ?? ''}"`,
|
|
324
|
+
);
|
|
325
|
+
}
|
|
326
|
+
|
|
327
|
+
// ── Test 8: Fallback to routingState when X-Forge-Session-Id missing ──
|
|
328
|
+
const fallbackSession = 'e2e-session-fallback';
|
|
329
|
+
const fallbackRouteId = 'e2e-route-fallback';
|
|
330
|
+
{
|
|
331
|
+
routingState.setRouting(fallbackSession, {
|
|
332
|
+
agentName: 'fallback-agent',
|
|
333
|
+
agentDescription: 'desc',
|
|
334
|
+
timestamp: Date.now(),
|
|
335
|
+
routeRequestId: fallbackRouteId,
|
|
336
|
+
});
|
|
337
|
+
// No X-Forge-* headers — server should fall back to routingState.getMostRecent().
|
|
338
|
+
const client = await newClient(port, authHeaders());
|
|
339
|
+
activeClients.push(client);
|
|
340
|
+
await client.callTool({
|
|
341
|
+
name: 'skill_invoke',
|
|
342
|
+
arguments: { skill_id: 'official-refactor', reason: 'fallback test' },
|
|
343
|
+
});
|
|
344
|
+
|
|
345
|
+
const rows = storage.querySkillInvocations({ session_id: fallbackSession });
|
|
346
|
+
const row = rows[0];
|
|
347
|
+
record(
|
|
348
|
+
'routingState fallback (no headers)',
|
|
349
|
+
rows.length === 1 &&
|
|
350
|
+
row?.route_request_id === fallbackRouteId &&
|
|
351
|
+
row?.agent_id === 'fallback-agent',
|
|
352
|
+
`rows=${rows.length}, route=${row?.route_request_id}, agent=${row?.agent_id}`,
|
|
353
|
+
);
|
|
354
|
+
routingState.clearRouting(fallbackSession);
|
|
355
|
+
}
|
|
356
|
+
|
|
357
|
+
// ── Test 9: mcp-anon-* fallback when neither header nor routingState ──
|
|
358
|
+
{
|
|
359
|
+
// routingState should be empty by now (we cleared the fallback above and
|
|
360
|
+
// no daemon is feeding it). Issue a call and look for an mcp-anon-* row.
|
|
361
|
+
const before = Date.now();
|
|
362
|
+
const client = await newClient(port, authHeaders());
|
|
363
|
+
activeClients.push(client);
|
|
364
|
+
await client.callTool({
|
|
365
|
+
name: 'skill_invoke',
|
|
366
|
+
arguments: { skill_id: 'official-debug', reason: 'anon test' },
|
|
367
|
+
});
|
|
368
|
+
|
|
369
|
+
// Fetch all rows and filter for mcp-anon-* with timestamp >= before.
|
|
370
|
+
// querySkillInvocations is keyed by session_id so we use a wide query.
|
|
371
|
+
const allRows = storage.querySkillInvocations({});
|
|
372
|
+
const anonRow = allRows.find(
|
|
373
|
+
(r) => r.session_id.startsWith('mcp-anon-') && r.timestamp >= before,
|
|
374
|
+
);
|
|
375
|
+
record(
|
|
376
|
+
'mcp-anon-* fallback (no header, no routingState)',
|
|
377
|
+
anonRow !== undefined && anonRow.success === 1,
|
|
378
|
+
`session_id=${anonRow?.session_id ?? '<none>'}`,
|
|
379
|
+
);
|
|
380
|
+
}
|
|
381
|
+
} catch (err) {
|
|
382
|
+
console.error('\n[FATAL] Unhandled error during tests:', err);
|
|
383
|
+
record('e2e-script', false, err instanceof Error ? err.message : String(err));
|
|
384
|
+
exitCode = 1;
|
|
385
|
+
} finally {
|
|
386
|
+
// Cleanup: clients first, then HTTP server, then storage, then tmp dir.
|
|
387
|
+
for (const c of activeClients) {
|
|
388
|
+
try {
|
|
389
|
+
await c.close();
|
|
390
|
+
} catch {
|
|
391
|
+
/* ignore */
|
|
392
|
+
}
|
|
393
|
+
}
|
|
394
|
+
const httpServer = (server as unknown as { server: import('node:net').Server | null }).server;
|
|
395
|
+
if (httpServer) {
|
|
396
|
+
(httpServer as unknown as { closeAllConnections?: () => void }).closeAllConnections?.();
|
|
397
|
+
await new Promise<void>((res) => httpServer.close(() => res()));
|
|
398
|
+
}
|
|
399
|
+
storage.close();
|
|
400
|
+
rmSync(tmp, { recursive: true, force: true });
|
|
401
|
+
restoreTestToken();
|
|
402
|
+
}
|
|
403
|
+
|
|
404
|
+
// ── Final report ─────────────────────────────────────────────────────────
|
|
405
|
+
const passed = results.filter((r) => r.pass).length;
|
|
406
|
+
const total = results.length;
|
|
407
|
+
console.log(`\n=== Result: ${passed}/${total} passed ===`);
|
|
408
|
+
if (passed !== total) {
|
|
409
|
+
console.log('\nFailures:');
|
|
410
|
+
for (const r of results.filter((x) => !x.pass)) {
|
|
411
|
+
console.log(` - ${r.step}: ${r.details}`);
|
|
412
|
+
}
|
|
413
|
+
}
|
|
414
|
+
|
|
415
|
+
process.exit(passed === total && exitCode === 0 ? 0 : 1);
|
|
416
|
+
}
|
|
417
|
+
|
|
418
|
+
main().catch((err) => {
|
|
419
|
+
console.error('E2E failed:', err);
|
|
420
|
+
restoreTestToken();
|
|
421
|
+
rmSync(tmp, { recursive: true, force: true });
|
|
422
|
+
process.exit(1);
|
|
423
|
+
});
|
|
@@ -0,0 +1,253 @@
|
|
|
1
|
+
#!/usr/bin/env tsx
|
|
2
|
+
/**
|
|
3
|
+
* E2E 验证:6 个方法论 Agent 的 prompt 质量
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
import { OFFICIAL_AGENTS, findOfficialAgent } from '../src/agents/index.js';
|
|
7
|
+
import { METHODOLOGY_PRESETS } from '../src/agents/methodologies/presets.js';
|
|
8
|
+
import { getPhase, getAllPhases, buildMethodologyAgent } from '../src/agents/methodologies/agent-builder.js';
|
|
9
|
+
import { OFFICIAL_SKILL_KEYWORDS } from '../src/skills/official-skills.js';
|
|
10
|
+
import { parseWorkflowMeta } from '../src/skills/tools/skill-invoke.js';
|
|
11
|
+
|
|
12
|
+
interface Issue {
|
|
13
|
+
agent: string;
|
|
14
|
+
severity: 'error' | 'warning' | 'info';
|
|
15
|
+
message: string;
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
const issues: Issue[] = [];
|
|
19
|
+
|
|
20
|
+
function record(agent: string, severity: Issue['severity'], message: string) {
|
|
21
|
+
issues.push({ agent, severity, message });
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
const PRESET_IDS = [
|
|
25
|
+
'bmad-feature-full',
|
|
26
|
+
'bmad-feature-quick',
|
|
27
|
+
'bmad-spec-only',
|
|
28
|
+
'harness-debug-full',
|
|
29
|
+
'harness-hotfix',
|
|
30
|
+
'harness-investigate',
|
|
31
|
+
];
|
|
32
|
+
|
|
33
|
+
const VALID_SKILLS = new Set(Object.keys(OFFICIAL_SKILL_KEYWORDS));
|
|
34
|
+
|
|
35
|
+
console.log('=== 方法论 Agent Prompt 质量 E2E 验证 ===\n');
|
|
36
|
+
|
|
37
|
+
// ── Check 1: 6 个 preset 都注册成功 ──────────────────────
|
|
38
|
+
console.log('-- Check 1: Preset 注册检查 --');
|
|
39
|
+
for (const id of PRESET_IDS) {
|
|
40
|
+
const agent = findOfficialAgent(id);
|
|
41
|
+
if (!agent) {
|
|
42
|
+
record(id, 'error', '未注册到 OFFICIAL_AGENTS');
|
|
43
|
+
} else {
|
|
44
|
+
console.log(`✓ ${id}: 已注册 (version=${agent.version}, skills=${agent.skills?.length ?? 0})`);
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
// ── Check 2: 每个 preset 的 phase 列表合法 ────────────────
|
|
49
|
+
console.log('\n-- Check 2: Phase 列表合法性 --');
|
|
50
|
+
for (const preset of METHODOLOGY_PRESETS) {
|
|
51
|
+
for (const phaseId of preset.phases) {
|
|
52
|
+
const phase = getPhase(phaseId);
|
|
53
|
+
if (!phase) {
|
|
54
|
+
record(preset.id, 'error', `phase ${phaseId} 不存在`);
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
console.log(`(已检查 ${METHODOLOGY_PRESETS.length} 个 preset 的 phase 列表)`);
|
|
59
|
+
|
|
60
|
+
// ── Check 3: prompt 内容质量 ─────────────────────────────
|
|
61
|
+
console.log('\n-- Check 3: prompt 内容质量 --');
|
|
62
|
+
for (const id of PRESET_IDS) {
|
|
63
|
+
const agent = findOfficialAgent(id);
|
|
64
|
+
if (!agent) continue;
|
|
65
|
+
const content = agent.content;
|
|
66
|
+
|
|
67
|
+
// 必须包含
|
|
68
|
+
if (!content.includes('软性建议') && !content.includes('软性')) {
|
|
69
|
+
record(id, 'warning', 'prompt 未明确"软性建议"语义');
|
|
70
|
+
}
|
|
71
|
+
if (content.length < 500) {
|
|
72
|
+
record(id, 'warning', `prompt 过短(${content.length} 字符),可能内容缺失`);
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
// 每个 phase 都该出现在 prompt 里
|
|
76
|
+
const preset = METHODOLOGY_PRESETS.find(p => p.id === id);
|
|
77
|
+
if (preset) {
|
|
78
|
+
for (const phaseId of preset.phases) {
|
|
79
|
+
const phase = getPhase(phaseId);
|
|
80
|
+
if (phase && !content.includes(`Phase: ${phase.id}`)) {
|
|
81
|
+
record(id, 'error', `prompt 缺少 "Phase: ${phase.id}" 标识`);
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
console.log(` ${id}: ${content.length} 字符`);
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
// ── Check 4: reason 格式约定一致 ─────────────────────────
|
|
89
|
+
// Prompt 里的 reason 用 <feature-slug> / <bug-slug> 当模板占位符。
|
|
90
|
+
// 运行时 LLM 会替换成实际 slug,再传给 skill_invoke。
|
|
91
|
+
// 因此校验时把占位符替换成示例 slug,再过 parseWorkflowMeta。
|
|
92
|
+
console.log('\n-- Check 4: reason 格式 --');
|
|
93
|
+
const TEMPLATE_PLACEHOLDERS = /<(feature-slug|bug-slug)>/g;
|
|
94
|
+
const PLACEHOLDER_SUBSTITUTE = 'example-slug';
|
|
95
|
+
|
|
96
|
+
for (const id of PRESET_IDS) {
|
|
97
|
+
const agent = findOfficialAgent(id);
|
|
98
|
+
if (!agent) continue;
|
|
99
|
+
|
|
100
|
+
// 提取所有 reason: "..." 出现
|
|
101
|
+
const reasonMatches = [...agent.content.matchAll(/reason:\s*"([^"]+)"/g)];
|
|
102
|
+
for (const match of reasonMatches) {
|
|
103
|
+
const reasonTemplate = match[1];
|
|
104
|
+
const reasonResolved = reasonTemplate.replace(TEMPLATE_PLACEHOLDERS, PLACEHOLDER_SUBSTITUTE);
|
|
105
|
+
|
|
106
|
+
// 子检查 4a:原始模板必须含 <feature-slug> 或 <bug-slug>(强制带 slug 元数据)
|
|
107
|
+
const hasSlugPlaceholder = reasonTemplate.includes('<feature-slug>') || reasonTemplate.includes('<bug-slug>');
|
|
108
|
+
if (!hasSlugPlaceholder) {
|
|
109
|
+
record(id, 'info', `reason "${reasonTemplate}" 缺少 <feature-slug>/<bug-slug> 占位符(仍可工作,但 slug 元数据会缺失)`);
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
// 子检查 4b:占位符替换后必须能被解析
|
|
113
|
+
const meta = parseWorkflowMeta(reasonResolved);
|
|
114
|
+
if (!meta.workflow) {
|
|
115
|
+
record(id, 'error', `reason "${reasonTemplate}" 替换占位符后 ("${reasonResolved}") 仍无法被 parseWorkflowMeta 解析`);
|
|
116
|
+
} else if (meta.workflow !== 'bmad' && meta.workflow !== 'harness') {
|
|
117
|
+
record(id, 'error', `reason "${reasonTemplate}" workflow=${meta.workflow}(应为 bmad/harness)`);
|
|
118
|
+
} else {
|
|
119
|
+
// 子检查 4c:解析出来的 workflow 应该跟 preset 主线一致
|
|
120
|
+
const preset = METHODOLOGY_PRESETS.find(p => p.id === id);
|
|
121
|
+
const presetWorkflow = id.startsWith('bmad-') ? 'bmad' : id.startsWith('harness-') ? 'harness' : null;
|
|
122
|
+
if (presetWorkflow && meta.workflow !== presetWorkflow) {
|
|
123
|
+
record(id, 'warning', `reason "${reasonTemplate}" workflow=${meta.workflow}(preset 主线=${presetWorkflow})`);
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
if (reasonMatches.length === 0) {
|
|
129
|
+
const preset = METHODOLOGY_PRESETS.find(p => p.id === id);
|
|
130
|
+
const needsSkill = preset?.phases.some(p => getPhase(p)?.skillId);
|
|
131
|
+
if (needsSkill) {
|
|
132
|
+
record(id, 'warning', '至少有 phase 需要调用 skill 但 prompt 里找不到 reason: "..." 字面量');
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
console.log(` ${id}: 发现 ${reasonMatches.length} 个 reason 引用`);
|
|
136
|
+
}
|
|
137
|
+
|
|
138
|
+
// ── Check 5: 所有引用的 skill 都真实存在 ─────────────────
|
|
139
|
+
console.log('\n-- Check 5: Skill 引用有效性 --');
|
|
140
|
+
let phaseSkillCount = 0;
|
|
141
|
+
for (const phase of getAllPhases()) {
|
|
142
|
+
if (phase.skillId) {
|
|
143
|
+
phaseSkillCount++;
|
|
144
|
+
if (!VALID_SKILLS.has(phase.skillId)) {
|
|
145
|
+
record(`phase:${phase.workflow}/${phase.id}`, 'error',
|
|
146
|
+
`skillId "${phase.skillId}" 不在 OFFICIAL_SKILL_KEYWORDS`);
|
|
147
|
+
}
|
|
148
|
+
}
|
|
149
|
+
}
|
|
150
|
+
console.log(` 共 ${getAllPhases().length} 个 phase,其中 ${phaseSkillCount} 个引用 skill`);
|
|
151
|
+
|
|
152
|
+
// ── Check 6: artifact 路径模板合理 ───────────────────────
|
|
153
|
+
console.log('\n-- Check 6: artifact 路径模板 --');
|
|
154
|
+
for (const phase of getAllPhases()) {
|
|
155
|
+
if (phase.artifactHint) {
|
|
156
|
+
// 检查路径是否在 docs/ 下(按用户决策)
|
|
157
|
+
if (!phase.artifactHint.startsWith('docs/') && !phase.artifactHint.startsWith('src/') && !phase.artifactHint.startsWith('tests/')) {
|
|
158
|
+
record(`phase:${phase.workflow}/${phase.id}`, 'warning',
|
|
159
|
+
`artifact 路径 "${phase.artifactHint}" 不在 docs/src/tests 下`);
|
|
160
|
+
}
|
|
161
|
+
// 检查是否含 <feature-slug> 或 <bug-slug> 占位符
|
|
162
|
+
if (!phase.artifactHint.includes('<')) {
|
|
163
|
+
record(`phase:${phase.workflow}/${phase.id}`, 'info',
|
|
164
|
+
`artifact 路径 "${phase.artifactHint}" 没有占位符(不可变路径?)`);
|
|
165
|
+
}
|
|
166
|
+
}
|
|
167
|
+
}
|
|
168
|
+
console.log(` (已检查所有 phase 的 artifactHint)`);
|
|
169
|
+
|
|
170
|
+
// ── Check 7: 跨方法论 Phase 没混淆 ───────────────────────
|
|
171
|
+
console.log('\n-- Check 7: 跨方法论组合 --');
|
|
172
|
+
for (const preset of METHODOLOGY_PRESETS) {
|
|
173
|
+
const workflows = new Set(preset.phases.map(p => getPhase(p)?.workflow));
|
|
174
|
+
// official preset 应该是单方法论的
|
|
175
|
+
if (workflows.size > 1) {
|
|
176
|
+
record(preset.id, 'info',
|
|
177
|
+
`跨方法论组合:${[...workflows].join(', ')}(official 预设建议单方法论)`);
|
|
178
|
+
}
|
|
179
|
+
}
|
|
180
|
+
console.log(` (已检查 ${METHODOLOGY_PRESETS.length} 个 preset 的 workflow 一致性)`);
|
|
181
|
+
|
|
182
|
+
// ── Check 8: Phase 的 skillId 跟 workflow 一致性 ──────────
|
|
183
|
+
console.log('\n-- Check 8: Skill 与 workflow 一致性 --');
|
|
184
|
+
for (const phase of getAllPhases()) {
|
|
185
|
+
if (phase.skillId) {
|
|
186
|
+
// 简单检查:bmad phase 的 skill 不该是纯 harness 类的
|
|
187
|
+
// (这是 heuristic,不是硬规则)
|
|
188
|
+
if (phase.workflow === 'bmad' && phase.skillId.includes('harness')) {
|
|
189
|
+
record(`phase:${phase.workflow}/${phase.id}`, 'warning',
|
|
190
|
+
`bmad phase 引用 harness skill "${phase.skillId}"`);
|
|
191
|
+
}
|
|
192
|
+
}
|
|
193
|
+
}
|
|
194
|
+
console.log(` (已检查所有 phase 的 skill 归属)`);
|
|
195
|
+
|
|
196
|
+
// ── Check 9: buildMethodologyAgent 是确定性的 ────────────
|
|
197
|
+
console.log('\n-- Check 9: buildMethodologyAgent 确定性 --');
|
|
198
|
+
const preset = METHODOLOGY_PRESETS[0];
|
|
199
|
+
const built1 = buildMethodologyAgent(preset);
|
|
200
|
+
const built2 = buildMethodologyAgent(preset);
|
|
201
|
+
if (built1.content !== built2.content) {
|
|
202
|
+
record(preset.id, 'error', 'buildMethodologyAgent 输出不确定');
|
|
203
|
+
} else {
|
|
204
|
+
console.log(` ✓ ${preset.id} 多次构建结果一致`);
|
|
205
|
+
}
|
|
206
|
+
|
|
207
|
+
// ── Check 10: harness 系列的 slug 占位符在整份 prompt 里保持一致 ──
|
|
208
|
+
// agent-builder 的"软性建议"行硬编码 `<feature-slug>`,但 harness phase body
|
|
209
|
+
// 全部使用 `<bug-slug>`。同一份 prompt 混用两种 slug 名会误导 LLM。
|
|
210
|
+
console.log('\n-- Check 10: slug 占位符在 prompt 内部一致 --');
|
|
211
|
+
for (const id of PRESET_IDS) {
|
|
212
|
+
const agent = findOfficialAgent(id);
|
|
213
|
+
if (!agent) continue;
|
|
214
|
+
const content = agent.content;
|
|
215
|
+
const hasFeatureSlug = content.includes('<feature-slug>');
|
|
216
|
+
const hasBugSlug = content.includes('<bug-slug>');
|
|
217
|
+
const isHarness = id.startsWith('harness-');
|
|
218
|
+
const isBmad = id.startsWith('bmad-');
|
|
219
|
+
|
|
220
|
+
if (isHarness && hasFeatureSlug) {
|
|
221
|
+
record(id, 'warning',
|
|
222
|
+
`harness preset 里出现 <feature-slug>(应为 <bug-slug>)— 可能来自 agent-builder 模板硬编码`);
|
|
223
|
+
}
|
|
224
|
+
if (isBmad && hasBugSlug) {
|
|
225
|
+
record(id, 'warning',
|
|
226
|
+
`bmad preset 里出现 <bug-slug>(应为 <feature-slug>)`);
|
|
227
|
+
}
|
|
228
|
+
if (hasFeatureSlug && hasBugSlug) {
|
|
229
|
+
record(id, 'warning',
|
|
230
|
+
`prompt 同时出现 <feature-slug> 和 <bug-slug>,会让 LLM 困惑`);
|
|
231
|
+
}
|
|
232
|
+
}
|
|
233
|
+
|
|
234
|
+
// ── 输出报告 ─────────────────────────────────────────────
|
|
235
|
+
const errors = issues.filter(i => i.severity === 'error');
|
|
236
|
+
const warnings = issues.filter(i => i.severity === 'warning');
|
|
237
|
+
const infos = issues.filter(i => i.severity === 'info');
|
|
238
|
+
|
|
239
|
+
console.log('\n=== 问题汇总 ===');
|
|
240
|
+
console.log(`Errors: ${errors.length}`);
|
|
241
|
+
console.log(`Warnings: ${warnings.length}`);
|
|
242
|
+
console.log(`Infos: ${infos.length}`);
|
|
243
|
+
|
|
244
|
+
if (issues.length > 0) {
|
|
245
|
+
console.log('\n=== 问题详情 ===');
|
|
246
|
+
for (const issue of issues) {
|
|
247
|
+
const icon = issue.severity === 'error' ? '✗' : issue.severity === 'warning' ? '⚠' : 'ℹ';
|
|
248
|
+
console.log(`${icon} [${issue.agent}] ${issue.message}`);
|
|
249
|
+
}
|
|
250
|
+
}
|
|
251
|
+
|
|
252
|
+
console.log(`\n=== Result: ${errors.length === 0 ? 'PASS' : 'FAIL'} ===`);
|
|
253
|
+
process.exit(errors.length === 0 ? 0 : 1);
|