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.
Files changed (429) hide show
  1. package/README.md +1573 -41
  2. package/bin/cli.js +140 -26
  3. package/bin/mcp-server.js +188 -0
  4. package/dist/src/commands/agent.d.ts +8 -0
  5. package/dist/src/commands/agent.d.ts.map +1 -0
  6. package/dist/src/commands/agent.js +819 -0
  7. package/dist/src/commands/agent.js.map +1 -0
  8. package/dist/src/commands/analyze.d.ts +19 -0
  9. package/dist/src/commands/analyze.d.ts.map +1 -0
  10. package/dist/src/commands/analyze.js +1823 -0
  11. package/dist/src/commands/analyze.js.map +1 -0
  12. package/dist/src/commands/claims.d.ts +10 -0
  13. package/dist/src/commands/claims.d.ts.map +1 -0
  14. package/dist/src/commands/claims.js +288 -0
  15. package/dist/src/commands/claims.js.map +1 -0
  16. package/dist/src/commands/completions.d.ts +10 -0
  17. package/dist/src/commands/completions.d.ts.map +1 -0
  18. package/dist/src/commands/completions.js +539 -0
  19. package/dist/src/commands/completions.js.map +1 -0
  20. package/dist/src/commands/config.d.ts +8 -0
  21. package/dist/src/commands/config.d.ts.map +1 -0
  22. package/dist/src/commands/config.js +406 -0
  23. package/dist/src/commands/config.js.map +1 -0
  24. package/dist/src/commands/daemon.d.ts +8 -0
  25. package/dist/src/commands/daemon.d.ts.map +1 -0
  26. package/dist/src/commands/daemon.js +593 -0
  27. package/dist/src/commands/daemon.js.map +1 -0
  28. package/dist/src/commands/deployment.d.ts +10 -0
  29. package/dist/src/commands/deployment.d.ts.map +1 -0
  30. package/dist/src/commands/deployment.js +289 -0
  31. package/dist/src/commands/deployment.js.map +1 -0
  32. package/dist/src/commands/doctor.d.ts +10 -0
  33. package/dist/src/commands/doctor.d.ts.map +1 -0
  34. package/dist/src/commands/doctor.js +448 -0
  35. package/dist/src/commands/doctor.js.map +1 -0
  36. package/dist/src/commands/embeddings.d.ts +18 -0
  37. package/dist/src/commands/embeddings.d.ts.map +1 -0
  38. package/dist/src/commands/embeddings.js +831 -0
  39. package/dist/src/commands/embeddings.js.map +1 -0
  40. package/dist/src/commands/hive-mind.d.ts +8 -0
  41. package/dist/src/commands/hive-mind.d.ts.map +1 -0
  42. package/dist/src/commands/hive-mind.js +844 -0
  43. package/dist/src/commands/hive-mind.js.map +1 -0
  44. package/dist/src/commands/hooks.d.ts +8 -0
  45. package/dist/src/commands/hooks.d.ts.map +1 -0
  46. package/dist/src/commands/hooks.js +2892 -0
  47. package/dist/src/commands/hooks.js.map +1 -0
  48. package/dist/src/commands/index.d.ts +89 -0
  49. package/dist/src/commands/index.d.ts.map +1 -0
  50. package/dist/src/commands/index.js +258 -0
  51. package/dist/src/commands/index.js.map +1 -0
  52. package/dist/src/commands/init.d.ts +8 -0
  53. package/dist/src/commands/init.d.ts.map +1 -0
  54. package/dist/src/commands/init.js +603 -0
  55. package/dist/src/commands/init.js.map +1 -0
  56. package/dist/src/commands/issues.d.ts +21 -0
  57. package/dist/src/commands/issues.d.ts.map +1 -0
  58. package/dist/src/commands/issues.js +567 -0
  59. package/dist/src/commands/issues.js.map +1 -0
  60. package/dist/src/commands/mcp.d.ts +11 -0
  61. package/dist/src/commands/mcp.d.ts.map +1 -0
  62. package/dist/src/commands/mcp.js +662 -0
  63. package/dist/src/commands/mcp.js.map +1 -0
  64. package/dist/src/commands/memory.d.ts +8 -0
  65. package/dist/src/commands/memory.d.ts.map +1 -0
  66. package/dist/src/commands/memory.js +1200 -0
  67. package/dist/src/commands/memory.js.map +1 -0
  68. package/dist/src/commands/migrate.d.ts +8 -0
  69. package/dist/src/commands/migrate.d.ts.map +1 -0
  70. package/dist/src/commands/migrate.js +398 -0
  71. package/dist/src/commands/migrate.js.map +1 -0
  72. package/dist/src/commands/neural.d.ts +10 -0
  73. package/dist/src/commands/neural.d.ts.map +1 -0
  74. package/dist/src/commands/neural.js +224 -0
  75. package/dist/src/commands/neural.js.map +1 -0
  76. package/dist/src/commands/performance.d.ts +10 -0
  77. package/dist/src/commands/performance.d.ts.map +1 -0
  78. package/dist/src/commands/performance.js +262 -0
  79. package/dist/src/commands/performance.js.map +1 -0
  80. package/dist/src/commands/plugins.d.ts +11 -0
  81. package/dist/src/commands/plugins.d.ts.map +1 -0
  82. package/dist/src/commands/plugins.js +630 -0
  83. package/dist/src/commands/plugins.js.map +1 -0
  84. package/dist/src/commands/process.d.ts +10 -0
  85. package/dist/src/commands/process.d.ts.map +1 -0
  86. package/dist/src/commands/process.js +641 -0
  87. package/dist/src/commands/process.js.map +1 -0
  88. package/dist/src/commands/progress.d.ts +11 -0
  89. package/dist/src/commands/progress.d.ts.map +1 -0
  90. package/dist/src/commands/progress.js +259 -0
  91. package/dist/src/commands/progress.js.map +1 -0
  92. package/dist/src/commands/providers.d.ts +10 -0
  93. package/dist/src/commands/providers.d.ts.map +1 -0
  94. package/dist/src/commands/providers.js +232 -0
  95. package/dist/src/commands/providers.js.map +1 -0
  96. package/dist/src/commands/route.d.ts +16 -0
  97. package/dist/src/commands/route.d.ts.map +1 -0
  98. package/dist/src/commands/route.js +813 -0
  99. package/dist/src/commands/route.js.map +1 -0
  100. package/dist/src/commands/security.d.ts +10 -0
  101. package/dist/src/commands/security.d.ts.map +1 -0
  102. package/dist/src/commands/security.js +261 -0
  103. package/dist/src/commands/security.js.map +1 -0
  104. package/dist/src/commands/session.d.ts +8 -0
  105. package/dist/src/commands/session.d.ts.map +1 -0
  106. package/dist/src/commands/session.js +750 -0
  107. package/dist/src/commands/session.js.map +1 -0
  108. package/dist/src/commands/start.d.ts +8 -0
  109. package/dist/src/commands/start.d.ts.map +1 -0
  110. package/dist/src/commands/start.js +418 -0
  111. package/dist/src/commands/start.js.map +1 -0
  112. package/dist/src/commands/status.d.ts +8 -0
  113. package/dist/src/commands/status.d.ts.map +1 -0
  114. package/dist/src/commands/status.js +584 -0
  115. package/dist/src/commands/status.js.map +1 -0
  116. package/dist/src/commands/swarm.d.ts +8 -0
  117. package/dist/src/commands/swarm.d.ts.map +1 -0
  118. package/dist/src/commands/swarm.js +726 -0
  119. package/dist/src/commands/swarm.js.map +1 -0
  120. package/dist/src/commands/task.d.ts +8 -0
  121. package/dist/src/commands/task.d.ts.map +1 -0
  122. package/dist/src/commands/task.js +671 -0
  123. package/dist/src/commands/task.js.map +1 -0
  124. package/dist/src/commands/transfer-store.d.ts +13 -0
  125. package/dist/src/commands/transfer-store.d.ts.map +1 -0
  126. package/dist/src/commands/transfer-store.js +428 -0
  127. package/dist/src/commands/transfer-store.js.map +1 -0
  128. package/dist/src/commands/workflow.d.ts +8 -0
  129. package/dist/src/commands/workflow.d.ts.map +1 -0
  130. package/dist/src/commands/workflow.js +617 -0
  131. package/dist/src/commands/workflow.js.map +1 -0
  132. package/dist/src/config-adapter.d.ts +15 -0
  133. package/dist/src/config-adapter.d.ts.map +1 -0
  134. package/dist/src/config-adapter.js +185 -0
  135. package/dist/src/config-adapter.js.map +1 -0
  136. package/dist/src/index.d.ts +57 -0
  137. package/dist/src/index.d.ts.map +1 -0
  138. package/dist/src/index.js +388 -0
  139. package/dist/src/index.js.map +1 -0
  140. package/dist/src/infrastructure/in-memory-repositories.d.ts +68 -0
  141. package/dist/src/infrastructure/in-memory-repositories.d.ts.map +1 -0
  142. package/dist/src/infrastructure/in-memory-repositories.js +264 -0
  143. package/dist/src/infrastructure/in-memory-repositories.js.map +1 -0
  144. package/dist/src/init/claudemd-generator.d.ts +15 -0
  145. package/dist/src/init/claudemd-generator.d.ts.map +1 -0
  146. package/dist/src/init/claudemd-generator.js +674 -0
  147. package/dist/src/init/claudemd-generator.js.map +1 -0
  148. package/dist/src/init/executor.d.ts +11 -0
  149. package/dist/src/init/executor.d.ts.map +1 -0
  150. package/dist/src/init/executor.js +763 -0
  151. package/dist/src/init/executor.js.map +1 -0
  152. package/dist/src/init/helpers-generator.d.ts +42 -0
  153. package/dist/src/init/helpers-generator.d.ts.map +1 -0
  154. package/dist/src/init/helpers-generator.js +613 -0
  155. package/dist/src/init/helpers-generator.js.map +1 -0
  156. package/dist/src/init/index.d.ts +12 -0
  157. package/dist/src/init/index.d.ts.map +1 -0
  158. package/dist/src/init/index.js +15 -0
  159. package/dist/src/init/index.js.map +1 -0
  160. package/dist/src/init/mcp-generator.d.ts +27 -0
  161. package/dist/src/init/mcp-generator.d.ts.map +1 -0
  162. package/dist/src/init/mcp-generator.js +100 -0
  163. package/dist/src/init/mcp-generator.js.map +1 -0
  164. package/dist/src/init/settings-generator.d.ts +14 -0
  165. package/dist/src/init/settings-generator.d.ts.map +1 -0
  166. package/dist/src/init/settings-generator.js +311 -0
  167. package/dist/src/init/settings-generator.js.map +1 -0
  168. package/dist/src/init/statusline-generator.d.ts +20 -0
  169. package/dist/src/init/statusline-generator.d.ts.map +1 -0
  170. package/dist/src/init/statusline-generator.js +369 -0
  171. package/dist/src/init/statusline-generator.js.map +1 -0
  172. package/dist/src/init/types.d.ts +246 -0
  173. package/dist/src/init/types.d.ts.map +1 -0
  174. package/dist/src/init/types.js +216 -0
  175. package/dist/src/init/types.js.map +1 -0
  176. package/dist/src/mcp-client.d.ts +92 -0
  177. package/dist/src/mcp-client.d.ts.map +1 -0
  178. package/dist/src/mcp-client.js +207 -0
  179. package/dist/src/mcp-client.js.map +1 -0
  180. package/dist/src/mcp-server.d.ts +158 -0
  181. package/dist/src/mcp-server.d.ts.map +1 -0
  182. package/dist/src/mcp-server.js +584 -0
  183. package/dist/src/mcp-server.js.map +1 -0
  184. package/dist/src/mcp-tools/agent-tools.d.ts +8 -0
  185. package/dist/src/mcp-tools/agent-tools.d.ts.map +1 -0
  186. package/dist/src/mcp-tools/agent-tools.js +426 -0
  187. package/dist/src/mcp-tools/agent-tools.js.map +1 -0
  188. package/dist/src/mcp-tools/analyze-tools.d.ts +38 -0
  189. package/dist/src/mcp-tools/analyze-tools.d.ts.map +1 -0
  190. package/dist/src/mcp-tools/analyze-tools.js +317 -0
  191. package/dist/src/mcp-tools/analyze-tools.js.map +1 -0
  192. package/dist/src/mcp-tools/config-tools.d.ts +8 -0
  193. package/dist/src/mcp-tools/config-tools.d.ts.map +1 -0
  194. package/dist/src/mcp-tools/config-tools.js +333 -0
  195. package/dist/src/mcp-tools/config-tools.js.map +1 -0
  196. package/dist/src/mcp-tools/hive-mind-tools.d.ts +8 -0
  197. package/dist/src/mcp-tools/hive-mind-tools.d.ts.map +1 -0
  198. package/dist/src/mcp-tools/hive-mind-tools.js +447 -0
  199. package/dist/src/mcp-tools/hive-mind-tools.js.map +1 -0
  200. package/dist/src/mcp-tools/hooks-tools.d.ts +41 -0
  201. package/dist/src/mcp-tools/hooks-tools.d.ts.map +1 -0
  202. package/dist/src/mcp-tools/hooks-tools.js +1836 -0
  203. package/dist/src/mcp-tools/hooks-tools.js.map +1 -0
  204. package/dist/src/mcp-tools/index.d.ts +20 -0
  205. package/dist/src/mcp-tools/index.d.ts.map +1 -0
  206. package/dist/src/mcp-tools/index.js +19 -0
  207. package/dist/src/mcp-tools/index.js.map +1 -0
  208. package/dist/src/mcp-tools/memory-tools.d.ts +8 -0
  209. package/dist/src/mcp-tools/memory-tools.d.ts.map +1 -0
  210. package/dist/src/mcp-tools/memory-tools.js +235 -0
  211. package/dist/src/mcp-tools/memory-tools.js.map +1 -0
  212. package/dist/src/mcp-tools/progress-tools.d.ts +14 -0
  213. package/dist/src/mcp-tools/progress-tools.d.ts.map +1 -0
  214. package/dist/src/mcp-tools/progress-tools.js +343 -0
  215. package/dist/src/mcp-tools/progress-tools.js.map +1 -0
  216. package/dist/src/mcp-tools/session-tools.d.ts +8 -0
  217. package/dist/src/mcp-tools/session-tools.d.ts.map +1 -0
  218. package/dist/src/mcp-tools/session-tools.js +315 -0
  219. package/dist/src/mcp-tools/session-tools.js.map +1 -0
  220. package/dist/src/mcp-tools/swarm-tools.d.ts +8 -0
  221. package/dist/src/mcp-tools/swarm-tools.d.ts.map +1 -0
  222. package/dist/src/mcp-tools/swarm-tools.js +102 -0
  223. package/dist/src/mcp-tools/swarm-tools.js.map +1 -0
  224. package/dist/src/mcp-tools/task-tools.d.ts +8 -0
  225. package/dist/src/mcp-tools/task-tools.d.ts.map +1 -0
  226. package/dist/src/mcp-tools/task-tools.js +302 -0
  227. package/dist/src/mcp-tools/task-tools.js.map +1 -0
  228. package/dist/src/mcp-tools/transfer-tools.d.ts +14 -0
  229. package/dist/src/mcp-tools/transfer-tools.d.ts.map +1 -0
  230. package/dist/src/mcp-tools/transfer-tools.js +396 -0
  231. package/dist/src/mcp-tools/transfer-tools.js.map +1 -0
  232. package/dist/src/mcp-tools/types.d.ts +31 -0
  233. package/dist/src/mcp-tools/types.d.ts.map +1 -0
  234. package/dist/src/mcp-tools/types.js +7 -0
  235. package/dist/src/mcp-tools/types.js.map +1 -0
  236. package/dist/src/mcp-tools/workflow-tools.d.ts +8 -0
  237. package/dist/src/mcp-tools/workflow-tools.d.ts.map +1 -0
  238. package/dist/src/mcp-tools/workflow-tools.js +481 -0
  239. package/dist/src/mcp-tools/workflow-tools.js.map +1 -0
  240. package/dist/src/memory/memory-initializer.d.ts +229 -0
  241. package/dist/src/memory/memory-initializer.d.ts.map +1 -0
  242. package/dist/src/memory/memory-initializer.js +1248 -0
  243. package/dist/src/memory/memory-initializer.js.map +1 -0
  244. package/dist/src/output.d.ts +133 -0
  245. package/dist/src/output.d.ts.map +1 -0
  246. package/dist/src/output.js +513 -0
  247. package/dist/src/output.js.map +1 -0
  248. package/dist/src/parser.d.ts +41 -0
  249. package/dist/src/parser.d.ts.map +1 -0
  250. package/dist/src/parser.js +353 -0
  251. package/dist/src/parser.js.map +1 -0
  252. package/dist/src/plugins/store/discovery.d.ts +73 -0
  253. package/dist/src/plugins/store/discovery.d.ts.map +1 -0
  254. package/dist/src/plugins/store/discovery.js +568 -0
  255. package/dist/src/plugins/store/discovery.js.map +1 -0
  256. package/dist/src/plugins/store/index.d.ts +76 -0
  257. package/dist/src/plugins/store/index.d.ts.map +1 -0
  258. package/dist/src/plugins/store/index.js +141 -0
  259. package/dist/src/plugins/store/index.js.map +1 -0
  260. package/dist/src/plugins/store/search.d.ts +46 -0
  261. package/dist/src/plugins/store/search.d.ts.map +1 -0
  262. package/dist/src/plugins/store/search.js +230 -0
  263. package/dist/src/plugins/store/search.js.map +1 -0
  264. package/dist/src/plugins/store/types.d.ts +274 -0
  265. package/dist/src/plugins/store/types.d.ts.map +1 -0
  266. package/dist/src/plugins/store/types.js +7 -0
  267. package/dist/src/plugins/store/types.js.map +1 -0
  268. package/dist/src/plugins/tests/demo-plugin-store.d.ts +7 -0
  269. package/dist/src/plugins/tests/demo-plugin-store.d.ts.map +1 -0
  270. package/dist/src/plugins/tests/demo-plugin-store.js +126 -0
  271. package/dist/src/plugins/tests/demo-plugin-store.js.map +1 -0
  272. package/dist/src/plugins/tests/standalone-test.d.ts +12 -0
  273. package/dist/src/plugins/tests/standalone-test.d.ts.map +1 -0
  274. package/dist/src/plugins/tests/standalone-test.js +188 -0
  275. package/dist/src/plugins/tests/standalone-test.js.map +1 -0
  276. package/dist/src/plugins/tests/test-plugin-store.d.ts +7 -0
  277. package/dist/src/plugins/tests/test-plugin-store.d.ts.map +1 -0
  278. package/dist/src/plugins/tests/test-plugin-store.js +206 -0
  279. package/dist/src/plugins/tests/test-plugin-store.js.map +1 -0
  280. package/dist/src/prompt.d.ts +44 -0
  281. package/dist/src/prompt.d.ts.map +1 -0
  282. package/dist/src/prompt.js +501 -0
  283. package/dist/src/prompt.js.map +1 -0
  284. package/dist/src/ruvector/ast-analyzer.d.ts +67 -0
  285. package/dist/src/ruvector/ast-analyzer.d.ts.map +1 -0
  286. package/dist/src/ruvector/ast-analyzer.js +277 -0
  287. package/dist/src/ruvector/ast-analyzer.js.map +1 -0
  288. package/dist/src/ruvector/coverage-router.d.ts +160 -0
  289. package/dist/src/ruvector/coverage-router.d.ts.map +1 -0
  290. package/dist/src/ruvector/coverage-router.js +529 -0
  291. package/dist/src/ruvector/coverage-router.js.map +1 -0
  292. package/dist/src/ruvector/coverage-tools.d.ts +33 -0
  293. package/dist/src/ruvector/coverage-tools.d.ts.map +1 -0
  294. package/dist/src/ruvector/coverage-tools.js +157 -0
  295. package/dist/src/ruvector/coverage-tools.js.map +1 -0
  296. package/dist/src/ruvector/diff-classifier.d.ts +175 -0
  297. package/dist/src/ruvector/diff-classifier.d.ts.map +1 -0
  298. package/dist/src/ruvector/diff-classifier.js +698 -0
  299. package/dist/src/ruvector/diff-classifier.js.map +1 -0
  300. package/dist/src/ruvector/graph-analyzer.d.ts +187 -0
  301. package/dist/src/ruvector/graph-analyzer.d.ts.map +1 -0
  302. package/dist/src/ruvector/graph-analyzer.js +929 -0
  303. package/dist/src/ruvector/graph-analyzer.js.map +1 -0
  304. package/dist/src/ruvector/index.d.ts +27 -0
  305. package/dist/src/ruvector/index.d.ts.map +1 -0
  306. package/dist/src/ruvector/index.js +53 -0
  307. package/dist/src/ruvector/index.js.map +1 -0
  308. package/dist/src/ruvector/q-learning-router.d.ts +211 -0
  309. package/dist/src/ruvector/q-learning-router.d.ts.map +1 -0
  310. package/dist/src/ruvector/q-learning-router.js +681 -0
  311. package/dist/src/ruvector/q-learning-router.js.map +1 -0
  312. package/dist/src/ruvector/vector-db.d.ts +69 -0
  313. package/dist/src/ruvector/vector-db.d.ts.map +1 -0
  314. package/dist/src/ruvector/vector-db.js +243 -0
  315. package/dist/src/ruvector/vector-db.js.map +1 -0
  316. package/dist/src/services/claim-service.d.ts +204 -0
  317. package/dist/src/services/claim-service.d.ts.map +1 -0
  318. package/dist/src/services/claim-service.js +818 -0
  319. package/dist/src/services/claim-service.js.map +1 -0
  320. package/dist/src/services/container-worker-pool.d.ts +197 -0
  321. package/dist/src/services/container-worker-pool.d.ts.map +1 -0
  322. package/dist/src/services/container-worker-pool.js +581 -0
  323. package/dist/src/services/container-worker-pool.js.map +1 -0
  324. package/dist/src/services/headless-worker-executor.d.ts +304 -0
  325. package/dist/src/services/headless-worker-executor.d.ts.map +1 -0
  326. package/dist/src/services/headless-worker-executor.js +997 -0
  327. package/dist/src/services/headless-worker-executor.js.map +1 -0
  328. package/dist/src/services/index.d.ts +13 -0
  329. package/dist/src/services/index.d.ts.map +1 -0
  330. package/dist/src/services/index.js +11 -0
  331. package/dist/src/services/index.js.map +1 -0
  332. package/dist/src/services/worker-daemon.d.ts +203 -0
  333. package/dist/src/services/worker-daemon.d.ts.map +1 -0
  334. package/dist/src/services/worker-daemon.js +745 -0
  335. package/dist/src/services/worker-daemon.js.map +1 -0
  336. package/dist/src/services/worker-queue.d.ts +194 -0
  337. package/dist/src/services/worker-queue.d.ts.map +1 -0
  338. package/dist/src/services/worker-queue.js +511 -0
  339. package/dist/src/services/worker-queue.js.map +1 -0
  340. package/dist/src/suggest.d.ts +53 -0
  341. package/dist/src/suggest.d.ts.map +1 -0
  342. package/dist/src/suggest.js +200 -0
  343. package/dist/src/suggest.js.map +1 -0
  344. package/dist/src/transfer/anonymization/index.d.ts +25 -0
  345. package/dist/src/transfer/anonymization/index.d.ts.map +1 -0
  346. package/dist/src/transfer/anonymization/index.js +175 -0
  347. package/dist/src/transfer/anonymization/index.js.map +1 -0
  348. package/dist/src/transfer/deploy-seraphine.d.ts +13 -0
  349. package/dist/src/transfer/deploy-seraphine.d.ts.map +1 -0
  350. package/dist/src/transfer/deploy-seraphine.js +205 -0
  351. package/dist/src/transfer/deploy-seraphine.js.map +1 -0
  352. package/dist/src/transfer/export.d.ts +25 -0
  353. package/dist/src/transfer/export.d.ts.map +1 -0
  354. package/dist/src/transfer/export.js +113 -0
  355. package/dist/src/transfer/export.js.map +1 -0
  356. package/dist/src/transfer/index.d.ts +12 -0
  357. package/dist/src/transfer/index.d.ts.map +1 -0
  358. package/dist/src/transfer/index.js +31 -0
  359. package/dist/src/transfer/index.js.map +1 -0
  360. package/dist/src/transfer/ipfs/client.d.ts +31 -0
  361. package/dist/src/transfer/ipfs/client.d.ts.map +1 -0
  362. package/dist/src/transfer/ipfs/client.js +74 -0
  363. package/dist/src/transfer/ipfs/client.js.map +1 -0
  364. package/dist/src/transfer/ipfs/upload.d.ts +95 -0
  365. package/dist/src/transfer/ipfs/upload.d.ts.map +1 -0
  366. package/dist/src/transfer/ipfs/upload.js +410 -0
  367. package/dist/src/transfer/ipfs/upload.js.map +1 -0
  368. package/dist/src/transfer/models/seraphine.d.ts +72 -0
  369. package/dist/src/transfer/models/seraphine.d.ts.map +1 -0
  370. package/dist/src/transfer/models/seraphine.js +373 -0
  371. package/dist/src/transfer/models/seraphine.js.map +1 -0
  372. package/dist/src/transfer/serialization/cfp.d.ts +49 -0
  373. package/dist/src/transfer/serialization/cfp.d.ts.map +1 -0
  374. package/dist/src/transfer/serialization/cfp.js +180 -0
  375. package/dist/src/transfer/serialization/cfp.js.map +1 -0
  376. package/dist/src/transfer/store/discovery.d.ts +84 -0
  377. package/dist/src/transfer/store/discovery.d.ts.map +1 -0
  378. package/dist/src/transfer/store/discovery.js +275 -0
  379. package/dist/src/transfer/store/discovery.js.map +1 -0
  380. package/dist/src/transfer/store/download.d.ts +70 -0
  381. package/dist/src/transfer/store/download.d.ts.map +1 -0
  382. package/dist/src/transfer/store/download.js +295 -0
  383. package/dist/src/transfer/store/download.js.map +1 -0
  384. package/dist/src/transfer/store/index.d.ts +84 -0
  385. package/dist/src/transfer/store/index.d.ts.map +1 -0
  386. package/dist/src/transfer/store/index.js +153 -0
  387. package/dist/src/transfer/store/index.js.map +1 -0
  388. package/dist/src/transfer/store/publish.d.ts +76 -0
  389. package/dist/src/transfer/store/publish.d.ts.map +1 -0
  390. package/dist/src/transfer/store/publish.js +262 -0
  391. package/dist/src/transfer/store/publish.js.map +1 -0
  392. package/dist/src/transfer/store/registry.d.ts +58 -0
  393. package/dist/src/transfer/store/registry.d.ts.map +1 -0
  394. package/dist/src/transfer/store/registry.js +285 -0
  395. package/dist/src/transfer/store/registry.js.map +1 -0
  396. package/dist/src/transfer/store/search.d.ts +54 -0
  397. package/dist/src/transfer/store/search.d.ts.map +1 -0
  398. package/dist/src/transfer/store/search.js +232 -0
  399. package/dist/src/transfer/store/search.js.map +1 -0
  400. package/dist/src/transfer/store/tests/standalone-test.d.ts +12 -0
  401. package/dist/src/transfer/store/tests/standalone-test.d.ts.map +1 -0
  402. package/dist/src/transfer/store/tests/standalone-test.js +190 -0
  403. package/dist/src/transfer/store/tests/standalone-test.js.map +1 -0
  404. package/dist/src/transfer/store/types.d.ts +193 -0
  405. package/dist/src/transfer/store/types.d.ts.map +1 -0
  406. package/dist/src/transfer/store/types.js +6 -0
  407. package/dist/src/transfer/store/types.js.map +1 -0
  408. package/dist/src/transfer/test-seraphine.d.ts +6 -0
  409. package/dist/src/transfer/test-seraphine.d.ts.map +1 -0
  410. package/dist/src/transfer/test-seraphine.js +105 -0
  411. package/dist/src/transfer/test-seraphine.js.map +1 -0
  412. package/dist/src/transfer/tests/test-store.d.ts +7 -0
  413. package/dist/src/transfer/tests/test-store.d.ts.map +1 -0
  414. package/dist/src/transfer/tests/test-store.js +214 -0
  415. package/dist/src/transfer/tests/test-store.js.map +1 -0
  416. package/dist/src/transfer/types.d.ts +245 -0
  417. package/dist/src/transfer/types.d.ts.map +1 -0
  418. package/dist/src/transfer/types.js +6 -0
  419. package/dist/src/transfer/types.js.map +1 -0
  420. package/dist/src/types.d.ts +198 -0
  421. package/dist/src/types.d.ts.map +1 -0
  422. package/dist/src/types.js +38 -0
  423. package/dist/src/types.js.map +1 -0
  424. package/dist/tsconfig.tsbuildinfo +1 -0
  425. package/package.json +33 -47
  426. package/dist/index.d.ts +0 -30
  427. package/dist/index.d.ts.map +0 -1
  428. package/dist/index.js +0 -30
  429. 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