@jjrawlins/cdk-diff-pr-github-action 0.0.1-beta → 0.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.jsii +475 -33
- package/.mergify.yml +102 -0
- package/API.md +351 -11
- package/README.md +223 -39
- package/lib/CdkDiffIamTemplate.d.ts +3 -1
- package/lib/CdkDiffIamTemplate.js +10 -5
- package/lib/CdkDiffStackWorkflow.d.ts +2 -2
- package/lib/CdkDiffStackWorkflow.js +19 -20
- package/lib/CdkDriftDetectionWorkflow.d.ts +32 -0
- package/lib/CdkDriftDetectionWorkflow.js +281 -0
- package/lib/CdkDriftIamTemplate.d.ts +10 -0
- package/lib/CdkDriftIamTemplate.js +77 -0
- package/lib/bin/cdk-changeset-script.js +3 -3
- package/lib/bin/cdk-drift-detection-script.d.ts +15 -0
- package/lib/bin/cdk-drift-detection-script.js +196 -0
- package/lib/bin/detect-drift.js +162 -0
- package/lib/index.d.ts +2 -0
- package/lib/index.js +3 -1
- package/package.json +7 -2
- package/sonar-project.properties +17 -0
- package/.junie/guidelines.md +0 -62
- package/.yalc/@jjrawlins/cdk-diff-pr-github-action/.jsii +0 -3917
- package/.yalc/@jjrawlins/cdk-diff-pr-github-action/.junie/guidelines.md +0 -62
- package/.yalc/@jjrawlins/cdk-diff-pr-github-action/.tool-versions +0 -3
- package/.yalc/@jjrawlins/cdk-diff-pr-github-action/API.md +0 -276
- package/.yalc/@jjrawlins/cdk-diff-pr-github-action/LICENSE +0 -202
- package/.yalc/@jjrawlins/cdk-diff-pr-github-action/README.md +0 -146
- package/.yalc/@jjrawlins/cdk-diff-pr-github-action/lib/CdkDiffIamTemplate.d.ts +0 -8
- package/.yalc/@jjrawlins/cdk-diff-pr-github-action/lib/CdkDiffIamTemplate.js +0 -96
- package/.yalc/@jjrawlins/cdk-diff-pr-github-action/lib/CdkDiffStackWorkflow.d.ts +0 -22
- package/.yalc/@jjrawlins/cdk-diff-pr-github-action/lib/CdkDiffStackWorkflow.js +0 -144
- package/.yalc/@jjrawlins/cdk-diff-pr-github-action/lib/bin/cdk-changeset-script.d.ts +0 -9
- package/.yalc/@jjrawlins/cdk-diff-pr-github-action/lib/bin/cdk-changeset-script.js +0 -256
- package/.yalc/@jjrawlins/cdk-diff-pr-github-action/lib/bin/describe-cfn-changeset.js +0 -204
- package/.yalc/@jjrawlins/cdk-diff-pr-github-action/lib/index.d.ts +0 -2
- package/.yalc/@jjrawlins/cdk-diff-pr-github-action/lib/index.js +0 -19
- package/.yalc/@jjrawlins/cdk-diff-pr-github-action/package.json +0 -137
- package/.yalc/@jjrawlins/cdk-diff-pr-github-action/yalc.lock +0 -10
- package/.yalc/@jjrawlins/cdk-diff-pr-github-action/yalc.sig +0 -1
- package/lib/bin/describe-cfn-changeset.d.ts +0 -1
- package/lib/bin/describe-cfn-changeset.js +0 -204
- package/yalc.lock +0 -10
- /package/{.yalc/@jjrawlins/cdk-diff-pr-github-action/lib/bin/describe-cfn-changeset.d.ts → lib/bin/detect-drift.d.ts} +0 -0
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
export interface Stack {
|
|
2
|
+
readonly stackName: string;
|
|
3
|
+
readonly driftDetectionRoleToAssumeRegion: string;
|
|
4
|
+
readonly driftDetectionRoleToAssumeArn: string;
|
|
5
|
+
readonly failOnDrift?: boolean;
|
|
6
|
+
}
|
|
7
|
+
export interface CdkDriftDetectionWorkflowProps {
|
|
8
|
+
readonly scriptOutputPath?: string;
|
|
9
|
+
readonly project: any;
|
|
10
|
+
readonly workflowName?: string;
|
|
11
|
+
readonly schedule?: string;
|
|
12
|
+
readonly createIssues?: boolean;
|
|
13
|
+
readonly oidcRoleArn: string;
|
|
14
|
+
readonly oidcRegion: string;
|
|
15
|
+
readonly stacks: Stack[];
|
|
16
|
+
readonly nodeVersion?: string;
|
|
17
|
+
/**
|
|
18
|
+
* Optional hook to append additional GitHub Actions steps after drift detection per stack.
|
|
19
|
+
* You can supply a static array of steps, or a factory that receives context and returns steps.
|
|
20
|
+
*/
|
|
21
|
+
/**
|
|
22
|
+
* Optional additional GitHub Action steps to run after drift detection for each stack.
|
|
23
|
+
* These steps run after results are uploaded for each stack. You can include
|
|
24
|
+
* any notifications you like (e.g., Slack). Provide explicit inputs (e.g., payload/markdown)
|
|
25
|
+
* directly in your step without relying on a pre-generated payload.
|
|
26
|
+
*/
|
|
27
|
+
readonly postGitHubSteps?: any;
|
|
28
|
+
}
|
|
29
|
+
export declare class CdkDriftDetectionWorkflow {
|
|
30
|
+
private static scriptCreated;
|
|
31
|
+
constructor(props: CdkDriftDetectionWorkflowProps);
|
|
32
|
+
}
|
|
@@ -0,0 +1,281 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var _a;
|
|
3
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
4
|
+
exports.CdkDriftDetectionWorkflow = void 0;
|
|
5
|
+
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
|
6
|
+
const github_1 = require("projen/lib/github");
|
|
7
|
+
const workflows_model_1 = require("projen/lib/github/workflows-model");
|
|
8
|
+
const cdk_drift_detection_script_1 = require("./bin/cdk-drift-detection-script");
|
|
9
|
+
const githubActionsAwsCredentialsVersion = 'v5';
|
|
10
|
+
const githubActionsCheckoutVersion = 'v5';
|
|
11
|
+
const githubActionsSetupNodeVersion = 'v5';
|
|
12
|
+
const githubActionsUploadArtifactVersion = 'v4';
|
|
13
|
+
const githubActionsDownloadArtifactVersion = 'v5';
|
|
14
|
+
const githubActionsGithubScriptVersion = 'v8';
|
|
15
|
+
class CdkDriftDetectionWorkflow {
|
|
16
|
+
constructor(props) {
|
|
17
|
+
const name = props.workflowName ?? 'drift-detection';
|
|
18
|
+
const fileName = toKebabCase(name) + '.yml';
|
|
19
|
+
const nodeVersion = props.nodeVersion ?? '24.x';
|
|
20
|
+
const createIssues = props.createIssues ?? true;
|
|
21
|
+
const project = props.project;
|
|
22
|
+
const scriptOutputPath = props.scriptOutputPath ?? '.github/workflows/scripts/detect-drift.ts';
|
|
23
|
+
// Only create the drift detection script once to avoid collisions
|
|
24
|
+
if (!CdkDriftDetectionWorkflow.scriptCreated) {
|
|
25
|
+
new cdk_drift_detection_script_1.CdkDriftDetectionScript({
|
|
26
|
+
project: props.project,
|
|
27
|
+
outputPath: scriptOutputPath,
|
|
28
|
+
});
|
|
29
|
+
CdkDriftDetectionWorkflow.scriptCreated = true;
|
|
30
|
+
}
|
|
31
|
+
const gh = project.github ?? new github_1.GitHub(project);
|
|
32
|
+
const workflow = new github_1.GithubWorkflow(gh, name, { fileName });
|
|
33
|
+
// triggers: schedule + manual dispatch with stack choice
|
|
34
|
+
const stackChoices = ['all', ...props.stacks.map((s) => s.stackName)];
|
|
35
|
+
workflow.on({
|
|
36
|
+
schedule: props.schedule ? [{ cron: props.schedule }] : undefined,
|
|
37
|
+
workflowDispatch: {
|
|
38
|
+
inputs: {
|
|
39
|
+
stack: {
|
|
40
|
+
description: "Stack to check for drift ('all' to run every stack)",
|
|
41
|
+
required: false,
|
|
42
|
+
type: 'choice',
|
|
43
|
+
options: stackChoices,
|
|
44
|
+
},
|
|
45
|
+
},
|
|
46
|
+
},
|
|
47
|
+
});
|
|
48
|
+
// One job per stack
|
|
49
|
+
const jobs = {};
|
|
50
|
+
for (const stack of props.stacks) {
|
|
51
|
+
const sanitizedStackName = toKebabCase(toGithubJobId(stack.stackName));
|
|
52
|
+
const originalStackName = stack.stackName;
|
|
53
|
+
const jobId = `drift-${sanitizedStackName}`;
|
|
54
|
+
const resultsFile = `drift-results-${sanitizedStackName}.json`;
|
|
55
|
+
const innerCond = "github.event_name == 'schedule' || !github.event.inputs.stack || github.event.inputs.stack == 'all' || github.event.inputs.stack == '" + sanitizedStackName + "' || github.event.inputs.stack == '" + originalStackName + "'";
|
|
56
|
+
const condExpr = '${{ ' + innerCond + ' }}';
|
|
57
|
+
const notCondExpr = '${{ !(' + innerCond + ') }}';
|
|
58
|
+
const rawPost = props.postGitHubSteps;
|
|
59
|
+
const postSteps = typeof rawPost === 'function' ? rawPost({ stack: sanitizedStackName }) : (rawPost ?? []);
|
|
60
|
+
jobs[jobId] = {
|
|
61
|
+
name: `Drift Detection - ${sanitizedStackName}`,
|
|
62
|
+
runsOn: ['ubuntu-latest'],
|
|
63
|
+
permissions: {
|
|
64
|
+
contents: workflows_model_1.JobPermission.READ,
|
|
65
|
+
idToken: workflows_model_1.JobPermission.WRITE,
|
|
66
|
+
issues: workflows_model_1.JobPermission.WRITE,
|
|
67
|
+
},
|
|
68
|
+
env: {
|
|
69
|
+
AWS_DEFAULT_REGION: stack.driftDetectionRoleToAssumeRegion,
|
|
70
|
+
AWS_REGION: stack.driftDetectionRoleToAssumeRegion,
|
|
71
|
+
DRIFT_DETECTION_OUTPUT: resultsFile,
|
|
72
|
+
STACK_ID: sanitizedStackName,
|
|
73
|
+
STACK_NAME: stack.stackName,
|
|
74
|
+
},
|
|
75
|
+
// No job-level condition; we gate steps so the job always completes and summary can run
|
|
76
|
+
steps: [
|
|
77
|
+
{ name: 'Skip (stack not selected)', if: notCondExpr, run: 'echo "Stack not selected; skipping drift detection for this job."' },
|
|
78
|
+
{ name: 'Checkout', if: condExpr, uses: `actions/checkout@${githubActionsCheckoutVersion}` },
|
|
79
|
+
{
|
|
80
|
+
name: 'Setup Node.js',
|
|
81
|
+
if: condExpr,
|
|
82
|
+
uses: `actions/setup-node@${githubActionsSetupNodeVersion}`,
|
|
83
|
+
with: { 'node-version': nodeVersion },
|
|
84
|
+
},
|
|
85
|
+
{ name: 'Install dependencies', if: condExpr, run: 'yarn install --frozen-lockfile || npm ci', env: { GITHUB_TOKEN: '${{ secrets.GITHUB_TOKEN }}' } },
|
|
86
|
+
{
|
|
87
|
+
name: 'AWS Credentials',
|
|
88
|
+
if: condExpr,
|
|
89
|
+
id: 'creds',
|
|
90
|
+
uses: `aws-actions/configure-aws-credentials@${githubActionsAwsCredentialsVersion}`,
|
|
91
|
+
with: {
|
|
92
|
+
'role-to-assume': props.oidcRoleArn,
|
|
93
|
+
'role-session-name': 'GitHubAction',
|
|
94
|
+
'aws-region': props.oidcRegion,
|
|
95
|
+
},
|
|
96
|
+
},
|
|
97
|
+
{
|
|
98
|
+
name: 'Assume Drift Detection Role',
|
|
99
|
+
if: condExpr,
|
|
100
|
+
uses: `aws-actions/configure-aws-credentials@${githubActionsAwsCredentialsVersion}`,
|
|
101
|
+
with: {
|
|
102
|
+
'role-to-assume': stack.driftDetectionRoleToAssumeArn,
|
|
103
|
+
'role-chaining': true,
|
|
104
|
+
'role-skip-session-tagging': true,
|
|
105
|
+
'aws-region': stack.driftDetectionRoleToAssumeRegion,
|
|
106
|
+
'aws-access-key-id': '${{ steps.creds.outputs.aws-access-key-id }}',
|
|
107
|
+
'aws-secret-access-key': '${{ steps.creds.outputs.aws-secret-access-key }}',
|
|
108
|
+
'aws-session-token': '${{ steps.creds.outputs.aws-session-token }}',
|
|
109
|
+
},
|
|
110
|
+
},
|
|
111
|
+
{
|
|
112
|
+
name: 'Detect drift',
|
|
113
|
+
if: condExpr,
|
|
114
|
+
id: 'drift',
|
|
115
|
+
continueOnError: true, // allow artifact upload and issue creation even when drift is detected
|
|
116
|
+
run: [
|
|
117
|
+
'set +e',
|
|
118
|
+
// Use the bundled script from this package
|
|
119
|
+
'node ./node_modules/@jjrawlins/cdk-diff-pr-github-action/lib/bin/detect-drift.js',
|
|
120
|
+
'status=$?',
|
|
121
|
+
'if [ -f "$DRIFT_DETECTION_OUTPUT" ]; then echo "Results file created: $DRIFT_DETECTION_OUTPUT"; fi',
|
|
122
|
+
// Expose useful outputs for downstream steps
|
|
123
|
+
"STACK_ARN=$(aws cloudformation describe-stacks --stack-name \"$STACK_NAME\" --query 'Stacks[0].StackId' --output text 2>/dev/null || true)",
|
|
124
|
+
'echo "stack-arn=$STACK_ARN" >> "$GITHUB_OUTPUT"',
|
|
125
|
+
'exit $status',
|
|
126
|
+
].join('\n'),
|
|
127
|
+
env: {
|
|
128
|
+
STACK_NAME: stack.stackName,
|
|
129
|
+
AWS_REGION: stack.driftDetectionRoleToAssumeRegion,
|
|
130
|
+
DRIFT_DETECTION_OUTPUT: resultsFile,
|
|
131
|
+
},
|
|
132
|
+
},
|
|
133
|
+
{
|
|
134
|
+
name: 'Upload results',
|
|
135
|
+
if: condExpr,
|
|
136
|
+
uses: `actions/upload-artifact@${githubActionsUploadArtifactVersion}`,
|
|
137
|
+
with: { 'name': `drift-results-${sanitizedStackName}`, 'path': resultsFile, 'if-no-files-found': 'ignore' },
|
|
138
|
+
},
|
|
139
|
+
...(createIssues
|
|
140
|
+
? [
|
|
141
|
+
{
|
|
142
|
+
name: 'Create Issue on Drift',
|
|
143
|
+
if: "always() && steps.drift.outcome == 'failure'",
|
|
144
|
+
id: 'issue',
|
|
145
|
+
uses: `actions/github-script@${githubActionsGithubScriptVersion}`,
|
|
146
|
+
with: { 'result-encoding': 'string', 'script': issueScript(sanitizedStackName, stack.driftDetectionRoleToAssumeRegion, resultsFile) },
|
|
147
|
+
},
|
|
148
|
+
]
|
|
149
|
+
: []),
|
|
150
|
+
...postSteps.map((step) => {
|
|
151
|
+
const s = { ...step };
|
|
152
|
+
// By default, only run extra notification steps when drift occurs
|
|
153
|
+
s.if = s.if ?? "always() && steps.drift.outcome == 'failure'";
|
|
154
|
+
return s;
|
|
155
|
+
}),
|
|
156
|
+
],
|
|
157
|
+
};
|
|
158
|
+
}
|
|
159
|
+
// summary aggregator job
|
|
160
|
+
jobs['drift-summary'] = {
|
|
161
|
+
name: 'Drift Detection Summary',
|
|
162
|
+
needs: Object.keys(jobs).filter((j) => j.startsWith('drift-') && j !== 'drift-summary'),
|
|
163
|
+
runsOn: ['ubuntu-latest'],
|
|
164
|
+
permissions: { contents: workflows_model_1.JobPermission.READ },
|
|
165
|
+
steps: [
|
|
166
|
+
{
|
|
167
|
+
name: 'Download all artifacts',
|
|
168
|
+
uses: `actions/download-artifact@${githubActionsDownloadArtifactVersion}`,
|
|
169
|
+
with: { path: 'drift-results' },
|
|
170
|
+
},
|
|
171
|
+
{ name: 'Generate summary', shell: 'bash', run: summaryScript() },
|
|
172
|
+
],
|
|
173
|
+
};
|
|
174
|
+
workflow.addJobs(jobs);
|
|
175
|
+
}
|
|
176
|
+
}
|
|
177
|
+
exports.CdkDriftDetectionWorkflow = CdkDriftDetectionWorkflow;
|
|
178
|
+
_a = JSII_RTTI_SYMBOL_1;
|
|
179
|
+
CdkDriftDetectionWorkflow[_a] = { fqn: "@jjrawlins/cdk-diff-pr-github-action.CdkDriftDetectionWorkflow", version: "0.0.1" };
|
|
180
|
+
CdkDriftDetectionWorkflow.scriptCreated = false;
|
|
181
|
+
function issueScript(stack, region, resultsFile) {
|
|
182
|
+
// Construct a plain JS script string (no template string nesting mishaps)
|
|
183
|
+
const lines = [
|
|
184
|
+
"const fs = require('fs');",
|
|
185
|
+
`const resultsFile = '${resultsFile}';`,
|
|
186
|
+
"if (!fs.existsSync(resultsFile)) { console.log('No results file found'); return 'NO_RESULTS'; }",
|
|
187
|
+
"const results = JSON.parse(fs.readFileSync(resultsFile, 'utf8'));",
|
|
188
|
+
"const driftedStacks = results.filter(r => r.driftStatus === 'DRIFTED');",
|
|
189
|
+
"if (driftedStacks.length === 0) { console.log('No drift detected'); return 'NO_DRIFT'; }",
|
|
190
|
+
`const title = 'Drift Detected in ${stack}';`,
|
|
191
|
+
`let body = '## Drift Detection Report\\n\\n' + '**Stack:** ${stack}\\n' + '**Region:** ${region}\\n' + '**Time:** ' + new Date().toISOString() + '\\n\\n';`,
|
|
192
|
+
"body += '### Summary\\n';",
|
|
193
|
+
"body += '- Total stacks checked: ' + results.length + '\\n';",
|
|
194
|
+
"body += '- Drifted stacks: ' + driftedStacks.length + '\\n\\n';",
|
|
195
|
+
"body += '### Drifted Stacks\\n';",
|
|
196
|
+
'for (const s of driftedStacks) {',
|
|
197
|
+
' const resources = s.driftedResources || [];',
|
|
198
|
+
" body += '#### ' + s.stackName + '\\n';",
|
|
199
|
+
" body += '- Drifted resources: ' + resources.length + '\\n';",
|
|
200
|
+
" for (const r of resources) { body += ' - ' + r.logicalResourceId + ' (' + r.resourceType + ')\\n'; }",
|
|
201
|
+
" body += '\\n';",
|
|
202
|
+
'}',
|
|
203
|
+
"body += '### Action Required\\n' + 'Please 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';",
|
|
204
|
+
'body += `[View workflow run](${context.serverUrl}/${context.repo.owner}/${context.repo.repo}/actions/runs/${context.runId})`;',
|
|
205
|
+
// List or update an issue with labels
|
|
206
|
+
`const issues = await github.rest.issues.listForRepo({ owner: context.repo.owner, repo: context.repo.repo, state: 'open', labels: ['drift-detection', '${stack}'] });`,
|
|
207
|
+
'let issueNumber;',
|
|
208
|
+
'if (issues.data.length === 0) {',
|
|
209
|
+
` const created = await github.rest.issues.create({ owner: context.repo.owner, repo: context.repo.repo, title, body, labels: ['drift-detection', '${stack}'] });`,
|
|
210
|
+
' issueNumber = created.data.number;',
|
|
211
|
+
'} else {',
|
|
212
|
+
' const issue = issues.data[0];',
|
|
213
|
+
' issueNumber = issue.number;',
|
|
214
|
+
' await github.rest.issues.createComment({ owner: context.repo.owner, repo: context.repo.repo, issue_number: issue.number, body });',
|
|
215
|
+
'}',
|
|
216
|
+
'return String(issueNumber ?? "");',
|
|
217
|
+
];
|
|
218
|
+
return lines.join('\n');
|
|
219
|
+
}
|
|
220
|
+
function summaryScript() {
|
|
221
|
+
return [
|
|
222
|
+
'#!/bin/bash',
|
|
223
|
+
'set -e',
|
|
224
|
+
'echo "## Drift Detection Summary" >> $GITHUB_STEP_SUMMARY',
|
|
225
|
+
'echo "" >> $GITHUB_STEP_SUMMARY',
|
|
226
|
+
'',
|
|
227
|
+
'total_stacks=0',
|
|
228
|
+
'total_drifted=0',
|
|
229
|
+
'total_errors=0',
|
|
230
|
+
'',
|
|
231
|
+
'shopt -s nullglob',
|
|
232
|
+
'for file in drift-results-*.json drift-results/*/drift-results-*.json; do',
|
|
233
|
+
' if [[ -f "$file" ]]; then',
|
|
234
|
+
' stack=$(basename "$file" | sed -E \"s/^drift-results-([^.]+)\\.json$/\\1/\")',
|
|
235
|
+
' echo "### Stack: $stack" >> $GITHUB_STEP_SUMMARY',
|
|
236
|
+
' jq -r \'' +
|
|
237
|
+
'. as $results |\n' +
|
|
238
|
+
'"- Total stacks: " + ($results | length | tostring) + "\\n" +\n' +
|
|
239
|
+
'"- Drifted: " + ([.[] | select(.driftStatus == "DRIFTED")] | length | tostring) + "\\n" +\n' +
|
|
240
|
+
'"- Errors: " + ([.[] | select(.error)] | length | tostring) + "\\n" +\n' +
|
|
241
|
+
'([.[] | select(.driftStatus == "DRIFTED")] | if length > 0 then "\\n**Drifted stacks:**\\n" + (map(" - " + .stackName + " (" + ((.driftedResources // []) | length | tostring) + " resources)") | join("\\n")) else "" end)\n' +
|
|
242
|
+
'\'' +
|
|
243
|
+
' "$file" >> $GITHUB_STEP_SUMMARY',
|
|
244
|
+
' echo "" >> $GITHUB_STEP_SUMMARY',
|
|
245
|
+
' total_stacks=$((total_stacks + $(jq \"length\" \"$file\")))',
|
|
246
|
+
' total_drifted=$((total_drifted + $(jq \"[.[] | select(.driftStatus == \\\"DRIFTED\\\")] | length\" \"$file\")))',
|
|
247
|
+
' total_errors=$((total_errors + $(jq \"[.[] | select(.error)] | length\" \"$file\")))',
|
|
248
|
+
' fi',
|
|
249
|
+
'done',
|
|
250
|
+
'',
|
|
251
|
+
'echo "### Overall Summary" >> $GITHUB_STEP_SUMMARY',
|
|
252
|
+
'echo "- Total stacks checked: $total_stacks" >> $GITHUB_STEP_SUMMARY',
|
|
253
|
+
'echo "- Total drifted stacks: $total_drifted" >> $GITHUB_STEP_SUMMARY',
|
|
254
|
+
'echo "- Total errors: $total_errors" >> $GITHUB_STEP_SUMMARY',
|
|
255
|
+
'',
|
|
256
|
+
'if [[ $total_drifted -gt 0 ]]; then',
|
|
257
|
+
' echo "" >> $GITHUB_STEP_SUMMARY',
|
|
258
|
+
' echo "⚠️ **Action required:** Drift detected in $total_drifted stacks" >> $GITHUB_STEP_SUMMARY',
|
|
259
|
+
'fi',
|
|
260
|
+
].join('\n');
|
|
261
|
+
}
|
|
262
|
+
function toKebabCase(s) {
|
|
263
|
+
return s.replace(/[^a-zA-Z0-9]+/g, '-').replace(/^-+|-+$/g, '').toLowerCase();
|
|
264
|
+
}
|
|
265
|
+
function toGithubJobId(s) {
|
|
266
|
+
// GitHub job_id must start with a letter or underscore and contain only A-Za-z0-9, '-', '_'
|
|
267
|
+
// 1) Replace any disallowed chars with '-'
|
|
268
|
+
let out = s.replace(/[^A-Za-z0-9_-]+/g, '-');
|
|
269
|
+
// 2) Collapse consecutive dashes
|
|
270
|
+
out = out.replace(/-+/g, '-');
|
|
271
|
+
// 3) Trim leading/trailing dashes (underscores are allowed at start)
|
|
272
|
+
out = out.replace(/^-+|-+$/g, '');
|
|
273
|
+
// 4) Lowercase for consistency (not required by GitHub but keeps things stable)
|
|
274
|
+
out = out.toLowerCase();
|
|
275
|
+
// 5) Ensure it starts with a letter or underscore
|
|
276
|
+
if (!out || !/^[a-z_]/i.test(out)) {
|
|
277
|
+
out = `s-${out}`;
|
|
278
|
+
}
|
|
279
|
+
return out;
|
|
280
|
+
}
|
|
281
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CdkDriftDetectionWorkflow.js","sourceRoot":"","sources":["../src/CdkDriftDetectionWorkflow.ts"],"names":[],"mappings":"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","sourcesContent":["import { GitHub, GithubWorkflow } from 'projen/lib/github';\nimport { JobPermission } from 'projen/lib/github/workflows-model';\nimport { CdkDriftDetectionScript } from './bin/cdk-drift-detection-script';\n\nconst githubActionsAwsCredentialsVersion = 'v5';\nconst githubActionsCheckoutVersion = 'v5';\nconst githubActionsSetupNodeVersion = 'v5';\nconst githubActionsUploadArtifactVersion = 'v4';\nconst githubActionsDownloadArtifactVersion = 'v5';\nconst githubActionsGithubScriptVersion = 'v8';\n\nexport interface Stack {\n  readonly stackName: string;\n  readonly driftDetectionRoleToAssumeRegion: string;\n  readonly driftDetectionRoleToAssumeArn: string;\n  readonly failOnDrift?: boolean; // if true, fail job when drift detected (default true)\n}\n\nexport interface CdkDriftDetectionWorkflowProps {\n  readonly scriptOutputPath?: string;\n  readonly project: any; // avoid exporting projen types in public API\n  readonly workflowName?: string; // workflow workflowName (also used to derive file workflowName)\n  readonly schedule?: string; // cron expression, e.g. '0 0 * * *'\n  readonly createIssues?: boolean; // create/update issue when drift detected on schedule (default true)\n  readonly oidcRoleArn: string; // default OIDC role ARN to assume for all stacks\n  readonly oidcRegion: string; // default OIDC region to assume for all stacks\n  readonly stacks: Stack[];\n  readonly nodeVersion?: string; // e.g., '24.x'\n  /**\n   * Optional hook to append additional GitHub Actions steps after drift detection per stack.\n   * You can supply a static array of steps, or a factory that receives context and returns steps.\n   */\n  /**\n   * Optional additional GitHub Action steps to run after drift detection for each stack.\n   * These steps run after results are uploaded for each stack. You can include\n   * any notifications you like (e.g., Slack). Provide explicit inputs (e.g., payload/markdown)\n   * directly in your step without relying on a pre-generated payload.\n   */\n  // NOTE: jsii does not support function types in public APIs; use 'any' here and accept either:\n  // - An array of GitHub steps, or\n  // - A function (ctx: { stack: string }) => GitHubStep[]\n  // The constructor handles both at runtime.\n  readonly postGitHubSteps?: any;\n}\n\ntype GitHubStep = {\n  name?: string;\n  id?: string;\n  if?: string;\n  uses?: string;\n  run?: string;\n  with?: Record<string, any>;\n  env?: Record<string, string>;\n  continueOnError?: boolean;\n  shell?: string;\n};\n\nexport class CdkDriftDetectionWorkflow {\n  private static scriptCreated = false;\n\n  constructor(props: CdkDriftDetectionWorkflowProps) {\n    const name = props.workflowName ?? 'drift-detection';\n    const fileName = toKebabCase(name) + '.yml';\n    const nodeVersion = props.nodeVersion ?? '24.x';\n    const createIssues = props.createIssues ?? true;\n    const project = props.project;\n    const scriptOutputPath= props.scriptOutputPath ?? '.github/workflows/scripts/detect-drift.ts';\n\n    // Only create the drift detection script once to avoid collisions\n    if (!CdkDriftDetectionWorkflow.scriptCreated) {\n      new CdkDriftDetectionScript({\n        project: props.project,\n        outputPath: scriptOutputPath,\n      });\n      CdkDriftDetectionWorkflow.scriptCreated = true;\n    }\n\n    const gh = (project as any).github ?? new GitHub(project);\n    const workflow = new GithubWorkflow(gh, name, { fileName });\n\n    // triggers: schedule + manual dispatch with stack choice\n    const stackChoices = ['all', ...props.stacks.map((s) => s.stackName)];\n    workflow.on({\n      schedule: props.schedule ? [{ cron: props.schedule }] : undefined,\n      workflowDispatch: {\n        inputs: {\n          stack: {\n            description: \"Stack to check for drift ('all' to run every stack)\",\n            required: false,\n            type: 'choice',\n            options: stackChoices,\n          },\n        },\n      },\n    });\n\n    // One job per stack\n    const jobs: Record<string, any> = {};\n\n    for (const stack of props.stacks) {\n      const sanitizedStackName = toKebabCase(toGithubJobId(stack.stackName));\n      const originalStackName = stack.stackName;\n      const jobId = `drift-${sanitizedStackName}`;\n      const resultsFile = `drift-results-${sanitizedStackName}.json`;\n      const innerCond = \"github.event_name == 'schedule' || !github.event.inputs.stack || github.event.inputs.stack == 'all' || github.event.inputs.stack == '\" + sanitizedStackName + \"' || github.event.inputs.stack == '\" + originalStackName + \"'\";\n      const condExpr = '${{ ' + innerCond + ' }}';\n      const notCondExpr = '${{ !(' + innerCond + ') }}';\n\n      const rawPost = props.postGitHubSteps;\n      const postSteps: GitHubStep[] = typeof rawPost === 'function' ? (rawPost as (ctx: { stack: string }) => GitHubStep[])({ stack: sanitizedStackName }) : (rawPost ?? []);\n\n      jobs[jobId] = {\n        name: `Drift Detection - ${sanitizedStackName}`,\n        runsOn: ['ubuntu-latest'],\n        permissions: {\n          contents: JobPermission.READ,\n          idToken: JobPermission.WRITE,\n          issues: JobPermission.WRITE,\n        },\n        env: {\n          AWS_DEFAULT_REGION: stack.driftDetectionRoleToAssumeRegion,\n          AWS_REGION: stack.driftDetectionRoleToAssumeRegion,\n          DRIFT_DETECTION_OUTPUT: resultsFile,\n          STACK_ID: sanitizedStackName,\n          STACK_NAME: stack.stackName,\n        },\n        // No job-level condition; we gate steps so the job always completes and summary can run\n        steps: [\n          { name: 'Skip (stack not selected)', if: notCondExpr, run: 'echo \"Stack not selected; skipping drift detection for this job.\"' },\n          { name: 'Checkout', if: condExpr, uses: `actions/checkout@${githubActionsCheckoutVersion}` },\n          {\n            name: 'Setup Node.js',\n            if: condExpr,\n            uses: `actions/setup-node@${githubActionsSetupNodeVersion}`,\n            with: { 'node-version': nodeVersion },\n          },\n          { name: 'Install dependencies', if: condExpr, run: 'yarn install --frozen-lockfile || npm ci', env: { GITHUB_TOKEN: '${{ secrets.GITHUB_TOKEN }}' } },\n          {\n            name: 'AWS Credentials',\n            if: condExpr,\n            id: 'creds',\n            uses: `aws-actions/configure-aws-credentials@${githubActionsAwsCredentialsVersion}`,\n            with: {\n              'role-to-assume': props.oidcRoleArn,\n              'role-session-name': 'GitHubAction',\n              'aws-region': props.oidcRegion,\n            },\n          },\n          {\n            name: 'Assume Drift Detection Role',\n            if: condExpr,\n            uses: `aws-actions/configure-aws-credentials@${githubActionsAwsCredentialsVersion}`,\n            with: {\n              'role-to-assume': stack.driftDetectionRoleToAssumeArn,\n              'role-chaining': true,\n              'role-skip-session-tagging': true,\n              'aws-region': stack.driftDetectionRoleToAssumeRegion,\n              'aws-access-key-id': '${{ steps.creds.outputs.aws-access-key-id }}',\n              'aws-secret-access-key': '${{ steps.creds.outputs.aws-secret-access-key }}',\n              'aws-session-token': '${{ steps.creds.outputs.aws-session-token }}',\n            },\n          },\n          {\n            name: 'Detect drift',\n            if: condExpr,\n            id: 'drift',\n            continueOnError: true, // allow artifact upload and issue creation even when drift is detected\n            run: [\n              'set +e',\n              // Use the bundled script from this package\n              'node ./node_modules/@jjrawlins/cdk-diff-pr-github-action/lib/bin/detect-drift.js',\n              'status=$?',\n              'if [ -f \"$DRIFT_DETECTION_OUTPUT\" ]; then echo \"Results file created: $DRIFT_DETECTION_OUTPUT\"; fi',\n              // Expose useful outputs for downstream steps\n              \"STACK_ARN=$(aws cloudformation describe-stacks --stack-name \\\"$STACK_NAME\\\" --query 'Stacks[0].StackId' --output text 2>/dev/null || true)\",\n              'echo \"stack-arn=$STACK_ARN\" >> \"$GITHUB_OUTPUT\"',\n              'exit $status',\n            ].join('\\n'),\n            env: {\n              STACK_NAME: stack.stackName,\n              AWS_REGION: stack.driftDetectionRoleToAssumeRegion,\n              DRIFT_DETECTION_OUTPUT: resultsFile,\n            },\n          },\n          {\n            name: 'Upload results',\n            if: condExpr,\n            uses: `actions/upload-artifact@${githubActionsUploadArtifactVersion}`,\n            with: { 'name': `drift-results-${sanitizedStackName}`, 'path': resultsFile, 'if-no-files-found': 'ignore' },\n          },\n          ...(\n            createIssues\n              ? [\n                {\n                  name: 'Create Issue on Drift',\n                  if: \"always() && steps.drift.outcome == 'failure'\",\n                  id: 'issue',\n                  uses: `actions/github-script@${githubActionsGithubScriptVersion}`,\n                  with: { 'result-encoding': 'string', 'script': issueScript(sanitizedStackName, stack.driftDetectionRoleToAssumeRegion, resultsFile) },\n                },\n              ]\n              : []\n          ),\n          ...postSteps.map((step) => {\n            const s: any = { ...(step as any) };\n            // By default, only run extra notification steps when drift occurs\n            s.if = s.if ?? \"always() && steps.drift.outcome == 'failure'\";\n            return s;\n          }),\n        ],\n      };\n    }\n\n    // summary aggregator job\n    jobs['drift-summary'] = {\n      name: 'Drift Detection Summary',\n      needs: Object.keys(jobs).filter((j) => j.startsWith('drift-') && j !== 'drift-summary'),\n      runsOn: ['ubuntu-latest'],\n      permissions: { contents: JobPermission.READ },\n      steps: [\n        {\n          name: 'Download all artifacts',\n          uses: `actions/download-artifact@${githubActionsDownloadArtifactVersion}`,\n          with: { path: 'drift-results' },\n        },\n        { name: 'Generate summary', shell: 'bash', run: summaryScript() },\n      ],\n    };\n\n    workflow.addJobs(jobs);\n  }\n}\n\n\nfunction issueScript(stack: string, region: string, resultsFile: string): string {\n  // Construct a plain JS script string (no template string nesting mishaps)\n  const lines = [\n    \"const fs = require('fs');\",\n    `const resultsFile = '${resultsFile}';`,\n    \"if (!fs.existsSync(resultsFile)) { console.log('No results file found'); return 'NO_RESULTS'; }\",\n    \"const results = JSON.parse(fs.readFileSync(resultsFile, 'utf8'));\",\n    \"const driftedStacks = results.filter(r => r.driftStatus === 'DRIFTED');\",\n    \"if (driftedStacks.length === 0) { console.log('No drift detected'); return 'NO_DRIFT'; }\",\n    `const title = 'Drift Detected in ${stack}';`,\n    `let body = '## Drift Detection Report\\\\n\\\\n' + '**Stack:** ${stack}\\\\n' + '**Region:** ${region}\\\\n' + '**Time:** ' + new Date().toISOString() + '\\\\n\\\\n';`,\n    \"body += '### Summary\\\\n';\",\n    \"body += '- Total stacks checked: ' + results.length + '\\\\n';\",\n    \"body += '- Drifted stacks: ' + driftedStacks.length + '\\\\n\\\\n';\",\n    \"body += '### Drifted Stacks\\\\n';\",\n    'for (const s of driftedStacks) {',\n    '  const resources = s.driftedResources || [];',\n    \"  body += '#### ' + s.stackName + '\\\\n';\",\n    \"  body += '- Drifted resources: ' + resources.length + '\\\\n';\",\n    \"  for (const r of resources) { body += '  - ' + r.logicalResourceId + ' (' + r.resourceType + ')\\\\n'; }\",\n    \"  body += '\\\\n';\",\n    '}',\n    \"body += '### Action Required\\\\n' + 'Please 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';\",\n    'body += `[View workflow run](${context.serverUrl}/${context.repo.owner}/${context.repo.repo}/actions/runs/${context.runId})`;',\n    // List or update an issue with labels\n    `const issues = await github.rest.issues.listForRepo({ owner: context.repo.owner, repo: context.repo.repo, state: 'open', labels: ['drift-detection', '${stack}'] });`,\n    'let issueNumber;',\n    'if (issues.data.length === 0) {',\n    `  const created = await github.rest.issues.create({ owner: context.repo.owner, repo: context.repo.repo, title, body, labels: ['drift-detection', '${stack}'] });`,\n    '  issueNumber = created.data.number;',\n    '} else {',\n    '  const issue = issues.data[0];',\n    '  issueNumber = issue.number;',\n    '  await github.rest.issues.createComment({ owner: context.repo.owner, repo: context.repo.repo, issue_number: issue.number, body });',\n    '}',\n    'return String(issueNumber ?? \"\");',\n  ];\n  return lines.join('\\n');\n}\n\n\nfunction summaryScript(): string {\n  return [\n    '#!/bin/bash',\n    'set -e',\n    'echo \"## Drift Detection Summary\" >> $GITHUB_STEP_SUMMARY',\n    'echo \"\" >> $GITHUB_STEP_SUMMARY',\n    '',\n    'total_stacks=0',\n    'total_drifted=0',\n    'total_errors=0',\n    '',\n    'shopt -s nullglob',\n    'for file in drift-results-*.json drift-results/*/drift-results-*.json; do',\n    '  if [[ -f \"$file\" ]]; then',\n    '    stack=$(basename \"$file\" | sed -E \\\"s/^drift-results-([^.]+)\\\\.json$/\\\\1/\\\")',\n    '    echo \"### Stack: $stack\" >> $GITHUB_STEP_SUMMARY',\n    '    jq -r \\'' +\n      '. as $results |\\n' +\n      '\"- Total stacks: \" + ($results | length | tostring) + \"\\\\n\" +\\n' +\n      '\"- Drifted: \" + ([.[] | select(.driftStatus == \"DRIFTED\")] | length | tostring) + \"\\\\n\" +\\n' +\n      '\"- Errors: \" + ([.[] | select(.error)] | length | tostring) + \"\\\\n\" +\\n' +\n      '([.[] | select(.driftStatus == \"DRIFTED\")] | if length > 0 then \"\\\\n**Drifted stacks:**\\\\n\" + (map(\"  - \" + .stackName + \" (\" + ((.driftedResources // []) | length | tostring) + \" resources)\") | join(\"\\\\n\")) else \"\" end)\\n' +\n    '\\'' +\n    ' \"$file\" >> $GITHUB_STEP_SUMMARY',\n    '    echo \"\" >> $GITHUB_STEP_SUMMARY',\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',\n    'done',\n    '',\n    'echo \"### Overall Summary\" >> $GITHUB_STEP_SUMMARY',\n    'echo \"- Total stacks checked: $total_stacks\" >> $GITHUB_STEP_SUMMARY',\n    'echo \"- Total drifted stacks: $total_drifted\" >> $GITHUB_STEP_SUMMARY',\n    'echo \"- Total errors: $total_errors\" >> $GITHUB_STEP_SUMMARY',\n    '',\n    'if [[ $total_drifted -gt 0 ]]; then',\n    '  echo \"\" >> $GITHUB_STEP_SUMMARY',\n    '  echo \"⚠️ **Action required:** Drift detected in $total_drifted stacks\" >> $GITHUB_STEP_SUMMARY',\n    'fi',\n  ].join('\\n');\n}\n\nfunction toKebabCase(s: string): string {\n  return s.replace(/[^a-zA-Z0-9]+/g, '-').replace(/^-+|-+$/g, '').toLowerCase();\n}\n\n\nfunction toGithubJobId(s: string): string {\n  // GitHub job_id must start with a letter or underscore and contain only A-Za-z0-9, '-', '_'\n  // 1) Replace any disallowed chars with '-'\n  let out = s.replace(/[^A-Za-z0-9_-]+/g, '-');\n  // 2) Collapse consecutive dashes\n  out = out.replace(/-+/g, '-');\n  // 3) Trim leading/trailing dashes (underscores are allowed at start)\n  out = out.replace(/^-+|-+$/g, '');\n  // 4) Lowercase for consistency (not required by GitHub but keeps things stable)\n  out = out.toLowerCase();\n  // 5) Ensure it starts with a letter or underscore\n  if (!out || !/^[a-z_]/i.test(out)) {\n    out = `s-${out}`;\n  }\n  return out;\n}\n"]}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
export interface CdkDriftIamTemplateProps {
|
|
2
|
+
readonly project: any;
|
|
3
|
+
readonly roleName: string;
|
|
4
|
+
readonly outputPath?: string;
|
|
5
|
+
readonly oidcRoleArn: string;
|
|
6
|
+
readonly oidcRegion: string;
|
|
7
|
+
}
|
|
8
|
+
export declare class CdkDriftIamTemplate {
|
|
9
|
+
constructor(props: CdkDriftIamTemplateProps);
|
|
10
|
+
}
|
|
@@ -0,0 +1,77 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var _a;
|
|
3
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
4
|
+
exports.CdkDriftIamTemplate = void 0;
|
|
5
|
+
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
|
6
|
+
const projen_1 = require("projen");
|
|
7
|
+
class CdkDriftIamTemplate {
|
|
8
|
+
constructor(props) {
|
|
9
|
+
const outputPath = props.outputPath ?? 'cdk-drift-workflow-iam-template.yaml';
|
|
10
|
+
props.project.addTask('deploy-cdkdrift-iam-template', {
|
|
11
|
+
description: 'Deploy the CDK Drift Detection IAM template via CloudFormation (accepts extra AWS CLI args, e.g., --parameter-overrides Key=Value...)',
|
|
12
|
+
receiveArgs: true,
|
|
13
|
+
exec: `aws cloudformation deploy --template-file ${outputPath} --stack-name cdk-drift-workflow-iam-role --capabilities CAPABILITY_NAMED_IAM`,
|
|
14
|
+
});
|
|
15
|
+
new projen_1.TextFile(props.project, outputPath, {
|
|
16
|
+
lines: [
|
|
17
|
+
"AWSTemplateFormatVersion: '2010-09-09'",
|
|
18
|
+
"Description: 'IAM role for CDK Drift Detection Workflow'",
|
|
19
|
+
'',
|
|
20
|
+
'Parameters:',
|
|
21
|
+
' GitHubOIDCRoleArn:',
|
|
22
|
+
' Type: String',
|
|
23
|
+
" Description: 'ARN of the existing GitHub OIDC role that can assume this drift role'",
|
|
24
|
+
` Default: '${props.oidcRoleArn}'`,
|
|
25
|
+
'',
|
|
26
|
+
'Resources:',
|
|
27
|
+
' # CloudFormation Drift Detection Role - minimal permissions for drift detection operations',
|
|
28
|
+
' CdkDriftRole:',
|
|
29
|
+
' Type: AWS::IAM::Role',
|
|
30
|
+
' Properties:',
|
|
31
|
+
" RoleName: '" + props.roleName + "'",
|
|
32
|
+
' AssumeRolePolicyDocument:',
|
|
33
|
+
" Version: '2012-10-17'",
|
|
34
|
+
' Statement:',
|
|
35
|
+
' - Effect: Allow',
|
|
36
|
+
' Principal:',
|
|
37
|
+
' AWS: !Ref GitHubOIDCRoleArn',
|
|
38
|
+
' Action: sts:AssumeRole',
|
|
39
|
+
' Condition:',
|
|
40
|
+
' StringEquals:',
|
|
41
|
+
" aws:RequestedRegion: '" + props.oidcRegion + "'",
|
|
42
|
+
' Policies:',
|
|
43
|
+
' - PolicyName: CloudFormationDriftAccess',
|
|
44
|
+
' PolicyDocument:',
|
|
45
|
+
" Version: '2012-10-17'",
|
|
46
|
+
' Statement:',
|
|
47
|
+
' # CloudFormation drift detection operations',
|
|
48
|
+
' - Effect: Allow',
|
|
49
|
+
' Action:',
|
|
50
|
+
' - cloudformation:DetectStackDrift',
|
|
51
|
+
' - cloudformation:DescribeStackDriftDetectionStatus',
|
|
52
|
+
' - cloudformation:DescribeStackResourceDrifts',
|
|
53
|
+
' - cloudformation:DescribeStacks',
|
|
54
|
+
' - cloudformation:ListStackResources',
|
|
55
|
+
' - cloudformation:DetectStackResourceDrift',
|
|
56
|
+
" Resource: '*'",
|
|
57
|
+
'',
|
|
58
|
+
'Outputs:',
|
|
59
|
+
' CdkDriftRoleArn:',
|
|
60
|
+
" Description: 'ARN of the CDK drift detection role'",
|
|
61
|
+
' Value: !GetAtt CdkDriftRole.Arn',
|
|
62
|
+
' Export:',
|
|
63
|
+
" Name: !Sub '${AWS::StackName}-CdkDriftRoleArn'",
|
|
64
|
+
'',
|
|
65
|
+
' CdkDriftRoleName:',
|
|
66
|
+
" Description: 'Name of the CDK drift detection role'",
|
|
67
|
+
' Value: !Ref CdkDriftRole',
|
|
68
|
+
' Export:',
|
|
69
|
+
" Name: !Sub '${AWS::StackName}-CdkDriftRoleName'",
|
|
70
|
+
],
|
|
71
|
+
});
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
exports.CdkDriftIamTemplate = CdkDriftIamTemplate;
|
|
75
|
+
_a = JSII_RTTI_SYMBOL_1;
|
|
76
|
+
CdkDriftIamTemplate[_a] = { fqn: "@jjrawlins/cdk-diff-pr-github-action.CdkDriftIamTemplate", version: "0.0.1" };
|
|
77
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiQ2RrRHJpZnRJYW1UZW1wbGF0ZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uL3NyYy9DZGtEcmlmdElhbVRlbXBsYXRlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7O0FBQUEsbUNBQWtDO0FBVWxDLE1BQWEsbUJBQW1CO0lBQzlCLFlBQVksS0FBK0I7UUFDekMsTUFBTSxVQUFVLEdBQUcsS0FBSyxDQUFDLFVBQVUsSUFBSSxzQ0FBc0MsQ0FBQztRQUU5RSxLQUFLLENBQUMsT0FBTyxDQUFDLE9BQU8sQ0FBQyw4QkFBOEIsRUFBRTtZQUNwRCxXQUFXLEVBQ1QsdUlBQXVJO1lBQ3pJLFdBQVcsRUFBRSxJQUFJO1lBQ2pCLElBQUksRUFDRiw2Q0FBNkMsVUFBVSwrRUFBK0U7U0FDekksQ0FBQyxDQUFDO1FBRUgsSUFBSSxpQkFBUSxDQUFDLEtBQUssQ0FBQyxPQUFPLEVBQUUsVUFBVSxFQUFFO1lBQ3RDLEtBQUssRUFBRTtnQkFDTCx3Q0FBd0M7Z0JBQ3hDLDBEQUEwRDtnQkFDMUQsRUFBRTtnQkFDRixhQUFhO2dCQUNiLHNCQUFzQjtnQkFDdEIsa0JBQWtCO2dCQUNsQix5RkFBeUY7Z0JBQ3pGLGlCQUFpQixLQUFLLENBQUMsV0FBVyxHQUFHO2dCQUNyQyxFQUFFO2dCQUNGLFlBQVk7Z0JBQ1osOEZBQThGO2dCQUM5RixpQkFBaUI7Z0JBQ2pCLDBCQUEwQjtnQkFDMUIsaUJBQWlCO2dCQUNqQixtQkFBbUIsR0FBRyxLQUFLLENBQUMsUUFBUSxHQUFHLEdBQUc7Z0JBQzFDLGlDQUFpQztnQkFDakMsK0JBQStCO2dCQUMvQixvQkFBb0I7Z0JBQ3BCLDJCQUEyQjtnQkFDM0Isd0JBQXdCO2dCQUN4QiwyQ0FBMkM7Z0JBQzNDLG9DQUFvQztnQkFDcEMsd0JBQXdCO2dCQUN4Qiw2QkFBNkI7Z0JBQzdCLHdDQUF3QyxHQUFHLEtBQUssQ0FBQyxVQUFVLEdBQUcsR0FBRztnQkFDakUsaUJBQWlCO2dCQUNqQixpREFBaUQ7Z0JBQ2pELDJCQUEyQjtnQkFDM0IsbUNBQW1DO2dCQUNuQyx3QkFBd0I7Z0JBQ3hCLDJEQUEyRDtnQkFDM0QsK0JBQStCO2dCQUMvQix5QkFBeUI7Z0JBQ3pCLHFEQUFxRDtnQkFDckQsc0VBQXNFO2dCQUN0RSxnRUFBZ0U7Z0JBQ2hFLG1EQUFtRDtnQkFDbkQsdURBQXVEO2dCQUN2RCw2REFBNkQ7Z0JBQzdELCtCQUErQjtnQkFDL0IsRUFBRTtnQkFDRixVQUFVO2dCQUNWLG9CQUFvQjtnQkFDcEIsd0RBQXdEO2dCQUN4RCxxQ0FBcUM7Z0JBQ3JDLGFBQWE7Z0JBQ2Isc0RBQXNEO2dCQUN0RCxFQUFFO2dCQUNGLHFCQUFxQjtnQkFDckIseURBQXlEO2dCQUN6RCw4QkFBOEI7Z0JBQzlCLGFBQWE7Z0JBQ2IsdURBQXVEO2FBQ3hEO1NBQ0YsQ0FBQyxDQUFDO0lBQ0wsQ0FBQzs7QUFyRUgsa0RBc0VDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgVGV4dEZpbGUgfSBmcm9tICdwcm9qZW4nO1xuXG5leHBvcnQgaW50ZXJmYWNlIENka0RyaWZ0SWFtVGVtcGxhdGVQcm9wcyB7XG4gIHJlYWRvbmx5IHByb2plY3Q6IGFueTtcbiAgcmVhZG9ubHkgcm9sZU5hbWU6IHN0cmluZztcbiAgcmVhZG9ubHkgb3V0cHV0UGF0aD86IHN0cmluZztcbiAgcmVhZG9ubHkgb2lkY1JvbGVBcm46IHN0cmluZztcbiAgcmVhZG9ubHkgb2lkY1JlZ2lvbjogc3RyaW5nO1xufVxuXG5leHBvcnQgY2xhc3MgQ2RrRHJpZnRJYW1UZW1wbGF0ZSB7XG4gIGNvbnN0cnVjdG9yKHByb3BzOiBDZGtEcmlmdElhbVRlbXBsYXRlUHJvcHMpIHtcbiAgICBjb25zdCBvdXRwdXRQYXRoID0gcHJvcHMub3V0cHV0UGF0aCA/PyAnY2RrLWRyaWZ0LXdvcmtmbG93LWlhbS10ZW1wbGF0ZS55YW1sJztcblxuICAgIHByb3BzLnByb2plY3QuYWRkVGFzaygnZGVwbG95LWNka2RyaWZ0LWlhbS10ZW1wbGF0ZScsIHtcbiAgICAgIGRlc2NyaXB0aW9uOlxuICAgICAgICAnRGVwbG95IHRoZSBDREsgRHJpZnQgRGV0ZWN0aW9uIElBTSB0ZW1wbGF0ZSB2aWEgQ2xvdWRGb3JtYXRpb24gKGFjY2VwdHMgZXh0cmEgQVdTIENMSSBhcmdzLCBlLmcuLCAtLXBhcmFtZXRlci1vdmVycmlkZXMgS2V5PVZhbHVlLi4uKScsXG4gICAgICByZWNlaXZlQXJnczogdHJ1ZSxcbiAgICAgIGV4ZWM6XG4gICAgICAgIGBhd3MgY2xvdWRmb3JtYXRpb24gZGVwbG95IC0tdGVtcGxhdGUtZmlsZSAke291dHB1dFBhdGh9IC0tc3RhY2stbmFtZSBjZGstZHJpZnQtd29ya2Zsb3ctaWFtLXJvbGUgLS1jYXBhYmlsaXRpZXMgQ0FQQUJJTElUWV9OQU1FRF9JQU1gLFxuICAgIH0pO1xuXG4gICAgbmV3IFRleHRGaWxlKHByb3BzLnByb2plY3QsIG91dHB1dFBhdGgsIHtcbiAgICAgIGxpbmVzOiBbXG4gICAgICAgIFwiQVdTVGVtcGxhdGVGb3JtYXRWZXJzaW9uOiAnMjAxMC0wOS0wOSdcIixcbiAgICAgICAgXCJEZXNjcmlwdGlvbjogJ0lBTSByb2xlIGZvciBDREsgRHJpZnQgRGV0ZWN0aW9uIFdvcmtmbG93J1wiLFxuICAgICAgICAnJyxcbiAgICAgICAgJ1BhcmFtZXRlcnM6JyxcbiAgICAgICAgJyAgR2l0SHViT0lEQ1JvbGVBcm46JyxcbiAgICAgICAgJyAgICBUeXBlOiBTdHJpbmcnLFxuICAgICAgICBcIiAgICBEZXNjcmlwdGlvbjogJ0FSTiBvZiB0aGUgZXhpc3RpbmcgR2l0SHViIE9JREMgcm9sZSB0aGF0IGNhbiBhc3N1bWUgdGhpcyBkcmlmdCByb2xlJ1wiLFxuICAgICAgICBgICAgIERlZmF1bHQ6ICcke3Byb3BzLm9pZGNSb2xlQXJufSdgLFxuICAgICAgICAnJyxcbiAgICAgICAgJ1Jlc291cmNlczonLFxuICAgICAgICAnICAjIENsb3VkRm9ybWF0aW9uIERyaWZ0IERldGVjdGlvbiBSb2xlIC0gbWluaW1hbCBwZXJtaXNzaW9ucyBmb3IgZHJpZnQgZGV0ZWN0aW9uIG9wZXJhdGlvbnMnLFxuICAgICAgICAnICBDZGtEcmlmdFJvbGU6JyxcbiAgICAgICAgJyAgICBUeXBlOiBBV1M6OklBTTo6Um9sZScsXG4gICAgICAgICcgICAgUHJvcGVydGllczonLFxuICAgICAgICBcIiAgICAgIFJvbGVOYW1lOiAnXCIgKyBwcm9wcy5yb2xlTmFtZSArIFwiJ1wiLFxuICAgICAgICAnICAgICAgQXNzdW1lUm9sZVBvbGljeURvY3VtZW50OicsXG4gICAgICAgIFwiICAgICAgICBWZXJzaW9uOiAnMjAxMi0xMC0xNydcIixcbiAgICAgICAgJyAgICAgICAgU3RhdGVtZW50OicsXG4gICAgICAgICcgICAgICAgICAgLSBFZmZlY3Q6IEFsbG93JyxcbiAgICAgICAgJyAgICAgICAgICAgIFByaW5jaXBhbDonLFxuICAgICAgICAnICAgICAgICAgICAgICBBV1M6ICFSZWYgR2l0SHViT0lEQ1JvbGVBcm4nLFxuICAgICAgICAnICAgICAgICAgICAgQWN0aW9uOiBzdHM6QXNzdW1lUm9sZScsXG4gICAgICAgICcgICAgICAgICAgICBDb25kaXRpb246JyxcbiAgICAgICAgJyAgICAgICAgICAgICAgU3RyaW5nRXF1YWxzOicsXG4gICAgICAgIFwiICAgICAgICAgICAgICAgIGF3czpSZXF1ZXN0ZWRSZWdpb246ICdcIiArIHByb3BzLm9pZGNSZWdpb24gKyBcIidcIixcbiAgICAgICAgJyAgICAgIFBvbGljaWVzOicsXG4gICAgICAgICcgICAgICAgIC0gUG9saWN5TmFtZTogQ2xvdWRGb3JtYXRpb25EcmlmdEFjY2VzcycsXG4gICAgICAgICcgICAgICAgICAgUG9saWN5RG9jdW1lbnQ6JyxcbiAgICAgICAgXCIgICAgICAgICAgICBWZXJzaW9uOiAnMjAxMi0xMC0xNydcIixcbiAgICAgICAgJyAgICAgICAgICAgIFN0YXRlbWVudDonLFxuICAgICAgICAnICAgICAgICAgICAgICAjIENsb3VkRm9ybWF0aW9uIGRyaWZ0IGRldGVjdGlvbiBvcGVyYXRpb25zJyxcbiAgICAgICAgJyAgICAgICAgICAgICAgLSBFZmZlY3Q6IEFsbG93JyxcbiAgICAgICAgJyAgICAgICAgICAgICAgICBBY3Rpb246JyxcbiAgICAgICAgJyAgICAgICAgICAgICAgICAgIC0gY2xvdWRmb3JtYXRpb246RGV0ZWN0U3RhY2tEcmlmdCcsXG4gICAgICAgICcgICAgICAgICAgICAgICAgICAtIGNsb3VkZm9ybWF0aW9uOkRlc2NyaWJlU3RhY2tEcmlmdERldGVjdGlvblN0YXR1cycsXG4gICAgICAgICcgICAgICAgICAgICAgICAgICAtIGNsb3VkZm9ybWF0aW9uOkRlc2NyaWJlU3RhY2tSZXNvdXJjZURyaWZ0cycsXG4gICAgICAgICcgICAgICAgICAgICAgICAgICAtIGNsb3VkZm9ybWF0aW9uOkRlc2NyaWJlU3RhY2tzJyxcbiAgICAgICAgJyAgICAgICAgICAgICAgICAgIC0gY2xvdWRmb3JtYXRpb246TGlzdFN0YWNrUmVzb3VyY2VzJyxcbiAgICAgICAgJyAgICAgICAgICAgICAgICAgIC0gY2xvdWRmb3JtYXRpb246RGV0ZWN0U3RhY2tSZXNvdXJjZURyaWZ0JyxcbiAgICAgICAgXCIgICAgICAgICAgICAgICAgUmVzb3VyY2U6ICcqJ1wiLFxuICAgICAgICAnJyxcbiAgICAgICAgJ091dHB1dHM6JyxcbiAgICAgICAgJyAgQ2RrRHJpZnRSb2xlQXJuOicsXG4gICAgICAgIFwiICAgIERlc2NyaXB0aW9uOiAnQVJOIG9mIHRoZSBDREsgZHJpZnQgZGV0ZWN0aW9uIHJvbGUnXCIsXG4gICAgICAgICcgICAgVmFsdWU6ICFHZXRBdHQgQ2RrRHJpZnRSb2xlLkFybicsXG4gICAgICAgICcgICAgRXhwb3J0OicsXG4gICAgICAgIFwiICAgICAgTmFtZTogIVN1YiAnJHtBV1M6OlN0YWNrTmFtZX0tQ2RrRHJpZnRSb2xlQXJuJ1wiLFxuICAgICAgICAnJyxcbiAgICAgICAgJyAgQ2RrRHJpZnRSb2xlTmFtZTonLFxuICAgICAgICBcIiAgICBEZXNjcmlwdGlvbjogJ05hbWUgb2YgdGhlIENESyBkcmlmdCBkZXRlY3Rpb24gcm9sZSdcIixcbiAgICAgICAgJyAgICBWYWx1ZTogIVJlZiBDZGtEcmlmdFJvbGUnLFxuICAgICAgICAnICAgIEV4cG9ydDonLFxuICAgICAgICBcIiAgICAgIE5hbWU6ICFTdWIgJyR7QVdTOjpTdGFja05hbWV9LUNka0RyaWZ0Um9sZU5hbWUnXCIsXG4gICAgICBdLFxuICAgIH0pO1xuICB9XG59XG4iXX0=
|
|
@@ -41,8 +41,8 @@ class CdkChangesetScript {
|
|
|
41
41
|
' const props: string[] = [];',
|
|
42
42
|
' for (const d of details) {',
|
|
43
43
|
" if (d?.Target?.Attribute !== 'Properties') continue;",
|
|
44
|
-
" const
|
|
45
|
-
' props.push(`- ${
|
|
44
|
+
" const workflowName = d?.Target?.Name ?? '-';",
|
|
45
|
+
' props.push(`- ${workflowName}`);',
|
|
46
46
|
' }',
|
|
47
47
|
" return props.join('<br>');",
|
|
48
48
|
'}',
|
|
@@ -253,4 +253,4 @@ class CdkChangesetScript {
|
|
|
253
253
|
}
|
|
254
254
|
}
|
|
255
255
|
exports.CdkChangesetScript = CdkChangesetScript;
|
|
256
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cdk-changeset-script.js","sourceRoot":"","sources":["../../src/bin/cdk-changeset-script.ts"],"names":[],"mappings":"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","sourcesContent":["import { TextFile } from 'projen';\nimport { AwsCdkTypeScriptApp } from 'projen/lib/awscdk';\n\ninterface CdkChangesetScriptProps {\n  project: AwsCdkTypeScriptApp;\n  outputPath?: string;\n}\n\nexport class CdkChangesetScript {\n  constructor(props: CdkChangesetScriptProps) {\n    const outputPath = props.outputPath ?? 'projenrc/describe-cfn-changeset.ts';\n\n    new TextFile(props.project, outputPath, {\n      lines: [\n        \"import { appendFile } from 'fs/promises';\",\n        \"import { CloudFormationClient, DescribeChangeSetCommand } from '@aws-sdk/client-cloudformation';\",\n        '',\n        \"type Change = NonNullable<Awaited<ReturnType<CloudFormationClient['send']>> extends any ? any : never>;\",\n        '',\n        '/**',\n        ' * Small sleep helper.',\n        ' */',\n        'const sleep = (ms: number) => new Promise((res) => setTimeout(res, ms));',\n        '',\n        '/**',\n        ' * Build the HTML color chip for an action.',\n        ' * Modify=blue, Add=green, Remove=red (match pretty_format.py).',\n        ' */',\n        'function actionChip(action?: string): string {',\n        '  // Use emoji instead of external images for reliability',\n        '  const emojiMap: Record<string, string> = {',\n        \"    Modify: '🔵',\",\n        \"    Add: '🟢',\",\n        \"    Remove: '🔴',\",\n        '  };',\n        \"  const mark = action ? (emojiMap[action] ?? '⚪') : '⚪';\",\n        \"  return `${mark} ${action ?? '-'}`;\",\n        '}',\n        '',\n        '/**',\n        \" * Extract changed property names only where Target.Attribute == 'Properties'\",\n        ' * joined by <br>, prefixed with \"- \".',\n        ' */',\n        'function changedPropertiesHTML(change: any): string {',\n        '  const details = change?.ResourceChange?.Details ?? [];',\n        '  const props: string[] = [];',\n        '  for (const d of details) {',\n        \"    if (d?.Target?.Attribute !== 'Properties') continue;\",\n        \"    const name = d?.Target?.Name ?? '-';\",\n        '    props.push(`- ${name}`);',\n        '  }',\n        \"  return props.join('<br>');\",\n        '}',\n        '',\n        '/**',\n        ' * Determine if a change should be ignored based on logical IDs and/or resource types.',\n        \" * - IGNORE_LOGICAL_IDS: comma-separated list of logical IDs to ignore (default includes 'CDKMetadata')\",\n        \" * - IGNORE_RESOURCE_TYPES: comma-separated list of resource types to ignore (e.g., 'AWS::CDK::Metadata')\",\n        ' */',\n        'function shouldIgnoreChange(change: any, ignoreIds: Set<string>, ignoreTypes: Set<string>): boolean {',\n        '  const rc = change?.ResourceChange ?? {};',\n        '  const logicalId = rc?.LogicalResourceId as string | undefined;',\n        '  const resourceType = rc?.ResourceType as string | undefined;',\n        '  if (logicalId && ignoreIds.has(logicalId)) return true;',\n        '  if (resourceType && ignoreTypes.has(resourceType)) return true;',\n        '  return false;',\n        '}',\n        '',\n        '/**',\n        ' * Generate the HTML body similar to pretty_format.py',\n        ' */',\n        'function buildHtml(stackName: string, changes: any[]): string {',\n        '  let body = `<h1>Change set</h1><h2>Stack Name: ${stackName}</h2><br>`;',\n        '  if ((changes?.length ?? 0) > 0) {',\n        \"    body += '<table><tr><th>Action&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</th><th>ID</th><th>Type</th><th>Replacement</th><th>Changed Properties</th></tr>';\",\n        '    for (const c of changes) {',\n        '      const rc = c?.ResourceChange ?? {};',\n        '      const action = actionChip(rc?.Action);',\n        \"      const logicalId = rc?.LogicalResourceId ?? '-';\",\n        \"      const type = rc?.ResourceType ?? '-';\",\n        \"      const replacement = rc?.Replacement ?? '-';\",\n        '      const details = changedPropertiesHTML(c);',\n        '',\n        \"      body += '<tr>';\",\n        '      body += `<td>${action}</td>`;',\n        '      body += `<td>${logicalId}</td>`;',\n        '      body += `<td>${type}</td>`;',\n        '      body += `<td>${replacement}</td>`;',\n        '      body += `<td>${details}</td>`;',\n        \"      body += '</tr>';\",\n        '    }',\n        \"    body += '</table>';\",\n        '  } else {',\n        \"    body += 'no change.';\",\n        '  }',\n        '  return body;',\n        '}',\n        '',\n        '/**',\n        ' * Poll DescribeChangeSet until a terminal status, then paginate to retrieve all Changes.',\n        ' */',\n        'async function getTerminalChangeSet(',\n        '  client: CloudFormationClient,',\n        '  stackName: string,',\n        '  changeSetName: string,',\n        '  maxAttempts = 60,',\n        '  delayMs = 3000,',\n        '): Promise<{ status?: string; statusReason?: string; changes: any[] }> {',\n        '  for (let attempt = 1; attempt <= maxAttempts; attempt++) {',\n        '    const resp = await client.send(new DescribeChangeSetCommand({',\n        '      StackName: stackName,',\n        '      ChangeSetName: changeSetName,',\n        '    }));',\n        '',\n        '    const status = resp.Status;',\n        '    const statusReason = resp.StatusReason;',\n        '',\n        \"    if (status === 'CREATE_COMPLETE' || status === 'FAILED') {\",\n        '      // Gather all pages',\n        '      const changes: any[] = [];',\n        '      if (resp.Changes) changes.push(...resp.Changes);',\n        '      let next = resp.NextToken;',\n        '      while (next) {',\n        '        const page = await client.send(new DescribeChangeSetCommand({',\n        '          StackName: stackName,',\n        '          ChangeSetName: changeSetName,',\n        '          NextToken: next,',\n        '        }));',\n        '        if (page.Changes) changes.push(...page.Changes);',\n        '        next = page.NextToken;',\n        '      }',\n        '      return { status, statusReason, changes };',\n        '    }',\n        '',\n        '    // Not terminal yet; wait and retry',\n        '    await sleep(delayMs);',\n        '  }',\n        '',\n        \"  throw new Error('Timed out waiting for change set to reach a terminal status.');\",\n        '}',\n        '',\n        'async function postGithubComment(url: string, token: string, body: string): Promise<void> {',\n        '  const res = await fetch(url, {',\n        \"    method: 'POST',\",\n        '    headers: {',\n        \"      'Authorization': `token ${token}`,\",\n        \"      'Content-Type': 'application/json',\",\n        \"      'Accept': 'application/vnd.github+json',\",\n        '    },',\n        '    body: JSON.stringify({ body }),',\n        '  });',\n        '',\n        '  if (!res.ok) {',\n        \"    const text = await res.text().catch(() => '');\",\n        '    throw new Error(`Failed to post GitHub comment: ${res.status} ${res.statusText} ${text}`);',\n        '  }',\n        '}',\n        '',\n        'async function appendStepSummary(summaryPath: string, content: string): Promise<void> {',\n        \"  await appendFile(summaryPath, `${content}\\\\n`, { encoding: 'utf8' });\",\n        '}',\n        '',\n        'async function main() {',\n        '  const {',\n        '    STACK_NAME,',\n        '    CHANGE_SET_NAME,',\n        '    AWS_REGION,',\n        '    GITHUB_TOKEN,',\n        '    GITHUB_COMMENT_URL,',\n        '    GITHUB_STEP_SUMMARY,',\n        '    IGNORE_LOGICAL_IDS,',\n        '    IGNORE_RESOURCE_TYPES,',\n        '  } = process.env;',\n        '',\n        '  if (!STACK_NAME) {',\n        \"    throw new Error('STACK_NAME is required');\",\n        '  }',\n        '  const region = AWS_REGION || process.env.AWS_DEFAULT_REGION;',\n        '  if (!region) {',\n        \"    throw new Error('AWS_REGION is required');\",\n        '  }',\n        '',\n        '  const changeSetName = CHANGE_SET_NAME || STACK_NAME;',\n        '',\n        '  const client = new CloudFormationClient({ region });',\n        '',\n        '  let status: string | undefined;',\n        '  let statusReason: string | undefined;',\n        '  let changes: any[] = [];',\n        '',\n        '  try {',\n        '    const result = await getTerminalChangeSet(client, STACK_NAME, changeSetName);',\n        '    status = result.status;',\n        '    statusReason = result.statusReason;',\n        '    changes = result.changes ?? [];',\n        '  } catch (err: any) {',\n        '    // If DescribeChangeSet fails entirely, surface the error',\n        \"    console.error('Error describing change set:', err?.message || err);\",\n        '    process.exitCode = 1;',\n        '    return;',\n        '  }',\n        '',\n        \"  // Apply ignores from env vars (IDs and types). Default ignore IDs include 'CDKMetadata'.\",\n        '  const ignoreIdSet = new Set(',\n        \"    (IGNORE_LOGICAL_IDS ?? 'CDKMetadata')\",\n        \"      .split(',')\",\n        '      .map(s => s.trim())',\n        '      .filter(Boolean),',\n        '  );',\n        '  const ignoreTypeSet = new Set(',\n        \"    (IGNORE_RESOURCE_TYPES ?? '')\",\n        \"      .split(',')\",\n        '      .map(s => s.trim())',\n        '      .filter(Boolean),',\n        '  );',\n        '  const filteredChanges = changes.filter(c => !shouldIgnoreChange(c, ignoreIdSet, ignoreTypeSet));',\n        '',\n        '  // Build HTML exactly like pretty_format.py logic (table when there are changes; \"no change.\" otherwise).',\n        '  const html = buildHtml(STACK_NAME, filteredChanges);',\n        '',\n        '  // Print to stdout',\n        '  // This allows capturing output or redirecting to a file if needed.',\n        '  console.log(html);',\n        '',\n        '  // Optionally append to GitHub Step Summary',\n        '  if (GITHUB_STEP_SUMMARY) {',\n        '    try {',\n        '      await appendStepSummary(GITHUB_STEP_SUMMARY, html);',\n        '      console.error(`Appended HTML to GITHUB_STEP_SUMMARY: ${GITHUB_STEP_SUMMARY}`);',\n        '    } catch (e: any) {',\n        \"      console.error('Failed to append to GITHUB_STEP_SUMMARY:', e?.message || e);\",\n        '    }',\n        '  }',\n        '',\n        '  // Optionally post a PR comment',\n        '  if (GITHUB_TOKEN && GITHUB_COMMENT_URL) {',\n        '    try {',\n        '      await postGithubComment(GITHUB_COMMENT_URL, GITHUB_TOKEN, html);',\n        \"      console.error('Posted HTML as a GitHub PR comment.');\",\n        '    } catch (e: any) {',\n        \"      console.error('Failed to post GitHub PR comment:', e?.message || e);\",\n        '      // Do not fail the whole script just for comment posting',\n        '    }',\n        '  }',\n        '',\n        \"  // Note: When status is FAILED due to \\\"didn't contain changes\\\", the HTML naturally says \\\"no change.\\\"\",\n        \"  if (status === 'FAILED' && statusReason) {\",\n        '    console.error(`Change set status: FAILED. Reason: ${statusReason}`);',\n        '  }',\n        '}',\n        '',\n        'main().catch((err) => {',\n        '  console.error(err);',\n        '  process.exitCode = 1;',\n        '});',\n      ],\n    });\n  }\n}\n"]}
|
|
256
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cdk-changeset-script.js","sourceRoot":"","sources":["../../src/bin/cdk-changeset-script.ts"],"names":[],"mappings":"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","sourcesContent":["import { TextFile } from 'projen';\nimport { AwsCdkTypeScriptApp } from 'projen/lib/awscdk';\n\ninterface CdkChangesetScriptProps {\n  project: AwsCdkTypeScriptApp;\n  outputPath?: string;\n}\n\nexport class CdkChangesetScript {\n  constructor(props: CdkChangesetScriptProps) {\n    const outputPath = props.outputPath ?? 'projenrc/describe-cfn-changeset.ts';\n\n    new TextFile(props.project, outputPath, {\n      lines: [\n        \"import { appendFile } from 'fs/promises';\",\n        \"import { CloudFormationClient, DescribeChangeSetCommand } from '@aws-sdk/client-cloudformation';\",\n        '',\n        \"type Change = NonNullable<Awaited<ReturnType<CloudFormationClient['send']>> extends any ? any : never>;\",\n        '',\n        '/**',\n        ' * Small sleep helper.',\n        ' */',\n        'const sleep = (ms: number) => new Promise((res) => setTimeout(res, ms));',\n        '',\n        '/**',\n        ' * Build the HTML color chip for an action.',\n        ' * Modify=blue, Add=green, Remove=red (match pretty_format.py).',\n        ' */',\n        'function actionChip(action?: string): string {',\n        '  // Use emoji instead of external images for reliability',\n        '  const emojiMap: Record<string, string> = {',\n        \"    Modify: '🔵',\",\n        \"    Add: '🟢',\",\n        \"    Remove: '🔴',\",\n        '  };',\n        \"  const mark = action ? (emojiMap[action] ?? '⚪') : '⚪';\",\n        \"  return `${mark} ${action ?? '-'}`;\",\n        '}',\n        '',\n        '/**',\n        \" * Extract changed property names only where Target.Attribute == 'Properties'\",\n        ' * joined by <br>, prefixed with \"- \".',\n        ' */',\n        'function changedPropertiesHTML(change: any): string {',\n        '  const details = change?.ResourceChange?.Details ?? [];',\n        '  const props: string[] = [];',\n        '  for (const d of details) {',\n        \"    if (d?.Target?.Attribute !== 'Properties') continue;\",\n        \"    const workflowName = d?.Target?.Name ?? '-';\",\n        '    props.push(`- ${workflowName}`);',\n        '  }',\n        \"  return props.join('<br>');\",\n        '}',\n        '',\n        '/**',\n        ' * Determine if a change should be ignored based on logical IDs and/or resource types.',\n        \" * - IGNORE_LOGICAL_IDS: comma-separated list of logical IDs to ignore (default includes 'CDKMetadata')\",\n        \" * - IGNORE_RESOURCE_TYPES: comma-separated list of resource types to ignore (e.g., 'AWS::CDK::Metadata')\",\n        ' */',\n        'function shouldIgnoreChange(change: any, ignoreIds: Set<string>, ignoreTypes: Set<string>): boolean {',\n        '  const rc = change?.ResourceChange ?? {};',\n        '  const logicalId = rc?.LogicalResourceId as string | undefined;',\n        '  const resourceType = rc?.ResourceType as string | undefined;',\n        '  if (logicalId && ignoreIds.has(logicalId)) return true;',\n        '  if (resourceType && ignoreTypes.has(resourceType)) return true;',\n        '  return false;',\n        '}',\n        '',\n        '/**',\n        ' * Generate the HTML body similar to pretty_format.py',\n        ' */',\n        'function buildHtml(stackName: string, changes: any[]): string {',\n        '  let body = `<h1>Change set</h1><h2>Stack Name: ${stackName}</h2><br>`;',\n        '  if ((changes?.length ?? 0) > 0) {',\n        \"    body += '<table><tr><th>Action&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</th><th>ID</th><th>Type</th><th>Replacement</th><th>Changed Properties</th></tr>';\",\n        '    for (const c of changes) {',\n        '      const rc = c?.ResourceChange ?? {};',\n        '      const action = actionChip(rc?.Action);',\n        \"      const logicalId = rc?.LogicalResourceId ?? '-';\",\n        \"      const type = rc?.ResourceType ?? '-';\",\n        \"      const replacement = rc?.Replacement ?? '-';\",\n        '      const details = changedPropertiesHTML(c);',\n        '',\n        \"      body += '<tr>';\",\n        '      body += `<td>${action}</td>`;',\n        '      body += `<td>${logicalId}</td>`;',\n        '      body += `<td>${type}</td>`;',\n        '      body += `<td>${replacement}</td>`;',\n        '      body += `<td>${details}</td>`;',\n        \"      body += '</tr>';\",\n        '    }',\n        \"    body += '</table>';\",\n        '  } else {',\n        \"    body += 'no change.';\",\n        '  }',\n        '  return body;',\n        '}',\n        '',\n        '/**',\n        ' * Poll DescribeChangeSet until a terminal status, then paginate to retrieve all Changes.',\n        ' */',\n        'async function getTerminalChangeSet(',\n        '  client: CloudFormationClient,',\n        '  stackName: string,',\n        '  changeSetName: string,',\n        '  maxAttempts = 60,',\n        '  delayMs = 3000,',\n        '): Promise<{ status?: string; statusReason?: string; changes: any[] }> {',\n        '  for (let attempt = 1; attempt <= maxAttempts; attempt++) {',\n        '    const resp = await client.send(new DescribeChangeSetCommand({',\n        '      StackName: stackName,',\n        '      ChangeSetName: changeSetName,',\n        '    }));',\n        '',\n        '    const status = resp.Status;',\n        '    const statusReason = resp.StatusReason;',\n        '',\n        \"    if (status === 'CREATE_COMPLETE' || status === 'FAILED') {\",\n        '      // Gather all pages',\n        '      const changes: any[] = [];',\n        '      if (resp.Changes) changes.push(...resp.Changes);',\n        '      let next = resp.NextToken;',\n        '      while (next) {',\n        '        const page = await client.send(new DescribeChangeSetCommand({',\n        '          StackName: stackName,',\n        '          ChangeSetName: changeSetName,',\n        '          NextToken: next,',\n        '        }));',\n        '        if (page.Changes) changes.push(...page.Changes);',\n        '        next = page.NextToken;',\n        '      }',\n        '      return { status, statusReason, changes };',\n        '    }',\n        '',\n        '    // Not terminal yet; wait and retry',\n        '    await sleep(delayMs);',\n        '  }',\n        '',\n        \"  throw new Error('Timed out waiting for change set to reach a terminal status.');\",\n        '}',\n        '',\n        'async function postGithubComment(url: string, token: string, body: string): Promise<void> {',\n        '  const res = await fetch(url, {',\n        \"    method: 'POST',\",\n        '    headers: {',\n        \"      'Authorization': `token ${token}`,\",\n        \"      'Content-Type': 'application/json',\",\n        \"      'Accept': 'application/vnd.github+json',\",\n        '    },',\n        '    body: JSON.stringify({ body }),',\n        '  });',\n        '',\n        '  if (!res.ok) {',\n        \"    const text = await res.text().catch(() => '');\",\n        '    throw new Error(`Failed to post GitHub comment: ${res.status} ${res.statusText} ${text}`);',\n        '  }',\n        '}',\n        '',\n        'async function appendStepSummary(summaryPath: string, content: string): Promise<void> {',\n        \"  await appendFile(summaryPath, `${content}\\\\n`, { encoding: 'utf8' });\",\n        '}',\n        '',\n        'async function main() {',\n        '  const {',\n        '    STACK_NAME,',\n        '    CHANGE_SET_NAME,',\n        '    AWS_REGION,',\n        '    GITHUB_TOKEN,',\n        '    GITHUB_COMMENT_URL,',\n        '    GITHUB_STEP_SUMMARY,',\n        '    IGNORE_LOGICAL_IDS,',\n        '    IGNORE_RESOURCE_TYPES,',\n        '  } = process.env;',\n        '',\n        '  if (!STACK_NAME) {',\n        \"    throw new Error('STACK_NAME is required');\",\n        '  }',\n        '  const region = AWS_REGION || process.env.AWS_DEFAULT_REGION;',\n        '  if (!region) {',\n        \"    throw new Error('AWS_REGION is required');\",\n        '  }',\n        '',\n        '  const changeSetName = CHANGE_SET_NAME || STACK_NAME;',\n        '',\n        '  const client = new CloudFormationClient({ region });',\n        '',\n        '  let status: string | undefined;',\n        '  let statusReason: string | undefined;',\n        '  let changes: any[] = [];',\n        '',\n        '  try {',\n        '    const result = await getTerminalChangeSet(client, STACK_NAME, changeSetName);',\n        '    status = result.status;',\n        '    statusReason = result.statusReason;',\n        '    changes = result.changes ?? [];',\n        '  } catch (err: any) {',\n        '    // If DescribeChangeSet fails entirely, surface the error',\n        \"    console.error('Error describing change set:', err?.message || err);\",\n        '    process.exitCode = 1;',\n        '    return;',\n        '  }',\n        '',\n        \"  // Apply ignores from env vars (IDs and types). Default ignore IDs include 'CDKMetadata'.\",\n        '  const ignoreIdSet = new Set(',\n        \"    (IGNORE_LOGICAL_IDS ?? 'CDKMetadata')\",\n        \"      .split(',')\",\n        '      .map(s => s.trim())',\n        '      .filter(Boolean),',\n        '  );',\n        '  const ignoreTypeSet = new Set(',\n        \"    (IGNORE_RESOURCE_TYPES ?? '')\",\n        \"      .split(',')\",\n        '      .map(s => s.trim())',\n        '      .filter(Boolean),',\n        '  );',\n        '  const filteredChanges = changes.filter(c => !shouldIgnoreChange(c, ignoreIdSet, ignoreTypeSet));',\n        '',\n        '  // Build HTML exactly like pretty_format.py logic (table when there are changes; \"no change.\" otherwise).',\n        '  const html = buildHtml(STACK_NAME, filteredChanges);',\n        '',\n        '  // Print to stdout',\n        '  // This allows capturing output or redirecting to a file if needed.',\n        '  console.log(html);',\n        '',\n        '  // Optionally append to GitHub Step Summary',\n        '  if (GITHUB_STEP_SUMMARY) {',\n        '    try {',\n        '      await appendStepSummary(GITHUB_STEP_SUMMARY, html);',\n        '      console.error(`Appended HTML to GITHUB_STEP_SUMMARY: ${GITHUB_STEP_SUMMARY}`);',\n        '    } catch (e: any) {',\n        \"      console.error('Failed to append to GITHUB_STEP_SUMMARY:', e?.message || e);\",\n        '    }',\n        '  }',\n        '',\n        '  // Optionally post a PR comment',\n        '  if (GITHUB_TOKEN && GITHUB_COMMENT_URL) {',\n        '    try {',\n        '      await postGithubComment(GITHUB_COMMENT_URL, GITHUB_TOKEN, html);',\n        \"      console.error('Posted HTML as a GitHub PR comment.');\",\n        '    } catch (e: any) {',\n        \"      console.error('Failed to post GitHub PR comment:', e?.message || e);\",\n        '      // Do not fail the whole script just for comment posting',\n        '    }',\n        '  }',\n        '',\n        \"  // Note: When status is FAILED due to \\\"didn't contain changes\\\", the HTML naturally says \\\"no change.\\\"\",\n        \"  if (status === 'FAILED' && statusReason) {\",\n        '    console.error(`Change set status: FAILED. Reason: ${statusReason}`);',\n        '  }',\n        '}',\n        '',\n        'main().catch((err) => {',\n        '  console.error(err);',\n        '  process.exitCode = 1;',\n        '});',\n      ],\n    });\n  }\n}\n"]}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
interface CdkDriftDetectionScriptProps {
|
|
2
|
+
project: any;
|
|
3
|
+
outputPath?: string;
|
|
4
|
+
}
|
|
5
|
+
/**
|
|
6
|
+
* Projen helper to emit the drift detection script into the repository so
|
|
7
|
+
* GitHub workflows can execute it from a stable location.
|
|
8
|
+
*
|
|
9
|
+
* This mirrors the pattern used by CdkChangesetScript, but writes the
|
|
10
|
+
* drift script (detect-drift.ts) under the workflows scripts directory by default.
|
|
11
|
+
*/
|
|
12
|
+
export declare class CdkDriftDetectionScript {
|
|
13
|
+
constructor(props: CdkDriftDetectionScriptProps);
|
|
14
|
+
}
|
|
15
|
+
export {};
|