1 | ;
|
2 | var _a;
|
3 | Object.defineProperty(exports, "__esModule", { value: true });
|
4 | exports.Arn = exports.ArnFormat = void 0;
|
5 | const jsiiDeprecationWarnings = require("../.warnings.jsii.js");
|
6 | const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
7 | const cfn_fn_1 = require("./cfn-fn");
|
8 | const token_1 = require("./token");
|
9 | const util_1 = require("./util");
|
10 | /**
|
11 | * An enum representing the various ARN formats that different services use.
|
12 | */
|
13 | var ArnFormat;
|
14 | (function (ArnFormat) {
|
15 | /**
|
16 | * This represents a format where there is no 'resourceName' part.
|
17 | * This format is used for S3 resources,
|
18 | * like 'arn:aws:s3:::bucket'.
|
19 | * Everything after the last colon is considered the 'resource',
|
20 | * even if it contains slashes,
|
21 | * like in 'arn:aws:s3:::bucket/object.zip'.
|
22 | */
|
23 | ArnFormat["NO_RESOURCE_NAME"] = "arn:aws:service:region:account:resource";
|
24 | /**
|
25 | * This represents a format where the 'resource' and 'resourceName'
|
26 | * parts are separated with a colon.
|
27 | * Like in: 'arn:aws:service:region:account:resource:resourceName'.
|
28 | * Everything after the last colon is considered the 'resourceName',
|
29 | * even if it contains slashes,
|
30 | * like in 'arn:aws:apigateway:region:account:resource:/test/mydemoresource/*'.
|
31 | */
|
32 | ArnFormat["COLON_RESOURCE_NAME"] = "arn:aws:service:region:account:resource:resourceName";
|
33 | /**
|
34 | * This represents a format where the 'resource' and 'resourceName'
|
35 | * parts are separated with a slash.
|
36 | * Like in: 'arn:aws:service:region:account:resource/resourceName'.
|
37 | * Everything after the separating slash is considered the 'resourceName',
|
38 | * even if it contains colons,
|
39 | * like in 'arn:aws:cognito-sync:region:account:identitypool/us-east-1:1a1a1a1a-ffff-1111-9999-12345678:bla'.
|
40 | */
|
41 | ArnFormat["SLASH_RESOURCE_NAME"] = "arn:aws:service:region:account:resource/resourceName";
|
42 | /**
|
43 | * This represents a format where the 'resource' and 'resourceName'
|
44 | * parts are seperated with a slash,
|
45 | * but there is also an additional slash after the colon separating 'account' from 'resource'.
|
46 | * Like in: 'arn:aws:service:region:account:/resource/resourceName'.
|
47 | * Note that the leading slash is _not_ included in the parsed 'resource' part.
|
48 | */
|
49 | ArnFormat["SLASH_RESOURCE_SLASH_RESOURCE_NAME"] = "arn:aws:service:region:account:/resource/resourceName";
|
50 | })(ArnFormat = exports.ArnFormat || (exports.ArnFormat = {}));
|
51 | class Arn {
|
52 | constructor() { }
|
53 | /**
|
54 | * Creates an ARN from components.
|
55 | *
|
56 | * If `partition`, `region` or `account` are not specified, the stack's
|
57 | * partition, region and account will be used.
|
58 | *
|
59 | * If any component is the empty string, an empty string will be inserted
|
60 | * into the generated ARN at the location that component corresponds to.
|
61 | *
|
62 | * The ARN will be formatted as follows:
|
63 | *
|
64 | * arn:{partition}:{service}:{region}:{account}:{resource}{sep}{resource-name}
|
65 | *
|
66 | * The required ARN pieces that are omitted will be taken from the stack that
|
67 | * the 'scope' is attached to. If all ARN pieces are supplied, the supplied scope
|
68 | * can be 'undefined'.
|
69 | */
|
70 | static format(components, stack) {
|
71 | try {
|
72 | jsiiDeprecationWarnings._aws_cdk_core_ArnComponents(components);
|
73 | jsiiDeprecationWarnings._aws_cdk_core_Stack(stack);
|
74 | }
|
75 | catch (error) {
|
76 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
77 | Error.captureStackTrace(error, this.format);
|
78 | }
|
79 | throw error;
|
80 | }
|
81 | const partition = components.partition ?? stack?.partition;
|
82 | const region = components.region ?? stack?.region;
|
83 | const account = components.account ?? stack?.account;
|
84 | // Catch both 'null' and 'undefined'
|
85 | if (partition == null || region == null || account == null) {
|
86 | throw new Error(`Arn.format: partition (${partition}), region (${region}), and account (${account}) must all be passed if stack is not passed.`);
|
87 | }
|
88 | const sep = components.sep ?? (components.arnFormat === ArnFormat.COLON_RESOURCE_NAME ? ':' : '/');
|
89 | const values = [
|
90 | 'arn', ':', partition, ':', components.service, ':', region, ':', account, ':',
|
91 | ...(components.arnFormat === ArnFormat.SLASH_RESOURCE_SLASH_RESOURCE_NAME ? ['/'] : []),
|
92 | components.resource,
|
93 | ];
|
94 | if (sep !== '/' && sep !== ':' && sep !== '') {
|
95 | throw new Error('resourcePathSep may only be ":", "/" or an empty string');
|
96 | }
|
97 | if (components.resourceName != null) {
|
98 | values.push(sep);
|
99 | values.push(components.resourceName);
|
100 | }
|
101 | return values.join('');
|
102 | }
|
103 | /**
|
104 | * Given an ARN, parses it and returns components.
|
105 | *
|
106 | * IF THE ARN IS A CONCRETE STRING...
|
107 | *
|
108 | * ...it will be parsed and validated. The separator (`sep`) will be set to '/'
|
109 | * if the 6th component includes a '/', in which case, `resource` will be set
|
110 | * to the value before the '/' and `resourceName` will be the rest. In case
|
111 | * there is no '/', `resource` will be set to the 6th components and
|
112 | * `resourceName` will be set to the rest of the string.
|
113 | *
|
114 | * IF THE ARN IS A TOKEN...
|
115 | *
|
116 | * ...it cannot be validated, since we don't have the actual value yet at the
|
117 | * time of this function call. You will have to supply `sepIfToken` and
|
118 | * whether or not ARNs of the expected format usually have resource names
|
119 | * in order to parse it properly. The resulting `ArnComponents` object will
|
120 | * contain tokens for the subexpressions of the ARN, not string literals.
|
121 | *
|
122 | * If the resource name could possibly contain the separator char, the actual
|
123 | * resource name cannot be properly parsed. This only occurs if the separator
|
124 | * char is '/', and happens for example for S3 object ARNs, IAM Role ARNs,
|
125 | * IAM OIDC Provider ARNs, etc. To properly extract the resource name from a
|
126 | * Tokenized ARN, you must know the resource type and call
|
127 | * `Arn.extractResourceName`.
|
128 | *
|
129 | * @param arn The ARN to parse
|
130 | * @param sepIfToken The separator used to separate resource from resourceName
|
131 | * @param hasName Whether there is a name component in the ARN at all. For
|
132 | * example, SNS Topics ARNs have the 'resource' component contain the topic
|
133 | * name, and no 'resourceName' component.
|
134 | *
|
135 | * @returns an ArnComponents object which allows access to the various
|
136 | * components of the ARN.
|
137 | *
|
138 | * @returns an ArnComponents object which allows access to the various
|
139 | * components of the ARN.
|
140 | *
|
141 | * @deprecated use split instead
|
142 | */
|
143 | static parse(arn, sepIfToken = '/', hasName = true) {
|
144 | try {
|
145 | jsiiDeprecationWarnings.print("@aws-cdk/core.Arn#parse", "use split instead");
|
146 | }
|
147 | catch (error) {
|
148 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
149 | Error.captureStackTrace(error, this.parse);
|
150 | }
|
151 | throw error;
|
152 | }
|
153 | let arnFormat;
|
154 | if (!hasName) {
|
155 | arnFormat = ArnFormat.NO_RESOURCE_NAME;
|
156 | }
|
157 | else {
|
158 | arnFormat = sepIfToken === '/' ? ArnFormat.SLASH_RESOURCE_NAME : ArnFormat.COLON_RESOURCE_NAME;
|
159 | }
|
160 | return this.split(arn, arnFormat);
|
161 | }
|
162 | /**
|
163 | * Splits the provided ARN into its components.
|
164 | * Works both if 'arn' is a string like 'arn:aws:s3:::bucket',
|
165 | * and a Token representing a dynamic CloudFormation expression
|
166 | * (in which case the returned components will also be dynamic CloudFormation expressions,
|
167 | * encoded as Tokens).
|
168 | *
|
169 | * @param arn the ARN to split into its components
|
170 | * @param arnFormat the expected format of 'arn' - depends on what format the service 'arn' represents uses
|
171 | */
|
172 | static split(arn, arnFormat) {
|
173 | try {
|
174 | jsiiDeprecationWarnings._aws_cdk_core_ArnFormat(arnFormat);
|
175 | }
|
176 | catch (error) {
|
177 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
178 | Error.captureStackTrace(error, this.split);
|
179 | }
|
180 | throw error;
|
181 | }
|
182 | const components = parseArnShape(arn);
|
183 | if (components === 'token') {
|
184 | return parseTokenArn(arn, arnFormat);
|
185 | }
|
186 | const [, partition, service, region, account, resourceTypeOrName, ...rest] = components;
|
187 | let resource;
|
188 | let resourceName;
|
189 | let sep;
|
190 | let resourcePartStartIndex = 0;
|
191 | let detectedArnFormat;
|
192 | let slashIndex = resourceTypeOrName.indexOf('/');
|
193 | if (slashIndex === 0) {
|
194 | // new-style ARNs are of the form 'arn:aws:s4:us-west-1:12345:/resource-type/resource-name'
|
195 | slashIndex = resourceTypeOrName.indexOf('/', 1);
|
196 | resourcePartStartIndex = 1;
|
197 | detectedArnFormat = ArnFormat.SLASH_RESOURCE_SLASH_RESOURCE_NAME;
|
198 | }
|
199 | if (slashIndex !== -1) {
|
200 | // the slash is only a separator if ArnFormat is not NO_RESOURCE_NAME
|
201 | if (arnFormat === ArnFormat.NO_RESOURCE_NAME) {
|
202 | sep = undefined;
|
203 | slashIndex = -1;
|
204 | detectedArnFormat = ArnFormat.NO_RESOURCE_NAME;
|
205 | }
|
206 | else {
|
207 | sep = '/';
|
208 | detectedArnFormat = resourcePartStartIndex === 0
|
209 | ? ArnFormat.SLASH_RESOURCE_NAME
|
210 | // need to repeat this here, as otherwise the compiler thinks 'detectedArnFormat' is not initialized in all paths
|
211 | : ArnFormat.SLASH_RESOURCE_SLASH_RESOURCE_NAME;
|
212 | }
|
213 | }
|
214 | else if (rest.length > 0) {
|
215 | sep = ':';
|
216 | slashIndex = -1;
|
217 | detectedArnFormat = ArnFormat.COLON_RESOURCE_NAME;
|
218 | }
|
219 | else {
|
220 | sep = undefined;
|
221 | detectedArnFormat = ArnFormat.NO_RESOURCE_NAME;
|
222 | }
|
223 | if (slashIndex !== -1) {
|
224 | resource = resourceTypeOrName.substring(resourcePartStartIndex, slashIndex);
|
225 | resourceName = resourceTypeOrName.substring(slashIndex + 1);
|
226 | }
|
227 | else {
|
228 | resource = resourceTypeOrName;
|
229 | }
|
230 | if (rest.length > 0) {
|
231 | if (!resourceName) {
|
232 | resourceName = '';
|
233 | }
|
234 | else {
|
235 | resourceName += ':';
|
236 | }
|
237 | resourceName += rest.join(':');
|
238 | }
|
239 | // "|| undefined" will cause empty strings to be treated as "undefined".
|
240 | // Optional ARN attributes (e.g. region, account) should return as empty string
|
241 | // if they are provided as such.
|
242 | return util_1.filterUndefined({
|
243 | service: service || undefined,
|
244 | resource: resource || undefined,
|
245 | partition: partition || undefined,
|
246 | region,
|
247 | account,
|
248 | resourceName,
|
249 | sep,
|
250 | arnFormat: detectedArnFormat,
|
251 | });
|
252 | }
|
253 | /**
|
254 | * Extract the full resource name from an ARN
|
255 | *
|
256 | * Necessary for resource names (paths) that may contain the separator, like
|
257 | * `arn:aws:iam::111111111111:role/path/to/role/name`.
|
258 | *
|
259 | * Only works if we statically know the expected `resourceType` beforehand, since we're going
|
260 | * to use that to split the string on ':<resourceType>/' (and take the right-hand side).
|
261 | *
|
262 | * We can't extract the 'resourceType' from the ARN at hand, because CloudFormation Expressions
|
263 | * only allow literals in the 'separator' argument to `{ Fn::Split }`, and so it can't be
|
264 | * `{ Fn::Select: [5, { Fn::Split: [':', ARN] }}`.
|
265 | *
|
266 | * Only necessary for ARN formats for which the type-name separator is `/`.
|
267 | */
|
268 | static extractResourceName(arn, resourceType) {
|
269 | const components = parseArnShape(arn);
|
270 | if (components === 'token') {
|
271 | return cfn_fn_1.Fn.select(1, cfn_fn_1.Fn.split(`:${resourceType}/`, arn));
|
272 | }
|
273 | // Apparently we could just parse this right away. Validate that we got the right
|
274 | // resource type (to notify authors of incorrect assumptions right away).
|
275 | const parsed = Arn.split(arn, ArnFormat.SLASH_RESOURCE_NAME);
|
276 | if (!token_1.Token.isUnresolved(parsed.resource) && parsed.resource !== resourceType) {
|
277 | throw new Error(`Expected resource type '${resourceType}' in ARN, got '${parsed.resource}' in '${arn}'`);
|
278 | }
|
279 | if (!parsed.resourceName) {
|
280 | throw new Error(`Expected resource name in ARN, didn't find one: '${arn}'`);
|
281 | }
|
282 | return parsed.resourceName;
|
283 | }
|
284 | }
|
285 | exports.Arn = Arn;
|
286 | _a = JSII_RTTI_SYMBOL_1;
|
287 | Arn[_a] = { fqn: "@aws-cdk/core.Arn", version: "1.204.0" };
|
288 | /**
|
289 | * Given a Token evaluating to ARN, parses it and returns components.
|
290 | *
|
291 | * The ARN cannot be validated, since we don't have the actual value yet
|
292 | * at the time of this function call. You will have to know the separator
|
293 | * and the type of ARN.
|
294 | *
|
295 | * The resulting `ArnComponents` object will contain tokens for the
|
296 | * subexpressions of the ARN, not string literals.
|
297 | *
|
298 | * WARNING: this function cannot properly parse the complete final
|
299 | * 'resourceName' part if it contains colons,
|
300 | * like 'arn:aws:cognito-sync:region:account:identitypool/us-east-1:1a1a1a1a-ffff-1111-9999-12345678:bla'.
|
301 | *
|
302 | * @param arnToken The input token that contains an ARN
|
303 | * @param arnFormat the expected format of 'arn' - depends on what format the service the ARN represents uses
|
304 | */
|
305 | function parseTokenArn(arnToken, arnFormat) {
|
306 | // ARN looks like:
|
307 | // arn:partition:service:region:account:resource
|
308 | // arn:partition:service:region:account:resource:resourceName
|
309 | // arn:partition:service:region:account:resource/resourceName
|
310 | // arn:partition:service:region:account:/resource/resourceName
|
311 | const components = cfn_fn_1.Fn.split(':', arnToken);
|
312 | const partition = cfn_fn_1.Fn.select(1, components).toString();
|
313 | const service = cfn_fn_1.Fn.select(2, components).toString();
|
314 | const region = cfn_fn_1.Fn.select(3, components).toString();
|
315 | const account = cfn_fn_1.Fn.select(4, components).toString();
|
316 | let resource;
|
317 | let resourceName;
|
318 | let sep;
|
319 | if (arnFormat === ArnFormat.NO_RESOURCE_NAME || arnFormat === ArnFormat.COLON_RESOURCE_NAME) {
|
320 | // we know that the 'resource' part will always be the 6th segment in this case
|
321 | resource = cfn_fn_1.Fn.select(5, components);
|
322 | if (arnFormat === ArnFormat.COLON_RESOURCE_NAME) {
|
323 | resourceName = cfn_fn_1.Fn.select(6, components);
|
324 | sep = ':';
|
325 | }
|
326 | else {
|
327 | resourceName = undefined;
|
328 | sep = undefined;
|
329 | }
|
330 | }
|
331 | else {
|
332 | // we know that the 'resource' and 'resourceName' parts are separated by slash here,
|
333 | // so we split the 6th segment from the colon-separated ones with a slash
|
334 | const lastComponents = cfn_fn_1.Fn.split('/', cfn_fn_1.Fn.select(5, components));
|
335 | if (arnFormat === ArnFormat.SLASH_RESOURCE_NAME) {
|
336 | resource = cfn_fn_1.Fn.select(0, lastComponents);
|
337 | resourceName = cfn_fn_1.Fn.select(1, lastComponents);
|
338 | }
|
339 | else {
|
340 | // arnFormat is ArnFormat.SLASH_RESOURCE_SLASH_RESOURCE_NAME,
|
341 | // which means there's an extra slash there at the beginning that we need to skip
|
342 | resource = cfn_fn_1.Fn.select(1, lastComponents);
|
343 | resourceName = cfn_fn_1.Fn.select(2, lastComponents);
|
344 | }
|
345 | sep = '/';
|
346 | }
|
347 | return { partition, service, region, account, resource, resourceName, sep, arnFormat };
|
348 | }
|
349 | /**
|
350 | * Validate that a string is either unparseable or looks mostly like an ARN
|
351 | */
|
352 | function parseArnShape(arn) {
|
353 | // assume anything that starts with 'arn:' is an ARN,
|
354 | // so we can report better errors
|
355 | const looksLikeArn = arn.startsWith('arn:');
|
356 | if (!looksLikeArn) {
|
357 | if (token_1.Token.isUnresolved(arn)) {
|
358 | return 'token';
|
359 | }
|
360 | else {
|
361 | throw new Error(`ARNs must start with "arn:" and have at least 6 components: ${arn}`);
|
362 | }
|
363 | }
|
364 | // If the ARN merely contains Tokens, but otherwise *looks* mostly like an ARN,
|
365 | // it's a string of the form 'arn:${partition}:service:${region}:${account}:resource/xyz'.
|
366 | // Parse fields out to the best of our ability.
|
367 | // Tokens won't contain ":", so this won't break them.
|
368 | const components = arn.split(':');
|
369 | const partition = components.length > 1 ? components[1] : undefined;
|
370 | if (!partition) {
|
371 | throw new Error('The `partition` component (2nd component) of an ARN is required: ' + arn);
|
372 | }
|
373 | const service = components.length > 2 ? components[2] : undefined;
|
374 | if (!service) {
|
375 | throw new Error('The `service` component (3rd component) of an ARN is required: ' + arn);
|
376 | }
|
377 | const resource = components.length > 5 ? components[5] : undefined;
|
378 | if (!resource) {
|
379 | throw new Error('The `resource` component (6th component) of an ARN is required: ' + arn);
|
380 | }
|
381 | // Region can be missing in global ARNs (such as used by IAM)
|
382 | // Account can be missing in some ARN types (such as used for S3 buckets)
|
383 | return components;
|
384 | }
|
385 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"arn.js","sourceRoot":"","sources":["arn.ts"],"names":[],"mappings":"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","sourcesContent":["import { Fn } from './cfn-fn';\nimport { Stack } from './stack';\nimport { Token } from './token';\nimport { filterUndefined } from './util';\n\n/**\n * An enum representing the various ARN formats that different services use.\n */\nexport enum ArnFormat {\n  /**\n   * This represents a format where there is no 'resourceName' part.\n   * This format is used for S3 resources,\n   * like 'arn:aws:s3:::bucket'.\n   * Everything after the last colon is considered the 'resource',\n   * even if it contains slashes,\n   * like in 'arn:aws:s3:::bucket/object.zip'.\n   */\n  NO_RESOURCE_NAME = 'arn:aws:service:region:account:resource',\n\n  /**\n   * This represents a format where the 'resource' and 'resourceName'\n   * parts are separated with a colon.\n   * Like in: 'arn:aws:service:region:account:resource:resourceName'.\n   * Everything after the last colon is considered the 'resourceName',\n   * even if it contains slashes,\n   * like in 'arn:aws:apigateway:region:account:resource:/test/mydemoresource/*'.\n   */\n  COLON_RESOURCE_NAME = 'arn:aws:service:region:account:resource:resourceName',\n\n  /**\n   * This represents a format where the 'resource' and 'resourceName'\n   * parts are separated with a slash.\n   * Like in: 'arn:aws:service:region:account:resource/resourceName'.\n   * Everything after the separating slash is considered the 'resourceName',\n   * even if it contains colons,\n   * like in 'arn:aws:cognito-sync:region:account:identitypool/us-east-1:1a1a1a1a-ffff-1111-9999-12345678:bla'.\n   */\n  SLASH_RESOURCE_NAME = 'arn:aws:service:region:account:resource/resourceName',\n\n  /**\n   * This represents a format where the 'resource' and 'resourceName'\n   * parts are seperated with a slash,\n   * but there is also an additional slash after the colon separating 'account' from 'resource'.\n   * Like in: 'arn:aws:service:region:account:/resource/resourceName'.\n   * Note that the leading slash is _not_ included in the parsed 'resource' part.\n   */\n  SLASH_RESOURCE_SLASH_RESOURCE_NAME = 'arn:aws:service:region:account:/resource/resourceName',\n}\n\nexport interface ArnComponents {\n  /**\n   * The partition that the resource is in. For standard AWS regions, the\n   * partition is aws. If you have resources in other partitions, the\n   * partition is aws-partitionname. For example, the partition for resources\n   * in the China (Beijing) region is aws-cn.\n   *\n   * @default The AWS partition the stack is deployed to.\n   */\n  readonly partition?: string;\n\n  /**\n   * The service namespace that identifies the AWS product (for example,\n   * 's3', 'iam', 'codepipline').\n   */\n  readonly service: string;\n\n  /**\n   * The region the resource resides in. Note that the ARNs for some resources\n   * do not require a region, so this component might be omitted.\n   *\n   * @default The region the stack is deployed to.\n   */\n  readonly region?: string;\n\n  /**\n   * The ID of the AWS account that owns the resource, without the hyphens.\n   * For example, 123456789012. Note that the ARNs for some resources don't\n   * require an account number, so this component might be omitted.\n   *\n   * @default The account the stack is deployed to.\n   */\n  readonly account?: string;\n\n  /**\n   * Resource type (e.g. \"table\", \"autoScalingGroup\", \"certificate\").\n   * For some resource types, e.g. S3 buckets, this field defines the bucket name.\n   */\n  readonly resource: string;\n\n  /**\n   * Separator between resource type and the resource.\n   *\n   * Can be either '/', ':' or an empty string. Will only be used if resourceName is defined.\n   * @default '/'\n   *\n   * @deprecated use arnFormat instead\n   */\n  readonly sep?: string;\n\n  /**\n   * Resource name or path within the resource (i.e. S3 bucket object key) or\n   * a wildcard such as ``\"*\"``. This is service-dependent.\n   */\n  readonly resourceName?: string;\n\n  /**\n   * The specific ARN format to use for this ARN value.\n   *\n   * @default - uses value of `sep` as the separator for formatting,\n   *   `ArnFormat.SLASH_RESOURCE_NAME` if that property was also not provided\n   */\n  readonly arnFormat?: ArnFormat;\n}\n\nexport class Arn {\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 static format(components: ArnComponents, stack?: Stack): string {\n    const partition = components.partition ?? stack?.partition;\n    const region = components.region ?? stack?.region;\n    const account = components.account ?? stack?.account;\n\n    // Catch both 'null' and 'undefined'\n    if (partition == null || region == null || account == null) {\n      throw new Error(`Arn.format: partition (${partition}), region (${region}), and account (${account}) must all be passed if stack is not passed.`);\n    }\n\n    const sep = components.sep ?? (components.arnFormat === ArnFormat.COLON_RESOURCE_NAME ? ':' : '/');\n\n    const values = [\n      'arn', ':', partition, ':', components.service, ':', region, ':', account, ':',\n      ...(components.arnFormat === ArnFormat.SLASH_RESOURCE_SLASH_RESOURCE_NAME ? ['/'] : []),\n      components.resource,\n    ];\n\n    if (sep !== '/' && sep !== ':' && sep !== '') {\n      throw new Error('resourcePathSep may only be \":\", \"/\" or an empty string');\n    }\n\n    if (components.resourceName != null) {\n      values.push(sep);\n      values.push(components.resourceName);\n    }\n\n    return values.join('');\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 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 split instead\n   */\n  public static parse(arn: string, sepIfToken: string = '/', hasName: boolean = true): ArnComponents {\n    let arnFormat: ArnFormat;\n    if (!hasName) {\n      arnFormat = ArnFormat.NO_RESOURCE_NAME;\n    } else {\n      arnFormat = sepIfToken === '/' ? ArnFormat.SLASH_RESOURCE_NAME : ArnFormat.COLON_RESOURCE_NAME;\n    }\n    return this.split(arn, arnFormat);\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 static split(arn: string, arnFormat: ArnFormat): ArnComponents {\n    const components = parseArnShape(arn);\n    if (components === 'token') {\n      return parseTokenArn(arn, arnFormat);\n    }\n\n    const [, partition, service, region, account, resourceTypeOrName, ...rest] = components;\n\n    let resource: string;\n    let resourceName: string | undefined;\n    let sep: string | undefined;\n    let resourcePartStartIndex = 0;\n    let detectedArnFormat: ArnFormat;\n\n    let slashIndex = resourceTypeOrName.indexOf('/');\n    if (slashIndex === 0) {\n      // new-style ARNs are of the form 'arn:aws:s4:us-west-1:12345:/resource-type/resource-name'\n      slashIndex = resourceTypeOrName.indexOf('/', 1);\n      resourcePartStartIndex = 1;\n      detectedArnFormat = ArnFormat.SLASH_RESOURCE_SLASH_RESOURCE_NAME;\n    }\n    if (slashIndex !== -1) {\n      // the slash is only a separator if ArnFormat is not NO_RESOURCE_NAME\n      if (arnFormat === ArnFormat.NO_RESOURCE_NAME) {\n        sep = undefined;\n        slashIndex = -1;\n        detectedArnFormat = ArnFormat.NO_RESOURCE_NAME;\n      } else {\n        sep = '/';\n        detectedArnFormat = resourcePartStartIndex === 0\n          ? ArnFormat.SLASH_RESOURCE_NAME\n          // need to repeat this here, as otherwise the compiler thinks 'detectedArnFormat' is not initialized in all paths\n          : ArnFormat.SLASH_RESOURCE_SLASH_RESOURCE_NAME;\n      }\n    } else if (rest.length > 0) {\n      sep = ':';\n      slashIndex = -1;\n      detectedArnFormat = ArnFormat.COLON_RESOURCE_NAME;\n    } else {\n      sep = undefined;\n      detectedArnFormat = ArnFormat.NO_RESOURCE_NAME;\n    }\n\n    if (slashIndex !== -1) {\n      resource = resourceTypeOrName.substring(resourcePartStartIndex, slashIndex);\n      resourceName = resourceTypeOrName.substring(slashIndex + 1);\n    } else {\n      resource = resourceTypeOrName;\n    }\n\n    if (rest.length > 0) {\n      if (!resourceName) {\n        resourceName = '';\n      } else {\n        resourceName += ':';\n      }\n\n      resourceName += rest.join(':');\n    }\n\n    // \"|| undefined\" will cause empty strings to be treated as \"undefined\".\n    // Optional ARN attributes (e.g. region, account) should return as empty string\n    // if they are provided as such.\n    return filterUndefined({\n      service: service || undefined,\n      resource: resource || undefined,\n      partition: partition || undefined,\n      region,\n      account,\n      resourceName,\n      sep,\n      arnFormat: detectedArnFormat,\n    });\n  }\n\n  /**\n   * Extract the full resource name from an ARN\n   *\n   * Necessary for resource names (paths) that may contain the separator, like\n   * `arn:aws:iam::111111111111:role/path/to/role/name`.\n   *\n   * Only works if we statically know the expected `resourceType` beforehand, since we're going\n   * to use that to split the string on ':<resourceType>/' (and take the right-hand side).\n   *\n   * We can't extract the 'resourceType' from the ARN at hand, because CloudFormation Expressions\n   * only allow literals in the 'separator' argument to `{ Fn::Split }`, and so it can't be\n   * `{ Fn::Select: [5, { Fn::Split: [':', ARN] }}`.\n   *\n   * Only necessary for ARN formats for which the type-name separator is `/`.\n   */\n  public static extractResourceName(arn: string, resourceType: string): string {\n    const components = parseArnShape(arn);\n    if (components === 'token') {\n      return Fn.select(1, Fn.split(`:${resourceType}/`, arn));\n    }\n\n    // Apparently we could just parse this right away. Validate that we got the right\n    // resource type (to notify authors of incorrect assumptions right away).\n    const parsed = Arn.split(arn, ArnFormat.SLASH_RESOURCE_NAME);\n    if (!Token.isUnresolved(parsed.resource) && parsed.resource !== resourceType) {\n      throw new Error(`Expected resource type '${resourceType}' in ARN, got '${parsed.resource}' in '${arn}'`);\n    }\n    if (!parsed.resourceName) {\n      throw new Error(`Expected resource name in ARN, didn't find one: '${arn}'`);\n    }\n    return parsed.resourceName;\n  }\n\n  private constructor() { }\n}\n\n/**\n * Given a Token evaluating to ARN, parses it and returns components.\n *\n * The ARN cannot be validated, since we don't have the actual value yet\n * at the time of this function call. You will have to know the separator\n * and the type of ARN.\n *\n * The resulting `ArnComponents` object will contain tokens for the\n * subexpressions of the ARN, not string literals.\n *\n * WARNING: this function cannot properly parse the complete final\n * 'resourceName' part if it contains colons,\n * like 'arn:aws:cognito-sync:region:account:identitypool/us-east-1:1a1a1a1a-ffff-1111-9999-12345678:bla'.\n *\n * @param arnToken The input token that contains an ARN\n * @param arnFormat the expected format of 'arn' - depends on what format the service the ARN represents uses\n */\nfunction parseTokenArn(arnToken: string, arnFormat: ArnFormat): ArnComponents {\n  // ARN looks like:\n  // arn:partition:service:region:account:resource\n  // arn:partition:service:region:account:resource:resourceName\n  // arn:partition:service:region:account:resource/resourceName\n  // arn:partition:service:region:account:/resource/resourceName\n\n  const components = Fn.split(':', arnToken);\n\n  const partition = Fn.select(1, components).toString();\n  const service = Fn.select(2, components).toString();\n  const region = Fn.select(3, components).toString();\n  const account = Fn.select(4, components).toString();\n  let resource: string;\n  let resourceName: string | undefined;\n  let sep: string | undefined;\n\n  if (arnFormat === ArnFormat.NO_RESOURCE_NAME || arnFormat === ArnFormat.COLON_RESOURCE_NAME) {\n    // we know that the 'resource' part will always be the 6th segment in this case\n    resource = Fn.select(5, components);\n    if (arnFormat === ArnFormat.COLON_RESOURCE_NAME) {\n      resourceName = Fn.select(6, components);\n      sep = ':';\n    } else {\n      resourceName = undefined;\n      sep = undefined;\n    }\n  } else {\n    // we know that the 'resource' and 'resourceName' parts are separated by slash here,\n    // so we split the 6th segment from the colon-separated ones with a slash\n    const lastComponents = Fn.split('/', Fn.select(5, components));\n\n    if (arnFormat === ArnFormat.SLASH_RESOURCE_NAME) {\n      resource = Fn.select(0, lastComponents);\n      resourceName = Fn.select(1, lastComponents);\n    } else {\n      // arnFormat is ArnFormat.SLASH_RESOURCE_SLASH_RESOURCE_NAME,\n      // which means there's an extra slash there at the beginning that we need to skip\n      resource = Fn.select(1, lastComponents);\n      resourceName = Fn.select(2, lastComponents);\n    }\n    sep = '/';\n  }\n\n  return { partition, service, region, account, resource, resourceName, sep, arnFormat };\n}\n\n/**\n * Validate that a string is either unparseable or looks mostly like an ARN\n */\nfunction parseArnShape(arn: string): 'token' | string[] {\n  // assume anything that starts with 'arn:' is an ARN,\n  // so we can report better errors\n  const looksLikeArn = arn.startsWith('arn:');\n\n  if (!looksLikeArn) {\n    if (Token.isUnresolved(arn)) {\n      return 'token';\n    } else {\n      throw new Error(`ARNs must start with \"arn:\" and have at least 6 components: ${arn}`);\n    }\n  }\n\n  // If the ARN merely contains Tokens, but otherwise *looks* mostly like an ARN,\n  // it's a string of the form 'arn:${partition}:service:${region}:${account}:resource/xyz'.\n  // Parse fields out to the best of our ability.\n  // Tokens won't contain \":\", so this won't break them.\n  const components = arn.split(':');\n\n  const partition = components.length > 1 ? components[1] : undefined;\n  if (!partition) {\n    throw new Error('The `partition` component (2nd component) of an ARN is required: ' + arn);\n  }\n\n  const service = components.length > 2 ? components[2] : undefined;\n  if (!service) {\n    throw new Error('The `service` component (3rd component) of an ARN is required: ' + arn);\n  }\n\n  const resource = components.length > 5 ? components[5] : undefined;\n  if (!resource) {\n    throw new Error('The `resource` component (6th component) of an ARN is required: ' + arn);\n  }\n\n  // Region can be missing in global ARNs (such as used by IAM)\n\n  // Account can be missing in some ARN types (such as used for S3 buckets)\n\n  return components;\n}\n"]} |
\ | No newline at end of file |