1 | ;
|
2 | var _a;
|
3 | Object.defineProperty(exports, "__esModule", { value: true });
|
4 | exports.rootPathTo = exports.Stack = exports.STACK_RESOURCE_LIMIT_CONTEXT = void 0;
|
5 | const jsiiDeprecationWarnings = require("../.warnings.jsii.js");
|
6 | const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
7 | const fs = require("fs");
|
8 | const path = require("path");
|
9 | const cxschema = require("@aws-cdk/cloud-assembly-schema");
|
10 | const cxapi = require("@aws-cdk/cx-api");
|
11 | const constructs_1 = require("constructs");
|
12 | const minimatch = require("minimatch");
|
13 | const annotations_1 = require("./annotations");
|
14 | const app_1 = require("./app");
|
15 | const arn_1 = require("./arn");
|
16 | const cfn_element_1 = require("./cfn-element");
|
17 | const cfn_fn_1 = require("./cfn-fn");
|
18 | const cfn_pseudo_1 = require("./cfn-pseudo");
|
19 | const cfn_resource_1 = require("./cfn-resource");
|
20 | const context_provider_1 = require("./context-provider");
|
21 | const feature_flags_1 = require("./feature-flags");
|
22 | const cloudformation_lang_1 = require("./private/cloudformation-lang");
|
23 | const logical_id_1 = require("./private/logical-id");
|
24 | const resolve_1 = require("./private/resolve");
|
25 | const uniqueid_1 = require("./private/uniqueid");
|
26 | // v2 - keep this import as a separate section to reduce merge conflict when forward merging with the v2 branch.
|
27 | // eslint-disable-next-line
|
28 | const construct_compat_1 = require("./construct-compat");
|
29 | const STACK_SYMBOL = Symbol.for('@aws-cdk/core.Stack');
|
30 | const MY_STACK_CACHE = Symbol.for('@aws-cdk/core.Stack.myStack');
|
31 | exports.STACK_RESOURCE_LIMIT_CONTEXT = '@aws-cdk/core:stackResourceLimit';
|
32 | const VALID_STACK_NAME_REGEX = /^[A-Za-z][A-Za-z0-9-]*$/;
|
33 | const MAX_RESOURCES = 500;
|
34 | /**
|
35 | * A root construct which represents a single CloudFormation stack.
|
36 | */
|
37 | class Stack extends construct_compat_1.Construct {
|
38 | /**
|
39 | * Creates a new stack.
|
40 | *
|
41 | * @param scope Parent of this stack, usually an `App` or a `Stage`, but could be any construct.
|
42 | * @param id The construct ID of this stack. If `stackName` is not explicitly
|
43 | * defined, this id (and any parent IDs) will be used to determine the
|
44 | * physical ID of the stack.
|
45 | * @param props Stack properties.
|
46 | */
|
47 | constructor(scope, id, props = {}) {
|
48 | try {
|
49 | jsiiDeprecationWarnings._aws_cdk_core_StackProps(props);
|
50 | }
|
51 | catch (error) {
|
52 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
53 | Error.captureStackTrace(error, Stack);
|
54 | }
|
55 | throw error;
|
56 | }
|
57 | // For unit test scope and id are optional for stacks, but we still want an App
|
58 | // as the parent because apps implement much of the synthesis logic.
|
59 | scope = scope ?? new app_1.App({
|
60 | autoSynth: false,
|
61 | outdir: fs_1.FileSystem.mkdtemp('cdk-test-app-'),
|
62 | });
|
63 | // "Default" is a "hidden id" from a `node.uniqueId` perspective
|
64 | id = id ?? 'Default';
|
65 | super(scope, id);
|
66 | this._missingContext = new Array();
|
67 | this._stackDependencies = {};
|
68 | this.templateOptions = {};
|
69 | Object.defineProperty(this, STACK_SYMBOL, { value: true });
|
70 | this._logicalIds = new logical_id_1.LogicalIDs();
|
71 | const { account, region, environment } = this.parseEnvironment(props.env);
|
72 | this.account = account;
|
73 | this.region = region;
|
74 | this.environment = environment;
|
75 | this.terminationProtection = props.terminationProtection;
|
76 | if (props.description !== undefined) {
|
77 | // Max length 1024 bytes
|
78 | // Typically 2 bytes per character, may be more for more exotic characters
|
79 | if (props.description.length > 512) {
|
80 | throw new Error(`Stack description must be <= 1024 bytes. Received description: '${props.description}'`);
|
81 | }
|
82 | this.templateOptions.description = props.description;
|
83 | }
|
84 | this._stackName = props.stackName ?? this.generateStackName();
|
85 | if (this._stackName.length > 128) {
|
86 | throw new Error(`Stack name must be <= 128 characters. Stack name: '${this._stackName}'`);
|
87 | }
|
88 | this.tags = new tag_manager_1.TagManager(cfn_resource_1.TagType.KEY_VALUE, 'aws:cdk:stack', props.tags);
|
89 | if (!VALID_STACK_NAME_REGEX.test(this.stackName)) {
|
90 | throw new Error(`Stack name must match the regular expression: ${VALID_STACK_NAME_REGEX.toString()}, got '${this.stackName}'`);
|
91 | }
|
92 | // the preferred behavior is to generate a unique id for this stack and use
|
93 | // it as the artifact ID in the assembly. this allows multiple stacks to use
|
94 | // the same name. however, this behavior is breaking for 1.x so it's only
|
95 | // applied under a feature flag which is applied automatically for new
|
96 | // projects created using `cdk init`.
|
97 | //
|
98 | // Also use the new behavior if we are using the new CI/CD-ready synthesizer; that way
|
99 | // people only have to flip one flag.
|
100 | const featureFlags = feature_flags_1.FeatureFlags.of(this);
|
101 | const stackNameDupeContext = featureFlags.isEnabled(cxapi.ENABLE_STACK_NAME_DUPLICATES_CONTEXT);
|
102 | const newStyleSynthesisContext = featureFlags.isEnabled(cxapi.NEW_STYLE_STACK_SYNTHESIS_CONTEXT);
|
103 | this.artifactId = (stackNameDupeContext || newStyleSynthesisContext)
|
104 | ? this.generateStackArtifactId()
|
105 | : this.stackName;
|
106 | this.templateFile = `${this.artifactId}.template.json`;
|
107 | // Not for nested stacks
|
108 | this._versionReportingEnabled = (props.analyticsReporting ?? this.node.tryGetContext(cxapi.ANALYTICS_REPORTING_ENABLED_CONTEXT))
|
109 | && !this.nestedStackParent;
|
110 | this.synthesizer = props.synthesizer ?? (newStyleSynthesisContext
|
111 | ? new stack_synthesizers_1.DefaultStackSynthesizer()
|
112 | : new stack_synthesizers_1.LegacyStackSynthesizer());
|
113 | this.synthesizer.bind(this);
|
114 | }
|
115 | /**
|
116 | * Return whether the given object is a Stack.
|
117 | *
|
118 | * We do attribute detection since we can't reliably use 'instanceof'.
|
119 | */
|
120 | static isStack(x) {
|
121 | return x !== null && typeof (x) === 'object' && STACK_SYMBOL in x;
|
122 | }
|
123 | /**
|
124 | * Looks up the first stack scope in which `construct` is defined. Fails if there is no stack up the tree.
|
125 | * @param construct The construct to start the search from.
|
126 | */
|
127 | static of(construct) {
|
128 | // we want this to be as cheap as possible. cache this result by mutating
|
129 | // the object. anecdotally, at the time of this writing, @aws-cdk/core unit
|
130 | // tests hit this cache 1,112 times, @aws-cdk/aws-cloudformation unit tests
|
131 | // hit this 2,435 times).
|
132 | const cache = construct[MY_STACK_CACHE];
|
133 | if (cache) {
|
134 | return cache;
|
135 | }
|
136 | else {
|
137 | const value = _lookup(construct);
|
138 | Object.defineProperty(construct, MY_STACK_CACHE, {
|
139 | enumerable: false,
|
140 | writable: false,
|
141 | configurable: false,
|
142 | value,
|
143 | });
|
144 | return value;
|
145 | }
|
146 | function _lookup(c) {
|
147 | if (Stack.isStack(c)) {
|
148 | return c;
|
149 | }
|
150 | const _scope = constructs_1.Node.of(c).scope;
|
151 | if (stage_1.Stage.isStage(c) || !_scope) {
|
152 | throw new Error(`${construct.constructor?.name ?? 'Construct'} at '${constructs_1.Node.of(construct).path}' should be created in the scope of a Stack, but no Stack found`);
|
153 | }
|
154 | return _lookup(_scope);
|
155 | }
|
156 | }
|
157 | /**
|
158 | * Resolve a tokenized value in the context of the current stack.
|
159 | */
|
160 | resolve(obj) {
|
161 | return resolve_1.resolve(obj, {
|
162 | scope: this,
|
163 | prefix: [],
|
164 | resolver: cloudformation_lang_1.CLOUDFORMATION_TOKEN_RESOLVER,
|
165 | preparing: false,
|
166 | });
|
167 | }
|
168 | /**
|
169 | * Convert an object, potentially containing tokens, to a JSON string
|
170 | */
|
171 | toJsonString(obj, space) {
|
172 | return cloudformation_lang_1.CloudFormationLang.toJSON(obj, space).toString();
|
173 | }
|
174 | /**
|
175 | * DEPRECATED
|
176 | * @deprecated use `reportMissingContextKey()`
|
177 | */
|
178 | reportMissingContext(report) {
|
179 | try {
|
180 | jsiiDeprecationWarnings.print("@aws-cdk/core.Stack#reportMissingContext", "use `reportMissingContextKey()`");
|
181 | }
|
182 | catch (error) {
|
183 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
184 | Error.captureStackTrace(error, this.reportMissingContext);
|
185 | }
|
186 | throw error;
|
187 | }
|
188 | if (!Object.values(cxschema.ContextProvider).includes(report.provider)) {
|
189 | throw new Error(`Unknown context provider requested in: ${JSON.stringify(report)}`);
|
190 | }
|
191 | this.reportMissingContextKey(report);
|
192 | }
|
193 | /**
|
194 | * Indicate that a context key was expected
|
195 | *
|
196 | * Contains instructions which will be emitted into the cloud assembly on how
|
197 | * the key should be supplied.
|
198 | *
|
199 | * @param report The set of parameters needed to obtain the context
|
200 | */
|
201 | reportMissingContextKey(report) {
|
202 | this._missingContext.push(report);
|
203 | }
|
204 | /**
|
205 | * Rename a generated logical identities
|
206 | *
|
207 | * To modify the naming scheme strategy, extend the `Stack` class and
|
208 | * override the `allocateLogicalId` method.
|
209 | */
|
210 | renameLogicalId(oldId, newId) {
|
211 | this._logicalIds.addRename(oldId, newId);
|
212 | }
|
213 | /**
|
214 | * Allocates a stack-unique CloudFormation-compatible logical identity for a
|
215 | * specific resource.
|
216 | *
|
217 | * This method is called when a `CfnElement` is created and used to render the
|
218 | * initial logical identity of resources. Logical ID renames are applied at
|
219 | * this stage.
|
220 | *
|
221 | * This method uses the protected method `allocateLogicalId` to render the
|
222 | * logical ID for an element. To modify the naming scheme, extend the `Stack`
|
223 | * class and override this method.
|
224 | *
|
225 | * @param element The CloudFormation element for which a logical identity is
|
226 | * needed.
|
227 | */
|
228 | getLogicalId(element) {
|
229 | try {
|
230 | jsiiDeprecationWarnings._aws_cdk_core_CfnElement(element);
|
231 | }
|
232 | catch (error) {
|
233 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
234 | Error.captureStackTrace(error, this.getLogicalId);
|
235 | }
|
236 | throw error;
|
237 | }
|
238 | const logicalId = this.allocateLogicalId(element);
|
239 | return this._logicalIds.applyRename(logicalId);
|
240 | }
|
241 | /**
|
242 | * Add a dependency between this stack and another stack.
|
243 | *
|
244 | * This can be used to define dependencies between any two stacks within an
|
245 | * app, and also supports nested stacks.
|
246 | */
|
247 | addDependency(target, reason) {
|
248 | try {
|
249 | jsiiDeprecationWarnings._aws_cdk_core_Stack(target);
|
250 | }
|
251 | catch (error) {
|
252 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
253 | Error.captureStackTrace(error, this.addDependency);
|
254 | }
|
255 | throw error;
|
256 | }
|
257 | deps_1.addDependency(this, target, reason);
|
258 | }
|
259 | /**
|
260 | * Return the stacks this stack depends on
|
261 | */
|
262 | get dependencies() {
|
263 | return Object.values(this._stackDependencies).map(x => x.stack);
|
264 | }
|
265 | /**
|
266 | * The concrete CloudFormation physical stack name.
|
267 | *
|
268 | * This is either the name defined explicitly in the `stackName` prop or
|
269 | * allocated based on the stack's location in the construct tree. Stacks that
|
270 | * are directly defined under the app use their construct `id` as their stack
|
271 | * name. Stacks that are defined deeper within the tree will use a hashed naming
|
272 | * scheme based on the construct path to ensure uniqueness.
|
273 | *
|
274 | * If you wish to obtain the deploy-time AWS::StackName intrinsic,
|
275 | * you can use `Aws.stackName` directly.
|
276 | */
|
277 | get stackName() {
|
278 | return this._stackName;
|
279 | }
|
280 | /**
|
281 | * The partition in which this stack is defined
|
282 | */
|
283 | get partition() {
|
284 | // Always return a non-scoped partition intrinsic. These will usually
|
285 | // be used to construct an ARN, but there are no cross-partition
|
286 | // calls anyway.
|
287 | return cfn_pseudo_1.Aws.PARTITION;
|
288 | }
|
289 | /**
|
290 | * The Amazon domain suffix for the region in which this stack is defined
|
291 | */
|
292 | get urlSuffix() {
|
293 | // Since URL Suffix always follows partition, it is unscoped like partition is.
|
294 | return cfn_pseudo_1.Aws.URL_SUFFIX;
|
295 | }
|
296 | /**
|
297 | * The ID of the stack
|
298 | *
|
299 | * @example
|
300 | * // After resolving, looks like
|
301 | * 'arn:aws:cloudformation:us-west-2:123456789012:stack/teststack/51af3dc0-da77-11e4-872e-1234567db123'
|
302 | */
|
303 | get stackId() {
|
304 | return new cfn_pseudo_1.ScopedAws(this).stackId;
|
305 | }
|
306 | /**
|
307 | * Returns the list of notification Amazon Resource Names (ARNs) for the current stack.
|
308 | */
|
309 | get notificationArns() {
|
310 | return new cfn_pseudo_1.ScopedAws(this).notificationArns;
|
311 | }
|
312 | /**
|
313 | * Indicates if this is a nested stack, in which case `parentStack` will include a reference to it's parent.
|
314 | */
|
315 | get nested() {
|
316 | return this.nestedStackResource !== undefined;
|
317 | }
|
318 | /**
|
319 | * Creates an ARN from components.
|
320 | *
|
321 | * If `partition`, `region` or `account` are not specified, the stack's
|
322 | * partition, region and account will be used.
|
323 | *
|
324 | * If any component is the empty string, an empty string will be inserted
|
325 | * into the generated ARN at the location that component corresponds to.
|
326 | *
|
327 | * The ARN will be formatted as follows:
|
328 | *
|
329 | * arn:{partition}:{service}:{region}:{account}:{resource}{sep}}{resource-name}
|
330 | *
|
331 | * The required ARN pieces that are omitted will be taken from the stack that
|
332 | * the 'scope' is attached to. If all ARN pieces are supplied, the supplied scope
|
333 | * can be 'undefined'.
|
334 | */
|
335 | formatArn(components) {
|
336 | try {
|
337 | jsiiDeprecationWarnings._aws_cdk_core_ArnComponents(components);
|
338 | }
|
339 | catch (error) {
|
340 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
341 | Error.captureStackTrace(error, this.formatArn);
|
342 | }
|
343 | throw error;
|
344 | }
|
345 | return arn_1.Arn.format(components, this);
|
346 | }
|
347 | /**
|
348 | * Given an ARN, parses it and returns components.
|
349 | *
|
350 | * IF THE ARN IS A CONCRETE STRING...
|
351 | *
|
352 | * ...it will be parsed and validated. The separator (`sep`) will be set to '/'
|
353 | * if the 6th component includes a '/', in which case, `resource` will be set
|
354 | * to the value before the '/' and `resourceName` will be the rest. In case
|
355 | * there is no '/', `resource` will be set to the 6th components and
|
356 | * `resourceName` will be set to the rest of the string.
|
357 | *
|
358 | * IF THE ARN IS A TOKEN...
|
359 | *
|
360 | * ...it cannot be validated, since we don't have the actual value yet at the
|
361 | * time of this function call. You will have to supply `sepIfToken` and
|
362 | * whether or not ARNs of the expected format usually have resource names
|
363 | * in order to parse it properly. The resulting `ArnComponents` object will
|
364 | * contain tokens for the subexpressions of the ARN, not string literals.
|
365 | *
|
366 | * If the resource name could possibly contain the separator char, the actual
|
367 | * resource name cannot be properly parsed. This only occurs if the separator
|
368 | * char is '/', and happens for example for S3 object ARNs, IAM Role ARNs,
|
369 | * IAM OIDC Provider ARNs, etc. To properly extract the resource name from a
|
370 | * Tokenized ARN, you must know the resource type and call
|
371 | * `Arn.extractResourceName`.
|
372 | *
|
373 | * @param arn The ARN string to parse
|
374 | * @param sepIfToken The separator used to separate resource from resourceName
|
375 | * @param hasName Whether there is a name component in the ARN at all. For
|
376 | * example, SNS Topics ARNs have the 'resource' component contain the topic
|
377 | * name, and no 'resourceName' component.
|
378 | *
|
379 | * @returns an ArnComponents object which allows access to the various
|
380 | * components of the ARN.
|
381 | *
|
382 | * @returns an ArnComponents object which allows access to the various
|
383 | * components of the ARN.
|
384 | *
|
385 | * @deprecated use splitArn instead
|
386 | */
|
387 | parseArn(arn, sepIfToken = '/', hasName = true) {
|
388 | try {
|
389 | jsiiDeprecationWarnings.print("@aws-cdk/core.Stack#parseArn", "use splitArn instead");
|
390 | }
|
391 | catch (error) {
|
392 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
393 | Error.captureStackTrace(error, this.parseArn);
|
394 | }
|
395 | throw error;
|
396 | }
|
397 | return arn_1.Arn.parse(arn, sepIfToken, hasName);
|
398 | }
|
399 | /**
|
400 | * Splits the provided ARN into its components.
|
401 | * Works both if 'arn' is a string like 'arn:aws:s3:::bucket',
|
402 | * and a Token representing a dynamic CloudFormation expression
|
403 | * (in which case the returned components will also be dynamic CloudFormation expressions,
|
404 | * encoded as Tokens).
|
405 | *
|
406 | * @param arn the ARN to split into its components
|
407 | * @param arnFormat the expected format of 'arn' - depends on what format the service 'arn' represents uses
|
408 | */
|
409 | splitArn(arn, arnFormat) {
|
410 | try {
|
411 | jsiiDeprecationWarnings._aws_cdk_core_ArnFormat(arnFormat);
|
412 | }
|
413 | catch (error) {
|
414 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
415 | Error.captureStackTrace(error, this.splitArn);
|
416 | }
|
417 | throw error;
|
418 | }
|
419 | return arn_1.Arn.split(arn, arnFormat);
|
420 | }
|
421 | /**
|
422 | * Returns the list of AZs that are available in the AWS environment
|
423 | * (account/region) associated with this stack.
|
424 | *
|
425 | * If the stack is environment-agnostic (either account and/or region are
|
426 | * tokens), this property will return an array with 2 tokens that will resolve
|
427 | * at deploy-time to the first two availability zones returned from CloudFormation's
|
428 | * `Fn::GetAZs` intrinsic function.
|
429 | *
|
430 | * If they are not available in the context, returns a set of dummy values and
|
431 | * reports them as missing, and let the CLI resolve them by calling EC2
|
432 | * `DescribeAvailabilityZones` on the target environment.
|
433 | *
|
434 | * To specify a different strategy for selecting availability zones override this method.
|
435 | */
|
436 | get availabilityZones() {
|
437 | // if account/region are tokens, we can't obtain AZs through the context
|
438 | // provider, so we fallback to use Fn::GetAZs. the current lowest common
|
439 | // denominator is 2 AZs across all AWS regions.
|
440 | const agnostic = token_1.Token.isUnresolved(this.account) || token_1.Token.isUnresolved(this.region);
|
441 | if (agnostic) {
|
442 | return this.node.tryGetContext(cxapi.AVAILABILITY_ZONE_FALLBACK_CONTEXT_KEY) || [
|
443 | cfn_fn_1.Fn.select(0, cfn_fn_1.Fn.getAzs()),
|
444 | cfn_fn_1.Fn.select(1, cfn_fn_1.Fn.getAzs()),
|
445 | ];
|
446 | }
|
447 | const value = context_provider_1.ContextProvider.getValue(this, {
|
448 | provider: cxschema.ContextProvider.AVAILABILITY_ZONE_PROVIDER,
|
449 | dummyValue: ['dummy1a', 'dummy1b', 'dummy1c'],
|
450 | }).value;
|
451 | if (!Array.isArray(value)) {
|
452 | throw new Error(`Provider ${cxschema.ContextProvider.AVAILABILITY_ZONE_PROVIDER} expects a list`);
|
453 | }
|
454 | return value;
|
455 | }
|
456 | /**
|
457 | * Register a file asset on this Stack
|
458 | *
|
459 | * @deprecated Use `stack.synthesizer.addFileAsset()` if you are calling,
|
460 | * and a different IStackSynthesizer class if you are implementing.
|
461 | */
|
462 | addFileAsset(asset) {
|
463 | try {
|
464 | jsiiDeprecationWarnings.print("@aws-cdk/core.Stack#addFileAsset", "Use `stack.synthesizer.addFileAsset()` if you are calling,\nand a different IStackSynthesizer class if you are implementing.");
|
465 | jsiiDeprecationWarnings._aws_cdk_core_FileAssetSource(asset);
|
466 | }
|
467 | catch (error) {
|
468 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
469 | Error.captureStackTrace(error, this.addFileAsset);
|
470 | }
|
471 | throw error;
|
472 | }
|
473 | return this.synthesizer.addFileAsset(asset);
|
474 | }
|
475 | /**
|
476 | * Register a docker image asset on this Stack
|
477 | *
|
478 | * @deprecated Use `stack.synthesizer.addDockerImageAsset()` if you are calling,
|
479 | * and a different `IStackSynthesizer` class if you are implementing.
|
480 | */
|
481 | addDockerImageAsset(asset) {
|
482 | try {
|
483 | jsiiDeprecationWarnings.print("@aws-cdk/core.Stack#addDockerImageAsset", "Use `stack.synthesizer.addDockerImageAsset()` if you are calling,\nand a different `IStackSynthesizer` class if you are implementing.");
|
484 | jsiiDeprecationWarnings._aws_cdk_core_DockerImageAssetSource(asset);
|
485 | }
|
486 | catch (error) {
|
487 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
488 | Error.captureStackTrace(error, this.addDockerImageAsset);
|
489 | }
|
490 | throw error;
|
491 | }
|
492 | return this.synthesizer.addDockerImageAsset(asset);
|
493 | }
|
494 | /**
|
495 | * If this is a nested stack, returns it's parent stack.
|
496 | */
|
497 | get nestedStackParent() {
|
498 | return this.nestedStackResource && Stack.of(this.nestedStackResource);
|
499 | }
|
500 | /**
|
501 | * Returns the parent of a nested stack.
|
502 | *
|
503 | * @deprecated use `nestedStackParent`
|
504 | */
|
505 | get parentStack() {
|
506 | try {
|
507 | jsiiDeprecationWarnings.print("@aws-cdk/core.Stack#parentStack", "use `nestedStackParent`");
|
508 | }
|
509 | catch (error) {
|
510 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
511 | Error.captureStackTrace(error, jsiiDeprecationWarnings.getPropertyDescriptor(this, "parentStack").get);
|
512 | }
|
513 | throw error;
|
514 | }
|
515 | return this.nestedStackParent;
|
516 | }
|
517 | /**
|
518 | * Add a Transform to this stack. A Transform is a macro that AWS
|
519 | * CloudFormation uses to process your template.
|
520 | *
|
521 | * Duplicate values are removed when stack is synthesized.
|
522 | *
|
523 | * @see https://docs.aws.amazon.com/AWSCloudFormation/latest/UserGuide/transform-section-structure.html
|
524 | * @param transform The transform to add
|
525 | *
|
526 | * @example
|
527 | * declare const stack: Stack;
|
528 | *
|
529 | * stack.addTransform('AWS::Serverless-2016-10-31')
|
530 | */
|
531 | addTransform(transform) {
|
532 | if (!this.templateOptions.transforms) {
|
533 | this.templateOptions.transforms = [];
|
534 | }
|
535 | this.templateOptions.transforms.push(transform);
|
536 | }
|
537 | /**
|
538 | * Called implicitly by the `addDependency` helper function in order to
|
539 | * realize a dependency between two top-level stacks at the assembly level.
|
540 | *
|
541 | * Use `stack.addDependency` to define the dependency between any two stacks,
|
542 | * and take into account nested stack relationships.
|
543 | *
|
544 | * @internal
|
545 | */
|
546 | _addAssemblyDependency(target, reason) {
|
547 | // defensive: we should never get here for nested stacks
|
548 | if (this.nested || target.nested) {
|
549 | throw new Error('Cannot add assembly-level dependencies for nested stacks');
|
550 | }
|
551 | reason = reason || 'dependency added using stack.addDependency()';
|
552 | const cycle = target.stackDependencyReasons(this);
|
553 | if (cycle !== undefined) {
|
554 | // eslint-disable-next-line max-len
|
555 | throw new Error(`'${target.node.path}' depends on '${this.node.path}' (${cycle.join(', ')}). Adding this dependency (${reason}) would create a cyclic reference.`);
|
556 | }
|
557 | let dep = this._stackDependencies[names_1.Names.uniqueId(target)];
|
558 | if (!dep) {
|
559 | dep = this._stackDependencies[names_1.Names.uniqueId(target)] = {
|
560 | stack: target,
|
561 | reasons: [],
|
562 | };
|
563 | }
|
564 | dep.reasons.push(reason);
|
565 | if (process.env.CDK_DEBUG_DEPS) {
|
566 | // eslint-disable-next-line no-console
|
567 | console.error(`[CDK_DEBUG_DEPS] stack "${this.node.path}" depends on "${target.node.path}" because: ${reason}`);
|
568 | }
|
569 | }
|
570 | /**
|
571 | * Synthesizes the cloudformation template into a cloud assembly.
|
572 | * @internal
|
573 | */
|
574 | _synthesizeTemplate(session, lookupRoleArn) {
|
575 | // In principle, stack synthesis is delegated to the
|
576 | // StackSynthesis object.
|
577 | //
|
578 | // However, some parts of synthesis currently use some private
|
579 | // methods on Stack, and I don't really see the value in refactoring
|
580 | // this right now, so some parts still happen here.
|
581 | const builder = session.assembly;
|
582 | const template = this._toCloudFormation();
|
583 | // write the CloudFormation template as a JSON file
|
584 | const outPath = path.join(builder.outdir, this.templateFile);
|
585 | if (this.maxResources > 0) {
|
586 | const resources = template.Resources || {};
|
587 | const numberOfResources = Object.keys(resources).length;
|
588 | if (numberOfResources > this.maxResources) {
|
589 | const counts = Object.entries(count(Object.values(resources).map((r) => `${r?.Type}`))).map(([type, c]) => `${type} (${c})`).join(', ');
|
590 | throw new Error(`Number of resources in stack '${this.node.path}': ${numberOfResources} is greater than allowed maximum of ${this.maxResources}: ${counts}`);
|
591 | }
|
592 | else if (numberOfResources >= (this.maxResources * 0.8)) {
|
593 | annotations_1.Annotations.of(this).addInfo(`Number of resources: ${numberOfResources} is approaching allowed maximum of ${this.maxResources}`);
|
594 | }
|
595 | }
|
596 | fs.writeFileSync(outPath, JSON.stringify(template, undefined, 1));
|
597 | for (const ctx of this._missingContext) {
|
598 | if (lookupRoleArn != null) {
|
599 | builder.addMissing({ ...ctx, props: { ...ctx.props, lookupRoleArn } });
|
600 | }
|
601 | else {
|
602 | builder.addMissing(ctx);
|
603 | }
|
604 | }
|
605 | }
|
606 | /**
|
607 | * Look up a fact value for the given fact for the region of this stack
|
608 | *
|
609 | * Will return a definite value only if the region of the current stack is resolved.
|
610 | * If not, a lookup map will be added to the stack and the lookup will be done at
|
611 | * CDK deployment time.
|
612 | *
|
613 | * What regions will be included in the lookup map is controlled by the
|
614 | * `@aws-cdk/core:target-partitions` context value: it must be set to a list
|
615 | * of partitions, and only regions from the given partitions will be included.
|
616 | * If no such context key is set, all regions will be included.
|
617 | *
|
618 | * This function is intended to be used by construct library authors. Application
|
619 | * builders can rely on the abstractions offered by construct libraries and do
|
620 | * not have to worry about regional facts.
|
621 | *
|
622 | * If `defaultValue` is not given, it is an error if the fact is unknown for
|
623 | * the given region.
|
624 | */
|
625 | regionalFact(factName, defaultValue) {
|
626 | if (!token_1.Token.isUnresolved(this.region)) {
|
627 | const ret = region_info_1.Fact.find(this.region, factName) ?? defaultValue;
|
628 | if (ret === undefined) {
|
629 | throw new Error(`region-info: don't know ${factName} for region ${this.region}. Use 'Fact.register' to provide this value.`);
|
630 | }
|
631 | return ret;
|
632 | }
|
633 | const partitions = constructs_1.Node.of(this).tryGetContext(cxapi.TARGET_PARTITIONS);
|
634 | if (partitions !== undefined && !Array.isArray(partitions)) {
|
635 | throw new Error(`Context value '${cxapi.TARGET_PARTITIONS}' should be a list of strings, got: ${JSON.stringify(cxapi.TARGET_PARTITIONS)}`);
|
636 | }
|
637 | const lookupMap = partitions ? region_info_1.RegionInfo.limitedRegionMap(factName, partitions) : region_info_1.RegionInfo.regionMap(factName);
|
638 | return region_lookup_1.deployTimeLookup(this, factName, lookupMap, defaultValue);
|
639 | }
|
640 | /**
|
641 | * Create a CloudFormation Export for a value
|
642 | *
|
643 | * Returns a string representing the corresponding `Fn.importValue()`
|
644 | * expression for this Export. You can control the name for the export by
|
645 | * passing the `name` option.
|
646 | *
|
647 | * If you don't supply a value for `name`, the value you're exporting must be
|
648 | * a Resource attribute (for example: `bucket.bucketName`) and it will be
|
649 | * given the same name as the automatic cross-stack reference that would be created
|
650 | * if you used the attribute in another Stack.
|
651 | *
|
652 | * One of the uses for this method is to *remove* the relationship between
|
653 | * two Stacks established by automatic cross-stack references. It will
|
654 | * temporarily ensure that the CloudFormation Export still exists while you
|
655 | * remove the reference from the consuming stack. After that, you can remove
|
656 | * the resource and the manual export.
|
657 | *
|
658 | * ## Example
|
659 | *
|
660 | * Here is how the process works. Let's say there are two stacks,
|
661 | * `producerStack` and `consumerStack`, and `producerStack` has a bucket
|
662 | * called `bucket`, which is referenced by `consumerStack` (perhaps because
|
663 | * an AWS Lambda Function writes into it, or something like that).
|
664 | *
|
665 | * It is not safe to remove `producerStack.bucket` because as the bucket is being
|
666 | * deleted, `consumerStack` might still be using it.
|
667 | *
|
668 | * Instead, the process takes two deployments:
|
669 | *
|
670 | * ### Deployment 1: break the relationship
|
671 | *
|
672 | * - Make sure `consumerStack` no longer references `bucket.bucketName` (maybe the consumer
|
673 | * stack now uses its own bucket, or it writes to an AWS DynamoDB table, or maybe you just
|
674 | * remove the Lambda Function altogether).
|
675 | * - In the `ProducerStack` class, call `this.exportValue(this.bucket.bucketName)`. This
|
676 | * will make sure the CloudFormation Export continues to exist while the relationship
|
677 | * between the two stacks is being broken.
|
678 | * - Deploy (this will effectively only change the `consumerStack`, but it's safe to deploy both).
|
679 | *
|
680 | * ### Deployment 2: remove the bucket resource
|
681 | *
|
682 | * - You are now free to remove the `bucket` resource from `producerStack`.
|
683 | * - Don't forget to remove the `exportValue()` call as well.
|
684 | * - Deploy again (this time only the `producerStack` will be changed -- the bucket will be deleted).
|
685 | */
|
686 | exportValue(exportedValue, options = {}) {
|
687 | try {
|
688 | jsiiDeprecationWarnings._aws_cdk_core_ExportValueOptions(options);
|
689 | }
|
690 | catch (error) {
|
691 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
692 | Error.captureStackTrace(error, this.exportValue);
|
693 | }
|
694 | throw error;
|
695 | }
|
696 | if (options.name) {
|
697 | new cfn_output_1.CfnOutput(this, `Export${options.name}`, {
|
698 | value: exportedValue,
|
699 | exportName: options.name,
|
700 | });
|
701 | return cfn_fn_1.Fn.importValue(options.name);
|
702 | }
|
703 | const resolvable = token_1.Tokenization.reverse(exportedValue);
|
704 | if (!resolvable || !reference_1.Reference.isReference(resolvable)) {
|
705 | throw new Error('exportValue: either supply \'name\' or make sure to export a resource attribute (like \'bucket.bucketName\')');
|
706 | }
|
707 | // if exportValue is being called manually (which is pre onPrepare) then the logicalId
|
708 | // could potentially be changed by a call to overrideLogicalId. This would cause our Export/Import
|
709 | // to have an incorrect id. For a better user experience, lock the logicalId and throw an error
|
710 | // if the user tries to override the id _after_ calling exportValue
|
711 | if (cfn_element_1.CfnElement.isCfnElement(resolvable.target)) {
|
712 | resolvable.target._lockLogicalId();
|
713 | }
|
714 | // "teleport" the value here, in case it comes from a nested stack. This will also
|
715 | // ensure the value is from our own scope.
|
716 | const exportable = refs_1.referenceNestedStackValueInParent(resolvable, this);
|
717 | // Ensure a singleton "Exports" scoping Construct
|
718 | // This mostly exists to trigger LogicalID munging, which would be
|
719 | // disabled if we parented constructs directly under Stack.
|
720 | // Also it nicely prevents likely construct name clashes
|
721 | const exportsScope = getCreateExportsScope(this);
|
722 | // Ensure a singleton CfnOutput for this value
|
723 | const resolved = this.resolve(exportable);
|
724 | const id = 'Output' + JSON.stringify(resolved);
|
725 | const exportName = generateExportName(exportsScope, id);
|
726 | if (token_1.Token.isUnresolved(exportName)) {
|
727 | throw new Error(`unresolved token in generated export name: ${JSON.stringify(this.resolve(exportName))}`);
|
728 | }
|
729 | const output = exportsScope.node.tryFindChild(id);
|
730 | if (!output) {
|
731 | new cfn_output_1.CfnOutput(exportsScope, id, { value: token_1.Token.asString(exportable), exportName });
|
732 | }
|
733 | return cfn_fn_1.Fn.importValue(exportName);
|
734 | }
|
735 | /**
|
736 | * Returns the naming scheme used to allocate logical IDs. By default, uses
|
737 | * the `HashedAddressingScheme` but this method can be overridden to customize
|
738 | * this behavior.
|
739 | *
|
740 | * In order to make sure logical IDs are unique and stable, we hash the resource
|
741 | * construct tree path (i.e. toplevel/secondlevel/.../myresource) and add it as
|
742 | * a suffix to the path components joined without a separator (CloudFormation
|
743 | * IDs only allow alphanumeric characters).
|
744 | *
|
745 | * The result will be:
|
746 | *
|
747 | * <path.join('')><md5(path.join('/')>
|
748 | * "human" "hash"
|
749 | *
|
750 | * If the "human" part of the ID exceeds 240 characters, we simply trim it so
|
751 | * the total ID doesn't exceed CloudFormation's 255 character limit.
|
752 | *
|
753 | * We only take 8 characters from the md5 hash (0.000005 chance of collision).
|
754 | *
|
755 | * Special cases:
|
756 | *
|
757 | * - If the path only contains a single component (i.e. it's a top-level
|
758 | * resource), we won't add the hash to it. The hash is not needed for
|
759 | * disamiguation and also, it allows for a more straightforward migration an
|
760 | * existing CloudFormation template to a CDK stack without logical ID changes
|
761 | * (or renames).
|
762 | * - For aesthetic reasons, if the last components of the path are the same
|
763 | * (i.e. `L1/L2/Pipeline/Pipeline`), they will be de-duplicated to make the
|
764 | * resulting human portion of the ID more pleasing: `L1L2Pipeline<HASH>`
|
765 | * instead of `L1L2PipelinePipeline<HASH>`
|
766 | * - If a component is named "Default" it will be omitted from the path. This
|
767 | * allows refactoring higher level abstractions around constructs without affecting
|
768 | * the IDs of already deployed resources.
|
769 | * - If a component is named "Resource" it will be omitted from the user-visible
|
770 | * path, but included in the hash. This reduces visual noise in the human readable
|
771 | * part of the identifier.
|
772 | *
|
773 | * @param cfnElement The element for which the logical ID is allocated.
|
774 | */
|
775 | allocateLogicalId(cfnElement) {
|
776 | try {
|
777 | jsiiDeprecationWarnings._aws_cdk_core_CfnElement(cfnElement);
|
778 | }
|
779 | catch (error) {
|
780 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
781 | Error.captureStackTrace(error, this.allocateLogicalId);
|
782 | }
|
783 | throw error;
|
784 | }
|
785 | const scopes = cfnElement.node.scopes;
|
786 | const stackIndex = scopes.indexOf(cfnElement.stack);
|
787 | const pathComponents = scopes.slice(stackIndex + 1).map(x => x.node.id);
|
788 | return uniqueid_1.makeUniqueId(pathComponents);
|
789 | }
|
790 | /**
|
791 | * Validate stack name
|
792 | *
|
793 | * CloudFormation stack names can include dashes in addition to the regular identifier
|
794 | * character classes, and we don't allow one of the magic markers.
|
795 | *
|
796 | * @internal
|
797 | */
|
798 | _validateId(name) {
|
799 | if (name && !VALID_STACK_NAME_REGEX.test(name)) {
|
800 | throw new Error(`Stack name must match the regular expression: ${VALID_STACK_NAME_REGEX.toString()}, got '${name}'`);
|
801 | }
|
802 | }
|
803 | /**
|
804 | * Returns the CloudFormation template for this stack by traversing
|
805 | * the tree and invoking _toCloudFormation() on all Entity objects.
|
806 | *
|
807 | * @internal
|
808 | */
|
809 | _toCloudFormation() {
|
810 | let transform;
|
811 | if (this.templateOptions.transform) {
|
812 | // eslint-disable-next-line max-len
|
813 | annotations_1.Annotations.of(this).addWarning('This stack is using the deprecated `templateOptions.transform` property. Consider switching to `addTransform()`.');
|
814 | this.addTransform(this.templateOptions.transform);
|
815 | }
|
816 | if (this.templateOptions.transforms) {
|
817 | if (this.templateOptions.transforms.length === 1) { // Extract single value
|
818 | transform = this.templateOptions.transforms[0];
|
819 | }
|
820 | else { // Remove duplicate values
|
821 | transform = Array.from(new Set(this.templateOptions.transforms));
|
822 | }
|
823 | }
|
824 | const template = {
|
825 | Description: this.templateOptions.description,
|
826 | Transform: transform,
|
827 | AWSTemplateFormatVersion: this.templateOptions.templateFormatVersion,
|
828 | Metadata: this.templateOptions.metadata,
|
829 | };
|
830 | const elements = cfnElements(this);
|
831 | const fragments = elements.map(e => this.resolve(e._toCloudFormation()));
|
832 | // merge in all CloudFormation fragments collected from the tree
|
833 | for (const fragment of fragments) {
|
834 | merge(template, fragment);
|
835 | }
|
836 | // resolve all tokens and remove all empties
|
837 | const ret = this.resolve(template) || {};
|
838 | this._logicalIds.assertAllRenamesApplied();
|
839 | return ret;
|
840 | }
|
841 | /**
|
842 | * Deprecated.
|
843 | *
|
844 | * @see https://github.com/aws/aws-cdk/pull/7187
|
845 | * @returns reference itself without any change
|
846 | * @deprecated cross reference handling has been moved to `App.prepare()`.
|
847 | */
|
848 | prepareCrossReference(_sourceStack, reference) {
|
849 | try {
|
850 | jsiiDeprecationWarnings.print("@aws-cdk/core.Stack#prepareCrossReference", "cross reference handling has been moved to `App.prepare()`.");
|
851 | jsiiDeprecationWarnings._aws_cdk_core_Stack(_sourceStack);
|
852 | jsiiDeprecationWarnings._aws_cdk_core_Reference(reference);
|
853 | }
|
854 | catch (error) {
|
855 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
856 | Error.captureStackTrace(error, this.prepareCrossReference);
|
857 | }
|
858 | throw error;
|
859 | }
|
860 | return reference;
|
861 | }
|
862 | /**
|
863 | * Determine the various stack environment attributes.
|
864 | *
|
865 | */
|
866 | parseEnvironment(env = {}) {
|
867 | // if an environment property is explicitly specified when the stack is
|
868 | // created, it will be used. if not, use tokens for account and region.
|
869 | //
|
870 | // (They do not need to be anchored to any construct like resource attributes
|
871 | // are, because we'll never Export/Fn::ImportValue them -- the only situation
|
872 | // in which Export/Fn::ImportValue would work is if the value are the same
|
873 | // between producer and consumer anyway, so we can just assume that they are).
|
874 | const containingAssembly = stage_1.Stage.of(this);
|
875 | const account = env.account ?? containingAssembly?.account ?? cfn_pseudo_1.Aws.ACCOUNT_ID;
|
876 | const region = env.region ?? containingAssembly?.region ?? cfn_pseudo_1.Aws.REGION;
|
877 | // this is the "aws://" env specification that will be written to the cloud assembly
|
878 | // manifest. it will use "unknown-account" and "unknown-region" to indicate
|
879 | // environment-agnosticness.
|
880 | const envAccount = !token_1.Token.isUnresolved(account) ? account : cxapi.UNKNOWN_ACCOUNT;
|
881 | const envRegion = !token_1.Token.isUnresolved(region) ? region : cxapi.UNKNOWN_REGION;
|
882 | return {
|
883 | account,
|
884 | region,
|
885 | environment: cxapi.EnvironmentUtils.format(envAccount, envRegion),
|
886 | };
|
887 | }
|
888 | /**
|
889 | * Maximum number of resources in the stack
|
890 | *
|
891 | * Set to 0 to mean "unlimited".
|
892 | */
|
893 | get maxResources() {
|
894 | const contextLimit = this.node.tryGetContext(exports.STACK_RESOURCE_LIMIT_CONTEXT);
|
895 | return contextLimit !== undefined ? parseInt(contextLimit, 10) : MAX_RESOURCES;
|
896 | }
|
897 | /**
|
898 | * Check whether this stack has a (transitive) dependency on another stack
|
899 | *
|
900 | * Returns the list of reasons on the dependency path, or undefined
|
901 | * if there is no dependency.
|
902 | */
|
903 | stackDependencyReasons(other) {
|
904 | if (this === other) {
|
905 | return [];
|
906 | }
|
907 | for (const dep of Object.values(this._stackDependencies)) {
|
908 | const ret = dep.stack.stackDependencyReasons(other);
|
909 | if (ret !== undefined) {
|
910 | return [...dep.reasons, ...ret];
|
911 | }
|
912 | }
|
913 | return undefined;
|
914 | }
|
915 | /**
|
916 | * Calculate the stack name based on the construct path
|
917 | *
|
918 | * The stack name is the name under which we'll deploy the stack,
|
919 | * and incorporates containing Stage names by default.
|
920 | *
|
921 | * Generally this looks a lot like how logical IDs are calculated.
|
922 | * The stack name is calculated based on the construct root path,
|
923 | * as follows:
|
924 | *
|
925 | * - Path is calculated with respect to containing App or Stage (if any)
|
926 | * - If the path is one component long just use that component, otherwise
|
927 | * combine them with a hash.
|
928 | *
|
929 | * Since the hash is quite ugly and we'd like to avoid it if possible -- but
|
930 | * we can't anymore in the general case since it has been written into legacy
|
931 | * stacks. The introduction of Stages makes it possible to make this nicer however.
|
932 | * When a Stack is nested inside a Stage, we use the path components below the
|
933 | * Stage, and prefix the path components of the Stage before it.
|
934 | */
|
935 | generateStackName() {
|
936 | const assembly = stage_1.Stage.of(this);
|
937 | const prefix = (assembly && assembly.stageName) ? `${assembly.stageName}-` : '';
|
938 | return `${prefix}${this.generateStackId(assembly)}`;
|
939 | }
|
940 | /**
|
941 | * The artifact ID for this stack
|
942 | *
|
943 | * Stack artifact ID is unique within the App's Cloud Assembly.
|
944 | */
|
945 | generateStackArtifactId() {
|
946 | return this.generateStackId(this.node.root);
|
947 | }
|
948 | /**
|
949 | * Generate an ID with respect to the given container construct.
|
950 | */
|
951 | generateStackId(container) {
|
952 | const rootPath = rootPathTo(this, container);
|
953 | const ids = rootPath.map(c => constructs_1.Node.of(c).id);
|
954 | // In unit tests our Stack (which is the only component) may not have an
|
955 | // id, so in that case just pretend it's "Stack".
|
956 | if (ids.length === 1 && !ids[0]) {
|
957 | throw new Error('unexpected: stack id must always be defined');
|
958 | }
|
959 | return makeStackName(ids);
|
960 | }
|
961 | /**
|
962 | * Indicates whether the stack requires bundling or not
|
963 | */
|
964 | get bundlingRequired() {
|
965 | const bundlingStacks = this.node.tryGetContext(cxapi.BUNDLING_STACKS) ?? ['*'];
|
966 | // bundlingStacks is of the form `Stage/Stack`, convert it to `Stage-Stack` before comparing to stack name
|
967 | return bundlingStacks.some(pattern => minimatch(this.stackName, pattern.replace('/', '-')));
|
968 | }
|
969 | }
|
970 | exports.Stack = Stack;
|
971 | _a = JSII_RTTI_SYMBOL_1;
|
972 | Stack[_a] = { fqn: "@aws-cdk/core.Stack", version: "1.204.0" };
|
973 | function merge(template, fragment) {
|
974 | for (const section of Object.keys(fragment)) {
|
975 | const src = fragment[section];
|
976 | // create top-level section if it doesn't exist
|
977 | const dest = template[section];
|
978 | if (!dest) {
|
979 | template[section] = src;
|
980 | }
|
981 | else {
|
982 | template[section] = mergeSection(section, dest, src);
|
983 | }
|
984 | }
|
985 | }
|
986 | function mergeSection(section, val1, val2) {
|
987 | switch (section) {
|
988 | case 'Description':
|
989 | return `${val1}\n${val2}`;
|
990 | case 'AWSTemplateFormatVersion':
|
991 | if (val1 != null && val2 != null && val1 !== val2) {
|
992 | throw new Error(`Conflicting CloudFormation template versions provided: '${val1}' and '${val2}`);
|
993 | }
|
994 | return val1 ?? val2;
|
995 | case 'Transform':
|
996 | return mergeSets(val1, val2);
|
997 | default:
|
998 | return mergeObjectsWithoutDuplicates(section, val1, val2);
|
999 | }
|
1000 | }
|
1001 | function mergeSets(val1, val2) {
|
1002 | const array1 = val1 == null ? [] : (Array.isArray(val1) ? val1 : [val1]);
|
1003 | const array2 = val2 == null ? [] : (Array.isArray(val2) ? val2 : [val2]);
|
1004 | for (const value of array2) {
|
1005 | if (!array1.includes(value)) {
|
1006 | array1.push(value);
|
1007 | }
|
1008 | }
|
1009 | return array1.length === 1 ? array1[0] : array1;
|
1010 | }
|
1011 | function mergeObjectsWithoutDuplicates(section, dest, src) {
|
1012 | if (typeof dest !== 'object') {
|
1013 | throw new Error(`Expecting ${JSON.stringify(dest)} to be an object`);
|
1014 | }
|
1015 | if (typeof src !== 'object') {
|
1016 | throw new Error(`Expecting ${JSON.stringify(src)} to be an object`);
|
1017 | }
|
1018 | // add all entities from source section to destination section
|
1019 | for (const id of Object.keys(src)) {
|
1020 | if (id in dest) {
|
1021 | throw new Error(`section '${section}' already contains '${id}'`);
|
1022 | }
|
1023 | dest[id] = src[id];
|
1024 | }
|
1025 | return dest;
|
1026 | }
|
1027 | /**
|
1028 | * Collect all CfnElements from a Stack.
|
1029 | *
|
1030 | * @param node Root node to collect all CfnElements from
|
1031 | * @param into Array to append CfnElements to
|
1032 | * @returns The same array as is being collected into
|
1033 | */
|
1034 | function cfnElements(node, into = []) {
|
1035 | if (cfn_element_1.CfnElement.isCfnElement(node)) {
|
1036 | into.push(node);
|
1037 | }
|
1038 | for (const child of constructs_1.Node.of(node).children) {
|
1039 | // Don't recurse into a substack
|
1040 | if (Stack.isStack(child)) {
|
1041 | continue;
|
1042 | }
|
1043 | cfnElements(child, into);
|
1044 | }
|
1045 | return into;
|
1046 | }
|
1047 | /**
|
1048 | * Return the construct root path of the given construct relative to the given ancestor
|
1049 | *
|
1050 | * If no ancestor is given or the ancestor is not found, return the entire root path.
|
1051 | */
|
1052 | function rootPathTo(construct, ancestor) {
|
1053 | const scopes = constructs_1.Node.of(construct).scopes;
|
1054 | for (let i = scopes.length - 2; i >= 0; i--) {
|
1055 | if (scopes[i] === ancestor) {
|
1056 | return scopes.slice(i + 1);
|
1057 | }
|
1058 | }
|
1059 | return scopes;
|
1060 | }
|
1061 | exports.rootPathTo = rootPathTo;
|
1062 | /**
|
1063 | * makeUniqueId, specialized for Stack names
|
1064 | *
|
1065 | * Stack names may contain '-', so we allow that character if the stack name
|
1066 | * has only one component. Otherwise we fall back to the regular "makeUniqueId"
|
1067 | * behavior.
|
1068 | */
|
1069 | function makeStackName(components) {
|
1070 | if (components.length === 1) {
|
1071 | return components[0];
|
1072 | }
|
1073 | return uniqueid_1.makeUniqueId(components);
|
1074 | }
|
1075 | function getCreateExportsScope(stack) {
|
1076 | const exportsName = 'Exports';
|
1077 | let stackExports = stack.node.tryFindChild(exportsName);
|
1078 | if (stackExports === undefined) {
|
1079 | stackExports = new construct_compat_1.Construct(stack, exportsName);
|
1080 | }
|
1081 | return stackExports;
|
1082 | }
|
1083 | function generateExportName(stackExports, id) {
|
1084 | const stackRelativeExports = feature_flags_1.FeatureFlags.of(stackExports).isEnabled(cxapi.STACK_RELATIVE_EXPORTS_CONTEXT);
|
1085 | const stack = Stack.of(stackExports);
|
1086 | const components = [
|
1087 | ...stackExports.node.scopes
|
1088 | .slice(stackRelativeExports ? stack.node.scopes.length : 2)
|
1089 | .map(c => c.node.id),
|
1090 | id,
|
1091 | ];
|
1092 | const prefix = stack.stackName ? stack.stackName + ':' : '';
|
1093 | const localPart = uniqueid_1.makeUniqueId(components);
|
1094 | const maxLength = 255;
|
1095 | return prefix + localPart.slice(Math.max(0, localPart.length - maxLength + prefix.length));
|
1096 | }
|
1097 | function count(xs) {
|
1098 | const ret = {};
|
1099 | for (const x of xs) {
|
1100 | if (x in ret) {
|
1101 | ret[x] += 1;
|
1102 | }
|
1103 | else {
|
1104 | ret[x] = 1;
|
1105 | }
|
1106 | }
|
1107 | return ret;
|
1108 | }
|
1109 | // These imports have to be at the end to prevent circular imports
|
1110 | const cfn_output_1 = require("./cfn-output");
|
1111 | const deps_1 = require("./deps");
|
1112 | const fs_1 = require("./fs");
|
1113 | const names_1 = require("./names");
|
1114 | const reference_1 = require("./reference");
|
1115 | const stack_synthesizers_1 = require("./stack-synthesizers");
|
1116 | const stage_1 = require("./stage");
|
1117 | const tag_manager_1 = require("./tag-manager");
|
1118 | const token_1 = require("./token");
|
1119 | const refs_1 = require("./private/refs");
|
1120 | const region_info_1 = require("@aws-cdk/region-info");
|
1121 | const region_lookup_1 = require("./private/region-lookup");
|
1122 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stack.js","sourceRoot":"","sources":["stack.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from 'fs';\nimport * as path from 'path';\nimport * as cxschema from '@aws-cdk/cloud-assembly-schema';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport { IConstruct, Construct, Node } from 'constructs';\nimport * as minimatch from 'minimatch';\nimport { Annotations } from './annotations';\nimport { App } from './app';\nimport { Arn, ArnComponents, ArnFormat } from './arn';\nimport { DockerImageAssetLocation, DockerImageAssetSource, FileAssetLocation, FileAssetSource } from './assets';\nimport { CfnElement } from './cfn-element';\nimport { Fn } from './cfn-fn';\nimport { Aws, ScopedAws } from './cfn-pseudo';\nimport { CfnResource, TagType } from './cfn-resource';\nimport { ISynthesisSession } from './construct-compat';\nimport { ContextProvider } from './context-provider';\nimport { Environment } from './environment';\nimport { FeatureFlags } from './feature-flags';\nimport { CLOUDFORMATION_TOKEN_RESOLVER, CloudFormationLang } from './private/cloudformation-lang';\nimport { LogicalIDs } from './private/logical-id';\nimport { resolve } from './private/resolve';\nimport { makeUniqueId } from './private/uniqueid';\n\n// v2 - keep this import as a separate section to reduce merge conflict when forward merging with the v2 branch.\n// eslint-disable-next-line\nimport { Construct as CoreConstruct } from './construct-compat';\n\nconst STACK_SYMBOL = Symbol.for('@aws-cdk/core.Stack');\nconst MY_STACK_CACHE = Symbol.for('@aws-cdk/core.Stack.myStack');\n\nexport const STACK_RESOURCE_LIMIT_CONTEXT = '@aws-cdk/core:stackResourceLimit';\n\nconst VALID_STACK_NAME_REGEX = /^[A-Za-z][A-Za-z0-9-]*$/;\n\nconst MAX_RESOURCES = 500;\n\nexport interface StackProps {\n  /**\n   * A description of the stack.\n   *\n   * @default - No description.\n   */\n  readonly description?: string;\n\n  /**\n   * The AWS environment (account/region) where this stack will be deployed.\n   *\n   * Set the `region`/`account` fields of `env` to either a concrete value to\n   * select the indicated environment (recommended for production stacks), or to\n   * the values of environment variables\n   * `CDK_DEFAULT_REGION`/`CDK_DEFAULT_ACCOUNT` to let the target environment\n   * depend on the AWS credentials/configuration that the CDK CLI is executed\n   * under (recommended for development stacks).\n   *\n   * If the `Stack` is instantiated inside a `Stage`, any undefined\n   * `region`/`account` fields from `env` will default to the same field on the\n   * encompassing `Stage`, if configured there.\n   *\n   * If either `region` or `account` are not set nor inherited from `Stage`, the\n   * Stack will be considered \"*environment-agnostic*\"\". Environment-agnostic\n   * stacks can be deployed to any environment but may not be able to take\n   * advantage of all features of the CDK. For example, they will not be able to\n   * use environmental context lookups such as `ec2.Vpc.fromLookup` and will not\n   * automatically translate Service Principals to the right format based on the\n   * environment's AWS partition, and other such enhancements.\n   *\n   * @example\n   *\n   * // Use a concrete account and region to deploy this stack to:\n   * // `.account` and `.region` will simply return these values.\n   * new Stack(app, 'Stack1', {\n   *   env: {\n   *     account: '123456789012',\n   *     region: 'us-east-1'\n   *   },\n   * });\n   *\n   * // Use the CLI's current credentials to determine the target environment:\n   * // `.account` and `.region` will reflect the account+region the CLI\n   * // is configured to use (based on the user CLI credentials)\n   * new Stack(app, 'Stack2', {\n   *   env: {\n   *     account: process.env.CDK_DEFAULT_ACCOUNT,\n   *     region: process.env.CDK_DEFAULT_REGION\n   *   },\n   * });\n   *\n   * // Define multiple stacks stage associated with an environment\n   * const myStage = new Stage(app, 'MyStage', {\n   *   env: {\n   *     account: '123456789012',\n   *     region: 'us-east-1'\n   *   }\n   * });\n   *\n   * // both of these stacks will use the stage's account/region:\n   * // `.account` and `.region` will resolve to the concrete values as above\n   * new MyStack(myStage, 'Stack1');\n   * new YourStack(myStage, 'Stack2');\n   *\n   * // Define an environment-agnostic stack:\n   * // `.account` and `.region` will resolve to `{ \"Ref\": \"AWS::AccountId\" }` and `{ \"Ref\": \"AWS::Region\" }` respectively.\n   * // which will only resolve to actual values by CloudFormation during deployment.\n   * new MyStack(app, 'Stack1');\n   *\n   * @default - The environment of the containing `Stage` if available,\n   * otherwise create the stack will be environment-agnostic.\n   */\n  readonly env?: Environment;\n\n  /**\n   * Name to deploy the stack with\n   *\n   * @default - Derived from construct path.\n   */\n  readonly stackName?: string;\n\n  /**\n   * Stack tags that will be applied to all the taggable resources and the stack itself.\n   *\n   * @default {}\n   */\n  readonly tags?: { [key: string]: string };\n\n  /**\n   * Synthesis method to use while deploying this stack\n   *\n   * @default - `DefaultStackSynthesizer` if the `@aws-cdk/core:newStyleStackSynthesis` feature flag\n   * is set, `LegacyStackSynthesizer` otherwise.\n   */\n  readonly synthesizer?: IStackSynthesizer;\n\n  /**\n   * Whether to enable termination protection for this stack.\n   *\n   * @default false\n   */\n  readonly terminationProtection?: boolean;\n\n  /**\n   * Include runtime versioning information in this Stack\n   *\n   * @default `analyticsReporting` setting of containing `App`, or value of\n   * 'aws:cdk:version-reporting' context key\n   */\n  readonly analyticsReporting?: boolean;\n}\n\n/**\n * A root construct which represents a single CloudFormation stack.\n */\nexport class Stack extends CoreConstruct implements ITaggable {\n  /**\n   * Return whether the given object is a Stack.\n   *\n   * We do attribute detection since we can't reliably use 'instanceof'.\n   */\n  public static isStack(x: any): x is Stack {\n    return x !== null && typeof(x) === 'object' && STACK_SYMBOL in x;\n  }\n\n  /**\n   * Looks up the first stack scope in which `construct` is defined. Fails if there is no stack up the tree.\n   * @param construct The construct to start the search from.\n   */\n  public static of(construct: IConstruct): Stack {\n    // we want this to be as cheap as possible. cache this result by mutating\n    // the object. anecdotally, at the time of this writing, @aws-cdk/core unit\n    // tests hit this cache 1,112 times, @aws-cdk/aws-cloudformation unit tests\n    // hit this 2,435 times).\n    const cache = (construct as any)[MY_STACK_CACHE] as Stack | undefined;\n    if (cache) {\n      return cache;\n    } else {\n      const value = _lookup(construct);\n      Object.defineProperty(construct, MY_STACK_CACHE, {\n        enumerable: false,\n        writable: false,\n        configurable: false,\n        value,\n      });\n      return value;\n    }\n\n    function _lookup(c: IConstruct): Stack {\n      if (Stack.isStack(c)) {\n        return c;\n      }\n\n      const _scope = Node.of(c).scope;\n      if (Stage.isStage(c) || !_scope) {\n        throw new Error(`${construct.constructor?.name ?? 'Construct'} at '${Node.of(construct).path}' should be created in the scope of a Stack, but no Stack found`);\n      }\n\n      return _lookup(_scope);\n    }\n  }\n\n  /**\n   * Tags to be applied to the stack.\n   */\n  public readonly tags: TagManager;\n\n  /**\n   * Options for CloudFormation template (like version, transform, description).\n   */\n  public readonly templateOptions: ITemplateOptions;\n\n  /**\n   * The AWS region into which this stack will be deployed (e.g. `us-west-2`).\n   *\n   * This value is resolved according to the following rules:\n   *\n   * 1. The value provided to `env.region` when the stack is defined. This can\n   *    either be a concerete region (e.g. `us-west-2`) or the `Aws.region`\n   *    token.\n   * 3. `Aws.region`, which is represents the CloudFormation intrinsic reference\n   *    `{ \"Ref\": \"AWS::Region\" }` encoded as a string token.\n   *\n   * Preferably, you should use the return value as an opaque string and not\n   * attempt to parse it to implement your logic. If you do, you must first\n   * check that it is a concerete value an not an unresolved token. If this\n   * value is an unresolved token (`Token.isUnresolved(stack.region)` returns\n   * `true`), this implies that the user wishes that this stack will synthesize\n   * into a **region-agnostic template**. In this case, your code should either\n   * fail (throw an error, emit a synth error using `Annotations.of(construct).addError()`) or\n   * implement some other region-agnostic behavior.\n   */\n  public readonly region: string;\n\n  /**\n   * The AWS account into which this stack will be deployed.\n   *\n   * This value is resolved according to the following rules:\n   *\n   * 1. The value provided to `env.account` when the stack is defined. This can\n   *    either be a concerete account (e.g. `585695031111`) or the\n   *    `Aws.accountId` token.\n   * 3. `Aws.accountId`, which represents the CloudFormation intrinsic reference\n   *    `{ \"Ref\": \"AWS::AccountId\" }` encoded as a string token.\n   *\n   * Preferably, you should use the return value as an opaque string and not\n   * attempt to parse it to implement your logic. If you do, you must first\n   * check that it is a concerete value an not an unresolved token. If this\n   * value is an unresolved token (`Token.isUnresolved(stack.account)` returns\n   * `true`), this implies that the user wishes that this stack will synthesize\n   * into a **account-agnostic template**. In this case, your code should either\n   * fail (throw an error, emit a synth error using `Annotations.of(construct).addError()`) or\n   * implement some other region-agnostic behavior.\n   */\n  public readonly account: string;\n\n  /**\n   * The environment coordinates in which this stack is deployed. In the form\n   * `aws://account/region`. Use `stack.account` and `stack.region` to obtain\n   * the specific values, no need to parse.\n   *\n   * You can use this value to determine if two stacks are targeting the same\n   * environment.\n   *\n   * If either `stack.account` or `stack.region` are not concrete values (e.g.\n   * `Aws.account` or `Aws.region`) the special strings `unknown-account` and/or\n   * `unknown-region` will be used respectively to indicate this stack is\n   * region/account-agnostic.\n   */\n  public readonly environment: string;\n\n  /**\n   * Whether termination protection is enabled for this stack.\n   */\n  public readonly terminationProtection?: boolean;\n\n  /**\n   * If this is a nested stack, this represents its `AWS::CloudFormation::Stack`\n   * resource. `undefined` for top-level (non-nested) stacks.\n   *\n   */\n  public readonly nestedStackResource?: CfnResource;\n\n  /**\n   * The name of the CloudFormation template file emitted to the output\n   * directory during synthesis.\n   *\n   * Example value: `MyStack.template.json`\n   */\n  public readonly templateFile: string;\n\n  /**\n   * The ID of the cloud assembly artifact for this stack.\n   */\n  public readonly artifactId: string;\n\n  /**\n   * Synthesis method for this stack\n   *\n   */\n  public readonly synthesizer: IStackSynthesizer;\n\n  /**\n   * Whether version reporting is enabled for this stack\n   *\n   * Controls whether the CDK Metadata resource is injected\n   *\n   * @internal\n   */\n  public readonly _versionReportingEnabled: boolean;\n\n  /**\n   * Logical ID generation strategy\n   */\n  private readonly _logicalIds: LogicalIDs;\n\n  /**\n   * Other stacks this stack depends on\n   */\n  private readonly _stackDependencies: { [uniqueId: string]: StackDependency };\n\n  /**\n   * Lists all missing contextual information.\n   * This is returned when the stack is synthesized under the 'missing' attribute\n   * and allows tooling to obtain the context and re-synthesize.\n   */\n  private readonly _missingContext: cxschema.MissingContext[];\n\n  private readonly _stackName: string;\n\n  /**\n   * Creates a new stack.\n   *\n   * @param scope Parent of this stack, usually an `App` or a `Stage`, but could be any construct.\n   * @param id The construct ID of this stack. If `stackName` is not explicitly\n   * defined, this id (and any parent IDs) will be used to determine the\n   * physical ID of the stack.\n   * @param props Stack properties.\n   */\n  public constructor(scope?: Construct, id?: string, props: StackProps = {}) {\n    // For unit test scope and id are optional for stacks, but we still want an App\n    // as the parent because apps implement much of the synthesis logic.\n    scope = scope ?? new App({\n      autoSynth: false,\n      outdir: FileSystem.mkdtemp('cdk-test-app-'),\n    });\n\n    // \"Default\" is a \"hidden id\" from a `node.uniqueId` perspective\n    id = id ?? 'Default';\n\n    super(scope, id);\n\n    this._missingContext = new Array<cxschema.MissingContext>();\n    this._stackDependencies = { };\n    this.templateOptions = { };\n\n    Object.defineProperty(this, STACK_SYMBOL, { value: true });\n\n    this._logicalIds = new LogicalIDs();\n\n    const { account, region, environment } = this.parseEnvironment(props.env);\n\n    this.account = account;\n    this.region = region;\n    this.environment = environment;\n    this.terminationProtection = props.terminationProtection;\n\n    if (props.description !== undefined) {\n      // Max length 1024 bytes\n      // Typically 2 bytes per character, may be more for more exotic characters\n      if (props.description.length > 512) {\n        throw new Error(`Stack description must be <= 1024 bytes. Received description: '${props.description}'`);\n      }\n      this.templateOptions.description = props.description;\n    }\n\n    this._stackName = props.stackName ?? this.generateStackName();\n    if (this._stackName.length > 128) {\n      throw new Error(`Stack name must be <= 128 characters. Stack name: '${this._stackName}'`);\n    }\n    this.tags = new TagManager(TagType.KEY_VALUE, 'aws:cdk:stack', props.tags);\n\n    if (!VALID_STACK_NAME_REGEX.test(this.stackName)) {\n      throw new Error(`Stack name must match the regular expression: ${VALID_STACK_NAME_REGEX.toString()}, got '${this.stackName}'`);\n    }\n\n    // the preferred behavior is to generate a unique id for this stack and use\n    // it as the artifact ID in the assembly. this allows multiple stacks to use\n    // the same name. however, this behavior is breaking for 1.x so it's only\n    // applied under a feature flag which is applied automatically for new\n    // projects created using `cdk init`.\n    //\n    // Also use the new behavior if we are using the new CI/CD-ready synthesizer; that way\n    // people only have to flip one flag.\n    const featureFlags = FeatureFlags.of(this);\n    const stackNameDupeContext = featureFlags.isEnabled(cxapi.ENABLE_STACK_NAME_DUPLICATES_CONTEXT);\n    const newStyleSynthesisContext = featureFlags.isEnabled(cxapi.NEW_STYLE_STACK_SYNTHESIS_CONTEXT);\n    this.artifactId = (stackNameDupeContext || newStyleSynthesisContext)\n      ? this.generateStackArtifactId()\n      : this.stackName;\n\n    this.templateFile = `${this.artifactId}.template.json`;\n\n    // Not for nested stacks\n    this._versionReportingEnabled = (props.analyticsReporting ?? this.node.tryGetContext(cxapi.ANALYTICS_REPORTING_ENABLED_CONTEXT))\n      && !this.nestedStackParent;\n\n    this.synthesizer = props.synthesizer ?? (newStyleSynthesisContext\n      ? new DefaultStackSynthesizer()\n      : new LegacyStackSynthesizer());\n    this.synthesizer.bind(this);\n  }\n\n  /**\n   * Resolve a tokenized value in the context of the current stack.\n   */\n  public resolve(obj: any): any {\n    return resolve(obj, {\n      scope: this,\n      prefix: [],\n      resolver: CLOUDFORMATION_TOKEN_RESOLVER,\n      preparing: false,\n    });\n  }\n\n  /**\n   * Convert an object, potentially containing tokens, to a JSON string\n   */\n  public toJsonString(obj: any, space?: number): string {\n    return CloudFormationLang.toJSON(obj, space).toString();\n  }\n\n  /**\n   * DEPRECATED\n   * @deprecated use `reportMissingContextKey()`\n   */\n  public reportMissingContext(report: cxapi.MissingContext) {\n    if (!Object.values(cxschema.ContextProvider).includes(report.provider as cxschema.ContextProvider)) {\n      throw new Error(`Unknown context provider requested in: ${JSON.stringify(report)}`);\n    }\n    this.reportMissingContextKey(report as cxschema.MissingContext);\n  }\n\n  /**\n   * Indicate that a context key was expected\n   *\n   * Contains instructions which will be emitted into the cloud assembly on how\n   * the key should be supplied.\n   *\n   * @param report The set of parameters needed to obtain the context\n   */\n  public reportMissingContextKey(report: cxschema.MissingContext) {\n    this._missingContext.push(report);\n  }\n\n  /**\n   * Rename a generated logical identities\n   *\n   * To modify the naming scheme strategy, extend the `Stack` class and\n   * override the `allocateLogicalId` method.\n   */\n  public renameLogicalId(oldId: string, newId: string) {\n    this._logicalIds.addRename(oldId, newId);\n  }\n\n  /**\n   * Allocates a stack-unique CloudFormation-compatible logical identity for a\n   * specific resource.\n   *\n   * This method is called when a `CfnElement` is created and used to render the\n   * initial logical identity of resources. Logical ID renames are applied at\n   * this stage.\n   *\n   * This method uses the protected method `allocateLogicalId` to render the\n   * logical ID for an element. To modify the naming scheme, extend the `Stack`\n   * class and override this method.\n   *\n   * @param element The CloudFormation element for which a logical identity is\n   * needed.\n   */\n  public getLogicalId(element: CfnElement): string {\n    const logicalId = this.allocateLogicalId(element);\n    return this._logicalIds.applyRename(logicalId);\n  }\n\n  /**\n   * Add a dependency between this stack and another stack.\n   *\n   * This can be used to define dependencies between any two stacks within an\n   * app, and also supports nested stacks.\n   */\n  public addDependency(target: Stack, reason?: string) {\n    addDependency(this, target, reason);\n  }\n\n  /**\n   * Return the stacks this stack depends on\n   */\n  public get dependencies(): Stack[] {\n    return Object.values(this._stackDependencies).map(x => x.stack);\n  }\n\n  /**\n   * The concrete CloudFormation physical stack name.\n   *\n   * This is either the name defined explicitly in the `stackName` prop or\n   * allocated based on the stack's location in the construct tree. Stacks that\n   * are directly defined under the app use their construct `id` as their stack\n   * name. Stacks that are defined deeper within the tree will use a hashed naming\n   * scheme based on the construct path to ensure uniqueness.\n   *\n   * If you wish to obtain the deploy-time AWS::StackName intrinsic,\n   * you can use `Aws.stackName` directly.\n   */\n  public get stackName(): string {\n    return this._stackName;\n  }\n\n  /**\n   * The partition in which this stack is defined\n   */\n  public get partition(): string {\n    // Always return a non-scoped partition intrinsic. These will usually\n    // be used to construct an ARN, but there are no cross-partition\n    // calls anyway.\n    return Aws.PARTITION;\n  }\n\n  /**\n   * The Amazon domain suffix for the region in which this stack is defined\n   */\n  public get urlSuffix(): string {\n    // Since URL Suffix always follows partition, it is unscoped like partition is.\n    return Aws.URL_SUFFIX;\n  }\n\n  /**\n   * The ID of the stack\n   *\n   * @example\n   * // After resolving, looks like\n   * 'arn:aws:cloudformation:us-west-2:123456789012:stack/teststack/51af3dc0-da77-11e4-872e-1234567db123'\n   */\n  public get stackId(): string {\n    return new ScopedAws(this).stackId;\n  }\n\n  /**\n   * Returns the list of notification Amazon Resource Names (ARNs) for the current stack.\n   */\n  public get notificationArns(): string[] {\n    return new ScopedAws(this).notificationArns;\n  }\n\n  /**\n   * Indicates if this is a nested stack, in which case `parentStack` will include a reference to it's parent.\n   */\n  public get nested(): boolean {\n    return this.nestedStackResource !== undefined;\n  }\n\n  /**\n   * Creates an ARN from components.\n   *\n   * If `partition`, `region` or `account` are not specified, the stack's\n   * partition, region and account will be used.\n   *\n   * If any component is the empty string, an empty string will be inserted\n   * into the generated ARN at the location that component corresponds to.\n   *\n   * The ARN will be formatted as follows:\n   *\n   *   arn:{partition}:{service}:{region}:{account}:{resource}{sep}}{resource-name}\n   *\n   * The required ARN pieces that are omitted will be taken from the stack that\n   * the 'scope' is attached to. If all ARN pieces are supplied, the supplied scope\n   * can be 'undefined'.\n   */\n  public formatArn(components: ArnComponents): string {\n    return Arn.format(components, this);\n  }\n\n  /**\n   * Given an ARN, parses it and returns components.\n   *\n   * IF THE ARN IS A CONCRETE STRING...\n   *\n   * ...it will be parsed and validated. The separator (`sep`) will be set to '/'\n   * if the 6th component includes a '/', in which case, `resource` will be set\n   * to the value before the '/' and `resourceName` will be the rest. In case\n   * there is no '/', `resource` will be set to the 6th components and\n   * `resourceName` will be set to the rest of the string.\n   *\n   * IF THE ARN IS A TOKEN...\n   *\n   * ...it cannot be validated, since we don't have the actual value yet at the\n   * time of this function call. You will have to supply `sepIfToken` and\n   * whether or not ARNs of the expected format usually have resource names\n   * in order to parse it properly. The resulting `ArnComponents` object will\n   * contain tokens for the subexpressions of the ARN, not string literals.\n   *\n   * If the resource name could possibly contain the separator char, the actual\n   * resource name cannot be properly parsed. This only occurs if the separator\n   * char is '/', and happens for example for S3 object ARNs, IAM Role ARNs,\n   * IAM OIDC Provider ARNs, etc. To properly extract the resource name from a\n   * Tokenized ARN, you must know the resource type and call\n   * `Arn.extractResourceName`.\n   *\n   * @param arn The ARN string to parse\n   * @param sepIfToken The separator used to separate resource from resourceName\n   * @param hasName Whether there is a name component in the ARN at all. For\n   * example, SNS Topics ARNs have the 'resource' component contain the topic\n   * name, and no 'resourceName' component.\n   *\n   * @returns an ArnComponents object which allows access to the various\n   * components of the ARN.\n   *\n   * @returns an ArnComponents object which allows access to the various\n   *      components of the ARN.\n   *\n   * @deprecated use splitArn instead\n   */\n  public parseArn(arn: string, sepIfToken: string = '/', hasName: boolean = true): ArnComponents {\n    return Arn.parse(arn, sepIfToken, hasName);\n  }\n\n  /**\n   * Splits the provided ARN into its components.\n   * Works both if 'arn' is a string like 'arn:aws:s3:::bucket',\n   * and a Token representing a dynamic CloudFormation expression\n   * (in which case the returned components will also be dynamic CloudFormation expressions,\n   * encoded as Tokens).\n   *\n   * @param arn the ARN to split into its components\n   * @param arnFormat the expected format of 'arn' - depends on what format the service 'arn' represents uses\n   */\n  public splitArn(arn: string, arnFormat: ArnFormat): ArnComponents {\n    return Arn.split(arn, arnFormat);\n  }\n\n  /**\n   * Returns the list of AZs that are available in the AWS environment\n   * (account/region) associated with this stack.\n   *\n   * If the stack is environment-agnostic (either account and/or region are\n   * tokens), this property will return an array with 2 tokens that will resolve\n   * at deploy-time to the first two availability zones returned from CloudFormation's\n   * `Fn::GetAZs` intrinsic function.\n   *\n   * If they are not available in the context, returns a set of dummy values and\n   * reports them as missing, and let the CLI resolve them by calling EC2\n   * `DescribeAvailabilityZones` on the target environment.\n   *\n   * To specify a different strategy for selecting availability zones override this method.\n   */\n  public get availabilityZones(): string[] {\n    // if account/region are tokens, we can't obtain AZs through the context\n    // provider, so we fallback to use Fn::GetAZs. the current lowest common\n    // denominator is 2 AZs across all AWS regions.\n    const agnostic = Token.isUnresolved(this.account) || Token.isUnresolved(this.region);\n    if (agnostic) {\n      return this.node.tryGetContext(cxapi.AVAILABILITY_ZONE_FALLBACK_CONTEXT_KEY) || [\n        Fn.select(0, Fn.getAzs()),\n        Fn.select(1, Fn.getAzs()),\n      ];\n    }\n\n    const value = ContextProvider.getValue(this, {\n      provider: cxschema.ContextProvider.AVAILABILITY_ZONE_PROVIDER,\n      dummyValue: ['dummy1a', 'dummy1b', 'dummy1c'],\n    }).value;\n\n    if (!Array.isArray(value)) {\n      throw new Error(`Provider ${cxschema.ContextProvider.AVAILABILITY_ZONE_PROVIDER} expects a list`);\n    }\n\n    return value;\n  }\n\n  /**\n   * Register a file asset on this Stack\n   *\n   * @deprecated Use `stack.synthesizer.addFileAsset()` if you are calling,\n   * and a different IStackSynthesizer class if you are implementing.\n   */\n  public addFileAsset(asset: FileAssetSource): FileAssetLocation {\n    return this.synthesizer.addFileAsset(asset);\n  }\n\n  /**\n   * Register a docker image asset on this Stack\n   *\n   * @deprecated Use `stack.synthesizer.addDockerImageAsset()` if you are calling,\n   * and a different `IStackSynthesizer` class if you are implementing.\n   */\n  public addDockerImageAsset(asset: DockerImageAssetSource): DockerImageAssetLocation {\n    return this.synthesizer.addDockerImageAsset(asset);\n  }\n\n  /**\n   * If this is a nested stack, returns it's parent stack.\n   */\n  public get nestedStackParent() {\n    return this.nestedStackResource && Stack.of(this.nestedStackResource);\n  }\n\n  /**\n   * Returns the parent of a nested stack.\n   *\n   * @deprecated use `nestedStackParent`\n   */\n  public get parentStack() {\n    return this.nestedStackParent;\n  }\n\n  /**\n   * Add a Transform to this stack. A Transform is a macro that AWS\n   * CloudFormation uses to process your template.\n   *\n   * Duplicate values are removed when stack is synthesized.\n   *\n   * @see https://docs.aws.amazon.com/AWSCloudFormation/latest/UserGuide/transform-section-structure.html\n   * @param transform The transform to add\n   *\n   * @example\n   * declare const stack: Stack;\n   *\n   * stack.addTransform('AWS::Serverless-2016-10-31')\n   */\n  public addTransform(transform: string) {\n    if (!this.templateOptions.transforms) {\n      this.templateOptions.transforms = [];\n    }\n    this.templateOptions.transforms.push(transform);\n  }\n\n  /**\n   * Called implicitly by the `addDependency` helper function in order to\n   * realize a dependency between two top-level stacks at the assembly level.\n   *\n   * Use `stack.addDependency` to define the dependency between any two stacks,\n   * and take into account nested stack relationships.\n   *\n   * @internal\n   */\n  public _addAssemblyDependency(target: Stack, reason?: string) {\n    // defensive: we should never get here for nested stacks\n    if (this.nested || target.nested) {\n      throw new Error('Cannot add assembly-level dependencies for nested stacks');\n    }\n\n    reason = reason || 'dependency added using stack.addDependency()';\n    const cycle = target.stackDependencyReasons(this);\n    if (cycle !== undefined) {\n      // eslint-disable-next-line max-len\n      throw new Error(`'${target.node.path}' depends on '${this.node.path}' (${cycle.join(', ')}). Adding this dependency (${reason}) would create a cyclic reference.`);\n    }\n\n    let dep = this._stackDependencies[Names.uniqueId(target)];\n    if (!dep) {\n      dep = this._stackDependencies[Names.uniqueId(target)] = {\n        stack: target,\n        reasons: [],\n      };\n    }\n\n    dep.reasons.push(reason);\n\n    if (process.env.CDK_DEBUG_DEPS) {\n      // eslint-disable-next-line no-console\n      console.error(`[CDK_DEBUG_DEPS] stack \"${this.node.path}\" depends on \"${target.node.path}\" because: ${reason}`);\n    }\n  }\n\n  /**\n   * Synthesizes the cloudformation template into a cloud assembly.\n   * @internal\n   */\n  public _synthesizeTemplate(session: ISynthesisSession, lookupRoleArn?: string): void {\n    // In principle, stack synthesis is delegated to the\n    // StackSynthesis object.\n    //\n    // However, some parts of synthesis currently use some private\n    // methods on Stack, and I don't really see the value in refactoring\n    // this right now, so some parts still happen here.\n    const builder = session.assembly;\n\n    const template = this._toCloudFormation();\n\n    // write the CloudFormation template as a JSON file\n    const outPath = path.join(builder.outdir, this.templateFile);\n\n    if (this.maxResources > 0) {\n      const resources = template.Resources || {};\n      const numberOfResources = Object.keys(resources).length;\n\n      if (numberOfResources > this.maxResources) {\n        const counts = Object.entries(count(Object.values(resources).map((r: any) => `${r?.Type}`))).map(([type, c]) => `${type} (${c})`).join(', ');\n        throw new Error(`Number of resources in stack '${this.node.path}': ${numberOfResources} is greater than allowed maximum of ${this.maxResources}: ${counts}`);\n      } else if (numberOfResources >= (this.maxResources * 0.8)) {\n        Annotations.of(this).addInfo(`Number of resources: ${numberOfResources} is approaching allowed maximum of ${this.maxResources}`);\n      }\n    }\n    fs.writeFileSync(outPath, JSON.stringify(template, undefined, 1));\n\n    for (const ctx of this._missingContext) {\n      if (lookupRoleArn != null) {\n        builder.addMissing({ ...ctx, props: { ...ctx.props, lookupRoleArn } });\n      } else {\n        builder.addMissing(ctx);\n      }\n    }\n  }\n\n  /**\n   * Look up a fact value for the given fact for the region of this stack\n   *\n   * Will return a definite value only if the region of the current stack is resolved.\n   * If not, a lookup map will be added to the stack and the lookup will be done at\n   * CDK deployment time.\n   *\n   * What regions will be included in the lookup map is controlled by the\n   * `@aws-cdk/core:target-partitions` context value: it must be set to a list\n   * of partitions, and only regions from the given partitions will be included.\n   * If no such context key is set, all regions will be included.\n   *\n   * This function is intended to be used by construct library authors. Application\n   * builders can rely on the abstractions offered by construct libraries and do\n   * not have to worry about regional facts.\n   *\n   * If `defaultValue` is not given, it is an error if the fact is unknown for\n   * the given region.\n   */\n  public regionalFact(factName: string, defaultValue?: string): string {\n    if (!Token.isUnresolved(this.region)) {\n      const ret = Fact.find(this.region, factName) ?? defaultValue;\n      if (ret === undefined) {\n        throw new Error(`region-info: don't know ${factName} for region ${this.region}. Use 'Fact.register' to provide this value.`);\n      }\n      return ret;\n    }\n\n    const partitions = Node.of(this).tryGetContext(cxapi.TARGET_PARTITIONS);\n    if (partitions !== undefined && !Array.isArray(partitions)) {\n      throw new Error(`Context value '${cxapi.TARGET_PARTITIONS}' should be a list of strings, got: ${JSON.stringify(cxapi.TARGET_PARTITIONS)}`);\n    }\n\n    const lookupMap = partitions ? RegionInfo.limitedRegionMap(factName, partitions) : RegionInfo.regionMap(factName);\n\n    return deployTimeLookup(this, factName, lookupMap, defaultValue);\n  }\n\n\n  /**\n   * Create a CloudFormation Export for a value\n   *\n   * Returns a string representing the corresponding `Fn.importValue()`\n   * expression for this Export. You can control the name for the export by\n   * passing the `name` option.\n   *\n   * If you don't supply a value for `name`, the value you're exporting must be\n   * a Resource attribute (for example: `bucket.bucketName`) and it will be\n   * given the same name as the automatic cross-stack reference that would be created\n   * if you used the attribute in another Stack.\n   *\n   * One of the uses for this method is to *remove* the relationship between\n   * two Stacks established by automatic cross-stack references. It will\n   * temporarily ensure that the CloudFormation Export still exists while you\n   * remove the reference from the consuming stack. After that, you can remove\n   * the resource and the manual export.\n   *\n   * ## Example\n   *\n   * Here is how the process works. Let's say there are two stacks,\n   * `producerStack` and `consumerStack`, and `producerStack` has a bucket\n   * called `bucket`, which is referenced by `consumerStack` (perhaps because\n   * an AWS Lambda Function writes into it, or something like that).\n   *\n   * It is not safe to remove `producerStack.bucket` because as the bucket is being\n   * deleted, `consumerStack` might still be using it.\n   *\n   * Instead, the process takes two deployments:\n   *\n   * ### Deployment 1: break the relationship\n   *\n   * - Make sure `consumerStack` no longer references `bucket.bucketName` (maybe the consumer\n   *   stack now uses its own bucket, or it writes to an AWS DynamoDB table, or maybe you just\n   *   remove the Lambda Function altogether).\n   * - In the `ProducerStack` class, call `this.exportValue(this.bucket.bucketName)`. This\n   *   will make sure the CloudFormation Export continues to exist while the relationship\n   *   between the two stacks is being broken.\n   * - Deploy (this will effectively only change the `consumerStack`, but it's safe to deploy both).\n   *\n   * ### Deployment 2: remove the bucket resource\n   *\n   * - You are now free to remove the `bucket` resource from `producerStack`.\n   * - Don't forget to remove the `exportValue()` call as well.\n   * - Deploy again (this time only the `producerStack` will be changed -- the bucket will be deleted).\n   */\n  public exportValue(exportedValue: any, options: ExportValueOptions = {}) {\n    if (options.name) {\n      new CfnOutput(this, `Export${options.name}`, {\n        value: exportedValue,\n        exportName: options.name,\n      });\n      return Fn.importValue(options.name);\n    }\n\n    const resolvable = Tokenization.reverse(exportedValue);\n    if (!resolvable || !Reference.isReference(resolvable)) {\n      throw new Error('exportValue: either supply \\'name\\' or make sure to export a resource attribute (like \\'bucket.bucketName\\')');\n    }\n\n    // if exportValue is being called manually (which is pre onPrepare) then the logicalId\n    // could potentially be changed by a call to overrideLogicalId. This would cause our Export/Import\n    // to have an incorrect id. For a better user experience, lock the logicalId and throw an error\n    // if the user tries to override the id _after_ calling exportValue\n    if (CfnElement.isCfnElement(resolvable.target)) {\n      resolvable.target._lockLogicalId();\n    }\n\n    // \"teleport\" the value here, in case it comes from a nested stack. This will also\n    // ensure the value is from our own scope.\n    const exportable = referenceNestedStackValueInParent(resolvable, this);\n\n    // Ensure a singleton \"Exports\" scoping Construct\n    // This mostly exists to trigger LogicalID munging, which would be\n    // disabled if we parented constructs directly under Stack.\n    // Also it nicely prevents likely construct name clashes\n    const exportsScope = getCreateExportsScope(this);\n\n    // Ensure a singleton CfnOutput for this value\n    const resolved = this.resolve(exportable);\n    const id = 'Output' + JSON.stringify(resolved);\n    const exportName = generateExportName(exportsScope, id);\n\n    if (Token.isUnresolved(exportName)) {\n      throw new Error(`unresolved token in generated export name: ${JSON.stringify(this.resolve(exportName))}`);\n    }\n\n    const output = exportsScope.node.tryFindChild(id) as CfnOutput;\n    if (!output) {\n      new CfnOutput(exportsScope, id, { value: Token.asString(exportable), exportName });\n    }\n\n    return Fn.importValue(exportName);\n  }\n\n  /**\n   * Returns the naming scheme used to allocate logical IDs. By default, uses\n   * the `HashedAddressingScheme` but this method can be overridden to customize\n   * this behavior.\n   *\n   * In order to make sure logical IDs are unique and stable, we hash the resource\n   * construct tree path (i.e. toplevel/secondlevel/.../myresource) and add it as\n   * a suffix to the path components joined without a separator (CloudFormation\n   * IDs only allow alphanumeric characters).\n   *\n   * The result will be:\n   *\n   *   <path.join('')><md5(path.join('/')>\n   *     \"human\"      \"hash\"\n   *\n   * If the \"human\" part of the ID exceeds 240 characters, we simply trim it so\n   * the total ID doesn't exceed CloudFormation's 255 character limit.\n   *\n   * We only take 8 characters from the md5 hash (0.000005 chance of collision).\n   *\n   * Special cases:\n   *\n   * - If the path only contains a single component (i.e. it's a top-level\n   *   resource), we won't add the hash to it. The hash is not needed for\n   *   disamiguation and also, it allows for a more straightforward migration an\n   *   existing CloudFormation template to a CDK stack without logical ID changes\n   *   (or renames).\n   * - For aesthetic reasons, if the last components of the path are the same\n   *   (i.e. `L1/L2/Pipeline/Pipeline`), they will be de-duplicated to make the\n   *   resulting human portion of the ID more pleasing: `L1L2Pipeline<HASH>`\n   *   instead of `L1L2PipelinePipeline<HASH>`\n   * - If a component is named \"Default\" it will be omitted from the path. This\n   *   allows refactoring higher level abstractions around constructs without affecting\n   *   the IDs of already deployed resources.\n   * - If a component is named \"Resource\" it will be omitted from the user-visible\n   *   path, but included in the hash. This reduces visual noise in the human readable\n   *   part of the identifier.\n   *\n   * @param cfnElement The element for which the logical ID is allocated.\n   */\n  protected allocateLogicalId(cfnElement: CfnElement): string {\n    const scopes = cfnElement.node.scopes;\n    const stackIndex = scopes.indexOf(cfnElement.stack);\n    const pathComponents = scopes.slice(stackIndex + 1).map(x => x.node.id);\n    return makeUniqueId(pathComponents);\n  }\n\n  /**\n   * Validate stack name\n   *\n   * CloudFormation stack names can include dashes in addition to the regular identifier\n   * character classes, and we don't allow one of the magic markers.\n   *\n   * @internal\n   */\n  protected _validateId(name: string) {\n    if (name && !VALID_STACK_NAME_REGEX.test(name)) {\n      throw new Error(`Stack name must match the regular expression: ${VALID_STACK_NAME_REGEX.toString()}, got '${name}'`);\n    }\n  }\n\n  /**\n   * Returns the CloudFormation template for this stack by traversing\n   * the tree and invoking _toCloudFormation() on all Entity objects.\n   *\n   * @internal\n   */\n  protected _toCloudFormation() {\n    let transform: string | string[] | undefined;\n\n    if (this.templateOptions.transform) {\n      // eslint-disable-next-line max-len\n      Annotations.of(this).addWarning('This stack is using the deprecated `templateOptions.transform` property. Consider switching to `addTransform()`.');\n      this.addTransform(this.templateOptions.transform);\n    }\n\n    if (this.templateOptions.transforms) {\n      if (this.templateOptions.transforms.length === 1) { // Extract single value\n        transform = this.templateOptions.transforms[0];\n      } else { // Remove duplicate values\n        transform = Array.from(new Set(this.templateOptions.transforms));\n      }\n    }\n\n    const template: any = {\n      Description: this.templateOptions.description,\n      Transform: transform,\n      AWSTemplateFormatVersion: this.templateOptions.templateFormatVersion,\n      Metadata: this.templateOptions.metadata,\n    };\n\n    const elements = cfnElements(this);\n    const fragments = elements.map(e => this.resolve(e._toCloudFormation()));\n\n    // merge in all CloudFormation fragments collected from the tree\n    for (const fragment of fragments) {\n      merge(template, fragment);\n    }\n\n    // resolve all tokens and remove all empties\n    const ret = this.resolve(template) || {};\n\n    this._logicalIds.assertAllRenamesApplied();\n\n    return ret;\n  }\n\n  /**\n   * Deprecated.\n   *\n   * @see https://github.com/aws/aws-cdk/pull/7187\n   * @returns reference itself without any change\n   * @deprecated cross reference handling has been moved to `App.prepare()`.\n   */\n  protected prepareCrossReference(_sourceStack: Stack, reference: Reference): IResolvable {\n    return reference;\n  }\n\n  /**\n   * Determine the various stack environment attributes.\n   *\n   */\n  private parseEnvironment(env: Environment = {}) {\n    // if an environment property is explicitly specified when the stack is\n    // created, it will be used. if not, use tokens for account and region.\n    //\n    // (They do not need to be anchored to any construct like resource attributes\n    // are, because we'll never Export/Fn::ImportValue them -- the only situation\n    // in which Export/Fn::ImportValue would work is if the value are the same\n    // between producer and consumer anyway, so we can just assume that they are).\n    const containingAssembly = Stage.of(this);\n    const account = env.account ?? containingAssembly?.account ?? Aws.ACCOUNT_ID;\n    const region = env.region ?? containingAssembly?.region ?? Aws.REGION;\n\n    // this is the \"aws://\" env specification that will be written to the cloud assembly\n    // manifest. it will use \"unknown-account\" and \"unknown-region\" to indicate\n    // environment-agnosticness.\n    const envAccount = !Token.isUnresolved(account) ? account : cxapi.UNKNOWN_ACCOUNT;\n    const envRegion = !Token.isUnresolved(region) ? region : cxapi.UNKNOWN_REGION;\n\n    return {\n      account,\n      region,\n      environment: cxapi.EnvironmentUtils.format(envAccount, envRegion),\n    };\n  }\n\n  /**\n   * Maximum number of resources in the stack\n   *\n   * Set to 0 to mean \"unlimited\".\n   */\n  private get maxResources(): number {\n    const contextLimit = this.node.tryGetContext(STACK_RESOURCE_LIMIT_CONTEXT);\n    return contextLimit !== undefined ? parseInt(contextLimit, 10) : MAX_RESOURCES;\n  }\n\n  /**\n   * Check whether this stack has a (transitive) dependency on another stack\n   *\n   * Returns the list of reasons on the dependency path, or undefined\n   * if there is no dependency.\n   */\n  private stackDependencyReasons(other: Stack): string[] | undefined {\n    if (this === other) { return []; }\n    for (const dep of Object.values(this._stackDependencies)) {\n      const ret = dep.stack.stackDependencyReasons(other);\n      if (ret !== undefined) {\n        return [...dep.reasons, ...ret];\n      }\n    }\n    return undefined;\n  }\n\n  /**\n   * Calculate the stack name based on the construct path\n   *\n   * The stack name is the name under which we'll deploy the stack,\n   * and incorporates containing Stage names by default.\n   *\n   * Generally this looks a lot like how logical IDs are calculated.\n   * The stack name is calculated based on the construct root path,\n   * as follows:\n   *\n   * - Path is calculated with respect to containing App or Stage (if any)\n   * - If the path is one component long just use that component, otherwise\n   *   combine them with a hash.\n   *\n   * Since the hash is quite ugly and we'd like to avoid it if possible -- but\n   * we can't anymore in the general case since it has been written into legacy\n   * stacks. The introduction of Stages makes it possible to make this nicer however.\n   * When a Stack is nested inside a Stage, we use the path components below the\n   * Stage, and prefix the path components of the Stage before it.\n   */\n  private generateStackName() {\n    const assembly = Stage.of(this);\n    const prefix = (assembly && assembly.stageName) ? `${assembly.stageName}-` : '';\n    return `${prefix}${this.generateStackId(assembly)}`;\n  }\n\n  /**\n   * The artifact ID for this stack\n   *\n   * Stack artifact ID is unique within the App's Cloud Assembly.\n   */\n  private generateStackArtifactId() {\n    return this.generateStackId(this.node.root);\n  }\n\n  /**\n   * Generate an ID with respect to the given container construct.\n   */\n  private generateStackId(container: IConstruct | undefined) {\n    const rootPath = rootPathTo(this, container);\n    const ids = rootPath.map(c => Node.of(c).id);\n\n    // In unit tests our Stack (which is the only component) may not have an\n    // id, so in that case just pretend it's \"Stack\".\n    if (ids.length === 1 && !ids[0]) {\n      throw new Error('unexpected: stack id must always be defined');\n    }\n\n    return makeStackName(ids);\n  }\n\n  /**\n   * Indicates whether the stack requires bundling or not\n   */\n  public get bundlingRequired() {\n    const bundlingStacks: string[] = this.node.tryGetContext(cxapi.BUNDLING_STACKS) ?? ['*'];\n\n    // bundlingStacks is of the form `Stage/Stack`, convert it to `Stage-Stack` before comparing to stack name\n    return bundlingStacks.some(pattern => minimatch(\n      this.stackName,\n      pattern.replace('/', '-'),\n    ));\n  }\n}\n\nfunction merge(template: any, fragment: any): void {\n  for (const section of Object.keys(fragment)) {\n    const src = fragment[section];\n\n    // create top-level section if it doesn't exist\n    const dest = template[section];\n    if (!dest) {\n      template[section] = src;\n    } else {\n      template[section] = mergeSection(section, dest, src);\n    }\n  }\n}\n\nfunction mergeSection(section: string, val1: any, val2: any): any {\n  switch (section) {\n    case 'Description':\n      return `${val1}\\n${val2}`;\n    case 'AWSTemplateFormatVersion':\n      if (val1 != null && val2 != null && val1 !== val2) {\n        throw new Error(`Conflicting CloudFormation template versions provided: '${val1}' and '${val2}`);\n      }\n      return val1 ?? val2;\n    case 'Transform':\n      return mergeSets(val1, val2);\n    default:\n      return mergeObjectsWithoutDuplicates(section, val1, val2);\n  }\n}\n\nfunction mergeSets(val1: any, val2: any): any {\n  const array1 = val1 == null ? [] : (Array.isArray(val1) ? val1 : [val1]);\n  const array2 = val2 == null ? [] : (Array.isArray(val2) ? val2 : [val2]);\n  for (const value of array2) {\n    if (!array1.includes(value)) {\n      array1.push(value);\n    }\n  }\n  return array1.length === 1 ? array1[0] : array1;\n}\n\nfunction mergeObjectsWithoutDuplicates(section: string, dest: any, src: any): any {\n  if (typeof dest !== 'object') {\n    throw new Error(`Expecting ${JSON.stringify(dest)} to be an object`);\n  }\n  if (typeof src !== 'object') {\n    throw new Error(`Expecting ${JSON.stringify(src)} to be an object`);\n  }\n\n  // add all entities from source section to destination section\n  for (const id of Object.keys(src)) {\n    if (id in dest) {\n      throw new Error(`section '${section}' already contains '${id}'`);\n    }\n    dest[id] = src[id];\n  }\n\n  return dest;\n}\n\n/**\n * CloudFormation template options for a stack.\n */\nexport interface ITemplateOptions {\n  /**\n   * Gets or sets the description of this stack.\n   * If provided, it will be included in the CloudFormation template's \"Description\" attribute.\n   */\n  description?: string;\n\n  /**\n   * Gets or sets the AWSTemplateFormatVersion field of the CloudFormation template.\n   */\n  templateFormatVersion?: string;\n\n  /**\n   * Gets or sets the top-level template transform for this stack (e.g. \"AWS::Serverless-2016-10-31\").\n   *\n   * @deprecated use `transforms` instead.\n   */\n  transform?: string;\n\n  /**\n   * Gets or sets the top-level template transform(s) for this stack (e.g. `[\"AWS::Serverless-2016-10-31\"]`).\n   */\n  transforms?: string[];\n\n  /**\n   * Metadata associated with the CloudFormation template.\n   */\n  metadata?: { [key: string]: any };\n}\n\n/**\n * Collect all CfnElements from a Stack.\n *\n * @param node Root node to collect all CfnElements from\n * @param into Array to append CfnElements to\n * @returns The same array as is being collected into\n */\nfunction cfnElements(node: IConstruct, into: CfnElement[] = []): CfnElement[] {\n  if (CfnElement.isCfnElement(node)) {\n    into.push(node);\n  }\n\n  for (const child of Node.of(node).children) {\n    // Don't recurse into a substack\n    if (Stack.isStack(child)) { continue; }\n\n    cfnElements(child, into);\n  }\n\n  return into;\n}\n\n/**\n * Return the construct root path of the given construct relative to the given ancestor\n *\n * If no ancestor is given or the ancestor is not found, return the entire root path.\n */\nexport function rootPathTo(construct: IConstruct, ancestor?: IConstruct): IConstruct[] {\n  const scopes = Node.of(construct).scopes;\n  for (let i = scopes.length - 2; i >= 0; i--) {\n    if (scopes[i] === ancestor) {\n      return scopes.slice(i + 1);\n    }\n  }\n  return scopes;\n}\n\n/**\n * makeUniqueId, specialized for Stack names\n *\n * Stack names may contain '-', so we allow that character if the stack name\n * has only one component. Otherwise we fall back to the regular \"makeUniqueId\"\n * behavior.\n */\nfunction makeStackName(components: string[]) {\n  if (components.length === 1) { return components[0]; }\n  return makeUniqueId(components);\n}\n\nfunction getCreateExportsScope(stack: Stack) {\n  const exportsName = 'Exports';\n  let stackExports = stack.node.tryFindChild(exportsName) as CoreConstruct;\n  if (stackExports === undefined) {\n    stackExports = new CoreConstruct(stack, exportsName);\n  }\n\n  return stackExports;\n}\n\nfunction generateExportName(stackExports: CoreConstruct, id: string) {\n  const stackRelativeExports = FeatureFlags.of(stackExports).isEnabled(cxapi.STACK_RELATIVE_EXPORTS_CONTEXT);\n  const stack = Stack.of(stackExports);\n\n  const components = [\n    ...stackExports.node.scopes\n      .slice(stackRelativeExports ? stack.node.scopes.length : 2)\n      .map(c => c.node.id),\n    id,\n  ];\n  const prefix = stack.stackName ? stack.stackName + ':' : '';\n  const localPart = makeUniqueId(components);\n  const maxLength = 255;\n  return prefix + localPart.slice(Math.max(0, localPart.length - maxLength + prefix.length));\n}\n\ninterface StackDependency {\n  stack: Stack;\n  reasons: string[];\n}\n\n/**\n * Options for the `stack.exportValue()` method\n */\nexport interface ExportValueOptions {\n  /**\n   * The name of the export to create\n   *\n   * @default - A name is automatically chosen\n   */\n  readonly name?: string;\n}\n\nfunction count(xs: string[]): Record<string, number> {\n  const ret: Record<string, number> = {};\n  for (const x of xs) {\n    if (x in ret) {\n      ret[x] += 1;\n    } else {\n      ret[x] = 1;\n    }\n  }\n  return ret;\n}\n\n// These imports have to be at the end to prevent circular imports\nimport { CfnOutput } from './cfn-output';\nimport { addDependency } from './deps';\nimport { FileSystem } from './fs';\nimport { Names } from './names';\nimport { Reference } from './reference';\nimport { IResolvable } from './resolvable';\nimport { DefaultStackSynthesizer, IStackSynthesizer, LegacyStackSynthesizer } from './stack-synthesizers';\nimport { Stage } from './stage';\nimport { ITaggable, TagManager } from './tag-manager';\nimport { Token, Tokenization } from './token';\nimport { referenceNestedStackValueInParent } from './private/refs';\nimport { Fact, RegionInfo } from '@aws-cdk/region-info';\nimport { deployTimeLookup } from './private/region-lookup';\n\n"]} |
\ | No newline at end of file |