mcoda 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (394) hide show
  1. package/.editorconfig +9 -0
  2. package/.eslintrc.cjs +12 -0
  3. package/.github/ISSUE_TEMPLATE/bug_report.md +29 -0
  4. package/.github/ISSUE_TEMPLATE/config.yml +5 -0
  5. package/.github/ISSUE_TEMPLATE/feature_request.md +19 -0
  6. package/.github/workflows/ci.yml +37 -0
  7. package/.github/workflows/nightly.yml +38 -0
  8. package/.github/workflows/release-dry-run.yml +40 -0
  9. package/.github/workflows/release-please.yml +22 -0
  10. package/.github/workflows/release.yml +149 -0
  11. package/.prettierrc +5 -0
  12. package/.release-please-manifest.json +8 -0
  13. package/CHANGELOG.md +7 -0
  14. package/CLA.md +42 -0
  15. package/CONTRIBUTING.md +38 -0
  16. package/LICENSE +21 -0
  17. package/README.md +314 -0
  18. package/docs/oss_publishing_plan.md +41 -0
  19. package/docs/pdr/.gitkeep +0 -0
  20. package/docs/quality_gates.md +32 -0
  21. package/docs/rfp/.gitkeep +0 -0
  22. package/docs/sds/sds.md +11963 -0
  23. package/docs/usage.md +72 -0
  24. package/openapi/gen-openapi.ts +1 -0
  25. package/openapi/generated/clients/.gitkeep +0 -0
  26. package/openapi/generated/types/.gitkeep +0 -0
  27. package/openapi/generated/types/index.ts +118 -0
  28. package/openapi/mcoda.yaml +2063 -0
  29. package/pack-mcoda.sh +88 -0
  30. package/package.json +46 -0
  31. package/packages/agents/CHANGELOG.md +7 -0
  32. package/packages/agents/LICENSE +21 -0
  33. package/packages/agents/README.md +9 -0
  34. package/packages/agents/package.json +41 -0
  35. package/packages/agents/src/AgentService/.gitkeep +0 -0
  36. package/packages/agents/src/AgentService/AgentService.d.ts +21 -0
  37. package/packages/agents/src/AgentService/AgentService.d.ts.map +1 -0
  38. package/packages/agents/src/AgentService/AgentService.js +141 -0
  39. package/packages/agents/src/AgentService/AgentService.ts +308 -0
  40. package/packages/agents/src/__tests__/AgentService.test.ts +284 -0
  41. package/packages/agents/src/adapters/AdapterTypes.d.ts +29 -0
  42. package/packages/agents/src/adapters/AdapterTypes.d.ts.map +1 -0
  43. package/packages/agents/src/adapters/AdapterTypes.js +1 -0
  44. package/packages/agents/src/adapters/AdapterTypes.ts +32 -0
  45. package/packages/agents/src/adapters/codex/.gitkeep +0 -0
  46. package/packages/agents/src/adapters/codex/CodexAdapter.d.ts +11 -0
  47. package/packages/agents/src/adapters/codex/CodexAdapter.d.ts.map +1 -0
  48. package/packages/agents/src/adapters/codex/CodexAdapter.js +43 -0
  49. package/packages/agents/src/adapters/codex/CodexAdapter.ts +63 -0
  50. package/packages/agents/src/adapters/codex/CodexCliRunner.ts +154 -0
  51. package/packages/agents/src/adapters/gemini/.gitkeep +0 -0
  52. package/packages/agents/src/adapters/gemini/GeminiAdapter.d.ts +11 -0
  53. package/packages/agents/src/adapters/gemini/GeminiAdapter.d.ts.map +1 -0
  54. package/packages/agents/src/adapters/gemini/GeminiAdapter.js +42 -0
  55. package/packages/agents/src/adapters/gemini/GeminiAdapter.ts +58 -0
  56. package/packages/agents/src/adapters/gemini/GeminiCliRunner.ts +75 -0
  57. package/packages/agents/src/adapters/local/.gitkeep +0 -0
  58. package/packages/agents/src/adapters/local/LocalAdapter.d.ts +11 -0
  59. package/packages/agents/src/adapters/local/LocalAdapter.d.ts.map +1 -0
  60. package/packages/agents/src/adapters/local/LocalAdapter.js +38 -0
  61. package/packages/agents/src/adapters/local/LocalAdapter.ts +43 -0
  62. package/packages/agents/src/adapters/ollama/OllamaCliAdapter.ts +58 -0
  63. package/packages/agents/src/adapters/ollama/OllamaCliRunner.ts +70 -0
  64. package/packages/agents/src/adapters/ollama/OllamaRemoteAdapter.ts +205 -0
  65. package/packages/agents/src/adapters/openai/.gitkeep +0 -0
  66. package/packages/agents/src/adapters/openai/OpenAiAdapter.d.ts +11 -0
  67. package/packages/agents/src/adapters/openai/OpenAiAdapter.d.ts.map +1 -0
  68. package/packages/agents/src/adapters/openai/OpenAiAdapter.js +51 -0
  69. package/packages/agents/src/adapters/openai/OpenAiAdapter.ts +56 -0
  70. package/packages/agents/src/adapters/openai/OpenAiCliAdapter.ts +62 -0
  71. package/packages/agents/src/adapters/qa/.gitkeep +0 -0
  72. package/packages/agents/src/adapters/qa/QaAdapter.d.ts +11 -0
  73. package/packages/agents/src/adapters/qa/QaAdapter.d.ts.map +1 -0
  74. package/packages/agents/src/adapters/qa/QaAdapter.js +37 -0
  75. package/packages/agents/src/adapters/qa/QaAdapter.ts +42 -0
  76. package/packages/agents/src/adapters/zhipu/ZhipuApiAdapter.ts +273 -0
  77. package/packages/agents/src/index.d.ts +8 -0
  78. package/packages/agents/src/index.d.ts.map +1 -0
  79. package/packages/agents/src/index.js +7 -0
  80. package/packages/agents/src/index.ts +11 -0
  81. package/packages/agents/tsconfig.json +14 -0
  82. package/packages/cli/CHANGELOG.md +7 -0
  83. package/packages/cli/LICENSE +21 -0
  84. package/packages/cli/README.md +23 -0
  85. package/packages/cli/package.json +61 -0
  86. package/packages/cli/src/__tests__/AgentsCommands.test.ts +137 -0
  87. package/packages/cli/src/__tests__/BacklogCommands.test.ts +40 -0
  88. package/packages/cli/src/__tests__/CodeReviewCommand.test.ts +594 -0
  89. package/packages/cli/src/__tests__/CreateTasksCommand.test.ts +40 -0
  90. package/packages/cli/src/__tests__/DocsCommands.test.ts +41 -0
  91. package/packages/cli/src/__tests__/EstimateCommands.test.ts +54 -0
  92. package/packages/cli/src/__tests__/JobsCommands.behavior.test.ts +311 -0
  93. package/packages/cli/src/__tests__/JobsCommands.test.ts +49 -0
  94. package/packages/cli/src/__tests__/MigrateTasksCommand.test.ts +36 -0
  95. package/packages/cli/src/__tests__/OpenapiCommands.test.ts +34 -0
  96. package/packages/cli/src/__tests__/OrderTasksCommand.test.ts +150 -0
  97. package/packages/cli/src/__tests__/PlanningCommands.test.ts +9 -0
  98. package/packages/cli/src/__tests__/QaTasksCommand.test.ts +58 -0
  99. package/packages/cli/src/__tests__/RefineTasksCommand.test.ts +63 -0
  100. package/packages/cli/src/__tests__/RoutingCommands.test.ts +302 -0
  101. package/packages/cli/src/__tests__/SetWorkspaceCommand.test.ts +18 -0
  102. package/packages/cli/src/__tests__/TaskShowCommands.test.ts +130 -0
  103. package/packages/cli/src/__tests__/TelemetryCommands.test.ts +35 -0
  104. package/packages/cli/src/__tests__/TestAgentCommand.test.ts +41 -0
  105. package/packages/cli/src/__tests__/UpdateCommands.test.ts +292 -0
  106. package/packages/cli/src/__tests__/WorkOnTasksCommand.test.ts +42 -0
  107. package/packages/cli/src/bin/.gitkeep +0 -0
  108. package/packages/cli/src/bin/McodaEntrypoint.ts +180 -0
  109. package/packages/cli/src/commands/agents/.gitkeep +0 -0
  110. package/packages/cli/src/commands/agents/AgentsCommands.ts +374 -0
  111. package/packages/cli/src/commands/agents/GatewayAgentCommand.ts +621 -0
  112. package/packages/cli/src/commands/agents/TestAgentCommand.ts +63 -0
  113. package/packages/cli/src/commands/backlog/.gitkeep +0 -0
  114. package/packages/cli/src/commands/backlog/BacklogCommands.ts +286 -0
  115. package/packages/cli/src/commands/backlog/OrderTasksCommand.ts +237 -0
  116. package/packages/cli/src/commands/backlog/TaskShowCommands.ts +289 -0
  117. package/packages/cli/src/commands/docs/.gitkeep +0 -0
  118. package/packages/cli/src/commands/docs/DocsCommands.ts +413 -0
  119. package/packages/cli/src/commands/estimate/EstimateCommands.ts +290 -0
  120. package/packages/cli/src/commands/jobs/.gitkeep +0 -0
  121. package/packages/cli/src/commands/jobs/JobsCommands.ts +595 -0
  122. package/packages/cli/src/commands/openapi/OpenapiCommands.ts +167 -0
  123. package/packages/cli/src/commands/planning/.gitkeep +0 -0
  124. package/packages/cli/src/commands/planning/CreateTasksCommand.ts +149 -0
  125. package/packages/cli/src/commands/planning/MigrateTasksCommand.ts +105 -0
  126. package/packages/cli/src/commands/planning/PlanningCommands.ts +1 -0
  127. package/packages/cli/src/commands/planning/QaTasksCommand.ts +320 -0
  128. package/packages/cli/src/commands/planning/RefineTasksCommand.ts +408 -0
  129. package/packages/cli/src/commands/review/CodeReviewCommand.ts +262 -0
  130. package/packages/cli/src/commands/routing/.gitkeep +0 -0
  131. package/packages/cli/src/commands/routing/RoutingCommands.ts +554 -0
  132. package/packages/cli/src/commands/telemetry/.gitkeep +0 -0
  133. package/packages/cli/src/commands/telemetry/TelemetryCommands.ts +348 -0
  134. package/packages/cli/src/commands/update/.gitkeep +0 -0
  135. package/packages/cli/src/commands/update/UpdateCommands.ts +301 -0
  136. package/packages/cli/src/commands/work/WorkOnTasksCommand.ts +264 -0
  137. package/packages/cli/src/commands/workspace/SetWorkspaceCommand.ts +132 -0
  138. package/packages/cli/src/index.ts +18 -0
  139. package/packages/cli/test/packaging_guardrails.test.js +75 -0
  140. package/packages/cli/tsconfig.json +20 -0
  141. package/packages/core/CHANGELOG.md +7 -0
  142. package/packages/core/LICENSE +21 -0
  143. package/packages/core/README.md +9 -0
  144. package/packages/core/package.json +45 -0
  145. package/packages/core/src/__tests__/SmokeClasses.test.ts +32 -0
  146. package/packages/core/src/api/AgentsApi.ts +219 -0
  147. package/packages/core/src/api/QaTasksApi.ts +38 -0
  148. package/packages/core/src/api/TasksApi.ts +35 -0
  149. package/packages/core/src/api/__tests__/AgentsApi.test.ts +203 -0
  150. package/packages/core/src/api/__tests__/QaTasksApi.test.ts +51 -0
  151. package/packages/core/src/api/__tests__/TasksApi.test.ts +56 -0
  152. package/packages/core/src/config/.gitkeep +0 -0
  153. package/packages/core/src/config/ConfigService.ts +1 -0
  154. package/packages/core/src/domain/dependencies/.gitkeep +0 -0
  155. package/packages/core/src/domain/dependencies/Dependency.ts +1 -0
  156. package/packages/core/src/domain/epics/.gitkeep +0 -0
  157. package/packages/core/src/domain/epics/Epic.ts +1 -0
  158. package/packages/core/src/domain/projects/.gitkeep +0 -0
  159. package/packages/core/src/domain/projects/Project.ts +1 -0
  160. package/packages/core/src/domain/tasks/.gitkeep +0 -0
  161. package/packages/core/src/domain/tasks/Task.ts +1 -0
  162. package/packages/core/src/domain/userStories/.gitkeep +0 -0
  163. package/packages/core/src/domain/userStories/UserStory.ts +1 -0
  164. package/packages/core/src/index.ts +27 -0
  165. package/packages/core/src/prompts/.gitkeep +0 -0
  166. package/packages/core/src/prompts/PdrPrompts.ts +23 -0
  167. package/packages/core/src/prompts/PromptLoader.ts +1 -0
  168. package/packages/core/src/prompts/SdsPrompts.ts +47 -0
  169. package/packages/core/src/services/agents/.gitkeep +0 -0
  170. package/packages/core/src/services/agents/AgentManagementService.ts +1 -0
  171. package/packages/core/src/services/agents/GatewayAgentService.ts +956 -0
  172. package/packages/core/src/services/agents/RoutingService.ts +461 -0
  173. package/packages/core/src/services/agents/__tests__/GatewayAgentService.test.ts +72 -0
  174. package/packages/core/src/services/agents/__tests__/RoutingService.test.ts +267 -0
  175. package/packages/core/src/services/agents/generated/RoutingApiClient.ts +89 -0
  176. package/packages/core/src/services/backlog/.gitkeep +0 -0
  177. package/packages/core/src/services/backlog/BacklogService.ts +580 -0
  178. package/packages/core/src/services/backlog/TaskOrderingService.ts +868 -0
  179. package/packages/core/src/services/backlog/__tests__/BacklogService.test.ts +219 -0
  180. package/packages/core/src/services/backlog/__tests__/TaskOrderingService.test.ts +268 -0
  181. package/packages/core/src/services/docs/.gitkeep +0 -0
  182. package/packages/core/src/services/docs/DocsService.ts +1913 -0
  183. package/packages/core/src/services/docs/__tests__/DocsService.test.ts +350 -0
  184. package/packages/core/src/services/estimate/EstimateService.ts +111 -0
  185. package/packages/core/src/services/estimate/VelocityService.ts +272 -0
  186. package/packages/core/src/services/estimate/__tests__/VelocityAndEstimate.test.ts +209 -0
  187. package/packages/core/src/services/estimate/types.ts +41 -0
  188. package/packages/core/src/services/execution/.gitkeep +0 -0
  189. package/packages/core/src/services/execution/ExecutionService.ts +1 -0
  190. package/packages/core/src/services/execution/QaFollowupService.ts +289 -0
  191. package/packages/core/src/services/execution/QaProfileService.ts +160 -0
  192. package/packages/core/src/services/execution/QaTasksService.ts +1303 -0
  193. package/packages/core/src/services/execution/TaskSelectionService.ts +362 -0
  194. package/packages/core/src/services/execution/TaskStateService.ts +64 -0
  195. package/packages/core/src/services/execution/WorkOnTasksService.ts +2023 -0
  196. package/packages/core/src/services/execution/__tests__/QaFollowupService.test.ts +58 -0
  197. package/packages/core/src/services/execution/__tests__/QaProfileService.test.ts +49 -0
  198. package/packages/core/src/services/execution/__tests__/QaTasksService.test.ts +157 -0
  199. package/packages/core/src/services/execution/__tests__/TaskSelectionService.test.ts +179 -0
  200. package/packages/core/src/services/execution/__tests__/TaskStateService.test.ts +51 -0
  201. package/packages/core/src/services/execution/__tests__/WorkOnTasksService.test.ts +285 -0
  202. package/packages/core/src/services/jobs/.gitkeep +0 -0
  203. package/packages/core/src/services/jobs/JobInsightsService.ts +355 -0
  204. package/packages/core/src/services/jobs/JobResumeService.ts +119 -0
  205. package/packages/core/src/services/jobs/JobService.ts +648 -0
  206. package/packages/core/src/services/jobs/JobsApiClient.ts +113 -0
  207. package/packages/core/src/services/jobs/__tests__/JobInsightsService.test.ts +17 -0
  208. package/packages/core/src/services/jobs/__tests__/JobResumeService.test.ts +45 -0
  209. package/packages/core/src/services/jobs/__tests__/JobService.test.ts +44 -0
  210. package/packages/core/src/services/openapi/OpenApiService.ts +558 -0
  211. package/packages/core/src/services/openapi/__tests__/OpenApiService.test.ts +57 -0
  212. package/packages/core/src/services/planning/.gitkeep +0 -0
  213. package/packages/core/src/services/planning/CreateTasksService.ts +1280 -0
  214. package/packages/core/src/services/planning/KeyHelpers.ts +80 -0
  215. package/packages/core/src/services/planning/PlanningService.ts +1 -0
  216. package/packages/core/src/services/planning/RefineTasksService.ts +1552 -0
  217. package/packages/core/src/services/planning/__tests__/CreateTasksService.test.ts +288 -0
  218. package/packages/core/src/services/planning/__tests__/KeyHelpers.test.ts +16 -0
  219. package/packages/core/src/services/planning/__tests__/RefineTasksService.test.ts +172 -0
  220. package/packages/core/src/services/review/CodeReviewService.ts +1386 -0
  221. package/packages/core/src/services/review/__tests__/CodeReviewService.test.ts +89 -0
  222. package/packages/core/src/services/system/SystemUpdateService.ts +177 -0
  223. package/packages/core/src/services/system/__tests__/SystemUpdateService.test.ts +40 -0
  224. package/packages/core/src/services/tasks/TaskApiResolver.ts +37 -0
  225. package/packages/core/src/services/tasks/TaskDetailService.ts +494 -0
  226. package/packages/core/src/services/tasks/__tests__/TaskApiResolver.test.ts +41 -0
  227. package/packages/core/src/services/tasks/__tests__/TaskDetailService.test.ts +178 -0
  228. package/packages/core/src/services/telemetry/.gitkeep +0 -0
  229. package/packages/core/src/services/telemetry/TelemetryService.ts +515 -0
  230. package/packages/core/src/services/telemetry/__tests__/TelemetryService.test.ts +160 -0
  231. package/packages/core/src/workspace/.gitkeep +0 -0
  232. package/packages/core/src/workspace/WorkspaceManager.ts +234 -0
  233. package/packages/core/tsconfig.json +20 -0
  234. package/packages/db/CHANGELOG.md +7 -0
  235. package/packages/db/LICENSE +21 -0
  236. package/packages/db/README.md +9 -0
  237. package/packages/db/package.json +42 -0
  238. package/packages/db/src/__tests__/GlobalRepository.test.ts +109 -0
  239. package/packages/db/src/__tests__/SchemaAlignment.test.ts +80 -0
  240. package/packages/db/src/__tests__/WorkspaceRepository.test.ts +19 -0
  241. package/packages/db/src/index.d.ts +6 -0
  242. package/packages/db/src/index.d.ts.map +1 -0
  243. package/packages/db/src/index.js +5 -0
  244. package/packages/db/src/index.ts +6 -0
  245. package/packages/db/src/migrations/global/.gitkeep +0 -0
  246. package/packages/db/src/migrations/global/GlobalMigrations.d.ts +9 -0
  247. package/packages/db/src/migrations/global/GlobalMigrations.d.ts.map +1 -0
  248. package/packages/db/src/migrations/global/GlobalMigrations.js +68 -0
  249. package/packages/db/src/migrations/global/GlobalMigrations.ts +336 -0
  250. package/packages/db/src/migrations/workspace/.gitkeep +0 -0
  251. package/packages/db/src/migrations/workspace/WorkspaceMigrations.d.ts +9 -0
  252. package/packages/db/src/migrations/workspace/WorkspaceMigrations.d.ts.map +1 -0
  253. package/packages/db/src/migrations/workspace/WorkspaceMigrations.js +251 -0
  254. package/packages/db/src/migrations/workspace/WorkspaceMigrations.ts +248 -0
  255. package/packages/db/src/repositories/global/.gitkeep +0 -0
  256. package/packages/db/src/repositories/global/GlobalRepository.d.ts +30 -0
  257. package/packages/db/src/repositories/global/GlobalRepository.d.ts.map +1 -0
  258. package/packages/db/src/repositories/global/GlobalRepository.js +209 -0
  259. package/packages/db/src/repositories/global/GlobalRepository.ts +492 -0
  260. package/packages/db/src/repositories/workspace/.gitkeep +0 -0
  261. package/packages/db/src/repositories/workspace/WorkspaceRepository.d.ts +282 -0
  262. package/packages/db/src/repositories/workspace/WorkspaceRepository.d.ts.map +1 -0
  263. package/packages/db/src/repositories/workspace/WorkspaceRepository.js +773 -0
  264. package/packages/db/src/repositories/workspace/WorkspaceRepository.ts +1511 -0
  265. package/packages/db/src/sqlite/connection.d.ts +11 -0
  266. package/packages/db/src/sqlite/connection.d.ts.map +1 -0
  267. package/packages/db/src/sqlite/connection.js +31 -0
  268. package/packages/db/src/sqlite/connection.ts +35 -0
  269. package/packages/db/src/sqlite/pragmas.d.ts +5 -0
  270. package/packages/db/src/sqlite/pragmas.d.ts.map +1 -0
  271. package/packages/db/src/sqlite/pragmas.js +6 -0
  272. package/packages/db/src/sqlite/pragmas.ts +10 -0
  273. package/packages/db/tsconfig.json +13 -0
  274. package/packages/generators/package.json +21 -0
  275. package/packages/generators/src/__tests__/Generators.test.ts +19 -0
  276. package/packages/generators/src/index.ts +1 -0
  277. package/packages/generators/src/openapi/generateTypes.ts +1 -0
  278. package/packages/generators/src/openapi/validateSchema.ts +1 -0
  279. package/packages/generators/src/scaffolding/docs/.gitkeep +0 -0
  280. package/packages/generators/src/scaffolding/docs/DocsScaffolder.ts +1 -0
  281. package/packages/generators/src/scaffolding/global/.gitkeep +0 -0
  282. package/packages/generators/src/scaffolding/global/GlobalScaffolder.ts +1 -0
  283. package/packages/generators/src/scaffolding/workspace/.gitkeep +0 -0
  284. package/packages/generators/src/scaffolding/workspace/WorkspaceScaffolder.ts +1 -0
  285. package/packages/generators/tsconfig.json +10 -0
  286. package/packages/integrations/CHANGELOG.md +7 -0
  287. package/packages/integrations/LICENSE +21 -0
  288. package/packages/integrations/README.md +9 -0
  289. package/packages/integrations/package.json +47 -0
  290. package/packages/integrations/src/docdex/.gitkeep +0 -0
  291. package/packages/integrations/src/docdex/DocdexClient.d.ts +50 -0
  292. package/packages/integrations/src/docdex/DocdexClient.d.ts.map +1 -0
  293. package/packages/integrations/src/docdex/DocdexClient.js +216 -0
  294. package/packages/integrations/src/docdex/DocdexClient.ts +261 -0
  295. package/packages/integrations/src/docdex/__tests__/DocdexClient.test.ts +29 -0
  296. package/packages/integrations/src/index.d.ts +2 -0
  297. package/packages/integrations/src/index.d.ts.map +1 -0
  298. package/packages/integrations/src/index.js +4 -0
  299. package/packages/integrations/src/index.ts +5 -0
  300. package/packages/integrations/src/issues/.gitkeep +0 -0
  301. package/packages/integrations/src/issues/IssuesClient.ts +1 -0
  302. package/packages/integrations/src/issues/__tests__/IssuesClient.test.ts +10 -0
  303. package/packages/integrations/src/qa/.gitkeep +0 -0
  304. package/packages/integrations/src/qa/ChromiumQaAdapter.ts +89 -0
  305. package/packages/integrations/src/qa/CliQaAdapter.ts +95 -0
  306. package/packages/integrations/src/qa/MaestroQaAdapter.ts +91 -0
  307. package/packages/integrations/src/qa/QaAdapter.ts +7 -0
  308. package/packages/integrations/src/qa/QaClient.ts +1 -0
  309. package/packages/integrations/src/qa/QaTypes.ts +26 -0
  310. package/packages/integrations/src/qa/__tests__/ChromiumQaAdapter.test.ts +30 -0
  311. package/packages/integrations/src/qa/__tests__/CliQaAdapter.test.ts +33 -0
  312. package/packages/integrations/src/qa/__tests__/MaestroQaAdapter.test.ts +30 -0
  313. package/packages/integrations/src/qa/index.ts +5 -0
  314. package/packages/integrations/src/system/SystemClient.ts +50 -0
  315. package/packages/integrations/src/system/__tests__/SystemClient.test.ts +40 -0
  316. package/packages/integrations/src/telemetry/TelemetryClient.ts +139 -0
  317. package/packages/integrations/src/telemetry/__tests__/TelemetryClient.test.ts +41 -0
  318. package/packages/integrations/src/vcs/.gitkeep +0 -0
  319. package/packages/integrations/src/vcs/VcsClient.ts +211 -0
  320. package/packages/integrations/src/vcs/__tests__/VcsClient.test.ts +26 -0
  321. package/packages/integrations/tsconfig.json +14 -0
  322. package/packages/shared/CHANGELOG.md +7 -0
  323. package/packages/shared/LICENSE +21 -0
  324. package/packages/shared/README.md +9 -0
  325. package/packages/shared/package.json +40 -0
  326. package/packages/shared/src/__tests__/CommandMetadata.test.ts +15 -0
  327. package/packages/shared/src/__tests__/ServiceShells.test.ts +16 -0
  328. package/packages/shared/src/crypto/.gitkeep +0 -0
  329. package/packages/shared/src/crypto/CryptoHelper.d.ts +15 -0
  330. package/packages/shared/src/crypto/CryptoHelper.d.ts.map +1 -0
  331. package/packages/shared/src/crypto/CryptoHelper.js +54 -0
  332. package/packages/shared/src/crypto/CryptoHelper.ts +57 -0
  333. package/packages/shared/src/errors/.gitkeep +0 -0
  334. package/packages/shared/src/errors/ErrorFactory.ts +1 -0
  335. package/packages/shared/src/index.d.ts +6 -0
  336. package/packages/shared/src/index.d.ts.map +1 -0
  337. package/packages/shared/src/index.js +4 -0
  338. package/packages/shared/src/index.ts +35 -0
  339. package/packages/shared/src/logging/.gitkeep +0 -0
  340. package/packages/shared/src/logging/Logger.ts +1 -0
  341. package/packages/shared/src/metadata/CommandMetadata.ts +165 -0
  342. package/packages/shared/src/openapi/.gitkeep +0 -0
  343. package/packages/shared/src/openapi/OpenApiTypes.d.ts +216 -0
  344. package/packages/shared/src/openapi/OpenApiTypes.d.ts.map +1 -0
  345. package/packages/shared/src/openapi/OpenApiTypes.js +1 -0
  346. package/packages/shared/src/openapi/OpenApiTypes.ts +312 -0
  347. package/packages/shared/src/paths/.gitkeep +0 -0
  348. package/packages/shared/src/paths/PathHelper.d.ts +12 -0
  349. package/packages/shared/src/paths/PathHelper.d.ts.map +1 -0
  350. package/packages/shared/src/paths/PathHelper.js +24 -0
  351. package/packages/shared/src/paths/PathHelper.ts +29 -0
  352. package/packages/shared/src/qa/QaProfile.ts +14 -0
  353. package/packages/shared/src/utils/.gitkeep +0 -0
  354. package/packages/shared/src/utils/UtilityService.ts +1 -0
  355. package/packages/shared/tsconfig.json +10 -0
  356. package/packages/testing/package.json +26 -0
  357. package/packages/testing/src/__tests__/TestingFakes.test.ts +15 -0
  358. package/packages/testing/src/cli/e2e/.gitkeep +0 -0
  359. package/packages/testing/src/cli/e2e/E2eSuite.ts +1 -0
  360. package/packages/testing/src/fakes/agents/.gitkeep +0 -0
  361. package/packages/testing/src/fakes/agents/FakeAgents.ts +1 -0
  362. package/packages/testing/src/fakes/docdex/.gitkeep +0 -0
  363. package/packages/testing/src/fakes/docdex/FakeDocdexClient.ts +1 -0
  364. package/packages/testing/src/fakes/qa/.gitkeep +0 -0
  365. package/packages/testing/src/fakes/qa/FakeQaClient.ts +1 -0
  366. package/packages/testing/src/fakes/vcs/.gitkeep +0 -0
  367. package/packages/testing/src/fakes/vcs/FakeVcsClient.ts +1 -0
  368. package/packages/testing/src/fixtures/db/.gitkeep +0 -0
  369. package/packages/testing/src/fixtures/db/DbFixtures.ts +1 -0
  370. package/packages/testing/src/fixtures/workspaces/.gitkeep +0 -0
  371. package/packages/testing/src/fixtures/workspaces/WorkspaceFixtures.ts +1 -0
  372. package/packages/testing/src/index.ts +1 -0
  373. package/packages/testing/tsconfig.json +10 -0
  374. package/pnpm-workspace.yaml +2 -0
  375. package/prompts/README.md +5 -0
  376. package/prompts/code-reviewer.md +23 -0
  377. package/prompts/code-writer.md +35 -0
  378. package/prompts/gateway-agent.md +27 -0
  379. package/prompts/qa-agent.md +21 -0
  380. package/release-please-config.json +39 -0
  381. package/scripts/build-all.ts +1 -0
  382. package/scripts/dev.ts +1 -0
  383. package/scripts/install-local-cli.sh +28 -0
  384. package/scripts/pack-npm-tarballs.js +63 -0
  385. package/scripts/release.ts +1 -0
  386. package/scripts/run-node-tests.js +37 -0
  387. package/tests/all.js +127 -0
  388. package/tests/api/openapi_spec.test.js +21 -0
  389. package/tests/artifacts.md +31 -0
  390. package/tests/component/cli_version.test.js +38 -0
  391. package/tests/integration/workspace_resolver.test.js +44 -0
  392. package/tests/unit/crypto_helper.test.js +36 -0
  393. package/tests/unit/path_helper.test.js +20 -0
  394. package/tsconfig.base.json +32 -0
