@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.
Files changed (43) hide show
  1. package/.jsii +475 -33
  2. package/.mergify.yml +102 -0
  3. package/API.md +351 -11
  4. package/README.md +223 -39
  5. package/lib/CdkDiffIamTemplate.d.ts +3 -1
  6. package/lib/CdkDiffIamTemplate.js +10 -5
  7. package/lib/CdkDiffStackWorkflow.d.ts +2 -2
  8. package/lib/CdkDiffStackWorkflow.js +19 -20
  9. package/lib/CdkDriftDetectionWorkflow.d.ts +32 -0
  10. package/lib/CdkDriftDetectionWorkflow.js +281 -0
  11. package/lib/CdkDriftIamTemplate.d.ts +10 -0
  12. package/lib/CdkDriftIamTemplate.js +77 -0
  13. package/lib/bin/cdk-changeset-script.js +3 -3
  14. package/lib/bin/cdk-drift-detection-script.d.ts +15 -0
  15. package/lib/bin/cdk-drift-detection-script.js +196 -0
  16. package/lib/bin/detect-drift.js +162 -0
  17. package/lib/index.d.ts +2 -0
  18. package/lib/index.js +3 -1
  19. package/package.json +7 -2
  20. package/sonar-project.properties +17 -0
  21. package/.junie/guidelines.md +0 -62
  22. package/.yalc/@jjrawlins/cdk-diff-pr-github-action/.jsii +0 -3917
  23. package/.yalc/@jjrawlins/cdk-diff-pr-github-action/.junie/guidelines.md +0 -62
  24. package/.yalc/@jjrawlins/cdk-diff-pr-github-action/.tool-versions +0 -3
  25. package/.yalc/@jjrawlins/cdk-diff-pr-github-action/API.md +0 -276
  26. package/.yalc/@jjrawlins/cdk-diff-pr-github-action/LICENSE +0 -202
  27. package/.yalc/@jjrawlins/cdk-diff-pr-github-action/README.md +0 -146
  28. package/.yalc/@jjrawlins/cdk-diff-pr-github-action/lib/CdkDiffIamTemplate.d.ts +0 -8
  29. package/.yalc/@jjrawlins/cdk-diff-pr-github-action/lib/CdkDiffIamTemplate.js +0 -96
  30. package/.yalc/@jjrawlins/cdk-diff-pr-github-action/lib/CdkDiffStackWorkflow.d.ts +0 -22
  31. package/.yalc/@jjrawlins/cdk-diff-pr-github-action/lib/CdkDiffStackWorkflow.js +0 -144
  32. package/.yalc/@jjrawlins/cdk-diff-pr-github-action/lib/bin/cdk-changeset-script.d.ts +0 -9
  33. package/.yalc/@jjrawlins/cdk-diff-pr-github-action/lib/bin/cdk-changeset-script.js +0 -256
  34. package/.yalc/@jjrawlins/cdk-diff-pr-github-action/lib/bin/describe-cfn-changeset.js +0 -204
  35. package/.yalc/@jjrawlins/cdk-diff-pr-github-action/lib/index.d.ts +0 -2
  36. package/.yalc/@jjrawlins/cdk-diff-pr-github-action/lib/index.js +0 -19
  37. package/.yalc/@jjrawlins/cdk-diff-pr-github-action/package.json +0 -137
  38. package/.yalc/@jjrawlins/cdk-diff-pr-github-action/yalc.lock +0 -10
  39. package/.yalc/@jjrawlins/cdk-diff-pr-github-action/yalc.sig +0 -1
  40. package/lib/bin/describe-cfn-changeset.d.ts +0 -1
  41. package/lib/bin/describe-cfn-changeset.js +0 -204
  42. package/yalc.lock +0 -10
  43. /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,196 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.CdkDriftDetectionScript = void 0;
