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