projen-pipelines 0.2.12 → 0.2.14

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 (49) hide show
  1. package/.jsii +5703 -1380
  2. package/API.md +5813 -1492
  3. package/README.md +253 -0
  4. package/docs/drift-detection.md +264 -0
  5. package/lib/assign-approver/base.js +1 -1
  6. package/lib/assign-approver/github.js +1 -1
  7. package/lib/awscdk/base.d.ts +21 -0
  8. package/lib/awscdk/base.js +246 -2
  9. package/lib/awscdk/bash.js +1 -1
  10. package/lib/awscdk/github.js +1 -1
  11. package/lib/awscdk/gitlab.js +1 -1
  12. package/lib/drift/base.d.ts +64 -0
  13. package/lib/drift/base.js +18 -0
  14. package/lib/drift/bash.d.ts +15 -0
  15. package/lib/drift/bash.js +170 -0
  16. package/lib/drift/detect-drift.d.ts +54 -0
  17. package/lib/drift/detect-drift.js +259 -0
  18. package/lib/drift/github.d.ts +21 -0
  19. package/lib/drift/github.js +232 -0
  20. package/lib/drift/gitlab.d.ts +20 -0
  21. package/lib/drift/gitlab.js +138 -0
  22. package/lib/drift/index.d.ts +5 -0
  23. package/lib/drift/index.js +22 -0
  24. package/lib/drift/step.d.ts +14 -0
  25. package/lib/drift/step.js +48 -0
  26. package/lib/index.d.ts +2 -0
  27. package/lib/index.js +3 -1
  28. package/lib/steps/artifact-steps.js +2 -2
  29. package/lib/steps/aws-assume-role.step.js +1 -1
  30. package/lib/steps/registries.js +2 -2
  31. package/lib/steps/step.d.ts +6 -1
  32. package/lib/steps/step.js +14 -10
  33. package/lib/versioning/computation.d.ts +63 -0
  34. package/lib/versioning/computation.js +121 -0
  35. package/lib/versioning/config.d.ts +41 -0
  36. package/lib/versioning/config.js +91 -0
  37. package/lib/versioning/index.d.ts +7 -0
  38. package/lib/versioning/index.js +46 -0
  39. package/lib/versioning/outputs.d.ts +87 -0
  40. package/lib/versioning/outputs.js +166 -0
  41. package/lib/versioning/setup.d.ts +30 -0
  42. package/lib/versioning/setup.js +165 -0
  43. package/lib/versioning/strategy.d.ts +21 -0
  44. package/lib/versioning/strategy.js +51 -0
  45. package/lib/versioning/types.d.ts +183 -0
  46. package/lib/versioning/types.js +3 -0
  47. package/lib/versioning/version-info.d.ts +106 -0
  48. package/lib/versioning/version-info.js +269 -0
  49. package/package.json +2 -1