4
+ const projen_1 = require("projen");
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
+ class CdkDriftDetectionScript {
13
+ constructor(props) {
14
+ const outputPath = props.outputPath ?? '.github/workflows/scripts/detect-drift.ts';
15
+ new projen_1.TextFile(props.project, outputPath, {
16
+ lines: [
17
+ "import {\n CloudFormationClient,\n DescribeStackDriftDetectionStatusCommand,\n DetectStackDriftCommand,\n DescribeStackResourceDriftsCommand,\n type DescribeStackResourceDriftsCommandOutput,\n type StackResourceDriftStatus,\n} from '@aws-sdk/client-cloudformation';",
18
+ '',
19
+ 'async function sleep(ms: number) {',
20
+ ' return new Promise((r) => setTimeout(r, ms));',
21
+ '}',
22
+ '',
23
+ 'async function main() {',
24
+ ' const stackName = process.env.STACK_NAME;',
25
+ ' if (!stackName) {',
26
+ " console.error('STACK_NAME env var is required');",
27
+ ' process.exit(1);',
28
+ ' }',
29
+ '',
30
+ ' // Region and credentials pulled from environment set by actions/configure-aws-credentials',
31
+ ' const client = new CloudFormationClient({});',
32
+ '',
33
+ ' const detect = await client.send(new DetectStackDriftCommand({ StackName: stackName }));',
34
+ ' if (!detect.StackDriftDetectionId) {',
35
+ " console.error('Failed to start drift detection');",
36
+ ' process.exit(1);',
37
+ ' }',
38
+ '',
39
+ ' const id = detect.StackDriftDetectionId;',
40
+ ' console.log(`Drift detection started: ${id}`);',
41
+ '',
42
+ " let detectionStatus = 'DETECTION_IN_PROGRESS';",
43
+ ' let stackDriftStatus: string | undefined;',
44
+ '',
45
+ " while (detectionStatus === 'DETECTION_IN_PROGRESS') {",
46
+ ' await sleep(5000);',
47
+ ' const res = await client.send(',
48
+ ' new DescribeStackDriftDetectionStatusCommand({ StackDriftDetectionId: id }),',
49
+ ' );',
50
+ " detectionStatus = res.DetectionStatus ?? 'UNKNOWN';",
51
+ ' stackDriftStatus = res.StackDriftStatus;',
52
+ ' console.log(`Detection status: ${detectionStatus}`);',
53
+ ' }',
54
+ '',
55
+ ' // Helper to build an HTML report of drifted resources',
56
+ ' const buildHtml = (stack: string, drifts: any[]): string => {',
57
+ ' let body = `<h1>Drift report</h1><h2>Stack Name: ${stack}</h2><br>`;',
58
+ ' if (drifts.length === 0) {',
59
+ " body += 'no drift.';",
60
+ ' return body;',
61
+ ' }',
62
+ " body += '<table>' +",
63
+ " '<tr><th>Status</th><th>ID</th><th>Type</th><th>Differences</th></tr>';",
64
+ ' for (const d of drifts) {',
65
+ " const status = d.StackResourceDriftStatus ?? '-';",
66
+ " const logicalId = d.LogicalResourceId ?? '-';",
67
+ " const type = d.ResourceType ?? '-';",
68
+ ' const diffs = (d.PropertyDifferences ?? []).map((pd: any) => {',
69
+ " const p = pd.PropertyPath ?? '-';",
70
+ " const t = pd.DifferenceType ?? '-';",
71
+ ' return `- ${t}: ${p}`;',
72
+ " }).join('<br>');",
73
+ " const statusEmoji = status === 'MODIFIED' ? '🟠' : status === 'DELETED' ? '🔴' : status === 'NOT_CHECKED' ? '⚪' : '🟢';",
74
+ " body += '<tr>' +",
75
+ ' `<td>${statusEmoji} ${status}</td>` +',
76
+ ' `<td>${logicalId}</td>` +',
77
+ ' `<td>${type}</td>` +',
78
+ ' `<td>${diffs}</td>` +',
79
+ " '</tr>';",
80
+ ' }',
81
+ " body += '</table>';",
82
+ ' return body;',
83
+ ' };',
84
+ '',
85
+ ' async function listDriftedResources(): Promise<any[]> {',
86
+ ' const results: any[] = [];',
87
+ ' // Only include resources that are not IN_SYNC',
88
+ " const filters: StackResourceDriftStatus[] = ['MODIFIED', 'DELETED', 'NOT_CHECKED'];",
89
+ ' let nextToken: string | undefined = undefined;',
90
+ ' do {',
91
+ ' const resp: DescribeStackResourceDriftsCommandOutput = await client.send(new DescribeStackResourceDriftsCommand({',
92
+ ' StackName: stackName,',
93
+ ' NextToken: nextToken,',
94
+ ' StackResourceDriftStatusFilters: filters,',
95
+ ' }));',
96
+ ' if (resp.StackResourceDrifts) results.push(...resp.StackResourceDrifts);',
97
+ ' nextToken = resp.NextToken;',
98
+ ' } while (nextToken);',
99
+ ' return results;',
100
+ ' }',
101
+ '',
102
+ ' async function postGithubComment(url: string, token: string, body: string): Promise<void> {',
103
+ ' const res = await fetch(url, {',
104
+ " method: 'POST',",
105
+ ' headers: {',
106
+ " 'Authorization': `token ${token}`,",
107
+ " 'Content-Type': 'application/json',",
108
+ " 'Accept': 'application/vnd.github+json',",
109
+ ' },',
110
+ ' body: JSON.stringify({ body }),',
111
+ ' });',
112
+ ' if (!res.ok) {',
113
+ ' const text = await res.text().catch(() => \'\');',
114
+ ' console.error(`Failed to post GitHub comment: ${res.status} ${res.statusText} ${text}`);',
115
+ ' }',
116
+ ' }',
117
+ '',
118
+ ' // When there is drift, collect details and post a PR comment + step summary',
119
+ ' const outputFile = process.env.DRIFT_DETECTION_OUTPUT;',
120
+ " if (stackDriftStatus !== 'IN_SYNC') {",
121
+ ' console.error(`Drift detected (status: ${stackDriftStatus})`);',
122
+ ' const drifts = await listDriftedResources();',
123
+ ' const html = buildHtml(stackName, drifts);',
124
+ '',
125
+ ' // Write machine-readable JSON if requested',
126
+ ' if (outputFile) {',
127
+ ' try {',
128
+ " const { writeFile } = await import('fs/promises');",
129
+ ' const result = [',
130
+ ' {',
131
+ ' stackName,',
132
+ ' driftStatus: stackDriftStatus,',
133
+ ' driftedResources: (drifts || []).map(d => ({',
134
+ ' logicalResourceId: d.LogicalResourceId,',
135
+ ' resourceType: d.ResourceType,',
136
+ ' stackResourceDriftStatus: d.StackResourceDriftStatus,',
137
+ ' propertyDifferences: d.PropertyDifferences,',
138
+ ' })),',
139
+ ' },',
140
+ ' ];',
141
+ " await writeFile(outputFile, JSON.stringify(result, null, 2), { encoding: 'utf8' });",
142
+ ' } catch (e: any) {',
143
+ " console.error('Failed to write drift JSON results:', e?.message || e);",
144
+ ' }',
145
+ ' }',
146
+ '',
147
+ ' // Print to stdout and append to summary if available',
148
+ ' console.log(html);',
149
+ ' const stepSummary = process.env.GITHUB_STEP_SUMMARY;',
150
+ ' if (stepSummary) {',
151
+ ' try {',
152
+ " const { appendFile } = await import('fs/promises');",
153
+ " await appendFile(stepSummary, `${html}\\n`, { encoding: 'utf8' });",
154
+ ' } catch (e: any) {',
155
+ " console.error('Failed to append to GITHUB_STEP_SUMMARY:', e?.message || e);",
156
+ ' }',
157
+ ' }',
158
+ '',
159
+ ' const commentUrl = process.env.GITHUB_COMMENT_URL;',
160
+ ' const token = process.env.GITHUB_TOKEN;',
161
+ ' if (commentUrl && token) {',
162
+ ' await postGithubComment(commentUrl, token, html);',
163
+ ' }',
164
+ '',
165
+ ' process.exit(1);',
166
+ ' }',
167
+ '',
168
+ ' // No drift case',
169
+ ' if (outputFile) {',
170
+ ' try {',
171
+ " const { writeFile } = await import('fs/promises');",
172
+ ' const result = [',
173
+ ' {',
174
+ ' stackName,',
175
+ " driftStatus: 'IN_SYNC',",
176
+ ' driftedResources: [],',
177
+ ' },',
178
+ ' ];',
179
+ " await writeFile(outputFile, JSON.stringify(result, null, 2), { encoding: 'utf8' });",
180
+ ' } catch (e: any) {',
181
+ " console.error('Failed to write drift JSON results:', e?.message || e);",
182
+ ' }',
183
+ ' }',
184
+ " console.log('No drift detected (IN_SYNC)');",
185
+ '}',
186
+ '',
187
+ 'main().catch((e) => {',
188
+ ' console.error(e);',
189
+ ' process.exit(1);',
190
+ '});',
191
+ ],
192
+ });
193
+ }
194
+ }
195
+ exports.CdkDriftDetectionScript = CdkDriftDetectionScript;
196
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cdk-drift-detection-script.js","sourceRoot":"","sources":["../../src/bin/cdk-drift-detection-script.ts"],"names":[],"mappings":"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","sourcesContent":["import { TextFile } from 'projen';\n\ninterface CdkDriftDetectionScriptProps {\n  // Avoid exporting projen types in public API\n  project: any;\n  outputPath?: string;\n}\n\n/**\n * Projen helper to emit the drift detection script into the repository so\n * GitHub workflows can execute it from a stable location.\n *\n * This mirrors the pattern used by CdkChangesetScript, but writes the\n * drift script (detect-drift.ts) under the workflows scripts directory by default.\n */\nexport class CdkDriftDetectionScript {\n  constructor(props: CdkDriftDetectionScriptProps) {\n    const outputPath = props.outputPath ?? '.github/workflows/scripts/detect-drift.ts';\n\n    new TextFile(props.project, outputPath, {\n      lines: [\n        \"import {\\n  CloudFormationClient,\\n  DescribeStackDriftDetectionStatusCommand,\\n  DetectStackDriftCommand,\\n  DescribeStackResourceDriftsCommand,\\n  type DescribeStackResourceDriftsCommandOutput,\\n  type StackResourceDriftStatus,\\n} from '@aws-sdk/client-cloudformation';\",\n        '',\n        'async function sleep(ms: number) {',\n        '  return new Promise((r) => setTimeout(r, ms));',\n        '}',\n        '',\n        'async function main() {',\n        '  const stackName = process.env.STACK_NAME;',\n        '  if (!stackName) {',\n        \"    console.error('STACK_NAME env var is required');\",\n        '    process.exit(1);',\n        '  }',\n        '',\n        '  // Region and credentials pulled from environment set by actions/configure-aws-credentials',\n        '  const client = new CloudFormationClient({});',\n        '',\n        '  const detect = await client.send(new DetectStackDriftCommand({ StackName: stackName }));',\n        '  if (!detect.StackDriftDetectionId) {',\n        \"    console.error('Failed to start drift detection');\",\n        '    process.exit(1);',\n        '  }',\n        '',\n        '  const id = detect.StackDriftDetectionId;',\n        '  console.log(`Drift detection started: ${id}`);',\n        '',\n        \"  let detectionStatus = 'DETECTION_IN_PROGRESS';\",\n        '  let stackDriftStatus: string | undefined;',\n        '',\n        \"  while (detectionStatus === 'DETECTION_IN_PROGRESS') {\",\n        '    await sleep(5000);',\n        '    const res = await client.send(',\n        '      new DescribeStackDriftDetectionStatusCommand({ StackDriftDetectionId: id }),',\n        '    );',\n        \"    detectionStatus = res.DetectionStatus ?? 'UNKNOWN';\",\n        '    stackDriftStatus = res.StackDriftStatus;',\n        '    console.log(`Detection status: ${detectionStatus}`);',\n        '  }',\n        '',\n        '  // Helper to build an HTML report of drifted resources',\n        '  const buildHtml = (stack: string, drifts: any[]): string => {',\n        '    let body = `<h1>Drift report</h1><h2>Stack Name: ${stack}</h2><br>`;',\n        '    if (drifts.length === 0) {',\n        \"      body += 'no drift.';\",\n        '      return body;',\n        '    }',\n        \"    body += '<table>' +\",\n        \"      '<tr><th>Status</th><th>ID</th><th>Type</th><th>Differences</th></tr>';\",\n        '    for (const d of drifts) {',\n        \"      const status = d.StackResourceDriftStatus ?? '-';\",\n        \"      const logicalId = d.LogicalResourceId ?? '-';\",\n        \"      const type = d.ResourceType ?? '-';\",\n        '      const diffs = (d.PropertyDifferences ?? []).map((pd: any) => {',\n        \"        const p = pd.PropertyPath ?? '-';\",\n        \"        const t = pd.DifferenceType ?? '-';\",\n        '        return `- ${t}: ${p}`;',\n        \"      }).join('<br>');\",\n        \"      const statusEmoji = status === 'MODIFIED' ? '🟠' : status === 'DELETED' ? '🔴' : status === 'NOT_CHECKED' ? '⚪' : '🟢';\",\n        \"      body += '<tr>' +\",\n        '        `<td>${statusEmoji} ${status}</td>` +',\n        '        `<td>${logicalId}</td>` +',\n        '        `<td>${type}</td>` +',\n        '        `<td>${diffs}</td>` +',\n        \"        '</tr>';\",\n        '    }',\n        \"    body += '</table>';\",\n        '    return body;',\n        '  };',\n        '',\n        '  async function listDriftedResources(): Promise<any[]> {',\n        '    const results: any[] = [];',\n        '    // Only include resources that are not IN_SYNC',\n        \"    const filters: StackResourceDriftStatus[] = ['MODIFIED', 'DELETED', 'NOT_CHECKED'];\",\n        '    let nextToken: string | undefined = undefined;',\n        '    do {',\n        '      const resp: DescribeStackResourceDriftsCommandOutput = await client.send(new DescribeStackResourceDriftsCommand({',\n        '        StackName: stackName,',\n        '        NextToken: nextToken,',\n        '        StackResourceDriftStatusFilters: filters,',\n        '      }));',\n        '      if (resp.StackResourceDrifts) results.push(...resp.StackResourceDrifts);',\n        '      nextToken = resp.NextToken;',\n        '    } while (nextToken);',\n        '    return results;',\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        '    if (!res.ok) {',\n        '      const text = await res.text().catch(() => \\'\\');',\n        '      console.error(`Failed to post GitHub comment: ${res.status} ${res.statusText} ${text}`);',\n        '    }',\n        '  }',\n        '',\n        '  // When there is drift, collect details and post a PR comment + step summary',\n        '  const outputFile = process.env.DRIFT_DETECTION_OUTPUT;',\n        \"  if (stackDriftStatus !== 'IN_SYNC') {\",\n        '    console.error(`Drift detected (status: ${stackDriftStatus})`);',\n        '    const drifts = await listDriftedResources();',\n        '    const html = buildHtml(stackName, drifts);',\n        '',\n        '    // Write machine-readable JSON if requested',\n        '    if (outputFile) {',\n        '      try {',\n        \"        const { writeFile } = await import('fs/promises');\",\n        '        const result = [',\n        '          {',\n        '            stackName,',\n        '            driftStatus: stackDriftStatus,',\n        '            driftedResources: (drifts || []).map(d => ({',\n        '              logicalResourceId: d.LogicalResourceId,',\n        '              resourceType: d.ResourceType,',\n        '              stackResourceDriftStatus: d.StackResourceDriftStatus,',\n        '              propertyDifferences: d.PropertyDifferences,',\n        '            })),',\n        '          },',\n        '        ];',\n        \"        await writeFile(outputFile, JSON.stringify(result, null, 2), { encoding: 'utf8' });\",\n        '      } catch (e: any) {',\n        \"        console.error('Failed to write drift JSON results:', e?.message || e);\",\n        '      }',\n        '    }',\n        '',\n        '    // Print to stdout and append to summary if available',\n        '    console.log(html);',\n        '    const stepSummary = process.env.GITHUB_STEP_SUMMARY;',\n        '    if (stepSummary) {',\n        '      try {',\n        \"        const { appendFile } = await import('fs/promises');\",\n        \"        await appendFile(stepSummary, `${html}\\\\n`, { encoding: 'utf8' });\",\n        '      } catch (e: any) {',\n        \"        console.error('Failed to append to GITHUB_STEP_SUMMARY:', e?.message || e);\",\n        '      }',\n        '    }',\n        '',\n        '    const commentUrl = process.env.GITHUB_COMMENT_URL;',\n        '    const token = process.env.GITHUB_TOKEN;',\n        '    if (commentUrl && token) {',\n        '      await postGithubComment(commentUrl, token, html);',\n        '    }',\n        '',\n        '    process.exit(1);',\n        '  }',\n        '',\n        '  // No drift case',\n        '  if (outputFile) {',\n        '    try {',\n        \"      const { writeFile } = await import('fs/promises');\",\n        '      const result = [',\n        '        {',\n        '          stackName,',\n        \"          driftStatus: 'IN_SYNC',\",\n        '          driftedResources: [],',\n        '        },',\n        '      ];',\n        \"      await writeFile(outputFile, JSON.stringify(result, null, 2), { encoding: 'utf8' });\",\n        '    } catch (e: any) {',\n        \"      console.error('Failed to write drift JSON results:', e?.message || e);\",\n        '    }',\n        '  }',\n        \"  console.log('No drift detected (IN_SYNC)');\",\n        '}',\n        '',\n        'main().catch((e) => {',\n        '  console.error(e);',\n        '  process.exit(1);',\n        '});',\n      ],\n    });\n  }\n}\n"]}
@@ -0,0 +1,162 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const client_cloudformation_1 = require("@aws-sdk/client-cloudformation");
4
+ async function sleep(ms) {
5
+ return new Promise((r) => setTimeout(r, ms));
6
+ }
7
+ async function main() {
8
+ const stackName = process.env.STACK_NAME;
9
+ if (!stackName) {
10
+ console.error('STACK_NAME env var is required');
11
+ process.exit(1);
12
+ }
13
+ // Region and credentials pulled from environment set by actions/configure-aws-credentials
14
+ const client = new client_cloudformation_1.CloudFormationClient({});
15
+ const detect = await client.send(new client_cloudformation_1.DetectStackDriftCommand({ StackName: stackName }));
16
+ if (!detect.StackDriftDetectionId) {
17
+ console.error('Failed to start drift detection');
18
+ process.exit(1);
19
+ }
20
+ const id = detect.StackDriftDetectionId;
21
+ console.log(`Drift detection started: ${id}`);
22
+ let detectionStatus = 'DETECTION_IN_PROGRESS';
23
+ let stackDriftStatus;
24
+ while (detectionStatus === 'DETECTION_IN_PROGRESS') {
25
+ await sleep(5000);
26
+ const res = await client.send(new client_cloudformation_1.DescribeStackDriftDetectionStatusCommand({ StackDriftDetectionId: id }));
27
+ detectionStatus = res.DetectionStatus ?? 'UNKNOWN';
28
+ stackDriftStatus = res.StackDriftStatus;
29
+ console.log(`Detection status: ${detectionStatus}`);
30
+ }
31
+ // Helper to build an HTML report of drifted resources
32
+ const buildHtml = (stack, drifts) => {
33
+ let body = `<h1>Drift report</h1><h2>Stack Name: ${stack}</h2><br>`;
34
+ if (drifts.length === 0) {
35
+ body += 'no drift.';
36
+ return body;
37
+ }
38
+ body += '<table>' +
39
+ '<tr><th>Status</th><th>ID</th><th>Type</th><th>Differences</th></tr>';
40
+ for (const d of drifts) {
41
+ const status = d.StackResourceDriftStatus ?? '-';
42
+ const logicalId = d.LogicalResourceId ?? '-';
43
+ const type = d.ResourceType ?? '-';
44
+ const diffs = (d.PropertyDifferences ?? []).map((pd) => {
45
+ const p = pd.PropertyPath ?? '-';
46
+ const t = pd.DifferenceType ?? '-';
47
+ return `- ${t}: ${p}`;
48
+ }).join('<br>');
49
+ const statusEmoji = status === 'MODIFIED' ? '🟠' : status === 'DELETED' ? '🔴' : status === 'NOT_CHECKED' ? '⚪' : '🟢';
50
+ body += '<tr>' +
51
+ `<td>${statusEmoji} ${status}</td>` +
52
+ `<td>${logicalId}</td>` +
53
+ `<td>${type}</td>` +
54
+ `<td>${diffs}</td>` +
55
+ '</tr>';
56
+ }
57
+ body += '</table>';
58
+ return body;
59
+ };
60
+ async function listDriftedResources() {
61
+ const results = [];
62
+ // Only include resources that are not IN_SYNC
63
+ const filters = ['MODIFIED', 'DELETED', 'NOT_CHECKED'];
64
+ let nextToken = undefined;
65
+ do {
66
+ const resp = await client.send(new client_cloudformation_1.DescribeStackResourceDriftsCommand({
67
+ StackName: stackName,
68
+ NextToken: nextToken,
69
+ StackResourceDriftStatusFilters: filters,
70
+ }));
71
+ if (resp.StackResourceDrifts)
72
+ results.push(...resp.StackResourceDrifts);
73
+ nextToken = resp.NextToken;
74
+ } while (nextToken);
75
+ return results;
76
+ }
77
+ async function postGithubComment(url, token, body) {
78
+ const res = await fetch(url, {
79
+ method: 'POST',
80
+ headers: {
81
+ 'Authorization': `token ${token}`,
82
+ 'Content-Type': 'application/json',
83
+ 'Accept': 'application/vnd.github+json',
84
+ },
85
+ body: JSON.stringify({ body }),
86
+ });
87
+ if (!res.ok) {
88
+ const text = await res.text().catch(() => '');
89
+ console.error(`Failed to post GitHub comment: ${res.status} ${res.statusText} ${text}`);
90
+ }
91
+ }
92
+ // When there is drift, collect details and post a PR comment + step summary
93
+ const outputFile = process.env.DRIFT_DETECTION_OUTPUT;
94
+ if (stackDriftStatus !== 'IN_SYNC') {
95
+ console.error(`Drift detected (status: ${stackDriftStatus})`);
96
+ const drifts = await listDriftedResources();
97
+ const html = buildHtml(stackName, drifts);
98
+ // Write machine-readable JSON if requested
99
+ if (outputFile) {
100
+ try {
101
+ const { writeFile } = await Promise.resolve().then(() => require('fs/promises'));
102
+ const result = [
103
+ {
104
+ stackName,
105
+ driftStatus: stackDriftStatus,
106
+ driftedResources: (drifts || []).map(d => ({
107
+ logicalResourceId: d.LogicalResourceId,
108
+ resourceType: d.ResourceType,
109
+ stackResourceDriftStatus: d.StackResourceDriftStatus,
110
+ propertyDifferences: d.PropertyDifferences,
111
+ })),
112
+ },
113
+ ];
114
+ await writeFile(outputFile, JSON.stringify(result, null, 2), { encoding: 'utf8' });
115
+ }
116
+ catch (e) {
117
+ console.error('Failed to write drift JSON results:', e?.message || e);
118
+ }
119
+ }
120
+ // Print to stdout and append to summary if available
121
+ console.log(html);
122
+ const stepSummary = process.env.GITHUB_STEP_SUMMARY;
123
+ if (stepSummary) {
124
+ try {
125
+ const { appendFile } = await Promise.resolve().then(() => require('fs/promises'));
126
+ await appendFile(stepSummary, `${html}\n`, { encoding: 'utf8' });
127
+ }
128
+ catch (e) {
129
+ console.error('Failed to append to GITHUB_STEP_SUMMARY:', e?.message || e);
130
+ }
131
+ }
132
+ const commentUrl = process.env.GITHUB_COMMENT_URL;
133
+ const token = process.env.GITHUB_TOKEN;
134
+ if (commentUrl && token) {
135
+ await postGithubComment(commentUrl, token, html);
136
+ }
137
+ process.exit(1);
138
+ }
139
+ // No drift case
140
+ if (outputFile) {
141
+ try {
142
+ const { writeFile } = await Promise.resolve().then(() => require('fs/promises'));
143
+ const result = [
144
+ {
145
+ stackName,
146
+ driftStatus: 'IN_SYNC',
147
+ driftedResources: [],
148
+ },
149
+ ];
150
+ await writeFile(outputFile, JSON.stringify(result, null, 2), { encoding: 'utf8' });
151
+ }
152
+ catch (e) {
153
+ console.error('Failed to write drift JSON results:', e?.message || e);
154
+ }
155
+ }
156
+ console.log('No drift detected (IN_SYNC)');
157
+ }
158
+ main().catch((e) => {
159
+ console.error(e);
160
+ process.exit(1);
161
+ });
162
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"detect-drift.js","sourceRoot":"","sources":["../../src/bin/detect-drift.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  CloudFormationClient,\n  DescribeStackDriftDetectionStatusCommand,\n  DetectStackDriftCommand,\n  DescribeStackResourceDriftsCommand,\n  type DescribeStackResourceDriftsCommandOutput,\n  type StackResourceDriftStatus,\n} from '@aws-sdk/client-cloudformation';\n\nasync function sleep(ms: number) {\n  return new Promise((r) => setTimeout(r, ms));\n}\n\nasync function main() {\n  const stackName = process.env.STACK_NAME;\n  if (!stackName) {\n    console.error('STACK_NAME env var is required');\n    process.exit(1);\n  }\n\n  // Region and credentials pulled from environment set by actions/configure-aws-credentials\n  const client = new CloudFormationClient({});\n\n  const detect = await client.send(new DetectStackDriftCommand({ StackName: stackName }));\n  if (!detect.StackDriftDetectionId) {\n    console.error('Failed to start drift detection');\n    process.exit(1);\n  }\n\n  const id = detect.StackDriftDetectionId;\n  console.log(`Drift detection started: ${id}`);\n\n  let detectionStatus = 'DETECTION_IN_PROGRESS';\n  let stackDriftStatus: string | undefined;\n\n  while (detectionStatus === 'DETECTION_IN_PROGRESS') {\n    await sleep(5000);\n    const res = await client.send(\n      new DescribeStackDriftDetectionStatusCommand({ StackDriftDetectionId: id }),\n    );\n    detectionStatus = res.DetectionStatus ?? 'UNKNOWN';\n    stackDriftStatus = res.StackDriftStatus;\n    console.log(`Detection status: ${detectionStatus}`);\n  }\n\n  // Helper to build an HTML report of drifted resources\n  const buildHtml = (stack: string, drifts: any[]): string => {\n    let body = `<h1>Drift report</h1><h2>Stack Name: ${stack}</h2><br>`;\n    if (drifts.length === 0) {\n      body += 'no drift.';\n      return body;\n    }\n    body += '<table>' +\n      '<tr><th>Status</th><th>ID</th><th>Type</th><th>Differences</th></tr>';\n    for (const d of drifts) {\n      const status = d.StackResourceDriftStatus ?? '-';\n      const logicalId = d.LogicalResourceId ?? '-';\n      const type = d.ResourceType ?? '-';\n      const diffs = (d.PropertyDifferences ?? []).map((pd: any) => {\n        const p = pd.PropertyPath ?? '-';\n        const t = pd.DifferenceType ?? '-';\n        return `- ${t}: ${p}`;\n      }).join('<br>');\n      const statusEmoji = status === 'MODIFIED' ? '🟠' : status === 'DELETED' ? '🔴' : status === 'NOT_CHECKED' ? '⚪' : '🟢';\n      body += '<tr>' +\n        `<td>${statusEmoji} ${status}</td>` +\n        `<td>${logicalId}</td>` +\n        `<td>${type}</td>` +\n        `<td>${diffs}</td>` +\n        '</tr>';\n    }\n    body += '</table>';\n    return body;\n  };\n\n  async function listDriftedResources(): Promise<any[]> {\n    const results: any[] = [];\n    // Only include resources that are not IN_SYNC\n    const filters: StackResourceDriftStatus[] = ['MODIFIED', 'DELETED', 'NOT_CHECKED'];\n    let nextToken: string | undefined = undefined;\n    do {\n      const resp: DescribeStackResourceDriftsCommandOutput = await client.send(new DescribeStackResourceDriftsCommand({\n        StackName: stackName,\n        NextToken: nextToken,\n        StackResourceDriftStatusFilters: filters,\n      }));\n      if (resp.StackResourceDrifts) results.push(...resp.StackResourceDrifts);\n      nextToken = resp.NextToken;\n    } while (nextToken);\n    return results;\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    if (!res.ok) {\n      const text = await res.text().catch(() => '');\n      console.error(`Failed to post GitHub comment: ${res.status} ${res.statusText} ${text}`);\n    }\n  }\n\n  // When there is drift, collect details and post a PR comment + step summary\n  const outputFile = process.env.DRIFT_DETECTION_OUTPUT;\n  if (stackDriftStatus !== 'IN_SYNC') {\n    console.error(`Drift detected (status: ${stackDriftStatus})`);\n    const drifts = await listDriftedResources();\n    const html = buildHtml(stackName, drifts);\n\n    // Write machine-readable JSON if requested\n    if (outputFile) {\n      try {\n        const { writeFile } = await import('fs/promises');\n        const result = [\n          {\n            stackName,\n            driftStatus: stackDriftStatus,\n            driftedResources: (drifts || []).map(d => ({\n              logicalResourceId: d.LogicalResourceId,\n              resourceType: d.ResourceType,\n              stackResourceDriftStatus: d.StackResourceDriftStatus,\n              propertyDifferences: d.PropertyDifferences,\n            })),\n          },\n        ];\n        await writeFile(outputFile, JSON.stringify(result, null, 2), { encoding: 'utf8' });\n      } catch (e: any) {\n        console.error('Failed to write drift JSON results:', e?.message || e);\n      }\n    }\n\n    // Print to stdout and append to summary if available\n    console.log(html);\n    const stepSummary = process.env.GITHUB_STEP_SUMMARY;\n    if (stepSummary) {\n      try {\n        const { appendFile } = await import('fs/promises');\n        await appendFile(stepSummary, `${html}\\n`, { encoding: 'utf8' });\n      } catch (e: any) {\n        console.error('Failed to append to GITHUB_STEP_SUMMARY:', e?.message || e);\n      }\n    }\n\n    const commentUrl = process.env.GITHUB_COMMENT_URL;\n    const token = process.env.GITHUB_TOKEN;\n    if (commentUrl && token) {\n      await postGithubComment(commentUrl, token, html);\n    }\n\n    process.exit(1);\n  }\n\n  // No drift case\n  if (outputFile) {\n    try {\n      const { writeFile } = await import('fs/promises');\n      const result = [\n        {\n          stackName,\n          driftStatus: 'IN_SYNC',\n          driftedResources: [],\n        },\n      ];\n      await writeFile(outputFile, JSON.stringify(result, null, 2), { encoding: 'utf8' });\n    } catch (e: any) {\n      console.error('Failed to write drift JSON results:', e?.message || e);\n    }\n  }\n  console.log('No drift detected (IN_SYNC)');\n}\n\nmain().catch((e) => {\n  console.error(e);\n  process.exit(1);\n});\n"]}
package/lib/index.d.ts CHANGED
@@ -1,2 +1,4 @@
1
1
  export * from './CdkDiffStackWorkflow';
