@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,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
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
|
-
|
|
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
|
-
"
|
|
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": "
|
|
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
|
package/.junie/guidelines.md
DELETED
|
@@ -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.
|