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,350 @@
1
+ import { describe, it } from "node:test";
2
+ import assert from "node:assert/strict";
3
+ import os from "node:os";
4
+ import path from "node:path";
5
+ import { promises as fs } from "node:fs";
6
+ import { Agent } from "@mcoda/shared";
7
+ import { DocdexClient } from "@mcoda/integrations";
8
+ import { DocsService } from "../DocsService.js";
9
+ import { JobService } from "../../jobs/JobService.js";
10
+ import { WorkspaceResolution } from "../../../workspace/WorkspaceManager.js";
11
+
12
+ // Disable sqlite usage in tests to avoid FK constraints from incomplete fixtures.
13
+ process.env.MCODA_DISABLE_DB = "1";
14
+
15
+ class FakeAgentService {
16
+ constructor(private agent: Agent, private response?: string) {}
17
+
18
+ async close(): Promise<void> {
19
+ // no-op
20
+ }
21
+
22
+ async resolveAgent(): Promise<Agent> {
23
+ return this.agent;
24
+ }
25
+
26
+ async getCapabilities(): Promise<string[]> {
27
+ return ["docdex_query", "doc_generation"];
28
+ }
29
+
30
+ async getPrompts() {
31
+ return { jobPrompt: "# Job\nFollow runbook", characterPrompt: "character" };
32
+ }
33
+
34
+ async invoke(_agentId: string, request: any) {
35
+ const jobId = request.metadata?.jobId ?? "job";
36
+ const output =
37
+ this.response ??
38
+ `# Product Design Review\n\n## Introduction\nGenerated draft for ${jobId}\n\n## Scope\nscope\n\n## Requirements & Constraints\n- goal\n\n## Architecture Overview\narch\n\n## Interfaces / APIs\napi\n\n## Non-Functional Requirements\nnfr\n\n## Risks & Mitigations\nrisk\n\n## Open Questions\nq`;
39
+ return {
40
+ output,
41
+ adapter: "fake",
42
+ metadata: { request },
43
+ };
44
+ }
45
+
46
+ async invokeStream() {
47
+ async function* generator() {
48
+ yield { output: "streamed-output", adapter: "fake" };
49
+ }
50
+ return generator();
51
+ }
52
+ }
53
+
54
+ class FakeRepo {
55
+ constructor(private agent: Agent) {}
56
+
57
+ async close(): Promise<void> {
58
+ // no-op
59
+ }
60
+
61
+ async getWorkspaceDefaults(): Promise<any[]> {
62
+ return [];
63
+ }
64
+
65
+ async listAgents(): Promise<Agent[]> {
66
+ return [this.agent];
67
+ }
68
+
69
+ async getAgentCapabilities(): Promise<string[]> {
70
+ return ["docdex_query", "doc_generation"];
71
+ }
72
+ }
73
+
74
+ class FakeRoutingService {
75
+ constructor(private agent: Agent) {}
76
+
77
+ async resolveAgentForCommand(params: { commandName: string; overrideAgentSlug?: string }): Promise<any> {
78
+ return {
79
+ agent: this.agent,
80
+ agentId: this.agent.id,
81
+ agentSlug: this.agent.slug,
82
+ model: this.agent.defaultModel,
83
+ capabilities: ["docdex_query", "doc_generation"],
84
+ healthStatus: "healthy",
85
+ source: params.overrideAgentSlug ? "override" : "workspace_default",
86
+ routingPreview: { workspaceId: "ws", commandName: params.commandName } as any,
87
+ };
88
+ }
89
+ }
90
+
91
+ class FailingDocdex {
92
+ async fetchDocumentById(): Promise<never> {
93
+ throw new Error("docdex down");
94
+ }
95
+ async ensureRegisteredFromFile(): Promise<never> {
96
+ throw new Error("docdex down");
97
+ }
98
+ async search(): Promise<any[]> {
99
+ throw new Error("docdex down");
100
+ }
101
+ }
102
+
103
+ describe("DocsService.generatePdr", () => {
104
+ it("writes a PDR and records job + telemetry artifacts", async () => {
105
+ process.env.MCODA_SKIP_PDR_VALIDATION = "1";
106
+ const workspaceRoot = await fs.mkdtemp(path.join(os.tmpdir(), "mcoda-workspace-"));
107
+ const workspace: WorkspaceResolution = {
108
+ workspaceRoot,
109
+ workspaceId: workspaceRoot,
110
+ mcodaDir: path.join(workspaceRoot, ".mcoda"),
111
+ id: workspaceRoot,
112
+ legacyWorkspaceIds: [],
113
+ workspaceDbPath: path.join(workspaceRoot, ".mcoda", "mcoda.db"),
114
+ globalDbPath: path.join(os.homedir(), ".mcoda", "mcoda.db"),
115
+ };
116
+ const agent: Agent = {
117
+ id: "agent-1",
118
+ slug: "fake",
119
+ adapter: "codex-api",
120
+ createdAt: new Date().toISOString(),
121
+ updatedAt: new Date().toISOString(),
122
+ };
123
+ const agentService = new FakeAgentService(agent);
124
+ const repo = new FakeRepo(agent);
125
+ const routingService = new FakeRoutingService(agent);
126
+ const jobService = new JobService(workspace);
127
+ const docdex = new DocdexClient({ workspaceRoot });
128
+ const service = new DocsService(workspace, {
129
+ agentService: agentService as any,
130
+ repo: repo as any,
131
+ routingService: routingService as any,
132
+ jobService,
133
+ docdex,
134
+ });
135
+
136
+ const rfpPath = path.join(workspaceRoot, "rfp.md");
137
+ await fs.writeFile(rfpPath, "- goal one\n- goal two\n", "utf8");
138
+
139
+ const result = await service.generatePdr({
140
+ workspace,
141
+ projectKey: "TEST",
142
+ rfpPath,
143
+ agentName: "fake",
144
+ agentStream: false,
145
+ dryRun: false,
146
+ json: false,
147
+ });
148
+
149
+ assert.ok(result.outputPath);
150
+ const content = await fs.readFile(result.outputPath ?? "", "utf8");
151
+ assert.match(content, /Product Design Review/);
152
+ const manifestPath = path.join(workspace.mcodaDir, "jobs", result.jobId, "manifest.json");
153
+ const manifest = JSON.parse(await fs.readFile(manifestPath, "utf8"));
154
+ assert.equal(manifest.state ?? manifest.status, "completed");
155
+
156
+ const tokenPath = path.join(workspace.mcodaDir, "token_usage.json");
157
+ const tokenUsage = JSON.parse(await fs.readFile(tokenPath, "utf8"));
158
+ assert.ok(Array.isArray(tokenUsage));
159
+ assert.ok(tokenUsage.length > 0);
160
+
161
+ await service.close();
162
+ });
163
+
164
+ it("supports dry-run without writing files", async () => {
165
+ process.env.MCODA_SKIP_PDR_VALIDATION = "1";
166
+ const workspaceRoot = await fs.mkdtemp(path.join(os.tmpdir(), "mcoda-workspace-"));
167
+ const workspace: WorkspaceResolution = {
168
+ workspaceRoot,
169
+ workspaceId: workspaceRoot,
170
+ mcodaDir: path.join(workspaceRoot, ".mcoda"),
171
+ id: workspaceRoot,
172
+ legacyWorkspaceIds: [],
173
+ workspaceDbPath: path.join(workspaceRoot, ".mcoda", "mcoda.db"),
174
+ globalDbPath: path.join(os.homedir(), ".mcoda", "mcoda.db"),
175
+ };
176
+ const agent: Agent = {
177
+ id: "agent-2",
178
+ slug: "fake",
179
+ adapter: "codex-api",
180
+ createdAt: new Date().toISOString(),
181
+ updatedAt: new Date().toISOString(),
182
+ };
183
+ const agentService = new FakeAgentService(agent);
184
+ const repo = new FakeRepo(agent);
185
+ const routingService = new FakeRoutingService(agent);
186
+ const jobService = new JobService(workspace);
187
+ const docdex = new DocdexClient({ workspaceRoot });
188
+ const service = new DocsService(workspace, {
189
+ agentService: agentService as any,
190
+ repo: repo as any,
191
+ routingService: routingService as any,
192
+ jobService,
193
+ docdex,
194
+ });
195
+ const rfpPath = path.join(workspaceRoot, "rfp.md");
196
+ await fs.writeFile(rfpPath, "- goal one\n", "utf8");
197
+
198
+ const result = await service.generatePdr({
199
+ workspace,
200
+ projectKey: "DRY",
201
+ rfpPath,
202
+ agentName: "fake",
203
+ agentStream: false,
204
+ dryRun: true,
205
+ json: true,
206
+ });
207
+
208
+ assert.ok(result.warnings.some((w) => w.toLowerCase().includes("dry run")));
209
+ const outputPath = result.outputPath ?? path.join(workspace.mcodaDir, "docs", "pdr");
210
+ await assert.rejects(fs.access(outputPath));
211
+ await service.close();
212
+ });
213
+
214
+ it("emits degraded warning when docdex unavailable", async () => {
215
+ process.env.MCODA_SKIP_PDR_VALIDATION = "1";
216
+ const workspaceRoot = await fs.mkdtemp(path.join(os.tmpdir(), "mcoda-workspace-"));
217
+ const workspace: WorkspaceResolution = {
218
+ workspaceRoot,
219
+ workspaceId: workspaceRoot,
220
+ mcodaDir: path.join(workspaceRoot, ".mcoda"),
221
+ id: workspaceRoot,
222
+ legacyWorkspaceIds: [],
223
+ workspaceDbPath: path.join(workspaceRoot, ".mcoda", "mcoda.db"),
224
+ globalDbPath: path.join(os.homedir(), ".mcoda", "mcoda.db"),
225
+ };
226
+ const agent: Agent = {
227
+ id: "agent-3",
228
+ slug: "fake",
229
+ adapter: "codex-api",
230
+ createdAt: new Date().toISOString(),
231
+ updatedAt: new Date().toISOString(),
232
+ };
233
+ const agentService = new FakeAgentService(agent);
234
+ const repo = new FakeRepo(agent);
235
+ const routingService = new FakeRoutingService(agent);
236
+ const jobService = new JobService(workspace);
237
+ const service = new DocsService(workspace, {
238
+ agentService: agentService as any,
239
+ repo: repo as any,
240
+ routingService: routingService as any,
241
+ jobService,
242
+ docdex: new FailingDocdex() as any,
243
+ });
244
+ const rfpPath = path.join(workspaceRoot, "rfp.md");
245
+ await fs.writeFile(rfpPath, "- goal\n", "utf8");
246
+
247
+ const result = await service.generatePdr({
248
+ workspace,
249
+ projectKey: "DOWN",
250
+ rfpPath,
251
+ agentName: "fake",
252
+ agentStream: false,
253
+ dryRun: true,
254
+ json: false,
255
+ });
256
+ assert.ok(result.warnings.some((w) => w.toLowerCase().includes("docdex")));
257
+ await service.close();
258
+ });
259
+
260
+ it("writes an SDS and records job + telemetry artifacts", async () => {
261
+ process.env.MCODA_SKIP_SDS_VALIDATION = "1";
262
+ const workspaceRoot = await fs.mkdtemp(path.join(os.tmpdir(), "mcoda-workspace-"));
263
+ const workspace: WorkspaceResolution = {
264
+ workspaceRoot,
265
+ workspaceId: workspaceRoot,
266
+ mcodaDir: path.join(workspaceRoot, ".mcoda"),
267
+ id: workspaceRoot,
268
+ legacyWorkspaceIds: [],
269
+ workspaceDbPath: path.join(workspaceRoot, ".mcoda", "mcoda.db"),
270
+ globalDbPath: path.join(os.homedir(), ".mcoda", "mcoda.db"),
271
+ };
272
+ const agent: Agent = {
273
+ id: "agent-sds",
274
+ slug: "fake-sds",
275
+ adapter: "codex-api",
276
+ createdAt: new Date().toISOString(),
277
+ updatedAt: new Date().toISOString(),
278
+ };
279
+ const sdsDraft = [
280
+ "# Software Design Specification",
281
+ "## Introduction",
282
+ "intro",
283
+ "## Goals & Scope",
284
+ "scope",
285
+ "## Architecture Overview",
286
+ "arch",
287
+ "## Components & Responsibilities",
288
+ "components",
289
+ "## Data Model & Persistence",
290
+ "data",
291
+ "## Interfaces & Contracts",
292
+ "interfaces",
293
+ "## Non-Functional Requirements",
294
+ "nfr",
295
+ "## Security & Compliance",
296
+ "security",
297
+ "## Failure Modes & Resilience",
298
+ "failures",
299
+ "## Risks & Mitigations",
300
+ "risks",
301
+ "## Assumptions",
302
+ "assumptions",
303
+ "## Open Questions",
304
+ "questions",
305
+ "## Acceptance Criteria",
306
+ "criteria",
307
+ ].join("\n");
308
+ const agentService = new FakeAgentService(agent, sdsDraft);
309
+ const repo = new FakeRepo(agent);
310
+ const routingService = new FakeRoutingService(agent);
311
+ const jobService = new JobService(workspace);
312
+ const docdex = new DocdexClient({ workspaceRoot });
313
+ await docdex.registerDocument({
314
+ docType: "PDR",
315
+ path: path.join(workspaceRoot, "pdr.md"),
316
+ content: "- goal from pdr",
317
+ metadata: { projectKey: "SDS" },
318
+ });
319
+ const service = new DocsService(workspace, {
320
+ agentService: agentService as any,
321
+ repo: repo as any,
322
+ routingService: routingService as any,
323
+ jobService,
324
+ docdex,
325
+ });
326
+
327
+ const result = await service.generateSds({
328
+ workspace,
329
+ projectKey: "SDS",
330
+ agentName: "fake-sds",
331
+ agentStream: false,
332
+ dryRun: false,
333
+ json: false,
334
+ force: true,
335
+ });
336
+
337
+ assert.ok(result.outputPath);
338
+ const content = await fs.readFile(result.outputPath ?? "", "utf8");
339
+ assert.match(content, /Software Design Specification/);
340
+ const manifestPath = path.join(workspace.mcodaDir, "jobs", result.jobId, "manifest.json");
341
+ const manifest = JSON.parse(await fs.readFile(manifestPath, "utf8"));
342
+ assert.equal(manifest.type, "sds_generate");
343
+
344
+ const tokenPath = path.join(workspace.mcodaDir, "token_usage.json");
345
+ const tokenUsage = JSON.parse(await fs.readFile(tokenPath, "utf8"));
346
+ assert.ok(tokenUsage.some((entry: any) => entry.commandName === "docs-sds-generate"));
347
+
348
+ await service.close();
349
+ });
350
+ });
@@ -0,0 +1,111 @@
1
+ import { BacklogService } from "../backlog/BacklogService.js";
2
+ import type { BacklogTotals } from "../backlog/BacklogService.js";
3
+ import { VelocityService } from "./VelocityService.js";
4
+ import type { EstimateDurations, EstimateEtas, EstimateOptions, EstimateResult } from "./types.js";
5
+ import type { EffectiveVelocity } from "./types.js";
6
+ import type { WorkspaceResolution } from "../../workspace/WorkspaceManager.js";
7
+
8
+ const HOURS_IN_MS = 3600 * 1000;
9
+
10
+ export class EstimateService {
11
+ private constructor(private workspace: WorkspaceResolution) {}
12
+
13
+ static async create(workspace: WorkspaceResolution): Promise<EstimateService> {
14
+ return new EstimateService(workspace);
15
+ }
16
+
17
+ private computeDurations(totals: BacklogTotals, velocity: EffectiveVelocity): EstimateDurations {
18
+ const safeDivide = (sp: number, spPerHour: number): number | null => {
19
+ if (!sp || sp <= 0) return 0;
20
+ if (!spPerHour || spPerHour <= 0) return null;
21
+ return sp / spPerHour;
22
+ };
23
+ const implementationHours = safeDivide(totals.implementation.story_points, velocity.implementationSpPerHour);
24
+ const reviewHours = safeDivide(totals.review.story_points, velocity.reviewSpPerHour);
25
+ const qaHours = safeDivide(totals.qa.story_points, velocity.qaSpPerHour);
26
+ const durationsList = [implementationHours, reviewHours, qaHours];
27
+ const hasNull = durationsList.some((value) => value === null);
28
+ const numeric = durationsList.filter((value) => value !== null) as number[];
29
+ const totalHours = hasNull || numeric.length === 0 ? null : Math.max(...numeric);
30
+ return {
31
+ implementationHours,
32
+ reviewHours,
33
+ qaHours,
34
+ totalHours,
35
+ };
36
+ }
37
+
38
+ private computeEtas(durations: EstimateDurations): EstimateEtas {
39
+ const now = Date.now();
40
+ const addHours = (hours: number | null | undefined): string | undefined => {
41
+ if (hours === null || hours === undefined || hours < 0) return undefined;
42
+ return new Date(now + hours * HOURS_IN_MS).toISOString();
43
+ };
44
+ const readyToReviewEta = durations.implementationHours !== null ? addHours(durations.implementationHours ?? undefined) : undefined;
45
+ const readyToQaEta =
46
+ durations.implementationHours !== null && durations.reviewHours !== null
47
+ ? addHours(Math.max(durations.implementationHours ?? 0, durations.reviewHours ?? 0))
48
+ : undefined;
49
+ const completeEta = durations.totalHours !== null ? addHours(durations.totalHours ?? undefined) : undefined;
50
+ return {
51
+ readyToReviewEta,
52
+ readyToQaEta,
53
+ completeEta,
54
+ };
55
+ }
56
+
57
+ async estimate(options: Omit<EstimateOptions, "workspace">): Promise<EstimateResult> {
58
+ const backlogService = await BacklogService.create(this.workspace);
59
+ let backlogTotals: BacklogTotals;
60
+ try {
61
+ const { summary } = await backlogService.getBacklog({
62
+ projectKey: options.projectKey,
63
+ epicKey: options.epicKey,
64
+ storyKey: options.storyKey,
65
+ assignee: options.assignee,
66
+ });
67
+ backlogTotals = summary.totals;
68
+ } finally {
69
+ await backlogService.close();
70
+ }
71
+
72
+ const velocityService = await VelocityService.create(this.workspace);
73
+ let effectiveVelocity: EffectiveVelocity;
74
+ try {
75
+ effectiveVelocity = await velocityService.getEffectiveVelocity({
76
+ projectKey: options.projectKey,
77
+ epicKey: options.epicKey,
78
+ storyKey: options.storyKey,
79
+ assignee: options.assignee,
80
+ mode: options.mode,
81
+ windowTasks: options.windowTasks,
82
+ spPerHourAll: options.spPerHourAll,
83
+ spPerHourReview: options.spPerHourReview,
84
+ spPerHourQa: options.spPerHourQa,
85
+ });
86
+ } finally {
87
+ await velocityService.close();
88
+ }
89
+
90
+ const durationsHours = this.computeDurations(backlogTotals, effectiveVelocity);
91
+ const etas = this.computeEtas(durationsHours);
92
+
93
+ return {
94
+ scope: {
95
+ workspaceId: this.workspace.workspaceId,
96
+ project: options.projectKey,
97
+ epic: options.epicKey,
98
+ story: options.storyKey,
99
+ assignee: options.assignee,
100
+ },
101
+ backlogTotals,
102
+ effectiveVelocity,
103
+ durationsHours,
104
+ etas,
105
+ };
106
+ }
107
+
108
+ async close(): Promise<void> {
109
+ // No long-lived resources to release yet, placeholder for symmetry with other services.
110
+ }
111
+ }