@jjrawlins/cdk-diff-pr-github-action 0.0.5-beta → 0.0.6-beta
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.
|
@@ -51,7 +51,9 @@ class CdkDriftDetectionWorkflow {
|
|
|
51
51
|
const short = stack.stackName ?? stageFromStack(stack.stackName);
|
|
52
52
|
const jobId = `drift-${short}`;
|
|
53
53
|
const resultsFile = `drift-results-${short}.json`;
|
|
54
|
-
const
|
|
54
|
+
const innerCond = "github.event_name == 'schedule' || !github.event.inputs.stage || github.event.inputs.stage == '" + short + "'";
|
|
55
|
+
const condExpr = "${{ " + innerCond + " }}";
|
|
56
|
+
const notCondExpr = "${{ !(" + innerCond + ") }}";
|
|
55
57
|
jobs[jobId] = {
|
|
56
58
|
name: `Drift Detection - ${short}`,
|
|
57
59
|
runsOn: ['ubuntu-latest'],
|
|
@@ -67,17 +69,20 @@ class CdkDriftDetectionWorkflow {
|
|
|
67
69
|
STAGE_NAME: short,
|
|
68
70
|
STACK_NAME: stack.stackName,
|
|
69
71
|
},
|
|
70
|
-
|
|
72
|
+
// No job-level condition; we gate steps so the job always completes and summary can run
|
|
71
73
|
steps: [
|
|
72
|
-
{ name: '
|
|
74
|
+
{ name: 'Skip (stage not selected)', if: notCondExpr, run: 'echo "Stage not selected; skipping drift detection for this job."' },
|
|
75
|
+
{ name: 'Checkout', if: condExpr, uses: `actions/checkout@${githubActionsCheckoutVersion}` },
|
|
73
76
|
{
|
|
74
77
|
name: 'Setup Node.js',
|
|
78
|
+
if: condExpr,
|
|
75
79
|
uses: `actions/setup-node@${githubActionsSetupNodeVersion}`,
|
|
76
80
|
with: { 'node-version': nodeVersion },
|
|
77
81
|
},
|
|
78
|
-
{ name: 'Install dependencies', run: 'yarn install --frozen-lockfile || npm ci', env: { GITHUB_TOKEN: '${{ secrets.GITHUB_TOKEN }}' } },
|
|
82
|
+
{ name: 'Install dependencies', if: condExpr, run: 'yarn install --frozen-lockfile || npm ci', env: { GITHUB_TOKEN: '${{ secrets.GITHUB_TOKEN }}' } },
|
|
79
83
|
{
|
|
80
84
|
name: 'AWS Credentials',
|
|
85
|
+
if: condExpr,
|
|
81
86
|
id: 'creds',
|
|
82
87
|
uses: `aws-actions/configure-aws-credentials@${githubActionsAwsCredentialsVersion}`,
|
|
83
88
|
with: {
|
|
@@ -88,6 +93,7 @@ class CdkDriftDetectionWorkflow {
|
|
|
88
93
|
},
|
|
89
94
|
{
|
|
90
95
|
name: 'Assume Drift Detection Role',
|
|
96
|
+
if: condExpr,
|
|
91
97
|
uses: `aws-actions/configure-aws-credentials@${githubActionsAwsCredentialsVersion}`,
|
|
92
98
|
with: {
|
|
93
99
|
'role-to-assume': stack.driftDetectionRoleToAssumeArn,
|
|
@@ -101,6 +107,7 @@ class CdkDriftDetectionWorkflow {
|
|
|
101
107
|
},
|
|
102
108
|
{
|
|
103
109
|
name: 'Detect drift',
|
|
110
|
+
if: condExpr,
|
|
104
111
|
id: 'drift',
|
|
105
112
|
continueOnError: true, // allow artifact upload and issue creation even when drift is detected
|
|
106
113
|
run: [
|
|
@@ -117,6 +124,7 @@ class CdkDriftDetectionWorkflow {
|
|
|
117
124
|
},
|
|
118
125
|
{
|
|
119
126
|
name: 'Upload results',
|
|
127
|
+
if: condExpr,
|
|
120
128
|
uses: `actions/upload-artifact@${githubActionsUploadArtifactVersion}`,
|
|
121
129
|
with: { name: `drift-results-${short}`, path: resultsFile, ifNoFilesFound: 'ignore' },
|
|
122
130
|
},
|
|
@@ -124,7 +132,7 @@ class CdkDriftDetectionWorkflow {
|
|
|
124
132
|
? [
|
|
125
133
|
{
|
|
126
134
|
name: 'Create Issue on Drift',
|
|
127
|
-
if: "always() && steps.drift.outcome == 'failure'
|
|
135
|
+
if: "always() && steps.drift.outcome == 'failure'",
|
|
128
136
|
uses: `actions/github-script@${githubActionsGithubScriptVersion}`,
|
|
129
137
|
with: { script: issueScript(short, stack.driftDetectionRoleToAssumeRegion, resultsFile) },
|
|
130
138
|
},
|
|
@@ -239,4 +247,4 @@ function summaryScript() {
|
|
|
239
247
|
function toKebabCase(s) {
|
|
240
248
|
return s.replace(/[^a-zA-Z0-9]+/g, '-').replace(/^-+|-+$/g, '').toLowerCase();
|
|
241
249
|
}
|
|
242
|
-
//# 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\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 stage choice\n    const stageChoices = props.stacks.map((s) => s.stackName ?? stageFromStack(s.stackName));\n    workflow.on({\n      schedule: props.schedule ? [{ cron: props.schedule }] : undefined,\n      workflowDispatch: {\n        inputs: {\n          stage: {\n            description: 'Stage to check for drift (leave empty for all)',\n            required: false,\n            type: 'choice',\n            options: stageChoices,\n          },\n        },\n      },\n    });\n\n    // One job per stage\n    const jobs: Record<string, any> = {};\n    for (const stack of props.stacks) {\n      const short = stack.stackName ?? stageFromStack(stack.stackName);\n      const jobId = `drift-${short}`;\n      const resultsFile = `drift-results-${short}.json`;\n      const ifCond = \"${{ github.event_name == 'schedule' || github.event.inputs.stage == '' || github.event.inputs.stage == '\" + short + \"' }}\";\n\n      jobs[jobId] = {\n        name: `Drift Detection - ${short}`,\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          STAGE_NAME: short,\n          STACK_NAME: stack.stackName,\n        },\n        if: ifCond,\n        steps: [\n          { name: 'Checkout', uses: `actions/checkout@${githubActionsCheckoutVersion}` },\n          {\n            name: 'Setup Node.js',\n            uses: `actions/setup-node@${githubActionsSetupNodeVersion}`,\n            with: { 'node-version': nodeVersion },\n          },\n          { name: 'Install dependencies', run: 'yarn install --frozen-lockfile || npm ci', env: { GITHUB_TOKEN: '${{ secrets.GITHUB_TOKEN }}' } },\n          {\n            name: 'AWS Credentials',\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            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            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              'if [ -f \"$DRIFT_DETECTION_OUTPUT\" ]; then echo \"Results file created: $DRIFT_DETECTION_OUTPUT\"; fi',\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            uses: `actions/upload-artifact@${githubActionsUploadArtifactVersion}`,\n            with: { name: `drift-results-${short}`, path: resultsFile, ifNoFilesFound: 'ignore' },\n          },\n          ...(\n            createIssues\n              ? [\n                {\n                  name: 'Create Issue on Drift',\n                  if: \"always() && steps.drift.outcome == 'failure' && github.event_name == 'schedule'\",\n                  uses: `actions/github-script@${githubActionsGithubScriptVersion}`,\n                  with: { script: issueScript(short, stack.driftDetectionRoleToAssumeRegion, resultsFile) },\n                },\n              ]\n              : []\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\nfunction stageFromStack(stackName: string): string {\n  const parts = stackName.split('-');\n  return parts[parts.length - 1] || stackName;\n}\n\nfunction issueScript(stage: 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; }\",\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; }\",\n    `const title = 'Drift Detected in ${stage}';`,\n    `let body = '## Drift Detection Report\\\\n\\\\n' + '**Stage:** ${stage}\\\\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', '${stage}'] });`,\n    'if (issues.data.length === 0) {',\n    `  await github.rest.issues.create({ owner: context.repo.owner, repo: context.repo.repo, title, body, labels: ['drift-detection', '${stage}'] });`,\n    '} else {',\n    '  const issue = issues.data[0];',\n    '  await github.rest.issues.createComment({ owner: context.repo.owner, repo: context.repo.repo, issue_number: issue.number, body });',\n    '}',\n  ];\n  return lines.join('\\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    '    stage=$(basename \"$file\" | sed -E \\\"s/^drift-results-([^.]+)\\\\.json$/\\\\1/\\\")',\n    '    echo \"### Stage: $stage\" >> $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"]}
|
|
250
|
+
//# 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\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 stage choice\n    const stageChoices = props.stacks.map((s) => s.stackName ?? stageFromStack(s.stackName));\n    workflow.on({\n      schedule: props.schedule ? [{ cron: props.schedule }] : undefined,\n      workflowDispatch: {\n        inputs: {\n          stage: {\n            description: 'Stage to check for drift (leave empty for all)',\n            required: false,\n            type: 'choice',\n            options: stageChoices,\n          },\n        },\n      },\n    });\n\n    // One job per stage\n    const jobs: Record<string, any> = {};\n    for (const stack of props.stacks) {\n      const short = stack.stackName ?? stageFromStack(stack.stackName);\n      const jobId = `drift-${short}`;\n      const resultsFile = `drift-results-${short}.json`;\n      const innerCond = \"github.event_name == 'schedule' || !github.event.inputs.stage || github.event.inputs.stage == '\" + short + \"'\";\n      const condExpr = \"${{ \" + innerCond + \" }}\";\n      const notCondExpr = \"${{ !(\" + innerCond + \") }}\";\n\n      jobs[jobId] = {\n        name: `Drift Detection - ${short}`,\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          STAGE_NAME: short,\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 (stage not selected)', if: notCondExpr, run: 'echo \"Stage 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              'if [ -f \"$DRIFT_DETECTION_OUTPUT\" ]; then echo \"Results file created: $DRIFT_DETECTION_OUTPUT\"; fi',\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-${short}`, path: resultsFile, ifNoFilesFound: 'ignore' },\n          },\n          ...(\n            createIssues\n              ? [\n                {\n                  name: 'Create Issue on Drift',\n                  if: \"always() && steps.drift.outcome == 'failure'\",\n                  uses: `actions/github-script@${githubActionsGithubScriptVersion}`,\n                  with: { script: issueScript(short, stack.driftDetectionRoleToAssumeRegion, resultsFile) },\n                },\n              ]\n              : []\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\nfunction stageFromStack(stackName: string): string {\n  const parts = stackName.split('-');\n  return parts[parts.length - 1] || stackName;\n}\n\nfunction issueScript(stage: 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; }\",\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; }\",\n    `const title = 'Drift Detected in ${stage}';`,\n    `let body = '## Drift Detection Report\\\\n\\\\n' + '**Stage:** ${stage}\\\\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', '${stage}'] });`,\n    'if (issues.data.length === 0) {',\n    `  await github.rest.issues.create({ owner: context.repo.owner, repo: context.repo.repo, title, body, labels: ['drift-detection', '${stage}'] });`,\n    '} else {',\n    '  const issue = issues.data[0];',\n    '  await github.rest.issues.createComment({ owner: context.repo.owner, repo: context.repo.repo, issue_number: issue.number, body });',\n    '}',\n  ];\n  return lines.join('\\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    '    stage=$(basename \"$file\" | sed -E \\\"s/^drift-results-([^.]+)\\\\.json$/\\\\1/\\\")',\n    '    echo \"### Stage: $stage\" >> $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"]}
|
|
@@ -52,6 +52,7 @@ class CdkDriftIamTemplate {
|
|
|
52
52
|
' - cloudformation:DescribeStackResourceDrifts',
|
|
53
53
|
' - cloudformation:DescribeStacks',
|
|
54
54
|
' - cloudformation:ListStackResources',
|
|
55
|
+
' - cloudformation:DetectStackResourceDrift',
|
|
55
56
|
" Resource: '*'",
|
|
56
57
|
'',
|
|
57
58
|
'Outputs:',
|
|
@@ -73,4 +74,4 @@ class CdkDriftIamTemplate {
|
|
|
73
74
|
exports.CdkDriftIamTemplate = CdkDriftIamTemplate;
|
|
74
75
|
_a = JSII_RTTI_SYMBOL_1;
|
|
75
76
|
CdkDriftIamTemplate[_a] = { fqn: "@jjrawlins/cdk-diff-pr-github-action.CdkDriftIamTemplate", version: "0.0.0" };
|
|
76
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
77
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiQ2RrRHJpZnRJYW1UZW1wbGF0ZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uL3NyYy9DZGtEcmlmdElhbVRlbXBsYXRlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7O0FBQUEsbUNBQWtDO0FBVWxDLE1BQWEsbUJBQW1CO0lBQzlCLFlBQVksS0FBK0I7UUFDekMsTUFBTSxVQUFVLEdBQUcsS0FBSyxDQUFDLFVBQVUsSUFBSSxzQ0FBc0MsQ0FBQztRQUU5RSxLQUFLLENBQUMsT0FBTyxDQUFDLE9BQU8sQ0FBQyw4QkFBOEIsRUFBRTtZQUNwRCxXQUFXLEVBQ1QsdUlBQXVJO1lBQ3pJLFdBQVcsRUFBRSxJQUFJO1lBQ2pCLElBQUksRUFDRiw2Q0FBNkMsVUFBVSwrRUFBK0U7U0FDekksQ0FBQyxDQUFDO1FBRUgsSUFBSSxpQkFBUSxDQUFDLEtBQUssQ0FBQyxPQUFPLEVBQUUsVUFBVSxFQUFFO1lBQ3RDLEtBQUssRUFBRTtnQkFDTCx3Q0FBd0M7Z0JBQ3hDLDBEQUEwRDtnQkFDMUQsRUFBRTtnQkFDRixhQUFhO2dCQUNiLHNCQUFzQjtnQkFDdEIsa0JBQWtCO2dCQUNsQix5RkFBeUY7Z0JBQ3pGLGlCQUFpQixLQUFLLENBQUMsV0FBVyxHQUFHO2dCQUNyQyxFQUFFO2dCQUNGLFlBQVk7Z0JBQ1osOEZBQThGO2dCQUM5RixpQkFBaUI7Z0JBQ2pCLDBCQUEwQjtnQkFDMUIsaUJBQWlCO2dCQUNqQixtQkFBbUIsR0FBRyxLQUFLLENBQUMsUUFBUSxHQUFHLEdBQUc7Z0JBQzFDLGlDQUFpQztnQkFDakMsK0JBQStCO2dCQUMvQixvQkFBb0I7Z0JBQ3BCLDJCQUEyQjtnQkFDM0Isd0JBQXdCO2dCQUN4QiwyQ0FBMkM7Z0JBQzNDLG9DQUFvQztnQkFDcEMsd0JBQXdCO2dCQUN4Qiw2QkFBNkI7Z0JBQzdCLHdDQUF3QyxHQUFHLEtBQUssQ0FBQyxVQUFVLEdBQUcsR0FBRztnQkFDakUsaUJBQWlCO2dCQUNqQixpREFBaUQ7Z0JBQ2pELDJCQUEyQjtnQkFDM0IsbUNBQW1DO2dCQUNuQyx3QkFBd0I7Z0JBQ3hCLDJEQUEyRDtnQkFDM0QsK0JBQStCO2dCQUMvQix5QkFBeUI7Z0JBQ3pCLHFEQUFxRDtnQkFDckQsc0VBQXNFO2dCQUN0RSxnRUFBZ0U7Z0JBQ2hFLG1EQUFtRDtnQkFDbkQsdURBQXVEO2dCQUN2RCw2REFBNkQ7Z0JBQzdELCtCQUErQjtnQkFDL0IsRUFBRTtnQkFDRixVQUFVO2dCQUNWLG9CQUFvQjtnQkFDcEIsd0RBQXdEO2dCQUN4RCxxQ0FBcUM7Z0JBQ3JDLGFBQWE7Z0JBQ2Isc0RBQXNEO2dCQUN0RCxFQUFFO2dCQUNGLHFCQUFxQjtnQkFDckIseURBQXlEO2dCQUN6RCw4QkFBOEI7Z0JBQzlCLGFBQWE7Z0JBQ2IsdURBQXVEO2FBQ3hEO1NBQ0YsQ0FBQyxDQUFDO0lBQ0wsQ0FBQzs7QUFyRUgsa0RBc0VDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgVGV4dEZpbGUgfSBmcm9tICdwcm9qZW4nO1xuXG5leHBvcnQgaW50ZXJmYWNlIENka0RyaWZ0SWFtVGVtcGxhdGVQcm9wcyB7XG4gIHJlYWRvbmx5IHByb2plY3Q6IGFueTtcbiAgcmVhZG9ubHkgcm9sZU5hbWU6IHN0cmluZztcbiAgcmVhZG9ubHkgb3V0cHV0UGF0aD86IHN0cmluZztcbiAgcmVhZG9ubHkgb2lkY1JvbGVBcm46IHN0cmluZztcbiAgcmVhZG9ubHkgb2lkY1JlZ2lvbjogc3RyaW5nO1xufVxuXG5leHBvcnQgY2xhc3MgQ2RrRHJpZnRJYW1UZW1wbGF0ZSB7XG4gIGNvbnN0cnVjdG9yKHByb3BzOiBDZGtEcmlmdElhbVRlbXBsYXRlUHJvcHMpIHtcbiAgICBjb25zdCBvdXRwdXRQYXRoID0gcHJvcHMub3V0cHV0UGF0aCA/PyAnY2RrLWRyaWZ0LXdvcmtmbG93LWlhbS10ZW1wbGF0ZS55YW1sJztcblxuICAgIHByb3BzLnByb2plY3QuYWRkVGFzaygnZGVwbG95LWNka2RyaWZ0LWlhbS10ZW1wbGF0ZScsIHtcbiAgICAgIGRlc2NyaXB0aW9uOlxuICAgICAgICAnRGVwbG95IHRoZSBDREsgRHJpZnQgRGV0ZWN0aW9uIElBTSB0ZW1wbGF0ZSB2aWEgQ2xvdWRGb3JtYXRpb24gKGFjY2VwdHMgZXh0cmEgQVdTIENMSSBhcmdzLCBlLmcuLCAtLXBhcmFtZXRlci1vdmVycmlkZXMgS2V5PVZhbHVlLi4uKScsXG4gICAgICByZWNlaXZlQXJnczogdHJ1ZSxcbiAgICAgIGV4ZWM6XG4gICAgICAgIGBhd3MgY2xvdWRmb3JtYXRpb24gZGVwbG95IC0tdGVtcGxhdGUtZmlsZSAke291dHB1dFBhdGh9IC0tc3RhY2stbmFtZSBjZGstZHJpZnQtd29ya2Zsb3ctaWFtLXJvbGUgLS1jYXBhYmlsaXRpZXMgQ0FQQUJJTElUWV9OQU1FRF9JQU1gLFxuICAgIH0pO1xuXG4gICAgbmV3IFRleHRGaWxlKHByb3BzLnByb2plY3QsIG91dHB1dFBhdGgsIHtcbiAgICAgIGxpbmVzOiBbXG4gICAgICAgIFwiQVdTVGVtcGxhdGVGb3JtYXRWZXJzaW9uOiAnMjAxMC0wOS0wOSdcIixcbiAgICAgICAgXCJEZXNjcmlwdGlvbjogJ0lBTSByb2xlIGZvciBDREsgRHJpZnQgRGV0ZWN0aW9uIFdvcmtmbG93J1wiLFxuICAgICAgICAnJyxcbiAgICAgICAgJ1BhcmFtZXRlcnM6JyxcbiAgICAgICAgJyAgR2l0SHViT0lEQ1JvbGVBcm46JyxcbiAgICAgICAgJyAgICBUeXBlOiBTdHJpbmcnLFxuICAgICAgICBcIiAgICBEZXNjcmlwdGlvbjogJ0FSTiBvZiB0aGUgZXhpc3RpbmcgR2l0SHViIE9JREMgcm9sZSB0aGF0IGNhbiBhc3N1bWUgdGhpcyBkcmlmdCByb2xlJ1wiLFxuICAgICAgICBgICAgIERlZmF1bHQ6ICcke3Byb3BzLm9pZGNSb2xlQXJufSdgLFxuICAgICAgICAnJyxcbiAgICAgICAgJ1Jlc291cmNlczonLFxuICAgICAgICAnICAjIENsb3VkRm9ybWF0aW9uIERyaWZ0IERldGVjdGlvbiBSb2xlIC0gbWluaW1hbCBwZXJtaXNzaW9ucyBmb3IgZHJpZnQgZGV0ZWN0aW9uIG9wZXJhdGlvbnMnLFxuICAgICAgICAnICBDZGtEcmlmdFJvbGU6JyxcbiAgICAgICAgJyAgICBUeXBlOiBBV1M6OklBTTo6Um9sZScsXG4gICAgICAgICcgICAgUHJvcGVydGllczonLFxuICAgICAgICBcIiAgICAgIFJvbGVOYW1lOiAnXCIgKyBwcm9wcy5yb2xlTmFtZSArIFwiJ1wiLFxuICAgICAgICAnICAgICAgQXNzdW1lUm9sZVBvbGljeURvY3VtZW50OicsXG4gICAgICAgIFwiICAgICAgICBWZXJzaW9uOiAnMjAxMi0xMC0xNydcIixcbiAgICAgICAgJyAgICAgICAgU3RhdGVtZW50OicsXG4gICAgICAgICcgICAgICAgICAgLSBFZmZlY3Q6IEFsbG93JyxcbiAgICAgICAgJyAgICAgICAgICAgIFByaW5jaXBhbDonLFxuICAgICAgICAnICAgICAgICAgICAgICBBV1M6ICFSZWYgR2l0SHViT0lEQ1JvbGVBcm4nLFxuICAgICAgICAnICAgICAgICAgICAgQWN0aW9uOiBzdHM6QXNzdW1lUm9sZScsXG4gICAgICAgICcgICAgICAgICAgICBDb25kaXRpb246JyxcbiAgICAgICAgJyAgICAgICAgICAgICAgU3RyaW5nRXF1YWxzOicsXG4gICAgICAgIFwiICAgICAgICAgICAgICAgIGF3czpSZXF1ZXN0ZWRSZWdpb246ICdcIiArIHByb3BzLm9pZGNSZWdpb24gKyBcIidcIixcbiAgICAgICAgJyAgICAgIFBvbGljaWVzOicsXG4gICAgICAgICcgICAgICAgIC0gUG9saWN5TmFtZTogQ2xvdWRGb3JtYXRpb25EcmlmdEFjY2VzcycsXG4gICAgICAgICcgICAgICAgICAgUG9saWN5RG9jdW1lbnQ6JyxcbiAgICAgICAgXCIgICAgICAgICAgICBWZXJzaW9uOiAnMjAxMi0xMC0xNydcIixcbiAgICAgICAgJyAgICAgICAgICAgIFN0YXRlbWVudDonLFxuICAgICAgICAnICAgICAgICAgICAgICAjIENsb3VkRm9ybWF0aW9uIGRyaWZ0IGRldGVjdGlvbiBvcGVyYXRpb25zJyxcbiAgICAgICAgJyAgICAgICAgICAgICAgLSBFZmZlY3Q6IEFsbG93JyxcbiAgICAgICAgJyAgICAgICAgICAgICAgICBBY3Rpb246JyxcbiAgICAgICAgJyAgICAgICAgICAgICAgICAgIC0gY2xvdWRmb3JtYXRpb246RGV0ZWN0U3RhY2tEcmlmdCcsXG4gICAgICAgICcgICAgICAgICAgICAgICAgICAtIGNsb3VkZm9ybWF0aW9uOkRlc2NyaWJlU3RhY2tEcmlmdERldGVjdGlvblN0YXR1cycsXG4gICAgICAgICcgICAgICAgICAgICAgICAgICAtIGNsb3VkZm9ybWF0aW9uOkRlc2NyaWJlU3RhY2tSZXNvdXJjZURyaWZ0cycsXG4gICAgICAgICcgICAgICAgICAgICAgICAgICAtIGNsb3VkZm9ybWF0aW9uOkRlc2NyaWJlU3RhY2tzJyxcbiAgICAgICAgJyAgICAgICAgICAgICAgICAgIC0gY2xvdWRmb3JtYXRpb246TGlzdFN0YWNrUmVzb3VyY2VzJyxcbiAgICAgICAgJyAgICAgICAgICAgICAgICAgIC0gY2xvdWRmb3JtYXRpb246RGV0ZWN0U3RhY2tSZXNvdXJjZURyaWZ0JyxcbiAgICAgICAgXCIgICAgICAgICAgICAgICAgUmVzb3VyY2U6ICcqJ1wiLFxuICAgICAgICAnJyxcbiAgICAgICAgJ091dHB1dHM6JyxcbiAgICAgICAgJyAgQ2RrRHJpZnRSb2xlQXJuOicsXG4gICAgICAgIFwiICAgIERlc2NyaXB0aW9uOiAnQVJOIG9mIHRoZSBDREsgZHJpZnQgZGV0ZWN0aW9uIHJvbGUnXCIsXG4gICAgICAgICcgICAgVmFsdWU6ICFHZXRBdHQgQ2RrRHJpZnRSb2xlLkFybicsXG4gICAgICAgICcgICAgRXhwb3J0OicsXG4gICAgICAgIFwiICAgICAgTmFtZTogIVN1YiAnJHtBV1M6OlN0YWNrTmFtZX0tQ2RrRHJpZnRSb2xlQXJuJ1wiLFxuICAgICAgICAnJyxcbiAgICAgICAgJyAgQ2RrRHJpZnRSb2xlTmFtZTonLFxuICAgICAgICBcIiAgICBEZXNjcmlwdGlvbjogJ05hbWUgb2YgdGhlIENESyBkcmlmdCBkZXRlY3Rpb24gcm9sZSdcIixcbiAgICAgICAgJyAgICBWYWx1ZTogIVJlZiBDZGtEcmlmdFJvbGUnLFxuICAgICAgICAnICAgIEV4cG9ydDonLFxuICAgICAgICBcIiAgICAgIE5hbWU6ICFTdWIgJyR7QVdTOjpTdGFja05hbWV9LUNka0RyaWZ0Um9sZU5hbWUnXCIsXG4gICAgICBdLFxuICAgIH0pO1xuICB9XG59XG4iXX0=
|