@jjrawlins/cdk-diff-pr-github-action 1.1.6 → 1.2.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/README.md CHANGED
@@ -107,9 +107,16 @@ If neither top‑level OIDC defaults nor all per‑stack values are supplied, th
107
107
 
108
108
  ## Usage: CdkDiffIamTemplate
109
109
 
110
- Emit an example IAM template you can deploy in your account for the Change Set workflow.
110
+ Emit an IAM template you can deploy in your account for the Change Set workflow. Supports two modes:
111
111
 
112
- ### With Projen
112
+ 1. **External OIDC Role** — Reference an existing GitHub OIDC role (original behavior)
113
+ 2. **Self-Contained** — Create the GitHub OIDC provider and role within the same template (new)
114
+
115
+ ### Option 1: Using an Existing OIDC Role (External)
116
+
117
+ Use this when you already have a GitHub OIDC provider and role set up in your account.
118
+
119
+ #### With Projen
113
120
 
114
121
  ```ts
115
122
  import { awscdk } from 'projen';
@@ -131,13 +138,7 @@ new CdkDiffIamTemplate({
131
138
  project.synth();
132
139
  ```
133
140
 
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
+ #### Without Projen (Standalone Generator)
141
142
 
142
143
  ```ts
143
144
  import { CdkDiffIamTemplateGenerator } from '@jjrawlins/cdk-diff-pr-github-action';
@@ -150,22 +151,116 @@ const template = CdkDiffIamTemplateGenerator.generateTemplate({
150
151
  });
151
152
 
152
153
  fs.writeFileSync('cdk-diff-iam-template.yaml', template);
154
+ ```
153
155
 
