@jjrawlins/cdk-diff-pr-github-action 0.0.7-beta → 0.0.9-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.
@@ -30,33 +30,35 @@ class CdkDriftDetectionWorkflow {
30
30
  }
31
31
  const gh = project.github ?? new github_1.GitHub(project);
32
32
  const workflow = new github_1.GithubWorkflow(gh, name, { fileName });
33
- // triggers: schedule + manual dispatch with stage choice
34
- const stageChoices = props.stacks.map((s) => s.stackName ?? stageFromStack(s.stackName));
33
+ // triggers: schedule + manual dispatch with stack choice
34
+ const stackChoices = props.stacks.map((s) => s.stackName);
35
35
  workflow.on({
36
36
  schedule: props.schedule ? [{ cron: props.schedule }] : undefined,
37
37
  workflowDispatch: {
38
38
  inputs: {
39
- stage: {
40
- description: 'Stage to check for drift (leave empty for all)',
39
+ stack: {
40
+ description: 'Stack to check for drift (leave empty for all)',
41
41
  required: false,
42
42
  type: 'choice',
43
- options: stageChoices,
43
+ options: stackChoices,
44
44
  },
45
45
  },
46
46
  },
47
47
  });
48
- // One job per stage
48
+ // One job per stack
49
49
  const jobs = {};
50
- const postSteps = Array.isArray(props.postGitHubSteps) ? props.postGitHubSteps : [];
51
50
  for (const stack of props.stacks) {
52
- const short = stack.stackName ?? stageFromStack(stack.stackName);
53
- const jobId = `drift-${short}`;
54
- const resultsFile = `drift-results-${short}.json`;
55
- const innerCond = "github.event_name == 'schedule' || !github.event.inputs.stage || github.event.inputs.stage == '" + short + "'";
51
+ const sanitizedStackName = toKebabCase(toGithubJobId(stack.stackName));
52
+ const originalStackName = stack.stackName;
53
+ const jobId = `drift-${sanitizedStackName}`;
54
+ const resultsFile = `drift-results-${sanitizedStackName}.json`;
55
+ const innerCond = "github.event_name == 'schedule' || !github.event.inputs.stack || github.event.inputs.stack == '" + sanitizedStackName + "' || github.event.inputs.stack == '" + originalStackName + "'";
56
56
  const condExpr = '${{ ' + innerCond + ' }}';
57
57
  const notCondExpr = '${{ !(' + innerCond + ') }}';
58
+ const rawPost = props.postGitHubSteps;
59
+ const postSteps = typeof rawPost === 'function' ? rawPost({ stack: sanitizedStackName }) : (rawPost ?? []);
58
60
  jobs[jobId] = {
59
- name: `Drift Detection - ${short}`,
61
+ name: `Drift Detection - ${sanitizedStackName}`,
60
62
  runsOn: ['ubuntu-latest'],
61
63
  permissions: {
62
64
  contents: workflows_model_1.JobPermission.READ,
@@ -67,12 +69,12 @@ class CdkDriftDetectionWorkflow {
67
69
  AWS_DEFAULT_REGION: stack.driftDetectionRoleToAssumeRegion,
68
70
  AWS_REGION: stack.driftDetectionRoleToAssumeRegion,
69
71
  DRIFT_DETECTION_OUTPUT: resultsFile,
70
- STAGE_NAME: short,
72
+ STAGE_NAME: sanitizedStackName,
71
73
  STACK_NAME: stack.stackName,
72
74
  },
73
75
  // No job-level condition; we gate steps so the job always completes and summary can run
74
76
  steps: [
75
- { name: 'Skip (stage not selected)', if: notCondExpr, run: 'echo "Stage not selected; skipping drift detection for this job."' },
77
+ { name: 'Skip (stack not selected)', if: notCondExpr, run: 'echo "Stage not selected; skipping drift detection for this job."' },
76
78
  { name: 'Checkout', if: condExpr, uses: `actions/checkout@${githubActionsCheckoutVersion}` },
77
79
  {
78
80
  name: 'Setup Node.js',
@@ -127,7 +129,7 @@ class CdkDriftDetectionWorkflow {
127
129
  name: 'Upload results',
128
130
  if: condExpr,
129
131
  uses: `actions/upload-artifact@${githubActionsUploadArtifactVersion}`,
130
- with: { name: `drift-results-${short}`, path: resultsFile, ifNoFilesFound: 'ignore' },
132
+ with: { name: `drift-results-${sanitizedStackName}`, path: resultsFile, ifNoFilesFound: 'ignore' },
131
133
  },
132
134
  {
133
135
  name: 'Prepare notification payload',
@@ -136,7 +138,7 @@ class CdkDriftDetectionWorkflow {
136
138
  uses: `actions/github-script@${githubActionsGithubScriptVersion}`,
137
139
  with: {
138
140
  'result-encoding': 'string',
139
- 'script': notificationScript(short, stack.driftDetectionRoleToAssumeRegion, resultsFile),
141
+ 'script': notificationScript(sanitizedStackName, stack.driftDetectionRoleToAssumeRegion, resultsFile),
140
142
  },
141
143
  },
142
144
  ...postSteps.map((step) => ({
@@ -150,7 +152,7 @@ class CdkDriftDetectionWorkflow {
150
152
  name: 'Create Issue on Drift',
151
153
  if: "always() && steps.drift.outcome == 'failure'",
152
154
  uses: `actions/github-script@${githubActionsGithubScriptVersion}`,
153
- with: { script: issueScript(short, stack.driftDetectionRoleToAssumeRegion, resultsFile) },
155
+ with: { script: issueScript(sanitizedStackName, stack.driftDetectionRoleToAssumeRegion, resultsFile) },
154
156
  },
155
157
  ]
156
158
  : []),
@@ -179,10 +181,6 @@ exports.CdkDriftDetectionWorkflow = CdkDriftDetectionWorkflow;
179
181
  _a = JSII_RTTI_SYMBOL_1;
180
182
  CdkDriftDetectionWorkflow[_a] = { fqn: "@jjrawlins/cdk-diff-pr-github-action.CdkDriftDetectionWorkflow", version: "0.0.0" };
181
183
  CdkDriftDetectionWorkflow.scriptCreated = false;
182
- function stageFromStack(stackName) {
183
- const parts = stackName.split('-');
184
- return parts[parts.length - 1] || stackName;
185
- }
186
184
  function issueScript(stage, region, resultsFile) {
187
185
  // Construct a plain JS script string (no template string nesting mishaps)
188
186
  const lines = [
@@ -224,8 +222,8 @@ function notificationScript(stage, region, resultsFile) {
224
222
  `const resultsFile = '${resultsFile}';`,
225
223
  'let payload;',
226
224
  'if (!fs.existsSync(resultsFile)) {',
227
- ' payload = { text: `Drift Detection (${stage}) - No results found`, blocks: [',
228
- " { type: 'section', text: { type: 'mrkdwn', text: `*Drift Detection (${stage})*` } },",
225
+ ' payload = { text: `Drift Detection (${stack}) - No results found`, blocks: [',
226
+ " { type: 'section', text: { type: 'mrkdwn', text: `*Drift Detection (${stack})*` } },",
229
227
  " { type: 'section', text: { type: 'mrkdwn', text: 'No results file found. The detection step may have been skipped.' } }",
230
228
  ' ] };',
231
229
  ' return JSON.stringify(payload);',
@@ -241,7 +239,7 @@ function notificationScript(stage, region, resultsFile) {
241
239
  ' linesArr.push(`• ${s.stackName} — ${count} resource(s) drifted`);',
242
240
  '}',
243
241
  'payload = {',
244
- ' text: `Drift Detection (${stage})`,',
242
+ ' text: `Drift Detection (${stack})`,',
245
243
  ' blocks: [',
246
244
  " { type: 'section', text: { type: 'mrkdwn', text: header } },",
247
245
  " { type: 'context', elements: [{ type: 'mrkdwn', text: summary }] },",
@@ -272,8 +270,8 @@ function summaryScript() {
272
270
  'shopt -s nullglob',
273
271
  'for file in drift-results-*.json drift-results/*/drift-results-*.json; do',
274
272
  ' if [[ -f "$file" ]]; then',
275
- ' stage=$(basename "$file" | sed -E \"s/^drift-results-([^.]+)\\.json$/\\1/\")',
276
- ' echo "### Stage: $stage" >> $GITHUB_STEP_SUMMARY',
273
+ ' stack=$(basename "$file" | sed -E \"s/^drift-results-([^.]+)\\.json$/\\1/\")',
274
+ ' echo "### Stage: $stack" >> $GITHUB_STEP_SUMMARY',
277
275
  ' jq -r \'' +
278
276
  '. as $results |\n' +
279
277
  '"- Total stacks: " + ($results | length | tostring) + "\\n" +\n' +
@@ -303,4 +301,20 @@ function summaryScript() {
303
301
  function toKebabCase(s) {
304
302
  return s.replace(/[^a-zA-Z0-9]+/g, '-').replace(/^-+|-+$/g, '').toLowerCase();
305
303
  }
306
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CdkDriftDetectionWorkflow.js","sourceRoot":"","sources":["../src/CdkDriftDetectionWorkflow.ts"],"names":[],"mappings":"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","sourcesContent":["import { GitHub, GithubWorkflow } from 'projen/lib/github';\nimport { JobPermission } from 'projen/lib/github/workflows-model';\nimport { CdkDriftDetectionScript } from './bin/cdk-drift-detection-script';\n\nconst githubActionsAwsCredentialsVersion = 'v5';\nconst githubActionsCheckoutVersion = 'v5';\nconst githubActionsSetupNodeVersion = 'v5';\nconst githubActionsUploadArtifactVersion = 'v4';\nconst githubActionsDownloadArtifactVersion = 'v5';\nconst githubActionsGithubScriptVersion = 'v8';\n\nexport interface Stack {\n  readonly stackName: string;\n  readonly driftDetectionRoleToAssumeRegion: string;\n  readonly driftDetectionRoleToAssumeArn: string;\n  readonly failOnDrift?: boolean; // if true, fail job when drift detected (default true)\n}\n\nexport interface CdkDriftDetectionWorkflowProps {\n  readonly scriptOutputPath?: string;\n  readonly project: any; // avoid exporting projen types in public API\n  readonly workflowName?: string; // workflow workflowName (also used to derive file workflowName)\n  readonly schedule?: string; // cron expression, e.g. '0 0 * * *'\n  readonly createIssues?: boolean; // create/update issue when drift detected on schedule (default true)\n  readonly oidcRoleArn: string; // default OIDC role ARN to assume for all stacks\n  readonly oidcRegion: string; // default OIDC region to assume for all stacks\n  readonly stacks: Stack[];\n  readonly nodeVersion?: string; // e.g., '24.x'\n  /**\n   * Optional hook to append additional GitHub Actions steps after drift detection per stack.\n   * You can supply a static array of steps, or a factory that receives context and returns steps.\n   */\n  /**\n   * Optional additional GitHub Action steps to run after drift detection for each stack.\n   * These steps are inserted after a 'Prepare notification payload' step which exposes\n   * a Slack-compatible JSON payload at `${{ steps.notify.outputs.result }}`.\n   */\n  readonly postGitHubSteps?: any[];\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\n    const postSteps: any[] = Array.isArray((props as any).postGitHubSteps) ? (props as any).postGitHubSteps : [];\n\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            name: 'Prepare notification payload',\n            if: condExpr,\n            id: 'notify',\n            uses: `actions/github-script@${githubActionsGithubScriptVersion}`,\n            with: {\n              'result-encoding': 'string',\n              'script': notificationScript(short, stack.driftDetectionRoleToAssumeRegion, resultsFile),\n            },\n          },\n          ...postSteps.map((step) => ({\n            // By default, only run extra notification steps when drift occurs\n            if: (step as any).if ?? \"always() && steps.drift.outcome == 'failure'\",\n            ...(step as any),\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 notificationScript(stage: string, region: string, resultsFile: string): string {\n  const lines = [\n    \"const fs = require('fs');\",\n    `const resultsFile = '${resultsFile}';`,\n    'let payload;',\n    'if (!fs.existsSync(resultsFile)) {',\n    '  payload = { text: `Drift Detection (${stage}) - No results found`, blocks: [',\n    \"    { type: 'section', text: { type: 'mrkdwn', text: `*Drift Detection (${stage})*` } },\",\n    \"    { type: 'section', text: { type: 'mrkdwn', text: 'No results file found. The detection step may have been skipped.' } }\",\n    '  ] };',\n    '  return JSON.stringify(payload);',\n    '}',\n    \"const results = JSON.parse(fs.readFileSync(resultsFile, 'utf8'));\",\n    \"const drifted = results.filter(r => r.driftStatus === 'DRIFTED');\",\n    'const errors = results.filter(r => r.error);',\n    `const header = \\`*Drift Detection (${stage})* — Region: ${region}\\`;`,\n    'const summary = `Total stacks: ${results.length} | Drifted: ${drifted.length} | Errors: ${errors.length}`;',\n    'const linesArr = [];',\n    'for (const s of drifted) {',\n    '  const count = (s.driftedResources || []).length;',\n    '  linesArr.push(`• ${s.stackName} — ${count} resource(s) drifted`);',\n    '}',\n    'payload = {',\n    '  text: `Drift Detection (${stage})`,',\n    '  blocks: [',\n    \"    { type: 'section', text: { type: 'mrkdwn', text: header } },\",\n    \"    { type: 'context', elements: [{ type: 'mrkdwn', text: summary }] },\",\n    '    ...(drifted.length > 0 ? [',\n    \"      { type: 'divider' },\",\n    \"      { type: 'section', text: { type: 'mrkdwn', text: '*Drifted Stacks:*\\n' + linesArr.join('\\n') } }\",\n    '    ] : [',\n    \"      { type: 'divider' },\",\n    \"      { type: 'section', text: { type: 'mrkdwn', text: 'No drift detected.' } }\",\n    '    ])',\n    '  ]',\n    '};',\n    'return JSON.stringify(payload);',\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"]}
304
+ function toGithubJobId(s) {
305
+ // GitHub job_id must start with a letter or underscore and contain only A-Za-z0-9, '-', '_'
306
+ // 1) Replace any disallowed chars with '-'
307
+ let out = s.replace(/[^A-Za-z0-9_-]+/g, '-');
308
+ // 2) Collapse consecutive dashes
309
+ out = out.replace(/-+/g, '-');
310
+ // 3) Trim leading/trailing dashes (underscores are allowed at start)
311
+ out = out.replace(/^-+|-+$/g, '');
312
+ // 4) Lowercase for consistency (not required by GitHub but keeps things stable)
313
+ out = out.toLowerCase();
314
+ // 5) Ensure it starts with a letter or underscore
315
+ if (!out || !/^[a-z_]/i.test(out)) {
316
+ out = `s-${out}`;
317
+ }
318
+ return out;
319
+ }
320
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CdkDriftDetectionWorkflow.js","sourceRoot":"","sources":["../src/CdkDriftDetectionWorkflow.ts"],"names":[],"mappings":"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","sourcesContent":["import { GitHub, GithubWorkflow } from 'projen/lib/github';\nimport { JobPermission } from 'projen/lib/github/workflows-model';\nimport { CdkDriftDetectionScript } from './bin/cdk-drift-detection-script';\n\nconst githubActionsAwsCredentialsVersion = 'v5';\nconst githubActionsCheckoutVersion = 'v5';\nconst githubActionsSetupNodeVersion = 'v5';\nconst githubActionsUploadArtifactVersion = 'v4';\nconst githubActionsDownloadArtifactVersion = 'v5';\nconst githubActionsGithubScriptVersion = 'v8';\n\nexport interface Stack {\n  readonly stackName: string;\n  readonly driftDetectionRoleToAssumeRegion: string;\n  readonly driftDetectionRoleToAssumeArn: string;\n  readonly failOnDrift?: boolean; // if true, fail job when drift detected (default true)\n}\n\nexport interface CdkDriftDetectionWorkflowProps {\n  readonly scriptOutputPath?: string;\n  readonly project: any; // avoid exporting projen types in public API\n  readonly workflowName?: string; // workflow workflowName (also used to derive file workflowName)\n  readonly schedule?: string; // cron expression, e.g. '0 0 * * *'\n  readonly createIssues?: boolean; // create/update issue when drift detected on schedule (default true)\n  readonly oidcRoleArn: string; // default OIDC role ARN to assume for all stacks\n  readonly oidcRegion: string; // default OIDC region to assume for all stacks\n  readonly stacks: Stack[];\n  readonly nodeVersion?: string; // e.g., '24.x'\n  /**\n   * Optional hook to append additional GitHub Actions steps after drift detection per stack.\n   * You can supply a static array of steps, or a factory that receives context and returns steps.\n   */\n  /**\n   * Optional additional GitHub Action steps to run after drift detection for each stack.\n   * These steps are inserted after a 'Prepare notification payload' step which exposes\n   * a Slack-compatible JSON payload at `${{ steps.notify.outputs.result }}`.\n   */\n  // NOTE: jsii does not support function types in public APIs; use 'any' here and accept either:\n  // - An array of GitHub steps, or\n  // - A function (ctx: { stack: string }) => GitHubStep[]\n  // The constructor handles both at runtime.\n  readonly postGitHubSteps?: any;\n}\n\ntype GitHubStep = {\n  name?: string;\n  id?: string;\n  if?: string;\n  uses?: string;\n  run?: string;\n  with?: Record<string, any>;\n  env?: Record<string, string>;\n  continueOnError?: boolean;\n  shell?: string;\n};\n\nexport class CdkDriftDetectionWorkflow {\n  private static scriptCreated = false;\n\n  constructor(props: CdkDriftDetectionWorkflowProps) {\n    const name = props.workflowName ?? 'drift-detection';\n    const fileName = toKebabCase(name) + '.yml';\n    const nodeVersion = props.nodeVersion ?? '24.x';\n    const createIssues = props.createIssues ?? true;\n    const project = props.project;\n    const scriptOutputPath= props.scriptOutputPath ?? '.github/workflows/scripts/detect-drift.ts';\n\n    // Only create the drift detection script once to avoid collisions\n    if (!CdkDriftDetectionWorkflow.scriptCreated) {\n      new CdkDriftDetectionScript({\n        project: props.project,\n        outputPath: scriptOutputPath,\n      });\n      CdkDriftDetectionWorkflow.scriptCreated = true;\n    }\n\n    const gh = (project as any).github ?? new GitHub(project);\n    const workflow = new GithubWorkflow(gh, name, { fileName });\n\n    // triggers: schedule + manual dispatch with stack choice\n    const stackChoices = props.stacks.map((s) => s.stackName);\n    workflow.on({\n      schedule: props.schedule ? [{ cron: props.schedule }] : undefined,\n      workflowDispatch: {\n        inputs: {\n          stack: {\n            description: 'Stack to check for drift (leave empty for all)',\n            required: false,\n            type: 'choice',\n            options: stackChoices,\n          },\n        },\n      },\n    });\n\n    // One job per stack\n    const jobs: Record<string, any> = {};\n\n    for (const stack of props.stacks) {\n      const sanitizedStackName = toKebabCase(toGithubJobId(stack.stackName));\n      const originalStackName = stack.stackName;\n      const jobId = `drift-${sanitizedStackName}`;\n      const resultsFile = `drift-results-${sanitizedStackName}.json`;\n      const innerCond = \"github.event_name == 'schedule' || !github.event.inputs.stack || github.event.inputs.stack == '\" + sanitizedStackName + \"' || github.event.inputs.stack == '\" + originalStackName + \"'\";\n      const condExpr = '${{ ' + innerCond + ' }}';\n      const notCondExpr = '${{ !(' + innerCond + ') }}';\n\n      const rawPost = props.postGitHubSteps;\n      const postSteps: GitHubStep[] = typeof rawPost === 'function' ? (rawPost as (ctx: { stack: string }) => GitHubStep[])({ stack: sanitizedStackName }) : (rawPost ?? []);\n\n      jobs[jobId] = {\n        name: `Drift Detection - ${sanitizedStackName}`,\n        runsOn: ['ubuntu-latest'],\n        permissions: {\n          contents: JobPermission.READ,\n          idToken: JobPermission.WRITE,\n          issues: JobPermission.WRITE,\n        },\n        env: {\n          AWS_DEFAULT_REGION: stack.driftDetectionRoleToAssumeRegion,\n          AWS_REGION: stack.driftDetectionRoleToAssumeRegion,\n          DRIFT_DETECTION_OUTPUT: resultsFile,\n          STAGE_NAME: sanitizedStackName,\n          STACK_NAME: stack.stackName,\n        },\n        // No job-level condition; we gate steps so the job always completes and summary can run\n        steps: [\n          { name: 'Skip (stack not selected)', if: notCondExpr, run: 'echo \"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-${sanitizedStackName}`, path: resultsFile, ifNoFilesFound: 'ignore' },\n          },\n          {\n            name: 'Prepare notification payload',\n            if: condExpr,\n            id: 'notify',\n            uses: `actions/github-script@${githubActionsGithubScriptVersion}`,\n            with: {\n              'result-encoding': 'string',\n              'script': notificationScript(sanitizedStackName, stack.driftDetectionRoleToAssumeRegion, resultsFile),\n            },\n          },\n          ...postSteps.map((step) => ({\n            // By default, only run extra notification steps when drift occurs\n            if: (step as any).if ?? \"always() && steps.drift.outcome == 'failure'\",\n            ...(step as any),\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(sanitizedStackName, 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\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 notificationScript(stage: string, region: string, resultsFile: string): string {\n  const lines = [\n    \"const fs = require('fs');\",\n    `const resultsFile = '${resultsFile}';`,\n    'let payload;',\n    'if (!fs.existsSync(resultsFile)) {',\n    '  payload = { text: `Drift Detection (${stack}) - No results found`, blocks: [',\n    \"    { type: 'section', text: { type: 'mrkdwn', text: `*Drift Detection (${stack})*` } },\",\n    \"    { type: 'section', text: { type: 'mrkdwn', text: 'No results file found. The detection step may have been skipped.' } }\",\n    '  ] };',\n    '  return JSON.stringify(payload);',\n    '}',\n    \"const results = JSON.parse(fs.readFileSync(resultsFile, 'utf8'));\",\n    \"const drifted = results.filter(r => r.driftStatus === 'DRIFTED');\",\n    'const errors = results.filter(r => r.error);',\n    `const header = \\`*Drift Detection (${stage})* — Region: ${region}\\`;`,\n    'const summary = `Total stacks: ${results.length} | Drifted: ${drifted.length} | Errors: ${errors.length}`;',\n    'const linesArr = [];',\n    'for (const s of drifted) {',\n    '  const count = (s.driftedResources || []).length;',\n    '  linesArr.push(`• ${s.stackName} — ${count} resource(s) drifted`);',\n    '}',\n    'payload = {',\n    '  text: `Drift Detection (${stack})`,',\n    '  blocks: [',\n    \"    { type: 'section', text: { type: 'mrkdwn', text: header } },\",\n    \"    { type: 'context', elements: [{ type: 'mrkdwn', text: summary }] },\",\n    '    ...(drifted.length > 0 ? [',\n    \"      { type: 'divider' },\",\n    \"      { type: 'section', text: { type: 'mrkdwn', text: '*Drifted Stacks:*\\n' + linesArr.join('\\n') } }\",\n    '    ] : [',\n    \"      { type: 'divider' },\",\n    \"      { type: 'section', text: { type: 'mrkdwn', text: 'No drift detected.' } }\",\n    '    ])',\n    '  ]',\n    '};',\n    'return JSON.stringify(payload);',\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    '    stack=$(basename \"$file\" | sed -E \\\"s/^drift-results-([^.]+)\\\\.json$/\\\\1/\\\")',\n    '    echo \"### Stage: $stack\" >> $GITHUB_STEP_SUMMARY',\n    '    jq -r \\'' +\n      '. as $results |\\n' +\n      '\"- Total stacks: \" + ($results | length | tostring) + \"\\\\n\" +\\n' +\n      '\"- Drifted: \" + ([.[] | select(.driftStatus == \"DRIFTED\")] | length | tostring) + \"\\\\n\" +\\n' +\n      '\"- Errors: \" + ([.[] | select(.error)] | length | tostring) + \"\\\\n\" +\\n' +\n      '([.[] | select(.driftStatus == \"DRIFTED\")] | if length > 0 then \"\\\\n**Drifted stacks:**\\\\n\" + (map(\"  - \" + .stackName + \" (\" + ((.driftedResources // []) | length | tostring) + \" resources)\") | join(\"\\\\n\")) else \"\" end)\\n' +\n    '\\'' +\n    ' \"$file\" >> $GITHUB_STEP_SUMMARY',\n    '    echo \"\" >> $GITHUB_STEP_SUMMARY',\n    '    total_stacks=$((total_stacks + $(jq \\\"length\\\" \\\"$file\\\")))',\n    '    total_drifted=$((total_drifted + $(jq \\\"[.[] | select(.driftStatus == \\\\\\\"DRIFTED\\\\\\\")] | length\\\" \\\"$file\\\")))',\n    '    total_errors=$((total_errors + $(jq \\\"[.[] | select(.error)] | length\\\" \\\"$file\\\")))',\n    '  fi',\n    'done',\n    '',\n    'echo \"### Overall Summary\" >> $GITHUB_STEP_SUMMARY',\n    'echo \"- Total stacks checked: $total_stacks\" >> $GITHUB_STEP_SUMMARY',\n    'echo \"- Total drifted stacks: $total_drifted\" >> $GITHUB_STEP_SUMMARY',\n    'echo \"- Total errors: $total_errors\" >> $GITHUB_STEP_SUMMARY',\n    '',\n    'if [[ $total_drifted -gt 0 ]]; then',\n    '  echo \"\" >> $GITHUB_STEP_SUMMARY',\n    '  echo \"⚠️ **Action required:** Drift detected in $total_drifted stacks\" >> $GITHUB_STEP_SUMMARY',\n    'fi',\n  ].join('\\n');\n}\n\nfunction toKebabCase(s: string): string {\n  return s.replace(/[^a-zA-Z0-9]+/g, '-').replace(/^-+|-+$/g, '').toLowerCase();\n}\n\n\nfunction toGithubJobId(s: string): string {\n  // GitHub job_id must start with a letter or underscore and contain only A-Za-z0-9, '-', '_'\n  // 1) Replace any disallowed chars with '-'\n  let out = s.replace(/[^A-Za-z0-9_-]+/g, '-');\n  // 2) Collapse consecutive dashes\n  out = out.replace(/-+/g, '-');\n  // 3) Trim leading/trailing dashes (underscores are allowed at start)\n  out = out.replace(/^-+|-+$/g, '');\n  // 4) Lowercase for consistency (not required by GitHub but keeps things stable)\n  out = out.toLowerCase();\n  // 5) Ensure it starts with a letter or underscore\n  if (!out || !/^[a-z_]/i.test(out)) {\n    out = `s-${out}`;\n  }\n  return out;\n}\n"]}
package/package.json CHANGED
@@ -102,7 +102,7 @@
102
102
  },
103
103
  "main": "lib/index.js",
104
104
  "license": "Apache-2.0",
105
- "version": "0.0.7-beta",
105
+ "version": "0.0.9-beta",
106
106
  "jest": {
107
107
  "coverageProvider": "v8",
108
108
  "testMatch": [