@rio-cloud/cdk-v2-constructs 4.0.2 → 4.1.1

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.
@@ -40,40 +40,81 @@ var PipelineType;
40
40
  * <p> This gradle command must be provided by the gradle project.
41
41
  */
42
42
  PipelineType["STANDARD_GRADLE"] = "STANDARD_GRADLE";
43
+ /**
44
+ * Self-mutating codepipeline deploying the provided AppStack with additional steps needed to access Kafka.
45
+ *
46
+ * Comes with a pre-defined buildspec.yaml for a gradle project. This buildspec.yaml is used for both the main and
47
+ * branch pipeline and includes the following steps
48
+ * - build and test of the artifact via command "./gradlew clean build".
49
+ * <p> This gradle command must be provided by the gradle project.
50
+ * - license-check and upload of license-check-file via command "./gradlew checkLicenses".
51
+ * <p> This gradle command must be provided by the gradle project.
52
+ * - create a self-signed certificate to make TLS communication possible, e.g. with a loadbalancer.
53
+ * - request a certificate to authenticate against Kafka via mutual TLS
54
+ * - build the container image via command "./gradlew jibBuildTar".
55
+ * <p> This gradle command must be provided by the gradle project.
56
+ */
43
57
  PipelineType["STANDARD_GRADLE_KAFKA"] = "STANDARD_GRADLE_KAFKA";
58
+ /**
59
+ * Self-mutating codepipeline deploying the provided AppStack.
60
+ *
61
+ * Comes with a pre-defined buildspec.yaml to synthesize the AppStack.
62
+ * No branch or vulnerability pipeline is created.
63
+ */
64
+ PipelineType["STANDARD_INFRASTRUCTURE"] = "STANDARD_INFRASTRUCTURE";
44
65
  })(PipelineType = exports.PipelineType || (exports.PipelineType = {}));
