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.
- package/.jsii +5703 -1380
- package/API.md +5813 -1492
- package/README.md +253 -0
- package/docs/drift-detection.md +264 -0
- package/lib/assign-approver/base.js +1 -1
- package/lib/assign-approver/github.js +1 -1
- package/lib/awscdk/base.d.ts +21 -0
- package/lib/awscdk/base.js +246 -2
- package/lib/awscdk/bash.js +1 -1
- package/lib/awscdk/github.js +1 -1
- package/lib/awscdk/gitlab.js +1 -1
- package/lib/drift/base.d.ts +64 -0
- package/lib/drift/base.js +18 -0
- package/lib/drift/bash.d.ts +15 -0
- package/lib/drift/bash.js +170 -0
- package/lib/drift/detect-drift.d.ts +54 -0
- package/lib/drift/detect-drift.js +259 -0
- package/lib/drift/github.d.ts +21 -0
- package/lib/drift/github.js +232 -0
- package/lib/drift/gitlab.d.ts +20 -0
- package/lib/drift/gitlab.js +138 -0
- package/lib/drift/index.d.ts +5 -0
- package/lib/drift/index.js +22 -0
- package/lib/drift/step.d.ts +14 -0
- package/lib/drift/step.js +48 -0
- package/lib/index.d.ts +2 -0
- package/lib/index.js +3 -1
- package/lib/steps/artifact-steps.js +2 -2
- package/lib/steps/aws-assume-role.step.js +1 -1
- package/lib/steps/registries.js +2 -2
- package/lib/steps/step.d.ts +6 -1
- package/lib/steps/step.js +14 -10
- package/lib/versioning/computation.d.ts +63 -0
- package/lib/versioning/computation.js +121 -0
- package/lib/versioning/config.d.ts +41 -0
- package/lib/versioning/config.js +91 -0
- package/lib/versioning/index.d.ts +7 -0
- package/lib/versioning/index.js +46 -0
- package/lib/versioning/outputs.d.ts +87 -0
- package/lib/versioning/outputs.js +166 -0
- package/lib/versioning/setup.d.ts +30 -0
- package/lib/versioning/setup.js +165 -0
- package/lib/versioning/strategy.d.ts +21 -0
- package/lib/versioning/strategy.js +51 -0
- package/lib/versioning/types.d.ts +183 -0
- package/lib/versioning/types.js +3 -0
- package/lib/versioning/version-info.d.ts +106 -0
- package/lib/versioning/version-info.js +269 -0
- 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":"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","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
|
+
}
|