reviewflow 3.39.3 → 3.40.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +21 -0
- package/dist/config/projectConfig.d.ts +1 -0
- package/dist/config/projectConfig.d.ts.map +1 -1
- package/dist/config/projectConfig.js +13 -0
- package/dist/config/projectConfig.js.map +1 -1
- package/dist/frameworks/config/configLoader.d.ts +1 -0
- package/dist/frameworks/config/configLoader.d.ts.map +1 -1
- package/dist/frameworks/config/configLoader.js +15 -1
- package/dist/frameworks/config/configLoader.js.map +1 -1
- package/dist/main/routes.d.ts.map +1 -1
- package/dist/main/routes.js +18 -2
- package/dist/main/routes.js.map +1 -1
- package/dist/modules/platform-integration/entities/webhookEvent/webhookEvent.d.ts +1 -0
- package/dist/modules/platform-integration/entities/webhookEvent/webhookEvent.d.ts.map +1 -1
- package/dist/modules/platform-integration/interface-adapters/controllers/webhook/diffSizeGuard.helper.d.ts +31 -0
- package/dist/modules/platform-integration/interface-adapters/controllers/webhook/diffSizeGuard.helper.d.ts.map +1 -0
- package/dist/modules/platform-integration/interface-adapters/controllers/webhook/diffSizeGuard.helper.js +57 -0
- package/dist/modules/platform-integration/interface-adapters/controllers/webhook/diffSizeGuard.helper.js.map +1 -0
- package/dist/modules/platform-integration/interface-adapters/controllers/webhook/github.controller.d.ts +4 -2
- package/dist/modules/platform-integration/interface-adapters/controllers/webhook/github.controller.d.ts.map +1 -1
- package/dist/modules/platform-integration/interface-adapters/controllers/webhook/github.controller.js +180 -182
- package/dist/modules/platform-integration/interface-adapters/controllers/webhook/github.controller.js.map +1 -1
- package/dist/modules/platform-integration/interface-adapters/controllers/webhook/gitlab.controller.d.ts +4 -2
- package/dist/modules/platform-integration/interface-adapters/controllers/webhook/gitlab.controller.d.ts.map +1 -1
- package/dist/modules/platform-integration/interface-adapters/controllers/webhook/gitlab.controller.js +179 -203
- package/dist/modules/platform-integration/interface-adapters/controllers/webhook/gitlab.controller.js.map +1 -1
- package/dist/modules/platform-integration/interface-adapters/controllers/webhook/webhookReply.d.ts +71 -0
- package/dist/modules/platform-integration/interface-adapters/controllers/webhook/webhookReply.d.ts.map +1 -0
- package/dist/modules/platform-integration/interface-adapters/controllers/webhook/webhookReply.js +104 -0
- package/dist/modules/platform-integration/interface-adapters/controllers/webhook/webhookReply.js.map +1 -0
- package/dist/modules/platform-integration/interface-adapters/gateways/changedFilesFetch.github.gateway.d.ts +10 -0
- package/dist/modules/platform-integration/interface-adapters/gateways/changedFilesFetch.github.gateway.d.ts.map +1 -0
- package/dist/modules/platform-integration/interface-adapters/gateways/changedFilesFetch.github.gateway.js +37 -0
- package/dist/modules/platform-integration/interface-adapters/gateways/changedFilesFetch.github.gateway.js.map +1 -0
- package/dist/modules/platform-integration/interface-adapters/gateways/changedFilesFetch.gitlab.gateway.d.ts +10 -0
- package/dist/modules/platform-integration/interface-adapters/gateways/changedFilesFetch.gitlab.gateway.d.ts.map +1 -0
- package/dist/modules/platform-integration/interface-adapters/gateways/changedFilesFetch.gitlab.gateway.js +41 -0
- package/dist/modules/platform-integration/interface-adapters/gateways/changedFilesFetch.gitlab.gateway.js.map +1 -0
- package/dist/modules/platform-integration/usecases/guardDiffSize.usecase.d.ts +26 -0
- package/dist/modules/platform-integration/usecases/guardDiffSize.usecase.d.ts.map +1 -0
- package/dist/modules/platform-integration/usecases/guardDiffSize.usecase.js +42 -0
- package/dist/modules/platform-integration/usecases/guardDiffSize.usecase.js.map +1 -0
- package/dist/modules/platform-integration/usecases/processReviewRequest.usecase.d.ts +39 -0
- package/dist/modules/platform-integration/usecases/processReviewRequest.usecase.d.ts.map +1 -0
- package/dist/modules/platform-integration/usecases/processReviewRequest.usecase.js +45 -0
- package/dist/modules/platform-integration/usecases/processReviewRequest.usecase.js.map +1 -0
- package/dist/modules/platform-integration/usecases/processWebhook.usecase.d.ts +16 -0
- package/dist/modules/platform-integration/usecases/processWebhook.usecase.d.ts.map +1 -1
- package/dist/modules/platform-integration/usecases/processWebhook.usecase.js +44 -1
- package/dist/modules/platform-integration/usecases/processWebhook.usecase.js.map +1 -1
- package/dist/modules/shared-kernel/entities/diffSizeGate/changedFilesFetch.gateway.d.ts +5 -0
- package/dist/modules/shared-kernel/entities/diffSizeGate/changedFilesFetch.gateway.d.ts.map +1 -0
- package/dist/modules/shared-kernel/entities/diffSizeGate/changedFilesFetch.gateway.js +2 -0
- package/dist/modules/shared-kernel/entities/diffSizeGate/changedFilesFetch.gateway.js.map +1 -0
- package/dist/modules/shared-kernel/entities/diffSizeGate/diffSizeGate.d.ts +17 -0
- package/dist/modules/shared-kernel/entities/diffSizeGate/diffSizeGate.d.ts.map +1 -0
- package/dist/modules/shared-kernel/entities/diffSizeGate/diffSizeGate.js +20 -0
- package/dist/modules/shared-kernel/entities/diffSizeGate/diffSizeGate.js.map +1 -0
- package/dist/modules/statistics-insights/usecases/insights/computeInsightsWithPersistence.usecase.d.ts.map +1 -1
- package/dist/modules/statistics-insights/usecases/insights/computeInsightsWithPersistence.usecase.js +46 -8
- package/dist/modules/statistics-insights/usecases/insights/computeInsightsWithPersistence.usecase.js.map +1 -1
- package/dist/modules/tracking/interface-adapters/controllers/http/mrTracking.routes.d.ts +2 -0
- package/dist/modules/tracking/interface-adapters/controllers/http/mrTracking.routes.d.ts.map +1 -1
- package/dist/modules/tracking/interface-adapters/controllers/http/mrTracking.routes.js +6 -3
- package/dist/modules/tracking/interface-adapters/controllers/http/mrTracking.routes.js.map +1 -1
- package/dist/modules/tracking/interface-adapters/presenters/mrDiffStats.presenter.d.ts +13 -0
- package/dist/modules/tracking/interface-adapters/presenters/mrDiffStats.presenter.d.ts.map +1 -0
- package/dist/modules/tracking/interface-adapters/presenters/mrDiffStats.presenter.js +28 -0
- package/dist/modules/tracking/interface-adapters/presenters/mrDiffStats.presenter.js.map +1 -0
- package/dist/tests/acceptance/180-quality-threshold-block-approval-iter-B.acceptance.test.js +3 -3
- package/dist/tests/acceptance/180-quality-threshold-block-approval-iter-B.acceptance.test.js.map +1 -1
- package/dist/tests/acceptance/180-quality-threshold-block-approval-iter-C.acceptance.test.js +26 -2
- package/dist/tests/acceptance/180-quality-threshold-block-approval-iter-C.acceptance.test.js.map +1 -1
- package/dist/tests/acceptance/197-trusted-actor-provenance-gate.acceptance.test.js +20 -12
- package/dist/tests/acceptance/197-trusted-actor-provenance-gate.acceptance.test.js.map +1 -1
- package/dist/tests/acceptance/200-webhook-event-idempotency.acceptance.test.js +16 -8
- package/dist/tests/acceptance/200-webhook-event-idempotency.acceptance.test.js.map +1 -1
- package/dist/tests/acceptance/209-mr-size-guard.acceptance.test.d.ts +19 -0
- package/dist/tests/acceptance/209-mr-size-guard.acceptance.test.d.ts.map +1 -0
- package/dist/tests/acceptance/209-mr-size-guard.acceptance.test.js +181 -0
- package/dist/tests/acceptance/209-mr-size-guard.acceptance.test.js.map +1 -0
- package/dist/tests/acceptance/46-github-followup-review-on-push.acceptance.test.js +10 -6
- package/dist/tests/acceptance/46-github-followup-review-on-push.acceptance.test.js.map +1 -1
- package/dist/tests/acceptance/73-process-webhook.acceptance.test.js +2 -0
- package/dist/tests/acceptance/73-process-webhook.acceptance.test.js.map +1 -1
- package/dist/tests/acceptance/73-stage4-controller-thinning.acceptance.test.d.ts +2 -0
- package/dist/tests/acceptance/73-stage4-controller-thinning.acceptance.test.d.ts.map +1 -0
- package/dist/tests/acceptance/73-stage4-controller-thinning.acceptance.test.js +226 -0
- package/dist/tests/acceptance/73-stage4-controller-thinning.acceptance.test.js.map +1 -0
- package/dist/tests/factories/changedFiles.factory.d.ts +6 -0
- package/dist/tests/factories/changedFiles.factory.d.ts.map +1 -0
- package/dist/tests/factories/changedFiles.factory.js +14 -0
- package/dist/tests/factories/changedFiles.factory.js.map +1 -0
- package/dist/tests/stubs/changedFilesFetch.stub.d.ts +12 -0
- package/dist/tests/stubs/changedFilesFetch.stub.d.ts.map +1 -0
- package/dist/tests/stubs/changedFilesFetch.stub.js +21 -0
- package/dist/tests/stubs/changedFilesFetch.stub.js.map +1 -0
- package/dist/tests/units/config/projectConfig.test.js +49 -0
- package/dist/tests/units/config/projectConfig.test.js.map +1 -1
- package/dist/tests/units/frameworks/config/configLoader.test.js +20 -0
- package/dist/tests/units/frameworks/config/configLoader.test.js.map +1 -1
- package/dist/tests/units/interface-adapters/controllers/webhook/github.controller.test.js +52 -70
- package/dist/tests/units/interface-adapters/controllers/webhook/github.controller.test.js.map +1 -1
- package/dist/tests/units/interface-adapters/controllers/webhook/gitlab.controller.test.js +57 -47
- package/dist/tests/units/interface-adapters/controllers/webhook/gitlab.controller.test.js.map +1 -1
- package/dist/tests/units/modules/platform-integration/interface-adapters/controllers/webhook/diffSizeGuard.helper.test.d.ts +2 -0
- package/dist/tests/units/modules/platform-integration/interface-adapters/controllers/webhook/diffSizeGuard.helper.test.d.ts.map +1 -0
- package/dist/tests/units/modules/platform-integration/interface-adapters/controllers/webhook/diffSizeGuard.helper.test.js +148 -0
- package/dist/tests/units/modules/platform-integration/interface-adapters/controllers/webhook/diffSizeGuard.helper.test.js.map +1 -0
- package/dist/tests/units/modules/platform-integration/interface-adapters/controllers/webhook/gitlabIdempotency.controller.test.js +13 -5
- package/dist/tests/units/modules/platform-integration/interface-adapters/controllers/webhook/gitlabIdempotency.controller.test.js.map +1 -1
- package/dist/tests/units/modules/platform-integration/interface-adapters/controllers/webhook/webhookReply.test.d.ts +2 -0
- package/dist/tests/units/modules/platform-integration/interface-adapters/controllers/webhook/webhookReply.test.d.ts.map +1 -0
- package/dist/tests/units/modules/platform-integration/interface-adapters/controllers/webhook/webhookReply.test.js +309 -0
- package/dist/tests/units/modules/platform-integration/interface-adapters/controllers/webhook/webhookReply.test.js.map +1 -0
- package/dist/tests/units/modules/platform-integration/interface-adapters/gateways/changedFilesFetch.github.gateway.test.d.ts +2 -0
- package/dist/tests/units/modules/platform-integration/interface-adapters/gateways/changedFilesFetch.github.gateway.test.d.ts.map +1 -0
- package/dist/tests/units/modules/platform-integration/interface-adapters/gateways/changedFilesFetch.github.gateway.test.js +52 -0
- package/dist/tests/units/modules/platform-integration/interface-adapters/gateways/changedFilesFetch.github.gateway.test.js.map +1 -0
- package/dist/tests/units/modules/platform-integration/interface-adapters/gateways/changedFilesFetch.gitlab.gateway.test.d.ts +2 -0
- package/dist/tests/units/modules/platform-integration/interface-adapters/gateways/changedFilesFetch.gitlab.gateway.test.d.ts.map +1 -0
- package/dist/tests/units/modules/platform-integration/interface-adapters/gateways/changedFilesFetch.gitlab.gateway.test.js +57 -0
- package/dist/tests/units/modules/platform-integration/interface-adapters/gateways/changedFilesFetch.gitlab.gateway.test.js.map +1 -0
- package/dist/tests/units/modules/platform-integration/usecases/guardDiffSize.usecase.test.d.ts +2 -0
- package/dist/tests/units/modules/platform-integration/usecases/guardDiffSize.usecase.test.d.ts.map +1 -0
- package/dist/tests/units/modules/platform-integration/usecases/guardDiffSize.usecase.test.js +91 -0
- package/dist/tests/units/modules/platform-integration/usecases/guardDiffSize.usecase.test.js.map +1 -0
- package/dist/tests/units/modules/platform-integration/usecases/processReviewRequest.usecase.test.d.ts +2 -0
- package/dist/tests/units/modules/platform-integration/usecases/processReviewRequest.usecase.test.d.ts.map +1 -0
- package/dist/tests/units/modules/platform-integration/usecases/processReviewRequest.usecase.test.js +170 -0
- package/dist/tests/units/modules/platform-integration/usecases/processReviewRequest.usecase.test.js.map +1 -0
- package/dist/tests/units/modules/platform-integration/usecases/processWebhook.usecase.test.js +99 -8
- package/dist/tests/units/modules/platform-integration/usecases/processWebhook.usecase.test.js.map +1 -1
- package/dist/tests/units/modules/shared-kernel/entities/diffSizeGate/diffSizeGate.test.d.ts +2 -0
- package/dist/tests/units/modules/shared-kernel/entities/diffSizeGate/diffSizeGate.test.d.ts.map +1 -0
- package/dist/tests/units/modules/shared-kernel/entities/diffSizeGate/diffSizeGate.test.js +67 -0
- package/dist/tests/units/modules/shared-kernel/entities/diffSizeGate/diffSizeGate.test.js.map +1 -0
- package/dist/tests/units/modules/tracking/interface-adapters/controllers/http/mrTracking.routes.test.js +66 -0
- package/dist/tests/units/modules/tracking/interface-adapters/controllers/http/mrTracking.routes.test.js.map +1 -1
- package/dist/tests/units/modules/tracking/interface-adapters/presenters/mrDiffStats.presenter.test.d.ts +2 -0
- package/dist/tests/units/modules/tracking/interface-adapters/presenters/mrDiffStats.presenter.test.d.ts.map +1 -0
- package/dist/tests/units/modules/tracking/interface-adapters/presenters/mrDiffStats.presenter.test.js +84 -0
- package/dist/tests/units/modules/tracking/interface-adapters/presenters/mrDiffStats.presenter.test.js.map +1 -0
- package/dist/tests/units/usecases/insights/computeInsightsWithPersistence.usecase.test.js +76 -0
- package/dist/tests/units/usecases/insights/computeInsightsWithPersistence.usecase.test.js.map +1 -1
- package/package.json +1 -1
package/dist/modules/platform-integration/interface-adapters/controllers/webhook/webhookReply.d.ts
ADDED
|
@@ -0,0 +1,71 @@
|
|
|
1
|
+
import type { ReviewRequestVerdict } from '../../../../../modules/platform-integration/usecases/processReviewRequest.usecase.js';
|
|
2
|
+
import type { BudgetStatus } from '../../../../../modules/token-accounting/entities/budget/budgetStatus.js';
|
|
3
|
+
export type WebhookReplyResult = {
|
|
4
|
+
kind: 'cleaned';
|
|
5
|
+
mergeRequestNumber: number;
|
|
6
|
+
jobCancelled: boolean;
|
|
7
|
+
trackingArchived: boolean;
|
|
8
|
+
} | {
|
|
9
|
+
kind: 'merged';
|
|
10
|
+
mergeRequestNumber: number;
|
|
11
|
+
} | {
|
|
12
|
+
kind: 'approved';
|
|
13
|
+
mergeRequestNumber: number;
|
|
14
|
+
} | {
|
|
15
|
+
kind: 'unapproved';
|
|
16
|
+
mergeRequestNumber: number;
|
|
17
|
+
reason: string;
|
|
18
|
+
} | {
|
|
19
|
+
kind: 'ignored-with-number';
|
|
20
|
+
mergeRequestNumber: number;
|
|
21
|
+
reason: string;
|
|
22
|
+
} | {
|
|
23
|
+
kind: 'ignored';
|
|
24
|
+
reason: string;
|
|
25
|
+
} | {
|
|
26
|
+
kind: 'rejected';
|
|
27
|
+
reason: string;
|
|
28
|
+
} | {
|
|
29
|
+
kind: 'queued';
|
|
30
|
+
jobId: string;
|
|
31
|
+
mergeRequestNumber: number;
|
|
32
|
+
} | {
|
|
33
|
+
kind: 'followup-queued';
|
|
34
|
+
jobId: string;
|
|
35
|
+
mergeRequestNumber: number;
|
|
36
|
+
} | {
|
|
37
|
+
kind: 'deduplicated';
|
|
38
|
+
jobId: string;
|
|
39
|
+
reason: string;
|
|
40
|
+
} | {
|
|
41
|
+
kind: 'pending-confirmation';
|
|
42
|
+
pendingId: string | null;
|
|
43
|
+
mergeRequestNumber: number;
|
|
44
|
+
} | {
|
|
45
|
+
kind: 'pending-confirmation-untrusted';
|
|
46
|
+
mergeRequestNumber: number;
|
|
47
|
+
};
|
|
48
|
+
export interface WebhookReplyOptions {
|
|
49
|
+
numberKey: 'mrNumber' | 'prNumber';
|
|
50
|
+
}
|
|
51
|
+
interface WebhookReplyTarget {
|
|
52
|
+
status(code: number): {
|
|
53
|
+
send(body: unknown): unknown;
|
|
54
|
+
};
|
|
55
|
+
}
|
|
56
|
+
export declare function sendWebhookReply(reply: WebhookReplyTarget, result: WebhookReplyResult, options: WebhookReplyOptions): void;
|
|
57
|
+
export interface ReviewRequestReplyOptions {
|
|
58
|
+
numberKey: 'mrNumber' | 'prNumber';
|
|
59
|
+
mergeRequestNumber: number;
|
|
60
|
+
jobId: string;
|
|
61
|
+
/**
|
|
62
|
+
* Initial reviews distinguish a freshly queued job from a deduplicated one
|
|
63
|
+
* (`'queued'` + a `deduplicated` body). Followup pushes collapse both outcomes
|
|
64
|
+
* into a single `'followup-queued'` reply, matching the historical wire contract.
|
|
65
|
+
*/
|
|
66
|
+
flow: 'initial' | 'followup';
|
|
67
|
+
onBudgetExceeded: (status: BudgetStatus) => void;
|
|
68
|
+
}
|
|
69
|
+
export declare function sendReviewRequestReply(reply: WebhookReplyTarget, verdict: ReviewRequestVerdict, options: ReviewRequestReplyOptions): void;
|
|
70
|
+
export {};
|
|
71
|
+
//# sourceMappingURL=webhookReply.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"webhookReply.d.ts","sourceRoot":"","sources":["../../../../../../src/modules/platform-integration/interface-adapters/controllers/webhook/webhookReply.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,oBAAoB,EAAE,MAAM,yEAAyE,CAAC;AACpH,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,4DAA4D,CAAC;AAE/F,MAAM,MAAM,kBAAkB,GAC1B;IACE,IAAI,EAAE,SAAS,CAAC;IAChB,kBAAkB,EAAE,MAAM,CAAC;IAC3B,YAAY,EAAE,OAAO,CAAC;IACtB,gBAAgB,EAAE,OAAO,CAAC;CAC3B,GACD;IAAE,IAAI,EAAE,QAAQ,CAAC;IAAC,kBAAkB,EAAE,MAAM,CAAA;CAAE,GAC9C;IAAE,IAAI,EAAE,UAAU,CAAC;IAAC,kBAAkB,EAAE,MAAM,CAAA;CAAE,GAChD;IAAE,IAAI,EAAE,YAAY,CAAC;IAAC,kBAAkB,EAAE,MAAM,CAAC;IAAC,MAAM,EAAE,MAAM,CAAA;CAAE,GAClE;IAAE,IAAI,EAAE,qBAAqB,CAAC;IAAC,kBAAkB,EAAE,MAAM,CAAC;IAAC,MAAM,EAAE,MAAM,CAAA;CAAE,GAC3E;IAAE,IAAI,EAAE,SAAS,CAAC;IAAC,MAAM,EAAE,MAAM,CAAA;CAAE,GACnC;IAAE,IAAI,EAAE,UAAU,CAAC;IAAC,MAAM,EAAE,MAAM,CAAA;CAAE,GACpC;IAAE,IAAI,EAAE,QAAQ,CAAC;IAAC,KAAK,EAAE,MAAM,CAAC;IAAC,kBAAkB,EAAE,MAAM,CAAA;CAAE,GAC7D;IAAE,IAAI,EAAE,iBAAiB,CAAC;IAAC,KAAK,EAAE,MAAM,CAAC;IAAC,kBAAkB,EAAE,MAAM,CAAA;CAAE,GACtE;IAAE,IAAI,EAAE,cAAc,CAAC;IAAC,KAAK,EAAE,MAAM,CAAC;IAAC,MAAM,EAAE,MAAM,CAAA;CAAE,GACvD;IAAE,IAAI,EAAE,sBAAsB,CAAC;IAAC,SAAS,EAAE,MAAM,GAAG,IAAI,CAAC;IAAC,kBAAkB,EAAE,MAAM,CAAA;CAAE,GACtF;IAAE,IAAI,EAAE,gCAAgC,CAAC;IAAC,kBAAkB,EAAE,MAAM,CAAA;CAAE,CAAC;AAE3E,MAAM,WAAW,mBAAmB;IAClC,SAAS,EAAE,UAAU,GAAG,UAAU,CAAC;CACpC;AAED,UAAU,kBAAkB;IAC1B,MAAM,CAAC,IAAI,EAAE,MAAM,GAAG;QAAE,IAAI,CAAC,IAAI,EAAE,OAAO,GAAG,OAAO,CAAA;KAAE,CAAC;CACxD;AAED,wBAAgB,gBAAgB,CAC9B,KAAK,EAAE,kBAAkB,EACzB,MAAM,EAAE,kBAAkB,EAC1B,OAAO,EAAE,mBAAmB,GAC3B,IAAI,CA0EN;AAED,MAAM,WAAW,yBAAyB;IACxC,SAAS,EAAE,UAAU,GAAG,UAAU,CAAC;IACnC,kBAAkB,EAAE,MAAM,CAAC;IAC3B,KAAK,EAAE,MAAM,CAAC;IACd;;;;OAIG;IACH,IAAI,EAAE,SAAS,GAAG,UAAU,CAAC;IAC7B,gBAAgB,EAAE,CAAC,MAAM,EAAE,YAAY,KAAK,IAAI,CAAC;CAClD;AAID,wBAAgB,sBAAsB,CACpC,KAAK,EAAE,kBAAkB,EACzB,OAAO,EAAE,oBAAoB,EAC7B,OAAO,EAAE,yBAAyB,GACjC,IAAI,CAgDN"}
|
package/dist/modules/platform-integration/interface-adapters/controllers/webhook/webhookReply.js
ADDED
|
@@ -0,0 +1,104 @@
|
|
|
1
|
+
export function sendWebhookReply(reply, result, options) {
|
|
2
|
+
const { numberKey } = options;
|
|
3
|
+
switch (result.kind) {
|
|
4
|
+
case 'cleaned':
|
|
5
|
+
reply.status(200).send({
|
|
6
|
+
status: 'cleaned',
|
|
7
|
+
[numberKey]: result.mergeRequestNumber,
|
|
8
|
+
jobCancelled: result.jobCancelled,
|
|
9
|
+
trackingArchived: result.trackingArchived,
|
|
10
|
+
});
|
|
11
|
+
return;
|
|
12
|
+
case 'merged':
|
|
13
|
+
reply.status(200).send({ status: 'merged', [numberKey]: result.mergeRequestNumber });
|
|
14
|
+
return;
|
|
15
|
+
case 'approved':
|
|
16
|
+
reply.status(200).send({ status: 'approved', [numberKey]: result.mergeRequestNumber });
|
|
17
|
+
return;
|
|
18
|
+
case 'unapproved':
|
|
19
|
+
reply.status(200).send({
|
|
20
|
+
status: 'unapproved',
|
|
21
|
+
[numberKey]: result.mergeRequestNumber,
|
|
22
|
+
reason: result.reason,
|
|
23
|
+
});
|
|
24
|
+
return;
|
|
25
|
+
case 'ignored-with-number':
|
|
26
|
+
reply.status(200).send({
|
|
27
|
+
status: 'ignored',
|
|
28
|
+
[numberKey]: result.mergeRequestNumber,
|
|
29
|
+
reason: result.reason,
|
|
30
|
+
});
|
|
31
|
+
return;
|
|
32
|
+
case 'ignored':
|
|
33
|
+
reply.status(200).send({ status: 'ignored', reason: result.reason });
|
|
34
|
+
return;
|
|
35
|
+
case 'rejected':
|
|
36
|
+
reply.status(200).send({ status: 'rejected', reason: result.reason });
|
|
37
|
+
return;
|
|
38
|
+
case 'queued':
|
|
39
|
+
reply.status(202).send({
|
|
40
|
+
status: 'queued',
|
|
41
|
+
jobId: result.jobId,
|
|
42
|
+
[numberKey]: result.mergeRequestNumber,
|
|
43
|
+
});
|
|
44
|
+
return;
|
|
45
|
+
case 'followup-queued':
|
|
46
|
+
reply.status(202).send({
|
|
47
|
+
status: 'followup-queued',
|
|
48
|
+
jobId: result.jobId,
|
|
49
|
+
[numberKey]: result.mergeRequestNumber,
|
|
50
|
+
});
|
|
51
|
+
return;
|
|
52
|
+
case 'deduplicated':
|
|
53
|
+
reply.status(200).send({
|
|
54
|
+
status: 'deduplicated',
|
|
55
|
+
jobId: result.jobId,
|
|
56
|
+
reason: result.reason,
|
|
57
|
+
});
|
|
58
|
+
return;
|
|
59
|
+
case 'pending-confirmation':
|
|
60
|
+
reply.status(202).send({
|
|
61
|
+
status: 'pending-confirmation',
|
|
62
|
+
pendingId: result.pendingId,
|
|
63
|
+
[numberKey]: result.mergeRequestNumber,
|
|
64
|
+
});
|
|
65
|
+
return;
|
|
66
|
+
case 'pending-confirmation-untrusted':
|
|
67
|
+
reply.status(202).send({
|
|
68
|
+
status: 'pending-confirmation',
|
|
69
|
+
reason: 'untrusted-actor',
|
|
70
|
+
[numberKey]: result.mergeRequestNumber,
|
|
71
|
+
});
|
|
72
|
+
return;
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
const DEDUPLICATED_REASON = 'Review already in progress or recently completed';
|
|
76
|
+
export function sendReviewRequestReply(reply, verdict, options) {
|
|
77
|
+
const { numberKey, mergeRequestNumber, jobId, flow, onBudgetExceeded } = options;
|
|
78
|
+
switch (verdict.type) {
|
|
79
|
+
case 'budget-exceeded':
|
|
80
|
+
onBudgetExceeded(verdict.status);
|
|
81
|
+
sendWebhookReply(reply, { kind: 'rejected', reason: 'budget-exceeded' }, { numberKey });
|
|
82
|
+
return;
|
|
83
|
+
case 'pending':
|
|
84
|
+
if (verdict.reason === 'untrusted-actor') {
|
|
85
|
+
sendWebhookReply(reply, { kind: 'pending-confirmation-untrusted', mergeRequestNumber }, { numberKey });
|
|
86
|
+
return;
|
|
87
|
+
}
|
|
88
|
+
sendWebhookReply(reply, { kind: 'pending-confirmation', pendingId: verdict.pendingId, mergeRequestNumber }, { numberKey });
|
|
89
|
+
return;
|
|
90
|
+
case 'queued':
|
|
91
|
+
sendWebhookReply(reply, flow === 'followup'
|
|
92
|
+
? { kind: 'followup-queued', jobId, mergeRequestNumber }
|
|
93
|
+
: { kind: 'queued', jobId, mergeRequestNumber }, { numberKey });
|
|
94
|
+
return;
|
|
95
|
+
case 'deduplicated':
|
|
96
|
+
if (flow === 'followup') {
|
|
97
|
+
sendWebhookReply(reply, { kind: 'followup-queued', jobId, mergeRequestNumber }, { numberKey });
|
|
98
|
+
return;
|
|
99
|
+
}
|
|
100
|
+
sendWebhookReply(reply, { kind: 'deduplicated', jobId, reason: DEDUPLICATED_REASON }, { numberKey });
|
|
101
|
+
return;
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
//# sourceMappingURL=webhookReply.js.map
|
package/dist/modules/platform-integration/interface-adapters/controllers/webhook/webhookReply.js.map
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"webhookReply.js","sourceRoot":"","sources":["../../../../../../src/modules/platform-integration/interface-adapters/controllers/webhook/webhookReply.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
import type { ChangedFilesFetchGateway } from '../../../../modules/shared-kernel/entities/diffSizeGate/changedFilesFetch.gateway.js';
|
|
2
|
+
import type { ChangedFile } from '../../../../modules/shared-kernel/entities/diffSizeGate/diffSizeGate.js';
|
|
3
|
+
import type { SimpleCommandExecutor } from '../../../../shared/foundation/commandExecutor.js';
|
|
4
|
+
export type CommandExecutor = SimpleCommandExecutor;
|
|
5
|
+
export declare class GitHubChangedFilesFetchGateway implements ChangedFilesFetchGateway {
|
|
6
|
+
private readonly executor;
|
|
7
|
+
constructor(executor: CommandExecutor);
|
|
8
|
+
fetchChangedFiles(projectPath: string, mergeRequestNumber: number): ChangedFile[] | null;
|
|
9
|
+
}
|
|
10
|
+
//# sourceMappingURL=changedFilesFetch.github.gateway.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"changedFilesFetch.github.gateway.d.ts","sourceRoot":"","sources":["../../../../../src/modules/platform-integration/interface-adapters/gateways/changedFilesFetch.github.gateway.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,wBAAwB,EAAE,MAAM,4EAA4E,CAAC;AAC3H,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,+DAA+D,CAAC;AACjG,OAAO,KAAK,EAAE,qBAAqB,EAAE,MAAM,wCAAwC,CAAC;AAEpF,MAAM,MAAM,eAAe,GAAG,qBAAqB,CAAC;AAyBpD,qBAAa,8BAA+B,YAAW,wBAAwB;IACjE,OAAO,CAAC,QAAQ,CAAC,QAAQ;gBAAR,QAAQ,EAAE,eAAe;IAEtD,iBAAiB,CAAC,WAAW,EAAE,MAAM,EAAE,kBAAkB,EAAE,MAAM,GAAG,WAAW,EAAE,GAAG,IAAI;CAczF"}
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
function readField(value, field) {
|
|
2
|
+
if (typeof value !== 'object' || value === null || !(field in value)) {
|
|
3
|
+
throw new Error(`GitHub pulls/files entry is missing "${field}"`);
|
|
4
|
+
}
|
|
5
|
+
return Reflect.get(value, field);
|
|
6
|
+
}
|
|
7
|
+
function toChangedFile(entry) {
|
|
8
|
+
const filename = readField(entry, 'filename');
|
|
9
|
+
const additions = readField(entry, 'additions');
|
|
10
|
+
const deletions = readField(entry, 'deletions');
|
|
11
|
+
if (typeof filename !== 'string' ||
|
|
12
|
+
typeof additions !== 'number' ||
|
|
13
|
+
typeof deletions !== 'number') {
|
|
14
|
+
throw new Error('GitHub pulls/files entry has invalid fields');
|
|
15
|
+
}
|
|
16
|
+
return { path: filename, additions, deletions };
|
|
17
|
+
}
|
|
18
|
+
export class GitHubChangedFilesFetchGateway {
|
|
19
|
+
executor;
|
|
20
|
+
constructor(executor) {
|
|
21
|
+
this.executor = executor;
|
|
22
|
+
}
|
|
23
|
+
fetchChangedFiles(projectPath, mergeRequestNumber) {
|
|
24
|
+
try {
|
|
25
|
+
const response = this.executor(`gh api --paginate repos/${projectPath}/pulls/${mergeRequestNumber}/files`);
|
|
26
|
+
const parsed = JSON.parse(response);
|
|
27
|
+
if (!Array.isArray(parsed)) {
|
|
28
|
+
return null;
|
|
29
|
+
}
|
|
30
|
+
return parsed.map(toChangedFile);
|
|
31
|
+
}
|
|
32
|
+
catch {
|
|
33
|
+
return null;
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
//# sourceMappingURL=changedFilesFetch.github.gateway.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"changedFilesFetch.github.gateway.js","sourceRoot":"","sources":["../../../../../src/modules/platform-integration/interface-adapters/gateways/changedFilesFetch.github.gateway.ts"],"names":[],"mappings":"AAMA,SAAS,SAAS,CAAC,KAAc,EAAE,KAAa;IAC9C,IAAI,OAAO,KAAK,KAAK,QAAQ,IAAI,KAAK,KAAK,IAAI,IAAI,CAAC,CAAC,KAAK,IAAI,KAAK,CAAC,EAAE,CAAC;QACrE,MAAM,IAAI,KAAK,CAAC,wCAAwC,KAAK,GAAG,CAAC,CAAC;IACpE,CAAC;IACD,OAAO,OAAO,CAAC,GAAG,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC;AACnC,CAAC;AAED,SAAS,aAAa,CAAC,KAAc;IACnC,MAAM,QAAQ,GAAG,SAAS,CAAC,KAAK,EAAE,UAAU,CAAC,CAAC;IAC9C,MAAM,SAAS,GAAG,SAAS,CAAC,KAAK,EAAE,WAAW,CAAC,CAAC;IAChD,MAAM,SAAS,GAAG,SAAS,CAAC,KAAK,EAAE,WAAW,CAAC,CAAC;IAEhD,IACE,OAAO,QAAQ,KAAK,QAAQ;QAC5B,OAAO,SAAS,KAAK,QAAQ;QAC7B,OAAO,SAAS,KAAK,QAAQ,EAC7B,CAAC;QACD,MAAM,IAAI,KAAK,CAAC,6CAA6C,CAAC,CAAC;IACjE,CAAC;IAED,OAAO,EAAE,IAAI,EAAE,QAAQ,EAAE,SAAS,EAAE,SAAS,EAAE,CAAC;AAClD,CAAC;AAED,MAAM,OAAO,8BAA8B;IACZ;IAA7B,YAA6B,QAAyB;QAAzB,aAAQ,GAAR,QAAQ,CAAiB;IAAG,CAAC;IAE1D,iBAAiB,CAAC,WAAmB,EAAE,kBAA0B;QAC/D,IAAI,CAAC;YACH,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAC5B,2BAA2B,WAAW,UAAU,kBAAkB,QAAQ,CAC3E,CAAC;YACF,MAAM,MAAM,GAAY,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC;YAC7C,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC;gBAC3B,OAAO,IAAI,CAAC;YACd,CAAC;YACD,OAAO,MAAM,CAAC,GAAG,CAAC,aAAa,CAAC,CAAC;QACnC,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,IAAI,CAAC;QACd,CAAC;IACH,CAAC;CACF"}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
import type { ChangedFilesFetchGateway } from '../../../../modules/shared-kernel/entities/diffSizeGate/changedFilesFetch.gateway.js';
|
|
2
|
+
import type { ChangedFile } from '../../../../modules/shared-kernel/entities/diffSizeGate/diffSizeGate.js';
|
|
3
|
+
import type { SimpleCommandExecutor } from '../../../../shared/foundation/commandExecutor.js';
|
|
4
|
+
export type CommandExecutor = SimpleCommandExecutor;
|
|
5
|
+
export declare class GitLabChangedFilesFetchGateway implements ChangedFilesFetchGateway {
|
|
6
|
+
private readonly executor;
|
|
7
|
+
constructor(executor: CommandExecutor);
|
|
8
|
+
fetchChangedFiles(projectPath: string, mergeRequestNumber: number): ChangedFile[] | null;
|
|
9
|
+
}
|
|
10
|
+
//# sourceMappingURL=changedFilesFetch.gitlab.gateway.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"changedFilesFetch.gitlab.gateway.d.ts","sourceRoot":"","sources":["../../../../../src/modules/platform-integration/interface-adapters/gateways/changedFilesFetch.gitlab.gateway.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,wBAAwB,EAAE,MAAM,4EAA4E,CAAC;AAC3H,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,+DAA+D,CAAC;AACjG,OAAO,KAAK,EAAE,qBAAqB,EAAE,MAAM,wCAAwC,CAAC;AAEpF,MAAM,MAAM,eAAe,GAAG,qBAAqB,CAAC;AAiCpD,qBAAa,8BAA+B,YAAW,wBAAwB;IACjE,OAAO,CAAC,QAAQ,CAAC,QAAQ;gBAAR,QAAQ,EAAE,eAAe;IAEtD,iBAAiB,CAAC,WAAW,EAAE,MAAM,EAAE,kBAAkB,EAAE,MAAM,GAAG,WAAW,EAAE,GAAG,IAAI;CASzF"}
|
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
function readField(value, field) {
|
|
2
|
+
if (typeof value !== 'object' || value === null || !(field in value)) {
|
|
3
|
+
throw new Error(`GitLab GraphQL response is missing "${field}"`);
|
|
4
|
+
}
|
|
5
|
+
return Reflect.get(value, field);
|
|
6
|
+
}
|
|
7
|
+
function toChangedFile(entry) {
|
|
8
|
+
const path = readField(entry, 'path');
|
|
9
|
+
const additions = readField(entry, 'additions');
|
|
10
|
+
const deletions = readField(entry, 'deletions');
|
|
11
|
+
if (typeof path !== 'string' || typeof additions !== 'number' || typeof deletions !== 'number') {
|
|
12
|
+
throw new Error('GitLab GraphQL diffStats entry has invalid fields');
|
|
13
|
+
}
|
|
14
|
+
return { path, additions, deletions };
|
|
15
|
+
}
|
|
16
|
+
function extractChangedFiles(response) {
|
|
17
|
+
const project = readField(readField(response, 'data'), 'project');
|
|
18
|
+
const mergeRequest = readField(project, 'mergeRequest');
|
|
19
|
+
const diffStats = readField(mergeRequest, 'diffStats');
|
|
20
|
+
if (!Array.isArray(diffStats)) {
|
|
21
|
+
throw new Error('GitLab GraphQL diffStats is not an array');
|
|
22
|
+
}
|
|
23
|
+
return diffStats.map(toChangedFile);
|
|
24
|
+
}
|
|
25
|
+
export class GitLabChangedFilesFetchGateway {
|
|
26
|
+
executor;
|
|
27
|
+
constructor(executor) {
|
|
28
|
+
this.executor = executor;
|
|
29
|
+
}
|
|
30
|
+
fetchChangedFiles(projectPath, mergeRequestNumber) {
|
|
31
|
+
try {
|
|
32
|
+
const query = `query { project(fullPath:"${projectPath}") { mergeRequest(iid:"${mergeRequestNumber}") { diffStats { path additions deletions } } } }`;
|
|
33
|
+
const response = this.executor(`glab api graphql -f query='${query}'`);
|
|
34
|
+
return extractChangedFiles(JSON.parse(response));
|
|
35
|
+
}
|
|
36
|
+
catch {
|
|
37
|
+
return null;
|
|
38
|
+
}
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
//# sourceMappingURL=changedFilesFetch.gitlab.gateway.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"changedFilesFetch.gitlab.gateway.js","sourceRoot":"","sources":["../../../../../src/modules/platform-integration/interface-adapters/gateways/changedFilesFetch.gitlab.gateway.ts"],"names":[],"mappings":"AAMA,SAAS,SAAS,CAAC,KAAc,EAAE,KAAa;IAC9C,IAAI,OAAO,KAAK,KAAK,QAAQ,IAAI,KAAK,KAAK,IAAI,IAAI,CAAC,CAAC,KAAK,IAAI,KAAK,CAAC,EAAE,CAAC;QACrE,MAAM,IAAI,KAAK,CAAC,uCAAuC,KAAK,GAAG,CAAC,CAAC;IACnE,CAAC;IACD,OAAO,OAAO,CAAC,GAAG,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC;AACnC,CAAC;AAED,SAAS,aAAa,CAAC,KAAc;IACnC,MAAM,IAAI,GAAG,SAAS,CAAC,KAAK,EAAE,MAAM,CAAC,CAAC;IACtC,MAAM,SAAS,GAAG,SAAS,CAAC,KAAK,EAAE,WAAW,CAAC,CAAC;IAChD,MAAM,SAAS,GAAG,SAAS,CAAC,KAAK,EAAE,WAAW,CAAC,CAAC;IAEhD,IAAI,OAAO,IAAI,KAAK,QAAQ,IAAI,OAAO,SAAS,KAAK,QAAQ,IAAI,OAAO,SAAS,KAAK,QAAQ,EAAE,CAAC;QAC/F,MAAM,IAAI,KAAK,CAAC,mDAAmD,CAAC,CAAC;IACvE,CAAC;IAED,OAAO,EAAE,IAAI,EAAE,SAAS,EAAE,SAAS,EAAE,CAAC;AACxC,CAAC;AAED,SAAS,mBAAmB,CAAC,QAAiB;IAC5C,MAAM,OAAO,GAAG,SAAS,CAAC,SAAS,CAAC,QAAQ,EAAE,MAAM,CAAC,EAAE,SAAS,CAAC,CAAC;IAClE,MAAM,YAAY,GAAG,SAAS,CAAC,OAAO,EAAE,cAAc,CAAC,CAAC;IACxD,MAAM,SAAS,GAAG,SAAS,CAAC,YAAY,EAAE,WAAW,CAAC,CAAC;IAEvD,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,SAAS,CAAC,EAAE,CAAC;QAC9B,MAAM,IAAI,KAAK,CAAC,0CAA0C,CAAC,CAAC;IAC9D,CAAC;IAED,OAAO,SAAS,CAAC,GAAG,CAAC,aAAa,CAAC,CAAC;AACtC,CAAC;AAED,MAAM,OAAO,8BAA8B;IACZ;IAA7B,YAA6B,QAAyB;QAAzB,aAAQ,GAAR,QAAQ,CAAiB;IAAG,CAAC;IAE1D,iBAAiB,CAAC,WAAmB,EAAE,kBAA0B;QAC/D,IAAI,CAAC;YACH,MAAM,KAAK,GAAG,6BAA6B,WAAW,0BAA0B,kBAAkB,mDAAmD,CAAC;YACtJ,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,8BAA8B,KAAK,GAAG,CAAC,CAAC;YACvE,OAAO,mBAAmB,CAAC,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC,CAAC;QACnD,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,IAAI,CAAC;QACd,CAAC;IACH,CAAC;CACF"}
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
import type { ChangedFilesFetchGateway } from '../../../modules/shared-kernel/entities/diffSizeGate/changedFilesFetch.gateway.js';
|
|
2
|
+
import type { UseCase } from '../../../shared/foundation/usecase.base.js';
|
|
3
|
+
interface GuardDiffSizeInput {
|
|
4
|
+
projectIdentifier: string;
|
|
5
|
+
mergeRequestNumber: number;
|
|
6
|
+
budget: number;
|
|
7
|
+
}
|
|
8
|
+
export type GuardDiffSizeVerdict = {
|
|
9
|
+
kind: 'allowed';
|
|
10
|
+
} | {
|
|
11
|
+
kind: 'blocked';
|
|
12
|
+
countedLines: number;
|
|
13
|
+
budget: number;
|
|
14
|
+
message: string;
|
|
15
|
+
};
|
|
16
|
+
interface GuardDiffSizeDependencies {
|
|
17
|
+
changedFilesFetchGateway: ChangedFilesFetchGateway;
|
|
18
|
+
}
|
|
19
|
+
export declare class GuardDiffSizeUseCase implements UseCase<GuardDiffSizeInput, GuardDiffSizeVerdict> {
|
|
20
|
+
private readonly dependencies;
|
|
21
|
+
constructor(dependencies: GuardDiffSizeDependencies);
|
|
22
|
+
execute(input: GuardDiffSizeInput): GuardDiffSizeVerdict;
|
|
23
|
+
private fetchChangedFiles;
|
|
24
|
+
}
|
|
25
|
+
export {};
|
|
26
|
+
//# sourceMappingURL=guardDiffSize.usecase.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"guardDiffSize.usecase.d.ts","sourceRoot":"","sources":["../../../../src/modules/platform-integration/usecases/guardDiffSize.usecase.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,wBAAwB,EAAE,MAAM,4EAA4E,CAAC;AAE3H,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,qCAAqC,CAAC;AAEnE,UAAU,kBAAkB;IAC1B,iBAAiB,EAAE,MAAM,CAAC;IAC1B,kBAAkB,EAAE,MAAM,CAAC;IAC3B,MAAM,EAAE,MAAM,CAAC;CAChB;AAED,MAAM,MAAM,oBAAoB,GAC5B;IAAE,IAAI,EAAE,SAAS,CAAA;CAAE,GACnB;IAAE,IAAI,EAAE,SAAS,CAAC;IAAC,YAAY,EAAE,MAAM,CAAC;IAAC,MAAM,EAAE,MAAM,CAAC;IAAC,OAAO,EAAE,MAAM,CAAA;CAAE,CAAC;AAE/E,UAAU,yBAAyB;IACjC,wBAAwB,EAAE,wBAAwB,CAAC;CACpD;AAYD,qBAAa,oBAAqB,YAAW,OAAO,CAAC,kBAAkB,EAAE,oBAAoB,CAAC;IAChF,OAAO,CAAC,QAAQ,CAAC,YAAY;gBAAZ,YAAY,EAAE,yBAAyB;IAEpE,OAAO,CAAC,KAAK,EAAE,kBAAkB,GAAG,oBAAoB;IAuBxD,OAAO,CAAC,iBAAiB;CAU1B"}
|
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
import { evaluateDiffSizeGate } from '../../../modules/shared-kernel/entities/diffSizeGate/diffSizeGate.js';
|
|
2
|
+
function buildSplitMessage(countedLines, budget) {
|
|
3
|
+
return (`Revue refusée : cette merge request fait ${countedLines} lignes comptées ` +
|
|
4
|
+
`(budget ${budget}). Pour faciliter la revue, découpez-la : ` +
|
|
5
|
+
'1) séparez les refactorings des nouvelles fonctionnalités, ' +
|
|
6
|
+
'2) extrayez les changements indépendants dans des MR dédiées, ' +
|
|
7
|
+
'3) limitez chaque MR à une seule intention.');
|
|
8
|
+
}
|
|
9
|
+
export class GuardDiffSizeUseCase {
|
|
10
|
+
dependencies;
|
|
11
|
+
constructor(dependencies) {
|
|
12
|
+
this.dependencies = dependencies;
|
|
13
|
+
}
|
|
14
|
+
execute(input) {
|
|
15
|
+
const files = this.fetchChangedFiles(input);
|
|
16
|
+
if (files === null) {
|
|
17
|
+
return { kind: 'allowed' };
|
|
18
|
+
}
|
|
19
|
+
const { oversized, countedLines, budget } = evaluateDiffSizeGate({
|
|
20
|
+
files,
|
|
21
|
+
budget: input.budget,
|
|
22
|
+
});
|
|
23
|
+
if (!oversized) {
|
|
24
|
+
return { kind: 'allowed' };
|
|
25
|
+
}
|
|
26
|
+
return {
|
|
27
|
+
kind: 'blocked',
|
|
28
|
+
countedLines,
|
|
29
|
+
budget,
|
|
30
|
+
message: buildSplitMessage(countedLines, budget),
|
|
31
|
+
};
|
|
32
|
+
}
|
|
33
|
+
fetchChangedFiles(input) {
|
|
34
|
+
try {
|
|
35
|
+
return this.dependencies.changedFilesFetchGateway.fetchChangedFiles(input.projectIdentifier, input.mergeRequestNumber);
|
|
36
|
+
}
|
|
37
|
+
catch {
|
|
38
|
+
return null;
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
//# sourceMappingURL=guardDiffSize.usecase.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"guardDiffSize.usecase.js","sourceRoot":"","sources":["../../../../src/modules/platform-integration/usecases/guardDiffSize.usecase.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,oBAAoB,EAAE,MAAM,+DAA+D,CAAC;AAiBrG,SAAS,iBAAiB,CAAC,YAAoB,EAAE,MAAc;IAC7D,OAAO,CACL,4CAA4C,YAAY,mBAAmB;QAC3E,WAAW,MAAM,4CAA4C;QAC7D,6DAA6D;QAC7D,gEAAgE;QAChE,6CAA6C,CAC9C,CAAC;AACJ,CAAC;AAED,MAAM,OAAO,oBAAoB;IACF;IAA7B,YAA6B,YAAuC;QAAvC,iBAAY,GAAZ,YAAY,CAA2B;IAAG,CAAC;IAExE,OAAO,CAAC,KAAyB;QAC/B,MAAM,KAAK,GAAG,IAAI,CAAC,iBAAiB,CAAC,KAAK,CAAC,CAAC;QAC5C,IAAI,KAAK,KAAK,IAAI,EAAE,CAAC;YACnB,OAAO,EAAE,IAAI,EAAE,SAAS,EAAE,CAAC;QAC7B,CAAC;QAED,MAAM,EAAE,SAAS,EAAE,YAAY,EAAE,MAAM,EAAE,GAAG,oBAAoB,CAAC;YAC/D,KAAK;YACL,MAAM,EAAE,KAAK,CAAC,MAAM;SACrB,CAAC,CAAC;QAEH,IAAI,CAAC,SAAS,EAAE,CAAC;YACf,OAAO,EAAE,IAAI,EAAE,SAAS,EAAE,CAAC;QAC7B,CAAC;QAED,OAAO;YACL,IAAI,EAAE,SAAS;YACf,YAAY;YACZ,MAAM;YACN,OAAO,EAAE,iBAAiB,CAAC,YAAY,EAAE,MAAM,CAAC;SACjD,CAAC;IACJ,CAAC;IAEO,iBAAiB,CAAC,KAAyB;QACjD,IAAI,CAAC;YACH,OAAO,IAAI,CAAC,YAAY,CAAC,wBAAwB,CAAC,iBAAiB,CACjE,KAAK,CAAC,iBAAiB,EACvB,KAAK,CAAC,kBAAkB,CACzB,CAAC;QACJ,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,IAAI,CAAC;QACd,CAAC;IACH,CAAC;CACF"}
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
import type { Logger } from 'pino';
|
|
2
|
+
import type { IsTrustedActorUseCase } from '../../../modules/platform-integration/usecases/isTrustedActor.usecase.js';
|
|
3
|
+
import type { ReviewJob } from '../../../modules/review-execution/entities/job/reviewJob.js';
|
|
4
|
+
import type { TriggerSource } from '../../../modules/review-execution/entities/pendingReviewRequest/pendingReviewRequest.schema.js';
|
|
5
|
+
import type { EnqueueReviewFunction, GateClaudeInvocationProcessor, GateClaudeInvocationUseCase } from '../../../modules/review-execution/usecases/gateClaudeInvocation.usecase.js';
|
|
6
|
+
import type { BudgetStatus } from '../../../modules/token-accounting/entities/budget/budgetStatus.js';
|
|
7
|
+
import type { EnforceBudgetUseCase } from '../../../modules/token-accounting/usecases/enforceBudget/enforceBudget.usecase.js';
|
|
8
|
+
export type ReviewRequestVerdict = {
|
|
9
|
+
type: 'budget-exceeded';
|
|
10
|
+
status: BudgetStatus;
|
|
11
|
+
} | {
|
|
12
|
+
type: 'pending';
|
|
13
|
+
pendingId: string | null;
|
|
14
|
+
reason?: 'untrusted-actor';
|
|
15
|
+
} | {
|
|
16
|
+
type: 'queued';
|
|
17
|
+
jobId: string;
|
|
18
|
+
} | {
|
|
19
|
+
type: 'deduplicated';
|
|
20
|
+
jobId: string;
|
|
21
|
+
};
|
|
22
|
+
export interface ProcessReviewRequestInput {
|
|
23
|
+
job: ReviewJob;
|
|
24
|
+
processor: GateClaudeInvocationProcessor;
|
|
25
|
+
triggerSource: TriggerSource;
|
|
26
|
+
localPaths: string[];
|
|
27
|
+
actorUsername: string;
|
|
28
|
+
projectPath: string;
|
|
29
|
+
gateActorTrust: boolean;
|
|
30
|
+
}
|
|
31
|
+
export interface ProcessReviewRequestDependencies {
|
|
32
|
+
enforceBudget: Pick<EnforceBudgetUseCase, 'execute'>;
|
|
33
|
+
gateClaudeInvocation?: Pick<GateClaudeInvocationUseCase, 'execute'>;
|
|
34
|
+
isTrustedActor?: Pick<IsTrustedActorUseCase, 'execute'>;
|
|
35
|
+
enqueue: EnqueueReviewFunction;
|
|
36
|
+
logger: Logger;
|
|
37
|
+
}
|
|
38
|
+
export declare function processReviewRequest(input: ProcessReviewRequestInput, deps: ProcessReviewRequestDependencies): Promise<ReviewRequestVerdict>;
|
|
39
|
+
//# sourceMappingURL=processReviewRequest.usecase.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"processReviewRequest.usecase.d.ts","sourceRoot":"","sources":["../../../../src/modules/platform-integration/usecases/processReviewRequest.usecase.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,MAAM,CAAC;AAEnC,OAAO,KAAK,EAAE,qBAAqB,EAAE,MAAM,mEAAmE,CAAC;AAC/G,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,sDAAsD,CAAC;AACtF,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,yFAAyF,CAAC;AAC7H,OAAO,KAAK,EACV,qBAAqB,EACrB,6BAA6B,EAC7B,2BAA2B,EAC5B,MAAM,qEAAqE,CAAC;AAC7E,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,4DAA4D,CAAC;AAC/F,OAAO,KAAK,EAAE,oBAAoB,EAAE,MAAM,4EAA4E,CAAC;AAEvH,MAAM,MAAM,oBAAoB,GAC5B;IAAE,IAAI,EAAE,iBAAiB,CAAC;IAAC,MAAM,EAAE,YAAY,CAAA;CAAE,GACjD;IAAE,IAAI,EAAE,SAAS,CAAC;IAAC,SAAS,EAAE,MAAM,GAAG,IAAI,CAAC;IAAC,MAAM,CAAC,EAAE,iBAAiB,CAAA;CAAE,GACzE;IAAE,IAAI,EAAE,QAAQ,CAAC;IAAC,KAAK,EAAE,MAAM,CAAA;CAAE,GACjC;IAAE,IAAI,EAAE,cAAc,CAAC;IAAC,KAAK,EAAE,MAAM,CAAA;CAAE,CAAC;AAE5C,MAAM,WAAW,yBAAyB;IACxC,GAAG,EAAE,SAAS,CAAC;IACf,SAAS,EAAE,6BAA6B,CAAC;IACzC,aAAa,EAAE,aAAa,CAAC;IAC7B,UAAU,EAAE,MAAM,EAAE,CAAC;IACrB,aAAa,EAAE,MAAM,CAAC;IACtB,WAAW,EAAE,MAAM,CAAC;IACpB,cAAc,EAAE,OAAO,CAAC;CACzB;AAED,MAAM,WAAW,gCAAgC;IAC/C,aAAa,EAAE,IAAI,CAAC,oBAAoB,EAAE,SAAS,CAAC,CAAC;IACrD,oBAAoB,CAAC,EAAE,IAAI,CAAC,2BAA2B,EAAE,SAAS,CAAC,CAAC;IACpE,cAAc,CAAC,EAAE,IAAI,CAAC,qBAAqB,EAAE,SAAS,CAAC,CAAC;IACxD,OAAO,EAAE,qBAAqB,CAAC;IAC/B,MAAM,EAAE,MAAM,CAAC;CAChB;AAiDD,wBAAsB,oBAAoB,CACxC,KAAK,EAAE,yBAAyB,EAChC,IAAI,EAAE,gCAAgC,GACrC,OAAO,CAAC,oBAAoB,CAAC,CAY/B"}
|
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
async function resolveActorTrust(input, deps) {
|
|
2
|
+
if (!input.gateActorTrust || !deps.isTrustedActor) {
|
|
3
|
+
return true;
|
|
4
|
+
}
|
|
5
|
+
return deps.isTrustedActor.execute({
|
|
6
|
+
username: input.actorUsername,
|
|
7
|
+
projectPath: input.projectPath,
|
|
8
|
+
});
|
|
9
|
+
}
|
|
10
|
+
async function gateAndClassify(gateClaudeInvocation, input, actorTrusted) {
|
|
11
|
+
const gateResult = await gateClaudeInvocation.execute({
|
|
12
|
+
job: input.job,
|
|
13
|
+
triggerSource: input.triggerSource,
|
|
14
|
+
processor: input.processor,
|
|
15
|
+
actorTrusted,
|
|
16
|
+
});
|
|
17
|
+
if (gateResult.status === 'pending') {
|
|
18
|
+
return { type: 'pending', pendingId: gateResult.pendingId };
|
|
19
|
+
}
|
|
20
|
+
if (gateResult.status === 'enqueued') {
|
|
21
|
+
return { type: 'queued', jobId: gateResult.jobId };
|
|
22
|
+
}
|
|
23
|
+
return { type: 'deduplicated', jobId: input.job.id };
|
|
24
|
+
}
|
|
25
|
+
async function rawEnqueueAndClassify(input, deps, actorTrusted) {
|
|
26
|
+
if (!actorTrusted) {
|
|
27
|
+
return { type: 'pending', pendingId: null, reason: 'untrusted-actor' };
|
|
28
|
+
}
|
|
29
|
+
const enqueued = await deps.enqueue(input.job, input.processor);
|
|
30
|
+
return enqueued
|
|
31
|
+
? { type: 'queued', jobId: input.job.id }
|
|
32
|
+
: { type: 'deduplicated', jobId: input.job.id };
|
|
33
|
+
}
|
|
34
|
+
export async function processReviewRequest(input, deps) {
|
|
35
|
+
const budgetDecision = await deps.enforceBudget.execute({ localPaths: input.localPaths });
|
|
36
|
+
if (!budgetDecision.accepted) {
|
|
37
|
+
return { type: 'budget-exceeded', status: budgetDecision.status };
|
|
38
|
+
}
|
|
39
|
+
const actorTrusted = await resolveActorTrust(input, deps);
|
|
40
|
+
if (deps.gateClaudeInvocation) {
|
|
41
|
+
return gateAndClassify(deps.gateClaudeInvocation, input, actorTrusted);
|
|
42
|
+
}
|
|
43
|
+
return rawEnqueueAndClassify(input, deps, actorTrusted);
|
|
44
|
+
}
|
|
45
|
+
//# sourceMappingURL=processReviewRequest.usecase.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"processReviewRequest.usecase.js","sourceRoot":"","sources":["../../../../src/modules/platform-integration/usecases/processReviewRequest.usecase.ts"],"names":[],"mappings":"AAqCA,KAAK,UAAU,iBAAiB,CAC9B,KAAgC,EAChC,IAAsC;IAEtC,IAAI,CAAC,KAAK,CAAC,cAAc,IAAI,CAAC,IAAI,CAAC,cAAc,EAAE,CAAC;QAClD,OAAO,IAAI,CAAC;IACd,CAAC;IACD,OAAO,IAAI,CAAC,cAAc,CAAC,OAAO,CAAC;QACjC,QAAQ,EAAE,KAAK,CAAC,aAAa;QAC7B,WAAW,EAAE,KAAK,CAAC,WAAW;KAC/B,CAAC,CAAC;AACL,CAAC;AAED,KAAK,UAAU,eAAe,CAC5B,oBAAkE,EAClE,KAAgC,EAChC,YAAqB;IAErB,MAAM,UAAU,GAAG,MAAM,oBAAoB,CAAC,OAAO,CAAC;QACpD,GAAG,EAAE,KAAK,CAAC,GAAG;QACd,aAAa,EAAE,KAAK,CAAC,aAAa;QAClC,SAAS,EAAE,KAAK,CAAC,SAAS;QAC1B,YAAY;KACb,CAAC,CAAC;IACH,IAAI,UAAU,CAAC,MAAM,KAAK,SAAS,EAAE,CAAC;QACpC,OAAO,EAAE,IAAI,EAAE,SAAS,EAAE,SAAS,EAAE,UAAU,CAAC,SAAS,EAAE,CAAC;IAC9D,CAAC;IACD,IAAI,UAAU,CAAC,MAAM,KAAK,UAAU,EAAE,CAAC;QACrC,OAAO,EAAE,IAAI,EAAE,QAAQ,EAAE,KAAK,EAAE,UAAU,CAAC,KAAK,EAAE,CAAC;IACrD,CAAC;IACD,OAAO,EAAE,IAAI,EAAE,cAAc,EAAE,KAAK,EAAE,KAAK,CAAC,GAAG,CAAC,EAAE,EAAE,CAAC;AACvD,CAAC;AAED,KAAK,UAAU,qBAAqB,CAClC,KAAgC,EAChC,IAAsC,EACtC,YAAqB;IAErB,IAAI,CAAC,YAAY,EAAE,CAAC;QAClB,OAAO,EAAE,IAAI,EAAE,SAAS,EAAE,SAAS,EAAE,IAAI,EAAE,MAAM,EAAE,iBAAiB,EAAE,CAAC;IACzE,CAAC;IACD,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,GAAG,EAAE,KAAK,CAAC,SAAS,CAAC,CAAC;IAChE,OAAO,QAAQ;QACb,CAAC,CAAC,EAAE,IAAI,EAAE,QAAQ,EAAE,KAAK,EAAE,KAAK,CAAC,GAAG,CAAC,EAAE,EAAE;QACzC,CAAC,CAAC,EAAE,IAAI,EAAE,cAAc,EAAE,KAAK,EAAE,KAAK,CAAC,GAAG,CAAC,EAAE,EAAE,CAAC;AACpD,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,oBAAoB,CACxC,KAAgC,EAChC,IAAsC;IAEtC,MAAM,cAAc,GAAG,MAAM,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC,EAAE,UAAU,EAAE,KAAK,CAAC,UAAU,EAAE,CAAC,CAAC;IAC1F,IAAI,CAAC,cAAc,CAAC,QAAQ,EAAE,CAAC;QAC7B,OAAO,EAAE,IAAI,EAAE,iBAAiB,EAAE,MAAM,EAAE,cAAc,CAAC,MAAM,EAAE,CAAC;IACpE,CAAC;IAED,MAAM,YAAY,GAAG,MAAM,iBAAiB,CAAC,KAAK,EAAE,IAAI,CAAC,CAAC;IAE1D,IAAI,IAAI,CAAC,oBAAoB,EAAE,CAAC;QAC9B,OAAO,eAAe,CAAC,IAAI,CAAC,oBAAoB,EAAE,KAAK,EAAE,YAAY,CAAC,CAAC;IACzE,CAAC;IACD,OAAO,qBAAqB,CAAC,KAAK,EAAE,IAAI,EAAE,YAAY,CAAC,CAAC;AAC1D,CAAC"}
|
|
@@ -1,7 +1,9 @@
|
|
|
1
1
|
import type { Logger } from 'pino';
|
|
2
2
|
import type { WebhookEvent } from '../../../modules/platform-integration/entities/webhookEvent/webhookEvent.js';
|
|
3
3
|
import type { HandleClose } from '../../../modules/review-execution/usecases/handleClose.usecase.js';
|
|
4
|
+
import type { QualityGateRejectionReason } from '../../../modules/tracking/entities/qualityGate/qualityGate.js';
|
|
4
5
|
import type { CheckFollowupNeededUseCase } from '../../../modules/tracking/usecases/tracking/checkFollowupNeeded.usecase.js';
|
|
6
|
+
import type { HandlePlatformApprovalUseCase } from '../../../modules/tracking/usecases/tracking/handlePlatformApproval.usecase.js';
|
|
5
7
|
import type { RecordPushUseCase } from '../../../modules/tracking/usecases/tracking/recordPush.usecase.js';
|
|
6
8
|
import type { TransitionStateUseCase } from '../../../modules/tracking/usecases/tracking/transitionState.usecase.js';
|
|
7
9
|
import type { RemoveWorktreeAction } from '../../../modules/worktree-management/entities/worktree/worktree.schema.js';
|
|
@@ -20,6 +22,18 @@ export type ProcessWebhookResult = {
|
|
|
20
22
|
type: 'followup-skipped';
|
|
21
23
|
mergeRequestNumber: number;
|
|
22
24
|
reason: string;
|
|
25
|
+
} | {
|
|
26
|
+
type: 'approved';
|
|
27
|
+
mergeRequestNumber: number;
|
|
28
|
+
} | {
|
|
29
|
+
type: 'approval-revoked';
|
|
30
|
+
mergeRequestNumber: number;
|
|
31
|
+
reason: QualityGateRejectionReason;
|
|
32
|
+
revokeMessage: string;
|
|
33
|
+
} | {
|
|
34
|
+
type: 'approval-ignored';
|
|
35
|
+
mergeRequestNumber: number;
|
|
36
|
+
reason: string;
|
|
23
37
|
} | {
|
|
24
38
|
type: 'ignored';
|
|
25
39
|
reason: string;
|
|
@@ -30,6 +44,8 @@ export interface ProcessWebhookDependencies {
|
|
|
30
44
|
recordPush: Pick<RecordPushUseCase, 'execute'>;
|
|
31
45
|
checkFollowupNeeded: Pick<CheckFollowupNeededUseCase, 'execute'>;
|
|
32
46
|
removeWorktree: RemoveWorktreeAction;
|
|
47
|
+
handlePlatformApproval: Pick<HandlePlatformApprovalUseCase, 'execute'>;
|
|
48
|
+
getQualityThreshold: (projectPath: string) => number | null;
|
|
33
49
|
logger: Logger;
|
|
34
50
|
}
|
|
35
51
|
export type ProcessWebhook = (event: WebhookEvent) => Promise<ProcessWebhookResult>;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"processWebhook.usecase.d.ts","sourceRoot":"","sources":["../../../../src/modules/platform-integration/usecases/processWebhook.usecase.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,MAAM,CAAC;AAEnC,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,sEAAsE,CAAC;AACzG,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,4DAA4D,CAAC;
|
|
1
|
+
{"version":3,"file":"processWebhook.usecase.d.ts","sourceRoot":"","sources":["../../../../src/modules/platform-integration/usecases/processWebhook.usecase.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,MAAM,CAAC;AAEnC,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,sEAAsE,CAAC;AACzG,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,4DAA4D,CAAC;AAE9F,OAAO,KAAK,EAAE,0BAA0B,EAAE,MAAM,wDAAwD,CAAC;AACzG,OAAO,KAAK,EAAE,0BAA0B,EAAE,MAAM,qEAAqE,CAAC;AACtH,OAAO,KAAK,EAAE,6BAA6B,EAAE,MAAM,wEAAwE,CAAC;AAC5H,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,4DAA4D,CAAC;AACpG,OAAO,KAAK,EAAE,sBAAsB,EAAE,MAAM,iEAAiE,CAAC;AAC9G,OAAO,KAAK,EAAE,oBAAoB,EAAE,MAAM,oEAAoE,CAAC;AAE/G,MAAM,MAAM,oBAAoB,GAC5B;IAAE,IAAI,EAAE,QAAQ,CAAC;IAAC,kBAAkB,EAAE,MAAM,CAAC;IAAC,YAAY,EAAE,OAAO,CAAC;IAAC,gBAAgB,EAAE,OAAO,CAAA;CAAE,GAChG;IAAE,IAAI,EAAE,QAAQ,CAAC;IAAC,kBAAkB,EAAE,MAAM,CAAA;CAAE,GAC9C;IAAE,IAAI,EAAE,mBAAmB,CAAC;IAAC,kBAAkB,EAAE,MAAM,CAAA;CAAE,GACzD;IAAE,IAAI,EAAE,kBAAkB,CAAC;IAAC,kBAAkB,EAAE,MAAM,CAAC;IAAC,MAAM,EAAE,MAAM,CAAA;CAAE,GACxE;IAAE,IAAI,EAAE,UAAU,CAAC;IAAC,kBAAkB,EAAE,MAAM,CAAA;CAAE,GAChD;IACE,IAAI,EAAE,kBAAkB,CAAC;IACzB,kBAAkB,EAAE,MAAM,CAAC;IAC3B,MAAM,EAAE,0BAA0B,CAAC;IACnC,aAAa,EAAE,MAAM,CAAC;CACvB,GACD;IAAE,IAAI,EAAE,kBAAkB,CAAC;IAAC,kBAAkB,EAAE,MAAM,CAAC;IAAC,MAAM,EAAE,MAAM,CAAA;CAAE,GACxE;IAAE,IAAI,EAAE,SAAS,CAAC;IAAC,MAAM,EAAE,MAAM,CAAA;CAAE,CAAC;AAExC,MAAM,WAAW,0BAA0B;IACzC,WAAW,EAAE,WAAW,CAAC;IACzB,eAAe,EAAE,IAAI,CAAC,sBAAsB,EAAE,SAAS,CAAC,CAAC;IACzD,UAAU,EAAE,IAAI,CAAC,iBAAiB,EAAE,SAAS,CAAC,CAAC;IAC/C,mBAAmB,EAAE,IAAI,CAAC,0BAA0B,EAAE,SAAS,CAAC,CAAC;IACjE,cAAc,EAAE,oBAAoB,CAAC;IACrC,sBAAsB,EAAE,IAAI,CAAC,6BAA6B,EAAE,SAAS,CAAC,CAAC;IACvE,mBAAmB,EAAE,CAAC,WAAW,EAAE,MAAM,KAAK,MAAM,GAAG,IAAI,CAAC;IAC5D,MAAM,EAAE,MAAM,CAAC;CAChB;AAED,MAAM,MAAM,cAAc,GAAG,CAAC,KAAK,EAAE,YAAY,KAAK,OAAO,CAAC,oBAAoB,CAAC,CAAC;AAqKpF,wBAAsB,cAAc,CAClC,KAAK,EAAE,YAAY,EACnB,IAAI,EAAE,0BAA0B,GAC/B,OAAO,CAAC,oBAAoB,CAAC,CAe/B"}
|
|
@@ -1,3 +1,4 @@
|
|
|
1
|
+
import { evaluateQualityGate } from '../../../modules/tracking/entities/qualityGate/qualityGate.js';
|
|
1
2
|
function buildMergeRequestId(event) {
|
|
2
3
|
return `${event.platform}-${event.projectPath}-${event.mergeRequestNumber}`;
|
|
3
4
|
}
|
|
@@ -75,6 +76,48 @@ function routeFollowup(event, deps) {
|
|
|
75
76
|
}
|
|
76
77
|
return { type: 'followup-eligible', mergeRequestNumber: event.mergeRequestNumber };
|
|
77
78
|
}
|
|
79
|
+
function routeApprove(event, deps) {
|
|
80
|
+
const mrId = buildMergeRequestId(event);
|
|
81
|
+
const threshold = deps.getQualityThreshold(event.localPath);
|
|
82
|
+
const transitionResult = deps.transitionState.execute({
|
|
83
|
+
projectPath: event.localPath,
|
|
84
|
+
mrId,
|
|
85
|
+
targetState: 'approved',
|
|
86
|
+
qualityCheck: (mr) => evaluateQualityGate({
|
|
87
|
+
latestScore: mr.latestScore,
|
|
88
|
+
blockingIssues: mr.openThreads,
|
|
89
|
+
threshold,
|
|
90
|
+
}),
|
|
91
|
+
});
|
|
92
|
+
if (transitionResult.ok) {
|
|
93
|
+
return { type: 'approved', mergeRequestNumber: event.mergeRequestNumber };
|
|
94
|
+
}
|
|
95
|
+
if (transitionResult.reason === 'quality-gate') {
|
|
96
|
+
const verdict = deps.handlePlatformApproval.execute({
|
|
97
|
+
projectPath: event.localPath,
|
|
98
|
+
mrId,
|
|
99
|
+
qualityThreshold: threshold,
|
|
100
|
+
});
|
|
101
|
+
if (verdict.kind === 'reverted') {
|
|
102
|
+
return {
|
|
103
|
+
type: 'approval-revoked',
|
|
104
|
+
mergeRequestNumber: event.mergeRequestNumber,
|
|
105
|
+
reason: verdict.reason,
|
|
106
|
+
revokeMessage: verdict.message,
|
|
107
|
+
};
|
|
108
|
+
}
|
|
109
|
+
return {
|
|
110
|
+
type: 'approval-ignored',
|
|
111
|
+
mergeRequestNumber: event.mergeRequestNumber,
|
|
112
|
+
reason: verdict.kind,
|
|
113
|
+
};
|
|
114
|
+
}
|
|
115
|
+
return {
|
|
116
|
+
type: 'approval-ignored',
|
|
117
|
+
mergeRequestNumber: event.mergeRequestNumber,
|
|
118
|
+
reason: transitionResult.reason,
|
|
119
|
+
};
|
|
120
|
+
}
|
|
78
121
|
export async function processWebhook(event, deps) {
|
|
79
122
|
switch (event.type) {
|
|
80
123
|
case 'close':
|
|
@@ -86,7 +129,7 @@ export async function processWebhook(event, deps) {
|
|
|
86
129
|
case 'review-requested':
|
|
87
130
|
return { type: 'ignored', reason: 'review-requested-handled-by-controller' };
|
|
88
131
|
case 'approve':
|
|
89
|
-
return
|
|
132
|
+
return routeApprove(event, deps);
|
|
90
133
|
case 'ignored':
|
|
91
134
|
return { type: 'ignored', reason: event.reason };
|
|
92
135
|
}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"processWebhook.usecase.js","sourceRoot":"","sources":["../../../../src/modules/platform-integration/usecases/processWebhook.usecase.ts"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"processWebhook.usecase.js","sourceRoot":"","sources":["../../../../src/modules/platform-integration/usecases/processWebhook.usecase.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
import type { ChangedFile } from '../../../../modules/shared-kernel/entities/diffSizeGate/diffSizeGate.js';
|
|
2
|
+
export interface ChangedFilesFetchGateway {
|
|
3
|
+
fetchChangedFiles(projectIdentifier: string, mergeRequestNumber: number): ChangedFile[] | null;
|
|
4
|
+
}
|
|
5
|
+
//# sourceMappingURL=changedFilesFetch.gateway.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"changedFilesFetch.gateway.d.ts","sourceRoot":"","sources":["../../../../../src/modules/shared-kernel/entities/diffSizeGate/changedFilesFetch.gateway.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,+DAA+D,CAAC;AAEjG,MAAM,WAAW,wBAAwB;IACvC,iBAAiB,CAAC,iBAAiB,EAAE,MAAM,EAAE,kBAAkB,EAAE,MAAM,GAAG,WAAW,EAAE,GAAG,IAAI,CAAC;CAChG"}
|