2
2
  export * from './CdkDiffIamTemplate';
3
+ export * from './CdkDriftIamTemplate';
4
+ export * from './CdkDriftDetectionWorkflow';
package/lib/index.js CHANGED
@@ -16,4 +16,6 @@ var __exportStar = (this && this.__exportStar) || function(m, exports) {
16
16
  Object.defineProperty(exports, "__esModule", { value: true });
17
17
  __exportStar(require("./CdkDiffStackWorkflow"), exports);
18
18
  __exportStar(require("./CdkDiffIamTemplate"), exports);
19
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7OztBQUFBLHlEQUF1QztBQUN2Qyx1REFBcUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL0Nka0RpZmZTdGFja1dvcmtmbG93JztcbmV4cG9ydCAqIGZyb20gJy4vQ2RrRGlmZklhbVRlbXBsYXRlJztcbiJdfQ==
19
+ __exportStar(require("./CdkDriftIamTemplate"), exports);
20
+ __exportStar(require("./CdkDriftDetectionWorkflow"), exports);
21
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7OztBQUFBLHlEQUF1QztBQUN2Qyx1REFBcUM7QUFDckMsd0RBQXNDO0FBQ3RDLDhEQUE0QyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vQ2RrRGlmZlN0YWNrV29ya2Zsb3cnO1xuZXhwb3J0ICogZnJvbSAnLi9DZGtEaWZmSWFtVGVtcGxhdGUnO1xuZXhwb3J0ICogZnJvbSAnLi9DZGtEcmlmdElhbVRlbXBsYXRlJztcbmV4cG9ydCAqIGZyb20gJy4vQ2RrRHJpZnREZXRlY3Rpb25Xb3JrZmxvdyc7XG4iXX0=
package/package.json CHANGED
@@ -19,11 +19,13 @@
19
19
  "package-all": "npx projen package-all",
20
20
  "package:js": "npx projen package:js",
21
21
  "post-compile": "npx projen post-compile",
22
+ "post-upgrade": "npx projen post-upgrade",
22
23
  "pre-compile": "npx projen pre-compile",
23
24
  "release": "npx projen release",
24
25
  "test": "npx projen test",
25
26
  "test:watch": "npx projen test:watch",
26
27
  "unbump": "npx projen unbump",
28
+ "upgrade": "npx projen upgrade",
27
29
  "watch": "npx projen watch",
28
30
  "projen": "npx projen"
29
31
  },
