UNPKG

30.8 kBJavaScriptView Raw
1"use strict";
2var _a;
3Object.defineProperty(exports, "__esModule", { value: true });
4exports.Resource = void 0;
5const jsiiDeprecationWarnings = require("../.warnings.jsii.js");
6const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
7const arn_1 = require("./arn");
8const cfn_resource_1 = require("./cfn-resource");
9const construct_compat_1 = require("./construct-compat");
10const lazy_1 = require("./lazy");
11const physical_name_generator_1 = require("./private/physical-name-generator");
12const reference_1 = require("./reference");
13const stack_1 = require("./stack");
14const token_1 = require("./token");
15const RESOURCE_SYMBOL = Symbol.for('@aws-cdk/core.Resource');
16/**
17 * A construct which represents an AWS resource.
18 */
19class 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}
195exports.Resource = Resource;
196_a = JSII_RTTI_SYMBOL_1;
197Resource[_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 */
207function 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