reviewflow 3.17.0 → 3.19.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 (189) hide show
  1. package/CHANGELOG.md +22 -0
  2. package/dist/config/projectConfig.d.ts +8 -0
  3. package/dist/config/projectConfig.d.ts.map +1 -1
  4. package/dist/config/projectConfig.js +59 -7
  5. package/dist/config/projectConfig.js.map +1 -1
  6. package/dist/dashboard/index.html +303 -175
  7. package/dist/dashboard/modules/budgetSettings.d.ts.map +1 -1
  8. package/dist/dashboard/modules/budgetSettings.js +2 -4
  9. package/dist/dashboard/modules/budgetSettings.js.map +1 -1
  10. package/dist/dashboard/modules/desktopNotifications.d.ts.map +1 -1
  11. package/dist/dashboard/modules/desktopNotifications.js +1 -0
  12. package/dist/dashboard/modules/desktopNotifications.js.map +1 -1
  13. package/dist/dashboard/modules/i18n.d.ts.map +1 -1
  14. package/dist/dashboard/modules/i18n.js +12 -0
  15. package/dist/dashboard/modules/i18n.js.map +1 -1
  16. package/dist/dashboard/modules/pendingReviews.d.ts +49 -0
  17. package/dist/dashboard/modules/pendingReviews.d.ts.map +1 -0
  18. package/dist/dashboard/modules/pendingReviews.js +80 -0
  19. package/dist/dashboard/modules/pendingReviews.js.map +1 -0
  20. package/dist/dashboard/styles.css +162 -8
  21. package/dist/frameworks/config/configLoader.d.ts +2 -0
  22. package/dist/frameworks/config/configLoader.d.ts.map +1 -1
  23. package/dist/frameworks/config/configLoader.js +27 -2
  24. package/dist/frameworks/config/configLoader.js.map +1 -1
  25. package/dist/main/routes.d.ts.map +1 -1
  26. package/dist/main/routes.js +57 -2
  27. package/dist/main/routes.js.map +1 -1
  28. package/dist/main/websocket.d.ts +1 -0
  29. package/dist/main/websocket.d.ts.map +1 -1
  30. package/dist/main/websocket.js +11 -0
  31. package/dist/main/websocket.js.map +1 -1
  32. package/dist/modules/cli-configuration/interface-adapters/controllers/http/projectConfig.routes.d.ts.map +1 -1
  33. package/dist/modules/cli-configuration/interface-adapters/controllers/http/projectConfig.routes.js +24 -6
  34. package/dist/modules/cli-configuration/interface-adapters/controllers/http/projectConfig.routes.js.map +1 -1
  35. package/dist/modules/platform-integration/interface-adapters/controllers/webhook/github.controller.d.ts +2 -0
  36. package/dist/modules/platform-integration/interface-adapters/controllers/webhook/github.controller.d.ts.map +1 -1
  37. package/dist/modules/platform-integration/interface-adapters/controllers/webhook/github.controller.js +54 -6
  38. package/dist/modules/platform-integration/interface-adapters/controllers/webhook/github.controller.js.map +1 -1
  39. package/dist/modules/platform-integration/interface-adapters/controllers/webhook/gitlab.controller.d.ts +6 -0
  40. package/dist/modules/platform-integration/interface-adapters/controllers/webhook/gitlab.controller.d.ts.map +1 -1
  41. package/dist/modules/platform-integration/interface-adapters/controllers/webhook/gitlab.controller.js +76 -21
  42. package/dist/modules/platform-integration/interface-adapters/controllers/webhook/gitlab.controller.js.map +1 -1
  43. package/dist/modules/review-execution/entities/pendingReviewRequest/pendingReviewRequest.gateway.d.ts +8 -0
  44. package/dist/modules/review-execution/entities/pendingReviewRequest/pendingReviewRequest.gateway.d.ts.map +1 -0
  45. package/dist/modules/review-execution/entities/pendingReviewRequest/pendingReviewRequest.gateway.js +2 -0
  46. package/dist/modules/review-execution/entities/pendingReviewRequest/pendingReviewRequest.gateway.js.map +1 -0
  47. package/dist/modules/review-execution/entities/pendingReviewRequest/pendingReviewRequest.guard.d.ts +29 -0
  48. package/dist/modules/review-execution/entities/pendingReviewRequest/pendingReviewRequest.guard.d.ts.map +1 -0
  49. package/dist/modules/review-execution/entities/pendingReviewRequest/pendingReviewRequest.guard.js +4 -0
  50. package/dist/modules/review-execution/entities/pendingReviewRequest/pendingReviewRequest.guard.js.map +1 -0
  51. package/dist/modules/review-execution/entities/pendingReviewRequest/pendingReviewRequest.schema.d.ts +94 -0
  52. package/dist/modules/review-execution/entities/pendingReviewRequest/pendingReviewRequest.schema.d.ts.map +1 -0
  53. package/dist/modules/review-execution/entities/pendingReviewRequest/pendingReviewRequest.schema.js +40 -0
  54. package/dist/modules/review-execution/entities/pendingReviewRequest/pendingReviewRequest.schema.js.map +1 -0
  55. package/dist/modules/review-execution/entities/progress/agentDefinition.type.d.ts +4 -0
  56. package/dist/modules/review-execution/entities/progress/agentDefinition.type.d.ts.map +1 -1
  57. package/dist/modules/review-execution/entities/progress/agentDefinition.type.js +42 -0
  58. package/dist/modules/review-execution/entities/progress/agentDefinition.type.js.map +1 -1
  59. package/dist/modules/review-execution/entities/progress/reviewFocus.type.d.ts +8 -0
  60. package/dist/modules/review-execution/entities/progress/reviewFocus.type.d.ts.map +1 -0
  61. package/dist/modules/review-execution/entities/progress/reviewFocus.type.js +30 -0
  62. package/dist/modules/review-execution/entities/progress/reviewFocus.type.js.map +1 -0
  63. package/dist/modules/review-execution/interface-adapters/controllers/http/pendingReviews.routes.d.ts +13 -0
  64. package/dist/modules/review-execution/interface-adapters/controllers/http/pendingReviews.routes.d.ts.map +1 -0
  65. package/dist/modules/review-execution/interface-adapters/controllers/http/pendingReviews.routes.js +34 -0
  66. package/dist/modules/review-execution/interface-adapters/controllers/http/pendingReviews.routes.js.map +1 -0
  67. package/dist/modules/review-execution/interface-adapters/gateways/pendingReviewRequest.fileSystem.gateway.d.ts +16 -0
  68. package/dist/modules/review-execution/interface-adapters/gateways/pendingReviewRequest.fileSystem.gateway.d.ts.map +1 -0
  69. package/dist/modules/review-execution/interface-adapters/gateways/pendingReviewRequest.fileSystem.gateway.js +81 -0
  70. package/dist/modules/review-execution/interface-adapters/gateways/pendingReviewRequest.fileSystem.gateway.js.map +1 -0
  71. package/dist/modules/review-execution/interface-adapters/presenters/pendingReview.presenter.d.ts +25 -0
  72. package/dist/modules/review-execution/interface-adapters/presenters/pendingReview.presenter.d.ts.map +1 -0
  73. package/dist/modules/review-execution/interface-adapters/presenters/pendingReview.presenter.js +41 -0
  74. package/dist/modules/review-execution/interface-adapters/presenters/pendingReview.presenter.js.map +1 -0
  75. package/dist/modules/review-execution/services/processorRegistry.d.ts +15 -0
  76. package/dist/modules/review-execution/services/processorRegistry.d.ts.map +1 -0
  77. package/dist/modules/review-execution/services/processorRegistry.js +23 -0
  78. package/dist/modules/review-execution/services/processorRegistry.js.map +1 -0
  79. package/dist/modules/review-execution/usecases/confirmPendingReview.usecase.d.ts +32 -0
  80. package/dist/modules/review-execution/usecases/confirmPendingReview.usecase.d.ts.map +1 -0
  81. package/dist/modules/review-execution/usecases/confirmPendingReview.usecase.js +29 -0
  82. package/dist/modules/review-execution/usecases/confirmPendingReview.usecase.js.map +1 -0
  83. package/dist/modules/review-execution/usecases/dismissPendingReview.usecase.d.ts +26 -0
  84. package/dist/modules/review-execution/usecases/dismissPendingReview.usecase.d.ts.map +1 -0
  85. package/dist/modules/review-execution/usecases/dismissPendingReview.usecase.js +22 -0
  86. package/dist/modules/review-execution/usecases/dismissPendingReview.usecase.js.map +1 -0
  87. package/dist/modules/review-execution/usecases/gateClaudeInvocation.usecase.d.ts +36 -0
  88. package/dist/modules/review-execution/usecases/gateClaudeInvocation.usecase.d.ts.map +1 -0
  89. package/dist/modules/review-execution/usecases/gateClaudeInvocation.usecase.js +36 -0
  90. package/dist/modules/review-execution/usecases/gateClaudeInvocation.usecase.js.map +1 -0
  91. package/dist/modules/review-execution/usecases/listPendingReviews.usecase.d.ts +11 -0
  92. package/dist/modules/review-execution/usecases/listPendingReviews.usecase.d.ts.map +1 -0
  93. package/dist/modules/review-execution/usecases/listPendingReviews.usecase.js +10 -0
  94. package/dist/modules/review-execution/usecases/listPendingReviews.usecase.js.map +1 -0
  95. package/dist/modules/tracking/interface-adapters/controllers/http/mrTrackingAdvanced.routes.d.ts +2 -0
  96. package/dist/modules/tracking/interface-adapters/controllers/http/mrTrackingAdvanced.routes.d.ts.map +1 -1
  97. package/dist/modules/tracking/interface-adapters/controllers/http/mrTrackingAdvanced.routes.js +22 -4
  98. package/dist/modules/tracking/interface-adapters/controllers/http/mrTrackingAdvanced.routes.js.map +1 -1
  99. package/dist/modules/tracking/usecases/tracking/checkFollowupNeeded.usecase.d.ts.map +1 -1
  100. package/dist/modules/tracking/usecases/tracking/checkFollowupNeeded.usecase.js +1 -3
  101. package/dist/modules/tracking/usecases/tracking/checkFollowupNeeded.usecase.js.map +1 -1
  102. package/dist/tests/acceptance/174-semi-auto-review-trigger-mode.acceptance.test.d.ts +2 -0
  103. package/dist/tests/acceptance/174-semi-auto-review-trigger-mode.acceptance.test.d.ts.map +1 -0
  104. package/dist/tests/acceptance/174-semi-auto-review-trigger-mode.acceptance.test.js +161 -0
  105. package/dist/tests/acceptance/174-semi-auto-review-trigger-mode.acceptance.test.js.map +1 -0
  106. package/dist/tests/acceptance/46-github-followup-review-on-push.acceptance.test.js +1 -0
  107. package/dist/tests/acceptance/46-github-followup-review-on-push.acceptance.test.js.map +1 -1
  108. package/dist/tests/acceptance/reviewFocus.acceptance.test.d.ts +2 -0
  109. package/dist/tests/acceptance/reviewFocus.acceptance.test.d.ts.map +1 -0
  110. package/dist/tests/acceptance/reviewFocus.acceptance.test.js +114 -0
  111. package/dist/tests/acceptance/reviewFocus.acceptance.test.js.map +1 -0
  112. package/dist/tests/factories/config.factory.d.ts.map +1 -1
  113. package/dist/tests/factories/config.factory.js +1 -0
  114. package/dist/tests/factories/config.factory.js.map +1 -1
  115. package/dist/tests/factories/pendingReviewRequest.factory.d.ts +5 -0
  116. package/dist/tests/factories/pendingReviewRequest.factory.d.ts.map +1 -0
  117. package/dist/tests/factories/pendingReviewRequest.factory.js +25 -0
  118. package/dist/tests/factories/pendingReviewRequest.factory.js.map +1 -0
  119. package/dist/tests/factories/projectConfig.factory.d.ts +21 -0
  120. package/dist/tests/factories/projectConfig.factory.d.ts.map +1 -0
  121. package/dist/tests/factories/projectConfig.factory.js +43 -0
  122. package/dist/tests/factories/projectConfig.factory.js.map +1 -0
  123. package/dist/tests/stubs/pendingReviewRequest.stub.d.ts +14 -0
  124. package/dist/tests/stubs/pendingReviewRequest.stub.d.ts.map +1 -0
  125. package/dist/tests/stubs/pendingReviewRequest.stub.js +29 -0
  126. package/dist/tests/stubs/pendingReviewRequest.stub.js.map +1 -0
  127. package/dist/tests/units/config/projectConfig.test.js +101 -1
  128. package/dist/tests/units/config/projectConfig.test.js.map +1 -1
  129. package/dist/tests/units/dashboard/modules/pendingReviews.test.d.ts +2 -0
  130. package/dist/tests/units/dashboard/modules/pendingReviews.test.d.ts.map +1 -0
  131. package/dist/tests/units/dashboard/modules/pendingReviews.test.js +60 -0
  132. package/dist/tests/units/dashboard/modules/pendingReviews.test.js.map +1 -0
  133. package/dist/tests/units/frameworks/config/configLoader.test.js +85 -0
  134. package/dist/tests/units/frameworks/config/configLoader.test.js.map +1 -1
  135. package/dist/tests/units/interface-adapters/controllers/webhook/github.controller.test.js +1 -0
  136. package/dist/tests/units/interface-adapters/controllers/webhook/github.controller.test.js.map +1 -1
  137. package/dist/tests/units/interface-adapters/controllers/webhook/gitlab.controller.test.js +1 -0
  138. package/dist/tests/units/interface-adapters/controllers/webhook/gitlab.controller.test.js.map +1 -1
  139. package/dist/tests/units/modules/cli-configuration/interface-adapters/controllers/http/projectConfig.routes.test.d.ts +2 -0
  140. package/dist/tests/units/modules/cli-configuration/interface-adapters/controllers/http/projectConfig.routes.test.d.ts.map +1 -0
  141. package/dist/tests/units/modules/cli-configuration/interface-adapters/controllers/http/projectConfig.routes.test.js +130 -0
  142. package/dist/tests/units/modules/cli-configuration/interface-adapters/controllers/http/projectConfig.routes.test.js.map +1 -0
  143. package/dist/tests/units/modules/review-execution/entities/pendingReviewRequest/pendingReviewRequest.guard.test.d.ts +2 -0
  144. package/dist/tests/units/modules/review-execution/entities/pendingReviewRequest/pendingReviewRequest.guard.test.d.ts.map +1 -0
  145. package/dist/tests/units/modules/review-execution/entities/pendingReviewRequest/pendingReviewRequest.guard.test.js +32 -0
  146. package/dist/tests/units/modules/review-execution/entities/pendingReviewRequest/pendingReviewRequest.guard.test.js.map +1 -0
  147. package/dist/tests/units/modules/review-execution/entities/progress/agentDefinition.type.test.d.ts +2 -0
  148. package/dist/tests/units/modules/review-execution/entities/progress/agentDefinition.type.test.d.ts.map +1 -0
  149. package/dist/tests/units/modules/review-execution/entities/progress/agentDefinition.type.test.js +108 -0
  150. package/dist/tests/units/modules/review-execution/entities/progress/agentDefinition.type.test.js.map +1 -0
  151. package/dist/tests/units/modules/review-execution/entities/progress/reviewFocus.type.test.d.ts +2 -0
  152. package/dist/tests/units/modules/review-execution/entities/progress/reviewFocus.type.test.d.ts.map +1 -0
  153. package/dist/tests/units/modules/review-execution/entities/progress/reviewFocus.type.test.js +73 -0
  154. package/dist/tests/units/modules/review-execution/entities/progress/reviewFocus.type.test.js.map +1 -0
  155. package/dist/tests/units/modules/review-execution/interface-adapters/controllers/http/pendingReviews.routes.test.d.ts +2 -0
  156. package/dist/tests/units/modules/review-execution/interface-adapters/controllers/http/pendingReviews.routes.test.d.ts.map +1 -0
  157. package/dist/tests/units/modules/review-execution/interface-adapters/controllers/http/pendingReviews.routes.test.js +114 -0
  158. package/dist/tests/units/modules/review-execution/interface-adapters/controllers/http/pendingReviews.routes.test.js.map +1 -0
  159. package/dist/tests/units/modules/review-execution/interface-adapters/gateways/pendingReviewRequest.fileSystem.gateway.test.d.ts +2 -0
  160. package/dist/tests/units/modules/review-execution/interface-adapters/gateways/pendingReviewRequest.fileSystem.gateway.test.d.ts.map +1 -0
  161. package/dist/tests/units/modules/review-execution/interface-adapters/gateways/pendingReviewRequest.fileSystem.gateway.test.js +67 -0
  162. package/dist/tests/units/modules/review-execution/interface-adapters/gateways/pendingReviewRequest.fileSystem.gateway.test.js.map +1 -0
  163. package/dist/tests/units/modules/review-execution/interface-adapters/presenters/pendingReview.presenter.test.d.ts +2 -0
  164. package/dist/tests/units/modules/review-execution/interface-adapters/presenters/pendingReview.presenter.test.d.ts.map +1 -0
  165. package/dist/tests/units/modules/review-execution/interface-adapters/presenters/pendingReview.presenter.test.js +40 -0
  166. package/dist/tests/units/modules/review-execution/interface-adapters/presenters/pendingReview.presenter.test.js.map +1 -0
  167. package/dist/tests/units/modules/review-execution/services/processorRegistry.test.d.ts +2 -0
  168. package/dist/tests/units/modules/review-execution/services/processorRegistry.test.d.ts.map +1 -0
  169. package/dist/tests/units/modules/review-execution/services/processorRegistry.test.js +54 -0
  170. package/dist/tests/units/modules/review-execution/services/processorRegistry.test.js.map +1 -0
  171. package/dist/tests/units/modules/review-execution/usecases/confirmPendingReview.usecase.test.d.ts +2 -0
  172. package/dist/tests/units/modules/review-execution/usecases/confirmPendingReview.usecase.test.d.ts.map +1 -0
  173. package/dist/tests/units/modules/review-execution/usecases/confirmPendingReview.usecase.test.js +86 -0
  174. package/dist/tests/units/modules/review-execution/usecases/confirmPendingReview.usecase.test.js.map +1 -0
  175. package/dist/tests/units/modules/review-execution/usecases/dismissPendingReview.usecase.test.d.ts +2 -0
  176. package/dist/tests/units/modules/review-execution/usecases/dismissPendingReview.usecase.test.d.ts.map +1 -0
  177. package/dist/tests/units/modules/review-execution/usecases/dismissPendingReview.usecase.test.js +50 -0
  178. package/dist/tests/units/modules/review-execution/usecases/dismissPendingReview.usecase.test.js.map +1 -0
  179. package/dist/tests/units/modules/review-execution/usecases/gateClaudeInvocation.usecase.test.d.ts +2 -0
  180. package/dist/tests/units/modules/review-execution/usecases/gateClaudeInvocation.usecase.test.d.ts.map +1 -0
  181. package/dist/tests/units/modules/review-execution/usecases/gateClaudeInvocation.usecase.test.js +134 -0
  182. package/dist/tests/units/modules/review-execution/usecases/gateClaudeInvocation.usecase.test.js.map +1 -0
  183. package/dist/tests/units/modules/review-execution/usecases/listPendingReviews.usecase.test.d.ts +2 -0
  184. package/dist/tests/units/modules/review-execution/usecases/listPendingReviews.usecase.test.d.ts.map +1 -0
  185. package/dist/tests/units/modules/review-execution/usecases/listPendingReviews.usecase.test.js +27 -0
  186. package/dist/tests/units/modules/review-execution/usecases/listPendingReviews.usecase.test.js.map +1 -0
  187. package/dist/tests/units/usecases/tracking/checkFollowupNeeded.usecase.test.js +46 -11
  188. package/dist/tests/units/usecases/tracking/checkFollowupNeeded.usecase.test.js.map +1 -1
  189. package/package.json +3 -2