@@ -102,7 +104,10 @@
102
104
  },
103
105
  "main": "lib/index.js",
104
106
  "license": "Apache-2.0",
105
- "version": "0.0.1-beta",
107
+ "publishConfig": {
108
+ "access": "public"
109
+ },
110
+ "version": "0.0.1",
106
111
  "jest": {
107
112
  "coverageProvider": "v8",
108
113
  "testMatch": [
@@ -149,7 +154,7 @@
149
154
  }
150
155
  },
151
156
  "types": "lib/index.d.ts",
152
- "stability": "stable",
157
+ "stability": "experimental",
153
158
  "jsii": {
154
159
  "outdir": "dist",
155
160
  "targets": {},
@@ -0,0 +1,17 @@
1
+ sonar.projectKey=JaysonRawlins_cdk-diff-pr-github-action
2
+ sonar.organization=jaysonrawlins
3
+
4
+
5
+ # This is the name and version displayed in the SonarCloud UI.
6
+ sonar.projectName=cdk-diff-pr-github-action
7
+ #sonar.projectVersion=1.0
8
+
9
+
10
+ # Path is relative to the sonar-project.properties file. Replace "\" by "/" on Windows.
11
+
12
+ # Encoding of the source code. Default is default system encoding
13
+ sonar.sourceEncoding=UTF-8
14
+ sonar.sources=src,test
15
+ sonar.host.url=https://sonarcloud.io
16
+ sonar.language=ts
17
+ sonar.exclusions=**/*.spec.ts,**/*.test.ts
@@ -1,62 +0,0 @@
1
- ### Guidelines
2
-
3
- ## AWS Credentials and Environment Setup
4
-
5
- Use the .env file if there is one present
6
-
7
- ### Steps
8
- 1) Load environment variables for the chosen environment:
9
- ```bash
10
- export $(grep -v '^#' .env | xargs)
11
- ```
12
-
13
- 2.) Verify credentials:
14
- ```bash
15
- aws sts get-caller-identity
16
- ```
17
-
18
- # Instructions for Projen Projects
19
-
20
- ## Projen manages project config as code. Look for `.projenrc.<ext>` to understand how the project is defined.
21
-
22
- ### Before you change anything
23
- - Locate `.projenrc.<ext>` (e.g., `.js`, `.ts`, `.py`).
24
- - Identify project type (Node, Python, etc.) and constructs used.
25
- - Check for existing Projen tasks that do what you need.
26
-
27
- ### Typical workflow
28
- 1) Update `.projenrc.*` with desired changes.
29
- 2) Synthesize files:
30
- ```bash
31
- npx projen
32
- ```
33
- 1) Review generated diffs before committing.
34
-
35
- ---
36
-
37
- ## Memory-Aware Workflow (Mem0)
38
-
39
- Always check relevant memories before starting each task, and update them as needed. Follow this lightweight loop:
40
-
41
- 1) Search for relevant memories
42
- - Use concise keywords from the task (1–3 words). Example queries: `projen test`, `deploy creds`, `pytest flags`.
43
- - If a relevant preference is found, apply it during the task.
44
-
45
- 2) Add new preferences when discovered
46
- - When you or the user clarify a repeatable preference or rule, store it as a memory in clear, actionable language.
47
-
48
- 3) Verify memory storage (quick check)
49
- - After adding a memory, immediately retrieve it with a related keyword to confirm it’s indexed and discoverable.
50
-
51
- 4) Re-run with preference when applicable
52
- - If the memory affects execution (e.g., which command to run), ensure the next step uses the remembered preference.
53
-
54
- ### Example (validated)
55
- - Preference: "I prefer npx projen test when there is a .projenrc.ts file present in the project."
56
- - Retrieval keyword: `projen test preference`
57
- - Expected outcome: The preference is found and informs task execution (e.g., run `npx projen test` when `.projenrc.ts` exists).
58
-
59
- ---
60
- ## Web Search
61
-
62
- Use the kagi mcp for web search related needs. Especially when trying to find the latest documents to solve the current task problem.