@jjrawlins/cdk-deploy-pr-github-action 0.0.6 → 0.0.8
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 +13 -13
- package/README.md +164 -1
- package/cdkdeployprgithubaction/jsii/jsii.go +2 -2
- package/cdkdeployprgithubaction/version +1 -1
- package/lib/CdkDeployDispatchWorkflow.js +1 -1
- package/lib/CdkDeployPipeline.js +18 -11
- package/package.json +1 -1
package/.jsii
CHANGED
|
@@ -3499,7 +3499,7 @@
|
|
|
3499
3499
|
},
|
|
3500
3500
|
"name": "@jjrawlins/cdk-deploy-pr-github-action",
|
|
3501
3501
|
"readme": {
|
|
3502
|
-
"markdown": "#
|
|
3502
|
+
"markdown": "# @jjrawlins/cdk-deploy-pr-github-action\n\nA [projen](https://projen.io/) construct that generates GitHub Actions workflows for AWS CDK deployments.\n\n## Features\n\n- **Automated deploy pipeline** (`deploy.yml`) — synth, publish assets, and deploy on push to main\n- **Manual dispatch workflow** (`deploy-dispatch.yml`) — deploy any previously published version to any environment\n- **Versioned cloud assemblies** — publishes `cdk.out` to GitHub Packages with auto-incrementing versions\n- **GitHub Releases** — each deploy creates a git tag and GitHub Release tied to the assembly version\n- **Multi-stage deployments** — parallel or sequential stages with `dependsOn` ordering\n- **GitHub Environments** — per-stage environment protection rules, secrets scoping, and deployment approvals\n- **Manual approval stages** — exclude stages from auto-deploy, only deployable via dispatch\n- **Per-stage IAM role overrides** — cross-account deployments with per-stage OIDC roles\n- **Concurrency groups** — prevents concurrent deployments to the same stage\n- **Rollback support** — redeploy any previous assembly version via the dispatch workflow\n\n## Installation\n\n```bash\nnpm install @jjrawlins/cdk-deploy-pr-github-action\n```\n\nAlso available on [PyPI](https://pypi.org/project/jjrawlins-cdk-deploy-pr-github-action/), [NuGet](https://www.nuget.org/packages/JJRawlins.CdkDeployPrGithubAction), and [Go](https://pkg.go.dev/github.com/JaysonRawlins/cdk-deploy-pr-github-action).\n\n## Usage\n\nIn your `.projenrc.ts`:\n\n```typescript\nimport { CdkDeployPipeline } from '@jjrawlins/cdk-deploy-pr-github-action';\n\nnew CdkDeployPipeline(project, {\n stackPrefix: 'MyApp',\n pkgNamespace: '@my-org',\n iamRoleArn: 'arn:aws:iam::111111111111:role/GitHubActionsOidcRole',\n iamRoleRegion: 'us-east-1',\n stages: [\n {\n name: 'dev',\n env: { account: '222222222222', region: 'us-east-1' },\n environment: 'development',\n },\n {\n name: 'staging',\n env: { account: '333333333333', region: 'us-east-1' },\n environment: 'staging',\n dependsOn: ['dev'],\n },\n {\n name: 'prod',\n env: { account: '444444444444', region: 'us-east-1' },\n environment: 'production',\n dependsOn: ['staging'],\n manualApproval: true, // Only deployable via dispatch\n },\n ],\n});\n```\n\nRun `npx projen` to generate the workflow files.\n\n## Generated Workflows\n\n### `deploy.yml`\n\nTriggered on push to main. Jobs:\n\n1. **synth** — checks out code, installs dependencies, runs `cdk synth`, uploads cloud assembly artifact\n2. **publish-assets** — publishes Lambda/container assets to AWS, versions and publishes the cloud assembly to GitHub Packages, creates a git tag and GitHub Release\n3. **deploy-{stage}** — one job per non-manual stage, deploys stacks using the cloud assembly artifact\n\n### `deploy-dispatch.yml`\n\nTriggered manually via `workflow_dispatch`. Inputs:\n\n- **environment** — target environment (dropdown of configured stages)\n- **version** — assembly version to deploy (e.g., `0.0.4`)\n\nDownloads the specified assembly version from GitHub Packages and deploys it. This enables:\n\n- Deploying to `manualApproval` stages (e.g., production)\n- Rolling back to any previous version\n- Ad-hoc deployments outside the normal CI/CD flow\n\n## Options\n\n### `CdkDeployPipelineOptions`\n\n| Option | Type | Default | Description |\n|--------|------|---------|-------------|\n| `pkgNamespace` | `string` | *required* | npm scope for assembly packages (e.g., `@my-org`) |\n| `stackPrefix` | `string` | *required* | Stack name prefix (e.g., `MyApp` -> `MyApp-dev`) |\n| `iamRoleArn` | `string` | *required* | Default OIDC role ARN for AWS credential assumption |\n| `stages` | `DeployStageOptions[]` | *required* | Deployment stages |\n| `iamRoleRegion` | `string` | `us-east-1` | Default AWS region for OIDC credential assumption |\n| `nodeVersion` | `string` | `24.x` | Node.js version for workflow runners |\n| `cdkCommand` | `string` | `npx cdk` | CDK CLI command prefix |\n| `installCommand` | `string` | `yarn install --check-files --frozen-lockfile` | Package install command |\n| `manualDeployment` | `boolean` | `true` | Generate the dispatch workflow |\n| `useGithubPackagesForAssembly` | `boolean` | `true` | Publish cloud assembly to GitHub Packages |\n| `branchName` | `string` | `main` | Branch that triggers deployments |\n\n### `DeployStageOptions`\n\n| Option | Type | Default | Description |\n|--------|------|---------|-------------|\n| `name` | `string` | *required* | Stage name (used in job IDs and stack names) |\n| `env` | `{ account, region }` | *required* | AWS target environment |\n| `environment` | `string` | stage name | GitHub Environment name |\n| `iamRoleArn` | `string` | pipeline default | Override OIDC role for this stage |\n| `iamRoleRegion` | `string` | pipeline default | Override AWS region for this stage |\n| `dependsOn` | `string[]` | `[]` | Stages that must complete first |\n| `stacks` | `string[]` | `['{stackPrefix}-{name}']` | CDK stack names to deploy |\n| `manualApproval` | `boolean` | `false` | Exclude from auto-deploy, dispatch only |\n\n## Examples\n\n### Parallel stages\n\nStages without `dependsOn` run in parallel after asset publishing:\n\n```typescript\nstages: [\n { name: 'us-east-1', env: usEast1Env },\n { name: 'eu-west-1', env: euWest1Env },\n]\n```\n\n### Cross-account with per-stage roles\n\n```typescript\nstages: [\n {\n name: 'dev',\n env: { account: '222222222222', region: 'us-east-1' },\n iamRoleArn: 'arn:aws:iam::222222222222:role/DevDeployRole',\n },\n {\n name: 'prod',\n env: { account: '333333333333', region: 'us-east-1' },\n iamRoleArn: 'arn:aws:iam::333333333333:role/ProdDeployRole',\n manualApproval: true,\n },\n]\n```\n\n### Rolling back\n\nEach deploy creates a GitHub Release with the assembly version. To roll back:\n\n```bash\ngh workflow run deploy-dispatch.yml -f environment=production -f version=0.0.3\n```\n\n## Prerequisites\n\n- AWS OIDC identity provider configured for GitHub Actions\n- IAM role with trust policy for GitHub Actions OIDC\n- GitHub Environments configured for deployment protection rules (optional)\n\n## License\n\nApache-2.0\n"
|
|
3503
3503
|
},
|
|
3504
3504
|
"repository": {
|
|
3505
3505
|
"type": "git",
|
|
@@ -3878,7 +3878,7 @@
|
|
|
3878
3878
|
"kind": "interface",
|
|
3879
3879
|
"locationInModule": {
|
|
3880
3880
|
"filename": "src/CdkDeployPipeline.ts",
|
|
3881
|
-
"line":
|
|
3881
|
+
"line": 354
|
|
3882
3882
|
},
|
|
3883
3883
|
"name": "DeployDispatchInternalOptions",
|
|
3884
3884
|
"properties": [
|
|
@@ -3890,7 +3890,7 @@
|
|
|
3890
3890
|
"immutable": true,
|
|
3891
3891
|
"locationInModule": {
|
|
3892
3892
|
"filename": "src/CdkDeployPipeline.ts",
|
|
3893
|
-
"line":
|
|
3893
|
+
"line": 357
|
|
3894
3894
|
},
|
|
3895
3895
|
"name": "appName",
|
|
3896
3896
|
"type": {
|
|
@@ -3905,7 +3905,7 @@
|
|
|
3905
3905
|
"immutable": true,
|
|
3906
3906
|
"locationInModule": {
|
|
3907
3907
|
"filename": "src/CdkDeployPipeline.ts",
|
|
3908
|
-
"line":
|
|
3908
|
+
"line": 362
|
|
3909
3909
|
},
|
|
3910
3910
|
"name": "cdkCommand",
|
|
3911
3911
|
"type": {
|
|
@@ -3920,7 +3920,7 @@
|
|
|
3920
3920
|
"immutable": true,
|
|
3921
3921
|
"locationInModule": {
|
|
3922
3922
|
"filename": "src/CdkDeployPipeline.ts",
|
|
3923
|
-
"line":
|
|
3923
|
+
"line": 359
|
|
3924
3924
|
},
|
|
3925
3925
|
"name": "iamRoleArn",
|
|
3926
3926
|
"type": {
|
|
@@ -3935,7 +3935,7 @@
|
|
|
3935
3935
|
"immutable": true,
|
|
3936
3936
|
"locationInModule": {
|
|
3937
3937
|
"filename": "src/CdkDeployPipeline.ts",
|
|
3938
|
-
"line":
|
|
3938
|
+
"line": 360
|
|
3939
3939
|
},
|
|
3940
3940
|
"name": "iamRoleRegion",
|
|
3941
3941
|
"type": {
|
|
@@ -3950,7 +3950,7 @@
|
|
|
3950
3950
|
"immutable": true,
|
|
3951
3951
|
"locationInModule": {
|
|
3952
3952
|
"filename": "src/CdkDeployPipeline.ts",
|
|
3953
|
-
"line":
|
|
3953
|
+
"line": 363
|
|
3954
3954
|
},
|
|
3955
3955
|
"name": "installCommand",
|
|
3956
3956
|
"type": {
|
|
@@ -3965,7 +3965,7 @@
|
|
|
3965
3965
|
"immutable": true,
|
|
3966
3966
|
"locationInModule": {
|
|
3967
3967
|
"filename": "src/CdkDeployPipeline.ts",
|
|
3968
|
-
"line":
|
|
3968
|
+
"line": 361
|
|
3969
3969
|
},
|
|
3970
3970
|
"name": "nodeVersion",
|
|
3971
3971
|
"type": {
|
|
@@ -3980,7 +3980,7 @@
|
|
|
3980
3980
|
"immutable": true,
|
|
3981
3981
|
"locationInModule": {
|
|
3982
3982
|
"filename": "src/CdkDeployPipeline.ts",
|
|
3983
|
-
"line":
|
|
3983
|
+
"line": 356
|
|
3984
3984
|
},
|
|
3985
3985
|
"name": "pkgNamespace",
|
|
3986
3986
|
"type": {
|
|
@@ -3995,7 +3995,7 @@
|
|
|
3995
3995
|
"immutable": true,
|
|
3996
3996
|
"locationInModule": {
|
|
3997
3997
|
"filename": "src/CdkDeployPipeline.ts",
|
|
3998
|
-
"line":
|
|
3998
|
+
"line": 358
|
|
3999
3999
|
},
|
|
4000
4000
|
"name": "stackPrefix",
|
|
4001
4001
|
"type": {
|
|
@@ -4010,7 +4010,7 @@
|
|
|
4010
4010
|
"immutable": true,
|
|
4011
4011
|
"locationInModule": {
|
|
4012
4012
|
"filename": "src/CdkDeployPipeline.ts",
|
|
4013
|
-
"line":
|
|
4013
|
+
"line": 355
|
|
4014
4014
|
},
|
|
4015
4015
|
"name": "stages",
|
|
4016
4016
|
"type": {
|
|
@@ -4197,6 +4197,6 @@
|
|
|
4197
4197
|
"symbolId": "src/types:DeployStageOptions"
|
|
4198
4198
|
}
|
|
4199
4199
|
},
|
|
4200
|
-
"version": "0.0.
|
|
4201
|
-
"fingerprint": "
|
|
4200
|
+
"version": "0.0.8",
|
|
4201
|
+
"fingerprint": "egbFOWI4RgOQ2mJGGcPSVGdJswsyJ/utU8J8LbVZTt4="
|
|
4202
4202
|
}
|
package/README.md
CHANGED
|
@@ -1 +1,164 @@
|
|
|
1
|
-
#
|
|
1
|
+
# @jjrawlins/cdk-deploy-pr-github-action
|
|
2
|
+
|
|
3
|
+
A [projen](https://projen.io/) construct that generates GitHub Actions workflows for AWS CDK deployments.
|
|
4
|
+
|
|
5
|
+
## Features
|
|
6
|
+
|
|
7
|
+
- **Automated deploy pipeline** (`deploy.yml`) — synth, publish assets, and deploy on push to main
|
|
8
|
+
- **Manual dispatch workflow** (`deploy-dispatch.yml`) — deploy any previously published version to any environment
|
|
9
|
+
- **Versioned cloud assemblies** — publishes `cdk.out` to GitHub Packages with auto-incrementing versions
|
|
10
|
+
- **GitHub Releases** — each deploy creates a git tag and GitHub Release tied to the assembly version
|
|
11
|
+
- **Multi-stage deployments** — parallel or sequential stages with `dependsOn` ordering
|
|
12
|
+
- **GitHub Environments** — per-stage environment protection rules, secrets scoping, and deployment approvals
|
|
13
|
+
- **Manual approval stages** — exclude stages from auto-deploy, only deployable via dispatch
|
|
14
|
+
- **Per-stage IAM role overrides** — cross-account deployments with per-stage OIDC roles
|
|
15
|
+
- **Concurrency groups** — prevents concurrent deployments to the same stage
|
|
16
|
+
- **Rollback support** — redeploy any previous assembly version via the dispatch workflow
|
|
17
|
+
|
|
18
|
+
## Installation
|
|
19
|
+
|
|
20
|
+
```bash
|
|
21
|
+
npm install @jjrawlins/cdk-deploy-pr-github-action
|
|
22
|
+
```
|
|
23
|
+
|
|
24
|
+
Also available on [PyPI](https://pypi.org/project/jjrawlins-cdk-deploy-pr-github-action/), [NuGet](https://www.nuget.org/packages/JJRawlins.CdkDeployPrGithubAction), and [Go](https://pkg.go.dev/github.com/JaysonRawlins/cdk-deploy-pr-github-action).
|
|
25
|
+
|
|
26
|
+
## Usage
|
|
27
|
+
|
|
28
|
+
In your `.projenrc.ts`:
|
|
29
|
+
|
|
30
|
+
```typescript
|
|
31
|
+
import { CdkDeployPipeline } from '@jjrawlins/cdk-deploy-pr-github-action';
|
|
32
|
+
|
|
33
|
+
new CdkDeployPipeline(project, {
|
|
34
|
+
stackPrefix: 'MyApp',
|
|
35
|
+
pkgNamespace: '@my-org',
|
|
36
|
+
iamRoleArn: 'arn:aws:iam::111111111111:role/GitHubActionsOidcRole',
|
|
37
|
+
iamRoleRegion: 'us-east-1',
|
|
38
|
+
stages: [
|
|
39
|
+
{
|
|
40
|
+
name: 'dev',
|
|
41
|
+
env: { account: '222222222222', region: 'us-east-1' },
|
|
42
|
+
environment: 'development',
|
|
43
|
+
},
|
|
44
|
+
{
|
|
45
|
+
name: 'staging',
|
|
46
|
+
env: { account: '333333333333', region: 'us-east-1' },
|
|
47
|
+
environment: 'staging',
|
|
48
|
+
dependsOn: ['dev'],
|
|
49
|
+
},
|
|
50
|
+
{
|
|
51
|
+
name: 'prod',
|
|
52
|
+
env: { account: '444444444444', region: 'us-east-1' },
|
|
53
|
+
environment: 'production',
|
|
54
|
+
dependsOn: ['staging'],
|
|
55
|
+
manualApproval: true, // Only deployable via dispatch
|
|
56
|
+
},
|
|
57
|
+
],
|
|
58
|
+
});
|
|
59
|
+
```
|
|
60
|
+
|
|
61
|
+
Run `npx projen` to generate the workflow files.
|
|
62
|
+
|
|
63
|
+
## Generated Workflows
|
|
64
|
+
|
|
65
|
+
### `deploy.yml`
|
|
66
|
+
|
|
67
|
+
Triggered on push to main. Jobs:
|
|
68
|
+
|
|
69
|
+
1. **synth** — checks out code, installs dependencies, runs `cdk synth`, uploads cloud assembly artifact
|
|
70
|
+
2. **publish-assets** — publishes Lambda/container assets to AWS, versions and publishes the cloud assembly to GitHub Packages, creates a git tag and GitHub Release
|
|
71
|
+
3. **deploy-{stage}** — one job per non-manual stage, deploys stacks using the cloud assembly artifact
|
|
72
|
+
|
|
73
|
+
### `deploy-dispatch.yml`
|
|
74
|
+
|
|
75
|
+
Triggered manually via `workflow_dispatch`. Inputs:
|
|
76
|
+
|
|
77
|
+
- **environment** — target environment (dropdown of configured stages)
|
|
78
|
+
- **version** — assembly version to deploy (e.g., `0.0.4`)
|
|
79
|
+
|
|
80
|
+
Downloads the specified assembly version from GitHub Packages and deploys it. This enables:
|
|
81
|
+
|
|
82
|
+
- Deploying to `manualApproval` stages (e.g., production)
|
|
83
|
+
- Rolling back to any previous version
|
|
84
|
+
- Ad-hoc deployments outside the normal CI/CD flow
|
|
85
|
+
|
|
86
|
+
## Options
|
|
87
|
+
|
|
88
|
+
### `CdkDeployPipelineOptions`
|
|
89
|
+
|
|
90
|
+
| Option | Type | Default | Description |
|
|
91
|
+
|--------|------|---------|-------------|
|
|
92
|
+
| `pkgNamespace` | `string` | *required* | npm scope for assembly packages (e.g., `@my-org`) |
|
|
93
|
+
| `stackPrefix` | `string` | *required* | Stack name prefix (e.g., `MyApp` -> `MyApp-dev`) |
|
|
94
|
+
| `iamRoleArn` | `string` | *required* | Default OIDC role ARN for AWS credential assumption |
|
|
95
|
+
| `stages` | `DeployStageOptions[]` | *required* | Deployment stages |
|
|
96
|
+
| `iamRoleRegion` | `string` | `us-east-1` | Default AWS region for OIDC credential assumption |
|
|
97
|
+
| `nodeVersion` | `string` | `24.x` | Node.js version for workflow runners |
|
|
98
|
+
| `cdkCommand` | `string` | `npx cdk` | CDK CLI command prefix |
|
|
99
|
+
| `installCommand` | `string` | `yarn install --check-files --frozen-lockfile` | Package install command |
|
|
100
|
+
| `manualDeployment` | `boolean` | `true` | Generate the dispatch workflow |
|
|
101
|
+
| `useGithubPackagesForAssembly` | `boolean` | `true` | Publish cloud assembly to GitHub Packages |
|
|
102
|
+
| `branchName` | `string` | `main` | Branch that triggers deployments |
|
|
103
|
+
|
|
104
|
+
### `DeployStageOptions`
|
|
105
|
+
|
|
106
|
+
| Option | Type | Default | Description |
|
|
107
|
+
|--------|------|---------|-------------|
|
|
108
|
+
| `name` | `string` | *required* | Stage name (used in job IDs and stack names) |
|
|
109
|
+
| `env` | `{ account, region }` | *required* | AWS target environment |
|
|
110
|
+
| `environment` | `string` | stage name | GitHub Environment name |
|
|
111
|
+
| `iamRoleArn` | `string` | pipeline default | Override OIDC role for this stage |
|
|
112
|
+
| `iamRoleRegion` | `string` | pipeline default | Override AWS region for this stage |
|
|
113
|
+
| `dependsOn` | `string[]` | `[]` | Stages that must complete first |
|
|
114
|
+
| `stacks` | `string[]` | `['{stackPrefix}-{name}']` | CDK stack names to deploy |
|
|
115
|
+
| `manualApproval` | `boolean` | `false` | Exclude from auto-deploy, dispatch only |
|
|
116
|
+
|
|
117
|
+
## Examples
|
|
118
|
+
|
|
119
|
+
### Parallel stages
|
|
120
|
+
|
|
121
|
+
Stages without `dependsOn` run in parallel after asset publishing:
|
|
122
|
+
|
|
123
|
+
```typescript
|
|
124
|
+
stages: [
|
|
125
|
+
{ name: 'us-east-1', env: usEast1Env },
|
|
126
|
+
{ name: 'eu-west-1', env: euWest1Env },
|
|
127
|
+
]
|
|
128
|
+
```
|
|
129
|
+
|
|
130
|
+
### Cross-account with per-stage roles
|
|
131
|
+
|
|
132
|
+
```typescript
|
|
133
|
+
stages: [
|
|
134
|
+
{
|
|
135
|
+
name: 'dev',
|
|
136
|
+
env: { account: '222222222222', region: 'us-east-1' },
|
|
137
|
+
iamRoleArn: 'arn:aws:iam::222222222222:role/DevDeployRole',
|
|
138
|
+
},
|
|
139
|
+
{
|
|
140
|
+
name: 'prod',
|
|
141
|
+
env: { account: '333333333333', region: 'us-east-1' },
|
|
142
|
+
iamRoleArn: 'arn:aws:iam::333333333333:role/ProdDeployRole',
|
|
143
|
+
manualApproval: true,
|
|
144
|
+
},
|
|
145
|
+
]
|
|
146
|
+
```
|
|
147
|
+
|
|
148
|
+
### Rolling back
|
|
149
|
+
|
|
150
|
+
Each deploy creates a GitHub Release with the assembly version. To roll back:
|
|
151
|
+
|
|
152
|
+
```bash
|
|
153
|
+
gh workflow run deploy-dispatch.yml -f environment=production -f version=0.0.3
|
|
154
|
+
```
|
|
155
|
+
|
|
156
|
+
## Prerequisites
|
|
157
|
+
|
|
158
|
+
- AWS OIDC identity provider configured for GitHub Actions
|
|
159
|
+
- IAM role with trust policy for GitHub Actions OIDC
|
|
160
|
+
- GitHub Environments configured for deployment protection rules (optional)
|
|
161
|
+
|
|
162
|
+
## License
|
|
163
|
+
|
|
164
|
+
Apache-2.0
|
|
@@ -13,7 +13,7 @@ import (
|
|
|
13
13
|
constructs "github.com/aws/constructs-go/constructs/v10/jsii"
|
|
14
14
|
)
|
|
15
15
|
|
|
16
|
-
//go:embed jjrawlins-cdk-deploy-pr-github-action-0.0.
|
|
16
|
+
//go:embed jjrawlins-cdk-deploy-pr-github-action-0.0.7.tgz
|
|
17
17
|
var tarball []byte
|
|
18
18
|
|
|
19
19
|
// Initialize loads the necessary packages in the @jsii/kernel to support the enclosing module.
|
|
@@ -24,5 +24,5 @@ func Initialize() {
|
|
|
24
24
|
constructs.Initialize()
|
|
25
25
|
|
|
26
26
|
// Load this library into the kernel
|
|
27
|
-
_jsii_.Load("@jjrawlins/cdk-deploy-pr-github-action", "0.0.
|
|
27
|
+
_jsii_.Load("@jjrawlins/cdk-deploy-pr-github-action", "0.0.7", tarball)
|
|
28
28
|
}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
0.0.
|
|
1
|
+
0.0.7
|
|
@@ -113,5 +113,5 @@ class CdkDeployDispatchWorkflow {
|
|
|
113
113
|
}
|
|
114
114
|
exports.CdkDeployDispatchWorkflow = CdkDeployDispatchWorkflow;
|
|
115
115
|
_a = JSII_RTTI_SYMBOL_1;
|
|
116
|
-
CdkDeployDispatchWorkflow[_a] = { fqn: "@jjrawlins/cdk-deploy-pr-github-action.CdkDeployDispatchWorkflow", version: "0.0.
|
|
116
|
+
CdkDeployDispatchWorkflow[_a] = { fqn: "@jjrawlins/cdk-deploy-pr-github-action.CdkDeployDispatchWorkflow", version: "0.0.8" };
|
|
117
117
|
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CdkDeployDispatchWorkflow.js","sourceRoot":"","sources":["../src/CdkDeployDispatchWorkflow.ts"],"names":[],"mappings":"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","sourcesContent":["import { GitHub, GithubWorkflow } from 'projen/lib/github';\nimport { JobPermission } from 'projen/lib/github/workflows-model';\nimport { DeployDispatchInternalOptions } from './CdkDeployPipeline';\nimport { toKebabCase } from './utils';\n\nconst CHECKOUT_VERSION = 'v5';\nconst SETUP_NODE_VERSION = 'v5';\nconst AWS_CREDENTIALS_VERSION = 'v5';\n\n/**\n * Generates a workflow_dispatch workflow for manual CDK deployments and rollbacks.\n *\n * Allows deploying any previously published version of the cloud assembly\n * to any configured environment. This enables:\n * - Manual promotion between environments\n * - Rollback to any previous version\n * - Ad-hoc deployments outside the normal CI/CD flow\n */\nexport class CdkDeployDispatchWorkflow {\n  constructor(project: any, options: DeployDispatchInternalOptions) {\n    const {\n      stages,\n      pkgNamespace,\n      appName,\n      stackPrefix,\n      iamRoleArn,\n      iamRoleRegion,\n      nodeVersion,\n      cdkCommand,\n      installCommand,\n    } = options;\n\n    const gh = project.github ?? new GitHub(project);\n    const workflow = new GithubWorkflow(gh, 'deploy-dispatch', {\n      fileName: 'deploy-dispatch.yml',\n    });\n\n    // Build environment choices from stages\n    const environmentChoices = stages.map(\n      (s) => s.environment ?? s.name,\n    );\n\n    workflow.on({\n      workflowDispatch: {\n        inputs: {\n          environment: {\n            description: 'Target environment to deploy to',\n            required: true,\n            type: 'choice',\n            options: environmentChoices,\n          },\n          version: {\n            description: 'Assembly version to deploy (e.g., 1.3.4)',\n            required: true,\n            type: 'string',\n          },\n        },\n      },\n    });\n\n    const jobs: Record<string, any> = {};\n\n    for (const stage of stages) {\n      const jobId = `deploy-${toKebabCase(stage.name)}`;\n      const stacks = stage.stacks ?? [`${stackPrefix}-${stage.name}`];\n      const githubEnv = stage.environment ?? stage.name;\n      const roleArn = stage.iamRoleArn ?? iamRoleArn;\n      const roleRegion =\n        stage.iamRoleRegion ?? iamRoleRegion ?? stage.env.region;\n\n      const deployCommand = stacks\n        .map((s) => `${cdkCommand} deploy ${s} --require-approval never --app cdk.out`)\n        .join(' && ');\n\n      jobs[jobId] = {\n        name: `Deploy ${stage.name} (manual)`,\n        if: `github.event.inputs.environment == '${githubEnv}'`,\n        runsOn: ['ubuntu-latest'],\n        environment: githubEnv,\n        concurrency: {\n          'group': `deploy-${toKebabCase(stage.name)}`,\n          'cancel-in-progress': false,\n        },\n        permissions: {\n          contents: JobPermission.READ,\n          packages: JobPermission.READ,\n          idToken: JobPermission.WRITE,\n        },\n        env: { CI: 'true' },\n        steps: [\n          {\n            name: 'Checkout',\n            uses: `actions/checkout@${CHECKOUT_VERSION}`,\n          },\n          {\n            name: 'Setup Node.js',\n            uses: `actions/setup-node@${SETUP_NODE_VERSION}`,\n            with: {\n              'node-version': nodeVersion,\n              'registry-url': 'https://npm.pkg.github.com',\n            },\n          },\n          {\n            name: 'Install dependencies',\n            run: installCommand,\n            env: { NODE_AUTH_TOKEN: '${{ secrets.GITHUB_TOKEN }}', GITHUB_TOKEN: '${{ secrets.GITHUB_TOKEN }}' },\n          },\n          {\n            name: 'Download versioned assembly',\n            run: `npm pack ${pkgNamespace}/${appName}@\\${{ github.event.inputs.version }} && tar -xzf *.tgz && mv package cdk.out`,\n            env: { NODE_AUTH_TOKEN: '${{ secrets.GITHUB_TOKEN }}', GITHUB_TOKEN: '${{ secrets.GITHUB_TOKEN }}' },\n          },\n          {\n            name: 'AWS Credentials',\n            uses: `aws-actions/configure-aws-credentials@${AWS_CREDENTIALS_VERSION}`,\n            with: {\n              'role-to-assume': roleArn,\n              'role-session-name': 'GitHubAction',\n              'aws-region': roleRegion,\n            },\n          },\n          {\n            name: `Deploy ${stage.name}`,\n            run: deployCommand,\n          },\n        ],\n      };\n    }\n\n    workflow.addJobs(jobs);\n  }\n}\n"]}
|
package/lib/CdkDeployPipeline.js
CHANGED
|
@@ -171,17 +171,24 @@ class CdkDeployPipeline {
|
|
|
171
171
|
run: [
|
|
172
172
|
'cd cdk.out && npm publish --registry=https://npm.pkg.github.com',
|
|
173
173
|
'VERSION=$(node -p "require(\'./package.json\').version")',
|
|
174
|
-
'echo "
|
|
175
|
-
'echo "" >> $GITHUB_STEP_SUMMARY',
|
|
176
|
-
`echo "**Package:** \\\`${pkgNamespace}/${appName}\\\`" >> $GITHUB_STEP_SUMMARY`,
|
|
177
|
-
'echo "**Version:** \\`$VERSION\\`" >> $GITHUB_STEP_SUMMARY',
|
|
178
|
-
'echo "" >> $GITHUB_STEP_SUMMARY',
|
|
179
|
-
'echo "To deploy this version manually:" >> $GITHUB_STEP_SUMMARY',
|
|
180
|
-
'echo "\\`\\`\\`" >> $GITHUB_STEP_SUMMARY',
|
|
181
|
-
'echo "gh workflow run deploy-dispatch.yml -f environment=<env> -f version=$VERSION" >> $GITHUB_STEP_SUMMARY',
|
|
182
|
-
'echo "\\`\\`\\`" >> $GITHUB_STEP_SUMMARY',
|
|
174
|
+
'echo "ASSEMBLY_VERSION=$VERSION" >> $GITHUB_ENV',
|
|
183
175
|
].join('\n'),
|
|
184
176
|
env: { NODE_AUTH_TOKEN: '${{ secrets.GITHUB_TOKEN }}', GITHUB_TOKEN: '${{ secrets.GITHUB_TOKEN }}' },
|
|
177
|
+
}, {
|
|
178
|
+
name: 'Tag and release assembly version',
|
|
179
|
+
run: [
|
|
180
|
+
'git tag "v${ASSEMBLY_VERSION}"',
|
|
181
|
+
'git push origin "v${ASSEMBLY_VERSION}"',
|
|
182
|
+
].join('\n'),
|
|
183
|
+
}, {
|
|
184
|
+
name: 'Create GitHub Release',
|
|
185
|
+
uses: 'softprops/action-gh-release@v2',
|
|
186
|
+
with: {
|
|
187
|
+
tag_name: 'v${{ env.ASSEMBLY_VERSION }}',
|
|
188
|
+
name: 'v${{ env.ASSEMBLY_VERSION }}',
|
|
189
|
+
body: `Assembly \`${pkgNamespace}/${appName}@\${{ env.ASSEMBLY_VERSION }}\` published to GitHub Packages.\n\nDeploy manually:\n\`\`\`\ngh workflow run deploy-dispatch.yml -f environment=<env> -f version=\${{ env.ASSEMBLY_VERSION }}\n\`\`\``,
|
|
190
|
+
generate_release_notes: true,
|
|
191
|
+
},
|
|
185
192
|
});
|
|
186
193
|
}
|
|
187
194
|
jobs['publish-assets'] = {
|
|
@@ -297,5 +304,5 @@ class CdkDeployPipeline {
|
|
|
297
304
|
}
|
|
298
305
|
exports.CdkDeployPipeline = CdkDeployPipeline;
|
|
299
306
|
_a = JSII_RTTI_SYMBOL_1;
|
|
300
|
-
CdkDeployPipeline[_a] = { fqn: "@jjrawlins/cdk-deploy-pr-github-action.CdkDeployPipeline", version: "0.0.
|
|
301
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CdkDeployPipeline.js","sourceRoot":"","sources":["../src/CdkDeployPipeline.ts"],"names":[],"mappings":"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","sourcesContent":["import { GitHub, GithubWorkflow } from 'projen/lib/github';\nimport { JobPermission } from 'projen/lib/github/workflows-model';\nimport { CdkDeployDispatchWorkflow } from './CdkDeployDispatchWorkflow';\nimport { CdkDeployPipelineOptions, DeployStageOptions } from './types';\nimport { toKebabCase, validateNoCycles } from './utils';\n\nconst CHECKOUT_VERSION = 'v5';\nconst SETUP_NODE_VERSION = 'v5';\nconst AWS_CREDENTIALS_VERSION = 'v5';\nconst UPLOAD_ARTIFACT_VERSION = 'v4';\nconst DOWNLOAD_ARTIFACT_VERSION = 'v5';\n\n/**\n * Generates GitHub Actions workflows for CDK deployments.\n *\n * Creates a `deploy.yml` workflow with:\n * - Synth job to compile and synthesize the CDK app\n * - Asset publish job to upload Lambda/container assets to AWS\n * - Per-stage deploy jobs with GitHub Environments, parallel execution, and concurrency groups\n *\n * Optionally creates a `deploy-dispatch.yml` workflow for manual deployments and rollbacks.\n */\nexport class CdkDeployPipeline {\n  constructor(project: any, options: CdkDeployPipelineOptions) {\n    const {\n      pkgNamespace,\n      stackPrefix,\n      iamRoleArn,\n      iamRoleRegion = 'us-east-1',\n      nodeVersion = '24.x',\n      cdkCommand = 'npx cdk',\n      installCommand = 'yarn install --check-files --frozen-lockfile',\n      stages,\n      manualDeployment = true,\n      useGithubPackagesForAssembly = true,\n      branchName = 'main',\n    } = options;\n\n    // Validate inputs\n    if (stages.length === 0) {\n      throw new Error('At least one deployment stage must be defined');\n    }\n\n    const stageNames = new Set(stages.map((s) => s.name));\n    for (const stage of stages) {\n      if (stage.dependsOn) {\n        for (const dep of stage.dependsOn) {\n          if (!stageNames.has(dep)) {\n            throw new Error(\n              `Stage '${stage.name}' depends on '${dep}', which is not a defined stage. Available: ${[...stageNames].join(', ')}`,\n            );\n          }\n          if (dep === stage.name) {\n            throw new Error(`Stage '${stage.name}' cannot depend on itself`);\n          }\n        }\n      }\n    }\n\n    // Validate that auto-deploy stages don't depend on manual-approval stages\n    for (const stage of stages) {\n      if (!stage.manualApproval && stage.dependsOn) {\n        for (const dep of stage.dependsOn) {\n          const depStage = stages.find((s) => s.name === dep);\n          if (depStage?.manualApproval) {\n            throw new Error(\n              `Stage '${stage.name}' depends on '${dep}', which has manualApproval enabled. Auto-deploy stages cannot depend on manual-approval stages.`,\n            );\n          }\n        }\n      }\n    }\n\n    validateNoCycles(stages);\n\n    if (useGithubPackagesForAssembly && !pkgNamespace) {\n      throw new Error(\n        'pkgNamespace is required when useGithubPackagesForAssembly is enabled',\n      );\n    }\n\n    // Create the deploy workflow\n    const gh = project.github ?? new GitHub(project);\n    const workflow = new GithubWorkflow(gh, 'deploy', {\n      fileName: 'deploy.yml',\n    });\n\n    workflow.on({\n      push: { branches: [branchName] },\n      workflowDispatch: {},\n    });\n\n    const appName = project.name ?? stackPrefix.toLowerCase();\n\n    // Build jobs object\n    const jobs: Record<string, any> = {};\n\n    // --- Synth Job ---\n    jobs.synth = {\n      name: 'Synthesize CDK application',\n      runsOn: ['ubuntu-latest'],\n      permissions: {\n        contents: JobPermission.READ,\n        packages: JobPermission.READ,\n      },\n      env: { CI: 'true' },\n      steps: [\n        {\n          name: 'Checkout',\n          uses: `actions/checkout@${CHECKOUT_VERSION}`,\n          with: { 'fetch-depth': 0 },\n        },\n        {\n          name: 'Setup Node.js',\n          uses: `actions/setup-node@${SETUP_NODE_VERSION}`,\n          with: {\n            'node-version': nodeVersion,\n            'registry-url': 'https://npm.pkg.github.com',\n          },\n        },\n        {\n          name: 'Install dependencies',\n          run: installCommand,\n          env: { NODE_AUTH_TOKEN: '${{ secrets.GITHUB_TOKEN }}', GITHUB_TOKEN: '${{ secrets.GITHUB_TOKEN }}' },\n        },\n        {\n          name: 'Synth',\n          run: `${cdkCommand} synth`,\n        },\n        {\n          name: 'Upload cloud assembly',\n          uses: `actions/upload-artifact@${UPLOAD_ARTIFACT_VERSION}`,\n          with: {\n            name: 'cloud-assembly',\n            path: 'cdk.out/',\n          },\n        },\n      ],\n    };\n\n    // --- Publish Assets Job ---\n    const publishSteps: any[] = [\n      {\n        name: 'Checkout',\n        uses: `actions/checkout@${CHECKOUT_VERSION}`,\n        with: { 'fetch-depth': 0 },\n      },\n      {\n        name: 'Setup Node.js',\n        uses: `actions/setup-node@${SETUP_NODE_VERSION}`,\n        with: {\n          'node-version': nodeVersion,\n          'registry-url': 'https://npm.pkg.github.com',\n        },\n      },\n      {\n        name: 'Install dependencies',\n        run: installCommand,\n        env: { NODE_AUTH_TOKEN: '${{ secrets.GITHUB_TOKEN }}', GITHUB_TOKEN: '${{ secrets.GITHUB_TOKEN }}' },\n      },\n      {\n        name: 'Download cloud assembly',\n        uses: `actions/download-artifact@${DOWNLOAD_ARTIFACT_VERSION}`,\n        with: { name: 'cloud-assembly', path: 'cdk.out/' },\n      },\n      {\n        name: 'AWS Credentials',\n        uses: `aws-actions/configure-aws-credentials@${AWS_CREDENTIALS_VERSION}`,\n        with: {\n          'role-to-assume': iamRoleArn,\n          'role-session-name': 'GitHubAction',\n          'aws-region': iamRoleRegion,\n        },\n      },\n      {\n        name: 'Publish assets',\n        run: 'for manifest in $(find cdk.out -name \"*-assets.json\"); do npx cdk-assets publish -p \"$manifest\"; done',\n      },\n    ];\n\n    if (useGithubPackagesForAssembly) {\n      publishSteps.push(\n        {\n          name: 'Configure git identity',\n          run: [\n            'git config user.name \"github-actions[bot]\"',\n            'git config user.email \"41898282+github-actions[bot]@users.noreply.github.com\"',\n          ].join('\\n'),\n        },\n        {\n          name: 'Create assembly package',\n          run: [\n            `PKG_NAME=\"${pkgNamespace}/${appName}\"`,\n            'LATEST=$(npm view \"$PKG_NAME\" version --registry=https://npm.pkg.github.com 2>/dev/null || echo \"0.0.0\")',\n            `echo '${JSON.stringify({ name: `${pkgNamespace}/${appName}`, publishConfig: { registry: 'https://npm.pkg.github.com' } })}' | jq --arg v \"$LATEST\" '. + {version: $v}' > cdk.out/package.json`,\n            'cd cdk.out && npm version --no-git-tag-version patch',\n          ].join('\\n'),\n          env: { NODE_AUTH_TOKEN: '${{ secrets.GITHUB_TOKEN }}', GITHUB_TOKEN: '${{ secrets.GITHUB_TOKEN }}' },\n        },\n        {\n          name: 'Publish assembly to GitHub Packages',\n          run: [\n            'cd cdk.out && npm publish --registry=https://npm.pkg.github.com',\n            'VERSION=$(node -p \"require(\\'./package.json\\').version\")',\n            'echo \"### Assembly Published\" >> $GITHUB_STEP_SUMMARY',\n            'echo \"\" >> $GITHUB_STEP_SUMMARY',\n            `echo \"**Package:** \\\\\\`${pkgNamespace}/${appName}\\\\\\`\" >> $GITHUB_STEP_SUMMARY`,\n            'echo \"**Version:** \\\\`$VERSION\\\\`\" >> $GITHUB_STEP_SUMMARY',\n            'echo \"\" >> $GITHUB_STEP_SUMMARY',\n            'echo \"To deploy this version manually:\" >> $GITHUB_STEP_SUMMARY',\n            'echo \"\\\\`\\\\`\\\\`\" >> $GITHUB_STEP_SUMMARY',\n            'echo \"gh workflow run deploy-dispatch.yml -f environment=<env> -f version=$VERSION\" >> $GITHUB_STEP_SUMMARY',\n            'echo \"\\\\`\\\\`\\\\`\" >> $GITHUB_STEP_SUMMARY',\n          ].join('\\n'),\n          env: { NODE_AUTH_TOKEN: '${{ secrets.GITHUB_TOKEN }}', GITHUB_TOKEN: '${{ secrets.GITHUB_TOKEN }}' },\n        },\n      );\n    }\n\n    jobs['publish-assets'] = {\n      name: 'Publish assets to AWS',\n      needs: ['synth'],\n      runsOn: ['ubuntu-latest'],\n      permissions: {\n        contents: JobPermission.WRITE,\n        packages: JobPermission.WRITE,\n        idToken: JobPermission.WRITE,\n      },\n      env: { CI: 'true' },\n      steps: publishSteps,\n    };\n\n    // --- Per-Stage Deploy Jobs ---\n    for (const stage of stages) {\n      if (stage.manualApproval) continue;\n      const jobId = `deploy-${toKebabCase(stage.name)}`;\n      const stacks =\n        stage.stacks ?? [`${stackPrefix}-${stage.name}`];\n      const githubEnv = stage.environment ?? stage.name;\n      const roleArn = stage.iamRoleArn ?? iamRoleArn;\n      const roleRegion =\n        stage.iamRoleRegion ?? iamRoleRegion ?? stage.env.region;\n\n      // Compute needs\n      let needs: string[];\n      if (stage.dependsOn && stage.dependsOn.length > 0) {\n        needs = stage.dependsOn.map(\n          (dep) => `deploy-${toKebabCase(dep)}`,\n        );\n      } else {\n        needs = ['publish-assets'];\n      }\n\n      const deployCommand = stacks\n        .map((s) => `${cdkCommand} deploy ${s} --require-approval never --app cdk.out`)\n        .join(' && ');\n\n      jobs[jobId] = {\n        name: `Deploy ${stage.name}`,\n        needs: needs,\n        runsOn: ['ubuntu-latest'],\n        environment: githubEnv,\n        concurrency: {\n          'group': `deploy-${toKebabCase(stage.name)}`,\n          'cancel-in-progress': false,\n        },\n        permissions: {\n          contents: JobPermission.READ,\n          packages: JobPermission.READ,\n          idToken: JobPermission.WRITE,\n        },\n        env: { CI: 'true' },\n        steps: [\n          {\n            name: 'Checkout',\n            uses: `actions/checkout@${CHECKOUT_VERSION}`,\n          },\n          {\n            name: 'Setup Node.js',\n            uses: `actions/setup-node@${SETUP_NODE_VERSION}`,\n            with: {\n              'node-version': nodeVersion,\n              'registry-url': 'https://npm.pkg.github.com',\n            },\n          },\n          {\n            name: 'Install dependencies',\n            run: installCommand,\n            env: { NODE_AUTH_TOKEN: '${{ secrets.GITHUB_TOKEN }}', GITHUB_TOKEN: '${{ secrets.GITHUB_TOKEN }}' },\n          },\n          {\n            name: 'Download cloud assembly',\n            uses: `actions/download-artifact@${DOWNLOAD_ARTIFACT_VERSION}`,\n            with: { name: 'cloud-assembly', path: 'cdk.out/' },\n          },\n          {\n            name: 'AWS Credentials',\n            uses: `aws-actions/configure-aws-credentials@${AWS_CREDENTIALS_VERSION}`,\n            with: {\n              'role-to-assume': roleArn,\n              'role-session-name': 'GitHubAction',\n              'aws-region': roleRegion,\n            },\n          },\n          {\n            name: `Deploy ${stage.name}`,\n            run: deployCommand,\n          },\n        ],\n      };\n\n      // Create projen task for local usage\n      project.addTask(`deploy:${stage.name}`, {\n        description: `Deploy to ${stage.name}`,\n        exec: stacks\n          .map((s) => `${cdkCommand} deploy ${s} --require-approval never`)\n          .join(' && '),\n      });\n    }\n\n    // Add all jobs to the workflow\n    workflow.addJobs(jobs);\n\n    // --- Dispatch Workflow ---\n    if (manualDeployment && useGithubPackagesForAssembly) {\n      new CdkDeployDispatchWorkflow(project, {\n        stages,\n        pkgNamespace,\n        appName,\n        stackPrefix,\n        iamRoleArn,\n        iamRoleRegion,\n        nodeVersion,\n        cdkCommand,\n        installCommand,\n      });\n    }\n  }\n}\n\n/**\n * Internal helper to build deploy dispatch workflow options from pipeline options.\n * Exported for use by CdkDeployDispatchWorkflow.\n */\nexport interface DeployDispatchInternalOptions {\n  readonly stages: DeployStageOptions[];\n  readonly pkgNamespace: string;\n  readonly appName: string;\n  readonly stackPrefix: string;\n  readonly iamRoleArn: string;\n  readonly iamRoleRegion: string;\n  readonly nodeVersion: string;\n  readonly cdkCommand: string;\n  readonly installCommand: string;\n}\n"]}
|
|
307
|
+
CdkDeployPipeline[_a] = { fqn: "@jjrawlins/cdk-deploy-pr-github-action.CdkDeployPipeline", version: "0.0.8" };
|
|
308
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CdkDeployPipeline.js","sourceRoot":"","sources":["../src/CdkDeployPipeline.ts"],"names":[],"mappings":";;;;;AAAA,8CAA2D;AAC3D,uEAAkE;AAClE,2EAAwE;AAExE,mCAAwD;AAExD,MAAM,gBAAgB,GAAG,IAAI,CAAC;AAC9B,MAAM,kBAAkB,GAAG,IAAI,CAAC;AAChC,MAAM,uBAAuB,GAAG,IAAI,CAAC;AACrC,MAAM,uBAAuB,GAAG,IAAI,CAAC;AACrC,MAAM,yBAAyB,GAAG,IAAI,CAAC;AAEvC;;;;;;;;;GASG;AACH,MAAa,iBAAiB;IAC5B,YAAY,OAAY,EAAE,OAAiC;QACzD,MAAM,EACJ,YAAY,EACZ,WAAW,EACX,UAAU,EACV,aAAa,GAAG,WAAW,EAC3B,WAAW,GAAG,MAAM,EACpB,UAAU,GAAG,SAAS,EACtB,cAAc,GAAG,8CAA8C,EAC/D,MAAM,EACN,gBAAgB,GAAG,IAAI,EACvB,4BAA4B,GAAG,IAAI,EACnC,UAAU,GAAG,MAAM,GACpB,GAAG,OAAO,CAAC;QAEZ,kBAAkB;QAClB,IAAI,MAAM,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACxB,MAAM,IAAI,KAAK,CAAC,+CAA+C,CAAC,CAAC;QACnE,CAAC;QAED,MAAM,UAAU,GAAG,IAAI,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC;QACtD,KAAK,MAAM,KAAK,IAAI,MAAM,EAAE,CAAC;YAC3B,IAAI,KAAK,CAAC,SAAS,EAAE,CAAC;gBACpB,KAAK,MAAM,GAAG,IAAI,KAAK,CAAC,SAAS,EAAE,CAAC;oBAClC,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC;wBACzB,MAAM,IAAI,KAAK,CACb,UAAU,KAAK,CAAC,IAAI,iBAAiB,GAAG,+CAA+C,CAAC,GAAG,UAAU,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CACpH,CAAC;oBACJ,CAAC;oBACD,IAAI,GAAG,KAAK,KAAK,CAAC,IAAI,EAAE,CAAC;wBACvB,MAAM,IAAI,KAAK,CAAC,UAAU,KAAK,CAAC,IAAI,2BAA2B,CAAC,CAAC;oBACnE,CAAC;gBACH,CAAC;YACH,CAAC;QACH,CAAC;QAED,0EAA0E;QAC1E,KAAK,MAAM,KAAK,IAAI,MAAM,EAAE,CAAC;YAC3B,IAAI,CAAC,KAAK,CAAC,cAAc,IAAI,KAAK,CAAC,SAAS,EAAE,CAAC;gBAC7C,KAAK,MAAM,GAAG,IAAI,KAAK,CAAC,SAAS,EAAE,CAAC;oBAClC,MAAM,QAAQ,GAAG,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,GAAG,CAAC,CAAC;oBACpD,IAAI,QAAQ,EAAE,cAAc,EAAE,CAAC;wBAC7B,MAAM,IAAI,KAAK,CACb,UAAU,KAAK,CAAC,IAAI,iBAAiB,GAAG,kGAAkG,CAC3I,CAAC;oBACJ,CAAC;gBACH,CAAC;YACH,CAAC;QACH,CAAC;QAED,IAAA,wBAAgB,EAAC,MAAM,CAAC,CAAC;QAEzB,IAAI,4BAA4B,IAAI,CAAC,YAAY,EAAE,CAAC;YAClD,MAAM,IAAI,KAAK,CACb,uEAAuE,CACxE,CAAC;QACJ,CAAC;QAED,6BAA6B;QAC7B,MAAM,EAAE,GAAG,OAAO,CAAC,MAAM,IAAI,IAAI,eAAM,CAAC,OAAO,CAAC,CAAC;QACjD,MAAM,QAAQ,GAAG,IAAI,uBAAc,CAAC,EAAE,EAAE,QAAQ,EAAE;YAChD,QAAQ,EAAE,YAAY;SACvB,CAAC,CAAC;QAEH,QAAQ,CAAC,EAAE,CAAC;YACV,IAAI,EAAE,EAAE,QAAQ,EAAE,CAAC,UAAU,CAAC,EAAE;YAChC,gBAAgB,EAAE,EAAE;SACrB,CAAC,CAAC;QAEH,MAAM,OAAO,GAAG,OAAO,CAAC,IAAI,IAAI,WAAW,CAAC,WAAW,EAAE,CAAC;QAE1D,oBAAoB;QACpB,MAAM,IAAI,GAAwB,EAAE,CAAC;QAErC,oBAAoB;QACpB,IAAI,CAAC,KAAK,GAAG;YACX,IAAI,EAAE,4BAA4B;YAClC,MAAM,EAAE,CAAC,eAAe,CAAC;YACzB,WAAW,EAAE;gBACX,QAAQ,EAAE,+BAAa,CAAC,IAAI;gBAC5B,QAAQ,EAAE,+BAAa,CAAC,IAAI;aAC7B;YACD,GAAG,EAAE,EAAE,EAAE,EAAE,MAAM,EAAE;YACnB,KAAK,EAAE;gBACL;oBACE,IAAI,EAAE,UAAU;oBAChB,IAAI,EAAE,oBAAoB,gBAAgB,EAAE;oBAC5C,IAAI,EAAE,EAAE,aAAa,EAAE,CAAC,EAAE;iBAC3B;gBACD;oBACE,IAAI,EAAE,eAAe;oBACrB,IAAI,EAAE,sBAAsB,kBAAkB,EAAE;oBAChD,IAAI,EAAE;wBACJ,cAAc,EAAE,WAAW;wBAC3B,cAAc,EAAE,4BAA4B;qBAC7C;iBACF;gBACD;oBACE,IAAI,EAAE,sBAAsB;oBAC5B,GAAG,EAAE,cAAc;oBACnB,GAAG,EAAE,EAAE,eAAe,EAAE,6BAA6B,EAAE,YAAY,EAAE,6BAA6B,EAAE;iBACrG;gBACD;oBACE,IAAI,EAAE,OAAO;oBACb,GAAG,EAAE,GAAG,UAAU,QAAQ;iBAC3B;gBACD;oBACE,IAAI,EAAE,uBAAuB;oBAC7B,IAAI,EAAE,2BAA2B,uBAAuB,EAAE;oBAC1D,IAAI,EAAE;wBACJ,IAAI,EAAE,gBAAgB;wBACtB,IAAI,EAAE,UAAU;qBACjB;iBACF;aACF;SACF,CAAC;QAEF,6BAA6B;QAC7B,MAAM,YAAY,GAAU;YAC1B;gBACE,IAAI,EAAE,UAAU;gBAChB,IAAI,EAAE,oBAAoB,gBAAgB,EAAE;gBAC5C,IAAI,EAAE,EAAE,aAAa,EAAE,CAAC,EAAE;aAC3B;YACD;gBACE,IAAI,EAAE,eAAe;gBACrB,IAAI,EAAE,sBAAsB,kBAAkB,EAAE;gBAChD,IAAI,EAAE;oBACJ,cAAc,EAAE,WAAW;oBAC3B,cAAc,EAAE,4BAA4B;iBAC7C;aACF;YACD;gBACE,IAAI,EAAE,sBAAsB;gBAC5B,GAAG,EAAE,cAAc;gBACnB,GAAG,EAAE,EAAE,eAAe,EAAE,6BAA6B,EAAE,YAAY,EAAE,6BAA6B,EAAE;aACrG;YACD;gBACE,IAAI,EAAE,yBAAyB;gBAC/B,IAAI,EAAE,6BAA6B,yBAAyB,EAAE;gBAC9D,IAAI,EAAE,EAAE,IAAI,EAAE,gBAAgB,EAAE,IAAI,EAAE,UAAU,EAAE;aACnD;YACD;gBACE,IAAI,EAAE,iBAAiB;gBACvB,IAAI,EAAE,yCAAyC,uBAAuB,EAAE;gBACxE,IAAI,EAAE;oBACJ,gBAAgB,EAAE,UAAU;oBAC5B,mBAAmB,EAAE,cAAc;oBACnC,YAAY,EAAE,aAAa;iBAC5B;aACF;YACD;gBACE,IAAI,EAAE,gBAAgB;gBACtB,GAAG,EAAE,uGAAuG;aAC7G;SACF,CAAC;QAEF,IAAI,4BAA4B,EAAE,CAAC;YACjC,YAAY,CAAC,IAAI,CACf;gBACE,IAAI,EAAE,wBAAwB;gBAC9B,GAAG,EAAE;oBACH,4CAA4C;oBAC5C,+EAA+E;iBAChF,CAAC,IAAI,CAAC,IAAI,CAAC;aACb,EACD;gBACE,IAAI,EAAE,yBAAyB;gBAC/B,GAAG,EAAE;oBACH,aAAa,YAAY,IAAI,OAAO,GAAG;oBACvC,0GAA0G;oBAC1G,SAAS,IAAI,CAAC,SAAS,CAAC,EAAE,IAAI,EAAE,GAAG,YAAY,IAAI,OAAO,EAAE,EAAE,aAAa,EAAE,EAAE,QAAQ,EAAE,4BAA4B,EAAE,EAAE,CAAC,qEAAqE;oBAC/L,sDAAsD;iBACvD,CAAC,IAAI,CAAC,IAAI,CAAC;gBACZ,GAAG,EAAE,EAAE,eAAe,EAAE,6BAA6B,EAAE,YAAY,EAAE,6BAA6B,EAAE;aACrG,EACD;gBACE,IAAI,EAAE,qCAAqC;gBAC3C,GAAG,EAAE;oBACH,iEAAiE;oBACjE,0DAA0D;oBAC1D,iDAAiD;iBAClD,CAAC,IAAI,CAAC,IAAI,CAAC;gBACZ,GAAG,EAAE,EAAE,eAAe,EAAE,6BAA6B,EAAE,YAAY,EAAE,6BAA6B,EAAE;aACrG,EACD;gBACE,IAAI,EAAE,kCAAkC;gBACxC,GAAG,EAAE;oBACH,gCAAgC;oBAChC,wCAAwC;iBACzC,CAAC,IAAI,CAAC,IAAI,CAAC;aACb,EACD;gBACE,IAAI,EAAE,uBAAuB;gBAC7B,IAAI,EAAE,gCAAgC;gBACtC,IAAI,EAAE;oBACJ,QAAQ,EAAE,8BAA8B;oBACxC,IAAI,EAAE,8BAA8B;oBACpC,IAAI,EAAE,cAAc,YAAY,IAAI,OAAO,qMAAqM;oBAChP,sBAAsB,EAAE,IAAI;iBAC7B;aACF,CACF,CAAC;QACJ,CAAC;QAED,IAAI,CAAC,gBAAgB,CAAC,GAAG;YACvB,IAAI,EAAE,uBAAuB;YAC7B,KAAK,EAAE,CAAC,OAAO,CAAC;YAChB,MAAM,EAAE,CAAC,eAAe,CAAC;YACzB,WAAW,EAAE;gBACX,QAAQ,EAAE,+BAAa,CAAC,KAAK;gBAC7B,QAAQ,EAAE,+BAAa,CAAC,KAAK;gBAC7B,OAAO,EAAE,+BAAa,CAAC,KAAK;aAC7B;YACD,GAAG,EAAE,EAAE,EAAE,EAAE,MAAM,EAAE;YACnB,KAAK,EAAE,YAAY;SACpB,CAAC;QAEF,gCAAgC;QAChC,KAAK,MAAM,KAAK,IAAI,MAAM,EAAE,CAAC;YAC3B,IAAI,KAAK,CAAC,cAAc;gBAAE,SAAS;YACnC,MAAM,KAAK,GAAG,UAAU,IAAA,mBAAW,EAAC,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC;YAClD,MAAM,MAAM,GACV,KAAK,CAAC,MAAM,IAAI,CAAC,GAAG,WAAW,IAAI,KAAK,CAAC,IAAI,EAAE,CAAC,CAAC;YACnD,MAAM,SAAS,GAAG,KAAK,CAAC,WAAW,IAAI,KAAK,CAAC,IAAI,CAAC;YAClD,MAAM,OAAO,GAAG,KAAK,CAAC,UAAU,IAAI,UAAU,CAAC;YAC/C,MAAM,UAAU,GACd,KAAK,CAAC,aAAa,IAAI,aAAa,IAAI,KAAK,CAAC,GAAG,CAAC,MAAM,CAAC;YAE3D,gBAAgB;YAChB,IAAI,KAAe,CAAC;YACpB,IAAI,KAAK,CAAC,SAAS,IAAI,KAAK,CAAC,SAAS,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBAClD,KAAK,GAAG,KAAK,CAAC,SAAS,CAAC,GAAG,CACzB,CAAC,GAAG,EAAE,EAAE,CAAC,UAAU,IAAA,mBAAW,EAAC,GAAG,CAAC,EAAE,CACtC,CAAC;YACJ,CAAC;iBAAM,CAAC;gBACN,KAAK,GAAG,CAAC,gBAAgB,CAAC,CAAC;YAC7B,CAAC;YAED,MAAM,aAAa,GAAG,MAAM;iBACzB,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,GAAG,UAAU,WAAW,CAAC,yCAAyC,CAAC;iBAC9E,IAAI,CAAC,MAAM,CAAC,CAAC;YAEhB,IAAI,CAAC,KAAK,CAAC,GAAG;gBACZ,IAAI,EAAE,UAAU,KAAK,CAAC,IAAI,EAAE;gBAC5B,KAAK,EAAE,KAAK;gBACZ,MAAM,EAAE,CAAC,eAAe,CAAC;gBACzB,WAAW,EAAE,SAAS;gBACtB,WAAW,EAAE;oBACX,OAAO,EAAE,UAAU,IAAA,mBAAW,EAAC,KAAK,CAAC,IAAI,CAAC,EAAE;oBAC5C,oBAAoB,EAAE,KAAK;iBAC5B;gBACD,WAAW,EAAE;oBACX,QAAQ,EAAE,+BAAa,CAAC,IAAI;oBAC5B,QAAQ,EAAE,+BAAa,CAAC,IAAI;oBAC5B,OAAO,EAAE,+BAAa,CAAC,KAAK;iBAC7B;gBACD,GAAG,EAAE,EAAE,EAAE,EAAE,MAAM,EAAE;gBACnB,KAAK,EAAE;oBACL;wBACE,IAAI,EAAE,UAAU;wBAChB,IAAI,EAAE,oBAAoB,gBAAgB,EAAE;qBAC7C;oBACD;wBACE,IAAI,EAAE,eAAe;wBACrB,IAAI,EAAE,sBAAsB,kBAAkB,EAAE;wBAChD,IAAI,EAAE;4BACJ,cAAc,EAAE,WAAW;4BAC3B,cAAc,EAAE,4BAA4B;yBAC7C;qBACF;oBACD;wBACE,IAAI,EAAE,sBAAsB;wBAC5B,GAAG,EAAE,cAAc;wBACnB,GAAG,EAAE,EAAE,eAAe,EAAE,6BAA6B,EAAE,YAAY,EAAE,6BAA6B,EAAE;qBACrG;oBACD;wBACE,IAAI,EAAE,yBAAyB;wBAC/B,IAAI,EAAE,6BAA6B,yBAAyB,EAAE;wBAC9D,IAAI,EAAE,EAAE,IAAI,EAAE,gBAAgB,EAAE,IAAI,EAAE,UAAU,EAAE;qBACnD;oBACD;wBACE,IAAI,EAAE,iBAAiB;wBACvB,IAAI,EAAE,yCAAyC,uBAAuB,EAAE;wBACxE,IAAI,EAAE;4BACJ,gBAAgB,EAAE,OAAO;4BACzB,mBAAmB,EAAE,cAAc;4BACnC,YAAY,EAAE,UAAU;yBACzB;qBACF;oBACD;wBACE,IAAI,EAAE,UAAU,KAAK,CAAC,IAAI,EAAE;wBAC5B,GAAG,EAAE,aAAa;qBACnB;iBACF;aACF,CAAC;YAEF,qCAAqC;YACrC,OAAO,CAAC,OAAO,CAAC,UAAU,KAAK,CAAC,IAAI,EAAE,EAAE;gBACtC,WAAW,EAAE,aAAa,KAAK,CAAC,IAAI,EAAE;gBACtC,IAAI,EAAE,MAAM;qBACT,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,GAAG,UAAU,WAAW,CAAC,2BAA2B,CAAC;qBAChE,IAAI,CAAC,MAAM,CAAC;aAChB,CAAC,CAAC;QACL,CAAC;QAED,+BAA+B;QAC/B,QAAQ,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;QAEvB,4BAA4B;QAC5B,IAAI,gBAAgB,IAAI,4BAA4B,EAAE,CAAC;YACrD,IAAI,qDAAyB,CAAC,OAAO,EAAE;gBACrC,MAAM;gBACN,YAAY;gBACZ,OAAO;gBACP,WAAW;gBACX,UAAU;gBACV,aAAa;gBACb,WAAW;gBACX,UAAU;gBACV,cAAc;aACf,CAAC,CAAC;QACL,CAAC;IACH,CAAC;;AApUH,8CAqUC","sourcesContent":["import { GitHub, GithubWorkflow } from 'projen/lib/github';\nimport { JobPermission } from 'projen/lib/github/workflows-model';\nimport { CdkDeployDispatchWorkflow } from './CdkDeployDispatchWorkflow';\nimport { CdkDeployPipelineOptions, DeployStageOptions } from './types';\nimport { toKebabCase, validateNoCycles } from './utils';\n\nconst CHECKOUT_VERSION = 'v5';\nconst SETUP_NODE_VERSION = 'v5';\nconst AWS_CREDENTIALS_VERSION = 'v5';\nconst UPLOAD_ARTIFACT_VERSION = 'v4';\nconst DOWNLOAD_ARTIFACT_VERSION = 'v5';\n\n/**\n * Generates GitHub Actions workflows for CDK deployments.\n *\n * Creates a `deploy.yml` workflow with:\n * - Synth job to compile and synthesize the CDK app\n * - Asset publish job to upload Lambda/container assets to AWS\n * - Per-stage deploy jobs with GitHub Environments, parallel execution, and concurrency groups\n *\n * Optionally creates a `deploy-dispatch.yml` workflow for manual deployments and rollbacks.\n */\nexport class CdkDeployPipeline {\n  constructor(project: any, options: CdkDeployPipelineOptions) {\n    const {\n      pkgNamespace,\n      stackPrefix,\n      iamRoleArn,\n      iamRoleRegion = 'us-east-1',\n      nodeVersion = '24.x',\n      cdkCommand = 'npx cdk',\n      installCommand = 'yarn install --check-files --frozen-lockfile',\n      stages,\n      manualDeployment = true,\n      useGithubPackagesForAssembly = true,\n      branchName = 'main',\n    } = options;\n\n    // Validate inputs\n    if (stages.length === 0) {\n      throw new Error('At least one deployment stage must be defined');\n    }\n\n    const stageNames = new Set(stages.map((s) => s.name));\n    for (const stage of stages) {\n      if (stage.dependsOn) {\n        for (const dep of stage.dependsOn) {\n          if (!stageNames.has(dep)) {\n            throw new Error(\n              `Stage '${stage.name}' depends on '${dep}', which is not a defined stage. Available: ${[...stageNames].join(', ')}`,\n            );\n          }\n          if (dep === stage.name) {\n            throw new Error(`Stage '${stage.name}' cannot depend on itself`);\n          }\n        }\n      }\n    }\n\n    // Validate that auto-deploy stages don't depend on manual-approval stages\n    for (const stage of stages) {\n      if (!stage.manualApproval && stage.dependsOn) {\n        for (const dep of stage.dependsOn) {\n          const depStage = stages.find((s) => s.name === dep);\n          if (depStage?.manualApproval) {\n            throw new Error(\n              `Stage '${stage.name}' depends on '${dep}', which has manualApproval enabled. Auto-deploy stages cannot depend on manual-approval stages.`,\n            );\n          }\n        }\n      }\n    }\n\n    validateNoCycles(stages);\n\n    if (useGithubPackagesForAssembly && !pkgNamespace) {\n      throw new Error(\n        'pkgNamespace is required when useGithubPackagesForAssembly is enabled',\n      );\n    }\n\n    // Create the deploy workflow\n    const gh = project.github ?? new GitHub(project);\n    const workflow = new GithubWorkflow(gh, 'deploy', {\n      fileName: 'deploy.yml',\n    });\n\n    workflow.on({\n      push: { branches: [branchName] },\n      workflowDispatch: {},\n    });\n\n    const appName = project.name ?? stackPrefix.toLowerCase();\n\n    // Build jobs object\n    const jobs: Record<string, any> = {};\n\n    // --- Synth Job ---\n    jobs.synth = {\n      name: 'Synthesize CDK application',\n      runsOn: ['ubuntu-latest'],\n      permissions: {\n        contents: JobPermission.READ,\n        packages: JobPermission.READ,\n      },\n      env: { CI: 'true' },\n      steps: [\n        {\n          name: 'Checkout',\n          uses: `actions/checkout@${CHECKOUT_VERSION}`,\n          with: { 'fetch-depth': 0 },\n        },\n        {\n          name: 'Setup Node.js',\n          uses: `actions/setup-node@${SETUP_NODE_VERSION}`,\n          with: {\n            'node-version': nodeVersion,\n            'registry-url': 'https://npm.pkg.github.com',\n          },\n        },\n        {\n          name: 'Install dependencies',\n          run: installCommand,\n          env: { NODE_AUTH_TOKEN: '${{ secrets.GITHUB_TOKEN }}', GITHUB_TOKEN: '${{ secrets.GITHUB_TOKEN }}' },\n        },\n        {\n          name: 'Synth',\n          run: `${cdkCommand} synth`,\n        },\n        {\n          name: 'Upload cloud assembly',\n          uses: `actions/upload-artifact@${UPLOAD_ARTIFACT_VERSION}`,\n          with: {\n            name: 'cloud-assembly',\n            path: 'cdk.out/',\n          },\n        },\n      ],\n    };\n\n    // --- Publish Assets Job ---\n    const publishSteps: any[] = [\n      {\n        name: 'Checkout',\n        uses: `actions/checkout@${CHECKOUT_VERSION}`,\n        with: { 'fetch-depth': 0 },\n      },\n      {\n        name: 'Setup Node.js',\n        uses: `actions/setup-node@${SETUP_NODE_VERSION}`,\n        with: {\n          'node-version': nodeVersion,\n          'registry-url': 'https://npm.pkg.github.com',\n        },\n      },\n      {\n        name: 'Install dependencies',\n        run: installCommand,\n        env: { NODE_AUTH_TOKEN: '${{ secrets.GITHUB_TOKEN }}', GITHUB_TOKEN: '${{ secrets.GITHUB_TOKEN }}' },\n      },\n      {\n        name: 'Download cloud assembly',\n        uses: `actions/download-artifact@${DOWNLOAD_ARTIFACT_VERSION}`,\n        with: { name: 'cloud-assembly', path: 'cdk.out/' },\n      },\n      {\n        name: 'AWS Credentials',\n        uses: `aws-actions/configure-aws-credentials@${AWS_CREDENTIALS_VERSION}`,\n        with: {\n          'role-to-assume': iamRoleArn,\n          'role-session-name': 'GitHubAction',\n          'aws-region': iamRoleRegion,\n        },\n      },\n      {\n        name: 'Publish assets',\n        run: 'for manifest in $(find cdk.out -name \"*-assets.json\"); do npx cdk-assets publish -p \"$manifest\"; done',\n      },\n    ];\n\n    if (useGithubPackagesForAssembly) {\n      publishSteps.push(\n        {\n          name: 'Configure git identity',\n          run: [\n            'git config user.name \"github-actions[bot]\"',\n            'git config user.email \"41898282+github-actions[bot]@users.noreply.github.com\"',\n          ].join('\\n'),\n        },\n        {\n          name: 'Create assembly package',\n          run: [\n            `PKG_NAME=\"${pkgNamespace}/${appName}\"`,\n            'LATEST=$(npm view \"$PKG_NAME\" version --registry=https://npm.pkg.github.com 2>/dev/null || echo \"0.0.0\")',\n            `echo '${JSON.stringify({ name: `${pkgNamespace}/${appName}`, publishConfig: { registry: 'https://npm.pkg.github.com' } })}' | jq --arg v \"$LATEST\" '. + {version: $v}' > cdk.out/package.json`,\n            'cd cdk.out && npm version --no-git-tag-version patch',\n          ].join('\\n'),\n          env: { NODE_AUTH_TOKEN: '${{ secrets.GITHUB_TOKEN }}', GITHUB_TOKEN: '${{ secrets.GITHUB_TOKEN }}' },\n        },\n        {\n          name: 'Publish assembly to GitHub Packages',\n          run: [\n            'cd cdk.out && npm publish --registry=https://npm.pkg.github.com',\n            'VERSION=$(node -p \"require(\\'./package.json\\').version\")',\n            'echo \"ASSEMBLY_VERSION=$VERSION\" >> $GITHUB_ENV',\n          ].join('\\n'),\n          env: { NODE_AUTH_TOKEN: '${{ secrets.GITHUB_TOKEN }}', GITHUB_TOKEN: '${{ secrets.GITHUB_TOKEN }}' },\n        },\n        {\n          name: 'Tag and release assembly version',\n          run: [\n            'git tag \"v${ASSEMBLY_VERSION}\"',\n            'git push origin \"v${ASSEMBLY_VERSION}\"',\n          ].join('\\n'),\n        },\n        {\n          name: 'Create GitHub Release',\n          uses: 'softprops/action-gh-release@v2',\n          with: {\n            tag_name: 'v${{ env.ASSEMBLY_VERSION }}',\n            name: 'v${{ env.ASSEMBLY_VERSION }}',\n            body: `Assembly \\`${pkgNamespace}/${appName}@\\${{ env.ASSEMBLY_VERSION }}\\` published to GitHub Packages.\\n\\nDeploy manually:\\n\\`\\`\\`\\ngh workflow run deploy-dispatch.yml -f environment=<env> -f version=\\${{ env.ASSEMBLY_VERSION }}\\n\\`\\`\\``,\n            generate_release_notes: true,\n          },\n        },\n      );\n    }\n\n    jobs['publish-assets'] = {\n      name: 'Publish assets to AWS',\n      needs: ['synth'],\n      runsOn: ['ubuntu-latest'],\n      permissions: {\n        contents: JobPermission.WRITE,\n        packages: JobPermission.WRITE,\n        idToken: JobPermission.WRITE,\n      },\n      env: { CI: 'true' },\n      steps: publishSteps,\n    };\n\n    // --- Per-Stage Deploy Jobs ---\n    for (const stage of stages) {\n      if (stage.manualApproval) continue;\n      const jobId = `deploy-${toKebabCase(stage.name)}`;\n      const stacks =\n        stage.stacks ?? [`${stackPrefix}-${stage.name}`];\n      const githubEnv = stage.environment ?? stage.name;\n      const roleArn = stage.iamRoleArn ?? iamRoleArn;\n      const roleRegion =\n        stage.iamRoleRegion ?? iamRoleRegion ?? stage.env.region;\n\n      // Compute needs\n      let needs: string[];\n      if (stage.dependsOn && stage.dependsOn.length > 0) {\n        needs = stage.dependsOn.map(\n          (dep) => `deploy-${toKebabCase(dep)}`,\n        );\n      } else {\n        needs = ['publish-assets'];\n      }\n\n      const deployCommand = stacks\n        .map((s) => `${cdkCommand} deploy ${s} --require-approval never --app cdk.out`)\n        .join(' && ');\n\n      jobs[jobId] = {\n        name: `Deploy ${stage.name}`,\n        needs: needs,\n        runsOn: ['ubuntu-latest'],\n        environment: githubEnv,\n        concurrency: {\n          'group': `deploy-${toKebabCase(stage.name)}`,\n          'cancel-in-progress': false,\n        },\n        permissions: {\n          contents: JobPermission.READ,\n          packages: JobPermission.READ,\n          idToken: JobPermission.WRITE,\n        },\n        env: { CI: 'true' },\n        steps: [\n          {\n            name: 'Checkout',\n            uses: `actions/checkout@${CHECKOUT_VERSION}`,\n          },\n          {\n            name: 'Setup Node.js',\n            uses: `actions/setup-node@${SETUP_NODE_VERSION}`,\n            with: {\n              'node-version': nodeVersion,\n              'registry-url': 'https://npm.pkg.github.com',\n            },\n          },\n          {\n            name: 'Install dependencies',\n            run: installCommand,\n            env: { NODE_AUTH_TOKEN: '${{ secrets.GITHUB_TOKEN }}', GITHUB_TOKEN: '${{ secrets.GITHUB_TOKEN }}' },\n          },\n          {\n            name: 'Download cloud assembly',\n            uses: `actions/download-artifact@${DOWNLOAD_ARTIFACT_VERSION}`,\n            with: { name: 'cloud-assembly', path: 'cdk.out/' },\n          },\n          {\n            name: 'AWS Credentials',\n            uses: `aws-actions/configure-aws-credentials@${AWS_CREDENTIALS_VERSION}`,\n            with: {\n              'role-to-assume': roleArn,\n              'role-session-name': 'GitHubAction',\n              'aws-region': roleRegion,\n            },\n          },\n          {\n            name: `Deploy ${stage.name}`,\n            run: deployCommand,\n          },\n        ],\n      };\n\n      // Create projen task for local usage\n      project.addTask(`deploy:${stage.name}`, {\n        description: `Deploy to ${stage.name}`,\n        exec: stacks\n          .map((s) => `${cdkCommand} deploy ${s} --require-approval never`)\n          .join(' && '),\n      });\n    }\n\n    // Add all jobs to the workflow\n    workflow.addJobs(jobs);\n\n    // --- Dispatch Workflow ---\n    if (manualDeployment && useGithubPackagesForAssembly) {\n      new CdkDeployDispatchWorkflow(project, {\n        stages,\n        pkgNamespace,\n        appName,\n        stackPrefix,\n        iamRoleArn,\n        iamRoleRegion,\n        nodeVersion,\n        cdkCommand,\n        installCommand,\n      });\n    }\n  }\n}\n\n/**\n * Internal helper to build deploy dispatch workflow options from pipeline options.\n * Exported for use by CdkDeployDispatchWorkflow.\n */\nexport interface DeployDispatchInternalOptions {\n  readonly stages: DeployStageOptions[];\n  readonly pkgNamespace: string;\n  readonly appName: string;\n  readonly stackPrefix: string;\n  readonly iamRoleArn: string;\n  readonly iamRoleRegion: string;\n  readonly nodeVersion: string;\n  readonly cdkCommand: string;\n  readonly installCommand: string;\n}\n"]}
|