154
- // Get the deploy command
155
- const deployCmd = CdkDiffIamTemplateGenerator.generateDeployCommand('cdk-diff-iam-template.yaml');
156
- console.log('Deploy with:', deployCmd);
156
+ ### Option 2: Self-Contained Template (Create OIDC Role)
157
+
158
+ Use 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`.
159
+
160
+ #### With Projen
161
+
162
+ ```ts
163
+ import { awscdk } from 'projen';
164
+ import { CdkDiffIamTemplate } from '@jjrawlins/cdk-diff-pr-github-action';
165
+
166
+ const project = new awscdk.AwsCdkConstructLibrary({
167
+ // ...
168
+ });
169
+
170
+ new CdkDiffIamTemplate({
171
+ project,
172
+ roleName: 'CdkChangesetRole',
173
+ createOidcRole: true,
174
+ oidcRoleName: 'GitHubOIDCRole', // Optional, default: 'GitHubOIDCRole'
175
+ githubOidc: {
176
+ owner: 'my-org', // GitHub org or username
177
+ repositories: ['infra-repo', 'app-repo'], // Repos allowed to assume roles
178
+ branches: ['main', 'release/*'], // Branch patterns (default: ['*'])
179
+ },
180
+ // Optional: Skip OIDC provider creation if it already exists
181
+ // skipOidcProviderCreation: true,
182
+ });
183
+
184
+ project.synth();
157
185
  ```
158
186
 
159
- ### What the template defines
187
+ #### Without Projen (Standalone Generator)
160
188
 
161
- - Parameter `GitHubOIDCRoleArn` with a default from `oidcRoleArn` — the ARN of your existing GitHub OIDC role allowed to assume the change set role.
162
- - IAM role `CdkChangesetRole` with minimal permissions for:
163
- - CloudFormation Change Set operations
164
- - Access to common CDK bootstrap S3 buckets and SSM parameters
165
- - `iam:PassRole` to `cloudformation.amazonaws.com`
166
- - Outputs exporting the role name and ARN.
189
+ ```ts
190
+ import { CdkDiffIamTemplateGenerator } from '@jjrawlins/cdk-diff-pr-github-action';
191
+ import * as fs from 'fs';
192
+
193
+ const template = CdkDiffIamTemplateGenerator.generateTemplate({
194
+ roleName: 'CdkChangesetRole',
195
+ createOidcRole: true,
196
+ oidcRoleName: 'GitHubOIDCRole',
197
+ githubOidc: {
198
+ owner: 'my-org',
199
+ repositories: ['infra-repo'],
200
+ branches: ['main'],
201
+ },
202
+ });
203
+
204
+ fs.writeFileSync('cdk-diff-iam-template.yaml', template);
205
+ ```
206
+
207
+ #### With Existing OIDC Provider (Skip Creation)
208
+
209
+ If your account already has a GitHub OIDC provider but you want the template to create the roles:
210
+
211
+ ```ts
212
+ new CdkDiffIamTemplate({
213
+ project,
214
+ roleName: 'CdkChangesetRole',
215
+ createOidcRole: true,
216
+ skipOidcProviderCreation: true, // Account already has OIDC provider
217
+ githubOidc: {
218
+ owner: 'my-org',
219
+ repositories: ['*'], // All repos in org
220
+ },
221
+ });
222
+ ```
223
+
224
+ ### Deploy Task
225
+
226
+ A Projen task is added for easy deployment:
227
+
228
+ ```bash
229
+ npx projen deploy-cdkdiff-iam-template -- --parameter-overrides GitHubOIDCRoleArn=... # plus any extra AWS CLI args
230
+ ```
167
231
 
168
- Use the created role ARN as `changesetRoleToAssumeArn` in `CdkDiffStackWorkflow`.
232
+ ### CdkDiffIamTemplate Props
233
+
234
+ | Property | Type | Description |
235
+ |----------|------|-------------|
236
+ | `roleName` | `string` | Name for the changeset IAM role (required) |
237
+ | `oidcRoleArn` | `string?` | ARN of existing GitHub OIDC role. Required when `createOidcRole` is false. |
238
+ | `oidcRegion` | `string?` | Region for OIDC trust condition. Required when `createOidcRole` is false. |
239
+ | `createOidcRole` | `boolean?` | Create OIDC role within template (default: false) |
240
+ | `oidcRoleName` | `string?` | Name of OIDC role to create (default: 'GitHubOIDCRole') |
241
+ | `githubOidc` | `GitHubOidcConfig?` | GitHub OIDC config. Required when `createOidcRole` is true. |
242
+ | `skipOidcProviderCreation` | `boolean?` | Skip OIDC provider if it exists (default: false) |
243
+ | `outputPath` | `string?` | Template output path (default: 'cdk-diff-workflow-iam-template.yaml') |
244
+
245
+ ### What the Template Creates
246
+
247
+ **External OIDC Role mode:**
248
+ - Parameter `GitHubOIDCRoleArn` — ARN of your existing GitHub OIDC role
249
+ - IAM role `CdkChangesetRole` with minimal permissions for change set operations
250
+ - Outputs: `CdkChangesetRoleArn`, `CdkChangesetRoleName`
251
+
252
+ **Self-Contained mode (`createOidcRole: true`):**
253
+ - GitHub OIDC Provider (unless `skipOidcProviderCreation: true`)
254
+ - IAM role `GitHubOIDCRole` with trust policy for GitHub Actions
255
+ - IAM role `CdkChangesetRole` with minimal permissions (trusts the OIDC role)
256
+ - Outputs: `GitHubOIDCProviderArn`, `GitHubOIDCRoleArn`, `GitHubOIDCRoleName`, `CdkChangesetRoleArn`, `CdkChangesetRoleName`
257
+
258
+ **Changeset Role Permissions:**
259
+ - CloudFormation Change Set operations
260
+ - Access to CDK bootstrap S3 buckets and SSM parameters
261
+ - `iam:PassRole` to `cloudformation.amazonaws.com`
262
+
263
+ Use the created changeset role ARN as `changesetRoleToAssumeArn` in `CdkDiffStackWorkflow`.
169
264
 
170
265
  ---
171
266
 
@@ -1,13 +1,44 @@
1
+ import { GitHubOidcConfig } from './CdkDiffIamTemplateStackSet';
1
2
  /**
2
3
  * Props for generating CDK Diff IAM templates (no Projen dependency)
3
4
  */
4
5
  export interface CdkDiffIamTemplateGeneratorProps {
5
- /** Name for the IAM role */
6
+ /** Name for the changeset IAM role */
6
7
  readonly roleName: string;
7
- /** ARN of the existing GitHub OIDC role that can assume this changeset role */
8
- readonly oidcRoleArn: string;
9
- /** Region for the OIDC trust condition */
10
- readonly oidcRegion: string;
8
+ /**
9
+ * ARN of the existing GitHub OIDC role that can assume this changeset role.
10
+ * Required when createOidcRole is false or undefined.
11
+ */
12
+ readonly oidcRoleArn?: string;
13
+ /**
14
+ * Region for the OIDC trust condition.
15
+ * Only used when oidcRoleArn is provided (external OIDC role).
16
+ */
17
+ readonly oidcRegion?: string;
18
+ /**
19
+ * Create a GitHub OIDC role within this template instead of using an existing one.
20
+ * When true, githubOidc configuration is required and oidcRoleArn is ignored.
21
+ * Default: false
22
+ */
23
+ readonly createOidcRole?: boolean;
24
+ /**
25
+ * Name of the GitHub OIDC role to create.
26
+ * Only used when createOidcRole is true.
27
+ * Default: 'GitHubOIDCRole'
28
+ */
29
+ readonly oidcRoleName?: string;
30
+ /**
31
+ * GitHub OIDC configuration for repo/branch restrictions.
32
+ * Required when createOidcRole is true.
33
+ */
34
+ readonly githubOidc?: GitHubOidcConfig;
35
+ /**
36
+ * Skip creating the OIDC provider (use existing one).
37
+ * Set to true if the account already has a GitHub OIDC provider.
38
+ * Only used when createOidcRole is true.
39
+ * Default: false
40
+ */
41
+ readonly skipOidcProviderCreation?: boolean;
11
42
  }
12
43
  /**
13
44
  * Pure generator class for CDK Diff IAM templates.
@@ -22,6 +53,21 @@ export declare class CdkDiffIamTemplateGenerator {
22
53
  * Generate the AWS CLI deploy command for the IAM template.
23
54
  */
24
55
  static generateDeployCommand(templatePath?: string): string;
56
+ /**
57
+ * Generate template that creates OIDC provider and role (self-contained)
58
+ */
59
+ private static generateTemplateWithOidcRole;
60
+ /**
61
+ * Generate template that uses an external OIDC role (original behavior)
62
+ */
63
+ private static generateTemplateWithExternalOidcRole;
64
+ private static generateOidcProviderLines;
65
+ private static generateOidcRoleLines;
66
+ private static buildSubjectClaims;
67
+ private static generateChangesetRoleWithOidcRef;
68
+ private static generateOidcProviderOutputLines;
69
+ private static generateOidcRoleOutputLines;
70
+ private static generateChangesetOutputLines;
25
71
  }
26
72
  /**
27
73
  * Props for the Projen-integrated CDK Diff IAM template construct
@@ -13,6 +13,69 @@ class CdkDiffIamTemplateGenerator {
13
13
  * Generate the CloudFormation IAM template as a YAML string.
14
14
  */
15
15
  static generateTemplate(props) {
16
+ const createOidcRole = props.createOidcRole ?? false;
17
+ // Validate props
18
+ if (createOidcRole) {
19
+ if (!props.githubOidc) {
20
+ throw new Error('githubOidc configuration is required when createOidcRole is true');
21
+ }
22
+ }
23
+ else {
24
+ if (!props.oidcRoleArn) {
25
+ throw new Error('oidcRoleArn is required when createOidcRole is false');
26
+ }
27
+ if (!props.oidcRegion) {
28
+ throw new Error('oidcRegion is required when createOidcRole is false');
29
+ }
30
+ }
31
+ if (createOidcRole) {
32
+ return this.generateTemplateWithOidcRole(props);
33
+ }
34
+ else {
35
+ return this.generateTemplateWithExternalOidcRole(props);
36
+ }
37
+ }
38
+ /**
39
+ * Generate the AWS CLI deploy command for the IAM template.
40
+ */
41
+ static generateDeployCommand(templatePath = 'cdk-diff-workflow-iam-template.yaml') {
42
+ return `aws cloudformation deploy --template-file ${templatePath} --stack-name cdk-diff-workflow-iam-role --capabilities CAPABILITY_NAMED_IAM`;
43
+ }
44
+ /**
45
+ * Generate template that creates OIDC provider and role (self-contained)
46
+ */
47
+ static generateTemplateWithOidcRole(props) {
48
+ const oidcRoleName = props.oidcRoleName ?? 'GitHubOIDCRole';
49
+ const skipOidcProvider = props.skipOidcProviderCreation ?? false;
50
+ const githubOidc = props.githubOidc;
51
+ const lines = [
52
+ "AWSTemplateFormatVersion: '2010-09-09'",
53
+ "Description: 'GitHub OIDC and IAM roles for CDK Diff Stack Workflow construct'",
54
+ '',
55
+ 'Resources:',
56
+ ];
57
+ // OIDC Provider (only if not skipping)
58
+ if (!skipOidcProvider) {
59
+ lines.push(...this.generateOidcProviderLines());
60
+ }
61
+ // OIDC Role
62
+ lines.push(...this.generateOidcRoleLines(oidcRoleName, githubOidc, skipOidcProvider));
63
+ // Changeset Role (trusts the created OIDC role)
64
+ lines.push(...this.generateChangesetRoleWithOidcRef(props.roleName));
65
+ // Outputs
66
+ lines.push('');
67
+ lines.push('Outputs:');
68
+ if (!skipOidcProvider) {
69
+ lines.push(...this.generateOidcProviderOutputLines());
70
+ }
71
+ lines.push(...this.generateOidcRoleOutputLines());
72
+ lines.push(...this.generateChangesetOutputLines());
73
+ return lines.join('\n');
74
+ }
75
+ /**
76
+ * Generate template that uses an external OIDC role (original behavior)
77
+ */
78
+ static generateTemplateWithExternalOidcRole(props) {
16
79
  const lines = [
17
80
  "AWSTemplateFormatVersion: '2010-09-09'",
18
81
  "Description: 'IAM role for CDK Diff Stack Workflow construct'",
@@ -94,16 +157,184 @@ class CdkDiffIamTemplateGenerator {
94
157
  ];
95
158
  return lines.join('\n');
96
159
  }
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`;
160
+ static generateOidcProviderLines() {
161
+ return [
162
+ ' # GitHub OIDC Provider',
163
+ ' GitHubOIDCProvider:',
164
+ ' Type: AWS::IAM::OIDCProvider',
165
+ ' Properties:',
166
+ ' Url: https://token.actions.githubusercontent.com',
167
+ ' ClientIdList:',
168
+ ' - sts.amazonaws.com',
169
+ ' ThumbprintList:',
170
+ ' - 6938fd4d98bab03faadb97b34396831e3780aea1',
171
+ ' - 1c58a3a8518e8759bf075b76b750d4f2df264fcd',
172
+ '',
173
+ ];
174
+ }
175
+ static generateOidcRoleLines(roleName, githubOidc, skipOidcProvider = false) {
176
+ const subjectClaims = this.buildSubjectClaims(githubOidc);
177
+ const lines = [
178
+ ' # GitHub OIDC Role - authenticates GitHub Actions workflows',
179
+ ' GitHubOIDCRole:',
180
+ ' Type: AWS::IAM::Role',
181
+ ];
182
+ // Only add DependsOn if we're creating the provider
183
+ if (!skipOidcProvider) {
184
+ lines.push(' DependsOn: GitHubOIDCProvider');
185
+ }
186
+ lines.push(' Properties:', " RoleName: '" + roleName + "'", ' AssumeRolePolicyDocument:', " Version: '2012-10-17'", ' Statement:', ' - Effect: Allow', ' Principal:', " Federated: !Sub 'arn:aws:iam::${AWS::AccountId}:oidc-provider/token.actions.githubusercontent.com'", ' Action: sts:AssumeRoleWithWebIdentity', ' Condition:', ' StringEquals:', " 'token.actions.githubusercontent.com:aud': 'sts.amazonaws.com'", ' StringLike:', " 'token.actions.githubusercontent.com:sub':");
187
+ // Add subject claims
188
+ for (const claim of subjectClaims) {
189
+ lines.push(` - '${claim}'`);
190
+ }
191
+ lines.push('');
192
+ return lines;
193
+ }
194
+ static buildSubjectClaims(githubOidc) {
195
+ const claims = [];
196
+ const branches = githubOidc.branches ?? ['*'];
197
+ for (const repo of githubOidc.repositories) {
198
+ if (repo === '*') {
199
+ // Wildcard repo - allow all repos with branch restrictions
200
+ for (const branch of branches) {
201
+ if (branch === '*') {
202
+ claims.push(`repo:${githubOidc.owner}/*`);
203
+ }
204
+ else {
205
+ claims.push(`repo:${githubOidc.owner}/*:ref:refs/heads/${branch}`);
206
+ }
207
+ }
208
+ }
209
+ else {
210
+ // Specific repo
211
+ for (const branch of branches) {
212
+ if (branch === '*') {
213
+ claims.push(`repo:${githubOidc.owner}/${repo}:*`);
214
+ }
215
+ else {
216
+ claims.push(`repo:${githubOidc.owner}/${repo}:ref:refs/heads/${branch}`);
217
+ }
218
+ }
219
+ }
220
+ }
221
+ // Add any additional claims
222
+ if (githubOidc.additionalClaims) {
223
+ for (const claim of githubOidc.additionalClaims) {
224
+ for (const repo of githubOidc.repositories) {
225
+ if (repo === '*') {
226
+ claims.push(`repo:${githubOidc.owner}/*:${claim}`);
227
+ }
228
+ else {
229
+ claims.push(`repo:${githubOidc.owner}/${repo}:${claim}`);
230
+ }
231
+ }
232
+ }
233
+ }
234
+ return claims;
235
+ }
236
+ static generateChangesetRoleWithOidcRef(roleName) {
237
+ return [
238
+ ' # CloudFormation ChangeSet Role - minimal permissions for changeset operations',
239
+ ' CdkChangesetRole:',
240
+ ' Type: AWS::IAM::Role',
241
+ ' DependsOn: GitHubOIDCRole',
242
+ ' Properties:',
243
+ " RoleName: '" + roleName + "'",
244
+ ' AssumeRolePolicyDocument:',
245
+ " Version: '2012-10-17'",
246
+ ' Statement:',
247
+ ' - Effect: Allow',
248
+ ' Principal:',
249
+ ' AWS: !GetAtt GitHubOIDCRole.Arn',
250
+ ' Action: sts:AssumeRole',
251
+ ' Policies:',
252
+ ' - PolicyName: CloudFormationChangeSetAccess',
253
+ ' PolicyDocument:',
254
+ " Version: '2012-10-17'",
255
+ ' Statement:',
256
+ ' # CloudFormation changeset operations',
257
+ ' - Effect: Allow',
258
+ ' Action:',
259
+ ' - cloudformation:CreateChangeSet',
260
+ ' - cloudformation:DescribeChangeSet',
261
+ ' - cloudformation:DeleteChangeSet',
262
+ ' - cloudformation:ListChangeSets',
263
+ ' - cloudformation:DescribeStacks',
264
+ " Resource: '*'",
265
+ ' # CDK bootstrap bucket access (for changeset creation)',
266
+ ' - Effect: Allow',
267
+ ' Action:',
268
+ ' - s3:GetObject',
269
+ ' - s3:PutObject',
270
+ ' - s3:DeleteObject',
271
+ ' - s3:ListBucket',
272
+ ' Resource:',
273
+ " - !Sub 'arn:aws:s3:::cdk-${AWS::AccountId}-${AWS::Region}-*'",
274
+ " - !Sub 'arn:aws:s3:::cdk-${AWS::AccountId}-${AWS::Region}-*/*'",
275
+ ' # CDK bootstrap parameter access',
276
+ ' - Effect: Allow',
277
+ ' Action:',
278
+ ' - ssm:GetParameter',
279
+ ' - ssm:GetParameters',
280
+ ' - ssm:GetParametersByPath',
281
+ " Resource: !Sub 'arn:aws:ssm:${AWS::Region}:${AWS::AccountId}:parameter/cdk-bootstrap/*'",
282
+ ' # IAM PassRole for CDK operations',
283
+ ' - Effect: Allow',
284
+ ' Action:',
285
+ ' - iam:PassRole',
286
+ " Resource: '*'",
287
+ ' Condition:',
288
+ ' StringEquals:',
289
+ " 'iam:PassedToService': 'cloudformation.amazonaws.com'",
290
+ '',
291
+ ];
292
+ }
293
+ static generateOidcProviderOutputLines() {
294
+ return [
295
+ ' GitHubOIDCProviderArn:',
296
+ " Description: 'ARN of the GitHub OIDC provider'",
297
+ ' Value: !GetAtt GitHubOIDCProvider.Arn',
298
+ ' Export:',
299
+ " Name: !Sub '${AWS::StackName}-GitHubOIDCProviderArn'",
300
+ '',
301
+ ];
302
+ }
303
+ static generateOidcRoleOutputLines() {
304
+ return [
305
+ ' GitHubOIDCRoleArn:',
306
+ " Description: 'ARN of the GitHub OIDC role'",
307
+ ' Value: !GetAtt GitHubOIDCRole.Arn',
308
+ ' Export:',
309
+ " Name: !Sub '${AWS::StackName}-GitHubOIDCRoleArn'",
310
+ '',
311
+ ' GitHubOIDCRoleName:',
312
+ " Description: 'Name of the GitHub OIDC role'",
313
+ ' Value: !Ref GitHubOIDCRole',
314
+ ' Export:',
315
+ " Name: !Sub '${AWS::StackName}-GitHubOIDCRoleName'",
316
+ '',
317
+ ];
318
+ }
319
+ static generateChangesetOutputLines() {
320
+ return [
321
+ ' CdkChangesetRoleArn:',
322
+ " Description: 'ARN of the CDK changeset role'",
323
+ ' Value: !GetAtt CdkChangesetRole.Arn',
324
+ ' Export:',
325
+ " Name: !Sub '${AWS::StackName}-CdkChangesetRoleArn'",
326
+ '',
327
+ ' CdkChangesetRoleName:',
328
+ " Description: 'Name of the CDK changeset role'",
329
+ ' Value: !Ref CdkChangesetRole',
330
+ ' Export:',
331
+ " Name: !Sub '${AWS::StackName}-CdkChangesetRoleName'",
332
+ ];
102
333
  }
