@jjrawlins/cdk-diff-pr-github-action 1.1.5 → 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/.jsii +94 -18
- package/API.md +148 -12
- package/README.md +115 -20
- package/lib/CdkDiffIamTemplate.d.ts +51 -5
- package/lib/CdkDiffIamTemplate.js +239 -8
- package/lib/CdkDiffIamTemplateStackSet.js +2 -2
- package/lib/CdkDiffStackWorkflow.js +1 -1
- package/lib/CdkDriftDetectionWorkflow.js +1 -1
- package/lib/CdkDriftIamTemplate.js +2 -2
- package/node_modules/@aws-sdk/client-cloudformation/package.json +6 -6
- package/node_modules/@aws-sdk/client-sso/package.json +5 -5
- package/node_modules/@aws-sdk/core/package.json +2 -2
- package/node_modules/@aws-sdk/credential-provider-env/package.json +2 -2
- package/node_modules/@aws-sdk/credential-provider-http/package.json +2 -2
- package/node_modules/@aws-sdk/credential-provider-ini/package.json +9 -9
- package/node_modules/@aws-sdk/credential-provider-login/package.json +3 -3
- package/node_modules/@aws-sdk/credential-provider-node/package.json +7 -7
- package/node_modules/@aws-sdk/credential-provider-process/package.json +2 -2
- package/node_modules/@aws-sdk/credential-provider-sso/package.json +4 -4
- package/node_modules/@aws-sdk/credential-provider-web-identity/package.json +3 -3
- package/node_modules/@aws-sdk/middleware-user-agent/package.json +3 -3
- package/node_modules/@aws-sdk/nested-clients/package.json +5 -5
- package/node_modules/@aws-sdk/token-providers/package.json +3 -3
- package/node_modules/@aws-sdk/util-endpoints/package.json +1 -1
- package/node_modules/@aws-sdk/util-user-agent-node/package.json +2 -2
- package/node_modules/@aws-sdk/xml-builder/package.json +2 -2
- package/package.json +3 -3
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/LICENSE +0 -201
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/README.md +0 -6
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-cjs/index.js +0 -415
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-cjs/lib/aws/partitions.json +0 -267
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-es/aws.js +0 -10
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-es/index.js +0 -6
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-es/lib/aws/index.js +0 -3
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-es/lib/aws/isVirtualHostableS3Bucket.js +0 -25
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-es/lib/aws/parseArn.js +0 -18
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-es/lib/aws/partition.js +0 -41
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-es/lib/aws/partitions.json +0 -267
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-es/lib/isIpAddress.js +0 -1
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-es/resolveDefaultAwsRegionalEndpointsConfig.js +0 -21
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-es/resolveEndpoint.js +0 -1
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-es/types/EndpointError.js +0 -1
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-es/types/EndpointRuleObject.js +0 -1
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-es/types/ErrorRuleObject.js +0 -1
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-es/types/RuleSetObject.js +0 -1
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-es/types/TreeRuleObject.js +0 -1
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-es/types/index.js +0 -6
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-es/types/shared.js +0 -1
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-types/aws.d.ts +0 -2
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-types/index.d.ts +0 -6
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-types/lib/aws/index.d.ts +0 -3
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-types/lib/aws/isVirtualHostableS3Bucket.d.ts +0 -5
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-types/lib/aws/parseArn.d.ts +0 -7
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-types/lib/aws/partition.d.ts +0 -38
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-types/lib/isIpAddress.d.ts +0 -1
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-types/resolveDefaultAwsRegionalEndpointsConfig.d.ts +0 -56
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-types/resolveEndpoint.d.ts +0 -1
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-types/ts3.4/aws.d.ts +0 -2
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-types/ts3.4/index.d.ts +0 -6
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-types/ts3.4/lib/aws/index.d.ts +0 -3
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-types/ts3.4/lib/aws/isVirtualHostableS3Bucket.d.ts +0 -4
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-types/ts3.4/lib/aws/parseArn.d.ts +0 -2
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-types/ts3.4/lib/aws/partition.d.ts +0 -28
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-types/ts3.4/lib/isIpAddress.d.ts +0 -1
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-types/ts3.4/resolveDefaultAwsRegionalEndpointsConfig.d.ts +0 -35
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-types/ts3.4/resolveEndpoint.d.ts +0 -1
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-types/ts3.4/types/EndpointError.d.ts +0 -1
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-types/ts3.4/types/EndpointRuleObject.d.ts +0 -6
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-types/ts3.4/types/ErrorRuleObject.d.ts +0 -1
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-types/ts3.4/types/RuleSetObject.d.ts +0 -5
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-types/ts3.4/types/TreeRuleObject.d.ts +0 -1
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-types/ts3.4/types/index.d.ts +0 -6
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-types/ts3.4/types/shared.d.ts +0 -12
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-types/types/EndpointError.d.ts +0 -1
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-types/types/EndpointRuleObject.d.ts +0 -1
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-types/types/ErrorRuleObject.d.ts +0 -1
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-types/types/RuleSetObject.d.ts +0 -1
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-types/types/TreeRuleObject.d.ts +0 -1
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-types/types/index.d.ts +0 -6
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/dist-types/types/shared.d.ts +0 -1
- package/node_modules/@aws-sdk/client-cloudformation/node_modules/@aws-sdk/util-endpoints/package.json +0 -60
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
|
|
110
|
+
Emit an IAM template you can deploy in your account for the Change Set workflow. Supports two modes:
|
|
111
111
|
|
|
112
|
-
|
|
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
|
-
|
|
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
|
-
|
|
155
|
-
|
|
156
|
-
|
|
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
|
-
|
|
187
|
+
#### Without Projen (Standalone Generator)
|
|
160
188
|
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
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
|
-
|
|
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
|
-
/**
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
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
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
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.
|
|
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.
|
|
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"]}
|