claude-flow 3.0.0-alpha.5 → 3.0.0-alpha.63
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/README.md +1573 -41
- package/bin/cli.js +140 -26
- package/bin/mcp-server.js +188 -0
- package/dist/src/commands/agent.d.ts +8 -0
- package/dist/src/commands/agent.d.ts.map +1 -0
- package/dist/src/commands/agent.js +819 -0
- package/dist/src/commands/agent.js.map +1 -0
- package/dist/src/commands/analyze.d.ts +19 -0
- package/dist/src/commands/analyze.d.ts.map +1 -0
- package/dist/src/commands/analyze.js +1823 -0
- package/dist/src/commands/analyze.js.map +1 -0
- package/dist/src/commands/claims.d.ts +10 -0
- package/dist/src/commands/claims.d.ts.map +1 -0
- package/dist/src/commands/claims.js +288 -0
- package/dist/src/commands/claims.js.map +1 -0
- package/dist/src/commands/completions.d.ts +10 -0
- package/dist/src/commands/completions.d.ts.map +1 -0
- package/dist/src/commands/completions.js +539 -0
- package/dist/src/commands/completions.js.map +1 -0
- package/dist/src/commands/config.d.ts +8 -0
- package/dist/src/commands/config.d.ts.map +1 -0
- package/dist/src/commands/config.js +406 -0
- package/dist/src/commands/config.js.map +1 -0
- package/dist/src/commands/daemon.d.ts +8 -0
- package/dist/src/commands/daemon.d.ts.map +1 -0
- package/dist/src/commands/daemon.js +593 -0
- package/dist/src/commands/daemon.js.map +1 -0
- package/dist/src/commands/deployment.d.ts +10 -0
- package/dist/src/commands/deployment.d.ts.map +1 -0
- package/dist/src/commands/deployment.js +289 -0
- package/dist/src/commands/deployment.js.map +1 -0
- package/dist/src/commands/doctor.d.ts +10 -0
- package/dist/src/commands/doctor.d.ts.map +1 -0
- package/dist/src/commands/doctor.js +448 -0
- package/dist/src/commands/doctor.js.map +1 -0
- package/dist/src/commands/embeddings.d.ts +18 -0
- package/dist/src/commands/embeddings.d.ts.map +1 -0
- package/dist/src/commands/embeddings.js +831 -0
- package/dist/src/commands/embeddings.js.map +1 -0
- package/dist/src/commands/hive-mind.d.ts +8 -0
- package/dist/src/commands/hive-mind.d.ts.map +1 -0
- package/dist/src/commands/hive-mind.js +844 -0
- package/dist/src/commands/hive-mind.js.map +1 -0
- package/dist/src/commands/hooks.d.ts +8 -0
- package/dist/src/commands/hooks.d.ts.map +1 -0
- package/dist/src/commands/hooks.js +2892 -0
- package/dist/src/commands/hooks.js.map +1 -0
- package/dist/src/commands/index.d.ts +89 -0
- package/dist/src/commands/index.d.ts.map +1 -0
- package/dist/src/commands/index.js +258 -0
- package/dist/src/commands/index.js.map +1 -0
- package/dist/src/commands/init.d.ts +8 -0
- package/dist/src/commands/init.d.ts.map +1 -0
- package/dist/src/commands/init.js +603 -0
- package/dist/src/commands/init.js.map +1 -0
- package/dist/src/commands/issues.d.ts +21 -0
- package/dist/src/commands/issues.d.ts.map +1 -0
- package/dist/src/commands/issues.js +567 -0
- package/dist/src/commands/issues.js.map +1 -0
- package/dist/src/commands/mcp.d.ts +11 -0
- package/dist/src/commands/mcp.d.ts.map +1 -0
- package/dist/src/commands/mcp.js +662 -0
- package/dist/src/commands/mcp.js.map +1 -0
- package/dist/src/commands/memory.d.ts +8 -0
- package/dist/src/commands/memory.d.ts.map +1 -0
- package/dist/src/commands/memory.js +1200 -0
- package/dist/src/commands/memory.js.map +1 -0
- package/dist/src/commands/migrate.d.ts +8 -0
- package/dist/src/commands/migrate.d.ts.map +1 -0
- package/dist/src/commands/migrate.js +398 -0
- package/dist/src/commands/migrate.js.map +1 -0
- package/dist/src/commands/neural.d.ts +10 -0
- package/dist/src/commands/neural.d.ts.map +1 -0
- package/dist/src/commands/neural.js +224 -0
- package/dist/src/commands/neural.js.map +1 -0
- package/dist/src/commands/performance.d.ts +10 -0
- package/dist/src/commands/performance.d.ts.map +1 -0
- package/dist/src/commands/performance.js +262 -0
- package/dist/src/commands/performance.js.map +1 -0
- package/dist/src/commands/plugins.d.ts +11 -0
- package/dist/src/commands/plugins.d.ts.map +1 -0
- package/dist/src/commands/plugins.js +630 -0
- package/dist/src/commands/plugins.js.map +1 -0
- package/dist/src/commands/process.d.ts +10 -0
- package/dist/src/commands/process.d.ts.map +1 -0
- package/dist/src/commands/process.js +641 -0
- package/dist/src/commands/process.js.map +1 -0
- package/dist/src/commands/progress.d.ts +11 -0
- package/dist/src/commands/progress.d.ts.map +1 -0
- package/dist/src/commands/progress.js +259 -0
- package/dist/src/commands/progress.js.map +1 -0
- package/dist/src/commands/providers.d.ts +10 -0
- package/dist/src/commands/providers.d.ts.map +1 -0
- package/dist/src/commands/providers.js +232 -0
- package/dist/src/commands/providers.js.map +1 -0
- package/dist/src/commands/route.d.ts +16 -0
- package/dist/src/commands/route.d.ts.map +1 -0
- package/dist/src/commands/route.js +813 -0
- package/dist/src/commands/route.js.map +1 -0
- package/dist/src/commands/security.d.ts +10 -0
- package/dist/src/commands/security.d.ts.map +1 -0
- package/dist/src/commands/security.js +261 -0
- package/dist/src/commands/security.js.map +1 -0
- package/dist/src/commands/session.d.ts +8 -0
- package/dist/src/commands/session.d.ts.map +1 -0
- package/dist/src/commands/session.js +750 -0
- package/dist/src/commands/session.js.map +1 -0
- package/dist/src/commands/start.d.ts +8 -0
- package/dist/src/commands/start.d.ts.map +1 -0
- package/dist/src/commands/start.js +418 -0
- package/dist/src/commands/start.js.map +1 -0
- package/dist/src/commands/status.d.ts +8 -0
- package/dist/src/commands/status.d.ts.map +1 -0
- package/dist/src/commands/status.js +584 -0
- package/dist/src/commands/status.js.map +1 -0
- package/dist/src/commands/swarm.d.ts +8 -0
- package/dist/src/commands/swarm.d.ts.map +1 -0
- package/dist/src/commands/swarm.js +726 -0
- package/dist/src/commands/swarm.js.map +1 -0
- package/dist/src/commands/task.d.ts +8 -0
- package/dist/src/commands/task.d.ts.map +1 -0
- package/dist/src/commands/task.js +671 -0
- package/dist/src/commands/task.js.map +1 -0
- package/dist/src/commands/transfer-store.d.ts +13 -0
- package/dist/src/commands/transfer-store.d.ts.map +1 -0
- package/dist/src/commands/transfer-store.js +428 -0
- package/dist/src/commands/transfer-store.js.map +1 -0
- package/dist/src/commands/workflow.d.ts +8 -0
- package/dist/src/commands/workflow.d.ts.map +1 -0
- package/dist/src/commands/workflow.js +617 -0
- package/dist/src/commands/workflow.js.map +1 -0
- package/dist/src/config-adapter.d.ts +15 -0
- package/dist/src/config-adapter.d.ts.map +1 -0
- package/dist/src/config-adapter.js +185 -0
- package/dist/src/config-adapter.js.map +1 -0
- package/dist/src/index.d.ts +57 -0
- package/dist/src/index.d.ts.map +1 -0
- package/dist/src/index.js +388 -0
- package/dist/src/index.js.map +1 -0
- package/dist/src/infrastructure/in-memory-repositories.d.ts +68 -0
- package/dist/src/infrastructure/in-memory-repositories.d.ts.map +1 -0
- package/dist/src/infrastructure/in-memory-repositories.js +264 -0
- package/dist/src/infrastructure/in-memory-repositories.js.map +1 -0
- package/dist/src/init/claudemd-generator.d.ts +15 -0
- package/dist/src/init/claudemd-generator.d.ts.map +1 -0
- package/dist/src/init/claudemd-generator.js +674 -0
- package/dist/src/init/claudemd-generator.js.map +1 -0
- package/dist/src/init/executor.d.ts +11 -0
- package/dist/src/init/executor.d.ts.map +1 -0
- package/dist/src/init/executor.js +763 -0
- package/dist/src/init/executor.js.map +1 -0
- package/dist/src/init/helpers-generator.d.ts +42 -0
- package/dist/src/init/helpers-generator.d.ts.map +1 -0
- package/dist/src/init/helpers-generator.js +613 -0
- package/dist/src/init/helpers-generator.js.map +1 -0
- package/dist/src/init/index.d.ts +12 -0
- package/dist/src/init/index.d.ts.map +1 -0
- package/dist/src/init/index.js +15 -0
- package/dist/src/init/index.js.map +1 -0
- package/dist/src/init/mcp-generator.d.ts +27 -0
- package/dist/src/init/mcp-generator.d.ts.map +1 -0
- package/dist/src/init/mcp-generator.js +100 -0
- package/dist/src/init/mcp-generator.js.map +1 -0
- package/dist/src/init/settings-generator.d.ts +14 -0
- package/dist/src/init/settings-generator.d.ts.map +1 -0
- package/dist/src/init/settings-generator.js +311 -0
- package/dist/src/init/settings-generator.js.map +1 -0
- package/dist/src/init/statusline-generator.d.ts +20 -0
- package/dist/src/init/statusline-generator.d.ts.map +1 -0
- package/dist/src/init/statusline-generator.js +369 -0
- package/dist/src/init/statusline-generator.js.map +1 -0
- package/dist/src/init/types.d.ts +246 -0
- package/dist/src/init/types.d.ts.map +1 -0
- package/dist/src/init/types.js +216 -0
- package/dist/src/init/types.js.map +1 -0
- package/dist/src/mcp-client.d.ts +92 -0
- package/dist/src/mcp-client.d.ts.map +1 -0
- package/dist/src/mcp-client.js +207 -0
- package/dist/src/mcp-client.js.map +1 -0
- package/dist/src/mcp-server.d.ts +158 -0
- package/dist/src/mcp-server.d.ts.map +1 -0
- package/dist/src/mcp-server.js +584 -0
- package/dist/src/mcp-server.js.map +1 -0
- package/dist/src/mcp-tools/agent-tools.d.ts +8 -0
- package/dist/src/mcp-tools/agent-tools.d.ts.map +1 -0
- package/dist/src/mcp-tools/agent-tools.js +426 -0
- package/dist/src/mcp-tools/agent-tools.js.map +1 -0
- package/dist/src/mcp-tools/analyze-tools.d.ts +38 -0
- package/dist/src/mcp-tools/analyze-tools.d.ts.map +1 -0
- package/dist/src/mcp-tools/analyze-tools.js +317 -0
- package/dist/src/mcp-tools/analyze-tools.js.map +1 -0
- package/dist/src/mcp-tools/config-tools.d.ts +8 -0
- package/dist/src/mcp-tools/config-tools.d.ts.map +1 -0
- package/dist/src/mcp-tools/config-tools.js +333 -0
- package/dist/src/mcp-tools/config-tools.js.map +1 -0
- package/dist/src/mcp-tools/hive-mind-tools.d.ts +8 -0
- package/dist/src/mcp-tools/hive-mind-tools.d.ts.map +1 -0
- package/dist/src/mcp-tools/hive-mind-tools.js +447 -0
- package/dist/src/mcp-tools/hive-mind-tools.js.map +1 -0
- package/dist/src/mcp-tools/hooks-tools.d.ts +41 -0
- package/dist/src/mcp-tools/hooks-tools.d.ts.map +1 -0
- package/dist/src/mcp-tools/hooks-tools.js +1836 -0
- package/dist/src/mcp-tools/hooks-tools.js.map +1 -0
- package/dist/src/mcp-tools/index.d.ts +20 -0
- package/dist/src/mcp-tools/index.d.ts.map +1 -0
- package/dist/src/mcp-tools/index.js +19 -0
- package/dist/src/mcp-tools/index.js.map +1 -0
- package/dist/src/mcp-tools/memory-tools.d.ts +8 -0
- package/dist/src/mcp-tools/memory-tools.d.ts.map +1 -0
- package/dist/src/mcp-tools/memory-tools.js +235 -0
- package/dist/src/mcp-tools/memory-tools.js.map +1 -0
- package/dist/src/mcp-tools/progress-tools.d.ts +14 -0
- package/dist/src/mcp-tools/progress-tools.d.ts.map +1 -0
- package/dist/src/mcp-tools/progress-tools.js +343 -0
- package/dist/src/mcp-tools/progress-tools.js.map +1 -0
- package/dist/src/mcp-tools/session-tools.d.ts +8 -0
- package/dist/src/mcp-tools/session-tools.d.ts.map +1 -0
- package/dist/src/mcp-tools/session-tools.js +315 -0
- package/dist/src/mcp-tools/session-tools.js.map +1 -0
- package/dist/src/mcp-tools/swarm-tools.d.ts +8 -0
- package/dist/src/mcp-tools/swarm-tools.d.ts.map +1 -0
- package/dist/src/mcp-tools/swarm-tools.js +102 -0
- package/dist/src/mcp-tools/swarm-tools.js.map +1 -0
- package/dist/src/mcp-tools/task-tools.d.ts +8 -0
- package/dist/src/mcp-tools/task-tools.d.ts.map +1 -0
- package/dist/src/mcp-tools/task-tools.js +302 -0
- package/dist/src/mcp-tools/task-tools.js.map +1 -0
- package/dist/src/mcp-tools/transfer-tools.d.ts +14 -0
- package/dist/src/mcp-tools/transfer-tools.d.ts.map +1 -0
- package/dist/src/mcp-tools/transfer-tools.js +396 -0
- package/dist/src/mcp-tools/transfer-tools.js.map +1 -0
- package/dist/src/mcp-tools/types.d.ts +31 -0
- package/dist/src/mcp-tools/types.d.ts.map +1 -0
- package/dist/src/mcp-tools/types.js +7 -0
- package/dist/src/mcp-tools/types.js.map +1 -0
- package/dist/src/mcp-tools/workflow-tools.d.ts +8 -0
- package/dist/src/mcp-tools/workflow-tools.d.ts.map +1 -0
- package/dist/src/mcp-tools/workflow-tools.js +481 -0
- package/dist/src/mcp-tools/workflow-tools.js.map +1 -0
- package/dist/src/memory/memory-initializer.d.ts +229 -0
- package/dist/src/memory/memory-initializer.d.ts.map +1 -0
- package/dist/src/memory/memory-initializer.js +1248 -0
- package/dist/src/memory/memory-initializer.js.map +1 -0
- package/dist/src/output.d.ts +133 -0
- package/dist/src/output.d.ts.map +1 -0
- package/dist/src/output.js +513 -0
- package/dist/src/output.js.map +1 -0
- package/dist/src/parser.d.ts +41 -0
- package/dist/src/parser.d.ts.map +1 -0
- package/dist/src/parser.js +353 -0
- package/dist/src/parser.js.map +1 -0
- package/dist/src/plugins/store/discovery.d.ts +73 -0
- package/dist/src/plugins/store/discovery.d.ts.map +1 -0
- package/dist/src/plugins/store/discovery.js +568 -0
- package/dist/src/plugins/store/discovery.js.map +1 -0
- package/dist/src/plugins/store/index.d.ts +76 -0
- package/dist/src/plugins/store/index.d.ts.map +1 -0
- package/dist/src/plugins/store/index.js +141 -0
- package/dist/src/plugins/store/index.js.map +1 -0
- package/dist/src/plugins/store/search.d.ts +46 -0
- package/dist/src/plugins/store/search.d.ts.map +1 -0
- package/dist/src/plugins/store/search.js +230 -0
- package/dist/src/plugins/store/search.js.map +1 -0
- package/dist/src/plugins/store/types.d.ts +274 -0
- package/dist/src/plugins/store/types.d.ts.map +1 -0
- package/dist/src/plugins/store/types.js +7 -0
- package/dist/src/plugins/store/types.js.map +1 -0
- package/dist/src/plugins/tests/demo-plugin-store.d.ts +7 -0
- package/dist/src/plugins/tests/demo-plugin-store.d.ts.map +1 -0
- package/dist/src/plugins/tests/demo-plugin-store.js +126 -0
- package/dist/src/plugins/tests/demo-plugin-store.js.map +1 -0
- package/dist/src/plugins/tests/standalone-test.d.ts +12 -0
- package/dist/src/plugins/tests/standalone-test.d.ts.map +1 -0
- package/dist/src/plugins/tests/standalone-test.js +188 -0
- package/dist/src/plugins/tests/standalone-test.js.map +1 -0
- package/dist/src/plugins/tests/test-plugin-store.d.ts +7 -0
- package/dist/src/plugins/tests/test-plugin-store.d.ts.map +1 -0
- package/dist/src/plugins/tests/test-plugin-store.js +206 -0
- package/dist/src/plugins/tests/test-plugin-store.js.map +1 -0
- package/dist/src/prompt.d.ts +44 -0
- package/dist/src/prompt.d.ts.map +1 -0
- package/dist/src/prompt.js +501 -0
- package/dist/src/prompt.js.map +1 -0
- package/dist/src/ruvector/ast-analyzer.d.ts +67 -0
- package/dist/src/ruvector/ast-analyzer.d.ts.map +1 -0
- package/dist/src/ruvector/ast-analyzer.js +277 -0
- package/dist/src/ruvector/ast-analyzer.js.map +1 -0
- package/dist/src/ruvector/coverage-router.d.ts +160 -0
- package/dist/src/ruvector/coverage-router.d.ts.map +1 -0
- package/dist/src/ruvector/coverage-router.js +529 -0
- package/dist/src/ruvector/coverage-router.js.map +1 -0
- package/dist/src/ruvector/coverage-tools.d.ts +33 -0
- package/dist/src/ruvector/coverage-tools.d.ts.map +1 -0
- package/dist/src/ruvector/coverage-tools.js +157 -0
- package/dist/src/ruvector/coverage-tools.js.map +1 -0
- package/dist/src/ruvector/diff-classifier.d.ts +175 -0
- package/dist/src/ruvector/diff-classifier.d.ts.map +1 -0
- package/dist/src/ruvector/diff-classifier.js +698 -0
- package/dist/src/ruvector/diff-classifier.js.map +1 -0
- package/dist/src/ruvector/graph-analyzer.d.ts +187 -0
- package/dist/src/ruvector/graph-analyzer.d.ts.map +1 -0
- package/dist/src/ruvector/graph-analyzer.js +929 -0
- package/dist/src/ruvector/graph-analyzer.js.map +1 -0
- package/dist/src/ruvector/index.d.ts +27 -0
- package/dist/src/ruvector/index.d.ts.map +1 -0
- package/dist/src/ruvector/index.js +53 -0
- package/dist/src/ruvector/index.js.map +1 -0
- package/dist/src/ruvector/q-learning-router.d.ts +211 -0
- package/dist/src/ruvector/q-learning-router.d.ts.map +1 -0
- package/dist/src/ruvector/q-learning-router.js +681 -0
- package/dist/src/ruvector/q-learning-router.js.map +1 -0
- package/dist/src/ruvector/vector-db.d.ts +69 -0
- package/dist/src/ruvector/vector-db.d.ts.map +1 -0
- package/dist/src/ruvector/vector-db.js +243 -0
- package/dist/src/ruvector/vector-db.js.map +1 -0
- package/dist/src/services/claim-service.d.ts +204 -0
- package/dist/src/services/claim-service.d.ts.map +1 -0
- package/dist/src/services/claim-service.js +818 -0
- package/dist/src/services/claim-service.js.map +1 -0
- package/dist/src/services/container-worker-pool.d.ts +197 -0
- package/dist/src/services/container-worker-pool.d.ts.map +1 -0
- package/dist/src/services/container-worker-pool.js +581 -0
- package/dist/src/services/container-worker-pool.js.map +1 -0
- package/dist/src/services/headless-worker-executor.d.ts +304 -0
- package/dist/src/services/headless-worker-executor.d.ts.map +1 -0
- package/dist/src/services/headless-worker-executor.js +997 -0
- package/dist/src/services/headless-worker-executor.js.map +1 -0
- package/dist/src/services/index.d.ts +13 -0
- package/dist/src/services/index.d.ts.map +1 -0
- package/dist/src/services/index.js +11 -0
- package/dist/src/services/index.js.map +1 -0
- package/dist/src/services/worker-daemon.d.ts +203 -0
- package/dist/src/services/worker-daemon.d.ts.map +1 -0
- package/dist/src/services/worker-daemon.js +745 -0
- package/dist/src/services/worker-daemon.js.map +1 -0
- package/dist/src/services/worker-queue.d.ts +194 -0
- package/dist/src/services/worker-queue.d.ts.map +1 -0
- package/dist/src/services/worker-queue.js +511 -0
- package/dist/src/services/worker-queue.js.map +1 -0
- package/dist/src/suggest.d.ts +53 -0
- package/dist/src/suggest.d.ts.map +1 -0
- package/dist/src/suggest.js +200 -0
- package/dist/src/suggest.js.map +1 -0
- package/dist/src/transfer/anonymization/index.d.ts +25 -0
- package/dist/src/transfer/anonymization/index.d.ts.map +1 -0
- package/dist/src/transfer/anonymization/index.js +175 -0
- package/dist/src/transfer/anonymization/index.js.map +1 -0
- package/dist/src/transfer/deploy-seraphine.d.ts +13 -0
- package/dist/src/transfer/deploy-seraphine.d.ts.map +1 -0
- package/dist/src/transfer/deploy-seraphine.js +205 -0
- package/dist/src/transfer/deploy-seraphine.js.map +1 -0
- package/dist/src/transfer/export.d.ts +25 -0
- package/dist/src/transfer/export.d.ts.map +1 -0
- package/dist/src/transfer/export.js +113 -0
- package/dist/src/transfer/export.js.map +1 -0
- package/dist/src/transfer/index.d.ts +12 -0
- package/dist/src/transfer/index.d.ts.map +1 -0
- package/dist/src/transfer/index.js +31 -0
- package/dist/src/transfer/index.js.map +1 -0
- package/dist/src/transfer/ipfs/client.d.ts +31 -0
- package/dist/src/transfer/ipfs/client.d.ts.map +1 -0
- package/dist/src/transfer/ipfs/client.js +74 -0
- package/dist/src/transfer/ipfs/client.js.map +1 -0
- package/dist/src/transfer/ipfs/upload.d.ts +95 -0
- package/dist/src/transfer/ipfs/upload.d.ts.map +1 -0
- package/dist/src/transfer/ipfs/upload.js +410 -0
- package/dist/src/transfer/ipfs/upload.js.map +1 -0
- package/dist/src/transfer/models/seraphine.d.ts +72 -0
- package/dist/src/transfer/models/seraphine.d.ts.map +1 -0
- package/dist/src/transfer/models/seraphine.js +373 -0
- package/dist/src/transfer/models/seraphine.js.map +1 -0
- package/dist/src/transfer/serialization/cfp.d.ts +49 -0
- package/dist/src/transfer/serialization/cfp.d.ts.map +1 -0
- package/dist/src/transfer/serialization/cfp.js +180 -0
- package/dist/src/transfer/serialization/cfp.js.map +1 -0
- package/dist/src/transfer/store/discovery.d.ts +84 -0
- package/dist/src/transfer/store/discovery.d.ts.map +1 -0
- package/dist/src/transfer/store/discovery.js +275 -0
- package/dist/src/transfer/store/discovery.js.map +1 -0
- package/dist/src/transfer/store/download.d.ts +70 -0
- package/dist/src/transfer/store/download.d.ts.map +1 -0
- package/dist/src/transfer/store/download.js +295 -0
- package/dist/src/transfer/store/download.js.map +1 -0
- package/dist/src/transfer/store/index.d.ts +84 -0
- package/dist/src/transfer/store/index.d.ts.map +1 -0
- package/dist/src/transfer/store/index.js +153 -0
- package/dist/src/transfer/store/index.js.map +1 -0
- package/dist/src/transfer/store/publish.d.ts +76 -0
- package/dist/src/transfer/store/publish.d.ts.map +1 -0
- package/dist/src/transfer/store/publish.js +262 -0
- package/dist/src/transfer/store/publish.js.map +1 -0
- package/dist/src/transfer/store/registry.d.ts +58 -0
- package/dist/src/transfer/store/registry.d.ts.map +1 -0
- package/dist/src/transfer/store/registry.js +285 -0
- package/dist/src/transfer/store/registry.js.map +1 -0
- package/dist/src/transfer/store/search.d.ts +54 -0
- package/dist/src/transfer/store/search.d.ts.map +1 -0
- package/dist/src/transfer/store/search.js +232 -0
- package/dist/src/transfer/store/search.js.map +1 -0
- package/dist/src/transfer/store/tests/standalone-test.d.ts +12 -0
- package/dist/src/transfer/store/tests/standalone-test.d.ts.map +1 -0
- package/dist/src/transfer/store/tests/standalone-test.js +190 -0
- package/dist/src/transfer/store/tests/standalone-test.js.map +1 -0
- package/dist/src/transfer/store/types.d.ts +193 -0
- package/dist/src/transfer/store/types.d.ts.map +1 -0
- package/dist/src/transfer/store/types.js +6 -0
- package/dist/src/transfer/store/types.js.map +1 -0
- package/dist/src/transfer/test-seraphine.d.ts +6 -0
- package/dist/src/transfer/test-seraphine.d.ts.map +1 -0
- package/dist/src/transfer/test-seraphine.js +105 -0
- package/dist/src/transfer/test-seraphine.js.map +1 -0
- package/dist/src/transfer/tests/test-store.d.ts +7 -0
- package/dist/src/transfer/tests/test-store.d.ts.map +1 -0
- package/dist/src/transfer/tests/test-store.js +214 -0
- package/dist/src/transfer/tests/test-store.js.map +1 -0
- package/dist/src/transfer/types.d.ts +245 -0
- package/dist/src/transfer/types.d.ts.map +1 -0
- package/dist/src/transfer/types.js +6 -0
- package/dist/src/transfer/types.js.map +1 -0
- package/dist/src/types.d.ts +198 -0
- package/dist/src/types.d.ts.map +1 -0
- package/dist/src/types.js +38 -0
- package/dist/src/types.js.map +1 -0
- package/dist/tsconfig.tsbuildinfo +1 -0
- package/package.json +33 -47
- package/dist/index.d.ts +0 -30
- package/dist/index.d.ts.map +0 -1
- package/dist/index.js +0 -30
- package/dist/index.js.map +0 -1
|
@@ -0,0 +1,818 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* V3 Collaborative Issue Claims Service
|
|
3
|
+
*
|
|
4
|
+
* Implements ADR-016: Collaborative Issue Claims for Human-Agent Workflows
|
|
5
|
+
*
|
|
6
|
+
* Features:
|
|
7
|
+
* - Issue claiming/releasing for humans and agents
|
|
8
|
+
* - Handoff mechanisms between humans and agents
|
|
9
|
+
* - Work stealing for idle agents
|
|
10
|
+
* - Load balancing across swarm
|
|
11
|
+
* - GitHub integration
|
|
12
|
+
*
|
|
13
|
+
* @see /v3/implementation/adrs/ADR-016-collaborative-issue-claims.md
|
|
14
|
+
*/
|
|
15
|
+
import { EventEmitter } from 'events';
|
|
16
|
+
import * as fs from 'fs';
|
|
17
|
+
import * as path from 'path';
|
|
18
|
+
import { execFileSync } from 'child_process';
|
|
19
|
+
// ============================================================================
|
|
20
|
+
// Default Configuration
|
|
21
|
+
// ============================================================================
|
|
22
|
+
const DEFAULT_CONFIG = {
|
|
23
|
+
staleThresholdMinutes: 30,
|
|
24
|
+
blockedThresholdMinutes: 60,
|
|
25
|
+
overloadThreshold: 5,
|
|
26
|
+
gracePeriodMinutes: 10,
|
|
27
|
+
minProgressToProtect: 75,
|
|
28
|
+
contestWindowMinutes: 5,
|
|
29
|
+
requireSameType: false,
|
|
30
|
+
allowCrossTypeSteal: [
|
|
31
|
+
['coder', 'debugger'],
|
|
32
|
+
['tester', 'reviewer'],
|
|
33
|
+
],
|
|
34
|
+
};
|
|
35
|
+
// ============================================================================
|
|
36
|
+
// Claim Service Implementation
|
|
37
|
+
// ============================================================================
|
|
38
|
+
export class ClaimService extends EventEmitter {
|
|
39
|
+
claims = new Map();
|
|
40
|
+
stealableInfo = new Map();
|
|
41
|
+
storagePath;
|
|
42
|
+
config;
|
|
43
|
+
eventLog = [];
|
|
44
|
+
constructor(projectRoot, config) {
|
|
45
|
+
super();
|
|
46
|
+
this.storagePath = path.join(projectRoot, '.claude-flow', 'claims');
|
|
47
|
+
this.config = { ...DEFAULT_CONFIG, ...config };
|
|
48
|
+
}
|
|
49
|
+
// ==========================================================================
|
|
50
|
+
// Initialization
|
|
51
|
+
// ==========================================================================
|
|
52
|
+
async initialize() {
|
|
53
|
+
// Ensure storage directory exists
|
|
54
|
+
if (!fs.existsSync(this.storagePath)) {
|
|
55
|
+
fs.mkdirSync(this.storagePath, { recursive: true });
|
|
56
|
+
}
|
|
57
|
+
// Load existing claims
|
|
58
|
+
await this.loadClaims();
|
|
59
|
+
}
|
|
60
|
+
async loadClaims() {
|
|
61
|
+
const claimsFile = path.join(this.storagePath, 'claims.json');
|
|
62
|
+
if (fs.existsSync(claimsFile)) {
|
|
63
|
+
try {
|
|
64
|
+
const data = JSON.parse(fs.readFileSync(claimsFile, 'utf-8'));
|
|
65
|
+
for (const claim of data.claims || []) {
|
|
66
|
+
claim.claimedAt = new Date(claim.claimedAt);
|
|
67
|
+
claim.statusChangedAt = new Date(claim.statusChangedAt);
|
|
68
|
+
if (claim.expiresAt)
|
|
69
|
+
claim.expiresAt = new Date(claim.expiresAt);
|
|
70
|
+
this.claims.set(claim.issueId, claim);
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
catch {
|
|
74
|
+
// Start fresh if file is corrupted
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
async saveClaims() {
|
|
79
|
+
const claimsFile = path.join(this.storagePath, 'claims.json');
|
|
80
|
+
const data = {
|
|
81
|
+
claims: Array.from(this.claims.values()),
|
|
82
|
+
savedAt: new Date().toISOString(),
|
|
83
|
+
};
|
|
84
|
+
fs.writeFileSync(claimsFile, JSON.stringify(data, null, 2));
|
|
85
|
+
}
|
|
86
|
+
// ==========================================================================
|
|
87
|
+
// Core Claiming
|
|
88
|
+
// ==========================================================================
|
|
89
|
+
async claim(issueId, claimant) {
|
|
90
|
+
// Check if already claimed
|
|
91
|
+
const existing = this.claims.get(issueId);
|
|
92
|
+
if (existing && existing.status !== 'stealable') {
|
|
93
|
+
return {
|
|
94
|
+
success: false,
|
|
95
|
+
error: `Issue ${issueId} is already claimed by ${this.formatClaimant(existing.claimant)}`,
|
|
96
|
+
};
|
|
97
|
+
}
|
|
98
|
+
const now = new Date();
|
|
99
|
+
const claim = {
|
|
100
|
+
issueId,
|
|
101
|
+
claimant,
|
|
102
|
+
claimedAt: now,
|
|
103
|
+
status: 'active',
|
|
104
|
+
statusChangedAt: now,
|
|
105
|
+
progress: 0,
|
|
106
|
+
};
|
|
107
|
+
this.claims.set(issueId, claim);
|
|
108
|
+
await this.saveClaims();
|
|
109
|
+
this.emitEvent({
|
|
110
|
+
type: 'issue:claimed',
|
|
111
|
+
timestamp: now,
|
|
112
|
+
issueId,
|
|
113
|
+
claimant,
|
|
114
|
+
previousClaimant: existing?.claimant,
|
|
115
|
+
});
|
|
116
|
+
return { success: true, claim };
|
|
117
|
+
}
|
|
118
|
+
async release(issueId, claimant) {
|
|
119
|
+
const claim = this.claims.get(issueId);
|
|
120
|
+
if (!claim) {
|
|
121
|
+
throw new Error(`Issue ${issueId} is not claimed`);
|
|
122
|
+
}
|
|
123
|
+
if (!this.isSameClaimant(claim.claimant, claimant)) {
|
|
124
|
+
throw new Error(`Issue ${issueId} is not claimed by ${this.formatClaimant(claimant)}`);
|
|
125
|
+
}
|
|
126
|
+
this.claims.delete(issueId);
|
|
127
|
+
this.stealableInfo.delete(issueId);
|
|
128
|
+
await this.saveClaims();
|
|
129
|
+
this.emitEvent({
|
|
130
|
+
type: 'issue:released',
|
|
131
|
+
timestamp: new Date(),
|
|
132
|
+
issueId,
|
|
133
|
+
claimant,
|
|
134
|
+
});
|
|
135
|
+
}
|
|
136
|
+
// ==========================================================================
|
|
137
|
+
// Handoffs
|
|
138
|
+
// ==========================================================================
|
|
139
|
+
async requestHandoff(issueId, from, to, reason) {
|
|
140
|
+
const claim = this.claims.get(issueId);
|
|
141
|
+
if (!claim) {
|
|
142
|
+
throw new Error(`Issue ${issueId} is not claimed`);
|
|
143
|
+
}
|
|
144
|
+
if (!this.isSameClaimant(claim.claimant, from)) {
|
|
145
|
+
throw new Error(`Issue ${issueId} is not claimed by ${this.formatClaimant(from)}`);
|
|
146
|
+
}
|
|
147
|
+
claim.status = 'handoff-pending';
|
|
148
|
+
claim.statusChangedAt = new Date();
|
|
149
|
+
claim.handoffTo = to;
|
|
150
|
+
claim.handoffReason = reason;
|
|
151
|
+
await this.saveClaims();
|
|
152
|
+
this.emitEvent({
|
|
153
|
+
type: 'issue:handoff:requested',
|
|
154
|
+
timestamp: new Date(),
|
|
155
|
+
issueId,
|
|
156
|
+
claimant: from,
|
|
157
|
+
data: { to, reason },
|
|
158
|
+
});
|
|
159
|
+
}
|
|
160
|
+
async acceptHandoff(issueId, claimant) {
|
|
161
|
+
const claim = this.claims.get(issueId);
|
|
162
|
+
if (!claim || claim.status !== 'handoff-pending') {
|
|
163
|
+
throw new Error(`No pending handoff for issue ${issueId}`);
|
|
164
|
+
}
|
|
165
|
+
if (!claim.handoffTo || !this.isSameClaimant(claim.handoffTo, claimant)) {
|
|
166
|
+
throw new Error(`Handoff not addressed to ${this.formatClaimant(claimant)}`);
|
|
167
|
+
}
|
|
168
|
+
const previousClaimant = claim.claimant;
|
|
169
|
+
claim.claimant = claimant;
|
|
170
|
+
claim.status = 'active';
|
|
171
|
+
claim.statusChangedAt = new Date();
|
|
172
|
+
delete claim.handoffTo;
|
|
173
|
+
delete claim.handoffReason;
|
|
174
|
+
await this.saveClaims();
|
|
175
|
+
this.emitEvent({
|
|
176
|
+
type: 'issue:handoff:accepted',
|
|
177
|
+
timestamp: new Date(),
|
|
178
|
+
issueId,
|
|
179
|
+
claimant,
|
|
180
|
+
previousClaimant,
|
|
181
|
+
});
|
|
182
|
+
}
|
|
183
|
+
async rejectHandoff(issueId, claimant, reason) {
|
|
184
|
+
const claim = this.claims.get(issueId);
|
|
185
|
+
if (!claim || claim.status !== 'handoff-pending') {
|
|
186
|
+
throw new Error(`No pending handoff for issue ${issueId}`);
|
|
187
|
+
}
|
|
188
|
+
if (!claim.handoffTo || !this.isSameClaimant(claim.handoffTo, claimant)) {
|
|
189
|
+
throw new Error(`Handoff not addressed to ${this.formatClaimant(claimant)}`);
|
|
190
|
+
}
|
|
191
|
+
claim.status = 'active';
|
|
192
|
+
claim.statusChangedAt = new Date();
|
|
193
|
+
delete claim.handoffTo;
|
|
194
|
+
delete claim.handoffReason;
|
|
195
|
+
await this.saveClaims();
|
|
196
|
+
this.emitEvent({
|
|
197
|
+
type: 'issue:handoff:rejected',
|
|
198
|
+
timestamp: new Date(),
|
|
199
|
+
issueId,
|
|
200
|
+
claimant,
|
|
201
|
+
data: { reason },
|
|
202
|
+
});
|
|
203
|
+
}
|
|
204
|
+
// ==========================================================================
|
|
205
|
+
// Status Updates
|
|
206
|
+
// ==========================================================================
|
|
207
|
+
async updateStatus(issueId, status, note) {
|
|
208
|
+
const claim = this.claims.get(issueId);
|
|
209
|
+
if (!claim) {
|
|
210
|
+
throw new Error(`Issue ${issueId} is not claimed`);
|
|
211
|
+
}
|
|
212
|
+
const previousStatus = claim.status;
|
|
213
|
+
claim.status = status;
|
|
214
|
+
claim.statusChangedAt = new Date();
|
|
215
|
+
if (status === 'blocked' && note) {
|
|
216
|
+
claim.blockReason = note;
|
|
217
|
+
}
|
|
218
|
+
if (status === 'completed') {
|
|
219
|
+
claim.progress = 100;
|
|
220
|
+
}
|
|
221
|
+
await this.saveClaims();
|
|
222
|
+
this.emitEvent({
|
|
223
|
+
type: 'issue:status:changed',
|
|
224
|
+
timestamp: new Date(),
|
|
225
|
+
issueId,
|
|
226
|
+
data: { previousStatus, newStatus: status, note },
|
|
227
|
+
});
|
|
228
|
+
}
|
|
229
|
+
async updateProgress(issueId, progress) {
|
|
230
|
+
const claim = this.claims.get(issueId);
|
|
231
|
+
if (!claim) {
|
|
232
|
+
throw new Error(`Issue ${issueId} is not claimed`);
|
|
233
|
+
}
|
|
234
|
+
claim.progress = Math.min(100, Math.max(0, progress));
|
|
235
|
+
await this.saveClaims();
|
|
236
|
+
}
|
|
237
|
+
async requestReview(issueId, reviewers) {
|
|
238
|
+
const claim = this.claims.get(issueId);
|
|
239
|
+
if (!claim) {
|
|
240
|
+
throw new Error(`Issue ${issueId} is not claimed`);
|
|
241
|
+
}
|
|
242
|
+
claim.status = 'review-requested';
|
|
243
|
+
claim.statusChangedAt = new Date();
|
|
244
|
+
await this.saveClaims();
|
|
245
|
+
this.emitEvent({
|
|
246
|
+
type: 'issue:review:requested',
|
|
247
|
+
timestamp: new Date(),
|
|
248
|
+
issueId,
|
|
249
|
+
claimant: claim.claimant,
|
|
250
|
+
data: { reviewers },
|
|
251
|
+
});
|
|
252
|
+
}
|
|
253
|
+
// ==========================================================================
|
|
254
|
+
// Work Stealing
|
|
255
|
+
// ==========================================================================
|
|
256
|
+
async markStealable(issueId, info) {
|
|
257
|
+
const claim = this.claims.get(issueId);
|
|
258
|
+
if (!claim) {
|
|
259
|
+
throw new Error(`Issue ${issueId} is not claimed`);
|
|
260
|
+
}
|
|
261
|
+
claim.status = 'stealable';
|
|
262
|
+
claim.statusChangedAt = new Date();
|
|
263
|
+
claim.context = info.context;
|
|
264
|
+
claim.progress = info.progress;
|
|
265
|
+
this.stealableInfo.set(issueId, info);
|
|
266
|
+
await this.saveClaims();
|
|
267
|
+
this.emitEvent({
|
|
268
|
+
type: 'issue:stealable',
|
|
269
|
+
timestamp: new Date(),
|
|
270
|
+
issueId,
|
|
271
|
+
claimant: claim.claimant,
|
|
272
|
+
data: { info },
|
|
273
|
+
});
|
|
274
|
+
}
|
|
275
|
+
async steal(issueId, stealer) {
|
|
276
|
+
const claim = this.claims.get(issueId);
|
|
277
|
+
if (!claim) {
|
|
278
|
+
return { success: false, error: `Issue ${issueId} is not claimed` };
|
|
279
|
+
}
|
|
280
|
+
if (claim.status !== 'stealable') {
|
|
281
|
+
return { success: false, error: `Issue ${issueId} is not stealable` };
|
|
282
|
+
}
|
|
283
|
+
const info = this.stealableInfo.get(issueId);
|
|
284
|
+
const previousOwner = claim.claimant;
|
|
285
|
+
// Check if steal is allowed
|
|
286
|
+
if (this.config.requireSameType && stealer.type === 'agent' && previousOwner.type === 'agent') {
|
|
287
|
+
if (stealer.agentType !== previousOwner.agentType) {
|
|
288
|
+
const allowed = this.config.allowCrossTypeSteal.some(pair => pair.includes(stealer.agentType) && pair.includes(previousOwner.agentType));
|
|
289
|
+
if (!allowed) {
|
|
290
|
+
return { success: false, error: `Cross-type steal not allowed` };
|
|
291
|
+
}
|
|
292
|
+
}
|
|
293
|
+
}
|
|
294
|
+
// Execute steal
|
|
295
|
+
claim.claimant = stealer;
|
|
296
|
+
claim.status = 'active';
|
|
297
|
+
claim.statusChangedAt = new Date();
|
|
298
|
+
claim.claimedAt = new Date();
|
|
299
|
+
this.stealableInfo.delete(issueId);
|
|
300
|
+
await this.saveClaims();
|
|
301
|
+
this.emitEvent({
|
|
302
|
+
type: 'issue:stolen',
|
|
303
|
+
timestamp: new Date(),
|
|
304
|
+
issueId,
|
|
305
|
+
claimant: stealer,
|
|
306
|
+
previousClaimant: previousOwner,
|
|
307
|
+
data: { context: info },
|
|
308
|
+
});
|
|
309
|
+
return { success: true, claim, previousOwner, context: info };
|
|
310
|
+
}
|
|
311
|
+
async getStealable(agentType) {
|
|
312
|
+
const stealable = [];
|
|
313
|
+
for (const claim of this.claims.values()) {
|
|
314
|
+
if (claim.status !== 'stealable')
|
|
315
|
+
continue;
|
|
316
|
+
const info = this.stealableInfo.get(claim.issueId);
|
|
317
|
+
if (agentType && info?.preferredTypes?.length) {
|
|
318
|
+
if (!info.preferredTypes.includes(agentType))
|
|
319
|
+
continue;
|
|
320
|
+
}
|
|
321
|
+
stealable.push(claim);
|
|
322
|
+
}
|
|
323
|
+
return stealable;
|
|
324
|
+
}
|
|
325
|
+
async contestSteal(issueId, originalClaimant, reason) {
|
|
326
|
+
const claim = this.claims.get(issueId);
|
|
327
|
+
if (!claim) {
|
|
328
|
+
throw new Error(`Issue ${issueId} is not claimed`);
|
|
329
|
+
}
|
|
330
|
+
this.emitEvent({
|
|
331
|
+
type: 'issue:steal:contested',
|
|
332
|
+
timestamp: new Date(),
|
|
333
|
+
issueId,
|
|
334
|
+
claimant: originalClaimant,
|
|
335
|
+
data: { reason, currentOwner: claim.claimant },
|
|
336
|
+
});
|
|
337
|
+
// Contest resolution would typically be handled by a coordinator or human
|
|
338
|
+
}
|
|
339
|
+
// ==========================================================================
|
|
340
|
+
// Load Balancing
|
|
341
|
+
// ==========================================================================
|
|
342
|
+
async getAgentLoad(agentId) {
|
|
343
|
+
const claims = [];
|
|
344
|
+
let blockedCount = 0;
|
|
345
|
+
for (const claim of this.claims.values()) {
|
|
346
|
+
if (claim.claimant.type === 'agent' && claim.claimant.agentId === agentId) {
|
|
347
|
+
claims.push(claim);
|
|
348
|
+
if (claim.status === 'blocked')
|
|
349
|
+
blockedCount++;
|
|
350
|
+
}
|
|
351
|
+
}
|
|
352
|
+
const agentType = claims[0]?.claimant.type === 'agent' ? claims[0].claimant.agentType : 'unknown';
|
|
353
|
+
return {
|
|
354
|
+
agentId,
|
|
355
|
+
agentType,
|
|
356
|
+
claimCount: claims.length,
|
|
357
|
+
maxClaims: this.config.overloadThreshold,
|
|
358
|
+
utilization: claims.length / this.config.overloadThreshold,
|
|
359
|
+
claims,
|
|
360
|
+
avgCompletionTime: 0, // Would need historical data
|
|
361
|
+
currentBlockedCount: blockedCount,
|
|
362
|
+
};
|
|
363
|
+
}
|
|
364
|
+
async rebalance(swarmId) {
|
|
365
|
+
const result = { moved: [], suggested: [] };
|
|
366
|
+
// Get all agent loads
|
|
367
|
+
const agentLoads = new Map();
|
|
368
|
+
const agentTypes = new Set();
|
|
369
|
+
for (const claim of this.claims.values()) {
|
|
370
|
+
if (claim.claimant.type !== 'agent')
|
|
371
|
+
continue;
|
|
372
|
+
const agentId = claim.claimant.agentId;
|
|
373
|
+
if (!agentLoads.has(agentId)) {
|
|
374
|
+
const load = await this.getAgentLoad(agentId);
|
|
375
|
+
agentLoads.set(agentId, load);
|
|
376
|
+
agentTypes.add(load.agentType);
|
|
377
|
+
}
|
|
378
|
+
}
|
|
379
|
+
// For each agent type, calculate average load
|
|
380
|
+
for (const agentType of agentTypes) {
|
|
381
|
+
const typeLoads = Array.from(agentLoads.values()).filter(l => l.agentType === agentType);
|
|
382
|
+
const avgLoad = typeLoads.reduce((sum, l) => sum + l.utilization, 0) / typeLoads.length;
|
|
383
|
+
const overloaded = typeLoads.filter(l => l.utilization > avgLoad * 1.5);
|
|
384
|
+
const underloaded = typeLoads.filter(l => l.utilization < avgLoad * 0.5);
|
|
385
|
+
// Generate suggestions
|
|
386
|
+
for (const over of overloaded) {
|
|
387
|
+
const lowProgressClaims = over.claims
|
|
388
|
+
.filter(c => c.progress < 25)
|
|
389
|
+
.sort((a, b) => a.progress - b.progress);
|
|
390
|
+
for (const claim of lowProgressClaims) {
|
|
391
|
+
const target = underloaded.find(u => u.claimCount < u.maxClaims);
|
|
392
|
+
if (target) {
|
|
393
|
+
result.suggested.push({
|
|
394
|
+
issueId: claim.issueId,
|
|
395
|
+
currentOwner: claim.claimant,
|
|
396
|
+
suggestedOwner: {
|
|
397
|
+
type: 'agent',
|
|
398
|
+
agentId: target.agentId,
|
|
399
|
+
agentType: target.agentType,
|
|
400
|
+
},
|
|
401
|
+
reason: 'Load balancing: redistributing work across swarm',
|
|
402
|
+
});
|
|
403
|
+
}
|
|
404
|
+
}
|
|
405
|
+
}
|
|
406
|
+
}
|
|
407
|
+
return result;
|
|
408
|
+
}
|
|
409
|
+
// ==========================================================================
|
|
410
|
+
// Queries
|
|
411
|
+
// ==========================================================================
|
|
412
|
+
async getClaimedBy(claimant) {
|
|
413
|
+
return Array.from(this.claims.values()).filter(c => this.isSameClaimant(c.claimant, claimant));
|
|
414
|
+
}
|
|
415
|
+
async getAvailableIssues(_filters) {
|
|
416
|
+
// This would integrate with GitHub API
|
|
417
|
+
// For now, return issues that are not claimed
|
|
418
|
+
return [];
|
|
419
|
+
}
|
|
420
|
+
async getIssueStatus(issueId) {
|
|
421
|
+
return this.claims.get(issueId) || null;
|
|
422
|
+
}
|
|
423
|
+
async getAllClaims() {
|
|
424
|
+
return Array.from(this.claims.values());
|
|
425
|
+
}
|
|
426
|
+
async getByStatus(status) {
|
|
427
|
+
return Array.from(this.claims.values()).filter(c => c.status === status);
|
|
428
|
+
}
|
|
429
|
+
// ==========================================================================
|
|
430
|
+
// Auto-Management
|
|
431
|
+
// ==========================================================================
|
|
432
|
+
async expireStale(maxAgeMinutes) {
|
|
433
|
+
const threshold = maxAgeMinutes ?? this.config.staleThresholdMinutes;
|
|
434
|
+
const now = Date.now();
|
|
435
|
+
const expired = [];
|
|
436
|
+
for (const claim of this.claims.values()) {
|
|
437
|
+
if (claim.status === 'stealable' || claim.status === 'completed')
|
|
438
|
+
continue;
|
|
439
|
+
const age = (now - claim.statusChangedAt.getTime()) / 60000;
|
|
440
|
+
if (age > threshold) {
|
|
441
|
+
// Mark as stealable
|
|
442
|
+
await this.markStealable(claim.issueId, {
|
|
443
|
+
reason: 'stale',
|
|
444
|
+
stealableAt: new Date(),
|
|
445
|
+
progress: claim.progress,
|
|
446
|
+
context: `Stale: No activity for ${Math.round(age)} minutes`,
|
|
447
|
+
});
|
|
448
|
+
expired.push(claim);
|
|
449
|
+
}
|
|
450
|
+
}
|
|
451
|
+
return expired;
|
|
452
|
+
}
|
|
453
|
+
// ==========================================================================
|
|
454
|
+
// Helpers
|
|
455
|
+
// ==========================================================================
|
|
456
|
+
formatClaimant(claimant) {
|
|
457
|
+
return claimant.type === 'human'
|
|
458
|
+
? `human:${claimant.name}`
|
|
459
|
+
: `agent:${claimant.agentType}:${claimant.agentId}`;
|
|
460
|
+
}
|
|
461
|
+
isSameClaimant(a, b) {
|
|
462
|
+
if (a.type !== b.type)
|
|
463
|
+
return false;
|
|
464
|
+
if (a.type === 'human' && b.type === 'human') {
|
|
465
|
+
return a.userId === b.userId;
|
|
466
|
+
}
|
|
467
|
+
if (a.type === 'agent' && b.type === 'agent') {
|
|
468
|
+
return a.agentId === b.agentId;
|
|
469
|
+
}
|
|
470
|
+
return false;
|
|
471
|
+
}
|
|
472
|
+
emitEvent(event) {
|
|
473
|
+
this.eventLog.push(event);
|
|
474
|
+
if (this.eventLog.length > 1000) {
|
|
475
|
+
this.eventLog = this.eventLog.slice(-500);
|
|
476
|
+
}
|
|
477
|
+
this.emit(event.type, event);
|
|
478
|
+
}
|
|
479
|
+
getEventLog(limit = 100) {
|
|
480
|
+
return this.eventLog.slice(-limit);
|
|
481
|
+
}
|
|
482
|
+
}
|
|
483
|
+
// ============================================================================
|
|
484
|
+
// GitHub Sync Implementation
|
|
485
|
+
// ============================================================================
|
|
486
|
+
const DEFAULT_GITHUB_CONFIG = {
|
|
487
|
+
enabled: false,
|
|
488
|
+
syncLabels: true,
|
|
489
|
+
claimLabel: 'claimed',
|
|
490
|
+
autoAssign: true,
|
|
491
|
+
commentOnClaim: true,
|
|
492
|
+
commentOnRelease: true,
|
|
493
|
+
};
|
|
494
|
+
// ============================================================================
|
|
495
|
+
// Input Validation (Security)
|
|
496
|
+
// ============================================================================
|
|
497
|
+
/**
|
|
498
|
+
* Validate GitHub repository format (owner/repo)
|
|
499
|
+
* Prevents command injection via malicious repo names
|
|
500
|
+
*/
|
|
501
|
+
function isValidRepo(repo) {
|
|
502
|
+
// owner/repo format: alphanumeric, hyphens, underscores, dots
|
|
503
|
+
return /^[\w.-]+\/[\w.-]+$/.test(repo) && repo.length <= 100;
|
|
504
|
+
}
|
|
505
|
+
/**
|
|
506
|
+
* Validate issue number (positive integer)
|
|
507
|
+
*/
|
|
508
|
+
function isValidIssueNumber(num) {
|
|
509
|
+
return Number.isInteger(num) && num > 0 && num < 1000000000;
|
|
510
|
+
}
|
|
511
|
+
/**
|
|
512
|
+
* Validate claimant name (GitHub username format)
|
|
513
|
+
* Prevents command injection via malicious usernames
|
|
514
|
+
*/
|
|
515
|
+
function isValidClaimantName(name) {
|
|
516
|
+
// GitHub usernames: alphanumeric, hyphens, max 39 chars
|
|
517
|
+
return /^[\w-]+$/.test(name) && name.length >= 1 && name.length <= 39;
|
|
518
|
+
}
|
|
519
|
+
/**
|
|
520
|
+
* Validate label name
|
|
521
|
+
* Prevents command injection via malicious label names
|
|
522
|
+
*/
|
|
523
|
+
function isValidLabel(label) {
|
|
524
|
+
// Labels: alphanumeric, hyphens, underscores, spaces, max 50 chars
|
|
525
|
+
return /^[\w\s-]+$/.test(label) && label.length >= 1 && label.length <= 50;
|
|
526
|
+
}
|
|
527
|
+
/**
|
|
528
|
+
* Sanitize error messages to prevent information disclosure
|
|
529
|
+
*/
|
|
530
|
+
function sanitizeError(error) {
|
|
531
|
+
const msg = error.message || 'Unknown error';
|
|
532
|
+
// Remove paths and sensitive details
|
|
533
|
+
return msg.replace(/\/[\w./-]+/g, '[path]').substring(0, 200);
|
|
534
|
+
}
|
|
535
|
+
export class GitHubSync {
|
|
536
|
+
config;
|
|
537
|
+
claimService;
|
|
538
|
+
constructor(claimService, config) {
|
|
539
|
+
this.claimService = claimService;
|
|
540
|
+
this.config = { ...DEFAULT_GITHUB_CONFIG, ...config };
|
|
541
|
+
}
|
|
542
|
+
/**
|
|
543
|
+
* Check if GitHub CLI is available
|
|
544
|
+
*/
|
|
545
|
+
isGhAvailable() {
|
|
546
|
+
try {
|
|
547
|
+
execFileSync('gh', ['--version'], { stdio: 'ignore' });
|
|
548
|
+
return true;
|
|
549
|
+
}
|
|
550
|
+
catch {
|
|
551
|
+
return false;
|
|
552
|
+
}
|
|
553
|
+
}
|
|
554
|
+
/**
|
|
555
|
+
* Get the current repository from git remote
|
|
556
|
+
*/
|
|
557
|
+
getRepo() {
|
|
558
|
+
if (this.config.repo) {
|
|
559
|
+
return isValidRepo(this.config.repo) ? this.config.repo : null;
|
|
560
|
+
}
|
|
561
|
+
try {
|
|
562
|
+
const remote = execFileSync('git', ['remote', 'get-url', 'origin'], { encoding: 'utf-8' }).trim();
|
|
563
|
+
const match = remote.match(/github\.com[/:]([\w.-]+\/[\w.-]+)/);
|
|
564
|
+
const repo = match ? match[1].replace('.git', '') : null;
|
|
565
|
+
return repo && isValidRepo(repo) ? repo : null;
|
|
566
|
+
}
|
|
567
|
+
catch {
|
|
568
|
+
return null;
|
|
569
|
+
}
|
|
570
|
+
}
|
|
571
|
+
/**
|
|
572
|
+
* Sync issues from GitHub
|
|
573
|
+
*/
|
|
574
|
+
async syncIssues(state = 'open') {
|
|
575
|
+
const errors = [];
|
|
576
|
+
const issues = [];
|
|
577
|
+
if (!this.isGhAvailable()) {
|
|
578
|
+
return { success: false, synced: 0, errors: ['GitHub CLI (gh) not installed'] };
|
|
579
|
+
}
|
|
580
|
+
const repo = this.getRepo();
|
|
581
|
+
if (!repo) {
|
|
582
|
+
return { success: false, synced: 0, errors: ['Could not determine GitHub repository'] };
|
|
583
|
+
}
|
|
584
|
+
// Validate state parameter (whitelist)
|
|
585
|
+
const validStates = ['open', 'closed', 'all'];
|
|
586
|
+
if (!validStates.includes(state)) {
|
|
587
|
+
return { success: false, synced: 0, errors: ['Invalid state parameter'] };
|
|
588
|
+
}
|
|
589
|
+
try {
|
|
590
|
+
const issuesJson = execFileSync('gh', [
|
|
591
|
+
'issue', 'list',
|
|
592
|
+
'--repo', repo,
|
|
593
|
+
'--state', state,
|
|
594
|
+
'--json', 'number,title,body,state,labels,assignees,url,createdAt,updatedAt',
|
|
595
|
+
'--limit', '100'
|
|
596
|
+
], { encoding: 'utf-8' });
|
|
597
|
+
const rawIssues = JSON.parse(issuesJson);
|
|
598
|
+
for (const raw of rawIssues) {
|
|
599
|
+
issues.push({
|
|
600
|
+
number: raw.number,
|
|
601
|
+
title: raw.title,
|
|
602
|
+
body: raw.body || '',
|
|
603
|
+
state: raw.state === 'OPEN' ? 'open' : 'closed',
|
|
604
|
+
labels: raw.labels?.map((l) => l.name) || [],
|
|
605
|
+
assignees: raw.assignees?.map((a) => a.login) || [],
|
|
606
|
+
url: raw.url,
|
|
607
|
+
createdAt: new Date(raw.createdAt),
|
|
608
|
+
updatedAt: new Date(raw.updatedAt),
|
|
609
|
+
});
|
|
610
|
+
}
|
|
611
|
+
return { success: true, synced: issues.length, errors, issues };
|
|
612
|
+
}
|
|
613
|
+
catch (error) {
|
|
614
|
+
errors.push(`Failed to fetch issues: ${sanitizeError(error)}`);
|
|
615
|
+
return { success: false, synced: 0, errors };
|
|
616
|
+
}
|
|
617
|
+
}
|
|
618
|
+
/**
|
|
619
|
+
* Sync a local claim to GitHub (add label/assignee/comment)
|
|
620
|
+
*/
|
|
621
|
+
async claimOnGitHub(issueNumber, claimant) {
|
|
622
|
+
const errors = [];
|
|
623
|
+
if (!this.config.enabled) {
|
|
624
|
+
return { success: true, synced: 0, errors: ['GitHub sync not enabled'] };
|
|
625
|
+
}
|
|
626
|
+
if (!this.isGhAvailable()) {
|
|
627
|
+
return { success: false, synced: 0, errors: ['GitHub CLI (gh) not installed'] };
|
|
628
|
+
}
|
|
629
|
+
// Validate issue number
|
|
630
|
+
if (!isValidIssueNumber(issueNumber)) {
|
|
631
|
+
return { success: false, synced: 0, errors: ['Invalid issue number'] };
|
|
632
|
+
}
|
|
633
|
+
const repo = this.getRepo();
|
|
634
|
+
if (!repo) {
|
|
635
|
+
return { success: false, synced: 0, errors: ['Could not determine repository'] };
|
|
636
|
+
}
|
|
637
|
+
// Validate claim label
|
|
638
|
+
if (!isValidLabel(this.config.claimLabel)) {
|
|
639
|
+
return { success: false, synced: 0, errors: ['Invalid claim label configuration'] };
|
|
640
|
+
}
|
|
641
|
+
try {
|
|
642
|
+
// Add claim label
|
|
643
|
+
if (this.config.syncLabels) {
|
|
644
|
+
try {
|
|
645
|
+
execFileSync('gh', [
|
|
646
|
+
'issue', 'edit', String(issueNumber),
|
|
647
|
+
'--repo', repo,
|
|
648
|
+
'--add-label', this.config.claimLabel
|
|
649
|
+
], { stdio: 'ignore' });
|
|
650
|
+
}
|
|
651
|
+
catch {
|
|
652
|
+
errors.push('Failed to add claim label (label may not exist)');
|
|
653
|
+
}
|
|
654
|
+
}
|
|
655
|
+
// Auto-assign if human claimant
|
|
656
|
+
if (this.config.autoAssign && claimant.type === 'human') {
|
|
657
|
+
if (!isValidClaimantName(claimant.name)) {
|
|
658
|
+
errors.push('Invalid claimant name format');
|
|
659
|
+
}
|
|
660
|
+
else {
|
|
661
|
+
try {
|
|
662
|
+
execFileSync('gh', [
|
|
663
|
+
'issue', 'edit', String(issueNumber),
|
|
664
|
+
'--repo', repo,
|
|
665
|
+
'--add-assignee', claimant.name
|
|
666
|
+
], { stdio: 'ignore' });
|
|
667
|
+
}
|
|
668
|
+
catch {
|
|
669
|
+
errors.push('Failed to assign issue');
|
|
670
|
+
}
|
|
671
|
+
}
|
|
672
|
+
}
|
|
673
|
+
// Add comment
|
|
674
|
+
if (this.config.commentOnClaim) {
|
|
675
|
+
const claimantStr = claimant.type === 'human'
|
|
676
|
+
? `@${claimant.name.replace(/[^a-zA-Z0-9_-]/g, '')}`
|
|
677
|
+
: `Agent: ${(claimant.agentType || 'unknown').replace(/[^a-zA-Z0-9_-]/g, '')}`;
|
|
678
|
+
const comment = `🤖 **Issue claimed** by ${claimantStr}\n\n_Coordinated by Claude Flow V3_`;
|
|
679
|
+
try {
|
|
680
|
+
execFileSync('gh', [
|
|
681
|
+
'issue', 'comment', String(issueNumber),
|
|
682
|
+
'--repo', repo,
|
|
683
|
+
'--body', comment
|
|
684
|
+
], { stdio: 'ignore' });
|
|
685
|
+
}
|
|
686
|
+
catch {
|
|
687
|
+
errors.push('Failed to add comment');
|
|
688
|
+
}
|
|
689
|
+
}
|
|
690
|
+
return { success: errors.length === 0, synced: 1, errors };
|
|
691
|
+
}
|
|
692
|
+
catch (error) {
|
|
693
|
+
errors.push(`GitHub sync failed: ${sanitizeError(error)}`);
|
|
694
|
+
return { success: false, synced: 0, errors };
|
|
695
|
+
}
|
|
696
|
+
}
|
|
697
|
+
/**
|
|
698
|
+
* Release claim on GitHub (remove label/assignee/comment)
|
|
699
|
+
*/
|
|
700
|
+
async releaseOnGitHub(issueNumber, claimant) {
|
|
701
|
+
const errors = [];
|
|
702
|
+
if (!this.config.enabled) {
|
|
703
|
+
return { success: true, synced: 0, errors: ['GitHub sync not enabled'] };
|
|
704
|
+
}
|
|
705
|
+
if (!this.isGhAvailable()) {
|
|
706
|
+
return { success: false, synced: 0, errors: ['GitHub CLI (gh) not installed'] };
|
|
707
|
+
}
|
|
708
|
+
// Validate issue number
|
|
709
|
+
if (!isValidIssueNumber(issueNumber)) {
|
|
710
|
+
return { success: false, synced: 0, errors: ['Invalid issue number'] };
|
|
711
|
+
}
|
|
712
|
+
const repo = this.getRepo();
|
|
713
|
+
if (!repo) {
|
|
714
|
+
return { success: false, synced: 0, errors: ['Could not determine repository'] };
|
|
715
|
+
}
|
|
716
|
+
// Validate claim label
|
|
717
|
+
if (!isValidLabel(this.config.claimLabel)) {
|
|
718
|
+
return { success: false, synced: 0, errors: ['Invalid claim label configuration'] };
|
|
719
|
+
}
|
|
720
|
+
try {
|
|
721
|
+
// Remove claim label
|
|
722
|
+
if (this.config.syncLabels) {
|
|
723
|
+
try {
|
|
724
|
+
execFileSync('gh', [
|
|
725
|
+
'issue', 'edit', String(issueNumber),
|
|
726
|
+
'--repo', repo,
|
|
727
|
+
'--remove-label', this.config.claimLabel
|
|
728
|
+
], { stdio: 'ignore' });
|
|
729
|
+
}
|
|
730
|
+
catch {
|
|
731
|
+
// Label might not exist
|
|
732
|
+
}
|
|
733
|
+
}
|
|
734
|
+
// Remove assignee if human claimant
|
|
735
|
+
if (this.config.autoAssign && claimant.type === 'human') {
|
|
736
|
+
if (isValidClaimantName(claimant.name)) {
|
|
737
|
+
try {
|
|
738
|
+
execFileSync('gh', [
|
|
739
|
+
'issue', 'edit', String(issueNumber),
|
|
740
|
+
'--repo', repo,
|
|
741
|
+
'--remove-assignee', claimant.name
|
|
742
|
+
], { stdio: 'ignore' });
|
|
743
|
+
}
|
|
744
|
+
catch {
|
|
745
|
+
errors.push('Failed to remove assignee');
|
|
746
|
+
}
|
|
747
|
+
}
|
|
748
|
+
}
|
|
749
|
+
// Add release comment
|
|
750
|
+
if (this.config.commentOnRelease) {
|
|
751
|
+
const claimantStr = claimant.type === 'human'
|
|
752
|
+
? `@${claimant.name.replace(/[^a-zA-Z0-9_-]/g, '')}`
|
|
753
|
+
: `Agent: ${(claimant.agentType || 'unknown').replace(/[^a-zA-Z0-9_-]/g, '')}`;
|
|
754
|
+
const comment = `🔓 **Issue released** by ${claimantStr}\n\n_This issue is now available for others to claim._`;
|
|
755
|
+
try {
|
|
756
|
+
execFileSync('gh', [
|
|
757
|
+
'issue', 'comment', String(issueNumber),
|
|
758
|
+
'--repo', repo,
|
|
759
|
+
'--body', comment
|
|
760
|
+
], { stdio: 'ignore' });
|
|
761
|
+
}
|
|
762
|
+
catch {
|
|
763
|
+
errors.push('Failed to add release comment');
|
|
764
|
+
}
|
|
765
|
+
}
|
|
766
|
+
return { success: errors.length === 0, synced: 1, errors };
|
|
767
|
+
}
|
|
768
|
+
catch (error) {
|
|
769
|
+
errors.push(`GitHub release sync failed: ${sanitizeError(error)}`);
|
|
770
|
+
return { success: false, synced: 0, errors };
|
|
771
|
+
}
|
|
772
|
+
}
|
|
773
|
+
/**
|
|
774
|
+
* Bulk sync all local claims to GitHub
|
|
775
|
+
*/
|
|
776
|
+
async syncAllClaimsToGitHub() {
|
|
777
|
+
const errors = [];
|
|
778
|
+
let synced = 0;
|
|
779
|
+
const claims = await this.claimService.getAllClaims();
|
|
780
|
+
for (const claim of claims) {
|
|
781
|
+
// Extract issue number from issueId (assumes format like "123" or "issue-123")
|
|
782
|
+
const issueMatch = claim.issueId.match(/(\d+)/);
|
|
783
|
+
if (issueMatch) {
|
|
784
|
+
const result = await this.claimOnGitHub(parseInt(issueMatch[1], 10), claim.claimant);
|
|
785
|
+
if (result.success)
|
|
786
|
+
synced++;
|
|
787
|
+
else
|
|
788
|
+
errors.push(...result.errors);
|
|
789
|
+
}
|
|
790
|
+
}
|
|
791
|
+
return { success: errors.length === 0, synced, errors };
|
|
792
|
+
}
|
|
793
|
+
/**
|
|
794
|
+
* Get GitHub issues that are claimed locally
|
|
795
|
+
*/
|
|
796
|
+
async getClaimedGitHubIssues() {
|
|
797
|
+
const syncResult = await this.syncIssues('open');
|
|
798
|
+
if (!syncResult.success || !syncResult.issues)
|
|
799
|
+
return [];
|
|
800
|
+
const localClaims = await this.claimService.getAllClaims();
|
|
801
|
+
const claimedIds = new Set(localClaims.map(c => {
|
|
802
|
+
const match = c.issueId.match(/(\d+)/);
|
|
803
|
+
return match ? parseInt(match[1], 10) : null;
|
|
804
|
+
}).filter(Boolean));
|
|
805
|
+
return syncResult.issues.filter(issue => claimedIds.has(issue.number));
|
|
806
|
+
}
|
|
807
|
+
}
|
|
808
|
+
// ============================================================================
|
|
809
|
+
// Factory
|
|
810
|
+
// ============================================================================
|
|
811
|
+
export function createClaimService(projectRoot, config) {
|
|
812
|
+
return new ClaimService(projectRoot, config);
|
|
813
|
+
}
|
|
814
|
+
export function createGitHubSync(claimService, config) {
|
|
815
|
+
return new GitHubSync(claimService, config);
|
|
816
|
+
}
|
|
817
|
+
export default ClaimService;
|
|
818
|
+
//# sourceMappingURL=claim-service.js.map
|