@@ -0,0 +1,259 @@
1
+ #!/usr/bin/env node
2
+ "use strict";
3
+ Object.defineProperty(exports, "__esModule", { value: true });
4
+ exports.DriftDetector = void 0;
5
+ const child_process_1 = require("child_process");
6
+ const fs_1 = require("fs");
7
+ class DriftDetector {
8
+ constructor(options) {
9
+ this.results = [];
10
+ this.options = {
11
+ timeout: 30,
12
+ failOnDrift: true,
13
+ ...options,
14
+ };
15
+ }
16
+ async run() {
17
+ console.log(`Starting drift detection in region ${this.options.region}`);
18
+ try {
19
+ const stacks = await this.getStacksToCheck();
20
+ for (const stackName of stacks) {
21
+ await this.checkStackDrift(stackName);
22
+ }
23
+ this.printSummary();
24
+ this.saveResults();
25
+ if (this.shouldFail()) {
26
+ process.exit(1);
27
+ }
28
+ }
29
+ catch (error) {
30
+ console.error('Fatal error during drift detection:', error);
31
+ process.exit(2);
32
+ }
33
+ }
34
+ async getStacksToCheck() {
35
+ if (this.options.stackNames && this.options.stackNames.length > 0) {
36
+ return this.options.stackNames;
37
+ }
38
+ console.log('Getting all stacks in the region...');
39
+ const output = (0, child_process_1.execSync)(`aws cloudformation list-stacks --region ${this.options.region} --stack-status-filter CREATE_COMPLETE UPDATE_COMPLETE --query 'StackSummaries[].StackName' --output json`, { encoding: 'utf8' });
40
+ return JSON.parse(output);
41
+ }
42
+ async checkStackDrift(stackName) {
43
+ console.log(`\nChecking drift for stack: ${stackName}`);
44
+ const result = {
45
+ stackName,
46
+ driftStatus: 'NOT_CHECKED',
47
+ };
48
+ try {
49
+ // Start drift detection
50
+ const driftId = await this.startDriftDetection(stackName);
51
+ console.log(`Started drift detection with ID: ${driftId}`);
52
+ // Wait for completion
53
+ const driftStatus = await this.waitForDriftDetection(driftId);
54
+ result.driftStatus = driftStatus;
55
+ if (driftStatus === 'DRIFTED') {
56
+ result.driftedResources = await this.getDriftedResources(stackName);
57
+ console.log(`DRIFT DETECTED in stack ${stackName}!`);
58
+ // Handle known drift errors
59
+ result.knownErrorsHandled = await this.handleKnownDriftErrors(result.driftedResources);
60
+ // Print drift details
61
+ this.printDriftDetails(result);
62
+ }
63
+ else if (driftStatus === 'IN_SYNC') {
64
+ console.log(`Stack ${stackName} is in sync`);
65
+ }
66
+ }
67
+ catch (error) {
68
+ console.error(`Error checking drift for stack ${stackName}:`, error.message);
69
+ result.error = error.message;
70
+ }
71
+ this.results.push(result);
72
+ }
73
+ async startDriftDetection(stackName) {
74
+ const output = (0, child_process_1.execSync)(`aws cloudformation detect-stack-drift --stack-name ${stackName} --region ${this.options.region} --query 'StackDriftDetectionId' --output text`, { encoding: 'utf8' });
75
+ return output.trim();
76
+ }
77
+ async waitForDriftDetection(driftId) {
78
+ const timeout = this.options.timeout * 60; // Convert to seconds
79
+ const startTime = Date.now();
80
+ while (true) {
81
+ const elapsed = Math.floor((Date.now() - startTime) / 1000);
82
+ if (elapsed > timeout) {
83
+ throw new Error(`Drift detection timed out after ${this.options.timeout} minutes`);
84
+ }
85
+ const output = (0, child_process_1.execSync)(`aws cloudformation describe-stack-drift-detection-status --stack-drift-detection-id ${driftId} --region ${this.options.region} --output json`, { encoding: 'utf8' });
86
+ const status = JSON.parse(output);
87
+ console.log(`Drift detection status: ${status.DetectionStatus} (${elapsed}s elapsed)`);
88
+ if (status.DetectionStatus === 'DETECTION_COMPLETE') {
89
+ return status.StackDriftStatus;
90
+ }
91
+ else if (status.DetectionStatus === 'DETECTION_FAILED') {
92
+ throw new Error(`Drift detection failed: ${status.DetectionStatusReason}`);
93
+ }
94
+ // Wait 10 seconds before checking again
95
+ await new Promise(resolve => setTimeout(resolve, 10000));
96
+ }
97
+ }
98
+ async getDriftedResources(stackName) {
99
+ const output = (0, child_process_1.execSync)(`aws cloudformation describe-stack-resource-drifts --stack-name ${stackName} --region ${this.options.region} --stack-resource-drift-status-filters MODIFIED DELETED --output json`, { encoding: 'utf8' });
100
+ const response = JSON.parse(output);
101
+ return response.StackResourceDrifts.map((drift) => ({
102
+ logicalResourceId: drift.LogicalResourceId,
103
+ resourceType: drift.ResourceType,
104
+ stackResourceDriftStatus: drift.StackResourceDriftStatus,
105
+ propertyDifferences: drift.PropertyDifferences?.map((diff) => ({
106
+ propertyPath: diff.PropertyPath,
107
+ expectedValue: diff.ExpectedValue,
108
+ actualValue: diff.ActualValue,
109
+ differenceType: diff.DifferenceType,
110
+ })),
111
+ }));
112
+ }
113
+ /**
114
+ * Handle known drift errors for specific resources
115
+ * This method can be extended later to implement custom logic for known issues
116
+ */
117
+ async handleKnownDriftErrors(driftedResources) {
118
+ const knownErrors = [];
119
+ if (!driftedResources) {
120
+ return knownErrors;
121
+ }
122
+ for (const _resource of driftedResources) {
123
+ // TODO: Implement custom logic here for known drift errors
124
+ // Example structure for handling known errors:
125
+ // Check for Lambda runtime drift (common issue)
126
+ // if (resource.resourceType === 'AWS::Lambda::Function' && resource.propertyDifferences) {
127
+ // const runtimeDrift = resource.propertyDifferences.find(diff =>
128
+ // diff.propertyPath === '/Runtime' || diff.propertyPath === 'Runtime',
129
+ // );
130
+ // if (runtimeDrift) {
131
+ // knownErrors.push({
132
+ // resourceId: resource.logicalResourceId,
133
+ // errorType: 'lambda-runtime-drift',
134
+ // originalError: runtimeDrift,
135
+ // handled: false, // Will be implemented later
136
+ // message: 'Lambda runtime drift detected - manual implementation needed',
137
+ // });
138
+ // }
139
+ // }
140
+ // // Check for auto-scaling related drift
141
+ // if (resource.resourceType.includes('AutoScaling') && resource.propertyDifferences) {
142
+ // knownErrors.push({
143
+ // resourceId: resource.logicalResourceId,
144
+ // errorType: 'autoscaling-drift',
145
+ // originalError: resource.propertyDifferences,
146
+ // handled: false, // Will be implemented later
147
+ // message: 'Auto-scaling drift detected - manual implementation needed',
148
+ // });
149
+ // }
150
+ // Add more patterns here as needed
151
+ }
152
+ return knownErrors;
153
+ }
154
+ printDriftDetails(result) {
155
+ if (!result.driftedResources || result.driftedResources.length === 0) {
156
+ return;
157
+ }
158
+ console.log('\nDrifted resources:');
159
+ console.log('=================');
160
+ for (const resource of result.driftedResources) {
161
+ console.log(`\n- ${resource.logicalResourceId} (${resource.resourceType})`);
162
+ console.log(` Status: ${resource.stackResourceDriftStatus}`);
163
+ if (resource.propertyDifferences) {
164
+ console.log(' Property differences:');
165
+ for (const diff of resource.propertyDifferences) {
166
+ console.log(` ${diff.propertyPath}:`);
167
+ console.log(` Expected: ${diff.expectedValue}`);
168
+ console.log(` Actual: ${diff.actualValue}`);
169
+ console.log(` Type: ${diff.differenceType}`);
170
+ }
171
+ }
172
+ }
173
+ }
174
+ printSummary() {
175
+ console.log('\n========== DRIFT DETECTION SUMMARY ==========');
176
+ const driftedStacks = this.results.filter(r => r.driftStatus === 'DRIFTED');
177
+ const syncedStacks = this.results.filter(r => r.driftStatus === 'IN_SYNC');
178
+ const errorStacks = this.results.filter(r => r.error);
179
+ console.log(`Total stacks checked: ${this.results.length}`);
180
+ console.log(`In sync: ${syncedStacks.length}`);
181
+ console.log(`Drifted: ${driftedStacks.length}`);
182
+ console.log(`Errors: ${errorStacks.length}`);
183
+ if (driftedStacks.length > 0) {
184
+ console.log('\nDrifted stacks:');
185
+ for (const stack of driftedStacks) {
186
+ const resourceCount = stack.driftedResources?.length || 0;
187
+ console.log(` - ${stack.stackName} (${resourceCount} resources)`);
188
+ }
189
+ }
190
+ if (errorStacks.length > 0) {
191
+ console.log('\nStacks with errors:');
192
+ for (const stack of errorStacks) {
193
+ console.log(` - ${stack.stackName}: ${stack.error}`);
194
+ }
195
+ }
196
+ }
197
+ saveResults() {
198
+ const outputFile = process.env.DRIFT_DETECTION_OUTPUT || 'drift-detection-results.json';
199
+ (0, fs_1.writeFileSync)(outputFile, JSON.stringify(this.results, null, 2));
200
+ console.log(`\nResults saved to: ${outputFile}`);
201
+ }
202
+ shouldFail() {
203
+ if (!this.options.failOnDrift) {
204
+ return false;
205
+ }
206
+ return this.results.some(r => r.driftStatus === 'DRIFTED');
207
+ }
208
+ }
209
+ exports.DriftDetector = DriftDetector;
210
+ // Parse command line arguments
211
+ function parseArgs() {
212
+ const args = process.argv.slice(2);
213
+ const options = {
214
+ region: process.env.AWS_REGION || 'us-east-1',
215
+ };
216
+ for (let i = 0; i < args.length; i++) {
217
+ switch (args[i]) {
218
+ case '--region':
219
+ options.region = args[++i];
220
+ break;
221
+ case '--stacks':
222
+ options.stackNames = args[++i].split(',');
223
+ break;
224
+ case '--timeout':
225
+ options.timeout = parseInt(args[++i]);
226
+ break;
227
+ case '--no-fail-on-drift':
228
+ options.failOnDrift = false;
229
+ break;
230
+ default:
231
+ console.error(`Unknown argument: ${args[i]}`);
232
+ printUsage();
233
+ process.exit(1);
234
+ }
235
+ }
236
+ return options;
237
+ }
238
+ function printUsage() {
239
+ console.log(`
240
+ Usage: detect-drift.ts [options]
241
+
242
+ Options:
243
+ --region <region> AWS region (default: us-east-1 or AWS_REGION env var)
244
+ --stacks <stack1,stack2> Comma-separated list of stack names (default: all stacks)
245
+ --timeout <minutes> Timeout in minutes (default: 30)
246
+ --no-fail-on-drift Don't exit with error code if drift is detected
247
+
248
+ Environment variables:
249
+ AWS_REGION Default AWS region
250
+ DRIFT_DETECTION_OUTPUT Output file path (default: drift-detection-results.json)
251
+ `);
252
+ }
253
+ // Main entry point
254
+ if (require.main === module) {
255
+ const options = parseArgs();
256
+ const detector = new DriftDetector(options);
257
+ detector.run().catch(console.error);
258
+ }
259
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"detect-drift.js","sourceRoot":"","sources":["../../src/drift/detect-drift.ts"],"names":[],"mappings":";;;;AAEA,iDAAyC;AACzC,2BAAmC;AAuCnC,MAAM,aAAa;IAIjB,YAAY,OAA8B;QAFzB,YAAO,GAAkB,EAAE,CAAC;QAG3C,IAAI,CAAC,OAAO,GAAG;YACb,OAAO,EAAE,EAAE;YACX,WAAW,EAAE,IAAI;YACjB,GAAG,OAAO;SACX,CAAC;IACJ,CAAC;IAEM,KAAK,CAAC,GAAG;QACd,OAAO,CAAC,GAAG,CAAC,sCAAsC,IAAI,CAAC,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC;QAEzE,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,gBAAgB,EAAE,CAAC;YAE7C,KAAK,MAAM,SAAS,IAAI,MAAM,EAAE,CAAC;gBAC/B,MAAM,IAAI,CAAC,eAAe,CAAC,SAAS,CAAC,CAAC;YACxC,CAAC;YAED,IAAI,CAAC,YAAY,EAAE,CAAC;YACpB,IAAI,CAAC,WAAW,EAAE,CAAC;YAEnB,IAAI,IAAI,CAAC,UAAU,EAAE,EAAE,CAAC;gBACtB,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;YAClB,CAAC;QACH,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO,CAAC,KAAK,CAAC,qCAAqC,EAAE,KAAK,CAAC,CAAC;YAC5D,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QAClB,CAAC;IACH,CAAC;IAEO,KAAK,CAAC,gBAAgB;QAC5B,IAAI,IAAI,CAAC,OAAO,CAAC,UAAU,IAAI,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAClE,OAAO,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC;QACjC,CAAC;QAED,OAAO,CAAC,GAAG,CAAC,qCAAqC,CAAC,CAAC;QACnD,MAAM,MAAM,GAAG,IAAA,wBAAQ,EACrB,2CAA2C,IAAI,CAAC,OAAO,CAAC,MAAM,2GAA2G,EACzK,EAAE,QAAQ,EAAE,MAAM,EAAE,CACrB,CAAC;QAEF,OAAO,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;IAC5B,CAAC;IAEO,KAAK,CAAC,eAAe,CAAC,SAAiB;QAC7C,OAAO,CAAC,GAAG,CAAC,+BAA+B,SAAS,EAAE,CAAC,CAAC;QAExD,MAAM,MAAM,GAAgB;YAC1B,SAAS;YACT,WAAW,EAAE,aAAa;SAC3B,CAAC;QAEF,IAAI,CAAC;YACH,wBAAwB;YACxB,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,mBAAmB,CAAC,SAAS,CAAC,CAAC;YAC1D,OAAO,CAAC,GAAG,CAAC,oCAAoC,OAAO,EAAE,CAAC,CAAC;YAE3D,sBAAsB;YACtB,MAAM,WAAW,GAAG,MAAM,IAAI,CAAC,qBAAqB,CAAC,OAAO,CAAC,CAAC;YAC9D,MAAM,CAAC,WAAW,GAAG,WAAW,CAAC;YAEjC,IAAI,WAAW,KAAK,SAAS,EAAE,CAAC;gBAC9B,MAAM,CAAC,gBAAgB,GAAG,MAAM,IAAI,CAAC,mBAAmB,CAAC,SAAS,CAAC,CAAC;gBACpE,OAAO,CAAC,GAAG,CAAC,2BAA2B,SAAS,GAAG,CAAC,CAAC;gBAErD,4BAA4B;gBAC5B,MAAM,CAAC,kBAAkB,GAAG,MAAM,IAAI,CAAC,sBAAsB,CAAC,MAAM,CAAC,gBAAgB,CAAC,CAAC;gBAEvF,sBAAsB;gBACtB,IAAI,CAAC,iBAAiB,CAAC,MAAM,CAAC,CAAC;YACjC,CAAC;iBAAM,IAAI,WAAW,KAAK,SAAS,EAAE,CAAC;gBACrC,OAAO,CAAC,GAAG,CAAC,SAAS,SAAS,aAAa,CAAC,CAAC;YAC/C,CAAC;QACH,CAAC;QAAC,OAAO,KAAU,EAAE,CAAC;YACpB,OAAO,CAAC,KAAK,CAAC,kCAAkC,SAAS,GAAG,EAAE,KAAK,CAAC,OAAO,CAAC,CAAC;YAC7E,MAAM,CAAC,KAAK,GAAG,KAAK,CAAC,OAAO,CAAC;QAC/B,CAAC;QAED,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;IAC5B,CAAC;IAEO,KAAK,CAAC,mBAAmB,CAAC,SAAiB;QACjD,MAAM,MAAM,GAAG,IAAA,wBAAQ,EACrB,sDAAsD,SAAS,aAAa,IAAI,CAAC,OAAO,CAAC,MAAM,gDAAgD,EAC/I,EAAE,QAAQ,EAAE,MAAM,EAAE,CACrB,CAAC;QAEF,OAAO,MAAM,CAAC,IAAI,EAAE,CAAC;IACvB,CAAC;IAEO,KAAK,CAAC,qBAAqB,CAAC,OAAe;QACjD,MAAM,OAAO,GAAG,IAAI,CAAC,OAAO,CAAC,OAAQ,GAAG,EAAE,CAAC,CAAC,qBAAqB;QACjE,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QAE7B,OAAO,IAAI,EAAE,CAAC;YACZ,MAAM,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,SAAS,CAAC,GAAG,IAAI,CAAC,CAAC;YAE5D,IAAI,OAAO,GAAG,OAAO,EAAE,CAAC;gBACtB,MAAM,IAAI,KAAK,CAAC,mCAAmC,IAAI,CAAC,OAAO,CAAC,OAAO,UAAU,CAAC,CAAC;YACrF,CAAC;YAED,MAAM,MAAM,GAAG,IAAA,wBAAQ,EACrB,uFAAuF,OAAO,aAAa,IAAI,CAAC,OAAO,CAAC,MAAM,gBAAgB,EAC9I,EAAE,QAAQ,EAAE,MAAM,EAAE,CACrB,CAAC;YAEF,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;YAClC,OAAO,CAAC,GAAG,CAAC,2BAA2B,MAAM,CAAC,eAAe,KAAK,OAAO,YAAY,CAAC,CAAC;YAEvF,IAAI,MAAM,CAAC,eAAe,KAAK,oBAAoB,EAAE,CAAC;gBACpD,OAAO,MAAM,CAAC,gBAAqE,CAAC;YACtF,CAAC;iBAAM,IAAI,MAAM,CAAC,eAAe,KAAK,kBAAkB,EAAE,CAAC;gBACzD,MAAM,IAAI,KAAK,CAAC,2BAA2B,MAAM,CAAC,qBAAqB,EAAE,CAAC,CAAC;YAC7E,CAAC;YAED,wCAAwC;YACxC,MAAM,IAAI,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC,UAAU,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC,CAAC;QAC3D,CAAC;IACH,CAAC;IAEO,KAAK,CAAC,mBAAmB,CAAC,SAAiB;QACjD,MAAM,MAAM,GAAG,IAAA,wBAAQ,EACrB,kEAAkE,SAAS,aAAa,IAAI,CAAC,OAAO,CAAC,MAAM,uEAAuE,EAClL,EAAE,QAAQ,EAAE,MAAM,EAAE,CACrB,CAAC;QAEF,MAAM,QAAQ,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;QACpC,OAAO,QAAQ,CAAC,mBAAmB,CAAC,GAAG,CAAC,CAAC,KAAU,EAAE,EAAE,CAAC,CAAC;YACvD,iBAAiB,EAAE,KAAK,CAAC,iBAAiB;YAC1C,YAAY,EAAE,KAAK,CAAC,YAAY;YAChC,wBAAwB,EAAE,KAAK,CAAC,wBAAwB;YACxD,mBAAmB,EAAE,KAAK,CAAC,mBAAmB,EAAE,GAAG,CAAC,CAAC,IAAS,EAAE,EAAE,CAAC,CAAC;gBAClE,YAAY,EAAE,IAAI,CAAC,YAAY;gBAC/B,aAAa,EAAE,IAAI,CAAC,aAAa;gBACjC,WAAW,EAAE,IAAI,CAAC,WAAW;gBAC7B,cAAc,EAAE,IAAI,CAAC,cAAc;aACpC,CAAC,CAAC;SACJ,CAAC,CAAC,CAAC;IACN,CAAC;IAED;;;OAGG;IACK,KAAK,CAAC,sBAAsB,CAAC,gBAAoC;QACvE,MAAM,WAAW,GAAuB,EAAE,CAAC;QAE3C,IAAI,CAAC,gBAAgB,EAAE,CAAC;YACtB,OAAO,WAAW,CAAC;QACrB,CAAC;QAED,KAAK,MAAM,SAAS,IAAI,gBAAgB,EAAE,CAAC;YAEzC,2DAA2D;YAC3D,+CAA+C;YAE/C,gDAAgD;YAChD,2FAA2F;YAC3F,mEAAmE;YACnE,2EAA2E;YAC3E,OAAO;YAEP,wBAAwB;YACxB,yBAAyB;YACzB,gDAAgD;YAChD,2CAA2C;YAC3C,qCAAqC;YACrC,qDAAqD;YACrD,iFAAiF;YACjF,UAAU;YACV,MAAM;YACN,IAAI;YAEJ,0CAA0C;YAC1C,uFAAuF;YACvF,uBAAuB;YACvB,8CAA8C;YAC9C,sCAAsC;YACtC,mDAAmD;YACnD,mDAAmD;YACnD,6EAA6E;YAC7E,QAAQ;YACR,IAAI;YAEJ,mCAAmC;QACrC,CAAC;QAED,OAAO,WAAW,CAAC;IACrB,CAAC;IAEO,iBAAiB,CAAC,MAAmB;QAC3C,IAAI,CAAC,MAAM,CAAC,gBAAgB,IAAI,MAAM,CAAC,gBAAgB,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACrE,OAAO;QACT,CAAC;QAED,OAAO,CAAC,GAAG,CAAC,sBAAsB,CAAC,CAAC;QACpC,OAAO,CAAC,GAAG,CAAC,mBAAmB,CAAC,CAAC;QAEjC,KAAK,MAAM,QAAQ,IAAI,MAAM,CAAC,gBAAgB,EAAE,CAAC;YAC/C,OAAO,CAAC,GAAG,CAAC,OAAO,QAAQ,CAAC,iBAAiB,KAAK,QAAQ,CAAC,YAAY,GAAG,CAAC,CAAC;YAC5E,OAAO,CAAC,GAAG,CAAC,aAAa,QAAQ,CAAC,wBAAwB,EAAE,CAAC,CAAC;YAE9D,IAAI,QAAQ,CAAC,mBAAmB,EAAE,CAAC;gBACjC,OAAO,CAAC,GAAG,CAAC,yBAAyB,CAAC,CAAC;gBACvC,KAAK,MAAM,IAAI,IAAI,QAAQ,CAAC,mBAAmB,EAAE,CAAC;oBAChD,OAAO,CAAC,GAAG,CAAC,OAAO,IAAI,CAAC,YAAY,GAAG,CAAC,CAAC;oBACzC,OAAO,CAAC,GAAG,CAAC,mBAAmB,IAAI,CAAC,aAAa,EAAE,CAAC,CAAC;oBACrD,OAAO,CAAC,GAAG,CAAC,iBAAiB,IAAI,CAAC,WAAW,EAAE,CAAC,CAAC;oBACjD,OAAO,CAAC,GAAG,CAAC,eAAe,IAAI,CAAC,cAAc,EAAE,CAAC,CAAC;gBACpD,CAAC;YACH,CAAC;QACH,CAAC;IACH,CAAC;IAEO,YAAY;QAClB,OAAO,CAAC,GAAG,CAAC,iDAAiD,CAAC,CAAC;QAE/D,MAAM,aAAa,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,WAAW,KAAK,SAAS,CAAC,CAAC;QAC5E,MAAM,YAAY,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,WAAW,KAAK,SAAS,CAAC,CAAC;QAC3E,MAAM,WAAW,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC;QAEtD,OAAO,CAAC,GAAG,CAAC,yBAAyB,IAAI,CAAC,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC;QAC5D,OAAO,CAAC,GAAG,CAAC,YAAY,YAAY,CAAC,MAAM,EAAE,CAAC,CAAC;QAC/C,OAAO,CAAC,GAAG,CAAC,YAAY,aAAa,CAAC,MAAM,EAAE,CAAC,CAAC;QAChD,OAAO,CAAC,GAAG,CAAC,WAAW,WAAW,CAAC,MAAM,EAAE,CAAC,CAAC;QAE7C,IAAI,aAAa,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAC7B,OAAO,CAAC,GAAG,CAAC,mBAAmB,CAAC,CAAC;YACjC,KAAK,MAAM,KAAK,IAAI,aAAa,EAAE,CAAC;gBAClC,MAAM,aAAa,GAAG,KAAK,CAAC,gBAAgB,EAAE,MAAM,IAAI,CAAC,CAAC;gBAC1D,OAAO,CAAC,GAAG,CAAC,OAAO,KAAK,CAAC,SAAS,KAAK,aAAa,aAAa,CAAC,CAAC;YACrE,CAAC;QACH,CAAC;QAED,IAAI,WAAW,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAC3B,OAAO,CAAC,GAAG,CAAC,uBAAuB,CAAC,CAAC;YACrC,KAAK,MAAM,KAAK,IAAI,WAAW,EAAE,CAAC;gBAChC,OAAO,CAAC,GAAG,CAAC,OAAO,KAAK,CAAC,SAAS,KAAK,KAAK,CAAC,KAAK,EAAE,CAAC,CAAC;YACxD,CAAC;QACH,CAAC;IACH,CAAC;IAEO,WAAW;QACjB,MAAM,UAAU,GAAG,OAAO,CAAC,GAAG,CAAC,sBAAsB,IAAI,8BAA8B,CAAC;QACxF,IAAA,kBAAa,EAAC,UAAU,EAAE,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;QACjE,OAAO,CAAC,GAAG,CAAC,uBAAuB,UAAU,EAAE,CAAC,CAAC;IACnD,CAAC;IAEO,UAAU;QAChB,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,WAAW,EAAE,CAAC;YAC9B,OAAO,KAAK,CAAC;QACf,CAAC;QAED,OAAO,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,WAAW,KAAK,SAAS,CAAC,CAAC;IAC7D,CAAC;CACF;AAwDQ,sCAAa;AAtDtB,+BAA+B;AAC/B,SAAS,SAAS;IAChB,MAAM,IAAI,GAAG,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;IACnC,MAAM,OAAO,GAA0B;QACrC,MAAM,EAAE,OAAO,CAAC,GAAG,CAAC,UAAU,IAAI,WAAW;KAC9C,CAAC;IAEF,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;QACrC,QAAQ,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC;YAChB,KAAK,UAAU;gBACb,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC,EAAE,CAAC,CAAC,CAAC;gBAC3B,MAAM;YACR,KAAK,UAAU;gBACb,OAAO,CAAC,UAAU,GAAG,IAAI,CAAC,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;gBAC1C,MAAM;YACR,KAAK,WAAW;gBACd,OAAO,CAAC,OAAO,GAAG,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;gBACtC,MAAM;YACR,KAAK,oBAAoB;gBACvB,OAAO,CAAC,WAAW,GAAG,KAAK,CAAC;gBAC5B,MAAM;YACR;gBACE,OAAO,CAAC,KAAK,CAAC,qBAAqB,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;gBAC9C,UAAU,EAAE,CAAC;gBACb,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QACpB,CAAC;IACH,CAAC;IAED,OAAO,OAAO,CAAC;AACjB,CAAC;AAED,SAAS,UAAU;IACjB,OAAO,CAAC,GAAG,CAAC;;;;;;;;;;;;CAYb,CAAC,CAAC;AACH,CAAC;AAED,mBAAmB;AACnB,IAAI,OAAO,CAAC,IAAI,KAAK,MAAM,EAAE,CAAC;IAC5B,MAAM,OAAO,GAAG,SAAS,EAAE,CAAC;IAC5B,MAAM,QAAQ,GAAG,IAAI,aAAa,CAAC,OAAO,CAAC,CAAC;IAC5C,QAAQ,CAAC,GAAG,EAAE,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;AACtC,CAAC","sourcesContent":["#!/usr/bin/env node\n\nimport { execSync } from 'child_process';\nimport { writeFileSync } from 'fs';\n\ninterface DriftDetectionOptions {\n  region: string;\n  stackNames?: string[];\n  timeout?: number;\n  failOnDrift?: boolean;\n}\n\ninterface DriftResult {\n  stackName: string;\n  driftStatus: 'IN_SYNC' | 'DRIFTED' | 'UNKNOWN' | 'NOT_CHECKED';\n  driftedResources?: DriftedResource[];\n  error?: string;\n  knownErrorsHandled?: KnownErrorResult[];\n}\n\ninterface KnownErrorResult {\n  readonly resourceId: string;\n  readonly errorType: string;\n  readonly originalError: any;\n  readonly handled: boolean;\n  readonly message?: string;\n}\n\ninterface DriftedResource {\n  readonly logicalResourceId: string;\n  readonly resourceType: string;\n  readonly stackResourceDriftStatus: string;\n  readonly propertyDifferences?: PropertyDifference[];\n}\n\ninterface PropertyDifference {\n  readonly propertyPath: string;\n  readonly expectedValue: string;\n  readonly actualValue: string;\n  readonly differenceType: string;\n}\n\nclass DriftDetector {\n  private readonly options: DriftDetectionOptions;\n  private readonly results: DriftResult[] = [];\n\n  constructor(options: DriftDetectionOptions) {\n    this.options = {\n      timeout: 30,\n      failOnDrift: true,\n      ...options,\n    };\n  }\n\n  public async run(): Promise<void> {\n    console.log(`Starting drift detection in region ${this.options.region}`);\n\n    try {\n      const stacks = await this.getStacksToCheck();\n\n      for (const stackName of stacks) {\n        await this.checkStackDrift(stackName);\n      }\n\n      this.printSummary();\n      this.saveResults();\n\n      if (this.shouldFail()) {\n        process.exit(1);\n      }\n    } catch (error) {\n      console.error('Fatal error during drift detection:', error);\n      process.exit(2);\n    }\n  }\n\n  private async getStacksToCheck(): Promise<string[]> {\n    if (this.options.stackNames && this.options.stackNames.length > 0) {\n      return this.options.stackNames;\n    }\n\n    console.log('Getting all stacks in the region...');\n    const output = execSync(\n      `aws cloudformation list-stacks --region ${this.options.region} --stack-status-filter CREATE_COMPLETE UPDATE_COMPLETE --query 'StackSummaries[].StackName' --output json`,\n      { encoding: 'utf8' },\n    );\n\n    return JSON.parse(output);\n  }\n\n  private async checkStackDrift(stackName: string): Promise<void> {\n    console.log(`\\nChecking drift for stack: ${stackName}`);\n\n    const result: DriftResult = {\n      stackName,\n      driftStatus: 'NOT_CHECKED',\n    };\n\n    try {\n      // Start drift detection\n      const driftId = await this.startDriftDetection(stackName);\n      console.log(`Started drift detection with ID: ${driftId}`);\n\n      // Wait for completion\n      const driftStatus = await this.waitForDriftDetection(driftId);\n      result.driftStatus = driftStatus;\n\n      if (driftStatus === 'DRIFTED') {\n        result.driftedResources = await this.getDriftedResources(stackName);\n        console.log(`DRIFT DETECTED in stack ${stackName}!`);\n\n        // Handle known drift errors\n        result.knownErrorsHandled = await this.handleKnownDriftErrors(result.driftedResources);\n\n        // Print drift details\n        this.printDriftDetails(result);\n      } else if (driftStatus === 'IN_SYNC') {\n        console.log(`Stack ${stackName} is in sync`);\n      }\n    } catch (error: any) {\n      console.error(`Error checking drift for stack ${stackName}:`, error.message);\n      result.error = error.message;\n    }\n\n    this.results.push(result);\n  }\n\n  private async startDriftDetection(stackName: string): Promise<string> {\n    const output = execSync(\n      `aws cloudformation detect-stack-drift --stack-name ${stackName} --region ${this.options.region} --query 'StackDriftDetectionId' --output text`,\n      { encoding: 'utf8' },\n    );\n\n    return output.trim();\n  }\n\n  private async waitForDriftDetection(driftId: string): Promise<'IN_SYNC' | 'DRIFTED' | 'UNKNOWN' | 'NOT_CHECKED'> {\n    const timeout = this.options.timeout! * 60; // Convert to seconds\n    const startTime = Date.now();\n\n    while (true) {\n      const elapsed = Math.floor((Date.now() - startTime) / 1000);\n\n      if (elapsed > timeout) {\n        throw new Error(`Drift detection timed out after ${this.options.timeout} minutes`);\n      }\n\n      const output = execSync(\n        `aws cloudformation describe-stack-drift-detection-status --stack-drift-detection-id ${driftId} --region ${this.options.region} --output json`,\n        { encoding: 'utf8' },\n      );\n\n      const status = JSON.parse(output);\n      console.log(`Drift detection status: ${status.DetectionStatus} (${elapsed}s elapsed)`);\n\n      if (status.DetectionStatus === 'DETECTION_COMPLETE') {\n        return status.StackDriftStatus as 'IN_SYNC' | 'DRIFTED' | 'UNKNOWN' | 'NOT_CHECKED';\n      } else if (status.DetectionStatus === 'DETECTION_FAILED') {\n        throw new Error(`Drift detection failed: ${status.DetectionStatusReason}`);\n      }\n\n      // Wait 10 seconds before checking again\n      await new Promise(resolve => setTimeout(resolve, 10000));\n    }\n  }\n\n  private async getDriftedResources(stackName: string): Promise<DriftedResource[]> {\n    const output = execSync(\n      `aws cloudformation describe-stack-resource-drifts --stack-name ${stackName} --region ${this.options.region} --stack-resource-drift-status-filters MODIFIED DELETED --output json`,\n      { encoding: 'utf8' },\n    );\n\n    const response = JSON.parse(output);\n    return response.StackResourceDrifts.map((drift: any) => ({\n      logicalResourceId: drift.LogicalResourceId,\n      resourceType: drift.ResourceType,\n      stackResourceDriftStatus: drift.StackResourceDriftStatus,\n      propertyDifferences: drift.PropertyDifferences?.map((diff: any) => ({\n        propertyPath: diff.PropertyPath,\n        expectedValue: diff.ExpectedValue,\n        actualValue: diff.ActualValue,\n        differenceType: diff.DifferenceType,\n      })),\n    }));\n  }\n\n  /**\n   * Handle known drift errors for specific resources\n   * This method can be extended later to implement custom logic for known issues\n   */\n  private async handleKnownDriftErrors(driftedResources?: DriftedResource[]): Promise<KnownErrorResult[]> {\n    const knownErrors: KnownErrorResult[] = [];\n\n    if (!driftedResources) {\n      return knownErrors;\n    }\n\n    for (const _resource of driftedResources) {\n\n      // TODO: Implement custom logic here for known drift errors\n      // Example structure for handling known errors:\n\n      // Check for Lambda runtime drift (common issue)\n      // if (resource.resourceType === 'AWS::Lambda::Function' && resource.propertyDifferences) {\n      //   const runtimeDrift = resource.propertyDifferences.find(diff =>\n      //     diff.propertyPath === '/Runtime' || diff.propertyPath === 'Runtime',\n      //   );\n\n      //   if (runtimeDrift) {\n      //     knownErrors.push({\n      //       resourceId: resource.logicalResourceId,\n      //       errorType: 'lambda-runtime-drift',\n      //       originalError: runtimeDrift,\n      //       handled: false, // Will be implemented later\n      //       message: 'Lambda runtime drift detected - manual implementation needed',\n      //     });\n      //   }\n      // }\n\n      // // Check for auto-scaling related drift\n      // if (resource.resourceType.includes('AutoScaling') && resource.propertyDifferences) {\n      //   knownErrors.push({\n      //     resourceId: resource.logicalResourceId,\n      //     errorType: 'autoscaling-drift',\n      //     originalError: resource.propertyDifferences,\n      //     handled: false, // Will be implemented later\n      //     message: 'Auto-scaling drift detected - manual implementation needed',\n      //   });\n      // }\n\n      // Add more patterns here as needed\n    }\n\n    return knownErrors;\n  }\n\n  private printDriftDetails(result: DriftResult): void {\n    if (!result.driftedResources || result.driftedResources.length === 0) {\n      return;\n    }\n\n    console.log('\\nDrifted resources:');\n    console.log('=================');\n\n    for (const resource of result.driftedResources) {\n      console.log(`\\n- ${resource.logicalResourceId} (${resource.resourceType})`);\n      console.log(`  Status: ${resource.stackResourceDriftStatus}`);\n\n      if (resource.propertyDifferences) {\n        console.log('  Property differences:');\n        for (const diff of resource.propertyDifferences) {\n          console.log(`    ${diff.propertyPath}:`);\n          console.log(`      Expected: ${diff.expectedValue}`);\n          console.log(`      Actual: ${diff.actualValue}`);\n          console.log(`      Type: ${diff.differenceType}`);\n        }\n      }\n    }\n  }\n\n  private printSummary(): void {\n    console.log('\\n========== DRIFT DETECTION SUMMARY ==========');\n\n    const driftedStacks = this.results.filter(r => r.driftStatus === 'DRIFTED');\n    const syncedStacks = this.results.filter(r => r.driftStatus === 'IN_SYNC');\n    const errorStacks = this.results.filter(r => r.error);\n\n    console.log(`Total stacks checked: ${this.results.length}`);\n    console.log(`In sync: ${syncedStacks.length}`);\n    console.log(`Drifted: ${driftedStacks.length}`);\n    console.log(`Errors: ${errorStacks.length}`);\n\n    if (driftedStacks.length > 0) {\n      console.log('\\nDrifted stacks:');\n      for (const stack of driftedStacks) {\n        const resourceCount = stack.driftedResources?.length || 0;\n        console.log(`  - ${stack.stackName} (${resourceCount} resources)`);\n      }\n    }\n\n    if (errorStacks.length > 0) {\n      console.log('\\nStacks with errors:');\n      for (const stack of errorStacks) {\n        console.log(`  - ${stack.stackName}: ${stack.error}`);\n      }\n    }\n  }\n\n  private saveResults(): void {\n    const outputFile = process.env.DRIFT_DETECTION_OUTPUT || 'drift-detection-results.json';\n    writeFileSync(outputFile, JSON.stringify(this.results, null, 2));\n    console.log(`\\nResults saved to: ${outputFile}`);\n  }\n\n  private shouldFail(): boolean {\n    if (!this.options.failOnDrift) {\n      return false;\n    }\n\n    return this.results.some(r => r.driftStatus === 'DRIFTED');\n  }\n}\n\n// Parse command line arguments\nfunction parseArgs(): DriftDetectionOptions {\n  const args = process.argv.slice(2);\n  const options: DriftDetectionOptions = {\n    region: process.env.AWS_REGION || 'us-east-1',\n  };\n\n  for (let i = 0; i < args.length; i++) {\n    switch (args[i]) {\n      case '--region':\n        options.region = args[++i];\n        break;\n      case '--stacks':\n        options.stackNames = args[++i].split(',');\n        break;\n      case '--timeout':\n        options.timeout = parseInt(args[++i]);\n        break;\n      case '--no-fail-on-drift':\n        options.failOnDrift = false;\n        break;\n      default:\n        console.error(`Unknown argument: ${args[i]}`);\n        printUsage();\n        process.exit(1);\n    }\n  }\n\n  return options;\n}\n\nfunction printUsage(): void {\n  console.log(`\nUsage: detect-drift.ts [options]\n\nOptions:\n  --region <region>           AWS region (default: us-east-1 or AWS_REGION env var)\n  --stacks <stack1,stack2>    Comma-separated list of stack names (default: all stacks)\n  --timeout <minutes>         Timeout in minutes (default: 30)\n  --no-fail-on-drift         Don't exit with error code if drift is detected\n\nEnvironment variables:\n  AWS_REGION                  Default AWS region\n  DRIFT_DETECTION_OUTPUT      Output file path (default: drift-detection-results.json)\n`);\n}\n\n// Main entry point\nif (require.main === module) {\n  const options = parseArgs();\n  const detector = new DriftDetector(options);\n  detector.run().catch(console.error);\n}\n\nexport { DriftDetector, DriftDetectionOptions, DriftResult, KnownErrorResult, DriftedResource };"]}
@@ -0,0 +1,21 @@
1
+ import { Project } from 'projen';
2
+ import { DriftDetectionWorkflow, DriftDetectionWorkflowOptions } from './base';
3
+ export interface GitHubDriftDetectionWorkflowOptions extends DriftDetectionWorkflowOptions {
4
+ /**
5
+ * Additional permissions for GitHub workflow
6
+ */
7
+ readonly permissions?: Record<string, string>;
8
+ /**
9
+ * Whether to create issues on drift detection
10
+ * @default false
11
+ */
12
+ readonly createIssues?: boolean;
13
+ }
14
+ export declare class GitHubDriftDetectionWorkflow extends DriftDetectionWorkflow {
15
+ private readonly permissions?;
16
+ private readonly createIssues;
17
+ private readonly workflow;
18
+ constructor(project: Project, options: GitHubDriftDetectionWorkflowOptions);
19
+ private generateIssueCreationScript;
20
+ private generateSummaryScript;
21
+ }
@@ -0,0 +1,232 @@
1
+ "use strict";
2
+ var _a;
3
+ Object.defineProperty(exports, "__esModule", { value: true });
4
+ exports.GitHubDriftDetectionWorkflow = void 0;
5
+ const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
6
+ const workflows_model_1 = require("projen/lib/github/workflows-model");
7
+ const base_1 = require("./base");
8
+ const step_1 = require("./step");
9
+ class GitHubDriftDetectionWorkflow extends base_1.DriftDetectionWorkflow {
10
+ constructor(project, options) {
11
+ super(project, options);
12
+ this.permissions = options.permissions;
13
+ this.createIssues = options.createIssues ?? false;
14
+ this.workflow = this.project.github.addWorkflow('drift-detection');
15
+ this.workflow.on({
16
+ schedule: [{
17
+ cron: this.schedule,
18
+ }],
19
+ workflowDispatch: {
20
+ inputs: {
21
+ stage: {
22
+ description: 'Stage to check for drift (leave empty for all)',
23
+ required: false,
24
+ type: 'choice',
25
+ options: this.stages.map(s => s.name),
26
+ },
27
+ },
28
+ },
29
+ });
30
+ // Add job for each stage
31
+ for (const stage of this.stages) {
32
+ const jobId = `drift-${stage.name}`.toLowerCase().replace(/[^a-z0-9-]/g, '-');
33
+ const driftStep = new step_1.DriftDetectionStep(this.project, stage).toGithub();
34
+ this.workflow.addJob(jobId, {
35
+ name: `Drift Detection - ${stage.name}`,
36
+ runsOn: ['ubuntu-latest'],
37
+ if: `\${{ github.event_name == 'schedule' || github.event.inputs.stage == '' || github.event.inputs.stage == '${stage.name}' }}`,
38
+ env: driftStep.env,
39
+ permissions: {
40
+ contents: workflows_model_1.JobPermission.READ,
41
+ ...(driftStep.permissions ?? {}),
42
+ ...(this.createIssues ? { issues: workflows_model_1.JobPermission.WRITE } : {}),
43
+ ...this.permissions,
44
+ },
45
+ steps: [
46
+ {
47
+ name: 'Checkout',
48
+ uses: 'actions/checkout@v4',
49
+ },
50
+ {
51
+ name: 'Setup Node.js',
52
+ uses: 'actions/setup-node@v4',
53
+ with: {
54
+ 'node-version': '20',
55
+ },
56
+ },
57
+ {
58
+ name: 'Install dependencies',
59
+ run: 'npm ci',
60
+ },
61
+ ...driftStep.steps,
62
+ {
63
+ name: 'Upload results',
64
+ uses: 'actions/upload-artifact@v4',
65
+ with: {
66
+ name: `drift-results-${stage.name}`,
67
+ path: `drift-results-${stage.name}.json`,
68
+ },
69
+ },
70
+ ...(this.createIssues ? [{
71
+ name: 'Create Issue on Drift',
72
+ if: 'steps.drift.outcome == \'failure\' && github.event_name == \'schedule\'',
73
+ uses: 'actions/github-script@v7',
74
+ with: {
75
+ script: this.generateIssueCreationScript(stage),
76
+ },
77
+ }] : []),
78
+ ],
79
+ });
80
+ }
81
+ // Add summary job
82
+ if (this.stages.length > 0) {
83
+ this.workflow.addJob('drift-summary', {
84
+ name: 'Drift Detection Summary',
85
+ runsOn: ['ubuntu-latest'],
86
+ permissions: {
87
+ contents: workflows_model_1.JobPermission.READ,
88
+ },
89
+ needs: this.stages.map(stage => `drift-${stage.name}`),
90
+ steps: [
91
+ {
92
+ name: 'Download all artifacts',
93
+ uses: 'actions/download-artifact@v4',
94
+ with: {
95
+ path: 'drift-results',
96
+ },
97
+ },
98
+ {
99
+ name: 'Generate summary',
100
+ run: this.generateSummaryScript(),
101
+ },
102
+ ],
103
+ });
104
+ }
105
+ }
106
+ generateIssueCreationScript(stage) {
107
+ return `
108
+ const fs = require('fs');
109
+ const resultsFile = 'drift-results-${stage.name}.json';
110
+
111
+ if (!fs.existsSync(resultsFile)) {
112
+ console.log('No results file found');
113
+ return;
114
+ }
115
+
116
+ const results = JSON.parse(fs.readFileSync(resultsFile, 'utf8'));
117
+ const driftedStacks = results.filter(r => r.driftStatus === 'DRIFTED');
118
+
119
+ if (driftedStacks.length === 0) {
120
+ console.log('No drift detected');
121
+ return;
122
+ }
123
+
124
+ const title = 'Drift Detected in ${stage.name}';
125
+ const body = \`## Drift Detection Report
126
+
127
+ **Stage:** ${stage.name}
128
+ **Region:** ${stage.region}
129
+ **Time:** \${new Date().toISOString()}
130
+
131
+ ### Summary
132
+ - Total stacks checked: \${results.length}
133
+ - Drifted stacks: \${driftedStacks.length}
134
+
135
+ ### Drifted Stacks
136
+ \${driftedStacks.map(stack => {
137
+ const resources = stack.driftedResources || [];
138
+ return \`#### \${stack.stackName}
139
+ - Drifted resources: \${resources.length}
140
+ \${resources.map(r => \` - \${r.logicalResourceId} (\${r.resourceType})\`).join('\\n')}
141
+ \`;
142
+ }).join('\\n')}
143
+
144
+ ### Action Required
145
+ Please review the drifted resources and either:
146
+ 1. Update the infrastructure code to match the actual state
147
+ 2. Restore the resources to match the expected state
148
+
149
+ [View workflow run](\${context.serverUrl}/\${context.repo.owner}/\${context.repo.repo}/actions/runs/\${context.runId})
150
+ \`;
151
+
152
+ // Check if issue already exists
153
+ const issues = await github.rest.issues.listForRepo({
154
+ owner: context.repo.owner,
155
+ repo: context.repo.repo,
156
+ state: 'open',
157
+ labels: ['drift-detection', '${stage.name}'],
158
+ });
159
+
160
+ if (issues.data.length === 0) {
161
+ await github.rest.issues.create({
162
+ owner: context.repo.owner,
163
+ repo: context.repo.repo,
164
+ title,
165
+ body,
166
+ labels: ['drift-detection', '${stage.name}'],
167
+ });
168
+ } else {
169
+ // Update existing issue
170
+ const issue = issues.data[0];
171
+ await github.rest.issues.createComment({
172
+ owner: context.repo.owner,
173
+ repo: context.repo.repo,
174
+ issue_number: issue.number,
175
+ body: body,
176
+ });
177
+ }
178
+ `;
179
+ }
180
+ generateSummaryScript() {
181
+ return `
182
+ #!/bin/bash
183
+ echo "## Drift Detection Summary" >> $GITHUB_STEP_SUMMARY
184
+ echo "" >> $GITHUB_STEP_SUMMARY
185
+
186
+ total_stacks=0
187
+ total_drifted=0
188
+ total_errors=0
189
+
190
+ for file in drift-results-*.json; do
191
+ if [[ -f "$file" ]]; then
192
+ stage=$(basename $(dirname "$file"))
193
+ echo "### Stage: $stage" >> $GITHUB_STEP_SUMMARY
194
+
195
+ # Parse JSON and generate summary
196
+ jq -r '
197
+ . as $results |
198
+ "- Total stacks: " + ($results | length | tostring) + "\\n" +
199
+ "- Drifted: " + ([$results[] | select(.driftStatus == "DRIFTED")] | length | tostring) + "\\n" +
200
+ "- Errors: " + ([$results[] | select(.error)] | length | tostring) + "\\n" +
201
+ ([$results[] | select(.driftStatus == "DRIFTED")] |
202
+ if length > 0 then
203
+ "\\n**Drifted stacks:**\\n" +
204
+ (map(" - " + .stackName + " (" + ((.driftedResources // []) | length | tostring) + " resources)") | join("\\n"))
205
+ else "" end)
206
+ ' "$file" >> $GITHUB_STEP_SUMMARY
207
+
208
+ echo "" >> $GITHUB_STEP_SUMMARY
209
+
210
+ # Count totals
211
+ total_stacks=$((total_stacks + $(jq 'length' "$file")))
212
+ total_drifted=$((total_drifted + $(jq '[.[] | select(.driftStatus == "DRIFTED")] | length' "$file")))
213
+ total_errors=$((total_errors + $(jq '[.[] | select(.error)] | length' "$file")))
214
+ fi
215
+ done
216
+
217
+ echo "### Overall Summary" >> $GITHUB_STEP_SUMMARY
218
+ echo "- Total stacks checked: $total_stacks" >> $GITHUB_STEP_SUMMARY
219
+ echo "- Total drifted stacks: $total_drifted" >> $GITHUB_STEP_SUMMARY
220
+ echo "- Total errors: $total_errors" >> $GITHUB_STEP_SUMMARY
221
+
222
+ if [[ $total_drifted -gt 0 ]]; then
223
+ echo "" >> $GITHUB_STEP_SUMMARY
224
+ echo "⚠️ **Action required:** Drift detected in $total_drifted stacks" >> $GITHUB_STEP_SUMMARY
225
+ fi
226
+ `;
227
+ }
228
+ }
229
+ exports.GitHubDriftDetectionWorkflow = GitHubDriftDetectionWorkflow;
230
+ _a = JSII_RTTI_SYMBOL_1;
231
+ GitHubDriftDetectionWorkflow[_a] = { fqn: "projen-pipelines.GitHubDriftDetectionWorkflow", version: "0.2.14" };
232
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"github.js","sourceRoot":"","sources":["../../src/drift/github.ts"],"names":[],"mappings":"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","sourcesContent":["import { Project } from 'projen';\nimport { GitHubProject, GithubWorkflow } from 'projen/lib/github';\nimport { JobPermission } from 'projen/lib/github/workflows-model';\nimport { DriftDetectionWorkflow, DriftDetectionWorkflowOptions, DriftDetectionStageOptions } from './base';\nimport { DriftDetectionStep } from './step';\n\nexport interface GitHubDriftDetectionWorkflowOptions extends DriftDetectionWorkflowOptions {\n  /**\n   * Additional permissions for GitHub workflow\n   */\n  readonly permissions?: Record<string, string>;\n\n  /**\n   * Whether to create issues on drift detection\n   * @default false\n   */\n  readonly createIssues?: boolean;\n}\n\nexport class GitHubDriftDetectionWorkflow extends DriftDetectionWorkflow {\n  private readonly permissions?: Record<string, string>;\n  private readonly createIssues: boolean;\n  private readonly workflow: GithubWorkflow;\n\n  constructor(project: Project, options: GitHubDriftDetectionWorkflowOptions) {\n    super(project, options);\n    this.permissions = options.permissions;\n    this.createIssues = options.createIssues ?? false;\n\n    this.workflow = (this.project as GitHubProject).github!.addWorkflow('drift-detection');\n    this.workflow.on({\n      schedule: [{\n        cron: this.schedule,\n      }],\n      workflowDispatch: {\n        inputs: {\n          stage: {\n            description: 'Stage to check for drift (leave empty for all)',\n            required: false,\n            type: 'choice',\n            options: this.stages.map(s => s.name),\n          },\n        },\n      },\n    });\n\n    // Add job for each stage\n    for (const stage of this.stages) {\n      const jobId = `drift-${stage.name}`.toLowerCase().replace(/[^a-z0-9-]/g, '-');\n\n      const driftStep = new DriftDetectionStep(this.project, stage).toGithub();\n\n      this.workflow.addJob(jobId, {\n        name: `Drift Detection - ${stage.name}`,\n        runsOn: ['ubuntu-latest'],\n        if: `\\${{ github.event_name == 'schedule' || github.event.inputs.stage == '' || github.event.inputs.stage == '${stage.name}' }}`,\n        env: driftStep.env,\n        permissions: {\n          contents: JobPermission.READ,\n          ...(driftStep.permissions ?? {}),\n          ...(this.createIssues ? { issues: JobPermission.WRITE } : {}),\n          ...this.permissions,\n        },\n        steps: [\n          {\n            name: 'Checkout',\n            uses: 'actions/checkout@v4',\n          },\n          {\n            name: 'Setup Node.js',\n            uses: 'actions/setup-node@v4',\n            with: {\n              'node-version': '20',\n            },\n          },\n          {\n            name: 'Install dependencies',\n            run: 'npm ci',\n          },\n          ...driftStep.steps,\n          {\n            name: 'Upload results',\n            uses: 'actions/upload-artifact@v4',\n            with: {\n              name: `drift-results-${stage.name}`,\n              path: `drift-results-${stage.name}.json`,\n            },\n          },\n          ...(this.createIssues ? [{\n            name: 'Create Issue on Drift',\n            if: 'steps.drift.outcome == \\'failure\\' && github.event_name == \\'schedule\\'',\n            uses: 'actions/github-script@v7',\n            with: {\n              script: this.generateIssueCreationScript(stage),\n            },\n          }] : []),\n        ],\n      });\n    }\n\n    // Add summary job\n    if (this.stages.length > 0) {\n      this.workflow.addJob('drift-summary', {\n        name: 'Drift Detection Summary',\n        runsOn: ['ubuntu-latest'],\n        permissions: {\n          contents: JobPermission.READ,\n        },\n        needs: this.stages.map(stage => `drift-${stage.name}`),\n        steps: [\n          {\n            name: 'Download all artifacts',\n            uses: 'actions/download-artifact@v4',\n            with: {\n              path: 'drift-results',\n            },\n          },\n          {\n            name: 'Generate summary',\n            run: this.generateSummaryScript(),\n          },\n        ],\n      });\n    }\n  }\n\n  private generateIssueCreationScript(stage: DriftDetectionStageOptions): string {\n    return `\nconst fs = require('fs');\nconst resultsFile = 'drift-results-${stage.name}.json';\n\nif (!fs.existsSync(resultsFile)) {\n  console.log('No results file found');\n  return;\n}\n\nconst results = JSON.parse(fs.readFileSync(resultsFile, 'utf8'));\nconst driftedStacks = results.filter(r => r.driftStatus === 'DRIFTED');\n\nif (driftedStacks.length === 0) {\n  console.log('No drift detected');\n  return;\n}\n\nconst title = 'Drift Detected in ${stage.name}';\nconst body = \\`## Drift Detection Report\n\n**Stage:** ${stage.name}\n**Region:** ${stage.region}\n**Time:** \\${new Date().toISOString()}\n\n### Summary\n- Total stacks checked: \\${results.length}\n- Drifted stacks: \\${driftedStacks.length}\n\n### Drifted Stacks\n\\${driftedStacks.map(stack => {\n  const resources = stack.driftedResources || [];\n  return \\`#### \\${stack.stackName}\n- Drifted resources: \\${resources.length}\n\\${resources.map(r => \\`  - \\${r.logicalResourceId} (\\${r.resourceType})\\`).join('\\\\n')}\n\\`;\n}).join('\\\\n')}\n\n### Action Required\nPlease review the drifted resources and either:\n1. Update the infrastructure code to match the actual state\n2. Restore the resources to match the expected state\n\n[View workflow run](\\${context.serverUrl}/\\${context.repo.owner}/\\${context.repo.repo}/actions/runs/\\${context.runId})\n\\`;\n\n// Check if issue already exists\nconst issues = await github.rest.issues.listForRepo({\n  owner: context.repo.owner,\n  repo: context.repo.repo,\n  state: 'open',\n  labels: ['drift-detection', '${stage.name}'],\n});\n\nif (issues.data.length === 0) {\n  await github.rest.issues.create({\n    owner: context.repo.owner,\n    repo: context.repo.repo,\n    title,\n    body,\n    labels: ['drift-detection', '${stage.name}'],\n  });\n} else {\n  // Update existing issue\n  const issue = issues.data[0];\n  await github.rest.issues.createComment({\n    owner: context.repo.owner,\n    repo: context.repo.repo,\n    issue_number: issue.number,\n    body: body,\n  });\n}\n`;\n  }\n\n  private generateSummaryScript(): string {\n    return `\n#!/bin/bash\necho \"## Drift Detection Summary\" >> $GITHUB_STEP_SUMMARY\necho \"\" >> $GITHUB_STEP_SUMMARY\n\ntotal_stacks=0\ntotal_drifted=0\ntotal_errors=0\n\nfor file in drift-results-*.json; do\n  if [[ -f \"$file\" ]]; then\n    stage=$(basename $(dirname \"$file\"))\n    echo \"### Stage: $stage\" >> $GITHUB_STEP_SUMMARY\n    \n    # Parse JSON and generate summary\n    jq -r '\n      . as $results |\n      \"- Total stacks: \" + ($results | length | tostring) + \"\\\\n\" +\n      \"- Drifted: \" + ([$results[] | select(.driftStatus == \"DRIFTED\")] | length | tostring) + \"\\\\n\" +\n      \"- Errors: \" + ([$results[] | select(.error)] | length | tostring) + \"\\\\n\" +\n      ([$results[] | select(.driftStatus == \"DRIFTED\")] | \n        if length > 0 then\n          \"\\\\n**Drifted stacks:**\\\\n\" + \n          (map(\"  - \" + .stackName + \" (\" + ((.driftedResources // []) | length | tostring) + \" resources)\") | join(\"\\\\n\"))\n        else \"\" end)\n    ' \"$file\" >> $GITHUB_STEP_SUMMARY\n    \n    echo \"\" >> $GITHUB_STEP_SUMMARY\n    \n    # Count totals\n    total_stacks=$((total_stacks + $(jq 'length' \"$file\")))\n    total_drifted=$((total_drifted + $(jq '[.[] | select(.driftStatus == \"DRIFTED\")] | length' \"$file\")))\n    total_errors=$((total_errors + $(jq '[.[] | select(.error)] | length' \"$file\")))\n  fi\ndone\n\necho \"### Overall Summary\" >> $GITHUB_STEP_SUMMARY\necho \"- Total stacks checked: $total_stacks\" >> $GITHUB_STEP_SUMMARY\necho \"- Total drifted stacks: $total_drifted\" >> $GITHUB_STEP_SUMMARY\necho \"- Total errors: $total_errors\" >> $GITHUB_STEP_SUMMARY\n\nif [[ $total_drifted -gt 0 ]]; then\n  echo \"\" >> $GITHUB_STEP_SUMMARY\n  echo \"⚠️ **Action required:** Drift detected in $total_drifted stacks\" >> $GITHUB_STEP_SUMMARY\nfi\n`;\n  }\n\n}"]}
@@ -0,0 +1,20 @@
1
+ import { Project } from 'projen';
2
+ import { DriftDetectionWorkflow, DriftDetectionWorkflowOptions } from './base';
3
+ export interface GitLabDriftDetectionWorkflowOptions extends DriftDetectionWorkflowOptions {
4
+ /**
5
+ * GitLab runner tags
6
+ */
7
+ readonly runnerTags?: string[];
8
+ /**
9
+ * Docker image to use for drift detection
10
+ * @default "node:18"
11
+ */
12
+ readonly image?: string;
13
+ }
14
+ export declare class GitLabDriftDetectionWorkflow extends DriftDetectionWorkflow {
15
+ private readonly runnerTags;
16
+ private readonly image;
17
+ private readonly config;
18
+ constructor(project: Project, options: GitLabDriftDetectionWorkflowOptions);
19
+ private generateSummaryScript;
20
+ }