103
334
  }
104
335
  exports.CdkDiffIamTemplateGenerator = CdkDiffIamTemplateGenerator;
105
336
  _a = JSII_RTTI_SYMBOL_1;
106
- CdkDiffIamTemplateGenerator[_a] = { fqn: "@jjrawlins/cdk-diff-pr-github-action.CdkDiffIamTemplateGenerator", version: "1.1.6" };
337
+ CdkDiffIamTemplateGenerator[_a] = { fqn: "@jjrawlins/cdk-diff-pr-github-action.CdkDiffIamTemplateGenerator", version: "1.2.0" };
107
338
  /**
108
339
  * Projen construct that emits a CloudFormation template with minimal IAM permissions
109
340
  * for the CDK Diff Stack Workflow.
@@ -126,5 +357,5 @@ class CdkDiffIamTemplate {
126
357
  }
127
358
  exports.CdkDiffIamTemplate = CdkDiffIamTemplate;
128
359
  _b = JSII_RTTI_SYMBOL_1;
129
- CdkDiffIamTemplate[_b] = { fqn: "@jjrawlins/cdk-diff-pr-github-action.CdkDiffIamTemplate", version: "1.1.6" };
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"]}
360
+ CdkDiffIamTemplate[_b] = { fqn: "@jjrawlins/cdk-diff-pr-github-action.CdkDiffIamTemplate", version: "1.2.0" };
361
+ //# 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\n    lines.push(...this.generateOidcRoleLines(oidcRoleName, githubOidc, skipOidcProvider));\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      \"                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  ): 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    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      \"                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"]}