UNPKG

49.3 kBJavaScriptView Raw
1"use strict";
2var _a;
3Object.defineProperty(exports, "__esModule", { value: true });
4exports.DefaultStackSynthesizer = exports.BOOTSTRAP_QUALIFIER_CONTEXT = void 0;
5const jsiiDeprecationWarnings = require("../../.warnings.jsii.js");
6const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
7const cxapi = require("@aws-cdk/cx-api");
8const cfn_fn_1 = require("../cfn-fn");
9const cfn_parameter_1 = require("../cfn-parameter");
10const cfn_rule_1 = require("../cfn-rule");
11const token_1 = require("../token");
12const _asset_manifest_builder_1 = require("./_asset-manifest-builder");
13const _shared_1 = require("./_shared");
14const stack_synthesizer_1 = require("./stack-synthesizer");
15exports.BOOTSTRAP_QUALIFIER_CONTEXT = '@aws-cdk/core:bootstrapQualifier';
16/* eslint-disable max-len */
17/**
18 * The minimum bootstrap stack version required by this app.
19 */
20const MIN_BOOTSTRAP_STACK_VERSION = 6;
21/**
22 * The minimum bootstrap stack version required
23 * to use the lookup role.
24 */
25const 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 */
41class 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}
218exports.DefaultStackSynthesizer = DefaultStackSynthesizer;
219_a = JSII_RTTI_SYMBOL_1;
220DefaultStackSynthesizer[_a] = { fqn: "@aws-cdk/core.DefaultStackSynthesizer", version: "1.204.0" };
221/**
222 * Default ARN qualifier
223 */
224DefaultStackSynthesizer.DEFAULT_QUALIFIER = 'hnb659fds';
225/**
226 * Default CloudFormation role ARN.
227 */
228DefaultStackSynthesizer.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 */
232DefaultStackSynthesizer.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 */
236DefaultStackSynthesizer.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 */
240DefaultStackSynthesizer.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 */
244DefaultStackSynthesizer.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 */
248DefaultStackSynthesizer.DEFAULT_IMAGE_ASSETS_REPOSITORY_NAME = 'cdk-${Qualifier}-container-assets-${AWS::AccountId}-${AWS::Region}';
249/**
250 * Default file assets bucket name
251 */
252DefaultStackSynthesizer.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 */
256DefaultStackSynthesizer.DEFAULT_FILE_ASSET_KEY_ARN_EXPORT_NAME = 'CdkBootstrap-${Qualifier}-FileAssetKeyArn';
257/**
258 * Default file asset prefix
259 */
260DefaultStackSynthesizer.DEFAULT_FILE_ASSET_PREFIX = '';
261/**
262 * Default Docker asset prefix
263 */
264DefaultStackSynthesizer.DEFAULT_DOCKER_ASSET_PREFIX = '';
265/**
266 * Default bootstrap stack version SSM parameter.
267 */
268DefaultStackSynthesizer.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 */
275function 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}
299function 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":";;;;;;AAAA,yCAAyC;AAEzC,sCAA+B;AAC/B,oDAAgD;AAChD,0CAAsC;AAGtC,oCAAiC;AACjC,uEAAiE;AACjE,uCAAmF;AACnF,2DAAuD;AAE1C,QAAA,2BAA2B,GAAG,kCAAkC,CAAC;AAE9E,4BAA4B;AAE5B;;GAEG;AACH,MAAM,2BAA2B,GAAG,CAAC,CAAC;AAEtC;;;GAGG;AACH,MAAM,uCAAuC,GAAG,CAAC,CAAC;AAgMlD;;;;;;;;;;;;;;GAcG;AACH,MAAa,uBAAwB,SAAQ,oCAAgB;IA4E3D,YAA6B,QAAsC,EAAE;QACnE,KAAK,EAAE,CAAC;QADmB,UAAK,GAAL,KAAK,CAAmC;QAF7D,kBAAa,GAAG,IAAI,8CAAoB,EAAE,CAAC;;;;;;+CA1ExC,uBAAuB;;;;QA8EhC,IAAI,CAAC,+BAA+B,GAAG,KAAK,CAAC,+BAA+B,IAAI,IAAI,CAAC;QAErF,KAAK,MAAM,GAAG,IAAI,KAAK,EAAE;YACvB,IAAI,KAAK,CAAC,cAAc,CAAC,GAAG,CAAC,EAAE;gBAC7B,eAAe,CAAC,GAAyC,CAAC,CAAC;aAC5D;SACF;QAED,SAAS,eAAe,CAA+C,GAAM;YAC3E,MAAM,IAAI,GAAG,KAAK,CAAC,GAAG,CAAC,CAAC;YACxB,IAAI,OAAO,IAAI,KAAK,QAAQ,IAAI,aAAK,CAAC,YAAY,CAAC,IAAI,CAAC,EAAE;gBACxD,MAAM,IAAI,KAAK,CAAC,gCAAgC,GAAG,+EAA+E,GAAG;oBACnI,cAAc;oBACd,KAAK,CAAC,uBAAuB,CAAC,cAAc;oBAC5C,KAAK,CAAC,uBAAuB,CAAC,eAAe;oBAC7C,KAAK,CAAC,uBAAuB,CAAC,iBAAiB;iBAChD,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC;aACf;QACH,CAAC;KACF;IAEM,IAAI,CAAC,KAAY;;;;;;;;;;QACtB,IAAI,IAAI,CAAC,MAAM,KAAK,SAAS,EAAE;YAC7B,MAAM,IAAI,KAAK,CAAC,wGAAwG,CAAC,CAAC;SAC3H;QAED,IAAI,CAAC,MAAM,GAAG,KAAK,CAAC;QAEpB,MAAM,SAAS,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,IAAI,KAAK,CAAC,IAAI,CAAC,aAAa,CAAC,mCAA2B,CAAC,IAAI,uBAAuB,CAAC,iBAAiB,CAAC;QAC7I,IAAI,CAAC,SAAS,GAAG,SAAS,CAAC;QAE3B,MAAM,IAAI,GAAG,IAAI,2BAAiB,CAAC,KAAK,EAAE,SAAS,CAAC,CAAC;QAErD,4BAA4B;QAC5B,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,KAAK,CAAC,oBAAoB,IAAI,uBAAuB,CAAC,+BAA+B,CAAC,CAAC;QAC9H,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,KAAK,CAAC,yBAAyB,IAAI,uBAAuB,CAAC,oCAAoC,CAAC,CAAC;QAC5I,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,KAAK,CAAC,aAAa,IAAI,uBAAuB,CAAC,uBAAuB,CAAC,CAAC;QACnH,IAAI,CAAC,+BAA+B,GAAG,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,KAAK,CAAC,2BAA2B,IAAI,uBAAuB,CAAC,+BAA+B,CAAC,CAAC;QAC1J,IAAI,CAAC,0BAA0B,GAAG,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,KAAK,CAAC,0BAA0B,IAAI,uBAAuB,CAAC,sCAAsC,CAAC,CAAC;QAC3J,IAAI,CAAC,2BAA2B,GAAG,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,KAAK,CAAC,2BAA2B,IAAI,uBAAuB,CAAC,uCAAuC,CAAC,CAAC;QAC9J,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,KAAK,CAAC,aAAa,IAAI,uBAAuB,CAAC,uBAAuB,CAAC,CAAC;QAClH,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,KAAK,CAAC,YAAY,IAAI,uBAAuB,CAAC,yBAAyB,CAAC,CAAC;QAClH,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,KAAK,CAAC,eAAe,IAAI,uBAAuB,CAAC,2BAA2B,CAAC,CAAC;QAC1H,IAAI,CAAC,iCAAiC,GAAG,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,KAAK,CAAC,iCAAiC,IAAI,uBAAuB,CAAC,6CAA6C,CAAC,CAAC;KAEpL;IAEM,YAAY,CAAC,KAAsB;;;;;;;;;;QACxC,qBAAW,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACxB,qBAAW,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QAC7B,qBAAW,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;QAE/B,OAAO,IAAI,CAAC,aAAa,CAAC,mBAAmB,CAAC,KAAK,EAAE,IAAI,CAAC,KAAK,EAAE,IAAI,CAAC,UAAU,EAAE,IAAI,CAAC,YAAY,EAAE;YACnG,aAAa,EAAE,IAAI,CAAC,0BAA0B;YAC9C,oBAAoB,EAAE,IAAI,CAAC,KAAK,CAAC,6BAA6B;SAC/D,CAAC,CAAC;KACJ;IAEM,mBAAmB,CAAC,KAA6B;;;;;;;;;;QACtD,qBAAW,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACxB,qBAAW,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC;QACjC,qBAAW,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;QAElC,OAAO,IAAI,CAAC,aAAa,CAAC,0BAA0B,CAAC,KAAK,EAAE,IAAI,CAAC,KAAK,EAAE,IAAI,CAAC,cAAc,EAAE,IAAI,CAAC,eAAe,EAAE;YACjH,aAAa,EAAE,IAAI,CAAC,2BAA2B;YAC/C,oBAAoB,EAAE,IAAI,CAAC,KAAK,CAAC,8BAA8B;SAChE,CAAC,CAAC;KACJ;IAES,uBAAuB,CAAC,KAAY,EAAE,OAA0B;;;;;;;;;;;QACxE,KAAK,CAAC,mBAAmB,CAAC,OAAO,EAAE,IAAI,CAAC,aAAa,CAAC,CAAC;KACxD;IAED;;OAEG;IACI,UAAU,CAAC,OAA0B;;;;;;;;;;QAC1C,qBAAW,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACxB,qBAAW,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QAE5B,kFAAkF;QAClF,4DAA4D;QAC5D,EAAE;QACF,gFAAgF;QAChF,oBAAoB;QACpB,IAAI,IAAI,CAAC,KAAK,CAAC,4BAA4B,IAAI,IAAI,EAAE;YACnD,uBAAuB,CAAC,IAAI,CAAC,KAAK,EAAE,2BAA2B,EAAW,IAAI,CAAC,iCAAiC,CAAC,CAAC;SACnH;QAED,IAAI,CAAC,uBAAuB,CAAC,IAAI,CAAC,KAAK,EAAE,OAAO,CAAC,CAAC;QAElD,MAAM,aAAa,GAAG,IAAI,CAAC,YAAY,CAAC,gCAAsB,CAAC,IAAI,CAAC,KAAK,EAAE,OAAO,CAAC,CAAC,CAAC;QAErF,MAAM,eAAe,GAAG,IAAI,CAAC,aAAa,CAAC,aAAa,CAAC,IAAI,CAAC,KAAK,EAAE,OAAO,EAAE;YAC5E,6BAA6B,EAAE,2BAA2B;YAC1D,iCAAiC,EAAE,IAAI,CAAC,iCAAiC;SAC1E,CAAC,CAAC;QAEH,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,KAAK,EAAE,OAAO,EAAE;YAC1C,oBAAoB,EAAE,IAAI,CAAC,KAAK,CAAC,oBAAoB;YACrD,aAAa,EAAE,IAAI,CAAC,cAAc;YAClC,8BAA8B,EAAE,IAAI,CAAC,+BAA+B;YACpE,2BAA2B,EAAE,aAAa,CAAC,2BAA2B;YACtE,6BAA6B,EAAE,2BAA2B;YAC1D,iCAAiC,EAAE,IAAI,CAAC,iCAAiC;YACzE,sBAAsB,EAAE,CAAC,eAAe,CAAC;YACzC,UAAU,EAAE,IAAI,CAAC,+BAA+B,IAAI,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC;gBACvE,GAAG,EAAE,IAAI,CAAC,aAAa;gBACvB,oBAAoB,EAAE,IAAI,CAAC,KAAK,CAAC,oBAAoB;gBACrD,6BAA6B,EAAE,uCAAuC;gBACtE,iCAAiC,EAAE,IAAI,CAAC,iCAAiC;aAC1E,CAAC,CAAC,CAAC,SAAS;SACd,CAAC,CAAC;KACJ;IAED;;OAEG;IACH,IAAW,aAAa;QACtB,IAAI,CAAC,IAAI,CAAC,cAAc,EAAE;YACxB,MAAM,IAAI,KAAK,CAAC,yFAAyF,CAAC,CAAC;SAC5G;QACD,OAAO,IAAI,CAAC,cAAc,CAAC;KAC5B;IAED;;OAEG;IACH,IAAW,8BAA8B;QACvC,IAAI,CAAC,IAAI,CAAC,+BAA+B,EAAE;YACzC,MAAM,IAAI,KAAK,CAAC,0GAA0G,CAAC,CAAC;SAC7H;QACD,OAAO,IAAI,CAAC,+BAA+B,CAAC;KAC7C;IAED,IAAc,KAAK;QACjB,OAAO,IAAI,CAAC,MAAM,CAAC;KACpB;;AAvNH,0DAwNC;;;AAvNC;;GAEG;AACoB,yCAAiB,GAAG,WAAW,CAAC;AAEvD;;GAEG;AACoB,uDAA+B,GAAG,mHAAmH,CAAC;AAE7K;;GAEG;AACoB,+CAAuB,GAAG,iHAAiH,CAAC;AAEnK;;GAEG;AACoB,8DAAsC,GAAG,0HAA0H,CAAC;AAE3L;;GAEG;AACoB,+DAAuC,GAAG,2HAA2H,CAAC;AAE7L;;GAEG;AACoB,+CAAuB,GAAG,iHAAiH,CAAC;AAEnK;;GAEG;AACoB,4DAAoC,GAAG,oEAAoE,CAAC;AAEnI;;GAEG;AACoB,uDAA+B,GAAG,0DAA0D,CAAC;AAEpH;;GAEG;AACoB,8DAAsC,GAAG,2CAA2C,CAAC;AAE5G;;GAEG;AACoB,iDAAyB,GAAG,EAAE,CAAC;AACtD;;GAEG;AACoB,mDAA2B,GAAG,EAAE,CAAC;AAExD;;GAEG;AACoB,qEAA6C,GAAG,qCAAqC,CAAC;AAgK/G;;;;;GAKG;AACH,SAAS,uBAAuB,CAAC,KAAY,EAAE,eAAuB,EAAE,iCAAyC;IAC/G,wGAAwG;IACxG,gGAAgG;IAChG,iFAAiF;IACjF,IAAI,KAAK,CAAC,IAAI,CAAC,YAAY,CAAC,kBAAkB,CAAC,EAAE;QAAE,OAAO;KAAE;IAE5D,MAAM,KAAK,GAAG,IAAI,4BAAY,CAAC,KAAK,EAAE,kBAAkB,EAAE;QACxD,IAAI,EAAE,oCAAoC;QAC1C,WAAW,EAAE,iHAAiH,KAAK,CAAC,sBAAsB,EAAE;QAC5J,OAAO,EAAE,iCAAiC;KAC3C,CAAC,CAAC;IAEH,yEAAyE;IACzE,0CAA0C;IAC1C,MAAM,WAAW,GAAG,KAAK,CAAC,CAAC,EAAE,eAAe,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;IAE/D,IAAI,kBAAO,CAAC,KAAK,EAAE,uBAAuB,EAAE;QAC1C,UAAU,EAAE;YACV;gBACE,MAAM,EAAE,WAAE,CAAC,YAAY,CAAC,WAAE,CAAC,iBAAiB,CAAC,WAAW,EAAE,KAAK,CAAC,aAAa,CAAC,CAAC;gBAC/E,iBAAiB,EAAE,+BAA+B,eAAe,6EAA6E;aAC/I;SACF;KACF,CAAC,CAAC;AACL,CAAC;AAED,SAAS,KAAK,CAAC,SAAiB,EAAE,OAAe;IAC/C,MAAM,GAAG,GAAG,IAAI,KAAK,EAAU,CAAC;IAChC,KAAK,IAAI,CAAC,GAAG,SAAS,EAAE,CAAC,GAAG,OAAO,EAAE,CAAC,EAAE,EAAE;QACxC,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;KACb;IACD,OAAO,GAAG,CAAC;AACb,CAAC","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