@aws-cdk/core
Version:
AWS Cloud Development Kit Core Library
167 lines • 20.6 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", { value: true });
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') {
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)}`);
}
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":";;AACA,8CAAiI;AAEjI,yCAA+E;AAC/E,2CAAuC;AAEvC,qGAAqG;AAErG,MAAM,QAAQ,GAAG,oBAAQ,CAAC,QAAQ,EAAE,CAAC;AAiBrC;;;;;;GAMG;AACH,SAAgB,OAAO,CAAC,GAAQ,EAAE,OAAwB;IACxD,MAAM,MAAM,GAAG,OAAO,CAAC,MAAM,IAAI,EAAE,CAAC;IACpC,MAAM,QAAQ,GAAG,GAAG,GAAG,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;IAExC;;OAEG;IACH,SAAS,WAAW,CAAC,UAAmB;QACtC,MAAM,SAAS,GAAG,UAAU,KAAK,SAAS,CAAC,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,MAAM,CAAC;QAE1F,IAAI,aAAyC,CAAC;QAE9C,MAAM,OAAO,GAAoB;YAC/B,SAAS,EAAE,OAAO,CAAC,SAAS;YAC5B,KAAK,EAAE,OAAO,CAAC,KAAK;YACpB,qBAAqB,CAAC,EAAE,IAAI,aAAa,GAAG,EAAE,CAAC,CAAC,CAAC;YACjD,OAAO,CAAC,CAAM,IAAI,OAAO,OAAO,CAAC,CAAC,EAAE,EAAE,GAAG,OAAO,EAAE,MAAM,EAAE,SAAS,EAAE,CAAC,CAAC,CAAC,CAAC;SAC1E,CAAC;QAEF,OAAO,CAAC,OAAO,EAAE,EAAE,WAAW,CAAC,CAAC,IAAI,OAAO,aAAa,CAAC,CAAC,CAAC,aAAa,CAAC,WAAW,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,EAAC,CAAC,CAAC;IAC5G,CAAC;IAED,uDAAuD;IACvD,IAAI,MAAM,CAAC,MAAM,GAAG,GAAG,EAAE;QACvB,MAAM,IAAI,KAAK,CAAC,+DAA+D,GAAG,QAAQ,CAAC,CAAC;KAC7F;IAED,EAAE;IACF,YAAY;IACZ,EAAE;IAEF,IAAI,OAAM,CAAC,GAAG,CAAC,KAAK,WAAW,EAAE;QAC/B,OAAO,SAAS,CAAC;KAClB;IAED,EAAE;IACF,OAAO;IACP,EAAE;IAEF,IAAI,GAAG,KAAK,IAAI,EAAE;QAChB,OAAO,IAAI,CAAC;KACb;IAED,EAAE;IACF,wDAAwD;IACxD,EAAE;IAEF,IAAI,OAAM,CAAC,GAAG,CAAC,KAAK,UAAU,EAAE;QAC9B,MAAM,IAAI,KAAK,CAAC,4FAA4F,QAAQ,aAAa,GAAG,EAAE,CAAC,CAAC;KACzI;IAED,EAAE;IACF,sDAAsD;IACtD,EAAE;IACF,IAAI,OAAM,CAAC,GAAG,CAAC,KAAK,QAAQ,EAAE;QAC5B,MAAM,GAAG,GAAG,sBAAW,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC;QACvC,IAAI,GAAG,CAAC,IAAI,EAAE,EAAE;YACd,MAAM,SAAS,GAAG,GAAG,CAAC,KAAK,CAAC,QAAQ,CAAC,WAAW,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC;YACjE,OAAO,OAAO,CAAC,QAAQ,CAAC,aAAa,CAAC,SAAS,EAAE,WAAW,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;SACpE;QACD,OAAO,GAAG,CAAC;KACZ;IAED,EAAE;IACF,+CAA+C;IAC/C,EAAE;IACF,IAAI,OAAM,CAAC,GAAG,CAAC,KAAK,QAAQ,EAAE;QAC5B,OAAO,kBAAkB,CAAC,GAAG,EAAE,WAAW,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;KAClD;IAED,EAAE;IACF,qBAAqB;IACrB,EAAE;IAEF,IAAI,OAAM,CAAC,GAAG,CAAC,KAAK,QAAQ,IAAI,GAAG,YAAY,IAAI,EAAE;QACnD,OAAO,GAAG,CAAC;KACZ;IAED,EAAE;IACF,wEAAwE;IACxE,EAAE;IAEF,IAAI,KAAK,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE;QACtB,IAAI,mCAAwB,CAAC,GAAG,CAAC,EAAE;YACjC,OAAO,OAAO,CAAC,QAAQ,CAAC,WAAW,CAAC,GAAG,EAAE,WAAW,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;SAC5D;QAED,MAAM,GAAG,GAAG,GAAG;aACZ,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,WAAW,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC;aAChD,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,OAAM,CAAC,CAAC,CAAC,KAAK,WAAW,CAAC,CAAC;QAE1C,OAAO,GAAG,CAAC;KACZ;IAED,EAAE;IACF,gEAAgE;IAChE,EAAE;IAEF,IAAI,qBAAU,CAAC,GAAG,CAAC,EAAE;QACnB,MAAM,CAAC,OAAO,EAAE,aAAa,CAAC,GAAG,WAAW,EAAE,CAAC;QAC/C,OAAO,OAAO,CAAC,QAAQ,CAAC,YAAY,CAAC,GAAG,EAAE,OAAO,EAAE,aAAa,CAAC,CAAC;KACnE;IAED,EAAE;IACF,oCAAoC;IACpC,EAAE;IAEF,4EAA4E;IAC5E,8EAA8E;IAC9E,qCAAqC;IACrC,IAAI,WAAW,CAAC,GAAG,CAAC,EAAE;QACpB,MAAM,IAAI,KAAK,CAAC,qCAAqC,GAAG,QAAQ,CAAC,CAAC;KACnE;IAED,MAAM,MAAM,GAAQ,EAAG,CAAC;IACxB,KAAK,MAAM,GAAG,IAAI,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE;QAClC,MAAM,WAAW,GAAG,WAAW,EAAE,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;QAClD,IAAI,OAAM,CAAC,WAAW,CAAC,KAAK,QAAQ,EAAE;YACpC,MAAM,IAAI,KAAK,CAAC,IAAI,GAAG,kFAAkF,IAAI,CAAC,SAAS,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC;SACzI;QAED,MAAM,KAAK,GAAG,WAAW,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC;QAEpD,iBAAiB;QACjB,IAAI,OAAM,CAAC,KAAK,CAAC,KAAK,WAAW,EAAE;YACjC,SAAS;SACV;QAED,MAAM,CAAC,WAAW,CAAC,GAAG,KAAK,CAAC;KAC7B;IAED,OAAO,MAAM,CAAC;AAChB,CAAC;AApID,0BAoIC;AAED;;GAEG;AACH,SAAgB,UAAU,CAAC,KAAiB,EAAE,EAAa;IACzD,MAAM,QAAQ,GAAG,IAAI,wBAAwB,CAAC,IAAI,yBAAY,EAAE,CAAC,CAAC;IAElE,OAAO,CAAC,EAAE,EAAE,EAAE,EAAE,KAAK,EAAE,MAAM,EAAE,EAAE,EAAE,QAAQ,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;IAEhE,OAAO,QAAQ,CAAC,MAAM,CAAC;AACzB,CAAC;AAND,gCAMC;AAED;;GAEG;AACH,MAAa,wBAAyB,SAAQ,iCAAoB;IAAlE;;QACmB,eAAU,GAAG,IAAI,GAAG,EAAe,CAAC;IAevD,CAAC;IAbQ,YAAY,CAAC,CAAc,EAAE,OAAwB,EAAE,aAA6B;QACzF,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;QACvB,OAAO,KAAK,CAAC,YAAY,CAAC,CAAC,EAAE,OAAO,EAAE,aAAa,CAAC,CAAC;IACvD,CAAC;IAEM,aAAa,CAAC,CAA2B,EAAE,OAAwB;QACxE,MAAM,GAAG,GAAG,KAAK,CAAC,aAAa,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC;QAC5C,OAAO,GAAG,CAAC;IACb,CAAC;IAED,IAAW,MAAM;QACf,OAAO,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;IACrC,CAAC;CACF;AAhBD,4DAgBC;AAED;;;;;GAKG;AACH,SAAS,WAAW,CAAC,CAAM;IACzB,OAAO,CAAC,CAAC,SAAS,KAAK,SAAS,IAAI,CAAC,CAAC,SAAS,KAAK,SAAS,CAAC;AAChE,CAAC;AAED,SAAS,kBAAkB,CAAC,CAAS,EAAE,OAAwB;IAC7D,MAAM,KAAK,GAAG,oBAAQ,CAAC,QAAQ,EAAE,CAAC,iBAAiB,CAAC,CAAC,CAAC,CAAC;IACvD,IAAI,KAAK,KAAK,SAAS,EAAE;QAAE,OAAO,CAAC,CAAC;KAAE;IACtC,OAAO,OAAO,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;AAChC,CAAC","sourcesContent":["import { IConstruct } from '../construct-compat';\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// 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,\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      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)}`);\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"]}