@jjrawlins/cdk-diff-pr-github-action 0.0.71 → 1.0.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 +982 -56
- package/API.md +1132 -186
- package/README.md +194 -14
- package/lib/CdkDiffIamTemplate.d.ts +36 -3
- package/lib/CdkDiffIamTemplate.js +116 -87
- package/lib/CdkDiffIamTemplateStackSet.d.ts +149 -0
- package/lib/CdkDiffIamTemplateStackSet.js +373 -0
- package/lib/CdkDiffStackWorkflow.js +1 -1
- package/lib/CdkDriftDetectionWorkflow.js +1 -1
- package/lib/CdkDriftIamTemplate.d.ts +36 -3
- package/lib/CdkDriftIamTemplate.js +92 -63
- package/lib/index.d.ts +1 -0
- package/lib/index.js +2 -1
- package/node_modules/@aws-sdk/client-sso/package.json +9 -9
- package/node_modules/@aws-sdk/core/package.json +5 -5
- package/node_modules/@aws-sdk/credential-provider-env/package.json +5 -5
- package/node_modules/@aws-sdk/credential-provider-http/package.json +5 -5
- package/node_modules/@aws-sdk/credential-provider-ini/package.json +12 -12
- package/node_modules/@aws-sdk/credential-provider-login/package.json +6 -6
- package/node_modules/@aws-sdk/credential-provider-node/package.json +10 -10
- package/node_modules/@aws-sdk/credential-provider-process/package.json +5 -5
- package/node_modules/@aws-sdk/credential-provider-sso/package.json +7 -7
- package/node_modules/@aws-sdk/credential-provider-web-identity/package.json +6 -6
- package/node_modules/@aws-sdk/middleware-host-header/package.json +4 -4
- package/node_modules/@aws-sdk/middleware-logger/package.json +4 -4
- package/node_modules/@aws-sdk/middleware-recursion-detection/package.json +4 -4
- package/node_modules/@aws-sdk/middleware-user-agent/package.json +5 -5
- package/node_modules/@aws-sdk/nested-clients/package.json +9 -9
- package/node_modules/@aws-sdk/region-config-resolver/package.json +4 -4
- package/node_modules/@aws-sdk/token-providers/package.json +3 -3
- package/node_modules/@aws-sdk/types/package.json +3 -3
- package/node_modules/@aws-sdk/util-locate-window/package.json +3 -3
- package/node_modules/@aws-sdk/util-user-agent-browser/package.json +4 -4
- package/node_modules/@aws-sdk/util-user-agent-node/package.json +5 -5
- package/node_modules/@aws-sdk/xml-builder/package.json +3 -3
- package/package.json +4 -4
package/README.md
CHANGED
|
@@ -1,17 +1,22 @@
|
|
|
1
1
|
# cdk-diff-pr-github-action
|
|
2
2
|
|
|
3
|
-
A
|
|
3
|
+
A library that provides GitHub workflows and IAM templates for:
|
|
4
4
|
- Creating CloudFormation Change Sets for your CDK stacks on pull requests and commenting a formatted diff back on the PR.
|
|
5
5
|
- Detecting CloudFormation drift on a schedule or manual trigger and producing a consolidated summary (optionally creating an issue).
|
|
6
|
+
- Deploying IAM roles across AWS Organizations using StackSets.
|
|
6
7
|
|
|
7
8
|
It also provides ready‑to‑deploy IAM templates with the minimal permissions required for each workflow.
|
|
8
9
|
|
|
9
|
-
|
|
10
|
+
**Works with or without Projen** — The StackSet generator can be used standalone in any Node.js project.
|
|
11
|
+
|
|
12
|
+
This package exposes five constructs:
|
|
10
13
|
|
|
11
14
|
- `CdkDiffStackWorkflow` — Generates one GitHub Actions workflow per stack to create a change set and render the diff back to the PR and Step Summary.
|
|
12
15
|
- `CdkDiffIamTemplate` — Emits a CloudFormation template file with minimal permissions for the Change Set workflow.
|
|
13
16
|
- `CdkDriftDetectionWorkflow` — Generates a GitHub Actions workflow to detect CloudFormation drift per stack, upload machine‑readable results, and aggregate a summary.
|
|
14
17
|
- `CdkDriftIamTemplate` — Emits a CloudFormation template file with minimal permissions for the Drift Detection workflow.
|
|
18
|
+
- `CdkDiffIamTemplateStackSet` — Creates a CloudFormation StackSet template for org-wide deployment of GitHub OIDC and IAM roles (Projen integration).
|
|
19
|
+
- `CdkDiffIamTemplateStackSetGenerator` — Pure generator class for StackSet templates (no Projen dependency).
|
|
15
20
|
|
|
16
21
|
## Quick start
|
|
17
22
|
|
|
@@ -102,7 +107,9 @@ If neither top‑level OIDC defaults nor all per‑stack values are supplied, th
|
|
|
102
107
|
|
|
103
108
|
## Usage: CdkDiffIamTemplate
|
|
104
109
|
|
|
105
|
-
Emit an example IAM template you can deploy in your account for the Change Set workflow
|
|
110
|
+
Emit an example IAM template you can deploy in your account for the Change Set workflow.
|
|
111
|
+
|
|
112
|
+
### With Projen
|
|
106
113
|
|
|
107
114
|
```ts
|
|
108
115
|
import { awscdk } from 'projen';
|
|
@@ -124,7 +131,33 @@ new CdkDiffIamTemplate({
|
|
|
124
131
|
project.synth();
|
|
125
132
|
```
|
|
126
133
|
|
|
127
|
-
|
|
134
|
+
A Projen task is also added:
|
|
135
|
+
|
|
136
|
+
```bash
|
|
137
|
+
npx projen deploy-cdkdiff-iam-template -- --parameter-overrides GitHubOIDCRoleArn=... # plus any extra AWS CLI args
|
|
138
|
+
```
|
|
139
|
+
|
|
140
|
+
### Without Projen (Standalone Generator)
|
|
141
|
+
|
|
142
|
+
```ts
|
|
143
|
+
import { CdkDiffIamTemplateGenerator } from '@jjrawlins/cdk-diff-pr-github-action';
|
|
144
|
+
import * as fs from 'fs';
|
|
145
|
+
|
|
146
|
+
const template = CdkDiffIamTemplateGenerator.generateTemplate({
|
|
147
|
+
roleName: 'cdk-diff-role',
|
|
148
|
+
oidcRoleArn: 'arn:aws:iam::123456789012:role/github-oidc-role',
|
|
149
|
+
oidcRegion: 'us-east-1',
|
|
150
|
+
});
|
|
151
|
+
|
|
152
|
+
fs.writeFileSync('cdk-diff-iam-template.yaml', template);
|
|
153
|
+
|
|
154
|
+
// Get the deploy command
|
|
155
|
+
const deployCmd = CdkDiffIamTemplateGenerator.generateDeployCommand('cdk-diff-iam-template.yaml');
|
|
156
|
+
console.log('Deploy with:', deployCmd);
|
|
157
|
+
```
|
|
158
|
+
|
|
159
|
+
### What the template defines
|
|
160
|
+
|
|
128
161
|
- Parameter `GitHubOIDCRoleArn` with a default from `oidcRoleArn` — the ARN of your existing GitHub OIDC role allowed to assume the change set role.
|
|
129
162
|
- IAM role `CdkChangesetRole` with minimal permissions for:
|
|
130
163
|
- CloudFormation Change Set operations
|
|
@@ -132,12 +165,6 @@ This writes `cdk-diff-workflow-iam-template.yaml` at the project root (or your c
|
|
|
132
165
|
- `iam:PassRole` to `cloudformation.amazonaws.com`
|
|
133
166
|
- Outputs exporting the role name and ARN.
|
|
134
167
|
|
|
135
|
-
A Projen task is also added:
|
|
136
|
-
|
|
137
|
-
```bash
|
|
138
|
-
npx projen deploy-cdkdiff-iam-template -- --parameter-overrides GitHubOIDCRoleArn=... # plus any extra AWS CLI args
|
|
139
|
-
```
|
|
140
|
-
|
|
141
168
|
Use the created role ARN as `changesetRoleToAssumeArn` in `CdkDiffStackWorkflow`.
|
|
142
169
|
|
|
143
170
|
---
|
|
@@ -276,7 +303,9 @@ Details:
|
|
|
276
303
|
|
|
277
304
|
## Usage: CdkDriftIamTemplate
|
|
278
305
|
|
|
279
|
-
Emit an example IAM template you can deploy in your account for the Drift Detection workflow
|
|
306
|
+
Emit an example IAM template you can deploy in your account for the Drift Detection workflow.
|
|
307
|
+
|
|
308
|
+
### With Projen
|
|
280
309
|
|
|
281
310
|
```ts
|
|
282
311
|
import { awscdk } from 'projen';
|
|
@@ -298,17 +327,168 @@ new CdkDriftIamTemplate({
|
|
|
298
327
|
project.synth();
|
|
299
328
|
```
|
|
300
329
|
|
|
301
|
-
|
|
330
|
+
A Projen task is also added:
|
|
331
|
+
|
|
332
|
+
```bash
|
|
333
|
+
npx projen deploy-cdkdrift-iam-template -- --parameter-overrides GitHubOIDCRoleArn=... # plus any extra AWS CLI args
|
|
334
|
+
```
|
|
335
|
+
|
|
336
|
+
### Without Projen (Standalone Generator)
|
|
337
|
+
|
|
338
|
+
```ts
|
|
339
|
+
import { CdkDriftIamTemplateGenerator } from '@jjrawlins/cdk-diff-pr-github-action';
|
|
340
|
+
import * as fs from 'fs';
|
|
341
|
+
|
|
342
|
+
const template = CdkDriftIamTemplateGenerator.generateTemplate({
|
|
343
|
+
roleName: 'cdk-drift-role',
|
|
344
|
+
oidcRoleArn: 'arn:aws:iam::123456789012:role/github-oidc-role',
|
|
345
|
+
oidcRegion: 'us-east-1',
|
|
346
|
+
});
|
|
347
|
+
|
|
348
|
+
fs.writeFileSync('cdk-drift-iam-template.yaml', template);
|
|
349
|
+
|
|
350
|
+
// Get the deploy command
|
|
351
|
+
const deployCmd = CdkDriftIamTemplateGenerator.generateDeployCommand('cdk-drift-iam-template.yaml');
|
|
352
|
+
console.log('Deploy with:', deployCmd);
|
|
353
|
+
```
|
|
354
|
+
|
|
355
|
+
### What the template defines
|
|
356
|
+
|
|
302
357
|
- Parameter `GitHubOIDCRoleArn` with a default from `oidcRoleArn` — the ARN of your existing GitHub OIDC role allowed to assume this drift role.
|
|
303
358
|
- IAM role `CdkDriftRole` with minimal permissions for CloudFormation drift detection operations.
|
|
304
359
|
- Outputs exporting the role name and ARN.
|
|
305
360
|
|
|
306
|
-
|
|
361
|
+
---
|
|
362
|
+
|
|
363
|
+
## Usage: CdkDiffIamTemplateStackSet (Org-Wide Deployment)
|
|
364
|
+
|
|
365
|
+
`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.
|
|
366
|
+
|
|
367
|
+
### Architecture
|
|
368
|
+
|
|
369
|
+
Each account in your organization gets:
|
|
370
|
+
- **GitHub OIDC Provider** — Authenticates GitHub Actions workflows
|
|
371
|
+
- **GitHubOIDCRole** — Trusts the OIDC provider with repo/branch restrictions
|
|
372
|
+
- **CdkChangesetRole** — For PR change set previews (trusts GitHubOIDCRole)
|
|
373
|
+
- **CdkDriftRole** — For drift detection (trusts GitHubOIDCRole)
|
|
374
|
+
|
|
375
|
+
This is a self-contained deployment with **no role chaining required**.
|
|
376
|
+
|
|
377
|
+
### With Projen
|
|
307
378
|
|
|
379
|
+
```ts
|
|
380
|
+
import { awscdk } from 'projen';
|
|
381
|
+
import { CdkDiffIamTemplateStackSet } from '@jjrawlins/cdk-diff-pr-github-action';
|
|
382
|
+
|
|
383
|
+
const project = new awscdk.AwsCdkConstructLibrary({ /* ... */ });
|
|
384
|
+
|
|
385
|
+
new CdkDiffIamTemplateStackSet({
|
|
386
|
+
project,
|
|
387
|
+
githubOidc: {
|
|
388
|
+
owner: 'my-org', // GitHub org or username
|
|
389
|
+
repositories: ['infra-repo', 'app-repo'], // Repos allowed to assume roles
|
|
390
|
+
branches: ['main', 'release/*'], // Branch patterns (default: ['*'])
|
|
391
|
+
},
|
|
392
|
+
targetOrganizationalUnitIds: ['ou-xxxx-xxxxxxxx'], // Target OUs
|
|
393
|
+
regions: ['us-east-1', 'eu-west-1'], // Target regions
|
|
394
|
+
// Optional settings:
|
|
395
|
+
// oidcRoleName: 'GitHubOIDCRole', // default
|
|
396
|
+
// changesetRoleName: 'CdkChangesetRole', // default
|
|
397
|
+
// driftRoleName: 'CdkDriftRole', // default
|
|
398
|
+
// roleSelection: StackSetRoleSelection.BOTH, // BOTH, CHANGESET_ONLY, or DRIFT_ONLY
|
|
399
|
+
// delegatedAdmin: true, // Use --call-as DELEGATED_ADMIN (default: true)
|
|
400
|
+
});
|
|
401
|
+
|
|
402
|
+
project.synth();
|
|
403
|
+
```
|
|
404
|
+
|
|
405
|
+
This creates:
|
|
406
|
+
- `cdk-diff-workflow-stackset-template.yaml` — CloudFormation template
|
|
407
|
+
- Projen tasks for StackSet management
|
|
408
|
+
|
|
409
|
+
**Projen tasks:**
|
|
308
410
|
```bash
|
|
309
|
-
npx projen
|
|
411
|
+
npx projen stackset-create # Create the StackSet
|
|
412
|
+
npx projen stackset-update # Update the StackSet template
|
|
413
|
+
npx projen stackset-deploy-instances # Deploy to target OUs/regions
|
|
414
|
+
npx projen stackset-delete-instances # Remove stack instances
|
|
415
|
+
npx projen stackset-delete # Delete the StackSet
|
|
416
|
+
npx projen stackset-describe # Show StackSet status
|
|
417
|
+
npx projen stackset-list-instances # List all instances
|
|
418
|
+
```
|
|
419
|
+
|
|
420
|
+
### Without Projen (Standalone Generator)
|
|
421
|
+
|
|
422
|
+
For non-Projen projects, use `CdkDiffIamTemplateStackSetGenerator` directly:
|
|
423
|
+
|
|
424
|
+
```ts
|
|
425
|
+
import {
|
|
426
|
+
CdkDiffIamTemplateStackSetGenerator
|
|
427
|
+
} from '@jjrawlins/cdk-diff-pr-github-action';
|
|
428
|
+
import * as fs from 'fs';
|
|
429
|
+
|
|
430
|
+
// Generate the CloudFormation template
|
|
431
|
+
const template = CdkDiffIamTemplateStackSetGenerator.generateTemplate({
|
|
432
|
+
githubOidc: {
|
|
433
|
+
owner: 'my-org',
|
|
434
|
+
repositories: ['infra-repo'],
|
|
435
|
+
branches: ['main'],
|
|
436
|
+
},
|
|
437
|
+
});
|
|
438
|
+
|
|
439
|
+
// Write to file
|
|
440
|
+
fs.writeFileSync('stackset-template.yaml', template);
|
|
441
|
+
|
|
442
|
+
// Get AWS CLI commands for StackSet operations
|
|
443
|
+
const commands = CdkDiffIamTemplateStackSetGenerator.generateCommands({
|
|
444
|
+
stackSetName: 'cdk-diff-workflow-iam-stackset',
|
|
445
|
+
templatePath: 'stackset-template.yaml',
|
|
446
|
+
targetOrganizationalUnitIds: ['ou-xxxx-xxxxxxxx'],
|
|
447
|
+
regions: ['us-east-1'],
|
|
448
|
+
});
|
|
449
|
+
|
|
450
|
+
console.log('Create StackSet:', commands['stackset-create']);
|
|
451
|
+
console.log('Deploy instances:', commands['stackset-deploy-instances']);
|
|
452
|
+
```
|
|
453
|
+
|
|
454
|
+
### GitHub Actions Workflow (Simplified)
|
|
455
|
+
|
|
456
|
+
With per-account OIDC, your workflow is simplified — no role chaining needed:
|
|
457
|
+
|
|
458
|
+
```yaml
|
|
459
|
+
jobs:
|
|
460
|
+
diff:
|
|
461
|
+
runs-on: ubuntu-latest
|
|
462
|
+
permissions:
|
|
463
|
+
id-token: write
|
|
464
|
+
contents: read
|
|
465
|
+
steps:
|
|
466
|
+
- uses: actions/checkout@v4
|
|
467
|
+
|
|
468
|
+
- uses: aws-actions/configure-aws-credentials@v4
|
|
469
|
+
with:
|
|
470
|
+
role-to-assume: arn:aws:iam::${{ env.ACCOUNT_ID }}:role/GitHubOIDCRole
|
|
471
|
+
aws-region: us-east-1
|
|
472
|
+
|
|
473
|
+
- name: Assume Changeset Role
|
|
474
|
+
run: |
|
|
475
|
+
CREDS=$(aws sts assume-role \
|
|
476
|
+
--role-arn arn:aws:iam::${{ env.ACCOUNT_ID }}:role/CdkChangesetRole \
|
|
477
|
+
--role-session-name changeset-session)
|
|
478
|
+
# Export credentials...
|
|
310
479
|
```
|
|
311
480
|
|
|
481
|
+
### GitHubOidcConfig options
|
|
482
|
+
|
|
483
|
+
| Property | Description |
|
|
484
|
+
|----------|-------------|
|
|
485
|
+
| `owner` | GitHub organization or username (required) |
|
|
486
|
+
| `repositories` | Array of repo names, or `['*']` for all repos (required) |
|
|
487
|
+
| `branches` | Array of branch patterns (default: `['*']`) |
|
|
488
|
+
| `additionalClaims` | Extra OIDC claims like `['pull_request', 'environment:production']` |
|
|
489
|
+
|
|
490
|
+
---
|
|
491
|
+
|
|
312
492
|
## Testing
|
|
313
493
|
|
|
314
494
|
This repository includes Jest tests that snapshot the synthesized outputs from Projen and assert that:
|
|
@@ -1,10 +1,43 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
1
|
+
/**
|
|
2
|
+
* Props for generating CDK Diff IAM templates (no Projen dependency)
|
|
3
|
+
*/
|
|
4
|
+
export interface CdkDiffIamTemplateGeneratorProps {
|
|
5
|
+
/** Name for the IAM role */
|
|
3
6
|
readonly roleName: string;
|
|
4
|
-
|
|
7
|
+
/** ARN of the existing GitHub OIDC role that can assume this changeset role */
|
|
5
8
|
readonly oidcRoleArn: string;
|
|
9
|
+
/** Region for the OIDC trust condition */
|
|
6
10
|
readonly oidcRegion: string;
|
|
7
11
|
}
|
|
12
|
+
/**
|
|
13
|
+
* Pure generator class for CDK Diff IAM templates.
|
|
14
|
+
* No Projen dependency - can be used in any project.
|
|
15
|
+
*/
|
|
16
|
+
export declare class CdkDiffIamTemplateGenerator {
|
|
17
|
+
/**
|
|
18
|
+
* Generate the CloudFormation IAM template as a YAML string.
|
|
19
|
+
*/
|
|
20
|
+
static generateTemplate(props: CdkDiffIamTemplateGeneratorProps): string;
|
|
21
|
+
/**
|
|
22
|
+
* Generate the AWS CLI deploy command for the IAM template.
|
|
23
|
+
*/
|
|
24
|
+
static generateDeployCommand(templatePath?: string): string;
|
|
25
|
+
}
|
|
26
|
+
/**
|
|
27
|
+
* Props for the Projen-integrated CDK Diff IAM template construct
|
|
28
|
+
*/
|
|
29
|
+
export interface CdkDiffIamTemplateProps extends CdkDiffIamTemplateGeneratorProps {
|
|
30
|
+
/** Projen project instance */
|
|
31
|
+
readonly project: any;
|
|
32
|
+
/** Output path for the template file (default: 'cdk-diff-workflow-iam-template.yaml') */
|
|
33
|
+
readonly outputPath?: string;
|
|
34
|
+
}
|
|
35
|
+
/**
|
|
36
|
+
* Projen construct that emits a CloudFormation template with minimal IAM permissions
|
|
37
|
+
* for the CDK Diff Stack Workflow.
|
|
38
|
+
*
|
|
39
|
+
* For non-Projen projects, use `CdkDiffIamTemplateGenerator` directly.
|
|
40
|
+
*/
|
|
8
41
|
export declare class CdkDiffIamTemplate {
|
|
9
42
|
constructor(props: CdkDiffIamTemplateProps);
|
|
10
43
|
}
|
|
@@ -1,101 +1,130 @@
|
|
|
1
1
|
"use strict";
|
|
2
|
-
var _a;
|
|
2
|
+
var _a, _b;
|
|
3
3
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
4
|
-
exports.CdkDiffIamTemplate = void 0;
|
|
4
|
+
exports.CdkDiffIamTemplate = exports.CdkDiffIamTemplateGenerator = void 0;
|
|
5
5
|
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
|
6
6
|
const projen_1 = require("projen");
|
|
7
|
+
/**
|
|
8
|
+
* Pure generator class for CDK Diff IAM templates.
|
|
9
|
+
* No Projen dependency - can be used in any project.
|
|
10
|
+
*/
|
|
11
|
+
class CdkDiffIamTemplateGenerator {
|
|
12
|
+
/**
|
|
13
|
+
* Generate the CloudFormation IAM template as a YAML string.
|
|
14
|
+
*/
|
|
15
|
+
static generateTemplate(props) {
|
|
16
|
+
const lines = [
|
|
17
|
+
"AWSTemplateFormatVersion: '2010-09-09'",
|
|
18
|
+
"Description: 'IAM role for CDK Diff Stack Workflow construct'",
|
|
19
|
+
'',
|
|
20
|
+
'Parameters:',
|
|
21
|
+
' GitHubOIDCRoleArn:',
|
|
22
|
+
' Type: String',
|
|
23
|
+
" Description: 'ARN of the existing GitHub OIDC role that can assume this changeset role'",
|
|
24
|
+
` Default: '${props.oidcRoleArn}'`,
|
|
25
|
+
'',
|
|
26
|
+
'Resources:',
|
|
27
|
+
' # CloudFormation ChangeSet Role - minimal permissions for changeset operations',
|
|
28
|
+
' CdkChangesetRole:',
|
|
29
|
+
' Type: AWS::IAM::Role',
|
|
30
|
+
' Properties:',
|
|
31
|
+
" RoleName: '" + props.roleName + "'",
|
|
32
|
+
' AssumeRolePolicyDocument:',
|
|
33
|
+
" Version: '2012-10-17'",
|
|
34
|
+
' Statement:',
|
|
35
|
+
' - Effect: Allow',
|
|
36
|
+
' Principal:',
|
|
37
|
+
' AWS: !Ref GitHubOIDCRoleArn',
|
|
38
|
+
' Action: sts:AssumeRole',
|
|
39
|
+
' Condition:',
|
|
40
|
+
' StringEquals:',
|
|
41
|
+
" aws:RequestedRegion: '" + props.oidcRegion + "'",
|
|
42
|
+
' Policies:',
|
|
43
|
+
' - PolicyName: CloudFormationChangeSetAccess',
|
|
44
|
+
' PolicyDocument:',
|
|
45
|
+
" Version: '2012-10-17'",
|
|
46
|
+
' Statement:',
|
|
47
|
+
' # CloudFormation changeset operations',
|
|
48
|
+
' - Effect: Allow',
|
|
49
|
+
' Action:',
|
|
50
|
+
' - cloudformation:CreateChangeSet',
|
|
51
|
+
' - cloudformation:DescribeChangeSet',
|
|
52
|
+
' - cloudformation:DeleteChangeSet',
|
|
53
|
+
' - cloudformation:ListChangeSets',
|
|
54
|
+
' - cloudformation:DescribeStacks',
|
|
55
|
+
" Resource: '*'",
|
|
56
|
+
' # CDK bootstrap bucket access (for changeset creation)',
|
|
57
|
+
' - Effect: Allow',
|
|
58
|
+
' Action:',
|
|
59
|
+
' - s3:GetObject',
|
|
60
|
+
' - s3:PutObject',
|
|
61
|
+
' - s3:DeleteObject',
|
|
62
|
+
' - s3:ListBucket',
|
|
63
|
+
' Resource:',
|
|
64
|
+
" - !Sub 'arn:aws:s3:::cdk-${AWS::AccountId}-${AWS::Region}-*'",
|
|
65
|
+
" - !Sub 'arn:aws:s3:::cdk-${AWS::AccountId}-${AWS::Region}-*/*'",
|
|
66
|
+
' # CDK bootstrap parameter access',
|
|
67
|
+
' - Effect: Allow',
|
|
68
|
+
' Action:',
|
|
69
|
+
' - ssm:GetParameter',
|
|
70
|
+
' - ssm:GetParameters',
|
|
71
|
+
' - ssm:GetParametersByPath',
|
|
72
|
+
" Resource: !Sub 'arn:aws:ssm:${AWS::Region}:${AWS::AccountId}:parameter/cdk-bootstrap/*'",
|
|
73
|
+
' # IAM PassRole for CDK operations',
|
|
74
|
+
' - Effect: Allow',
|
|
75
|
+
' Action:',
|
|
76
|
+
' - iam:PassRole',
|
|
77
|
+
" Resource: '*'",
|
|
78
|
+
' Condition:',
|
|
79
|
+
' StringEquals:',
|
|
80
|
+
" 'iam:PassedToService': 'cloudformation.amazonaws.com'",
|
|
81
|
+
'',
|
|
82
|
+
'Outputs:',
|
|
83
|
+
' CdkChangesetRoleArn:',
|
|
84
|
+
" Description: 'ARN of the CDK changeset role'",
|
|
85
|
+
' Value: !GetAtt CdkChangesetRole.Arn',
|
|
86
|
+
' Export:',
|
|
87
|
+
" Name: !Sub '${AWS::StackName}-CdkChangesetRoleArn'",
|
|
88
|
+
'',
|
|
89
|
+
' CdkChangesetRoleName:',
|
|
90
|
+
" Description: 'Name of the CDK changeset role'",
|
|
91
|
+
' Value: !Ref CdkChangesetRole',
|
|
92
|
+
' Export:',
|
|
93
|
+
" Name: !Sub '${AWS::StackName}-CdkChangesetRoleName'",
|
|
94
|
+
];
|
|
95
|
+
return lines.join('\n');
|
|
96
|
+
}
|
|
97
|
+
/**
|
|
98
|
+
* Generate the AWS CLI deploy command for the IAM template.
|
|
99
|
+
*/
|
|
100
|
+
static generateDeployCommand(templatePath = 'cdk-diff-workflow-iam-template.yaml') {
|
|
101
|
+
return `aws cloudformation deploy --template-file ${templatePath} --stack-name cdk-diff-workflow-iam-role --capabilities CAPABILITY_NAMED_IAM`;
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
exports.CdkDiffIamTemplateGenerator = CdkDiffIamTemplateGenerator;
|
|
105
|
+
_a = JSII_RTTI_SYMBOL_1;
|
|
106
|
+
CdkDiffIamTemplateGenerator[_a] = { fqn: "@jjrawlins/cdk-diff-pr-github-action.CdkDiffIamTemplateGenerator", version: "1.0.0" };
|
|
107
|
+
/**
|
|
108
|
+
* Projen construct that emits a CloudFormation template with minimal IAM permissions
|
|
109
|
+
* for the CDK Diff Stack Workflow.
|
|
110
|
+
*
|
|
111
|
+
* For non-Projen projects, use `CdkDiffIamTemplateGenerator` directly.
|
|
112
|
+
*/
|
|
7
113
|
class CdkDiffIamTemplate {
|
|
8
114
|
constructor(props) {
|
|
9
115
|
const outputPath = props.outputPath ?? 'cdk-diff-workflow-iam-template.yaml';
|
|
116
|
+
// Generate template using the generator
|
|
117
|
+
const template = CdkDiffIamTemplateGenerator.generateTemplate(props);
|
|
118
|
+
new projen_1.TextFile(props.project, outputPath, { lines: template.split('\n') });
|
|
119
|
+
// Add deploy task
|
|
10
120
|
props.project.addTask('deploy-cdkdiff-iam-template', {
|
|
11
121
|
description: 'Deploy the CDK Diff IAM template via CloudFormation (accepts extra AWS CLI args, e.g., --parameter-overrides Key=Value...)',
|
|
12
122
|
receiveArgs: true,
|
|
13
|
-
exec:
|
|
14
|
-
});
|
|
15
|
-
new projen_1.TextFile(props.project, outputPath, {
|
|
16
|
-
lines: [
|
|
17
|
-
"AWSTemplateFormatVersion: '2010-09-09'",
|
|
18
|
-
"Description: 'IAM role for CDK Diff Stack Workflow construct'",
|
|
19
|
-
'',
|
|
20
|
-
'Parameters:',
|
|
21
|
-
' GitHubOIDCRoleArn:',
|
|
22
|
-
' Type: String',
|
|
23
|
-
" Description: 'ARN of the existing GitHub OIDC role that can assume this changeset role'",
|
|
24
|
-
` Default: '${props.oidcRoleArn}'`,
|
|
25
|
-
'',
|
|
26
|
-
'Resources:',
|
|
27
|
-
' # CloudFormation ChangeSet Role - minimal permissions for changeset operations',
|
|
28
|
-
' CdkChangesetRole:',
|
|
29
|
-
' Type: AWS::IAM::Role',
|
|
30
|
-
' Properties:',
|
|
31
|
-
" RoleName: '" + props.roleName + "'",
|
|
32
|
-
' AssumeRolePolicyDocument:',
|
|
33
|
-
" Version: '2012-10-17'",
|
|
34
|
-
' Statement:',
|
|
35
|
-
' - Effect: Allow',
|
|
36
|
-
' Principal:',
|
|
37
|
-
' AWS: !Ref GitHubOIDCRoleArn',
|
|
38
|
-
' Action: sts:AssumeRole',
|
|
39
|
-
' Condition:',
|
|
40
|
-
' StringEquals:',
|
|
41
|
-
" aws:RequestedRegion: '" + props.oidcRegion + "'",
|
|
42
|
-
' Policies:',
|
|
43
|
-
' - PolicyName: CloudFormationChangeSetAccess',
|
|
44
|
-
' PolicyDocument:',
|
|
45
|
-
" Version: '2012-10-17'",
|
|
46
|
-
' Statement:',
|
|
47
|
-
' # CloudFormation changeset operations',
|
|
48
|
-
' - Effect: Allow',
|
|
49
|
-
' Action:',
|
|
50
|
-
' - cloudformation:CreateChangeSet',
|
|
51
|
-
' - cloudformation:DescribeChangeSet',
|
|
52
|
-
' - cloudformation:DeleteChangeSet',
|
|
53
|
-
' - cloudformation:ListChangeSets',
|
|
54
|
-
' - cloudformation:DescribeStacks',
|
|
55
|
-
" Resource: '*'",
|
|
56
|
-
' # CDK bootstrap bucket access (for changeset creation)',
|
|
57
|
-
' - Effect: Allow',
|
|
58
|
-
' Action:',
|
|
59
|
-
' - s3:GetObject',
|
|
60
|
-
' - s3:PutObject',
|
|
61
|
-
' - s3:DeleteObject',
|
|
62
|
-
' - s3:ListBucket',
|
|
63
|
-
' Resource:',
|
|
64
|
-
" - !Sub 'arn:aws:s3:::cdk-${AWS::AccountId}-${AWS::Region}-*'",
|
|
65
|
-
" - !Sub 'arn:aws:s3:::cdk-${AWS::AccountId}-${AWS::Region}-*/*'",
|
|
66
|
-
' # CDK bootstrap parameter access',
|
|
67
|
-
' - Effect: Allow',
|
|
68
|
-
' Action:',
|
|
69
|
-
' - ssm:GetParameter',
|
|
70
|
-
' - ssm:GetParameters',
|
|
71
|
-
' - ssm:GetParametersByPath',
|
|
72
|
-
" Resource: !Sub 'arn:aws:ssm:${AWS::Region}:${AWS::AccountId}:parameter/cdk-bootstrap/*'",
|
|
73
|
-
' # IAM PassRole for CDK operations',
|
|
74
|
-
' - Effect: Allow',
|
|
75
|
-
' Action:',
|
|
76
|
-
' - iam:PassRole',
|
|
77
|
-
" Resource: '*'",
|
|
78
|
-
' Condition:',
|
|
79
|
-
' StringEquals:',
|
|
80
|
-
" 'iam:PassedToService': 'cloudformation.amazonaws.com'",
|
|
81
|
-
'',
|
|
82
|
-
'Outputs:',
|
|
83
|
-
' CdkChangesetRoleArn:',
|
|
84
|
-
" Description: 'ARN of the CDK changeset role'",
|
|
85
|
-
' Value: !GetAtt CdkChangesetRole.Arn',
|
|
86
|
-
' Export:',
|
|
87
|
-
" Name: !Sub '${AWS::StackName}-CdkChangesetRoleArn'",
|
|
88
|
-
'',
|
|
89
|
-
' CdkChangesetRoleName:',
|
|
90
|
-
" Description: 'Name of the CDK changeset role'",
|
|
91
|
-
' Value: !Ref CdkChangesetRole',
|
|
92
|
-
' Export:',
|
|
93
|
-
" Name: !Sub '${AWS::StackName}-CdkChangesetRoleName'",
|
|
94
|
-
],
|
|
123
|
+
exec: CdkDiffIamTemplateGenerator.generateDeployCommand(outputPath),
|
|
95
124
|
});
|
|
96
125
|
}
|
|
97
126
|
}
|
|
98
127
|
exports.CdkDiffIamTemplate = CdkDiffIamTemplate;
|
|
99
|
-
|
|
100
|
-
CdkDiffIamTemplate[
|
|
101
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CdkDiffIamTemplate.js","sourceRoot":"","sources":["../src/CdkDiffIamTemplate.ts"],"names":[],"mappings":";;;;;AAAA,mCAAkC;AAUlC,MAAa,kBAAkB;IAC7B,YAAY,KAA8B;QACxC,MAAM,UAAU,GAAG,KAAK,CAAC,UAAU,IAAI,qCAAqC,CAAC;QAC7E,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC,6BAA6B,EAAE;YACnD,WAAW,EAAE,4HAA4H;YACzI,WAAW,EAAE,IAAI;YACjB,IAAI,EAAE,6CAA6C,UAAU,8EAA8E;SAC5I,CAAC,CAAC;QAEH,IAAI,iBAAQ,CAAC,KAAK,CAAC,OAAO,EAAE,UAAU,EAAE;YACtC,KAAK,EAAE;gBACL,wCAAwC;gBACxC,+DAA+D;gBAC/D,EAAE;gBACF,aAAa;gBACb,sBAAsB;gBACtB,kBAAkB;gBAClB,6FAA6F;gBAC7F,iBAAiB,KAAK,CAAC,WAAW,GAAG;gBACrC,EAAE;gBACF,YAAY;gBACZ,kFAAkF;gBAClF,qBAAqB;gBACrB,0BAA0B;gBAC1B,iBAAiB;gBACjB,mBAAmB,GAAG,KAAK,CAAC,QAAQ,GAAG,GAAG;gBAC1C,iCAAiC;gBACjC,+BAA+B;gBAC/B,oBAAoB;gBACpB,2BAA2B;gBAC3B,wBAAwB;gBACxB,2CAA2C;gBAC3C,oCAAoC;gBACpC,wBAAwB;gBACxB,6BAA6B;gBAC7B,wCAAwC,GAAG,KAAK,CAAC,UAAU,GAAG,GAAG;gBACjE,iBAAiB;gBACjB,qDAAqD;gBACrD,2BAA2B;gBAC3B,mCAAmC;gBACnC,wBAAwB;gBACxB,qDAAqD;gBACrD,+BAA+B;gBAC/B,yBAAyB;gBACzB,oDAAoD;gBACpD,sDAAsD;gBACtD,oDAAoD;gBACpD,mDAAmD;gBACnD,mDAAmD;gBACnD,+BAA+B;gBAC/B,sEAAsE;gBACtE,+BAA+B;gBAC/B,yBAAyB;gBACzB,kCAAkC;gBAClC,kCAAkC;gBAClC,qCAAqC;gBACrC,mCAAmC;gBACnC,2BAA2B;gBAC3B,gFAAgF;gBAChF,kFAAkF;gBAClF,gDAAgD;gBAChD,+BAA+B;gBAC/B,yBAAyB;gBACzB,sCAAsC;gBACtC,uCAAuC;gBACvC,6CAA6C;gBAC7C,yGAAyG;gBACzG,iDAAiD;gBACjD,+BAA+B;gBAC/B,yBAAyB;gBACzB,kCAAkC;gBAClC,+BAA+B;gBAC/B,4BAA4B;gBAC5B,iCAAiC;gBACjC,2EAA2E;gBAC3E,EAAE;gBACF,UAAU;gBACV,wBAAwB;gBACxB,kDAAkD;gBAClD,yCAAyC;gBACzC,aAAa;gBACb,0DAA0D;gBAC1D,EAAE;gBACF,yBAAyB;gBACzB,mDAAmD;gBACnD,kCAAkC;gBAClC,aAAa;gBACb,2DAA2D;aAC5D;SACF,CAAC,CAAC;IACL,CAAC;;AA1FH,gDA2FC","sourcesContent":["import { TextFile } from 'projen';\n\nexport interface CdkDiffIamTemplateProps {\n  readonly project: any;\n  readonly roleName: string;\n  readonly outputPath?: string;\n  readonly oidcRoleArn: string;\n  readonly oidcRegion: string;\n}\n\nexport class CdkDiffIamTemplate {\n  constructor(props: CdkDiffIamTemplateProps) {\n    const outputPath = props.outputPath ?? 'cdk-diff-workflow-iam-template.yaml';\n    props.project.addTask('deploy-cdkdiff-iam-template', {\n      description: 'Deploy the CDK Diff IAM template via CloudFormation (accepts extra AWS CLI args, e.g., --parameter-overrides Key=Value...)',\n      receiveArgs: true,\n      exec: `aws cloudformation deploy --template-file ${outputPath} --stack-name cdk-diff-workflow-iam-role --capabilities CAPABILITY_NAMED_IAM`,\n    });\n\n    new TextFile(props.project, outputPath, {\n      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        \"                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  }\n}\n"]}
|
|
128
|
+
_b = JSII_RTTI_SYMBOL_1;
|
|
129
|
+
CdkDiffIamTemplate[_b] = { fqn: "@jjrawlins/cdk-diff-pr-github-action.CdkDiffIamTemplate", version: "1.0.0" };
|
|
130
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CdkDiffIamTemplate.js","sourceRoot":"","sources":["../src/CdkDiffIamTemplate.ts"],"names":[],"mappings":";;;;;AAAA,mCAAkC;AAclC;;;GAGG;AACH,MAAa,2BAA2B;IACtC;;OAEG;IACH,MAAM,CAAC,gBAAgB,CAAC,KAAuC;QAC7D,MAAM,KAAK,GAAG;YACZ,wCAAwC;YACxC,+DAA+D;YAC/D,EAAE;YACF,aAAa;YACb,sBAAsB;YACtB,kBAAkB;YAClB,6FAA6F;YAC7F,iBAAiB,KAAK,CAAC,WAAW,GAAG;YACrC,EAAE;YACF,YAAY;YACZ,kFAAkF;YAClF,qBAAqB;YACrB,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,qDAAqD;YACrD,2BAA2B;YAC3B,mCAAmC;YACnC,wBAAwB;YACxB,qDAAqD;YACrD,+BAA+B;YAC/B,yBAAyB;YACzB,oDAAoD;YACpD,sDAAsD;YACtD,oDAAoD;YACpD,mDAAmD;YACnD,mDAAmD;YACnD,+BAA+B;YAC/B,sEAAsE;YACtE,+BAA+B;YAC/B,yBAAyB;YACzB,kCAAkC;YAClC,kCAAkC;YAClC,qCAAqC;YACrC,mCAAmC;YACnC,2BAA2B;YAC3B,gFAAgF;YAChF,kFAAkF;YAClF,gDAAgD;YAChD,+BAA+B;YAC/B,yBAAyB;YACzB,sCAAsC;YACtC,uCAAuC;YACvC,6CAA6C;YAC7C,yGAAyG;YACzG,iDAAiD;YACjD,+BAA+B;YAC/B,yBAAyB;YACzB,kCAAkC;YAClC,+BAA+B;YAC/B,4BAA4B;YAC5B,iCAAiC;YACjC,2EAA2E;YAC3E,EAAE;YACF,UAAU;YACV,wBAAwB;YACxB,kDAAkD;YAClD,yCAAyC;YACzC,aAAa;YACb,0DAA0D;YAC1D,EAAE;YACF,yBAAyB;YACzB,mDAAmD;YACnD,kCAAkC;YAClC,aAAa;YACb,2DAA2D;SAC5D,CAAC;QAEF,OAAO,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IAC1B,CAAC;IAED;;OAEG;IACH,MAAM,CAAC,qBAAqB,CAAC,eAAuB,qCAAqC;QACvF,OAAO,6CAA6C,YAAY,8EAA8E,CAAC;IACjJ,CAAC;;AA7FH,kEA8FC;;;AAYD;;;;;GAKG;AACH,MAAa,kBAAkB;IAC7B,YAAY,KAA8B;QACxC,MAAM,UAAU,GAAG,KAAK,CAAC,UAAU,IAAI,qCAAqC,CAAC;QAE7E,wCAAwC;QACxC,MAAM,QAAQ,GAAG,2BAA2B,CAAC,gBAAgB,CAAC,KAAK,CAAC,CAAC;QACrE,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,6BAA6B,EAAE;YACnD,WAAW,EACT,4HAA4H;YAC9H,WAAW,EAAE,IAAI;YACjB,IAAI,EAAE,2BAA2B,CAAC,qBAAqB,CAAC,UAAU,CAAC;SACpE,CAAC,CAAC;IACL,CAAC;;AAfH,gDAgBC","sourcesContent":["import { TextFile } from 'projen';\n\n/**\n * Props for generating CDK Diff IAM templates (no Projen dependency)\n */\nexport interface CdkDiffIamTemplateGeneratorProps {\n  /** Name for the IAM role */\n  readonly roleName: string;\n  /** ARN of the existing GitHub OIDC role that can assume this changeset 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 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 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      \"                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  /**\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/**\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"]}
|