projen-pipelines 0.0.17 → 0.0.18
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/lib/{pipeline.d.ts → awscdk/base.d.ts} +10 -15
- package/lib/awscdk/base.js +284 -0
- package/lib/awscdk/github.d.ts +22 -0
- package/lib/awscdk/github.js +189 -0
- package/lib/{engine → awscdk}/index.js +1 -1
- package/lib/index.d.ts +1 -2
- package/lib/index.js +2 -3
- package/package.json +1 -1
- package/lib/engine/base.d.ts +0 -23
- package/lib/engine/base.js +0 -13
- package/lib/engine/github.d.ts +0 -20
- package/lib/engine/github.js +0 -180
- package/lib/pipeline.js +0 -277
- /package/lib/{engine → awscdk}/index.d.ts +0 -0
|
@@ -1,5 +1,4 @@
|
|
|
1
1
|
import { Component, awscdk } from 'projen';
|
|
2
|
-
import { BaseEngine, GithubEngineConfig } from './engine';
|
|
3
2
|
/**
|
|
4
3
|
* The Environment interface is designed to hold AWS related information
|
|
5
4
|
* for a specific deployment environment within your infrastructure.
|
|
@@ -78,14 +77,6 @@ export interface CDKPipelineOptions {
|
|
|
78
77
|
* @default CONTINUOUS_DELIVERY
|
|
79
78
|
*/
|
|
80
79
|
readonly deploymentType?: DeploymentType;
|
|
81
|
-
/**
|
|
82
|
-
* This field determines the CI/CD tooling that will be used to run the pipeline. The component
|
|
83
|
-
* will render workflows for the given system. Options include GitHub and GitLab.
|
|
84
|
-
*
|
|
85
|
-
* @default - tries to derive it from the projects configuration
|
|
86
|
-
*/
|
|
87
|
-
readonly engine?: PipelineEngine;
|
|
88
|
-
readonly githubConfig?: GithubEngineConfig;
|
|
89
80
|
readonly preInstallCommands?: string[];
|
|
90
81
|
readonly preSynthCommands?: string[];
|
|
91
82
|
readonly postSynthCommands?: string[];
|
|
@@ -94,13 +85,17 @@ export interface CDKPipelineOptions {
|
|
|
94
85
|
* The CDKPipeline class extends the Component class and sets up the necessary configuration for deploying AWS CDK (Cloud Development Kit) applications across multiple stages.
|
|
95
86
|
* It also manages tasks such as publishing CDK assets, bumping version based on git tags, and cleaning up conflicting tasks.
|
|
96
87
|
*/
|
|
97
|
-
export declare class CDKPipeline extends Component {
|
|
98
|
-
|
|
99
|
-
private
|
|
88
|
+
export declare abstract class CDKPipeline extends Component {
|
|
89
|
+
protected app: awscdk.AwsCdkTypeScriptApp;
|
|
90
|
+
private baseOptions;
|
|
100
91
|
readonly stackPrefix: string;
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
92
|
+
constructor(app: awscdk.AwsCdkTypeScriptApp, baseOptions: CDKPipelineOptions);
|
|
93
|
+
protected getInstallCommands(): string[];
|
|
94
|
+
protected getInstallPackageCommands(packageName: string, runPreInstallCommands?: boolean): string[];
|
|
95
|
+
protected getSynthCommands(): string[];
|
|
96
|
+
protected getAssetUploadCommands(needsVersionedArtifacts: boolean): string[];
|
|
97
|
+
protected getDeployCommands(stageName: string): string[];
|
|
98
|
+
protected getDiffCommands(stageName: string): string[];
|
|
104
99
|
/**
|
|
105
100
|
* This method generates the entry point for the application, including interfaces and classes
|
|
106
101
|
* necessary to set up the pipeline and define the AWS CDK stacks for different environments.
|
|
@@ -0,0 +1,284 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.CDKPipeline = exports.DeploymentType = exports.PipelineEngine = void 0;
|
|
4
|
+
const projen_1 = require("projen");
|
|
5
|
+
const common_1 = require("projen/lib/common");
|
|
6
|
+
const javascript_1 = require("projen/lib/javascript");
|
|
7
|
+
/**
|
|
8
|
+
* The CI/CD tooling used to run your pipeline.
|
|
9
|
+
* The component will render workflows for the given system
|
|
10
|
+
*/
|
|
11
|
+
var PipelineEngine;
|
|
12
|
+
(function (PipelineEngine) {
|
|
13
|
+
/** Create GitHub actions */
|
|
14
|
+
PipelineEngine[PipelineEngine["GITHUB"] = 0] = "GITHUB";
|
|
15
|
+
/** Create a .gitlab-ci.yaml file */
|
|
16
|
+
PipelineEngine[PipelineEngine["GITLAB"] = 1] = "GITLAB";
|
|
17
|
+
// /** Create AWS CodeCatalyst workflows */
|
|
18
|
+
// CODE_CATALYST,
|
|
19
|
+
})(PipelineEngine || (exports.PipelineEngine = PipelineEngine = {}));
|
|
20
|
+
/**
|
|
21
|
+
* Describes the type of pipeline that will be created
|
|
22
|
+
*/
|
|
23
|
+
var DeploymentType;
|
|
24
|
+
(function (DeploymentType) {
|
|
25
|
+
/** Deploy every commit as far as possible; hopefully into production */
|
|
26
|
+
DeploymentType[DeploymentType["CONTINUOUS_DEPLOYMENT"] = 0] = "CONTINUOUS_DEPLOYMENT";
|
|
27
|
+
/** Build every commit and prepare all assets for a later deployment */
|
|
28
|
+
DeploymentType[DeploymentType["CONTINUOUS_DELIVERY"] = 1] = "CONTINUOUS_DELIVERY";
|
|
29
|
+
})(DeploymentType || (exports.DeploymentType = DeploymentType = {}));
|
|
30
|
+
/**
|
|
31
|
+
* The CDKPipeline class extends the Component class and sets up the necessary configuration for deploying AWS CDK (Cloud Development Kit) applications across multiple stages.
|
|
32
|
+
* It also manages tasks such as publishing CDK assets, bumping version based on git tags, and cleaning up conflicting tasks.
|
|
33
|
+
*/
|
|
34
|
+
class CDKPipeline extends projen_1.Component {
|
|
35
|
+
constructor(app, baseOptions) {
|
|
36
|
+
var _a, _b, _c;
|
|
37
|
+
super(app);
|
|
38
|
+
this.app = app;
|
|
39
|
+
this.baseOptions = baseOptions;
|
|
40
|
+
// Add development dependencies
|
|
41
|
+
this.app.addDevDeps('@types/standard-version', 'standard-version', 'cdk-assets');
|
|
42
|
+
// this.app.addDeps(
|
|
43
|
+
// );
|
|
44
|
+
this.stackPrefix = (_a = baseOptions.stackPrefix) !== null && _a !== void 0 ? _a : app.name;
|
|
45
|
+
// Removes the compiled cloud assembly before each synth
|
|
46
|
+
(_b = this.project.tasks.tryFind('synth')) === null || _b === void 0 ? void 0 : _b.prependExec(`rm -rf ${this.app.cdkConfig.cdkout}`);
|
|
47
|
+
(_c = this.project.tasks.tryFind('synth:silent')) === null || _c === void 0 ? void 0 : _c.prependExec(`rm -rf ${this.app.cdkConfig.cdkout}`);
|
|
48
|
+
// Remove tasks that might conflict with the pipeline process
|
|
49
|
+
this.project.removeTask('deploy');
|
|
50
|
+
this.project.removeTask('diff');
|
|
51
|
+
this.project.removeTask('destroy');
|
|
52
|
+
this.project.removeTask('watch');
|
|
53
|
+
// Creates different deployment stages
|
|
54
|
+
if (baseOptions.personalStage) {
|
|
55
|
+
this.createPersonalStage();
|
|
56
|
+
}
|
|
57
|
+
if (baseOptions.featureStages) {
|
|
58
|
+
this.createFeatureStage();
|
|
59
|
+
}
|
|
60
|
+
for (const stage of baseOptions.stages) {
|
|
61
|
+
this.createPipelineStage(stage);
|
|
62
|
+
}
|
|
63
|
+
// Creates tasks to handle the release process
|
|
64
|
+
this.createReleaseTasks();
|
|
65
|
+
// Creates a specialized CDK App class
|
|
66
|
+
this.createApplicationEntrypoint();
|
|
67
|
+
}
|
|
68
|
+
getInstallCommands() {
|
|
69
|
+
var _a;
|
|
70
|
+
return [
|
|
71
|
+
...((_a = this.baseOptions.preInstallCommands) !== null && _a !== void 0 ? _a : []),
|
|
72
|
+
`npx projen ${this.app.package.installCiTask.name}`,
|
|
73
|
+
];
|
|
74
|
+
}
|
|
75
|
+
getInstallPackageCommands(packageName, runPreInstallCommands = false) {
|
|
76
|
+
var _a;
|
|
77
|
+
const commands = runPreInstallCommands ? (_a = this.baseOptions.preInstallCommands) !== null && _a !== void 0 ? _a : [] : [];
|
|
78
|
+
switch (this.app.package.packageManager) {
|
|
79
|
+
case javascript_1.NodePackageManager.YARN:
|
|
80
|
+
case javascript_1.NodePackageManager.YARN2:
|
|
81
|
+
commands.push(`yarn add ${packageName}`);
|
|
82
|
+
break;
|
|
83
|
+
case javascript_1.NodePackageManager.NPM:
|
|
84
|
+
commands.push(`npm install ${packageName}`);
|
|
85
|
+
break;
|
|
86
|
+
default:
|
|
87
|
+
throw new Error('No install scripts for packageManager: ' + this.app.package.packageManager);
|
|
88
|
+
}
|
|
89
|
+
return commands;
|
|
90
|
+
}
|
|
91
|
+
getSynthCommands() {
|
|
92
|
+
var _a, _b;
|
|
93
|
+
return [
|
|
94
|
+
...this.getInstallCommands(),
|
|
95
|
+
...((_a = this.baseOptions.preSynthCommands) !== null && _a !== void 0 ? _a : []),
|
|
96
|
+
'npx projen build',
|
|
97
|
+
...((_b = this.baseOptions.postSynthCommands) !== null && _b !== void 0 ? _b : []),
|
|
98
|
+
];
|
|
99
|
+
}
|
|
100
|
+
getAssetUploadCommands(needsVersionedArtifacts) {
|
|
101
|
+
return [
|
|
102
|
+
...this.getInstallCommands(),
|
|
103
|
+
'npx projen publish:assets',
|
|
104
|
+
...(needsVersionedArtifacts ? [
|
|
105
|
+
'npx projen bump',
|
|
106
|
+
'npx projen release:push-assembly',
|
|
107
|
+
] : []),
|
|
108
|
+
];
|
|
109
|
+
}
|
|
110
|
+
getDeployCommands(stageName) {
|
|
111
|
+
return [
|
|
112
|
+
`npx projen deploy:${stageName}`,
|
|
113
|
+
];
|
|
114
|
+
}
|
|
115
|
+
getDiffCommands(stageName) {
|
|
116
|
+
return [
|
|
117
|
+
`npx projen diff:${stageName}`,
|
|
118
|
+
];
|
|
119
|
+
}
|
|
120
|
+
/**
|
|
121
|
+
* This method generates the entry point for the application, including interfaces and classes
|
|
122
|
+
* necessary to set up the pipeline and define the AWS CDK stacks for different environments.
|
|
123
|
+
*/
|
|
124
|
+
createApplicationEntrypoint() {
|
|
125
|
+
let propsCode = '';
|
|
126
|
+
let appCode = '';
|
|
127
|
+
if (this.baseOptions.personalStage) {
|
|
128
|
+
propsCode += ` /** This function will be used to generate a personal stack. */
|
|
129
|
+
providePersonalStack: (app: App, stackId: string, props: PipelineAppStackProps) => Stack;
|
|
130
|
+
`;
|
|
131
|
+
appCode += ` // If the environment variable USER is set and a function is provided for creating a personal stack, it is called with necessary arguments.
|
|
132
|
+
if (props.providePersonalStack && process.env.USER) {
|
|
133
|
+
const stageName = 'personal-' + process.env.USER.toLowerCase().replace(/\\\//g, '-');
|
|
134
|
+
props.providePersonalStack(this, '${this.stackPrefix}-personal', { env: ${JSON.stringify(this.baseOptions.personalStage.env)}, stackName: \`${this.stackPrefix}-\${stageName}\`, stageName });
|
|
135
|
+
}
|
|
136
|
+
`;
|
|
137
|
+
}
|
|
138
|
+
if (this.baseOptions.featureStages) {
|
|
139
|
+
propsCode += ` /** This function will be used to generate a feature stack. */
|
|
140
|
+
provideFeatureStack: (app: App, stackId: string, props: PipelineAppStackProps) => Stack;
|
|
141
|
+
`;
|
|
142
|
+
appCode += ` // If the environment variable BRANCH is set and a function is provided for creating a feature stack, it is called with necessary arguments.
|
|
143
|
+
if (props.provideFeatureStack && process.env.BRANCH) {
|
|
144
|
+
const stageName = 'feature-' + process.env.BRANCH.toLowerCase().replace(/\\\//g, '-');
|
|
145
|
+
props.provideFeatureStack(this, '${this.stackPrefix}-feature', { env: ${JSON.stringify(this.baseOptions.featureStages.env)}, stackName: \`${this.stackPrefix}-\${stageName}\`, stageName });
|
|
146
|
+
}
|
|
147
|
+
`;
|
|
148
|
+
}
|
|
149
|
+
for (const stage of this.baseOptions.stages) {
|
|
150
|
+
const nameUpperFirst = `${stage.name.charAt(0).toUpperCase()}${stage.name.substring(1)}`;
|
|
151
|
+
propsCode += ` /** This function will be used to generate a ${stage.name} stack. */
|
|
152
|
+
provide${nameUpperFirst}Stack: (app: App, stackId: string, props: PipelineAppStackProps) => Stack;
|
|
153
|
+
`;
|
|
154
|
+
appCode += ` // If a function is provided for creating a ${stage.name} stack, it is called with necessary arguments.
|
|
155
|
+
if (props.provide${nameUpperFirst}Stack) {
|
|
156
|
+
props.provide${nameUpperFirst}Stack(this, '${this.stackPrefix}-${stage.name}', { env: ${JSON.stringify(stage.env)}, stackName: '${this.stackPrefix}-${stage.name}', stageName: '${stage.name}' });
|
|
157
|
+
}
|
|
158
|
+
`;
|
|
159
|
+
}
|
|
160
|
+
const appFile = new projen_1.TextFile(this.project, `${this.app.srcdir}/app.ts`);
|
|
161
|
+
appFile.addLine(`// ${common_1.PROJEN_MARKER}
|
|
162
|
+
/* eslint-disable */
|
|
163
|
+
import { App, AppProps, Stack, StackProps } from 'aws-cdk-lib';
|
|
164
|
+
|
|
165
|
+
/**
|
|
166
|
+
* PipelineAppProps is an extension of AppProps, which is part of the AWS CDK core.
|
|
167
|
+
* It includes optional functions to provide AWS Stacks for different stages.
|
|
168
|
+
*
|
|
169
|
+
* Use these functions to instantiate your application stacks with the parameters for
|
|
170
|
+
* each stage
|
|
171
|
+
*/
|
|
172
|
+
export interface PipelineAppProps extends AppProps {
|
|
173
|
+
${propsCode}
|
|
174
|
+
}
|
|
175
|
+
|
|
176
|
+
/**
|
|
177
|
+
* PipelineAppStackProps is an extension of StackProps, which is part of the AWS CDK core.
|
|
178
|
+
* It includes an additional property to specify the stage name.
|
|
179
|
+
*/
|
|
180
|
+
export interface PipelineAppStackProps extends StackProps {
|
|
181
|
+
stageName: string;
|
|
182
|
+
}
|
|
183
|
+
|
|
184
|
+
/**
|
|
185
|
+
* The PipelineApp class extends the App class from AWS CDK and overrides the constructor to support
|
|
186
|
+
* different stages of the application (development, production, personal, feature) by invoking the provided
|
|
187
|
+
* stack-providing functions from the props.
|
|
188
|
+
*/
|
|
189
|
+
export class PipelineApp extends App {
|
|
190
|
+
constructor(props: PipelineAppProps) {
|
|
191
|
+
super(props);
|
|
192
|
+
|
|
193
|
+
${appCode}
|
|
194
|
+
|
|
195
|
+
}
|
|
196
|
+
}
|
|
197
|
+
`);
|
|
198
|
+
}
|
|
199
|
+
/**
|
|
200
|
+
* This method sets up tasks to publish CDK assets to all accounts and handle versioning, including bumping the version
|
|
201
|
+
* based on the latest git tag and pushing the CDK assembly to the package repository.
|
|
202
|
+
*/
|
|
203
|
+
createReleaseTasks() {
|
|
204
|
+
// Task to publish the CDK assets to all accounts
|
|
205
|
+
this.project.addTask('publish:assets', {
|
|
206
|
+
steps: this.baseOptions.stages.map(stage => ({
|
|
207
|
+
exec: `npx cdk-assets -p ${this.app.cdkConfig.cdkout}/${this.stackPrefix}-${stage.name}.assets.json publish`,
|
|
208
|
+
})),
|
|
209
|
+
});
|
|
210
|
+
this.project.addTask('bump', {
|
|
211
|
+
description: 'Bumps version based on latest git tag',
|
|
212
|
+
steps: [
|
|
213
|
+
{
|
|
214
|
+
exec: 'pipelines-release bump',
|
|
215
|
+
},
|
|
216
|
+
{
|
|
217
|
+
exec: 'git push --tags',
|
|
218
|
+
},
|
|
219
|
+
],
|
|
220
|
+
});
|
|
221
|
+
this.project.addTask('release:push-assembly', {
|
|
222
|
+
steps: [
|
|
223
|
+
{
|
|
224
|
+
exec: `pipelines-release create-manifest "${this.app.cdkConfig.cdkout}" "${this.baseOptions.pkgNamespace}"`,
|
|
225
|
+
},
|
|
226
|
+
{
|
|
227
|
+
cwd: this.app.cdkConfig.cdkout,
|
|
228
|
+
exec: 'npm version --no-git-tag-version from-git',
|
|
229
|
+
},
|
|
230
|
+
{
|
|
231
|
+
cwd: this.app.cdkConfig.cdkout,
|
|
232
|
+
exec: 'npm publish',
|
|
233
|
+
},
|
|
234
|
+
],
|
|
235
|
+
});
|
|
236
|
+
}
|
|
237
|
+
/**
|
|
238
|
+
* This method sets up tasks for the personal deployment stage, including deployment, watching for changes,
|
|
239
|
+
* comparing changes (diff), and destroying the stack when no longer needed.
|
|
240
|
+
*/
|
|
241
|
+
createPersonalStage() {
|
|
242
|
+
this.project.addTask('deploy:personal', {
|
|
243
|
+
exec: `cdk deploy ${this.stackPrefix}-personal`,
|
|
244
|
+
});
|
|
245
|
+
this.project.addTask('watch:personal', {
|
|
246
|
+
exec: `cdk deploy --watch --hotswap ${this.stackPrefix}-personal`,
|
|
247
|
+
});
|
|
248
|
+
this.project.addTask('diff:personal', {
|
|
249
|
+
exec: `cdk diff ${this.stackPrefix}-personal`,
|
|
250
|
+
});
|
|
251
|
+
this.project.addTask('destroy:personal', {
|
|
252
|
+
exec: `cdk destroy ${this.stackPrefix}-personal`,
|
|
253
|
+
});
|
|
254
|
+
}
|
|
255
|
+
/**
|
|
256
|
+
* This method sets up tasks for the feature deployment stage, including deployment, comparing changes (diff),
|
|
257
|
+
* and destroying the stack when no longer needed.
|
|
258
|
+
*/
|
|
259
|
+
createFeatureStage() {
|
|
260
|
+
this.project.addTask('deploy:feature', {
|
|
261
|
+
exec: `cdk --progress events --require-approval never deploy ${this.stackPrefix}-feature`,
|
|
262
|
+
});
|
|
263
|
+
this.project.addTask('diff:feature', {
|
|
264
|
+
exec: `cdk diff ${this.stackPrefix}-feature`,
|
|
265
|
+
});
|
|
266
|
+
this.project.addTask('destroy:feature', {
|
|
267
|
+
exec: `cdk destroy ${this.stackPrefix}-feature`,
|
|
268
|
+
});
|
|
269
|
+
}
|
|
270
|
+
/**
|
|
271
|
+
* This method sets up tasks for the general pipeline stages (dev, prod), including deployment and comparing changes (diff).
|
|
272
|
+
* @param {DeployStageOptions} stage - The stage to create
|
|
273
|
+
*/
|
|
274
|
+
createPipelineStage(stage) {
|
|
275
|
+
this.project.addTask(`deploy:${stage.name}`, {
|
|
276
|
+
exec: `cdk --app ${this.app.cdkConfig.cdkout} --progress events --require-approval never deploy ${this.stackPrefix}-${stage.name}`,
|
|
277
|
+
});
|
|
278
|
+
this.project.addTask(`diff:${stage.name}`, {
|
|
279
|
+
exec: `cdk --app ${this.app.cdkConfig.cdkout} diff ${this.stackPrefix}-${stage.name}`,
|
|
280
|
+
});
|
|
281
|
+
}
|
|
282
|
+
}
|
|
283
|
+
exports.CDKPipeline = CDKPipeline;
|
|
284
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"base.js","sourceRoot":"","sources":["../../src/awscdk/base.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, TextFile, awscdk } from 'projen';\nimport { PROJEN_MARKER } from 'projen/lib/common';\nimport { NodePackageManager } from 'projen/lib/javascript';\n\n/**\n * The Environment interface is designed to hold AWS related information\n * for a specific deployment environment within your infrastructure.\n * Each environment requires a specific account and region for its resources.\n */\nexport interface Environment {\n  /**\n   * The AWS Account ID associated with the environment. It's important because\n   * different services or features could have distinct permissions and settings\n   * in different accounts.\n   */\n  readonly account: string;\n\n  /**\n   * The AWS Region for the environment. This determines where your resources\n   * are created and where your application will run. It can affect latency,\n   * availability, and pricing.\n   */\n  readonly region: string;\n}\n\n/**\n * The CI/CD tooling used to run your pipeline.\n * The component will render workflows for the given system\n */\nexport enum PipelineEngine {\n  /** Create GitHub actions */\n  GITHUB,\n  /** Create a .gitlab-ci.yaml file */\n  GITLAB,\n  // /** Create AWS CodeCatalyst workflows */\n  // CODE_CATALYST,\n}\n\n/**\n * Describes the type of pipeline that will be created\n */\nexport enum DeploymentType {\n  /** Deploy every commit as far as possible; hopefully into production */\n  CONTINUOUS_DEPLOYMENT,\n  /** Build every commit and prepare all assets for a later deployment */\n  CONTINUOUS_DELIVERY,\n}\n\nexport interface DeploymentStage {\n  readonly name: string;\n  readonly env: Environment;\n  readonly manualApproval?: boolean;\n}\n\n/**\n * The CDKPipelineOptions interface is designed to provide configuration\n * options for a CDK (Cloud Development Kit) pipeline. It allows the definition\n * of settings such as the stack prefix and package namespace to be used in the\n * AWS stack, along with the environments configuration to be used.\n */\nexport interface CDKPipelineOptions {\n\n  /**\n   * This field is used to define a prefix for the AWS Stack resources created\n   * during the pipeline's operation.\n   *\n   * @default project name\n   */\n  readonly stackPrefix?: string;\n\n  /**\n   * This field determines the NPM namespace to be used when packaging CDK cloud\n   * assemblies. A namespace helps group related resources together, providing\n   * better organization and ease of management.\n   */\n  readonly pkgNamespace: string;\n\n  readonly stages: DeploymentStage[];\n\n  readonly personalStage?: {\n    readonly env: Environment;\n  };\n\n  readonly featureStages?: {\n    readonly env: Environment;\n  };\n\n  /**\n   * This field specifies the type of pipeline to create. If set to CONTINUOUS_DEPLOYMENT,\n   * every commit is deployed as far as possible, hopefully into production. If set to\n   * CONTINUOUS_DELIVERY, every commit is built and all assets are prepared for a later deployment.\n   *\n   * @default CONTINUOUS_DELIVERY\n   */\n  readonly deploymentType?: DeploymentType;\n\n  readonly preInstallCommands?: string[];\n  readonly preSynthCommands?: string[];\n  readonly postSynthCommands?: string[];\n\n}\n\n/**\n * The CDKPipeline class extends the Component class and sets up the necessary configuration for deploying AWS CDK (Cloud Development Kit) applications across multiple stages.\n * It also manages tasks such as publishing CDK assets, bumping version based on git tags, and cleaning up conflicting tasks.\n */\nexport abstract class CDKPipeline extends Component {\n\n  public readonly stackPrefix: string;\n\n  constructor(protected app: awscdk.AwsCdkTypeScriptApp, private baseOptions: CDKPipelineOptions) {\n    super(app);\n\n    // Add development dependencies\n    this.app.addDevDeps(\n      '@types/standard-version',\n      'standard-version',\n      'cdk-assets',\n    );\n    // this.app.addDeps(\n    // );\n\n    this.stackPrefix = baseOptions.stackPrefix ?? app.name;\n\n    // Removes the compiled cloud assembly before each synth\n    this.project.tasks.tryFind('synth')?.prependExec(`rm -rf ${this.app.cdkConfig.cdkout}`);\n    this.project.tasks.tryFind('synth:silent')?.prependExec(`rm -rf ${this.app.cdkConfig.cdkout}`);\n\n    // Remove tasks that might conflict with the pipeline process\n    this.project.removeTask('deploy');\n    this.project.removeTask('diff');\n    this.project.removeTask('destroy');\n    this.project.removeTask('watch');\n\n    // Creates different deployment stages\n    if (baseOptions.personalStage) {\n      this.createPersonalStage();\n    }\n    if (baseOptions.featureStages) {\n      this.createFeatureStage();\n    }\n    for (const stage of baseOptions.stages) {\n      this.createPipelineStage(stage);\n    }\n\n    // Creates tasks to handle the release process\n    this.createReleaseTasks();\n\n    // Creates a specialized CDK App class\n    this.createApplicationEntrypoint();\n\n  }\n\n  protected getInstallCommands(): string[] {\n    return [\n      ...(this.baseOptions.preInstallCommands ?? []),\n      `npx projen ${this.app.package.installCiTask.name}`,\n    ];\n  }\n\n  protected getInstallPackageCommands(packageName: string, runPreInstallCommands: boolean = false): string[] {\n    const commands = runPreInstallCommands ? this.baseOptions.preInstallCommands ?? [] : [];\n\n    switch (this.app.package.packageManager) {\n      case NodePackageManager.YARN:\n      case NodePackageManager.YARN2:\n        commands.push(`yarn add ${packageName}`);\n        break;\n      case NodePackageManager.NPM:\n        commands.push(`npm install ${packageName}`);\n        break;\n      default:\n        throw new Error('No install scripts for packageManager: ' + this.app.package.packageManager);\n    }\n    return commands;\n  }\n\n  protected getSynthCommands(): string[] {\n    return [\n      ...this.getInstallCommands(),\n      ...(this.baseOptions.preSynthCommands ?? []),\n      'npx projen build',\n      ...(this.baseOptions.postSynthCommands ?? []),\n    ];\n  }\n\n  protected getAssetUploadCommands(needsVersionedArtifacts: boolean): string[] {\n    return [\n      ...this.getInstallCommands(),\n      'npx projen publish:assets',\n      ...(needsVersionedArtifacts ? [\n        'npx projen bump',\n        'npx projen release:push-assembly',\n      ] : []),\n    ];\n  }\n\n  protected getDeployCommands(stageName: string): string[] {\n    return [\n      `npx projen deploy:${stageName}`,\n    ];\n  }\n\n  protected getDiffCommands(stageName: string): string[] {\n    return [\n      `npx projen diff:${stageName}`,\n    ];\n  }\n\n  /**\n   * This method generates the entry point for the application, including interfaces and classes\n   * necessary to set up the pipeline and define the AWS CDK stacks for different environments.\n   */\n  private createApplicationEntrypoint() {\n    let propsCode = '';\n    let appCode = '';\n\n    if (this.baseOptions.personalStage) {\n      propsCode += `  /** This function will be used to generate a personal stack. */\n  providePersonalStack: (app: App, stackId: string, props: PipelineAppStackProps) => Stack;\n`;\n      appCode += `    // If the environment variable USER is set and a function is provided for creating a personal stack, it is called with necessary arguments.\n    if (props.providePersonalStack && process.env.USER) {\n      const stageName = 'personal-' + process.env.USER.toLowerCase().replace(/\\\\\\//g, '-');\n      props.providePersonalStack(this, '${this.stackPrefix}-personal', { env: ${JSON.stringify(this.baseOptions.personalStage.env)}, stackName: \\`${this.stackPrefix}-\\${stageName}\\`, stageName });\n    }\n`;\n    }\n\n    if (this.baseOptions.featureStages) {\n      propsCode += `  /** This function will be used to generate a feature stack. */\n  provideFeatureStack: (app: App, stackId: string, props: PipelineAppStackProps) => Stack;\n`;\n      appCode += `    // If the environment variable BRANCH is set and a function is provided for creating a feature stack, it is called with necessary arguments.\n    if (props.provideFeatureStack && process.env.BRANCH) {\n      const stageName = 'feature-' + process.env.BRANCH.toLowerCase().replace(/\\\\\\//g, '-');\n      props.provideFeatureStack(this, '${this.stackPrefix}-feature', { env: ${JSON.stringify(this.baseOptions.featureStages.env)}, stackName: \\`${this.stackPrefix}-\\${stageName}\\`, stageName });\n    }\n`;\n    }\n\n    for (const stage of this.baseOptions.stages) {\n      const nameUpperFirst = `${stage.name.charAt(0).toUpperCase()}${stage.name.substring(1)}`;\n\n      propsCode += `  /** This function will be used to generate a ${stage.name} stack. */\n  provide${nameUpperFirst}Stack: (app: App, stackId: string, props: PipelineAppStackProps) => Stack;\n`;\n      appCode += `    // If a function is provided for creating a ${stage.name} stack, it is called with necessary arguments.\n    if (props.provide${nameUpperFirst}Stack) {\n      props.provide${nameUpperFirst}Stack(this, '${this.stackPrefix}-${stage.name}', { env: ${JSON.stringify(stage.env)}, stackName: '${this.stackPrefix}-${stage.name}', stageName: '${stage.name}' });\n    }\n`;\n    }\n\n    const appFile = new TextFile(this.project, `${this.app.srcdir}/app.ts`);\n    appFile.addLine(`// ${PROJEN_MARKER}\n/* eslint-disable */\nimport { App, AppProps, Stack, StackProps } from 'aws-cdk-lib';\n\n/**\n * PipelineAppProps is an extension of AppProps, which is part of the AWS CDK core.\n * It includes optional functions to provide AWS Stacks for different stages.\n *\n * Use these functions to instantiate your application stacks with the parameters for\n * each stage\n */\nexport interface PipelineAppProps extends AppProps {\n${propsCode}\n}\n\n/**\n * PipelineAppStackProps is an extension of StackProps, which is part of the AWS CDK core.\n * It includes an additional property to specify the stage name.\n */\nexport interface PipelineAppStackProps extends StackProps {\n  stageName: string;\n}\n\n/**\n * The PipelineApp class extends the App class from AWS CDK and overrides the constructor to support\n * different stages of the application (development, production, personal, feature) by invoking the provided\n * stack-providing functions from the props.\n */\nexport class PipelineApp extends App {\n  constructor(props: PipelineAppProps) {\n    super(props);\n\n${appCode}\n\n  }\n}\n`);\n  }\n\n  /**\n   * This method sets up tasks to publish CDK assets to all accounts and handle versioning, including bumping the version\n   * based on the latest git tag and pushing the CDK assembly to the package repository.\n   */\n  private createReleaseTasks() {\n    // Task to publish the CDK assets to all accounts\n    this.project.addTask('publish:assets', {\n      steps: this.baseOptions.stages.map(stage => ({\n        exec: `npx cdk-assets -p ${this.app.cdkConfig.cdkout}/${this.stackPrefix}-${stage.name}.assets.json publish`,\n      })),\n    });\n\n    this.project.addTask('bump', {\n      description: 'Bumps version based on latest git tag',\n      steps: [\n        {\n          exec: 'pipelines-release bump',\n        },\n        {\n          exec: 'git push --tags',\n        },\n      ],\n    });\n    this.project.addTask('release:push-assembly', {\n      steps: [\n        {\n          exec: `pipelines-release create-manifest \"${this.app.cdkConfig.cdkout}\"  \"${this.baseOptions.pkgNamespace}\"`,\n        },\n        {\n          cwd: this.app.cdkConfig.cdkout,\n          exec: 'npm version --no-git-tag-version from-git',\n        },\n        {\n          cwd: this.app.cdkConfig.cdkout,\n          exec: 'npm publish',\n        },\n      ],\n    });\n  }\n\n  /**\n   * This method sets up tasks for the personal deployment stage, including deployment, watching for changes,\n   * comparing changes (diff), and destroying the stack when no longer needed.\n   */\n  private createPersonalStage() {\n    this.project.addTask('deploy:personal', {\n      exec: `cdk deploy ${this.stackPrefix}-personal`,\n    });\n    this.project.addTask('watch:personal', {\n      exec: `cdk deploy --watch --hotswap ${this.stackPrefix}-personal`,\n    });\n    this.project.addTask('diff:personal', {\n      exec: `cdk diff ${this.stackPrefix}-personal`,\n    });\n    this.project.addTask('destroy:personal', {\n      exec: `cdk destroy ${this.stackPrefix}-personal`,\n    });\n  }\n\n  /**\n   * This method sets up tasks for the feature deployment stage, including deployment, comparing changes (diff),\n   * and destroying the stack when no longer needed.\n   */\n  private createFeatureStage() {\n    this.project.addTask('deploy:feature', {\n      exec: `cdk --progress events --require-approval never deploy ${this.stackPrefix}-feature`,\n    });\n    this.project.addTask('diff:feature', {\n      exec: `cdk diff ${this.stackPrefix}-feature`,\n    });\n    this.project.addTask('destroy:feature', {\n      exec: `cdk destroy ${this.stackPrefix}-feature`,\n    });\n  }\n\n  /**\n   * This method sets up tasks for the general pipeline stages (dev, prod), including deployment and comparing changes (diff).\n   * @param {DeployStageOptions} stage - The stage to create\n   */\n  private createPipelineStage(stage: DeploymentStage) {\n    this.project.addTask(`deploy:${stage.name}`, {\n      exec: `cdk --app ${this.app.cdkConfig.cdkout} --progress events --require-approval never deploy ${this.stackPrefix}-${stage.name}`,\n    });\n    this.project.addTask(`diff:${stage.name}`, {\n      exec: `cdk --app ${this.app.cdkConfig.cdkout} diff ${this.stackPrefix}-${stage.name}`,\n    });\n  }\n}"]}
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
import { awscdk } from 'projen';
|
|
2
|
+
import { CDKPipeline, CDKPipelineOptions, DeploymentStage } from './base';
|
|
3
|
+
export interface GithubCDKPipelineOptions extends CDKPipelineOptions {
|
|
4
|
+
readonly iamRoleArns: {
|
|
5
|
+
readonly default?: string;
|
|
6
|
+
readonly synth?: string;
|
|
7
|
+
readonly assetPublishing?: string;
|
|
8
|
+
readonly deployment?: {
|
|
9
|
+
[stage: string]: string;
|
|
10
|
+
};
|
|
11
|
+
};
|
|
12
|
+
}
|
|
13
|
+
export declare class GithubCDKPipeline extends CDKPipeline {
|
|
14
|
+
private options;
|
|
15
|
+
readonly needsVersionedArtifacts: boolean;
|
|
16
|
+
private deploymentWorkflow;
|
|
17
|
+
private deploymentStages;
|
|
18
|
+
constructor(app: awscdk.AwsCdkTypeScriptApp, options: GithubCDKPipelineOptions);
|
|
19
|
+
private createSynth;
|
|
20
|
+
createAssetUpload(): void;
|
|
21
|
+
createDeployment(stage: DeploymentStage): void;
|
|
22
|
+
}
|
|
@@ -0,0 +1,189 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.GithubCDKPipeline = void 0;
|
|
4
|
+
const workflows_model_1 = require("projen/lib/github/workflows-model");
|
|
5
|
+
const base_1 = require("./base");
|
|
6
|
+
class GithubCDKPipeline extends base_1.CDKPipeline {
|
|
7
|
+
constructor(app, options) {
|
|
8
|
+
super(app, options);
|
|
9
|
+
this.options = options;
|
|
10
|
+
this.deploymentStages = [];
|
|
11
|
+
this.deploymentWorkflow = this.app.github.addWorkflow('deploy');
|
|
12
|
+
this.deploymentWorkflow.on({
|
|
13
|
+
push: {
|
|
14
|
+
branches: ['main'], // TODO use defaultReleaseBranch
|
|
15
|
+
},
|
|
16
|
+
workflowDispatch: {},
|
|
17
|
+
});
|
|
18
|
+
this.needsVersionedArtifacts = this.options.stages.find(s => s.manualApproval === true) !== undefined;
|
|
19
|
+
this.createSynth();
|
|
20
|
+
this.createAssetUpload();
|
|
21
|
+
for (const stage of options.stages) {
|
|
22
|
+
this.createDeployment(stage);
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
createSynth() {
|
|
26
|
+
var _a;
|
|
27
|
+
const steps = [{
|
|
28
|
+
name: 'Checkout',
|
|
29
|
+
uses: 'actions/checkout@v3',
|
|
30
|
+
}];
|
|
31
|
+
if ((_a = this.options.iamRoleArns) === null || _a === void 0 ? void 0 : _a.synth) {
|
|
32
|
+
steps.push({
|
|
33
|
+
name: 'AWS Credentials',
|
|
34
|
+
uses: 'aws-actions/configure-aws-credentials@master',
|
|
35
|
+
with: {
|
|
36
|
+
'role-to-assume': this.options.iamRoleArns.synth,
|
|
37
|
+
'role-session-name': 'GitHubAction',
|
|
38
|
+
'aws-region': 'us-east-1',
|
|
39
|
+
},
|
|
40
|
+
});
|
|
41
|
+
}
|
|
42
|
+
steps.push(...this.getSynthCommands().map(cmd => ({
|
|
43
|
+
run: cmd,
|
|
44
|
+
})));
|
|
45
|
+
steps.push({
|
|
46
|
+
uses: 'actions/upload-artifact@v3',
|
|
47
|
+
with: {
|
|
48
|
+
name: 'cloud-assembly',
|
|
49
|
+
path: `${this.app.cdkConfig.cdkout}/`,
|
|
50
|
+
},
|
|
51
|
+
});
|
|
52
|
+
this.deploymentWorkflow.addJob('synth', {
|
|
53
|
+
name: 'Synth CDK application',
|
|
54
|
+
runsOn: ['ubuntu-latest'],
|
|
55
|
+
env: {
|
|
56
|
+
CI: 'true',
|
|
57
|
+
},
|
|
58
|
+
permissions: { idToken: workflows_model_1.JobPermission.WRITE, contents: workflows_model_1.JobPermission.READ },
|
|
59
|
+
steps,
|
|
60
|
+
});
|
|
61
|
+
}
|
|
62
|
+
createAssetUpload() {
|
|
63
|
+
var _a, _b, _c;
|
|
64
|
+
this.deploymentWorkflow.addJob('assetUpload', {
|
|
65
|
+
name: 'Publish assets to AWS',
|
|
66
|
+
needs: ['synth'],
|
|
67
|
+
runsOn: ['ubuntu-latest'],
|
|
68
|
+
env: {
|
|
69
|
+
CI: 'true',
|
|
70
|
+
},
|
|
71
|
+
permissions: { idToken: workflows_model_1.JobPermission.WRITE, contents: this.needsVersionedArtifacts ? workflows_model_1.JobPermission.WRITE : workflows_model_1.JobPermission.READ },
|
|
72
|
+
steps: [{
|
|
73
|
+
name: 'Checkout',
|
|
74
|
+
uses: 'actions/checkout@v3',
|
|
75
|
+
with: {
|
|
76
|
+
'fetch-depth': 0,
|
|
77
|
+
},
|
|
78
|
+
}, {
|
|
79
|
+
name: 'Setup GIT identity',
|
|
80
|
+
run: 'git config --global user.name "github-actions" && git config --global user.email "github-actions@github.com"',
|
|
81
|
+
}, {
|
|
82
|
+
name: 'AWS Credentials',
|
|
83
|
+
uses: 'aws-actions/configure-aws-credentials@master',
|
|
84
|
+
with: {
|
|
85
|
+
'role-to-assume': (_b = (_a = this.options.iamRoleArns) === null || _a === void 0 ? void 0 : _a.assetPublishing) !== null && _b !== void 0 ? _b : (_c = this.options.iamRoleArns) === null || _c === void 0 ? void 0 : _c.default,
|
|
86
|
+
'role-session-name': 'GitHubAction',
|
|
87
|
+
'aws-region': 'us-east-1',
|
|
88
|
+
},
|
|
89
|
+
}, {
|
|
90
|
+
uses: 'actions/download-artifact@v3',
|
|
91
|
+
with: {
|
|
92
|
+
name: 'cloud-assembly',
|
|
93
|
+
path: `${this.app.cdkConfig.cdkout}/`,
|
|
94
|
+
},
|
|
95
|
+
},
|
|
96
|
+
...this.getAssetUploadCommands(this.needsVersionedArtifacts).map(cmd => ({
|
|
97
|
+
run: cmd,
|
|
98
|
+
}))],
|
|
99
|
+
});
|
|
100
|
+
}
|
|
101
|
+
createDeployment(stage) {
|
|
102
|
+
var _a, _b, _c, _d, _e, _f, _g, _h;
|
|
103
|
+
if (stage.manualApproval === true) {
|
|
104
|
+
// Create new workflow for deployment
|
|
105
|
+
const stageWorkflow = this.app.github.addWorkflow(`release-${stage.name}`);
|
|
106
|
+
stageWorkflow.on({
|
|
107
|
+
workflowDispatch: {
|
|
108
|
+
inputs: {
|
|
109
|
+
version: {
|
|
110
|
+
description: 'Package version',
|
|
111
|
+
required: true,
|
|
112
|
+
},
|
|
113
|
+
},
|
|
114
|
+
},
|
|
115
|
+
});
|
|
116
|
+
stageWorkflow.addJob('deploy', {
|
|
117
|
+
name: `Release stage ${stage.name} to AWS`,
|
|
118
|
+
runsOn: ['ubuntu-latest'],
|
|
119
|
+
env: {
|
|
120
|
+
CI: 'true',
|
|
121
|
+
},
|
|
122
|
+
permissions: { idToken: workflows_model_1.JobPermission.WRITE, contents: workflows_model_1.JobPermission.READ },
|
|
123
|
+
steps: [{
|
|
124
|
+
name: 'Checkout',
|
|
125
|
+
uses: 'actions/checkout@v3',
|
|
126
|
+
}, {
|
|
127
|
+
name: 'AWS Credentials',
|
|
128
|
+
uses: 'aws-actions/configure-aws-credentials@master',
|
|
129
|
+
with: {
|
|
130
|
+
'role-to-assume': (_c = (_b = (_a = this.options.iamRoleArns) === null || _a === void 0 ? void 0 : _a.deployment) === null || _b === void 0 ? void 0 : _b[stage.name]) !== null && _c !== void 0 ? _c : (_d = this.options.iamRoleArns) === null || _d === void 0 ? void 0 : _d.default,
|
|
131
|
+
'role-session-name': 'GitHubAction',
|
|
132
|
+
'aws-region': stage.env.region,
|
|
133
|
+
},
|
|
134
|
+
},
|
|
135
|
+
...this.getInstallCommands().map(cmd => ({
|
|
136
|
+
run: cmd,
|
|
137
|
+
})),
|
|
138
|
+
...this.getInstallPackageCommands(`${this.options.pkgNamespace}/${this.app.name}@\${{github.event.inputs.version}}`).map(cmd => ({
|
|
139
|
+
run: cmd,
|
|
140
|
+
})),
|
|
141
|
+
{
|
|
142
|
+
run: `mv ./node_modules/${this.options.pkgNamespace}/${this.app.name} ${this.app.cdkConfig.cdkout}`,
|
|
143
|
+
},
|
|
144
|
+
...this.getDeployCommands(stage.name).map(cmd => ({
|
|
145
|
+
run: cmd,
|
|
146
|
+
}))],
|
|
147
|
+
});
|
|
148
|
+
}
|
|
149
|
+
else {
|
|
150
|
+
// Add deployment to CI/CD workflow
|
|
151
|
+
this.deploymentWorkflow.addJob(`deploy-${stage.name}`, {
|
|
152
|
+
name: `Deploy stage ${stage.name} to AWS`,
|
|
153
|
+
needs: this.deploymentStages.length > 0 ? ['assetUpload', `deploy-${this.deploymentStages.at(-1)}`] : ['assetUpload'],
|
|
154
|
+
runsOn: ['ubuntu-latest'],
|
|
155
|
+
env: {
|
|
156
|
+
CI: 'true',
|
|
157
|
+
},
|
|
158
|
+
permissions: { idToken: workflows_model_1.JobPermission.WRITE, contents: workflows_model_1.JobPermission.READ },
|
|
159
|
+
steps: [{
|
|
160
|
+
name: 'Checkout',
|
|
161
|
+
uses: 'actions/checkout@v3',
|
|
162
|
+
}, {
|
|
163
|
+
name: 'AWS Credentials',
|
|
164
|
+
uses: 'aws-actions/configure-aws-credentials@master',
|
|
165
|
+
with: {
|
|
166
|
+
'role-to-assume': (_g = (_f = (_e = this.options.iamRoleArns) === null || _e === void 0 ? void 0 : _e.deployment) === null || _f === void 0 ? void 0 : _f[stage.name]) !== null && _g !== void 0 ? _g : (_h = this.options.iamRoleArns) === null || _h === void 0 ? void 0 : _h.default,
|
|
167
|
+
'role-session-name': 'GitHubAction',
|
|
168
|
+
'aws-region': stage.env.region,
|
|
169
|
+
},
|
|
170
|
+
}, {
|
|
171
|
+
uses: 'actions/download-artifact@v3',
|
|
172
|
+
with: {
|
|
173
|
+
name: 'cloud-assembly',
|
|
174
|
+
path: `${this.app.cdkConfig.cdkout}/`,
|
|
175
|
+
},
|
|
176
|
+
},
|
|
177
|
+
...this.getInstallCommands().map(cmd => ({
|
|
178
|
+
run: cmd,
|
|
179
|
+
})),
|
|
180
|
+
...this.getDeployCommands(stage.name).map(cmd => ({
|
|
181
|
+
run: cmd,
|
|
182
|
+
}))],
|
|
183
|
+
});
|
|
184
|
+
this.deploymentStages.push(stage.name);
|
|
185
|
+
}
|
|
186
|
+
}
|
|
187
|
+
}
|
|
188
|
+
exports.GithubCDKPipeline = GithubCDKPipeline;
|
|
189
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"github.js","sourceRoot":"","sources":["../../src/awscdk/github.ts"],"names":[],"mappings":";;;AAEA,uEAA2E;AAC3E,iCAA0E;AAY1E,MAAa,iBAAkB,SAAQ,kBAAW;IAOhD,YAAY,GAA+B,EAAU,OAAiC;QACpF,KAAK,CAAC,GAAG,EAAE,OAAO,CAAC,CAAC;QAD+B,YAAO,GAAP,OAAO,CAA0B;QAF9E,qBAAgB,GAAa,EAAE,CAAC;QAKtC,IAAI,CAAC,kBAAkB,GAAG,IAAI,CAAC,GAAG,CAAC,MAAO,CAAC,WAAW,CAAC,QAAQ,CAAC,CAAC;QACjE,IAAI,CAAC,kBAAkB,CAAC,EAAE,CAAC;YACzB,IAAI,EAAE;gBACJ,QAAQ,EAAE,CAAC,MAAM,CAAC,EAAE,gCAAgC;aACrD;YACD,gBAAgB,EAAE,EAAE;SACrB,CAAC,CAAC;QAEH,IAAI,CAAC,uBAAuB,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,cAAc,KAAK,IAAI,CAAC,KAAK,SAAS,CAAC;QAEtG,IAAI,CAAC,WAAW,EAAE,CAAC;QAEnB,IAAI,CAAC,iBAAiB,EAAE,CAAC;QAEzB,KAAK,MAAM,KAAK,IAAI,OAAO,CAAC,MAAM,EAAE;YAClC,IAAI,CAAC,gBAAgB,CAAC,KAAK,CAAC,CAAC;SAC9B;IACH,CAAC;IAEO,WAAW;;QACjB,MAAM,KAAK,GAAc,CAAC;gBACxB,IAAI,EAAE,UAAU;gBAChB,IAAI,EAAE,qBAAqB;aAC5B,CAAC,CAAC;QAEH,IAAI,MAAA,IAAI,CAAC,OAAO,CAAC,WAAW,0CAAE,KAAK,EAAE;YACnC,KAAK,CAAC,IAAI,CAAC;gBACT,IAAI,EAAE,iBAAiB;gBACvB,IAAI,EAAE,8CAA8C;gBACpD,IAAI,EAAE;oBACJ,gBAAgB,EAAE,IAAI,CAAC,OAAO,CAAC,WAAW,CAAC,KAAK;oBAChD,mBAAmB,EAAE,cAAc;oBACnC,YAAY,EAAE,WAAW;iBAC1B;aACF,CAAC,CAAC;SACJ;QAED,KAAK,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,gBAAgB,EAAE,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;YAChD,GAAG,EAAE,GAAG;SACT,CAAC,CAAC,CAAC,CAAC;QAEL,KAAK,CAAC,IAAI,CAAC;YACT,IAAI,EAAE,4BAA4B;YAClC,IAAI,EAAE;gBACJ,IAAI,EAAE,gBAAgB;gBACtB,IAAI,EAAE,GAAG,IAAI,CAAC,GAAG,CAAC,SAAS,CAAC,MAAM,GAAG;aACtC;SACF,CAAC,CAAC;QAEH,IAAI,CAAC,kBAAkB,CAAC,MAAM,CAAC,OAAO,EAAE;YACtC,IAAI,EAAE,uBAAuB;YAC7B,MAAM,EAAE,CAAC,eAAe,CAAC;YACzB,GAAG,EAAE;gBACH,EAAE,EAAE,MAAM;aACX;YACD,WAAW,EAAE,EAAE,OAAO,EAAE,+BAAa,CAAC,KAAK,EAAE,QAAQ,EAAE,+BAAa,CAAC,IAAI,EAAE;YAC3E,KAAK;SACN,CAAC,CAAC;IACL,CAAC;IAEM,iBAAiB;;QACtB,IAAI,CAAC,kBAAkB,CAAC,MAAM,CAAC,aAAa,EAAE;YAC5C,IAAI,EAAE,uBAAuB;YAC7B,KAAK,EAAE,CAAC,OAAO,CAAC;YAChB,MAAM,EAAE,CAAC,eAAe,CAAC;YACzB,GAAG,EAAE;gBACH,EAAE,EAAE,MAAM;aACX;YACD,WAAW,EAAE,EAAE,OAAO,EAAE,+BAAa,CAAC,KAAK,EAAE,QAAQ,EAAE,IAAI,CAAC,uBAAuB,CAAC,CAAC,CAAC,+BAAa,CAAC,KAAK,CAAC,CAAC,CAAC,+BAAa,CAAC,IAAI,EAAE;YAChI,KAAK,EAAE,CAAC;oBACN,IAAI,EAAE,UAAU;oBAChB,IAAI,EAAE,qBAAqB;oBAC3B,IAAI,EAAE;wBACJ,aAAa,EAAE,CAAC;qBACjB;iBACF,EAAE;oBACD,IAAI,EAAE,oBAAoB;oBAC1B,GAAG,EAAE,8GAA8G;iBACpH,EAAE;oBACD,IAAI,EAAE,iBAAiB;oBACvB,IAAI,EAAE,8CAA8C;oBACpD,IAAI,EAAE;wBACJ,gBAAgB,EAAE,MAAA,MAAA,IAAI,CAAC,OAAO,CAAC,WAAW,0CAAE,eAAe,mCAAI,MAAA,IAAI,CAAC,OAAO,CAAC,WAAW,0CAAE,OAAO;wBAChG,mBAAmB,EAAE,cAAc;wBACnC,YAAY,EAAE,WAAW;qBAC1B;iBACF,EAAE;oBACD,IAAI,EAAE,8BAA8B;oBACpC,IAAI,EAAE;wBACJ,IAAI,EAAE,gBAAgB;wBACtB,IAAI,EAAE,GAAG,IAAI,CAAC,GAAG,CAAC,SAAS,CAAC,MAAM,GAAG;qBACtC;iBACF;gBACD,GAAG,IAAI,CAAC,sBAAsB,CAAC,IAAI,CAAC,uBAAuB,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;oBACvE,GAAG,EAAE,GAAG;iBACT,CAAC,CAAC,CAAC;SACL,CAAC,CAAC;IACL,CAAC;IAEM,gBAAgB,CAAC,KAAsB;;QAC5C,IAAI,KAAK,CAAC,cAAc,KAAK,IAAI,EAAE;YACjC,qCAAqC;YACrC,MAAM,aAAa,GAAG,IAAI,CAAC,GAAG,CAAC,MAAO,CAAC,WAAW,CAAC,WAAW,KAAK,CAAC,IAAI,EAAE,CAAC,CAAC;YAC5E,aAAa,CAAC,EAAE,CAAC;gBACf,gBAAgB,EAAE;oBAChB,MAAM,EAAE;wBACN,OAAO,EAAE;4BACP,WAAW,EAAE,iBAAiB;4BAC9B,QAAQ,EAAE,IAAI;yBACf;qBACF;iBACF;aACF,CAAC,CAAC;YACH,aAAa,CAAC,MAAM,CAAC,QAAQ,EAAE;gBAC7B,IAAI,EAAE,iBAAiB,KAAK,CAAC,IAAI,SAAS;gBAC1C,MAAM,EAAE,CAAC,eAAe,CAAC;gBACzB,GAAG,EAAE;oBACH,EAAE,EAAE,MAAM;iBACX;gBACD,WAAW,EAAE,EAAE,OAAO,EAAE,+BAAa,CAAC,KAAK,EAAE,QAAQ,EAAE,+BAAa,CAAC,IAAI,EAAE;gBAC3E,KAAK,EAAE,CAAC;wBACN,IAAI,EAAE,UAAU;wBAChB,IAAI,EAAE,qBAAqB;qBAC5B,EAAE;wBACD,IAAI,EAAE,iBAAiB;wBACvB,IAAI,EAAE,8CAA8C;wBACpD,IAAI,EAAE;4BACJ,gBAAgB,EAAE,MAAA,MAAA,MAAA,IAAI,CAAC,OAAO,CAAC,WAAW,0CAAE,UAAU,0CAAG,KAAK,CAAC,IAAI,CAAC,mCAAI,MAAA,IAAI,CAAC,OAAO,CAAC,WAAW,0CAAE,OAAO;4BACzG,mBAAmB,EAAE,cAAc;4BACnC,YAAY,EAAE,KAAK,CAAC,GAAG,CAAC,MAAM;yBAC/B;qBACF;oBACD,GAAG,IAAI,CAAC,kBAAkB,EAAE,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;wBACvC,GAAG,EAAE,GAAG;qBACT,CAAC,CAAC;oBACH,GAAG,IAAI,CAAC,yBAAyB,CAAC,GAAG,IAAI,CAAC,OAAO,CAAC,YAAY,IAAI,IAAI,CAAC,GAAG,CAAC,IAAI,oCAAoC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;wBAC/H,GAAG,EAAE,GAAG;qBACT,CAAC,CAAC;oBACH;wBACE,GAAG,EAAE,qBAAqB,IAAI,CAAC,OAAO,CAAC,YAAY,IAAI,IAAI,CAAC,GAAG,CAAC,IAAI,IAAI,IAAI,CAAC,GAAG,CAAC,SAAS,CAAC,MAAM,EAAE;qBACpG;oBACD,GAAG,IAAI,CAAC,iBAAiB,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;wBAChD,GAAG,EAAE,GAAG;qBACT,CAAC,CAAC,CAAC;aACL,CAAC,CAAC;SAEJ;aAAM;YACL,mCAAmC;YACnC,IAAI,CAAC,kBAAkB,CAAC,MAAM,CAAC,UAAU,KAAK,CAAC,IAAI,EAAE,EAAE;gBACrD,IAAI,EAAE,gBAAgB,KAAK,CAAC,IAAI,SAAS;gBACzC,KAAK,EAAE,IAAI,CAAC,gBAAgB,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,aAAa,EAAE,UAAU,IAAI,CAAC,gBAAgB,CAAC,EAAE,CAAC,CAAC,CAAC,CAAE,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,aAAa,CAAC;gBACtH,MAAM,EAAE,CAAC,eAAe,CAAC;gBACzB,GAAG,EAAE;oBACH,EAAE,EAAE,MAAM;iBACX;gBACD,WAAW,EAAE,EAAE,OAAO,EAAE,+BAAa,CAAC,KAAK,EAAE,QAAQ,EAAE,+BAAa,CAAC,IAAI,EAAE;gBAC3E,KAAK,EAAE,CAAC;wBACN,IAAI,EAAE,UAAU;wBAChB,IAAI,EAAE,qBAAqB;qBAC5B,EAAE;wBACD,IAAI,EAAE,iBAAiB;wBACvB,IAAI,EAAE,8CAA8C;wBACpD,IAAI,EAAE;4BACJ,gBAAgB,EAAE,MAAA,MAAA,MAAA,IAAI,CAAC,OAAO,CAAC,WAAW,0CAAE,UAAU,0CAAG,KAAK,CAAC,IAAI,CAAC,mCAAI,MAAA,IAAI,CAAC,OAAO,CAAC,WAAW,0CAAE,OAAO;4BACzG,mBAAmB,EAAE,cAAc;4BACnC,YAAY,EAAE,KAAK,CAAC,GAAG,CAAC,MAAM;yBAC/B;qBACF,EAAE;wBACD,IAAI,EAAE,8BAA8B;wBACpC,IAAI,EAAE;4BACJ,IAAI,EAAE,gBAAgB;4BACtB,IAAI,EAAE,GAAG,IAAI,CAAC,GAAG,CAAC,SAAS,CAAC,MAAM,GAAG;yBACtC;qBACF;oBACD,GAAG,IAAI,CAAC,kBAAkB,EAAE,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;wBACvC,GAAG,EAAE,GAAG;qBACT,CAAC,CAAC;oBACH,GAAG,IAAI,CAAC,iBAAiB,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;wBAChD,GAAG,EAAE,GAAG;qBACT,CAAC,CAAC,CAAC;aACL,CAAC,CAAC;YACH,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;SACxC;IACH,CAAC;CACF;AAlMD,8CAkMC","sourcesContent":["import { awscdk } from 'projen';\nimport { GithubWorkflow } from 'projen/lib/github';\nimport { JobPermission, JobStep } from 'projen/lib/github/workflows-model';\nimport { CDKPipeline, CDKPipelineOptions, DeploymentStage } from './base';\n\n\nexport interface GithubCDKPipelineOptions extends CDKPipelineOptions {\n  readonly iamRoleArns: {\n    readonly default?: string;\n    readonly synth?: string;\n    readonly assetPublishing?: string;\n    readonly deployment?: { [stage: string]: string };\n  };\n}\n\nexport class GithubCDKPipeline extends CDKPipeline {\n\n  public readonly needsVersionedArtifacts: boolean;\n\n  private deploymentWorkflow: GithubWorkflow;\n  private deploymentStages: string[] = [];\n\n  constructor(app: awscdk.AwsCdkTypeScriptApp, private options: GithubCDKPipelineOptions) {\n    super(app, options);\n\n    this.deploymentWorkflow = this.app.github!.addWorkflow('deploy');\n    this.deploymentWorkflow.on({\n      push: {\n        branches: ['main'], // TODO use defaultReleaseBranch\n      },\n      workflowDispatch: {},\n    });\n\n    this.needsVersionedArtifacts = this.options.stages.find(s => s.manualApproval === true) !== undefined;\n\n    this.createSynth();\n\n    this.createAssetUpload();\n\n    for (const stage of options.stages) {\n      this.createDeployment(stage);\n    }\n  }\n\n  private createSynth(): void {\n    const steps: JobStep[] = [{\n      name: 'Checkout',\n      uses: 'actions/checkout@v3',\n    }];\n\n    if (this.options.iamRoleArns?.synth) {\n      steps.push({\n        name: 'AWS Credentials',\n        uses: 'aws-actions/configure-aws-credentials@master',\n        with: {\n          'role-to-assume': this.options.iamRoleArns.synth,\n          'role-session-name': 'GitHubAction',\n          'aws-region': 'us-east-1',\n        },\n      });\n    }\n\n    steps.push(...this.getSynthCommands().map(cmd => ({\n      run: cmd,\n    })));\n\n    steps.push({\n      uses: 'actions/upload-artifact@v3',\n      with: {\n        name: 'cloud-assembly',\n        path: `${this.app.cdkConfig.cdkout}/`,\n      },\n    });\n\n    this.deploymentWorkflow.addJob('synth', {\n      name: 'Synth CDK application',\n      runsOn: ['ubuntu-latest'],\n      env: {\n        CI: 'true',\n      },\n      permissions: { idToken: JobPermission.WRITE, contents: JobPermission.READ },\n      steps,\n    });\n  }\n\n  public createAssetUpload(): void {\n    this.deploymentWorkflow.addJob('assetUpload', {\n      name: 'Publish assets to AWS',\n      needs: ['synth'],\n      runsOn: ['ubuntu-latest'],\n      env: {\n        CI: 'true',\n      },\n      permissions: { idToken: JobPermission.WRITE, contents: this.needsVersionedArtifacts ? JobPermission.WRITE : JobPermission.READ },\n      steps: [{\n        name: 'Checkout',\n        uses: 'actions/checkout@v3',\n        with: {\n          'fetch-depth': 0,\n        },\n      }, {\n        name: 'Setup GIT identity',\n        run: 'git config --global user.name \"github-actions\" && git config --global user.email \"github-actions@github.com\"',\n      }, {\n        name: 'AWS Credentials',\n        uses: 'aws-actions/configure-aws-credentials@master',\n        with: {\n          'role-to-assume': this.options.iamRoleArns?.assetPublishing ?? this.options.iamRoleArns?.default,\n          'role-session-name': 'GitHubAction',\n          'aws-region': 'us-east-1',\n        },\n      }, {\n        uses: 'actions/download-artifact@v3',\n        with: {\n          name: 'cloud-assembly',\n          path: `${this.app.cdkConfig.cdkout}/`,\n        },\n      },\n      ...this.getAssetUploadCommands(this.needsVersionedArtifacts).map(cmd => ({\n        run: cmd,\n      }))],\n    });\n  }\n\n  public createDeployment(stage: DeploymentStage): void {\n    if (stage.manualApproval === true) {\n      // Create new workflow for deployment\n      const stageWorkflow = this.app.github!.addWorkflow(`release-${stage.name}`);\n      stageWorkflow.on({\n        workflowDispatch: {\n          inputs: {\n            version: {\n              description: 'Package version',\n              required: true,\n            },\n          },\n        },\n      });\n      stageWorkflow.addJob('deploy', {\n        name: `Release stage ${stage.name} to AWS`,\n        runsOn: ['ubuntu-latest'],\n        env: {\n          CI: 'true',\n        },\n        permissions: { idToken: JobPermission.WRITE, contents: JobPermission.READ },\n        steps: [{\n          name: 'Checkout',\n          uses: 'actions/checkout@v3',\n        }, {\n          name: 'AWS Credentials',\n          uses: 'aws-actions/configure-aws-credentials@master',\n          with: {\n            'role-to-assume': this.options.iamRoleArns?.deployment?.[stage.name] ?? this.options.iamRoleArns?.default,\n            'role-session-name': 'GitHubAction',\n            'aws-region': stage.env.region,\n          },\n        },\n        ...this.getInstallCommands().map(cmd => ({\n          run: cmd,\n        })),\n        ...this.getInstallPackageCommands(`${this.options.pkgNamespace}/${this.app.name}@\\${{github.event.inputs.version}}`).map(cmd => ({\n          run: cmd,\n        })),\n        {\n          run: `mv ./node_modules/${this.options.pkgNamespace}/${this.app.name} ${this.app.cdkConfig.cdkout}`,\n        },\n        ...this.getDeployCommands(stage.name).map(cmd => ({\n          run: cmd,\n        }))],\n      });\n\n    } else {\n      // Add deployment to CI/CD workflow\n      this.deploymentWorkflow.addJob(`deploy-${stage.name}`, {\n        name: `Deploy stage ${stage.name} to AWS`,\n        needs: this.deploymentStages.length > 0 ? ['assetUpload', `deploy-${this.deploymentStages.at(-1)!}`] : ['assetUpload'],\n        runsOn: ['ubuntu-latest'],\n        env: {\n          CI: 'true',\n        },\n        permissions: { idToken: JobPermission.WRITE, contents: JobPermission.READ },\n        steps: [{\n          name: 'Checkout',\n          uses: 'actions/checkout@v3',\n        }, {\n          name: 'AWS Credentials',\n          uses: 'aws-actions/configure-aws-credentials@master',\n          with: {\n            'role-to-assume': this.options.iamRoleArns?.deployment?.[stage.name] ?? this.options.iamRoleArns?.default,\n            'role-session-name': 'GitHubAction',\n            'aws-region': stage.env.region,\n          },\n        }, {\n          uses: 'actions/download-artifact@v3',\n          with: {\n            name: 'cloud-assembly',\n            path: `${this.app.cdkConfig.cdkout}/`,\n          },\n        },\n        ...this.getInstallCommands().map(cmd => ({\n          run: cmd,\n        })),\n        ...this.getDeployCommands(stage.name).map(cmd => ({\n          run: cmd,\n        }))],\n      });\n      this.deploymentStages.push(stage.name);\n    }\n  }\n}\n"]}
|
|
@@ -16,4 +16,4 @@ var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
|
16
16
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
17
17
|
__exportStar(require("./base"), exports);
|
|
18
18
|
__exportStar(require("./github"), exports);
|
|
19
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
19
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvYXdzY2RrL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQSx5Q0FBdUI7QUFDdkIsMkNBQXlCIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9iYXNlJztcbmV4cG9ydCAqIGZyb20gJy4vZ2l0aHViJzsiXX0=
|
package/lib/index.d.ts
CHANGED
|
@@ -1,2 +1 @@
|
|
|
1
|
-
export * from './
|
|
2
|
-
export * from './engine';
|
|
1
|
+
export * from './awscdk';
|
package/lib/index.js
CHANGED
|
@@ -14,6 +14,5 @@ var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
|
14
14
|
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
|
|
15
15
|
};
|
|
16
16
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
17
|
-
__exportStar(require("./
|
|
18
|
-
|
|
19
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7OztBQUFBLDZDQUEyQjtBQUMzQiwyQ0FBeUIiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL3BpcGVsaW5lJztcbmV4cG9ydCAqIGZyb20gJy4vZW5naW5lJzsiXX0=
|
|
17
|
+
__exportStar(require("./awscdk"), exports);
|
|
18
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7OztBQUFBLDJDQUF5QiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vYXdzY2RrJzsiXX0=
|
package/package.json
CHANGED
package/lib/engine/base.d.ts
DELETED
|
@@ -1,23 +0,0 @@
|
|
|
1
|
-
import { awscdk } from 'projen';
|
|
2
|
-
import { CDKPipeline, CDKPipelineOptions, DeploymentStage } from '../pipeline';
|
|
3
|
-
export interface SynthStageOptions {
|
|
4
|
-
readonly commands: string[];
|
|
5
|
-
}
|
|
6
|
-
export interface AssetUploadStageOptions {
|
|
7
|
-
readonly commands: string[];
|
|
8
|
-
}
|
|
9
|
-
export interface DeployStageOptions {
|
|
10
|
-
readonly config: DeploymentStage;
|
|
11
|
-
readonly installCommands: string[];
|
|
12
|
-
readonly deployCommands: string[];
|
|
13
|
-
}
|
|
14
|
-
export declare abstract class BaseEngine {
|
|
15
|
-
protected app: awscdk.AwsCdkTypeScriptApp;
|
|
16
|
-
protected props: CDKPipelineOptions;
|
|
17
|
-
protected pipeline: CDKPipeline;
|
|
18
|
-
abstract readonly needsVersionedArtifacts: boolean;
|
|
19
|
-
constructor(app: awscdk.AwsCdkTypeScriptApp, props: CDKPipelineOptions, pipeline: CDKPipeline);
|
|
20
|
-
abstract createSynth(options: SynthStageOptions): void;
|
|
21
|
-
abstract createAssetUpload(options: AssetUploadStageOptions): void;
|
|
22
|
-
abstract createDeployment(options: DeployStageOptions): void;
|
|
23
|
-
}
|
package/lib/engine/base.js
DELETED
|
@@ -1,13 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.BaseEngine = void 0;
|
|
4
|
-
class BaseEngine {
|
|
5
|
-
constructor(app, props, pipeline) {
|
|
6
|
-
this.app = app;
|
|
7
|
-
this.props = props;
|
|
8
|
-
this.pipeline = pipeline;
|
|
9
|
-
//
|
|
10
|
-
}
|
|
11
|
-
}
|
|
12
|
-
exports.BaseEngine = BaseEngine;
|
|
13
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmFzZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9lbmdpbmUvYmFzZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFpQkEsTUFBc0IsVUFBVTtJQUk5QixZQUFzQixHQUErQixFQUFZLEtBQXlCLEVBQVksUUFBcUI7UUFBckcsUUFBRyxHQUFILEdBQUcsQ0FBNEI7UUFBWSxVQUFLLEdBQUwsS0FBSyxDQUFvQjtRQUFZLGFBQVEsR0FBUixRQUFRLENBQWE7UUFDekgsRUFBRTtJQUNKLENBQUM7Q0FNRjtBQVpELGdDQVlDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgYXdzY2RrIH0gZnJvbSAncHJvamVuJztcbmltcG9ydCB7IENES1BpcGVsaW5lLCBDREtQaXBlbGluZU9wdGlvbnMsIERlcGxveW1lbnRTdGFnZSB9IGZyb20gJy4uL3BpcGVsaW5lJztcblxuZXhwb3J0IGludGVyZmFjZSBTeW50aFN0YWdlT3B0aW9ucyB7XG4gIHJlYWRvbmx5IGNvbW1hbmRzOiBzdHJpbmdbXTtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBBc3NldFVwbG9hZFN0YWdlT3B0aW9ucyB7XG4gIHJlYWRvbmx5IGNvbW1hbmRzOiBzdHJpbmdbXTtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBEZXBsb3lTdGFnZU9wdGlvbnMge1xuICByZWFkb25seSBjb25maWc6IERlcGxveW1lbnRTdGFnZTtcbiAgcmVhZG9ubHkgaW5zdGFsbENvbW1hbmRzOiBzdHJpbmdbXTtcbiAgcmVhZG9ubHkgZGVwbG95Q29tbWFuZHM6IHN0cmluZ1tdO1xufVxuXG5leHBvcnQgYWJzdHJhY3QgY2xhc3MgQmFzZUVuZ2luZSB7XG5cbiAgcHVibGljIGFic3RyYWN0IHJlYWRvbmx5IG5lZWRzVmVyc2lvbmVkQXJ0aWZhY3RzOiBib29sZWFuO1xuXG4gIGNvbnN0cnVjdG9yKHByb3RlY3RlZCBhcHA6IGF3c2Nkay5Bd3NDZGtUeXBlU2NyaXB0QXBwLCBwcm90ZWN0ZWQgcHJvcHM6IENES1BpcGVsaW5lT3B0aW9ucywgcHJvdGVjdGVkIHBpcGVsaW5lOiBDREtQaXBlbGluZSkge1xuICAgIC8vXG4gIH1cblxuICBhYnN0cmFjdCBjcmVhdGVTeW50aChvcHRpb25zOiBTeW50aFN0YWdlT3B0aW9ucyk6IHZvaWQ7XG4gIGFic3RyYWN0IGNyZWF0ZUFzc2V0VXBsb2FkKG9wdGlvbnM6IEFzc2V0VXBsb2FkU3RhZ2VPcHRpb25zKTogdm9pZDtcbiAgYWJzdHJhY3QgY3JlYXRlRGVwbG95bWVudChvcHRpb25zOiBEZXBsb3lTdGFnZU9wdGlvbnMpOiB2b2lkO1xuXG59Il19
|
package/lib/engine/github.d.ts
DELETED
|
@@ -1,20 +0,0 @@
|
|
|
1
|
-
import { awscdk } from 'projen';
|
|
2
|
-
import { AssetUploadStageOptions, BaseEngine, DeployStageOptions, SynthStageOptions } from './base';
|
|
3
|
-
import { CDKPipeline, CDKPipelineOptions } from '../pipeline';
|
|
4
|
-
export interface GithubEngineConfig {
|
|
5
|
-
readonly defaultAwsRoleArn?: string;
|
|
6
|
-
readonly awsRoleArnForSynth?: string;
|
|
7
|
-
readonly awsRoleArnForAssetPublishing?: string;
|
|
8
|
-
readonly awsRoleArnForDeployment?: {
|
|
9
|
-
[stage: string]: string;
|
|
10
|
-
};
|
|
11
|
-
}
|
|
12
|
-
export declare class GitHubEngine extends BaseEngine {
|
|
13
|
-
readonly needsVersionedArtifacts: boolean;
|
|
14
|
-
private deploymentWorkflow;
|
|
15
|
-
private deploymentStages;
|
|
16
|
-
constructor(app: awscdk.AwsCdkTypeScriptApp, props: CDKPipelineOptions, pipeline: CDKPipeline);
|
|
17
|
-
createSynth(options: SynthStageOptions): void;
|
|
18
|
-
createAssetUpload(options: AssetUploadStageOptions): void;
|
|
19
|
-
createDeployment(options: DeployStageOptions): void;
|
|
20
|
-
}
|
package/lib/engine/github.js
DELETED
|
@@ -1,180 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.GitHubEngine = void 0;
|
|
4
|
-
const workflows_model_1 = require("projen/lib/github/workflows-model");
|
|
5
|
-
const base_1 = require("./base");
|
|
6
|
-
class GitHubEngine extends base_1.BaseEngine {
|
|
7
|
-
constructor(app, props, pipeline) {
|
|
8
|
-
super(app, props, pipeline);
|
|
9
|
-
this.deploymentStages = [];
|
|
10
|
-
this.deploymentWorkflow = this.app.github.addWorkflow('deploy');
|
|
11
|
-
this.deploymentWorkflow.on({
|
|
12
|
-
push: {
|
|
13
|
-
branches: ['main'], // TODO use defaultReleaseBranch
|
|
14
|
-
},
|
|
15
|
-
workflowDispatch: {},
|
|
16
|
-
});
|
|
17
|
-
this.needsVersionedArtifacts = this.props.stages.find(s => s.manualApproval === true) !== undefined;
|
|
18
|
-
}
|
|
19
|
-
createSynth(options) {
|
|
20
|
-
var _a;
|
|
21
|
-
const steps = [{
|
|
22
|
-
name: 'Checkout',
|
|
23
|
-
uses: 'actions/checkout@v3',
|
|
24
|
-
}];
|
|
25
|
-
if ((_a = this.props.githubConfig) === null || _a === void 0 ? void 0 : _a.awsRoleArnForSynth) {
|
|
26
|
-
steps.push({
|
|
27
|
-
name: 'AWS Credentials',
|
|
28
|
-
uses: 'aws-actions/configure-aws-credentials@master',
|
|
29
|
-
with: {
|
|
30
|
-
'role-to-assume': this.props.githubConfig.awsRoleArnForSynth,
|
|
31
|
-
'role-session-name': 'GitHubAction',
|
|
32
|
-
'aws-region': 'us-east-1',
|
|
33
|
-
},
|
|
34
|
-
});
|
|
35
|
-
}
|
|
36
|
-
steps.push(...options.commands.map(cmd => ({
|
|
37
|
-
run: cmd,
|
|
38
|
-
})));
|
|
39
|
-
steps.push({
|
|
40
|
-
uses: 'actions/upload-artifact@v3',
|
|
41
|
-
with: {
|
|
42
|
-
name: 'cloud-assembly',
|
|
43
|
-
path: `${this.app.cdkConfig.cdkout}/`,
|
|
44
|
-
},
|
|
45
|
-
});
|
|
46
|
-
this.deploymentWorkflow.addJob('synth', {
|
|
47
|
-
name: 'Synth CDK application',
|
|
48
|
-
runsOn: ['ubuntu-latest'],
|
|
49
|
-
env: {
|
|
50
|
-
CI: 'true',
|
|
51
|
-
},
|
|
52
|
-
permissions: { idToken: workflows_model_1.JobPermission.WRITE, contents: workflows_model_1.JobPermission.READ },
|
|
53
|
-
steps,
|
|
54
|
-
});
|
|
55
|
-
}
|
|
56
|
-
createAssetUpload(options) {
|
|
57
|
-
var _a, _b, _c;
|
|
58
|
-
this.deploymentWorkflow.addJob('assetUpload', {
|
|
59
|
-
name: 'Publish assets to AWS',
|
|
60
|
-
needs: ['synth'],
|
|
61
|
-
runsOn: ['ubuntu-latest'],
|
|
62
|
-
env: {
|
|
63
|
-
CI: 'true',
|
|
64
|
-
},
|
|
65
|
-
permissions: { idToken: workflows_model_1.JobPermission.WRITE, contents: this.needsVersionedArtifacts ? workflows_model_1.JobPermission.WRITE : workflows_model_1.JobPermission.READ },
|
|
66
|
-
steps: [{
|
|
67
|
-
name: 'Checkout',
|
|
68
|
-
uses: 'actions/checkout@v3',
|
|
69
|
-
with: {
|
|
70
|
-
'fetch-depth': 0,
|
|
71
|
-
},
|
|
72
|
-
}, {
|
|
73
|
-
name: 'Setup GIT identity',
|
|
74
|
-
run: 'git config --global user.name "projen pipeline" && git config --global user.email "info@taimos.de"',
|
|
75
|
-
}, {
|
|
76
|
-
name: 'AWS Credentials',
|
|
77
|
-
uses: 'aws-actions/configure-aws-credentials@master',
|
|
78
|
-
with: {
|
|
79
|
-
'role-to-assume': (_b = (_a = this.props.githubConfig) === null || _a === void 0 ? void 0 : _a.awsRoleArnForAssetPublishing) !== null && _b !== void 0 ? _b : (_c = this.props.githubConfig) === null || _c === void 0 ? void 0 : _c.defaultAwsRoleArn,
|
|
80
|
-
'role-session-name': 'GitHubAction',
|
|
81
|
-
'aws-region': 'us-east-1',
|
|
82
|
-
},
|
|
83
|
-
}, {
|
|
84
|
-
uses: 'actions/download-artifact@v3',
|
|
85
|
-
with: {
|
|
86
|
-
name: 'cloud-assembly',
|
|
87
|
-
path: `${this.app.cdkConfig.cdkout}/`,
|
|
88
|
-
},
|
|
89
|
-
},
|
|
90
|
-
...options.commands.map(cmd => ({
|
|
91
|
-
run: cmd,
|
|
92
|
-
}))],
|
|
93
|
-
});
|
|
94
|
-
}
|
|
95
|
-
createDeployment(options) {
|
|
96
|
-
var _a, _b, _c, _d, _e, _f, _g, _h;
|
|
97
|
-
if (options.config.manualApproval === true) {
|
|
98
|
-
// Create new workflow for deployment
|
|
99
|
-
const stageWorkflow = this.app.github.addWorkflow(`release-${options.config.name}`);
|
|
100
|
-
stageWorkflow.on({
|
|
101
|
-
workflowDispatch: {
|
|
102
|
-
inputs: {
|
|
103
|
-
version: {
|
|
104
|
-
description: 'Package version',
|
|
105
|
-
required: true,
|
|
106
|
-
},
|
|
107
|
-
},
|
|
108
|
-
},
|
|
109
|
-
});
|
|
110
|
-
stageWorkflow.addJob('deploy', {
|
|
111
|
-
name: `Release stage ${options.config.name} to AWS`,
|
|
112
|
-
runsOn: ['ubuntu-latest'],
|
|
113
|
-
env: {
|
|
114
|
-
CI: 'true',
|
|
115
|
-
},
|
|
116
|
-
permissions: { idToken: workflows_model_1.JobPermission.WRITE, contents: workflows_model_1.JobPermission.READ },
|
|
117
|
-
steps: [{
|
|
118
|
-
name: 'Checkout',
|
|
119
|
-
uses: 'actions/checkout@v3',
|
|
120
|
-
}, {
|
|
121
|
-
name: 'AWS Credentials',
|
|
122
|
-
uses: 'aws-actions/configure-aws-credentials@master',
|
|
123
|
-
with: {
|
|
124
|
-
'role-to-assume': (_c = (_b = (_a = this.props.githubConfig) === null || _a === void 0 ? void 0 : _a.awsRoleArnForDeployment) === null || _b === void 0 ? void 0 : _b[options.config.name]) !== null && _c !== void 0 ? _c : (_d = this.props.githubConfig) === null || _d === void 0 ? void 0 : _d.defaultAwsRoleArn,
|
|
125
|
-
'role-session-name': 'GitHubAction',
|
|
126
|
-
'aws-region': options.config.env.region,
|
|
127
|
-
},
|
|
128
|
-
},
|
|
129
|
-
...options.installCommands.map(cmd => ({
|
|
130
|
-
run: cmd,
|
|
131
|
-
})),
|
|
132
|
-
{
|
|
133
|
-
run: `yarn add ${this.props.pkgNamespace}/${this.app.name}@\${{github.event.inputs.version}} && mv ./node_modules/${this.props.pkgNamespace}/${this.app.name} ${this.app.cdkConfig.cdkout}`,
|
|
134
|
-
},
|
|
135
|
-
...options.deployCommands.map(cmd => ({
|
|
136
|
-
run: cmd,
|
|
137
|
-
}))],
|
|
138
|
-
});
|
|
139
|
-
}
|
|
140
|
-
else {
|
|
141
|
-
// Add deployment to CI/CD workflow
|
|
142
|
-
this.deploymentWorkflow.addJob(`deploy-${options.config.name}`, {
|
|
143
|
-
name: `Deploy stage ${options.config.name} to AWS`,
|
|
144
|
-
needs: this.deploymentStages.length > 0 ? ['assetUpload', `deploy-${this.deploymentStages.at(-1)}`] : ['assetUpload'],
|
|
145
|
-
runsOn: ['ubuntu-latest'],
|
|
146
|
-
env: {
|
|
147
|
-
CI: 'true',
|
|
148
|
-
},
|
|
149
|
-
permissions: { idToken: workflows_model_1.JobPermission.WRITE, contents: workflows_model_1.JobPermission.READ },
|
|
150
|
-
steps: [{
|
|
151
|
-
name: 'Checkout',
|
|
152
|
-
uses: 'actions/checkout@v3',
|
|
153
|
-
}, {
|
|
154
|
-
name: 'AWS Credentials',
|
|
155
|
-
uses: 'aws-actions/configure-aws-credentials@master',
|
|
156
|
-
with: {
|
|
157
|
-
'role-to-assume': (_g = (_f = (_e = this.props.githubConfig) === null || _e === void 0 ? void 0 : _e.awsRoleArnForDeployment) === null || _f === void 0 ? void 0 : _f[options.config.name]) !== null && _g !== void 0 ? _g : (_h = this.props.githubConfig) === null || _h === void 0 ? void 0 : _h.defaultAwsRoleArn,
|
|
158
|
-
'role-session-name': 'GitHubAction',
|
|
159
|
-
'aws-region': options.config.env.region,
|
|
160
|
-
},
|
|
161
|
-
}, {
|
|
162
|
-
uses: 'actions/download-artifact@v3',
|
|
163
|
-
with: {
|
|
164
|
-
name: 'cloud-assembly',
|
|
165
|
-
path: `${this.app.cdkConfig.cdkout}/`,
|
|
166
|
-
},
|
|
167
|
-
},
|
|
168
|
-
...options.installCommands.map(cmd => ({
|
|
169
|
-
run: cmd,
|
|
170
|
-
})),
|
|
171
|
-
...options.deployCommands.map(cmd => ({
|
|
172
|
-
run: cmd,
|
|
173
|
-
}))],
|
|
174
|
-
});
|
|
175
|
-
this.deploymentStages.push(options.config.name);
|
|
176
|
-
}
|
|
177
|
-
}
|
|
178
|
-
}
|
|
179
|
-
exports.GitHubEngine = GitHubEngine;
|
|
180
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"github.js","sourceRoot":"","sources":["../../src/engine/github.ts"],"names":[],"mappings":"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","sourcesContent":["import { awscdk } from 'projen';\nimport { GithubWorkflow } from 'projen/lib/github';\nimport { JobPermission, JobStep } from 'projen/lib/github/workflows-model';\nimport { AssetUploadStageOptions, BaseEngine, DeployStageOptions, SynthStageOptions } from './base';\nimport { CDKPipeline, CDKPipelineOptions } from '../pipeline';\n\nexport interface GithubEngineConfig {\n  readonly defaultAwsRoleArn?: string;\n  readonly awsRoleArnForSynth?: string;\n  readonly awsRoleArnForAssetPublishing?: string;\n  readonly awsRoleArnForDeployment?: { [stage: string]: string };\n}\n\nexport class GitHubEngine extends BaseEngine {\n\n  public readonly needsVersionedArtifacts: boolean;\n\n  private deploymentWorkflow: GithubWorkflow;\n  private deploymentStages: string[] = [];\n\n  constructor(app: awscdk.AwsCdkTypeScriptApp, props: CDKPipelineOptions, pipeline: CDKPipeline) {\n    super(app, props, pipeline);\n\n    this.deploymentWorkflow = this.app.github!.addWorkflow('deploy');\n    this.deploymentWorkflow.on({\n      push: {\n        branches: ['main'], // TODO use defaultReleaseBranch\n      },\n      workflowDispatch: {},\n    });\n\n    this.needsVersionedArtifacts = this.props.stages.find(s => s.manualApproval === true) !== undefined;\n  }\n\n  public createSynth(options: SynthStageOptions): void {\n    const steps: JobStep[] = [{\n      name: 'Checkout',\n      uses: 'actions/checkout@v3',\n    }];\n\n    if (this.props.githubConfig?.awsRoleArnForSynth) {\n      steps.push({\n        name: 'AWS Credentials',\n        uses: 'aws-actions/configure-aws-credentials@master',\n        with: {\n          'role-to-assume': this.props.githubConfig.awsRoleArnForSynth,\n          'role-session-name': 'GitHubAction',\n          'aws-region': 'us-east-1',\n        },\n      });\n    }\n\n    steps.push(...options.commands.map(cmd => ({\n      run: cmd,\n    })));\n\n    steps.push({\n      uses: 'actions/upload-artifact@v3',\n      with: {\n        name: 'cloud-assembly',\n        path: `${this.app.cdkConfig.cdkout}/`,\n      },\n    });\n\n    this.deploymentWorkflow.addJob('synth', {\n      name: 'Synth CDK application',\n      runsOn: ['ubuntu-latest'],\n      env: {\n        CI: 'true',\n      },\n      permissions: { idToken: JobPermission.WRITE, contents: JobPermission.READ },\n      steps,\n    });\n  }\n\n  public createAssetUpload(options: AssetUploadStageOptions): void {\n    this.deploymentWorkflow.addJob('assetUpload', {\n      name: 'Publish assets to AWS',\n      needs: ['synth'],\n      runsOn: ['ubuntu-latest'],\n      env: {\n        CI: 'true',\n      },\n      permissions: { idToken: JobPermission.WRITE, contents: this.needsVersionedArtifacts ? JobPermission.WRITE : JobPermission.READ },\n      steps: [{\n        name: 'Checkout',\n        uses: 'actions/checkout@v3',\n        with: {\n          'fetch-depth': 0,\n        },\n      }, {\n        name: 'Setup GIT identity',\n        run: 'git config --global user.name \"projen pipeline\" && git config --global user.email \"info@taimos.de\"',\n      }, {\n        name: 'AWS Credentials',\n        uses: 'aws-actions/configure-aws-credentials@master',\n        with: {\n          'role-to-assume': this.props.githubConfig?.awsRoleArnForAssetPublishing ?? this.props.githubConfig?.defaultAwsRoleArn,\n          'role-session-name': 'GitHubAction',\n          'aws-region': 'us-east-1',\n        },\n      }, {\n        uses: 'actions/download-artifact@v3',\n        with: {\n          name: 'cloud-assembly',\n          path: `${this.app.cdkConfig.cdkout}/`,\n        },\n      },\n      ...options.commands.map(cmd => ({\n        run: cmd,\n      }))],\n    });\n  }\n\n  public createDeployment(options: DeployStageOptions): void {\n    if (options.config.manualApproval === true) {\n      // Create new workflow for deployment\n      const stageWorkflow = this.app.github!.addWorkflow(`release-${options.config.name}`);\n      stageWorkflow.on({\n        workflowDispatch: {\n          inputs: {\n            version: {\n              description: 'Package version',\n              required: true,\n            },\n          },\n        },\n      });\n      stageWorkflow.addJob('deploy', {\n        name: `Release stage ${options.config.name} to AWS`,\n        runsOn: ['ubuntu-latest'],\n        env: {\n          CI: 'true',\n        },\n        permissions: { idToken: JobPermission.WRITE, contents: JobPermission.READ },\n        steps: [{\n          name: 'Checkout',\n          uses: 'actions/checkout@v3',\n        }, {\n          name: 'AWS Credentials',\n          uses: 'aws-actions/configure-aws-credentials@master',\n          with: {\n            'role-to-assume': this.props.githubConfig?.awsRoleArnForDeployment?.[options.config.name] ?? this.props.githubConfig?.defaultAwsRoleArn,\n            'role-session-name': 'GitHubAction',\n            'aws-region': options.config.env.region,\n          },\n        },\n        ...options.installCommands.map(cmd => ({\n          run: cmd,\n        })),\n        {\n          run: `yarn add ${this.props.pkgNamespace}/${this.app.name}@\\${{github.event.inputs.version}} && mv ./node_modules/${this.props.pkgNamespace}/${this.app.name} ${this.app.cdkConfig.cdkout}`,\n        },\n        ...options.deployCommands.map(cmd => ({\n          run: cmd,\n        }))],\n      });\n\n    } else {\n      // Add deployment to CI/CD workflow\n      this.deploymentWorkflow.addJob(`deploy-${options.config.name}`, {\n        name: `Deploy stage ${options.config.name} to AWS`,\n        needs: this.deploymentStages.length > 0 ? ['assetUpload', `deploy-${this.deploymentStages.at(-1)!}`] : ['assetUpload'],\n        runsOn: ['ubuntu-latest'],\n        env: {\n          CI: 'true',\n        },\n        permissions: { idToken: JobPermission.WRITE, contents: JobPermission.READ },\n        steps: [{\n          name: 'Checkout',\n          uses: 'actions/checkout@v3',\n        }, {\n          name: 'AWS Credentials',\n          uses: 'aws-actions/configure-aws-credentials@master',\n          with: {\n            'role-to-assume': this.props.githubConfig?.awsRoleArnForDeployment?.[options.config.name] ?? this.props.githubConfig?.defaultAwsRoleArn,\n            'role-session-name': 'GitHubAction',\n            'aws-region': options.config.env.region,\n          },\n        }, {\n          uses: 'actions/download-artifact@v3',\n          with: {\n            name: 'cloud-assembly',\n            path: `${this.app.cdkConfig.cdkout}/`,\n          },\n        },\n        ...options.installCommands.map(cmd => ({\n          run: cmd,\n        })),\n        ...options.deployCommands.map(cmd => ({\n          run: cmd,\n        }))],\n      });\n      this.deploymentStages.push(options.config.name);\n    }\n  }\n}"]}
|
package/lib/pipeline.js
DELETED
|
@@ -1,277 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.CDKPipeline = exports.DeploymentType = exports.PipelineEngine = void 0;
|
|
4
|
-
const projen_1 = require("projen");
|
|
5
|
-
const common_1 = require("projen/lib/common");
|
|
6
|
-
const engine_1 = require("./engine");
|
|
7
|
-
/**
|
|
8
|
-
* The CI/CD tooling used to run your pipeline.
|
|
9
|
-
* The component will render workflows for the given system
|
|
10
|
-
*/
|
|
11
|
-
var PipelineEngine;
|
|
12
|
-
(function (PipelineEngine) {
|
|
13
|
-
/** Create GitHub actions */
|
|
14
|
-
PipelineEngine[PipelineEngine["GITHUB"] = 0] = "GITHUB";
|
|
15
|
-
/** Create a .gitlab-ci.yaml file */
|
|
16
|
-
PipelineEngine[PipelineEngine["GITLAB"] = 1] = "GITLAB";
|
|
17
|
-
// /** Create AWS CodeCatalyst workflows */
|
|
18
|
-
// CODE_CATALYST,
|
|
19
|
-
})(PipelineEngine || (exports.PipelineEngine = PipelineEngine = {}));
|
|
20
|
-
/**
|
|
21
|
-
* Describes the type of pipeline that will be created
|
|
22
|
-
*/
|
|
23
|
-
var DeploymentType;
|
|
24
|
-
(function (DeploymentType) {
|
|
25
|
-
/** Deploy every commit as far as possible; hopefully into production */
|
|
26
|
-
DeploymentType[DeploymentType["CONTINUOUS_DEPLOYMENT"] = 0] = "CONTINUOUS_DEPLOYMENT";
|
|
27
|
-
/** Build every commit and prepare all assets for a later deployment */
|
|
28
|
-
DeploymentType[DeploymentType["CONTINUOUS_DELIVERY"] = 1] = "CONTINUOUS_DELIVERY";
|
|
29
|
-
})(DeploymentType || (exports.DeploymentType = DeploymentType = {}));
|
|
30
|
-
/**
|
|
31
|
-
* The CDKPipeline class extends the Component class and sets up the necessary configuration for deploying AWS CDK (Cloud Development Kit) applications across multiple stages.
|
|
32
|
-
* It also manages tasks such as publishing CDK assets, bumping version based on git tags, and cleaning up conflicting tasks.
|
|
33
|
-
*/
|
|
34
|
-
class CDKPipeline extends projen_1.Component {
|
|
35
|
-
constructor(app, props) {
|
|
36
|
-
var _a, _b, _c;
|
|
37
|
-
super(app);
|
|
38
|
-
this.app = app;
|
|
39
|
-
this.props = props;
|
|
40
|
-
// Add development dependencies
|
|
41
|
-
this.app.addDevDeps('@types/standard-version', 'standard-version', 'cdk-assets');
|
|
42
|
-
// this.app.addDeps(
|
|
43
|
-
// );
|
|
44
|
-
this.stackPrefix = (_a = props.stackPrefix) !== null && _a !== void 0 ? _a : app.name;
|
|
45
|
-
// Create engine instance to use
|
|
46
|
-
switch (props.engine) {
|
|
47
|
-
case PipelineEngine.GITHUB:
|
|
48
|
-
this.engine = new engine_1.GitHubEngine(app, props, this);
|
|
49
|
-
break;
|
|
50
|
-
default:
|
|
51
|
-
throw new Error('Invalid engine');
|
|
52
|
-
}
|
|
53
|
-
// Removes the compiled cloud assembly before each synth
|
|
54
|
-
(_b = this.project.tasks.tryFind('synth')) === null || _b === void 0 ? void 0 : _b.prependExec(`rm -rf ${this.app.cdkConfig.cdkout}`);
|
|
55
|
-
(_c = this.project.tasks.tryFind('synth:silent')) === null || _c === void 0 ? void 0 : _c.prependExec(`rm -rf ${this.app.cdkConfig.cdkout}`);
|
|
56
|
-
// Remove tasks that might conflict with the pipeline process
|
|
57
|
-
this.project.removeTask('deploy');
|
|
58
|
-
this.project.removeTask('diff');
|
|
59
|
-
this.project.removeTask('destroy');
|
|
60
|
-
this.project.removeTask('watch');
|
|
61
|
-
this.createSynthStage();
|
|
62
|
-
// Creates different deployment stages
|
|
63
|
-
if (props.personalStage) {
|
|
64
|
-
this.createPersonalStage();
|
|
65
|
-
}
|
|
66
|
-
if (props.featureStages) {
|
|
67
|
-
this.createFeatureStage();
|
|
68
|
-
}
|
|
69
|
-
for (const stage of props.stages) {
|
|
70
|
-
this.createPipelineStage(stage);
|
|
71
|
-
}
|
|
72
|
-
// Creates tasks to handle the release process
|
|
73
|
-
this.createReleaseTasks();
|
|
74
|
-
// Creates a specialized CDK App class
|
|
75
|
-
this.createApplicationEntrypoint();
|
|
76
|
-
}
|
|
77
|
-
createSynthStage() {
|
|
78
|
-
var _a, _b, _c, _d;
|
|
79
|
-
this.engine.createSynth({
|
|
80
|
-
commands: [
|
|
81
|
-
...((_a = this.props.preInstallCommands) !== null && _a !== void 0 ? _a : []),
|
|
82
|
-
`npx projen ${this.app.package.installCiTask.name}`,
|
|
83
|
-
...((_b = this.props.preSynthCommands) !== null && _b !== void 0 ? _b : []),
|
|
84
|
-
'npx projen build',
|
|
85
|
-
...((_c = this.props.postSynthCommands) !== null && _c !== void 0 ? _c : []),
|
|
86
|
-
],
|
|
87
|
-
});
|
|
88
|
-
this.engine.createAssetUpload({
|
|
89
|
-
commands: [
|
|
90
|
-
...((_d = this.props.preInstallCommands) !== null && _d !== void 0 ? _d : []),
|
|
91
|
-
`npx projen ${this.app.package.installCiTask.name}`,
|
|
92
|
-
'npx projen publish:assets',
|
|
93
|
-
...(this.engine.needsVersionedArtifacts ? [
|
|
94
|
-
'npx projen bump',
|
|
95
|
-
'npx projen release:push-assembly',
|
|
96
|
-
] : []),
|
|
97
|
-
],
|
|
98
|
-
});
|
|
99
|
-
}
|
|
100
|
-
/**
|
|
101
|
-
* This method generates the entry point for the application, including interfaces and classes
|
|
102
|
-
* necessary to set up the pipeline and define the AWS CDK stacks for different environments.
|
|
103
|
-
*/
|
|
104
|
-
createApplicationEntrypoint() {
|
|
105
|
-
let propsCode = '';
|
|
106
|
-
let appCode = '';
|
|
107
|
-
if (this.props.personalStage) {
|
|
108
|
-
propsCode += ` /** This function will be used to generate a personal stack. */
|
|
109
|
-
providePersonalStack: (app: App, stackId: string, props: PipelineAppStackProps) => Stack;
|
|
110
|
-
`;
|
|
111
|
-
appCode += ` // If the environment variable USER is set and a function is provided for creating a personal stack, it is called with necessary arguments.
|
|
112
|
-
if (props.providePersonalStack && process.env.USER) {
|
|
113
|
-
const stageName = 'personal-' + process.env.USER.toLowerCase().replace(/\\\//g, '-');
|
|
114
|
-
props.providePersonalStack(this, '${this.stackPrefix}-personal', { env: ${JSON.stringify(this.props.personalStage.env)}, stackName: \`${this.stackPrefix}-\${stageName}\`, stageName });
|
|
115
|
-
}
|
|
116
|
-
`;
|
|
117
|
-
}
|
|
118
|
-
if (this.props.featureStages) {
|
|
119
|
-
propsCode += ` /** This function will be used to generate a feature stack. */
|
|
120
|
-
provideFeatureStack: (app: App, stackId: string, props: PipelineAppStackProps) => Stack;
|
|
121
|
-
`;
|
|
122
|
-
appCode += ` // If the environment variable BRANCH is set and a function is provided for creating a feature stack, it is called with necessary arguments.
|
|
123
|
-
if (props.provideFeatureStack && process.env.BRANCH) {
|
|
124
|
-
const stageName = 'feature-' + process.env.BRANCH.toLowerCase().replace(/\\\//g, '-');
|
|
125
|
-
props.provideFeatureStack(this, '${this.stackPrefix}-feature', { env: ${JSON.stringify(this.props.featureStages.env)}, stackName: \`${this.stackPrefix}-\${stageName}\`, stageName });
|
|
126
|
-
}
|
|
127
|
-
`;
|
|
128
|
-
}
|
|
129
|
-
for (const stage of this.props.stages) {
|
|
130
|
-
const nameUpperFirst = `${stage.name.charAt(0).toUpperCase()}${stage.name.substring(1)}`;
|
|
131
|
-
propsCode += ` /** This function will be used to generate a ${stage.name} stack. */
|
|
132
|
-
provide${nameUpperFirst}Stack: (app: App, stackId: string, props: PipelineAppStackProps) => Stack;
|
|
133
|
-
`;
|
|
134
|
-
appCode += ` // If a function is provided for creating a ${stage.name} stack, it is called with necessary arguments.
|
|
135
|
-
if (props.provide${nameUpperFirst}Stack) {
|
|
136
|
-
props.provide${nameUpperFirst}Stack(this, '${this.stackPrefix}-${stage.name}', { env: ${JSON.stringify(stage.env)}, stackName: '${this.stackPrefix}-${stage.name}', stageName: '${stage.name}' });
|
|
137
|
-
}
|
|
138
|
-
`;
|
|
139
|
-
}
|
|
140
|
-
const appFile = new projen_1.TextFile(this.project, `${this.app.srcdir}/app.ts`);
|
|
141
|
-
appFile.addLine(`// ${common_1.PROJEN_MARKER}
|
|
142
|
-
/* eslint-disable */
|
|
143
|
-
import { App, AppProps, Stack, StackProps } from 'aws-cdk-lib';
|
|
144
|
-
|
|
145
|
-
/**
|
|
146
|
-
* PipelineAppProps is an extension of AppProps, which is part of the AWS CDK core.
|
|
147
|
-
* It includes optional functions to provide AWS Stacks for different stages.
|
|
148
|
-
*
|
|
149
|
-
* Use these functions to instantiate your application stacks with the parameters for
|
|
150
|
-
* each stage
|
|
151
|
-
*/
|
|
152
|
-
export interface PipelineAppProps extends AppProps {
|
|
153
|
-
${propsCode}
|
|
154
|
-
}
|
|
155
|
-
|
|
156
|
-
/**
|
|
157
|
-
* PipelineAppStackProps is an extension of StackProps, which is part of the AWS CDK core.
|
|
158
|
-
* It includes an additional property to specify the stage name.
|
|
159
|
-
*/
|
|
160
|
-
export interface PipelineAppStackProps extends StackProps {
|
|
161
|
-
stageName: string;
|
|
162
|
-
}
|
|
163
|
-
|
|
164
|
-
/**
|
|
165
|
-
* The PipelineApp class extends the App class from AWS CDK and overrides the constructor to support
|
|
166
|
-
* different stages of the application (development, production, personal, feature) by invoking the provided
|
|
167
|
-
* stack-providing functions from the props.
|
|
168
|
-
*/
|
|
169
|
-
export class PipelineApp extends App {
|
|
170
|
-
constructor(props: PipelineAppProps) {
|
|
171
|
-
super(props);
|
|
172
|
-
|
|
173
|
-
${appCode}
|
|
174
|
-
|
|
175
|
-
}
|
|
176
|
-
}
|
|
177
|
-
`);
|
|
178
|
-
}
|
|
179
|
-
/**
|
|
180
|
-
* This method sets up tasks to publish CDK assets to all accounts and handle versioning, including bumping the version
|
|
181
|
-
* based on the latest git tag and pushing the CDK assembly to the package repository.
|
|
182
|
-
*/
|
|
183
|
-
createReleaseTasks() {
|
|
184
|
-
// Task to publish the CDK assets to all accounts
|
|
185
|
-
this.project.addTask('publish:assets', {
|
|
186
|
-
steps: this.props.stages.map(stage => ({
|
|
187
|
-
exec: `npx cdk-assets -p ${this.app.cdkConfig.cdkout}/${this.stackPrefix}-${stage.name}.assets.json publish`,
|
|
188
|
-
})),
|
|
189
|
-
});
|
|
190
|
-
this.project.addTask('bump', {
|
|
191
|
-
description: 'Bumps version based on latest git tag',
|
|
192
|
-
steps: [
|
|
193
|
-
{
|
|
194
|
-
exec: 'pipelines-release bump',
|
|
195
|
-
},
|
|
196
|
-
{
|
|
197
|
-
exec: 'git push --tags',
|
|
198
|
-
},
|
|
199
|
-
],
|
|
200
|
-
});
|
|
201
|
-
this.project.addTask('release:push-assembly', {
|
|
202
|
-
steps: [
|
|
203
|
-
{
|
|
204
|
-
exec: `pipelines-release create-manifest "${this.app.cdkConfig.cdkout}" "${this.props.pkgNamespace}"`,
|
|
205
|
-
},
|
|
206
|
-
{
|
|
207
|
-
cwd: this.app.cdkConfig.cdkout,
|
|
208
|
-
exec: 'npm version --no-git-tag-version from-git',
|
|
209
|
-
},
|
|
210
|
-
{
|
|
211
|
-
cwd: this.app.cdkConfig.cdkout,
|
|
212
|
-
exec: 'npm publish',
|
|
213
|
-
},
|
|
214
|
-
],
|
|
215
|
-
});
|
|
216
|
-
}
|
|
217
|
-
/**
|
|
218
|
-
* This method sets up tasks for the personal deployment stage, including deployment, watching for changes,
|
|
219
|
-
* comparing changes (diff), and destroying the stack when no longer needed.
|
|
220
|
-
*/
|
|
221
|
-
createPersonalStage() {
|
|
222
|
-
this.project.addTask('deploy:personal', {
|
|
223
|
-
exec: `cdk deploy ${this.stackPrefix}-personal`,
|
|
224
|
-
});
|
|
225
|
-
this.project.addTask('watch:personal', {
|
|
226
|
-
exec: `cdk deploy --watch --hotswap ${this.stackPrefix}-personal`,
|
|
227
|
-
});
|
|
228
|
-
this.project.addTask('diff:personal', {
|
|
229
|
-
exec: `cdk diff ${this.stackPrefix}-personal`,
|
|
230
|
-
});
|
|
231
|
-
this.project.addTask('destroy:personal', {
|
|
232
|
-
exec: `cdk destroy ${this.stackPrefix}-personal`,
|
|
233
|
-
});
|
|
234
|
-
}
|
|
235
|
-
/**
|
|
236
|
-
* This method sets up tasks for the feature deployment stage, including deployment, comparing changes (diff),
|
|
237
|
-
* and destroying the stack when no longer needed.
|
|
238
|
-
*/
|
|
239
|
-
createFeatureStage() {
|
|
240
|
-
this.project.addTask('deploy:feature', {
|
|
241
|
-
exec: `cdk --progress events --require-approval never deploy ${this.stackPrefix}-feature`,
|
|
242
|
-
});
|
|
243
|
-
this.project.addTask('diff:feature', {
|
|
244
|
-
exec: `cdk diff ${this.stackPrefix}-feature`,
|
|
245
|
-
});
|
|
246
|
-
this.project.addTask('destroy:feature', {
|
|
247
|
-
exec: `cdk destroy ${this.stackPrefix}-feature`,
|
|
248
|
-
});
|
|
249
|
-
}
|
|
250
|
-
/**
|
|
251
|
-
* This method sets up tasks for the general pipeline stages (dev, prod), including deployment and comparing changes (diff).
|
|
252
|
-
* @param {DeployStageOptions} stage - The stage to create
|
|
253
|
-
*/
|
|
254
|
-
createPipelineStage(stage) {
|
|
255
|
-
var _a;
|
|
256
|
-
this.project.addTask(`deploy:${stage.name}`, {
|
|
257
|
-
exec: `cdk --app ${this.app.cdkConfig.cdkout} --progress events --require-approval never deploy ${this.stackPrefix}-${stage.name}`,
|
|
258
|
-
});
|
|
259
|
-
this.project.addTask(`diff:${stage.name}`, {
|
|
260
|
-
exec: `cdk --app ${this.app.cdkConfig.cdkout} diff ${this.stackPrefix}-${stage.name}`,
|
|
261
|
-
});
|
|
262
|
-
this.engine.createDeployment({
|
|
263
|
-
config: stage,
|
|
264
|
-
installCommands: [
|
|
265
|
-
...((_a = this.props.preInstallCommands) !== null && _a !== void 0 ? _a : []),
|
|
266
|
-
`npx projen ${this.app.package.installCiTask.name}`,
|
|
267
|
-
],
|
|
268
|
-
deployCommands: [
|
|
269
|
-
// TODO pre deploy steps
|
|
270
|
-
`npx projen deploy:${stage.name}`,
|
|
271
|
-
// TODO post deploy steps
|
|
272
|
-
],
|
|
273
|
-
});
|
|
274
|
-
}
|
|
275
|
-
}
|
|
276
|
-
exports.CDKPipeline = CDKPipeline;
|
|
277
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pipeline.js","sourceRoot":"","sources":["../src/pipeline.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, TextFile, awscdk } from 'projen';\nimport { PROJEN_MARKER } from 'projen/lib/common';\nimport { BaseEngine, GitHubEngine, GithubEngineConfig } from './engine';\n\n/**\n * The Environment interface is designed to hold AWS related information\n * for a specific deployment environment within your infrastructure.\n * Each environment requires a specific account and region for its resources.\n */\nexport interface Environment {\n  /**\n   * The AWS Account ID associated with the environment. It's important because\n   * different services or features could have distinct permissions and settings\n   * in different accounts.\n   */\n  readonly account: string;\n\n  /**\n   * The AWS Region for the environment. This determines where your resources\n   * are created and where your application will run. It can affect latency,\n   * availability, and pricing.\n   */\n  readonly region: string;\n}\n\n/**\n * The CI/CD tooling used to run your pipeline.\n * The component will render workflows for the given system\n */\nexport enum PipelineEngine {\n  /** Create GitHub actions */\n  GITHUB,\n  /** Create a .gitlab-ci.yaml file */\n  GITLAB,\n  // /** Create AWS CodeCatalyst workflows */\n  // CODE_CATALYST,\n}\n\n/**\n * Describes the type of pipeline that will be created\n */\nexport enum DeploymentType {\n  /** Deploy every commit as far as possible; hopefully into production */\n  CONTINUOUS_DEPLOYMENT,\n  /** Build every commit and prepare all assets for a later deployment */\n  CONTINUOUS_DELIVERY,\n}\n\nexport interface DeploymentStage {\n  readonly name: string;\n  readonly env: Environment;\n  readonly manualApproval?: boolean;\n}\n\n/**\n * The CDKPipelineOptions interface is designed to provide configuration\n * options for a CDK (Cloud Development Kit) pipeline. It allows the definition\n * of settings such as the stack prefix and package namespace to be used in the\n * AWS stack, along with the environments configuration to be used.\n */\nexport interface CDKPipelineOptions {\n\n  /**\n   * This field is used to define a prefix for the AWS Stack resources created\n   * during the pipeline's operation.\n   *\n   * @default project name\n   */\n  readonly stackPrefix?: string;\n\n  /**\n   * This field determines the NPM namespace to be used when packaging CDK cloud\n   * assemblies. A namespace helps group related resources together, providing\n   * better organization and ease of management.\n   */\n  readonly pkgNamespace: string;\n\n  readonly stages: DeploymentStage[];\n\n  readonly personalStage?: {\n    readonly env: Environment;\n  };\n\n  readonly featureStages?: {\n    readonly env: Environment;\n  };\n\n  /**\n   * This field specifies the type of pipeline to create. If set to CONTINUOUS_DEPLOYMENT,\n   * every commit is deployed as far as possible, hopefully into production. If set to\n   * CONTINUOUS_DELIVERY, every commit is built and all assets are prepared for a later deployment.\n   *\n   * @default CONTINUOUS_DELIVERY\n   */\n  readonly deploymentType?: DeploymentType;\n\n  /**\n   * This field determines the CI/CD tooling that will be used to run the pipeline. The component\n   * will render workflows for the given system. Options include GitHub and GitLab.\n   *\n   * @default - tries to derive it from the projects configuration\n   */\n  readonly engine?: PipelineEngine;\n\n  readonly githubConfig?: GithubEngineConfig;\n\n  readonly preInstallCommands?: string[];\n  readonly preSynthCommands?: string[];\n  readonly postSynthCommands?: string[];\n\n}\n\n/**\n * The CDKPipeline class extends the Component class and sets up the necessary configuration for deploying AWS CDK (Cloud Development Kit) applications across multiple stages.\n * It also manages tasks such as publishing CDK assets, bumping version based on git tags, and cleaning up conflicting tasks.\n */\nexport class CDKPipeline extends Component {\n\n  public readonly stackPrefix: string;\n  public readonly engine: BaseEngine;\n\n  constructor(private app: awscdk.AwsCdkTypeScriptApp, private props: CDKPipelineOptions) {\n    super(app);\n\n    // Add development dependencies\n    this.app.addDevDeps(\n      '@types/standard-version',\n      'standard-version',\n      'cdk-assets',\n    );\n    // this.app.addDeps(\n    // );\n\n    this.stackPrefix = props.stackPrefix ?? app.name;\n\n    // Create engine instance to use\n    switch (props.engine) {\n      case PipelineEngine.GITHUB:\n        this.engine = new GitHubEngine(app, props, this);\n        break;\n      default:\n        throw new Error('Invalid engine');\n    }\n\n    // Removes the compiled cloud assembly before each synth\n    this.project.tasks.tryFind('synth')?.prependExec(`rm -rf ${this.app.cdkConfig.cdkout}`);\n    this.project.tasks.tryFind('synth:silent')?.prependExec(`rm -rf ${this.app.cdkConfig.cdkout}`);\n\n    // Remove tasks that might conflict with the pipeline process\n    this.project.removeTask('deploy');\n    this.project.removeTask('diff');\n    this.project.removeTask('destroy');\n    this.project.removeTask('watch');\n\n    this.createSynthStage();\n\n    // Creates different deployment stages\n    if (props.personalStage) {\n      this.createPersonalStage();\n    }\n    if (props.featureStages) {\n      this.createFeatureStage();\n    }\n    for (const stage of props.stages) {\n      this.createPipelineStage(stage);\n    }\n\n    // Creates tasks to handle the release process\n    this.createReleaseTasks();\n\n    // Creates a specialized CDK App class\n    this.createApplicationEntrypoint();\n\n  }\n\n  private createSynthStage() {\n    this.engine.createSynth({\n      commands: [\n        ...(this.props.preInstallCommands ?? []),\n        `npx projen ${this.app.package.installCiTask.name}`,\n        ...(this.props.preSynthCommands ?? []),\n        'npx projen build',\n        ...(this.props.postSynthCommands ?? []),\n      ],\n    });\n    this.engine.createAssetUpload({\n      commands: [\n        ...(this.props.preInstallCommands ?? []),\n        `npx projen ${this.app.package.installCiTask.name}`,\n        'npx projen publish:assets',\n        ...(this.engine.needsVersionedArtifacts ? [\n          'npx projen bump',\n          'npx projen release:push-assembly',\n        ] : []),\n      ],\n    });\n  }\n\n  /**\n   * This method generates the entry point for the application, including interfaces and classes\n   * necessary to set up the pipeline and define the AWS CDK stacks for different environments.\n   */\n  private createApplicationEntrypoint() {\n    let propsCode = '';\n    let appCode = '';\n\n    if (this.props.personalStage) {\n      propsCode += `  /** This function will be used to generate a personal stack. */\n  providePersonalStack: (app: App, stackId: string, props: PipelineAppStackProps) => Stack;\n`;\n      appCode += `    // If the environment variable USER is set and a function is provided for creating a personal stack, it is called with necessary arguments.\n    if (props.providePersonalStack && process.env.USER) {\n      const stageName = 'personal-' + process.env.USER.toLowerCase().replace(/\\\\\\//g, '-');\n      props.providePersonalStack(this, '${this.stackPrefix}-personal', { env: ${JSON.stringify(this.props.personalStage.env)}, stackName: \\`${this.stackPrefix}-\\${stageName}\\`, stageName });\n    }\n`;\n    }\n\n    if (this.props.featureStages) {\n      propsCode += `  /** This function will be used to generate a feature stack. */\n  provideFeatureStack: (app: App, stackId: string, props: PipelineAppStackProps) => Stack;\n`;\n      appCode += `    // If the environment variable BRANCH is set and a function is provided for creating a feature stack, it is called with necessary arguments.\n    if (props.provideFeatureStack && process.env.BRANCH) {\n      const stageName = 'feature-' + process.env.BRANCH.toLowerCase().replace(/\\\\\\//g, '-');\n      props.provideFeatureStack(this, '${this.stackPrefix}-feature', { env: ${JSON.stringify(this.props.featureStages.env)}, stackName: \\`${this.stackPrefix}-\\${stageName}\\`, stageName });\n    }\n`;\n    }\n\n    for (const stage of this.props.stages) {\n      const nameUpperFirst = `${stage.name.charAt(0).toUpperCase()}${stage.name.substring(1)}`;\n\n      propsCode += `  /** This function will be used to generate a ${stage.name} stack. */\n  provide${nameUpperFirst}Stack: (app: App, stackId: string, props: PipelineAppStackProps) => Stack;\n`;\n      appCode += `    // If a function is provided for creating a ${stage.name} stack, it is called with necessary arguments.\n    if (props.provide${nameUpperFirst}Stack) {\n      props.provide${nameUpperFirst}Stack(this, '${this.stackPrefix}-${stage.name}', { env: ${JSON.stringify(stage.env)}, stackName: '${this.stackPrefix}-${stage.name}', stageName: '${stage.name}' });\n    }\n`;\n    }\n\n    const appFile = new TextFile(this.project, `${this.app.srcdir}/app.ts`);\n    appFile.addLine(`// ${PROJEN_MARKER}\n/* eslint-disable */\nimport { App, AppProps, Stack, StackProps } from 'aws-cdk-lib';\n\n/**\n * PipelineAppProps is an extension of AppProps, which is part of the AWS CDK core.\n * It includes optional functions to provide AWS Stacks for different stages.\n *\n * Use these functions to instantiate your application stacks with the parameters for\n * each stage\n */\nexport interface PipelineAppProps extends AppProps {\n${propsCode}\n}\n\n/**\n * PipelineAppStackProps is an extension of StackProps, which is part of the AWS CDK core.\n * It includes an additional property to specify the stage name.\n */\nexport interface PipelineAppStackProps extends StackProps {\n  stageName: string;\n}\n\n/**\n * The PipelineApp class extends the App class from AWS CDK and overrides the constructor to support\n * different stages of the application (development, production, personal, feature) by invoking the provided\n * stack-providing functions from the props.\n */\nexport class PipelineApp extends App {\n  constructor(props: PipelineAppProps) {\n    super(props);\n\n${appCode}\n\n  }\n}\n`);\n  }\n\n  /**\n   * This method sets up tasks to publish CDK assets to all accounts and handle versioning, including bumping the version\n   * based on the latest git tag and pushing the CDK assembly to the package repository.\n   */\n  private createReleaseTasks() {\n    // Task to publish the CDK assets to all accounts\n    this.project.addTask('publish:assets', {\n      steps: this.props.stages.map(stage => ({\n        exec: `npx cdk-assets -p ${this.app.cdkConfig.cdkout}/${this.stackPrefix}-${stage.name}.assets.json publish`,\n      })),\n    });\n\n    this.project.addTask('bump', {\n      description: 'Bumps version based on latest git tag',\n      steps: [\n        {\n          exec: 'pipelines-release bump',\n        },\n        {\n          exec: 'git push --tags',\n        },\n      ],\n    });\n    this.project.addTask('release:push-assembly', {\n      steps: [\n        {\n          exec: `pipelines-release create-manifest \"${this.app.cdkConfig.cdkout}\"  \"${this.props.pkgNamespace}\"`,\n        },\n        {\n          cwd: this.app.cdkConfig.cdkout,\n          exec: 'npm version --no-git-tag-version from-git',\n        },\n        {\n          cwd: this.app.cdkConfig.cdkout,\n          exec: 'npm publish',\n        },\n      ],\n    });\n  }\n\n  /**\n   * This method sets up tasks for the personal deployment stage, including deployment, watching for changes,\n   * comparing changes (diff), and destroying the stack when no longer needed.\n   */\n  private createPersonalStage() {\n    this.project.addTask('deploy:personal', {\n      exec: `cdk deploy ${this.stackPrefix}-personal`,\n    });\n    this.project.addTask('watch:personal', {\n      exec: `cdk deploy --watch --hotswap ${this.stackPrefix}-personal`,\n    });\n    this.project.addTask('diff:personal', {\n      exec: `cdk diff ${this.stackPrefix}-personal`,\n    });\n    this.project.addTask('destroy:personal', {\n      exec: `cdk destroy ${this.stackPrefix}-personal`,\n    });\n  }\n\n  /**\n   * This method sets up tasks for the feature deployment stage, including deployment, comparing changes (diff),\n   * and destroying the stack when no longer needed.\n   */\n  private createFeatureStage() {\n    this.project.addTask('deploy:feature', {\n      exec: `cdk --progress events --require-approval never deploy ${this.stackPrefix}-feature`,\n    });\n    this.project.addTask('diff:feature', {\n      exec: `cdk diff ${this.stackPrefix}-feature`,\n    });\n    this.project.addTask('destroy:feature', {\n      exec: `cdk destroy ${this.stackPrefix}-feature`,\n    });\n  }\n\n  /**\n   * This method sets up tasks for the general pipeline stages (dev, prod), including deployment and comparing changes (diff).\n   * @param {DeployStageOptions} stage - The stage to create\n   */\n  private createPipelineStage(stage: DeploymentStage) {\n    this.project.addTask(`deploy:${stage.name}`, {\n      exec: `cdk --app ${this.app.cdkConfig.cdkout} --progress events --require-approval never deploy ${this.stackPrefix}-${stage.name}`,\n    });\n    this.project.addTask(`diff:${stage.name}`, {\n      exec: `cdk --app ${this.app.cdkConfig.cdkout} diff ${this.stackPrefix}-${stage.name}`,\n    });\n\n    this.engine.createDeployment({\n      config: stage,\n      installCommands: [\n        ...(this.props.preInstallCommands ?? []),\n        `npx projen ${this.app.package.installCiTask.name}`,\n      ],\n      deployCommands: [\n        // TODO pre deploy steps\n        `npx projen deploy:${stage.name}`,\n        // TODO post deploy steps\n      ],\n    });\n  }\n}"]}
|
|
File without changes
|