UNPKG

@aws-cdk/core

Version:

AWS Cloud Development Kit Core Library

194 lines 28.1 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.Arn = void 0; const cfn_fn_1 = require("./cfn-fn"); const token_1 = require("./token"); const util_1 = require("./util"); /** * */ class Arn { /** * Creates an ARN from components. * * If `partition`, `region` or `account` are not specified, the stack's * partition, region and account will be used. * * If any component is the empty string, an empty string will be inserted * into the generated ARN at the location that component corresponds to. * * The ARN will be formatted as follows: * * arn:{partition}:{service}:{region}:{account}:{resource}{sep}{resource-name} * * The required ARN pieces that are omitted will be taken from the stack that * the 'scope' is attached to. If all ARN pieces are supplied, the supplied scope * can be 'undefined'. */ static format(components, stack) { const partition = components.partition !== undefined ? components.partition : stack.partition; const region = components.region !== undefined ? components.region : stack.region; const account = components.account !== undefined ? components.account : stack.account; const sep = components.sep !== undefined ? components.sep : '/'; const values = ['arn', ':', partition, ':', components.service, ':', region, ':', account, ':', components.resource]; if (sep !== '/' && sep !== ':' && sep !== '') { throw new Error('resourcePathSep may only be ":", "/" or an empty string'); } if (components.resourceName != null) { values.push(sep); values.push(components.resourceName); } return values.join(''); } /** * Given an ARN, parses it and returns components. * * If the ARN is a concrete string, it will be parsed and validated. The * separator (`sep`) will be set to '/' if the 6th component includes a '/', * in which case, `resource` will be set to the value before the '/' and * `resourceName` will be the rest. In case there is no '/', `resource` will * be set to the 6th components and `resourceName` will be set to the rest * of the string. * * If the ARN includes tokens (or is a token), the ARN cannot be validated, * since we don't have the actual value yet at the time of this function * call. You will have to know the separator and the type of ARN. The * resulting `ArnComponents` object will contain tokens for the * subexpressions of the ARN, not string literals. In this case this * function cannot properly parse the complete final resourceName (path) out * of ARNs that use '/' to both separate the 'resource' from the * 'resourceName' AND to subdivide the resourceName further. For example, in * S3 ARNs: * * arn:aws:s3:::my_corporate_bucket/path/to/exampleobject.png * * After parsing the resourceName will not contain * 'path/to/exampleobject.png' but simply 'path'. This is a limitation * because there is no slicing functionality in CloudFormation templates. * * @param arn The ARN to parse. * @param sepIfToken The separator used to separate resource from resourceName. * @param hasName Whether there is a name component in the ARN at all. * @returns an ArnComponents object which allows access to the various * components of the ARN. */ static parse(arn, sepIfToken = '/', hasName = true) { const components = arn.split(':'); const looksLikeArn = arn.startsWith('arn:') && components.length >= 6 && components.length <= 7; if (token_1.Token.isUnresolved(arn) && !looksLikeArn) { return parseToken(arn, sepIfToken, hasName); } // If the ARN merely contains Tokens, but otherwise *looks* mostly like an ARN, // it's a string of the form 'arn:${partition}:service:${region}:${account}:abc/xyz'. // Parse fields out to the best of our ability. // Tokens won't contain ":", so this won't break them. if (components.length < 6) { throw new Error('ARNs must have at least 6 components: ' + arn); } const [arnPrefix, partition, service, region, account, sixth, ...rest] = components; if (arnPrefix !== 'arn') { throw new Error('ARNs must start with "arn:": ' + arn); } if (!service) { throw new Error('The `service` component (3rd component) is required: ' + arn); } if (!sixth) { throw new Error('The `resource` component (6th component) is required: ' + arn); } let resource; let resourceName; let sep; let sepIndex = sixth.indexOf('/'); if (sepIndex !== -1) { sep = '/'; } else if (rest.length > 0) { sep = ':'; sepIndex = -1; } if (sepIndex !== -1) { resource = sixth.substr(0, sepIndex); resourceName = sixth.substr(sepIndex + 1); } else { resource = sixth; } if (rest.length > 0) { if (!resourceName) { resourceName = ''; } else { resourceName += ':'; } resourceName += rest.join(':'); } // "|| undefined" will cause empty strings to be treated as "undefined". // Optional ARN attributes (e.g. region, account) should return as empty string // if they are provided as such. return util_1.filterUndefined({ service: service || undefined, resource: resource || undefined, partition: partition || undefined, region, account, resourceName, sep, }); } constructor() { } } exports.Arn = Arn; /** * Given a Token evaluating to ARN, parses it and returns components. * * The ARN cannot be validated, since we don't have the actual value yet * at the time of this function call. You will have to know the separator * and the type of ARN. * * The resulting `ArnComponents` object will contain tokens for the * subexpressions of the ARN, not string literals. * * WARNING: this function cannot properly parse the complete final * resourceName (path) out of ARNs that use '/' to both separate the * 'resource' from the 'resourceName' AND to subdivide the resourceName * further. For example, in S3 ARNs: * * arn:aws:s3:::my_corporate_bucket/path/to/exampleobject.png * * After parsing the resourceName will not contain 'path/to/exampleobject.png' * but simply 'path'. This is a limitation because there is no slicing * functionality in CloudFormation templates. * * @param arnToken The input token that contains an ARN * @param sep The separator used to separate resource from resourceName * @param hasName Whether there is a name component in the ARN at all. * For example, SNS Topics ARNs have the 'resource' component contain the * topic name, and no 'resourceName' component. * @returns an ArnComponents object which allows access to the various * components of the ARN. */ function parseToken(arnToken, sep = '/', hasName = true) { // Arn ARN looks like: // arn:partition:service:region:account-id:resource // arn:partition:service:region:account-id:resourcetype/resource // arn:partition:service:region:account-id:resourcetype:resource // We need the 'hasName' argument because {Fn::Select}ing a nonexistent field // throws an error. const components = cfn_fn_1.Fn.split(':', arnToken); const partition = cfn_fn_1.Fn.select(1, components).toString(); const service = cfn_fn_1.Fn.select(2, components).toString(); const region = cfn_fn_1.Fn.select(3, components).toString(); const account = cfn_fn_1.Fn.select(4, components).toString(); if (sep === ':') { const resource = cfn_fn_1.Fn.select(5, components).toString(); const resourceName = hasName ? cfn_fn_1.Fn.select(6, components).toString() : undefined; return { partition, service, region, account, resource, resourceName, sep }; } else { const lastComponents = cfn_fn_1.Fn.split(sep, cfn_fn_1.Fn.select(5, components)); const resource = cfn_fn_1.Fn.select(0, lastComponents).toString(); const resourceName = hasName ? cfn_fn_1.Fn.select(1, lastComponents).toString() : undefined; return { partition, service, region, account, resource, resourceName, sep }; } } //# 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\nexport interface ArnComponents {\n                                                                                                                                                                                                                                                                                                                                                      \n  readonly partition?: string;\n\n                                                                                                                      \n  readonly service: string;\n\n                                                                                                                                                                                                                    \n  readonly region?: string;\n\n                                                                                                                                                                                                                                                                                                \n  readonly account?: string;\n\n                                                                                                                                                                    \n  readonly resource: string;\n\n                                                                                                                                                                                       \n  readonly sep?: string;\n\n                                                                                                                                                     \n  readonly resourceName?: string;\n}\n\nexport class Arn {\n                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                      \n  public static format(components: ArnComponents, stack: Stack): string {\n    const partition = components.partition !== undefined ? components.partition : stack.partition;\n    const region = components.region !== undefined ? components.region : stack.region;\n    const account = components.account !== undefined ? components.account : stack.account;\n    const sep = components.sep !== undefined ? components.sep : '/';\n\n    const values = ['arn', ':', partition, ':', components.service, ':', region, ':', account, ':', components.resource];\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  public static parse(arn: string, sepIfToken: string = '/', hasName: boolean = true): ArnComponents {\n    const components = arn.split(':') as Array<string | undefined>;\n    const looksLikeArn = arn.startsWith('arn:') && components.length >= 6 && components.length <= 7;\n    if (Token.isUnresolved(arn) && !looksLikeArn) {\n      return parseToken(arn, sepIfToken, hasName);\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}:abc/xyz'.\n    // Parse fields out to the best of our ability.\n    // Tokens won't contain \":\", so this won't break them.\n\n    if (components.length < 6) {\n      throw new Error('ARNs must have at least 6 components: ' + arn);\n    }\n\n    const [arnPrefix, partition, service, region, account, sixth, ...rest] = components;\n\n    if (arnPrefix !== 'arn') {\n      throw new Error('ARNs must start with \"arn:\": ' + arn);\n    }\n\n    if (!service) {\n      throw new Error('The `service` component (3rd component) is required: ' + arn);\n    }\n\n    if (!sixth) {\n      throw new Error('The `resource` component (6th component) is required: ' + arn);\n    }\n\n    let resource: string;\n    let resourceName: string | undefined;\n    let sep: string | undefined;\n\n    let sepIndex = sixth.indexOf('/');\n    if (sepIndex !== -1) {\n      sep = '/';\n    } else if (rest.length > 0) {\n      sep = ':';\n      sepIndex = -1;\n    }\n\n    if (sepIndex !== -1) {\n      resource = sixth.substr(0, sepIndex);\n      resourceName = sixth.substr(sepIndex + 1);\n    } else {\n      resource = sixth;\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    });\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 (path) out of ARNs that use '/' to both separate the\n * 'resource' from the 'resourceName' AND to subdivide the resourceName\n * further. For example, in S3 ARNs:\n *\n *    arn:aws:s3:::my_corporate_bucket/path/to/exampleobject.png\n *\n * After parsing the resourceName will not contain 'path/to/exampleobject.png'\n * but simply 'path'. This is a limitation because there is no slicing\n * functionality in CloudFormation templates.\n *\n * @param arnToken The input token that contains an ARN\n * @param sep The separator used to separate resource from resourceName\n * @param hasName Whether there is a name component in the ARN at all.\n * For example, SNS Topics ARNs have the 'resource' component contain the\n * topic name, and no 'resourceName' component.\n * @returns an ArnComponents object which allows access to the various\n * components of the ARN.\n */\nfunction parseToken(arnToken: string, sep: string = '/', hasName: boolean = true): ArnComponents {\n  // Arn ARN looks like:\n  // arn:partition:service:region:account-id:resource\n  // arn:partition:service:region:account-id:resourcetype/resource\n  // arn:partition:service:region:account-id:resourcetype:resource\n\n  // We need the 'hasName' argument because {Fn::Select}ing a nonexistent field\n  // throws an error.\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\n  if (sep === ':') {\n    const resource = Fn.select(5, components).toString();\n    const resourceName = hasName ? Fn.select(6, components).toString() : undefined;\n\n    return { partition, service, region, account, resource, resourceName, sep };\n  } else {\n    const lastComponents = Fn.split(sep, Fn.select(5, components));\n\n    const resource = Fn.select(0, lastComponents).toString();\n    const resourceName = hasName ? Fn.select(1, lastComponents).toString() : undefined;\n\n    return { partition, service, region, account, resource, resourceName, sep };\n  }\n}\n"]}