UNPKG

@aws-cdk/core

Version:

AWS Cloud Development Kit Core Library

169 lines 21.2 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.RememberingTokenResolver = exports.findTokens = exports.resolve = void 0; const resolvable_1 = require("../resolvable"); const encoding_1 = require("./encoding"); const token_map_1 = require("./token-map"); // This file should not be exported to consumers, resolving should happen through Construct.resolve() const tokenMap = token_map_1.TokenMap.instance(); /** * Resolves an object by evaluating all tokens and removing any undefined or empty objects or arrays. * Values can only be primitives, arrays or tokens. Other objects (i.e. with methods) will be rejected. * * @param obj The object to resolve. * @param prefix Prefix key path components for diagnostics. */ function resolve(obj, options) { const prefix = options.prefix || []; const pathName = '/' + prefix.join('/'); /** * Make a new resolution context */ function makeContext(appendPath) { const newPrefix = appendPath !== undefined ? prefix.concat([appendPath]) : options.prefix; let postProcessor; const context = { preparing: options.preparing, scope: options.scope, registerPostProcessor(pp) { postProcessor = pp; }, resolve(x) { return resolve(x, { ...options, prefix: newPrefix }); }, }; return [context, { postProcess(x) { return postProcessor ? postProcessor.postProcess(x, context) : x; } }]; } // protect against cyclic references by limiting depth. if (prefix.length > 200) { throw new Error('Unable to resolve object tree with circular reference. Path: ' + pathName); } // // undefined // if (typeof (obj) === 'undefined') { return undefined; } // // null // if (obj === null) { return null; } // // functions - not supported (only tokens are supported) // if (typeof (obj) === 'function') { throw new Error(`Trying to resolve a non-data object. Only token are supported for lazy evaluation. Path: ${pathName}. Object: ${obj}`); } // // string - potentially replace all stringified Tokens // if (typeof (obj) === 'string') { const str = encoding_1.TokenString.forString(obj); if (str.test()) { const fragments = str.split(tokenMap.lookupToken.bind(tokenMap)); return options.resolver.resolveString(fragments, makeContext()[0]); } return obj; } // // number - potentially decode Tokenized number // if (typeof (obj) === 'number') { return resolveNumberToken(obj, makeContext()[0]); } // // primitives - as-is // if (typeof (obj) !== 'object' || obj instanceof Date) { return obj; } // // arrays - resolve all values, remove undefined and remove empty arrays // if (Array.isArray(obj)) { if (encoding_1.containsListTokenElement(obj)) { return options.resolver.resolveList(obj, makeContext()[0]); } const arr = obj .map((x, i) => makeContext(`${i}`)[0].resolve(x)) .filter(x => typeof (x) !== 'undefined'); return arr; } // // tokens - invoke 'resolve' and continue to resolve recursively // if (encoding_1.unresolved(obj)) { const [context, postProcessor] = makeContext(); return options.resolver.resolveToken(obj, context, postProcessor); } // // objects - deep-resolve all values // // Must not be a Construct at this point, otherwise you probably made a typo // mistake somewhere and resolve will get into an infinite loop recursing into // child.parent <---> parent.children if (isConstruct(obj)) { throw new Error('Trying to resolve() a Construct at ' + pathName); } const result = {}; for (const key of Object.keys(obj)) { const resolvedKey = makeContext()[0].resolve(key); if (typeof (resolvedKey) !== 'string') { // eslint-disable-next-line max-len throw new Error(`"${key}" is used as the key in a map so must resolve to a string, but it resolves to: ${JSON.stringify(resolvedKey)}. Consider using "CfnJson" to delay resolution to deployment-time`); } const value = makeContext(key)[0].resolve(obj[key]); // skip undefined if (typeof (value) === 'undefined') { continue; } result[resolvedKey] = value; } return result; } exports.resolve = resolve; /** * Find all Tokens that are used in the given structure */ function findTokens(scope, fn) { const resolver = new RememberingTokenResolver(new resolvable_1.StringConcat()); resolve(fn(), { scope, prefix: [], resolver, preparing: true }); return resolver.tokens; } exports.findTokens = findTokens; /** * Remember all Tokens encountered while resolving */ class RememberingTokenResolver extends resolvable_1.DefaultTokenResolver { constructor() { super(...arguments); this.tokensSeen = new Set(); } resolveToken(t, context, postProcessor) { this.tokensSeen.add(t); return super.resolveToken(t, context, postProcessor); } resolveString(s, context) { const ret = super.resolveString(s, context); return ret; } get tokens() { return Array.from(this.tokensSeen); } } exports.RememberingTokenResolver = RememberingTokenResolver; /** * Determine whether an object is a Construct * * Not in 'construct.ts' because that would lead to a dependency cycle via 'uniqueid.ts', * and this is a best-effort protection against a common programming mistake anyway. */ function isConstruct(x) { return x._children !== undefined && x._metadata !== undefined; } function resolveNumberToken(x, context) { const token = token_map_1.TokenMap.instance().lookupNumberToken(x); if (token === undefined) { return x; } return context.resolve(token); } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"resolve.js","sourceRoot":"","sources":["resolve.ts"],"names":[],"mappings":"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","sourcesContent":["import { IConstruct } from 'constructs';\nimport { DefaultTokenResolver, IPostProcessor, IResolvable, IResolveContext, ITokenResolver, StringConcat } from '../resolvable';\nimport { TokenizedStringFragments } from '../string-fragments';\nimport { containsListTokenElement, TokenString, unresolved } from './encoding';\nimport { TokenMap } from './token-map';\n\n// v2 - leave this as a separate section so it reduces merge conflicts when compat is removed\n// eslint-disable-next-line import/order\nimport { IConstruct as ICoreConstruct } from '../construct-compat';\n\n// This file should not be exported to consumers, resolving should happen through Construct.resolve()\n\nconst tokenMap = TokenMap.instance();\n\n/**\n * Options to the resolve() operation\n *\n * NOT the same as the ResolveContext; ResolveContext is exposed to Token\n * implementors and resolution hooks, whereas this struct is just to bundle\n * a number of things that would otherwise be arguments to resolve() in a\n * readable way.\n */\nexport interface IResolveOptions {\n  scope: IConstruct;\n  preparing: boolean;\n  resolver: ITokenResolver;\n  prefix?: string[];\n}\n\n/**\n * Resolves an object by evaluating all tokens and removing any undefined or empty objects or arrays.\n * Values can only be primitives, arrays or tokens. Other objects (i.e. with methods) will be rejected.\n *\n * @param obj The object to resolve.\n * @param prefix Prefix key path components for diagnostics.\n */\nexport function resolve(obj: any, options: IResolveOptions): any {\n  const prefix = options.prefix || [];\n  const pathName = '/' + prefix.join('/');\n\n  /**\n   * Make a new resolution context\n   */\n  function makeContext(appendPath?: string): [IResolveContext, IPostProcessor] {\n    const newPrefix = appendPath !== undefined ? prefix.concat([appendPath]) : options.prefix;\n\n    let postProcessor: IPostProcessor | undefined;\n\n    const context: IResolveContext = {\n      preparing: options.preparing,\n      scope: options.scope as ICoreConstruct,\n      registerPostProcessor(pp) { postProcessor = pp; },\n      resolve(x: any) { return resolve(x, { ...options, prefix: newPrefix }); },\n    };\n\n    return [context, { postProcess(x) { return postProcessor ? postProcessor.postProcess(x, context) : x; } }];\n  }\n\n  // protect against cyclic references by limiting depth.\n  if (prefix.length > 200) {\n    throw new Error('Unable to resolve object tree with circular reference. Path: ' + pathName);\n  }\n\n  //\n  // undefined\n  //\n\n  if (typeof(obj) === 'undefined') {\n    return undefined;\n  }\n\n  //\n  // null\n  //\n\n  if (obj === null) {\n    return null;\n  }\n\n  //\n  // functions - not supported (only tokens are supported)\n  //\n\n  if (typeof(obj) === 'function') {\n    throw new Error(`Trying to resolve a non-data object. Only token are supported for lazy evaluation. Path: ${pathName}. Object: ${obj}`);\n  }\n\n  //\n  // string - potentially replace all stringified Tokens\n  //\n  if (typeof(obj) === 'string') {\n    const str = TokenString.forString(obj);\n    if (str.test()) {\n      const fragments = str.split(tokenMap.lookupToken.bind(tokenMap));\n      return options.resolver.resolveString(fragments, makeContext()[0]);\n    }\n    return obj;\n  }\n\n  //\n  // number - potentially decode Tokenized number\n  //\n  if (typeof(obj) === 'number') {\n    return resolveNumberToken(obj, makeContext()[0]);\n  }\n\n  //\n  // primitives - as-is\n  //\n\n  if (typeof(obj) !== 'object' || obj instanceof Date) {\n    return obj;\n  }\n\n  //\n  // arrays - resolve all values, remove undefined and remove empty arrays\n  //\n\n  if (Array.isArray(obj)) {\n    if (containsListTokenElement(obj)) {\n      return options.resolver.resolveList(obj, makeContext()[0]);\n    }\n\n    const arr = obj\n      .map((x, i) => makeContext(`${i}`)[0].resolve(x))\n      .filter(x => typeof(x) !== 'undefined');\n\n    return arr;\n  }\n\n  //\n  // tokens - invoke 'resolve' and continue to resolve recursively\n  //\n\n  if (unresolved(obj)) {\n    const [context, postProcessor] = makeContext();\n    return options.resolver.resolveToken(obj, context, postProcessor);\n  }\n\n  //\n  // objects - deep-resolve all values\n  //\n\n  // Must not be a Construct at this point, otherwise you probably made a typo\n  // mistake somewhere and resolve will get into an infinite loop recursing into\n  // child.parent <---> parent.children\n  if (isConstruct(obj)) {\n    throw new Error('Trying to resolve() a Construct at ' + pathName);\n  }\n\n  const result: any = { };\n  for (const key of Object.keys(obj)) {\n    const resolvedKey = makeContext()[0].resolve(key);\n    if (typeof(resolvedKey) !== 'string') {\n      // eslint-disable-next-line max-len\n      throw new Error(`\"${key}\" is used as the key in a map so must resolve to a string, but it resolves to: ${JSON.stringify(resolvedKey)}. Consider using \"CfnJson\" to delay resolution to deployment-time`);\n    }\n\n    const value = makeContext(key)[0].resolve(obj[key]);\n\n    // skip undefined\n    if (typeof(value) === 'undefined') {\n      continue;\n    }\n\n    result[resolvedKey] = value;\n  }\n\n  return result;\n}\n\n/**\n * Find all Tokens that are used in the given structure\n */\nexport function findTokens(scope: IConstruct, fn: () => any): IResolvable[] {\n  const resolver = new RememberingTokenResolver(new StringConcat());\n\n  resolve(fn(), { scope, prefix: [], resolver, preparing: true });\n\n  return resolver.tokens;\n}\n\n/**\n * Remember all Tokens encountered while resolving\n */\nexport class RememberingTokenResolver extends DefaultTokenResolver {\n  private readonly tokensSeen = new Set<IResolvable>();\n\n  public resolveToken(t: IResolvable, context: IResolveContext, postProcessor: IPostProcessor) {\n    this.tokensSeen.add(t);\n    return super.resolveToken(t, context, postProcessor);\n  }\n\n  public resolveString(s: TokenizedStringFragments, context: IResolveContext) {\n    const ret = super.resolveString(s, context);\n    return ret;\n  }\n\n  public get tokens(): IResolvable[] {\n    return Array.from(this.tokensSeen);\n  }\n}\n\n/**\n * Determine whether an object is a Construct\n *\n * Not in 'construct.ts' because that would lead to a dependency cycle via 'uniqueid.ts',\n * and this is a best-effort protection against a common programming mistake anyway.\n */\nfunction isConstruct(x: any): boolean {\n  return x._children !== undefined && x._metadata !== undefined;\n}\n\nfunction resolveNumberToken(x: number, context: IResolveContext): any {\n  const token = TokenMap.instance().lookupNumberToken(x);\n  if (token === undefined) { return x; }\n  return context.resolve(token);\n}\n"]}