1 | ;
|
2 | var _a;
|
3 | Object.defineProperty(exports, "__esModule", { value: true });
|
4 | exports.Resource = void 0;
|
5 | const jsiiDeprecationWarnings = require("../.warnings.jsii.js");
|
6 | const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
7 | const arn_1 = require("./arn");
|
8 | const cfn_resource_1 = require("./cfn-resource");
|
9 | const construct_compat_1 = require("./construct-compat");
|
10 | const lazy_1 = require("./lazy");
|
11 | const physical_name_generator_1 = require("./private/physical-name-generator");
|
12 | const reference_1 = require("./reference");
|
13 | const stack_1 = require("./stack");
|
14 | const token_1 = require("./token");
|
15 | const RESOURCE_SYMBOL = Symbol.for('@aws-cdk/core.Resource');
|
16 | /**
|
17 | * A construct which represents an AWS resource.
|
18 | */
|
19 | class Resource extends construct_compat_1.Construct {
|
20 | constructor(scope, id, props = {}) {
|
21 | super(scope, id);
|
22 | try {
|
23 | jsiiDeprecationWarnings._aws_cdk_core_ResourceProps(props);
|
24 | }
|
25 | catch (error) {
|
26 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
27 | Error.captureStackTrace(error, Resource);
|
28 | }
|
29 | throw error;
|
30 | }
|
31 | if ((props.account !== undefined || props.region !== undefined) && props.environmentFromArn !== undefined) {
|
32 | throw new Error(`Supply at most one of 'account'/'region' (${props.account}/${props.region}) and 'environmentFromArn' (${props.environmentFromArn})`);
|
33 | }
|
34 | Object.defineProperty(this, RESOURCE_SYMBOL, { value: true });
|
35 | this.stack = stack_1.Stack.of(this);
|
36 | const parsedArn = props.environmentFromArn ?
|
37 | // Since we only want the region and account, NO_RESOURE_NAME is good enough
|
38 | this.stack.splitArn(props.environmentFromArn, arn_1.ArnFormat.NO_RESOURCE_NAME)
|
39 | : undefined;
|
40 | this.env = {
|
41 | account: props.account ?? parsedArn?.account ?? this.stack.account,
|
42 | region: props.region ?? parsedArn?.region ?? this.stack.region,
|
43 | };
|
44 | let physicalName = props.physicalName;
|
45 | if (props.physicalName && physical_name_generator_1.isGeneratedWhenNeededMarker(props.physicalName)) {
|
46 | // auto-generate only if cross-env is required
|
47 | this._physicalName = undefined;
|
48 | this._allowCrossEnvironment = true;
|
49 | physicalName = lazy_1.Lazy.string({ produce: () => this._physicalName });
|
50 | }
|
51 | else if (props.physicalName && !token_1.Token.isUnresolved(props.physicalName)) {
|
52 | // concrete value specified by the user
|
53 | this._physicalName = props.physicalName;
|
54 | this._allowCrossEnvironment = true;
|
55 | }
|
56 | else {
|
57 | // either undefined (deploy-time) or has tokens, which means we can't use for cross-env
|
58 | this._physicalName = props.physicalName;
|
59 | this._allowCrossEnvironment = false;
|
60 | }
|
61 | if (physicalName === undefined) {
|
62 | physicalName = token_1.Token.asString(undefined);
|
63 | }
|
64 | this.physicalName = physicalName;
|
65 | }
|
66 | /**
|
67 | * Check whether the given construct is a Resource
|
68 | */
|
69 | static isResource(construct) {
|
70 | try {
|
71 | jsiiDeprecationWarnings._aws_cdk_core_IConstruct(construct);
|
72 | }
|
73 | catch (error) {
|
74 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
75 | Error.captureStackTrace(error, this.isResource);
|
76 | }
|
77 | throw error;
|
78 | }
|
79 | return construct !== null && typeof (construct) === 'object' && RESOURCE_SYMBOL in construct;
|
80 | }
|
81 | /**
|
82 | * Called when this resource is referenced across environments
|
83 | * (account/region) to order to request that a physical name will be generated
|
84 | * for this resource during synthesis, so the resource can be referenced
|
85 | * through it's absolute name/arn.
|
86 | *
|
87 | * @internal
|
88 | */
|
89 | _enableCrossEnvironment() {
|
90 | if (!this._allowCrossEnvironment) {
|
91 | // error out - a deploy-time name cannot be used across environments
|
92 | throw new Error(`Cannot use resource '${this.node.path}' in a cross-environment fashion, ` +
|
93 | "the resource's physical name must be explicit set or use `PhysicalName.GENERATE_IF_NEEDED`");
|
94 | }
|
95 | if (!this._physicalName) {
|
96 | this._physicalName = this.generatePhysicalName();
|
97 | }
|
98 | }
|
99 | /**
|
100 | * Apply the given removal policy to this resource
|
101 | *
|
102 | * The Removal Policy controls what happens to this resource when it stops
|
103 | * being managed by CloudFormation, either because you've removed it from the
|
104 | * CDK application or because you've made a change that requires the resource
|
105 | * to be replaced.
|
106 | *
|
107 | * The resource can be deleted (`RemovalPolicy.DESTROY`), or left in your AWS
|
108 | * account for data recovery and cleanup later (`RemovalPolicy.RETAIN`).
|
109 | */
|
110 | applyRemovalPolicy(policy) {
|
111 | try {
|
112 | jsiiDeprecationWarnings._aws_cdk_core_RemovalPolicy(policy);
|
113 | }
|
114 | catch (error) {
|
115 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
116 | Error.captureStackTrace(error, this.applyRemovalPolicy);
|
117 | }
|
118 | throw error;
|
119 | }
|
120 | const child = this.node.defaultChild;
|
121 | if (!child || !cfn_resource_1.CfnResource.isCfnResource(child)) {
|
122 | throw new Error('Cannot apply RemovalPolicy: no child or not a CfnResource. Apply the removal policy on the CfnResource directly.');
|
123 | }
|
124 | child.applyRemovalPolicy(policy);
|
125 | }
|
126 | generatePhysicalName() {
|
127 | return physical_name_generator_1.generatePhysicalName(this);
|
128 | }
|
129 | /**
|
130 | * Returns an environment-sensitive token that should be used for the
|
131 | * resource's "name" attribute (e.g. `bucket.bucketName`).
|
132 | *
|
133 | * Normally, this token will resolve to `nameAttr`, but if the resource is
|
134 | * referenced across environments, it will be resolved to `this.physicalName`,
|
135 | * which will be a concrete name.
|
136 | *
|
137 | * @param nameAttr The CFN attribute which resolves to the resource's name.
|
138 | * Commonly this is the resource's `ref`.
|
139 | */
|
140 | getResourceNameAttribute(nameAttr) {
|
141 | return mimicReference(nameAttr, {
|
142 | produce: (context) => {
|
143 | const consumingStack = stack_1.Stack.of(context.scope);
|
144 | if (this.stack.environment !== consumingStack.environment) {
|
145 | this._enableCrossEnvironment();
|
146 | return this.physicalName;
|
147 | }
|
148 | else {
|
149 | return nameAttr;
|
150 | }
|
151 | },
|
152 | });
|
153 | }
|
154 | /**
|
155 | * Returns an environment-sensitive token that should be used for the
|
156 | * resource's "ARN" attribute (e.g. `bucket.bucketArn`).
|
157 | *
|
158 | * Normally, this token will resolve to `arnAttr`, but if the resource is
|
159 | * referenced across environments, `arnComponents` will be used to synthesize
|
160 | * a concrete ARN with the resource's physical name. Make sure to reference
|
161 | * `this.physicalName` in `arnComponents`.
|
162 | *
|
163 | * @param arnAttr The CFN attribute which resolves to the ARN of the resource.
|
164 | * Commonly it will be called "Arn" (e.g. `resource.attrArn`), but sometimes
|
165 | * it's the CFN resource's `ref`.
|
166 | * @param arnComponents The format of the ARN of this resource. You must
|
167 | * reference `this.physicalName` somewhere within the ARN in order for
|
168 | * cross-environment references to work.
|
169 | *
|
170 | */
|
171 | getResourceArnAttribute(arnAttr, arnComponents) {
|
172 | try {
|
173 | jsiiDeprecationWarnings._aws_cdk_core_ArnComponents(arnComponents);
|
174 | }
|
175 | catch (error) {
|
176 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
177 | Error.captureStackTrace(error, this.getResourceArnAttribute);
|
178 | }
|
179 | throw error;
|
180 | }
|
181 | return mimicReference(arnAttr, {
|
182 | produce: (context) => {
|
183 | const consumingStack = stack_1.Stack.of(context.scope);
|
184 | if (this.stack.environment !== consumingStack.environment) {
|
185 | this._enableCrossEnvironment();
|
186 | return this.stack.formatArn(arnComponents);
|
187 | }
|
188 | else {
|
189 | return arnAttr;
|
190 | }
|
191 | },
|
192 | });
|
193 | }
|
194 | }
|
195 | exports.Resource = Resource;
|
196 | _a = JSII_RTTI_SYMBOL_1;
|
197 | Resource[_a] = { fqn: "@aws-cdk/core.Resource", version: "1.204.0" };
|
198 | /**
|
199 | * Produce a Lazy that is also a Reference (if the base value is a Reference).
|
200 | *
|
201 | * If the given value is a Reference (or resolves to a Reference), return a new
|
202 | * Reference that mimics the same target and display name, but resolves using
|
203 | * the logic of the passed lazy.
|
204 | *
|
205 | * If the given value is NOT a Reference, just return a simple Lazy.
|
206 | */
|
207 | function mimicReference(refSource, producer) {
|
208 | const reference = token_1.Tokenization.reverse(refSource, {
|
209 | // If this is an ARN concatenation, just fail to extract a reference.
|
210 | failConcat: false,
|
211 | });
|
212 | if (!reference_1.Reference.isReference(reference)) {
|
213 | return lazy_1.Lazy.uncachedString(producer);
|
214 | }
|
215 | return token_1.Token.asString(new class extends reference_1.Reference {
|
216 | resolve(context) {
|
217 | return producer.produce(context);
|
218 | }
|
219 | }(reference, reference.target, reference.displayName));
|
220 | }
|
221 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"resource.js","sourceRoot":"","sources":["resource.ts"],"names":[],"mappings":"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","sourcesContent":["import { ArnComponents, ArnFormat } from './arn';\nimport { CfnResource } from './cfn-resource';\nimport { IConstruct, Construct as CoreConstruct } from './construct-compat';\nimport { IStringProducer, Lazy } from './lazy';\nimport { generatePhysicalName, isGeneratedWhenNeededMarker } from './private/physical-name-generator';\nimport { Reference } from './reference';\nimport { RemovalPolicy } from './removal-policy';\nimport { IResolveContext } from './resolvable';\nimport { Stack } from './stack';\nimport { Token, Tokenization } from './token';\n\n// v2 - leave this as a separate section so it reduces merge conflicts when compat is removed\n// eslint-disable-next-line import/order\nimport { Construct } from 'constructs';\n\nconst RESOURCE_SYMBOL = Symbol.for('@aws-cdk/core.Resource');\n\n/**\n * Represents the environment a given resource lives in.\n * Used as the return value for the {@link IResource.env} property.\n */\nexport interface ResourceEnvironment {\n  /**\n   * The AWS account ID that this resource belongs to.\n   * Since this can be a Token\n   * (for example, when the account is CloudFormation's AWS::AccountId intrinsic),\n   * make sure to use Token.compareStrings()\n   * instead of just comparing the values for equality.\n   */\n  readonly account: string;\n\n  /**\n   * The AWS region that this resource belongs to.\n   * Since this can be a Token\n   * (for example, when the region is CloudFormation's AWS::Region intrinsic),\n   * make sure to use Token.compareStrings()\n   * instead of just comparing the values for equality.\n   */\n  readonly region: string;\n}\n\n/**\n * Interface for the Resource construct.\n */\nexport interface IResource extends IConstruct {\n  /**\n   * The stack in which this resource is defined.\n   */\n  readonly stack: Stack;\n\n  /**\n   * The environment this resource belongs to.\n   * For resources that are created and managed by the CDK\n   * (generally, those created by creating new class instances like Role, Bucket, etc.),\n   * this is always the same as the environment of the stack they belong to;\n   * however, for imported resources\n   * (those obtained from static methods like fromRoleArn, fromBucketName, etc.),\n   * that might be different than the stack they were imported into.\n   */\n  readonly env: ResourceEnvironment;\n\n  /**\n   * Apply the given removal policy to this resource\n   *\n   * The Removal Policy controls what happens to this resource when it stops\n   * being managed by CloudFormation, either because you've removed it from the\n   * CDK application or because you've made a change that requires the resource\n   * to be replaced.\n   *\n   * The resource can be deleted (`RemovalPolicy.DESTROY`), or left in your AWS\n   * account for data recovery and cleanup later (`RemovalPolicy.RETAIN`).\n   */\n  applyRemovalPolicy(policy: RemovalPolicy): void;\n}\n\n/**\n * Construction properties for {@link Resource}.\n */\nexport interface ResourceProps {\n  /**\n   * The value passed in by users to the physical name prop of the resource.\n   *\n   * - `undefined` implies that a physical name will be allocated by\n   *   CloudFormation during deployment.\n   * - a concrete value implies a specific physical name\n   * - `PhysicalName.GENERATE_IF_NEEDED` is a marker that indicates that a physical will only be generated\n   *   by the CDK if it is needed for cross-environment references. Otherwise, it will be allocated by CloudFormation.\n   *\n   * @default - The physical name will be allocated by CloudFormation at deployment time\n   */\n  readonly physicalName?: string;\n\n  /**\n   * The AWS account ID this resource belongs to.\n   *\n   * @default - the resource is in the same account as the stack it belongs to\n   */\n  readonly account?: string;\n\n  /**\n   * The AWS region this resource belongs to.\n   *\n   * @default - the resource is in the same region as the stack it belongs to\n   */\n  readonly region?: string;\n\n  /**\n   * ARN to deduce region and account from\n   *\n   * The ARN is parsed and the account and region are taken from the ARN.\n   * This should be used for imported resources.\n   *\n   * Cannot be supplied together with either `account` or `region`.\n   *\n   * @default - take environment from `account`, `region` parameters, or use Stack environment.\n   */\n  readonly environmentFromArn?: string;\n}\n\n/**\n * A construct which represents an AWS resource.\n */\nexport abstract class Resource extends CoreConstruct implements IResource {\n  /**\n   * Check whether the given construct is a Resource\n   */\n  public static isResource(construct: IConstruct): construct is CfnResource {\n    return construct !== null && typeof(construct) === 'object' && RESOURCE_SYMBOL in construct;\n  }\n\n  public readonly stack: Stack;\n  public readonly env: ResourceEnvironment;\n\n  /**\n   * Returns a string-encoded token that resolves to the physical name that\n   * should be passed to the CloudFormation resource.\n   *\n   * This value will resolve to one of the following:\n   * - a concrete value (e.g. `\"my-awesome-bucket\"`)\n   * - `undefined`, when a name should be generated by CloudFormation\n   * - a concrete name generated automatically during synthesis, in\n   *   cross-environment scenarios.\n   *\n   */\n  protected readonly physicalName: string;\n\n  private _physicalName: string | undefined;\n  private readonly _allowCrossEnvironment: boolean;\n\n  constructor(scope: Construct, id: string, props: ResourceProps = {}) {\n    super(scope, id);\n\n    if ((props.account !== undefined || props.region !== undefined) && props.environmentFromArn !== undefined) {\n      throw new Error(`Supply at most one of 'account'/'region' (${props.account}/${props.region}) and 'environmentFromArn' (${props.environmentFromArn})`);\n    }\n\n    Object.defineProperty(this, RESOURCE_SYMBOL, { value: true });\n\n    this.stack = Stack.of(this);\n\n    const parsedArn = props.environmentFromArn ?\n      // Since we only want the region and account, NO_RESOURE_NAME is good enough\n      this.stack.splitArn(props.environmentFromArn, ArnFormat.NO_RESOURCE_NAME)\n      : undefined;\n    this.env = {\n      account: props.account ?? parsedArn?.account ?? this.stack.account,\n      region: props.region ?? parsedArn?.region ?? this.stack.region,\n    };\n\n    let physicalName = props.physicalName;\n\n    if (props.physicalName && isGeneratedWhenNeededMarker(props.physicalName)) {\n      // auto-generate only if cross-env is required\n      this._physicalName = undefined;\n      this._allowCrossEnvironment = true;\n      physicalName = Lazy.string({ produce: () => this._physicalName });\n    } else if (props.physicalName && !Token.isUnresolved(props.physicalName)) {\n      // concrete value specified by the user\n      this._physicalName = props.physicalName;\n      this._allowCrossEnvironment = true;\n    } else {\n      // either undefined (deploy-time) or has tokens, which means we can't use for cross-env\n      this._physicalName = props.physicalName;\n      this._allowCrossEnvironment = false;\n    }\n\n    if (physicalName === undefined) {\n      physicalName = Token.asString(undefined);\n    }\n\n    this.physicalName = physicalName;\n  }\n\n  /**\n   * Called when this resource is referenced across environments\n   * (account/region) to order to request that a physical name will be generated\n   * for this resource during synthesis, so the resource can be referenced\n   * through it's absolute name/arn.\n   *\n   * @internal\n   */\n  public _enableCrossEnvironment(): void {\n    if (!this._allowCrossEnvironment) {\n      // error out - a deploy-time name cannot be used across environments\n      throw new Error(`Cannot use resource '${this.node.path}' in a cross-environment fashion, ` +\n        \"the resource's physical name must be explicit set or use `PhysicalName.GENERATE_IF_NEEDED`\");\n    }\n\n    if (!this._physicalName) {\n      this._physicalName = this.generatePhysicalName();\n    }\n  }\n\n  /**\n   * Apply the given removal policy to this resource\n   *\n   * The Removal Policy controls what happens to this resource when it stops\n   * being managed by CloudFormation, either because you've removed it from the\n   * CDK application or because you've made a change that requires the resource\n   * to be replaced.\n   *\n   * The resource can be deleted (`RemovalPolicy.DESTROY`), or left in your AWS\n   * account for data recovery and cleanup later (`RemovalPolicy.RETAIN`).\n   */\n  public applyRemovalPolicy(policy: RemovalPolicy) {\n    const child = this.node.defaultChild;\n    if (!child || !CfnResource.isCfnResource(child)) {\n      throw new Error('Cannot apply RemovalPolicy: no child or not a CfnResource. Apply the removal policy on the CfnResource directly.');\n    }\n    child.applyRemovalPolicy(policy);\n  }\n\n  protected generatePhysicalName(): string {\n    return generatePhysicalName(this);\n  }\n\n  /**\n   * Returns an environment-sensitive token that should be used for the\n   * resource's \"name\" attribute (e.g. `bucket.bucketName`).\n   *\n   * Normally, this token will resolve to `nameAttr`, but if the resource is\n   * referenced across environments, it will be resolved to `this.physicalName`,\n   * which will be a concrete name.\n   *\n   * @param nameAttr The CFN attribute which resolves to the resource's name.\n   * Commonly this is the resource's `ref`.\n   */\n  protected getResourceNameAttribute(nameAttr: string) {\n    return mimicReference(nameAttr, {\n      produce: (context: IResolveContext) => {\n        const consumingStack = Stack.of(context.scope);\n\n        if (this.stack.environment !== consumingStack.environment) {\n          this._enableCrossEnvironment();\n          return this.physicalName;\n        } else {\n          return nameAttr;\n        }\n      },\n    });\n  }\n\n  /**\n   * Returns an environment-sensitive token that should be used for the\n   * resource's \"ARN\" attribute (e.g. `bucket.bucketArn`).\n   *\n   * Normally, this token will resolve to `arnAttr`, but if the resource is\n   * referenced across environments, `arnComponents` will be used to synthesize\n   * a concrete ARN with the resource's physical name. Make sure to reference\n   * `this.physicalName` in `arnComponents`.\n   *\n   * @param arnAttr The CFN attribute which resolves to the ARN of the resource.\n   * Commonly it will be called \"Arn\" (e.g. `resource.attrArn`), but sometimes\n   * it's the CFN resource's `ref`.\n   * @param arnComponents The format of the ARN of this resource. You must\n   * reference `this.physicalName` somewhere within the ARN in order for\n   * cross-environment references to work.\n   *\n   */\n  protected getResourceArnAttribute(arnAttr: string, arnComponents: ArnComponents) {\n    return mimicReference(arnAttr, {\n      produce: (context: IResolveContext) => {\n        const consumingStack = Stack.of(context.scope);\n        if (this.stack.environment !== consumingStack.environment) {\n          this._enableCrossEnvironment();\n          return this.stack.formatArn(arnComponents);\n        } else {\n          return arnAttr;\n        }\n      },\n    });\n  }\n}\n\n/**\n * Produce a Lazy that is also a Reference (if the base value is a Reference).\n *\n * If the given value is a Reference (or resolves to a Reference), return a new\n * Reference that mimics the same target and display name, but resolves using\n * the logic of the passed lazy.\n *\n * If the given value is NOT a Reference, just return a simple Lazy.\n */\nfunction mimicReference(refSource: any, producer: IStringProducer): string {\n  const reference = Tokenization.reverse(refSource, {\n    // If this is an ARN concatenation, just fail to extract a reference.\n    failConcat: false,\n  });\n  if (!Reference.isReference(reference)) {\n    return Lazy.uncachedString(producer);\n  }\n\n  return Token.asString(new class extends Reference {\n    resolve(context: IResolveContext) {\n      return producer.produce(context);\n    }\n  }(reference, reference.target, reference.displayName));\n}\n"]} |
\ | No newline at end of file |