@jjrawlins/cdk-deploy-pr-github-action 0.0.7 → 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 CHANGED
@@ -3499,7 +3499,7 @@
3499
3499
  },
3500
3500
  "name": "@jjrawlins/cdk-deploy-pr-github-action",
3501
3501
  "readme": {
3502
- "markdown": "# replace this"
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",
@@ -4197,6 +4197,6 @@
4197
4197
  "symbolId": "src/types:DeployStageOptions"
4198
4198
  }
4199
4199
  },
4200
- "version": "0.0.7",
4201
- "fingerprint": "JorqNri7czyLlkvZH3NnaTWu7jyIpJTK+Tpalky7+0Y="
4200
+ "version": "0.0.8",
4201
+ "fingerprint": "egbFOWI4RgOQ2mJGGcPSVGdJswsyJ/utU8J8LbVZTt4="
4202
4202
  }
package/README.md CHANGED
@@ -1 +1,164 @@
1
- # replace this
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.6.tgz
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.6", tarball)
27
+ _jsii_.Load("@jjrawlins/cdk-deploy-pr-github-action", "0.0.7", tarball)
28
28
  }
@@ -1 +1 @@
1
- 0.0.6
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.7" };
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"]}
@@ -304,5 +304,5 @@ class CdkDeployPipeline {
304
304
  }
305
305
  exports.CdkDeployPipeline = CdkDeployPipeline;
306
306
  _a = JSII_RTTI_SYMBOL_1;
307
- CdkDeployPipeline[_a] = { fqn: "@jjrawlins/cdk-deploy-pr-github-action.CdkDeployPipeline", version: "0.0.7" };
307
+ CdkDeployPipeline[_a] = { fqn: "@jjrawlins/cdk-deploy-pr-github-action.CdkDeployPipeline", version: "0.0.8" };
308
308
  //# 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_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"]}
package/package.json CHANGED
@@ -91,7 +91,7 @@
91
91
  "publishConfig": {
92
92
  "access": "public"
93
93
  },
94
- "version": "0.0.7",
94
+ "version": "0.0.8",
95
95
  "jest": {
96
96
  "coverageProvider": "v8",
97
97
  "testMatch": [