UNPKG

@aws-cdk/core

Version:

AWS Cloud Development Kit Core Library

198 lines 21.2 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.TokenMap = void 0; const token_1 = require("../token"); const encoding_1 = require("./encoding"); const glob = global; const STRING_SYMBOL = Symbol.for('@aws-cdk/core.TokenMap.STRING'); const LIST_SYMBOL = Symbol.for('@aws-cdk/core.TokenMap.LIST'); const NUMBER_SYMBOL = Symbol.for('@aws-cdk/core.TokenMap.NUMBER'); /** * Central place where we keep a mapping from Tokens to their String representation * * The string representation is used to embed token into strings, * and stored to be able to reverse that mapping. * * All instances of TokenStringMap share the same storage, so that this process * works even when different copies of the library are loaded. */ class TokenMap { constructor() { this.stringTokenMap = new Map(); this.numberTokenMap = new Map(); /** * Counter to assign unique IDs to tokens * * Start at a random number to prevent people from accidentally taking * dependencies on token values between runs. * * This is most prominent in tests, where people will write: * * ```ts * sha256(JSON.stringify({ ...some structure that can contain tokens ... })) * ``` * * This should have been: * * ```ts * sha256(JSON.stringify(stack.resolve({ ...some structure that can contain tokens ... }))) * ``` * * The hash value is hard to inspect for correctness. It will LOOK consistent * during testing, but will break as soon as someone stringifies another * token before the run. * * By changing the starting number for tokens, we ensure that the hash is almost * guaranteed to be different during a few test runs, so the hashing of unresolved * tokens can be detected. */ this.tokenCounter = Math.floor(Math.random() * 10); } /** * Singleton instance of the token string map */ static instance() { if (!glob.__cdkTokenMap) { glob.__cdkTokenMap = new TokenMap(); } return glob.__cdkTokenMap; } /** * Generate a unique string for this Token, returning a key * * Every call for the same Token will produce a new unique string, no * attempt is made to deduplicate. Token objects should cache the * value themselves, if required. * * The token can choose (part of) its own representation string with a * hint. This may be used to produce aesthetically pleasing and * recognizable token representations for humans. */ registerString(token, displayHint) { return cachedValue(token, STRING_SYMBOL, () => { const key = this.registerStringKey(token, displayHint); return `${encoding_1.BEGIN_STRING_TOKEN_MARKER}${key}${encoding_1.END_TOKEN_MARKER}`; }); } /** * Generate a unique string for this Token, returning a key */ registerList(token, displayHint) { return cachedValue(token, LIST_SYMBOL, () => { const key = this.registerStringKey(token, displayHint); return [`${encoding_1.BEGIN_LIST_TOKEN_MARKER}${key}${encoding_1.END_TOKEN_MARKER}`]; }); } /** * Create a unique number representation for this Token and return it */ registerNumber(token) { return cachedValue(token, NUMBER_SYMBOL, () => { return this.registerNumberKey(token); }); } /** * Lookup a token from an encoded value */ tokenFromEncoding(x) { if (token_1.isResolvableObject(x)) { return x; } if (typeof x === 'string') { return this.lookupString(x); } if (Array.isArray(x)) { return this.lookupList(x); } if (token_1.Token.isUnresolved(x)) { return x; } return undefined; } /** * Reverse a string representation into a Token object */ lookupString(s) { const fragments = this.splitString(s); if (fragments.tokens.length > 0 && fragments.length === 1) { return fragments.firstToken; } return undefined; } /** * Reverse a string representation into a Token object */ lookupList(xs) { if (xs.length !== 1) { return undefined; } const str = encoding_1.TokenString.forListToken(xs[0]); const fragments = str.split(this.lookupToken.bind(this)); if (fragments.length === 1) { return fragments.firstToken; } return undefined; } /** * Split a string into literals and Tokens */ splitString(s) { const str = encoding_1.TokenString.forString(s); return str.split(this.lookupToken.bind(this)); } /** * Reverse a number encoding into a Token, or undefined if the number wasn't a Token */ lookupNumberToken(x) { const tokenIndex = encoding_1.extractTokenDouble(x); if (tokenIndex === undefined) { return undefined; } const t = this.numberTokenMap.get(tokenIndex); if (t === undefined) { throw new Error('Encoded representation of unknown number Token found'); } return t; } /** * Find a Token by key. * * This excludes the token markers. */ lookupToken(key) { const token = this.stringTokenMap.get(key); if (!token) { throw new Error(`Unrecognized token key: ${key}`); } return token; } registerStringKey(token, displayHint) { const counter = this.tokenCounter++; const representation = (displayHint || 'TOKEN').replace(new RegExp(`[^${encoding_1.VALID_KEY_CHARS}]`, 'g'), '.'); const key = `${representation}.${counter}`; this.stringTokenMap.set(key, token); return key; } registerNumberKey(token) { const counter = this.tokenCounter++; const dbl = encoding_1.createTokenDouble(counter); // Register in the number map, as well as a string representation of that token // in the string map. this.numberTokenMap.set(counter, token); this.stringTokenMap.set(`${dbl}`, token); return dbl; } } exports.TokenMap = TokenMap; /** * Get a cached value for an object, storing it on the object in a symbol */ function cachedValue(x, sym, prod) { let cached = x[sym]; if (cached === undefined) { cached = prod(); Object.defineProperty(x, sym, { value: cached }); } return cached; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"token-map.js","sourceRoot":"","sources":["token-map.ts"],"names":[],"mappings":"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","sourcesContent":["import { IResolvable } from '../resolvable';\nimport { TokenizedStringFragments } from '../string-fragments';\nimport { isResolvableObject, Token } from '../token';\nimport {\n  BEGIN_LIST_TOKEN_MARKER, BEGIN_STRING_TOKEN_MARKER, createTokenDouble,\n  END_TOKEN_MARKER, extractTokenDouble, TokenString, VALID_KEY_CHARS,\n} from './encoding';\n\nconst glob = global as any;\n\nconst STRING_SYMBOL = Symbol.for('@aws-cdk/core.TokenMap.STRING');\nconst LIST_SYMBOL = Symbol.for('@aws-cdk/core.TokenMap.LIST');\nconst NUMBER_SYMBOL = Symbol.for('@aws-cdk/core.TokenMap.NUMBER');\n\n/**\n * Central place where we keep a mapping from Tokens to their String representation\n *\n * The string representation is used to embed token into strings,\n * and stored to be able to reverse that mapping.\n *\n * All instances of TokenStringMap share the same storage, so that this process\n * works even when different copies of the library are loaded.\n */\nexport class TokenMap {\n  /**\n   * Singleton instance of the token string map\n   */\n  public static instance(): TokenMap {\n    if (!glob.__cdkTokenMap) {\n      glob.__cdkTokenMap = new TokenMap();\n    }\n    return glob.__cdkTokenMap;\n  }\n\n  private readonly stringTokenMap = new Map<string, IResolvable>();\n  private readonly numberTokenMap = new Map<number, IResolvable>();\n\n  /**\n   * Counter to assign unique IDs to tokens\n   *\n   * Start at a random number to prevent people from accidentally taking\n   * dependencies on token values between runs.\n   *\n   * This is most prominent in tests, where people will write:\n   *\n   * ```ts\n   * sha256(JSON.stringify({ ...some structure that can contain tokens ... }))\n   * ```\n   *\n   * This should have been:\n   *\n   * ```ts\n   * sha256(JSON.stringify(stack.resolve({ ...some structure that can contain tokens ... })))\n   * ```\n   *\n   * The hash value is hard to inspect for correctness. It will LOOK consistent\n   * during testing, but will break as soon as someone stringifies another\n   * token before the run.\n   *\n   * By changing the starting number for tokens, we ensure that the hash is almost\n   * guaranteed to be different during a few test runs, so the hashing of unresolved\n   * tokens can be detected.\n   */\n  private tokenCounter = Math.floor(Math.random() * 10);\n\n  /**\n   * Generate a unique string for this Token, returning a key\n   *\n   * Every call for the same Token will produce a new unique string, no\n   * attempt is made to deduplicate. Token objects should cache the\n   * value themselves, if required.\n   *\n   * The token can choose (part of) its own representation string with a\n   * hint. This may be used to produce aesthetically pleasing and\n   * recognizable token representations for humans.\n   */\n  public registerString(token: IResolvable, displayHint?: string): string {\n    return cachedValue(token, STRING_SYMBOL, () => {\n      const key = this.registerStringKey(token, displayHint);\n      return `${BEGIN_STRING_TOKEN_MARKER}${key}${END_TOKEN_MARKER}`;\n    });\n  }\n\n  /**\n   * Generate a unique string for this Token, returning a key\n   */\n  public registerList(token: IResolvable, displayHint?: string): string[] {\n    return cachedValue(token, LIST_SYMBOL, () => {\n      const key = this.registerStringKey(token, displayHint);\n      return [`${BEGIN_LIST_TOKEN_MARKER}${key}${END_TOKEN_MARKER}`];\n    });\n  }\n\n  /**\n   * Create a unique number representation for this Token and return it\n   */\n  public registerNumber(token: IResolvable): number {\n    return cachedValue(token, NUMBER_SYMBOL, () => {\n      return this.registerNumberKey(token);\n    });\n  }\n\n  /**\n   * Lookup a token from an encoded value\n   */\n  public tokenFromEncoding(x: any): IResolvable | undefined {\n    if (isResolvableObject(x)) { return x; }\n    if (typeof x === 'string') { return this.lookupString(x); }\n    if (Array.isArray(x)) { return this.lookupList(x); }\n    if (Token.isUnresolved(x)) { return x; }\n    return undefined;\n  }\n\n  /**\n   * Reverse a string representation into a Token object\n   */\n  public lookupString(s: string): IResolvable | undefined {\n    const fragments = this.splitString(s);\n    if (fragments.tokens.length > 0 && fragments.length === 1) {\n      return fragments.firstToken;\n    }\n    return undefined;\n  }\n\n  /**\n   * Reverse a string representation into a Token object\n   */\n  public lookupList(xs: string[]): IResolvable | undefined {\n    if (xs.length !== 1) { return undefined; }\n    const str = TokenString.forListToken(xs[0]);\n    const fragments = str.split(this.lookupToken.bind(this));\n    if (fragments.length === 1) {\n      return fragments.firstToken;\n    }\n    return undefined;\n  }\n\n  /**\n   * Split a string into literals and Tokens\n   */\n  public splitString(s: string): TokenizedStringFragments {\n    const str = TokenString.forString(s);\n    return str.split(this.lookupToken.bind(this));\n  }\n\n  /**\n   * Reverse a number encoding into a Token, or undefined if the number wasn't a Token\n   */\n  public lookupNumberToken(x: number): IResolvable | undefined {\n    const tokenIndex = extractTokenDouble(x);\n    if (tokenIndex === undefined) { return undefined; }\n    const t = this.numberTokenMap.get(tokenIndex);\n    if (t === undefined) { throw new Error('Encoded representation of unknown number Token found'); }\n    return t;\n  }\n\n  /**\n   * Find a Token by key.\n   *\n   * This excludes the token markers.\n   */\n  public lookupToken(key: string): IResolvable {\n    const token = this.stringTokenMap.get(key);\n    if (!token) {\n      throw new Error(`Unrecognized token key: ${key}`);\n    }\n    return token;\n  }\n\n  private registerStringKey(token: IResolvable, displayHint?: string): string {\n    const counter = this.tokenCounter++;\n    const representation = (displayHint || 'TOKEN').replace(new RegExp(`[^${VALID_KEY_CHARS}]`, 'g'), '.');\n    const key = `${representation}.${counter}`;\n    this.stringTokenMap.set(key, token);\n    return key;\n  }\n\n  private registerNumberKey(token: IResolvable): number {\n    const counter = this.tokenCounter++;\n    const dbl = createTokenDouble(counter);\n    // Register in the number map, as well as a string representation of that token\n    // in the string map.\n    this.numberTokenMap.set(counter, token);\n    this.stringTokenMap.set(`${dbl}`, token);\n    return dbl;\n  }\n}\n\n/**\n * Get a cached value for an object, storing it on the object in a symbol\n */\nfunction cachedValue<A extends object, B>(x: A, sym: symbol, prod: () => B) {\n  let cached = (x as any)[sym as any];\n  if (cached === undefined) {\n    cached = prod();\n    Object.defineProperty(x, sym, { value: cached });\n  }\n  return cached;\n}\n"]}