@@ -0,0 +1,34 @@
1
+ export const pendingReviewsRoutes = async (fastify, opts) => {
2
+ const { listPendingReviews, confirmPendingReview, dismissPendingReview, presenter } = opts;
3
+ fastify.get('/api/pending-reviews', async () => {
4
+ const pendingList = await listPendingReviews.execute();
5
+ return { pendingReviews: pendingList.map((entry) => presenter.present(entry)) };
6
+ });
7
+ fastify.post('/api/pending-reviews/:pendingId/confirm', async (request, reply) => {
8
+ const { pendingId } = request.params;
9
+ const result = await confirmPendingReview.execute({ pendingId });
10
+ if (result.status === 'not-found') {
11
+ reply.code(404);
12
+ return { status: 'not-found' };
13
+ }
14
+ if (result.status === 'already-running') {
15
+ reply.code(409);
16
+ return { status: 'already-running', message: result.message };
17
+ }
18
+ return { status: 'confirmed', jobId: result.jobId };
19
+ });
20
+ fastify.delete('/api/pending-reviews/:pendingId', async (request, reply) => {
21
+ const { pendingId } = request.params;
22
+ const result = await dismissPendingReview.execute({ pendingId });
23
+ if (result.status === 'not-found') {
24
+ reply.code(404);
25
+ return { status: 'not-found' };
26
+ }
27
+ if (result.status === 'already-running') {
28
+ reply.code(409);
29
+ return { status: 'already-running', message: result.message };
30
+ }
31
+ return { status: 'dismissed' };
32
+ });
33
+ };
34
+ //# sourceMappingURL=pendingReviews.routes.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"pendingReviews.routes.js","sourceRoot":"","sources":["../../../../../../src/modules/review-execution/interface-adapters/controllers/http/pendingReviews.routes.ts"],"names":[],"mappings":"AAiBA,MAAM,CAAC,MAAM,oBAAoB,GAAoD,KAAK,EACxF,OAAO,EACP,IAAI,EACJ,EAAE;IACF,MAAM,EAAE,kBAAkB,EAAE,oBAAoB,EAAE,oBAAoB,EAAE,SAAS,EAAE,GAAG,IAAI,CAAC;IAE3F,OAAO,CAAC,GAAG,CAAC,sBAAsB,EAAE,KAAK,IAAI,EAAE;QAC7C,MAAM,WAAW,GAAG,MAAM,kBAAkB,CAAC,OAAO,EAAE,CAAC;QACvD,OAAO,EAAE,cAAc,EAAE,WAAW,CAAC,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,SAAS,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC;IAClF,CAAC,CAAC,CAAC;IAEH,OAAO,CAAC,IAAI,CACV,yCAAyC,EACzC,KAAK,EAAE,OAAO,EAAE,KAAK,EAAE,EAAE;QACvB,MAAM,EAAE,SAAS,EAAE,GAAG,OAAO,CAAC,MAAM,CAAC;QACrC,MAAM,MAAM,GAAG,MAAM,oBAAoB,CAAC,OAAO,CAAC,EAAE,SAAS,EAAE,CAAC,CAAC;QACjE,IAAI,MAAM,CAAC,MAAM,KAAK,WAAW,EAAE,CAAC;YAClC,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YAChB,OAAO,EAAE,MAAM,EAAE,WAAW,EAAE,CAAC;QACjC,CAAC;QACD,IAAI,MAAM,CAAC,MAAM,KAAK,iBAAiB,EAAE,CAAC;YACxC,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YAChB,OAAO,EAAE,MAAM,EAAE,iBAAiB,EAAE,OAAO,EAAE,MAAM,CAAC,OAAO,EAAE,CAAC;QAChE,CAAC;QACD,OAAO,EAAE,MAAM,EAAE,WAAW,EAAE,KAAK,EAAE,MAAM,CAAC,KAAK,EAAE,CAAC;IACtD,CAAC,CACF,CAAC;IAEF,OAAO,CAAC,MAAM,CACZ,iCAAiC,EACjC,KAAK,EAAE,OAAO,EAAE,KAAK,EAAE,EAAE;QACvB,MAAM,EAAE,SAAS,EAAE,GAAG,OAAO,CAAC,MAAM,CAAC;QACrC,MAAM,MAAM,GAAG,MAAM,oBAAoB,CAAC,OAAO,CAAC,EAAE,SAAS,EAAE,CAAC,CAAC;QACjE,IAAI,MAAM,CAAC,MAAM,KAAK,WAAW,EAAE,CAAC;YAClC,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YAChB,OAAO,EAAE,MAAM,EAAE,WAAW,EAAE,CAAC;QACjC,CAAC;QACD,IAAI,MAAM,CAAC,MAAM,KAAK,iBAAiB,EAAE,CAAC;YACxC,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YAChB,OAAO,EAAE,MAAM,EAAE,iBAAiB,EAAE,OAAO,EAAE,MAAM,CAAC,OAAO,EAAE,CAAC;QAChE,CAAC;QACD,OAAO,EAAE,MAAM,EAAE,WAAW,EAAE,CAAC;IACjC,CAAC,CACF,CAAC;AACJ,CAAC,CAAC"}
@@ -0,0 +1,16 @@
1
+ import type { PendingReviewRequestGateway } from '../../../../modules/review-execution/entities/pendingReviewRequest/pendingReviewRequest.gateway.js';
2
+ import type { PendingReviewRequest } from '../../../../modules/review-execution/entities/pendingReviewRequest/pendingReviewRequest.schema.js';
3
+ export interface PendingReviewRequestFileSystemGatewayOptions {
4
+ rootDir?: string;
5
+ }
6
+ export declare class PendingReviewRequestFileSystemGateway implements PendingReviewRequestGateway {
7
+ private readonly rootDir;
8
+ constructor(options?: PendingReviewRequestFileSystemGatewayOptions);
9
+ private ensureRootDir;
10
+ private filePath;
11
+ save(pending: PendingReviewRequest): Promise<void>;
12
+ load(pendingReviewRequestId: string): Promise<PendingReviewRequest | null>;
13
+ listAll(): Promise<PendingReviewRequest[]>;
14
+ delete(pendingReviewRequestId: string): Promise<boolean>;
15
+ }
16
+ //# sourceMappingURL=pendingReviewRequest.fileSystem.gateway.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"pendingReviewRequest.fileSystem.gateway.d.ts","sourceRoot":"","sources":["../../../../../src/modules/review-execution/interface-adapters/gateways/pendingReviewRequest.fileSystem.gateway.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,2BAA2B,EAAE,MAAM,0FAA0F,CAAC;AAE5I,OAAO,KAAK,EAAE,oBAAoB,EAAE,MAAM,yFAAyF,CAAC;AAGpI,MAAM,WAAW,4CAA4C;IAC3D,OAAO,CAAC,EAAE,MAAM,CAAC;CAClB;AAMD,qBAAa,qCAAsC,YAAW,2BAA2B;IACvF,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAS;gBAErB,OAAO,GAAE,4CAAiD;IAItE,OAAO,CAAC,aAAa;IAMrB,OAAO,CAAC,QAAQ;IAIV,IAAI,CAAC,OAAO,EAAE,oBAAoB,GAAG,OAAO,CAAC,IAAI,CAAC;IAKlD,IAAI,CAAC,sBAAsB,EAAE,MAAM,GAAG,OAAO,CAAC,oBAAoB,GAAG,IAAI,CAAC;IAkB1E,OAAO,IAAI,OAAO,CAAC,oBAAoB,EAAE,CAAC;IAsB1C,MAAM,CAAC,sBAAsB,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC;CAY/D"}
@@ -0,0 +1,81 @@
1
+ import { existsSync, mkdirSync, readFileSync, readdirSync, writeFileSync, unlinkSync } from 'node:fs';
2
+ import { homedir } from 'node:os';
3
+ import { join } from 'node:path';
4
+ import { pendingReviewRequestGuard } from '../../../../modules/review-execution/entities/pendingReviewRequest/pendingReviewRequest.guard.js';
5
+ import { sanitizeJobId } from '../../../../shared/services/mcpJobContext.js';
6
+ function defaultRootDir() {
7
+ return join(homedir(), '.claude-review', 'pending');
8
+ }
9
+ export class PendingReviewRequestFileSystemGateway {
10
+ rootDir;
11
+ constructor(options = {}) {
12
+ this.rootDir = options.rootDir ?? defaultRootDir();
13
+ }
14
+ ensureRootDir() {
15
+ if (!existsSync(this.rootDir)) {
16
+ mkdirSync(this.rootDir, { recursive: true });
17
+ }
18
+ }
19
+ filePath(pendingReviewRequestId) {
20
+ return join(this.rootDir, `${sanitizeJobId(pendingReviewRequestId)}.json`);
21
+ }
22
+ async save(pending) {
23
+ this.ensureRootDir();
24
+ writeFileSync(this.filePath(pending.pendingReviewRequestId), `${JSON.stringify(pending, null, 2)}\n`);
25
+ }
26
+ async load(pendingReviewRequestId) {
27
+ const filePath = this.filePath(pendingReviewRequestId);
28
+ if (!existsSync(filePath)) {
29
+ return null;
30
+ }
31
+ try {
32
+ const raw = readFileSync(filePath, 'utf-8');
33
+ const parsed = JSON.parse(raw);
34
+ const result = pendingReviewRequestGuard.safeParse(parsed);
35
+ if (!result.success) {
36
+ return null;
37
+ }
38
+ return result.data;
39
+ }
40
+ catch {
41
+ return null;
42
+ }
43
+ }
44
+ async listAll() {
45
+ if (!existsSync(this.rootDir)) {
46
+ return [];
47
+ }
48
+ const entries = readdirSync(this.rootDir);
49
+ const collected = [];
50
+ for (const entry of entries) {
51
+ if (!entry.endsWith('.json'))
52
+ continue;
53
+ try {
54
+ const raw = readFileSync(join(this.rootDir, entry), 'utf-8');
55
+ const parsed = JSON.parse(raw);
56
+ const result = pendingReviewRequestGuard.safeParse(parsed);
57
+ if (result.success) {
58
+ collected.push(result.data);
59
+ }
60
+ }
61
+ catch {
62
+ // Skip unreadable / invalid files; the file system gateway is forgiving.
63
+ }
64
+ }
65
+ return collected;
66
+ }
67
+ async delete(pendingReviewRequestId) {
68
+ const filePath = this.filePath(pendingReviewRequestId);
69
+ if (!existsSync(filePath)) {
70
+ return false;
71
+ }
72
+ try {
73
+ unlinkSync(filePath);
74
+ return true;
75
+ }
76
+ catch {
77
+ return false;
78
+ }
79
+ }
80
+ }
81
+ //# sourceMappingURL=pendingReviewRequest.fileSystem.gateway.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"pendingReviewRequest.fileSystem.gateway.js","sourceRoot":"","sources":["../../../../../src/modules/review-execution/interface-adapters/gateways/pendingReviewRequest.fileSystem.gateway.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,SAAS,EAAE,YAAY,EAAE,WAAW,EAAE,aAAa,EAAE,UAAU,EAAE,MAAM,SAAS,CAAC;AACtG,OAAO,EAAE,OAAO,EAAE,MAAM,SAAS,CAAC;AAClC,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AAEjC,OAAO,EAAE,yBAAyB,EAAE,MAAM,wFAAwF,CAAC;AAEnI,OAAO,EAAE,aAAa,EAAE,MAAM,oCAAoC,CAAC;AAMnE,SAAS,cAAc;IACrB,OAAO,IAAI,CAAC,OAAO,EAAE,EAAE,gBAAgB,EAAE,SAAS,CAAC,CAAC;AACtD,CAAC;AAED,MAAM,OAAO,qCAAqC;IAC/B,OAAO,CAAS;IAEjC,YAAY,UAAwD,EAAE;QACpE,IAAI,CAAC,OAAO,GAAG,OAAO,CAAC,OAAO,IAAI,cAAc,EAAE,CAAC;IACrD,CAAC;IAEO,aAAa;QACnB,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,OAAO,CAAC,EAAE,CAAC;YAC9B,SAAS,CAAC,IAAI,CAAC,OAAO,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;QAC/C,CAAC;IACH,CAAC;IAEO,QAAQ,CAAC,sBAA8B;QAC7C,OAAO,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,GAAG,aAAa,CAAC,sBAAsB,CAAC,OAAO,CAAC,CAAC;IAC7E,CAAC;IAED,KAAK,CAAC,IAAI,CAAC,OAA6B;QACtC,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,aAAa,CAAC,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,sBAAsB,CAAC,EAAE,GAAG,IAAI,CAAC,SAAS,CAAC,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC,IAAI,CAAC,CAAC;IACxG,CAAC;IAED,KAAK,CAAC,IAAI,CAAC,sBAA8B;QACvC,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,sBAAsB,CAAC,CAAC;QACvD,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,EAAE,CAAC;YAC1B,OAAO,IAAI,CAAC;QACd,CAAC;QACD,IAAI,CAAC;YACH,MAAM,GAAG,GAAG,YAAY,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;YAC5C,MAAM,MAAM,GAAY,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;YACxC,MAAM,MAAM,GAAG,yBAAyB,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC;YAC3D,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE,CAAC;gBACpB,OAAO,IAAI,CAAC;YACd,CAAC;YACD,OAAO,MAAM,CAAC,IAAI,CAAC;QACrB,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,IAAI,CAAC;QACd,CAAC;IACH,CAAC;IAED,KAAK,CAAC,OAAO;QACX,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,OAAO,CAAC,EAAE,CAAC;YAC9B,OAAO,EAAE,CAAC;QACZ,CAAC;QACD,MAAM,OAAO,GAAG,WAAW,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAC1C,MAAM,SAAS,GAA2B,EAAE,CAAC;QAC7C,KAAK,MAAM,KAAK,IAAI,OAAO,EAAE,CAAC;YAC5B,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,OAAO,CAAC;gBAAE,SAAS;YACvC,IAAI,CAAC;gBACH,MAAM,GAAG,GAAG,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,KAAK,CAAC,EAAE,OAAO,CAAC,CAAC;gBAC7D,MAAM,MAAM,GAAY,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;gBACxC,MAAM,MAAM,GAAG,yBAAyB,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC;gBAC3D,IAAI,MAAM,CAAC,OAAO,EAAE,CAAC;oBACnB,SAAS,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBAC9B,CAAC;YACH,CAAC;YAAC,MAAM,CAAC;gBACP,yEAAyE;YAC3E,CAAC;QACH,CAAC;QACD,OAAO,SAAS,CAAC;IACnB,CAAC;IAED,KAAK,CAAC,MAAM,CAAC,sBAA8B;QACzC,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,sBAAsB,CAAC,CAAC;QACvD,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,EAAE,CAAC;YAC1B,OAAO,KAAK,CAAC;QACf,CAAC;QACD,IAAI,CAAC;YACH,UAAU,CAAC,QAAQ,CAAC,CAAC;YACrB,OAAO,IAAI,CAAC;QACd,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,KAAK,CAAC;QACf,CAAC;IACH,CAAC;CACF"}
@@ -0,0 +1,25 @@
1
+ import type { Presenter } from '../../../../shared/foundation/presenter.base.js';
2
+ import type { PendingReviewRequest } from '../../../../modules/review-execution/entities/pendingReviewRequest/pendingReviewRequest.schema.js';
3
+ export interface PendingReviewViewModel {
4
+ identifier: string;
5
+ mrNumber: number;
6
+ displayTitle: string;
7
+ projectPath: string;
8
+ mrUrl: string;
9
+ jobTypeLabel: string;
10
+ triggerSourceLabel: string;
11
+ createdAtRelative: string;
12
+ confirmActionUrl: string;
13
+ dismissActionUrl: string;
14
+ }
15
+ export interface PendingReviewPresenterOptions {
16
+ now?: () => Date;
17
+ }
18
+ export declare class PendingReviewPresenter implements Presenter<PendingReviewRequest, PendingReviewViewModel> {
19
+ private readonly now;
20
+ constructor(options?: PendingReviewPresenterOptions);
21
+ present(pending: PendingReviewRequest): PendingReviewViewModel;
22
+ private formatDisplayTitle;
23
+ private formatRelative;
24
+ }
25
+ //# sourceMappingURL=pendingReview.presenter.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"pendingReview.presenter.d.ts","sourceRoot":"","sources":["../../../../../src/modules/review-execution/interface-adapters/presenters/pendingReview.presenter.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,uCAAuC,CAAC;AAEvE,OAAO,KAAK,EACV,oBAAoB,EAErB,MAAM,yFAAyF,CAAC;AAEjG,MAAM,WAAW,sBAAsB;IACrC,UAAU,EAAE,MAAM,CAAC;IACnB,QAAQ,EAAE,MAAM,CAAC;IACjB,YAAY,EAAE,MAAM,CAAC;IACrB,WAAW,EAAE,MAAM,CAAC;IACpB,KAAK,EAAE,MAAM,CAAC;IACd,YAAY,EAAE,MAAM,CAAC;IACrB,kBAAkB,EAAE,MAAM,CAAC;IAC3B,iBAAiB,EAAE,MAAM,CAAC;IAC1B,gBAAgB,EAAE,MAAM,CAAC;IACzB,gBAAgB,EAAE,MAAM,CAAC;CAC1B;AAED,MAAM,WAAW,6BAA6B;IAC5C,GAAG,CAAC,EAAE,MAAM,IAAI,CAAC;CAClB;AAQD,qBAAa,sBAAuB,YAAW,SAAS,CAAC,oBAAoB,EAAE,sBAAsB,CAAC;IACpG,OAAO,CAAC,QAAQ,CAAC,GAAG,CAAa;gBAErB,OAAO,GAAE,6BAAkC;IAIvD,OAAO,CAAC,OAAO,EAAE,oBAAoB,GAAG,sBAAsB;IAe9D,OAAO,CAAC,kBAAkB;IAO1B,OAAO,CAAC,cAAc;CAQvB"}
@@ -0,0 +1,41 @@
1
+ import { Duration } from '../../../../modules/shared-kernel/entities/shared/duration.valueObject.js';
2
+ const TRIGGER_SOURCE_LABELS = {
3
+ 'webhook-initial': 'Webhook',
4
+ 'webhook-followup': 'Webhook',
5
+ 'dashboard-manual': 'Manuel',
6
+ };
7
+ export class PendingReviewPresenter {
8
+ now;
9
+ constructor(options = {}) {
10
+ this.now = options.now ?? (() => new Date());
11
+ }
12
+ present(pending) {
13
+ return {
14
+ identifier: pending.pendingReviewRequestId,
15
+ mrNumber: pending.job.mrNumber,
16
+ displayTitle: this.formatDisplayTitle(pending),
17
+ projectPath: pending.job.projectPath,
18
+ mrUrl: pending.job.mrUrl,
19
+ jobTypeLabel: pending.jobType === 'followup' ? 'Followup' : 'Review',
20
+ triggerSourceLabel: TRIGGER_SOURCE_LABELS[pending.triggerSource],
21
+ createdAtRelative: this.formatRelative(pending.createdAt),
22
+ confirmActionUrl: `/api/pending-reviews/${pending.pendingReviewRequestId}/confirm`,
23
+ dismissActionUrl: `/api/pending-reviews/${pending.pendingReviewRequestId}`,
24
+ };
25
+ }
26
+ formatDisplayTitle(pending) {
27
+ const title = pending.job.title ?? '';
28
+ return title.length > 0
29
+ ? `MR !${pending.job.mrNumber} - ${title}`
30
+ : `MR !${pending.job.mrNumber}`;
31
+ }
32
+ formatRelative(isoString) {
33
+ const createdAtMs = new Date(isoString).getTime();
34
+ if (Number.isNaN(createdAtMs)) {
35
+ return '';
36
+ }
37
+ const elapsedMs = Math.max(0, this.now().getTime() - createdAtMs);
38
+ return `il y a ${Duration.fromMilliseconds(elapsedMs).formatted}`;
39
+ }
40
+ }
41
+ //# sourceMappingURL=pendingReview.presenter.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"pendingReview.presenter.js","sourceRoot":"","sources":["../../../../../src/modules/review-execution/interface-adapters/presenters/pendingReview.presenter.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,QAAQ,EAAE,MAAM,iEAAiE,CAAC;AAuB3F,MAAM,qBAAqB,GAAkC;IAC3D,iBAAiB,EAAE,SAAS;IAC5B,kBAAkB,EAAE,SAAS;IAC7B,kBAAkB,EAAE,QAAQ;CAC7B,CAAC;AAEF,MAAM,OAAO,sBAAsB;IAChB,GAAG,CAAa;IAEjC,YAAY,UAAyC,EAAE;QACrD,IAAI,CAAC,GAAG,GAAG,OAAO,CAAC,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC,IAAI,IAAI,EAAE,CAAC,CAAC;IAC/C,CAAC;IAED,OAAO,CAAC,OAA6B;QACnC,OAAO;YACL,UAAU,EAAE,OAAO,CAAC,sBAAsB;YAC1C,QAAQ,EAAE,OAAO,CAAC,GAAG,CAAC,QAAQ;YAC9B,YAAY,EAAE,IAAI,CAAC,kBAAkB,CAAC,OAAO,CAAC;YAC9C,WAAW,EAAE,OAAO,CAAC,GAAG,CAAC,WAAW;YACpC,KAAK,EAAE,OAAO,CAAC,GAAG,CAAC,KAAK;YACxB,YAAY,EAAE,OAAO,CAAC,OAAO,KAAK,UAAU,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,QAAQ;YACpE,kBAAkB,EAAE,qBAAqB,CAAC,OAAO,CAAC,aAAa,CAAC;YAChE,iBAAiB,EAAE,IAAI,CAAC,cAAc,CAAC,OAAO,CAAC,SAAS,CAAC;YACzD,gBAAgB,EAAE,wBAAwB,OAAO,CAAC,sBAAsB,UAAU;YAClF,gBAAgB,EAAE,wBAAwB,OAAO,CAAC,sBAAsB,EAAE;SAC3E,CAAC;IACJ,CAAC;IAEO,kBAAkB,CAAC,OAA6B;QACtD,MAAM,KAAK,GAAG,OAAO,CAAC,GAAG,CAAC,KAAK,IAAI,EAAE,CAAC;QACtC,OAAO,KAAK,CAAC,MAAM,GAAG,CAAC;YACrB,CAAC,CAAC,OAAO,OAAO,CAAC,GAAG,CAAC,QAAQ,MAAM,KAAK,EAAE;YAC1C,CAAC,CAAC,OAAO,OAAO,CAAC,GAAG,CAAC,QAAQ,EAAE,CAAC;IACpC,CAAC;IAEO,cAAc,CAAC,SAAiB;QACtC,MAAM,WAAW,GAAG,IAAI,IAAI,CAAC,SAAS,CAAC,CAAC,OAAO,EAAE,CAAC;QAClD,IAAI,MAAM,CAAC,KAAK,CAAC,WAAW,CAAC,EAAE,CAAC;YAC9B,OAAO,EAAE,CAAC;QACZ,CAAC;QACD,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,GAAG,EAAE,CAAC,OAAO,EAAE,GAAG,WAAW,CAAC,CAAC;QAClE,OAAO,UAAU,QAAQ,CAAC,gBAAgB,CAAC,SAAS,CAAC,CAAC,SAAS,EAAE,CAAC;IACpE,CAAC;CACF"}
@@ -0,0 +1,15 @@
1
+ import type { ReviewJob } from '../../../frameworks/queue/pQueueAdapter.js';
2
+ import type { PendingReviewRequest, TriggerSource } from '../../../modules/review-execution/entities/pendingReviewRequest/pendingReviewRequest.schema.js';
3
+ import type { GateClaudeInvocationProcessor } from '../../../modules/review-execution/usecases/gateClaudeInvocation.usecase.js';
4
+ export type ProcessorBuilder = (job: ReviewJob) => GateClaudeInvocationProcessor;
5
+ export interface ProcessorKey {
6
+ triggerSource: TriggerSource;
7
+ platform: 'gitlab' | 'github';
8
+ jobType: 'review' | 'followup';
9
+ }
10
+ export declare class ProcessorRegistry {
11
+ private readonly builders;
12
+ register(key: ProcessorKey, builder: ProcessorBuilder): void;
13
+ resolve(pending: PendingReviewRequest): GateClaudeInvocationProcessor;
14
+ }
15
+ //# sourceMappingURL=processorRegistry.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"processorRegistry.d.ts","sourceRoot":"","sources":["../../../../src/modules/review-execution/services/processorRegistry.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,qCAAqC,CAAC;AACrE,OAAO,KAAK,EACV,oBAAoB,EACpB,aAAa,EACd,MAAM,yFAAyF,CAAC;AACjG,OAAO,KAAK,EAAE,6BAA6B,EAAE,MAAM,qEAAqE,CAAC;AAEzH,MAAM,MAAM,gBAAgB,GAAG,CAAC,GAAG,EAAE,SAAS,KAAK,6BAA6B,CAAC;AAEjF,MAAM,WAAW,YAAY;IAC3B,aAAa,EAAE,aAAa,CAAC;IAC7B,QAAQ,EAAE,QAAQ,GAAG,QAAQ,CAAC;IAC9B,OAAO,EAAE,QAAQ,GAAG,UAAU,CAAC;CAChC;AAMD,qBAAa,iBAAiB;IAC5B,OAAO,CAAC,QAAQ,CAAC,QAAQ,CAAuC;IAEhE,QAAQ,CAAC,GAAG,EAAE,YAAY,EAAE,OAAO,EAAE,gBAAgB,GAAG,IAAI;IAI5D,OAAO,CAAC,OAAO,EAAE,oBAAoB,GAAG,6BAA6B;CAetE"}
@@ -0,0 +1,23 @@
1
+ function buildRegistryKey(key) {
2
+ return `${key.triggerSource}:${key.platform}:${key.jobType}`;
3
+ }
4
+ export class ProcessorRegistry {
5
+ builders = new Map();
6
+ register(key, builder) {
7
+ this.builders.set(buildRegistryKey(key), builder);
8
+ }
9
+ resolve(pending) {
10
+ const key = buildRegistryKey({
11
+ triggerSource: pending.triggerSource,
12
+ platform: pending.platform,
13
+ jobType: pending.jobType,
14
+ });
15
+ const builder = this.builders.get(key);
16
+ if (!builder) {
17
+ const available = [...this.builders.keys()].join(', ') || '(none)';
18
+ throw new Error(`No processor builder registered for key "${key}". Available keys: ${available}`);
19
+ }
20
+ return builder(pending.job);
21
+ }
22
+ }
23
+ //# sourceMappingURL=processorRegistry.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"processorRegistry.js","sourceRoot":"","sources":["../../../../src/modules/review-execution/services/processorRegistry.ts"],"names":[],"mappings":"AAeA,SAAS,gBAAgB,CAAC,GAAiB;IACzC,OAAO,GAAG,GAAG,CAAC,aAAa,IAAI,GAAG,CAAC,QAAQ,IAAI,GAAG,CAAC,OAAO,EAAE,CAAC;AAC/D,CAAC;AAED,MAAM,OAAO,iBAAiB;IACX,QAAQ,GAAG,IAAI,GAAG,EAA4B,CAAC;IAEhE,QAAQ,CAAC,GAAiB,EAAE,OAAyB;QACnD,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,gBAAgB,CAAC,GAAG,CAAC,EAAE,OAAO,CAAC,CAAC;IACpD,CAAC;IAED,OAAO,CAAC,OAA6B;QACnC,MAAM,GAAG,GAAG,gBAAgB,CAAC;YAC3B,aAAa,EAAE,OAAO,CAAC,aAAa;YACpC,QAAQ,EAAE,OAAO,CAAC,QAAQ;YAC1B,OAAO,EAAE,OAAO,CAAC,OAAO;SACzB,CAAC,CAAC;QACH,MAAM,OAAO,GAAG,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;QACvC,IAAI,CAAC,OAAO,EAAE,CAAC;YACb,MAAM,SAAS,GAAG,CAAC,GAAG,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,QAAQ,CAAC;YACnE,MAAM,IAAI,KAAK,CACb,4CAA4C,GAAG,sBAAsB,SAAS,EAAE,CACjF,CAAC;QACJ,CAAC;QACD,OAAO,OAAO,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;IAC9B,CAAC;CACF"}
@@ -0,0 +1,32 @@
1
+ import type { Logger } from 'pino';
2
+ import type { PendingReviewRequestGateway } from '../../../modules/review-execution/entities/pendingReviewRequest/pendingReviewRequest.gateway.js';
3
+ import type { PendingReviewRequest } from '../../../modules/review-execution/entities/pendingReviewRequest/pendingReviewRequest.schema.js';
4
+ import type { EnqueueReviewFunction, GateClaudeInvocationProcessor } from '../../../modules/review-execution/usecases/gateClaudeInvocation.usecase.js';
5
+ export interface ConfirmQueuePort {
6
+ hasActiveJob(jobId: string): boolean;
7
+ getJobStatus(jobId: string): 'queued' | 'running' | 'completed' | 'failed' | null;
8
+ }
9
+ export interface ConfirmPendingReviewDependencies {
10
+ pendingReviewRequestGateway: PendingReviewRequestGateway;
11
+ queuePort: ConfirmQueuePort;
12
+ enqueue: EnqueueReviewFunction;
13
+ resolveProcessor: (pending: PendingReviewRequest) => GateClaudeInvocationProcessor;
14
+ logger: Logger;
15
+ }
16
+ export type ConfirmPendingReviewResult = {
17
+ status: 'confirmed';
18
+ jobId: string;
19
+ } | {
20
+ status: 'not-found';
21
+ } | {
22
+ status: 'already-running';
23
+ message: string;
24
+ };
25
+ export declare class ConfirmPendingReviewUseCase {
26
+ private readonly deps;
27
+ constructor(deps: ConfirmPendingReviewDependencies);
28
+ execute(input: {
29
+ pendingId: string;
30
+ }): Promise<ConfirmPendingReviewResult>;
31
+ }
32
+ //# sourceMappingURL=confirmPendingReview.usecase.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"confirmPendingReview.usecase.d.ts","sourceRoot":"","sources":["../../../../src/modules/review-execution/usecases/confirmPendingReview.usecase.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,MAAM,CAAC;AAEnC,OAAO,KAAK,EAAE,2BAA2B,EAAE,MAAM,0FAA0F,CAAC;AAC5I,OAAO,KAAK,EAAE,oBAAoB,EAAE,MAAM,yFAAyF,CAAC;AACpI,OAAO,KAAK,EACV,qBAAqB,EACrB,6BAA6B,EAC9B,MAAM,qEAAqE,CAAC;AAE7E,MAAM,WAAW,gBAAgB;IAC/B,YAAY,CAAC,KAAK,EAAE,MAAM,GAAG,OAAO,CAAC;IACrC,YAAY,CAAC,KAAK,EAAE,MAAM,GAAG,QAAQ,GAAG,SAAS,GAAG,WAAW,GAAG,QAAQ,GAAG,IAAI,CAAC;CACnF;AAED,MAAM,WAAW,gCAAgC;IAC/C,2BAA2B,EAAE,2BAA2B,CAAC;IACzD,SAAS,EAAE,gBAAgB,CAAC;IAC5B,OAAO,EAAE,qBAAqB,CAAC;IAK/B,gBAAgB,EAAE,CAAC,OAAO,EAAE,oBAAoB,KAAK,6BAA6B,CAAC;IACnF,MAAM,EAAE,MAAM,CAAC;CAChB;AAED,MAAM,MAAM,0BAA0B,GAClC;IAAE,MAAM,EAAE,WAAW,CAAC;IAAC,KAAK,EAAE,MAAM,CAAA;CAAE,GACtC;IAAE,MAAM,EAAE,WAAW,CAAA;CAAE,GACvB;IAAE,MAAM,EAAE,iBAAiB,CAAC;IAAC,OAAO,EAAE,MAAM,CAAA;CAAE,CAAC;AAInD,qBAAa,2BAA2B;IAC1B,OAAO,CAAC,QAAQ,CAAC,IAAI;gBAAJ,IAAI,EAAE,gCAAgC;IAE7D,OAAO,CAAC,KAAK,EAAE;QAAE,SAAS,EAAE,MAAM,CAAA;KAAE,GAAG,OAAO,CAAC,0BAA0B,CAAC;CAyBjF"}
@@ -0,0 +1,29 @@
1
+ const ALREADY_RUNNING_MESSAGE = 'Cette review est déjà en cours';
2
+ export class ConfirmPendingReviewUseCase {
3
+ deps;
4
+ constructor(deps) {
5
+ this.deps = deps;
6
+ }
7
+ async execute(input) {
8
+ const { pendingReviewRequestGateway, queuePort, enqueue, resolveProcessor, logger } = this.deps;
9
+ const pending = await pendingReviewRequestGateway.load(input.pendingId);
10
+ if (!pending) {
11
+ return { status: 'not-found' };
12
+ }
13
+ if (queuePort.hasActiveJob(pending.job.id)) {
14
+ logger.info({ pendingId: input.pendingId, jobId: pending.job.id }, 'Pending review confirm rejected: already running');
15
+ return { status: 'already-running', message: ALREADY_RUNNING_MESSAGE };
16
+ }
17
+ const processor = resolveProcessor(pending);
18
+ const job = pending.job;
19
+ const enqueued = await enqueue(job, processor);
20
+ if (!enqueued) {
21
+ logger.warn({ pendingId: input.pendingId, jobId: job.id }, 'Pending review confirm refused by queue');
22
+ return { status: 'already-running', message: ALREADY_RUNNING_MESSAGE };
23
+ }
24
+ await pendingReviewRequestGateway.delete(input.pendingId);
25
+ logger.info({ pendingId: input.pendingId, jobId: job.id }, 'Pending review confirmed and enqueued');
26
+ return { status: 'confirmed', jobId: job.id };
27
+ }
28
+ }
29
+ //# sourceMappingURL=confirmPendingReview.usecase.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"confirmPendingReview.usecase.js","sourceRoot":"","sources":["../../../../src/modules/review-execution/usecases/confirmPendingReview.usecase.ts"],"names":[],"mappings":"AA+BA,MAAM,uBAAuB,GAAG,gCAAgC,CAAC;AAEjE,MAAM,OAAO,2BAA2B;IACT;IAA7B,YAA6B,IAAsC;QAAtC,SAAI,GAAJ,IAAI,CAAkC;IAAG,CAAC;IAEvE,KAAK,CAAC,OAAO,CAAC,KAA4B;QACxC,MAAM,EAAE,2BAA2B,EAAE,SAAS,EAAE,OAAO,EAAE,gBAAgB,EAAE,MAAM,EAAE,GAAG,IAAI,CAAC,IAAI,CAAC;QAEhG,MAAM,OAAO,GAAG,MAAM,2BAA2B,CAAC,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;QACxE,IAAI,CAAC,OAAO,EAAE,CAAC;YACb,OAAO,EAAE,MAAM,EAAE,WAAW,EAAE,CAAC;QACjC,CAAC;QAED,IAAI,SAAS,CAAC,YAAY,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE,CAAC;YAC3C,MAAM,CAAC,IAAI,CAAC,EAAE,SAAS,EAAE,KAAK,CAAC,SAAS,EAAE,KAAK,EAAE,OAAO,CAAC,GAAG,CAAC,EAAE,EAAE,EAAE,kDAAkD,CAAC,CAAC;YACvH,OAAO,EAAE,MAAM,EAAE,iBAAiB,EAAE,OAAO,EAAE,uBAAuB,EAAE,CAAC;QACzE,CAAC;QAED,MAAM,SAAS,GAAG,gBAAgB,CAAC,OAAO,CAAC,CAAC;QAC5C,MAAM,GAAG,GAAc,OAAO,CAAC,GAAG,CAAC;QACnC,MAAM,QAAQ,GAAG,MAAM,OAAO,CAAC,GAAG,EAAE,SAAS,CAAC,CAAC;QAC/C,IAAI,CAAC,QAAQ,EAAE,CAAC;YACd,MAAM,CAAC,IAAI,CAAC,EAAE,SAAS,EAAE,KAAK,CAAC,SAAS,EAAE,KAAK,EAAE,GAAG,CAAC,EAAE,EAAE,EAAE,yCAAyC,CAAC,CAAC;YACtG,OAAO,EAAE,MAAM,EAAE,iBAAiB,EAAE,OAAO,EAAE,uBAAuB,EAAE,CAAC;QACzE,CAAC;QAED,MAAM,2BAA2B,CAAC,MAAM,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;QAC1D,MAAM,CAAC,IAAI,CAAC,EAAE,SAAS,EAAE,KAAK,CAAC,SAAS,EAAE,KAAK,EAAE,GAAG,CAAC,EAAE,EAAE,EAAE,uCAAuC,CAAC,CAAC;QACpG,OAAO,EAAE,MAAM,EAAE,WAAW,EAAE,KAAK,EAAE,GAAG,CAAC,EAAE,EAAE,CAAC;IAChD,CAAC;CACF"}
@@ -0,0 +1,26 @@
1
+ import type { Logger } from 'pino';
2
+ import type { PendingReviewRequestGateway } from '../../../modules/review-execution/entities/pendingReviewRequest/pendingReviewRequest.gateway.js';
3
+ export interface DismissQueuePort {
4
+ hasActiveJob(jobId: string): boolean;
5
+ }
6
+ export interface DismissPendingReviewDependencies {
7
+ pendingReviewRequestGateway: PendingReviewRequestGateway;
8
+ queuePort: DismissQueuePort;
9
+ logger: Logger;
10
+ }
11
+ export type DismissPendingReviewResult = {
12
+ status: 'dismissed';
13
+ } | {
14
+ status: 'not-found';
15
+ } | {
16
+ status: 'already-running';
17
+ message: string;
18
+ };
19
+ export declare class DismissPendingReviewUseCase {
20
+ private readonly deps;
21
+ constructor(deps: DismissPendingReviewDependencies);
22
+ execute(input: {
23
+ pendingId: string;
24
+ }): Promise<DismissPendingReviewResult>;
25
+ }
26
+ //# sourceMappingURL=dismissPendingReview.usecase.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"dismissPendingReview.usecase.d.ts","sourceRoot":"","sources":["../../../../src/modules/review-execution/usecases/dismissPendingReview.usecase.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,MAAM,CAAC;AACnC,OAAO,KAAK,EAAE,2BAA2B,EAAE,MAAM,0FAA0F,CAAC;AAE5I,MAAM,WAAW,gBAAgB;IAC/B,YAAY,CAAC,KAAK,EAAE,MAAM,GAAG,OAAO,CAAC;CACtC;AAED,MAAM,WAAW,gCAAgC;IAC/C,2BAA2B,EAAE,2BAA2B,CAAC;IACzD,SAAS,EAAE,gBAAgB,CAAC;IAC5B,MAAM,EAAE,MAAM,CAAC;CAChB;AAED,MAAM,MAAM,0BAA0B,GAClC;IAAE,MAAM,EAAE,WAAW,CAAA;CAAE,GACvB;IAAE,MAAM,EAAE,WAAW,CAAA;CAAE,GACvB;IAAE,MAAM,EAAE,iBAAiB,CAAC;IAAC,OAAO,EAAE,MAAM,CAAA;CAAE,CAAC;AAInD,qBAAa,2BAA2B;IAC1B,OAAO,CAAC,QAAQ,CAAC,IAAI;gBAAJ,IAAI,EAAE,gCAAgC;IAE7D,OAAO,CAAC,KAAK,EAAE;QAAE,SAAS,EAAE,MAAM,CAAA;KAAE,GAAG,OAAO,CAAC,0BAA0B,CAAC;CAiBjF"}
@@ -0,0 +1,22 @@
1
+ const ALREADY_RUNNING_DISMISS_MESSAGE = "Cette review est déjà en cours, impossible de l'ignorer";
2
+ export class DismissPendingReviewUseCase {
3
+ deps;
4
+ constructor(deps) {
5
+ this.deps = deps;
6
+ }
7
+ async execute(input) {
8
+ const { pendingReviewRequestGateway, queuePort, logger } = this.deps;
9
+ const pending = await pendingReviewRequestGateway.load(input.pendingId);
10
+ if (!pending) {
11
+ return { status: 'not-found' };
12
+ }
13
+ if (queuePort.hasActiveJob(pending.job.id)) {
14
+ logger.info({ pendingId: input.pendingId, jobId: pending.job.id }, 'Pending review dismiss rejected: already running');
15
+ return { status: 'already-running', message: ALREADY_RUNNING_DISMISS_MESSAGE };
16
+ }
17
+ await pendingReviewRequestGateway.delete(input.pendingId);
18
+ logger.info({ pendingId: input.pendingId }, 'Pending review dismissed');
19
+ return { status: 'dismissed' };
20
+ }
21
+ }
22
+ //# sourceMappingURL=dismissPendingReview.usecase.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"dismissPendingReview.usecase.js","sourceRoot":"","sources":["../../../../src/modules/review-execution/usecases/dismissPendingReview.usecase.ts"],"names":[],"mappings":"AAkBA,MAAM,+BAA+B,GAAG,yDAAyD,CAAC;AAElG,MAAM,OAAO,2BAA2B;IACT;IAA7B,YAA6B,IAAsC;QAAtC,SAAI,GAAJ,IAAI,CAAkC;IAAG,CAAC;IAEvE,KAAK,CAAC,OAAO,CAAC,KAA4B;QACxC,MAAM,EAAE,2BAA2B,EAAE,SAAS,EAAE,MAAM,EAAE,GAAG,IAAI,CAAC,IAAI,CAAC;QAErE,MAAM,OAAO,GAAG,MAAM,2BAA2B,CAAC,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;QACxE,IAAI,CAAC,OAAO,EAAE,CAAC;YACb,OAAO,EAAE,MAAM,EAAE,WAAW,EAAE,CAAC;QACjC,CAAC;QAED,IAAI,SAAS,CAAC,YAAY,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE,CAAC;YAC3C,MAAM,CAAC,IAAI,CAAC,EAAE,SAAS,EAAE,KAAK,CAAC,SAAS,EAAE,KAAK,EAAE,OAAO,CAAC,GAAG,CAAC,EAAE,EAAE,EAAE,kDAAkD,CAAC,CAAC;YACvH,OAAO,EAAE,MAAM,EAAE,iBAAiB,EAAE,OAAO,EAAE,+BAA+B,EAAE,CAAC;QACjF,CAAC;QAED,MAAM,2BAA2B,CAAC,MAAM,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;QAC1D,MAAM,CAAC,IAAI,CAAC,EAAE,SAAS,EAAE,KAAK,CAAC,SAAS,EAAE,EAAE,0BAA0B,CAAC,CAAC;QACxE,OAAO,EAAE,MAAM,EAAE,WAAW,EAAE,CAAC;IACjC,CAAC;CACF"}
@@ -0,0 +1,36 @@
1
+ import type { Logger } from 'pino';
2
+ import type { ReviewJob } from '../../../frameworks/queue/pQueueAdapter.js';
3
+ import type { PendingReviewRequestGateway } from '../../../modules/review-execution/entities/pendingReviewRequest/pendingReviewRequest.gateway.js';
4
+ import type { PendingReviewRequest, TriggerSource } from '../../../modules/review-execution/entities/pendingReviewRequest/pendingReviewRequest.schema.js';
5
+ export type TriggerMode = 'full-auto' | 'semi-auto';
6
+ export type GateClaudeInvocationProcessor = (job: ReviewJob, signal: AbortSignal) => Promise<void>;
7
+ export type EnqueueReviewFunction = (job: ReviewJob, processor: GateClaudeInvocationProcessor) => Promise<boolean>;
8
+ export interface GateClaudeInvocationDependencies {
9
+ triggerMode: TriggerMode;
10
+ pendingReviewRequestGateway: PendingReviewRequestGateway;
11
+ enqueue: EnqueueReviewFunction;
12
+ broadcastPendingChanged: (pending: PendingReviewRequest) => void;
13
+ logger: Logger;
14
+ clock?: () => Date;
15
+ }
16
+ export interface GateClaudeInvocationInput {
17
+ job: ReviewJob;
18
+ triggerSource: TriggerSource;
19
+ processor: GateClaudeInvocationProcessor;
20
+ }
21
+ export type GateClaudeInvocationResult = {
22
+ status: 'enqueued';
23
+ jobId: string;
24
+ } | {
25
+ status: 'pending';
26
+ pendingId: string;
27
+ } | {
28
+ status: 'rejected';
29
+ reason: string;
30
+ };
31
+ export declare class GateClaudeInvocationUseCase {
32
+ private readonly deps;
33
+ constructor(deps: GateClaudeInvocationDependencies);
34
+ execute(input: GateClaudeInvocationInput): Promise<GateClaudeInvocationResult>;
35
+ }
36
+ //# sourceMappingURL=gateClaudeInvocation.usecase.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"gateClaudeInvocation.usecase.d.ts","sourceRoot":"","sources":["../../../../src/modules/review-execution/usecases/gateClaudeInvocation.usecase.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,MAAM,CAAC;AACnC,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,qCAAqC,CAAC;AAErE,OAAO,KAAK,EAAE,2BAA2B,EAAE,MAAM,0FAA0F,CAAC;AAC5I,OAAO,KAAK,EACV,oBAAoB,EACpB,aAAa,EACd,MAAM,yFAAyF,CAAC;AAEjG,MAAM,MAAM,WAAW,GAAG,WAAW,GAAG,WAAW,CAAC;AAEpD,MAAM,MAAM,6BAA6B,GAAG,CAC1C,GAAG,EAAE,SAAS,EACd,MAAM,EAAE,WAAW,KAChB,OAAO,CAAC,IAAI,CAAC,CAAC;AAEnB,MAAM,MAAM,qBAAqB,GAAG,CAClC,GAAG,EAAE,SAAS,EACd,SAAS,EAAE,6BAA6B,KACrC,OAAO,CAAC,OAAO,CAAC,CAAC;AAEtB,MAAM,WAAW,gCAAgC;IAC/C,WAAW,EAAE,WAAW,CAAC;IACzB,2BAA2B,EAAE,2BAA2B,CAAC;IACzD,OAAO,EAAE,qBAAqB,CAAC;IAC/B,uBAAuB,EAAE,CAAC,OAAO,EAAE,oBAAoB,KAAK,IAAI,CAAC;IACjE,MAAM,EAAE,MAAM,CAAC;IACf,KAAK,CAAC,EAAE,MAAM,IAAI,CAAC;CACpB;AAED,MAAM,WAAW,yBAAyB;IACxC,GAAG,EAAE,SAAS,CAAC;IACf,aAAa,EAAE,aAAa,CAAC;IAC7B,SAAS,EAAE,6BAA6B,CAAC;CAC1C;AAED,MAAM,MAAM,0BAA0B,GAClC;IAAE,MAAM,EAAE,UAAU,CAAC;IAAC,KAAK,EAAE,MAAM,CAAA;CAAE,GACrC;IAAE,MAAM,EAAE,SAAS,CAAC;IAAC,SAAS,EAAE,MAAM,CAAA;CAAE,GACxC;IAAE,MAAM,EAAE,UAAU,CAAC;IAAC,MAAM,EAAE,MAAM,CAAA;CAAE,CAAC;AAM3C,qBAAa,2BAA2B;IAC1B,OAAO,CAAC,QAAQ,CAAC,IAAI;gBAAJ,IAAI,EAAE,gCAAgC;IAE7D,OAAO,CAAC,KAAK,EAAE,yBAAyB,GAAG,OAAO,CAAC,0BAA0B,CAAC;CAgCrF"}
@@ -0,0 +1,36 @@
1
+ import { sanitizeJobId } from '../../../shared/services/mcpJobContext.js';
2
+ function buildPendingId(jobId) {
3
+ return `pending-${sanitizeJobId(jobId)}`;
4
+ }
5
+ export class GateClaudeInvocationUseCase {
6
+ deps;
7
+ constructor(deps) {
8
+ this.deps = deps;
9
+ }
10
+ async execute(input) {
11
+ const { triggerMode, pendingReviewRequestGateway, enqueue, broadcastPendingChanged, logger } = this.deps;
12
+ if (triggerMode === 'full-auto') {
13
+ const enqueued = await enqueue(input.job, input.processor);
14
+ if (!enqueued) {
15
+ logger.info({ jobId: input.job.id }, 'Job rejected by queue (deduplicated or already active)');
16
+ return { status: 'rejected', reason: 'Queue refused the job (deduplicated or already active)' };
17
+ }
18
+ return { status: 'enqueued', jobId: input.job.id };
19
+ }
20
+ const pendingId = buildPendingId(input.job.id);
21
+ const createdAt = (this.deps.clock ?? (() => new Date()))().toISOString();
22
+ const pending = {
23
+ pendingReviewRequestId: pendingId,
24
+ job: input.job,
25
+ jobType: input.job.jobType ?? 'review',
26
+ platform: input.job.platform,
27
+ triggerSource: input.triggerSource,
28
+ createdAt,
29
+ };
30
+ await pendingReviewRequestGateway.save(pending);
31
+ broadcastPendingChanged(pending);
32
+ logger.info({ pendingId, jobId: input.job.id, triggerSource: input.triggerSource }, 'Review parked for human confirmation (semi-auto trigger mode)');
33
+ return { status: 'pending', pendingId };
34
+ }
35
+ }
36
+ //# sourceMappingURL=gateClaudeInvocation.usecase.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"gateClaudeInvocation.usecase.js","sourceRoot":"","sources":["../../../../src/modules/review-execution/usecases/gateClaudeInvocation.usecase.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,aAAa,EAAE,MAAM,oCAAoC,CAAC;AAuCnE,SAAS,cAAc,CAAC,KAAa;IACnC,OAAO,WAAW,aAAa,CAAC,KAAK,CAAC,EAAE,CAAC;AAC3C,CAAC;AAED,MAAM,OAAO,2BAA2B;IACT;IAA7B,YAA6B,IAAsC;QAAtC,SAAI,GAAJ,IAAI,CAAkC;IAAG,CAAC;IAEvE,KAAK,CAAC,OAAO,CAAC,KAAgC;QAC5C,MAAM,EAAE,WAAW,EAAE,2BAA2B,EAAE,OAAO,EAAE,uBAAuB,EAAE,MAAM,EAAE,GAAG,IAAI,CAAC,IAAI,CAAC;QAEzG,IAAI,WAAW,KAAK,WAAW,EAAE,CAAC;YAChC,MAAM,QAAQ,GAAG,MAAM,OAAO,CAAC,KAAK,CAAC,GAAG,EAAE,KAAK,CAAC,SAAS,CAAC,CAAC;YAC3D,IAAI,CAAC,QAAQ,EAAE,CAAC;gBACd,MAAM,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,KAAK,CAAC,GAAG,CAAC,EAAE,EAAE,EAAE,wDAAwD,CAAC,CAAC;gBAC/F,OAAO,EAAE,MAAM,EAAE,UAAU,EAAE,MAAM,EAAE,wDAAwD,EAAE,CAAC;YAClG,CAAC;YACD,OAAO,EAAE,MAAM,EAAE,UAAU,EAAE,KAAK,EAAE,KAAK,CAAC,GAAG,CAAC,EAAE,EAAE,CAAC;QACrD,CAAC;QAED,MAAM,SAAS,GAAG,cAAc,CAAC,KAAK,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;QAC/C,MAAM,SAAS,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,IAAI,CAAC,GAAG,EAAE,CAAC,IAAI,IAAI,EAAE,CAAC,CAAC,EAAE,CAAC,WAAW,EAAE,CAAC;QAC1E,MAAM,OAAO,GAAyB;YACpC,sBAAsB,EAAE,SAAS;YACjC,GAAG,EAAE,KAAK,CAAC,GAAG;YACd,OAAO,EAAE,KAAK,CAAC,GAAG,CAAC,OAAO,IAAI,QAAQ;YACtC,QAAQ,EAAE,KAAK,CAAC,GAAG,CAAC,QAAQ;YAC5B,aAAa,EAAE,KAAK,CAAC,aAAa;YAClC,SAAS;SACV,CAAC;QAEF,MAAM,2BAA2B,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAChD,uBAAuB,CAAC,OAAO,CAAC,CAAC;QACjC,MAAM,CAAC,IAAI,CACT,EAAE,SAAS,EAAE,KAAK,EAAE,KAAK,CAAC,GAAG,CAAC,EAAE,EAAE,aAAa,EAAE,KAAK,CAAC,aAAa,EAAE,EACtE,+DAA+D,CAChE,CAAC;QAEF,OAAO,EAAE,MAAM,EAAE,SAAS,EAAE,SAAS,EAAE,CAAC;IAC1C,CAAC;CACF"}
@@ -0,0 +1,11 @@
1
+ import type { PendingReviewRequestGateway } from '../../../modules/review-execution/entities/pendingReviewRequest/pendingReviewRequest.gateway.js';
2
+ import type { PendingReviewRequest } from '../../../modules/review-execution/entities/pendingReviewRequest/pendingReviewRequest.schema.js';
3
+ export interface ListPendingReviewsDependencies {
4
+ pendingReviewRequestGateway: PendingReviewRequestGateway;
5
+ }
6
+ export declare class ListPendingReviewsUseCase {
7
+ private readonly deps;
8
+ constructor(deps: ListPendingReviewsDependencies);
9
+ execute(): Promise<PendingReviewRequest[]>;
10
+ }
11
+ //# sourceMappingURL=listPendingReviews.usecase.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"listPendingReviews.usecase.d.ts","sourceRoot":"","sources":["../../../../src/modules/review-execution/usecases/listPendingReviews.usecase.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,2BAA2B,EAAE,MAAM,0FAA0F,CAAC;AAC5I,OAAO,KAAK,EAAE,oBAAoB,EAAE,MAAM,yFAAyF,CAAC;AAEpI,MAAM,WAAW,8BAA8B;IAC7C,2BAA2B,EAAE,2BAA2B,CAAC;CAC1D;AAED,qBAAa,yBAAyB;IACxB,OAAO,CAAC,QAAQ,CAAC,IAAI;gBAAJ,IAAI,EAAE,8BAA8B;IAE3D,OAAO,IAAI,OAAO,CAAC,oBAAoB,EAAE,CAAC;CAGjD"}
@@ -0,0 +1,10 @@
1
+ export class ListPendingReviewsUseCase {
2
+ deps;
3
+ constructor(deps) {
4
+ this.deps = deps;
5
+ }
6
+ async execute() {
7
+ return this.deps.pendingReviewRequestGateway.listAll();
8
+ }
9
+ }
10
+ //# sourceMappingURL=listPendingReviews.usecase.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"listPendingReviews.usecase.js","sourceRoot":"","sources":["../../../../src/modules/review-execution/usecases/listPendingReviews.usecase.ts"],"names":[],"mappings":"AAOA,MAAM,OAAO,yBAAyB;IACP;IAA7B,YAA6B,IAAoC;QAApC,SAAI,GAAJ,IAAI,CAAgC;IAAG,CAAC;IAErE,KAAK,CAAC,OAAO;QACX,OAAO,IAAI,CAAC,IAAI,CAAC,2BAA2B,CAAC,OAAO,EAAE,CAAC;IACzD,CAAC;CACF"}
@@ -1,5 +1,6 @@
1
1
  import type { FastifyPluginAsync } from 'fastify';
