UNPKG

147 kBJavaScriptView Raw
1"use strict";
2var _a;
3Object.defineProperty(exports, "__esModule", { value: true });
4exports.rootPathTo = exports.Stack = exports.STACK_RESOURCE_LIMIT_CONTEXT = void 0;
5const jsiiDeprecationWarnings = require("../.warnings.jsii.js");
6const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
7const fs = require("fs");
8const path = require("path");
9const cxschema = require("@aws-cdk/cloud-assembly-schema");
10const cxapi = require("@aws-cdk/cx-api");
11const constructs_1 = require("constructs");
12const minimatch = require("minimatch");
13const annotations_1 = require("./annotations");
14const app_1 = require("./app");
15const arn_1 = require("./arn");
16const cfn_element_1 = require("./cfn-element");
17const cfn_fn_1 = require("./cfn-fn");
18const cfn_pseudo_1 = require("./cfn-pseudo");
19const cfn_resource_1 = require("./cfn-resource");
20const context_provider_1 = require("./context-provider");
21const feature_flags_1 = require("./feature-flags");
22const cloudformation_lang_1 = require("./private/cloudformation-lang");
23const logical_id_1 = require("./private/logical-id");
24const resolve_1 = require("./private/resolve");
25const 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
28const construct_compat_1 = require("./construct-compat");
29const STACK_SYMBOL = Symbol.for('@aws-cdk/core.Stack');
30const MY_STACK_CACHE = Symbol.for('@aws-cdk/core.Stack.myStack');
31exports.STACK_RESOURCE_LIMIT_CONTEXT = '@aws-cdk/core:stackResourceLimit';
32const VALID_STACK_NAME_REGEX = /^[A-Za-z][A-Za-z0-9-]*$/;
33const MAX_RESOURCES = 500;
34/**
35 * A root construct which represents a single CloudFormation stack.
36 */
37class 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}
970exports.Stack = Stack;
971_a = JSII_RTTI_SYMBOL_1;
972Stack[_a] = { fqn: "@aws-cdk/core.Stack", version: "1.204.0" };
973function 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}
986function 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}
1001function 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}
1011function 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 */
1034function 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 */
1052function 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}
1061exports.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 */
1069function makeStackName(components) {
1070 if (components.length === 1) {
1071 return components[0];
1072 }
1073 return uniqueid_1.makeUniqueId(components);
1074}
1075function 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}
1083function 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}
1097function 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
1110const cfn_output_1 = require("./cfn-output");
1111const deps_1 = require("./deps");
1112const fs_1 = require("./fs");
1113const names_1 = require("./names");
1114const reference_1 = require("./reference");
1115const stack_synthesizers_1 = require("./stack-synthesizers");
1116const stage_1 = require("./stage");
1117const tag_manager_1 = require("./tag-manager");
1118const token_1 = require("./token");
1119const refs_1 = require("./private/refs");
1120const region_info_1 = require("@aws-cdk/region-info");
1121const 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