1 | ;
|
2 | var _a;
|
3 | Object.defineProperty(exports, "__esModule", { value: true });
|
4 | exports.DefaultStackSynthesizer = exports.BOOTSTRAP_QUALIFIER_CONTEXT = void 0;
|
5 | const jsiiDeprecationWarnings = require("../../.warnings.jsii.js");
|
6 | const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
7 | const cxapi = require("@aws-cdk/cx-api");
|
8 | const cfn_fn_1 = require("../cfn-fn");
|
9 | const cfn_parameter_1 = require("../cfn-parameter");
|
10 | const cfn_rule_1 = require("../cfn-rule");
|
11 | const token_1 = require("../token");
|
12 | const _asset_manifest_builder_1 = require("./_asset-manifest-builder");
|
13 | const _shared_1 = require("./_shared");
|
14 | const stack_synthesizer_1 = require("./stack-synthesizer");
|
15 | exports.BOOTSTRAP_QUALIFIER_CONTEXT = '@aws-cdk/core:bootstrapQualifier';
|
16 | /* eslint-disable max-len */
|
17 | /**
|
18 | * The minimum bootstrap stack version required by this app.
|
19 | */
|
20 | const MIN_BOOTSTRAP_STACK_VERSION = 6;
|
21 | /**
|
22 | * The minimum bootstrap stack version required
|
23 | * to use the lookup role.
|
24 | */
|
25 | const MIN_LOOKUP_ROLE_BOOTSTRAP_STACK_VERSION = 8;
|
26 | /**
|
27 | * Uses conventionally named roles and asset storage locations
|
28 | *
|
29 | * This synthesizer:
|
30 | *
|
31 | * - Supports cross-account deployments (the CLI can have credentials to one
|
32 | * account, and you can still deploy to another account by assuming roles with
|
33 | * well-known names in the other account).
|
34 | * - Supports the **CDK Pipelines** library.
|
35 | *
|
36 | * Requires the environment to have been bootstrapped with Bootstrap Stack V2
|
37 | * (also known as "modern bootstrap stack"). The synthesizer adds a version
|
38 | * check to the template, to make sure the bootstrap stack is recent enough
|
39 | * to support all features expected by this synthesizer.
|
40 | */
|
41 | class DefaultStackSynthesizer extends stack_synthesizer_1.StackSynthesizer {
|
42 | constructor(props = {}) {
|
43 | super();
|
44 | this.props = props;
|
45 | this.assetManifest = new _asset_manifest_builder_1.AssetManifestBuilder();
|
46 | try {
|
47 | jsiiDeprecationWarnings._aws_cdk_core_DefaultStackSynthesizerProps(props);
|
48 | }
|
49 | catch (error) {
|
50 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
51 | Error.captureStackTrace(error, DefaultStackSynthesizer);
|
52 | }
|
53 | throw error;
|
54 | }
|
55 | this.useLookupRoleForStackOperations = props.useLookupRoleForStackOperations ?? true;
|
56 | for (const key in props) {
|
57 | if (props.hasOwnProperty(key)) {
|
58 | validateNoToken(key);
|
59 | }
|
60 | }
|
61 | function validateNoToken(key) {
|
62 | const prop = props[key];
|
63 | if (typeof prop === 'string' && token_1.Token.isUnresolved(prop)) {
|
64 | throw new Error(`DefaultSynthesizer property '${key}' cannot contain tokens; only the following placeholder strings are allowed: ` + [
|
65 | '${Qualifier}',
|
66 | cxapi.EnvironmentPlaceholders.CURRENT_REGION,
|
67 | cxapi.EnvironmentPlaceholders.CURRENT_ACCOUNT,
|
68 | cxapi.EnvironmentPlaceholders.CURRENT_PARTITION,
|
69 | ].join(', '));
|
70 | }
|
71 | }
|
72 | }
|
73 | bind(stack) {
|
74 | try {
|
75 | jsiiDeprecationWarnings._aws_cdk_core_Stack(stack);
|
76 | }
|
77 | catch (error) {
|
78 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
79 | Error.captureStackTrace(error, this.bind);
|
80 | }
|
81 | throw error;
|
82 | }
|
83 | if (this._stack !== undefined) {
|
84 | throw new Error('A StackSynthesizer can only be used for one Stack: create a new instance to use with a different Stack');
|
85 | }
|
86 | this._stack = stack;
|
87 | const qualifier = this.props.qualifier ?? stack.node.tryGetContext(exports.BOOTSTRAP_QUALIFIER_CONTEXT) ?? DefaultStackSynthesizer.DEFAULT_QUALIFIER;
|
88 | this.qualifier = qualifier;
|
89 | const spec = new _shared_1.StringSpecializer(stack, qualifier);
|
90 | /* eslint-disable max-len */
|
91 | this.bucketName = spec.specialize(this.props.fileAssetsBucketName ?? DefaultStackSynthesizer.DEFAULT_FILE_ASSETS_BUCKET_NAME);
|
92 | this.repositoryName = spec.specialize(this.props.imageAssetsRepositoryName ?? DefaultStackSynthesizer.DEFAULT_IMAGE_ASSETS_REPOSITORY_NAME);
|
93 | this._deployRoleArn = spec.specialize(this.props.deployRoleArn ?? DefaultStackSynthesizer.DEFAULT_DEPLOY_ROLE_ARN);
|
94 | this._cloudFormationExecutionRoleArn = spec.specialize(this.props.cloudFormationExecutionRole ?? DefaultStackSynthesizer.DEFAULT_CLOUDFORMATION_ROLE_ARN);
|
95 | this.fileAssetPublishingRoleArn = spec.specialize(this.props.fileAssetPublishingRoleArn ?? DefaultStackSynthesizer.DEFAULT_FILE_ASSET_PUBLISHING_ROLE_ARN);
|
96 | this.imageAssetPublishingRoleArn = spec.specialize(this.props.imageAssetPublishingRoleArn ?? DefaultStackSynthesizer.DEFAULT_IMAGE_ASSET_PUBLISHING_ROLE_ARN);
|
97 | this.lookupRoleArn = spec.specialize(this.props.lookupRoleArn ?? DefaultStackSynthesizer.DEFAULT_LOOKUP_ROLE_ARN);
|
98 | this.bucketPrefix = spec.specialize(this.props.bucketPrefix ?? DefaultStackSynthesizer.DEFAULT_FILE_ASSET_PREFIX);
|
99 | this.dockerTagPrefix = spec.specialize(this.props.dockerTagPrefix ?? DefaultStackSynthesizer.DEFAULT_DOCKER_ASSET_PREFIX);
|
100 | this.bootstrapStackVersionSsmParameter = spec.qualifierOnly(this.props.bootstrapStackVersionSsmParameter ?? DefaultStackSynthesizer.DEFAULT_BOOTSTRAP_STACK_VERSION_SSM_PARAMETER);
|
101 | }
|
102 | addFileAsset(asset) {
|
103 | try {
|
104 | jsiiDeprecationWarnings._aws_cdk_core_FileAssetSource(asset);
|
105 | }
|
106 | catch (error) {
|
107 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
108 | Error.captureStackTrace(error, this.addFileAsset);
|
109 | }
|
110 | throw error;
|
111 | }
|
112 | _shared_1.assertBound(this.stack);
|
113 | _shared_1.assertBound(this.bucketName);
|
114 | _shared_1.assertBound(this.bucketPrefix);
|
115 | return this.assetManifest.addFileAssetDefault(asset, this.stack, this.bucketName, this.bucketPrefix, {
|
116 | assumeRoleArn: this.fileAssetPublishingRoleArn,
|
117 | assumeRoleExternalId: this.props.fileAssetPublishingExternalId,
|
118 | });
|
119 | }
|
120 | addDockerImageAsset(asset) {
|
121 | try {
|
122 | jsiiDeprecationWarnings._aws_cdk_core_DockerImageAssetSource(asset);
|
123 | }
|
124 | catch (error) {
|
125 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
126 | Error.captureStackTrace(error, this.addDockerImageAsset);
|
127 | }
|
128 | throw error;
|
129 | }
|
130 | _shared_1.assertBound(this.stack);
|
131 | _shared_1.assertBound(this.repositoryName);
|
132 | _shared_1.assertBound(this.dockerTagPrefix);
|
133 | return this.assetManifest.addDockerImageAssetDefault(asset, this.stack, this.repositoryName, this.dockerTagPrefix, {
|
134 | assumeRoleArn: this.imageAssetPublishingRoleArn,
|
135 | assumeRoleExternalId: this.props.imageAssetPublishingExternalId,
|
136 | });
|
137 | }
|
138 | synthesizeStackTemplate(stack, session) {
|
139 | try {
|
140 | jsiiDeprecationWarnings._aws_cdk_core_Stack(stack);
|
141 | jsiiDeprecationWarnings._aws_cdk_core_ISynthesisSession(session);
|
142 | }
|
143 | catch (error) {
|
144 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
145 | Error.captureStackTrace(error, this.synthesizeStackTemplate);
|
146 | }
|
147 | throw error;
|
148 | }
|
149 | stack._synthesizeTemplate(session, this.lookupRoleArn);
|
150 | }
|
151 | /**
|
152 | * Synthesize the associated stack to the session
|
153 | */
|
154 | synthesize(session) {
|
155 | try {
|
156 | jsiiDeprecationWarnings._aws_cdk_core_ISynthesisSession(session);
|
157 | }
|
158 | catch (error) {
|
159 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
160 | Error.captureStackTrace(error, this.synthesize);
|
161 | }
|
162 | throw error;
|
163 | }
|
164 | _shared_1.assertBound(this.stack);
|
165 | _shared_1.assertBound(this.qualifier);
|
166 | // Must be done here -- if it's done in bind() (called in the Stack's constructor)
|
167 | // then it will become impossible to set context after that.
|
168 | //
|
169 | // If it's done AFTER _synthesizeTemplate(), then the template won't contain the
|
170 | // right constructs.
|
171 | if (this.props.generateBootstrapVersionRule ?? true) {
|
172 | addBootstrapVersionRule(this.stack, MIN_BOOTSTRAP_STACK_VERSION, this.bootstrapStackVersionSsmParameter);
|
173 | }
|
174 | this.synthesizeStackTemplate(this.stack, session);
|
175 | const templateAsset = this.addFileAsset(_shared_1.stackTemplateFileAsset(this.stack, session));
|
176 | const assetManifestId = this.assetManifest.writeManifest(this.stack, session, {
|
177 | requiresBootstrapStackVersion: MIN_BOOTSTRAP_STACK_VERSION,
|
178 | bootstrapStackVersionSsmParameter: this.bootstrapStackVersionSsmParameter,
|
179 | });
|
180 | this.emitStackArtifact(this.stack, session, {
|
181 | assumeRoleExternalId: this.props.deployRoleExternalId,
|
182 | assumeRoleArn: this._deployRoleArn,
|
183 | cloudFormationExecutionRoleArn: this._cloudFormationExecutionRoleArn,
|
184 | stackTemplateAssetObjectUrl: templateAsset.s3ObjectUrlWithPlaceholders,
|
185 | requiresBootstrapStackVersion: MIN_BOOTSTRAP_STACK_VERSION,
|
186 | bootstrapStackVersionSsmParameter: this.bootstrapStackVersionSsmParameter,
|
187 | additionalDependencies: [assetManifestId],
|
188 | lookupRole: this.useLookupRoleForStackOperations && this.lookupRoleArn ? {
|
189 | arn: this.lookupRoleArn,
|
190 | assumeRoleExternalId: this.props.lookupRoleExternalId,
|
191 | requiresBootstrapStackVersion: MIN_LOOKUP_ROLE_BOOTSTRAP_STACK_VERSION,
|
192 | bootstrapStackVersionSsmParameter: this.bootstrapStackVersionSsmParameter,
|
193 | } : undefined,
|
194 | });
|
195 | }
|
196 | /**
|
197 | * Returns the ARN of the deploy Role.
|
198 | */
|
199 | get deployRoleArn() {
|
200 | if (!this._deployRoleArn) {
|
201 | throw new Error('deployRoleArn getter can only be called after the synthesizer has been bound to a Stack');
|
202 | }
|
203 | return this._deployRoleArn;
|
204 | }
|
205 | /**
|
206 | * Returns the ARN of the CFN execution Role.
|
207 | */
|
208 | get cloudFormationExecutionRoleArn() {
|
209 | if (!this._cloudFormationExecutionRoleArn) {
|
210 | throw new Error('cloudFormationExecutionRoleArn getter can only be called after the synthesizer has been bound to a Stack');
|
211 | }
|
212 | return this._cloudFormationExecutionRoleArn;
|
213 | }
|
214 | get stack() {
|
215 | return this._stack;
|
216 | }
|
217 | }
|
218 | exports.DefaultStackSynthesizer = DefaultStackSynthesizer;
|
219 | _a = JSII_RTTI_SYMBOL_1;
|
220 | DefaultStackSynthesizer[_a] = { fqn: "@aws-cdk/core.DefaultStackSynthesizer", version: "1.204.0" };
|
221 | /**
|
222 | * Default ARN qualifier
|
223 | */
|
224 | DefaultStackSynthesizer.DEFAULT_QUALIFIER = 'hnb659fds';
|
225 | /**
|
226 | * Default CloudFormation role ARN.
|
227 | */
|
228 | DefaultStackSynthesizer.DEFAULT_CLOUDFORMATION_ROLE_ARN = 'arn:${AWS::Partition}:iam::${AWS::AccountId}:role/cdk-${Qualifier}-cfn-exec-role-${AWS::AccountId}-${AWS::Region}';
|
229 | /**
|
230 | * Default deploy role ARN.
|
231 | */
|
232 | DefaultStackSynthesizer.DEFAULT_DEPLOY_ROLE_ARN = 'arn:${AWS::Partition}:iam::${AWS::AccountId}:role/cdk-${Qualifier}-deploy-role-${AWS::AccountId}-${AWS::Region}';
|
233 | /**
|
234 | * Default asset publishing role ARN for file (S3) assets.
|
235 | */
|
236 | DefaultStackSynthesizer.DEFAULT_FILE_ASSET_PUBLISHING_ROLE_ARN = 'arn:${AWS::Partition}:iam::${AWS::AccountId}:role/cdk-${Qualifier}-file-publishing-role-${AWS::AccountId}-${AWS::Region}';
|
237 | /**
|
238 | * Default asset publishing role ARN for image (ECR) assets.
|
239 | */
|
240 | DefaultStackSynthesizer.DEFAULT_IMAGE_ASSET_PUBLISHING_ROLE_ARN = 'arn:${AWS::Partition}:iam::${AWS::AccountId}:role/cdk-${Qualifier}-image-publishing-role-${AWS::AccountId}-${AWS::Region}';
|
241 | /**
|
242 | * Default lookup role ARN for missing values.
|
243 | */
|
244 | DefaultStackSynthesizer.DEFAULT_LOOKUP_ROLE_ARN = 'arn:${AWS::Partition}:iam::${AWS::AccountId}:role/cdk-${Qualifier}-lookup-role-${AWS::AccountId}-${AWS::Region}';
|
245 | /**
|
246 | * Default image assets repository name
|
247 | */
|
248 | DefaultStackSynthesizer.DEFAULT_IMAGE_ASSETS_REPOSITORY_NAME = 'cdk-${Qualifier}-container-assets-${AWS::AccountId}-${AWS::Region}';
|
249 | /**
|
250 | * Default file assets bucket name
|
251 | */
|
252 | DefaultStackSynthesizer.DEFAULT_FILE_ASSETS_BUCKET_NAME = 'cdk-${Qualifier}-assets-${AWS::AccountId}-${AWS::Region}';
|
253 | /**
|
254 | * Name of the CloudFormation Export with the asset key name
|
255 | */
|
256 | DefaultStackSynthesizer.DEFAULT_FILE_ASSET_KEY_ARN_EXPORT_NAME = 'CdkBootstrap-${Qualifier}-FileAssetKeyArn';
|
257 | /**
|
258 | * Default file asset prefix
|
259 | */
|
260 | DefaultStackSynthesizer.DEFAULT_FILE_ASSET_PREFIX = '';
|
261 | /**
|
262 | * Default Docker asset prefix
|
263 | */
|
264 | DefaultStackSynthesizer.DEFAULT_DOCKER_ASSET_PREFIX = '';
|
265 | /**
|
266 | * Default bootstrap stack version SSM parameter.
|
267 | */
|
268 | DefaultStackSynthesizer.DEFAULT_BOOTSTRAP_STACK_VERSION_SSM_PARAMETER = '/cdk-bootstrap/${Qualifier}/version';
|
269 | /**
|
270 | * Add a CfnRule to the Stack which checks the current version of the bootstrap stack this template is targeting
|
271 | *
|
272 | * The CLI normally checks this, but in a pipeline the CLI is not involved
|
273 | * so we encode this rule into the template in a way that CloudFormation will check it.
|
274 | */
|
275 | function addBootstrapVersionRule(stack, requiredVersion, bootstrapStackVersionSsmParameter) {
|
276 | // Because of https://github.com/aws/aws-cdk/blob/master/packages/assert-internal/lib/synth-utils.ts#L74
|
277 | // synthesize() may be called more than once on a stack in unit tests, and the below would break
|
278 | // if we execute it a second time. Guard against the constructs already existing.
|
279 | if (stack.node.tryFindChild('BootstrapVersion')) {
|
280 | return;
|
281 | }
|
282 | const param = new cfn_parameter_1.CfnParameter(stack, 'BootstrapVersion', {
|
283 | type: 'AWS::SSM::Parameter::Value<String>',
|
284 | description: `Version of the CDK Bootstrap resources in this environment, automatically retrieved from SSM Parameter Store. ${cxapi.SSMPARAM_NO_INVALIDATE}`,
|
285 | default: bootstrapStackVersionSsmParameter,
|
286 | });
|
287 | // There is no >= check in CloudFormation, so we have to check the number
|
288 | // is NOT in [1, 2, 3, ... <required> - 1]
|
289 | const oldVersions = range(1, requiredVersion).map(n => `${n}`);
|
290 | new cfn_rule_1.CfnRule(stack, 'CheckBootstrapVersion', {
|
291 | assertions: [
|
292 | {
|
293 | assert: cfn_fn_1.Fn.conditionNot(cfn_fn_1.Fn.conditionContains(oldVersions, param.valueAsString)),
|
294 | assertDescription: `CDK bootstrap stack version ${requiredVersion} required. Please run 'cdk bootstrap' with a recent version of the CDK CLI.`,
|
295 | },
|
296 | ],
|
297 | });
|
298 | }
|
299 | function range(startIncl, endExcl) {
|
300 | const ret = new Array();
|
301 | for (let i = startIncl; i < endExcl; i++) {
|
302 | ret.push(i);
|
303 | }
|
304 | return ret;
|
305 | }
|
306 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"default-synthesizer.js","sourceRoot":"","sources":["default-synthesizer.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cxapi from '@aws-cdk/cx-api';\nimport { DockerImageAssetLocation, DockerImageAssetSource, FileAssetLocation, FileAssetSource } from '../assets';\nimport { Fn } from '../cfn-fn';\nimport { CfnParameter } from '../cfn-parameter';\nimport { CfnRule } from '../cfn-rule';\nimport { ISynthesisSession } from '../construct-compat';\nimport { Stack } from '../stack';\nimport { Token } from '../token';\nimport { AssetManifestBuilder } from './_asset-manifest-builder';\nimport { assertBound, StringSpecializer, stackTemplateFileAsset } from './_shared';\nimport { StackSynthesizer } from './stack-synthesizer';\n\nexport const BOOTSTRAP_QUALIFIER_CONTEXT = '@aws-cdk/core:bootstrapQualifier';\n\n/* eslint-disable max-len */\n\n/**\n * The minimum bootstrap stack version required by this app.\n */\nconst MIN_BOOTSTRAP_STACK_VERSION = 6;\n\n/**\n * The minimum bootstrap stack version required\n * to use the lookup role.\n */\nconst MIN_LOOKUP_ROLE_BOOTSTRAP_STACK_VERSION = 8;\n\n/**\n * Configuration properties for DefaultStackSynthesizer\n */\nexport interface DefaultStackSynthesizerProps {\n  /**\n   * Name of the S3 bucket to hold file assets\n   *\n   * You must supply this if you have given a non-standard name to the staging bucket.\n   *\n   * The placeholders `${Qualifier}`, `${AWS::AccountId}` and `${AWS::Region}` will\n   * be replaced with the values of qualifier and the stack's account and region,\n   * respectively.\n   *\n   * @default DefaultStackSynthesizer.DEFAULT_FILE_ASSETS_BUCKET_NAME\n   */\n  readonly fileAssetsBucketName?: string;\n\n  /**\n   * Name of the ECR repository to hold Docker Image assets\n   *\n   * You must supply this if you have given a non-standard name to the ECR repository.\n   *\n   * The placeholders `${Qualifier}`, `${AWS::AccountId}` and `${AWS::Region}` will\n   * be replaced with the values of qualifier and the stack's account and region,\n   * respectively.\n   *\n   * @default DefaultStackSynthesizer.DEFAULT_IMAGE_ASSETS_REPOSITORY_NAME\n   */\n  readonly imageAssetsRepositoryName?: string;\n\n  /**\n   * The role to use to publish file assets to the S3 bucket in this environment\n   *\n   * You must supply this if you have given a non-standard name to the publishing role.\n   *\n   * The placeholders `${Qualifier}`, `${AWS::AccountId}` and `${AWS::Region}` will\n   * be replaced with the values of qualifier and the stack's account and region,\n   * respectively.\n   *\n   * @default DefaultStackSynthesizer.DEFAULT_FILE_ASSET_PUBLISHING_ROLE_ARN\n   */\n  readonly fileAssetPublishingRoleArn?: string;\n\n  /**\n   * External ID to use when assuming role for file asset publishing\n   *\n   * @default - No external ID\n   */\n  readonly fileAssetPublishingExternalId?: string;\n\n  /**\n   * The role to use to publish image assets to the ECR repository in this environment\n   *\n   * You must supply this if you have given a non-standard name to the publishing role.\n   *\n   * The placeholders `${Qualifier}`, `${AWS::AccountId}` and `${AWS::Region}` will\n   * be replaced with the values of qualifier and the stack's account and region,\n   * respectively.\n   *\n   * @default DefaultStackSynthesizer.DEFAULT_IMAGE_ASSET_PUBLISHING_ROLE_ARN\n   */\n  readonly imageAssetPublishingRoleArn?: string;\n\n  /**\n   * The role to use to look up values from the target AWS account during synthesis\n   *\n   * @default - None\n   */\n  readonly lookupRoleArn?: string;\n\n  /**\n   * External ID to use when assuming lookup role\n   *\n   * @default - No external ID\n   */\n  readonly lookupRoleExternalId?: string;\n\n  /**\n   * Use the bootstrapped lookup role for (read-only) stack operations\n   *\n   * Use the lookup role when performing a `cdk diff`. If set to `false`, the\n   * `deploy role` credentials will be used to perform a `cdk diff`.\n   *\n   * Requires bootstrap stack version 8.\n   *\n   * @default true\n   */\n  readonly useLookupRoleForStackOperations?: boolean;\n\n  /**\n   * External ID to use when assuming role for image asset publishing\n   *\n   * @default - No external ID\n   */\n  readonly imageAssetPublishingExternalId?: string;\n\n  /**\n   * External ID to use when assuming role for cloudformation deployments\n   *\n   * @default - No external ID\n   */\n  readonly deployRoleExternalId?: string;\n\n  /**\n   * The role to assume to initiate a deployment in this environment\n   *\n   * You must supply this if you have given a non-standard name to the publishing role.\n   *\n   * The placeholders `${Qualifier}`, `${AWS::AccountId}` and `${AWS::Region}` will\n   * be replaced with the values of qualifier and the stack's account and region,\n   * respectively.\n   *\n   * @default DefaultStackSynthesizer.DEFAULT_DEPLOY_ROLE_ARN\n   */\n  readonly deployRoleArn?: string;\n\n  /**\n   * The role CloudFormation will assume when deploying the Stack\n   *\n   * You must supply this if you have given a non-standard name to the execution role.\n   *\n   * The placeholders `${Qualifier}`, `${AWS::AccountId}` and `${AWS::Region}` will\n   * be replaced with the values of qualifier and the stack's account and region,\n   * respectively.\n   *\n   * @default DefaultStackSynthesizer.DEFAULT_CLOUDFORMATION_ROLE_ARN\n   */\n  readonly cloudFormationExecutionRole?: string;\n\n  /**\n   * Name of the CloudFormation Export with the asset key name\n   *\n   * You must supply this if you have given a non-standard name to the KMS key export\n   *\n   * The placeholders `${Qualifier}`, `${AWS::AccountId}` and `${AWS::Region}` will\n   * be replaced with the values of qualifier and the stack's account and region,\n   * respectively.\n   *\n   * @default DefaultStackSynthesizer.DEFAULT_FILE_ASSET_KEY_ARN_EXPORT_NAME\n   * @deprecated This property is not used anymore\n   */\n  readonly fileAssetKeyArnExportName?: string;\n\n  /**\n   * Qualifier to disambiguate multiple environments in the same account\n   *\n   * You can use this and leave the other naming properties empty if you have deployed\n   * the bootstrap environment with standard names but only differnet qualifiers.\n   *\n   * @default - Value of context key '@aws-cdk/core:bootstrapQualifier' if set, otherwise `DefaultStackSynthesizer.DEFAULT_QUALIFIER`\n   */\n  readonly qualifier?: string;\n\n  /**\n   * Whether to add a Rule to the stack template verifying the bootstrap stack version\n   *\n   * This generally should be left set to `true`, unless you explicitly\n   * want to be able to deploy to an unbootstrapped environment.\n   *\n   * @default true\n   */\n  readonly generateBootstrapVersionRule?: boolean;\n\n  /**\n   * bucketPrefix to use while storing S3 Assets\n   *\n   * @default - DefaultStackSynthesizer.DEFAULT_FILE_ASSET_PREFIX\n   */\n  readonly bucketPrefix?: string;\n\n  /**\n   * A prefix to use while tagging and uploading Docker images to ECR.\n   *\n   * This does not add any separators - the source hash will be appended to\n   * this string directly.\n   *\n   * @default - DefaultStackSynthesizer.DEFAULT_DOCKER_ASSET_PREFIX\n   */\n  readonly dockerTagPrefix?: string;\n\n  /**\n   * Bootstrap stack version SSM parameter.\n   *\n   * The placeholder `${Qualifier}` will be replaced with the value of qualifier.\n   *\n   * @default DefaultStackSynthesizer.DEFAULT_BOOTSTRAP_STACK_VERSION_SSM_PARAMETER\n   */\n  readonly bootstrapStackVersionSsmParameter?: string;\n}\n\n/**\n * Uses conventionally named roles and asset storage locations\n *\n * This synthesizer:\n *\n * - Supports cross-account deployments (the CLI can have credentials to one\n *   account, and you can still deploy to another account by assuming roles with\n *   well-known names in the other account).\n * - Supports the **CDK Pipelines** library.\n *\n * Requires the environment to have been bootstrapped with Bootstrap Stack V2\n * (also known as \"modern bootstrap stack\"). The synthesizer adds a version\n * check to the template, to make sure the bootstrap stack is recent enough\n * to support all features expected by this synthesizer.\n */\nexport class DefaultStackSynthesizer extends StackSynthesizer {\n  /**\n   * Default ARN qualifier\n   */\n  public static readonly DEFAULT_QUALIFIER = 'hnb659fds';\n\n  /**\n   * Default CloudFormation role ARN.\n   */\n  public static readonly DEFAULT_CLOUDFORMATION_ROLE_ARN = 'arn:${AWS::Partition}:iam::${AWS::AccountId}:role/cdk-${Qualifier}-cfn-exec-role-${AWS::AccountId}-${AWS::Region}';\n\n  /**\n   * Default deploy role ARN.\n   */\n  public static readonly DEFAULT_DEPLOY_ROLE_ARN = 'arn:${AWS::Partition}:iam::${AWS::AccountId}:role/cdk-${Qualifier}-deploy-role-${AWS::AccountId}-${AWS::Region}';\n\n  /**\n   * Default asset publishing role ARN for file (S3) assets.\n   */\n  public static readonly DEFAULT_FILE_ASSET_PUBLISHING_ROLE_ARN = 'arn:${AWS::Partition}:iam::${AWS::AccountId}:role/cdk-${Qualifier}-file-publishing-role-${AWS::AccountId}-${AWS::Region}';\n\n  /**\n   * Default asset publishing role ARN for image (ECR) assets.\n   */\n  public static readonly DEFAULT_IMAGE_ASSET_PUBLISHING_ROLE_ARN = 'arn:${AWS::Partition}:iam::${AWS::AccountId}:role/cdk-${Qualifier}-image-publishing-role-${AWS::AccountId}-${AWS::Region}';\n\n  /**\n   * Default lookup role ARN for missing values.\n   */\n  public static readonly DEFAULT_LOOKUP_ROLE_ARN = 'arn:${AWS::Partition}:iam::${AWS::AccountId}:role/cdk-${Qualifier}-lookup-role-${AWS::AccountId}-${AWS::Region}';\n\n  /**\n   * Default image assets repository name\n   */\n  public static readonly DEFAULT_IMAGE_ASSETS_REPOSITORY_NAME = 'cdk-${Qualifier}-container-assets-${AWS::AccountId}-${AWS::Region}';\n\n  /**\n   * Default file assets bucket name\n   */\n  public static readonly DEFAULT_FILE_ASSETS_BUCKET_NAME = 'cdk-${Qualifier}-assets-${AWS::AccountId}-${AWS::Region}';\n\n  /**\n   * Name of the CloudFormation Export with the asset key name\n   */\n  public static readonly DEFAULT_FILE_ASSET_KEY_ARN_EXPORT_NAME = 'CdkBootstrap-${Qualifier}-FileAssetKeyArn';\n\n  /**\n   * Default file asset prefix\n   */\n  public static readonly DEFAULT_FILE_ASSET_PREFIX = '';\n  /**\n   * Default Docker asset prefix\n   */\n  public static readonly DEFAULT_DOCKER_ASSET_PREFIX = '';\n\n  /**\n   * Default bootstrap stack version SSM parameter.\n   */\n  public static readonly DEFAULT_BOOTSTRAP_STACK_VERSION_SSM_PARAMETER = '/cdk-bootstrap/${Qualifier}/version';\n\n  private _stack?: Stack;\n  private bucketName?: string;\n  private repositoryName?: string;\n  private _deployRoleArn?: string;\n  private _cloudFormationExecutionRoleArn?: string;\n  private fileAssetPublishingRoleArn?: string;\n  private imageAssetPublishingRoleArn?: string;\n  private lookupRoleArn?: string;\n  private useLookupRoleForStackOperations: boolean;\n  private qualifier?: string;\n  private bucketPrefix?: string;\n  private dockerTagPrefix?: string;\n  private bootstrapStackVersionSsmParameter?: string;\n\n  private assetManifest = new AssetManifestBuilder();\n\n  constructor(private readonly props: DefaultStackSynthesizerProps = {}) {\n    super();\n    this.useLookupRoleForStackOperations = props.useLookupRoleForStackOperations ?? true;\n\n    for (const key in props) {\n      if (props.hasOwnProperty(key)) {\n        validateNoToken(key as keyof DefaultStackSynthesizerProps);\n      }\n    }\n\n    function validateNoToken<A extends keyof DefaultStackSynthesizerProps>(key: A) {\n      const prop = props[key];\n      if (typeof prop === 'string' && Token.isUnresolved(prop)) {\n        throw new Error(`DefaultSynthesizer property '${key}' cannot contain tokens; only the following placeholder strings are allowed: ` + [\n          '${Qualifier}',\n          cxapi.EnvironmentPlaceholders.CURRENT_REGION,\n          cxapi.EnvironmentPlaceholders.CURRENT_ACCOUNT,\n          cxapi.EnvironmentPlaceholders.CURRENT_PARTITION,\n        ].join(', '));\n      }\n    }\n  }\n\n  public bind(stack: Stack): void {\n    if (this._stack !== undefined) {\n      throw new Error('A StackSynthesizer can only be used for one Stack: create a new instance to use with a different Stack');\n    }\n\n    this._stack = stack;\n\n    const qualifier = this.props.qualifier ?? stack.node.tryGetContext(BOOTSTRAP_QUALIFIER_CONTEXT) ?? DefaultStackSynthesizer.DEFAULT_QUALIFIER;\n    this.qualifier = qualifier;\n\n    const spec = new StringSpecializer(stack, qualifier);\n\n    /* eslint-disable max-len */\n    this.bucketName = spec.specialize(this.props.fileAssetsBucketName ?? DefaultStackSynthesizer.DEFAULT_FILE_ASSETS_BUCKET_NAME);\n    this.repositoryName = spec.specialize(this.props.imageAssetsRepositoryName ?? DefaultStackSynthesizer.DEFAULT_IMAGE_ASSETS_REPOSITORY_NAME);\n    this._deployRoleArn = spec.specialize(this.props.deployRoleArn ?? DefaultStackSynthesizer.DEFAULT_DEPLOY_ROLE_ARN);\n    this._cloudFormationExecutionRoleArn = spec.specialize(this.props.cloudFormationExecutionRole ?? DefaultStackSynthesizer.DEFAULT_CLOUDFORMATION_ROLE_ARN);\n    this.fileAssetPublishingRoleArn = spec.specialize(this.props.fileAssetPublishingRoleArn ?? DefaultStackSynthesizer.DEFAULT_FILE_ASSET_PUBLISHING_ROLE_ARN);\n    this.imageAssetPublishingRoleArn = spec.specialize(this.props.imageAssetPublishingRoleArn ?? DefaultStackSynthesizer.DEFAULT_IMAGE_ASSET_PUBLISHING_ROLE_ARN);\n    this.lookupRoleArn = spec.specialize(this.props.lookupRoleArn ?? DefaultStackSynthesizer.DEFAULT_LOOKUP_ROLE_ARN);\n    this.bucketPrefix = spec.specialize(this.props.bucketPrefix ?? DefaultStackSynthesizer.DEFAULT_FILE_ASSET_PREFIX);\n    this.dockerTagPrefix = spec.specialize(this.props.dockerTagPrefix ?? DefaultStackSynthesizer.DEFAULT_DOCKER_ASSET_PREFIX);\n    this.bootstrapStackVersionSsmParameter = spec.qualifierOnly(this.props.bootstrapStackVersionSsmParameter ?? DefaultStackSynthesizer.DEFAULT_BOOTSTRAP_STACK_VERSION_SSM_PARAMETER);\n    /* eslint-enable max-len */\n  }\n\n  public addFileAsset(asset: FileAssetSource): FileAssetLocation {\n    assertBound(this.stack);\n    assertBound(this.bucketName);\n    assertBound(this.bucketPrefix);\n\n    return this.assetManifest.addFileAssetDefault(asset, this.stack, this.bucketName, this.bucketPrefix, {\n      assumeRoleArn: this.fileAssetPublishingRoleArn,\n      assumeRoleExternalId: this.props.fileAssetPublishingExternalId,\n    });\n  }\n\n  public addDockerImageAsset(asset: DockerImageAssetSource): DockerImageAssetLocation {\n    assertBound(this.stack);\n    assertBound(this.repositoryName);\n    assertBound(this.dockerTagPrefix);\n\n    return this.assetManifest.addDockerImageAssetDefault(asset, this.stack, this.repositoryName, this.dockerTagPrefix, {\n      assumeRoleArn: this.imageAssetPublishingRoleArn,\n      assumeRoleExternalId: this.props.imageAssetPublishingExternalId,\n    });\n  }\n\n  protected synthesizeStackTemplate(stack: Stack, session: ISynthesisSession) {\n    stack._synthesizeTemplate(session, this.lookupRoleArn);\n  }\n\n  /**\n   * Synthesize the associated stack to the session\n   */\n  public synthesize(session: ISynthesisSession): void {\n    assertBound(this.stack);\n    assertBound(this.qualifier);\n\n    // Must be done here -- if it's done in bind() (called in the Stack's constructor)\n    // then it will become impossible to set context after that.\n    //\n    // If it's done AFTER _synthesizeTemplate(), then the template won't contain the\n    // right constructs.\n    if (this.props.generateBootstrapVersionRule ?? true) {\n      addBootstrapVersionRule(this.stack, MIN_BOOTSTRAP_STACK_VERSION, <string> this.bootstrapStackVersionSsmParameter);\n    }\n\n    this.synthesizeStackTemplate(this.stack, session);\n\n    const templateAsset = this.addFileAsset(stackTemplateFileAsset(this.stack, session));\n\n    const assetManifestId = this.assetManifest.writeManifest(this.stack, session, {\n      requiresBootstrapStackVersion: MIN_BOOTSTRAP_STACK_VERSION,\n      bootstrapStackVersionSsmParameter: this.bootstrapStackVersionSsmParameter,\n    });\n\n    this.emitStackArtifact(this.stack, session, {\n      assumeRoleExternalId: this.props.deployRoleExternalId,\n      assumeRoleArn: this._deployRoleArn,\n      cloudFormationExecutionRoleArn: this._cloudFormationExecutionRoleArn,\n      stackTemplateAssetObjectUrl: templateAsset.s3ObjectUrlWithPlaceholders,\n      requiresBootstrapStackVersion: MIN_BOOTSTRAP_STACK_VERSION,\n      bootstrapStackVersionSsmParameter: this.bootstrapStackVersionSsmParameter,\n      additionalDependencies: [assetManifestId],\n      lookupRole: this.useLookupRoleForStackOperations && this.lookupRoleArn ? {\n        arn: this.lookupRoleArn,\n        assumeRoleExternalId: this.props.lookupRoleExternalId,\n        requiresBootstrapStackVersion: MIN_LOOKUP_ROLE_BOOTSTRAP_STACK_VERSION,\n        bootstrapStackVersionSsmParameter: this.bootstrapStackVersionSsmParameter,\n      } : undefined,\n    });\n  }\n\n  /**\n   * Returns the ARN of the deploy Role.\n   */\n  public get deployRoleArn(): string {\n    if (!this._deployRoleArn) {\n      throw new Error('deployRoleArn getter can only be called after the synthesizer has been bound to a Stack');\n    }\n    return this._deployRoleArn;\n  }\n\n  /**\n   * Returns the ARN of the CFN execution Role.\n   */\n  public get cloudFormationExecutionRoleArn(): string {\n    if (!this._cloudFormationExecutionRoleArn) {\n      throw new Error('cloudFormationExecutionRoleArn getter can only be called after the synthesizer has been bound to a Stack');\n    }\n    return this._cloudFormationExecutionRoleArn;\n  }\n\n  protected get stack(): Stack | undefined {\n    return this._stack;\n  }\n}\n\n/**\n * Add a CfnRule to the Stack which checks the current version of the bootstrap stack this template is targeting\n *\n * The CLI normally checks this, but in a pipeline the CLI is not involved\n * so we encode this rule into the template in a way that CloudFormation will check it.\n */\nfunction addBootstrapVersionRule(stack: Stack, requiredVersion: number, bootstrapStackVersionSsmParameter: string) {\n  // Because of https://github.com/aws/aws-cdk/blob/master/packages/assert-internal/lib/synth-utils.ts#L74\n  // synthesize() may be called more than once on a stack in unit tests, and the below would break\n  // if we execute it a second time. Guard against the constructs already existing.\n  if (stack.node.tryFindChild('BootstrapVersion')) { return; }\n\n  const param = new CfnParameter(stack, 'BootstrapVersion', {\n    type: 'AWS::SSM::Parameter::Value<String>',\n    description: `Version of the CDK Bootstrap resources in this environment, automatically retrieved from SSM Parameter Store. ${cxapi.SSMPARAM_NO_INVALIDATE}`,\n    default: bootstrapStackVersionSsmParameter,\n  });\n\n  // There is no >= check in CloudFormation, so we have to check the number\n  // is NOT in [1, 2, 3, ... <required> - 1]\n  const oldVersions = range(1, requiredVersion).map(n => `${n}`);\n\n  new CfnRule(stack, 'CheckBootstrapVersion', {\n    assertions: [\n      {\n        assert: Fn.conditionNot(Fn.conditionContains(oldVersions, param.valueAsString)),\n        assertDescription: `CDK bootstrap stack version ${requiredVersion} required. Please run 'cdk bootstrap' with a recent version of the CDK CLI.`,\n      },\n    ],\n  });\n}\n\nfunction range(startIncl: number, endExcl: number) {\n  const ret = new Array<number>();\n  for (let i = startIncl; i < endExcl; i++) {\n    ret.push(i);\n  }\n  return ret;\n}\n\n"]} |
\ | No newline at end of file |