2
2
  import type { RepositoryConfig } from '../../../../../config/loader.js';
3
+ import type { GateClaudeInvocationUseCase } from '../../../../../modules/review-execution/usecases/gateClaudeInvocation.usecase.js';
3
4
  import type { ClaudeInvokerDependencies } from '../../../../../frameworks/claude/claudeInvoker.js';
4
5
  import type { ReviewRequestTrackingGateway } from '../../gateways/reviewRequestTracking.gateway.js';
5
6
  import type { RecordReviewCompletionUseCase } from '../../../../../modules/tracking/usecases/tracking/recordReviewCompletion.usecase.js';
@@ -27,6 +28,7 @@ export interface MrTrackingAdvancedRoutesOptions {
27
28
  enforceBudget: Pick<EnforceBudgetUseCase, 'execute'>;
28
29
  broadcastBudgetExceeded: (payload: BudgetExceededPayload) => void;
29
30
  claudeInvokerDeps?: ClaudeInvokerDependencies;
31
+ gateClaudeInvocation?: GateClaudeInvocationUseCase;
30
32
  logger: Logger;
31
33
  }
32
34
  export declare const mrTrackingAdvancedRoutes: FastifyPluginAsync<MrTrackingAdvancedRoutesOptions>;
@@ -1 +1 @@
1
- {"version":3,"file":"mrTrackingAdvanced.routes.d.ts","sourceRoot":"","sources":["../../../../../../src/modules/tracking/interface-adapters/controllers/http/mrTrackingAdvanced.routes.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,kBAAkB,EAAE,MAAM,SAAS,CAAC;AAClD,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,oBAAoB,CAAC;AAM3D,OAAO,KAAK,EAAE,yBAAyB,EAAE,MAAM,sCAAsC,CAAC;AACtF,OAAO,KAAK,EAAE,4BAA4B,EAAE,MAAM,iDAAiD,CAAC;AACpG,OAAO,KAAK,EAAE,6BAA6B,EAAE,MAAM,wEAAwE,CAAC;AAC5H,OAAO,KAAK,EAAE,kBAAkB,EAAE,MAAM,6DAA6D,CAAC;AAItG,OAAO,KAAK,EAAE,8BAA8B,EAAE,MAAM,4FAA4F,CAAC;AACjJ,OAAO,KAAK,EAAE,wBAAwB,EAAE,MAAM,0FAA0F,CAAC;AACzI,OAAO,KAAK,EAAE,wBAAwB,EAAE,MAAM,0FAA0F,CAAC;AACzI,OAAO,KAAK,EAAE,8BAA8B,EAAE,MAAM,gGAAgG,CAAC;AACrJ,OAAO,KAAK,EAAE,8BAA8B,EAAE,MAAM,gGAAgG,CAAC;AAErJ,OAAO,KAAK,EAAE,2BAA2B,EAAE,MAAM,4FAA4F,CAAC;AAC9I,OAAO,KAAK,EAAE,2BAA2B,EAAE,MAAM,4FAA4F,CAAC;AAC9I,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,MAAM,CAAC;AACnC,OAAO,KAAK,EAAE,oBAAoB,EAAE,MAAM,4EAA4E,CAAC;AACvH,OAAO,KAAK,EAAE,qBAAqB,EAAE,MAAM,qBAAqB,CAAC;AAEjE,KAAK,QAAQ,GAAG,QAAQ,GAAG,QAAQ,CAAC;AAEpC,MAAM,WAAW,+BAA+B;IAC9C,eAAe,EAAE,MAAM,gBAAgB,EAAE,CAAC;IAC1C,4BAA4B,EAAE,4BAA4B,CAAC;IAC3D,oBAAoB,EAAE,8BAA8B,CAAC;IACrD,yBAAyB,EAAE,CACzB,QAAQ,EAAE,QAAQ,KACf,wBAAwB,GAAG,wBAAwB,CAAC;IACzD,+BAA+B,EAAE,CAC/B,QAAQ,EAAE,QAAQ,KACf,8BAA8B,GAAG,8BAA8B,CAAC;IACrE,4BAA4B,EAAE,CAC5B,QAAQ,EAAE,QAAQ,KACf,2BAA2B,GAAG,2BAA2B,CAAC;IAC/D,wBAAwB,EAAE,CAAC,QAAQ,EAAE,QAAQ,KAAK,kBAAkB,CAAC;IACrE,sBAAsB,EAAE,6BAA6B,CAAC;IACtD,aAAa,EAAE,IAAI,CAAC,oBAAoB,EAAE,SAAS,CAAC,CAAC;IACrD,uBAAuB,EAAE,CAAC,OAAO,EAAE,qBAAqB,KAAK,IAAI,CAAC;IAClE,iBAAiB,CAAC,EAAE,yBAAyB,CAAC;IAC9C,MAAM,EAAE,MAAM,CAAC;CAChB;AAeD,eAAO,MAAM,wBAAwB,EAAE,kBAAkB,CAAC,+BAA+B,CAiZxF,CAAC"}
1
+ {"version":3,"file":"mrTrackingAdvanced.routes.d.ts","sourceRoot":"","sources":["../../../../../../src/modules/tracking/interface-adapters/controllers/http/mrTrackingAdvanced.routes.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,kBAAkB,EAAE,MAAM,SAAS,CAAC;AAClD,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,oBAAoB,CAAC;AAG3D,OAAO,KAAK,EAAE,2BAA2B,EAAE,MAAM,qEAAqE,CAAC;AAIvH,OAAO,KAAK,EAAE,yBAAyB,EAAE,MAAM,sCAAsC,CAAC;AACtF,OAAO,KAAK,EAAE,4BAA4B,EAAE,MAAM,iDAAiD,CAAC;AACpG,OAAO,KAAK,EAAE,6BAA6B,EAAE,MAAM,wEAAwE,CAAC;AAC5H,OAAO,KAAK,EAAE,kBAAkB,EAAE,MAAM,6DAA6D,CAAC;AAItG,OAAO,KAAK,EAAE,8BAA8B,EAAE,MAAM,4FAA4F,CAAC;AACjJ,OAAO,KAAK,EAAE,wBAAwB,EAAE,MAAM,0FAA0F,CAAC;AACzI,OAAO,KAAK,EAAE,wBAAwB,EAAE,MAAM,0FAA0F,CAAC;AACzI,OAAO,KAAK,EAAE,8BAA8B,EAAE,MAAM,gGAAgG,CAAC;AACrJ,OAAO,KAAK,EAAE,8BAA8B,EAAE,MAAM,gGAAgG,CAAC;AAErJ,OAAO,KAAK,EAAE,2BAA2B,EAAE,MAAM,4FAA4F,CAAC;AAC9I,OAAO,KAAK,EAAE,2BAA2B,EAAE,MAAM,4FAA4F,CAAC;AAC9I,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,MAAM,CAAC;AACnC,OAAO,KAAK,EAAE,oBAAoB,EAAE,MAAM,4EAA4E,CAAC;AACvH,OAAO,KAAK,EAAE,qBAAqB,EAAE,MAAM,qBAAqB,CAAC;AAEjE,KAAK,QAAQ,GAAG,QAAQ,GAAG,QAAQ,CAAC;AAEpC,MAAM,WAAW,+BAA+B;IAC9C,eAAe,EAAE,MAAM,gBAAgB,EAAE,CAAC;IAC1C,4BAA4B,EAAE,4BAA4B,CAAC;IAC3D,oBAAoB,EAAE,8BAA8B,CAAC;IACrD,yBAAyB,EAAE,CACzB,QAAQ,EAAE,QAAQ,KACf,wBAAwB,GAAG,wBAAwB,CAAC;IACzD,+BAA+B,EAAE,CAC/B,QAAQ,EAAE,QAAQ,KACf,8BAA8B,GAAG,8BAA8B,CAAC;IACrE,4BAA4B,EAAE,CAC5B,QAAQ,EAAE,QAAQ,KACf,2BAA2B,GAAG,2BAA2B,CAAC;IAC/D,wBAAwB,EAAE,CAAC,QAAQ,EAAE,QAAQ,KAAK,kBAAkB,CAAC;IACrE,sBAAsB,EAAE,6BAA6B,CAAC;IACtD,aAAa,EAAE,IAAI,CAAC,oBAAoB,EAAE,SAAS,CAAC,CAAC;IACrD,uBAAuB,EAAE,CAAC,OAAO,EAAE,qBAAqB,KAAK,IAAI,CAAC;IAClE,iBAAiB,CAAC,EAAE,yBAAyB,CAAC;IAC9C,oBAAoB,CAAC,EAAE,2BAA2B,CAAC;IACnD,MAAM,EAAE,MAAM,CAAC;CAChB;AAeD,eAAO,MAAM,wBAAwB,EAAE,kBAAkB,CAAC,+BAA+B,CAuaxF,CAAC"}
@@ -18,7 +18,7 @@ function validateProjectPath(path) {
18
18
  return { valid: true, path: trimmed };
19
19
  }
20
20
  export const mrTrackingAdvancedRoutes = async (fastify, opts) => {
21
- const { getRepositories, reviewRequestTrackingGateway, reviewContextGateway: contextGateway, threadFetchGatewayFactory, diffMetadataFetchGatewayFactory, diffStatsFetchGatewayFactory, createSyncThreadsUseCase, recordReviewCompletion, enforceBudget, broadcastBudgetExceeded, claudeInvokerDeps, logger, } = opts;
21
+ const { getRepositories, reviewRequestTrackingGateway, reviewContextGateway: contextGateway, threadFetchGatewayFactory, diffMetadataFetchGatewayFactory, diffStatsFetchGatewayFactory, createSyncThreadsUseCase, recordReviewCompletion, enforceBudget, broadcastBudgetExceeded, claudeInvokerDeps, gateClaudeInvocation, logger, } = opts;
22
22
  fastify.post('/api/mr-tracking/followup', async (request, reply) => {
23
23
  const body = request.body;
24
24
  const { mrId, projectPath } = body;
@@ -74,7 +74,7 @@ export const mrTrackingAdvancedRoutes = async (fastify, opts) => {
74
74
  reply.code(200);
75
75
  return { status: 'rejected', reason: 'budget-exceeded' };
76
76
  }
77
- const enqueued = await enqueueReview({
77
+ const manualFollowupJob = {
78
78
  id: jobId,
79
79
  platform: platform,
80
80
  projectPath: gitProjectPath,
@@ -85,7 +85,8 @@ export const mrTrackingAdvancedRoutes = async (fastify, opts) => {
85
85
  sourceBranch: 'unknown',
86
86
  targetBranch: 'unknown',
87
87
  jobType: 'followup',
88
- }, async (job, signal) => {
88
+ };
89
+ const manualFollowupProcessor = async (job, signal) => {
89
90
  sendNotification('Review followup started', `MR !${job.mrNumber}`, logger);
90
91
  // Use injected gateways to fetch threads + diff metadata.
91
92
  const threadFetchGateway = threadFetchGatewayFactory(job.platform);
@@ -192,7 +193,24 @@ export const mrTrackingAdvancedRoutes = async (fastify, opts) => {
192
193
  else if (!result.cancelled) {
193
194
  sendNotification('Review followup failed', `MR !${job.mrNumber}`, logger);
194
195
  }
195
- });
196
+ };
197
+ if (gateClaudeInvocation) {
198
+ const gateResult = await gateClaudeInvocation.execute({
199
+ job: manualFollowupJob,
200
+ triggerSource: 'dashboard-manual',
201
+ processor: manualFollowupProcessor,
202
+ });
203
+ if (gateResult.status === 'pending') {
204
+ logInfo('Manual followup parked for human confirmation', { mrId, mrNumber, pendingId: gateResult.pendingId });
205
+ return { success: true, status: 'pending-confirmation', pendingId: gateResult.pendingId };
206
+ }
207
+ if (gateResult.status === 'rejected') {
208
+ return { success: false, error: 'Review already in progress or recently performed' };
209
+ }
210
+ logInfo('Followup triggered manually', { mrId, mrNumber, skill });
211
+ return { success: true, jobId, message: 'Followup review in progress' };
212
+ }
213
+ const enqueued = await enqueueReview(manualFollowupJob, manualFollowupProcessor);
196
214
  if (!enqueued) {
197
215
  return { success: false, error: 'Review already in progress or recently performed' };
198
216
  }