@jjrawlins/cdk-diff-pr-github-action 1.7.2 → 1.8.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.
- package/.jsii +46 -8
- package/API.md +38 -0
- package/README.md +40 -0
- package/cdkdiffprgithubaction/jsii/jsii.go +2 -2
- package/cdkdiffprgithubaction/version +1 -1
- package/lib/CdkDiffIamTemplate.js +2 -2
- package/lib/CdkDiffIamTemplateStackSet.js +2 -2
- package/lib/CdkDiffStackWorkflow.d.ts +10 -0
- package/lib/CdkDiffStackWorkflow.js +16 -5
- package/lib/CdkDriftDetectionWorkflow.d.ts +10 -0
- package/lib/CdkDriftDetectionWorkflow.js +11 -4
- package/lib/CdkDriftIamTemplate.js +2 -2
- package/package.json +1 -1
package/.jsii
CHANGED
|
@@ -3507,7 +3507,7 @@
|
|
|
3507
3507
|
},
|
|
3508
3508
|
"name": "@jjrawlins/cdk-diff-pr-github-action",
|
|
3509
3509
|
"readme": {
|
|
3510
|
-
"markdown": "# cdk-diff-pr-github-action\n\nA [Projen](https://projen.io/) construct library that surfaces **CloudFormation change set diffs and drift status directly on your pull requests** so reviewers can see exactly what will change before merging.\n\n## Why this exists\n\n`cdk diff` output disappears into CI logs that nobody reads. Meanwhile, a single property change on an RDS instance or EC2 \"pet\" server can trigger a **resource replacement** — destroying the database or instance and recreating it from scratch. If that replacement slips through code review unnoticed, the result is data loss and downtime.\n\nThis construct was built to make those dangerous changes impossible to miss:\n\n- **Replacement column front and center** — Every change set row shows whether CloudFormation will modify the resource in place or **replace** it, with before/after property values so reviewers can understand *why*.\n- **Comment appears on the PR itself** — No digging through workflow logs. The diff table is posted (and updated in place) as a PR comment and in the GitHub Step Summary.\n- **Drift banner** — If the stack has drifted from its template, a warning banner is prepended to the comment so reviewers know the baseline is already out of sync.\n\nIf you have ever lost an EC2 instance, an RDS database, or an ElastiCache cluster to an unexpected CloudFormation replacement, this tool is for you.\n\n---\n\nA library that provides GitHub workflows and IAM templates for:\n- Creating CloudFormation Change Sets for your CDK stacks on pull requests and commenting a formatted diff back on the PR.\n- Detecting CloudFormation drift on a schedule or manual trigger and producing a consolidated summary (optionally creating an issue).\n- Deploying IAM roles across AWS Organizations using StackSets.\n\nIt also provides ready-to-deploy IAM templates with the minimal permissions required for each workflow.\n\n**Works with or without Projen** -- The StackSet generator can be used standalone in any Node.js project.\n\nThis package exposes five constructs:\n\n- `CdkDiffStackWorkflow` — Generates one GitHub Actions workflow per stack to create a change set and render the diff back to the PR and Step Summary.\n- `CdkDiffIamTemplate` — Emits a CloudFormation template file with minimal permissions for the Change Set workflow.\n- `CdkDriftDetectionWorkflow` — Generates a GitHub Actions workflow to detect CloudFormation drift per stack, upload machine‑readable results, and aggregate a summary.\n- `CdkDriftIamTemplate` — Emits a CloudFormation template file with minimal permissions for the Drift Detection workflow.\n- `CdkDiffIamTemplateStackSet` — Creates a CloudFormation StackSet template for org-wide deployment of GitHub OIDC and IAM roles (Projen integration).\n- `CdkDiffIamTemplateStackSetGenerator` — Pure generator class for StackSet templates (no Projen dependency).\n\n## Quick start\n\n1) Add the constructs to your Projen project (in `.projenrc.ts`).\n2) Synthesize with `npx projen`.\n3) Commit the generated files.\n4) Open a pull request or run the drift detection workflow.\n\n## End-to-end example\n\nA realistic setup for a CDK Pipelines project with multiple stages and accounts. This generates one diff workflow per stack, a shared IAM template, and a scheduled drift detection workflow.\n\n```ts\n// .projenrc.ts\nimport { awscdk } from 'projen';\nimport {\n CdkDiffStackWorkflow,\n CdkDiffIamTemplate,\n CdkDriftDetectionWorkflow,\n} from '@jjrawlins/cdk-diff-pr-github-action';\n\nconst project = new awscdk.AwsCdkTypeScriptApp({\n name: 'my-data-platform',\n defaultReleaseBranch: 'main',\n cdkVersion: '2.170.0',\n github: true,\n});\n\n// --- Change Set Diff Workflows (one per stack) ---\n// stackName is the CDK construct path used with `cdk deploy <target>`.\n// For CDK Pipelines, this is typically: pipeline/Stage/Stack\n// The construct automatically resolves the real CloudFormation stack name\n// from cdk.out at runtime (e.g., \"my-stage-dev-my-stack\").\n\nnew CdkDiffStackWorkflow({\n project,\n oidcRoleArn: 'arn:aws:iam::111111111111:role/GitHubOIDCRole',\n oidcRegion: 'us-east-1',\n stacks: [\n {\n stackName: 'my-pipeline/dev/DatabaseStack',\n changesetRoleToAssumeArn: 'arn:aws:iam::111111111111:role/CdkChangesetRole',\n changesetRoleToAssumeRegion: 'us-east-1',\n },\n {\n stackName: 'my-pipeline/dev/ComputeStack',\n changesetRoleToAssumeArn: 'arn:aws:iam::111111111111:role/CdkChangesetRole',\n changesetRoleToAssumeRegion: 'us-east-1',\n },\n {\n // Cross-account: prod stacks can use a different OIDC role and region\n stackName: 'my-pipeline/prod/DatabaseStack',\n changesetRoleToAssumeArn: 'arn:aws:iam::222222222222:role/CdkChangesetRole',\n changesetRoleToAssumeRegion: 'us-east-1',\n oidcRoleArn: 'arn:aws:iam::222222222222:role/GitHubOIDCRole',\n oidcRegion: 'us-east-1',\n },\n ],\n});\n\n// --- IAM Template (deploy once per account) ---\nnew CdkDiffIamTemplate({\n project,\n roleName: 'CdkChangesetRole',\n createOidcRole: true,\n githubOidc: {\n owner: 'my-org',\n repositories: ['my-data-platform'],\n branches: ['main'],\n },\n});\n\n// --- Drift Detection (scheduled + manual) ---\nnew CdkDriftDetectionWorkflow({\n project,\n schedule: '0 6 * * 1', // Every Monday at 6 AM UTC\n oidcRoleArn: 'arn:aws:iam::111111111111:role/GitHubOIDCRole',\n oidcRegion: 'us-east-1',\n stacks: [\n {\n stackName: 'dev-DatabaseStack',\n driftDetectionRoleToAssumeArn: 'arn:aws:iam::111111111111:role/CdkDriftRole',\n driftDetectionRoleToAssumeRegion: 'us-east-1',\n },\n ],\n});\n\nproject.synth();\n```\n\nAfter `npx projen`, this generates:\n- `.github/workflows/diff-my-pipeline-dev-databasestack.yml`\n- `.github/workflows/diff-my-pipeline-dev-computestack.yml`\n- `.github/workflows/diff-my-pipeline-prod-databasestack.yml`\n- `.github/workflows/scripts/describe-cfn-changeset.ts`\n- `.github/workflows/drift-detection.yml`\n- `.github/workflows/scripts/detect-drift.ts`\n- `cdk-diff-workflow-iam-template.yaml`\n\nWhen a pull request is opened, each diff workflow runs automatically and posts a comment like this:\n\n| Action | ID | Type | Replacement | Details |\n|--------|-----|------|-------------|---------|\n| 🔵 Modify | MyDatabase | AWS::RDS::DBInstance | **True** | 🔵 **DBInstanceClass**: `db.t3.medium` -> `db.t3.large` |\n| 🔵 Modify | MyFunction | AWS::Lambda::Function | False | 🔵 **Runtime**: `nodejs18.x` -> `nodejs20.x` |\n\nThe **Replacement: True** on the RDS instance is exactly the kind of change this tool is designed to catch before it reaches production.\n\n## Usage: CdkDiffStackWorkflow\n\n`CdkDiffStackWorkflow` renders a workflow per stack named `diff-<StackName>.yml` under `.github/workflows/`. It also generates a helper script at `.github/workflows/scripts/describe-cfn-changeset.ts` that formats the change set output and takes care of posting the PR comment and Step Summary.\n\nExample `.projenrc.ts`:\n\n```ts\nimport { awscdk } from 'projen';\nimport { CdkDiffStackWorkflow } from '@jjrawlins/cdk-diff-pr-github-action';\n\nconst project = new awscdk.AwsCdkConstructLibrary({\n // ... your usual settings ...\n workflowName: 'my-lib',\n defaultReleaseBranch: 'main',\n cdkVersion: '2.85.0',\n github: true,\n});\n\nnew CdkDiffStackWorkflow({\n project,\n stacks: [\n {\n stackName: 'MyAppStack',\n changesetRoleToAssumeArn: 'arn:aws:iam::123456789012:role/cdk-diff-role',\n changesetRoleToAssumeRegion: 'us-east-1',\n // Optional per‑stack OIDC override (if not using the defaults below)\n // oidcRoleArn: 'arn:aws:iam::123456789012:role/github-oidc-role',\n // oidcRegion: 'us-east-1',\n },\n ],\n // Default OIDC role/region used by all stacks unless overridden per‑stack\n oidcRoleArn: 'arn:aws:iam::123456789012:role/github-oidc-role',\n oidcRegion: 'us-east-1',\n // Optional: Node version used in the workflow (default: '24.x')\n // nodeVersion: '24.x',\n // Optional: Yarn command to run CDK (default: 'cdk')\n // cdkYarnCommand: 'cdk',\n // Optional: Where to place the helper script (default: '.github/workflows/scripts/describe-cfn-changeset.ts')\n // scriptOutputPath: '.github/workflows/scripts/describe-cfn-changeset.ts',\n});\n\nproject.synth();\n```\n\n### CdkDiffStackWorkflow props\n- `project` (required) — Your Projen project instance.\n- `stacks` (required) — Array of stack entries.\n- `oidcRoleArn` (required unless provided per‑stack) — Default OIDC role ARN.\n- `oidcRegion` (required unless provided per‑stack) — Default OIDC region.\n- `nodeVersion` (optional, default `'24.x'`) — Node.js version for the workflow runner.\n- `cdkYarnCommand` (optional, default `'cdk'`) — Yarn script/command to invoke CDK.\n- `scriptOutputPath` (optional, default `'.github/workflows/scripts/describe-cfn-changeset.ts'`) — Where to write the helper script.\n\nIf neither top‑level OIDC defaults nor all per‑stack values are supplied, the construct throws a helpful error.\n\n### Stack item fields\n- `stackName` (required) — The CDK stack name to create the change set for.\n- `changesetRoleToAssumeArn` (required) — The ARN of the role used to create the change set (role chaining after OIDC).\n- `changesetRoleToAssumeRegion` (required) — The region for that role.\n- `oidcRoleArn` (optional) — Per‑stack override for the OIDC role.\n- `oidcRegion` (optional) — Per‑stack override for the OIDC region.\n\n### What gets generated\n- `.github/workflows/diff-<StackName>.yml` — One workflow per stack, triggered on PR open/sync/reopen.\n- `.github/workflows/scripts/describe-cfn-changeset.ts` — A helper script that:\n - Polls `DescribeChangeSet` until terminal\n - Filters out ignorable logical IDs or resource types using environment variables `IGNORE_LOGICAL_IDS` and `IGNORE_RESOURCE_TYPES`\n - Renders an HTML table with actions, logical IDs, types, replacements, and changed properties\n - **Checks cached drift status** via `DescribeStacks` — if the stack has drifted, a warning banner with drifted resource details is prepended to the output (non-fatal; degrades gracefully if IAM permissions are missing)\n - Prints the HTML and appends to the GitHub Step Summary\n - **Upserts the PR comment** — uses an HTML marker (`<!-- cdk-diff:stack:STACK_NAME -->`) to find and update an existing comment instead of creating duplicates on every push\n\n### Change Set Output Format\n\nThe change set script uses the CloudFormation `IncludePropertyValues` API feature to show **actual before/after values** for changed properties, not just property names.\n\n**Example PR comment:**\n\n> **Warning: Stack has drifted (2 resources out of sync)**\n>\n> Last drift check: 2025-01-15T10:30:00.000Z\n>\n> <details><summary>View drifted resources</summary>\n>\n> | Resource | Type | Drift Status |\n> |----------|------|-------------|\n> | MySecurityGroup | AWS::EC2::SecurityGroup | MODIFIED |\n> | OldBucket | AWS::S3::Bucket | DELETED |\n> </details>\n\n| Action | ID | Type | Replacement | Details |\n|--------|-----|------|-------------|---------|\n| 🔵 Modify | MyDatabase | AWS::RDS::DBInstance | **True** | 🔵 **DBInstanceClass**: `db.t3.medium` -> `db.t3.large` |\n| 🔵 Modify | MyLambdaFunction | AWS::Lambda::Function | False | 🔵 **Runtime**: `nodejs18.x` -> `nodejs20.x` |\n| 🟢 Add | NewSecurityGroup | AWS::EC2::SecurityGroup | - | |\n| 🔴 Remove | OldRole | AWS::IAM::Role | - | |\n\n**Features:**\n- **Replacement column** highlights when CloudFormation will **destroy and recreate** a resource — critical for stateful resources like databases, EC2 instances, and file systems\n- **Drift banner** — if the stack has drifted, a warning with drifted resource details appears above the change set table so reviewers know the baseline state\n- **Comment upsert** — each stack's comment is updated in place on subsequent pushes instead of creating duplicates; uses an HTML marker for reliable find-and-replace\n- **Color-coded indicators**: 🟢 Added, 🔵 Modified, 🔴 Removed\n- **Inline values for small changes**: Shows `before -> after` directly in the table\n- **Collapsible details for large values**: IAM policies, tags, and other large JSON values are wrapped in expandable `<details>` elements to keep the table readable\n- **All attribute types supported**: Properties, Tags, Metadata, etc.\n- **HTML-escaped values**: Prevents XSS from property values\n\n### Environment variables used by the change set script\n- `STACK_NAME` (required) — Stack name to describe.\n- `CHANGE_SET_NAME` (default: same as `STACK_NAME`).\n- `AWS_REGION` — Region for CloudFormation API calls. The workflow sets this via the credentials action(s).\n- `GITHUB_TOKEN` (optional) — If set with `GITHUB_COMMENT_URL`, posts a PR comment.\n- `GITHUB_COMMENT_URL` (optional) — PR comments URL.\n- `GITHUB_STEP_SUMMARY` (optional) — When present, appends the HTML to the step summary file.\n- `IGNORE_LOGICAL_IDS` (optional) — Comma‑separated logical IDs to ignore (default includes `CDKMetadata`).\n- `IGNORE_RESOURCE_TYPES` (optional) — Comma‑separated resource types to ignore (e.g., `AWS::CDK::Metadata`).\n\n## Usage: CdkDiffIamTemplate\n\nEmit an IAM template you can deploy in your account for the Change Set workflow. Supports two modes:\n\n1. **External OIDC Role** — Reference an existing GitHub OIDC role (original behavior)\n2. **Self-Contained** — Create the GitHub OIDC provider and role within the same template (new)\n\n### Option 1: Using an Existing OIDC Role (External)\n\nUse this when you already have a GitHub OIDC provider and role set up in your account.\n\n#### With Projen\n\n```ts\nimport { awscdk } from 'projen';\nimport { CdkDiffIamTemplate } from '@jjrawlins/cdk-diff-pr-github-action';\n\nconst project = new awscdk.AwsCdkConstructLibrary({\n // ...\n});\n\nnew CdkDiffIamTemplate({\n project,\n roleName: 'cdk-diff-role',\n oidcRoleArn: 'arn:aws:iam::123456789012:role/github-oidc-role',\n oidcRegion: 'us-east-1',\n // Optional: custom output path (default: 'cdk-diff-workflow-iam-template.yaml')\n // outputPath: 'infra/cdk-diff-iam.yaml',\n});\n\nproject.synth();\n```\n\n#### Without Projen (Standalone Generator)\n\n```ts\nimport { CdkDiffIamTemplateGenerator } from '@jjrawlins/cdk-diff-pr-github-action';\nimport * as fs from 'fs';\n\nconst template = CdkDiffIamTemplateGenerator.generateTemplate({\n roleName: 'cdk-diff-role',\n oidcRoleArn: 'arn:aws:iam::123456789012:role/github-oidc-role',\n oidcRegion: 'us-east-1',\n});\n\nfs.writeFileSync('cdk-diff-iam-template.yaml', template);\n```\n\n### Option 2: Self-Contained Template (Create OIDC Role)\n\nUse this when you want a single template that creates everything needed — the GitHub OIDC provider, OIDC role, and changeset role. This simplifies deployment and pairs well with the `CdkDiffStackWorkflow`.\n\n#### With Projen\n\n```ts\nimport { awscdk } from 'projen';\nimport { CdkDiffIamTemplate } from '@jjrawlins/cdk-diff-pr-github-action';\n\nconst project = new awscdk.AwsCdkConstructLibrary({\n // ...\n});\n\nnew CdkDiffIamTemplate({\n project,\n roleName: 'CdkChangesetRole',\n createOidcRole: true,\n oidcRoleName: 'GitHubOIDCRole', // Optional, default: 'GitHubOIDCRole'\n githubOidc: {\n owner: 'my-org', // GitHub org or username\n repositories: ['infra-repo', 'app-repo'], // Repos allowed to assume roles\n branches: ['main', 'release/*'], // Branch patterns (default: ['*'])\n },\n // Optional: Skip OIDC provider creation if it already exists\n // skipOidcProviderCreation: true,\n});\n\nproject.synth();\n```\n\n#### Without Projen (Standalone Generator)\n\n```ts\nimport { CdkDiffIamTemplateGenerator } from '@jjrawlins/cdk-diff-pr-github-action';\nimport * as fs from 'fs';\n\nconst template = CdkDiffIamTemplateGenerator.generateTemplate({\n roleName: 'CdkChangesetRole',\n createOidcRole: true,\n oidcRoleName: 'GitHubOIDCRole',\n githubOidc: {\n owner: 'my-org',\n repositories: ['infra-repo'],\n branches: ['main'],\n },\n});\n\nfs.writeFileSync('cdk-diff-iam-template.yaml', template);\n```\n\n#### With Existing OIDC Provider (Skip Creation)\n\nIf your account already has a GitHub OIDC provider but you want the template to create the roles:\n\n```ts\nnew CdkDiffIamTemplate({\n project,\n roleName: 'CdkChangesetRole',\n createOidcRole: true,\n skipOidcProviderCreation: true, // Account already has OIDC provider\n githubOidc: {\n owner: 'my-org',\n repositories: ['*'], // All repos in org\n },\n});\n```\n\n### Deploy Task\n\nA Projen task is added for easy deployment:\n\n```bash\nnpx projen deploy-cdkdiff-iam-template -- --parameter-overrides GitHubOIDCRoleArn=... # plus any extra AWS CLI args\n```\n\n### CdkDiffIamTemplate Props\n\n| Property | Type | Description |\n|----------|------|-------------|\n| `roleName` | `string` | Name for the changeset IAM role (required) |\n| `oidcRoleArn` | `string?` | ARN of existing GitHub OIDC role. Required when `createOidcRole` is false. |\n| `oidcRegion` | `string?` | Region for OIDC trust condition. Required when `createOidcRole` is false. |\n| `createOidcRole` | `boolean?` | Create OIDC role within template (default: false) |\n| `oidcRoleName` | `string?` | Name of OIDC role to create (default: 'GitHubOIDCRole') |\n| `githubOidc` | `GitHubOidcConfig?` | GitHub OIDC config. Required when `createOidcRole` is true. |\n| `skipOidcProviderCreation` | `boolean?` | Skip OIDC provider if it exists (default: false) |\n| `outputPath` | `string?` | Template output path (default: 'cdk-diff-workflow-iam-template.yaml') |\n\n### What the Template Creates\n\n**External OIDC Role mode:**\n- Parameter `GitHubOIDCRoleArn` — ARN of your existing GitHub OIDC role\n- IAM role `CdkChangesetRole` with minimal permissions for change set operations\n- Outputs: `CdkChangesetRoleArn`, `CdkChangesetRoleName`\n\n**Self-Contained mode (`createOidcRole: true`):**\n- GitHub OIDC Provider (unless `skipOidcProviderCreation: true`)\n- IAM role `GitHubOIDCRole` with trust policy for GitHub Actions\n- IAM role `CdkChangesetRole` with minimal permissions (trusts the OIDC role)\n- Outputs: `GitHubOIDCProviderArn`, `GitHubOIDCRoleArn`, `GitHubOIDCRoleName`, `CdkChangesetRoleArn`, `CdkChangesetRoleName`\n\n**Changeset Role Permissions:**\n- CloudFormation Change Set operations\n- Access to CDK bootstrap S3 buckets and SSM parameters\n- `iam:PassRole` to `cloudformation.amazonaws.com`\n\nUse the created changeset role ARN as `changesetRoleToAssumeArn` in `CdkDiffStackWorkflow`.\n\n---\n\n## Usage: CdkDriftDetectionWorkflow\n\n`CdkDriftDetectionWorkflow` creates a single workflow file (default `drift-detection.yml`) that can run on a schedule and via manual dispatch. It generates a helper script at `.github/workflows/scripts/detect-drift.ts` (by default) that uses AWS SDK v3 to run drift detection, write optional machine‑readable JSON, and print an HTML report for the Step Summary.\n\nExample `.projenrc.ts`:\n\n```ts\nimport { awscdk } from 'projen';\nimport { CdkDriftDetectionWorkflow } from '@jjrawlins/cdk-diff-pr-github-action';\n\nconst project = new awscdk.AwsCdkConstructLibrary({ github: true, /* ... */ });\n\nnew CdkDriftDetectionWorkflow({\n project,\n workflowName: 'Drift Detection', // optional; file name derived as 'drift-detection.yml'\n schedule: '0 1 * * *', // optional cron\n createIssues: true, // default true; create/update issue when drift detected on schedule\n oidcRoleArn: 'arn:aws:iam::123456789012:role/github-oidc-role',\n oidcRegion: 'us-east-1',\n // Optional: Node version (default '24.x')\n // nodeVersion: '24.x',\n // Optional: Where to place the helper script (default '.github/workflows/scripts/detect-drift.ts')\n // scriptOutputPath: '.github/workflows/scripts/detect-drift.ts',\n stacks: [\n {\n stackName: 'MyAppStack-Prod',\n driftDetectionRoleToAssumeArn: 'arn:aws:iam::123456789012:role/cdk-drift-role',\n driftDetectionRoleToAssumeRegion: 'us-east-1',\n // failOnDrift: true, // optional (default true)\n },\n ],\n});\n\nproject.synth();\n```\n\n### CdkDriftDetectionWorkflow props\n- `project` (required) — Your Projen project instance.\n- `stacks` (required) — Array of stacks to check.\n- `oidcRoleArn` (required) — Default OIDC role ARN used before chaining into per‑stack drift roles.\n- `oidcRegion` (required) — Default OIDC region.\n- `workflowName` (optional, default `'drift-detection'`) — Human‑friendly workflow name; the file name is derived in kebab‑case.\n- `schedule` (optional) — Cron expression for automatic runs.\n- `createIssues` (optional, default `true`) — When true, scheduled runs will create/update a GitHub issue if drift is detected.\n- `nodeVersion` (optional, default `'24.x'`) — Node.js version for the runner.\n- `scriptOutputPath` (optional, default `'.github/workflows/scripts/detect-drift.ts'`) — Where to write the helper script.\n\n### Per‑stack fields\n- `stackName` (required) — The full CloudFormation stack name.\n- `driftDetectionRoleToAssumeArn` (required) — Role to assume (after OIDC) for making drift API calls.\n- `driftDetectionRoleToAssumeRegion` (required) — Region for that role and API calls.\n- `failOnDrift` (optional, default `true`) — Intended to fail the detection step on drift. The provided script exits with non‑zero when drift is found; the job continues to allow artifact upload and issue creation.\n\n### What gets generated\n- `.github/workflows/<kebab(workflowName)>.yml` — A workflow with one job per stack plus a final summary job.\n- `.github/workflows/scripts/detect-drift.ts` — Helper script that:\n - Starts drift detection and polls until completion\n - Lists non‑`IN_SYNC` resources and builds an HTML report\n - Writes optional JSON to `DRIFT_DETECTION_OUTPUT` when set\n - Prints to stdout and appends to the GitHub Step Summary when available\n\n### Artifacts and summary\n- Each stack job uploads `drift-results-<stack>.json` (if produced).\n- A final `Drift Detection Summary` job downloads all artifacts and prints a consolidated summary.\n\n### Manual dispatch\n- The workflow exposes an input named `stack` with choices including each configured stack and an `all` option.\n- Choose a specific stack to run drift detection for that stack only, or select `all` (or leave the input empty) to run all stacks.\n\nNote: The default workflow does not post PR comments for drift. It can create/update an Issue on scheduled runs when `createIssues` is `true`.\n\n### Post-notification steps (e.g., Slack)\n\nYou can add your own GitHub Action steps to run after the drift detection step for each stack using `postGitHubSteps`.\nProvide your own Slack payload/markdown (this library no longer generates a payload step for you).\n\nOption A: slackapi/slack-github-action (Incoming Webhook, official syntax)\n\n```ts\nnew CdkDriftDetectionWorkflow({\n project,\n oidcRoleArn: 'arn:aws:iam::123456789012:role/github-oidc-role',\n oidcRegion: 'us-east-1',\n stacks: [/* ... */],\n postGitHubSteps: ({ stack }) => {\n // Build a descriptive name per stack\n const name = `Notify Slack (${stack} post-drift)`;\n const step = {\n name,\n uses: 'slackapi/slack-github-action@v2.1.1',\n // by default, post steps run only when drift is detected; you can override `if`\n if: \"always() && steps.drift.outcome == 'failure'\",\n // Use official inputs: webhook + webhook-type, and a YAML payload with blocks\n with: {\n webhook: '${{ secrets.CDK_NOTIFICATIONS_SLACK_WEBHOOK }}',\n 'webhook-type': 'incoming-webhook',\n payload: [\n 'text: \"** ${{ env.STACK_NAME }} ** has drifted!\"',\n 'blocks:',\n ' - type: \"section\"',\n ' text:',\n ' type: \"mrkdwn\"',\n ' text: \"*Stack:* ${{ env.STACK_NAME }} (region ${{ env.AWS_REGION }}) has drifted:exclamation:\"',\n ' - type: \"section\"',\n ' fields:',\n ' - type: \"mrkdwn\"',\n ' text: \"*Stack ARN*\\\\n${{ steps.drift.outputs.stack-arn }}\"',\n ' - type: \"mrkdwn\"',\n ' text: \"*Issue*\\\\n<${{ github.server_url }}/${{ github.repository }}/issues/${{ steps.issue.outputs.result }}|#${{ steps.issue.outputs.result }}>\"',\n ].join('\\n'),\n },\n };\n return [step];\n },\n});\n```\n\nNote: The Issue link requires `createIssues: true` (default) so that the `Create Issue on Drift` step runs before this Slack step and exposes `steps.issue.outputs.result`. This library orders the steps accordingly.\n\nDetails:\n- `postGitHubSteps` can be:\n - an array of step objects, or\n - a factory function `({ stack }) => step | step[]`.\n- Each step you provide is inserted after the results are uploaded.\n- Default condition: if you do not set `if` on your step, it will default to `always() && steps.drift.outcome == 'failure'`.\n- Available context/env you can use:\n - `${{ env.STACK_NAME }}`, `${{ env.DRIFT_DETECTION_OUTPUT }}`\n - `${{ steps.drift.outcome }}` — success/failure of the detect step\n - `${{ steps.drift.outputs.stack-arn }}` — Stack ARN resolved at runtime\n - `${{ steps.issue.outputs.result }}` — Issue number if the workflow created/found one (empty when not applicable)\n```\n\n## Usage: CdkDriftIamTemplate\n\nEmit an example IAM template you can deploy in your account for the Drift Detection workflow.\n\n### With Projen\n\n```ts\nimport { awscdk } from 'projen';\nimport { CdkDriftIamTemplate } from '@jjrawlins/cdk-diff-pr-github-action';\n\nconst project = new awscdk.AwsCdkConstructLibrary({\n // ...\n});\n\nnew CdkDriftIamTemplate({\n project,\n roleName: 'cdk-drift-role',\n oidcRoleArn: 'arn:aws:iam::123456789012:role/github-oidc-role',\n oidcRegion: 'us-east-1',\n // Optional: custom output path (default: 'cdk-drift-workflow-iam-template.yaml')\n // outputPath: 'infra/cdk-drift-iam.yaml',\n});\n\nproject.synth();\n```\n\nA Projen task is also added:\n\n```bash\nnpx projen deploy-cdkdrift-iam-template -- --parameter-overrides GitHubOIDCRoleArn=... # plus any extra AWS CLI args\n```\n\n### Without Projen (Standalone Generator)\n\n```ts\nimport { CdkDriftIamTemplateGenerator } from '@jjrawlins/cdk-diff-pr-github-action';\nimport * as fs from 'fs';\n\nconst template = CdkDriftIamTemplateGenerator.generateTemplate({\n roleName: 'cdk-drift-role',\n oidcRoleArn: 'arn:aws:iam::123456789012:role/github-oidc-role',\n oidcRegion: 'us-east-1',\n});\n\nfs.writeFileSync('cdk-drift-iam-template.yaml', template);\n\n// Get the deploy command\nconst deployCmd = CdkDriftIamTemplateGenerator.generateDeployCommand('cdk-drift-iam-template.yaml');\nconsole.log('Deploy with:', deployCmd);\n```\n\n### What the template defines\n\n- Parameter `GitHubOIDCRoleArn` with a default from `oidcRoleArn` — the ARN of your existing GitHub OIDC role allowed to assume this drift role.\n- IAM role `CdkDriftRole` with minimal permissions for CloudFormation drift detection operations.\n- Outputs exporting the role name and ARN.\n\n---\n\n## Usage: CdkDiffIamTemplateStackSet (Org-Wide Deployment)\n\n`CdkDiffIamTemplateStackSet` creates a CloudFormation StackSet template for deploying GitHub OIDC provider, OIDC role, and CDK diff/drift IAM roles across an entire AWS Organization. This is the recommended approach for organizations that want to enable CDK diff/drift workflows across multiple accounts.\n\n### Architecture\n\nEach account in your organization gets:\n- **GitHub OIDC Provider** — Authenticates GitHub Actions workflows\n- **GitHubOIDCRole** — Trusts the OIDC provider with repo/branch restrictions\n- **CdkChangesetRole** — For PR change set previews (trusts GitHubOIDCRole)\n- **CdkDriftRole** — For drift detection (trusts GitHubOIDCRole)\n\nThis is a self-contained deployment with **no role chaining required**.\n\n### With Projen\n\n```ts\nimport { awscdk } from 'projen';\nimport { CdkDiffIamTemplateStackSet } from '@jjrawlins/cdk-diff-pr-github-action';\n\nconst project = new awscdk.AwsCdkConstructLibrary({ /* ... */ });\n\nnew CdkDiffIamTemplateStackSet({\n project,\n githubOidc: {\n owner: 'my-org', // GitHub org or username\n repositories: ['infra-repo', 'app-repo'], // Repos allowed to assume roles\n branches: ['main', 'release/*'], // Branch patterns (default: ['*'])\n },\n targetOrganizationalUnitIds: ['ou-xxxx-xxxxxxxx'], // Target OUs\n regions: ['us-east-1', 'eu-west-1'], // Target regions\n // Optional settings:\n // oidcRoleName: 'GitHubOIDCRole', // default\n // changesetRoleName: 'CdkChangesetRole', // default\n // driftRoleName: 'CdkDriftRole', // default\n // roleSelection: StackSetRoleSelection.BOTH, // BOTH, CHANGESET_ONLY, or DRIFT_ONLY\n // delegatedAdmin: true, // Use --call-as DELEGATED_ADMIN (default: true)\n});\n\nproject.synth();\n```\n\nThis creates:\n- `cdk-diff-workflow-stackset-template.yaml` — CloudFormation template\n- Projen tasks for StackSet management\n\n**Projen tasks:**\n```bash\nnpx projen stackset-create # Create the StackSet\nnpx projen stackset-update # Update the StackSet template\nnpx projen stackset-deploy-instances # Deploy to target OUs/regions\nnpx projen stackset-delete-instances # Remove stack instances\nnpx projen stackset-delete # Delete the StackSet\nnpx projen stackset-describe # Show StackSet status\nnpx projen stackset-list-instances # List all instances\n```\n\n### Without Projen (Standalone Generator)\n\nFor non-Projen projects, use `CdkDiffIamTemplateStackSetGenerator` directly:\n\n```ts\nimport {\n CdkDiffIamTemplateStackSetGenerator\n} from '@jjrawlins/cdk-diff-pr-github-action';\nimport * as fs from 'fs';\n\n// Generate the CloudFormation template\nconst template = CdkDiffIamTemplateStackSetGenerator.generateTemplate({\n githubOidc: {\n owner: 'my-org',\n repositories: ['infra-repo'],\n branches: ['main'],\n },\n});\n\n// Write to file\nfs.writeFileSync('stackset-template.yaml', template);\n\n// Get AWS CLI commands for StackSet operations\nconst commands = CdkDiffIamTemplateStackSetGenerator.generateCommands({\n stackSetName: 'cdk-diff-workflow-iam-stackset',\n templatePath: 'stackset-template.yaml',\n targetOrganizationalUnitIds: ['ou-xxxx-xxxxxxxx'],\n regions: ['us-east-1'],\n});\n\nconsole.log('Create StackSet:', commands['stackset-create']);\nconsole.log('Deploy instances:', commands['stackset-deploy-instances']);\n```\n\n### GitHub Actions Workflow (Simplified)\n\nWith per-account OIDC, your workflow is simplified — no role chaining needed:\n\n```yaml\njobs:\n diff:\n runs-on: ubuntu-latest\n permissions:\n id-token: write\n contents: read\n steps:\n - uses: actions/checkout@v4\n\n - uses: aws-actions/configure-aws-credentials@v4\n with:\n role-to-assume: arn:aws:iam::${{ env.ACCOUNT_ID }}:role/GitHubOIDCRole\n aws-region: us-east-1\n\n - name: Assume Changeset Role\n run: |\n CREDS=$(aws sts assume-role \\\n --role-arn arn:aws:iam::${{ env.ACCOUNT_ID }}:role/CdkChangesetRole \\\n --role-session-name changeset-session)\n # Export credentials...\n```\n\n### GitHubOidcConfig options\n\n| Property | Description |\n|----------|-------------|\n| `owner` | GitHub organization or username (required) |\n| `repositories` | Array of repo names, or `['*']` for all repos (required) |\n| `branches` | Array of branch patterns (default: `['*']`) |\n| `additionalClaims` | Extra OIDC claims like `['pull_request', 'environment:production']` |\n\n---\n\n## Testing\n\nThis repository includes Jest tests that snapshot the synthesized outputs from Projen and assert that:\n- Diff workflows are created per stack and contain all expected steps.\n- Drift detection workflow produces one job per stack and a summary job.\n- Only one helper script file is generated per workflow type.\n- Per‑stack OIDC overrides (where supported) are respected.\n- Helpful validation errors are thrown for missing OIDC settings.\n- The IAM template files contain the expected resources and outputs.\n\nRun tests with:\n\n```bash\nyarn test\n```\n\n## Notes\n- This package assumes your repository is configured with GitHub Actions and that you have a GitHub OIDC role configured in AWS.\n- The generated scripts use the AWS SDK v3 for CloudFormation and, where applicable, the GitHub REST API.\n"
|
|
3510
|
+
"markdown": "# cdk-diff-pr-github-action\n\nA [Projen](https://projen.io/) construct library that surfaces **CloudFormation change set diffs and drift status directly on your pull requests** so reviewers can see exactly what will change before merging.\n\n## Why this exists\n\n`cdk diff` output disappears into CI logs that nobody reads. Meanwhile, a single property change on an RDS instance or EC2 \"pet\" server can trigger a **resource replacement** — destroying the database or instance and recreating it from scratch. If that replacement slips through code review unnoticed, the result is data loss and downtime.\n\nThis construct was built to make those dangerous changes impossible to miss:\n\n- **Replacement column front and center** — Every change set row shows whether CloudFormation will modify the resource in place or **replace** it, with before/after property values so reviewers can understand *why*.\n- **Comment appears on the PR itself** — No digging through workflow logs. The diff table is posted (and updated in place) as a PR comment and in the GitHub Step Summary.\n- **Drift banner** — If the stack has drifted from its template, a warning banner is prepended to the comment so reviewers know the baseline is already out of sync.\n\nIf you have ever lost an EC2 instance, an RDS database, or an ElastiCache cluster to an unexpected CloudFormation replacement, this tool is for you.\n\n---\n\nA library that provides GitHub workflows and IAM templates for:\n- Creating CloudFormation Change Sets for your CDK stacks on pull requests and commenting a formatted diff back on the PR.\n- Detecting CloudFormation drift on a schedule or manual trigger and producing a consolidated summary (optionally creating an issue).\n- Deploying IAM roles across AWS Organizations using StackSets.\n\nIt also provides ready-to-deploy IAM templates with the minimal permissions required for each workflow.\n\n**Works with or without Projen** -- The StackSet generator can be used standalone in any Node.js project.\n\nThis package exposes five constructs:\n\n- `CdkDiffStackWorkflow` — Generates one GitHub Actions workflow per stack to create a change set and render the diff back to the PR and Step Summary.\n- `CdkDiffIamTemplate` — Emits a CloudFormation template file with minimal permissions for the Change Set workflow.\n- `CdkDriftDetectionWorkflow` — Generates a GitHub Actions workflow to detect CloudFormation drift per stack, upload machine‑readable results, and aggregate a summary.\n- `CdkDriftIamTemplate` — Emits a CloudFormation template file with minimal permissions for the Drift Detection workflow.\n- `CdkDiffIamTemplateStackSet` — Creates a CloudFormation StackSet template for org-wide deployment of GitHub OIDC and IAM roles (Projen integration).\n- `CdkDiffIamTemplateStackSetGenerator` — Pure generator class for StackSet templates (no Projen dependency).\n\n## Quick start\n\n1) Add the constructs to your Projen project (in `.projenrc.ts`).\n2) Synthesize with `npx projen`.\n3) Commit the generated files.\n4) Open a pull request or run the drift detection workflow.\n\n## End-to-end example\n\nA realistic setup for a CDK Pipelines project with multiple stages and accounts. This generates one diff workflow per stack, a shared IAM template, and a scheduled drift detection workflow.\n\n```ts\n// .projenrc.ts\nimport { awscdk } from 'projen';\nimport {\n CdkDiffStackWorkflow,\n CdkDiffIamTemplate,\n CdkDriftDetectionWorkflow,\n} from '@jjrawlins/cdk-diff-pr-github-action';\n\nconst project = new awscdk.AwsCdkTypeScriptApp({\n name: 'my-data-platform',\n defaultReleaseBranch: 'main',\n cdkVersion: '2.170.0',\n github: true,\n});\n\n// --- Change Set Diff Workflows (one per stack) ---\n// stackName is the CDK construct path used with `cdk deploy <target>`.\n// For CDK Pipelines, this is typically: pipeline/Stage/Stack\n// The construct automatically resolves the real CloudFormation stack name\n// from cdk.out at runtime (e.g., \"my-stage-dev-my-stack\").\n\nnew CdkDiffStackWorkflow({\n project,\n oidcRoleArn: 'arn:aws:iam::111111111111:role/GitHubOIDCRole',\n oidcRegion: 'us-east-1',\n stacks: [\n {\n stackName: 'my-pipeline/dev/DatabaseStack',\n changesetRoleToAssumeArn: 'arn:aws:iam::111111111111:role/CdkChangesetRole',\n changesetRoleToAssumeRegion: 'us-east-1',\n },\n {\n stackName: 'my-pipeline/dev/ComputeStack',\n changesetRoleToAssumeArn: 'arn:aws:iam::111111111111:role/CdkChangesetRole',\n changesetRoleToAssumeRegion: 'us-east-1',\n },\n {\n // Cross-account: prod stacks can use a different OIDC role and region\n stackName: 'my-pipeline/prod/DatabaseStack',\n changesetRoleToAssumeArn: 'arn:aws:iam::222222222222:role/CdkChangesetRole',\n changesetRoleToAssumeRegion: 'us-east-1',\n oidcRoleArn: 'arn:aws:iam::222222222222:role/GitHubOIDCRole',\n oidcRegion: 'us-east-1',\n },\n ],\n});\n\n// --- IAM Template (deploy once per account) ---\nnew CdkDiffIamTemplate({\n project,\n roleName: 'CdkChangesetRole',\n createOidcRole: true,\n githubOidc: {\n owner: 'my-org',\n repositories: ['my-data-platform'],\n branches: ['main'],\n },\n});\n\n// --- Drift Detection (scheduled + manual) ---\nnew CdkDriftDetectionWorkflow({\n project,\n schedule: '0 6 * * 1', // Every Monday at 6 AM UTC\n oidcRoleArn: 'arn:aws:iam::111111111111:role/GitHubOIDCRole',\n oidcRegion: 'us-east-1',\n stacks: [\n {\n stackName: 'dev-DatabaseStack',\n driftDetectionRoleToAssumeArn: 'arn:aws:iam::111111111111:role/CdkDriftRole',\n driftDetectionRoleToAssumeRegion: 'us-east-1',\n },\n ],\n});\n\nproject.synth();\n```\n\nAfter `npx projen`, this generates:\n- `.github/workflows/diff-my-pipeline-dev-databasestack.yml`\n- `.github/workflows/diff-my-pipeline-dev-computestack.yml`\n- `.github/workflows/diff-my-pipeline-prod-databasestack.yml`\n- `.github/workflows/scripts/describe-cfn-changeset.ts`\n- `.github/workflows/drift-detection.yml`\n- `.github/workflows/scripts/detect-drift.ts`\n- `cdk-diff-workflow-iam-template.yaml`\n\nWhen a pull request is opened, each diff workflow runs automatically and posts a comment like this:\n\n| Action | ID | Type | Replacement | Details |\n|--------|-----|------|-------------|---------|\n| 🔵 Modify | MyDatabase | AWS::RDS::DBInstance | **True** | 🔵 **DBInstanceClass**: `db.t3.medium` -> `db.t3.large` |\n| 🔵 Modify | MyFunction | AWS::Lambda::Function | False | 🔵 **Runtime**: `nodejs18.x` -> `nodejs20.x` |\n\nThe **Replacement: True** on the RDS instance is exactly the kind of change this tool is designed to catch before it reaches production.\n\n## Usage: CdkDiffStackWorkflow\n\n`CdkDiffStackWorkflow` renders a workflow per stack named `diff-<StackName>.yml` under `.github/workflows/`. It also generates a helper script at `.github/workflows/scripts/describe-cfn-changeset.ts` that formats the change set output and takes care of posting the PR comment and Step Summary.\n\nExample `.projenrc.ts`:\n\n```ts\nimport { awscdk } from 'projen';\nimport { CdkDiffStackWorkflow } from '@jjrawlins/cdk-diff-pr-github-action';\n\nconst project = new awscdk.AwsCdkConstructLibrary({\n // ... your usual settings ...\n workflowName: 'my-lib',\n defaultReleaseBranch: 'main',\n cdkVersion: '2.85.0',\n github: true,\n});\n\nnew CdkDiffStackWorkflow({\n project,\n stacks: [\n {\n stackName: 'MyAppStack',\n changesetRoleToAssumeArn: 'arn:aws:iam::123456789012:role/cdk-diff-role',\n changesetRoleToAssumeRegion: 'us-east-1',\n // Optional per‑stack OIDC override (if not using the defaults below)\n // oidcRoleArn: 'arn:aws:iam::123456789012:role/github-oidc-role',\n // oidcRegion: 'us-east-1',\n },\n ],\n // Default OIDC role/region used by all stacks unless overridden per‑stack\n oidcRoleArn: 'arn:aws:iam::123456789012:role/github-oidc-role',\n oidcRegion: 'us-east-1',\n // Optional: Node version used in the workflow (default: '24.x')\n // nodeVersion: '24.x',\n // Optional: Yarn command to run CDK (default: 'cdk')\n // cdkYarnCommand: 'cdk',\n // Optional: Where to place the helper script (default: '.github/workflows/scripts/describe-cfn-changeset.ts')\n // scriptOutputPath: '.github/workflows/scripts/describe-cfn-changeset.ts',\n});\n\nproject.synth();\n```\n\n### CdkDiffStackWorkflow props\n- `project` (required) — Your Projen project instance.\n- `stacks` (required) — Array of stack entries.\n- `oidcRoleArn` (required unless provided per‑stack) — Default OIDC role ARN.\n- `oidcRegion` (required unless provided per‑stack) — Default OIDC region.\n- `nodeVersion` (optional, default `'24.x'`) — Node.js version for the workflow runner.\n- `cdkYarnCommand` (optional, default `'cdk'`) — Yarn script/command to invoke CDK.\n- `scriptOutputPath` (optional, default `'.github/workflows/scripts/describe-cfn-changeset.ts'`) — Where to write the helper script.\n- `workingDirectory` (optional) — Subdirectory where the CDK app lives (e.g., `'infra'`). Sets `defaults.run.working-directory` on all jobs so `install`, `synth`, and `deploy` steps run in that directory. The describe-changeset script path is automatically prefixed with `$GITHUB_WORKSPACE/` and `NODE_PATH` is set to resolve modules from the working directory.\n\nIf neither top‑level OIDC defaults nor all per‑stack values are supplied, the construct throws a helpful error.\n\n### Stack item fields\n- `stackName` (required) — The CDK stack name to create the change set for.\n- `changesetRoleToAssumeArn` (required) — The ARN of the role used to create the change set (role chaining after OIDC).\n- `changesetRoleToAssumeRegion` (required) — The region for that role.\n- `oidcRoleArn` (optional) — Per‑stack override for the OIDC role.\n- `oidcRegion` (optional) — Per‑stack override for the OIDC region.\n\n### What gets generated\n- `.github/workflows/diff-<StackName>.yml` — One workflow per stack, triggered on PR open/sync/reopen.\n- `.github/workflows/scripts/describe-cfn-changeset.ts` — A helper script that:\n - Polls `DescribeChangeSet` until terminal\n - Filters out ignorable logical IDs or resource types using environment variables `IGNORE_LOGICAL_IDS` and `IGNORE_RESOURCE_TYPES`\n - Renders an HTML table with actions, logical IDs, types, replacements, and changed properties\n - **Checks cached drift status** via `DescribeStacks` — if the stack has drifted, a warning banner with drifted resource details is prepended to the output (non-fatal; degrades gracefully if IAM permissions are missing)\n - Prints the HTML and appends to the GitHub Step Summary\n - **Upserts the PR comment** — uses an HTML marker (`<!-- cdk-diff:stack:STACK_NAME -->`) to find and update an existing comment instead of creating duplicates on every push\n\n### Change Set Output Format\n\nThe change set script uses the CloudFormation `IncludePropertyValues` API feature to show **actual before/after values** for changed properties, not just property names.\n\n**Example PR comment:**\n\n> **Warning: Stack has drifted (2 resources out of sync)**\n>\n> Last drift check: 2025-01-15T10:30:00.000Z\n>\n> <details><summary>View drifted resources</summary>\n>\n> | Resource | Type | Drift Status |\n> |----------|------|-------------|\n> | MySecurityGroup | AWS::EC2::SecurityGroup | MODIFIED |\n> | OldBucket | AWS::S3::Bucket | DELETED |\n> </details>\n\n| Action | ID | Type | Replacement | Details |\n|--------|-----|------|-------------|---------|\n| 🔵 Modify | MyDatabase | AWS::RDS::DBInstance | **True** | 🔵 **DBInstanceClass**: `db.t3.medium` -> `db.t3.large` |\n| 🔵 Modify | MyLambdaFunction | AWS::Lambda::Function | False | 🔵 **Runtime**: `nodejs18.x` -> `nodejs20.x` |\n| 🟢 Add | NewSecurityGroup | AWS::EC2::SecurityGroup | - | |\n| 🔴 Remove | OldRole | AWS::IAM::Role | - | |\n\n**Features:**\n- **Replacement column** highlights when CloudFormation will **destroy and recreate** a resource — critical for stateful resources like databases, EC2 instances, and file systems\n- **Drift banner** — if the stack has drifted, a warning with drifted resource details appears above the change set table so reviewers know the baseline state\n- **Comment upsert** — each stack's comment is updated in place on subsequent pushes instead of creating duplicates; uses an HTML marker for reliable find-and-replace\n- **Color-coded indicators**: 🟢 Added, 🔵 Modified, 🔴 Removed\n- **Inline values for small changes**: Shows `before -> after` directly in the table\n- **Collapsible details for large values**: IAM policies, tags, and other large JSON values are wrapped in expandable `<details>` elements to keep the table readable\n- **All attribute types supported**: Properties, Tags, Metadata, etc.\n- **HTML-escaped values**: Prevents XSS from property values\n\n### Environment variables used by the change set script\n- `STACK_NAME` (required) — Stack name to describe.\n- `CHANGE_SET_NAME` (default: same as `STACK_NAME`).\n- `AWS_REGION` — Region for CloudFormation API calls. The workflow sets this via the credentials action(s).\n- `GITHUB_TOKEN` (optional) — If set with `GITHUB_COMMENT_URL`, posts a PR comment.\n- `GITHUB_COMMENT_URL` (optional) — PR comments URL.\n- `GITHUB_STEP_SUMMARY` (optional) — When present, appends the HTML to the step summary file.\n- `IGNORE_LOGICAL_IDS` (optional) — Comma‑separated logical IDs to ignore (default includes `CDKMetadata`).\n- `IGNORE_RESOURCE_TYPES` (optional) — Comma‑separated resource types to ignore (e.g., `AWS::CDK::Metadata`).\n\n## Usage: CdkDiffIamTemplate\n\nEmit an IAM template you can deploy in your account for the Change Set workflow. Supports two modes:\n\n1. **External OIDC Role** — Reference an existing GitHub OIDC role (original behavior)\n2. **Self-Contained** — Create the GitHub OIDC provider and role within the same template (new)\n\n### Option 1: Using an Existing OIDC Role (External)\n\nUse this when you already have a GitHub OIDC provider and role set up in your account.\n\n#### With Projen\n\n```ts\nimport { awscdk } from 'projen';\nimport { CdkDiffIamTemplate } from '@jjrawlins/cdk-diff-pr-github-action';\n\nconst project = new awscdk.AwsCdkConstructLibrary({\n // ...\n});\n\nnew CdkDiffIamTemplate({\n project,\n roleName: 'cdk-diff-role',\n oidcRoleArn: 'arn:aws:iam::123456789012:role/github-oidc-role',\n oidcRegion: 'us-east-1',\n // Optional: custom output path (default: 'cdk-diff-workflow-iam-template.yaml')\n // outputPath: 'infra/cdk-diff-iam.yaml',\n});\n\nproject.synth();\n```\n\n#### Without Projen (Standalone Generator)\n\n```ts\nimport { CdkDiffIamTemplateGenerator } from '@jjrawlins/cdk-diff-pr-github-action';\nimport * as fs from 'fs';\n\nconst template = CdkDiffIamTemplateGenerator.generateTemplate({\n roleName: 'cdk-diff-role',\n oidcRoleArn: 'arn:aws:iam::123456789012:role/github-oidc-role',\n oidcRegion: 'us-east-1',\n});\n\nfs.writeFileSync('cdk-diff-iam-template.yaml', template);\n```\n\n### Option 2: Self-Contained Template (Create OIDC Role)\n\nUse this when you want a single template that creates everything needed — the GitHub OIDC provider, OIDC role, and changeset role. This simplifies deployment and pairs well with the `CdkDiffStackWorkflow`.\n\n#### With Projen\n\n```ts\nimport { awscdk } from 'projen';\nimport { CdkDiffIamTemplate } from '@jjrawlins/cdk-diff-pr-github-action';\n\nconst project = new awscdk.AwsCdkConstructLibrary({\n // ...\n});\n\nnew CdkDiffIamTemplate({\n project,\n roleName: 'CdkChangesetRole',\n createOidcRole: true,\n oidcRoleName: 'GitHubOIDCRole', // Optional, default: 'GitHubOIDCRole'\n githubOidc: {\n owner: 'my-org', // GitHub org or username\n repositories: ['infra-repo', 'app-repo'], // Repos allowed to assume roles\n branches: ['main', 'release/*'], // Branch patterns (default: ['*'])\n },\n // Optional: Skip OIDC provider creation if it already exists\n // skipOidcProviderCreation: true,\n});\n\nproject.synth();\n```\n\n#### Without Projen (Standalone Generator)\n\n```ts\nimport { CdkDiffIamTemplateGenerator } from '@jjrawlins/cdk-diff-pr-github-action';\nimport * as fs from 'fs';\n\nconst template = CdkDiffIamTemplateGenerator.generateTemplate({\n roleName: 'CdkChangesetRole',\n createOidcRole: true,\n oidcRoleName: 'GitHubOIDCRole',\n githubOidc: {\n owner: 'my-org',\n repositories: ['infra-repo'],\n branches: ['main'],\n },\n});\n\nfs.writeFileSync('cdk-diff-iam-template.yaml', template);\n```\n\n#### With Existing OIDC Provider (Skip Creation)\n\nIf your account already has a GitHub OIDC provider but you want the template to create the roles:\n\n```ts\nnew CdkDiffIamTemplate({\n project,\n roleName: 'CdkChangesetRole',\n createOidcRole: true,\n skipOidcProviderCreation: true, // Account already has OIDC provider\n githubOidc: {\n owner: 'my-org',\n repositories: ['*'], // All repos in org\n },\n});\n```\n\n### Deploy Task\n\nA Projen task is added for easy deployment:\n\n```bash\nnpx projen deploy-cdkdiff-iam-template -- --parameter-overrides GitHubOIDCRoleArn=... # plus any extra AWS CLI args\n```\n\n### CdkDiffIamTemplate Props\n\n| Property | Type | Description |\n|----------|------|-------------|\n| `roleName` | `string` | Name for the changeset IAM role (required) |\n| `oidcRoleArn` | `string?` | ARN of existing GitHub OIDC role. Required when `createOidcRole` is false. |\n| `oidcRegion` | `string?` | Region for OIDC trust condition. Required when `createOidcRole` is false. |\n| `createOidcRole` | `boolean?` | Create OIDC role within template (default: false) |\n| `oidcRoleName` | `string?` | Name of OIDC role to create (default: 'GitHubOIDCRole') |\n| `githubOidc` | `GitHubOidcConfig?` | GitHub OIDC config. Required when `createOidcRole` is true. |\n| `skipOidcProviderCreation` | `boolean?` | Skip OIDC provider if it exists (default: false) |\n| `outputPath` | `string?` | Template output path (default: 'cdk-diff-workflow-iam-template.yaml') |\n\n### What the Template Creates\n\n**External OIDC Role mode:**\n- Parameter `GitHubOIDCRoleArn` — ARN of your existing GitHub OIDC role\n- IAM role `CdkChangesetRole` with minimal permissions for change set operations\n- Outputs: `CdkChangesetRoleArn`, `CdkChangesetRoleName`\n\n**Self-Contained mode (`createOidcRole: true`):**\n- GitHub OIDC Provider (unless `skipOidcProviderCreation: true`)\n- IAM role `GitHubOIDCRole` with trust policy for GitHub Actions\n- IAM role `CdkChangesetRole` with minimal permissions (trusts the OIDC role)\n- Outputs: `GitHubOIDCProviderArn`, `GitHubOIDCRoleArn`, `GitHubOIDCRoleName`, `CdkChangesetRoleArn`, `CdkChangesetRoleName`\n\n**Changeset Role Permissions:**\n- CloudFormation Change Set operations\n- Access to CDK bootstrap S3 buckets and SSM parameters\n- `iam:PassRole` to `cloudformation.amazonaws.com`\n\nUse the created changeset role ARN as `changesetRoleToAssumeArn` in `CdkDiffStackWorkflow`.\n\n---\n\n## Usage: CdkDriftDetectionWorkflow\n\n`CdkDriftDetectionWorkflow` creates a single workflow file (default `drift-detection.yml`) that can run on a schedule and via manual dispatch. It generates a helper script at `.github/workflows/scripts/detect-drift.ts` (by default) that uses AWS SDK v3 to run drift detection, write optional machine‑readable JSON, and print an HTML report for the Step Summary.\n\nExample `.projenrc.ts`:\n\n```ts\nimport { awscdk } from 'projen';\nimport { CdkDriftDetectionWorkflow } from '@jjrawlins/cdk-diff-pr-github-action';\n\nconst project = new awscdk.AwsCdkConstructLibrary({ github: true, /* ... */ });\n\nnew CdkDriftDetectionWorkflow({\n project,\n workflowName: 'Drift Detection', // optional; file name derived as 'drift-detection.yml'\n schedule: '0 1 * * *', // optional cron\n createIssues: true, // default true; create/update issue when drift detected on schedule\n oidcRoleArn: 'arn:aws:iam::123456789012:role/github-oidc-role',\n oidcRegion: 'us-east-1',\n // Optional: Node version (default '24.x')\n // nodeVersion: '24.x',\n // Optional: Where to place the helper script (default '.github/workflows/scripts/detect-drift.ts')\n // scriptOutputPath: '.github/workflows/scripts/detect-drift.ts',\n stacks: [\n {\n stackName: 'MyAppStack-Prod',\n driftDetectionRoleToAssumeArn: 'arn:aws:iam::123456789012:role/cdk-drift-role',\n driftDetectionRoleToAssumeRegion: 'us-east-1',\n // failOnDrift: true, // optional (default true)\n },\n ],\n});\n\nproject.synth();\n```\n\n### CdkDriftDetectionWorkflow props\n- `project` (required) — Your Projen project instance.\n- `stacks` (required) — Array of stacks to check.\n- `oidcRoleArn` (required) — Default OIDC role ARN used before chaining into per‑stack drift roles.\n- `oidcRegion` (required) — Default OIDC region.\n- `workflowName` (optional, default `'drift-detection'`) — Human‑friendly workflow name; the file name is derived in kebab‑case.\n- `schedule` (optional) — Cron expression for automatic runs.\n- `createIssues` (optional, default `true`) — When true, scheduled runs will create/update a GitHub issue if drift is detected.\n- `nodeVersion` (optional, default `'24.x'`) — Node.js version for the runner.\n- `scriptOutputPath` (optional, default `'.github/workflows/scripts/detect-drift.ts'`) — Where to write the helper script.\n- `workingDirectory` (optional) — Subdirectory where the CDK app lives (e.g., `'infra'`). Sets `defaults.run.working-directory` on all jobs. Artifact upload and issue-script paths are automatically prefixed.\n\n### Per‑stack fields\n- `stackName` (required) — The full CloudFormation stack name.\n- `driftDetectionRoleToAssumeArn` (required) — Role to assume (after OIDC) for making drift API calls.\n- `driftDetectionRoleToAssumeRegion` (required) — Region for that role and API calls.\n- `failOnDrift` (optional, default `true`) — Intended to fail the detection step on drift. The provided script exits with non‑zero when drift is found; the job continues to allow artifact upload and issue creation.\n\n### What gets generated\n- `.github/workflows/<kebab(workflowName)>.yml` — A workflow with one job per stack plus a final summary job.\n- `.github/workflows/scripts/detect-drift.ts` — Helper script that:\n - Starts drift detection and polls until completion\n - Lists non‑`IN_SYNC` resources and builds an HTML report\n - Writes optional JSON to `DRIFT_DETECTION_OUTPUT` when set\n - Prints to stdout and appends to the GitHub Step Summary when available\n\n### Artifacts and summary\n- Each stack job uploads `drift-results-<stack>.json` (if produced).\n- A final `Drift Detection Summary` job downloads all artifacts and prints a consolidated summary.\n\n### Manual dispatch\n- The workflow exposes an input named `stack` with choices including each configured stack and an `all` option.\n- Choose a specific stack to run drift detection for that stack only, or select `all` (or leave the input empty) to run all stacks.\n\nNote: The default workflow does not post PR comments for drift. It can create/update an Issue on scheduled runs when `createIssues` is `true`.\n\n### Post-notification steps (e.g., Slack)\n\nYou can add your own GitHub Action steps to run after the drift detection step for each stack using `postGitHubSteps`.\nProvide your own Slack payload/markdown (this library no longer generates a payload step for you).\n\nOption A: slackapi/slack-github-action (Incoming Webhook, official syntax)\n\n```ts\nnew CdkDriftDetectionWorkflow({\n project,\n oidcRoleArn: 'arn:aws:iam::123456789012:role/github-oidc-role',\n oidcRegion: 'us-east-1',\n stacks: [/* ... */],\n postGitHubSteps: ({ stack }) => {\n // Build a descriptive name per stack\n const name = `Notify Slack (${stack} post-drift)`;\n const step = {\n name,\n uses: 'slackapi/slack-github-action@v2.1.1',\n // by default, post steps run only when drift is detected; you can override `if`\n if: \"always() && steps.drift.outcome == 'failure'\",\n // Use official inputs: webhook + webhook-type, and a YAML payload with blocks\n with: {\n webhook: '${{ secrets.CDK_NOTIFICATIONS_SLACK_WEBHOOK }}',\n 'webhook-type': 'incoming-webhook',\n payload: [\n 'text: \"** ${{ env.STACK_NAME }} ** has drifted!\"',\n 'blocks:',\n ' - type: \"section\"',\n ' text:',\n ' type: \"mrkdwn\"',\n ' text: \"*Stack:* ${{ env.STACK_NAME }} (region ${{ env.AWS_REGION }}) has drifted:exclamation:\"',\n ' - type: \"section\"',\n ' fields:',\n ' - type: \"mrkdwn\"',\n ' text: \"*Stack ARN*\\\\n${{ steps.drift.outputs.stack-arn }}\"',\n ' - type: \"mrkdwn\"',\n ' text: \"*Issue*\\\\n<${{ github.server_url }}/${{ github.repository }}/issues/${{ steps.issue.outputs.result }}|#${{ steps.issue.outputs.result }}>\"',\n ].join('\\n'),\n },\n };\n return [step];\n },\n});\n```\n\nNote: The Issue link requires `createIssues: true` (default) so that the `Create Issue on Drift` step runs before this Slack step and exposes `steps.issue.outputs.result`. This library orders the steps accordingly.\n\nDetails:\n- `postGitHubSteps` can be:\n - an array of step objects, or\n - a factory function `({ stack }) => step | step[]`.\n- Each step you provide is inserted after the results are uploaded.\n- Default condition: if you do not set `if` on your step, it will default to `always() && steps.drift.outcome == 'failure'`.\n- Available context/env you can use:\n - `${{ env.STACK_NAME }}`, `${{ env.DRIFT_DETECTION_OUTPUT }}`\n - `${{ steps.drift.outcome }}` — success/failure of the detect step\n - `${{ steps.drift.outputs.stack-arn }}` — Stack ARN resolved at runtime\n - `${{ steps.issue.outputs.result }}` — Issue number if the workflow created/found one (empty when not applicable)\n```\n\n## Usage: CdkDriftIamTemplate\n\nEmit an example IAM template you can deploy in your account for the Drift Detection workflow.\n\n### With Projen\n\n```ts\nimport { awscdk } from 'projen';\nimport { CdkDriftIamTemplate } from '@jjrawlins/cdk-diff-pr-github-action';\n\nconst project = new awscdk.AwsCdkConstructLibrary({\n // ...\n});\n\nnew CdkDriftIamTemplate({\n project,\n roleName: 'cdk-drift-role',\n oidcRoleArn: 'arn:aws:iam::123456789012:role/github-oidc-role',\n oidcRegion: 'us-east-1',\n // Optional: custom output path (default: 'cdk-drift-workflow-iam-template.yaml')\n // outputPath: 'infra/cdk-drift-iam.yaml',\n});\n\nproject.synth();\n```\n\nA Projen task is also added:\n\n```bash\nnpx projen deploy-cdkdrift-iam-template -- --parameter-overrides GitHubOIDCRoleArn=... # plus any extra AWS CLI args\n```\n\n### Without Projen (Standalone Generator)\n\n```ts\nimport { CdkDriftIamTemplateGenerator } from '@jjrawlins/cdk-diff-pr-github-action';\nimport * as fs from 'fs';\n\nconst template = CdkDriftIamTemplateGenerator.generateTemplate({\n roleName: 'cdk-drift-role',\n oidcRoleArn: 'arn:aws:iam::123456789012:role/github-oidc-role',\n oidcRegion: 'us-east-1',\n});\n\nfs.writeFileSync('cdk-drift-iam-template.yaml', template);\n\n// Get the deploy command\nconst deployCmd = CdkDriftIamTemplateGenerator.generateDeployCommand('cdk-drift-iam-template.yaml');\nconsole.log('Deploy with:', deployCmd);\n```\n\n### What the template defines\n\n- Parameter `GitHubOIDCRoleArn` with a default from `oidcRoleArn` — the ARN of your existing GitHub OIDC role allowed to assume this drift role.\n- IAM role `CdkDriftRole` with minimal permissions for CloudFormation drift detection operations.\n- Outputs exporting the role name and ARN.\n\n---\n\n## Usage: CdkDiffIamTemplateStackSet (Org-Wide Deployment)\n\n`CdkDiffIamTemplateStackSet` creates a CloudFormation StackSet template for deploying GitHub OIDC provider, OIDC role, and CDK diff/drift IAM roles across an entire AWS Organization. This is the recommended approach for organizations that want to enable CDK diff/drift workflows across multiple accounts.\n\n### Architecture\n\nEach account in your organization gets:\n- **GitHub OIDC Provider** — Authenticates GitHub Actions workflows\n- **GitHubOIDCRole** — Trusts the OIDC provider with repo/branch restrictions\n- **CdkChangesetRole** — For PR change set previews (trusts GitHubOIDCRole)\n- **CdkDriftRole** — For drift detection (trusts GitHubOIDCRole)\n\nThis is a self-contained deployment with **no role chaining required**.\n\n### With Projen\n\n```ts\nimport { awscdk } from 'projen';\nimport { CdkDiffIamTemplateStackSet } from '@jjrawlins/cdk-diff-pr-github-action';\n\nconst project = new awscdk.AwsCdkConstructLibrary({ /* ... */ });\n\nnew CdkDiffIamTemplateStackSet({\n project,\n githubOidc: {\n owner: 'my-org', // GitHub org or username\n repositories: ['infra-repo', 'app-repo'], // Repos allowed to assume roles\n branches: ['main', 'release/*'], // Branch patterns (default: ['*'])\n },\n targetOrganizationalUnitIds: ['ou-xxxx-xxxxxxxx'], // Target OUs\n regions: ['us-east-1', 'eu-west-1'], // Target regions\n // Optional settings:\n // oidcRoleName: 'GitHubOIDCRole', // default\n // changesetRoleName: 'CdkChangesetRole', // default\n // driftRoleName: 'CdkDriftRole', // default\n // roleSelection: StackSetRoleSelection.BOTH, // BOTH, CHANGESET_ONLY, or DRIFT_ONLY\n // delegatedAdmin: true, // Use --call-as DELEGATED_ADMIN (default: true)\n});\n\nproject.synth();\n```\n\nThis creates:\n- `cdk-diff-workflow-stackset-template.yaml` — CloudFormation template\n- Projen tasks for StackSet management\n\n**Projen tasks:**\n```bash\nnpx projen stackset-create # Create the StackSet\nnpx projen stackset-update # Update the StackSet template\nnpx projen stackset-deploy-instances # Deploy to target OUs/regions\nnpx projen stackset-delete-instances # Remove stack instances\nnpx projen stackset-delete # Delete the StackSet\nnpx projen stackset-describe # Show StackSet status\nnpx projen stackset-list-instances # List all instances\n```\n\n### Without Projen (Standalone Generator)\n\nFor non-Projen projects, use `CdkDiffIamTemplateStackSetGenerator` directly:\n\n```ts\nimport {\n CdkDiffIamTemplateStackSetGenerator\n} from '@jjrawlins/cdk-diff-pr-github-action';\nimport * as fs from 'fs';\n\n// Generate the CloudFormation template\nconst template = CdkDiffIamTemplateStackSetGenerator.generateTemplate({\n githubOidc: {\n owner: 'my-org',\n repositories: ['infra-repo'],\n branches: ['main'],\n },\n});\n\n// Write to file\nfs.writeFileSync('stackset-template.yaml', template);\n\n// Get AWS CLI commands for StackSet operations\nconst commands = CdkDiffIamTemplateStackSetGenerator.generateCommands({\n stackSetName: 'cdk-diff-workflow-iam-stackset',\n templatePath: 'stackset-template.yaml',\n targetOrganizationalUnitIds: ['ou-xxxx-xxxxxxxx'],\n regions: ['us-east-1'],\n});\n\nconsole.log('Create StackSet:', commands['stackset-create']);\nconsole.log('Deploy instances:', commands['stackset-deploy-instances']);\n```\n\n### GitHub Actions Workflow (Simplified)\n\nWith per-account OIDC, your workflow is simplified — no role chaining needed:\n\n```yaml\njobs:\n diff:\n runs-on: ubuntu-latest\n permissions:\n id-token: write\n contents: read\n steps:\n - uses: actions/checkout@v4\n\n - uses: aws-actions/configure-aws-credentials@v4\n with:\n role-to-assume: arn:aws:iam::${{ env.ACCOUNT_ID }}:role/GitHubOIDCRole\n aws-region: us-east-1\n\n - name: Assume Changeset Role\n run: |\n CREDS=$(aws sts assume-role \\\n --role-arn arn:aws:iam::${{ env.ACCOUNT_ID }}:role/CdkChangesetRole \\\n --role-session-name changeset-session)\n # Export credentials...\n```\n\n### GitHubOidcConfig options\n\n| Property | Description |\n|----------|-------------|\n| `owner` | GitHub organization or username (required) |\n| `repositories` | Array of repo names, or `['*']` for all repos (required) |\n| `branches` | Array of branch patterns (default: `['*']`) |\n| `additionalClaims` | Extra OIDC claims like `['pull_request', 'environment:production']` |\n\n---\n\n## Testing\n\nThis repository includes Jest tests that snapshot the synthesized outputs from Projen and assert that:\n- Diff workflows are created per stack and contain all expected steps.\n- Drift detection workflow produces one job per stack and a summary job.\n- Only one helper script file is generated per workflow type.\n- Per‑stack OIDC overrides (where supported) are respected.\n- Helpful validation errors are thrown for missing OIDC settings.\n- The IAM template files contain the expected resources and outputs.\n\nRun tests with:\n\n```bash\nyarn test\n```\n\n## Monorepo support\n\nIf your CDK app lives in a subdirectory (e.g., `infra/`), use the `workingDirectory` option:\n\n```ts\nnew CdkDiffStackWorkflow({\n project,\n workingDirectory: 'infra',\n oidcRoleArn: 'arn:aws:iam::111111111111:role/GitHubOIDCRole',\n oidcRegion: 'us-east-1',\n stacks: [\n {\n stackName: 'MyStack-dev',\n changesetRoleToAssumeArn: 'arn:aws:iam::222222222222:role/CdkChangesetRole',\n changesetRoleToAssumeRegion: 'us-east-1',\n },\n ],\n});\n\nnew CdkDriftDetectionWorkflow({\n project,\n workingDirectory: 'infra',\n oidcRoleArn: 'arn:aws:iam::111111111111:role/GitHubOIDCRole',\n oidcRegion: 'us-east-1',\n stacks: [\n {\n stackName: 'MyStack-dev',\n driftDetectionRoleToAssumeArn: 'arn:aws:iam::222222222222:role/CdkDriftRole',\n driftDetectionRoleToAssumeRegion: 'us-east-1',\n },\n ],\n});\n```\n\nThis sets `defaults.run.working-directory: infra` on all workflow jobs so that `install`, `synth`, and `deploy` steps run in the correct directory. The describe-changeset and detect-drift scripts are referenced with absolute paths so they resolve correctly from the subdirectory.\n\n**Note:** Your `infra/package.json` must include `@aws-sdk/client-cloudformation` as a dependency for the describe-changeset script to resolve modules at runtime.\n\n## Notes\n- This package assumes your repository is configured with GitHub Actions and that you have a GitHub OIDC role configured in AWS.\n- The generated scripts use the AWS SDK v3 for CloudFormation and, where applicable, the GitHub REST API.\n"
|
|
3511
3511
|
},
|
|
3512
3512
|
"repository": {
|
|
3513
3513
|
"type": "git",
|
|
@@ -4472,7 +4472,7 @@
|
|
|
4472
4472
|
},
|
|
4473
4473
|
"locationInModule": {
|
|
4474
4474
|
"filename": "src/CdkDiffStackWorkflow.ts",
|
|
4475
|
-
"line":
|
|
4475
|
+
"line": 41
|
|
4476
4476
|
},
|
|
4477
4477
|
"parameters": [
|
|
4478
4478
|
{
|
|
@@ -4486,7 +4486,7 @@
|
|
|
4486
4486
|
"kind": "class",
|
|
4487
4487
|
"locationInModule": {
|
|
4488
4488
|
"filename": "src/CdkDiffStackWorkflow.ts",
|
|
4489
|
-
"line":
|
|
4489
|
+
"line": 38
|
|
4490
4490
|
},
|
|
4491
4491
|
"name": "CdkDiffStackWorkflow",
|
|
4492
4492
|
"symbolId": "src/CdkDiffStackWorkflow:CdkDiffStackWorkflow"
|
|
@@ -4619,6 +4619,25 @@
|
|
|
4619
4619
|
"type": {
|
|
4620
4620
|
"primitive": "string"
|
|
4621
4621
|
}
|
|
4622
|
+
},
|
|
4623
|
+
{
|
|
4624
|
+
"abstract": true,
|
|
4625
|
+
"docs": {
|
|
4626
|
+
"default": "- repository root",
|
|
4627
|
+
"remarks": "Useful for monorepos where infrastructure lives in a subdirectory (e.g., 'infra').\n\nWhen set, all workflow run steps will use `defaults.run.working-directory`\nand script paths will be adjusted to use absolute references.",
|
|
4628
|
+
"stability": "experimental",
|
|
4629
|
+
"summary": "Working directory for the CDK app, relative to the repository root."
|
|
4630
|
+
},
|
|
4631
|
+
"immutable": true,
|
|
4632
|
+
"locationInModule": {
|
|
4633
|
+
"filename": "src/CdkDiffStackWorkflow.ts",
|
|
4634
|
+
"line": 36
|
|
4635
|
+
},
|
|
4636
|
+
"name": "workingDirectory",
|
|
4637
|
+
"optional": true,
|
|
4638
|
+
"type": {
|
|
4639
|
+
"primitive": "string"
|
|
4640
|
+
}
|
|
4622
4641
|
}
|
|
4623
4642
|
],
|
|
4624
4643
|
"symbolId": "src/CdkDiffStackWorkflow:CdkDiffStackWorkflowProps"
|
|
@@ -4635,7 +4654,7 @@
|
|
|
4635
4654
|
},
|
|
4636
4655
|
"locationInModule": {
|
|
4637
4656
|
"filename": "src/CdkDriftDetectionWorkflow.ts",
|
|
4638
|
-
"line":
|
|
4657
|
+
"line": 75
|
|
4639
4658
|
},
|
|
4640
4659
|
"parameters": [
|
|
4641
4660
|
{
|
|
@@ -4649,7 +4668,7 @@
|
|
|
4649
4668
|
"kind": "class",
|
|
4650
4669
|
"locationInModule": {
|
|
4651
4670
|
"filename": "src/CdkDriftDetectionWorkflow.ts",
|
|
4652
|
-
"line":
|
|
4671
|
+
"line": 72
|
|
4653
4672
|
},
|
|
4654
4673
|
"name": "CdkDriftDetectionWorkflow",
|
|
4655
4674
|
"symbolId": "src/CdkDriftDetectionWorkflow:CdkDriftDetectionWorkflow"
|
|
@@ -4777,7 +4796,7 @@
|
|
|
4777
4796
|
"immutable": true,
|
|
4778
4797
|
"locationInModule": {
|
|
4779
4798
|
"filename": "src/CdkDriftDetectionWorkflow.ts",
|
|
4780
|
-
"line":
|
|
4799
|
+
"line": 57
|
|
4781
4800
|
},
|
|
4782
4801
|
"name": "postGitHubSteps",
|
|
4783
4802
|
"optional": true,
|
|
@@ -4832,6 +4851,25 @@
|
|
|
4832
4851
|
"type": {
|
|
4833
4852
|
"primitive": "string"
|
|
4834
4853
|
}
|
|
4854
|
+
},
|
|
4855
|
+
{
|
|
4856
|
+
"abstract": true,
|
|
4857
|
+
"docs": {
|
|
4858
|
+
"default": "- repository root",
|
|
4859
|
+
"remarks": "Useful for monorepos where infrastructure lives in a subdirectory (e.g., 'infra').\n\nWhen set, all workflow run steps will use `defaults.run.working-directory`\nand artifact/script paths will be adjusted accordingly.",
|
|
4860
|
+
"stability": "experimental",
|
|
4861
|
+
"summary": "Working directory for the CDK app, relative to the repository root."
|
|
4862
|
+
},
|
|
4863
|
+
"immutable": true,
|
|
4864
|
+
"locationInModule": {
|
|
4865
|
+
"filename": "src/CdkDriftDetectionWorkflow.ts",
|
|
4866
|
+
"line": 42
|
|
4867
|
+
},
|
|
4868
|
+
"name": "workingDirectory",
|
|
4869
|
+
"optional": true,
|
|
4870
|
+
"type": {
|
|
4871
|
+
"primitive": "string"
|
|
4872
|
+
}
|
|
4835
4873
|
}
|
|
4836
4874
|
],
|
|
4837
4875
|
"symbolId": "src/CdkDriftDetectionWorkflow:CdkDriftDetectionWorkflowProps"
|
|
@@ -5363,6 +5401,6 @@
|
|
|
5363
5401
|
"symbolId": "src/CdkDiffIamTemplateStackSet:StackSetRoleSelection"
|
|
5364
5402
|
}
|
|
5365
5403
|
},
|
|
5366
|
-
"version": "1.
|
|
5367
|
-
"fingerprint": "
|
|
5404
|
+
"version": "1.8.0",
|
|
5405
|
+
"fingerprint": "gOPm1rvgC24rKxRefMcHRBmTZTxskniMczkswaBzoY4="
|
|
5368
5406
|
}
|
package/API.md
CHANGED
|
@@ -797,6 +797,7 @@ const cdkDiffStackWorkflowProps: CdkDiffStackWorkflowProps = { ... }
|
|
|
797
797
|
| <code><a href="#@jjrawlins/cdk-diff-pr-github-action.CdkDiffStackWorkflowProps.property.oidcRegion">oidcRegion</a></code> | <code>string</code> | *No description.* |
|
|
798
798
|
| <code><a href="#@jjrawlins/cdk-diff-pr-github-action.CdkDiffStackWorkflowProps.property.oidcRoleArn">oidcRoleArn</a></code> | <code>string</code> | *No description.* |
|
|
799
799
|
| <code><a href="#@jjrawlins/cdk-diff-pr-github-action.CdkDiffStackWorkflowProps.property.scriptOutputPath">scriptOutputPath</a></code> | <code>string</code> | *No description.* |
|
|
800
|
+
| <code><a href="#@jjrawlins/cdk-diff-pr-github-action.CdkDiffStackWorkflowProps.property.workingDirectory">workingDirectory</a></code> | <code>string</code> | Working directory for the CDK app, relative to the repository root. |
|
|
800
801
|
|
|
801
802
|
---
|
|
802
803
|
|
|
@@ -870,6 +871,24 @@ public readonly scriptOutputPath: string;
|
|
|
870
871
|
|
|
871
872
|
---
|
|
872
873
|
|
|
874
|
+
##### `workingDirectory`<sup>Optional</sup> <a name="workingDirectory" id="@jjrawlins/cdk-diff-pr-github-action.CdkDiffStackWorkflowProps.property.workingDirectory"></a>
|
|
875
|
+
|
|
876
|
+
```typescript
|
|
877
|
+
public readonly workingDirectory: string;
|
|
878
|
+
```
|
|
879
|
+
|
|
880
|
+
- *Type:* string
|
|
881
|
+
- *Default:* repository root
|
|
882
|
+
|
|
883
|
+
Working directory for the CDK app, relative to the repository root.
|
|
884
|
+
|
|
885
|
+
Useful for monorepos where infrastructure lives in a subdirectory (e.g., 'infra').
|
|
886
|
+
|
|
887
|
+
When set, all workflow run steps will use `defaults.run.working-directory`
|
|
888
|
+
and script paths will be adjusted to use absolute references.
|
|
889
|
+
|
|
890
|
+
---
|
|
891
|
+
|
|
873
892
|
### CdkDriftDetectionWorkflowProps <a name="CdkDriftDetectionWorkflowProps" id="@jjrawlins/cdk-diff-pr-github-action.CdkDriftDetectionWorkflowProps"></a>
|
|
874
893
|
|
|
875
894
|
#### Initializer <a name="Initializer" id="@jjrawlins/cdk-diff-pr-github-action.CdkDriftDetectionWorkflowProps.Initializer"></a>
|
|
@@ -894,6 +913,7 @@ const cdkDriftDetectionWorkflowProps: CdkDriftDetectionWorkflowProps = { ... }
|
|
|
894
913
|
| <code><a href="#@jjrawlins/cdk-diff-pr-github-action.CdkDriftDetectionWorkflowProps.property.schedule">schedule</a></code> | <code>string</code> | *No description.* |
|
|
895
914
|
| <code><a href="#@jjrawlins/cdk-diff-pr-github-action.CdkDriftDetectionWorkflowProps.property.scriptOutputPath">scriptOutputPath</a></code> | <code>string</code> | *No description.* |
|
|
896
915
|
| <code><a href="#@jjrawlins/cdk-diff-pr-github-action.CdkDriftDetectionWorkflowProps.property.workflowName">workflowName</a></code> | <code>string</code> | *No description.* |
|
|
916
|
+
| <code><a href="#@jjrawlins/cdk-diff-pr-github-action.CdkDriftDetectionWorkflowProps.property.workingDirectory">workingDirectory</a></code> | <code>string</code> | Working directory for the CDK app, relative to the repository root. |
|
|
897
917
|
|
|
898
918
|
---
|
|
899
919
|
|
|
@@ -1003,6 +1023,24 @@ public readonly workflowName: string;
|
|
|
1003
1023
|
|
|
1004
1024
|
---
|
|
1005
1025
|
|
|
1026
|
+
##### `workingDirectory`<sup>Optional</sup> <a name="workingDirectory" id="@jjrawlins/cdk-diff-pr-github-action.CdkDriftDetectionWorkflowProps.property.workingDirectory"></a>
|
|
1027
|
+
|
|
1028
|
+
```typescript
|
|
1029
|
+
public readonly workingDirectory: string;
|
|
1030
|
+
```
|
|
1031
|
+
|
|
1032
|
+
- *Type:* string
|
|
1033
|
+
- *Default:* repository root
|
|
1034
|
+
|
|
1035
|
+
Working directory for the CDK app, relative to the repository root.
|
|
1036
|
+
|
|
1037
|
+
Useful for monorepos where infrastructure lives in a subdirectory (e.g., 'infra').
|
|
1038
|
+
|
|
1039
|
+
When set, all workflow run steps will use `defaults.run.working-directory`
|
|
1040
|
+
and artifact/script paths will be adjusted accordingly.
|
|
1041
|
+
|
|
1042
|
+
---
|
|
1043
|
+
|
|
1006
1044
|
### CdkDriftIamTemplateGeneratorProps <a name="CdkDriftIamTemplateGeneratorProps" id="@jjrawlins/cdk-diff-pr-github-action.CdkDriftIamTemplateGeneratorProps"></a>
|
|
1007
1045
|
|
|
1008
1046
|
Props for generating CDK Drift IAM templates (no Projen dependency).
|
package/README.md
CHANGED
|
@@ -193,6 +193,7 @@ project.synth();
|
|
|
193
193
|
- `nodeVersion` (optional, default `'24.x'`) — Node.js version for the workflow runner.
|
|
194
194
|
- `cdkYarnCommand` (optional, default `'cdk'`) — Yarn script/command to invoke CDK.
|
|
195
195
|
- `scriptOutputPath` (optional, default `'.github/workflows/scripts/describe-cfn-changeset.ts'`) — Where to write the helper script.
|
|
196
|
+
- `workingDirectory` (optional) — Subdirectory where the CDK app lives (e.g., `'infra'`). Sets `defaults.run.working-directory` on all jobs so `install`, `synth`, and `deploy` steps run in that directory. The describe-changeset script path is automatically prefixed with `$GITHUB_WORKSPACE/` and `NODE_PATH` is set to resolve modules from the working directory.
|
|
196
197
|
|
|
197
198
|
If neither top‑level OIDC defaults nor all per‑stack values are supplied, the construct throws a helpful error.
|
|
198
199
|
|
|
@@ -463,6 +464,7 @@ project.synth();
|
|
|
463
464
|
- `createIssues` (optional, default `true`) — When true, scheduled runs will create/update a GitHub issue if drift is detected.
|
|
464
465
|
- `nodeVersion` (optional, default `'24.x'`) — Node.js version for the runner.
|
|
465
466
|
- `scriptOutputPath` (optional, default `'.github/workflows/scripts/detect-drift.ts'`) — Where to write the helper script.
|
|
467
|
+
- `workingDirectory` (optional) — Subdirectory where the CDK app lives (e.g., `'infra'`). Sets `defaults.run.working-directory` on all jobs. Artifact upload and issue-script paths are automatically prefixed.
|
|
466
468
|
|
|
467
469
|
### Per‑stack fields
|
|
468
470
|
- `stackName` (required) — The full CloudFormation stack name.
|
|
@@ -753,6 +755,44 @@ Run tests with:
|
|
|
753
755
|
yarn test
|
|
754
756
|
```
|
|
755
757
|
|
|
758
|
+
## Monorepo support
|
|
759
|
+
|
|
760
|
+
If your CDK app lives in a subdirectory (e.g., `infra/`), use the `workingDirectory` option:
|
|
761
|
+
|
|
762
|
+
```ts
|
|
763
|
+
new CdkDiffStackWorkflow({
|
|
764
|
+
project,
|
|
765
|
+
workingDirectory: 'infra',
|
|
766
|
+
oidcRoleArn: 'arn:aws:iam::111111111111:role/GitHubOIDCRole',
|
|
767
|
+
oidcRegion: 'us-east-1',
|
|
768
|
+
stacks: [
|
|
769
|
+
{
|
|
770
|
+
stackName: 'MyStack-dev',
|
|
771
|
+
changesetRoleToAssumeArn: 'arn:aws:iam::222222222222:role/CdkChangesetRole',
|
|
772
|
+
changesetRoleToAssumeRegion: 'us-east-1',
|
|
773
|
+
},
|
|
774
|
+
],
|
|
775
|
+
});
|
|
776
|
+
|
|
777
|
+
new CdkDriftDetectionWorkflow({
|
|
778
|
+
project,
|
|
779
|
+
workingDirectory: 'infra',
|
|
780
|
+
oidcRoleArn: 'arn:aws:iam::111111111111:role/GitHubOIDCRole',
|
|
781
|
+
oidcRegion: 'us-east-1',
|
|
782
|
+
stacks: [
|
|
783
|
+
{
|
|
784
|
+
stackName: 'MyStack-dev',
|
|
785
|
+
driftDetectionRoleToAssumeArn: 'arn:aws:iam::222222222222:role/CdkDriftRole',
|
|
786
|
+
driftDetectionRoleToAssumeRegion: 'us-east-1',
|
|
787
|
+
},
|
|
788
|
+
],
|
|
789
|
+
});
|
|
790
|
+
```
|
|
791
|
+
|
|
792
|
+
This sets `defaults.run.working-directory: infra` on all workflow jobs so that `install`, `synth`, and `deploy` steps run in the correct directory. The describe-changeset and detect-drift scripts are referenced with absolute paths so they resolve correctly from the subdirectory.
|
|
793
|
+
|
|
794
|
+
**Note:** Your `infra/package.json` must include `@aws-sdk/client-cloudformation` as a dependency for the describe-changeset script to resolve modules at runtime.
|
|
795
|
+
|
|
756
796
|
## Notes
|
|
757
797
|
- This package assumes your repository is configured with GitHub Actions and that you have a GitHub OIDC role configured in AWS.
|
|
758
798
|
- The generated scripts use the AWS SDK v3 for CloudFormation and, where applicable, the GitHub REST API.
|
|
@@ -13,7 +13,7 @@ import (
|
|
|
13
13
|
constructs "github.com/aws/constructs-go/constructs/v10/jsii"
|
|
14
14
|
)
|
|
15
15
|
|
|
16
|
-
//go:embed jjrawlins-cdk-diff-pr-github-action-1.7.
|
|
16
|
+
//go:embed jjrawlins-cdk-diff-pr-github-action-1.7.2.tgz
|
|
17
17
|
var tarball []byte
|
|
18
18
|
|
|
19
19
|
// Initialize loads the necessary packages in the @jsii/kernel to support the enclosing module.
|
|
@@ -24,5 +24,5 @@ func Initialize() {
|
|
|
24
24
|
constructs.Initialize()
|
|
25
25
|
|
|
26
26
|
// Load this library into the kernel
|
|
27
|
-
_jsii_.Load("@jjrawlins/cdk-diff-pr-github-action", "1.7.
|
|
27
|
+
_jsii_.Load("@jjrawlins/cdk-diff-pr-github-action", "1.7.2", tarball)
|
|
28
28
|
}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
1.7.
|
|
1
|
+
1.7.2
|
|
@@ -397,7 +397,7 @@ class CdkDiffIamTemplateGenerator {
|
|
|
397
397
|
}
|
|
398
398
|
exports.CdkDiffIamTemplateGenerator = CdkDiffIamTemplateGenerator;
|
|
399
399
|
_a = JSII_RTTI_SYMBOL_1;
|
|
400
|
-
CdkDiffIamTemplateGenerator[_a] = { fqn: "@jjrawlins/cdk-diff-pr-github-action.CdkDiffIamTemplateGenerator", version: "1.
|
|
400
|
+
CdkDiffIamTemplateGenerator[_a] = { fqn: "@jjrawlins/cdk-diff-pr-github-action.CdkDiffIamTemplateGenerator", version: "1.8.0" };
|
|
401
401
|
/**
|
|
402
402
|
* Projen construct that emits a CloudFormation template with minimal IAM permissions
|
|
403
403
|
* for the CDK Diff Stack Workflow.
|
|
@@ -420,5 +420,5 @@ class CdkDiffIamTemplate {
|
|
|
420
420
|
}
|
|
421
421
|
exports.CdkDiffIamTemplate = CdkDiffIamTemplate;
|
|
422
422
|
_b = JSII_RTTI_SYMBOL_1;
|
|
423
|
-
CdkDiffIamTemplate[_b] = { fqn: "@jjrawlins/cdk-diff-pr-github-action.CdkDiffIamTemplate", version: "1.
|
|
423
|
+
CdkDiffIamTemplate[_b] = { fqn: "@jjrawlins/cdk-diff-pr-github-action.CdkDiffIamTemplate", version: "1.8.0" };
|
|
424
424
|
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CdkDiffIamTemplate.js","sourceRoot":"","sources":["../src/CdkDiffIamTemplate.ts"],"names":[],"mappings":"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","sourcesContent":["import { TextFile } from 'projen';\nimport { GitHubOidcConfig } from './CdkDiffIamTemplateStackSet';\n\n/**\n * Props for generating CDK Diff IAM templates (no Projen dependency)\n */\nexport interface CdkDiffIamTemplateGeneratorProps {\n  /** Name for the changeset IAM role */\n  readonly roleName: string;\n\n  /**\n   * ARN of the existing GitHub OIDC role that can assume this changeset role.\n   * Required when createOidcRole is false or undefined.\n   */\n  readonly oidcRoleArn?: string;\n\n  /**\n   * Region for the OIDC trust condition.\n   * Only used when oidcRoleArn is provided (external OIDC role).\n   */\n  readonly oidcRegion?: string;\n\n  /**\n   * Create a GitHub OIDC role within this template instead of using an existing one.\n   * When true, githubOidc configuration is required and oidcRoleArn is ignored.\n   * Default: false\n   */\n  readonly createOidcRole?: boolean;\n\n  /**\n   * Name of the GitHub OIDC role to create.\n   * Only used when createOidcRole is true.\n   * Default: 'GitHubOIDCRole'\n   */\n  readonly oidcRoleName?: string;\n\n  /**\n   * GitHub OIDC configuration for repo/branch restrictions.\n   * Required when createOidcRole is true.\n   */\n  readonly githubOidc?: GitHubOidcConfig;\n\n  /**\n   * Skip creating the OIDC provider (use existing one).\n   * Set to true if the account already has a GitHub OIDC provider.\n   * Only used when createOidcRole is true.\n   * Default: false\n   */\n  readonly skipOidcProviderCreation?: boolean;\n}\n\n/**\n * Pure generator class for CDK Diff IAM templates.\n * No Projen dependency - can be used in any project.\n */\nexport class CdkDiffIamTemplateGenerator {\n  /**\n   * Generate the CloudFormation IAM template as a YAML string.\n   */\n  static generateTemplate(props: CdkDiffIamTemplateGeneratorProps): string {\n    const createOidcRole = props.createOidcRole ?? false;\n\n    // Validate props\n    if (createOidcRole) {\n      if (!props.githubOidc) {\n        throw new Error('githubOidc configuration is required when createOidcRole is true');\n      }\n    } else {\n      if (!props.oidcRoleArn) {\n        throw new Error('oidcRoleArn is required when createOidcRole is false');\n      }\n      if (!props.oidcRegion) {\n        throw new Error('oidcRegion is required when createOidcRole is false');\n      }\n    }\n\n    if (createOidcRole) {\n      return this.generateTemplateWithOidcRole(props);\n    } else {\n      return this.generateTemplateWithExternalOidcRole(props);\n    }\n  }\n\n  /**\n   * Generate the AWS CLI deploy command for the IAM template.\n   */\n  static generateDeployCommand(templatePath: string = 'cdk-diff-workflow-iam-template.yaml'): string {\n    return `aws cloudformation deploy --template-file ${templatePath} --stack-name cdk-diff-workflow-iam-role --capabilities CAPABILITY_NAMED_IAM`;\n  }\n\n  /**\n   * Generate template that creates OIDC provider and role (self-contained)\n   */\n  private static generateTemplateWithOidcRole(props: CdkDiffIamTemplateGeneratorProps): string {\n    const oidcRoleName = props.oidcRoleName ?? 'GitHubOIDCRole';\n    const skipOidcProvider = props.skipOidcProviderCreation ?? false;\n    const githubOidc = props.githubOidc!;\n\n    const lines: string[] = [\n      \"AWSTemplateFormatVersion: '2010-09-09'\",\n      \"Description: 'GitHub OIDC and IAM roles for CDK Diff Stack Workflow construct'\",\n      '',\n      'Resources:',\n    ];\n\n    // OIDC Provider (only if not skipping)\n    if (!skipOidcProvider) {\n      lines.push(...this.generateOidcProviderLines());\n    }\n\n    // OIDC Role (needs permission to assume the changeset role)\n    lines.push(...this.generateOidcRoleLines(oidcRoleName, githubOidc, skipOidcProvider, props.roleName));\n\n    // Changeset Role (trusts the created OIDC role)\n    lines.push(...this.generateChangesetRoleWithOidcRef(props.roleName));\n\n    // Outputs\n    lines.push('');\n    lines.push('Outputs:');\n\n    if (!skipOidcProvider) {\n      lines.push(...this.generateOidcProviderOutputLines());\n    }\n\n    lines.push(...this.generateOidcRoleOutputLines());\n    lines.push(...this.generateChangesetOutputLines());\n\n    return lines.join('\\n');\n  }\n\n  /**\n   * Generate template that uses an external OIDC role (original behavior)\n   */\n  private static generateTemplateWithExternalOidcRole(props: CdkDiffIamTemplateGeneratorProps): string {\n    const lines = [\n      \"AWSTemplateFormatVersion: '2010-09-09'\",\n      \"Description: 'IAM role for CDK Diff Stack Workflow construct'\",\n      '',\n      'Parameters:',\n      '  GitHubOIDCRoleArn:',\n      '    Type: String',\n      \"    Description: 'ARN of the existing GitHub OIDC role that can assume this changeset role'\",\n      `    Default: '${props.oidcRoleArn}'`,\n      '',\n      'Resources:',\n      '  # CloudFormation ChangeSet Role - minimal permissions for changeset operations',\n      '  CdkChangesetRole:',\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: CloudFormationChangeSetAccess',\n      '          PolicyDocument:',\n      \"            Version: '2012-10-17'\",\n      '            Statement:',\n      '              # CloudFormation changeset operations',\n      '              - Effect: Allow',\n      '                Action:',\n      '                  - cloudformation:CreateChangeSet',\n      '                  - cloudformation:DescribeChangeSet',\n      '                  - cloudformation:DeleteChangeSet',\n      '                  - cloudformation:ListChangeSets',\n      '                  - cloudformation:DescribeStacks',\n      '                  - cloudformation:DescribeStackResourceDrifts',\n      '                  - cloudformation:GetTemplate',\n      \"                Resource: '*'\",\n      '              # CDK bootstrap bucket access (for changeset creation)',\n      '              - Effect: Allow',\n      '                Action:',\n      '                  - s3:GetObject',\n      '                  - s3:PutObject',\n      '                  - s3:DeleteObject',\n      '                  - s3:ListBucket',\n      '                Resource:',\n      \"                  - !Sub 'arn:aws:s3:::cdk-${AWS::AccountId}-${AWS::Region}-*'\",\n      \"                  - !Sub 'arn:aws:s3:::cdk-${AWS::AccountId}-${AWS::Region}-*/*'\",\n      '              # CDK bootstrap parameter access',\n      '              - Effect: Allow',\n      '                Action:',\n      '                  - ssm:GetParameter',\n      '                  - ssm:GetParameters',\n      '                  - ssm:GetParametersByPath',\n      \"                Resource: !Sub 'arn:aws:ssm:${AWS::Region}:${AWS::AccountId}:parameter/cdk-bootstrap/*'\",\n      '              # IAM PassRole for CDK operations',\n      '              - Effect: Allow',\n      '                Action:',\n      '                  - iam:PassRole',\n      \"                Resource: '*'\",\n      '                Condition:',\n      '                  StringEquals:',\n      \"                    'iam:PassedToService': 'cloudformation.amazonaws.com'\",\n      '',\n      'Outputs:',\n      '  CdkChangesetRoleArn:',\n      \"    Description: 'ARN of the CDK changeset role'\",\n      '    Value: !GetAtt CdkChangesetRole.Arn',\n      '    Export:',\n      \"      Name: !Sub '${AWS::StackName}-CdkChangesetRoleArn'\",\n      '',\n      '  CdkChangesetRoleName:',\n      \"    Description: 'Name of the CDK changeset role'\",\n      '    Value: !Ref CdkChangesetRole',\n      '    Export:',\n      \"      Name: !Sub '${AWS::StackName}-CdkChangesetRoleName'\",\n    ];\n\n    return lines.join('\\n');\n  }\n\n  private static generateOidcProviderLines(): string[] {\n    return [\n      '  # GitHub OIDC Provider',\n      '  GitHubOIDCProvider:',\n      '    Type: AWS::IAM::OIDCProvider',\n      '    Properties:',\n      '      Url: https://token.actions.githubusercontent.com',\n      '      ClientIdList:',\n      '        - sts.amazonaws.com',\n      '      ThumbprintList:',\n      '        - 6938fd4d98bab03faadb97b34396831e3780aea1',\n      '        - 1c58a3a8518e8759bf075b76b750d4f2df264fcd',\n      '',\n    ];\n  }\n\n  private static generateOidcRoleLines(\n    roleName: string,\n    githubOidc: GitHubOidcConfig,\n    skipOidcProvider: boolean = false,\n    changesetRoleName?: string,\n  ): string[] {\n    const subjectClaims = this.buildSubjectClaims(githubOidc);\n\n    const lines = [\n      '  # GitHub OIDC Role - authenticates GitHub Actions workflows',\n      '  GitHubOIDCRole:',\n      '    Type: AWS::IAM::Role',\n    ];\n\n    // Only add DependsOn if we're creating the provider\n    if (!skipOidcProvider) {\n      lines.push('    DependsOn: GitHubOIDCProvider');\n    }\n\n    lines.push(\n      '    Properties:',\n      \"      RoleName: '\" + roleName + \"'\",\n      '      AssumeRolePolicyDocument:',\n      \"        Version: '2012-10-17'\",\n      '        Statement:',\n      '          - Effect: Allow',\n      '            Principal:',\n      \"              Federated: !Sub 'arn:aws:iam::${AWS::AccountId}:oidc-provider/token.actions.githubusercontent.com'\",\n      '            Action: sts:AssumeRoleWithWebIdentity',\n      '            Condition:',\n      '              StringEquals:',\n      \"                'token.actions.githubusercontent.com:aud': 'sts.amazonaws.com'\",\n      '              StringLike:',\n      \"                'token.actions.githubusercontent.com:sub':\",\n    );\n\n    // Add subject claims\n    for (const claim of subjectClaims) {\n      lines.push(`                  - '${claim}'`);\n    }\n\n    // Add policies for assuming changeset role and CDK bootstrap roles, plus CDK diff permissions\n    lines.push('      Policies:');\n\n    // Policy 1: AssumeChangesetRole - allows assuming changeset role and CDK bootstrap roles\n    lines.push('        - PolicyName: AssumeChangesetRole');\n    lines.push('          PolicyDocument:');\n    lines.push(\"            Version: '2012-10-17'\");\n    lines.push('            Statement:');\n    lines.push('              - Effect: Allow');\n    lines.push('                Action: sts:AssumeRole');\n    lines.push('                Resource:');\n    if (changesetRoleName) {\n      lines.push(`                  - !Sub 'arn:aws:iam::\\${AWS::AccountId}:role/${changesetRoleName}'`);\n    }\n    // CDK bootstrap roles\n    lines.push('                  # CDK bootstrap roles');\n    lines.push(\"                  - !Sub 'arn:aws:iam::${AWS::AccountId}:role/cdk-hnb659fds-deploy-role-${AWS::AccountId}-${AWS::Region}'\");\n    lines.push(\"                  - !Sub 'arn:aws:iam::${AWS::AccountId}:role/cdk-hnb659fds-file-publishing-role-${AWS::AccountId}-${AWS::Region}'\");\n    lines.push(\"                  - !Sub 'arn:aws:iam::${AWS::AccountId}:role/cdk-hnb659fds-image-publishing-role-${AWS::AccountId}-${AWS::Region}'\");\n    lines.push(\"                  - !Sub 'arn:aws:iam::${AWS::AccountId}:role/cdk-hnb659fds-lookup-role-${AWS::AccountId}-${AWS::Region}'\");\n\n    // Policy 2: CdkDiffPermissions - direct permissions for CDK diff operations\n    lines.push('        - PolicyName: CdkDiffPermissions');\n    lines.push('          PolicyDocument:');\n    lines.push(\"            Version: '2012-10-17'\");\n    lines.push('            Statement:');\n    lines.push('              # CDK bootstrap parameter access (needed for cdk deploy/diff)');\n    lines.push('              - Effect: Allow');\n    lines.push('                Action:');\n    lines.push('                  - ssm:GetParameter');\n    lines.push('                  - ssm:GetParameters');\n    lines.push(\"                Resource: !Sub 'arn:aws:ssm:${AWS::Region}:${AWS::AccountId}:parameter/cdk-bootstrap/*'\");\n    lines.push('              # CloudFormation operations for changeset creation');\n    lines.push('              - Effect: Allow');\n    lines.push('                Action:');\n    lines.push('                  - cloudformation:CreateChangeSet');\n    lines.push('                  - cloudformation:DescribeChangeSet');\n    lines.push('                  - cloudformation:DeleteChangeSet');\n    lines.push('                  - cloudformation:ListChangeSets');\n    lines.push('                  - cloudformation:DescribeStacks');\n    lines.push('                  - cloudformation:DescribeStackResourceDrifts');\n    lines.push('                  - cloudformation:GetTemplate');\n    lines.push(\"                Resource: '*'\");\n    lines.push('              # CDK bootstrap bucket access');\n    lines.push('              - Effect: Allow');\n    lines.push('                Action:');\n    lines.push('                  - s3:GetObject');\n    lines.push('                  - s3:PutObject');\n    lines.push('                  - s3:ListBucket');\n    lines.push('                  - s3:GetBucketLocation');\n    lines.push('                Resource:');\n    lines.push(\"                  - !Sub 'arn:aws:s3:::cdk-hnb659fds-assets-${AWS::AccountId}-${AWS::Region}'\");\n    lines.push(\"                  - !Sub 'arn:aws:s3:::cdk-hnb659fds-assets-${AWS::AccountId}-${AWS::Region}/*'\");\n    lines.push('              # IAM PassRole for CDK operations');\n    lines.push('              - Effect: Allow');\n    lines.push('                Action:');\n    lines.push('                  - iam:PassRole');\n    lines.push(\"                Resource: !Sub 'arn:aws:iam::${AWS::AccountId}:role/cdk-hnb659fds-*'\");\n    lines.push('                Condition:');\n    lines.push('                  StringEquals:');\n    lines.push(\"                    'iam:PassedToService': 'cloudformation.amazonaws.com'\");\n\n    lines.push('');\n    return lines;\n  }\n\n  private static buildSubjectClaims(githubOidc: GitHubOidcConfig): string[] {\n    const claims: string[] = [];\n    const branches = githubOidc.branches ?? ['*'];\n\n    for (const repo of githubOidc.repositories) {\n      if (repo === '*') {\n        // Wildcard repo - allow all repos with branch restrictions\n        for (const branch of branches) {\n          if (branch === '*') {\n            claims.push(`repo:${githubOidc.owner}/*`);\n          } else {\n            claims.push(`repo:${githubOidc.owner}/*:ref:refs/heads/${branch}`);\n          }\n        }\n      } else {\n        // Specific repo\n        for (const branch of branches) {\n          if (branch === '*') {\n            claims.push(`repo:${githubOidc.owner}/${repo}:*`);\n          } else {\n            claims.push(`repo:${githubOidc.owner}/${repo}:ref:refs/heads/${branch}`);\n          }\n        }\n      }\n    }\n\n    // Add any additional claims\n    if (githubOidc.additionalClaims) {\n      for (const claim of githubOidc.additionalClaims) {\n        for (const repo of githubOidc.repositories) {\n          if (repo === '*') {\n            claims.push(`repo:${githubOidc.owner}/*:${claim}`);\n          } else {\n            claims.push(`repo:${githubOidc.owner}/${repo}:${claim}`);\n          }\n        }\n      }\n    }\n\n    return claims;\n  }\n\n  private static generateChangesetRoleWithOidcRef(roleName: string): string[] {\n    return [\n      '  # CloudFormation ChangeSet Role - minimal permissions for changeset operations',\n      '  CdkChangesetRole:',\n      '    Type: AWS::IAM::Role',\n      '    DependsOn: GitHubOIDCRole',\n      '    Properties:',\n      \"      RoleName: '\" + roleName + \"'\",\n      '      AssumeRolePolicyDocument:',\n      \"        Version: '2012-10-17'\",\n      '        Statement:',\n      '          - Effect: Allow',\n      '            Principal:',\n      '              AWS: !GetAtt GitHubOIDCRole.Arn',\n      '            Action: sts:AssumeRole',\n      '      Policies:',\n      '        - PolicyName: CloudFormationChangeSetAccess',\n      '          PolicyDocument:',\n      \"            Version: '2012-10-17'\",\n      '            Statement:',\n      '              # CloudFormation changeset operations',\n      '              - Effect: Allow',\n      '                Action:',\n      '                  - cloudformation:CreateChangeSet',\n      '                  - cloudformation:DescribeChangeSet',\n      '                  - cloudformation:DeleteChangeSet',\n      '                  - cloudformation:ListChangeSets',\n      '                  - cloudformation:DescribeStacks',\n      '                  - cloudformation:DescribeStackResourceDrifts',\n      '                  - cloudformation:GetTemplate',\n      \"                Resource: '*'\",\n      '              # CDK bootstrap bucket access (for changeset creation)',\n      '              - Effect: Allow',\n      '                Action:',\n      '                  - s3:GetObject',\n      '                  - s3:PutObject',\n      '                  - s3:DeleteObject',\n      '                  - s3:ListBucket',\n      '                Resource:',\n      \"                  - !Sub 'arn:aws:s3:::cdk-${AWS::AccountId}-${AWS::Region}-*'\",\n      \"                  - !Sub 'arn:aws:s3:::cdk-${AWS::AccountId}-${AWS::Region}-*/*'\",\n      '              # CDK bootstrap parameter access',\n      '              - Effect: Allow',\n      '                Action:',\n      '                  - ssm:GetParameter',\n      '                  - ssm:GetParameters',\n      '                  - ssm:GetParametersByPath',\n      \"                Resource: !Sub 'arn:aws:ssm:${AWS::Region}:${AWS::AccountId}:parameter/cdk-bootstrap/*'\",\n      '              # IAM PassRole for CDK operations',\n      '              - Effect: Allow',\n      '                Action:',\n      '                  - iam:PassRole',\n      \"                Resource: '*'\",\n      '                Condition:',\n      '                  StringEquals:',\n      \"                    'iam:PassedToService': 'cloudformation.amazonaws.com'\",\n      '',\n    ];\n  }\n\n  private static generateOidcProviderOutputLines(): string[] {\n    return [\n      '  GitHubOIDCProviderArn:',\n      \"    Description: 'ARN of the GitHub OIDC provider'\",\n      '    Value: !GetAtt GitHubOIDCProvider.Arn',\n      '    Export:',\n      \"      Name: !Sub '${AWS::StackName}-GitHubOIDCProviderArn'\",\n      '',\n    ];\n  }\n\n  private static generateOidcRoleOutputLines(): string[] {\n    return [\n      '  GitHubOIDCRoleArn:',\n      \"    Description: 'ARN of the GitHub OIDC role'\",\n      '    Value: !GetAtt GitHubOIDCRole.Arn',\n      '    Export:',\n      \"      Name: !Sub '${AWS::StackName}-GitHubOIDCRoleArn'\",\n      '',\n      '  GitHubOIDCRoleName:',\n      \"    Description: 'Name of the GitHub OIDC role'\",\n      '    Value: !Ref GitHubOIDCRole',\n      '    Export:',\n      \"      Name: !Sub '${AWS::StackName}-GitHubOIDCRoleName'\",\n      '',\n    ];\n  }\n\n  private static generateChangesetOutputLines(): string[] {\n    return [\n      '  CdkChangesetRoleArn:',\n      \"    Description: 'ARN of the CDK changeset role'\",\n      '    Value: !GetAtt CdkChangesetRole.Arn',\n      '    Export:',\n      \"      Name: !Sub '${AWS::StackName}-CdkChangesetRoleArn'\",\n      '',\n      '  CdkChangesetRoleName:',\n      \"    Description: 'Name of the CDK changeset role'\",\n      '    Value: !Ref CdkChangesetRole',\n      '    Export:',\n      \"      Name: !Sub '${AWS::StackName}-CdkChangesetRoleName'\",\n    ];\n  }\n}\n\n/**\n * Props for the Projen-integrated CDK Diff IAM template construct\n */\nexport interface CdkDiffIamTemplateProps extends CdkDiffIamTemplateGeneratorProps {\n  /** Projen project instance */\n  readonly project: any;\n  /** Output path for the template file (default: 'cdk-diff-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 Diff Stack Workflow.\n *\n * For non-Projen projects, use `CdkDiffIamTemplateGenerator` directly.\n */\nexport class CdkDiffIamTemplate {\n  constructor(props: CdkDiffIamTemplateProps) {\n    const outputPath = props.outputPath ?? 'cdk-diff-workflow-iam-template.yaml';\n\n    // Generate template using the generator\n    const template = CdkDiffIamTemplateGenerator.generateTemplate(props);\n    new TextFile(props.project, outputPath, { lines: template.split('\\n') });\n\n    // Add deploy task\n    props.project.addTask('deploy-cdkdiff-iam-template', {\n      description:\n        'Deploy the CDK Diff IAM template via CloudFormation (accepts extra AWS CLI args, e.g., --parameter-overrides Key=Value...)',\n      receiveArgs: true,\n      exec: CdkDiffIamTemplateGenerator.generateDeployCommand(outputPath),\n    });\n  }\n}\n"]}
|
|
@@ -344,7 +344,7 @@ class CdkDiffIamTemplateStackSetGenerator {
|
|
|
344
344
|
}
|
|
345
345
|
exports.CdkDiffIamTemplateStackSetGenerator = CdkDiffIamTemplateStackSetGenerator;
|
|
346
346
|
_a = JSII_RTTI_SYMBOL_1;
|
|
347
|
-
CdkDiffIamTemplateStackSetGenerator[_a] = { fqn: "@jjrawlins/cdk-diff-pr-github-action.CdkDiffIamTemplateStackSetGenerator", version: "1.
|
|
347
|
+
CdkDiffIamTemplateStackSetGenerator[_a] = { fqn: "@jjrawlins/cdk-diff-pr-github-action.CdkDiffIamTemplateStackSetGenerator", version: "1.8.0" };
|
|
348
348
|
/**
|
|
349
349
|
* Projen construct that creates a CloudFormation StackSet template for org-wide deployment of
|
|
350
350
|
* GitHub OIDC provider, OIDC role, and CDK Diff/Drift IAM roles.
|
|
@@ -389,5 +389,5 @@ class CdkDiffIamTemplateStackSet {
|
|
|
389
389
|
}
|
|
390
390
|
exports.CdkDiffIamTemplateStackSet = CdkDiffIamTemplateStackSet;
|
|
391
391
|
_b = JSII_RTTI_SYMBOL_1;
|
|
392
|
-
CdkDiffIamTemplateStackSet[_b] = { fqn: "@jjrawlins/cdk-diff-pr-github-action.CdkDiffIamTemplateStackSet", version: "1.
|
|
392
|
+
CdkDiffIamTemplateStackSet[_b] = { fqn: "@jjrawlins/cdk-diff-pr-github-action.CdkDiffIamTemplateStackSet", version: "1.8.0" };
|
|
393
393
|
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CdkDiffIamTemplateStackSet.js","sourceRoot":"","sources":["../src/CdkDiffIamTemplateStackSet.ts"],"names":[],"mappings":"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","sourcesContent":["import { TextFile } from 'projen';\n\n/**\n * Which roles to include in the StackSet\n */\nexport enum StackSetRoleSelection {\n  /** Include only the changeset role (CdkChangesetRole) */\n  CHANGESET_ONLY = 'CHANGESET_ONLY',\n  /** Include only the drift role (CdkDriftRole) */\n  DRIFT_ONLY = 'DRIFT_ONLY',\n  /** Include both roles (default) */\n  BOTH = 'BOTH',\n}\n\n/**\n * Configuration for StackSet auto-deployment\n */\nexport interface StackSetAutoDeployment {\n  /** Enable auto-deployment to new accounts in target OUs (default: true) */\n  readonly enabled?: boolean;\n  /** Retain stacks when account leaves OU (default: false) */\n  readonly retainStacksOnAccountRemoval?: boolean;\n}\n\n/**\n * GitHub repository restrictions for OIDC authentication\n */\nexport interface GitHubOidcConfig {\n  /**\n   * GitHub organization or username (e.g., 'my-org' or 'my-username')\n   */\n  readonly owner: string;\n\n  /**\n   * Repository names allowed to assume the role (e.g., ['repo1', 'repo2'])\n   * Use ['*'] to allow all repos in the organization\n   */\n  readonly repositories: string[];\n\n  /**\n   * Branch patterns allowed (e.g., ['main', 'release/*'])\n   * Default: ['*'] (all branches)\n   */\n  readonly branches?: string[];\n\n  /**\n   * Additional subject claims for fine-grained access\n   * e.g., ['pull_request', 'environment:production']\n   */\n  readonly additionalClaims?: string[];\n}\n\n/**\n * Props for generating StackSet templates (no Projen dependency)\n */\nexport interface CdkDiffIamTemplateStackSetGeneratorProps {\n  /** GitHub OIDC configuration for repo/branch restrictions */\n  readonly githubOidc: GitHubOidcConfig;\n\n  /** Name of the GitHub OIDC role (default: 'GitHubOIDCRole') */\n  readonly oidcRoleName?: string;\n\n  /** Name of the CdkChangesetRole (default: 'CdkChangesetRole') */\n  readonly changesetRoleName?: string;\n\n  /** Name of the CdkDriftRole (default: 'CdkDriftRole') */\n  readonly driftRoleName?: string;\n\n  /** Which roles to include (default: BOTH) */\n  readonly roleSelection?: StackSetRoleSelection;\n\n  /** Description for the StackSet */\n  readonly description?: string;\n\n  /**\n   * Skip creating the OIDC provider (use existing one).\n   * Set to true if accounts already have a GitHub OIDC provider.\n   * The template will reference the existing provider by ARN.\n   * Default: false\n   */\n  readonly skipOidcProviderCreation?: boolean;\n}\n\n/**\n * Props for generating StackSet CLI commands (no Projen dependency)\n */\nexport interface CdkDiffIamTemplateStackSetCommandsProps {\n  /** Name of the StackSet (default: 'cdk-diff-workflow-iam-stackset') */\n  readonly stackSetName?: string;\n\n  /** Path to the template file (default: 'cdk-diff-workflow-stackset-template.yaml') */\n  readonly templatePath?: string;\n\n  /** Target OUs for deployment (e.g., ['ou-xxxx-xxxxxxxx', 'r-xxxx']) */\n  readonly targetOrganizationalUnitIds?: string[];\n\n  /** Target regions for deployment (e.g., ['us-east-1', 'eu-west-1']) */\n  readonly regions?: string[];\n\n  /** Auto-deployment configuration */\n  readonly autoDeployment?: StackSetAutoDeployment;\n\n  /**\n   * Whether to use delegated admin mode for StackSet operations.\n   * If true, adds --call-as DELEGATED_ADMIN to commands.\n   * Default: true\n   */\n  readonly delegatedAdmin?: boolean;\n}\n\n/**\n * Pure generator class for StackSet templates and commands.\n * No Projen dependency - can be used in any project.\n */\nexport class CdkDiffIamTemplateStackSetGenerator {\n  /**\n   * Generate the CloudFormation StackSet template as a YAML string.\n   */\n  static generateTemplate(props: CdkDiffIamTemplateStackSetGeneratorProps): string {\n    const oidcRoleName = props.oidcRoleName ?? 'GitHubOIDCRole';\n    const changesetRoleName = props.changesetRoleName ?? 'CdkChangesetRole';\n    const driftRoleName = props.driftRoleName ?? 'CdkDriftRole';\n    const roleSelection = props.roleSelection ?? StackSetRoleSelection.BOTH;\n    const skipOidcProvider = props.skipOidcProviderCreation ?? false;\n\n    const lines = this.generateTemplateLines(\n      props.githubOidc,\n      oidcRoleName,\n      changesetRoleName,\n      driftRoleName,\n      roleSelection,\n      props.description,\n      skipOidcProvider,\n    );\n\n    return lines.join('\\n');\n  }\n\n  /**\n   * Generate AWS CLI commands for StackSet operations.\n   * Returns a map of command names to shell commands.\n   */\n  static generateCommands(props: CdkDiffIamTemplateStackSetCommandsProps = {}): Record<string, string> {\n    const stackSetName = props.stackSetName ?? 'cdk-diff-workflow-iam-stackset';\n    const templatePath = props.templatePath ?? 'cdk-diff-workflow-stackset-template.yaml';\n    const regions = props.regions ?? ['us-east-1'];\n    const targetOUs = props.targetOrganizationalUnitIds ?? [];\n    const autoDeployEnabled = props.autoDeployment?.enabled ?? true;\n    const retainStacks = props.autoDeployment?.retainStacksOnAccountRemoval ?? false;\n    const delegatedAdmin = props.delegatedAdmin ?? true;\n    const callAs = delegatedAdmin ? ' --call-as DELEGATED_ADMIN' : '';\n\n    const ouList = targetOUs.length > 0 ? targetOUs.join(',') : '<OU_IDS>';\n    const regionList = regions.join(' ');\n\n    return {\n      'stackset-create': `aws cloudformation create-stack-set --stack-set-name ${stackSetName} --template-body file://${templatePath} --capabilities CAPABILITY_NAMED_IAM --permission-model SERVICE_MANAGED --auto-deployment Enabled=${autoDeployEnabled},RetainStacksOnAccountRemoval=${retainStacks}${callAs}`,\n      'stackset-update': `aws cloudformation update-stack-set --stack-set-name ${stackSetName} --template-body file://${templatePath} --capabilities CAPABILITY_NAMED_IAM${callAs}`,\n      'stackset-deploy-instances': `aws cloudformation create-stack-instances --stack-set-name ${stackSetName} --deployment-targets OrganizationalUnitIds=${ouList} --regions ${regionList}${callAs}`,\n      'stackset-delete-instances': `aws cloudformation delete-stack-instances --stack-set-name ${stackSetName} --deployment-targets OrganizationalUnitIds=${ouList} --regions ${regionList} --no-retain-stacks${callAs}`,\n      'stackset-delete': `aws cloudformation delete-stack-set --stack-set-name ${stackSetName}${callAs}`,\n      'stackset-describe': `aws cloudformation describe-stack-set --stack-set-name ${stackSetName}${callAs}`,\n      'stackset-list-instances': `aws cloudformation list-stack-instances --stack-set-name ${stackSetName}${callAs}`,\n    };\n  }\n\n  private static generateTemplateLines(\n    githubOidc: GitHubOidcConfig,\n    oidcRoleName: string,\n    changesetRoleName: string,\n    driftRoleName: string,\n    roleSelection: StackSetRoleSelection,\n    description?: string,\n    skipOidcProvider: boolean = false,\n  ): string[] {\n    const lines: string[] = [];\n    const desc = description ?? 'GitHub OIDC and IAM roles for CDK Diff/Drift workflows (StackSet deployment)';\n\n    const includeChangeset =\n      roleSelection === StackSetRoleSelection.BOTH || roleSelection === StackSetRoleSelection.CHANGESET_ONLY;\n    const includeDrift =\n      roleSelection === StackSetRoleSelection.BOTH || roleSelection === StackSetRoleSelection.DRIFT_ONLY;\n\n    // Header\n    lines.push(\"AWSTemplateFormatVersion: '2010-09-09'\");\n    lines.push(`Description: '${desc}'`);\n    lines.push('');\n\n    // Resources\n    lines.push('Resources:');\n\n    // OIDC Provider (only if not skipping)\n    if (!skipOidcProvider) {\n      lines.push(...this.generateOidcProviderLines());\n    }\n\n    // OIDC Role (needs permissions to assume changeset/drift roles)\n    const targetRoleNames: string[] = [];\n    if (includeChangeset) targetRoleNames.push(changesetRoleName);\n    if (includeDrift) targetRoleNames.push(driftRoleName);\n    lines.push(...this.generateOidcRoleLines(oidcRoleName, githubOidc, skipOidcProvider, targetRoleNames));\n\n    // Changeset/Drift roles\n    if (includeChangeset) {\n      lines.push(...this.generateChangesetRoleLines(changesetRoleName));\n    }\n\n    if (includeDrift) {\n      lines.push(...this.generateDriftRoleLines(driftRoleName));\n    }\n\n    // Outputs\n    lines.push('');\n    lines.push('Outputs:');\n    if (!skipOidcProvider) {\n      lines.push(...this.generateOidcOutputLines());\n    }\n\n    // Always output the OIDC role ARN\n    lines.push(...this.generateOidcRoleOutputLines());\n\n    if (includeChangeset) {\n      lines.push(...this.generateChangesetOutputLines());\n    }\n\n    if (includeDrift) {\n      lines.push(...this.generateDriftOutputLines());\n    }\n\n    return lines;\n  }\n\n  private static generateOidcProviderLines(): string[] {\n    return [\n      '  # GitHub OIDC Provider',\n      '  GitHubOIDCProvider:',\n      '    Type: AWS::IAM::OIDCProvider',\n      '    Properties:',\n      '      Url: https://token.actions.githubusercontent.com',\n      '      ClientIdList:',\n      '        - sts.amazonaws.com',\n      '      ThumbprintList:',\n      '        - 6938fd4d98bab03faadb97b34396831e3780aea1',\n      '        - 1c58a3a8518e8759bf075b76b750d4f2df264fcd',\n      '',\n    ];\n  }\n\n  private static generateOidcRoleLines(\n    roleName: string,\n    githubOidc: GitHubOidcConfig,\n    skipOidcProvider: boolean = false,\n    targetRoleNames: string[] = [],\n  ): string[] {\n    const subjectClaims = this.buildSubjectClaims(githubOidc);\n\n    const lines = [\n      '  # GitHub OIDC Role - authenticates GitHub Actions workflows',\n      '  GitHubOIDCRole:',\n      '    Type: AWS::IAM::Role',\n    ];\n\n    // Only add DependsOn if we're creating the provider\n    if (!skipOidcProvider) {\n      lines.push('    DependsOn: GitHubOIDCProvider');\n    }\n\n    lines.push(\n      '    Properties:',\n      \"      RoleName: '\" + roleName + \"'\",\n      '      AssumeRolePolicyDocument:',\n      \"        Version: '2012-10-17'\",\n      '        Statement:',\n      '          - Effect: Allow',\n      '            Principal:',\n      \"              Federated: !Sub 'arn:aws:iam::${AWS::AccountId}:oidc-provider/token.actions.githubusercontent.com'\",\n      '            Action: sts:AssumeRoleWithWebIdentity',\n      '            Condition:',\n      '              StringEquals:',\n      \"                'token.actions.githubusercontent.com:aud': 'sts.amazonaws.com'\",\n      '              StringLike:',\n      \"                'token.actions.githubusercontent.com:sub':\",\n    );\n\n    // Add subject claims\n    for (const claim of subjectClaims) {\n      lines.push(`                  - '${claim}'`);\n    }\n\n    // Add policy to allow assuming the target roles (changeset/drift)\n    if (targetRoleNames.length > 0) {\n      lines.push('      Policies:');\n      lines.push('        - PolicyName: AssumeTargetRoles');\n      lines.push('          PolicyDocument:');\n      lines.push(\"            Version: '2012-10-17'\");\n      lines.push('            Statement:');\n      lines.push('              - Effect: Allow');\n      lines.push('                Action: sts:AssumeRole');\n      lines.push('                Resource:');\n      for (const targetRoleName of targetRoleNames) {\n        lines.push(`                  - !Sub 'arn:aws:iam::\\${AWS::AccountId}:role/${targetRoleName}'`);\n      }\n    }\n\n    lines.push('');\n    return lines;\n  }\n\n  private static buildSubjectClaims(githubOidc: GitHubOidcConfig): string[] {\n    const claims: string[] = [];\n    const branches = githubOidc.branches ?? ['*'];\n\n    for (const repo of githubOidc.repositories) {\n      if (repo === '*') {\n        // Wildcard repo - allow all repos with branch restrictions\n        for (const branch of branches) {\n          if (branch === '*') {\n            claims.push(`repo:${githubOidc.owner}/*`);\n          } else {\n            claims.push(`repo:${githubOidc.owner}/*:ref:refs/heads/${branch}`);\n          }\n        }\n      } else {\n        // Specific repo\n        for (const branch of branches) {\n          if (branch === '*') {\n            claims.push(`repo:${githubOidc.owner}/${repo}:*`);\n          } else {\n            claims.push(`repo:${githubOidc.owner}/${repo}:ref:refs/heads/${branch}`);\n          }\n        }\n      }\n    }\n\n    // Add any additional claims\n    if (githubOidc.additionalClaims) {\n      for (const claim of githubOidc.additionalClaims) {\n        for (const repo of githubOidc.repositories) {\n          if (repo === '*') {\n            claims.push(`repo:${githubOidc.owner}/*:${claim}`);\n          } else {\n            claims.push(`repo:${githubOidc.owner}/${repo}:${claim}`);\n          }\n        }\n      }\n    }\n\n    return claims;\n  }\n\n  private static generateChangesetRoleLines(roleName: string): string[] {\n    return [\n      '  # CloudFormation ChangeSet Role - minimal permissions for changeset operations',\n      '  CdkChangesetRole:',\n      '    Type: AWS::IAM::Role',\n      '    DependsOn: GitHubOIDCRole',\n      '    Properties:',\n      \"      RoleName: '\" + roleName + \"'\",\n      '      AssumeRolePolicyDocument:',\n      \"        Version: '2012-10-17'\",\n      '        Statement:',\n      '          - Effect: Allow',\n      '            Principal:',\n      '              AWS: !GetAtt GitHubOIDCRole.Arn',\n      '            Action: sts:AssumeRole',\n      '      Policies:',\n      '        - PolicyName: CloudFormationChangeSetAccess',\n      '          PolicyDocument:',\n      \"            Version: '2012-10-17'\",\n      '            Statement:',\n      '              # CloudFormation changeset operations',\n      '              - Effect: Allow',\n      '                Action:',\n      '                  - cloudformation:CreateChangeSet',\n      '                  - cloudformation:DescribeChangeSet',\n      '                  - cloudformation:DeleteChangeSet',\n      '                  - cloudformation:ListChangeSets',\n      '                  - cloudformation:DescribeStacks',\n      '                  - cloudformation:DescribeStackResourceDrifts',\n      \"                Resource: '*'\",\n      '              # CDK bootstrap bucket access (for changeset creation)',\n      '              - Effect: Allow',\n      '                Action:',\n      '                  - s3:GetObject',\n      '                  - s3:PutObject',\n      '                  - s3:DeleteObject',\n      '                  - s3:ListBucket',\n      '                Resource:',\n      \"                  - !Sub 'arn:aws:s3:::cdk-${AWS::AccountId}-${AWS::Region}-*'\",\n      \"                  - !Sub 'arn:aws:s3:::cdk-${AWS::AccountId}-${AWS::Region}-*/*'\",\n      '              # CDK bootstrap parameter access',\n      '              - Effect: Allow',\n      '                Action:',\n      '                  - ssm:GetParameter',\n      '                  - ssm:GetParameters',\n      '                  - ssm:GetParametersByPath',\n      \"                Resource: !Sub 'arn:aws:ssm:${AWS::Region}:${AWS::AccountId}:parameter/cdk-bootstrap/*'\",\n      '              # IAM PassRole for CDK operations',\n      '              - Effect: Allow',\n      '                Action:',\n      '                  - iam:PassRole',\n      \"                Resource: '*'\",\n      '                Condition:',\n      '                  StringEquals:',\n      \"                    'iam:PassedToService': 'cloudformation.amazonaws.com'\",\n      '',\n    ];\n  }\n\n  private static generateDriftRoleLines(roleName: string): string[] {\n    return [\n      '  # CloudFormation Drift Detection Role - minimal permissions for drift detection operations',\n      '  CdkDriftRole:',\n      '    Type: AWS::IAM::Role',\n      '    DependsOn: GitHubOIDCRole',\n      '    Properties:',\n      \"      RoleName: '\" + roleName + \"'\",\n      '      AssumeRolePolicyDocument:',\n      \"        Version: '2012-10-17'\",\n      '        Statement:',\n      '          - Effect: Allow',\n      '            Principal:',\n      '              AWS: !GetAtt GitHubOIDCRole.Arn',\n      '            Action: sts:AssumeRole',\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    ];\n  }\n\n  private static generateOidcOutputLines(): string[] {\n    return [\n      '  GitHubOIDCProviderArn:',\n      \"    Description: 'ARN of the GitHub OIDC provider'\",\n      '    Value: !GetAtt GitHubOIDCProvider.Arn',\n      '    Export:',\n      \"      Name: !Sub '${AWS::StackName}-GitHubOIDCProviderArn'\",\n      '',\n    ];\n  }\n\n  private static generateOidcRoleOutputLines(): string[] {\n    return [\n      '  GitHubOIDCRoleArn:',\n      \"    Description: 'ARN of the GitHub OIDC role'\",\n      '    Value: !GetAtt GitHubOIDCRole.Arn',\n      '    Export:',\n      \"      Name: !Sub '${AWS::StackName}-GitHubOIDCRoleArn'\",\n      '',\n      '  GitHubOIDCRoleName:',\n      \"    Description: 'Name of the GitHub OIDC role'\",\n      '    Value: !Ref GitHubOIDCRole',\n      '    Export:',\n      \"      Name: !Sub '${AWS::StackName}-GitHubOIDCRoleName'\",\n      '',\n    ];\n  }\n\n  private static generateChangesetOutputLines(): string[] {\n    return [\n      '  CdkChangesetRoleArn:',\n      \"    Description: 'ARN of the CDK changeset role'\",\n      '    Value: !GetAtt CdkChangesetRole.Arn',\n      '    Export:',\n      \"      Name: !Sub '${AWS::StackName}-CdkChangesetRoleArn'\",\n      '',\n      '  CdkChangesetRoleName:',\n      \"    Description: 'Name of the CDK changeset role'\",\n      '    Value: !Ref CdkChangesetRole',\n      '    Export:',\n      \"      Name: !Sub '${AWS::StackName}-CdkChangesetRoleName'\",\n      '',\n    ];\n  }\n\n  private static generateDriftOutputLines(): string[] {\n    return [\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}\n\n/**\n * Props for the Projen-integrated StackSet construct\n */\nexport interface CdkDiffIamTemplateStackSetProps extends CdkDiffIamTemplateStackSetGeneratorProps {\n  /** Projen project instance */\n  readonly project: any;\n\n  /** Name of the StackSet (default: 'cdk-diff-workflow-iam-stackset') */\n  readonly stackSetName?: string;\n\n  /** Output path for the template file (default: 'cdk-diff-workflow-stackset-template.yaml') */\n  readonly outputPath?: string;\n\n  /** Target OUs for deployment (e.g., ['ou-xxxx-xxxxxxxx', 'r-xxxx']) */\n  readonly targetOrganizationalUnitIds?: string[];\n\n  /** Target regions for deployment (e.g., ['us-east-1', 'eu-west-1']) */\n  readonly regions?: string[];\n\n  /** Auto-deployment configuration */\n  readonly autoDeployment?: StackSetAutoDeployment;\n\n  /**\n   * Whether to use delegated admin mode for StackSet operations.\n   * If true, adds --call-as DELEGATED_ADMIN to commands.\n   * If false, assumes running from the management account.\n   * Default: true\n   */\n  readonly delegatedAdmin?: boolean;\n}\n\n/**\n * Projen construct that creates a CloudFormation StackSet template for org-wide deployment of\n * GitHub OIDC provider, OIDC role, and CDK Diff/Drift IAM roles.\n *\n * This provides a self-contained per-account deployment with no role chaining required.\n *\n * For non-Projen projects, use `CdkDiffIamTemplateStackSetGenerator` directly.\n */\nexport class CdkDiffIamTemplateStackSet {\n  constructor(props: CdkDiffIamTemplateStackSetProps) {\n    const outputPath = props.outputPath ?? 'cdk-diff-workflow-stackset-template.yaml';\n    const stackSetName = props.stackSetName ?? 'cdk-diff-workflow-iam-stackset';\n\n    // Generate template using the generator\n    const template = CdkDiffIamTemplateStackSetGenerator.generateTemplate(props);\n    new TextFile(props.project, outputPath, { lines: template.split('\\n') });\n\n    // Generate commands and add as Projen tasks\n    const commands = CdkDiffIamTemplateStackSetGenerator.generateCommands({\n      stackSetName,\n      templatePath: outputPath,\n      targetOrganizationalUnitIds: props.targetOrganizationalUnitIds,\n      regions: props.regions,\n      autoDeployment: props.autoDeployment,\n      delegatedAdmin: props.delegatedAdmin,\n    });\n\n    const taskDescriptions: Record<string, string> = {\n      'stackset-create': 'Create the StackSet for org-wide IAM role deployment',\n      'stackset-update': 'Update the StackSet template',\n      'stackset-deploy-instances':\n        'Deploy stack instances to target OUs and regions (pass --deployment-targets OrganizationalUnitIds=<ou-ids> to override)',\n      'stackset-delete-instances': 'Delete stack instances from target OUs and regions',\n      'stackset-delete': 'Delete the StackSet (requires all instances to be deleted first)',\n      'stackset-describe': 'Describe the StackSet status and configuration',\n      'stackset-list-instances': 'List all stack instances and their statuses',\n    };\n\n    for (const [taskName, command] of Object.entries(commands)) {\n      props.project.addTask(taskName, {\n        description: taskDescriptions[taskName],\n        receiveArgs: taskName !== 'stackset-delete',\n        exec: command,\n      });\n    }\n  }\n}\n"]}
|
|
@@ -13,6 +13,16 @@ export interface CdkDiffStackWorkflowProps {
|
|
|
13
13
|
readonly nodeVersion?: string;
|
|
14
14
|
readonly cdkYarnCommand?: string;
|
|
15
15
|
readonly scriptOutputPath?: string;
|
|
16
|
+
/**
|
|
17
|
+
* Working directory for the CDK app, relative to the repository root.
|
|
18
|
+
* Useful for monorepos where infrastructure lives in a subdirectory (e.g., 'infra').
|
|
19
|
+
*
|
|
20
|
+
* When set, all workflow run steps will use `defaults.run.working-directory`
|
|
21
|
+
* and script paths will be adjusted to use absolute references.
|
|
22
|
+
*
|
|
23
|
+
* @default - repository root
|
|
24
|
+
*/
|
|
25
|
+
readonly workingDirectory?: string;
|
|
16
26
|
}
|
|
17
27
|
export declare class CdkDiffStackWorkflow {
|
|
18
28
|
private static scriptCreated;
|
|
@@ -17,6 +17,9 @@ class CdkDiffStackWorkflow {
|
|
|
17
17
|
const cdkYarnCommand = props.cdkYarnCommand ?? 'cdk';
|
|
18
18
|
const nodeVersion = props.nodeVersion ?? '24.x';
|
|
19
19
|
const scriptOutputPath = props.scriptOutputPath ?? '.github/workflows/scripts/describe-cfn-changeset.ts';
|
|
20
|
+
// Normalize working directory: strip trailing slashes
|
|
21
|
+
const wd = props.workingDirectory?.replace(/\/+$/, '');
|
|
22
|
+
const defaults = wd ? { run: { workingDirectory: wd } } : undefined;
|
|
20
23
|
// Validate OIDC configuration
|
|
21
24
|
this.validateOidcConfiguration(props);
|
|
22
25
|
// Only create the changeset script once to avoid collisions
|
|
@@ -34,7 +37,7 @@ class CdkDiffStackWorkflow {
|
|
|
34
37
|
const fileName = `${workflowName}.yml`;
|
|
35
38
|
const gh = props.project.github ?? new github_1.GitHub(props.project);
|
|
36
39
|
const diffDeployWorkflow = new github_1.GithubWorkflow(gh, workflowName, { fileName });
|
|
37
|
-
this.createWorkflowForStack(diffDeployWorkflow, stack, cdkYarnCommand, nodeVersion, scriptOutputPath, props.oidcRoleArn, props.oidcRegion);
|
|
40
|
+
this.createWorkflowForStack(diffDeployWorkflow, stack, cdkYarnCommand, nodeVersion, scriptOutputPath, defaults, props.oidcRoleArn, props.oidcRegion);
|
|
38
41
|
}
|
|
39
42
|
}
|
|
40
43
|
validateOidcConfiguration(props) {
|
|
@@ -51,10 +54,12 @@ class CdkDiffStackWorkflow {
|
|
|
51
54
|
throw new Error('Either provide default oidcRegion or specify oidcRegion for each stack');
|
|
52
55
|
}
|
|
53
56
|
}
|
|
54
|
-
createWorkflowForStack(workflow, stack, cdkYarnCommand, nodeVersion, scriptOutputPath, defaultOidcRoleArn, defaultOidcRegion) {
|
|
57
|
+
createWorkflowForStack(workflow, stack, cdkYarnCommand, nodeVersion, scriptOutputPath, defaults, defaultOidcRoleArn, defaultOidcRegion) {
|
|
55
58
|
// Sanitize stack name for CloudFormation API calls (must match [a-zA-Z][-a-zA-Z0-9]*, max 128 chars)
|
|
56
59
|
// Original stack.stackName is preserved only for the CDK deploy target and step display names
|
|
57
60
|
const sanitizedStackName = sanitizeForCloudFormation(stack.stackName);
|
|
61
|
+
// When working directory is set, the describe-changeset step overrides back to repo root
|
|
62
|
+
// because the script needs root-level ts-node, tsconfig.json, and node_modules (e.g. @aws-sdk)
|
|
58
63
|
workflow.on({
|
|
59
64
|
pullRequest: {
|
|
60
65
|
types: ['opened', 'synchronize', 'reopened'],
|
|
@@ -63,6 +68,7 @@ class CdkDiffStackWorkflow {
|
|
|
63
68
|
workflow.addJobs({
|
|
64
69
|
diff: {
|
|
65
70
|
runsOn: ['ubuntu-latest'],
|
|
71
|
+
...(defaults && { defaults }),
|
|
66
72
|
permissions: {
|
|
67
73
|
idToken: workflows_model_1.JobPermission.WRITE,
|
|
68
74
|
contents: workflows_model_1.JobPermission.READ,
|
|
@@ -158,8 +164,13 @@ class CdkDiffStackWorkflow {
|
|
|
158
164
|
},
|
|
159
165
|
{
|
|
160
166
|
name: `Describe change set for ${stack.stackName}`,
|
|
161
|
-
run:
|
|
167
|
+
run: defaults
|
|
168
|
+
? `npx ts-node --transpile-only \$GITHUB_WORKSPACE/${scriptOutputPath}`
|
|
169
|
+
: `npx ts-node ${scriptOutputPath}`,
|
|
162
170
|
env: {
|
|
171
|
+
...(defaults && {
|
|
172
|
+
NODE_PATH: `\${{ github.workspace }}/${defaults.run.workingDirectory}/node_modules`,
|
|
173
|
+
}),
|
|
163
174
|
STACK_NAME: '${{ steps.create-changeset.outputs.cf-stack-name }}',
|
|
164
175
|
CHANGE_SET_NAME: sanitizedStackName,
|
|
165
176
|
GITHUB_COMMENT_URL: '${{ github.event.pull_request.comments_url }}',
|
|
@@ -177,7 +188,7 @@ class CdkDiffStackWorkflow {
|
|
|
177
188
|
}
|
|
178
189
|
exports.CdkDiffStackWorkflow = CdkDiffStackWorkflow;
|
|
179
190
|
_a = JSII_RTTI_SYMBOL_1;
|
|
180
|
-
CdkDiffStackWorkflow[_a] = { fqn: "@jjrawlins/cdk-diff-pr-github-action.CdkDiffStackWorkflow", version: "1.
|
|
191
|
+
CdkDiffStackWorkflow[_a] = { fqn: "@jjrawlins/cdk-diff-pr-github-action.CdkDiffStackWorkflow", version: "1.8.0" };
|
|
181
192
|
CdkDiffStackWorkflow.scriptCreated = false;
|
|
182
193
|
function sanitizeForFileName(name) {
|
|
183
194
|
return name.replace(/[^a-zA-Z0-9]+/g, '-').replace(/^-+|-+$/g, '').toLowerCase();
|
|
@@ -213,4 +224,4 @@ function buildWorkflowName(prefix, sanitizedName, maxLength = MAX_WORKFLOW_FILEN
|
|
|
213
224
|
// Truncate from the beginning to keep the stack name (end) intact
|
|
214
225
|
return `${prefix}${sanitizedName.slice(-available).replace(/^-+/, '')}`;
|
|
215
226
|
}
|
|
216
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CdkDiffStackWorkflow.js","sourceRoot":"","sources":["../src/CdkDiffStackWorkflow.ts"],"names":[],"mappings":"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","sourcesContent":["import { GitHub, GithubWorkflow } from 'projen/lib/github';\nimport { JobPermission } from 'projen/lib/github/workflows-model';\nimport { CdkChangesetScript } from './bin/cdk-changeset-script';\n\nconst githubActionsAwsCredentialsVersion = 'v4';\nconst githubActionsCheckoutVersion = 'v4';\nconst githubActionsSetupNodeVersion = 'v4';\n\nconst MAX_WORKFLOW_FILENAME_LENGTH = 255;\n\nexport interface CdkDiffStack {\n  readonly stackName: string;\n  readonly changesetRoleToAssumeArn: string;\n  readonly changesetRoleToAssumeRegion: string;\n  readonly oidcRoleArn?: string; // Optional override for OIDC role\n  readonly oidcRegion?: string; // Optional override for OIDC region\n}\n\nexport interface CdkDiffStackWorkflowProps {\n  readonly project: any; // avoid exporting projen types in public API\n  readonly stacks: CdkDiffStack[];\n  readonly oidcRoleArn?: string; // Default OIDC role ARN for all stacks (or each stack must have its own)\n  readonly oidcRegion?: string; // Default OIDC region for all stacks (or each stack must have its own)\n  readonly nodeVersion?: string;\n  readonly cdkYarnCommand?: string;\n  readonly scriptOutputPath?: string;\n}\nexport class CdkDiffStackWorkflow {\n  private static scriptCreated = false;\n\n  constructor(props: CdkDiffStackWorkflowProps) {\n    const cdkYarnCommand = props.cdkYarnCommand ?? 'cdk';\n    const nodeVersion = props.nodeVersion ?? '24.x';\n    const scriptOutputPath = props.scriptOutputPath ?? '.github/workflows/scripts/describe-cfn-changeset.ts';\n\n    // Validate OIDC configuration\n    this.validateOidcConfiguration(props);\n\n    // Only create the changeset script once to avoid collisions\n    if (!CdkDiffStackWorkflow.scriptCreated) {\n      new CdkChangesetScript({\n        project: props.project,\n        outputPath: scriptOutputPath,\n      });\n      CdkDiffStackWorkflow.scriptCreated = true;\n    }\n\n    // Create a separate workflow for each stack\n    for (const stack of props.stacks) {\n      const sanitizedStackName = sanitizeForFileName(stack.stackName);\n      const workflowName = buildWorkflowName('diff-', sanitizedStackName);\n      const fileName = `${workflowName}.yml`;\n      const gh = (props as any).project.github ?? new GitHub((props as any).project);\n      const diffDeployWorkflow = new GithubWorkflow(gh, workflowName, { fileName });\n\n      this.createWorkflowForStack(diffDeployWorkflow, stack, cdkYarnCommand, nodeVersion, scriptOutputPath, props.oidcRoleArn, props.oidcRegion);\n    }\n  }\n\n  private validateOidcConfiguration(props: CdkDiffStackWorkflowProps): 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  private createWorkflowForStack(\n    workflow: GithubWorkflow,\n    stack: CdkDiffStack,\n    cdkYarnCommand: string,\n    nodeVersion: string,\n    scriptOutputPath: string,\n    defaultOidcRoleArn?: string,\n    defaultOidcRegion?: string,\n  ) {\n    // Sanitize stack name for CloudFormation API calls (must match [a-zA-Z][-a-zA-Z0-9]*, max 128 chars)\n    // Original stack.stackName is preserved only for the CDK deploy target and step display names\n    const sanitizedStackName = sanitizeForCloudFormation(stack.stackName);\n\n    workflow.on({\n      pullRequest: {\n        types: ['opened', 'synchronize', 'reopened'],\n      },\n    });\n    workflow.addJobs({\n      diff: {\n        runsOn: ['ubuntu-latest'],\n        permissions: {\n          idToken: JobPermission.WRITE,\n          contents: JobPermission.READ,\n          issues: JobPermission.WRITE,\n          pullRequests: JobPermission.WRITE,\n          packages: JobPermission.READ,\n        },\n        env: {\n          GITHUB_TOKEN: '${{ secrets.GITHUB_TOKEN }}',\n        },\n        steps: [\n          {\n            uses: `actions/checkout@${githubActionsCheckoutVersion}`,\n          },\n          {\n            uses: `actions/setup-node@${githubActionsSetupNodeVersion}`,\n            with: {\n              'node-version': nodeVersion,\n              'registry-url': 'https://npm.pkg.github.com',\n              'scope': '@${{ github.repository_owner }}',\n              'token': '${{ secrets.GITHUB_TOKEN }}',\n            },\n          },\n          {\n            name: 'Install dependencies',\n            run: [\n              'if [ -f yarn.lock ]; then',\n              '  yarn install --frozen-lockfile',\n              'elif [ -f package-lock.json ]; then',\n              '  npm ci',\n              'else',\n              '  echo \"No lock file found (yarn.lock or package-lock.json)\" && exit 1',\n              'fi',\n            ].join('\\n'),\n            env: {\n              NODE_AUTH_TOKEN: '${{ secrets.GITHUB_TOKEN }}',\n            },\n          },\n          {\n            uses: `aws-actions/configure-aws-credentials@${githubActionsAwsCredentialsVersion}`,\n            with: {\n              'role-to-assume': stack.oidcRoleArn ?? defaultOidcRoleArn,\n              'aws-region': stack.oidcRegion ?? defaultOidcRegion,\n            },\n          },\n          {\n            name: `Create Changeset for ${stack.stackName}`,\n            id: 'create-changeset',\n            run: [\n              'set -o pipefail',\n              `yarn ${cdkYarnCommand} deploy ${stack.stackName} --no-execute --change-set-name ${sanitizedStackName} --require-approval never`,\n              '',\n              '# Look up real CF stack name from cdk.out manifest (authoritative source)',\n              'CF_STACK_NAME=\"\"',\n              'for manifest in cdk.out/*/manifest.json cdk.out/manifest.json; do',\n              '  [ -f \"$manifest\" ] || continue',\n              `  name=$(jq -r --arg dp \"${stack.stackName}\" \\\\`,\n              '    \\'.artifacts | to_entries[] | select(.value.type == \"aws:cloudformation:stack\" and .value.displayName == $dp) | (.value.properties.stackName // .key)\\' \\\\',\n              '    \"$manifest\")',\n              '  if [ -n \"$name\" ] && [ \"$name\" != \"null\" ]; then',\n              '    CF_STACK_NAME=\"$name\"',\n              '    break',\n              '  fi',\n              'done',\n              'if [ -z \"$CF_STACK_NAME\" ]; then',\n              '  echo \"Warning: Could not resolve CF stack name from cdk.out, falling back to sanitized name\"',\n              `  CF_STACK_NAME=\"${sanitizedStackName}\"`,\n              'fi',\n              'echo \"cf-stack-name=${CF_STACK_NAME}\" >> \"$GITHUB_OUTPUT\"',\n            ].join('\\n'),\n          },\n          {\n            id: 'creds',\n            uses: `aws-actions/configure-aws-credentials@${githubActionsAwsCredentialsVersion}`,\n            with: {\n              'role-to-assume': stack.oidcRoleArn ?? defaultOidcRoleArn,\n              'aws-region': stack.oidcRegion ?? defaultOidcRegion,\n            },\n          },\n          {\n            name: 'Assume CloudFormation Changeset Role',\n            id: 'changeset-role',\n            uses: `aws-actions/configure-aws-credentials@${githubActionsAwsCredentialsVersion}`,\n            with: {\n              'role-to-assume': stack.changesetRoleToAssumeArn,\n              'role-chaining': true,\n              'role-skip-session-tagging': true,\n              'aws-region': stack.changesetRoleToAssumeRegion,\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: `Describe change set for ${stack.stackName}`,\n            run: `npx ts-node ${scriptOutputPath}`,\n            env: {\n              STACK_NAME: '${{ steps.create-changeset.outputs.cf-stack-name }}',\n              CHANGE_SET_NAME: sanitizedStackName,\n              GITHUB_COMMENT_URL: '${{ github.event.pull_request.comments_url }}',\n              GITHUB_TOKEN: '${{ secrets.GITHUB_TOKEN }}',\n            },\n          },\n          {\n            name: `Delete changeset for ${stack.stackName}`,\n            run: `aws cloudformation delete-change-set --change-set-name ${sanitizedStackName} --stack-name \"\\${{ steps.create-changeset.outputs.cf-stack-name }}\" --region ${stack.changesetRoleToAssumeRegion}`,\n          },\n        ],\n      },\n    });\n  }\n}\n\nexport function sanitizeForFileName(name: string): string {\n  return name.replace(/[^a-zA-Z0-9]+/g, '-').replace(/^-+|-+$/g, '').toLowerCase();\n}\n\nexport const MAX_CF_IDENTIFIER_LENGTH = 128;\n\nexport function sanitizeForCloudFormation(name: string): string {\n  // CloudFormation stack/changeset names: [a-zA-Z][-a-zA-Z0-9]*, max 128 chars\n  let sanitized = name.replace(/[^a-zA-Z0-9]+/g, '-').replace(/^-+|-+$/g, '');\n  // Ensure it starts with a letter\n  if (!sanitized || !/^[a-zA-Z]/.test(sanitized)) {\n    sanitized = `cs-${sanitized}`;\n  }\n  // Truncate from the beginning to keep the stack name (end) intact\n  if (sanitized.length > MAX_CF_IDENTIFIER_LENGTH) {\n    sanitized = sanitized.slice(-MAX_CF_IDENTIFIER_LENGTH).replace(/^-+/, '');\n    // Re-check starts with a letter after truncation\n    if (!/^[a-zA-Z]/.test(sanitized)) {\n      sanitized = `cs-${sanitized.slice(0, MAX_CF_IDENTIFIER_LENGTH - 3)}`;\n    }\n  }\n  return sanitized;\n}\n\nfunction buildWorkflowName(prefix: string, sanitizedName: string, maxLength: number = MAX_WORKFLOW_FILENAME_LENGTH): string {\n  const ext = '.yml';\n  const full = `${prefix}${sanitizedName}${ext}`;\n  if (full.length <= maxLength) {\n    return `${prefix}${sanitizedName}`;\n  }\n  const available = maxLength - prefix.length - ext.length;\n  if (available <= 0) {\n    throw new Error(`Workflow prefix and extension exceed maximum filename length of ${maxLength}`);\n  }\n  // Truncate from the beginning to keep the stack name (end) intact\n  return `${prefix}${sanitizedName.slice(-available).replace(/^-+/, '')}`;\n}\n"]}
|
|
227
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CdkDiffStackWorkflow.js","sourceRoot":"","sources":["../src/CdkDiffStackWorkflow.ts"],"names":[],"mappings":"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","sourcesContent":["import { GitHub, GithubWorkflow } from 'projen/lib/github';\nimport { JobPermission } from 'projen/lib/github/workflows-model';\nimport { CdkChangesetScript } from './bin/cdk-changeset-script';\n\nconst githubActionsAwsCredentialsVersion = 'v4';\nconst githubActionsCheckoutVersion = 'v4';\nconst githubActionsSetupNodeVersion = 'v4';\n\nconst MAX_WORKFLOW_FILENAME_LENGTH = 255;\n\nexport interface CdkDiffStack {\n  readonly stackName: string;\n  readonly changesetRoleToAssumeArn: string;\n  readonly changesetRoleToAssumeRegion: string;\n  readonly oidcRoleArn?: string; // Optional override for OIDC role\n  readonly oidcRegion?: string; // Optional override for OIDC region\n}\n\nexport interface CdkDiffStackWorkflowProps {\n  readonly project: any; // avoid exporting projen types in public API\n  readonly stacks: CdkDiffStack[];\n  readonly oidcRoleArn?: string; // Default OIDC role ARN for all stacks (or each stack must have its own)\n  readonly oidcRegion?: string; // Default OIDC region for all stacks (or each stack must have its own)\n  readonly nodeVersion?: string;\n  readonly cdkYarnCommand?: string;\n  readonly scriptOutputPath?: string;\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 script paths will be adjusted to use absolute references.\n   *\n   * @default - repository root\n   */\n  readonly workingDirectory?: string;\n}\nexport class CdkDiffStackWorkflow {\n  private static scriptCreated = false;\n\n  constructor(props: CdkDiffStackWorkflowProps) {\n    const cdkYarnCommand = props.cdkYarnCommand ?? 'cdk';\n    const nodeVersion = props.nodeVersion ?? '24.x';\n    const scriptOutputPath = props.scriptOutputPath ?? '.github/workflows/scripts/describe-cfn-changeset.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    // Validate OIDC configuration\n    this.validateOidcConfiguration(props);\n\n    // Only create the changeset script once to avoid collisions\n    if (!CdkDiffStackWorkflow.scriptCreated) {\n      new CdkChangesetScript({\n        project: props.project,\n        outputPath: scriptOutputPath,\n      });\n      CdkDiffStackWorkflow.scriptCreated = true;\n    }\n\n    // Create a separate workflow for each stack\n    for (const stack of props.stacks) {\n      const sanitizedStackName = sanitizeForFileName(stack.stackName);\n      const workflowName = buildWorkflowName('diff-', sanitizedStackName);\n      const fileName = `${workflowName}.yml`;\n      const gh = (props as any).project.github ?? new GitHub((props as any).project);\n      const diffDeployWorkflow = new GithubWorkflow(gh, workflowName, { fileName });\n\n      this.createWorkflowForStack(\n        diffDeployWorkflow, stack, cdkYarnCommand, nodeVersion,\n        scriptOutputPath, defaults, props.oidcRoleArn, props.oidcRegion,\n      );\n    }\n  }\n\n  private validateOidcConfiguration(props: CdkDiffStackWorkflowProps): 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  private createWorkflowForStack(\n    workflow: GithubWorkflow,\n    stack: CdkDiffStack,\n    cdkYarnCommand: string,\n    nodeVersion: string,\n    scriptOutputPath: string,\n    defaults: { run: { workingDirectory: string } } | undefined,\n    defaultOidcRoleArn?: string,\n    defaultOidcRegion?: string,\n  ) {\n    // Sanitize stack name for CloudFormation API calls (must match [a-zA-Z][-a-zA-Z0-9]*, max 128 chars)\n    // Original stack.stackName is preserved only for the CDK deploy target and step display names\n    const sanitizedStackName = sanitizeForCloudFormation(stack.stackName);\n\n    // When working directory is set, the describe-changeset step overrides back to repo root\n    // because the script needs root-level ts-node, tsconfig.json, and node_modules (e.g. @aws-sdk)\n\n    workflow.on({\n      pullRequest: {\n        types: ['opened', 'synchronize', 'reopened'],\n      },\n    });\n    workflow.addJobs({\n      diff: {\n        runsOn: ['ubuntu-latest'],\n        ...(defaults && { defaults }),\n        permissions: {\n          idToken: JobPermission.WRITE,\n          contents: JobPermission.READ,\n          issues: JobPermission.WRITE,\n          pullRequests: JobPermission.WRITE,\n          packages: JobPermission.READ,\n        },\n        env: {\n          GITHUB_TOKEN: '${{ secrets.GITHUB_TOKEN }}',\n        },\n        steps: [\n          {\n            uses: `actions/checkout@${githubActionsCheckoutVersion}`,\n          },\n          {\n            uses: `actions/setup-node@${githubActionsSetupNodeVersion}`,\n            with: {\n              'node-version': nodeVersion,\n              'registry-url': 'https://npm.pkg.github.com',\n              'scope': '@${{ github.repository_owner }}',\n              'token': '${{ secrets.GITHUB_TOKEN }}',\n            },\n          },\n          {\n            name: 'Install dependencies',\n            run: [\n              'if [ -f yarn.lock ]; then',\n              '  yarn install --frozen-lockfile',\n              'elif [ -f package-lock.json ]; then',\n              '  npm ci',\n              'else',\n              '  echo \"No lock file found (yarn.lock or package-lock.json)\" && exit 1',\n              'fi',\n            ].join('\\n'),\n            env: {\n              NODE_AUTH_TOKEN: '${{ secrets.GITHUB_TOKEN }}',\n            },\n          },\n          {\n            uses: `aws-actions/configure-aws-credentials@${githubActionsAwsCredentialsVersion}`,\n            with: {\n              'role-to-assume': stack.oidcRoleArn ?? defaultOidcRoleArn,\n              'aws-region': stack.oidcRegion ?? defaultOidcRegion,\n            },\n          },\n          {\n            name: `Create Changeset for ${stack.stackName}`,\n            id: 'create-changeset',\n            run: [\n              'set -o pipefail',\n              `yarn ${cdkYarnCommand} deploy ${stack.stackName} --no-execute --change-set-name ${sanitizedStackName} --require-approval never`,\n              '',\n              '# Look up real CF stack name from cdk.out manifest (authoritative source)',\n              'CF_STACK_NAME=\"\"',\n              'for manifest in cdk.out/*/manifest.json cdk.out/manifest.json; do',\n              '  [ -f \"$manifest\" ] || continue',\n              `  name=$(jq -r --arg dp \"${stack.stackName}\" \\\\`,\n              '    \\'.artifacts | to_entries[] | select(.value.type == \"aws:cloudformation:stack\" and .value.displayName == $dp) | (.value.properties.stackName // .key)\\' \\\\',\n              '    \"$manifest\")',\n              '  if [ -n \"$name\" ] && [ \"$name\" != \"null\" ]; then',\n              '    CF_STACK_NAME=\"$name\"',\n              '    break',\n              '  fi',\n              'done',\n              'if [ -z \"$CF_STACK_NAME\" ]; then',\n              '  echo \"Warning: Could not resolve CF stack name from cdk.out, falling back to sanitized name\"',\n              `  CF_STACK_NAME=\"${sanitizedStackName}\"`,\n              'fi',\n              'echo \"cf-stack-name=${CF_STACK_NAME}\" >> \"$GITHUB_OUTPUT\"',\n            ].join('\\n'),\n          },\n          {\n            id: 'creds',\n            uses: `aws-actions/configure-aws-credentials@${githubActionsAwsCredentialsVersion}`,\n            with: {\n              'role-to-assume': stack.oidcRoleArn ?? defaultOidcRoleArn,\n              'aws-region': stack.oidcRegion ?? defaultOidcRegion,\n            },\n          },\n          {\n            name: 'Assume CloudFormation Changeset Role',\n            id: 'changeset-role',\n            uses: `aws-actions/configure-aws-credentials@${githubActionsAwsCredentialsVersion}`,\n            with: {\n              'role-to-assume': stack.changesetRoleToAssumeArn,\n              'role-chaining': true,\n              'role-skip-session-tagging': true,\n              'aws-region': stack.changesetRoleToAssumeRegion,\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: `Describe change set for ${stack.stackName}`,\n            run: defaults\n              ? `npx ts-node --transpile-only \\$GITHUB_WORKSPACE/${scriptOutputPath}`\n              : `npx ts-node ${scriptOutputPath}`,\n            env: {\n              ...(defaults && {\n                NODE_PATH: `\\${{ github.workspace }}/${defaults.run.workingDirectory}/node_modules`,\n              }),\n              STACK_NAME: '${{ steps.create-changeset.outputs.cf-stack-name }}',\n              CHANGE_SET_NAME: sanitizedStackName,\n              GITHUB_COMMENT_URL: '${{ github.event.pull_request.comments_url }}',\n              GITHUB_TOKEN: '${{ secrets.GITHUB_TOKEN }}',\n            },\n          },\n          {\n            name: `Delete changeset for ${stack.stackName}`,\n            run: `aws cloudformation delete-change-set --change-set-name ${sanitizedStackName} --stack-name \"\\${{ steps.create-changeset.outputs.cf-stack-name }}\" --region ${stack.changesetRoleToAssumeRegion}`,\n          },\n        ],\n      },\n    });\n  }\n}\n\nexport function sanitizeForFileName(name: string): string {\n  return name.replace(/[^a-zA-Z0-9]+/g, '-').replace(/^-+|-+$/g, '').toLowerCase();\n}\n\nexport const MAX_CF_IDENTIFIER_LENGTH = 128;\n\nexport function sanitizeForCloudFormation(name: string): string {\n  // CloudFormation stack/changeset names: [a-zA-Z][-a-zA-Z0-9]*, max 128 chars\n  let sanitized = name.replace(/[^a-zA-Z0-9]+/g, '-').replace(/^-+|-+$/g, '');\n  // Ensure it starts with a letter\n  if (!sanitized || !/^[a-zA-Z]/.test(sanitized)) {\n    sanitized = `cs-${sanitized}`;\n  }\n  // Truncate from the beginning to keep the stack name (end) intact\n  if (sanitized.length > MAX_CF_IDENTIFIER_LENGTH) {\n    sanitized = sanitized.slice(-MAX_CF_IDENTIFIER_LENGTH).replace(/^-+/, '');\n    // Re-check starts with a letter after truncation\n    if (!/^[a-zA-Z]/.test(sanitized)) {\n      sanitized = `cs-${sanitized.slice(0, MAX_CF_IDENTIFIER_LENGTH - 3)}`;\n    }\n  }\n  return sanitized;\n}\n\nfunction buildWorkflowName(prefix: string, sanitizedName: string, maxLength: number = MAX_WORKFLOW_FILENAME_LENGTH): string {\n  const ext = '.yml';\n  const full = `${prefix}${sanitizedName}${ext}`;\n  if (full.length <= maxLength) {\n    return `${prefix}${sanitizedName}`;\n  }\n  const available = maxLength - prefix.length - ext.length;\n  if (available <= 0) {\n    throw new Error(`Workflow prefix and extension exceed maximum filename length of ${maxLength}`);\n  }\n  // Truncate from the beginning to keep the stack name (end) intact\n  return `${prefix}${sanitizedName.slice(-available).replace(/^-+/, '')}`;\n}\n"]}
|
|
@@ -16,6 +16,16 @@ export interface CdkDriftDetectionWorkflowProps {
|
|
|
16
16
|
readonly oidcRegion?: string;
|
|
17
17
|
readonly stacks: Stack[];
|
|
18
18
|
readonly nodeVersion?: string;
|
|
19
|
+
/**
|
|
20
|
+
* Working directory for the CDK app, relative to the repository root.
|
|
21
|
+
* Useful for monorepos where infrastructure lives in a subdirectory (e.g., 'infra').
|
|
22
|
+
*
|
|
23
|
+
* When set, all workflow run steps will use `defaults.run.working-directory`
|
|
24
|
+
* and artifact/script paths will be adjusted accordingly.
|
|
25
|
+
*
|
|
26
|
+
* @default - repository root
|
|
27
|
+
*/
|
|
28
|
+
readonly workingDirectory?: string;
|
|
19
29
|
/**
|
|
20
30
|
* Optional hook to append additional GitHub Actions steps after drift detection per stack.
|
|
21
31
|
* You can supply a static array of steps, or a factory that receives context and returns steps.
|
|
@@ -23,6 +23,9 @@ class CdkDriftDetectionWorkflow {
|
|
|
23
23
|
const createIssues = props.createIssues ?? true;
|
|
24
24
|
const project = props.project;
|
|
25
25
|
const scriptOutputPath = props.scriptOutputPath ?? '.github/workflows/scripts/detect-drift.ts';
|
|
26
|
+
// Normalize working directory: strip trailing slashes
|
|
27
|
+
const wd = props.workingDirectory?.replace(/\/+$/, '');
|
|
28
|
+
const defaults = wd ? { run: { workingDirectory: wd } } : undefined;
|
|
26
29
|
// Only create the drift detection script once to avoid collisions
|
|
27
30
|
if (!CdkDriftDetectionWorkflow.scriptCreated) {
|
|
28
31
|
new cdk_drift_detection_script_1.CdkDriftDetectionScript({
|
|
@@ -60,9 +63,13 @@ class CdkDriftDetectionWorkflow {
|
|
|
60
63
|
const notCondExpr = '${{ !(' + innerCond + ') }}';
|
|
61
64
|
const rawPost = props.postGitHubSteps;
|
|
62
65
|
const postSteps = typeof rawPost === 'function' ? rawPost({ stack: sanitizedStackName }) : (rawPost ?? []);
|
|
66
|
+
// Prefix results file path for uses: steps (which ignore defaults.run.working-directory)
|
|
67
|
+
const artifactResultsPath = wd ? `${wd}/${resultsFile}` : resultsFile;
|
|
68
|
+
const issueResultsPath = wd ? `${wd}/${resultsFile}` : resultsFile;
|
|
63
69
|
jobs[jobId] = {
|
|
64
70
|
name: `Drift Detection - ${sanitizedStackName}`,
|
|
65
71
|
runsOn: ['ubuntu-latest'],
|
|
72
|
+
...(defaults && { defaults }),
|
|
66
73
|
permissions: {
|
|
67
74
|
contents: workflows_model_1.JobPermission.READ,
|
|
68
75
|
idToken: workflows_model_1.JobPermission.WRITE,
|
|
@@ -137,7 +144,7 @@ class CdkDriftDetectionWorkflow {
|
|
|
137
144
|
name: 'Upload results',
|
|
138
145
|
if: condExpr,
|
|
139
146
|
uses: `actions/upload-artifact@${githubActionsUploadArtifactVersion}`,
|
|
140
|
-
with: { 'name': `drift-results-${sanitizedStackName}`, 'path':
|
|
147
|
+
with: { 'name': `drift-results-${sanitizedStackName}`, 'path': artifactResultsPath, 'if-no-files-found': 'ignore' },
|
|
141
148
|
},
|
|
142
149
|
...(createIssues
|
|
143
150
|
? [
|
|
@@ -146,7 +153,7 @@ class CdkDriftDetectionWorkflow {
|
|
|
146
153
|
if: "always() && steps.drift.outcome == 'failure'",
|
|
147
154
|
id: 'issue',
|
|
148
155
|
uses: `actions/github-script@${githubActionsGithubScriptVersion}`,
|
|
149
|
-
with: { 'result-encoding': 'string', 'script': issueScript(sanitizedStackName, stack.driftDetectionRoleToAssumeRegion,
|
|
156
|
+
with: { 'result-encoding': 'string', 'script': issueScript(sanitizedStackName, stack.driftDetectionRoleToAssumeRegion, issueResultsPath) },
|
|
150
157
|
},
|
|
151
158
|
]
|
|
152
159
|
: []),
|
|
@@ -193,7 +200,7 @@ class CdkDriftDetectionWorkflow {
|
|
|
193
200
|
}
|
|
194
201
|
exports.CdkDriftDetectionWorkflow = CdkDriftDetectionWorkflow;
|
|
195
202
|
_a = JSII_RTTI_SYMBOL_1;
|
|
196
|
-
CdkDriftDetectionWorkflow[_a] = { fqn: "@jjrawlins/cdk-diff-pr-github-action.CdkDriftDetectionWorkflow", version: "1.
|
|
203
|
+
CdkDriftDetectionWorkflow[_a] = { fqn: "@jjrawlins/cdk-diff-pr-github-action.CdkDriftDetectionWorkflow", version: "1.8.0" };
|
|
197
204
|
CdkDriftDetectionWorkflow.scriptCreated = false;
|
|
198
205
|
function issueScript(stack, region, resultsFile) {
|
|
199
206
|
// Construct a plain JS script string (no template string nesting mishaps)
|
|
@@ -307,4 +314,4 @@ function toGithubJobId(s) {
|
|
|
307
314
|
}
|
|
308
315
|
return out;
|
|
309
316
|
}
|
|
310
|
-
//# 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   * 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    // 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      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          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': resultsFile, '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, resultsFile) },\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"]}
|
|
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"]}
|
|
@@ -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.
|
|
82
|
+
CdkDriftIamTemplateGenerator[_a] = { fqn: "@jjrawlins/cdk-diff-pr-github-action.CdkDriftIamTemplateGenerator", version: "1.8.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.
|
|
105
|
+
CdkDriftIamTemplate[_b] = { fqn: "@jjrawlins/cdk-diff-pr-github-action.CdkDriftIamTemplate", version: "1.8.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"]}
|