@aws-cdk/core
Version:
AWS Cloud Development Kit Core Library
191 lines • 27.2 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", { value: true });
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. 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.
*
* @returns an ArnComponents object which allows access to the various
* components of the ARN.
*/
static parse(arn, sepIfToken = '/', hasName = true) {
if (token_1.Token.isUnresolved(arn)) {
return parseToken(arn, sepIfToken, hasName);
}
const components = arn.split(':');
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   * 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  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\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 !== 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   * Given an ARN, parses it and returns components.\n   *\n   * If the ARN is a concrete string, it will be parsed and validated. The\n   * separator (`sep`) will be set to '/' if the 6th component includes a '/',\n   * in which case, `resource` will be set to the value before the '/' and\n   * `resourceName` will be the rest. In case there is no '/', `resource` will\n   * be set to the 6th components and `resourceName` will be set to the rest\n   * of the string.\n   *\n   * If the ARN includes tokens (or is a token), the ARN cannot be validated,\n   * since we don't have the actual value yet at the time of this function\n   * call. You will have to know the separator and the type of ARN. The\n   * resulting `ArnComponents` object will contain tokens for the\n   * subexpressions of the ARN, not string literals. In this case this\n   * function cannot properly parse the complete final resourceName (path) out\n   * of ARNs that use '/' to both separate the 'resource' from the\n   * 'resourceName' AND to subdivide the resourceName further. For example, in\n   * S3 ARNs:\n   *\n   *    arn:aws:s3:::my_corporate_bucket/path/to/exampleobject.png\n   *\n   * After parsing the resourceName will not contain\n   * 'path/to/exampleobject.png' but simply 'path'. This is a limitation\n   * because there is no slicing functionality in CloudFormation templates.\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  public static parse(arn: string, sepIfToken: string = '/', hasName: boolean = true): ArnComponents {\n    if (Token.isUnresolved(arn)) {\n      return parseToken(arn, sepIfToken, hasName);\n    }\n\n    const components = arn.split(':') as Array<string | undefined>;\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"]}