@aws-cdk/core
Version:
AWS Cloud Development Kit Core Library
89 lines • 11.7 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", { value: true });
exports.isGeneratedWhenNeededMarker = exports.GeneratedWhenNeededMarker = exports.generatePhysicalName = void 0;
const crypto = require("crypto");
const constructs_1 = require("constructs");
const stack_1 = require("../stack");
const token_1 = require("../token");
const token_map_1 = require("./token-map");
function generatePhysicalName(resource) {
const stack = stack_1.Stack.of(resource);
const stackPart = new PrefixNamePart(stack.stackName, 25);
const idPart = new SuffixNamePart(constructs_1.Node.of(resource).uniqueId, 24);
const region = stack.region;
if (token_1.Token.isUnresolved(region) || !region) {
throw new Error(`Cannot generate a physical name for ${constructs_1.Node.of(resource).path}, because the region is un-resolved or missing`);
}
const account = stack.account;
if (token_1.Token.isUnresolved(account) || !account) {
throw new Error(`Cannot generate a physical name for ${constructs_1.Node.of(resource).path}, because the account is un-resolved or missing`);
}
const parts = [stackPart, idPart]
.map(part => part.generate());
const hashLength = 12;
const sha256 = crypto.createHash('sha256')
.update(stackPart.bareStr)
.update(idPart.bareStr)
.update(region)
.update(account);
const hash = sha256.digest('hex').slice(0, hashLength);
const ret = [...parts, hash].join('');
return ret.toLowerCase();
}
exports.generatePhysicalName = generatePhysicalName;
class NamePart {
constructor(bareStr) {
this.bareStr = bareStr;
}
}
class PrefixNamePart extends NamePart {
constructor(bareStr, prefixLength) {
super(bareStr);
this.prefixLength = prefixLength;
}
generate() {
return this.bareStr.slice(0, this.prefixLength);
}
}
class SuffixNamePart extends NamePart {
constructor(str, suffixLength) {
super(str);
this.suffixLength = suffixLength;
}
generate() {
const strLen = this.bareStr.length;
const startIndex = Math.max(strLen - this.suffixLength, 0);
return this.bareStr.slice(startIndex, strLen);
}
}
const GENERATE_IF_NEEDED_SYMBOL = Symbol.for('@aws-cdk/core.<private>.GenerateIfNeeded');
/**
* This marker token is used by PhysicalName.GENERATE_IF_NEEDED. When that token is passed to the
* physicalName property of a Resource, it triggers different behavior in the Resource constructor
* that will allow emission of a generated physical name (when the resource is used across
* environments) or undefined (when the resource is not shared).
*
* This token throws an Error when it is resolved, as a way to prevent inadvertent mis-uses of it.
*/
class GeneratedWhenNeededMarker {
constructor() {
this.creationStack = [];
Object.defineProperty(this, GENERATE_IF_NEEDED_SYMBOL, { value: true });
}
resolve(_ctx) {
throw new Error('Invalid physical name passed to CloudFormation. Use "this.physicalName" instead');
}
toString() {
return 'PhysicalName.GENERATE_IF_NEEDED';
}
}
exports.GeneratedWhenNeededMarker = GeneratedWhenNeededMarker;
/**
* Checks whether a stringified token resolves to a `GeneratedWhenNeededMarker`.
*/
function isGeneratedWhenNeededMarker(val) {
const token = token_map_1.TokenMap.instance().lookupString(val);
return !!token && GENERATE_IF_NEEDED_SYMBOL in token;
}
exports.isGeneratedWhenNeededMarker = isGeneratedWhenNeededMarker;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"physical-name-generator.js","sourceRoot":"","sources":["physical-name-generator.ts"],"names":[],"mappings":"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","sourcesContent":["import * as crypto from 'crypto';\nimport { Node } from 'constructs';\nimport { IResolvable, IResolveContext } from '../resolvable';\nimport { IResource } from '../resource';\nimport { Stack } from '../stack';\nimport { Token } from '../token';\nimport { TokenMap } from './token-map';\n\nexport function generatePhysicalName(resource: IResource): string {\n  const stack = Stack.of(resource);\n  const stackPart = new PrefixNamePart(stack.stackName, 25);\n  const idPart = new SuffixNamePart(Node.of(resource).uniqueId, 24);\n\n  const region: string = stack.region;\n  if (Token.isUnresolved(region) || !region) {\n    throw new Error(`Cannot generate a physical name for ${Node.of(resource).path}, because the region is un-resolved or missing`);\n  }\n\n  const account: string = stack.account;\n  if (Token.isUnresolved(account) || !account) {\n    throw new Error(`Cannot generate a physical name for ${Node.of(resource).path}, because the account is un-resolved or missing`);\n  }\n\n  const parts = [stackPart, idPart]\n    .map(part => part.generate());\n\n  const hashLength = 12;\n  const sha256 = crypto.createHash('sha256')\n    .update(stackPart.bareStr)\n    .update(idPart.bareStr)\n    .update(region)\n    .update(account);\n  const hash = sha256.digest('hex').slice(0, hashLength);\n\n  const ret = [...parts, hash].join('');\n\n  return ret.toLowerCase();\n}\n\nabstract class NamePart {\n  public readonly bareStr: string;\n\n  constructor(bareStr: string) {\n    this.bareStr = bareStr;\n  }\n\n  public abstract generate(): string;\n}\n\nclass PrefixNamePart extends NamePart {\n  constructor(bareStr: string, private readonly prefixLength: number) {\n    super(bareStr);\n  }\n\n  public generate(): string {\n    return this.bareStr.slice(0, this.prefixLength);\n  }\n}\n\nclass SuffixNamePart extends NamePart {\n  constructor(str: string, private readonly suffixLength: number) {\n    super(str);\n  }\n\n  public generate(): string {\n    const strLen = this.bareStr.length;\n    const startIndex = Math.max(strLen - this.suffixLength, 0);\n    return this.bareStr.slice(startIndex, strLen);\n  }\n}\n\nconst GENERATE_IF_NEEDED_SYMBOL = Symbol.for('@aws-cdk/core.<private>.GenerateIfNeeded');\n\n/**\n * This marker token is used by PhysicalName.GENERATE_IF_NEEDED. When that token is passed to the\n * physicalName property of a Resource, it triggers different behavior in the Resource constructor\n * that will allow emission of a generated physical name (when the resource is used across\n * environments) or undefined (when the resource is not shared).\n *\n * This token throws an Error when it is resolved, as a way to prevent inadvertent mis-uses of it.\n */\nexport class GeneratedWhenNeededMarker implements IResolvable {\n  public readonly creationStack: string[] = [];\n\n  constructor() {\n    Object.defineProperty(this, GENERATE_IF_NEEDED_SYMBOL, { value: true });\n  }\n\n  public resolve(_ctx: IResolveContext): never {\n    throw new Error('Invalid physical name passed to CloudFormation. Use \"this.physicalName\" instead');\n  }\n\n  public toString(): string {\n    return 'PhysicalName.GENERATE_IF_NEEDED';\n  }\n}\n\n/**\n * Checks whether a stringified token resolves to a `GeneratedWhenNeededMarker`.\n */\nexport function isGeneratedWhenNeededMarker(val: string): boolean {\n  const token = TokenMap.instance().lookupString(val);\n  return !!token && GENERATE_IF_NEEDED_SYMBOL in token;\n}\n"]}