@@ -0,0 +1,286 @@
1
+ import path from "node:path";
2
+ import { BacklogService, BacklogSummary, WorkspaceResolver } from "@mcoda/core";
3
+
4
+ interface ParsedArgs {
5
+ workspaceRoot?: string;
6
+ project?: string;
7
+ epic?: string;
8
+ story?: string;
9
+ assignee?: string;
10
+ statuses?: string[];
11
+ orderDependencies: boolean;
12
+ json: boolean;
13
+ verbose: boolean;
14
+ }
15
+
16
+ const usage = `mcoda backlog \\
17
+ [--workspace-root <PATH>] \\
18
+ [--project <PROJECT_KEY>] \\
19
+ [--epic <EPIC_KEY>] \\
20
+ [--story <STORY_KEY>] \\
21
+ [--assignee <USER>] \\
22
+ [--status <STATUS_FILTER>] \\
23
+ [--order dependencies] # dependency-aware ordering (topological, most depended-on first) \\
24
+ [--json] \\
25
+ [--verbose]`;
26
+
27
+ const parseStatuses = (value: string | undefined): string[] | undefined => {
28
+ if (!value) return undefined;
29
+ return value
30
+ .split(",")
31
+ .map((s) => s.trim())
32
+ .filter(Boolean)
33
+ .map((s) => s.toLowerCase());
34
+ };
35
+
36
+ export const parseBacklogArgs = (argv: string[]): ParsedArgs => {
37
+ const args: ParsedArgs = {
38
+ orderDependencies: false,
39
+ json: false,
40
+ verbose: false,
41
+ };
42
+
43
+ for (let i = 0; i < argv.length; i += 1) {
44
+ const arg = argv[i];
45
+ if (!arg.startsWith("-")) continue;
46
+ switch (arg) {
47
+ case "--workspace-root":
48
+ args.workspaceRoot = argv[i + 1] ? path.resolve(argv[i + 1]) : undefined;
49
+ i += 1;
50
+ break;
51
+ case "--project":
52
+ args.project = argv[i + 1];
53
+ i += 1;
54
+ break;
55
+ case "--epic":
56
+ args.epic = argv[i + 1];
57
+ i += 1;
58
+ break;
59
+ case "--story":
60
+ args.story = argv[i + 1];
61
+ i += 1;
62
+ break;
63
+ case "--assignee":
64
+ args.assignee = argv[i + 1];
65
+ i += 1;
66
+ break;
67
+ case "--status": {
68
+ const value = argv[i + 1];
69
+ if (value && !value.startsWith("-")) {
70
+ args.statuses = parseStatuses(value);
71
+ i += 1;
72
+ }
73
+ break;
74
+ }
75
+ default:
76
+ if (arg.startsWith("--project=")) {
77
+ args.project = arg.split("=")[1];
78
+ } else if (arg.startsWith("--epic=")) {
79
+ args.epic = arg.split("=")[1];
80
+ } else if (arg.startsWith("--story=")) {
81
+ args.story = arg.split("=")[1];
82
+ } else if (arg.startsWith("--assignee=")) {
83
+ args.assignee = arg.split("=")[1];
84
+ } else if (arg.startsWith("--status=")) {
85
+ args.statuses = parseStatuses(arg.split("=")[1]);
86
+ } else if (arg === "--order") {
87
+ const value = argv[i + 1];
88
+ if (value === "dependencies") {
89
+ args.orderDependencies = true;
90
+ i += 1;
91
+ }
92
+ } else if (arg === "--order=dependencies") {
93
+ args.orderDependencies = true;
94
+ } else if (arg === "--json") {
95
+ args.json = true;
96
+ } else if (arg === "--verbose") {
97
+ args.verbose = true;
98
+ } else if (arg === "--help" || arg === "-h") {
99
+ // eslint-disable-next-line no-console
100
+ console.log(usage);
101
+ process.exit(0);
102
+ }
103
+ break;
104
+ }
105
+ }
106
+
107
+ return args;
108
+ };
109
+
110
+ const pad = (value: string, width: number): string => value.padEnd(width, " ");
111
+
112
+ const formatTable = (headers: string[], rows: string[][]): string => {
113
+ const widths = headers.map((header, idx) => {
114
+ return Math.max(header.length, ...rows.map((row) => (row[idx] ?? "").length));
115
+ });
116
+ const headerLine = headers.map((h, idx) => pad(h, widths[idx])).join(" | ");
117
+ const sepLine = widths.map((w) => "-".repeat(w)).join("-+-");
118
+ const body = rows.map((row) => row.map((cell, idx) => pad(cell ?? "", widths[idx])).join(" | ")).join("\n");
119
+ return [headerLine, sepLine, body].filter(Boolean).join("\n");
120
+ };
121
+
122
+ const truncate = (value: string | undefined, max = 100): string => {
123
+ if (!value) return "";
124
+ if (value.length <= max) return value;
125
+ return `${value.slice(0, max - 3)}...`;
126
+ };
127
+
128
+ const formatNumber = (value: number | null | undefined): string => {
129
+ if (value === null || value === undefined) return "-";
130
+ if (Number.isInteger(value)) return `${value}`;
131
+ return value.toFixed(2);
132
+ };
133
+
134
+ const renderSummary = (summary: BacklogSummary): void => {
135
+ const rows = [
136
+ ["Implementation", `${summary.totals.implementation.tasks}`, formatNumber(summary.totals.implementation.story_points)],
137
+ ["Review", `${summary.totals.review.tasks}`, formatNumber(summary.totals.review.story_points)],
138
+ ["QA", `${summary.totals.qa.tasks}`, formatNumber(summary.totals.qa.story_points)],
139
+ ["Done", `${summary.totals.done.tasks}`, formatNumber(summary.totals.done.story_points)],
140
+ ];
141
+ // eslint-disable-next-line no-console
142
+ console.log("Summary (tasks / SP):");
143
+ // eslint-disable-next-line no-console
144
+ console.log(formatTable(["LANE", "TASKS", "SP"], rows));
145
+ };
146
+
147
+ const renderEpics = (summary: BacklogSummary): void => {
148
+ if (summary.epics.length === 0) {
149
+ // eslint-disable-next-line no-console
150
+ console.log("\nEpics: none");
151
+ return;
152
+ }
153
+ const rows = summary.epics.map((epic) => [
154
+ epic.epic_key,
155
+ truncate(epic.title, 40),
156
+ formatNumber(epic.priority),
157
+ formatNumber(epic.totals.implementation.story_points),
158
+ formatNumber(epic.totals.review.story_points),
159
+ formatNumber(epic.totals.qa.story_points),
160
+ formatNumber(epic.totals.done.story_points),
161
+ `${epic.totals.implementation.tasks + epic.totals.review.tasks + epic.totals.qa.tasks + epic.totals.done.tasks}`,
162
+ truncate(epic.description, 100),
163
+ ]);
164
+ // eslint-disable-next-line no-console
165
+ console.log("\nEpics:");
166
+ // eslint-disable-next-line no-console
167
+ console.log(
168
+ formatTable(
169
+ ["EPIC_KEY", "TITLE", "PRIORITY", "IMPL_SP", "REVIEW_SP", "QA_SP", "DONE_SP", "TASKS", "DESC"],
170
+ rows,
171
+ ),
172
+ );
173
+ };
174
+
175
+ const renderStories = (summary: BacklogSummary): void => {
176
+ const stories = summary.epics.flatMap((epic) => epic.stories);
177
+ if (stories.length === 0) {
178
+ // eslint-disable-next-line no-console
179
+ console.log("\nStories: none");
180
+ return;
181
+ }
182
+ const rows = stories.map((story) => [
183
+ story.user_story_key,
184
+ story.epic_key,
185
+ truncate(story.title, 40),
186
+ formatNumber(story.priority),
187
+ story.status ?? "-",
188
+ formatNumber(story.totals.implementation.story_points),
189
+ formatNumber(story.totals.review.story_points),
190
+ formatNumber(story.totals.qa.story_points),
191
+ formatNumber(story.totals.done.story_points),
192
+ `${story.totals.implementation.tasks + story.totals.review.tasks + story.totals.qa.tasks + story.totals.done.tasks}`,
193
+ truncate(story.description, 100),
194
+ ]);
195
+ // eslint-disable-next-line no-console
196
+ console.log("\nStories:");
197
+ // eslint-disable-next-line no-console
198
+ console.log(
199
+ formatTable(
200
+ ["STORY_KEY", "EPIC_KEY", "TITLE", "PRIORITY", "STATUS", "IMPL_SP", "REVIEW_SP", "QA_SP", "DONE_SP", "TASKS", "DESC"],
201
+ rows,
202
+ ),
203
+ );
204
+ };
205
+
206
+ const renderTasks = (summary: BacklogSummary): void => {
207
+ if (summary.tasks.length === 0) {
208
+ // eslint-disable-next-line no-console
209
+ console.log("\nTasks: none");
210
+ return;
211
+ }
212
+ const rows = summary.tasks.map((task) => [
213
+ task.task_key,
214
+ task.epic_key,
215
+ task.user_story_key,
216
+ task.status,
217
+ formatNumber(task.story_points),
218
+ formatNumber(task.priority),
219
+ task.assignee ?? "-",
220
+ truncate(task.dependency_keys.join(", "), 80),
221
+ truncate(task.description, 100),
222
+ ]);
223
+ // eslint-disable-next-line no-console
224
+ console.log("\nTasks:");
225
+ // eslint-disable-next-line no-console
226
+ console.log(
227
+ formatTable(
228
+ ["TASK_KEY", "EPIC_KEY", "STORY_KEY", "STATUS", "SP", "PRIORITY", "ASSIGNEE", "DEPENDS_ON", "DESC"],
229
+ rows,
230
+ ),
231
+ );
232
+ };
233
+
234
+ const renderBacklog = (summary: BacklogSummary): void => {
235
+ renderSummary(summary);
236
+ renderEpics(summary);
237
+ renderStories(summary);
238
+ renderTasks(summary);
239
+ };
240
+
241
+ export class BacklogCommands {
242
+ static async run(argv: string[]): Promise<void> {
243
+ const parsed = parseBacklogArgs(argv);
244
+ const workspace = await WorkspaceResolver.resolveWorkspace({
245
+ cwd: process.cwd(),
246
+ explicitWorkspace: parsed.workspaceRoot,
247
+ });
248
+
249
+ let service: BacklogService | undefined;
250
+ try {
251
+ service = await BacklogService.create(workspace);
252
+ const { summary, warnings } = await service.getBacklog({
253
+ projectKey: parsed.project,
254
+ epicKey: parsed.epic,
255
+ storyKey: parsed.story,
256
+ assignee: parsed.assignee,
257
+ statuses: parsed.statuses,
258
+ orderByDependencies: parsed.orderDependencies,
259
+ verbose: parsed.verbose,
260
+ });
261
+
262
+ if (parsed.json) {
263
+ // eslint-disable-next-line no-console
264
+ console.log(JSON.stringify(summary, null, 2));
265
+ } else {
266
+ renderBacklog(summary);
267
+ if (parsed.verbose && warnings.length > 0) {
268
+ // eslint-disable-next-line no-console
269
+ console.error("\nWarnings:");
270
+ for (const warning of warnings) {
271
+ // eslint-disable-next-line no-console
272
+ console.error(`- ${warning}`);
273
+ }
274
+ }
275
+ }
276
+ } catch (error) {
277
+ // eslint-disable-next-line no-console
278
+ console.error(`backlog failed: ${(error as Error).message}`);
279
+ process.exitCode = 1;
280
+ } finally {
281
+ if (service) {
282
+ await service.close();
283
+ }
284
+ }
285
+ }
286
+ }
@@ -0,0 +1,237 @@
1
+ import path from "node:path";
2
+ import { TaskOrderingService, WorkspaceResolver } from "@mcoda/core";
3
+
4
+ interface ParsedArgs {
5
+ workspaceRoot?: string;
6
+ project?: string;
7
+ epic?: string;
8
+ status?: string[];
9
+ includeBlocked: boolean;
10
+ agentName?: string;
11
+ agentStream?: boolean;
12
+ json: boolean;
13
+ }
14
+
15
+ const usage = `mcoda order-tasks \\
16
+ [--workspace-root <PATH>] \\
17
+ --project <PROJECT_KEY> \\
18
+ [--epic <EPIC_KEY>] \\
19
+ [--status <STATUS_FILTER>] \\
20
+ [--include-blocked] \\
21
+ [--agent <NAME>] \\
22
+ [--agent-stream <true|false>] \\
23
+ [--json]`;
24
+
25
+ const parseBooleanFlag = (value: string | undefined, defaultValue: boolean): boolean => {
26
+ if (value === undefined) return defaultValue;
27
+ const normalized = value.toLowerCase();
28
+ if (["false", "0", "no"].includes(normalized)) return false;
29
+ if (["true", "1", "yes"].includes(normalized)) return true;
30
+ return defaultValue;
31
+ };
32
+
33
+ const parseStatuses = (value?: string): string[] | undefined => {
34
+ if (!value) return undefined;
35
+ const parts = value
36
+ .split(",")
37
+ .map((s) => s.trim())
38
+ .filter(Boolean);
39
+ return parts.length ? parts : undefined;
40
+ };
41
+
42
+ export const parseOrderTasksArgs = (argv: string[]): ParsedArgs => {
43
+ const parsed: ParsedArgs = {
44
+ includeBlocked: false,
45
+ json: false,
46
+ };
47
+ for (let i = 0; i < argv.length; i += 1) {
48
+ const arg = argv[i];
49
+ if (arg.startsWith("--status=")) {
50
+ parsed.status = parseStatuses(arg.split("=")[1]);
51
+ continue;
52
+ }
53
+ if (arg.startsWith("--agent-stream=")) {
54
+ parsed.agentStream = parseBooleanFlag(arg.split("=")[1], true);
55
+ continue;
56
+ }
57
+ switch (arg) {
58
+ case "--workspace-root":
59
+ parsed.workspaceRoot = argv[i + 1] ? path.resolve(argv[i + 1]) : undefined;
60
+ i += 1;
61
+ break;
62
+ case "--project":
63
+ parsed.project = argv[i + 1];
64
+ i += 1;
65
+ break;
66
+ case "--epic":
67
+ parsed.epic = argv[i + 1];
68
+ i += 1;
69
+ break;
70
+ case "--status":
71
+ parsed.status = parseStatuses(argv[i + 1]);
72
+ i += 1;
73
+ break;
74
+ case "--include-blocked":
75
+ parsed.includeBlocked = true;
76
+ break;
77
+ case "--agent":
78
+ parsed.agentName = argv[i + 1];
79
+ i += 1;
80
+ break;
81
+ case "--agent-stream": {
82
+ const next = argv[i + 1];
83
+ if (next && !next.startsWith("-")) {
84
+ parsed.agentStream = parseBooleanFlag(next, true);
85
+ i += 1;
86
+ } else {
87
+ parsed.agentStream = true;
88
+ }
89
+ break;
90
+ }
91
+ case "--json":
92
+ parsed.json = true;
93
+ break;
94
+ case "--help":
95
+ case "-h":
96
+ // eslint-disable-next-line no-console
97
+ console.log(usage);
98
+ process.exit(0);
99
+ break;
100
+ default:
101
+ if (arg.startsWith("--project=")) {
102
+ parsed.project = arg.split("=")[1];
103
+ } else if (arg.startsWith("--epic=")) {
104
+ parsed.epic = arg.split("=")[1];
105
+ } else if (arg === "--include-blocked=true") {
106
+ parsed.includeBlocked = true;
107
+ } else if (arg === "--json=true") {
108
+ parsed.json = true;
109
+ }
110
+ break;
111
+ }
112
+ }
113
+ return parsed;
114
+ };
115
+
116
+ const pad = (value: string, width: number): string => value.padEnd(width, " ");
117
+
118
+ const formatTable = (headers: string[], rows: string[][]): string => {
119
+ const widths = headers.map((header, idx) => Math.max(header.length, ...rows.map((row) => (row[idx] ?? "").length)));
120
+ const headerLine = headers.map((h, idx) => pad(h, widths[idx])).join(" | ");
121
+ const sepLine = widths.map((w) => "-".repeat(w)).join("-+-");
122
+ const body = rows.map((row) => row.map((cell, idx) => pad(cell ?? "", widths[idx])).join(" | ")).join("\n");
123
+ return [headerLine, sepLine, body].filter(Boolean).join("\n");
124
+ };
125
+
126
+ const formatImpact = (value?: { direct: number; total: number }): string => {
127
+ if (!value) return "-";
128
+ return `${value.direct}/${value.total}`;
129
+ };
130
+
131
+ const renderOrder = (
132
+ ordered: Awaited<ReturnType<TaskOrderingService["orderTasks"]>>["ordered"],
133
+ blocked: Awaited<ReturnType<TaskOrderingService["orderTasks"]>>["blocked"],
134
+ includeBlocked: boolean,
135
+ warnings: string[],
136
+ ): void => {
137
+ const rows = ordered.map((task) => [
138
+ `${task.priority}`,
139
+ task.taskKey,
140
+ task.status,
141
+ task.storyPoints === null || task.storyPoints === undefined ? "-" : `${task.storyPoints}`,
142
+ task.epicKey,
143
+ task.storyKey,
144
+ formatImpact(task.dependencyImpact),
145
+ task.blocked ? "yes" : "",
146
+ task.title.length > 60 ? `${task.title.slice(0, 57)}...` : task.title,
147
+ ]);
148
+
149
+ // eslint-disable-next-line no-console
150
+ console.log(
151
+ formatTable(
152
+ ["PRIORITY", "TASK", "STATUS", "SP", "EPIC", "STORY", "DEP_IMPACT", "BLOCKED", "TITLE"],
153
+ rows,
154
+ ),
155
+ );
156
+
157
+ if (!includeBlocked && blocked.length > 0) {
158
+ // eslint-disable-next-line no-console
159
+ console.log("\nBlocked tasks (excluded from ordering):");
160
+ const blockedRows = blocked.map((task) => [
161
+ `${task.priority}`,
162
+ task.taskKey,
163
+ task.status,
164
+ task.storyPoints === null || task.storyPoints === undefined ? "-" : `${task.storyPoints}`,
165
+ task.epicKey,
166
+ task.storyKey,
167
+ formatImpact(task.dependencyImpact),
168
+ (task.blockedBy ?? []).join(", "),
169
+ task.title.length > 60 ? `${task.title.slice(0, 57)}...` : task.title,
170
+ ]);
171
+ // eslint-disable-next-line no-console
172
+ console.log(
173
+ formatTable(
174
+ ["PRIORITY", "TASK", "STATUS", "SP", "EPIC", "STORY", "DEP_IMPACT", "BLOCKED_BY", "TITLE"],
175
+ blockedRows,
176
+ ),
177
+ );
178
+ }
179
+
180
+ if (warnings.length > 0) {
181
+ // eslint-disable-next-line no-console
182
+ console.warn("\nWarnings:");
183
+ warnings.forEach((w) => {
184
+ // eslint-disable-next-line no-console
185
+ console.warn(`- ${w}`);
186
+ });
187
+ }
188
+ };
189
+
190
+ export class OrderTasksCommand {
191
+ static async run(argv: string[]): Promise<void> {
192
+ const parsed = parseOrderTasksArgs(argv);
193
+ const workspace = await WorkspaceResolver.resolveWorkspace({
194
+ cwd: process.cwd(),
195
+ explicitWorkspace: parsed.workspaceRoot,
196
+ });
197
+ if (!parsed.project) {
198
+ // eslint-disable-next-line no-console
199
+ console.error("order-tasks requires --project <PROJECT_KEY>");
200
+ process.exitCode = 1;
201
+ return;
202
+ }
203
+ const service = await TaskOrderingService.create(workspace);
204
+ try {
205
+ const result = await service.orderTasks({
206
+ projectKey: parsed.project,
207
+ epicKey: parsed.epic,
208
+ statusFilter: parsed.status,
209
+ includeBlocked: parsed.includeBlocked,
210
+ agentName: parsed.agentName,
211
+ agentStream: parsed.agentStream,
212
+ });
213
+ if (parsed.json) {
214
+ const payload: Record<string, unknown> = {
215
+ order: parsed.includeBlocked ? result.ordered : result.ordered.filter((t) => !t.blocked),
216
+ };
217
+ if (result.warnings.length > 0) {
218
+ payload.warnings = result.warnings;
219
+ }
220
+ // eslint-disable-next-line no-console
221
+ console.log(JSON.stringify(payload, null, 2));
222
+ return;
223
+ }
224
+ renderOrder(result.ordered, result.blocked, parsed.includeBlocked, result.warnings);
225
+ } catch (error) {
226
+ // eslint-disable-next-line no-console
227
+ console.error(`order-tasks failed: ${(error as Error).message}`);
228
+ process.exitCode = 1;
229
+ } finally {
230
+ try {
231
+ await service.close();
232
+ } catch {
233
+ // ignore close errors (e.g., database already closed)
234
+ }
235
+ }
236
+ }
237
+ }