@jjrawlins/cdk-diff-pr-github-action 1.8.0 → 1.9.0

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.
@@ -61,8 +61,11 @@ class CdkDriftDetectionWorkflow {
61
61
  const innerCond = "github.event_name == 'schedule' || !github.event.inputs.stack || github.event.inputs.stack == 'all' || github.event.inputs.stack == '" + sanitizedStackName + "' || github.event.inputs.stack == '" + originalStackName + "'";
62
62
  const condExpr = '${{ ' + innerCond + ' }}';
63
63
  const notCondExpr = '${{ !(' + innerCond + ') }}';
64
+ const stepCtx = { stack: sanitizedStackName, workingDirectory: wd };
65
+ const rawPre = props.preGitHubSteps;
66
+ const preSteps = typeof rawPre === 'function' ? rawPre(stepCtx) : (rawPre ?? []);
64
67
  const rawPost = props.postGitHubSteps;
65
- const postSteps = typeof rawPost === 'function' ? rawPost({ stack: sanitizedStackName }) : (rawPost ?? []);
68
+ const postSteps = typeof rawPost === 'function' ? rawPost(stepCtx) : (rawPost ?? []);
66
69
  // Prefix results file path for uses: steps (which ignore defaults.run.working-directory)
67
70
  const artifactResultsPath = wd ? `${wd}/${resultsFile}` : resultsFile;
68
71
  const issueResultsPath = wd ? `${wd}/${resultsFile}` : resultsFile;
@@ -93,6 +96,11 @@ class CdkDriftDetectionWorkflow {
93
96
  with: { 'node-version': nodeVersion },
94
97
  },
95
98
  { name: 'Install dependencies', if: condExpr, run: 'yarn install --frozen-lockfile || npm ci', env: { GITHUB_TOKEN: '${{ secrets.GITHUB_TOKEN }}' } },
99
+ ...preSteps.map((step) => {
100
+ const s = { ...step };
101
+ s.if = s.if ?? condExpr;
102
+ return s;
103
+ }),
96
104
  {
97
105
  name: 'AWS Credentials',
98
106
  if: condExpr,
@@ -200,7 +208,7 @@ class CdkDriftDetectionWorkflow {
200
208
  }
201
209
  exports.CdkDriftDetectionWorkflow = CdkDriftDetectionWorkflow;
202
210
  _a = JSII_RTTI_SYMBOL_1;
203
- CdkDriftDetectionWorkflow[_a] = { fqn: "@jjrawlins/cdk-diff-pr-github-action.CdkDriftDetectionWorkflow", version: "1.8.0" };
211
+ CdkDriftDetectionWorkflow[_a] = { fqn: "@jjrawlins/cdk-diff-pr-github-action.CdkDriftDetectionWorkflow", version: "1.9.0" };
204
212
  CdkDriftDetectionWorkflow.scriptCreated = false;
205
213
  function issueScript(stack, region, resultsFile) {
206
214
  // Construct a plain JS script string (no template string nesting mishaps)
@@ -314,4 +322,4 @@ function toGithubJobId(s) {
314
322
  }
315
323
  return out;
316
324
  }
317
- //# 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\nconst MAX_WORKFLOW_FILENAME_LENGTH = 255;\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  readonly oidcRoleArn?: string; // Optional override for OIDC role\n  readonly oidcRegion?: string; // Optional override for OIDC region\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 (or each stack must have its own)\n  readonly oidcRegion?: string; // default OIDC region to assume for all stacks (or each stack must have its own)\n  readonly stacks: Stack[];\n  readonly nodeVersion?: string; // e.g., '24.x'\n  /**\n   * Working directory for the CDK app, relative to the repository root.\n   * Useful for monorepos where infrastructure lives in a subdirectory (e.g., 'infra').\n   *\n   * When set, all workflow run steps will use `defaults.run.working-directory`\n   * and artifact/script paths will be adjusted accordingly.\n   *\n   * @default - repository root\n   */\n  readonly workingDirectory?: string;\n  /**\n   * Optional hook to append additional GitHub Actions steps after drift detection per stack.\n   * You can supply a static array of steps, or a factory that receives context and returns steps.\n   */\n  /**\n   * Optional additional GitHub Action steps to run after drift detection for each stack.\n   * These steps run after results are uploaded for each stack. You can include\n   * any notifications you like (e.g., Slack). Provide explicit inputs (e.g., payload/markdown)\n   * directly in your step without relying on a pre-generated payload.\n   */\n  // NOTE: jsii does not support function types in public APIs; use 'any' here and accept either:\n  // - An array of GitHub steps, or\n  // - A function (ctx: { stack: string }) => GitHubStep[]\n  // The constructor handles both at runtime.\n  readonly postGitHubSteps?: any;\n}\n\ntype GitHubStep = {\n  name?: string;\n  id?: string;\n  if?: string;\n  uses?: string;\n  run?: string;\n  with?: Record<string, any>;\n  env?: Record<string, string>;\n  continueOnError?: boolean;\n  shell?: string;\n};\n\nexport class CdkDriftDetectionWorkflow {\n  private static scriptCreated = false;\n\n  constructor(props: CdkDriftDetectionWorkflowProps) {\n    // Validate OIDC configuration\n    this.validateOidcConfiguration(props);\n\n    const name = props.workflowName ?? 'drift-detection';\n    const fileName = buildWorkflowFileName(toKebabCase(name));\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    // Normalize working directory: strip trailing slashes\n    const wd = props.workingDirectory?.replace(/\\/+$/, '');\n    const defaults = wd ? { run: { workingDirectory: wd } } : undefined;\n\n    // Only create the drift detection script once to avoid collisions\n    if (!CdkDriftDetectionWorkflow.scriptCreated) {\n      new CdkDriftDetectionScript({\n        project: props.project,\n        outputPath: scriptOutputPath,\n      });\n      CdkDriftDetectionWorkflow.scriptCreated = true;\n    }\n\n    const gh = (project as any).github ?? new GitHub(project);\n    const workflow = new GithubWorkflow(gh, name, { fileName });\n\n    // triggers: schedule + manual dispatch with stack choice\n    const stackChoices = ['all', ...props.stacks.map((s) => s.stackName)];\n    workflow.on({\n      schedule: props.schedule ? [{ cron: props.schedule }] : undefined,\n      workflowDispatch: {\n        inputs: {\n          stack: {\n            description: \"Stack to check for drift ('all' to run every stack)\",\n            required: false,\n            type: 'choice',\n            options: stackChoices,\n          },\n        },\n      },\n    });\n\n    // One job per stack\n    const jobs: Record<string, any> = {};\n\n    for (const stack of props.stacks) {\n      const sanitizedStackName = toKebabCase(toGithubJobId(stack.stackName));\n      const originalStackName = stack.stackName;\n      const jobId = `drift-${sanitizedStackName}`;\n      const resultsFile = `drift-results-${sanitizedStackName}.json`;\n      const innerCond = \"github.event_name == 'schedule' || !github.event.inputs.stack || github.event.inputs.stack == 'all' || github.event.inputs.stack == '\" + sanitizedStackName + \"' || github.event.inputs.stack == '\" + originalStackName + \"'\";\n      const condExpr = '${{ ' + innerCond + ' }}';\n      const notCondExpr = '${{ !(' + innerCond + ') }}';\n\n      const rawPost = props.postGitHubSteps;\n      const postSteps: GitHubStep[] = typeof rawPost === 'function' ? (rawPost as (ctx: { stack: string }) => GitHubStep[])({ stack: sanitizedStackName }) : (rawPost ?? []);\n\n      // Prefix results file path for uses: steps (which ignore defaults.run.working-directory)\n      const artifactResultsPath = wd ? `${wd}/${resultsFile}` : resultsFile;\n      const issueResultsPath = wd ? `${wd}/${resultsFile}` : resultsFile;\n\n      jobs[jobId] = {\n        name: `Drift Detection - ${sanitizedStackName}`,\n        runsOn: ['ubuntu-latest'],\n        ...(defaults && { defaults }),\n        permissions: {\n          contents: JobPermission.READ,\n          idToken: JobPermission.WRITE,\n          issues: JobPermission.WRITE,\n        },\n        env: {\n          AWS_DEFAULT_REGION: stack.driftDetectionRoleToAssumeRegion,\n          AWS_REGION: stack.driftDetectionRoleToAssumeRegion,\n          DRIFT_DETECTION_OUTPUT: resultsFile,\n          STACK_ID: sanitizedStackName,\n          STACK_NAME: stack.stackName,\n        },\n        // No job-level condition; we gate steps so the job always completes and summary can run\n        steps: [\n          { name: 'Skip (stack not selected)', if: notCondExpr, run: 'echo \"Stack not selected; skipping drift detection for this job.\"' },\n          { name: 'Checkout', if: condExpr, uses: `actions/checkout@${githubActionsCheckoutVersion}` },\n          {\n            name: 'Setup Node.js',\n            if: condExpr,\n            uses: `actions/setup-node@${githubActionsSetupNodeVersion}`,\n            with: { 'node-version': nodeVersion },\n          },\n          { name: 'Install dependencies', if: condExpr, run: 'yarn install --frozen-lockfile || npm ci', env: { GITHUB_TOKEN: '${{ secrets.GITHUB_TOKEN }}' } },\n          {\n            name: 'AWS Credentials',\n            if: condExpr,\n            id: 'creds',\n            uses: `aws-actions/configure-aws-credentials@${githubActionsAwsCredentialsVersion}`,\n            with: {\n              'role-to-assume': stack.oidcRoleArn ?? props.oidcRoleArn,\n              'role-session-name': 'GitHubAction',\n              'aws-region': stack.oidcRegion ?? props.oidcRegion,\n            },\n          },\n          {\n            name: 'Assume Drift Detection Role',\n            if: condExpr,\n            uses: `aws-actions/configure-aws-credentials@${githubActionsAwsCredentialsVersion}`,\n            with: {\n              'role-to-assume': stack.driftDetectionRoleToAssumeArn,\n              'role-chaining': true,\n              'role-skip-session-tagging': true,\n              'aws-region': stack.driftDetectionRoleToAssumeRegion,\n              'aws-access-key-id': '${{ steps.creds.outputs.aws-access-key-id }}',\n              'aws-secret-access-key': '${{ steps.creds.outputs.aws-secret-access-key }}',\n              'aws-session-token': '${{ steps.creds.outputs.aws-session-token }}',\n            },\n          },\n          {\n            name: 'Detect drift',\n            if: condExpr,\n            id: 'drift',\n            continueOnError: true, // allow artifact upload and issue creation even when drift is detected\n            run: [\n              'set +e',\n              // Use the bundled script from this package\n              'node ./node_modules/@jjrawlins/cdk-diff-pr-github-action/lib/bin/detect-drift.js',\n              'status=$?',\n              'if [ -f \"$DRIFT_DETECTION_OUTPUT\" ]; then echo \"Results file created: $DRIFT_DETECTION_OUTPUT\"; fi',\n              // Expose useful outputs for downstream steps\n              \"STACK_ARN=$(aws cloudformation describe-stacks --stack-name \\\"$STACK_NAME\\\" --query 'Stacks[0].StackId' --output text 2>/dev/null || true)\",\n              'echo \"stack-arn=$STACK_ARN\" >> \"$GITHUB_OUTPUT\"',\n              'exit $status',\n            ].join('\\n'),\n            env: {\n              STACK_NAME: stack.stackName,\n              AWS_REGION: stack.driftDetectionRoleToAssumeRegion,\n              DRIFT_DETECTION_OUTPUT: resultsFile,\n            },\n          },\n          {\n            name: 'Upload results',\n            if: condExpr,\n            uses: `actions/upload-artifact@${githubActionsUploadArtifactVersion}`,\n            with: { 'name': `drift-results-${sanitizedStackName}`, 'path': artifactResultsPath, 'if-no-files-found': 'ignore' },\n          },\n          ...(\n            createIssues\n              ? [\n                {\n                  name: 'Create Issue on Drift',\n                  if: \"always() && steps.drift.outcome == 'failure'\",\n                  id: 'issue',\n                  uses: `actions/github-script@${githubActionsGithubScriptVersion}`,\n                  with: { 'result-encoding': 'string', 'script': issueScript(sanitizedStackName, stack.driftDetectionRoleToAssumeRegion, issueResultsPath) },\n                },\n              ]\n              : []\n          ),\n          ...postSteps.map((step) => {\n            const s: any = { ...(step as any) };\n            // By default, only run extra notification steps when drift occurs\n            s.if = s.if ?? \"always() && steps.drift.outcome == 'failure'\";\n            return s;\n          }),\n        ],\n      };\n    }\n\n    // summary aggregator job\n    jobs['drift-summary'] = {\n      name: 'Drift Detection Summary',\n      needs: Object.keys(jobs).filter((j) => j.startsWith('drift-') && j !== 'drift-summary'),\n      runsOn: ['ubuntu-latest'],\n      permissions: { contents: JobPermission.READ },\n      steps: [\n        {\n          name: 'Download all artifacts',\n          uses: `actions/download-artifact@${githubActionsDownloadArtifactVersion}`,\n          with: { path: 'drift-results' },\n        },\n        { name: 'Generate summary', shell: 'bash', run: summaryScript() },\n      ],\n    };\n\n    workflow.addJobs(jobs);\n  }\n\n  private validateOidcConfiguration(props: CdkDriftDetectionWorkflowProps): void {\n    const hasDefaultOidcRole = !!props.oidcRoleArn;\n    const hasDefaultOidcRegion = !!props.oidcRegion;\n\n    // Check if all stacks have their own OIDC configuration\n    const allStacksHaveOidcRole = props.stacks.every(stack => !!stack.oidcRoleArn);\n    const allStacksHaveOidcRegion = props.stacks.every(stack => !!stack.oidcRegion);\n\n    // Either defaults must be provided OR all stacks must have their own OIDC config\n    if (!hasDefaultOidcRole && !allStacksHaveOidcRole) {\n      throw new Error('Either provide default oidcRoleArn or specify oidcRoleArn for each stack');\n    }\n\n    if (!hasDefaultOidcRegion && !allStacksHaveOidcRegion) {\n      throw new Error('Either provide default oidcRegion or specify oidcRegion for each stack');\n    }\n  }\n}\n\n\nfunction issueScript(stack: string, region: string, resultsFile: string): string {\n  // Construct a plain JS script string (no template string nesting mishaps)\n  const lines = [\n    \"const fs = require('fs');\",\n    `const resultsFile = '${resultsFile}';`,\n    \"if (!fs.existsSync(resultsFile)) { console.log('No results file found'); return 'NO_RESULTS'; }\",\n    \"const results = JSON.parse(fs.readFileSync(resultsFile, 'utf8'));\",\n    \"const driftedStacks = results.filter(r => r.driftStatus === 'DRIFTED');\",\n    \"if (driftedStacks.length === 0) { console.log('No drift detected'); return 'NO_DRIFT'; }\",\n    `const title = 'Drift Detected in ${stack}';`,\n    `let body = '## Drift Detection Report\\\\n\\\\n' + '**Stack:** ${stack}\\\\n' + '**Region:** ${region}\\\\n' + '**Time:** ' + new Date().toISOString() + '\\\\n\\\\n';`,\n    \"body += '### Summary\\\\n';\",\n    \"body += '- Total stacks checked: ' + results.length + '\\\\n';\",\n    \"body += '- Drifted stacks: ' + driftedStacks.length + '\\\\n\\\\n';\",\n    \"body += '### Drifted Stacks\\\\n';\",\n    'for (const s of driftedStacks) {',\n    '  const resources = s.driftedResources || [];',\n    \"  body += '#### ' + s.stackName + '\\\\n';\",\n    \"  body += '- Drifted resources: ' + resources.length + '\\\\n';\",\n    \"  for (const r of resources) { body += '  - ' + r.logicalResourceId + ' (' + r.resourceType + ')\\\\n'; }\",\n    \"  body += '\\\\n';\",\n    '}',\n    \"body += '### Action Required\\\\n' + 'Please review the drifted resources and either:\\\\n1. Update the infrastructure code to match the actual state\\\\n2. Restore the resources to match the expected state\\\\n\\\\n';\",\n    'body += `[View workflow run](${context.serverUrl}/${context.repo.owner}/${context.repo.repo}/actions/runs/${context.runId})`;',\n    // List or update an issue with labels\n    `const issues = await github.rest.issues.listForRepo({ owner: context.repo.owner, repo: context.repo.repo, state: 'open', labels: ['drift-detection', '${stack}'] });`,\n    'let issueNumber;',\n    'if (issues.data.length === 0) {',\n    `  const created = await github.rest.issues.create({ owner: context.repo.owner, repo: context.repo.repo, title, body, labels: ['drift-detection', '${stack}'] });`,\n    '  issueNumber = created.data.number;',\n    '} else {',\n    '  const issue = issues.data[0];',\n    '  issueNumber = issue.number;',\n    '  await github.rest.issues.createComment({ owner: context.repo.owner, repo: context.repo.repo, issue_number: issue.number, body });',\n    '}',\n    'return String(issueNumber ?? \"\");',\n  ];\n  return lines.join('\\n');\n}\n\n\nfunction summaryScript(): string {\n  return [\n    '#!/bin/bash',\n    'set -e',\n    'echo \"## Drift Detection Summary\" >> $GITHUB_STEP_SUMMARY',\n    'echo \"\" >> $GITHUB_STEP_SUMMARY',\n    '',\n    'total_stacks=0',\n    'total_drifted=0',\n    'total_errors=0',\n    '',\n    'shopt -s nullglob',\n    'for file in drift-results-*.json drift-results/*/drift-results-*.json; do',\n    '  if [[ -f \"$file\" ]]; then',\n    '    stack=$(basename \"$file\" | sed -E \\\"s/^drift-results-([^.]+)\\\\.json$/\\\\1/\\\")',\n    '    echo \"### Stack: $stack\" >> $GITHUB_STEP_SUMMARY',\n    '    jq -r \\'' +\n      '. as $results |\\n' +\n      '\"- Total stacks: \" + ($results | length | tostring) + \"\\\\n\" +\\n' +\n      '\"- Drifted: \" + ([.[] | select(.driftStatus == \"DRIFTED\")] | length | tostring) + \"\\\\n\" +\\n' +\n      '\"- Errors: \" + ([.[] | select(.error)] | length | tostring) + \"\\\\n\" +\\n' +\n      '([.[] | select(.driftStatus == \"DRIFTED\")] | if length > 0 then \"\\\\n**Drifted stacks:**\\\\n\" + (map(\"  - \" + .stackName + \" (\" + ((.driftedResources // []) | length | tostring) + \" resources)\") | join(\"\\\\n\")) else \"\" end)\\n' +\n    '\\'' +\n    ' \"$file\" >> $GITHUB_STEP_SUMMARY',\n    '    echo \"\" >> $GITHUB_STEP_SUMMARY',\n    '    total_stacks=$((total_stacks + $(jq \\\"length\\\" \\\"$file\\\")))',\n    '    total_drifted=$((total_drifted + $(jq \\\"[.[] | select(.driftStatus == \\\\\\\"DRIFTED\\\\\\\")] | length\\\" \\\"$file\\\")))',\n    '    total_errors=$((total_errors + $(jq \\\"[.[] | select(.error)] | length\\\" \\\"$file\\\")))',\n    '  fi',\n    'done',\n    '',\n    'echo \"### Overall Summary\" >> $GITHUB_STEP_SUMMARY',\n    'echo \"- Total stacks checked: $total_stacks\" >> $GITHUB_STEP_SUMMARY',\n    'echo \"- Total drifted stacks: $total_drifted\" >> $GITHUB_STEP_SUMMARY',\n    'echo \"- Total errors: $total_errors\" >> $GITHUB_STEP_SUMMARY',\n    '',\n    'if [[ $total_drifted -gt 0 ]]; then',\n    '  echo \"\" >> $GITHUB_STEP_SUMMARY',\n    '  echo \"⚠️ **Action required:** Drift detected in $total_drifted stacks\" >> $GITHUB_STEP_SUMMARY',\n    'fi',\n  ].join('\\n');\n}\n\nfunction toKebabCase(s: string): string {\n  return s.replace(/[^a-zA-Z0-9]+/g, '-').replace(/^-+|-+$/g, '').toLowerCase();\n}\n\n\nfunction buildWorkflowFileName(baseName: string, ext: string = '.yml', maxLength: number = MAX_WORKFLOW_FILENAME_LENGTH): string {\n  const full = `${baseName}${ext}`;\n  if (full.length <= maxLength) {\n    return full;\n  }\n  const available = maxLength - ext.length;\n  if (available <= 0) {\n    throw new Error(`Workflow extension exceeds maximum filename length of ${maxLength}`);\n  }\n  // Truncate from the beginning to keep the end intact\n  return baseName.slice(-available).replace(/^-+/, '') + ext;\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"]}
325
+ //# 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\nconst MAX_WORKFLOW_FILENAME_LENGTH = 255;\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  readonly oidcRoleArn?: string; // Optional override for OIDC role\n  readonly oidcRegion?: string; // Optional override for OIDC region\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 (or each stack must have its own)\n  readonly oidcRegion?: string; // default OIDC region to assume for all stacks (or each stack must have its own)\n  readonly stacks: Stack[];\n  readonly nodeVersion?: string; // e.g., '24.x'\n  /**\n   * Working directory for the CDK app, relative to the repository root.\n   * Useful for monorepos where infrastructure lives in a subdirectory (e.g., 'infra').\n   *\n   * When set, all workflow run steps will use `defaults.run.working-directory`\n   * and artifact/script paths will be adjusted accordingly.\n   *\n   * @default - repository root\n   */\n  readonly workingDirectory?: string;\n  /**\n   * Optional hook to append additional GitHub Actions steps after drift detection per stack.\n   * You can supply a static array of steps, or a factory that receives context and returns steps.\n   */\n  /**\n   * Optional additional GitHub Action steps to run after drift detection for each stack.\n   * These steps run after results are uploaded for each stack. You can include\n   * any notifications you like (e.g., Slack). Provide explicit inputs (e.g., payload/markdown)\n   * directly in your step without relying on a pre-generated payload.\n   */\n  // NOTE: jsii does not support function types in public APIs; use 'any' here and accept either:\n  // - An array of GitHub steps, or\n  // - A function (ctx: { stack: string; workingDirectory?: string }) => GitHubStep[]\n  // The constructor handles both at runtime.\n  readonly postGitHubSteps?: any;\n  /**\n   * Additional GitHub Actions steps to run before drift detection (after install, before AWS creds).\n   * Accepts a static array of steps, or a factory function receiving context:\n   * `(ctx: { stack: string; workingDirectory?: string }) => GitHubStep[]`\n   *\n   * When `workingDirectory` is set, all `run:` steps inherit that directory.\n   * To run a step at the repository root, add `working-directory: '.'` to that step.\n   *\n   * Pre-steps automatically receive the stack-selection condition (`if`) so they\n   * only run when the stack is selected via dispatch.\n   */\n  readonly preGitHubSteps?: 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    // Validate OIDC configuration\n    this.validateOidcConfiguration(props);\n\n    const name = props.workflowName ?? 'drift-detection';\n    const fileName = buildWorkflowFileName(toKebabCase(name));\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    // Normalize working directory: strip trailing slashes\n    const wd = props.workingDirectory?.replace(/\\/+$/, '');\n    const defaults = wd ? { run: { workingDirectory: wd } } : undefined;\n\n    // Only create the drift detection script once to avoid collisions\n    if (!CdkDriftDetectionWorkflow.scriptCreated) {\n      new CdkDriftDetectionScript({\n        project: props.project,\n        outputPath: scriptOutputPath,\n      });\n      CdkDriftDetectionWorkflow.scriptCreated = true;\n    }\n\n    const gh = (project as any).github ?? new GitHub(project);\n    const workflow = new GithubWorkflow(gh, name, { fileName });\n\n    // triggers: schedule + manual dispatch with stack choice\n    const stackChoices = ['all', ...props.stacks.map((s) => s.stackName)];\n    workflow.on({\n      schedule: props.schedule ? [{ cron: props.schedule }] : undefined,\n      workflowDispatch: {\n        inputs: {\n          stack: {\n            description: \"Stack to check for drift ('all' to run every stack)\",\n            required: false,\n            type: 'choice',\n            options: stackChoices,\n          },\n        },\n      },\n    });\n\n    // One job per stack\n    const jobs: Record<string, any> = {};\n\n    for (const stack of props.stacks) {\n      const sanitizedStackName = toKebabCase(toGithubJobId(stack.stackName));\n      const originalStackName = stack.stackName;\n      const jobId = `drift-${sanitizedStackName}`;\n      const resultsFile = `drift-results-${sanitizedStackName}.json`;\n      const innerCond = \"github.event_name == 'schedule' || !github.event.inputs.stack || github.event.inputs.stack == 'all' || github.event.inputs.stack == '\" + sanitizedStackName + \"' || github.event.inputs.stack == '\" + originalStackName + \"'\";\n      const condExpr = '${{ ' + innerCond + ' }}';\n      const notCondExpr = '${{ !(' + innerCond + ') }}';\n\n      const stepCtx = { stack: sanitizedStackName, workingDirectory: wd };\n      const rawPre = props.preGitHubSteps;\n      const preSteps: GitHubStep[] = typeof rawPre === 'function' ? rawPre(stepCtx) : (rawPre ?? []);\n      const rawPost = props.postGitHubSteps;\n      const postSteps: GitHubStep[] = typeof rawPost === 'function' ? rawPost(stepCtx) : (rawPost ?? []);\n\n      // Prefix results file path for uses: steps (which ignore defaults.run.working-directory)\n      const artifactResultsPath = wd ? `${wd}/${resultsFile}` : resultsFile;\n      const issueResultsPath = wd ? `${wd}/${resultsFile}` : resultsFile;\n\n      jobs[jobId] = {\n        name: `Drift Detection - ${sanitizedStackName}`,\n        runsOn: ['ubuntu-latest'],\n        ...(defaults && { defaults }),\n        permissions: {\n          contents: JobPermission.READ,\n          idToken: JobPermission.WRITE,\n          issues: JobPermission.WRITE,\n        },\n        env: {\n          AWS_DEFAULT_REGION: stack.driftDetectionRoleToAssumeRegion,\n          AWS_REGION: stack.driftDetectionRoleToAssumeRegion,\n          DRIFT_DETECTION_OUTPUT: resultsFile,\n          STACK_ID: sanitizedStackName,\n          STACK_NAME: stack.stackName,\n        },\n        // No job-level condition; we gate steps so the job always completes and summary can run\n        steps: [\n          { name: 'Skip (stack not selected)', if: notCondExpr, run: 'echo \"Stack not selected; skipping drift detection for this job.\"' },\n          { name: 'Checkout', if: condExpr, uses: `actions/checkout@${githubActionsCheckoutVersion}` },\n          {\n            name: 'Setup Node.js',\n            if: condExpr,\n            uses: `actions/setup-node@${githubActionsSetupNodeVersion}`,\n            with: { 'node-version': nodeVersion },\n          },\n          { name: 'Install dependencies', if: condExpr, run: 'yarn install --frozen-lockfile || npm ci', env: { GITHUB_TOKEN: '${{ secrets.GITHUB_TOKEN }}' } },\n          ...preSteps.map((step) => {\n            const s: any = { ...(step as any) };\n            s.if = s.if ?? condExpr;\n            return s;\n          }),\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': stack.oidcRoleArn ?? props.oidcRoleArn,\n              'role-session-name': 'GitHubAction',\n              'aws-region': stack.oidcRegion ?? props.oidcRegion,\n            },\n          },\n          {\n            name: 'Assume Drift Detection Role',\n            if: condExpr,\n            uses: `aws-actions/configure-aws-credentials@${githubActionsAwsCredentialsVersion}`,\n            with: {\n              'role-to-assume': stack.driftDetectionRoleToAssumeArn,\n              'role-chaining': true,\n              'role-skip-session-tagging': true,\n              'aws-region': stack.driftDetectionRoleToAssumeRegion,\n              'aws-access-key-id': '${{ steps.creds.outputs.aws-access-key-id }}',\n              'aws-secret-access-key': '${{ steps.creds.outputs.aws-secret-access-key }}',\n              'aws-session-token': '${{ steps.creds.outputs.aws-session-token }}',\n            },\n          },\n          {\n            name: 'Detect drift',\n            if: condExpr,\n            id: 'drift',\n            continueOnError: true, // allow artifact upload and issue creation even when drift is detected\n            run: [\n              'set +e',\n              // Use the bundled script from this package\n              'node ./node_modules/@jjrawlins/cdk-diff-pr-github-action/lib/bin/detect-drift.js',\n              'status=$?',\n              'if [ -f \"$DRIFT_DETECTION_OUTPUT\" ]; then echo \"Results file created: $DRIFT_DETECTION_OUTPUT\"; fi',\n              // Expose useful outputs for downstream steps\n              \"STACK_ARN=$(aws cloudformation describe-stacks --stack-name \\\"$STACK_NAME\\\" --query 'Stacks[0].StackId' --output text 2>/dev/null || true)\",\n              'echo \"stack-arn=$STACK_ARN\" >> \"$GITHUB_OUTPUT\"',\n              'exit $status',\n            ].join('\\n'),\n            env: {\n              STACK_NAME: stack.stackName,\n              AWS_REGION: stack.driftDetectionRoleToAssumeRegion,\n              DRIFT_DETECTION_OUTPUT: resultsFile,\n            },\n          },\n          {\n            name: 'Upload results',\n            if: condExpr,\n            uses: `actions/upload-artifact@${githubActionsUploadArtifactVersion}`,\n            with: { 'name': `drift-results-${sanitizedStackName}`, 'path': artifactResultsPath, 'if-no-files-found': 'ignore' },\n          },\n          ...(\n            createIssues\n              ? [\n                {\n                  name: 'Create Issue on Drift',\n                  if: \"always() && steps.drift.outcome == 'failure'\",\n                  id: 'issue',\n                  uses: `actions/github-script@${githubActionsGithubScriptVersion}`,\n                  with: { 'result-encoding': 'string', 'script': issueScript(sanitizedStackName, stack.driftDetectionRoleToAssumeRegion, issueResultsPath) },\n                },\n              ]\n              : []\n          ),\n          ...postSteps.map((step) => {\n            const s: any = { ...(step as any) };\n            // By default, only run extra notification steps when drift occurs\n            s.if = s.if ?? \"always() && steps.drift.outcome == 'failure'\";\n            return s;\n          }),\n        ],\n      };\n    }\n\n    // summary aggregator job\n    jobs['drift-summary'] = {\n      name: 'Drift Detection Summary',\n      needs: Object.keys(jobs).filter((j) => j.startsWith('drift-') && j !== 'drift-summary'),\n      runsOn: ['ubuntu-latest'],\n      permissions: { contents: JobPermission.READ },\n      steps: [\n        {\n          name: 'Download all artifacts',\n          uses: `actions/download-artifact@${githubActionsDownloadArtifactVersion}`,\n          with: { path: 'drift-results' },\n        },\n        { name: 'Generate summary', shell: 'bash', run: summaryScript() },\n      ],\n    };\n\n    workflow.addJobs(jobs);\n  }\n\n  private validateOidcConfiguration(props: CdkDriftDetectionWorkflowProps): void {\n    const hasDefaultOidcRole = !!props.oidcRoleArn;\n    const hasDefaultOidcRegion = !!props.oidcRegion;\n\n    // Check if all stacks have their own OIDC configuration\n    const allStacksHaveOidcRole = props.stacks.every(stack => !!stack.oidcRoleArn);\n    const allStacksHaveOidcRegion = props.stacks.every(stack => !!stack.oidcRegion);\n\n    // Either defaults must be provided OR all stacks must have their own OIDC config\n    if (!hasDefaultOidcRole && !allStacksHaveOidcRole) {\n      throw new Error('Either provide default oidcRoleArn or specify oidcRoleArn for each stack');\n    }\n\n    if (!hasDefaultOidcRegion && !allStacksHaveOidcRegion) {\n      throw new Error('Either provide default oidcRegion or specify oidcRegion for each stack');\n    }\n  }\n}\n\n\nfunction issueScript(stack: string, region: string, resultsFile: string): string {\n  // Construct a plain JS script string (no template string nesting mishaps)\n  const lines = [\n    \"const fs = require('fs');\",\n    `const resultsFile = '${resultsFile}';`,\n    \"if (!fs.existsSync(resultsFile)) { console.log('No results file found'); return 'NO_RESULTS'; }\",\n    \"const results = JSON.parse(fs.readFileSync(resultsFile, 'utf8'));\",\n    \"const driftedStacks = results.filter(r => r.driftStatus === 'DRIFTED');\",\n    \"if (driftedStacks.length === 0) { console.log('No drift detected'); return 'NO_DRIFT'; }\",\n    `const title = 'Drift Detected in ${stack}';`,\n    `let body = '## Drift Detection Report\\\\n\\\\n' + '**Stack:** ${stack}\\\\n' + '**Region:** ${region}\\\\n' + '**Time:** ' + new Date().toISOString() + '\\\\n\\\\n';`,\n    \"body += '### Summary\\\\n';\",\n    \"body += '- Total stacks checked: ' + results.length + '\\\\n';\",\n    \"body += '- Drifted stacks: ' + driftedStacks.length + '\\\\n\\\\n';\",\n    \"body += '### Drifted Stacks\\\\n';\",\n    'for (const s of driftedStacks) {',\n    '  const resources = s.driftedResources || [];',\n    \"  body += '#### ' + s.stackName + '\\\\n';\",\n    \"  body += '- Drifted resources: ' + resources.length + '\\\\n';\",\n    \"  for (const r of resources) { body += '  - ' + r.logicalResourceId + ' (' + r.resourceType + ')\\\\n'; }\",\n    \"  body += '\\\\n';\",\n    '}',\n    \"body += '### Action Required\\\\n' + 'Please review the drifted resources and either:\\\\n1. Update the infrastructure code to match the actual state\\\\n2. Restore the resources to match the expected state\\\\n\\\\n';\",\n    'body += `[View workflow run](${context.serverUrl}/${context.repo.owner}/${context.repo.repo}/actions/runs/${context.runId})`;',\n    // List or update an issue with labels\n    `const issues = await github.rest.issues.listForRepo({ owner: context.repo.owner, repo: context.repo.repo, state: 'open', labels: ['drift-detection', '${stack}'] });`,\n    'let issueNumber;',\n    'if (issues.data.length === 0) {',\n    `  const created = await github.rest.issues.create({ owner: context.repo.owner, repo: context.repo.repo, title, body, labels: ['drift-detection', '${stack}'] });`,\n    '  issueNumber = created.data.number;',\n    '} else {',\n    '  const issue = issues.data[0];',\n    '  issueNumber = issue.number;',\n    '  await github.rest.issues.createComment({ owner: context.repo.owner, repo: context.repo.repo, issue_number: issue.number, body });',\n    '}',\n    'return String(issueNumber ?? \"\");',\n  ];\n  return lines.join('\\n');\n}\n\n\nfunction summaryScript(): string {\n  return [\n    '#!/bin/bash',\n    'set -e',\n    'echo \"## Drift Detection Summary\" >> $GITHUB_STEP_SUMMARY',\n    'echo \"\" >> $GITHUB_STEP_SUMMARY',\n    '',\n    'total_stacks=0',\n    'total_drifted=0',\n    'total_errors=0',\n    '',\n    'shopt -s nullglob',\n    'for file in drift-results-*.json drift-results/*/drift-results-*.json; do',\n    '  if [[ -f \"$file\" ]]; then',\n    '    stack=$(basename \"$file\" | sed -E \\\"s/^drift-results-([^.]+)\\\\.json$/\\\\1/\\\")',\n    '    echo \"### Stack: $stack\" >> $GITHUB_STEP_SUMMARY',\n    '    jq -r \\'' +\n      '. as $results |\\n' +\n      '\"- Total stacks: \" + ($results | length | tostring) + \"\\\\n\" +\\n' +\n      '\"- Drifted: \" + ([.[] | select(.driftStatus == \"DRIFTED\")] | length | tostring) + \"\\\\n\" +\\n' +\n      '\"- Errors: \" + ([.[] | select(.error)] | length | tostring) + \"\\\\n\" +\\n' +\n      '([.[] | select(.driftStatus == \"DRIFTED\")] | if length > 0 then \"\\\\n**Drifted stacks:**\\\\n\" + (map(\"  - \" + .stackName + \" (\" + ((.driftedResources // []) | length | tostring) + \" resources)\") | join(\"\\\\n\")) else \"\" end)\\n' +\n    '\\'' +\n    ' \"$file\" >> $GITHUB_STEP_SUMMARY',\n    '    echo \"\" >> $GITHUB_STEP_SUMMARY',\n    '    total_stacks=$((total_stacks + $(jq \\\"length\\\" \\\"$file\\\")))',\n    '    total_drifted=$((total_drifted + $(jq \\\"[.[] | select(.driftStatus == \\\\\\\"DRIFTED\\\\\\\")] | length\\\" \\\"$file\\\")))',\n    '    total_errors=$((total_errors + $(jq \\\"[.[] | select(.error)] | length\\\" \\\"$file\\\")))',\n    '  fi',\n    'done',\n    '',\n    'echo \"### Overall Summary\" >> $GITHUB_STEP_SUMMARY',\n    'echo \"- Total stacks checked: $total_stacks\" >> $GITHUB_STEP_SUMMARY',\n    'echo \"- Total drifted stacks: $total_drifted\" >> $GITHUB_STEP_SUMMARY',\n    'echo \"- Total errors: $total_errors\" >> $GITHUB_STEP_SUMMARY',\n    '',\n    'if [[ $total_drifted -gt 0 ]]; then',\n    '  echo \"\" >> $GITHUB_STEP_SUMMARY',\n    '  echo \"⚠️ **Action required:** Drift detected in $total_drifted stacks\" >> $GITHUB_STEP_SUMMARY',\n    'fi',\n  ].join('\\n');\n}\n\nfunction toKebabCase(s: string): string {\n  return s.replace(/[^a-zA-Z0-9]+/g, '-').replace(/^-+|-+$/g, '').toLowerCase();\n}\n\n\nfunction buildWorkflowFileName(baseName: string, ext: string = '.yml', maxLength: number = MAX_WORKFLOW_FILENAME_LENGTH): string {\n  const full = `${baseName}${ext}`;\n  if (full.length <= maxLength) {\n    return full;\n  }\n  const available = maxLength - ext.length;\n  if (available <= 0) {\n    throw new Error(`Workflow extension exceeds maximum filename length of ${maxLength}`);\n  }\n  // Truncate from the beginning to keep the end intact\n  return baseName.slice(-available).replace(/^-+/, '') + ext;\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"]}
@@ -79,7 +79,7 @@ class CdkDriftIamTemplateGenerator {
79
79
  }
80
80
  exports.CdkDriftIamTemplateGenerator = CdkDriftIamTemplateGenerator;
81
81
  _a = JSII_RTTI_SYMBOL_1;
82
- CdkDriftIamTemplateGenerator[_a] = { fqn: "@jjrawlins/cdk-diff-pr-github-action.CdkDriftIamTemplateGenerator", version: "1.8.0" };
82
+ CdkDriftIamTemplateGenerator[_a] = { fqn: "@jjrawlins/cdk-diff-pr-github-action.CdkDriftIamTemplateGenerator", version: "1.9.0" };
83
83
  /**
84
84
  * Projen construct that emits a CloudFormation template with minimal IAM permissions
85
85
  * for the CDK Drift Detection Workflow.
@@ -102,5 +102,5 @@ class CdkDriftIamTemplate {
102
102
  }
103
103
  exports.CdkDriftIamTemplate = CdkDriftIamTemplate;
104
104
  _b = JSII_RTTI_SYMBOL_1;
105
- CdkDriftIamTemplate[_b] = { fqn: "@jjrawlins/cdk-diff-pr-github-action.CdkDriftIamTemplate", version: "1.8.0" };
105
+ CdkDriftIamTemplate[_b] = { fqn: "@jjrawlins/cdk-diff-pr-github-action.CdkDriftIamTemplate", version: "1.9.0" };
106
106
  //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CdkDriftIamTemplate.js","sourceRoot":"","sources":["../src/CdkDriftIamTemplate.ts"],"names":[],"mappings":";;;;;AAAA,mCAAkC;AAclC;;;GAGG;AACH,MAAa,4BAA4B;IACvC;;OAEG;IACH,MAAM,CAAC,gBAAgB,CAAC,KAAwC;QAC9D,MAAM,KAAK,GAAG;YACZ,wCAAwC;YACxC,0DAA0D;YAC1D,EAAE;YACF,aAAa;YACb,sBAAsB;YACtB,kBAAkB;YAClB,yFAAyF;YACzF,iBAAiB,KAAK,CAAC,WAAW,GAAG;YACrC,EAAE;YACF,YAAY;YACZ,8FAA8F;YAC9F,iBAAiB;YACjB,0BAA0B;YAC1B,iBAAiB;YACjB,mBAAmB,GAAG,KAAK,CAAC,QAAQ,GAAG,GAAG;YAC1C,iCAAiC;YACjC,+BAA+B;YAC/B,oBAAoB;YACpB,2BAA2B;YAC3B,wBAAwB;YACxB,2CAA2C;YAC3C,oCAAoC;YACpC,wBAAwB;YACxB,6BAA6B;YAC7B,wCAAwC,GAAG,KAAK,CAAC,UAAU,GAAG,GAAG;YACjE,iBAAiB;YACjB,iDAAiD;YACjD,2BAA2B;YAC3B,mCAAmC;YACnC,wBAAwB;YACxB,2DAA2D;YAC3D,+BAA+B;YAC/B,yBAAyB;YACzB,qDAAqD;YACrD,sEAAsE;YACtE,gEAAgE;YAChE,mDAAmD;YACnD,uDAAuD;YACvD,6DAA6D;YAC7D,+BAA+B;YAC/B,EAAE;YACF,UAAU;YACV,oBAAoB;YACpB,wDAAwD;YACxD,qCAAqC;YACrC,aAAa;YACb,sDAAsD;YACtD,EAAE;YACF,qBAAqB;YACrB,yDAAyD;YACzD,8BAA8B;YAC9B,aAAa;YACb,uDAAuD;SACxD,CAAC;QAEF,OAAO,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IAC1B,CAAC;IAED;;OAEG;IACH,MAAM,CAAC,qBAAqB,CAAC,eAAuB,sCAAsC;QACxF,OAAO,6CAA6C,YAAY,+EAA+E,CAAC;IAClJ,CAAC;;AArEH,oEAsEC;;;AAYD;;;;;GAKG;AACH,MAAa,mBAAmB;IAC9B,YAAY,KAA+B;QACzC,MAAM,UAAU,GAAG,KAAK,CAAC,UAAU,IAAI,sCAAsC,CAAC;QAE9E,wCAAwC;QACxC,MAAM,QAAQ,GAAG,4BAA4B,CAAC,gBAAgB,CAAC,KAAK,CAAC,CAAC;QACtE,IAAI,iBAAQ,CAAC,KAAK,CAAC,OAAO,EAAE,UAAU,EAAE,EAAE,KAAK,EAAE,QAAQ,CAAC,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAEzE,kBAAkB;QAClB,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC,8BAA8B,EAAE;YACpD,WAAW,EACT,uIAAuI;YACzI,WAAW,EAAE,IAAI;YACjB,IAAI,EAAE,4BAA4B,CAAC,qBAAqB,CAAC,UAAU,CAAC;SACrE,CAAC,CAAC;IACL,CAAC;;AAfH,kDAgBC","sourcesContent":["import { TextFile } from 'projen';\n\n/**\n * Props for generating CDK Drift IAM templates (no Projen dependency)\n */\nexport interface CdkDriftIamTemplateGeneratorProps {\n  /** Name for the IAM role */\n  readonly roleName: string;\n  /** ARN of the existing GitHub OIDC role that can assume this drift role */\n  readonly oidcRoleArn: string;\n  /** Region for the OIDC trust condition */\n  readonly oidcRegion: string;\n}\n\n/**\n * Pure generator class for CDK Drift IAM templates.\n * No Projen dependency - can be used in any project.\n */\nexport class CdkDriftIamTemplateGenerator {\n  /**\n   * Generate the CloudFormation IAM template as a YAML string.\n   */\n  static generateTemplate(props: CdkDriftIamTemplateGeneratorProps): string {\n    const lines = [\n      \"AWSTemplateFormatVersion: '2010-09-09'\",\n      \"Description: 'IAM role for CDK Drift Detection Workflow'\",\n      '',\n      'Parameters:',\n      '  GitHubOIDCRoleArn:',\n      '    Type: String',\n      \"    Description: 'ARN of the existing GitHub OIDC role that can assume this drift role'\",\n      `    Default: '${props.oidcRoleArn}'`,\n      '',\n      'Resources:',\n      '  # CloudFormation Drift Detection Role - minimal permissions for drift detection operations',\n      '  CdkDriftRole:',\n      '    Type: AWS::IAM::Role',\n      '    Properties:',\n      \"      RoleName: '\" + props.roleName + \"'\",\n      '      AssumeRolePolicyDocument:',\n      \"        Version: '2012-10-17'\",\n      '        Statement:',\n      '          - Effect: Allow',\n      '            Principal:',\n      '              AWS: !Ref GitHubOIDCRoleArn',\n      '            Action: sts:AssumeRole',\n      '            Condition:',\n      '              StringEquals:',\n      \"                aws:RequestedRegion: '\" + props.oidcRegion + \"'\",\n      '      Policies:',\n      '        - PolicyName: CloudFormationDriftAccess',\n      '          PolicyDocument:',\n      \"            Version: '2012-10-17'\",\n      '            Statement:',\n      '              # CloudFormation drift detection operations',\n      '              - Effect: Allow',\n      '                Action:',\n      '                  - cloudformation:DetectStackDrift',\n      '                  - cloudformation:DescribeStackDriftDetectionStatus',\n      '                  - cloudformation:DescribeStackResourceDrifts',\n      '                  - cloudformation:DescribeStacks',\n      '                  - cloudformation:ListStackResources',\n      '                  - cloudformation:DetectStackResourceDrift',\n      \"                Resource: '*'\",\n      '',\n      'Outputs:',\n      '  CdkDriftRoleArn:',\n      \"    Description: 'ARN of the CDK drift detection role'\",\n      '    Value: !GetAtt CdkDriftRole.Arn',\n      '    Export:',\n      \"      Name: !Sub '${AWS::StackName}-CdkDriftRoleArn'\",\n      '',\n      '  CdkDriftRoleName:',\n      \"    Description: 'Name of the CDK drift detection role'\",\n      '    Value: !Ref CdkDriftRole',\n      '    Export:',\n      \"      Name: !Sub '${AWS::StackName}-CdkDriftRoleName'\",\n    ];\n\n    return lines.join('\\n');\n  }\n\n  /**\n   * Generate the AWS CLI deploy command for the IAM template.\n   */\n  static generateDeployCommand(templatePath: string = 'cdk-drift-workflow-iam-template.yaml'): string {\n    return `aws cloudformation deploy --template-file ${templatePath} --stack-name cdk-drift-workflow-iam-role --capabilities CAPABILITY_NAMED_IAM`;\n  }\n}\n\n/**\n * Props for the Projen-integrated CDK Drift IAM template construct\n */\nexport interface CdkDriftIamTemplateProps extends CdkDriftIamTemplateGeneratorProps {\n  /** Projen project instance */\n  readonly project: any;\n  /** Output path for the template file (default: 'cdk-drift-workflow-iam-template.yaml') */\n  readonly outputPath?: string;\n}\n\n/**\n * Projen construct that emits a CloudFormation template with minimal IAM permissions\n * for the CDK Drift Detection Workflow.\n *\n * For non-Projen projects, use `CdkDriftIamTemplateGenerator` directly.\n */\nexport class CdkDriftIamTemplate {\n  constructor(props: CdkDriftIamTemplateProps) {\n    const outputPath = props.outputPath ?? 'cdk-drift-workflow-iam-template.yaml';\n\n    // Generate template using the generator\n    const template = CdkDriftIamTemplateGenerator.generateTemplate(props);\n    new TextFile(props.project, outputPath, { lines: template.split('\\n') });\n\n    // Add deploy task\n    props.project.addTask('deploy-cdkdrift-iam-template', {\n      description:\n        'Deploy the CDK Drift Detection IAM template via CloudFormation (accepts extra AWS CLI args, e.g., --parameter-overrides Key=Value...)',\n      receiveArgs: true,\n      exec: CdkDriftIamTemplateGenerator.generateDeployCommand(outputPath),\n    });\n  }\n}\n"]}
package/package.json CHANGED
@@ -110,7 +110,7 @@
110
110
  "publishConfig": {
111
111
  "access": "public"
112
112
  },
113
- "version": "1.8.0",
113
+ "version": "1.9.0",
114
114
  "jest": {
115
115
  "coverageProvider": "v8",
116
116
  "testMatch": [