@jjrawlins/cdk-deploy-pr-github-action 0.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,274 @@
1
+ "use strict";
2
+ var _a;
3
+ Object.defineProperty(exports, "__esModule", { value: true });
4
+ exports.CdkDeployPipeline = void 0;
5
+ const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
6
+ const github_1 = require("projen/lib/github");
7
+ const workflows_model_1 = require("projen/lib/github/workflows-model");
8
+ const CdkDeployDispatchWorkflow_1 = require("./CdkDeployDispatchWorkflow");
9
+ const utils_1 = require("./utils");
10
+ const CHECKOUT_VERSION = 'v5';
11
+ const SETUP_NODE_VERSION = 'v5';
12
+ const AWS_CREDENTIALS_VERSION = 'v5';
13
+ const UPLOAD_ARTIFACT_VERSION = 'v4';
14
+ const DOWNLOAD_ARTIFACT_VERSION = 'v5';
15
+ /**
16
+ * Generates GitHub Actions workflows for CDK deployments.
17
+ *
18
+ * Creates a `deploy.yml` workflow with:
19
+ * - Synth job to compile and synthesize the CDK app
20
+ * - Asset publish job to upload Lambda/container assets to AWS
21
+ * - Per-stage deploy jobs with GitHub Environments, parallel execution, and concurrency groups
22
+ *
23
+ * Optionally creates a `deploy-dispatch.yml` workflow for manual deployments and rollbacks.
24
+ */
25
+ class CdkDeployPipeline {
26
+ constructor(project, options) {
27
+ const { pkgNamespace, stackPrefix, iamRoleArn, iamRoleRegion = 'us-east-1', nodeVersion = '24.x', cdkCommand = 'npx cdk', installCommand = 'yarn install --check-files --frozen-lockfile', stages, manualDeployment = true, useGithubPackagesForAssembly = true, branchName = 'main', } = options;
28
+ // Validate inputs
29
+ if (stages.length === 0) {
30
+ throw new Error('At least one deployment stage must be defined');
31
+ }
32
+ const stageNames = new Set(stages.map((s) => s.name));
33
+ for (const stage of stages) {
34
+ if (stage.dependsOn) {
35
+ for (const dep of stage.dependsOn) {
36
+ if (!stageNames.has(dep)) {
37
+ throw new Error(`Stage '${stage.name}' depends on '${dep}', which is not a defined stage. Available: ${[...stageNames].join(', ')}`);
38
+ }
39
+ if (dep === stage.name) {
40
+ throw new Error(`Stage '${stage.name}' cannot depend on itself`);
41
+ }
42
+ }
43
+ }
44
+ }
45
+ (0, utils_1.validateNoCycles)(stages);
46
+ if (useGithubPackagesForAssembly && !pkgNamespace) {
47
+ throw new Error('pkgNamespace is required when useGithubPackagesForAssembly is enabled');
48
+ }
49
+ // Create the deploy workflow
50
+ const gh = project.github ?? new github_1.GitHub(project);
51
+ const workflow = new github_1.GithubWorkflow(gh, 'deploy', {
52
+ fileName: 'deploy.yml',
53
+ });
54
+ workflow.on({
55
+ push: { branches: [branchName] },
56
+ workflowDispatch: {},
57
+ });
58
+ const appName = project.name ?? stackPrefix.toLowerCase();
59
+ // Build jobs object
60
+ const jobs = {};
61
+ // --- Synth Job ---
62
+ jobs.synth = {
63
+ name: 'Synthesize CDK application',
64
+ runsOn: ['ubuntu-latest'],
65
+ permissions: {
66
+ contents: workflows_model_1.JobPermission.READ,
67
+ packages: workflows_model_1.JobPermission.READ,
68
+ },
69
+ env: { CI: 'true' },
70
+ steps: [
71
+ {
72
+ name: 'Checkout',
73
+ uses: `actions/checkout@${CHECKOUT_VERSION}`,
74
+ with: { 'fetch-depth': 0 },
75
+ },
76
+ {
77
+ name: 'Setup Node.js',
78
+ uses: `actions/setup-node@${SETUP_NODE_VERSION}`,
79
+ with: {
80
+ 'node-version': nodeVersion,
81
+ 'registry-url': 'https://npm.pkg.github.com',
82
+ },
83
+ },
84
+ {
85
+ name: 'Install dependencies',
86
+ run: installCommand,
87
+ env: { NODE_AUTH_TOKEN: '${{ secrets.GITHUB_TOKEN }}', GITHUB_TOKEN: '${{ secrets.GITHUB_TOKEN }}' },
88
+ },
89
+ {
90
+ name: 'Synth',
91
+ run: `${cdkCommand} synth`,
92
+ },
93
+ {
94
+ name: 'Upload cloud assembly',
95
+ uses: `actions/upload-artifact@${UPLOAD_ARTIFACT_VERSION}`,
96
+ with: {
97
+ name: 'cloud-assembly',
98
+ path: 'cdk.out/',
99
+ },
100
+ },
101
+ ],
102
+ };
103
+ // --- Publish Assets Job ---
104
+ const publishSteps = [
105
+ {
106
+ name: 'Checkout',
107
+ uses: `actions/checkout@${CHECKOUT_VERSION}`,
108
+ with: { 'fetch-depth': 0 },
109
+ },
110
+ {
111
+ name: 'Setup Node.js',
112
+ uses: `actions/setup-node@${SETUP_NODE_VERSION}`,
113
+ with: {
114
+ 'node-version': nodeVersion,
115
+ 'registry-url': 'https://npm.pkg.github.com',
116
+ },
117
+ },
118
+ {
119
+ name: 'Install dependencies',
120
+ run: installCommand,
121
+ env: { NODE_AUTH_TOKEN: '${{ secrets.GITHUB_TOKEN }}', GITHUB_TOKEN: '${{ secrets.GITHUB_TOKEN }}' },
122
+ },
123
+ {
124
+ name: 'Download cloud assembly',
125
+ uses: `actions/download-artifact@${DOWNLOAD_ARTIFACT_VERSION}`,
126
+ with: { name: 'cloud-assembly', path: 'cdk.out/' },
127
+ },
128
+ {
129
+ name: 'AWS Credentials',
130
+ uses: `aws-actions/configure-aws-credentials@${AWS_CREDENTIALS_VERSION}`,
131
+ with: {
132
+ 'role-to-assume': iamRoleArn,
133
+ 'role-session-name': 'GitHubAction',
134
+ 'aws-region': iamRoleRegion,
135
+ },
136
+ },
137
+ {
138
+ name: 'Publish assets',
139
+ run: 'npx cdk-assets publish -p cdk.out/**/*-assets.json',
140
+ },
141
+ ];
142
+ if (useGithubPackagesForAssembly) {
143
+ publishSteps.push({
144
+ name: 'Configure git identity',
145
+ run: [
146
+ 'git config user.name "github-actions[bot]"',
147
+ 'git config user.email "41898282+github-actions[bot]@users.noreply.github.com"',
148
+ ].join('\n'),
149
+ }, {
150
+ name: 'Create assembly package',
151
+ run: [
152
+ `echo '${JSON.stringify({ name: `${pkgNamespace}/${appName}`, version: '0.0.0' })}' > cdk.out/package.json`,
153
+ 'cd cdk.out && npm version --no-git-tag-version from-git || npm version --no-git-tag-version patch',
154
+ ].join('\n'),
155
+ env: { NODE_AUTH_TOKEN: '${{ secrets.GITHUB_TOKEN }}', GITHUB_TOKEN: '${{ secrets.GITHUB_TOKEN }}' },
156
+ }, {
157
+ name: 'Publish assembly to GitHub Packages',
158
+ run: 'cd cdk.out && npm publish',
159
+ env: { NODE_AUTH_TOKEN: '${{ secrets.GITHUB_TOKEN }}', GITHUB_TOKEN: '${{ secrets.GITHUB_TOKEN }}' },
160
+ });
161
+ }
162
+ jobs['publish-assets'] = {
163
+ name: 'Publish assets to AWS',
164
+ needs: ['synth'],
165
+ runsOn: ['ubuntu-latest'],
166
+ permissions: {
167
+ contents: workflows_model_1.JobPermission.WRITE,
168
+ packages: workflows_model_1.JobPermission.WRITE,
169
+ idToken: workflows_model_1.JobPermission.WRITE,
170
+ },
171
+ env: { CI: 'true' },
172
+ steps: publishSteps,
173
+ };
174
+ // --- Per-Stage Deploy Jobs ---
175
+ for (const stage of stages) {
176
+ const jobId = `deploy-${(0, utils_1.toKebabCase)(stage.name)}`;
177
+ const stacks = stage.stacks ?? [`${stackPrefix}-${stage.name}`];
178
+ const githubEnv = stage.environment ?? stage.name;
179
+ const roleArn = stage.iamRoleArn ?? iamRoleArn;
180
+ const roleRegion = stage.iamRoleRegion ?? iamRoleRegion ?? stage.env.region;
181
+ // Compute needs
182
+ let needs;
183
+ if (stage.dependsOn && stage.dependsOn.length > 0) {
184
+ needs = stage.dependsOn.map((dep) => `deploy-${(0, utils_1.toKebabCase)(dep)}`);
185
+ }
186
+ else {
187
+ needs = ['publish-assets'];
188
+ }
189
+ const deployCommand = stacks
190
+ .map((s) => `${cdkCommand} deploy ${s} --require-approval never --app cdk.out`)
191
+ .join(' && ');
192
+ jobs[jobId] = {
193
+ name: `Deploy ${stage.name}`,
194
+ needs: needs,
195
+ runsOn: ['ubuntu-latest'],
196
+ environment: githubEnv,
197
+ concurrency: {
198
+ 'group': `deploy-${(0, utils_1.toKebabCase)(stage.name)}`,
199
+ 'cancel-in-progress': false,
200
+ },
201
+ permissions: {
202
+ contents: workflows_model_1.JobPermission.READ,
203
+ packages: workflows_model_1.JobPermission.READ,
204
+ idToken: workflows_model_1.JobPermission.WRITE,
205
+ },
206
+ env: { CI: 'true' },
207
+ steps: [
208
+ {
209
+ name: 'Checkout',
210
+ uses: `actions/checkout@${CHECKOUT_VERSION}`,
211
+ },
212
+ {
213
+ name: 'Setup Node.js',
214
+ uses: `actions/setup-node@${SETUP_NODE_VERSION}`,
215
+ with: {
216
+ 'node-version': nodeVersion,
217
+ 'registry-url': 'https://npm.pkg.github.com',
218
+ },
219
+ },
220
+ {
221
+ name: 'Install dependencies',
222
+ run: installCommand,
223
+ env: { NODE_AUTH_TOKEN: '${{ secrets.GITHUB_TOKEN }}', GITHUB_TOKEN: '${{ secrets.GITHUB_TOKEN }}' },
224
+ },
225
+ {
226
+ name: 'Download cloud assembly',
227
+ uses: `actions/download-artifact@${DOWNLOAD_ARTIFACT_VERSION}`,
228
+ with: { name: 'cloud-assembly', path: 'cdk.out/' },
229
+ },
230
+ {
231
+ name: 'AWS Credentials',
232
+ uses: `aws-actions/configure-aws-credentials@${AWS_CREDENTIALS_VERSION}`,
233
+ with: {
234
+ 'role-to-assume': roleArn,
235
+ 'role-session-name': 'GitHubAction',
236
+ 'aws-region': roleRegion,
237
+ },
238
+ },
239
+ {
240
+ name: `Deploy ${stage.name}`,
241
+ run: deployCommand,
242
+ },
243
+ ],
244
+ };
245
+ // Create projen task for local usage
246
+ project.addTask(`deploy:${stage.name}`, {
247
+ description: `Deploy to ${stage.name}`,
248
+ exec: stacks
249
+ .map((s) => `${cdkCommand} deploy ${s} --require-approval never`)
250
+ .join(' && '),
251
+ });
252
+ }
253
+ // Add all jobs to the workflow
254
+ workflow.addJobs(jobs);
255
+ // --- Dispatch Workflow ---
256
+ if (manualDeployment && useGithubPackagesForAssembly) {
257
+ new CdkDeployDispatchWorkflow_1.CdkDeployDispatchWorkflow(project, {
258
+ stages,
259
+ pkgNamespace,
260
+ appName,
261
+ stackPrefix,
262
+ iamRoleArn,
263
+ iamRoleRegion,
264
+ nodeVersion,
265
+ cdkCommand,
266
+ installCommand,
267
+ });
268
+ }
269
+ }
270
+ }
271
+ exports.CdkDeployPipeline = CdkDeployPipeline;
272
+ _a = JSII_RTTI_SYMBOL_1;
273
+ CdkDeployPipeline[_a] = { fqn: "@jjrawlins/cdk-deploy-pr-github-action.CdkDeployPipeline", version: "0.0.0" };
274
+ //# 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    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: 'npx cdk-assets publish -p cdk.out/**/*-assets.json',\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            `echo '${JSON.stringify({ name: `${pkgNamespace}/${appName}`, version: '0.0.0' })}' > cdk.out/package.json`,\n            'cd cdk.out && npm version --no-git-tag-version from-git || 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: 'cd cdk.out && npm publish',\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      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/lib/index.d.ts ADDED
@@ -0,0 +1,4 @@
1
+ export * from './types';
2
+ export * from './CdkDeployPipeline';
3
+ export * from './CdkDeployDispatchWorkflow';
4
+ export * from './utils';
package/lib/index.js ADDED
@@ -0,0 +1,21 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __exportStar = (this && this.__exportStar) || function(m, exports) {
14
+ for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
15
+ };
16
+ Object.defineProperty(exports, "__esModule", { value: true });
17
+ __exportStar(require("./types"), exports);
18
+ __exportStar(require("./CdkDeployPipeline"), exports);
19
+ __exportStar(require("./CdkDeployDispatchWorkflow"), exports);
20
+ __exportStar(require("./utils"), exports);
21
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7OztBQUFBLDBDQUF3QjtBQUN4QixzREFBb0M7QUFDcEMsOERBQTRDO0FBQzVDLDBDQUF3QiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vdHlwZXMnO1xuZXhwb3J0ICogZnJvbSAnLi9DZGtEZXBsb3lQaXBlbGluZSc7XG5leHBvcnQgKiBmcm9tICcuL0Nka0RlcGxveURpc3BhdGNoV29ya2Zsb3cnO1xuZXhwb3J0ICogZnJvbSAnLi91dGlscyc7XG4iXX0=
package/lib/types.d.ts ADDED
@@ -0,0 +1,94 @@
1
+ /**
2
+ * AWS environment target for a deployment stage.
3
+ */
4
+ export interface AwsEnvironment {
5
+ /** AWS account ID */
6
+ readonly account: string;
7
+ /** AWS region */
8
+ readonly region: string;
9
+ }
10
+ /**
11
+ * Configuration for a single deployment stage.
12
+ */
13
+ export interface DeployStageOptions {
14
+ /** Stage name (used as suffix in job IDs and stack names, e.g., 'Sandbox', 'Production') */
15
+ readonly name: string;
16
+ /** AWS target environment (account + region) */
17
+ readonly env: AwsEnvironment;
18
+ /**
19
+ * GitHub Environment name for protection rules, secrets scoping, and deployment approvals.
20
+ * @default - uses the stage name
21
+ */
22
+ readonly environment?: string;
23
+ /**
24
+ * Override the default OIDC role ARN for this stage.
25
+ * Useful for cross-account deployments where each account has its own role.
26
+ */
27
+ readonly iamRoleArn?: string;
28
+ /**
29
+ * Override the default AWS region for OIDC credential assumption.
30
+ * @default - uses the pipeline-level iamRoleRegion or the stage env.region
31
+ */
32
+ readonly iamRoleRegion?: string;
33
+ /**
34
+ * Stage names that must complete successfully before this stage runs.
35
+ * Stages with no dependsOn run in parallel after the publish-assets job.
36
+ *
37
+ * @example ['Sandbox', 'Dev'] // waits for both before deploying
38
+ * @default - depends only on publish-assets (runs as soon as assets are ready)
39
+ */
40
+ readonly dependsOn?: string[];
41
+ /**
42
+ * Specific CDK stack names to deploy in this stage.
43
+ * @default - ['{stackPrefix}-{stageName}']
44
+ */
45
+ readonly stacks?: string[];
46
+ }
47
+ /**
48
+ * Configuration for the CDK deploy pipeline.
49
+ */
50
+ export interface CdkDeployPipelineOptions {
51
+ /** npm scope for versioned cloud assembly packages (e.g., '@defiance-digital') */
52
+ readonly pkgNamespace: string;
53
+ /** Stack name prefix. Stage names are appended with a dash (e.g., 'MyApp' -> 'MyApp-Production') */
54
+ readonly stackPrefix: string;
55
+ /** Default OIDC role ARN for AWS credential assumption */
56
+ readonly iamRoleArn: string;
57
+ /**
58
+ * Default AWS region for OIDC credential assumption.
59
+ * @default 'us-east-1'
60
+ */
61
+ readonly iamRoleRegion?: string;
62
+ /**
63
+ * Node.js version for workflow runners.
64
+ * @default '24.x'
65
+ */
66
+ readonly nodeVersion?: string;
67
+ /**
68
+ * CDK CLI command prefix.
69
+ * @default 'npx cdk'
70
+ */
71
+ readonly cdkCommand?: string;
72
+ /**
73
+ * Package install command.
74
+ * @default 'yarn install --check-files --frozen-lockfile'
75
+ */
76
+ readonly installCommand?: string;
77
+ /** Deployment stages in declaration order */
78
+ readonly stages: DeployStageOptions[];
79
+ /**
80
+ * Generate a workflow_dispatch workflow for manual deployments and rollbacks.
81
+ * @default true
82
+ */
83
+ readonly manualDeployment?: boolean;
84
+ /**
85
+ * Version and publish cloud assembly to GitHub Packages for rollback support.
86
+ * @default true
87
+ */
88
+ readonly useGithubPackagesForAssembly?: boolean;
89
+ /**
90
+ * Branch that triggers deployments on push.
91
+ * @default 'main'
92
+ */
93
+ readonly branchName?: string;
94
+ }
package/lib/types.js ADDED
@@ -0,0 +1,3 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQVdTIGVudmlyb25tZW50IHRhcmdldCBmb3IgYSBkZXBsb3ltZW50IHN0YWdlLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIEF3c0Vudmlyb25tZW50IHtcbiAgLyoqIEFXUyBhY2NvdW50IElEICovXG4gIHJlYWRvbmx5IGFjY291bnQ6IHN0cmluZztcbiAgLyoqIEFXUyByZWdpb24gKi9cbiAgcmVhZG9ubHkgcmVnaW9uOiBzdHJpbmc7XG59XG5cbi8qKlxuICogQ29uZmlndXJhdGlvbiBmb3IgYSBzaW5nbGUgZGVwbG95bWVudCBzdGFnZS5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBEZXBsb3lTdGFnZU9wdGlvbnMge1xuICAvKiogU3RhZ2UgbmFtZSAodXNlZCBhcyBzdWZmaXggaW4gam9iIElEcyBhbmQgc3RhY2sgbmFtZXMsIGUuZy4sICdTYW5kYm94JywgJ1Byb2R1Y3Rpb24nKSAqL1xuICByZWFkb25seSBuYW1lOiBzdHJpbmc7XG5cbiAgLyoqIEFXUyB0YXJnZXQgZW52aXJvbm1lbnQgKGFjY291bnQgKyByZWdpb24pICovXG4gIHJlYWRvbmx5IGVudjogQXdzRW52aXJvbm1lbnQ7XG5cbiAgLyoqXG4gICAqIEdpdEh1YiBFbnZpcm9ubWVudCBuYW1lIGZvciBwcm90ZWN0aW9uIHJ1bGVzLCBzZWNyZXRzIHNjb3BpbmcsIGFuZCBkZXBsb3ltZW50IGFwcHJvdmFscy5cbiAgICogQGRlZmF1bHQgLSB1c2VzIHRoZSBzdGFnZSBuYW1lXG4gICAqL1xuICByZWFkb25seSBlbnZpcm9ubWVudD86IHN0cmluZztcblxuICAvKipcbiAgICogT3ZlcnJpZGUgdGhlIGRlZmF1bHQgT0lEQyByb2xlIEFSTiBmb3IgdGhpcyBzdGFnZS5cbiAgICogVXNlZnVsIGZvciBjcm9zcy1hY2NvdW50IGRlcGxveW1lbnRzIHdoZXJlIGVhY2ggYWNjb3VudCBoYXMgaXRzIG93biByb2xlLlxuICAgKi9cbiAgcmVhZG9ubHkgaWFtUm9sZUFybj86IHN0cmluZztcblxuICAvKipcbiAgICogT3ZlcnJpZGUgdGhlIGRlZmF1bHQgQVdTIHJlZ2lvbiBmb3IgT0lEQyBjcmVkZW50aWFsIGFzc3VtcHRpb24uXG4gICAqIEBkZWZhdWx0IC0gdXNlcyB0aGUgcGlwZWxpbmUtbGV2ZWwgaWFtUm9sZVJlZ2lvbiBvciB0aGUgc3RhZ2UgZW52LnJlZ2lvblxuICAgKi9cbiAgcmVhZG9ubHkgaWFtUm9sZVJlZ2lvbj86IHN0cmluZztcblxuICAvKipcbiAgICogU3RhZ2UgbmFtZXMgdGhhdCBtdXN0IGNvbXBsZXRlIHN1Y2Nlc3NmdWxseSBiZWZvcmUgdGhpcyBzdGFnZSBydW5zLlxuICAgKiBTdGFnZXMgd2l0aCBubyBkZXBlbmRzT24gcnVuIGluIHBhcmFsbGVsIGFmdGVyIHRoZSBwdWJsaXNoLWFzc2V0cyBqb2IuXG4gICAqXG4gICAqIEBleGFtcGxlIFsnU2FuZGJveCcsICdEZXYnXSAvLyB3YWl0cyBmb3IgYm90aCBiZWZvcmUgZGVwbG95aW5nXG4gICAqIEBkZWZhdWx0IC0gZGVwZW5kcyBvbmx5IG9uIHB1Ymxpc2gtYXNzZXRzIChydW5zIGFzIHNvb24gYXMgYXNzZXRzIGFyZSByZWFkeSlcbiAgICovXG4gIHJlYWRvbmx5IGRlcGVuZHNPbj86IHN0cmluZ1tdO1xuXG4gIC8qKlxuICAgKiBTcGVjaWZpYyBDREsgc3RhY2sgbmFtZXMgdG8gZGVwbG95IGluIHRoaXMgc3RhZ2UuXG4gICAqIEBkZWZhdWx0IC0gWyd7c3RhY2tQcmVmaXh9LXtzdGFnZU5hbWV9J11cbiAgICovXG4gIHJlYWRvbmx5IHN0YWNrcz86IHN0cmluZ1tdO1xufVxuXG4vKipcbiAqIENvbmZpZ3VyYXRpb24gZm9yIHRoZSBDREsgZGVwbG95IHBpcGVsaW5lLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIENka0RlcGxveVBpcGVsaW5lT3B0aW9ucyB7XG4gIC8qKiBucG0gc2NvcGUgZm9yIHZlcnNpb25lZCBjbG91ZCBhc3NlbWJseSBwYWNrYWdlcyAoZS5nLiwgJ0BkZWZpYW5jZS1kaWdpdGFsJykgKi9cbiAgcmVhZG9ubHkgcGtnTmFtZXNwYWNlOiBzdHJpbmc7XG5cbiAgLyoqIFN0YWNrIG5hbWUgcHJlZml4LiBTdGFnZSBuYW1lcyBhcmUgYXBwZW5kZWQgd2l0aCBhIGRhc2ggKGUuZy4sICdNeUFwcCcgLT4gJ015QXBwLVByb2R1Y3Rpb24nKSAqL1xuICByZWFkb25seSBzdGFja1ByZWZpeDogc3RyaW5nO1xuXG4gIC8qKiBEZWZhdWx0IE9JREMgcm9sZSBBUk4gZm9yIEFXUyBjcmVkZW50aWFsIGFzc3VtcHRpb24gKi9cbiAgcmVhZG9ubHkgaWFtUm9sZUFybjogc3RyaW5nO1xuXG4gIC8qKlxuICAgKiBEZWZhdWx0IEFXUyByZWdpb24gZm9yIE9JREMgY3JlZGVudGlhbCBhc3N1bXB0aW9uLlxuICAgKiBAZGVmYXVsdCAndXMtZWFzdC0xJ1xuICAgKi9cbiAgcmVhZG9ubHkgaWFtUm9sZVJlZ2lvbj86IHN0cmluZztcblxuICAvKipcbiAgICogTm9kZS5qcyB2ZXJzaW9uIGZvciB3b3JrZmxvdyBydW5uZXJzLlxuICAgKiBAZGVmYXVsdCAnMjQueCdcbiAgICovXG4gIHJlYWRvbmx5IG5vZGVWZXJzaW9uPzogc3RyaW5nO1xuXG4gIC8qKlxuICAgKiBDREsgQ0xJIGNvbW1hbmQgcHJlZml4LlxuICAgKiBAZGVmYXVsdCAnbnB4IGNkaydcbiAgICovXG4gIHJlYWRvbmx5IGNka0NvbW1hbmQ/OiBzdHJpbmc7XG5cbiAgLyoqXG4gICAqIFBhY2thZ2UgaW5zdGFsbCBjb21tYW5kLlxuICAgKiBAZGVmYXVsdCAneWFybiBpbnN0YWxsIC0tY2hlY2stZmlsZXMgLS1mcm96ZW4tbG9ja2ZpbGUnXG4gICAqL1xuICByZWFkb25seSBpbnN0YWxsQ29tbWFuZD86IHN0cmluZztcblxuICAvKiogRGVwbG95bWVudCBzdGFnZXMgaW4gZGVjbGFyYXRpb24gb3JkZXIgKi9cbiAgcmVhZG9ubHkgc3RhZ2VzOiBEZXBsb3lTdGFnZU9wdGlvbnNbXTtcblxuICAvKipcbiAgICogR2VuZXJhdGUgYSB3b3JrZmxvd19kaXNwYXRjaCB3b3JrZmxvdyBmb3IgbWFudWFsIGRlcGxveW1lbnRzIGFuZCByb2xsYmFja3MuXG4gICAqIEBkZWZhdWx0IHRydWVcbiAgICovXG4gIHJlYWRvbmx5IG1hbnVhbERlcGxveW1lbnQ/OiBib29sZWFuO1xuXG4gIC8qKlxuICAgKiBWZXJzaW9uIGFuZCBwdWJsaXNoIGNsb3VkIGFzc2VtYmx5IHRvIEdpdEh1YiBQYWNrYWdlcyBmb3Igcm9sbGJhY2sgc3VwcG9ydC5cbiAgICogQGRlZmF1bHQgdHJ1ZVxuICAgKi9cbiAgcmVhZG9ubHkgdXNlR2l0aHViUGFja2FnZXNGb3JBc3NlbWJseT86IGJvb2xlYW47XG5cbiAgLyoqXG4gICAqIEJyYW5jaCB0aGF0IHRyaWdnZXJzIGRlcGxveW1lbnRzIG9uIHB1c2guXG4gICAqIEBkZWZhdWx0ICdtYWluJ1xuICAgKi9cbiAgcmVhZG9ubHkgYnJhbmNoTmFtZT86IHN0cmluZztcbn1cbiJdfQ==
package/lib/utils.d.ts ADDED
@@ -0,0 +1,17 @@
1
+ /**
2
+ * Convert a string to kebab-case for use in file names and job IDs.
3
+ */
4
+ export declare function toKebabCase(s: string): string;
5
+ /**
6
+ * Convert a string to a valid GitHub Actions job ID.
7
+ * Job IDs must start with a letter or underscore and contain only alphanumerics, hyphens, and underscores.
8
+ */
9
+ export declare function toGithubJobId(s: string): string;
10
+ /**
11
+ * Validate that there are no circular dependencies in the stage graph.
12
+ * @throws Error if a cycle is detected
13
+ */
14
+ export declare function validateNoCycles(stages: {
15
+ name: string;
16
+ dependsOn?: string[];
17
+ }[]): void;
package/lib/utils.js ADDED
@@ -0,0 +1,58 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.toKebabCase = toKebabCase;
4
+ exports.toGithubJobId = toGithubJobId;
5
+ exports.validateNoCycles = validateNoCycles;
6
+ /**
7
+ * Convert a string to kebab-case for use in file names and job IDs.
8
+ */
9
+ function toKebabCase(s) {
10
+ return s
11
+ .replace(/[^a-zA-Z0-9]+/g, '-')
12
+ .replace(/^-+|-+$/g, '')
13
+ .toLowerCase();
14
+ }
15
+ /**
16
+ * Convert a string to a valid GitHub Actions job ID.
17
+ * Job IDs must start with a letter or underscore and contain only alphanumerics, hyphens, and underscores.
18
+ */
19
+ function toGithubJobId(s) {
20
+ let out = s.replace(/[^A-Za-z0-9_-]+/g, '-');
21
+ out = out.replace(/-+/g, '-');
22
+ out = out.replace(/^-+|-+$/g, '');
23
+ out = out.toLowerCase();
24
+ if (!out || !/^[a-z_]/i.test(out)) {
25
+ out = `s-${out}`;
26
+ }
27
+ return out;
28
+ }
29
+ /**
30
+ * Validate that there are no circular dependencies in the stage graph.
31
+ * @throws Error if a cycle is detected
32
+ */
33
+ function validateNoCycles(stages) {
34
+ const visited = new Set();
35
+ const inStack = new Set();
36
+ const stageMap = new Map(stages.map((s) => [s.name, s]));
37
+ function visit(name, path) {
38
+ if (inStack.has(name)) {
39
+ throw new Error(`Circular dependency detected: ${[...path, name].join(' -> ')}`);
40
+ }
41
+ if (visited.has(name)) {
42
+ return;
43
+ }
44
+ inStack.add(name);
45
+ const stage = stageMap.get(name);
46
+ if (stage?.dependsOn) {
47
+ for (const dep of stage.dependsOn) {
48
+ visit(dep, [...path, name]);
49
+ }
50
+ }
51
+ inStack.delete(name);
52
+ visited.add(name);
53
+ }
54
+ for (const stage of stages) {
55
+ visit(stage.name, []);
56
+ }
57
+ }
58
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXRpbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvdXRpbHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFHQSxrQ0FLQztBQU1ELHNDQVNDO0FBTUQsNENBOEJDO0FBM0REOztHQUVHO0FBQ0gsU0FBZ0IsV0FBVyxDQUFDLENBQVM7SUFDbkMsT0FBTyxDQUFDO1NBQ0wsT0FBTyxDQUFDLGdCQUFnQixFQUFFLEdBQUcsQ0FBQztTQUM5QixPQUFPLENBQUMsVUFBVSxFQUFFLEVBQUUsQ0FBQztTQUN2QixXQUFXLEVBQUUsQ0FBQztBQUNuQixDQUFDO0FBRUQ7OztHQUdHO0FBQ0gsU0FBZ0IsYUFBYSxDQUFDLENBQVM7SUFDckMsSUFBSSxHQUFHLEdBQUcsQ0FBQyxDQUFDLE9BQU8sQ0FBQyxrQkFBa0IsRUFBRSxHQUFHLENBQUMsQ0FBQztJQUM3QyxHQUFHLEdBQUcsR0FBRyxDQUFDLE9BQU8sQ0FBQyxLQUFLLEVBQUUsR0FBRyxDQUFDLENBQUM7SUFDOUIsR0FBRyxHQUFHLEdBQUcsQ0FBQyxPQUFPLENBQUMsVUFBVSxFQUFFLEVBQUUsQ0FBQyxDQUFDO0lBQ2xDLEdBQUcsR0FBRyxHQUFHLENBQUMsV0FBVyxFQUFFLENBQUM7SUFDeEIsSUFBSSxDQUFDLEdBQUcsSUFBSSxDQUFDLFVBQVUsQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLEVBQUUsQ0FBQztRQUNsQyxHQUFHLEdBQUcsS0FBSyxHQUFHLEVBQUUsQ0FBQztJQUNuQixDQUFDO0lBQ0QsT0FBTyxHQUFHLENBQUM7QUFDYixDQUFDO0FBRUQ7OztHQUdHO0FBQ0gsU0FBZ0IsZ0JBQWdCLENBQzlCLE1BQWdEO0lBRWhELE1BQU0sT0FBTyxHQUFHLElBQUksR0FBRyxFQUFVLENBQUM7SUFDbEMsTUFBTSxPQUFPLEdBQUcsSUFBSSxHQUFHLEVBQVUsQ0FBQztJQUNsQyxNQUFNLFFBQVEsR0FBRyxJQUFJLEdBQUcsQ0FBQyxNQUFNLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxDQUFDLENBQUMsQ0FBQyxJQUFJLEVBQUUsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDO0lBRXpELFNBQVMsS0FBSyxDQUFDLElBQVksRUFBRSxJQUFjO1FBQ3pDLElBQUksT0FBTyxDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFDO1lBQ3RCLE1BQU0sSUFBSSxLQUFLLENBQ2IsaUNBQWlDLENBQUMsR0FBRyxJQUFJLEVBQUUsSUFBSSxDQUFDLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxFQUFFLENBQ2hFLENBQUM7UUFDSixDQUFDO1FBQ0QsSUFBSSxPQUFPLENBQUMsR0FBRyxDQUFDLElBQUksQ0FBQyxFQUFFLENBQUM7WUFDdEIsT0FBTztRQUNULENBQUM7UUFDRCxPQUFPLENBQUMsR0FBRyxDQUFDLElBQUksQ0FBQyxDQUFDO1FBQ2xCLE1BQU0sS0FBSyxHQUFHLFFBQVEsQ0FBQyxHQUFHLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDakMsSUFBSSxLQUFLLEVBQUUsU0FBUyxFQUFFLENBQUM7WUFDckIsS0FBSyxNQUFNLEdBQUcsSUFBSSxLQUFLLENBQUMsU0FBUyxFQUFFLENBQUM7Z0JBQ2xDLEtBQUssQ0FBQyxHQUFHLEVBQUUsQ0FBQyxHQUFHLElBQUksRUFBRSxJQUFJLENBQUMsQ0FBQyxDQUFDO1lBQzlCLENBQUM7UUFDSCxDQUFDO1FBQ0QsT0FBTyxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUNyQixPQUFPLENBQUMsR0FBRyxDQUFDLElBQUksQ0FBQyxDQUFDO0lBQ3BCLENBQUM7SUFFRCxLQUFLLE1BQU0sS0FBSyxJQUFJLE1BQU0sRUFBRSxDQUFDO1FBQzNCLEtBQUssQ0FBQyxLQUFLLENBQUMsSUFBSSxFQUFFLEVBQUUsQ0FBQyxDQUFDO0lBQ3hCLENBQUM7QUFDSCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBDb252ZXJ0IGEgc3RyaW5nIHRvIGtlYmFiLWNhc2UgZm9yIHVzZSBpbiBmaWxlIG5hbWVzIGFuZCBqb2IgSURzLlxuICovXG5leHBvcnQgZnVuY3Rpb24gdG9LZWJhYkNhc2Uoczogc3RyaW5nKTogc3RyaW5nIHtcbiAgcmV0dXJuIHNcbiAgICAucmVwbGFjZSgvW15hLXpBLVowLTldKy9nLCAnLScpXG4gICAgLnJlcGxhY2UoL14tK3wtKyQvZywgJycpXG4gICAgLnRvTG93ZXJDYXNlKCk7XG59XG5cbi8qKlxuICogQ29udmVydCBhIHN0cmluZyB0byBhIHZhbGlkIEdpdEh1YiBBY3Rpb25zIGpvYiBJRC5cbiAqIEpvYiBJRHMgbXVzdCBzdGFydCB3aXRoIGEgbGV0dGVyIG9yIHVuZGVyc2NvcmUgYW5kIGNvbnRhaW4gb25seSBhbHBoYW51bWVyaWNzLCBoeXBoZW5zLCBhbmQgdW5kZXJzY29yZXMuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiB0b0dpdGh1YkpvYklkKHM6IHN0cmluZyk6IHN0cmluZyB7XG4gIGxldCBvdXQgPSBzLnJlcGxhY2UoL1teQS1aYS16MC05Xy1dKy9nLCAnLScpO1xuICBvdXQgPSBvdXQucmVwbGFjZSgvLSsvZywgJy0nKTtcbiAgb3V0ID0gb3V0LnJlcGxhY2UoL14tK3wtKyQvZywgJycpO1xuICBvdXQgPSBvdXQudG9Mb3dlckNhc2UoKTtcbiAgaWYgKCFvdXQgfHwgIS9eW2Etel9dL2kudGVzdChvdXQpKSB7XG4gICAgb3V0ID0gYHMtJHtvdXR9YDtcbiAgfVxuICByZXR1cm4gb3V0O1xufVxuXG4vKipcbiAqIFZhbGlkYXRlIHRoYXQgdGhlcmUgYXJlIG5vIGNpcmN1bGFyIGRlcGVuZGVuY2llcyBpbiB0aGUgc3RhZ2UgZ3JhcGguXG4gKiBAdGhyb3dzIEVycm9yIGlmIGEgY3ljbGUgaXMgZGV0ZWN0ZWRcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIHZhbGlkYXRlTm9DeWNsZXMoXG4gIHN0YWdlczogeyBuYW1lOiBzdHJpbmc7IGRlcGVuZHNPbj86IHN0cmluZ1tdIH1bXSxcbik6IHZvaWQge1xuICBjb25zdCB2aXNpdGVkID0gbmV3IFNldDxzdHJpbmc+KCk7XG4gIGNvbnN0IGluU3RhY2sgPSBuZXcgU2V0PHN0cmluZz4oKTtcbiAgY29uc3Qgc3RhZ2VNYXAgPSBuZXcgTWFwKHN0YWdlcy5tYXAoKHMpID0+IFtzLm5hbWUsIHNdKSk7XG5cbiAgZnVuY3Rpb24gdmlzaXQobmFtZTogc3RyaW5nLCBwYXRoOiBzdHJpbmdbXSk6IHZvaWQge1xuICAgIGlmIChpblN0YWNrLmhhcyhuYW1lKSkge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKFxuICAgICAgICBgQ2lyY3VsYXIgZGVwZW5kZW5jeSBkZXRlY3RlZDogJHtbLi4ucGF0aCwgbmFtZV0uam9pbignIC0+ICcpfWAsXG4gICAgICApO1xuICAgIH1cbiAgICBpZiAodmlzaXRlZC5oYXMobmFtZSkpIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG4gICAgaW5TdGFjay5hZGQobmFtZSk7XG4gICAgY29uc3Qgc3RhZ2UgPSBzdGFnZU1hcC5nZXQobmFtZSk7XG4gICAgaWYgKHN0YWdlPy5kZXBlbmRzT24pIHtcbiAgICAgIGZvciAoY29uc3QgZGVwIG9mIHN0YWdlLmRlcGVuZHNPbikge1xuICAgICAgICB2aXNpdChkZXAsIFsuLi5wYXRoLCBuYW1lXSk7XG4gICAgICB9XG4gICAgfVxuICAgIGluU3RhY2suZGVsZXRlKG5hbWUpO1xuICAgIHZpc2l0ZWQuYWRkKG5hbWUpO1xuICB9XG5cbiAgZm9yIChjb25zdCBzdGFnZSBvZiBzdGFnZXMpIHtcbiAgICB2aXNpdChzdGFnZS5uYW1lLCBbXSk7XG4gIH1cbn1cbiJdfQ==