45
66
  class PipelineStack extends aws_cdk_lib_1.Stack {
46
67
  constructor(scope, id, props) {
47
- var _b, _c, _d, _e, _f;
48
68
  super(scope, id, props);
49
- // Creating the main pipeline
50
- const bucket = s3.Bucket.fromBucketName(this, 'TriggerBucket', aws_cdk_lib_1.Fn.importValue('bitbucket-integration-s3-bucket'));
51
- const hostedZone = rio.RioLandingZone.getDefaultHostedZone(this);
52
- const inputMasterTriggerFile = pipelines.CodePipelineSource.s3(bucket, `${props.serviceName}/trigger/${rio.TriggerKey.MASTER_REPO}`, { trigger: codepipeline_actions.S3Trigger.EVENTS });
53
- const inputBranchTriggerFile = pipelines.CodePipelineSource.s3(bucket, `${props.serviceName}/trigger/${rio.TriggerKey.BRANCH_REPO}`, { trigger: codepipeline_actions.S3Trigger.EVENTS });
54
- const ossLicensesBucketParameter = ssm.StringParameter.fromStringParameterAttributes(this, 'OssLicensesBucketParameter', {
55
- parameterName: '/config/oss-licenses/bucket-name',
56
- });
57
- const ossLicensesBucket = s3.Bucket.fromBucketName(this, 'OssLicensesBucket', ossLicensesBucketParameter.stringValue);
58
- const serviceKeystorePassword = ssm.StringParameter.fromSecureStringParameterAttributes(this, 'KeystorePassword', {
69
+ // populate external parameters
70
+ this.bucket = s3.Bucket.fromBucketName(this, 'TriggerBucket', aws_cdk_lib_1.Fn.importValue('bitbucket-integration-s3-bucket'));
71
+ this.hostedZone = rio.RioLandingZone.getDefaultHostedZone(this);
72
+ this.inputMasterTriggerFile = pipelines.CodePipelineSource.s3(this.bucket, `${props.serviceName}/trigger/${rio.TriggerKey.MASTER_REPO}`, { trigger: codepipeline_actions.S3Trigger.EVENTS });
73
+ this.inputBranchTriggerFile = pipelines.CodePipelineSource.s3(this.bucket, `${props.serviceName}/trigger/${rio.TriggerKey.BRANCH_REPO}`, { trigger: codepipeline_actions.S3Trigger.EVENTS });
74
+ this.ossLicensesBucketParameter = ssm.StringParameter.fromStringParameterAttributes(this, 'OssLicensesBucketParameter', { parameterName: '/config/oss-licenses/bucket-name' });
75
+ this.ossLicensesBucket = s3.Bucket.fromBucketName(this, 'OssLicensesBucket', this.ossLicensesBucketParameter.stringValue);
76
+ this.serviceKeystorePassword = ssm.StringParameter.fromSecureStringParameterAttributes(this, 'KeystorePassword', {
59
77
  parameterName: `/config/${props.serviceName}/service-keystore-password`,
60
78
  version: 1,
61
79
  });
62
- const accountNameParameter = ssm.StringParameter.fromStringParameterName(this, 'AccountNameParameter', '/config/account/name');
63
- const nistDataMirrorUrl = ssm.StringParameter.fromSecureStringParameterAttributes(this, 'NistDataMirrorUrl', {
64
- parameterName: '/config/nist-data-mirror/url', version: 1,
80
+ this.accountNameParameter = ssm.StringParameter.fromStringParameterName(this, 'AccountNameParameter', '/config/account/name');
81
+ this.nistDataMirrorUrl = ssm.StringParameter.fromSecureStringParameterAttributes(this, 'NistDataMirrorUrl', {
82
+ parameterName: '/config/nist-data-mirror/url',
83
+ version: 1,
84
+ });
85
+ this.addMainPipeline(props);
86
+ if (props.pipelineType !== PipelineType.STANDARD_INFRASTRUCTURE) {
87
+ this.addBranchPipeline(props);
88
+ this.addVulnerabilityPipeline(props);
89
+ }
90
+ const [feature, packageName] = __dirname.split(path.sep).reverse();
91
+ new index_1.Claidometer(this, 'Claidometer', {
92
+ product: index_1.ClaidometerProduct.CDK_CONTRIBUTION,
93
+ packageName,
94
+ feature,
95
+ version: '0.0.5',
96
+ });
97
+ }
98
+ static addLifecycleRuleArtifactBucket(pipeline) {
99
+ const artifactsBucket = pipeline.pipeline.artifactBucket;
100
+ artifactsBucket.addLifecycleRule({
101
+ enabled: true,
102
+ expiration: aws_cdk_lib_1.Duration.days(90),
65
103
  });
66
- const gradlebuild = new build_project_1.RioGradleCodeBuildProject(this, 'GradleBuild', {
67
- input: inputMasterTriggerFile,
104
+ }
105
+ addMainPipeline(props) {
106
+ var _b, _c;
107
+ const mainBuild = new build_project_1.RioCodeBuildProject(this, 'MainBuild', {
108
+ input: this.inputMasterTriggerFile,
68
109
  cdkOutDirectory: 'infrastructure/cdk.out',
69
110
  partialBuildSpec: this.loadBuildSpecFromFile(((_b = props === null || props === void 0 ? void 0 : props.codeBuildOptions) === null || _b === void 0 ? void 0 : _b.partialBuildSpecPath) ||
70
111
  this.resolveDefaultBuildSpec(props.pipelineType)),
71
112
  buildEnvironment: (_c = props === null || props === void 0 ? void 0 : props.codeBuildOptions) === null || _c === void 0 ? void 0 : _c.buildEnvironment,
72
- hostedZoneName: hostedZone.zoneName,
113
+ hostedZoneName: this.hostedZone.zoneName,
73
114
  serviceName: props.serviceName,
74
115
  });
75
116
  const pipeline = new pipelines.CodePipeline(this, 'Pipeline', {
76
- synth: gradlebuild,
117
+ synth: mainBuild,
77
118
  });
78
119
  // CloudAssemblyArtifacts must be smaller than 256MB. Due to the gradle artifact we are exceeding this limit and pipeline fails in deployment step.
79
120
  // Issue is known and not fixed for already some years. See: https://github.com/aws/aws-cdk/issues/9917
@@ -97,7 +138,7 @@ class PipelineStack extends aws_cdk_lib_1.Stack {
97
138
  pre: props.secretsDeployStepProps ? [
98
139
  new rio.RioSecretsDeployStep(this, 'DeploySecrets', {
99
140
  ...props.secretsDeployStepProps,
100
- input: inputMasterTriggerFile,
141
+ input: this.inputMasterTriggerFile,
101
142
  }),
102
143
  ] : undefined,
103
144
  });
@@ -105,22 +146,31 @@ class PipelineStack extends aws_cdk_lib_1.Stack {
105
146
  const rule = new events.Rule(this, 'PipelineSchedule', { schedule: events.Schedule.expression('cron(0 9 ? * 2#1 *)') });
106
147
  rule.addTarget(new targets.CodePipeline(pipeline.pipeline));
107
148
  PipelineStack.addLifecycleRuleArtifactBucket(pipeline);
108
- ossLicensesBucket.grantReadWrite(gradlebuild);
109
- ossLicensesBucketParameter.grantRead(gradlebuild);
110
- serviceKeystorePassword.grantRead(gradlebuild);
111
- accountNameParameter.grantRead(gradlebuild);
149
+ this.ossLicensesBucket.grantReadWrite(mainBuild);
150
+ this.ossLicensesBucketParameter.grantRead(mainBuild);
151
+ if (props.pipelineType !== PipelineType.STANDARD_INFRASTRUCTURE) {
152
+ this.serviceKeystorePassword.grantRead(mainBuild);
153
+ }
154
+ this.accountNameParameter.grantRead(mainBuild);
112
155
  if (props.pipelineType === PipelineType.STANDARD_GRADLE_KAFKA) {
113
- this.grantPermissionsForKafkaIntegration(gradlebuild);
156
+ this.grantPermissionsForKafkaIntegration(mainBuild, 'KafkaIntegrationPolicy');
114
157
  }
115
158
  pipeline.pipeline.artifactBucket.grantReadWrite(stripAssetsStep.project);
116
- // Branch pipeline
117
- const branchBuild = new build_project_1.RioGradleCodeBuildProject(this, 'BranchBuild', {
118
- input: inputBranchTriggerFile,
159
+ new datadog_monitors_1.DatadogPipelineErrorAlert(this, 'DatadogPipelineErrorMonitor', {
160
+ serviceName: props.serviceName,
161
+ pipelineName: this.renamePipelineToLowerCase(pipeline),
162
+ accountId: this.account,
163
+ });
164
+ }
165
+ addBranchPipeline(props) {
166
+ var _b, _c;
167
+ const branchBuild = new build_project_1.RioCodeBuildProject(this, 'BranchBuild', {
168
+ input: this.inputBranchTriggerFile,
119
169
  cdkOutDirectory: 'infrastructure/cdk.out',
120
- partialBuildSpec: this.loadBuildSpecFromFile(((_d = props === null || props === void 0 ? void 0 : props.codeBuildOptions) === null || _d === void 0 ? void 0 : _d.partialBuildSpecPath) ||
170
+ partialBuildSpec: this.loadBuildSpecFromFile(((_b = props === null || props === void 0 ? void 0 : props.codeBuildOptions) === null || _b === void 0 ? void 0 : _b.partialBuildSpecPath) ||
121
171
  this.resolveDefaultBuildSpec(props.pipelineType)),
122
- buildEnvironment: (_e = props === null || props === void 0 ? void 0 : props.codeBuildOptions) === null || _e === void 0 ? void 0 : _e.buildEnvironment,
123
- hostedZoneName: hostedZone.zoneName,
172
+ buildEnvironment: (_c = props === null || props === void 0 ? void 0 : props.codeBuildOptions) === null || _c === void 0 ? void 0 : _c.buildEnvironment,
173
+ hostedZoneName: this.hostedZone.zoneName,
124
174
  serviceName: props.serviceName,
125
175
  });
126
176
  const branchPipeline = new pipelines.CodePipeline(this, 'BranchPipeline', {
@@ -130,16 +180,21 @@ class PipelineStack extends aws_cdk_lib_1.Stack {
130
180
  branchPipeline.addWave('BranchBuild').addPost(branchBuild);
131
181
  branchPipeline.buildPipeline();
132
182
  PipelineStack.addLifecycleRuleArtifactBucket(branchPipeline);
133
- ossLicensesBucket.grantReadWrite(branchBuild);
134
- ossLicensesBucketParameter.grantRead(branchBuild);
135
- serviceKeystorePassword.grantRead(branchBuild);
136
- accountNameParameter.grantRead(branchBuild);
137
- // Vulnerability Pipeline
138
- const vulnerabilityBuild = new build_project_1.RioGradleCodeBuildProject(this, 'VulnerabilityChecks', {
139
- input: inputMasterTriggerFile,
183
+ this.ossLicensesBucket.grantReadWrite(branchBuild);
184
+ this.ossLicensesBucketParameter.grantRead(branchBuild);
185
+ this.serviceKeystorePassword.grantRead(branchBuild);
186
+ this.accountNameParameter.grantRead(branchBuild);
187
+ if (props.pipelineType === PipelineType.STANDARD_GRADLE_KAFKA) {
188
+ this.grantPermissionsForKafkaIntegration(branchBuild, 'BranchKafkaIntegrationPolicy');
189
+ }
190
+ }
191
+ addVulnerabilityPipeline(props) {
192
+ var _b;
193
+ const vulnerabilityBuild = new build_project_1.RioCodeBuildProject(this, 'VulnerabilityChecks', {
194
+ input: this.inputMasterTriggerFile,
140
195
  partialBuildSpec: this.loadBuildSpecFromFile(path.resolve(__dirname, 'buildspecs/buildspec-vulnerability-checks.yaml')),
141
- buildEnvironment: (_f = props === null || props === void 0 ? void 0 : props.codeBuildOptions) === null || _f === void 0 ? void 0 : _f.buildEnvironment,
142
- hostedZoneName: hostedZone.zoneName,
196
+ buildEnvironment: (_b = props === null || props === void 0 ? void 0 : props.codeBuildOptions) === null || _b === void 0 ? void 0 : _b.buildEnvironment,
197
+ hostedZoneName: this.hostedZone.zoneName,
143
198
  serviceName: props.serviceName,
144
199
  });
145
200
  const vulnerabilityPipeline = new pipelines.CodePipeline(this, 'VulnerabilityPipeline', {
@@ -148,45 +203,30 @@ class PipelineStack extends aws_cdk_lib_1.Stack {
148
203
  });
149
204
  vulnerabilityPipeline.addWave('VulnerabilityChecks').addPost(vulnerabilityBuild);
150
205
  vulnerabilityPipeline.buildPipeline();
151
- nistDataMirrorUrl.grantRead(vulnerabilityBuild);
206
+ this.nistDataMirrorUrl.grantRead(vulnerabilityBuild);
152
207
  const vulnerabilityRule = new events.Rule(this, 'VulnerabilityPipelineSchedule', { schedule: events.Schedule.expression('cron(0 5 ? * MON-FRI *)') });
153
208
  vulnerabilityRule.addTarget(new targets.CodePipeline(vulnerabilityPipeline.pipeline));
154
209
  PipelineStack.addLifecycleRuleArtifactBucket(vulnerabilityPipeline);
155
- new datadog_monitors_1.DatadogPipelineErrorAlert(this, 'DatadogPipelineErrorMonitor', {
156
- serviceName: props.serviceName,
157
- pipelineName: this.renamePipelineToLowerCase(pipeline),
158
- accountId: this.account,
159
- });
160
210
  new datadog_monitors_1.DataDogPipelineErrorAlertForVulnerabilityChecks(this, 'DataDogPipelineErrorForVulnerabilityChecksMonitor', {
161
211
  serviceName: props.serviceName,
162
212
  pipelineName: this.renamePipelineToLowerCase(vulnerabilityPipeline),
163
213
  accountId: this.account,
164
214
  });
165
- const [feature, packageName] = __dirname.split(path.sep).reverse();
166
- new index_1.Claidometer(this, 'Claidometer', {
167
- product: index_1.ClaidometerProduct.CDK_CONTRIBUTION,
168
- packageName,
169
- feature,
170
- version: '0.0.3',
171
- });
172
- }
173
- static addLifecycleRuleArtifactBucket(pipeline) {
174
- const artifactsBucket = pipeline.pipeline.artifactBucket;
175
- artifactsBucket.addLifecycleRule({
176
- enabled: true,
177
- expiration: aws_cdk_lib_1.Duration.days(90),
178
- });
179
215
  }
180
216
  resolveDefaultBuildSpec(pipelineType) {
181
217
  switch (pipelineType) {
182
- case PipelineType.STANDARD_GRADLE: return path.resolve(__dirname, 'buildspecs/buildspec.yaml');
183
- case PipelineType.STANDARD_GRADLE_KAFKA: return path.resolve(__dirname, 'buildspecs/buildspec_kafka.yaml');
218
+ case PipelineType.STANDARD_GRADLE:
219
+ return path.resolve(__dirname, 'buildspecs/buildspec.yaml');
220
+ case PipelineType.STANDARD_GRADLE_KAFKA:
221
+ return path.resolve(__dirname, 'buildspecs/buildspec_kafka.yaml');
222
+ case PipelineType.STANDARD_INFRASTRUCTURE:
223
+ return path.resolve(__dirname, 'buildspecs/buildspec_infrastructure.yaml');
184
224
  }
185
225
  }
186
- grantPermissionsForKafkaIntegration(buildProject) {
226
+ grantPermissionsForKafkaIntegration(buildProject, policyId) {
187
227
  const buildProjectRole = buildProject.project.role;
188
228
  if (buildProjectRole) {
189
- buildProjectRole.addManagedPolicy(aws_iam_1.ManagedPolicy.fromManagedPolicyArn(this, 'KafkaIntegrationPolicy', aws_cdk_lib_1.Fn.importValue('kafka-integration-policy-arn')));
229
+ buildProjectRole.addManagedPolicy(aws_iam_1.ManagedPolicy.fromManagedPolicyArn(this, policyId, aws_cdk_lib_1.Fn.importValue('kafka-integration-policy-arn')));
190
230
  }
191
231
  else {
192
232
  throw Error(`Could not add managed policy for kafka-integration to CodeBuild project of ${buildProject.id} due to missing role.`);
@@ -209,4 +249,4 @@ class PipelineStack extends aws_cdk_lib_1.Stack {
209
249
  exports.PipelineStack = PipelineStack;
210
250
  _a = JSII_RTTI_SYMBOL_1;
211
251
  PipelineStack[_a] = { fqn: "@rio-cloud/cdk-v2-constructs.PipelineStack", version: "0.0.0" };
212
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pipeline-stack.js","sourceRoot":"","sources":["../../../../src/contributions/team-transport-two/pipeline/pipeline-stack.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from 'fs';\nimport * as path from 'path';\nimport * as cdk from 'aws-cdk-lib';\nimport { Duration, Fn, Reference, Stack, StackProps } from 'aws-cdk-lib';\nimport { BuildEnvironment, BuildSpec } from 'aws-cdk-lib/aws-codebuild';\nimport * as codepipeline from 'aws-cdk-lib/aws-codepipeline';\nimport * as codepipeline_actions from 'aws-cdk-lib/aws-codepipeline-actions';\nimport * as events from 'aws-cdk-lib/aws-events';\nimport * as targets from 'aws-cdk-lib/aws-events-targets';\nimport { ManagedPolicy } from 'aws-cdk-lib/aws-iam';\nimport * as s3 from 'aws-cdk-lib/aws-s3';\nimport * as ssm from 'aws-cdk-lib/aws-ssm';\nimport * as pipelines from 'aws-cdk-lib/pipelines';\nimport { CodeBuildStep } from 'aws-cdk-lib/pipelines';\nimport { Construct } from 'constructs';\nimport * as yaml from 'js-yaml';\nimport * as rio from '../../../index';\nimport { Claidometer, ClaidometerProduct } from '../../../index';\nimport { UpperToLower } from '../../../watchful/upperToLower';\nimport { Application } from './application-stage';\nimport { RioGradleCodeBuildProject } from './build-project';\nimport { DatadogPipelineErrorAlert, DataDogPipelineErrorAlertForVulnerabilityChecks } from './datadog-monitors';\n\n/**\n  * @struct\n */\nexport interface PipelineStackProps extends StackProps {\n  /**\n   * The name of the service\n   * - This has to be the same name as the repository as the s3 trigger file created by the bitbucket-integration uses this as a prefix\n   * - The serviceName is available as an environment variable SERVICE_NAME in the CodeBuildStep\n   * - The serviceName is available in the AppStackProps in the appStackFactory\n   * - The serviceName is available as stackName property in the AppStackProps in the appStackFactory. I.e. the stack is named after the service\n   * - The serviceName is handed over to all Datadog monitors\n   * - RIO convention is that uploaded license-check-files are named after the service, i.e. serviceName.txt\n   */\n  readonly serviceName: string;\n\n  /**\n   * Provider for appStack to be deployed.\n   *\n   * This acts as a wrapper for all resources you want to deploy via the pipeline.\n   */\n  readonly appStackFactory: IAppStackFactory;\n\n  /**\n   * Defines which kind of pipeline is deployed.\n   */\n  readonly pipelineType: PipelineType;\n\n  /**\n   * Path to secrets file containing encrypted secrets.\n   *\n   * The RioSecretsDeployStep is added prior to deployment of the AppStack if a path is provided.\n   *\n   * No secrets are deployed if no path is provided.\n   */\n  readonly secretsDeployStepProps?: PipelineStackRioSecretsDeployStepProps;\n\n  /**\n   * CodeBuild options overriding the rio-specific defaults.\n   */\n  readonly codeBuildOptions?: CodeBuildOptions;\n}\n\n/**\n * Provider interface to create the AppStack within the specific scope\n *\n * Simply place the AppStack creation into the create-method:\n * @example\n *\n * // Use e.g. within ApplicationProps as JSONObject:\n * { create: (construct, props) => new Stack(construct, 'AppStack', props) }\n *\n * // or simply as class interface within your stack implementation:\n * class PipelineStage extends Stage implements IAppStackFactory {\n *   constructor(scope: Construct, id: string, props: StageProps) {\n *     super(scope, id, props);\n *     // your stage definitiom\n *   }\n *\n *   create(scope: Construct, props: AppStackProps): void {\n *     // app stack initiation goes in here\n *   }\n * }\n */\nexport interface IAppStackFactory {\n  /**\n   * Factory method, being invoked with the specific scope during pipeline instantiation\n   *\n   * @param scope the parent construct for the app stack\n   * @param props the app stacks properties\n   */\n  create(scope: Construct, props: AppStackProps ): void;\n}\n\nexport interface AppStackProps extends StackProps {\n  readonly serviceName: string;\n  readonly version: string;\n}\n\n/**\n * Selected properties being handed over to RioSecretsDeployStep.\n *\n * @see rio.RioSecretsDeployStep\n * @see rio.RioSecretsDeployStepProps\n */\nexport interface PipelineStackRioSecretsDeployStepProps {\n  readonly secretsFilePath: string;\n}\n\nexport enum PipelineType {\n\n  /**\n   * Self-mutating codepipeline deploying the provided AppStack. Be aware: The provided service won't have access to Kafka.\n   *\n   * Comes with a pre-defined buildspec.yaml for a gradle project. This buildspec.yaml is used for both the main and\n   * branch pipeline and includes the following steps\n   * - build and test of the artifact via command \"./gradlew clean build\".\n   * <p> This gradle command must be provided by the gradle project.\n   * - license-check and upload of license-check-file via command \"./gradlew checkLicenses\".\n   * <p> This gradle command must be provided by the gradle project.\n   * - create a self-signed certificate to make TLS communication possible, e.g. with a loadbalancer.\n   * - build the container image via command \"./gradlew jibBuildTar\".\n   * <p> This gradle command must be provided by the gradle project.\n   */\n  STANDARD_GRADLE = 'STANDARD_GRADLE',\n  STANDARD_GRADLE_KAFKA = 'STANDARD_GRADLE_KAFKA',\n}\n\n/**\n * @param partialBuildSpecPath Path to a custom build-spec file to use for the main and branch pipeline.\n * A pre-defined buildspec.yaml fitting the pipelineType is used if not provided.\n * @param buildEnvironment Custom CodeBuild build environment overriding the rio-specific defaults. The defaults are:\n *          buildImage: AMAZON_LINUX_2_4\n *          computeType: LARGE\n */\nexport interface CodeBuildOptions {\n  readonly partialBuildSpecPath?: string;\n  readonly buildEnvironment?: BuildEnvironment;\n}\n\nexport class PipelineStack extends Stack {\n  private static addLifecycleRuleArtifactBucket(pipeline: pipelines.CodePipeline) {\n    const artifactsBucket = pipeline.pipeline.artifactBucket;\n    (artifactsBucket as s3.Bucket).addLifecycleRule({\n      enabled: true,\n      expiration: Duration.days(90),\n    });\n  }\n\n\n  constructor(scope: Construct, id: string, props: PipelineStackProps) {\n    super(scope, id, props);\n\n    // Creating the main pipeline\n    const bucket = s3.Bucket.fromBucketName(this, 'TriggerBucket', Fn.importValue('bitbucket-integration-s3-bucket'));\n    const hostedZone = rio.RioLandingZone.getDefaultHostedZone(this);\n    const inputMasterTriggerFile = pipelines.CodePipelineSource.s3(bucket, `${props.serviceName}/trigger/${rio.TriggerKey.MASTER_REPO}`, { trigger: codepipeline_actions.S3Trigger.EVENTS });\n    const inputBranchTriggerFile = pipelines.CodePipelineSource.s3(bucket, `${props.serviceName}/trigger/${rio.TriggerKey.BRANCH_REPO}`, { trigger: codepipeline_actions.S3Trigger.EVENTS });\n\n    const ossLicensesBucketParameter = ssm.StringParameter.fromStringParameterAttributes(\n      this,\n      'OssLicensesBucketParameter',\n      {\n        parameterName: '/config/oss-licenses/bucket-name',\n      },\n    );\n    const ossLicensesBucket = s3.Bucket.fromBucketName(\n      this,\n      'OssLicensesBucket',\n      ossLicensesBucketParameter.stringValue,\n    );\n    const serviceKeystorePassword = ssm.StringParameter.fromSecureStringParameterAttributes(this, 'KeystorePassword', {\n      parameterName: `/config/${props.serviceName}/service-keystore-password`,\n      version: 1,\n    });\n    const accountNameParameter = ssm.StringParameter.fromStringParameterName(\n      this,\n      'AccountNameParameter',\n      '/config/account/name',\n    );\n    const nistDataMirrorUrl = ssm.StringParameter.fromSecureStringParameterAttributes(\n      this,\n      'NistDataMirrorUrl',\n      {\n        parameterName: '/config/nist-data-mirror/url', version: 1,\n      },\n    );\n\n    const gradlebuild = new RioGradleCodeBuildProject(this, 'GradleBuild', {\n      input: inputMasterTriggerFile,\n      cdkOutDirectory: 'infrastructure/cdk.out',\n      partialBuildSpec: this.loadBuildSpecFromFile(props?.codeBuildOptions?.partialBuildSpecPath ||\n          this.resolveDefaultBuildSpec( props.pipelineType)),\n      buildEnvironment: props?.codeBuildOptions?.buildEnvironment,\n      hostedZoneName: hostedZone.zoneName,\n      serviceName: props.serviceName,\n    });\n\n    const pipeline = new pipelines.CodePipeline(this, 'Pipeline', {\n      synth: gradlebuild,\n    });\n\n    // CloudAssemblyArtifacts must be smaller than 256MB. Due to the gradle artifact we are exceeding this limit and pipeline fails in deployment step.\n    // Issue is known and not fixed for already some years. See: https://github.com/aws/aws-cdk/issues/9917\n    // Dirty fix: Assets are already uploaded and thus not needed. Remove them from pipeline artifact.\n    const stripAssetsStep = new CodeBuildStep(\n      'StripAssetsFromAssembly', {\n        input: pipeline.cloudAssemblyFileSet,\n        commands: [\n          'S3_PATH=${CODEBUILD_SOURCE_VERSION#\"arn:aws:s3:::\"}',\n          'ZIP_ARCHIVE=$(basename $S3_PATH)',\n          'rm -rfv asset.*',\n          'zip -r -q -A $ZIP_ARCHIVE *',\n          'aws s3 cp $ZIP_ARCHIVE s3://$S3_PATH',\n        ],\n      },\n    );\n\n    pipeline.addWave('BeforeDeploy', { pre: [stripAssetsStep] });\n\n    pipeline.addStage(new Application(this, 'DeployApplicationStacks', {\n      appStackFactory: props.appStackFactory,\n      serviceName: props.serviceName,\n      version: codepipeline.GlobalVariables.executionId,\n    }), {\n      pre: props.secretsDeployStepProps ? [\n        new rio.RioSecretsDeployStep(this, 'DeploySecrets', {\n          ...props.secretsDeployStepProps,\n          input: inputMasterTriggerFile,\n        }),\n      ] : undefined,\n    });\n    pipeline.buildPipeline();\n\n    const rule = new events.Rule(this, 'PipelineSchedule', { schedule: events.Schedule.expression('cron(0 9 ? * 2#1 *)') });\n    rule.addTarget(new targets.CodePipeline(pipeline.pipeline));\n\n    PipelineStack.addLifecycleRuleArtifactBucket(pipeline);\n    ossLicensesBucket.grantReadWrite(gradlebuild);\n    ossLicensesBucketParameter.grantRead(gradlebuild);\n    serviceKeystorePassword.grantRead(gradlebuild);\n    accountNameParameter.grantRead(gradlebuild);\n    if (props.pipelineType === PipelineType.STANDARD_GRADLE_KAFKA) {\n      this.grantPermissionsForKafkaIntegration(gradlebuild);\n    }\n    pipeline.pipeline.artifactBucket.grantReadWrite(stripAssetsStep.project);\n\n    // Branch pipeline\n    const branchBuild = new RioGradleCodeBuildProject(this, 'BranchBuild', {\n      input: inputBranchTriggerFile,\n      cdkOutDirectory: 'infrastructure/cdk.out',\n      partialBuildSpec: this.loadBuildSpecFromFile(props?.codeBuildOptions?.partialBuildSpecPath ||\n          this.resolveDefaultBuildSpec(props.pipelineType)),\n      buildEnvironment: props?.codeBuildOptions?.buildEnvironment,\n      hostedZoneName: hostedZone.zoneName,\n      serviceName: props.serviceName,\n    });\n    const branchPipeline = new pipelines.CodePipeline(this, 'BranchPipeline', {\n      synth: branchBuild,\n      selfMutation: false,\n    });\n    branchPipeline.addWave('BranchBuild').addPost(branchBuild);\n    branchPipeline.buildPipeline();\n\n    PipelineStack.addLifecycleRuleArtifactBucket(branchPipeline);\n    ossLicensesBucket.grantReadWrite(branchBuild);\n    ossLicensesBucketParameter.grantRead(branchBuild);\n    serviceKeystorePassword.grantRead(branchBuild);\n    accountNameParameter.grantRead(branchBuild);\n\n    // Vulnerability Pipeline\n    const vulnerabilityBuild = new RioGradleCodeBuildProject(this, 'VulnerabilityChecks', {\n      input: inputMasterTriggerFile,\n      partialBuildSpec: this.loadBuildSpecFromFile(path.resolve(__dirname, 'buildspecs/buildspec-vulnerability-checks.yaml')),\n      buildEnvironment: props?.codeBuildOptions?.buildEnvironment,\n      hostedZoneName: hostedZone.zoneName,\n      serviceName: props.serviceName,\n    });\n    const vulnerabilityPipeline = new pipelines.CodePipeline(this, 'VulnerabilityPipeline', {\n      synth: vulnerabilityBuild,\n      selfMutation: false,\n    });\n    vulnerabilityPipeline.addWave('VulnerabilityChecks').addPost(vulnerabilityBuild);\n    vulnerabilityPipeline.buildPipeline();\n    nistDataMirrorUrl.grantRead(vulnerabilityBuild);\n\n    const vulnerabilityRule = new events.Rule(this, 'VulnerabilityPipelineSchedule', { schedule: events.Schedule.expression('cron(0 5 ? * MON-FRI *)') });\n    vulnerabilityRule.addTarget(new targets.CodePipeline(vulnerabilityPipeline.pipeline));\n\n    PipelineStack.addLifecycleRuleArtifactBucket(vulnerabilityPipeline);\n\n    new DatadogPipelineErrorAlert(this, 'DatadogPipelineErrorMonitor', {\n      serviceName: props.serviceName,\n      pipelineName: this.renamePipelineToLowerCase(pipeline),\n      accountId: this.account,\n    });\n\n    new DataDogPipelineErrorAlertForVulnerabilityChecks(this, 'DataDogPipelineErrorForVulnerabilityChecksMonitor', {\n      serviceName: props.serviceName,\n      pipelineName: this.renamePipelineToLowerCase(vulnerabilityPipeline),\n      accountId: this.account,\n    });\n\n    const [feature, packageName] = __dirname.split(path.sep).reverse();\n    new Claidometer(this, 'Claidometer', {\n      product: ClaidometerProduct.CDK_CONTRIBUTION,\n      packageName,\n      feature,\n      version: '0.0.3',\n    });\n  }\n\n  private resolveDefaultBuildSpec(pipelineType: PipelineType) {\n    switch (pipelineType) {\n      case PipelineType.STANDARD_GRADLE: return path.resolve(__dirname, 'buildspecs/buildspec.yaml');\n      case PipelineType.STANDARD_GRADLE_KAFKA: return path.resolve(__dirname, 'buildspecs/buildspec_kafka.yaml');\n    }\n  }\n\n  private grantPermissionsForKafkaIntegration(buildProject: RioGradleCodeBuildProject) {\n    const buildProjectRole = buildProject.project.role;\n    if (buildProjectRole) {\n      buildProjectRole.addManagedPolicy(\n        ManagedPolicy.fromManagedPolicyArn(this,\n          'KafkaIntegrationPolicy',\n          Fn.importValue('kafka-integration-policy-arn'),\n        ),\n      );\n    } else {\n      throw Error(`Could not add managed policy for kafka-integration to CodeBuild project of ${buildProject.id} due to missing role.`);\n    }\n  }\n\n  private loadBuildSpecFromFile(buildSpecPath: string): BuildSpec {\n    return BuildSpec.fromObject(yaml.load(fs.readFileSync(buildSpecPath, { encoding: 'utf-8' })) as Object);\n  }\n\n  private renamePipelineToLowerCase(pipeline: pipelines.CodePipeline): Reference {\n    const upperToLowerCr = new cdk.CustomResource(this, `${pipeline.node.id}-UpperToLower`, {\n      resourceType: 'Custom::UpperToLower',\n      serviceToken: UpperToLower.getOrCreate(this).provider.serviceToken,\n      properties: {\n        Upper: pipeline.pipeline.pipelineName,\n      },\n    });\n    return upperToLowerCr.getAtt('Lower');\n  }\n}\n"]}
252
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pipeline-stack.js","sourceRoot":"","sources":["../../../../src/contributions/team-transport-two/pipeline/pipeline-stack.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from 'fs';\nimport * as path from 'path';\nimport * as cdk from 'aws-cdk-lib';\nimport { Duration, Fn, Reference, Stack, StackProps } from 'aws-cdk-lib';\nimport { BuildEnvironment, BuildSpec } from 'aws-cdk-lib/aws-codebuild';\nimport * as codepipeline from 'aws-cdk-lib/aws-codepipeline';\nimport * as codepipeline_actions from 'aws-cdk-lib/aws-codepipeline-actions';\nimport * as events from 'aws-cdk-lib/aws-events';\nimport * as targets from 'aws-cdk-lib/aws-events-targets';\nimport { ManagedPolicy } from 'aws-cdk-lib/aws-iam';\nimport { IHostedZone } from 'aws-cdk-lib/aws-route53';\nimport * as s3 from 'aws-cdk-lib/aws-s3';\nimport { IBucket } from 'aws-cdk-lib/aws-s3';\nimport * as ssm from 'aws-cdk-lib/aws-ssm';\nimport { IStringParameter } from 'aws-cdk-lib/aws-ssm';\nimport * as pipelines from 'aws-cdk-lib/pipelines';\nimport { CodeBuildStep, CodePipelineSource } from 'aws-cdk-lib/pipelines';\nimport { Construct } from 'constructs';\nimport * as yaml from 'js-yaml';\nimport * as rio from '../../../index';\nimport { Claidometer, ClaidometerProduct } from '../../../index';\nimport { UpperToLower } from '../../../watchful/upperToLower';\nimport { Application } from './application-stage';\nimport { RioCodeBuildProject } from './build-project';\nimport { DatadogPipelineErrorAlert, DataDogPipelineErrorAlertForVulnerabilityChecks } from './datadog-monitors';\n\n/**\n * @struct\n */\nexport interface PipelineStackProps extends StackProps {\n  /**\n   * The name of the service\n   * - This has to be the same name as the repository as the s3 trigger file created by the bitbucket-integration uses this as a prefix\n   * - The serviceName is available as an environment variable SERVICE_NAME in the CodeBuildStep\n   * - The serviceName is available in the AppStackProps in the appStackFactory\n   * - The serviceName is available as stackName property in the AppStackProps in the appStackFactory. I.e. the stack is named after the service\n   * - The serviceName is handed over to all Datadog monitors\n   * - RIO convention is that uploaded license-check-files are named after the service, i.e. serviceName.txt\n   */\n  readonly serviceName: string;\n\n  /**\n   * Provider for appStack to be deployed.\n   *\n   * This acts as a wrapper for all resources you want to deploy via the pipeline.\n   */\n  readonly appStackFactory: IAppStackFactory;\n\n  /**\n   * Defines which kind of pipeline is deployed.\n   */\n  readonly pipelineType: PipelineType;\n\n  /**\n   * Path to secrets file containing encrypted secrets.\n   *\n   * The RioSecretsDeployStep is added prior to deployment of the AppStack if a path is provided.\n   *\n   * No secrets are deployed if no path is provided.\n   */\n  readonly secretsDeployStepProps?: PipelineStackRioSecretsDeployStepProps;\n\n  /**\n   * CodeBuild options overriding the rio-specific defaults.\n   */\n  readonly codeBuildOptions?: CodeBuildOptions;\n}\n\n/**\n * Provider interface to create the AppStack within the specific scope\n *\n * Simply place the AppStack creation into the create-method:\n * @example\n *\n * // Use e.g. within ApplicationProps as JSONObject:\n * { create: (construct, props) => new Stack(construct, 'AppStack', props) }\n *\n * // or simply as class interface within your stack implementation:\n * class PipelineStage extends Stage implements IAppStackFactory {\n *   constructor(scope: Construct, id: string, props: StageProps) {\n *     super(scope, id, props);\n *     // your stage definition\n *   }\n *\n *   create(scope: Construct, props: AppStackProps): void {\n *     // app stack initiation goes in here\n *   }\n * }\n */\nexport interface IAppStackFactory {\n  /**\n   * Factory method, being invoked with the specific scope during pipeline instantiation\n   *\n   * @param scope the parent construct for the app stack\n   * @param props the app stacks properties\n   */\n  create(scope: Construct, props: AppStackProps): void;\n}\n\nexport interface AppStackProps extends StackProps {\n  readonly serviceName: string;\n  readonly version: string;\n}\n\n/**\n * Selected properties being handed over to RioSecretsDeployStep.\n *\n * @see rio.RioSecretsDeployStep\n * @see rio.RioSecretsDeployStepProps\n */\nexport interface PipelineStackRioSecretsDeployStepProps {\n  readonly secretsFilePath: string;\n}\n\nexport enum PipelineType {\n\n  /**\n   * Self-mutating codepipeline deploying the provided AppStack. Be aware: The provided service won't have access to Kafka.\n   *\n   * Comes with a pre-defined buildspec.yaml for a gradle project. This buildspec.yaml is used for both the main and\n   * branch pipeline and includes the following steps\n   * - build and test of the artifact via command \"./gradlew clean build\".\n   * <p> This gradle command must be provided by the gradle project.\n   * - license-check and upload of license-check-file via command \"./gradlew checkLicenses\".\n   * <p> This gradle command must be provided by the gradle project.\n   * - create a self-signed certificate to make TLS communication possible, e.g. with a loadbalancer.\n   * - build the container image via command \"./gradlew jibBuildTar\".\n   * <p> This gradle command must be provided by the gradle project.\n   */\n  STANDARD_GRADLE = 'STANDARD_GRADLE',\n  /**\n   * Self-mutating codepipeline deploying the provided AppStack with additional steps needed to access Kafka.\n   *\n   * Comes with a pre-defined buildspec.yaml for a gradle project. This buildspec.yaml is used for both the main and\n   * branch pipeline and includes the following steps\n   * - build and test of the artifact via command \"./gradlew clean build\".\n   * <p> This gradle command must be provided by the gradle project.\n   * - license-check and upload of license-check-file via command \"./gradlew checkLicenses\".\n   * <p> This gradle command must be provided by the gradle project.\n   * - create a self-signed certificate to make TLS communication possible, e.g. with a loadbalancer.\n   * - request a certificate to authenticate against Kafka via mutual TLS\n   * - build the container image via command \"./gradlew jibBuildTar\".\n   * <p> This gradle command must be provided by the gradle project.\n   */\n  STANDARD_GRADLE_KAFKA = 'STANDARD_GRADLE_KAFKA',\n  /**\n   * Self-mutating codepipeline deploying the provided AppStack.\n   *\n   * Comes with a pre-defined buildspec.yaml to synthesize the AppStack.\n   * No branch or vulnerability pipeline is created.\n   */\n  STANDARD_INFRASTRUCTURE = 'STANDARD_INFRASTRUCTURE',\n}\n\n/**\n * @param partialBuildSpecPath Path to a custom build-spec file to use for the main and branch pipeline.\n * A pre-defined buildspec.yaml fitting the pipelineType is used if not provided.\n * @param buildEnvironment Custom CodeBuild build environment overriding the rio-specific defaults. The defaults are:\n *          buildImage: AMAZON_LINUX_2_4\n *          computeType: LARGE\n */\nexport interface CodeBuildOptions {\n  readonly partialBuildSpecPath?: string;\n  readonly buildEnvironment?: BuildEnvironment;\n}\n\nexport class PipelineStack extends Stack {\n  private static addLifecycleRuleArtifactBucket(pipeline: pipelines.CodePipeline) {\n    const artifactsBucket = pipeline.pipeline.artifactBucket;\n    (artifactsBucket as s3.Bucket).addLifecycleRule({\n      enabled: true,\n      expiration: Duration.days(90),\n    });\n  }\n\n  private readonly bucket: IBucket;\n  private readonly hostedZone: IHostedZone;\n  private readonly inputMasterTriggerFile: CodePipelineSource;\n  private readonly inputBranchTriggerFile: CodePipelineSource;\n  private readonly ossLicensesBucketParameter: IStringParameter;\n  private readonly ossLicensesBucket: IBucket;\n  private readonly serviceKeystorePassword: IStringParameter;\n  private readonly accountNameParameter: IStringParameter;\n  private readonly nistDataMirrorUrl: IStringParameter;\n\n  constructor(scope: Construct, id: string, props: PipelineStackProps) {\n    super(scope, id, props);\n\n    // populate external parameters\n    this.bucket = s3.Bucket.fromBucketName(this, 'TriggerBucket', Fn.importValue('bitbucket-integration-s3-bucket'));\n    this.hostedZone = rio.RioLandingZone.getDefaultHostedZone(this);\n    this.inputMasterTriggerFile = pipelines.CodePipelineSource.s3(this.bucket, `${props.serviceName}/trigger/${rio.TriggerKey.MASTER_REPO}`, { trigger: codepipeline_actions.S3Trigger.EVENTS });\n    this.inputBranchTriggerFile = pipelines.CodePipelineSource.s3(this.bucket, `${props.serviceName}/trigger/${rio.TriggerKey.BRANCH_REPO}`, { trigger: codepipeline_actions.S3Trigger.EVENTS });\n\n    this.ossLicensesBucketParameter = ssm.StringParameter.fromStringParameterAttributes(this, 'OssLicensesBucketParameter', { parameterName: '/config/oss-licenses/bucket-name' });\n    this.ossLicensesBucket = s3.Bucket.fromBucketName(this, 'OssLicensesBucket', this.ossLicensesBucketParameter.stringValue);\n    this.serviceKeystorePassword = ssm.StringParameter.fromSecureStringParameterAttributes(this, 'KeystorePassword', {\n      parameterName: `/config/${props.serviceName}/service-keystore-password`,\n      version: 1,\n    });\n    this.accountNameParameter = ssm.StringParameter.fromStringParameterName(this, 'AccountNameParameter', '/config/account/name');\n    this.nistDataMirrorUrl = ssm.StringParameter.fromSecureStringParameterAttributes(this, 'NistDataMirrorUrl', {\n      parameterName: '/config/nist-data-mirror/url',\n      version: 1,\n    });\n\n    this.addMainPipeline(props);\n\n    if (props.pipelineType !== PipelineType.STANDARD_INFRASTRUCTURE) {\n      this.addBranchPipeline(props);\n      this.addVulnerabilityPipeline(props);\n    }\n\n    const [feature, packageName] = __dirname.split(path.sep).reverse();\n    new Claidometer(this, 'Claidometer', {\n      product: ClaidometerProduct.CDK_CONTRIBUTION,\n      packageName,\n      feature,\n      version: '0.0.5',\n    });\n  }\n\n  private addMainPipeline(props: PipelineStackProps) {\n    const mainBuild = new RioCodeBuildProject(this, 'MainBuild', {\n      input: this.inputMasterTriggerFile,\n      cdkOutDirectory: 'infrastructure/cdk.out',\n      partialBuildSpec: this.loadBuildSpecFromFile(props?.codeBuildOptions?.partialBuildSpecPath ||\n        this.resolveDefaultBuildSpec(props.pipelineType)),\n      buildEnvironment: props?.codeBuildOptions?.buildEnvironment,\n      hostedZoneName: this.hostedZone.zoneName,\n      serviceName: props.serviceName,\n    });\n\n    const pipeline = new pipelines.CodePipeline(this, 'Pipeline', {\n      synth: mainBuild,\n    });\n\n    // CloudAssemblyArtifacts must be smaller than 256MB. Due to the gradle artifact we are exceeding this limit and pipeline fails in deployment step.\n    // Issue is known and not fixed for already some years. See: https://github.com/aws/aws-cdk/issues/9917\n    // Dirty fix: Assets are already uploaded and thus not needed. Remove them from pipeline artifact.\n    const stripAssetsStep = new CodeBuildStep(\n      'StripAssetsFromAssembly', {\n        input: pipeline.cloudAssemblyFileSet,\n        commands: [\n          'S3_PATH=${CODEBUILD_SOURCE_VERSION#\"arn:aws:s3:::\"}',\n          'ZIP_ARCHIVE=$(basename $S3_PATH)',\n          'rm -rfv asset.*',\n          'zip -r -q -A $ZIP_ARCHIVE *',\n          'aws s3 cp $ZIP_ARCHIVE s3://$S3_PATH',\n        ],\n      },\n    );\n\n    pipeline.addWave('BeforeDeploy', { pre: [stripAssetsStep] });\n\n    pipeline.addStage(new Application(this, 'DeployApplicationStacks', {\n      appStackFactory: props.appStackFactory,\n      serviceName: props.serviceName,\n      version: codepipeline.GlobalVariables.executionId,\n    }), {\n      pre: props.secretsDeployStepProps ? [\n        new rio.RioSecretsDeployStep(this, 'DeploySecrets', {\n          ...props.secretsDeployStepProps,\n          input: this.inputMasterTriggerFile,\n        }),\n      ] : undefined,\n    });\n    pipeline.buildPipeline();\n\n    const rule = new events.Rule(this, 'PipelineSchedule', { schedule: events.Schedule.expression('cron(0 9 ? * 2#1 *)') });\n    rule.addTarget(new targets.CodePipeline(pipeline.pipeline));\n\n    PipelineStack.addLifecycleRuleArtifactBucket(pipeline);\n    this.ossLicensesBucket.grantReadWrite(mainBuild);\n    this.ossLicensesBucketParameter.grantRead(mainBuild);\n    if (props.pipelineType !== PipelineType.STANDARD_INFRASTRUCTURE) {\n      this.serviceKeystorePassword.grantRead(mainBuild);\n    }\n    this.accountNameParameter.grantRead(mainBuild);\n    if (props.pipelineType === PipelineType.STANDARD_GRADLE_KAFKA) {\n      this.grantPermissionsForKafkaIntegration(mainBuild, 'KafkaIntegrationPolicy');\n    }\n    pipeline.pipeline.artifactBucket.grantReadWrite(stripAssetsStep.project);\n\n    new DatadogPipelineErrorAlert(this, 'DatadogPipelineErrorMonitor', {\n      serviceName: props.serviceName,\n      pipelineName: this.renamePipelineToLowerCase(pipeline),\n      accountId: this.account,\n    });\n  }\n\n  private addBranchPipeline(props: PipelineStackProps) {\n    const branchBuild = new RioCodeBuildProject(this, 'BranchBuild', {\n      input: this.inputBranchTriggerFile,\n      cdkOutDirectory: 'infrastructure/cdk.out',\n      partialBuildSpec: this.loadBuildSpecFromFile(props?.codeBuildOptions?.partialBuildSpecPath ||\n        this.resolveDefaultBuildSpec(props.pipelineType)),\n      buildEnvironment: props?.codeBuildOptions?.buildEnvironment,\n      hostedZoneName: this.hostedZone.zoneName,\n      serviceName: props.serviceName,\n    });\n    const branchPipeline = new pipelines.CodePipeline(this, 'BranchPipeline', {\n      synth: branchBuild,\n      selfMutation: false,\n    });\n    branchPipeline.addWave('BranchBuild').addPost(branchBuild);\n    branchPipeline.buildPipeline();\n\n    PipelineStack.addLifecycleRuleArtifactBucket(branchPipeline);\n    this.ossLicensesBucket.grantReadWrite(branchBuild);\n    this.ossLicensesBucketParameter.grantRead(branchBuild);\n    this.serviceKeystorePassword.grantRead(branchBuild);\n    this.accountNameParameter.grantRead(branchBuild);\n    if (props.pipelineType === PipelineType.STANDARD_GRADLE_KAFKA) {\n      this.grantPermissionsForKafkaIntegration(branchBuild, 'BranchKafkaIntegrationPolicy');\n    }\n  }\n\n  private addVulnerabilityPipeline(props: PipelineStackProps) {\n    const vulnerabilityBuild = new RioCodeBuildProject(this, 'VulnerabilityChecks', {\n      input: this.inputMasterTriggerFile,\n      partialBuildSpec: this.loadBuildSpecFromFile(path.resolve(__dirname, 'buildspecs/buildspec-vulnerability-checks.yaml')),\n      buildEnvironment: props?.codeBuildOptions?.buildEnvironment,\n      hostedZoneName: this.hostedZone.zoneName,\n      serviceName: props.serviceName,\n    });\n    const vulnerabilityPipeline = new pipelines.CodePipeline(this, 'VulnerabilityPipeline', {\n      synth: vulnerabilityBuild,\n      selfMutation: false,\n    });\n    vulnerabilityPipeline.addWave('VulnerabilityChecks').addPost(vulnerabilityBuild);\n    vulnerabilityPipeline.buildPipeline();\n    this.nistDataMirrorUrl.grantRead(vulnerabilityBuild);\n\n    const vulnerabilityRule = new events.Rule(this, 'VulnerabilityPipelineSchedule', { schedule: events.Schedule.expression('cron(0 5 ? * MON-FRI *)') });\n    vulnerabilityRule.addTarget(new targets.CodePipeline(vulnerabilityPipeline.pipeline));\n\n    PipelineStack.addLifecycleRuleArtifactBucket(vulnerabilityPipeline);\n    new DataDogPipelineErrorAlertForVulnerabilityChecks(this, 'DataDogPipelineErrorForVulnerabilityChecksMonitor', {\n      serviceName: props.serviceName,\n      pipelineName: this.renamePipelineToLowerCase(vulnerabilityPipeline),\n      accountId: this.account,\n    });\n  }\n\n  private resolveDefaultBuildSpec(pipelineType: PipelineType) {\n    switch (pipelineType) {\n      case PipelineType.STANDARD_GRADLE:\n        return path.resolve(__dirname, 'buildspecs/buildspec.yaml');\n      case PipelineType.STANDARD_GRADLE_KAFKA:\n        return path.resolve(__dirname, 'buildspecs/buildspec_kafka.yaml');\n      case PipelineType.STANDARD_INFRASTRUCTURE:\n        return path.resolve(__dirname, 'buildspecs/buildspec_infrastructure.yaml');\n    }\n  }\n\n  private grantPermissionsForKafkaIntegration(buildProject: RioCodeBuildProject, policyId: string) {\n    const buildProjectRole = buildProject.project.role;\n    if (buildProjectRole) {\n      buildProjectRole.addManagedPolicy(\n        ManagedPolicy.fromManagedPolicyArn(this,\n          policyId,\n          Fn.importValue('kafka-integration-policy-arn'),\n        ),\n      );\n    } else {\n      throw Error(`Could not add managed policy for kafka-integration to CodeBuild project of ${buildProject.id} due to missing role.`);\n    }\n  }\n\n  private loadBuildSpecFromFile(buildSpecPath: string): BuildSpec {\n    return BuildSpec.fromObject(yaml.load(fs.readFileSync(buildSpecPath, { encoding: 'utf-8' })) as Object);\n  }\n\n  private renamePipelineToLowerCase(pipeline: pipelines.CodePipeline): Reference {\n    const upperToLowerCr = new cdk.CustomResource(this, `${pipeline.node.id}-UpperToLower`, {\n      resourceType: 'Custom::UpperToLower',\n      serviceToken: UpperToLower.getOrCreate(this).provider.serviceToken,\n      properties: {\n        Upper: pipeline.pipeline.pipelineName,\n      },\n    });\n    return upperToLowerCr.getAtt('Lower');\n  }\n}\n"]}
package/package.json CHANGED
@@ -15,7 +15,7 @@
15
15
  ],
16
16
  "main": "lib/index.js",
17
17
  "license": "Apache-2.0",
18
- "version": "4.0.2",
18
+ "version": "4.1.1",
19
19
  "types": "lib/index.d.ts",
20
20
  "stability": "stable",
21
21
  "jsii": {
package/version.json CHANGED
@@ -1,3 +1,3 @@
1
1
  {
2
- "version": "4.0.2"
2
+ "version": "4.1.1"
3
3
  }
@@ -1,216 +0,0 @@
1
- # Changelog
2
-
3
- All notable changes to this project will be documented in this file.
4
-
5
- The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
6
- and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
7
-
8
-
9
- ## [2.0.1] - 2020-08-29
10
- ### Fixed
11
- - Fix issue with `process.argv` when used with interpreters (`coffee`, `ts-node`, etc.), #150.
12
-
13
-
14
- ## [2.0.0] - 2020-08-14
15
- ### Changed
16
- - Full rewrite. Now port from python 3.9.0 & more precise following.
17
- See [doc](./doc) for difference and migration info.
18
- - node.js 10+ required
19
- - Removed most of local docs in favour of original ones.
20
-
21
-
22
- ## [1.0.10] - 2018-02-15
23
- ### Fixed
24
- - Use .concat instead of + for arrays, #122.
25
-
26
-
27
- ## [1.0.9] - 2016-09-29
28
- ### Changed
29
- - Rerelease after 1.0.8 - deps cleanup.
30
-
31
-
32
- ## [1.0.8] - 2016-09-29
33
- ### Changed
34
- - Maintenance (deps bump, fix node 6.5+ tests, coverage report).
35
-
36
-
37
- ## [1.0.7] - 2016-03-17
38
- ### Changed
39
- - Teach `addArgument` to accept string arg names. #97, @tomxtobin.
40
-
41
-
42
- ## [1.0.6] - 2016-02-06
43
- ### Changed
44
- - Maintenance: moved to eslint & updated CS.
45
-
46
-
47
- ## [1.0.5] - 2016-02-05
48
- ### Changed
49
- - Removed lodash dependency to significantly reduce install size.
50
- Thanks to @mourner.
51
-
52
-
53
- ## [1.0.4] - 2016-01-17
54
- ### Changed
55
- - Maintenance: lodash update to 4.0.0.
56
-
57
-
58
- ## [1.0.3] - 2015-10-27
59
- ### Fixed
60
- - Fix parse `=` in args: `--examplepath="C:\myfolder\env=x64"`. #84, @CatWithApple.
61
-
62
-
63
- ## [1.0.2] - 2015-03-22
64
- ### Changed
65
- - Relaxed lodash version dependency.
66
-
67
-
68
- ## [1.0.1] - 2015-02-20
69
- ### Changed
70
- - Changed dependencies to be compatible with ancient nodejs.
71
-
72
-
73
- ## [1.0.0] - 2015-02-19
74
- ### Changed
75
- - Maintenance release.
76
- - Replaced `underscore` with `lodash`.
77
- - Bumped version to 1.0.0 to better reflect semver meaning.
78
- - HISTORY.md -> CHANGELOG.md
79
-
80
-
81
- ## [0.1.16] - 2013-12-01
82
- ### Changed
83
- - Maintenance release. Updated dependencies and docs.
84
-
85
-
86
- ## [0.1.15] - 2013-05-13
87
- ### Fixed
88
- - Fixed #55, @trebor89
89
-
90
-
91
- ## [0.1.14] - 2013-05-12
92
- ### Fixed
93
- - Fixed #62, @maxtaco
94
-
95
-
96
- ## [0.1.13] - 2013-04-08
97
- ### Changed
98
- - Added `.npmignore` to reduce package size
99
-
100
-
101
- ## [0.1.12] - 2013-02-10
102
- ### Fixed
103
- - Fixed conflictHandler (#46), @hpaulj
104
-
105
-
106
- ## [0.1.11] - 2013-02-07
107
- ### Added
108
- - Added 70+ tests (ported from python), @hpaulj
109
- - Added conflictHandler, @applepicke
110
- - Added fromfilePrefixChar, @hpaulj
111
-
112
- ### Fixed
113
- - Multiple bugfixes, @hpaulj
114
-
115
-
116
- ## [0.1.10] - 2012-12-30
117
- ### Added
118
- - Added [mutual exclusion](http://docs.python.org/dev/library/argparse.html#mutual-exclusion)
119
- support, thanks to @hpaulj
120
-
121
- ### Fixed
122
- - Fixed options check for `storeConst` & `appendConst` actions, thanks to @hpaulj
123
-
124
-
125
- ## [0.1.9] - 2012-12-27
126
- ### Fixed
127
- - Fixed option dest interferens with other options (issue #23), thanks to @hpaulj
128
- - Fixed default value behavior with `*` positionals, thanks to @hpaulj
129
- - Improve `getDefault()` behavior, thanks to @hpaulj
130
- - Improve negative argument parsing, thanks to @hpaulj
131
-
132
-
133
- ## [0.1.8] - 2012-12-01
134
- ### Fixed
135
- - Fixed parser parents (issue #19), thanks to @hpaulj
136
- - Fixed negative argument parse (issue #20), thanks to @hpaulj
137
-
138
-
139
- ## [0.1.7] - 2012-10-14
140
- ### Fixed
141
- - Fixed 'choices' argument parse (issue #16)
142
- - Fixed stderr output (issue #15)
143
-
144
-
145
- ## [0.1.6] - 2012-09-09
146
- ### Fixed
147
- - Fixed check for conflict of options (thanks to @tomxtobin)
148
-
149
-
150
- ## [0.1.5] - 2012-09-03
151
- ### Fixed
152
- - Fix parser #setDefaults method (thanks to @tomxtobin)
153
-
154
-
155
- ## [0.1.4] - 2012-07-30
156
- ### Fixed
157
- - Fixed pseudo-argument support (thanks to @CGamesPlay)
158
- - Fixed addHelp default (should be true), if not set (thanks to @benblank)
159
-
160
-
161
- ## [0.1.3] - 2012-06-27
162
- ### Fixed
163
- - Fixed formatter api name: Formatter -> HelpFormatter
164
-
165
-
166
- ## [0.1.2] - 2012-05-29
167
- ### Fixed
168
- - Removed excess whitespace in help
169
- - Fixed error reporting, when parcer with subcommands
170
- called with empty arguments
171
-
172
- ### Added
173
- - Added basic tests
174
-
175
-
176
- ## [0.1.1] - 2012-05-23
177
- ### Fixed
178
- - Fixed line wrapping in help formatter
179
- - Added better error reporting on invalid arguments
180
-
181
-
182
- ## [0.1.0] - 2012-05-16
183
- ### Added
184
- - First release.
185
-
186
-
187
- [2.0.1]: https://github.com/nodeca/argparse/compare/2.0.0...2.0.1
188
- [2.0.0]: https://github.com/nodeca/argparse/compare/1.0.10...2.0.0
189
- [1.0.10]: https://github.com/nodeca/argparse/compare/1.0.9...1.0.10
190
- [1.0.9]: https://github.com/nodeca/argparse/compare/1.0.8...1.0.9
191
- [1.0.8]: https://github.com/nodeca/argparse/compare/1.0.7...1.0.8
192
- [1.0.7]: https://github.com/nodeca/argparse/compare/1.0.6...1.0.7
193
- [1.0.6]: https://github.com/nodeca/argparse/compare/1.0.5...1.0.6
194
- [1.0.5]: https://github.com/nodeca/argparse/compare/1.0.4...1.0.5
195
- [1.0.4]: https://github.com/nodeca/argparse/compare/1.0.3...1.0.4
196
- [1.0.3]: https://github.com/nodeca/argparse/compare/1.0.2...1.0.3
197
- [1.0.2]: https://github.com/nodeca/argparse/compare/1.0.1...1.0.2
198
- [1.0.1]: https://github.com/nodeca/argparse/compare/1.0.0...1.0.1
199
- [1.0.0]: https://github.com/nodeca/argparse/compare/0.1.16...1.0.0
200
- [0.1.16]: https://github.com/nodeca/argparse/compare/0.1.15...0.1.16
201
- [0.1.15]: https://github.com/nodeca/argparse/compare/0.1.14...0.1.15
202
- [0.1.14]: https://github.com/nodeca/argparse/compare/0.1.13...0.1.14
203
- [0.1.13]: https://github.com/nodeca/argparse/compare/0.1.12...0.1.13
204
- [0.1.12]: https://github.com/nodeca/argparse/compare/0.1.11...0.1.12
205
- [0.1.11]: https://github.com/nodeca/argparse/compare/0.1.10...0.1.11
206
- [0.1.10]: https://github.com/nodeca/argparse/compare/0.1.9...0.1.10
207
- [0.1.9]: https://github.com/nodeca/argparse/compare/0.1.8...0.1.9
208
- [0.1.8]: https://github.com/nodeca/argparse/compare/0.1.7...0.1.8
209
- [0.1.7]: https://github.com/nodeca/argparse/compare/0.1.6...0.1.7
210
- [0.1.6]: https://github.com/nodeca/argparse/compare/0.1.5...0.1.6
211
- [0.1.5]: https://github.com/nodeca/argparse/compare/0.1.4...0.1.5
212
- [0.1.4]: https://github.com/nodeca/argparse/compare/0.1.3...0.1.4
213
- [0.1.3]: https://github.com/nodeca/argparse/compare/0.1.2...0.1.3
214
- [0.1.2]: https://github.com/nodeca/argparse/compare/0.1.1...0.1.2
215
- [0.1.1]: https://github.com/nodeca/argparse/compare/0.1.0...0.1.1
216
- [0.1.0]: https://github.com/nodeca/argparse/releases/tag/0.1.0