UNPKG

49.9 kBJavaScriptView Raw
1"use strict";
2var _a;
3Object.defineProperty(exports, "__esModule", { value: true });
4exports.Arn = exports.ArnFormat = void 0;
5const jsiiDeprecationWarnings = require("../.warnings.jsii.js");
6const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
7const cfn_fn_1 = require("./cfn-fn");
8const token_1 = require("./token");
9const util_1 = require("./util");
10/**
11 * An enum representing the various ARN formats that different services use.
12 */
13var 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 = {}));
51class 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}
285exports.Arn = Arn;
286_a = JSII_RTTI_SYMBOL_1;
287Arn[_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 */
305function 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 */
352function 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