Socket
Socket
Sign inDemoInstall

@aws-cdk/cdk

Package Overview
Dependencies
Maintainers
4
Versions
42
Alerts
File Explorer

Advanced tools

Socket logo

Install Socket

Detect and block malicious and high-risk dependencies

Install

@aws-cdk/cdk - npm Package Compare versions

Comparing version 0.13.0 to 0.14.0

3

lib/app.js

@@ -43,2 +43,3 @@ "use strict";

const result = {
version: cxapi.PROTO_RESPONSE_VERSION,
stacks: this.synthesizeStacks(Object.keys(this.stacks)),

@@ -185,2 +186,2 @@ runtime: this.collectRuntimeInformation()

}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"app.js","sourceRoot":"","sources":["app.ts"],"names":[],"mappings":"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","sourcesContent":["import cxapi = require('@aws-cdk/cx-api');\nimport fs = require('fs');\nimport path = require('path');\nimport { Stack } from './cloudformation/stack';\nimport { Construct, MetadataEntry, PATH_SEP, Root } from './core/construct';\nimport { resolve } from './core/tokens';\n\n/**\n * Represents a CDK program.\n */\nexport class App extends Root {\n  /**\n   * Initializes a CDK application.\n   * @param request Optional toolkit request (e.g. for tests)\n   */\n  constructor() {\n    super();\n    this.loadContext();\n  }\n\n  private get stacks() {\n    const out: { [name: string]: Stack } = { };\n    for (const child of this.children) {\n      if (!(child instanceof Stack)) {\n        throw new Error(`The child ${child.toString()} of Program must be a Stack`);\n      }\n\n      out[child.id] = child as Stack;\n    }\n    return out;\n  }\n\n  /**\n   * Runs the program. Output is written to output directory as specified in the request.\n   */\n  public run(): void {\n    const outdir = process.env[cxapi.OUTDIR_ENV];\n    if (!outdir) {\n      process.stderr.write(`ERROR: The environment variable \"${cxapi.OUTDIR_ENV}\" is not defined\\n`);\n      process.stderr.write('AWS CDK Toolkit (>= 0.11.0) is required in order to interact with this program.\\n');\n      process.exit(1);\n      return;\n    }\n\n    const result: cxapi.SynthesizeResponse = {\n      stacks: this.synthesizeStacks(Object.keys(this.stacks)),\n      runtime: this.collectRuntimeInformation()\n    };\n\n    const outfile = path.join(outdir, cxapi.OUTFILE_NAME);\n    fs.writeFileSync(outfile, JSON.stringify(result, undefined, 2));\n  }\n\n  /**\n   * Synthesize and validate a single stack\n   * @param stackName The name of the stack to synthesize\n   */\n  public synthesizeStack(stackName: string): cxapi.SynthesizedStack {\n    const stack = this.getStack(stackName);\n\n    // first, validate this stack and stop if there are errors.\n    const errors = stack.validateTree();\n    if (errors.length > 0) {\n      const errorList = errors.map(e => `[${e.source.path}] ${e.message}`).join('\\n  ');\n      throw new Error(`Stack validation failed with the following errors:\\n  ${errorList}`);\n    }\n\n    const account = stack.env.account || 'unknown-account';\n    const region = stack.env.region || 'unknown-region';\n\n    const environment: cxapi.Environment = {\n      name: `${account}/${region}`,\n      account,\n      region\n    };\n\n    const missing = Object.keys(stack.missingContext).length ? stack.missingContext : undefined;\n    return {\n      name: stack.id,\n      environment,\n      missing,\n      template: stack.toCloudFormation(),\n      metadata: this.collectMetadata(stack)\n    };\n  }\n\n  /**\n   * Synthesizes multiple stacks\n   */\n  public synthesizeStacks(stackNames: string[]): cxapi.SynthesizedStack[] {\n    const ret: cxapi.SynthesizedStack[] = [];\n    for (const stackName of stackNames) {\n      ret.push(this.synthesizeStack(stackName));\n    }\n    return ret;\n  }\n\n  /**\n   * Returns metadata for all constructs in the stack.\n   */\n  public collectMetadata(stack: Stack) {\n    const output: { [id: string]: MetadataEntry[] } = { };\n\n    visit(stack);\n\n    // add app-level metadata under \".\"\n    if (this.metadata.length > 0) {\n      output[PATH_SEP] = this.metadata;\n    }\n\n    return output;\n\n    function visit(node: Construct) {\n      if (node.metadata.length > 0) {\n        // Make the path absolute\n        output[PATH_SEP + node.path] = node.metadata.map(md => resolve(md) as MetadataEntry);\n      }\n\n      for (const child of node.children) {\n        visit(child);\n      }\n    }\n  }\n\n  private collectRuntimeInformation(): cxapi.AppRuntime {\n    const libraries: { [name: string]: string } = {};\n\n    for (const fileName of Object.keys(require.cache)) {\n      const pkg = findNpmPackage(fileName);\n      if (pkg && !pkg.private) {\n        libraries[pkg.name] = pkg.version;\n      }\n    }\n\n    return { libraries };\n  }\n\n  private getStack(stackname: string) {\n    if (stackname == null) {\n      throw new Error('Stack name must be defined');\n    }\n\n    const stack = this.stacks[stackname];\n    if (!stack) {\n      throw new Error(`Cannot find stack ${stackname}`);\n    }\n    return stack;\n  }\n\n  private loadContext() {\n    const contextJson = process.env[cxapi.CONTEXT_ENV];\n    const context = !contextJson ? { } : JSON.parse(contextJson);\n    for (const key of Object.keys(context)) {\n      this.setContext(key, context[key]);\n    }\n  }\n}\n\n/**\n * Determines which NPM module a given loaded javascript file is from.\n *\n * The only infromation that is available locally is a list of Javascript files,\n * and every source file is associated with a search path to resolve the further\n * ``require`` calls made from there, which includes its own directory on disk,\n * and parent directories - for example:\n *\n * [ '...repo/packages/aws-cdk-resources/lib/cfn/node_modules',\n *   '...repo/packages/aws-cdk-resources/lib/node_modules',\n *   '...repo/packages/aws-cdk-resources/node_modules',\n *   '...repo/packages/node_modules',\n *   // etc...\n * ]\n *\n * We are looking for ``package.json`` that is anywhere in the tree, except it's\n * in the parent directory, not in the ``node_modules`` directory. For this\n * reason, we strip the ``/node_modules`` suffix off each path and use regular\n * module resolution to obtain a reference to ``package.json``.\n *\n * @param fileName a javascript file name.\n * @returns the NPM module infos (aka ``package.json`` contents), or\n *      ``undefined`` if the lookup was unsuccessful.\n */\nfunction findNpmPackage(fileName: string): { name: string, version: string, private?: boolean } | undefined {\n  const mod = require.cache[fileName];\n  const paths = mod.paths.map(stripNodeModules);\n\n  try {\n    const packagePath = require.resolve('package.json', { paths });\n    return require(packagePath);\n  } catch (e) {\n    return undefined;\n  }\n\n  /**\n   * @param s a path.\n   * @returns ``s`` with any terminating ``/node_modules``\n   *      (or ``\\\\node_modules``) stripped off.)\n   */\n  function stripNodeModules(s: string): string {\n    if (s.endsWith('/node_modules') || s.endsWith('\\\\node_modules')) {\n      // /node_modules is 13 characters\n      return s.substr(0, s.length - 13);\n    }\n    return s;\n  }\n}\n"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"app.js","sourceRoot":"","sources":["app.ts"],"names":[],"mappings":"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","sourcesContent":["import cxapi = require('@aws-cdk/cx-api');\nimport fs = require('fs');\nimport path = require('path');\nimport { Stack } from './cloudformation/stack';\nimport { Construct, MetadataEntry, PATH_SEP, Root } from './core/construct';\nimport { resolve } from './core/tokens';\n\n/**\n * Represents a CDK program.\n */\nexport class App extends Root {\n  /**\n   * Initializes a CDK application.\n   * @param request Optional toolkit request (e.g. for tests)\n   */\n  constructor() {\n    super();\n    this.loadContext();\n  }\n\n  private get stacks() {\n    const out: { [name: string]: Stack } = { };\n    for (const child of this.children) {\n      if (!(child instanceof Stack)) {\n        throw new Error(`The child ${child.toString()} of Program must be a Stack`);\n      }\n\n      out[child.id] = child as Stack;\n    }\n    return out;\n  }\n\n  /**\n   * Runs the program. Output is written to output directory as specified in the request.\n   */\n  public run(): void {\n    const outdir = process.env[cxapi.OUTDIR_ENV];\n    if (!outdir) {\n      process.stderr.write(`ERROR: The environment variable \"${cxapi.OUTDIR_ENV}\" is not defined\\n`);\n      process.stderr.write('AWS CDK Toolkit (>= 0.11.0) is required in order to interact with this program.\\n');\n      process.exit(1);\n      return;\n    }\n\n    const result: cxapi.SynthesizeResponse = {\n      version: cxapi.PROTO_RESPONSE_VERSION,\n      stacks: this.synthesizeStacks(Object.keys(this.stacks)),\n      runtime: this.collectRuntimeInformation()\n    };\n\n    const outfile = path.join(outdir, cxapi.OUTFILE_NAME);\n    fs.writeFileSync(outfile, JSON.stringify(result, undefined, 2));\n  }\n\n  /**\n   * Synthesize and validate a single stack\n   * @param stackName The name of the stack to synthesize\n   */\n  public synthesizeStack(stackName: string): cxapi.SynthesizedStack {\n    const stack = this.getStack(stackName);\n\n    // first, validate this stack and stop if there are errors.\n    const errors = stack.validateTree();\n    if (errors.length > 0) {\n      const errorList = errors.map(e => `[${e.source.path}] ${e.message}`).join('\\n  ');\n      throw new Error(`Stack validation failed with the following errors:\\n  ${errorList}`);\n    }\n\n    const account = stack.env.account || 'unknown-account';\n    const region = stack.env.region || 'unknown-region';\n\n    const environment: cxapi.Environment = {\n      name: `${account}/${region}`,\n      account,\n      region\n    };\n\n    const missing = Object.keys(stack.missingContext).length ? stack.missingContext : undefined;\n    return {\n      name: stack.id,\n      environment,\n      missing,\n      template: stack.toCloudFormation(),\n      metadata: this.collectMetadata(stack)\n    };\n  }\n\n  /**\n   * Synthesizes multiple stacks\n   */\n  public synthesizeStacks(stackNames: string[]): cxapi.SynthesizedStack[] {\n    const ret: cxapi.SynthesizedStack[] = [];\n    for (const stackName of stackNames) {\n      ret.push(this.synthesizeStack(stackName));\n    }\n    return ret;\n  }\n\n  /**\n   * Returns metadata for all constructs in the stack.\n   */\n  public collectMetadata(stack: Stack) {\n    const output: { [id: string]: MetadataEntry[] } = { };\n\n    visit(stack);\n\n    // add app-level metadata under \".\"\n    if (this.metadata.length > 0) {\n      output[PATH_SEP] = this.metadata;\n    }\n\n    return output;\n\n    function visit(node: Construct) {\n      if (node.metadata.length > 0) {\n        // Make the path absolute\n        output[PATH_SEP + node.path] = node.metadata.map(md => resolve(md) as MetadataEntry);\n      }\n\n      for (const child of node.children) {\n        visit(child);\n      }\n    }\n  }\n\n  private collectRuntimeInformation(): cxapi.AppRuntime {\n    const libraries: { [name: string]: string } = {};\n\n    for (const fileName of Object.keys(require.cache)) {\n      const pkg = findNpmPackage(fileName);\n      if (pkg && !pkg.private) {\n        libraries[pkg.name] = pkg.version;\n      }\n    }\n\n    return { libraries };\n  }\n\n  private getStack(stackname: string) {\n    if (stackname == null) {\n      throw new Error('Stack name must be defined');\n    }\n\n    const stack = this.stacks[stackname];\n    if (!stack) {\n      throw new Error(`Cannot find stack ${stackname}`);\n    }\n    return stack;\n  }\n\n  private loadContext() {\n    const contextJson = process.env[cxapi.CONTEXT_ENV];\n    const context = !contextJson ? { } : JSON.parse(contextJson);\n    for (const key of Object.keys(context)) {\n      this.setContext(key, context[key]);\n    }\n  }\n}\n\n/**\n * Determines which NPM module a given loaded javascript file is from.\n *\n * The only infromation that is available locally is a list of Javascript files,\n * and every source file is associated with a search path to resolve the further\n * ``require`` calls made from there, which includes its own directory on disk,\n * and parent directories - for example:\n *\n * [ '...repo/packages/aws-cdk-resources/lib/cfn/node_modules',\n *   '...repo/packages/aws-cdk-resources/lib/node_modules',\n *   '...repo/packages/aws-cdk-resources/node_modules',\n *   '...repo/packages/node_modules',\n *   // etc...\n * ]\n *\n * We are looking for ``package.json`` that is anywhere in the tree, except it's\n * in the parent directory, not in the ``node_modules`` directory. For this\n * reason, we strip the ``/node_modules`` suffix off each path and use regular\n * module resolution to obtain a reference to ``package.json``.\n *\n * @param fileName a javascript file name.\n * @returns the NPM module infos (aka ``package.json`` contents), or\n *      ``undefined`` if the lookup was unsuccessful.\n */\nfunction findNpmPackage(fileName: string): { name: string, version: string, private?: boolean } | undefined {\n  const mod = require.cache[fileName];\n  const paths = mod.paths.map(stripNodeModules);\n\n  try {\n    const packagePath = require.resolve('package.json', { paths });\n    return require(packagePath);\n  } catch (e) {\n    return undefined;\n  }\n\n  /**\n   * @param s a path.\n   * @returns ``s`` with any terminating ``/node_modules``\n   *      (or ``\\\\node_modules``) stripped off.)\n   */\n  function stripNodeModules(s: string): string {\n    if (s.endsWith('/node_modules') || s.endsWith('\\\\node_modules')) {\n      // /node_modules is 13 characters\n      return s.substr(0, s.length - 13);\n    }\n    return s;\n  }\n}\n"]}

@@ -68,2 +68,5 @@ import { CloudFormationToken } from './cloudformation-token';

export declare class FnJoin extends Fn {
private readonly delimiter;
private readonly listOfValues;
private _resolvedValues?;
/**

@@ -76,2 +79,9 @@ * Creates an ``Fn::Join`` function.

constructor(delimiter: string, listOfValues: any[]);
resolve(): any;
/**
* Optimization: if an Fn::Join is nested in another one and they share the same delimiter, then flatten it up. Also,
* if two concatenated elements are literal strings (not tokens), then pre-concatenate them with the delimiter, to
* generate shorter output.
*/
private resolveValues;
}

@@ -82,3 +92,2 @@ /**

export declare class FnConcat extends FnJoin {
private readonly listOfValues;
/**

@@ -85,0 +94,0 @@ * Creates an ``Fn::Join`` function with an empty delimiter.

"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
const tokens_1 = require("../core/tokens");
const cloudformation_token_1 = require("./cloudformation-token");

@@ -96,4 +97,48 @@ // tslint:disable:max-line-length

}
super('Fn::Join', [delimiter, listOfValues]);
// Passing the values as a token, optimization requires resolving stringified tokens, we should be deferred until
// this token is itself being resolved.
super('Fn::Join', [delimiter, new tokens_1.Token(() => this.resolveValues())]);
this.delimiter = delimiter;
this.listOfValues = listOfValues;
}
resolve() {
if (this.resolveValues().length === 1) {
return this.resolveValues()[0];
}
return super.resolve();
}
/**
* Optimization: if an Fn::Join is nested in another one and they share the same delimiter, then flatten it up. Also,
* if two concatenated elements are literal strings (not tokens), then pre-concatenate them with the delimiter, to
* generate shorter output.
*/
resolveValues() {
if (this._resolvedValues) {
return this._resolvedValues;
}
const resolvedValues = [...this.listOfValues.map(e => tokens_1.resolve(e))];
let i = 0;
while (i < resolvedValues.length) {
const el = resolvedValues[i];
if (isFnJoinIntrinsicWithSameDelimiter.call(this, el)) {
resolvedValues.splice(i, 1, ...el['Fn::Join'][1]);
}
else if (i > 0 && isPlainString(resolvedValues[i - 1]) && isPlainString(resolvedValues[i])) {
resolvedValues[i - 1] += this.delimiter + resolvedValues[i];
resolvedValues.splice(i, 1);
}
else {
i += 1;
}
}
return this._resolvedValues = resolvedValues;
function isFnJoinIntrinsicWithSameDelimiter(obj) {
return cloudformation_token_1.isIntrinsic(obj)
&& Object.keys(obj)[0] === 'Fn::Join'
&& obj['Fn::Join'][0] === this.delimiter;
}
function isPlainString(obj) {
return typeof obj === 'string' && !tokens_1.unresolved(obj);
}
}
}

@@ -110,26 +155,3 @@ exports.FnJoin = FnJoin;

constructor(...listOfValues) {
// Optimization: if any of the input arguments is also a FnConcat,
// splice their list of values into the current FnConcat. 'instanceof'
// can fail, but we do not depend depend on this for correctness.
//
// Do the same for resolved intrinsics, so we can detect this
// happening both at Token as well as at CloudFormation level.
let i = 0;
while (i < listOfValues.length) {
const el = listOfValues[i];
if (el instanceof FnConcat) {
listOfValues.splice(i, 1, ...el.listOfValues);
i += el.listOfValues.length;
}
else if (isConcatIntrinsic(el)) {
const values = concatIntrinsicValues(el);
listOfValues.splice(i, 1, ...values);
i += values;
}
else {
i++;
}
}
super('', listOfValues);
this.listOfValues = listOfValues;
}

@@ -139,14 +161,2 @@ }

/**
* Return whether the given object represents a CloudFormation intrinsic that is the result of a FnConcat resolution
*/
function isConcatIntrinsic(x) {
return cloudformation_token_1.isIntrinsic(x) && Object.keys(x)[0] === 'Fn::Join' && x['Fn::Join'][0] === '';
}
/**
* Return the concatted values of the concat intrinsic
*/
function concatIntrinsicValues(x) {
return x['Fn::Join'][1];
}
/**
* The intrinsic function ``Fn::Select`` returns a single object from a list of objects by index.

@@ -418,2 +428,2 @@ */

exports.FnValueOfAll = FnValueOfAll;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fn.js","sourceRoot":"","sources":["fn.ts"],"names":[],"mappings":"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","sourcesContent":["import { CloudFormationToken, isIntrinsic } from './cloudformation-token';\n// tslint:disable:max-line-length\n\n/**\n * CloudFormation intrinsic functions.\n * http://docs.aws.amazon.com/AWSCloudFormation/latest/UserGuide/intrinsic-function-reference.html\n */\nexport class Fn extends CloudFormationToken {\n  constructor(name: string, value: any) {\n    super(() => ({ [name]: value }));\n  }\n}\n\n/**\n * The intrinsic function ``Fn::FindInMap`` returns the value corresponding to keys in a two-level\n * map that is declared in the Mappings section.\n */\nexport class FnFindInMap extends Fn {\n  /**\n   * Creates an ``Fn::FindInMap`` function.\n   * @param mapName The logical name of a mapping declared in the Mappings section that contains the keys and values.\n   * @param topLevelKey The top-level key name. Its value is a list of key-value pairs.\n   * @param secondLevelKey The second-level key name, which is set to one of the keys from the list assigned to TopLevelKey.\n   */\n  constructor(mapName: string, topLevelKey: any, secondLevelKey: any) {\n    super('Fn::FindInMap', [ mapName, topLevelKey, secondLevelKey ]);\n  }\n}\n\n/**\n * The ``Fn::GetAtt`` intrinsic function returns the value of an attribute from a resource in the template.\n */\nexport class FnGetAtt extends Fn {\n  /**\n   * Creates a ``Fn::GetAtt`` function.\n   * @param logicalNameOfResource The logical name (also called logical ID) of the resource that contains the attribute that you want.\n   * @param attributeName The name of the resource-specific attribute whose value you want. See the resource's reference page for details about the attributes available for that resource type.\n   */\n  constructor(logicalNameOfResource: string, attributeName: string) {\n    super('Fn::GetAtt', [ logicalNameOfResource, attributeName ]);\n  }\n}\n\n/**\n * The intrinsic function ``Fn::GetAZs`` returns an array that lists Availability Zones for a\n * specified region. Because customers have access to different Availability Zones, the intrinsic\n * function ``Fn::GetAZs`` enables template authors to write templates that adapt to the calling\n * user's access. That way you don't have to hard-code a full list of Availability Zones for a\n * specified region.\n */\nexport class FnGetAZs extends Fn {\n  /**\n   * Creates an ``Fn::GetAZs`` function.\n   * @param region The name of the region for which you want to get the Availability Zones.\n   *         You can use the AWS::Region pseudo parameter to specify the region in\n   *         which the stack is created. Specifying an empty string is equivalent to\n   *         specifying AWS::Region.\n   */\n  constructor(region?: string) {\n    super('Fn::GetAZs', region || '');\n  }\n}\n\n/**\n * The intrinsic function ``Fn::ImportValue`` returns the value of an output exported by another stack.\n * You typically use this function to create cross-stack references. In the following example\n * template snippets, Stack A exports VPC security group values and Stack B imports them.\n */\nexport class FnImportValue extends Fn {\n  /**\n   * Creates an ``Fn::ImportValue`` function.\n   * @param sharedValueToImport The stack output value that you want to import.\n   */\n  constructor(sharedValueToImport: string) {\n    super('Fn::ImportValue', sharedValueToImport);\n  }\n}\n\n/**\n * The intrinsic function ``Fn::Join`` appends a set of values into a single value, separated by\n * the specified delimiter. If a delimiter is the empty string, the set of values are concatenated\n * with no delimiter.\n */\nexport class FnJoin extends Fn {\n  /**\n   * Creates an ``Fn::Join`` function.\n   * @param delimiter The value you want to occur between fragments. The delimiter will occur between fragments only.\n   *          It will not terminate the final value.\n   * @param listOfValues The list of values you want combined.\n   */\n  constructor(delimiter: string, listOfValues: any[]) {\n    if (listOfValues.length === 0) {\n      throw new Error(`FnJoin requires at least one value to be provided`);\n    }\n    super('Fn::Join', [ delimiter, listOfValues ]);\n  }\n}\n\n/**\n * Alias for ``FnJoin('', listOfValues)``.\n */\nexport class FnConcat extends FnJoin {\n  private readonly listOfValues: any[];\n\n  /**\n   * Creates an ``Fn::Join`` function with an empty delimiter.\n   * @param listOfValues The list of values to concatenate.\n   */\n  constructor(...listOfValues: any[]) {\n    // Optimization: if any of the input arguments is also a FnConcat,\n    // splice their list of values into the current FnConcat. 'instanceof'\n    // can fail, but we do not depend depend on this for correctness.\n    //\n    // Do the same for resolved intrinsics, so we can detect this\n    // happening both at Token as well as at CloudFormation level.\n\n    let i = 0;\n    while (i < listOfValues.length) {\n      const el = listOfValues[i];\n      if (el instanceof FnConcat) {\n        listOfValues.splice(i, 1, ...el.listOfValues);\n        i += el.listOfValues.length;\n      } else if (isConcatIntrinsic(el)) {\n        const values = concatIntrinsicValues(el);\n        listOfValues.splice(i, 1, ...values);\n        i += values;\n      } else {\n        i++;\n      }\n    }\n\n    super('', listOfValues);\n    this.listOfValues = listOfValues;\n  }\n}\n\n/**\n * Return whether the given object represents a CloudFormation intrinsic that is the result of a FnConcat resolution\n */\nfunction isConcatIntrinsic(x: any) {\n  return isIntrinsic(x) && Object.keys(x)[0] === 'Fn::Join' && x['Fn::Join'][0] === '';\n}\n\n/**\n * Return the concatted values of the concat intrinsic\n */\nfunction concatIntrinsicValues(x: any) {\n  return x['Fn::Join'][1];\n}\n\n/**\n * The intrinsic function ``Fn::Select`` returns a single object from a list of objects by index.\n */\nexport class FnSelect extends Fn {\n  /**\n   * Creates an ``Fn::Select`` function.\n   * @param index The index of the object to retrieve. This must be a value from zero to N-1, where N represents the number of elements in the array.\n   * @param array The list of objects to select from. This list must not be null, nor can it have null entries.\n   */\n  constructor(index: number, array: any) {\n    super('Fn::Select', [ index, array ]);\n  }\n}\n\n/**\n * To split a string into a list of string values so that you can select an element from the\n * resulting string list, use the ``Fn::Split`` intrinsic function. Specify the location of splits\n * with a delimiter, such as , (a comma). After you split a string, use the ``Fn::Select`` function\n * to pick a specific element.\n */\nexport class FnSplit extends Fn {\n  /**\n   * Create an ``Fn::Split`` function.\n   * @param delimiter A string value that determines where the source string is divided.\n   * @param source The string value that you want to split.\n   */\n  constructor(delimiter: string, source: any) {\n    super('Fn::Split', [ delimiter, source ]);\n  }\n}\n\n/**\n * The intrinsic function ``Fn::Sub`` substitutes variables in an input string with values that\n * you specify. In your templates, you can use this function to construct commands or outputs\n * that include values that aren't available until you create or update a stack.\n */\nexport class FnSub extends Fn {\n  /**\n   * Creates an ``Fn::Sub`` function.\n   * @param body A string with variables that AWS CloudFormation substitutes with their\n   *       associated values at runtime. Write variables as ${MyVarName}. Variables\n   *       can be template parameter names, resource logical IDs, resource attributes,\n   *       or a variable in a key-value map. If you specify only template parameter names,\n   *       resource logical IDs, and resource attributes, don't specify a key-value map.\n   * @param variables The name of a variable that you included in the String parameter.\n   *          The value that AWS CloudFormation substitutes for the associated variable name at runtime.\n   */\n  constructor(body: string, variables?: { [key: string]: any }) {\n    super('Fn::Sub', variables ? [body, variables] : body);\n  }\n}\n\n/**\n * The intrinsic function ``Fn::Base64`` returns the Base64 representation of the input string.\n * This function is typically used to pass encoded data to Amazon EC2 instances by way of\n * the UserData property.\n */\nexport class FnBase64 extends Fn {\n\n  /**\n   * Creates an ``Fn::Base64`` function.\n   * @param data The string value you want to convert to Base64.\n   */\n  constructor(data: any) {\n    super('Fn::Base64', data);\n  }\n}\n\n/**\n * The intrinsic function ``Fn::Cidr`` returns the specified Cidr address block.\n */\nexport class FnCidr extends Fn {\n  /**\n   * Creates an ``Fn::Cidr`` function.\n   * @param ipBlock  The user-specified default Cidr address block.\n   * @param count  The number of subnets' Cidr block wanted. Count can be 1 to 256.\n   * @param sizeMask The digit covered in the subnet.\n   */\n  constructor(ipBlock: any, count: any, sizeMask?: any) {\n    if (count < 1 || count > 256) {\n      throw new Error(`Fn::Cidr's count attribute must be betwen 1 and 256, ${count} was provided.`);\n    }\n    super('Fn::Cidr', [ipBlock, count, sizeMask]);\n  }\n}\n\n/**\n * You can use intrinsic functions, such as ``Fn::If``, ``Fn::Equals``, and ``Fn::Not``, to conditionally\n * create stack resources. These conditions are evaluated based on input parameters that you\n * declare when you create or update a stack. After you define all your conditions, you can\n * associate them with resources or resource properties in the Resources and Outputs sections\n * of a template.\n *\n * You define all conditions in the Conditions section of a template except for ``Fn::If`` conditions.\n * You can use the ``Fn::If`` condition in the metadata attribute, update policy attribute, and property\n * values in the Resources section and Outputs sections of a template.\n *\n * You might use conditions when you want to reuse a template that can create resources in different\n * contexts, such as a test environment versus a production environment. In your template, you can\n * add an EnvironmentType input parameter, which accepts either prod or test as inputs. For the\n * production environment, you might include Amazon EC2 instances with certain capabilities;\n * however, for the test environment, you want to use less capabilities to save costs. With\n * conditions, you can define which resources are created and how they're configured for each\n * environment type.\n */\nexport class FnCondition extends Fn {\n\n}\n\n/**\n * Returns true if all the specified conditions evaluate to true, or returns false if any one\n *  of the conditions evaluates to false. ``Fn::And`` acts as an AND operator. The minimum number of\n * conditions that you can include is 2, and the maximum is 10.\n */\nexport class FnAnd extends FnCondition {\n  constructor(...condition: FnCondition[]) {\n    super('Fn::And', condition);\n  }\n}\n\n/**\n * Compares if two values are equal. Returns true if the two values are equal or false\n * if they aren't.\n */\nexport class FnEquals extends FnCondition {\n  /**\n   * Creates an ``Fn::Equals`` condition function.\n   * @param lhs A value of any type that you want to compare.\n   * @param rhs A value of any type that you want to compare.\n   */\n  constructor(lhs: any, rhs: any) {\n    super('Fn::Equals', [ lhs, rhs ]);\n  }\n}\n\n/**\n * Returns one value if the specified condition evaluates to true and another value if the\n * specified condition evaluates to false. Currently, AWS CloudFormation supports the ``Fn::If``\n * intrinsic function in the metadata attribute, update policy attribute, and property values\n * in the Resources section and Outputs sections of a template. You can use the AWS::NoValue\n * pseudo parameter as a return value to remove the corresponding property.\n */\nexport class FnIf extends FnCondition {\n  /**\n   * Creates an ``Fn::If`` condition function.\n   * @param condition A reference to a condition in the Conditions section. Use the condition's name to reference it.\n   * @param valueIfTrue A value to be returned if the specified condition evaluates to true.\n   * @param valueIfFalse A value to be returned if the specified condition evaluates to false.\n   */\n  constructor(condition: string, valueIfTrue: any, valueIfFalse: any) {\n    super('Fn::If', [ condition, valueIfTrue, valueIfFalse ]);\n  }\n}\n\n/**\n * Returns true for a condition that evaluates to false or returns false for a condition that evaluates to true.\n * ``Fn::Not`` acts as a NOT operator.\n */\nexport class FnNot extends FnCondition {\n  /**\n   * Creates an ``Fn::Not`` condition function.\n   * @param condition A condition such as ``Fn::Equals`` that evaluates to true or false.\n   */\n  constructor(condition: FnCondition) {\n    super('Fn::Not', [ condition ]);\n  }\n}\n\n/**\n * Returns true if any one of the specified conditions evaluate to true, or returns false if\n * all of the conditions evaluates to false. ``Fn::Or`` acts as an OR operator. The minimum number\n * of conditions that you can include is 2, and the maximum is 10.\n */\nexport class FnOr extends FnCondition {\n  /**\n   * Creates an ``Fn::Or`` condition function.\n   * @param condition A condition that evaluates to true or false.\n   */\n  constructor(...condition: FnCondition[]) {\n    super('Fn::Or', condition);\n  }\n}\n\n/**\n * Returns true if a specified string matches at least one value in a list of strings.\n */\nexport class FnContains extends FnCondition {\n  /**\n   * Creates an ``Fn::Contains`` function.\n   * @param listOfStrings A list of strings, such as \"A\", \"B\", \"C\".\n   * @param value A string, such as \"A\", that you want to compare against a list of strings.\n   */\n  constructor(listOfStrings: any, value: string) {\n    super('Fn::Contains', [ listOfStrings, value ]);\n  }\n}\n\n/**\n * Returns true if a specified string matches all values in a list.\n */\nexport class FnEachMemberEquals extends FnCondition {\n  /**\n   * Creates an ``Fn::EachMemberEquals`` function.\n   * @param listOfStrings A list of strings, such as \"A\", \"B\", \"C\".\n   * @param value A string, such as \"A\", that you want to compare against a list of strings.\n   */\n  constructor(listOfStrings: any, value: string) {\n    super('Fn::EachMemberEquals', [ listOfStrings, value ]);\n  }\n}\n\n/**\n * Returns true if each member in a list of strings matches at least one value in a second\n * list of strings.\n */\nexport class FnEachMemberIn extends FnCondition {\n  /**\n   * Creates an ``Fn::EachMemberIn`` function.\n   * @param stringsToCheck A list of strings, such as \"A\", \"B\", \"C\". AWS CloudFormation checks whether each member in the strings_to_check parameter is in the strings_to_match parameter.\n   * @param stringsToMatch A list of strings, such as \"A\", \"B\", \"C\". Each member in the strings_to_match parameter is compared against the members of the strings_to_check parameter.\n   */\n  constructor(stringsToCheck: any, stringsToMatch: any) {\n    super('Fn::EachMemberIn', [ [stringsToCheck], stringsToMatch ]);\n  }\n}\n\n/**\n * Returns all values for a specified parameter type.\n */\nexport class FnRefAll extends FnCondition {\n  /**\n   * Creates an ``Fn::RefAll`` function.\n   * @param parameterType An AWS-specific parameter type, such as AWS::EC2::SecurityGroup::Id or\n   *            AWS::EC2::VPC::Id. For more information, see Parameters in the AWS\n   *            CloudFormation User Guide.\n   */\n  constructor(parameterType: string) {\n    super('Fn::RefAll', parameterType);\n  }\n}\n\n/**\n * Returns an attribute value or list of values for a specific parameter and attribute.\n */\nexport class FnValueOf extends FnCondition {\n  /**\n   * Creates an ``Fn::ValueOf`` function.\n   * @param parameterOrLogicalId The name of a parameter for which you want to retrieve attribute values. The parameter must be declared in the Parameters section of the template.\n   * @param attribute The name of an attribute from which you want to retrieve a value.\n   */\n  constructor(parameterOrLogicalId: string, attribute: string) {\n    super('Fn::ValueOf', [ parameterOrLogicalId, attribute ]);\n  }\n}\n\n/**\n * Returns a list of all attribute values for a given parameter type and attribute.\n */\nexport class FnValueOfAll extends FnCondition {\n  /**\n   * Creates an ``Fn::ValueOfAll`` function.\n   * @param parameterType An AWS-specific parameter type, such as AWS::EC2::SecurityGroup::Id or AWS::EC2::VPC::Id. For more information, see Parameters in the AWS CloudFormation User Guide.\n   * @param attribute The name of an attribute from which you want to retrieve a value. For more information about attributes, see Supported Attributes.\n   */\n  constructor(parameterType: string, attribute: string) {\n    super('Fn::ValueOfAll', [ parameterType, attribute ]);\n  }\n}\n"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fn.js","sourceRoot":"","sources":["fn.ts"],"names":[],"mappings":"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","sourcesContent":["import { resolve, Token, unresolved } from '../core/tokens';\nimport { CloudFormationToken, isIntrinsic } from './cloudformation-token';\n// tslint:disable:max-line-length\n\n/**\n * CloudFormation intrinsic functions.\n * http://docs.aws.amazon.com/AWSCloudFormation/latest/UserGuide/intrinsic-function-reference.html\n */\nexport class Fn extends CloudFormationToken {\n  constructor(name: string, value: any) {\n    super(() => ({ [name]: value }));\n  }\n}\n\n/**\n * The intrinsic function ``Fn::FindInMap`` returns the value corresponding to keys in a two-level\n * map that is declared in the Mappings section.\n */\nexport class FnFindInMap extends Fn {\n  /**\n   * Creates an ``Fn::FindInMap`` function.\n   * @param mapName The logical name of a mapping declared in the Mappings section that contains the keys and values.\n   * @param topLevelKey The top-level key name. Its value is a list of key-value pairs.\n   * @param secondLevelKey The second-level key name, which is set to one of the keys from the list assigned to TopLevelKey.\n   */\n  constructor(mapName: string, topLevelKey: any, secondLevelKey: any) {\n    super('Fn::FindInMap', [ mapName, topLevelKey, secondLevelKey ]);\n  }\n}\n\n/**\n * The ``Fn::GetAtt`` intrinsic function returns the value of an attribute from a resource in the template.\n */\nexport class FnGetAtt extends Fn {\n  /**\n   * Creates a ``Fn::GetAtt`` function.\n   * @param logicalNameOfResource The logical name (also called logical ID) of the resource that contains the attribute that you want.\n   * @param attributeName The name of the resource-specific attribute whose value you want. See the resource's reference page for details about the attributes available for that resource type.\n   */\n  constructor(logicalNameOfResource: string, attributeName: string) {\n    super('Fn::GetAtt', [ logicalNameOfResource, attributeName ]);\n  }\n}\n\n/**\n * The intrinsic function ``Fn::GetAZs`` returns an array that lists Availability Zones for a\n * specified region. Because customers have access to different Availability Zones, the intrinsic\n * function ``Fn::GetAZs`` enables template authors to write templates that adapt to the calling\n * user's access. That way you don't have to hard-code a full list of Availability Zones for a\n * specified region.\n */\nexport class FnGetAZs extends Fn {\n  /**\n   * Creates an ``Fn::GetAZs`` function.\n   * @param region The name of the region for which you want to get the Availability Zones.\n   *         You can use the AWS::Region pseudo parameter to specify the region in\n   *         which the stack is created. Specifying an empty string is equivalent to\n   *         specifying AWS::Region.\n   */\n  constructor(region?: string) {\n    super('Fn::GetAZs', region || '');\n  }\n}\n\n/**\n * The intrinsic function ``Fn::ImportValue`` returns the value of an output exported by another stack.\n * You typically use this function to create cross-stack references. In the following example\n * template snippets, Stack A exports VPC security group values and Stack B imports them.\n */\nexport class FnImportValue extends Fn {\n  /**\n   * Creates an ``Fn::ImportValue`` function.\n   * @param sharedValueToImport The stack output value that you want to import.\n   */\n  constructor(sharedValueToImport: string) {\n    super('Fn::ImportValue', sharedValueToImport);\n  }\n}\n\n/**\n * The intrinsic function ``Fn::Join`` appends a set of values into a single value, separated by\n * the specified delimiter. If a delimiter is the empty string, the set of values are concatenated\n * with no delimiter.\n */\nexport class FnJoin extends Fn {\n  private readonly delimiter: string;\n  private readonly listOfValues: any[];\n  // Cache for the result of resolveValues() - since it otherwise would be computed several times\n  private _resolvedValues?: any[];\n\n  /**\n   * Creates an ``Fn::Join`` function.\n   * @param delimiter The value you want to occur between fragments. The delimiter will occur between fragments only.\n   *          It will not terminate the final value.\n   * @param listOfValues The list of values you want combined.\n   */\n  constructor(delimiter: string, listOfValues: any[]) {\n    if (listOfValues.length === 0) {\n      throw new Error(`FnJoin requires at least one value to be provided`);\n    }\n    // Passing the values as a token, optimization requires resolving stringified tokens, we should be deferred until\n    // this token is itself being resolved.\n    super('Fn::Join', [ delimiter, new Token(() => this.resolveValues()) ]);\n    this.delimiter = delimiter;\n    this.listOfValues = listOfValues;\n  }\n\n  public resolve(): any {\n    if (this.resolveValues().length === 1) {\n      return this.resolveValues()[0];\n    }\n    return super.resolve();\n  }\n\n  /**\n   * Optimization: if an Fn::Join is nested in another one and they share the same delimiter, then flatten it up. Also,\n   * if two concatenated elements are literal strings (not tokens), then pre-concatenate them with the delimiter, to\n   * generate shorter output.\n   */\n  private resolveValues() {\n    if (this._resolvedValues) { return this._resolvedValues; }\n\n    const resolvedValues = [...this.listOfValues.map(e => resolve(e))];\n    let i = 0;\n    while (i < resolvedValues.length) {\n      const el = resolvedValues[i];\n      if (isFnJoinIntrinsicWithSameDelimiter.call(this, el)) {\n        resolvedValues.splice(i, 1, ...el['Fn::Join'][1]);\n      } else if (i > 0 && isPlainString(resolvedValues[i - 1]) && isPlainString(resolvedValues[i])) {\n        resolvedValues[i - 1] += this.delimiter + resolvedValues[i];\n        resolvedValues.splice(i, 1);\n      } else {\n        i += 1;\n      }\n    }\n\n    return this._resolvedValues = resolvedValues;\n\n    function isFnJoinIntrinsicWithSameDelimiter(this: FnJoin, obj: any): boolean {\n      return isIntrinsic(obj)\n        && Object.keys(obj)[0] === 'Fn::Join'\n        && obj['Fn::Join'][0] === this.delimiter;\n    }\n\n    function isPlainString(obj: any): boolean {\n      return typeof obj === 'string' && !unresolved(obj);\n    }\n  }\n}\n\n/**\n * Alias for ``FnJoin('', listOfValues)``.\n */\nexport class FnConcat extends FnJoin {\n  /**\n   * Creates an ``Fn::Join`` function with an empty delimiter.\n   * @param listOfValues The list of values to concatenate.\n   */\n  constructor(...listOfValues: any[]) {\n    super('', listOfValues);\n  }\n}\n\n/**\n * The intrinsic function ``Fn::Select`` returns a single object from a list of objects by index.\n */\nexport class FnSelect extends Fn {\n  /**\n   * Creates an ``Fn::Select`` function.\n   * @param index The index of the object to retrieve. This must be a value from zero to N-1, where N represents the number of elements in the array.\n   * @param array The list of objects to select from. This list must not be null, nor can it have null entries.\n   */\n  constructor(index: number, array: any) {\n    super('Fn::Select', [ index, array ]);\n  }\n}\n\n/**\n * To split a string into a list of string values so that you can select an element from the\n * resulting string list, use the ``Fn::Split`` intrinsic function. Specify the location of splits\n * with a delimiter, such as , (a comma). After you split a string, use the ``Fn::Select`` function\n * to pick a specific element.\n */\nexport class FnSplit extends Fn {\n  /**\n   * Create an ``Fn::Split`` function.\n   * @param delimiter A string value that determines where the source string is divided.\n   * @param source The string value that you want to split.\n   */\n  constructor(delimiter: string, source: any) {\n    super('Fn::Split', [ delimiter, source ]);\n  }\n}\n\n/**\n * The intrinsic function ``Fn::Sub`` substitutes variables in an input string with values that\n * you specify. In your templates, you can use this function to construct commands or outputs\n * that include values that aren't available until you create or update a stack.\n */\nexport class FnSub extends Fn {\n  /**\n   * Creates an ``Fn::Sub`` function.\n   * @param body A string with variables that AWS CloudFormation substitutes with their\n   *       associated values at runtime. Write variables as ${MyVarName}. Variables\n   *       can be template parameter names, resource logical IDs, resource attributes,\n   *       or a variable in a key-value map. If you specify only template parameter names,\n   *       resource logical IDs, and resource attributes, don't specify a key-value map.\n   * @param variables The name of a variable that you included in the String parameter.\n   *          The value that AWS CloudFormation substitutes for the associated variable name at runtime.\n   */\n  constructor(body: string, variables?: { [key: string]: any }) {\n    super('Fn::Sub', variables ? [body, variables] : body);\n  }\n}\n\n/**\n * The intrinsic function ``Fn::Base64`` returns the Base64 representation of the input string.\n * This function is typically used to pass encoded data to Amazon EC2 instances by way of\n * the UserData property.\n */\nexport class FnBase64 extends Fn {\n\n  /**\n   * Creates an ``Fn::Base64`` function.\n   * @param data The string value you want to convert to Base64.\n   */\n  constructor(data: any) {\n    super('Fn::Base64', data);\n  }\n}\n\n/**\n * The intrinsic function ``Fn::Cidr`` returns the specified Cidr address block.\n */\nexport class FnCidr extends Fn {\n  /**\n   * Creates an ``Fn::Cidr`` function.\n   * @param ipBlock  The user-specified default Cidr address block.\n   * @param count  The number of subnets' Cidr block wanted. Count can be 1 to 256.\n   * @param sizeMask The digit covered in the subnet.\n   */\n  constructor(ipBlock: any, count: any, sizeMask?: any) {\n    if (count < 1 || count > 256) {\n      throw new Error(`Fn::Cidr's count attribute must be betwen 1 and 256, ${count} was provided.`);\n    }\n    super('Fn::Cidr', [ipBlock, count, sizeMask]);\n  }\n}\n\n/**\n * You can use intrinsic functions, such as ``Fn::If``, ``Fn::Equals``, and ``Fn::Not``, to conditionally\n * create stack resources. These conditions are evaluated based on input parameters that you\n * declare when you create or update a stack. After you define all your conditions, you can\n * associate them with resources or resource properties in the Resources and Outputs sections\n * of a template.\n *\n * You define all conditions in the Conditions section of a template except for ``Fn::If`` conditions.\n * You can use the ``Fn::If`` condition in the metadata attribute, update policy attribute, and property\n * values in the Resources section and Outputs sections of a template.\n *\n * You might use conditions when you want to reuse a template that can create resources in different\n * contexts, such as a test environment versus a production environment. In your template, you can\n * add an EnvironmentType input parameter, which accepts either prod or test as inputs. For the\n * production environment, you might include Amazon EC2 instances with certain capabilities;\n * however, for the test environment, you want to use less capabilities to save costs. With\n * conditions, you can define which resources are created and how they're configured for each\n * environment type.\n */\nexport class FnCondition extends Fn {\n\n}\n\n/**\n * Returns true if all the specified conditions evaluate to true, or returns false if any one\n *  of the conditions evaluates to false. ``Fn::And`` acts as an AND operator. The minimum number of\n * conditions that you can include is 2, and the maximum is 10.\n */\nexport class FnAnd extends FnCondition {\n  constructor(...condition: FnCondition[]) {\n    super('Fn::And', condition);\n  }\n}\n\n/**\n * Compares if two values are equal. Returns true if the two values are equal or false\n * if they aren't.\n */\nexport class FnEquals extends FnCondition {\n  /**\n   * Creates an ``Fn::Equals`` condition function.\n   * @param lhs A value of any type that you want to compare.\n   * @param rhs A value of any type that you want to compare.\n   */\n  constructor(lhs: any, rhs: any) {\n    super('Fn::Equals', [ lhs, rhs ]);\n  }\n}\n\n/**\n * Returns one value if the specified condition evaluates to true and another value if the\n * specified condition evaluates to false. Currently, AWS CloudFormation supports the ``Fn::If``\n * intrinsic function in the metadata attribute, update policy attribute, and property values\n * in the Resources section and Outputs sections of a template. You can use the AWS::NoValue\n * pseudo parameter as a return value to remove the corresponding property.\n */\nexport class FnIf extends FnCondition {\n  /**\n   * Creates an ``Fn::If`` condition function.\n   * @param condition A reference to a condition in the Conditions section. Use the condition's name to reference it.\n   * @param valueIfTrue A value to be returned if the specified condition evaluates to true.\n   * @param valueIfFalse A value to be returned if the specified condition evaluates to false.\n   */\n  constructor(condition: string, valueIfTrue: any, valueIfFalse: any) {\n    super('Fn::If', [ condition, valueIfTrue, valueIfFalse ]);\n  }\n}\n\n/**\n * Returns true for a condition that evaluates to false or returns false for a condition that evaluates to true.\n * ``Fn::Not`` acts as a NOT operator.\n */\nexport class FnNot extends FnCondition {\n  /**\n   * Creates an ``Fn::Not`` condition function.\n   * @param condition A condition such as ``Fn::Equals`` that evaluates to true or false.\n   */\n  constructor(condition: FnCondition) {\n    super('Fn::Not', [ condition ]);\n  }\n}\n\n/**\n * Returns true if any one of the specified conditions evaluate to true, or returns false if\n * all of the conditions evaluates to false. ``Fn::Or`` acts as an OR operator. The minimum number\n * of conditions that you can include is 2, and the maximum is 10.\n */\nexport class FnOr extends FnCondition {\n  /**\n   * Creates an ``Fn::Or`` condition function.\n   * @param condition A condition that evaluates to true or false.\n   */\n  constructor(...condition: FnCondition[]) {\n    super('Fn::Or', condition);\n  }\n}\n\n/**\n * Returns true if a specified string matches at least one value in a list of strings.\n */\nexport class FnContains extends FnCondition {\n  /**\n   * Creates an ``Fn::Contains`` function.\n   * @param listOfStrings A list of strings, such as \"A\", \"B\", \"C\".\n   * @param value A string, such as \"A\", that you want to compare against a list of strings.\n   */\n  constructor(listOfStrings: any, value: string) {\n    super('Fn::Contains', [ listOfStrings, value ]);\n  }\n}\n\n/**\n * Returns true if a specified string matches all values in a list.\n */\nexport class FnEachMemberEquals extends FnCondition {\n  /**\n   * Creates an ``Fn::EachMemberEquals`` function.\n   * @param listOfStrings A list of strings, such as \"A\", \"B\", \"C\".\n   * @param value A string, such as \"A\", that you want to compare against a list of strings.\n   */\n  constructor(listOfStrings: any, value: string) {\n    super('Fn::EachMemberEquals', [ listOfStrings, value ]);\n  }\n}\n\n/**\n * Returns true if each member in a list of strings matches at least one value in a second\n * list of strings.\n */\nexport class FnEachMemberIn extends FnCondition {\n  /**\n   * Creates an ``Fn::EachMemberIn`` function.\n   * @param stringsToCheck A list of strings, such as \"A\", \"B\", \"C\". AWS CloudFormation checks whether each member in the strings_to_check parameter is in the strings_to_match parameter.\n   * @param stringsToMatch A list of strings, such as \"A\", \"B\", \"C\". Each member in the strings_to_match parameter is compared against the members of the strings_to_check parameter.\n   */\n  constructor(stringsToCheck: any, stringsToMatch: any) {\n    super('Fn::EachMemberIn', [ [stringsToCheck], stringsToMatch ]);\n  }\n}\n\n/**\n * Returns all values for a specified parameter type.\n */\nexport class FnRefAll extends FnCondition {\n  /**\n   * Creates an ``Fn::RefAll`` function.\n   * @param parameterType An AWS-specific parameter type, such as AWS::EC2::SecurityGroup::Id or\n   *            AWS::EC2::VPC::Id. For more information, see Parameters in the AWS\n   *            CloudFormation User Guide.\n   */\n  constructor(parameterType: string) {\n    super('Fn::RefAll', parameterType);\n  }\n}\n\n/**\n * Returns an attribute value or list of values for a specific parameter and attribute.\n */\nexport class FnValueOf extends FnCondition {\n  /**\n   * Creates an ``Fn::ValueOf`` function.\n   * @param parameterOrLogicalId The name of a parameter for which you want to retrieve attribute values. The parameter must be declared in the Parameters section of the template.\n   * @param attribute The name of an attribute from which you want to retrieve a value.\n   */\n  constructor(parameterOrLogicalId: string, attribute: string) {\n    super('Fn::ValueOf', [ parameterOrLogicalId, attribute ]);\n  }\n}\n\n/**\n * Returns a list of all attribute values for a given parameter type and attribute.\n */\nexport class FnValueOfAll extends FnCondition {\n  /**\n   * Creates an ``Fn::ValueOfAll`` function.\n   * @param parameterType An AWS-specific parameter type, such as AWS::EC2::SecurityGroup::Id or AWS::EC2::VPC::Id. For more information, see Parameters in the AWS CloudFormation User Guide.\n   * @param attribute The name of an attribute from which you want to retrieve a value. For more information about attributes, see Supported Attributes.\n   */\n  constructor(parameterType: string, attribute: string) {\n    super('Fn::ValueOfAll', [ parameterType, attribute ]);\n  }\n}\n"]}

@@ -41,7 +41,2 @@ "use strict";

this.properties = props.properties || {};
// 'name' is a special property included for resource constructs and passed
// as 'name', but we don't want it to be serialized into the template.
if (this.properties.name) {
delete this.properties.name;
}
}

@@ -225,2 +220,2 @@ /**

exports.deepMerge = deepMerge;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"resource.js","sourceRoot":"","sources":["resource.ts"],"names":[],"mappings":"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","sourcesContent":["import { Construct } from '../core/construct';\nimport { capitalizePropertyNames, ignoreEmpty } from '../core/util';\nimport { CloudFormationToken } from './cloudformation-token';\nimport { Condition } from './condition';\nimport { CreationPolicy, DeletionPolicy, UpdatePolicy } from './resource-policy';\nimport { IDependable, Referenceable, StackElement } from './stack';\n\nexport interface ResourceProps {\n  /**\n   * CloudFormation resource type.\n   */\n  type: string;\n\n  /**\n   * CloudFormation properties.\n   */\n  properties?: any;\n}\n\n/**\n * Represents a CloudFormation resource.\n */\nexport class Resource extends Referenceable {\n  /**\n   * A decoration used to create a CloudFormation attribute property.\n   * @param customName Custom name for the attribute (default is the name of the property)\n   * NOTE: we return \"any\" here to satistfy jsii, which doesn't support lambdas.\n   */\n  public static attribute(customName?: string): any {\n    return (prototype: any, key: string) => {\n      const name = customName || key;\n      Object.defineProperty(prototype, key, {\n        get() {\n          return (this as any).getAtt(name);\n        }\n      });\n    };\n  }\n\n  /**\n   * Options for this resource, such as condition, update policy etc.\n   */\n  public readonly options: ResourceOptions = {};\n\n  /**\n   * AWS resource type.\n   */\n  public readonly resourceType: string;\n\n  /**\n   * AWS resource property overrides.\n   *\n   * During synthesis, the method \"renderProperties(this.overrides)\" is called\n   * with this object, and merged on top of the output of\n   * \"renderProperties(this.properties)\".\n   *\n   * Derived classes should expose a strongly-typed version of this object as\n   * a public property called `propertyOverrides`.\n   */\n  protected readonly untypedPropertyOverrides: any = { };\n\n  /**\n   * AWS resource properties.\n   *\n   * This object is rendered via a call to \"renderProperties(this.properties)\".\n   */\n  protected readonly properties: any;\n\n  /**\n   * An object to be merged on top of the entire resource definition.\n   */\n  private readonly rawOverrides: any = { };\n\n  private dependsOn = new Array<IDependable>();\n\n  /**\n   * Creates a resource construct.\n   * @param resourceType The CloudFormation type of this resource (e.g. AWS::DynamoDB::Table)\n   */\n  constructor(parent: Construct, name: string, props: ResourceProps) {\n    super(parent, name);\n\n    if (!props.type) {\n      throw new Error('The `type` property is required');\n    }\n\n    this.resourceType = props.type;\n    this.properties = props.properties || { };\n\n    // 'name' is a special property included for resource constructs and passed\n    // as 'name', but we don't want it to be serialized into the template.\n    if (this.properties.name) {\n      delete this.properties.name;\n    }\n  }\n\n  /**\n   * Returns a token for an runtime attribute of this resource.\n   * Ideally, use generated attribute accessors (e.g. `resource.arn`), but this can be used for future compatibility\n   * in case there is no generated attribute.\n   * @param attributeName The name of the attribute.\n   */\n  public getAtt(attributeName: string) {\n    return new CloudFormationToken({ 'Fn::GetAtt': [this.logicalId, attributeName] }, `${this.logicalId}.${attributeName}`);\n  }\n\n  /**\n   * Adds a dependency on another resource.\n   * @param other The other resource.\n   */\n  public addDependency(...other: IDependable[]) {\n    this.dependsOn.push(...other);\n  }\n\n  /**\n   * Adds an override to the synthesized CloudFormation resource. To add a\n   * property override, either use `addPropertyOverride` or prefix `path` with\n   * \"Properties.\" (i.e. `Properties.TopicName`).\n   *\n   * @param path  The path of the property, you can use dot notation to\n   *        override values in complex types. Any intermdediate keys\n   *        will be created as needed.\n   * @param value The value. Could be primitive or complex.\n   */\n  public addOverride(path: string, value: any) {\n    const parts = path.split('.');\n    let curr: any = this.rawOverrides;\n\n    while (parts.length > 1) {\n      const key = parts.shift()!;\n\n      // if we can't recurse further or the previous value is not an\n      // object overwrite it with an object.\n      const isObject = curr[key] != null && typeof(curr[key]) === 'object' && !Array.isArray(curr[key]);\n      if (!isObject) {\n        curr[key] = { };\n      }\n\n      curr = curr[key];\n    }\n\n    const lastKey = parts.shift()!;\n    curr[lastKey] = value;\n  }\n\n  /**\n   * Syntactic sugar for `addOverride(path, undefined)`.\n   * @param path The path of the value to delete\n   */\n  public addDeletionOverride(path: string) {\n    this.addOverride(path, undefined);\n  }\n\n  /**\n   * Adds an override to a resource property.\n   *\n   * Syntactic sugar for `addOverride(\"Properties.<...>\", value)`.\n   *\n   * @param propertyPath The path of the property\n   * @param value The value\n   */\n  public addPropertyOverride(propertyPath: string, value: any) {\n    this.addOverride(`Properties.${propertyPath}`, value);\n  }\n\n  /**\n   * Adds an override that deletes the value of a property from the resource definition.\n   * @param propertyPath The path to the property.\n   */\n  public addPropertyDeletionOverride(propertyPath: string) {\n    this.addPropertyOverride(propertyPath, undefined);\n  }\n\n  /**\n   * Emits CloudFormation for this resource.\n   */\n  public toCloudFormation(): object {\n    try {\n      // merge property overrides onto properties and then render (and validate).\n      const properties = this.renderProperties(deepMerge(this.properties || { }, this.untypedPropertyOverrides));\n\n      return {\n        Resources: {\n          [this.logicalId]: deepMerge({\n            Type: this.resourceType,\n            Properties: ignoreEmpty(properties),\n            DependsOn: ignoreEmpty(this.renderDependsOn()),\n            CreationPolicy:  capitalizePropertyNames(this.options.creationPolicy),\n            UpdatePolicy: capitalizePropertyNames(this.options.updatePolicy),\n            DeletionPolicy: capitalizePropertyNames(this.options.deletionPolicy),\n            Metadata: ignoreEmpty(this.options.metadata),\n            Condition: this.options.condition && this.options.condition.logicalId\n          }, this.rawOverrides)\n        }\n      };\n    } catch (e) {\n      // Change message\n      e.message = `While synthesizing ${this.path}: ${e.message}`;\n      // Adjust stack trace (make it look like node built it, too...)\n      const creationStack = ['--- resource created at ---', ...this.creationStackTrace].join('\\n  at ');\n      const problemTrace = e.stack.substr(e.stack.indexOf(e.message) + e.message.length);\n      e.stack = `${e.message}\\n  ${creationStack}\\n  --- problem discovered at ---${problemTrace}`;\n      // Re-throw\n      throw e;\n    }\n  }\n\n  protected renderProperties(properties: any): { [key: string]: any } {\n    return properties;\n  }\n\n  private renderDependsOn() {\n    const logicalIDs = new Set<string>();\n    for (const d of this.dependsOn) {\n      addDependency(d);\n    }\n\n    return Array.from(logicalIDs);\n\n    function addDependency(d: IDependable) {\n      d.dependencyElements.forEach(dep => {\n        const logicalId = (dep as StackElement).logicalId;\n        if (logicalId) {\n          logicalIDs.add(logicalId);\n        }\n      });\n\n      // break if dependencyElements include only 'd', which means we reached a terminal.\n      if (d.dependencyElements.length === 1 && d.dependencyElements[0] === d) {\n        return;\n      } else {\n        d.dependencyElements.forEach(dep => addDependency(dep));\n      }\n    }\n  }\n}\n\nexport interface ResourceOptions {\n  /**\n   * A condition to associate with this resource. This means that only if the condition evaluates to 'true' when the stack\n   * is deployed, the resource will be included. This is provided to allow CDK projects to produce legacy templates, but noramlly\n   * there is no need to use it in CDK projects.\n   */\n  condition?: Condition;\n\n  /**\n   * Associate the CreationPolicy attribute with a resource to prevent its status from reaching create complete until\n   * AWS CloudFormation receives a specified number of success signals or the timeout period is exceeded. To signal a\n   * resource, you can use the cfn-signal helper script or SignalResource API. AWS CloudFormation publishes valid signals\n   * to the stack events so that you track the number of signals sent.\n   */\n  creationPolicy?: CreationPolicy;\n\n  /**\n   * With the DeletionPolicy attribute you can preserve or (in some cases) backup a resource when its stack is deleted.\n   * You specify a DeletionPolicy attribute for each resource that you want to control. If a resource has no DeletionPolicy\n   * attribute, AWS CloudFormation deletes the resource by default. Note that this capability also applies to update operations\n   * that lead to resources being removed.\n   */\n  deletionPolicy?: DeletionPolicy;\n\n  /**\n   * Use the UpdatePolicy attribute to specify how AWS CloudFormation handles updates to the AWS::AutoScaling::AutoScalingGroup\n   * resource. AWS CloudFormation invokes one of three update policies depending on the type of change you make or whether a\n   * scheduled action is associated with the Auto Scaling group.\n   */\n  updatePolicy?: UpdatePolicy;\n\n  /**\n   * Metadata associated with the CloudFormation resource. This is not the same as the construct metadata which can be added\n   * using construct.addMetadata(), but would not appear in the CloudFormation template automatically.\n   */\n  metadata?: { [key: string]: any };\n}\n\n/**\n * Merges `source` into `target`, overriding any existing values.\n * `null`s will cause a value to be deleted.\n */\nexport function deepMerge(target: any, source: any) {\n  if (typeof(source) !== 'object' || typeof(target) !== 'object') {\n    throw new Error(`Invalid usage. Both source (${JSON.stringify(source)}) and target (${JSON.stringify(target)}) must be objects`);\n  }\n\n  for (const key of Object.keys(source)) {\n    const value = source[key];\n    if (typeof(value) === 'object' && value != null && !Array.isArray(value)) {\n      // if the value at the target is not an object, override it with an\n      // object so we can continue the recursion\n      if (typeof(target[key]) !== 'object') {\n        target[key] = { };\n      }\n\n      deepMerge(target[key], value);\n\n      // if the result of the merge is an empty object, it's because the\n      // eventual value we assigned is `undefined`, and there are no\n      // sibling concrete values alongside, so we can delete this tree.\n      const output = target[key];\n      if (typeof(output) === 'object' && Object.keys(output).length === 0) {\n        delete target[key];\n      }\n    } else if (value === undefined) {\n      delete target[key];\n    } else {\n      target[key] = value;\n    }\n  }\n\n  return target;\n}\n"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"resource.js","sourceRoot":"","sources":["resource.ts"],"names":[],"mappings":"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","sourcesContent":["import { Construct } from '../core/construct';\nimport { capitalizePropertyNames, ignoreEmpty } from '../core/util';\nimport { CloudFormationToken } from './cloudformation-token';\nimport { Condition } from './condition';\nimport { CreationPolicy, DeletionPolicy, UpdatePolicy } from './resource-policy';\nimport { IDependable, Referenceable, StackElement } from './stack';\n\nexport interface ResourceProps {\n  /**\n   * CloudFormation resource type.\n   */\n  type: string;\n\n  /**\n   * CloudFormation properties.\n   */\n  properties?: any;\n}\n\n/**\n * Represents a CloudFormation resource.\n */\nexport class Resource extends Referenceable {\n  /**\n   * A decoration used to create a CloudFormation attribute property.\n   * @param customName Custom name for the attribute (default is the name of the property)\n   * NOTE: we return \"any\" here to satistfy jsii, which doesn't support lambdas.\n   */\n  public static attribute(customName?: string): any {\n    return (prototype: any, key: string) => {\n      const name = customName || key;\n      Object.defineProperty(prototype, key, {\n        get() {\n          return (this as any).getAtt(name);\n        }\n      });\n    };\n  }\n\n  /**\n   * Options for this resource, such as condition, update policy etc.\n   */\n  public readonly options: ResourceOptions = {};\n\n  /**\n   * AWS resource type.\n   */\n  public readonly resourceType: string;\n\n  /**\n   * AWS resource property overrides.\n   *\n   * During synthesis, the method \"renderProperties(this.overrides)\" is called\n   * with this object, and merged on top of the output of\n   * \"renderProperties(this.properties)\".\n   *\n   * Derived classes should expose a strongly-typed version of this object as\n   * a public property called `propertyOverrides`.\n   */\n  protected readonly untypedPropertyOverrides: any = { };\n\n  /**\n   * AWS resource properties.\n   *\n   * This object is rendered via a call to \"renderProperties(this.properties)\".\n   */\n  protected readonly properties: any;\n\n  /**\n   * An object to be merged on top of the entire resource definition.\n   */\n  private readonly rawOverrides: any = { };\n\n  private dependsOn = new Array<IDependable>();\n\n  /**\n   * Creates a resource construct.\n   * @param resourceType The CloudFormation type of this resource (e.g. AWS::DynamoDB::Table)\n   */\n  constructor(parent: Construct, name: string, props: ResourceProps) {\n    super(parent, name);\n\n    if (!props.type) {\n      throw new Error('The `type` property is required');\n    }\n\n    this.resourceType = props.type;\n    this.properties = props.properties || { };\n  }\n\n  /**\n   * Returns a token for an runtime attribute of this resource.\n   * Ideally, use generated attribute accessors (e.g. `resource.arn`), but this can be used for future compatibility\n   * in case there is no generated attribute.\n   * @param attributeName The name of the attribute.\n   */\n  public getAtt(attributeName: string) {\n    return new CloudFormationToken({ 'Fn::GetAtt': [this.logicalId, attributeName] }, `${this.logicalId}.${attributeName}`);\n  }\n\n  /**\n   * Adds a dependency on another resource.\n   * @param other The other resource.\n   */\n  public addDependency(...other: IDependable[]) {\n    this.dependsOn.push(...other);\n  }\n\n  /**\n   * Adds an override to the synthesized CloudFormation resource. To add a\n   * property override, either use `addPropertyOverride` or prefix `path` with\n   * \"Properties.\" (i.e. `Properties.TopicName`).\n   *\n   * @param path  The path of the property, you can use dot notation to\n   *        override values in complex types. Any intermdediate keys\n   *        will be created as needed.\n   * @param value The value. Could be primitive or complex.\n   */\n  public addOverride(path: string, value: any) {\n    const parts = path.split('.');\n    let curr: any = this.rawOverrides;\n\n    while (parts.length > 1) {\n      const key = parts.shift()!;\n\n      // if we can't recurse further or the previous value is not an\n      // object overwrite it with an object.\n      const isObject = curr[key] != null && typeof(curr[key]) === 'object' && !Array.isArray(curr[key]);\n      if (!isObject) {\n        curr[key] = { };\n      }\n\n      curr = curr[key];\n    }\n\n    const lastKey = parts.shift()!;\n    curr[lastKey] = value;\n  }\n\n  /**\n   * Syntactic sugar for `addOverride(path, undefined)`.\n   * @param path The path of the value to delete\n   */\n  public addDeletionOverride(path: string) {\n    this.addOverride(path, undefined);\n  }\n\n  /**\n   * Adds an override to a resource property.\n   *\n   * Syntactic sugar for `addOverride(\"Properties.<...>\", value)`.\n   *\n   * @param propertyPath The path of the property\n   * @param value The value\n   */\n  public addPropertyOverride(propertyPath: string, value: any) {\n    this.addOverride(`Properties.${propertyPath}`, value);\n  }\n\n  /**\n   * Adds an override that deletes the value of a property from the resource definition.\n   * @param propertyPath The path to the property.\n   */\n  public addPropertyDeletionOverride(propertyPath: string) {\n    this.addPropertyOverride(propertyPath, undefined);\n  }\n\n  /**\n   * Emits CloudFormation for this resource.\n   */\n  public toCloudFormation(): object {\n    try {\n      // merge property overrides onto properties and then render (and validate).\n      const properties = this.renderProperties(deepMerge(this.properties || { }, this.untypedPropertyOverrides));\n\n      return {\n        Resources: {\n          [this.logicalId]: deepMerge({\n            Type: this.resourceType,\n            Properties: ignoreEmpty(properties),\n            DependsOn: ignoreEmpty(this.renderDependsOn()),\n            CreationPolicy:  capitalizePropertyNames(this.options.creationPolicy),\n            UpdatePolicy: capitalizePropertyNames(this.options.updatePolicy),\n            DeletionPolicy: capitalizePropertyNames(this.options.deletionPolicy),\n            Metadata: ignoreEmpty(this.options.metadata),\n            Condition: this.options.condition && this.options.condition.logicalId\n          }, this.rawOverrides)\n        }\n      };\n    } catch (e) {\n      // Change message\n      e.message = `While synthesizing ${this.path}: ${e.message}`;\n      // Adjust stack trace (make it look like node built it, too...)\n      const creationStack = ['--- resource created at ---', ...this.creationStackTrace].join('\\n  at ');\n      const problemTrace = e.stack.substr(e.stack.indexOf(e.message) + e.message.length);\n      e.stack = `${e.message}\\n  ${creationStack}\\n  --- problem discovered at ---${problemTrace}`;\n      // Re-throw\n      throw e;\n    }\n  }\n\n  protected renderProperties(properties: any): { [key: string]: any } {\n    return properties;\n  }\n\n  private renderDependsOn() {\n    const logicalIDs = new Set<string>();\n    for (const d of this.dependsOn) {\n      addDependency(d);\n    }\n\n    return Array.from(logicalIDs);\n\n    function addDependency(d: IDependable) {\n      d.dependencyElements.forEach(dep => {\n        const logicalId = (dep as StackElement).logicalId;\n        if (logicalId) {\n          logicalIDs.add(logicalId);\n        }\n      });\n\n      // break if dependencyElements include only 'd', which means we reached a terminal.\n      if (d.dependencyElements.length === 1 && d.dependencyElements[0] === d) {\n        return;\n      } else {\n        d.dependencyElements.forEach(dep => addDependency(dep));\n      }\n    }\n  }\n}\n\nexport interface ResourceOptions {\n  /**\n   * A condition to associate with this resource. This means that only if the condition evaluates to 'true' when the stack\n   * is deployed, the resource will be included. This is provided to allow CDK projects to produce legacy templates, but noramlly\n   * there is no need to use it in CDK projects.\n   */\n  condition?: Condition;\n\n  /**\n   * Associate the CreationPolicy attribute with a resource to prevent its status from reaching create complete until\n   * AWS CloudFormation receives a specified number of success signals or the timeout period is exceeded. To signal a\n   * resource, you can use the cfn-signal helper script or SignalResource API. AWS CloudFormation publishes valid signals\n   * to the stack events so that you track the number of signals sent.\n   */\n  creationPolicy?: CreationPolicy;\n\n  /**\n   * With the DeletionPolicy attribute you can preserve or (in some cases) backup a resource when its stack is deleted.\n   * You specify a DeletionPolicy attribute for each resource that you want to control. If a resource has no DeletionPolicy\n   * attribute, AWS CloudFormation deletes the resource by default. Note that this capability also applies to update operations\n   * that lead to resources being removed.\n   */\n  deletionPolicy?: DeletionPolicy;\n\n  /**\n   * Use the UpdatePolicy attribute to specify how AWS CloudFormation handles updates to the AWS::AutoScaling::AutoScalingGroup\n   * resource. AWS CloudFormation invokes one of three update policies depending on the type of change you make or whether a\n   * scheduled action is associated with the Auto Scaling group.\n   */\n  updatePolicy?: UpdatePolicy;\n\n  /**\n   * Metadata associated with the CloudFormation resource. This is not the same as the construct metadata which can be added\n   * using construct.addMetadata(), but would not appear in the CloudFormation template automatically.\n   */\n  metadata?: { [key: string]: any };\n}\n\n/**\n * Merges `source` into `target`, overriding any existing values.\n * `null`s will cause a value to be deleted.\n */\nexport function deepMerge(target: any, source: any) {\n  if (typeof(source) !== 'object' || typeof(target) !== 'object') {\n    throw new Error(`Invalid usage. Both source (${JSON.stringify(source)}) and target (${JSON.stringify(target)}) must be objects`);\n  }\n\n  for (const key of Object.keys(source)) {\n    const value = source[key];\n    if (typeof(value) === 'object' && value != null && !Array.isArray(value)) {\n      // if the value at the target is not an object, override it with an\n      // object so we can continue the recursion\n      if (typeof(target[key]) !== 'object') {\n        target[key] = { };\n      }\n\n      deepMerge(target[key], value);\n\n      // if the result of the merge is an empty object, it's because the\n      // eventual value we assigned is `undefined`, and there are no\n      // sibling concrete values alongside, so we can delete this tree.\n      const output = target[key];\n      if (typeof(output) === 'object' && Object.keys(output).length === 0) {\n        delete target[key];\n      }\n    } else if (value === undefined) {\n      delete target[key];\n    } else {\n      target[key] = value;\n    }\n  }\n\n  return target;\n}\n"]}
import { Construct } from './core/construct';
declare type ContextProviderProps = {
[key: string]: any;
};
/**

@@ -12,25 +15,22 @@ * Base class for the model side of context providers

export declare class ContextProvider {
private context;
private readonly context;
private readonly provider;
private readonly stack;
constructor(context: Construct);
private readonly props;
constructor(context: Construct, provider: string, props?: ContextProviderProps);
readonly key: string;
/**
* Read a provider value and verify it is not `null`
*/
getValue(defaultValue: any): any;
/**
* Read a provider value, verifying it's a string
* @param provider The name of the context provider
* @param scope The scope (e.g. account/region) for the value
* @param args Any arguments
* @param defaultValue The value to return if there is no value defined for this context key
*/
getStringValue(provider: string, scope: undefined | string[], args: string[], defaultValue: string): string;
getStringValue(defaultValue: string): string;
/**
* Read a provider value, verifying it's a list
* @param provider The name of the context provider
* @param scope The scope (e.g. account/region) for the value
* @param args Any arguments
* @param defaultValue The value to return if there is no value defined for this context key
*/
getStringListValue(provider: string, scope: undefined | string[], args: string[], defaultValue: string[]): string[];
/**
* Helper function to wrap up account and region into a scope tuple
*/
accountRegionScope(providerDescription: string): undefined | string[];
getStringListValue(defaultValue: string[]): string[];
}

@@ -48,2 +48,8 @@ /**

}
export interface SSMParameterProviderProps {
/**
* The name of the parameter to lookup
*/
parameterName: string;
}
/**

@@ -54,7 +60,8 @@ * Context provider that will read values from the SSM parameter store in the indicated account and region

private provider;
constructor(context: Construct);
constructor(context: Construct, props: SSMParameterProviderProps);
/**
* Return the SSM parameter string with the indicated key
*/
getString(parameterName: string, defaultValue?: string): any;
parameterValue(): any;
}
export {};

@@ -16,29 +16,54 @@ "use strict";

class ContextProvider {
constructor(context) {
constructor(context, provider, props = {}) {
this.context = context;
this.provider = provider;
this.stack = stack_1.Stack.find(context);
this.props = Object.assign({ account: this.stack.env.account, region: this.stack.env.region }, props);
}
get key() {
const propStrings = propsToArray(this.props);
return `${this.provider}:${propStrings.join(':')}`;
}
/**
* Read a provider value and verify it is not `null`
*/
getValue(defaultValue) {
// if account or region is not defined this is probably a test mode, so we just
// return the default value
if (!this.props.account || !this.props.region) {
this.context.addError(formatMissingScopeError(this.provider, this.props));
return defaultValue;
}
const value = this.context.getContext(this.key);
if (value != null) {
return value;
}
this.stack.reportMissingContext(this.key, {
provider: this.provider,
props: this.props,
});
return defaultValue;
}
/**
* Read a provider value, verifying it's a string
* @param provider The name of the context provider
* @param scope The scope (e.g. account/region) for the value
* @param args Any arguments
* @param defaultValue The value to return if there is no value defined for this context key
*/
getStringValue(provider, scope, args, defaultValue) {
getStringValue(defaultValue) {
// if scope is undefined, this is probably a test mode, so we just
// return the default value
if (!scope) {
this.context.addError(formatMissingScopeError(provider, args));
if (!this.props.account || !this.props.region) {
this.context.addError(formatMissingScopeError(this.provider, this.props));
return defaultValue;
}
const key = colonQuote([provider].concat(scope).concat(args)).join(':');
const value = this.context.getContext(key);
const value = this.context.getContext(this.key);
if (value != null) {
if (typeof value !== 'string') {
throw new TypeError(`Expected context parameter '${key}' to be a string, but got '${value}'`);
throw new TypeError(`Expected context parameter '${this.key}' to be a string, but got '${JSON.stringify(value)}'`);
}
return value;
}
this.stack.reportMissingContext(key, { provider, scope, args });
this.stack.reportMissingContext(this.key, {
provider: this.provider,
props: this.props,
});
return defaultValue;

@@ -48,42 +73,25 @@ }

* Read a provider value, verifying it's a list
* @param provider The name of the context provider
* @param scope The scope (e.g. account/region) for the value
* @param args Any arguments
* @param defaultValue The value to return if there is no value defined for this context key
*/
getStringListValue(provider, scope, args, defaultValue) {
getStringListValue(defaultValue) {
// if scope is undefined, this is probably a test mode, so we just
// return the default value and report an error so this in not accidentally used
// in the toolkit
if (!scope) {
// tslint:disable-next-line:max-line-length
this.context.addError(formatMissingScopeError(provider, args));
if (!this.props.account || !this.props.region) {
this.context.addError(formatMissingScopeError(this.provider, this.props));
return defaultValue;
}
const key = colonQuote([provider].concat(scope).concat(args)).join(':');
const value = this.context.getContext(key);
const value = this.context.getContext(this.key);
if (value != null) {
if (!value.map) {
throw new Error(`Context value '${key}' is supposed to be a list, got '${value}'`);
throw new Error(`Context value '${this.key}' is supposed to be a list, got '${JSON.stringify(value)}'`);
}
return value;
}
this.stack.reportMissingContext(key, { provider, scope, args });
this.stack.reportMissingContext(this.key, {
provider: this.provider,
props: this.props,
});
return defaultValue;
}
/**
* Helper function to wrap up account and region into a scope tuple
*/
accountRegionScope(providerDescription) {
const stack = stack_1.Stack.find(this.context);
if (!stack) {
throw new Error(`${providerDescription}: construct must be in a stack`);
}
const account = stack.env.account;
const region = stack.env.region;
if (account == null || region == null) {
return undefined;
}
return [account, region];
}
}

@@ -98,3 +106,3 @@ exports.ContextProvider = ContextProvider;

function colonQuote(xs) {
return xs.map(x => x.replace('$', '$$').replace(':', '$:'));
return xs.replace('$', '$$').replace(':', '$:');
}

@@ -106,3 +114,3 @@ /**

constructor(context) {
this.provider = new ContextProvider(context);
this.provider = new ContextProvider(context, AVAILABILITY_ZONES_PROVIDER);
}

@@ -113,3 +121,3 @@ /**

get availabilityZones() {
return this.provider.getStringListValue(AVAILABILITY_ZONES_PROVIDER, this.provider.accountRegionScope('AvailabilityZoneProvider'), [], ['dummy1a', 'dummy1b', 'dummy1c']);
return this.provider.getStringListValue(['dummy1a', 'dummy1b', 'dummy1c']);
}

@@ -122,4 +130,4 @@ }

class SSMParameterProvider {
constructor(context) {
this.provider = new ContextProvider(context);
constructor(context, props) {
this.provider = new ContextProvider(context, SSM_PARAMETER_PROVIDER, props);
}

@@ -129,14 +137,11 @@ /**

*/
getString(parameterName, defaultValue = "dummy") {
const scope = this.provider.accountRegionScope('SSMParameterProvider');
return this.provider.getStringValue(SSM_PARAMETER_PROVIDER, scope, [parameterName], defaultValue);
parameterValue() {
return this.provider.getStringValue('dummy');
}
}
exports.SSMParameterProvider = SSMParameterProvider;
function formatMissingScopeError(provider, args) {
function formatMissingScopeError(provider, props) {
let s = `Cannot determine scope for context provider ${provider}`;
if (args.length > 0) {
s += JSON.stringify(args);
}
s += '.';
const propsString = Object.keys(props).map(key => (`${key}=${props[key]}`));
s += ` with props: ${propsString}.`;
s += '\n';

@@ -146,2 +151,26 @@ s += 'This usually happens when AWS credentials are not available and the default account/region cannot be determined.';

}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"context.js","sourceRoot":"","sources":["context.ts"],"names":[],"mappings":"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","sourcesContent":["import { Stack } from './cloudformation/stack';\nimport { Construct } from './core/construct';\n\nconst AVAILABILITY_ZONES_PROVIDER = 'availability-zones';\nconst SSM_PARAMETER_PROVIDER = 'ssm';\n\n/**\n * Base class for the model side of context providers\n *\n * Instances of this class communicate with context provider plugins in the 'cdk\n * toolkit' via context variables (input), outputting specialized queries for\n * more context variables (output).\n *\n * ContextProvider needs access to a Construct to hook into the context mechanism.\n */\nexport class ContextProvider {\n\n  private readonly stack: Stack;\n\n  constructor(private context: Construct) {\n    this.stack = Stack.find(context);\n  }\n\n  /**\n   * Read a provider value, verifying it's a string\n   * @param provider The name of the context provider\n   * @param scope The scope (e.g. account/region) for the value\n   * @param args Any arguments\n   * @param defaultValue The value to return if there is no value defined for this context key\n   */\n  public getStringValue(\n    provider: string,\n    scope: undefined | string[],\n    args: string[],\n    defaultValue: string): string {\n    // if scope is undefined, this is probably a test mode, so we just\n    // return the default value\n    if (!scope) {\n      this.context.addError(formatMissingScopeError(provider, args));\n      return defaultValue;\n    }\n    const key = colonQuote([provider].concat(scope).concat(args)).join(':');\n    const value = this.context.getContext(key);\n    if (value != null) {\n      if (typeof value !== 'string') {\n        throw new TypeError(`Expected context parameter '${key}' to be a string, but got '${value}'`);\n      }\n      return value;\n    }\n\n    this.stack.reportMissingContext(key, { provider, scope, args });\n    return defaultValue;\n  }\n\n  /**\n   * Read a provider value, verifying it's a list\n   * @param provider The name of the context provider\n   * @param scope The scope (e.g. account/region) for the value\n   * @param args Any arguments\n   * @param defaultValue The value to return if there is no value defined for this context key\n   */\n  public getStringListValue(\n    provider: string,\n    scope: undefined | string[],\n    args: string[],\n    defaultValue: string[]): string[] {\n    // if scope is undefined, this is probably a test mode, so we just\n    // return the default value and report an error so this in not accidentally used\n    // in the toolkit\n    if (!scope) {\n      // tslint:disable-next-line:max-line-length\n      this.context.addError(formatMissingScopeError(provider, args));\n      return defaultValue;\n    }\n\n    const key = colonQuote([provider].concat(scope).concat(args)).join(':');\n    const value = this.context.getContext(key);\n\n    if (value != null) {\n      if (!value.map) {\n        throw new Error(`Context value '${key}' is supposed to be a list, got '${value}'`);\n      }\n      return value;\n    }\n\n    this.stack.reportMissingContext(key, { provider, scope, args });\n    return defaultValue;\n  }\n\n  /**\n   * Helper function to wrap up account and region into a scope tuple\n   */\n  public accountRegionScope(providerDescription: string): undefined | string[] {\n    const stack = Stack.find(this.context);\n    if (!stack) {\n      throw new Error(`${providerDescription}: construct must be in a stack`);\n    }\n\n    const account = stack.env.account;\n    const region = stack.env.region;\n\n    if (account == null || region == null) {\n      return undefined;\n    }\n\n    return [account, region];\n  }\n}\n\n/**\n * Quote colons in all strings so that we can undo the quoting at a later point\n *\n * We'll use $ as a quoting character, for no particularly good reason other\n * than that \\ is going to lead to quoting hell when the keys are stored in JSON.\n */\nfunction colonQuote(xs: string[]): string[] {\n  return xs.map(x => x.replace('$', '$$').replace(':', '$:'));\n}\n\n/**\n * Context provider that will return the availability zones for the current account and region\n */\nexport class AvailabilityZoneProvider {\n  private provider: ContextProvider;\n\n  constructor(context: Construct) {\n    this.provider = new ContextProvider(context);\n  }\n\n  /**\n   * Return the list of AZs for the current account and region\n   */\n  public get availabilityZones(): string[] {\n    return this.provider.getStringListValue(AVAILABILITY_ZONES_PROVIDER,\n                        this.provider.accountRegionScope('AvailabilityZoneProvider'),\n                        [],\n                        ['dummy1a', 'dummy1b', 'dummy1c']);\n  }\n}\n\n/**\n * Context provider that will read values from the SSM parameter store in the indicated account and region\n */\nexport class SSMParameterProvider {\n  private provider: ContextProvider;\n\n  constructor(context: Construct) {\n    this.provider = new ContextProvider(context);\n  }\n\n  /**\n   * Return the SSM parameter string with the indicated key\n   */\n  public getString(parameterName: string, defaultValue: string = \"dummy\"): any {\n    const scope = this.provider.accountRegionScope('SSMParameterProvider');\n    return this.provider.getStringValue(SSM_PARAMETER_PROVIDER, scope, [parameterName], defaultValue);\n  }\n}\n\nfunction formatMissingScopeError(provider: string, args: string[]) {\n  let s = `Cannot determine scope for context provider ${provider}`;\n  if (args.length > 0) {\n    s += JSON.stringify(args);\n  }\n  s += '.';\n  s += '\\n';\n  s += 'This usually happens when AWS credentials are not available and the default account/region cannot be determined.';\n  return s;\n}\n"]}
function propsToArray(props) {
const propArray = [];
const keys = Object.keys(props);
keys.sort();
for (const key of keys) {
switch (typeof props[key]) {
case 'object': {
const childObjStrs = propsToArray(props[key]);
const qualifiedChildStr = childObjStrs.map(child => (`${key}.${child}`)).join(':');
propArray.push(qualifiedChildStr);
break;
}
case 'string': {
propArray.push(`${key}=${colonQuote(props[key])}`);
break;
}
default: {
propArray.push(`${key}=${JSON.stringify(props[key])}`);
break;
}
}
}
return propArray;
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"context.js","sourceRoot":"","sources":["context.ts"],"names":[],"mappings":";;AAAA,kDAA+C;AAG/C,MAAM,2BAA2B,GAAG,oBAAoB,CAAC;AACzD,MAAM,sBAAsB,GAAG,KAAK,CAAC;AAGrC;;;;;;;;GAQG;AACH,MAAa,eAAe;IAK1B,YACmB,OAAkB,EAClB,QAAgB,EACjC,QAA8B,EAAE;QAFf,YAAO,GAAP,OAAO,CAAW;QAClB,aAAQ,GAAR,QAAQ,CAAQ;QAEjC,IAAI,CAAC,KAAK,GAAG,aAAK,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QACjC,IAAI,CAAC,KAAK,mBACR,OAAO,EAAE,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,OAAO,EAC/B,MAAM,EAAE,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,MAAM,IAC1B,KAAK,CACT,CAAC;IACJ,CAAC;IAED,IAAW,GAAG;QACZ,MAAM,WAAW,GAAa,YAAY,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACvD,OAAO,GAAG,IAAI,CAAC,QAAQ,IAAI,WAAW,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC;IACrD,CAAC;IAED;;OAEG;IACI,QAAQ,CAAC,YAAiB;QAC/B,+EAA+E;QAC/E,2BAA2B;QAC3B,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,OAAO,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,EAAE;YAC7C,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,uBAAuB,CAAC,IAAI,CAAC,QAAQ,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC;YAC1E,OAAO,YAAY,CAAC;SACrB;QAED,MAAM,KAAK,GAAG,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QAEhD,IAAI,KAAK,IAAI,IAAI,EAAE;YACjB,OAAO,KAAK,CAAC;SACd;QAED,IAAI,CAAC,KAAK,CAAC,oBAAoB,CAAC,IAAI,CAAC,GAAG,EAAE;YACxC,QAAQ,EAAE,IAAI,CAAC,QAAQ;YACvB,KAAK,EAAE,IAAI,CAAC,KAAK;SAClB,CAAC,CAAC;QACH,OAAO,YAAY,CAAC;IACtB,CAAC;IACD;;;OAGG;IACI,cAAc,CAAE,YAAoB;QACzC,kEAAkE;QAClE,2BAA2B;QAC3B,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,OAAO,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,EAAE;YAC7C,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,uBAAuB,CAAC,IAAI,CAAC,QAAQ,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC;YAC1E,OAAO,YAAY,CAAC;SACrB;QAED,MAAM,KAAK,GAAG,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QAEhD,IAAI,KAAK,IAAI,IAAI,EAAE;YACjB,IAAI,OAAO,KAAK,KAAK,QAAQ,EAAE;gBAC7B,MAAM,IAAI,SAAS,CAAC,+BAA+B,IAAI,CAAC,GAAG,8BAA8B,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;aACpH;YACD,OAAO,KAAK,CAAC;SACd;QAED,IAAI,CAAC,KAAK,CAAC,oBAAoB,CAAC,IAAI,CAAC,GAAG,EAAE;YACxC,QAAQ,EAAE,IAAI,CAAC,QAAQ;YACvB,KAAK,EAAE,IAAI,CAAC,KAAK;SAClB,CAAC,CAAC;QACH,OAAO,YAAY,CAAC;IACtB,CAAC;IAED;;;OAGG;IACI,kBAAkB,CACvB,YAAsB;QACpB,kEAAkE;QAClE,gFAAgF;QAChF,iBAAiB;QACjB,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,OAAO,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,EAAE;YAC7C,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,uBAAuB,CAAC,IAAI,CAAC,QAAQ,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC;YAC1E,OAAO,YAAY,CAAC;SACrB;QAED,MAAM,KAAK,GAAG,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QAEhD,IAAI,KAAK,IAAI,IAAI,EAAE;YACjB,IAAI,CAAC,KAAK,CAAC,GAAG,EAAE;gBACd,MAAM,IAAI,KAAK,CAAC,kBAAkB,IAAI,CAAC,GAAG,oCAAoC,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;aACzG;YACD,OAAO,KAAK,CAAC;SACd;QAED,IAAI,CAAC,KAAK,CAAC,oBAAoB,CAAC,IAAI,CAAC,GAAG,EAAE;YACxC,QAAQ,EAAE,IAAI,CAAC,QAAQ;YACvB,KAAK,EAAE,IAAI,CAAC,KAAK;SAClB,CAAC,CAAC;QAEH,OAAO,YAAY,CAAC;IACtB,CAAC;CACJ;AAvGD,0CAuGC;AAED;;;;;GAKG;AACH,SAAS,UAAU,CAAC,EAAU;IAC5B,OAAO,EAAE,CAAC,OAAO,CAAC,GAAG,EAAE,IAAI,CAAC,CAAC,OAAO,CAAC,GAAG,EAAE,IAAI,CAAC,CAAC;AAClD,CAAC;AAED;;GAEG;AACH,MAAa,wBAAwB;IAGnC,YAAY,OAAkB;QAC5B,IAAI,CAAC,QAAQ,GAAG,IAAI,eAAe,CAAC,OAAO,EAAE,2BAA2B,CAAC,CAAC;IAC5E,CAAC;IAED;;OAEG;IACH,IAAW,iBAAiB;QAE1B,OAAO,IAAI,CAAC,QAAQ,CAAC,kBAAkB,CAAC,CAAC,SAAS,EAAE,SAAS,EAAE,SAAS,CAAC,CAAC,CAAC;IAC7E,CAAC;CACF;AAdD,4DAcC;AAQD;;GAEG;AACH,MAAa,oBAAoB;IAG/B,YAAY,OAAkB,EAAE,KAAgC;QAC9D,IAAI,CAAC,QAAQ,GAAG,IAAI,eAAe,CAAC,OAAO,EAAE,sBAAsB,EAAE,KAAK,CAAC,CAAC;IAC9E,CAAC;IAED;;OAEG;IACI,cAAc;QACnB,OAAO,IAAI,CAAC,QAAQ,CAAC,cAAc,CAAC,OAAO,CAAC,CAAC;IAC/C,CAAC;CACF;AAbD,oDAaC;AAED,SAAS,uBAAuB,CAAC,QAAgB,EAAE,KAA8B;IAC/E,IAAI,CAAC,GAAG,+CAA+C,QAAQ,EAAE,CAAC;IAClE,MAAM,WAAW,GAAG,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,GAAG,CAAE,GAAG,CAAC,EAAE,CAAC,CAAC,GAAG,GAAG,IAAI,KAAK,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC;IAC7E,CAAC,IAAI,gBAAgB,WAAW,GAAG,CAAC;IACpC,CAAC,IAAI,IAAI,CAAC;IACV,CAAC,IAAI,kHAAkH,CAAC;IACxH,OAAO,CAAC,CAAC;AACX,CAAC;AAED,SAAS,YAAY,CAAC,KAA2B;IAC/C,MAAM,SAAS,GAAa,EAAE,CAAC;IAC/B,MAAM,IAAI,GAAG,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IAChC,IAAI,CAAC,IAAI,EAAE,CAAC;IACZ,KAAK,MAAM,GAAG,IAAI,IAAI,EAAE;QACtB,QAAQ,OAAO,KAAK,CAAC,GAAG,CAAC,EAAE;YACzB,KAAK,QAAQ,CAAC,CAAC;gBACb,MAAM,YAAY,GAAG,YAAY,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC;gBAC9C,MAAM,iBAAiB,GAAG,YAAY,CAAC,GAAG,CAAE,KAAK,CAAC,EAAE,CAAC,CAAC,GAAG,GAAG,IAAI,KAAK,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;gBACpF,SAAS,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;gBAClC,MAAM;aACP;YACD,KAAK,QAAQ,CAAC,CAAC;gBACb,SAAS,CAAC,IAAI,CAAC,GAAG,GAAG,IAAI,UAAU,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC,CAAC;gBACnD,MAAM;aACP;YACD,OAAO,CAAC,CAAC;gBACP,SAAS,CAAC,IAAI,CAAC,GAAG,GAAG,IAAI,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC,CAAC;gBACvD,MAAM;aACP;SACF;KACF;IACD,OAAO,SAAS,CAAC;AACnB,CAAC","sourcesContent":["import { Stack } from './cloudformation/stack';\nimport { Construct } from './core/construct';\n\nconst AVAILABILITY_ZONES_PROVIDER = 'availability-zones';\nconst SSM_PARAMETER_PROVIDER = 'ssm';\n\ntype ContextProviderProps = {[key: string]: any};\n/**\n * Base class for the model side of context providers\n *\n * Instances of this class communicate with context provider plugins in the 'cdk\n * toolkit' via context variables (input), outputting specialized queries for\n * more context variables (output).\n *\n * ContextProvider needs access to a Construct to hook into the context mechanism.\n */\nexport class ContextProvider {\n\n  private readonly stack: Stack;\n  private readonly props: ContextProviderProps;\n\n  constructor(\n    private readonly context: Construct,\n    private readonly provider: string,\n    props: ContextProviderProps = {}) {\n    this.stack = Stack.find(context);\n    this.props = {\n      account: this.stack.env.account,\n      region: this.stack.env.region,\n      ...props,\n    };\n  }\n\n  public get key(): string {\n    const propStrings: string[] = propsToArray(this.props);\n    return `${this.provider}:${propStrings.join(':')}`;\n  }\n\n  /**\n   * Read a provider value and verify it is not `null`\n   */\n  public getValue(defaultValue: any): any {\n    // if account or region is not defined this is probably a test mode, so we just\n    // return the default value\n    if (!this.props.account || !this.props.region) {\n      this.context.addError(formatMissingScopeError(this.provider, this.props));\n      return defaultValue;\n    }\n\n    const value = this.context.getContext(this.key);\n\n    if (value != null) {\n      return value;\n    }\n\n    this.stack.reportMissingContext(this.key, {\n      provider: this.provider,\n      props: this.props,\n    });\n    return defaultValue;\n  }\n  /**\n   * Read a provider value, verifying it's a string\n   * @param defaultValue The value to return if there is no value defined for this context key\n   */\n  public getStringValue( defaultValue: string): string {\n    // if scope is undefined, this is probably a test mode, so we just\n    // return the default value\n    if (!this.props.account || !this.props.region) {\n      this.context.addError(formatMissingScopeError(this.provider, this.props));\n      return defaultValue;\n    }\n\n    const value = this.context.getContext(this.key);\n\n    if (value != null) {\n      if (typeof value !== 'string') {\n        throw new TypeError(`Expected context parameter '${this.key}' to be a string, but got '${JSON.stringify(value)}'`);\n      }\n      return value;\n    }\n\n    this.stack.reportMissingContext(this.key, {\n      provider: this.provider,\n      props: this.props,\n    });\n    return defaultValue;\n  }\n\n  /**\n   * Read a provider value, verifying it's a list\n   * @param defaultValue The value to return if there is no value defined for this context key\n   */\n  public getStringListValue(\n    defaultValue: string[]): string[] {\n      // if scope is undefined, this is probably a test mode, so we just\n      // return the default value and report an error so this in not accidentally used\n      // in the toolkit\n      if (!this.props.account || !this.props.region) {\n        this.context.addError(formatMissingScopeError(this.provider, this.props));\n        return defaultValue;\n      }\n\n      const value = this.context.getContext(this.key);\n\n      if (value != null) {\n        if (!value.map) {\n          throw new Error(`Context value '${this.key}' is supposed to be a list, got '${JSON.stringify(value)}'`);\n        }\n        return value;\n      }\n\n      this.stack.reportMissingContext(this.key, {\n        provider: this.provider,\n        props: this.props,\n      });\n\n      return defaultValue;\n    }\n}\n\n/**\n * Quote colons in all strings so that we can undo the quoting at a later point\n *\n * We'll use $ as a quoting character, for no particularly good reason other\n * than that \\ is going to lead to quoting hell when the keys are stored in JSON.\n */\nfunction colonQuote(xs: string): string {\n  return xs.replace('$', '$$').replace(':', '$:');\n}\n\n/**\n * Context provider that will return the availability zones for the current account and region\n */\nexport class AvailabilityZoneProvider {\n  private provider: ContextProvider;\n\n  constructor(context: Construct) {\n    this.provider = new ContextProvider(context, AVAILABILITY_ZONES_PROVIDER);\n  }\n\n  /**\n   * Return the list of AZs for the current account and region\n   */\n  public get availabilityZones(): string[] {\n\n    return this.provider.getStringListValue(['dummy1a', 'dummy1b', 'dummy1c']);\n  }\n}\n\nexport interface SSMParameterProviderProps {\n  /**\n   * The name of the parameter to lookup\n   */\n  parameterName: string;\n}\n/**\n * Context provider that will read values from the SSM parameter store in the indicated account and region\n */\nexport class SSMParameterProvider {\n  private provider: ContextProvider;\n\n  constructor(context: Construct, props: SSMParameterProviderProps) {\n    this.provider = new ContextProvider(context, SSM_PARAMETER_PROVIDER, props);\n  }\n\n  /**\n   * Return the SSM parameter string with the indicated key\n   */\n  public parameterValue(): any {\n    return this.provider.getStringValue('dummy');\n  }\n}\n\nfunction formatMissingScopeError(provider: string, props: {[key: string]: string}) {\n  let s = `Cannot determine scope for context provider ${provider}`;\n  const propsString = Object.keys(props).map( key => (`${key}=${props[key]}`));\n  s += ` with props: ${propsString}.`;\n  s += '\\n';\n  s += 'This usually happens when AWS credentials are not available and the default account/region cannot be determined.';\n  return s;\n}\n\nfunction propsToArray(props: {[key: string]: any}): string[] {\n  const propArray: string[] = [];\n  const keys = Object.keys(props);\n  keys.sort();\n  for (const key of keys) {\n    switch (typeof props[key]) {\n      case 'object': {\n        const childObjStrs = propsToArray(props[key]);\n        const qualifiedChildStr = childObjStrs.map( child => (`${key}.${child}`)).join(':');\n        propArray.push(qualifiedChildStr);\n        break;\n      }\n      case 'string': {\n        propArray.push(`${key}=${colonQuote(props[key])}`);\n        break;\n      }\n      default: {\n        propArray.push(`${key}=${JSON.stringify(props[key])}`);\n        break;\n      }\n    }\n  }\n  return propArray;\n}\n"]}

@@ -76,3 +76,3 @@ /**

*/
export declare function unresolved(obj: any): obj is Token;
export declare function unresolved(obj: any): boolean;
/**

@@ -79,0 +79,0 @@ * Resolves an object by evaluating all tokens and removing any undefined or empty objects or arrays.

@@ -362,2 +362,2 @@ "use strict";

}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tokens.js","sourceRoot":"","sources":["tokens.ts"],"names":[],"mappings":"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","sourcesContent":["import { Construct } from \"./construct\";\n\n/**\n * If objects has a function property by this name, they will be considered tokens, and this\n * function will be called to resolve the value for this object.\n */\nexport const RESOLVE_METHOD = 'resolve';\n\n/**\n * Represents a special or lazily-evaluated value.\n *\n * Can be used to delay evaluation of a certain value in case, for example,\n * that it requires some context or late-bound data. Can also be used to\n * mark values that need special processing at document rendering time.\n *\n * Tokens can be embedded into strings while retaining their original\n * semantics.\n */\nexport class Token {\n  private tokenKey?: string;\n\n  /**\n   * Creates a token that resolves to `value`.\n   *\n   * If value is a function, the function is evaluated upon resolution and\n   * the value it returns will be used as the token's value.\n   *\n   * displayName is used to represent the Token when it's embedded into a string; it\n   * will look something like this:\n   *\n   *    \"embedded in a larger string is ${Token[DISPLAY_NAME.123]}\"\n   *\n   * This value is used as a hint to humans what the meaning of the Token is,\n   * and does not have any effect on the evaluation.\n   *\n   * Must contain only alphanumeric and simple separator characters (_.:-).\n   *\n   * @param valueOrFunction What this token will evaluate to, literal or function.\n   * @param displayName A human-readable display hint for this Token\n   */\n  constructor(private readonly valueOrFunction?: any, private readonly displayName?: string) {\n  }\n\n  /**\n   * @returns The resolved value for this token.\n   */\n  public resolve(): any {\n    let value = this.valueOrFunction;\n    if (typeof(value) === 'function') {\n      value = value();\n    }\n\n    return value;\n  }\n\n  /**\n   * Return a reversible string representation of this token\n   *\n   * If the Token is initialized with a literal, the stringified value of the\n   * literal is returned. Otherwise, a special quoted string representation\n   * of the Token is returned that can be embedded into other strings.\n   *\n   * Strings with quoted Tokens in them can be restored back into\n   * complex values with the Tokens restored by calling `resolve()`\n   * on the string.\n   */\n  public toString(): string {\n    const valueType = typeof this.valueOrFunction;\n    // Optimization: if we can immediately resolve this, don't bother\n    // registering a Token.\n    if (valueType === 'string' || valueType === 'number' || valueType === 'boolean') {\n      return this.valueOrFunction.toString();\n    }\n\n    if (this.tokenKey === undefined) {\n      this.tokenKey = TOKEN_STRING_MAP.register(this, this.displayName);\n    }\n    return this.tokenKey;\n  }\n\n  /**\n   * Turn this Token into JSON\n   *\n   * This gets called by JSON.stringify(). We want to prohibit this, because\n   * it's not possible to do this properly, so we just throw an error here.\n   */\n  public toJSON(): any {\n    // tslint:disable-next-line:max-line-length\n    throw new Error('JSON.stringify() cannot be applied to structure with a Token in it. Use a document-specific stringification method instead.');\n  }\n\n  /**\n   * Return a concated version of this Token in a string context\n   *\n   * The default implementation of this combines strings, but specialized\n   * implements of Token can return a more appropriate value.\n   */\n  public concat(left: any | undefined, right: any | undefined): Token {\n    const parts = [left, resolve(this), right].filter(x => x !== undefined);\n    return new Token(parts.map(x => `${x}`).join(''));\n  }\n}\n\n/**\n * Returns true if obj is a token (i.e. has the resolve() method or is a string\n * that includes token markers).\n * @param obj The object to test.\n */\nexport function unresolved(obj: any): obj is Token {\n  if (typeof(obj) === 'string') {\n    return TOKEN_STRING_MAP.createTokenString(obj).test();\n  } else {\n    return typeof(obj[RESOLVE_METHOD]) === 'function';\n  }\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, prefix?: string[]): any {\n  const path = prefix || [ ];\n  const pathName = '/' + path.join('/');\n\n  // protect against cyclic references by limiting depth.\n  if (path.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    return TOKEN_STRING_MAP.resolveMarkers(obj as string);\n  }\n\n  //\n  // primitives - as-is\n  //\n\n  if (typeof(obj) !== 'object' || obj instanceof Date) {\n    return obj;\n  }\n\n  //\n  // tokens - invoke 'resolve' and continue to resolve recursively\n  //\n\n  if (unresolved(obj)) {\n    const value = obj[RESOLVE_METHOD]();\n    return resolve(value, path);\n  }\n\n  //\n  // arrays - resolve all values, remove undefined and remove empty arrays\n  //\n\n  if (Array.isArray(obj)) {\n    const arr = obj\n      .map((x, i) => resolve(x, path.concat(i.toString())))\n      .filter(x => typeof(x) !== 'undefined');\n\n    return arr;\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 type\n  // mistake somewhere and resolve will get into an infinite loop recursing into\n  // child.parent <---> parent.children\n  if (obj instanceof Construct) {\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 = resolve(key);\n    if (typeof(resolvedKey) !== 'string') {\n      throw new Error(`The key \"${key}\" has been resolved to ${JSON.stringify(resolvedKey)} but must be resolvable to a string`);\n    }\n\n    const value = resolve(obj[key], path.concat(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 * 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\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 */\nclass TokenStringMap {\n  private readonly tokenMap: {[key: string]: Token};\n\n  constructor() {\n    const glob = global as any;\n    this.tokenMap = glob.__cdkTokenMap = glob.__cdkTokenMap || {};\n  }\n\n  /**\n   * Generating 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 register(token: Token, representationHint?: string): string {\n    const counter = Object.keys(this.tokenMap).length;\n    const representation = representationHint || `TOKEN`;\n\n    const key = `${representation}.${counter}`;\n    if (new RegExp(`[^${VALID_KEY_CHARS}]`).exec(key)) {\n      throw new Error(`Invalid characters in token representation: ${key}`);\n    }\n\n    this.tokenMap[key] = token;\n    return `${BEGIN_TOKEN_MARKER}${key}${END_TOKEN_MARKER}`;\n  }\n\n  /**\n   * Returns a `TokenString` for this string.\n   */\n  public createTokenString(s: string) {\n    return new TokenString(s, BEGIN_TOKEN_MARKER, `[${VALID_KEY_CHARS}]+`, END_TOKEN_MARKER);\n  }\n\n  /**\n   * Replace any Token markers in this string with their resolved values\n   */\n  public resolveMarkers(s: string): any {\n    const str = this.createTokenString(s);\n    const fragments = str.split(this.lookupToken.bind(this));\n    return fragments.join();\n  }\n\n  /**\n   * Find a Token by key\n   */\n  public lookupToken(key: string): Token {\n    if (!(key in this.tokenMap)) {\n      throw new Error(`Unrecognized token key: ${key}`);\n    }\n\n    return this.tokenMap[key];\n  }\n}\n\nconst BEGIN_TOKEN_MARKER = '${Token[';\nconst END_TOKEN_MARKER = ']}';\nconst VALID_KEY_CHARS = 'a-zA-Z0-9:._-';\n\n/**\n * Singleton instance of the token string map\n */\nconst TOKEN_STRING_MAP = new TokenStringMap();\n\n/**\n * Interface that Token joiners implement\n */\nexport interface ITokenJoiner {\n  /**\n   * The name of the joiner.\n   *\n   * Must be unique per joiner: this value will be used to assert that there\n   * is exactly only type of joiner in a join operation.\n   */\n  id: string;\n\n  /**\n   * Return the language intrinsic that will combine the strings in the given engine\n   */\n  join(fragments: any[]): any;\n}\n\n/**\n * A string with markers in it that can be resolved to external values\n */\nclass TokenString {\n  private pattern: string;\n\n  constructor(\n    private readonly str: string,\n    private readonly beginMarker: string,\n    private readonly idPattern: string,\n    private readonly endMarker: string) {\n    this.pattern = `${regexQuote(this.beginMarker)}(${this.idPattern})${regexQuote(this.endMarker)}`;\n  }\n\n  /**\n   * Split string on markers, substituting markers with Tokens\n   */\n  public split(lookup: (id: string) => Token): TokenStringFragments {\n    const re = new RegExp(this.pattern, 'g');\n    const ret = new TokenStringFragments();\n\n    let rest = 0;\n    let m = re.exec(this.str);\n    while (m) {\n      if (m.index > rest) {\n        ret.addString(this.str.substring(rest, m.index));\n      }\n\n      ret.addToken(lookup(m[1]));\n\n      rest = re.lastIndex;\n      m = re.exec(this.str);\n    }\n\n    if (rest < this.str.length) {\n      ret.addString(this.str.substring(rest));\n    }\n\n    return ret;\n  }\n\n  /**\n   * Indicates if this string includes tokens.\n   */\n  public test(): boolean {\n    const re = new RegExp(this.pattern, 'g');\n    return re.test(this.str);\n  }\n}\n\n/**\n * Result of the split of a string with Tokens\n *\n * Either a literal part of the string, or an unresolved Token.\n */\ntype StringFragment = { type: 'string'; str: string };\ntype TokenFragment = { type: 'token'; token: Token };\ntype Fragment =  StringFragment | TokenFragment;\n\n/**\n * Fragments of a string with markers\n */\nclass TokenStringFragments {\n  private readonly fragments = new Array<Fragment>();\n\n  public values(): any[] {\n    return this.fragments.map(f => f.type === 'token' ? resolve(f.token) : f.str);\n  }\n\n  public addString(str: string) {\n    this.fragments.push({ type: 'string', str });\n  }\n\n  public addToken(token: Token) {\n    this.fragments.push({ type: 'token', token });\n  }\n\n  /**\n   * Combine the resolved string fragments using the Tokens to join.\n   *\n   * Resolves the result.\n   */\n  public join(): any {\n    if (this.fragments.length === 0) { return ''; }\n    if (this.fragments.length === 1) { return resolveFragment(this.fragments[0]); }\n\n    const first = this.fragments[0];\n\n    let i;\n    let token: Token;\n\n    if (first.type === 'token') {\n      token = first.token;\n      i = 1;\n    } else {\n      // We never have two strings in a row\n      token = (this.fragments[1] as TokenFragment).token.concat(first.str, undefined);\n      i = 2;\n    }\n\n    while (i < this.fragments.length) {\n      token = token.concat(undefined, resolveFragment(this.fragments[i]));\n      i++;\n    }\n\n    return resolve(token);\n  }\n}\n\n/**\n * Resolve the value from a single fragment\n */\nfunction resolveFragment(fragment: Fragment): any {\n  return fragment.type === 'string' ? fragment.str : resolve(fragment.token);\n}\n\n/**\n * Quote a string for use in a regex\n */\nfunction regexQuote(s: string) {\n  return s.replace(/[.?*+^$[\\]\\\\(){}|-]/g, \"\\\\$&\");\n}\n"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tokens.js","sourceRoot":"","sources":["tokens.ts"],"names":[],"mappings":"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","sourcesContent":["import { Construct } from \"./construct\";\n\n/**\n * If objects has a function property by this name, they will be considered tokens, and this\n * function will be called to resolve the value for this object.\n */\nexport const RESOLVE_METHOD = 'resolve';\n\n/**\n * Represents a special or lazily-evaluated value.\n *\n * Can be used to delay evaluation of a certain value in case, for example,\n * that it requires some context or late-bound data. Can also be used to\n * mark values that need special processing at document rendering time.\n *\n * Tokens can be embedded into strings while retaining their original\n * semantics.\n */\nexport class Token {\n  private tokenKey?: string;\n\n  /**\n   * Creates a token that resolves to `value`.\n   *\n   * If value is a function, the function is evaluated upon resolution and\n   * the value it returns will be used as the token's value.\n   *\n   * displayName is used to represent the Token when it's embedded into a string; it\n   * will look something like this:\n   *\n   *    \"embedded in a larger string is ${Token[DISPLAY_NAME.123]}\"\n   *\n   * This value is used as a hint to humans what the meaning of the Token is,\n   * and does not have any effect on the evaluation.\n   *\n   * Must contain only alphanumeric and simple separator characters (_.:-).\n   *\n   * @param valueOrFunction What this token will evaluate to, literal or function.\n   * @param displayName A human-readable display hint for this Token\n   */\n  constructor(private readonly valueOrFunction?: any, private readonly displayName?: string) {\n  }\n\n  /**\n   * @returns The resolved value for this token.\n   */\n  public resolve(): any {\n    let value = this.valueOrFunction;\n    if (typeof(value) === 'function') {\n      value = value();\n    }\n\n    return value;\n  }\n\n  /**\n   * Return a reversible string representation of this token\n   *\n   * If the Token is initialized with a literal, the stringified value of the\n   * literal is returned. Otherwise, a special quoted string representation\n   * of the Token is returned that can be embedded into other strings.\n   *\n   * Strings with quoted Tokens in them can be restored back into\n   * complex values with the Tokens restored by calling `resolve()`\n   * on the string.\n   */\n  public toString(): string {\n    const valueType = typeof this.valueOrFunction;\n    // Optimization: if we can immediately resolve this, don't bother\n    // registering a Token.\n    if (valueType === 'string' || valueType === 'number' || valueType === 'boolean') {\n      return this.valueOrFunction.toString();\n    }\n\n    if (this.tokenKey === undefined) {\n      this.tokenKey = TOKEN_STRING_MAP.register(this, this.displayName);\n    }\n    return this.tokenKey;\n  }\n\n  /**\n   * Turn this Token into JSON\n   *\n   * This gets called by JSON.stringify(). We want to prohibit this, because\n   * it's not possible to do this properly, so we just throw an error here.\n   */\n  public toJSON(): any {\n    // tslint:disable-next-line:max-line-length\n    throw new Error('JSON.stringify() cannot be applied to structure with a Token in it. Use a document-specific stringification method instead.');\n  }\n\n  /**\n   * Return a concated version of this Token in a string context\n   *\n   * The default implementation of this combines strings, but specialized\n   * implements of Token can return a more appropriate value.\n   */\n  public concat(left: any | undefined, right: any | undefined): Token {\n    const parts = [left, resolve(this), right].filter(x => x !== undefined);\n    return new Token(parts.map(x => `${x}`).join(''));\n  }\n}\n\n/**\n * Returns true if obj is a token (i.e. has the resolve() method or is a string\n * that includes token markers).\n * @param obj The object to test.\n */\nexport function unresolved(obj: any): boolean {\n  if (typeof(obj) === 'string') {\n    return TOKEN_STRING_MAP.createTokenString(obj).test();\n  } else {\n    return typeof(obj[RESOLVE_METHOD]) === 'function';\n  }\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, prefix?: string[]): any {\n  const path = prefix || [ ];\n  const pathName = '/' + path.join('/');\n\n  // protect against cyclic references by limiting depth.\n  if (path.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    return TOKEN_STRING_MAP.resolveMarkers(obj as string);\n  }\n\n  //\n  // primitives - as-is\n  //\n\n  if (typeof(obj) !== 'object' || obj instanceof Date) {\n    return obj;\n  }\n\n  //\n  // tokens - invoke 'resolve' and continue to resolve recursively\n  //\n\n  if (unresolved(obj)) {\n    const value = obj[RESOLVE_METHOD]();\n    return resolve(value, path);\n  }\n\n  //\n  // arrays - resolve all values, remove undefined and remove empty arrays\n  //\n\n  if (Array.isArray(obj)) {\n    const arr = obj\n      .map((x, i) => resolve(x, path.concat(i.toString())))\n      .filter(x => typeof(x) !== 'undefined');\n\n    return arr;\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 type\n  // mistake somewhere and resolve will get into an infinite loop recursing into\n  // child.parent <---> parent.children\n  if (obj instanceof Construct) {\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 = resolve(key);\n    if (typeof(resolvedKey) !== 'string') {\n      throw new Error(`The key \"${key}\" has been resolved to ${JSON.stringify(resolvedKey)} but must be resolvable to a string`);\n    }\n\n    const value = resolve(obj[key], path.concat(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 * 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\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 */\nclass TokenStringMap {\n  private readonly tokenMap: {[key: string]: Token};\n\n  constructor() {\n    const glob = global as any;\n    this.tokenMap = glob.__cdkTokenMap = glob.__cdkTokenMap || {};\n  }\n\n  /**\n   * Generating 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 register(token: Token, representationHint?: string): string {\n    const counter = Object.keys(this.tokenMap).length;\n    const representation = representationHint || `TOKEN`;\n\n    const key = `${representation}.${counter}`;\n    if (new RegExp(`[^${VALID_KEY_CHARS}]`).exec(key)) {\n      throw new Error(`Invalid characters in token representation: ${key}`);\n    }\n\n    this.tokenMap[key] = token;\n    return `${BEGIN_TOKEN_MARKER}${key}${END_TOKEN_MARKER}`;\n  }\n\n  /**\n   * Returns a `TokenString` for this string.\n   */\n  public createTokenString(s: string) {\n    return new TokenString(s, BEGIN_TOKEN_MARKER, `[${VALID_KEY_CHARS}]+`, END_TOKEN_MARKER);\n  }\n\n  /**\n   * Replace any Token markers in this string with their resolved values\n   */\n  public resolveMarkers(s: string): any {\n    const str = this.createTokenString(s);\n    const fragments = str.split(this.lookupToken.bind(this));\n    return fragments.join();\n  }\n\n  /**\n   * Find a Token by key\n   */\n  public lookupToken(key: string): Token {\n    if (!(key in this.tokenMap)) {\n      throw new Error(`Unrecognized token key: ${key}`);\n    }\n\n    return this.tokenMap[key];\n  }\n}\n\nconst BEGIN_TOKEN_MARKER = '${Token[';\nconst END_TOKEN_MARKER = ']}';\nconst VALID_KEY_CHARS = 'a-zA-Z0-9:._-';\n\n/**\n * Singleton instance of the token string map\n */\nconst TOKEN_STRING_MAP = new TokenStringMap();\n\n/**\n * Interface that Token joiners implement\n */\nexport interface ITokenJoiner {\n  /**\n   * The name of the joiner.\n   *\n   * Must be unique per joiner: this value will be used to assert that there\n   * is exactly only type of joiner in a join operation.\n   */\n  id: string;\n\n  /**\n   * Return the language intrinsic that will combine the strings in the given engine\n   */\n  join(fragments: any[]): any;\n}\n\n/**\n * A string with markers in it that can be resolved to external values\n */\nclass TokenString {\n  private pattern: string;\n\n  constructor(\n    private readonly str: string,\n    private readonly beginMarker: string,\n    private readonly idPattern: string,\n    private readonly endMarker: string) {\n    this.pattern = `${regexQuote(this.beginMarker)}(${this.idPattern})${regexQuote(this.endMarker)}`;\n  }\n\n  /**\n   * Split string on markers, substituting markers with Tokens\n   */\n  public split(lookup: (id: string) => Token): TokenStringFragments {\n    const re = new RegExp(this.pattern, 'g');\n    const ret = new TokenStringFragments();\n\n    let rest = 0;\n    let m = re.exec(this.str);\n    while (m) {\n      if (m.index > rest) {\n        ret.addString(this.str.substring(rest, m.index));\n      }\n\n      ret.addToken(lookup(m[1]));\n\n      rest = re.lastIndex;\n      m = re.exec(this.str);\n    }\n\n    if (rest < this.str.length) {\n      ret.addString(this.str.substring(rest));\n    }\n\n    return ret;\n  }\n\n  /**\n   * Indicates if this string includes tokens.\n   */\n  public test(): boolean {\n    const re = new RegExp(this.pattern, 'g');\n    return re.test(this.str);\n  }\n}\n\n/**\n * Result of the split of a string with Tokens\n *\n * Either a literal part of the string, or an unresolved Token.\n */\ntype StringFragment = { type: 'string'; str: string };\ntype TokenFragment = { type: 'token'; token: Token };\ntype Fragment =  StringFragment | TokenFragment;\n\n/**\n * Fragments of a string with markers\n */\nclass TokenStringFragments {\n  private readonly fragments = new Array<Fragment>();\n\n  public values(): any[] {\n    return this.fragments.map(f => f.type === 'token' ? resolve(f.token) : f.str);\n  }\n\n  public addString(str: string) {\n    this.fragments.push({ type: 'string', str });\n  }\n\n  public addToken(token: Token) {\n    this.fragments.push({ type: 'token', token });\n  }\n\n  /**\n   * Combine the resolved string fragments using the Tokens to join.\n   *\n   * Resolves the result.\n   */\n  public join(): any {\n    if (this.fragments.length === 0) { return ''; }\n    if (this.fragments.length === 1) { return resolveFragment(this.fragments[0]); }\n\n    const first = this.fragments[0];\n\n    let i;\n    let token: Token;\n\n    if (first.type === 'token') {\n      token = first.token;\n      i = 1;\n    } else {\n      // We never have two strings in a row\n      token = (this.fragments[1] as TokenFragment).token.concat(first.str, undefined);\n      i = 2;\n    }\n\n    while (i < this.fragments.length) {\n      token = token.concat(undefined, resolveFragment(this.fragments[i]));\n      i++;\n    }\n\n    return resolve(token);\n  }\n}\n\n/**\n * Resolve the value from a single fragment\n */\nfunction resolveFragment(fragment: Fragment): any {\n  return fragment.type === 'string' ? fragment.str : resolve(fragment.token);\n}\n\n/**\n * Quote a string for use in a regex\n */\nfunction regexQuote(s: string) {\n  return s.replace(/[.?*+^$[\\]\\\\(){}|-]/g, \"\\\\$&\");\n}\n"]}
{
"_args": [
[
"cli-color@0.1.7",
"/codebuild/output/src566918775/src/packages/@aws-cdk/cdk"
]
],
"_from": "cli-color@0.1.7",
"_from": "cli-color@~0.1.6",
"_id": "cli-color@0.1.7",

@@ -15,10 +9,10 @@ "_inBundle": false,

"_requested": {
"type": "version",
"type": "range",
"registry": true,
"raw": "cli-color@0.1.7",
"raw": "cli-color@~0.1.6",
"name": "cli-color",
"escapedName": "cli-color",
"rawSpec": "0.1.7",
"rawSpec": "~0.1.6",
"saveSpec": null,
"fetchSpec": "0.1.7"
"fetchSpec": "~0.1.6"
},

@@ -28,5 +22,6 @@ "_requiredBy": [

],
"_resolved": "https://registry.npmjs.org/cli-color/-/cli-color-0.1.7.tgz",
"_spec": "0.1.7",
"_where": "/codebuild/output/src566918775/src/packages/@aws-cdk/cdk",
"_resolved": "http://registry.npmjs.org/cli-color/-/cli-color-0.1.7.tgz",
"_shasum": "adc3200fa471cc211b0da7f566b71e98b9d67347",
"_spec": "cli-color@~0.1.6",
"_where": "/codebuild/output/src393726274/src/packages/@aws-cdk/cdk/node_modules/json-diff",
"author": {

@@ -41,5 +36,7 @@ "name": "Mariusz Nowak",

},
"bundleDependencies": false,
"dependencies": {
"es5-ext": "0.8.x"
},
"deprecated": false,
"description": "Colors, formatting and other tools for the console",

@@ -46,0 +43,0 @@ "devDependencies": {

{
"_args": [
[
"difflib@0.2.4",
"/codebuild/output/src566918775/src/packages/@aws-cdk/cdk"
]
],
"_from": "difflib@0.2.4",
"_from": "difflib@~0.2.1",
"_id": "difflib@0.2.4",

@@ -15,10 +9,10 @@ "_inBundle": false,

"_requested": {
"type": "version",
"type": "range",
"registry": true,
"raw": "difflib@0.2.4",
"raw": "difflib@~0.2.1",
"name": "difflib",
"escapedName": "difflib",
"rawSpec": "0.2.4",
"rawSpec": "~0.2.1",
"saveSpec": null,
"fetchSpec": "0.2.4"
"fetchSpec": "~0.2.1"
},

@@ -29,4 +23,5 @@ "_requiredBy": [

"_resolved": "https://registry.npmjs.org/difflib/-/difflib-0.2.4.tgz",
"_spec": "0.2.4",
"_where": "/codebuild/output/src566918775/src/packages/@aws-cdk/cdk",
"_shasum": "b5e30361a6db023176d562892db85940a718f47e",
"_spec": "difflib@~0.2.1",
"_where": "/codebuild/output/src393726274/src/packages/@aws-cdk/cdk/node_modules/json-diff",
"author": {

@@ -39,5 +34,7 @@ "name": "Xueqiao Xu",

},
"bundleDependencies": false,
"dependencies": {
"heap": ">= 0.2.0"
},
"deprecated": false,
"description": "text diff library ported from Python's difflib module",

@@ -51,2 +48,5 @@ "devDependencies": {

},
"engines": {
"node": "*"
},
"homepage": "https://github.com/qiao/difflib.js",

@@ -53,0 +53,0 @@ "keywords": [

{
"_args": [
[
"dreamopt@0.6.0",
"/codebuild/output/src566918775/src/packages/@aws-cdk/cdk"
]
],
"_from": "dreamopt@0.6.0",
"_from": "dreamopt@~0.6.0",
"_id": "dreamopt@0.6.0",

@@ -15,10 +9,10 @@ "_inBundle": false,

"_requested": {
"type": "version",
"type": "range",
"registry": true,
"raw": "dreamopt@0.6.0",
"raw": "dreamopt@~0.6.0",
"name": "dreamopt",
"escapedName": "dreamopt",
"rawSpec": "0.6.0",
"rawSpec": "~0.6.0",
"saveSpec": null,
"fetchSpec": "0.6.0"
"fetchSpec": "~0.6.0"
},

@@ -29,4 +23,5 @@ "_requiredBy": [

"_resolved": "https://registry.npmjs.org/dreamopt/-/dreamopt-0.6.0.tgz",
"_spec": "0.6.0",
"_where": "/codebuild/output/src566918775/src/packages/@aws-cdk/cdk",
"_shasum": "d813ccdac8d39d8ad526775514a13dda664d6b4b",
"_spec": "dreamopt@~0.6.0",
"_where": "/codebuild/output/src393726274/src/packages/@aws-cdk/cdk/node_modules/json-diff",
"author": {

@@ -39,5 +34,7 @@ "name": "Andrey Tarantsov",

},
"bundleDependencies": false,
"dependencies": {
"wordwrap": ">=0.0.2"
},
"deprecated": false,
"description": "Command-line parser with readable syntax from your sweetest dreams",

@@ -44,0 +41,0 @@ "devDependencies": {

{
"_args": [
[
"es5-ext@0.8.2",
"/codebuild/output/src566918775/src/packages/@aws-cdk/cdk"
]
],
"_from": "es5-ext@0.8.2",
"_from": "es5-ext@0.8.x",
"_id": "es5-ext@0.8.2",

@@ -15,10 +9,10 @@ "_inBundle": false,

"_requested": {
"type": "version",
"type": "range",
"registry": true,
"raw": "es5-ext@0.8.2",
"raw": "es5-ext@0.8.x",
"name": "es5-ext",
"escapedName": "es5-ext",
"rawSpec": "0.8.2",
"rawSpec": "0.8.x",
"saveSpec": null,
"fetchSpec": "0.8.2"
"fetchSpec": "0.8.x"
},

@@ -29,4 +23,5 @@ "_requiredBy": [

"_resolved": "https://registry.npmjs.org/es5-ext/-/es5-ext-0.8.2.tgz",
"_spec": "0.8.2",
"_where": "/codebuild/output/src566918775/src/packages/@aws-cdk/cdk",
"_shasum": "aba8d9e1943a895ac96837a62a39b3f55ecd94ab",
"_spec": "es5-ext@0.8.x",
"_where": "/codebuild/output/src393726274/src/packages/@aws-cdk/cdk/node_modules/cli-color",
"author": {

@@ -41,3 +36,5 @@ "name": "Mariusz Nowak",

},
"bundleDependencies": false,
"dependencies": {},
"deprecated": false,
"description": "ECMAScript5 extensions",

@@ -44,0 +41,0 @@ "devDependencies": {

{
"_args": [
[
"heap@0.2.6",
"/codebuild/output/src566918775/src/packages/@aws-cdk/cdk"
]
],
"_from": "heap@0.2.6",
"_from": "heap@>= 0.2.0",
"_id": "heap@0.2.6",

@@ -15,10 +9,10 @@ "_inBundle": false,

"_requested": {
"type": "version",
"type": "range",
"registry": true,
"raw": "heap@0.2.6",
"raw": "heap@>= 0.2.0",
"name": "heap",
"escapedName": "heap",
"rawSpec": "0.2.6",
"rawSpec": ">= 0.2.0",
"saveSpec": null,
"fetchSpec": "0.2.6"
"fetchSpec": ">= 0.2.0"
},

@@ -29,4 +23,5 @@ "_requiredBy": [

"_resolved": "https://registry.npmjs.org/heap/-/heap-0.2.6.tgz",
"_spec": "0.2.6",
"_where": "/codebuild/output/src566918775/src/packages/@aws-cdk/cdk",
"_shasum": "087e1f10b046932fc8594dd9e6d378afc9d1e5ac",
"_spec": "heap@>= 0.2.0",
"_where": "/codebuild/output/src393726274/src/packages/@aws-cdk/cdk/node_modules/difflib",
"author": {

@@ -39,3 +34,5 @@ "name": "Xueqiao Xu",

},
"bundleDependencies": false,
"dependencies": {},
"deprecated": false,
"description": "binary heap (priority queue) algorithms (ported from Python's heapq module)",

@@ -42,0 +39,0 @@ "devDependencies": {

{
"_args": [
[
"js-base64@2.4.9",
"/codebuild/output/src566918775/src/packages/@aws-cdk/cdk"
]
],
"_from": "js-base64@2.4.9",
"_from": "js-base64@^2.4.5",
"_id": "js-base64@2.4.9",

@@ -15,10 +9,10 @@ "_inBundle": false,

"_requested": {
"type": "version",
"type": "range",
"registry": true,
"raw": "js-base64@2.4.9",
"raw": "js-base64@^2.4.5",
"name": "js-base64",
"escapedName": "js-base64",
"rawSpec": "2.4.9",
"rawSpec": "^2.4.5",
"saveSpec": null,
"fetchSpec": "2.4.9"
"fetchSpec": "^2.4.5"
},

@@ -29,4 +23,5 @@ "_requiredBy": [

"_resolved": "https://registry.npmjs.org/js-base64/-/js-base64-2.4.9.tgz",
"_spec": "2.4.9",
"_where": "/codebuild/output/src566918775/src/packages/@aws-cdk/cdk",
"_shasum": "748911fb04f48a60c4771b375cac45a80df11c03",
"_spec": "js-base64@^2.4.5",
"_where": "/codebuild/output/src393726274/src/packages/@aws-cdk/cdk",
"author": {

@@ -38,2 +33,4 @@ "name": "Dan Kogai"

},
"bundleDependencies": false,
"deprecated": false,
"description": "Yet another Base64 transcoder in pure-JS",

@@ -40,0 +37,0 @@ "devDependencies": {

{
"_args": [
[
"json-diff@0.3.1",
"/codebuild/output/src566918775/src/packages/@aws-cdk/cdk"
]
],
"_from": "json-diff@0.3.1",
"_from": "json-diff@^0.3.1",
"_id": "json-diff@0.3.1",

@@ -15,10 +9,10 @@ "_inBundle": false,

"_requested": {
"type": "version",
"type": "range",
"registry": true,
"raw": "json-diff@0.3.1",
"raw": "json-diff@^0.3.1",
"name": "json-diff",
"escapedName": "json-diff",
"rawSpec": "0.3.1",
"rawSpec": "^0.3.1",
"saveSpec": null,
"fetchSpec": "0.3.1"
"fetchSpec": "^0.3.1"
},

@@ -28,5 +22,6 @@ "_requiredBy": [

],
"_resolved": "https://registry.npmjs.org/json-diff/-/json-diff-0.3.1.tgz",
"_spec": "0.3.1",
"_where": "/codebuild/output/src566918775/src/packages/@aws-cdk/cdk",
"_resolved": "http://registry.npmjs.org/json-diff/-/json-diff-0.3.1.tgz",
"_shasum": "6dbc3ae2d25e075a7fd71bcd9874458666fb681b",
"_spec": "json-diff@^0.3.1",
"_where": "/codebuild/output/src393726274/src/packages/@aws-cdk/cdk",
"author": {

@@ -42,2 +37,3 @@ "name": "Andrey Tarantsov",

},
"bundleDependencies": false,
"dependencies": {

@@ -48,2 +44,3 @@ "cli-color": "~0.1.6",

},
"deprecated": false,
"description": "JSON diff",

@@ -50,0 +47,0 @@ "devDependencies": {

{
"_args": [
[
"wordwrap@1.0.0",
"/codebuild/output/src566918775/src/packages/@aws-cdk/cdk"
]
],
"_from": "wordwrap@1.0.0",
"_from": "wordwrap@>=0.0.2",
"_id": "wordwrap@1.0.0",

@@ -15,10 +9,10 @@ "_inBundle": false,

"_requested": {
"type": "version",
"type": "range",
"registry": true,
"raw": "wordwrap@1.0.0",
"raw": "wordwrap@>=0.0.2",
"name": "wordwrap",
"escapedName": "wordwrap",
"rawSpec": "1.0.0",
"rawSpec": ">=0.0.2",
"saveSpec": null,
"fetchSpec": "1.0.0"
"fetchSpec": ">=0.0.2"
},

@@ -29,4 +23,5 @@ "_requiredBy": [

"_resolved": "https://registry.npmjs.org/wordwrap/-/wordwrap-1.0.0.tgz",
"_spec": "1.0.0",
"_where": "/codebuild/output/src566918775/src/packages/@aws-cdk/cdk",
"_shasum": "27584810891456a4171c8d0226441ade90cbcaeb",
"_spec": "wordwrap@>=0.0.2",
"_where": "/codebuild/output/src393726274/src/packages/@aws-cdk/cdk/node_modules/dreamopt",
"author": {

@@ -40,2 +35,4 @@ "name": "James Halliday",

},
"bundleDependencies": false,
"deprecated": false,
"description": "Wrap those words. Show them at what columns to start and stop.",

@@ -42,0 +39,0 @@ "devDependencies": {

{
"name": "@aws-cdk/cdk",
"version": "0.13.0",
"version": "0.14.0",
"description": "AWS Cloud Development Kit Core Library",

@@ -56,8 +56,11 @@ "main": "lib/index.js",

"@types/js-base64": "^2.3.1",
"cdk-build-tools": "^0.13.0",
"cfn2ts": "^0.13.0",
"pkglint": "^0.13.0"
"@types/lodash": "^4.14.117",
"cdk-build-tools": "^0.14.0",
"cfn2ts": "^0.14.0",
"fast-check": "^1.7.0",
"lodash": "^4.17.11",
"pkglint": "^0.14.0"
},
"dependencies": {
"@aws-cdk/cx-api": "^0.13.0",
"@aws-cdk/cx-api": "^0.14.0",
"js-base64": "^2.4.5",

@@ -64,0 +67,0 @@ "json-diff": "^0.3.1"

@@ -9,14 +9,3 @@ "use strict";

});
test.deepEqual(lib_1.resolve(arn), { 'Fn::Join': ['',
['arn',
':',
{ Ref: 'AWS::Partition' },
':',
'sqs',
':',
{ Ref: 'AWS::Region' },
':',
{ Ref: 'AWS::AccountId' },
':',
'myqueuename']] });
test.deepEqual(lib_1.resolve(arn), lib_1.resolve(new lib_1.FnConcat('arn', ':', { Ref: 'AWS::Partition' }, ':', 'sqs', ':', { Ref: 'AWS::Region' }, ':', { Ref: 'AWS::AccountId' }, ':', 'myqueuename')));
test.done();

@@ -33,16 +22,3 @@ },

});
test.deepEqual(lib_1.resolve(arn), { 'Fn::Join': ['',
['arn',
':',
'aws-cn',
':',
'dynamodb',
':',
'us-east-1',
':',
'123456789012',
':',
'table',
'/',
'mytable/stream/label']] });
test.deepEqual(lib_1.resolve(arn), lib_1.resolve(new lib_1.FnConcat('arn', ':', 'aws-cn', ':', 'dynamodb', ':', 'us-east-1', ':', '123456789012', ':', 'table', '/', 'mytable/stream/label')));
test.done();

@@ -58,20 +34,3 @@ },

});
test.deepEqual(lib_1.resolve(arn), {
'Fn::Join': [
'',
[
'arn',
':',
'aws-cn',
':',
's3',
':',
'',
':',
'',
':',
'my-bucket',
]
]
});
test.deepEqual(lib_1.resolve(arn), lib_1.resolve(new lib_1.FnConcat('arn', ':', 'aws-cn', ':', 's3', ':', '', ':', '', ':', 'my-bucket')));
test.done();

@@ -86,16 +45,3 @@ },

});
test.deepEqual(lib_1.resolve(arn), { 'Fn::Join': ['',
['arn',
':',
{ Ref: 'AWS::Partition' },
':',
'codedeploy',
':',
{ Ref: 'AWS::Region' },
':',
{ Ref: 'AWS::AccountId' },
':',
'application',
':',
'WordPress_App']] });
test.deepEqual(lib_1.resolve(arn), lib_1.resolve(new lib_1.FnConcat('arn', ':', { Ref: 'AWS::Partition' }, ':', 'codedeploy', ':', { Ref: 'AWS::Region' }, ':', { Ref: 'AWS::AccountId' }, ':', 'application', ':', 'WordPress_App')));
test.done();

@@ -199,2 +145,2 @@ },

};
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"test.arn.js","sourceRoot":"","sources":["test.arn.ts"],"names":[],"mappings":"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","sourcesContent":["import { Test } from 'nodeunit';\nimport { ArnComponents, ArnUtils, resolve, Token } from '../../lib';\n\nexport = {\n  'create from components with defaults'(test: Test) {\n    const arn = ArnUtils.fromComponents({\n      service: 'sqs',\n      resource: 'myqueuename'\n    });\n\n    test.deepEqual(resolve(arn), { 'Fn::Join':\n    [ '',\n      [ 'arn',\n      ':',\n      { Ref: 'AWS::Partition' },\n      ':',\n      'sqs',\n      ':',\n      { Ref: 'AWS::Region' },\n      ':',\n      { Ref: 'AWS::AccountId' },\n      ':',\n      'myqueuename' ] ] });\n    test.done();\n  },\n\n  'create from components with specific values for the various components'(test: Test) {\n    const arn = ArnUtils.fromComponents({\n      service: 'dynamodb',\n      resource: 'table',\n      account: '123456789012',\n      region: 'us-east-1',\n      partition: 'aws-cn',\n      resourceName: 'mytable/stream/label'\n    });\n\n    test.deepEqual(resolve(arn), { 'Fn::Join':\n    [ '',\n      [ 'arn',\n      ':',\n      'aws-cn',\n      ':',\n      'dynamodb',\n      ':',\n      'us-east-1',\n      ':',\n      '123456789012',\n      ':',\n      'table',\n      '/',\n      'mytable/stream/label' ] ] });\n    test.done();\n  },\n\n  'allow empty string in components'(test: Test) {\n    const arn = ArnUtils.fromComponents({\n      service: 's3',\n      resource: 'my-bucket',\n      account: '',\n      region: '',\n      partition: 'aws-cn',\n    });\n\n    test.deepEqual(resolve(arn), {\n      'Fn::Join': [\n        '',\n        [\n          'arn',\n          ':',\n          'aws-cn',\n          ':',\n          's3',\n          ':',\n          '',\n          ':',\n          '',\n          ':',\n          'my-bucket',\n        ]\n      ]\n    });\n\n    test.done();\n  },\n\n  'resourcePathSep can be set to \":\" instead of the default \"/\"'(test: Test) {\n    const arn = ArnUtils.fromComponents({\n      service: 'codedeploy',\n      resource: 'application',\n      sep: ':',\n      resourceName: 'WordPress_App'\n    });\n\n    test.deepEqual(resolve(arn), { 'Fn::Join':\n    [ '',\n      [ 'arn',\n      ':',\n      { Ref: 'AWS::Partition' },\n      ':',\n      'codedeploy',\n      ':',\n      { Ref: 'AWS::Region' },\n      ':',\n      { Ref: 'AWS::AccountId' },\n      ':',\n      'application',\n      ':',\n      'WordPress_App' ] ] });\n    test.done();\n  },\n\n  'fails if resourcePathSep is neither \":\" nor \"/\"'(test: Test) {\n    test.throws(() => ArnUtils.fromComponents({\n      service: 'foo',\n      resource: 'bar',\n      sep: 'x' }));\n    test.done();\n  },\n\n  'Arn.parse(s)': {\n\n    'fails': {\n      'if doesn\\'t start with \"arn:\"'(test: Test) {\n        test.throws(() => ArnUtils.parse(\"barn:foo:x:a:1:2\"), /ARNs must start with \"arn:\": barn:foo/);\n        test.done();\n      },\n\n      'if the ARN doesnt have enough components'(test: Test) {\n        test.throws(() => ArnUtils.parse('arn:is:too:short'), /ARNs must have at least 6 components: arn:is:too:short/);\n        test.done();\n      },\n\n      'if \"service\" is not specified'(test: Test) {\n        test.throws(() => ArnUtils.parse('arn:aws::4:5:6'), /The `service` component \\(3rd component\\) is required/);\n        test.done();\n      },\n\n      'if \"resource\" is not specified'(test: Test) {\n        test.throws(() => ArnUtils.parse('arn:aws:service:::'), /The `resource` component \\(6th component\\) is required/);\n        test.done();\n      }\n    },\n\n    'various successful parses'(test: Test) {\n      const tests: { [arn: string]: ArnComponents } = {\n        'arn:aws:a4b:region:accountid:resourcetype/resource': {\n          partition: 'aws',\n          service: 'a4b',\n          region: 'region',\n          account: 'accountid',\n          resource: 'resourcetype',\n          resourceName: 'resource',\n          sep: '/'\n        },\n        'arn:aws:apigateway:us-east-1::a123456789012bc3de45678901f23a45:/test/mydemoresource/*': {\n          partition: 'aws',\n          service: 'apigateway',\n          region: 'us-east-1',\n          resource: 'a123456789012bc3de45678901f23a45',\n          sep: ':',\n          resourceName: '/test/mydemoresource/*'\n        },\n        'arn:aws-cn:cloud9::123456789012:environment:81e900317347585a0601e04c8d52eaEX': {\n          partition: 'aws-cn',\n          service: 'cloud9',\n          account: '123456789012',\n          resource: 'environment',\n          resourceName: '81e900317347585a0601e04c8d52eaEX',\n          sep: ':'\n        },\n        'arn::cognito-sync:::identitypool/us-east-1:1a1a1a1a-ffff-1111-9999-12345678:bla': {\n          service: 'cognito-sync',\n          resource: 'identitypool',\n          resourceName: 'us-east-1:1a1a1a1a-ffff-1111-9999-12345678:bla',\n          sep: '/'\n        },\n        'arn:aws:s3:::my_corporate_bucket': {\n          partition: 'aws',\n          service: 's3',\n          resource: 'my_corporate_bucket'\n        }\n      };\n\n      Object.keys(tests).forEach(arn => {\n        const expected = tests[arn];\n        test.deepEqual(ArnUtils.parse(arn), expected, arn);\n      });\n\n      test.done();\n    },\n\n    'a Token with : separator'(test: Test) {\n      const theToken = { Ref: 'SomeParameter' };\n      const parsed = ArnUtils.parseToken(new Token(() => theToken), ':');\n\n      test.deepEqual(resolve(parsed.partition), { 'Fn::Select': [ 1, { 'Fn::Split': [ ':', theToken ]} ]});\n      test.deepEqual(resolve(parsed.service), { 'Fn::Select': [ 2, { 'Fn::Split': [ ':', theToken ]} ]});\n      test.deepEqual(resolve(parsed.region), { 'Fn::Select': [ 3, { 'Fn::Split': [ ':', theToken ]} ]});\n      test.deepEqual(resolve(parsed.account), { 'Fn::Select': [ 4, { 'Fn::Split': [ ':', theToken ]} ]});\n      test.deepEqual(resolve(parsed.resource), { 'Fn::Select': [ 5, { 'Fn::Split': [ ':', theToken ]} ]});\n      test.deepEqual(resolve(parsed.resourceName), { 'Fn::Select': [ 6, { 'Fn::Split': [ ':', theToken ]} ]});\n      test.equal(parsed.sep, ':');\n\n      test.done();\n    },\n\n    'a Token with / separator'(test: Test) {\n      const theToken = { Ref: 'SomeParameter' };\n      const parsed = ArnUtils.parseToken(new Token(() => theToken));\n\n      test.equal(parsed.sep, '/');\n\n      // tslint:disable-next-line:max-line-length\n      test.deepEqual(resolve(parsed.resource), { 'Fn::Select': [ 0, { 'Fn::Split': [ '/', { 'Fn::Select': [ 5, { 'Fn::Split': [ ':', theToken ]} ]} ]} ]});\n      // tslint:disable-next-line:max-line-length\n      test.deepEqual(resolve(parsed.resourceName), { 'Fn::Select': [ 1, { 'Fn::Split': [ '/', { 'Fn::Select': [ 5, { 'Fn::Split': [ ':', theToken ]} ]} ]} ]});\n\n      test.done();\n    }\n  },\n\n};\n"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"test.arn.js","sourceRoot":"","sources":["test.arn.ts"],"names":[],"mappings":"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","sourcesContent":["import { Test } from 'nodeunit';\nimport { ArnComponents, ArnUtils, FnConcat, resolve, Token } from '../../lib';\n\nexport = {\n  'create from components with defaults'(test: Test) {\n    const arn = ArnUtils.fromComponents({\n      service: 'sqs',\n      resource: 'myqueuename'\n    });\n\n    test.deepEqual(resolve(arn),\n                   resolve(new FnConcat('arn',\n                                        ':',\n                                        { Ref: 'AWS::Partition' },\n                                        ':',\n                                        'sqs',\n                                        ':',\n                                        { Ref: 'AWS::Region' },\n                                        ':',\n                                        { Ref: 'AWS::AccountId' },\n                                        ':',\n                                        'myqueuename')));\n    test.done();\n  },\n\n  'create from components with specific values for the various components'(test: Test) {\n    const arn = ArnUtils.fromComponents({\n      service: 'dynamodb',\n      resource: 'table',\n      account: '123456789012',\n      region: 'us-east-1',\n      partition: 'aws-cn',\n      resourceName: 'mytable/stream/label'\n    });\n\n    test.deepEqual(resolve(arn),\n                   resolve(new FnConcat('arn',\n                                        ':',\n                                        'aws-cn',\n                                        ':',\n                                        'dynamodb',\n                                        ':',\n                                        'us-east-1',\n                                        ':',\n                                        '123456789012',\n                                        ':',\n                                        'table',\n                                        '/',\n                                        'mytable/stream/label')));\n    test.done();\n  },\n\n  'allow empty string in components'(test: Test) {\n    const arn = ArnUtils.fromComponents({\n      service: 's3',\n      resource: 'my-bucket',\n      account: '',\n      region: '',\n      partition: 'aws-cn',\n    });\n\n    test.deepEqual(resolve(arn),\n                   resolve(new FnConcat('arn',\n                                        ':',\n                                        'aws-cn',\n                                        ':',\n                                        's3',\n                                        ':',\n                                        '',\n                                        ':',\n                                        '',\n                                        ':',\n                                        'my-bucket')));\n\n    test.done();\n  },\n\n  'resourcePathSep can be set to \":\" instead of the default \"/\"'(test: Test) {\n    const arn = ArnUtils.fromComponents({\n      service: 'codedeploy',\n      resource: 'application',\n      sep: ':',\n      resourceName: 'WordPress_App'\n    });\n\n    test.deepEqual(resolve(arn),\n                   resolve(new FnConcat('arn',\n                                        ':',\n                                        { Ref: 'AWS::Partition' },\n                                        ':',\n                                        'codedeploy',\n                                        ':',\n                                        { Ref: 'AWS::Region' },\n                                        ':',\n                                        { Ref: 'AWS::AccountId' },\n                                        ':',\n                                        'application',\n                                        ':',\n                                        'WordPress_App')));\n    test.done();\n  },\n\n  'fails if resourcePathSep is neither \":\" nor \"/\"'(test: Test) {\n    test.throws(() => ArnUtils.fromComponents({\n      service: 'foo',\n      resource: 'bar',\n      sep: 'x' }));\n    test.done();\n  },\n\n  'Arn.parse(s)': {\n\n    'fails': {\n      'if doesn\\'t start with \"arn:\"'(test: Test) {\n        test.throws(() => ArnUtils.parse(\"barn:foo:x:a:1:2\"), /ARNs must start with \"arn:\": barn:foo/);\n        test.done();\n      },\n\n      'if the ARN doesnt have enough components'(test: Test) {\n        test.throws(() => ArnUtils.parse('arn:is:too:short'), /ARNs must have at least 6 components: arn:is:too:short/);\n        test.done();\n      },\n\n      'if \"service\" is not specified'(test: Test) {\n        test.throws(() => ArnUtils.parse('arn:aws::4:5:6'), /The `service` component \\(3rd component\\) is required/);\n        test.done();\n      },\n\n      'if \"resource\" is not specified'(test: Test) {\n        test.throws(() => ArnUtils.parse('arn:aws:service:::'), /The `resource` component \\(6th component\\) is required/);\n        test.done();\n      }\n    },\n\n    'various successful parses'(test: Test) {\n      const tests: { [arn: string]: ArnComponents } = {\n        'arn:aws:a4b:region:accountid:resourcetype/resource': {\n          partition: 'aws',\n          service: 'a4b',\n          region: 'region',\n          account: 'accountid',\n          resource: 'resourcetype',\n          resourceName: 'resource',\n          sep: '/'\n        },\n        'arn:aws:apigateway:us-east-1::a123456789012bc3de45678901f23a45:/test/mydemoresource/*': {\n          partition: 'aws',\n          service: 'apigateway',\n          region: 'us-east-1',\n          resource: 'a123456789012bc3de45678901f23a45',\n          sep: ':',\n          resourceName: '/test/mydemoresource/*'\n        },\n        'arn:aws-cn:cloud9::123456789012:environment:81e900317347585a0601e04c8d52eaEX': {\n          partition: 'aws-cn',\n          service: 'cloud9',\n          account: '123456789012',\n          resource: 'environment',\n          resourceName: '81e900317347585a0601e04c8d52eaEX',\n          sep: ':'\n        },\n        'arn::cognito-sync:::identitypool/us-east-1:1a1a1a1a-ffff-1111-9999-12345678:bla': {\n          service: 'cognito-sync',\n          resource: 'identitypool',\n          resourceName: 'us-east-1:1a1a1a1a-ffff-1111-9999-12345678:bla',\n          sep: '/'\n        },\n        'arn:aws:s3:::my_corporate_bucket': {\n          partition: 'aws',\n          service: 's3',\n          resource: 'my_corporate_bucket'\n        }\n      };\n\n      Object.keys(tests).forEach(arn => {\n        const expected = tests[arn];\n        test.deepEqual(ArnUtils.parse(arn), expected, arn);\n      });\n\n      test.done();\n    },\n\n    'a Token with : separator'(test: Test) {\n      const theToken = { Ref: 'SomeParameter' };\n      const parsed = ArnUtils.parseToken(new Token(() => theToken), ':');\n\n      test.deepEqual(resolve(parsed.partition), { 'Fn::Select': [ 1, { 'Fn::Split': [ ':', theToken ]} ]});\n      test.deepEqual(resolve(parsed.service), { 'Fn::Select': [ 2, { 'Fn::Split': [ ':', theToken ]} ]});\n      test.deepEqual(resolve(parsed.region), { 'Fn::Select': [ 3, { 'Fn::Split': [ ':', theToken ]} ]});\n      test.deepEqual(resolve(parsed.account), { 'Fn::Select': [ 4, { 'Fn::Split': [ ':', theToken ]} ]});\n      test.deepEqual(resolve(parsed.resource), { 'Fn::Select': [ 5, { 'Fn::Split': [ ':', theToken ]} ]});\n      test.deepEqual(resolve(parsed.resourceName), { 'Fn::Select': [ 6, { 'Fn::Split': [ ':', theToken ]} ]});\n      test.equal(parsed.sep, ':');\n\n      test.done();\n    },\n\n    'a Token with / separator'(test: Test) {\n      const theToken = { Ref: 'SomeParameter' };\n      const parsed = ArnUtils.parseToken(new Token(() => theToken));\n\n      test.equal(parsed.sep, '/');\n\n      // tslint:disable-next-line:max-line-length\n      test.deepEqual(resolve(parsed.resource), { 'Fn::Select': [ 0, { 'Fn::Split': [ '/', { 'Fn::Select': [ 5, { 'Fn::Split': [ ':', theToken ]} ]} ]} ]});\n      // tslint:disable-next-line:max-line-length\n      test.deepEqual(resolve(parsed.resourceName), { 'Fn::Select': [ 1, { 'Fn::Split': [ '/', { 'Fn::Select': [ 5, { 'Fn::Split': [ ':', theToken ]} ]} ]} ]});\n\n      test.done();\n    }\n  },\n\n};\n"]}
"use strict";
const fc = require("fast-check");
const _ = require("lodash");
const nodeunit = require("nodeunit");
const fn = require("../../lib/cloudformation/fn");
const tokens_1 = require("../../lib/core/tokens");
function asyncTest(cb) {
return async (test) => {
let error;
try {
await cb(test);
}
catch (e) {
error = e;
}
finally {
test.doesNotThrow(() => {
if (error) {
throw error;
}
});
test.done();
}
};
}
const nonEmptyString = fc.string(1, 16);
const tokenish = fc.array(nonEmptyString, 2, 2).map(arr => ({ [arr[0]]: arr[1] }));
const anyValue = fc.oneof(nonEmptyString, tokenish);
module.exports = nodeunit.testCase({
'Fn::Join': {
FnJoin: {
'rejects empty list of arguments to join'(test) {
test.throws(() => new fn.FnJoin('.', []));
test.done();
}
}
},
'resolves to the value if only one value is joined': asyncTest(async () => {
await fc.assert(fc.property(fc.string(), anyValue, (delimiter, value) => _.isEqual(tokens_1.resolve(new fn.FnJoin(delimiter, [value])), value)), { verbose: true });
}),
'pre-concatenates string literals': asyncTest(async () => {
await fc.assert(fc.property(fc.string(), fc.array(nonEmptyString, 1, 15), (delimiter, values) => tokens_1.resolve(new fn.FnJoin(delimiter, values)) === values.join(delimiter)), { verbose: true });
}),
'pre-concatenates around tokens': asyncTest(async () => {
await fc.assert(fc.property(fc.string(), fc.array(nonEmptyString, 1, 3), tokenish, fc.array(nonEmptyString, 1, 3), (delimiter, prefix, obj, suffix) => _.isEqual(tokens_1.resolve(new fn.FnJoin(delimiter, [...prefix, obj, ...suffix])), { 'Fn::Join': [delimiter, [prefix.join(delimiter), obj, suffix.join(delimiter)]] })), { verbose: true, seed: 1539874645005, path: "0:0:0:0:0:0:0:0:0" });
}),
'flattens joins nested under joins with same delimiter': asyncTest(async () => {
await fc.assert(fc.property(fc.string(), fc.array(anyValue), fc.array(anyValue, 1, 3), fc.array(anyValue), (delimiter, prefix, nested, suffix) =>
// Gonna test
_.isEqual(tokens_1.resolve(new fn.FnJoin(delimiter, [...prefix, new fn.FnJoin(delimiter, nested), ...suffix])), tokens_1.resolve(new fn.FnJoin(delimiter, [...prefix, ...nested, ...suffix])))), { verbose: true });
}),
'does not flatten joins nested under joins with different delimiter': asyncTest(async () => {
await fc.assert(fc.property(fc.string(), fc.string(), fc.array(anyValue, 1, 3), fc.array(tokenish, 2, 3), fc.array(anyValue, 3), (delimiter1, delimiter2, prefix, nested, suffix) => {
fc.pre(delimiter1 !== delimiter2);
const join = new fn.FnJoin(delimiter1, [...prefix, new fn.FnJoin(delimiter2, nested), ...suffix]);
const resolved = tokens_1.resolve(join);
return resolved['Fn::Join'][1].find((e) => typeof e === 'object'
&& ('Fn::Join' in e)
&& e['Fn::Join'][0] === delimiter2) != null;
}), { verbose: true });
}),
},
});
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGVzdC5mbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbInRlc3QuZm4udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBLHFDQUFzQztBQUN0QyxrREFBbUQ7QUFFbkQsaUJBQVMsUUFBUSxDQUFDLFFBQVEsQ0FBQztJQUN6QixVQUFVLEVBQUU7UUFDVix5Q0FBeUMsQ0FBQyxJQUFtQjtZQUMzRCxJQUFJLENBQUMsTUFBTSxDQUFDLEdBQUcsRUFBRSxDQUFDLElBQUksRUFBRSxDQUFDLE1BQU0sQ0FBQyxHQUFHLEVBQUUsRUFBRSxDQUFDLENBQUMsQ0FBQztZQUMxQyxJQUFJLENBQUMsSUFBSSxFQUFFLENBQUM7UUFDZCxDQUFDO0tBQ0Y7Q0FDRixDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgbm9kZXVuaXQgPSByZXF1aXJlKCdub2RldW5pdCcpO1xuaW1wb3J0IGZuID0gcmVxdWlyZSgnLi4vLi4vbGliL2Nsb3VkZm9ybWF0aW9uL2ZuJyk7XG5cbmV4cG9ydCA9IG5vZGV1bml0LnRlc3RDYXNlKHtcbiAgJ0ZuOjpKb2luJzoge1xuICAgICdyZWplY3RzIGVtcHR5IGxpc3Qgb2YgYXJndW1lbnRzIHRvIGpvaW4nKHRlc3Q6IG5vZGV1bml0LlRlc3QpIHtcbiAgICAgIHRlc3QudGhyb3dzKCgpID0+IG5ldyBmbi5GbkpvaW4oJy4nLCBbXSkpO1xuICAgICAgdGVzdC5kb25lKCk7XG4gICAgfVxuICB9XG59KTtcbiJdfQ==
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"test.fn.js","sourceRoot":"","sources":["test.fn.ts"],"names":[],"mappings":"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","sourcesContent":["import fc = require('fast-check');\nimport _ = require('lodash');\nimport nodeunit = require('nodeunit');\nimport fn = require('../../lib/cloudformation/fn');\nimport { resolve } from '../../lib/core/tokens';\n\nfunction asyncTest(cb: (test: nodeunit.Test) => Promise<void>): (test: nodeunit.Test) => void {\n  return async (test: nodeunit.Test) => {\n    let error: Error;\n    try {\n      await cb(test);\n    } catch (e) {\n      error = e;\n    } finally {\n      test.doesNotThrow(() => {\n        if (error) { throw error; }\n      });\n      test.done();\n    }\n  };\n}\n\nconst nonEmptyString = fc.string(1, 16);\nconst tokenish = fc.array(nonEmptyString, 2, 2).map(arr => ({ [arr[0]]: arr[1] }));\nconst anyValue = fc.oneof<any>(nonEmptyString, tokenish);\n\nexport = nodeunit.testCase({\n  FnJoin: {\n    'rejects empty list of arguments to join'(test: nodeunit.Test) {\n      test.throws(() => new fn.FnJoin('.', []));\n      test.done();\n    },\n    'resolves to the value if only one value is joined': asyncTest(async () => {\n      await fc.assert(\n        fc.property(\n          fc.string(), anyValue,\n          (delimiter, value) => _.isEqual(resolve(new fn.FnJoin(delimiter, [value])), value)\n        ),\n        { verbose: true }\n      );\n    }),\n    'pre-concatenates string literals': asyncTest(async () => {\n      await fc.assert(\n        fc.property(\n          fc.string(), fc.array(nonEmptyString, 1, 15),\n          (delimiter, values) => resolve(new fn.FnJoin(delimiter, values)) === values.join(delimiter)\n        ),\n        { verbose: true }\n      );\n    }),\n    'pre-concatenates around tokens': asyncTest(async () => {\n      await fc.assert(\n        fc.property(\n          fc.string(), fc.array(nonEmptyString, 1, 3), tokenish, fc.array(nonEmptyString, 1, 3),\n          (delimiter, prefix, obj, suffix) =>\n            _.isEqual(resolve(new fn.FnJoin(delimiter, [...prefix, obj, ...suffix])),\n                      { 'Fn::Join': [delimiter, [prefix.join(delimiter), obj, suffix.join(delimiter)]] })\n        ),\n        { verbose: true, seed: 1539874645005, path: \"0:0:0:0:0:0:0:0:0\" }\n      );\n    }),\n    'flattens joins nested under joins with same delimiter': asyncTest(async () => {\n      await fc.assert(\n        fc.property(\n          fc.string(), fc.array(anyValue),\n                      fc.array(anyValue, 1, 3),\n                      fc.array(anyValue),\n          (delimiter, prefix, nested, suffix) =>\n            // Gonna test\n            _.isEqual(resolve(new fn.FnJoin(delimiter, [...prefix, new fn.FnJoin(delimiter, nested), ...suffix])),\n                      resolve(new fn.FnJoin(delimiter, [...prefix, ...nested, ...suffix])))\n        ),\n        { verbose: true }\n      );\n    }),\n    'does not flatten joins nested under joins with different delimiter': asyncTest(async () => {\n      await fc.assert(\n        fc.property(\n          fc.string(), fc.string(),\n          fc.array(anyValue, 1, 3),\n          fc.array(tokenish, 2, 3),\n          fc.array(anyValue, 3),\n          (delimiter1, delimiter2, prefix, nested, suffix) => {\n            fc.pre(delimiter1 !== delimiter2);\n            const join = new fn.FnJoin(delimiter1, [...prefix, new fn.FnJoin(delimiter2, nested), ...suffix]);\n            const resolved = resolve(join);\n            return resolved['Fn::Join'][1].find((e: any) => typeof e === 'object'\n                                                        && ('Fn::Join' in e)\n                                                        && e['Fn::Join'][0] === delimiter2) != null;\n          }\n        ),\n        { verbose: true }\n      );\n    }),\n  },\n});\n"]}

@@ -15,3 +15,2 @@ import { Test } from 'nodeunit';

'the "type" property is required when creating a resource'(test: Test): void;
'the "name" property is deleted when synthesizing into a CloudFormation resource'(test: Test): void;
'removal policy is a high level abstraction of deletion policy used by l2'(test: Test): void;

@@ -18,0 +17,0 @@ 'addDependency adds all dependencyElements of dependent constructs'(test: Test): void;

@@ -193,3 +193,3 @@ "use strict";

// GIVEN
const token = new lib_1.CloudFormationToken(() => 'woof woof');
const token = new lib_1.CloudFormationToken(() => ({ woof: 'woof' }));
// WHEN

@@ -200,3 +200,3 @@ const stringified = `The dog says: ${token}`;

test.deepEqual(resolved, {
'Fn::Join': ['', ['The dog says: ', 'woof woof']]
'Fn::Join': ['', ['The dog says: ', { woof: 'woof' }]]
});

@@ -284,2 +284,2 @@ test.done();

};
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"test.tokens.js","sourceRoot":"","sources":["test.tokens.ts"],"names":[],"mappings":"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","sourcesContent":["import { Test } from 'nodeunit';\nimport { CloudFormationToken, resolve, Token, unresolved } from '../../lib';\nimport { evaluateCFN } from '../cloudformation/evaluate-cfn';\n\nexport = {\n  'resolve a plain old object should just return the object'(test: Test) {\n    const obj = { PlainOldObject: 123, Array: [ 1, 2, 3 ] };\n    test.deepEqual(resolve(obj), obj);\n    test.done();\n  },\n\n  'if a value is an object with a token value, it will be evaluated'(test: Test) {\n    const obj = {\n      RegularValue: 'hello',\n      LazyValue: new Token('World')\n    };\n\n    test.deepEqual(resolve(obj), {\n      RegularValue: 'hello',\n      LazyValue: 'World'\n    });\n\n    test.done();\n  },\n\n  'tokens are evaluated anywhere in the object tree'(test: Test) {\n    const obj = new Promise1();\n    const actual = resolve({ Obj: obj });\n\n    test.deepEqual(actual, {\n      Obj: [\n        {\n        Data: {\n          stringProp: \"hello\",\n          numberProp: 1234\n        },\n        Recurse: 42\n        },\n        {\n        Data: {\n          stringProp: \"hello\",\n          numberProp: 1234\n        },\n        Recurse: 42\n        }\n      ]\n    });\n\n    test.done();\n  },\n\n  'tokens are evaluated recursively'(test: Test) {\n    const obj = new Promise1();\n    const actual = resolve(new Token(() => ({ Obj: obj })));\n\n    test.deepEqual(actual, {\n      Obj: [\n        {\n        Data: {\n          stringProp: \"hello\",\n          numberProp: 1234\n        },\n        Recurse: 42\n        },\n        {\n        Data: {\n          stringProp: \"hello\",\n          numberProp: 1234\n        },\n        Recurse: 42\n        }\n      ]\n    });\n\n    test.done();\n  },\n\n  'empty arrays or objects are kept'(test: Test) {\n    test.deepEqual(resolve({ }), { });\n    test.deepEqual(resolve([ ]), [ ]);\n\n    const obj = {\n      Prop1: 1234,\n      Prop2: { },\n      Prop3: [ ],\n      Prop4: 'hello',\n      Prop5: {\n        PropA: { },\n        PropB: {\n          PropC: [ undefined, undefined ],\n          PropD: 'Yoohoo'\n        }\n      }\n    };\n\n    test.deepEqual(resolve(obj), {\n      Prop1: 1234,\n      Prop2: { },\n      Prop3: [ ],\n      Prop4: 'hello',\n      Prop5: {\n        PropA: { },\n        PropB: {\n          PropC: [ ],\n          PropD: 'Yoohoo'\n        }\n      }\n    });\n\n    test.done();\n  },\n\n  'if an object has a \"resolve\" property that is not a function, it is not considered a token'(test: Test) {\n    test.deepEqual(resolve({ a_token: { resolve: () => 78787 }}), { a_token: 78787 });\n    test.deepEqual(resolve({ not_a_token: { resolve: 12 } }),   { not_a_token: { resolve: 12 } });\n    test.done();\n  },\n\n  // tslint:disable-next-line:max-line-length\n  'if a resolvable object inherits from a class that is also resolvable, the \"constructor\" function will not get in the way (uses Object.keys instead of \"for in\")'(test: Test) {\n    test.deepEqual(resolve({ prop: new DataType() }), { prop: { foo: 12, goo: 'hello' } });\n    test.done();\n  },\n\n  'isToken(obj) can be used to determine if an object is a token'(test: Test) {\n    test.ok(unresolved({ resolve: () => 123 }));\n    test.ok(unresolved({ a: 1, b: 2, resolve: () => 'hello' }));\n    test.ok(!unresolved({ a: 1, b: 2, resolve: 3 }));\n    test.done();\n  },\n\n  'Token can be used to create tokens that contain a constant value'(test: Test) {\n    test.equal(resolve(new Token(12)), 12);\n    test.equal(resolve(new Token('hello')), 'hello');\n    test.deepEqual(resolve(new Token([ 'hi', 'there' ])), [ 'hi', 'there' ]);\n    test.done();\n  },\n\n  'resolving leaves a Date object in working order'(test: Test) {\n    const date = new Date('2000-01-01');\n    const resolved = resolve(date);\n\n    test.equal(date.toString(), resolved.toString());\n    test.done();\n  },\n\n  'tokens can be stringified and evaluated to conceptual value'(test: Test) {\n    // GIVEN\n    const token = new Token(() => 'woof woof');\n\n    // WHEN\n    const stringified = `The dog says: ${token}`;\n    const resolved = resolve(stringified);\n\n    // THEN\n    test.deepEqual(evaluateCFN(resolved), 'The dog says: woof woof');\n    test.done();\n  },\n\n  'Tokens stringification and reversing of CloudFormation Tokens is implemented using Fn::Join'(test: Test) {\n    // GIVEN\n    const token = new CloudFormationToken(() => 'woof woof');\n\n    // WHEN\n    const stringified = `The dog says: ${token}`;\n    const resolved = resolve(stringified);\n\n    // THEN\n    test.deepEqual(resolved, {\n      'Fn::Join': ['', ['The dog says: ', 'woof woof']]\n    });\n    test.done();\n  },\n\n  'Doubly nested strings evaluate correctly in scalar context'(test: Test) {\n    // GIVEN\n    const token1 = new Token(() => \"world\");\n    const token2 = new Token(() => `hello ${token1}`);\n\n    // WHEN\n    const resolved1 = resolve(token2.toString());\n    const resolved2 = resolve(token2);\n\n    // THEN\n    test.deepEqual(evaluateCFN(resolved1), \"hello world\");\n    test.deepEqual(evaluateCFN(resolved2), \"hello world\");\n\n    test.done();\n  },\n\n  'integer Tokens can be stringified and evaluate to conceptual value'(test: Test) {\n    // GIVEN\n    for (const token of literalTokensThatResolveTo(1)) {\n      // WHEN\n      const stringified = `the number is ${token}`;\n      const resolved = resolve(stringified);\n\n      // THEN\n      test.deepEqual(evaluateCFN(resolved), 'the number is 1');\n    }\n    test.done();\n  },\n\n  'intrinsic Tokens can be stringified and evaluate to conceptual value'(test: Test) {\n    // GIVEN\n    for (const bucketName of cloudFormationTokensThatResolveTo({ Ref: 'MyBucket' })) {\n      // WHEN\n      const resolved = resolve(`my bucket is named ${bucketName}`);\n\n      // THEN\n      const context = {MyBucket: 'TheName'};\n      test.equal(evaluateCFN(resolved, context), 'my bucket is named TheName');\n    }\n\n    test.done();\n  },\n\n  'tokens resolve properly in initial position'(test: Test) {\n    // GIVEN\n    for (const token of tokensThatResolveTo('Hello')) {\n      // WHEN\n      const resolved = resolve(`${token} world`);\n\n      // THEN\n      test.equal(evaluateCFN(resolved), 'Hello world');\n    }\n\n    test.done();\n  },\n\n  'side-by-side Tokens resolve correctly'(test: Test) {\n    // GIVEN\n    for (const token1 of tokensThatResolveTo('Hello ')) {\n      for (const token2 of tokensThatResolveTo('world')) {\n        // WHEN\n        const resolved = resolve(`${token1}${token2}`);\n\n        // THEN\n        test.equal(evaluateCFN(resolved), 'Hello world');\n      }\n    }\n\n    test.done();\n  },\n\n  'tokens can be used in hash keys but must resolve to a string'(test: Test) {\n    // GIVEN\n    const token = new Token(() => 'I am a string');\n\n    // WHEN\n    const s = {\n      [token.toString()]: `boom ${token}`\n    };\n\n    // THEN\n    test.deepEqual(resolve(s), { 'I am a string': 'boom I am a string' });\n    test.done();\n  },\n\n  'fails if token in a hash key resolves to a non-string'(test: Test) {\n    // GIVEN\n    const token = new CloudFormationToken({ Ref: 'Other' });\n\n    // WHEN\n    const s = {\n      [token.toString()]: `boom ${token}`\n    };\n\n    // THEN\n    test.throws(() => resolve(s), 'The key \"${Token[TOKEN.19]}\" has been resolved to {\"Ref\":\"Other\"} but must be resolvable to a string');\n    test.done();\n  }\n};\n\nclass Promise2 extends Token {\n  public resolve() {\n    return {\n      Data: {\n        stringProp: 'hello',\n        numberProp: 1234,\n      },\n      Recurse: new Token(() => 42)\n    };\n  }\n}\n\nclass Promise1 extends Token {\n  public p2 = [ new Promise2(), new Promise2() ];\n\n  public resolve() {\n    return this.p2;\n  }\n}\n\nclass BaseDataType {\n  constructor(readonly foo: number) {\n  }\n}\n\nclass DataType extends BaseDataType {\n  public goo = 'hello';\n\n  constructor() {\n    super(12);\n  }\n}\n\n/**\n * Return various flavors of Tokens that resolve to the given value\n */\nfunction literalTokensThatResolveTo(value: any): Token[] {\n  return [\n    new Token(value),\n    new Token(() => value)\n  ];\n}\n\n/**\n * Return various flavors of Tokens that resolve to the given value\n */\nfunction cloudFormationTokensThatResolveTo(value: any): Token[] {\n  return [\n    new CloudFormationToken(value),\n    new CloudFormationToken(() => value)\n  ];\n}\n\n/**\n * Return Tokens in both flavors that resolve to the given string\n */\nfunction tokensThatResolveTo(value: string): Token[] {\n  return literalTokensThatResolveTo(value).concat(cloudFormationTokensThatResolveTo(value));\n}\n"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"test.tokens.js","sourceRoot":"","sources":["test.tokens.ts"],"names":[],"mappings":"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","sourcesContent":["import { Test } from 'nodeunit';\nimport { CloudFormationToken, resolve, Token, unresolved } from '../../lib';\nimport { evaluateCFN } from '../cloudformation/evaluate-cfn';\n\nexport = {\n  'resolve a plain old object should just return the object'(test: Test) {\n    const obj = { PlainOldObject: 123, Array: [ 1, 2, 3 ] };\n    test.deepEqual(resolve(obj), obj);\n    test.done();\n  },\n\n  'if a value is an object with a token value, it will be evaluated'(test: Test) {\n    const obj = {\n      RegularValue: 'hello',\n      LazyValue: new Token('World')\n    };\n\n    test.deepEqual(resolve(obj), {\n      RegularValue: 'hello',\n      LazyValue: 'World'\n    });\n\n    test.done();\n  },\n\n  'tokens are evaluated anywhere in the object tree'(test: Test) {\n    const obj = new Promise1();\n    const actual = resolve({ Obj: obj });\n\n    test.deepEqual(actual, {\n      Obj: [\n        {\n        Data: {\n          stringProp: \"hello\",\n          numberProp: 1234\n        },\n        Recurse: 42\n        },\n        {\n        Data: {\n          stringProp: \"hello\",\n          numberProp: 1234\n        },\n        Recurse: 42\n        }\n      ]\n    });\n\n    test.done();\n  },\n\n  'tokens are evaluated recursively'(test: Test) {\n    const obj = new Promise1();\n    const actual = resolve(new Token(() => ({ Obj: obj })));\n\n    test.deepEqual(actual, {\n      Obj: [\n        {\n        Data: {\n          stringProp: \"hello\",\n          numberProp: 1234\n        },\n        Recurse: 42\n        },\n        {\n        Data: {\n          stringProp: \"hello\",\n          numberProp: 1234\n        },\n        Recurse: 42\n        }\n      ]\n    });\n\n    test.done();\n  },\n\n  'empty arrays or objects are kept'(test: Test) {\n    test.deepEqual(resolve({ }), { });\n    test.deepEqual(resolve([ ]), [ ]);\n\n    const obj = {\n      Prop1: 1234,\n      Prop2: { },\n      Prop3: [ ],\n      Prop4: 'hello',\n      Prop5: {\n        PropA: { },\n        PropB: {\n          PropC: [ undefined, undefined ],\n          PropD: 'Yoohoo'\n        }\n      }\n    };\n\n    test.deepEqual(resolve(obj), {\n      Prop1: 1234,\n      Prop2: { },\n      Prop3: [ ],\n      Prop4: 'hello',\n      Prop5: {\n        PropA: { },\n        PropB: {\n          PropC: [ ],\n          PropD: 'Yoohoo'\n        }\n      }\n    });\n\n    test.done();\n  },\n\n  'if an object has a \"resolve\" property that is not a function, it is not considered a token'(test: Test) {\n    test.deepEqual(resolve({ a_token: { resolve: () => 78787 }}), { a_token: 78787 });\n    test.deepEqual(resolve({ not_a_token: { resolve: 12 } }),   { not_a_token: { resolve: 12 } });\n    test.done();\n  },\n\n  // tslint:disable-next-line:max-line-length\n  'if a resolvable object inherits from a class that is also resolvable, the \"constructor\" function will not get in the way (uses Object.keys instead of \"for in\")'(test: Test) {\n    test.deepEqual(resolve({ prop: new DataType() }), { prop: { foo: 12, goo: 'hello' } });\n    test.done();\n  },\n\n  'isToken(obj) can be used to determine if an object is a token'(test: Test) {\n    test.ok(unresolved({ resolve: () => 123 }));\n    test.ok(unresolved({ a: 1, b: 2, resolve: () => 'hello' }));\n    test.ok(!unresolved({ a: 1, b: 2, resolve: 3 }));\n    test.done();\n  },\n\n  'Token can be used to create tokens that contain a constant value'(test: Test) {\n    test.equal(resolve(new Token(12)), 12);\n    test.equal(resolve(new Token('hello')), 'hello');\n    test.deepEqual(resolve(new Token([ 'hi', 'there' ])), [ 'hi', 'there' ]);\n    test.done();\n  },\n\n  'resolving leaves a Date object in working order'(test: Test) {\n    const date = new Date('2000-01-01');\n    const resolved = resolve(date);\n\n    test.equal(date.toString(), resolved.toString());\n    test.done();\n  },\n\n  'tokens can be stringified and evaluated to conceptual value'(test: Test) {\n    // GIVEN\n    const token = new Token(() => 'woof woof');\n\n    // WHEN\n    const stringified = `The dog says: ${token}`;\n    const resolved = resolve(stringified);\n\n    // THEN\n    test.deepEqual(evaluateCFN(resolved), 'The dog says: woof woof');\n    test.done();\n  },\n\n  'Tokens stringification and reversing of CloudFormation Tokens is implemented using Fn::Join'(test: Test) {\n    // GIVEN\n    const token = new CloudFormationToken(() => ({ woof: 'woof' }));\n\n    // WHEN\n    const stringified = `The dog says: ${token}`;\n    const resolved = resolve(stringified);\n\n    // THEN\n    test.deepEqual(resolved, {\n      'Fn::Join': ['', ['The dog says: ', { woof: 'woof' }]]\n    });\n    test.done();\n  },\n\n  'Doubly nested strings evaluate correctly in scalar context'(test: Test) {\n    // GIVEN\n    const token1 = new Token(() => \"world\");\n    const token2 = new Token(() => `hello ${token1}`);\n\n    // WHEN\n    const resolved1 = resolve(token2.toString());\n    const resolved2 = resolve(token2);\n\n    // THEN\n    test.deepEqual(evaluateCFN(resolved1), \"hello world\");\n    test.deepEqual(evaluateCFN(resolved2), \"hello world\");\n\n    test.done();\n  },\n\n  'integer Tokens can be stringified and evaluate to conceptual value'(test: Test) {\n    // GIVEN\n    for (const token of literalTokensThatResolveTo(1)) {\n      // WHEN\n      const stringified = `the number is ${token}`;\n      const resolved = resolve(stringified);\n\n      // THEN\n      test.deepEqual(evaluateCFN(resolved), 'the number is 1');\n    }\n    test.done();\n  },\n\n  'intrinsic Tokens can be stringified and evaluate to conceptual value'(test: Test) {\n    // GIVEN\n    for (const bucketName of cloudFormationTokensThatResolveTo({ Ref: 'MyBucket' })) {\n      // WHEN\n      const resolved = resolve(`my bucket is named ${bucketName}`);\n\n      // THEN\n      const context = {MyBucket: 'TheName'};\n      test.equal(evaluateCFN(resolved, context), 'my bucket is named TheName');\n    }\n\n    test.done();\n  },\n\n  'tokens resolve properly in initial position'(test: Test) {\n    // GIVEN\n    for (const token of tokensThatResolveTo('Hello')) {\n      // WHEN\n      const resolved = resolve(`${token} world`);\n\n      // THEN\n      test.equal(evaluateCFN(resolved), 'Hello world');\n    }\n\n    test.done();\n  },\n\n  'side-by-side Tokens resolve correctly'(test: Test) {\n    // GIVEN\n    for (const token1 of tokensThatResolveTo('Hello ')) {\n      for (const token2 of tokensThatResolveTo('world')) {\n        // WHEN\n        const resolved = resolve(`${token1}${token2}`);\n\n        // THEN\n        test.equal(evaluateCFN(resolved), 'Hello world');\n      }\n    }\n\n    test.done();\n  },\n\n  'tokens can be used in hash keys but must resolve to a string'(test: Test) {\n    // GIVEN\n    const token = new Token(() => 'I am a string');\n\n    // WHEN\n    const s = {\n      [token.toString()]: `boom ${token}`\n    };\n\n    // THEN\n    test.deepEqual(resolve(s), { 'I am a string': 'boom I am a string' });\n    test.done();\n  },\n\n  'fails if token in a hash key resolves to a non-string'(test: Test) {\n    // GIVEN\n    const token = new CloudFormationToken({ Ref: 'Other' });\n\n    // WHEN\n    const s = {\n      [token.toString()]: `boom ${token}`\n    };\n\n    // THEN\n    test.throws(() => resolve(s), 'The key \"${Token[TOKEN.19]}\" has been resolved to {\"Ref\":\"Other\"} but must be resolvable to a string');\n    test.done();\n  }\n};\n\nclass Promise2 extends Token {\n  public resolve() {\n    return {\n      Data: {\n        stringProp: 'hello',\n        numberProp: 1234,\n      },\n      Recurse: new Token(() => 42)\n    };\n  }\n}\n\nclass Promise1 extends Token {\n  public p2 = [ new Promise2(), new Promise2() ];\n\n  public resolve() {\n    return this.p2;\n  }\n}\n\nclass BaseDataType {\n  constructor(readonly foo: number) {\n  }\n}\n\nclass DataType extends BaseDataType {\n  public goo = 'hello';\n\n  constructor() {\n    super(12);\n  }\n}\n\n/**\n * Return various flavors of Tokens that resolve to the given value\n */\nfunction literalTokensThatResolveTo(value: any): Token[] {\n  return [\n    new Token(value),\n    new Token(() => value)\n  ];\n}\n\n/**\n * Return various flavors of Tokens that resolve to the given value\n */\nfunction cloudFormationTokensThatResolveTo(value: any): Token[] {\n  return [\n    new CloudFormationToken(value),\n    new CloudFormationToken(() => value)\n  ];\n}\n\n/**\n * Return Tokens in both flavors that resolve to the given string\n */\nfunction tokensThatResolveTo(value: string): Token[] {\n  return literalTokensThatResolveTo(value).concat(cloudFormationTokensThatResolveTo(value));\n}\n"]}

@@ -74,3 +74,5 @@ "use strict";

delete response.runtime;
test.deepEqual(response, { stacks: [{ name: 'stack1',
test.deepEqual(response, {
version: '0.14.0',
stacks: [{ name: 'stack1',
environment: { name: '12345/us-east-1',

@@ -87,3 +89,4 @@ account: '12345',

s1c2r1D1791C01: { Type: 'ResourceType1' },
s1c2r25F685FFF: { Type: 'ResourceType2' } } } }] });
s1c2r25F685FFF: { Type: 'ResourceType2' } } } }]
});
test.done();

@@ -194,10 +197,15 @@ },

this.reportMissingContext('missing-context-key', {
provider: 'ctx-provider',
args: ['arg1', 'arg2'],
scope: ['scope1', 'scope2']
provider: 'fake',
props: {
account: '12345689012',
region: 'ab-north-1',
},
});
this.reportMissingContext('missing-context-key-2', {
provider: 'ctx-provider',
args: ['arg1', 'arg2'],
scope: ['scope1', 'scope2']
provider: 'fake2',
props: {
foo: 'bar',
account: '12345689012',
region: 'ab-south-1',
},
});

@@ -211,23 +219,16 @@ }

"missing-context-key": {
provider: "ctx-provider",
args: [
"arg1",
"arg2"
],
scope: [
"scope1",
"scope2"
]
provider: 'fake',
props: {
account: '12345689012',
region: 'ab-north-1',
},
},
"missing-context-key-2": {
provider: "ctx-provider",
args: [
"arg1",
"arg2"
],
scope: [
"scope1",
"scope2"
]
}
provider: 'fake2',
props: {
account: '12345689012',
region: 'ab-south-1',
foo: 'bar',
},
},
});

@@ -237,2 +238,2 @@ test.done();

};
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"test.app.js","sourceRoot":"","sources":["test.app.ts"],"names":[],"mappings":"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","sourcesContent":["import cxapi = require('@aws-cdk/cx-api');\nimport fs = require('fs');\nimport { Test } from 'nodeunit';\nimport os = require('os');\nimport path = require('path');\nimport { Construct, Resource, Stack, StackProps } from '../lib';\nimport { App } from '../lib/app';\n\nfunction withApp(context: { [key: string]: any } | undefined, block: (app: App) => void) {\n  const outdir = fs.mkdtempSync(path.join(os.tmpdir(), 'cdk-app-test'));\n  process.env[cxapi.OUTDIR_ENV] = outdir;\n\n  if (context) {\n    process.env[cxapi.CONTEXT_ENV] = JSON.stringify(context);\n  } else {\n    delete process.env[cxapi.CONTEXT_ENV];\n  }\n\n  const app = new App();\n  block(app);\n\n  app.run();\n\n  const outfile = path.join(outdir, cxapi.OUTFILE_NAME);\n  const response = JSON.parse(fs.readFileSync(outfile).toString());\n  fs.unlinkSync(outfile);\n  fs.rmdirSync(outdir);\n  return response;\n}\n\nfunction synth(context?: { [key: string]: any }): cxapi.SynthesizeResponse {\n  return withApp(context, app => {\n    const stack1 = new Stack(app, 'stack1', { env: { account: '12345', region: 'us-east-1' } });\n    new Resource(stack1, 's1c1', { type: 'DummyResource', properties: { Prop1: 'Prop1' } });\n    const r2 = new Resource(stack1, 's1c2', { type: 'DummyResource', properties: { Foo: 123 } });\n\n    const stack2 = new Stack(app, 'stack2');\n    new Resource(stack2, 's2c1', { type: 'DummyResource', properties: { Prog2: 'Prog2' } });\n    const c1 = new MyConstruct(stack2, 's1c2');\n\n    // add some metadata\n    stack1.addMetadata('meta', 111);\n    r2.addWarning('warning1');\n    r2.addWarning('warning2');\n    c1.addMetadata('meta', { key: 'value' });\n    app.addMetadata('applevel', 123); // apps can also have metadata\n  });\n}\n\nfunction synthStack(name: string, includeMetadata: boolean = false, context?: any): cxapi.SynthesizedStack {\n  const response = synth(context);\n  const stack = response.stacks.find(s => s.name === name);\n  if (!stack) {\n    throw new Error(`Stack ${name} not found`);\n  }\n\n  if (!includeMetadata) {\n    delete stack.metadata;\n  }\n\n  return stack;\n}\n\nexport = {\n  'synthesizes all stacks and returns synthesis result'(test: Test) {\n    const response = synth();\n\n    // clean up metadata so assertion will be sane\n    response.stacks.forEach(s => delete s.metadata);\n    delete response.runtime;\n\n    test.deepEqual(response, { stacks:\n      [ { name: 'stack1',\n          environment:\n           { name: '12345/us-east-1',\n             account: '12345',\n             region: 'us-east-1' },\n          template:\n           { Resources:\n              { s1c1: { Type: 'DummyResource', Properties: { Prop1: 'Prop1' } },\n                s1c2: { Type: 'DummyResource', Properties: { Foo: 123 } } } } },\n        { name: 'stack2',\n          environment:\n           { name: 'unknown-account/unknown-region',\n             account: 'unknown-account',\n             region: 'unknown-region' },\n          template:\n           { Resources:\n              { s2c1: { Type: 'DummyResource', Properties: { Prog2: 'Prog2' } },\n                s1c2r1D1791C01: { Type: 'ResourceType1' },\n                s1c2r25F685FFF: { Type: 'ResourceType2' } } } } ] });\n    test.done();\n  },\n\n  'synth(name) can be used programmatically'(test: Test) {\n    const prog = new App();\n    const stack = new Stack(prog, 'MyStack');\n    new Resource(stack, 'MyResource', { type: 'MyResourceType' });\n\n    let throws;\n    try {\n      prog.synthesizeStacks(['foo']);\n    } catch (e) {\n      throws = e.message;\n    }\n    test.ok(throws.indexOf('Cannot find stack foo') !== -1);\n\n    test.deepEqual(prog.synthesizeStack('MyStack').template,\n      { Resources: { MyResource: { Type: 'MyResourceType' } } });\n    test.done();\n  },\n\n  'synth(name) also collects metadata from all constructs in the stack'(test: Test) {\n    const stack = synthStack('stack1', true);\n\n    const output = stack.metadata;\n    stripStackTraces(output);\n\n    test.ok(output['/'], 'app-level metadata is included under \".\"');\n    test.equal(output['/'][0].type, 'applevel');\n    test.equal(output['/'][0].data, 123);\n\n    test.ok(output['/stack1'], 'the construct \"stack1\" has metadata');\n    test.equal(output['/stack1'][0].type, 'meta');\n    test.equal(output['/stack1'][0].data, 111);\n    test.ok(output['/stack1'][0].trace.length > 0, 'trace contains multiple entries');\n\n    test.ok(output['/stack1/s1c2']);\n    test.equal(output['/stack1/s1c2'].length, 2, 'two entries');\n    test.equal(output['/stack1/s1c2'][0].data, 'warning1');\n\n    const stack2 = synthStack('stack2', true);\n    const output2 = stack2.metadata;\n\n    test.ok(output2['/stack2/s1c2']);\n    test.equal(output2['/stack2/s1c2'][0].type, 'meta');\n    test.deepEqual(output2['/stack2/s1c2'][0].data, { key: 'value' });\n\n    test.done();\n  },\n\n  'context can be passed through CDK_CONTEXT'(test: Test) {\n    process.env[cxapi.CONTEXT_ENV] = JSON.stringify({\n      key1: 'val1',\n      key2: 'val2'\n    });\n    const prog = new App();\n    test.deepEqual(prog.getContext('key1'), 'val1');\n    test.deepEqual(prog.getContext('key2'), 'val2');\n    test.done();\n  },\n\n  'context from the command line can be used when creating the stack'(test: Test) {\n    const output = synthStack('stack2', false, { ctx1: 'HELLO' });\n\n    test.deepEqual(output.template, {\n      Resources: {\n        s2c1: {\n        Type: \"DummyResource\",\n        Properties: {\n          Prog2: \"Prog2\"\n        }\n        },\n        s1c2r1D1791C01: {\n        Type: \"ResourceType1\"\n        },\n        s1c2r25F685FFF: {\n        Type: \"ResourceType2\",\n        Properties: {\n          FromContext: \"HELLO\"\n        }\n        }\n      }\n    });\n    test.done();\n  },\n\n  'setContext(k,v) can be used to set context programmatically'(test: Test) {\n    const prog = new App();\n    prog.setContext('foo', 'bar');\n    test.deepEqual(prog.getContext('foo'), 'bar');\n    test.done();\n  },\n\n  'setContext(k,v) cannot be called after stacks have been added because stacks may use the context'(test: Test) {\n    const prog = new App();\n    new Stack(prog, 's1');\n    test.throws(() => prog.setContext('foo', 'bar'));\n    test.done();\n  },\n\n  'app.synthesizeStack(stack) performs validation first (app.validateAll()) and if there are errors, it returns the errors'(test: Test) {\n\n    class Child extends Construct {\n      public validate() {\n        return [ `Error from ${this.id}` ];\n      }\n    }\n\n    class Parent extends Stack {\n\n    }\n\n    const app = new App();\n\n    const parent = new Parent(app, 'Parent');\n    new Child(parent, 'C1');\n    new Child(parent, 'C2');\n\n    test.throws(() => {\n      app.synthesizeStacks(['Parent']);\n    }, /Stack validation failed with the following errors/);\n\n    test.done();\n  },\n\n  'app.synthesizeStack(stack) will return a list of missing contextual information'(test: Test) {\n    class MyStack extends Stack {\n      constructor(parent: App, name: string, props?: StackProps) {\n        super(parent, name, props);\n\n        this.reportMissingContext('missing-context-key', {\n          provider: 'ctx-provider',\n          args: [ 'arg1', 'arg2' ],\n          scope: [ 'scope1', 'scope2' ]\n        });\n\n        this.reportMissingContext('missing-context-key-2', {\n          provider: 'ctx-provider',\n          args: [ 'arg1', 'arg2' ],\n          scope: [ 'scope1', 'scope2' ]\n        });\n      }\n    }\n\n    const response = withApp(undefined, app => {\n      new MyStack(app, 'MyStack');\n    });\n\n    test.deepEqual(response.stacks[0].missing, {\n      \"missing-context-key\": {\n      provider: \"ctx-provider\",\n      args: [\n        \"arg1\",\n        \"arg2\"\n      ],\n      scope: [\n        \"scope1\",\n        \"scope2\"\n      ]\n      },\n      \"missing-context-key-2\": {\n      provider: \"ctx-provider\",\n      args: [\n        \"arg1\",\n        \"arg2\"\n      ],\n      scope: [\n        \"scope1\",\n        \"scope2\"\n      ]\n      }\n    });\n\n    test.done();\n  },\n};\n\nclass MyConstruct extends Construct {\n  constructor(parent: Construct, name: string) {\n    super(parent, name);\n\n    new Resource(this, 'r1', { type: 'ResourceType1' });\n    new Resource(this, 'r2', { type: 'ResourceType2', properties: { FromContext: this.getContext('ctx1') } });\n  }\n}\n\n/**\n * Strip stack traces from metadata\n */\nfunction stripStackTraces(meta: cxapi.StackMetadata) {\n  for (const key of Object.keys(meta)) {\n    meta[key] = meta[key].filter(entry => entry.type !== 'aws:cdk:logicalId');\n  }\n}\n"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"test.app.js","sourceRoot":"","sources":["test.app.ts"],"names":[],"mappings":"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","sourcesContent":["import cxapi = require('@aws-cdk/cx-api');\nimport fs = require('fs');\nimport { Test } from 'nodeunit';\nimport os = require('os');\nimport path = require('path');\nimport { Construct, Resource, Stack, StackProps } from '../lib';\nimport { App } from '../lib/app';\n\nfunction withApp(context: { [key: string]: any } | undefined, block: (app: App) => void) {\n  const outdir = fs.mkdtempSync(path.join(os.tmpdir(), 'cdk-app-test'));\n  process.env[cxapi.OUTDIR_ENV] = outdir;\n\n  if (context) {\n    process.env[cxapi.CONTEXT_ENV] = JSON.stringify(context);\n  } else {\n    delete process.env[cxapi.CONTEXT_ENV];\n  }\n\n  const app = new App();\n  block(app);\n\n  app.run();\n\n  const outfile = path.join(outdir, cxapi.OUTFILE_NAME);\n  const response = JSON.parse(fs.readFileSync(outfile).toString());\n  fs.unlinkSync(outfile);\n  fs.rmdirSync(outdir);\n  return response;\n}\n\nfunction synth(context?: { [key: string]: any }): cxapi.SynthesizeResponse {\n  return withApp(context, app => {\n    const stack1 = new Stack(app, 'stack1', { env: { account: '12345', region: 'us-east-1' } });\n    new Resource(stack1, 's1c1', { type: 'DummyResource', properties: { Prop1: 'Prop1' } });\n    const r2 = new Resource(stack1, 's1c2', { type: 'DummyResource', properties: { Foo: 123 } });\n\n    const stack2 = new Stack(app, 'stack2');\n    new Resource(stack2, 's2c1', { type: 'DummyResource', properties: { Prog2: 'Prog2' } });\n    const c1 = new MyConstruct(stack2, 's1c2');\n\n    // add some metadata\n    stack1.addMetadata('meta', 111);\n    r2.addWarning('warning1');\n    r2.addWarning('warning2');\n    c1.addMetadata('meta', { key: 'value' });\n    app.addMetadata('applevel', 123); // apps can also have metadata\n  });\n}\n\nfunction synthStack(name: string, includeMetadata: boolean = false, context?: any): cxapi.SynthesizedStack {\n  const response = synth(context);\n  const stack = response.stacks.find(s => s.name === name);\n  if (!stack) {\n    throw new Error(`Stack ${name} not found`);\n  }\n\n  if (!includeMetadata) {\n    delete stack.metadata;\n  }\n\n  return stack;\n}\n\nexport = {\n  'synthesizes all stacks and returns synthesis result'(test: Test) {\n    const response = synth();\n\n    // clean up metadata so assertion will be sane\n    response.stacks.forEach(s => delete s.metadata);\n    delete response.runtime;\n\n    test.deepEqual(response, {\n      version: '0.14.0',\n      stacks:\n      [ { name: 'stack1',\n          environment:\n           { name: '12345/us-east-1',\n             account: '12345',\n             region: 'us-east-1' },\n          template:\n           { Resources:\n              { s1c1: { Type: 'DummyResource', Properties: { Prop1: 'Prop1' } },\n                s1c2: { Type: 'DummyResource', Properties: { Foo: 123 } } } } },\n        { name: 'stack2',\n          environment:\n           { name: 'unknown-account/unknown-region',\n             account: 'unknown-account',\n             region: 'unknown-region' },\n          template:\n           { Resources:\n              { s2c1: { Type: 'DummyResource', Properties: { Prog2: 'Prog2' } },\n                s1c2r1D1791C01: { Type: 'ResourceType1' },\n                s1c2r25F685FFF: { Type: 'ResourceType2' } } } } ] });\n    test.done();\n  },\n\n  'synth(name) can be used programmatically'(test: Test) {\n    const prog = new App();\n    const stack = new Stack(prog, 'MyStack');\n    new Resource(stack, 'MyResource', { type: 'MyResourceType' });\n\n    let throws;\n    try {\n      prog.synthesizeStacks(['foo']);\n    } catch (e) {\n      throws = e.message;\n    }\n    test.ok(throws.indexOf('Cannot find stack foo') !== -1);\n\n    test.deepEqual(prog.synthesizeStack('MyStack').template,\n      { Resources: { MyResource: { Type: 'MyResourceType' } } });\n    test.done();\n  },\n\n  'synth(name) also collects metadata from all constructs in the stack'(test: Test) {\n    const stack = synthStack('stack1', true);\n\n    const output = stack.metadata;\n    stripStackTraces(output);\n\n    test.ok(output['/'], 'app-level metadata is included under \".\"');\n    test.equal(output['/'][0].type, 'applevel');\n    test.equal(output['/'][0].data, 123);\n\n    test.ok(output['/stack1'], 'the construct \"stack1\" has metadata');\n    test.equal(output['/stack1'][0].type, 'meta');\n    test.equal(output['/stack1'][0].data, 111);\n    test.ok(output['/stack1'][0].trace.length > 0, 'trace contains multiple entries');\n\n    test.ok(output['/stack1/s1c2']);\n    test.equal(output['/stack1/s1c2'].length, 2, 'two entries');\n    test.equal(output['/stack1/s1c2'][0].data, 'warning1');\n\n    const stack2 = synthStack('stack2', true);\n    const output2 = stack2.metadata;\n\n    test.ok(output2['/stack2/s1c2']);\n    test.equal(output2['/stack2/s1c2'][0].type, 'meta');\n    test.deepEqual(output2['/stack2/s1c2'][0].data, { key: 'value' });\n\n    test.done();\n  },\n\n  'context can be passed through CDK_CONTEXT'(test: Test) {\n    process.env[cxapi.CONTEXT_ENV] = JSON.stringify({\n      key1: 'val1',\n      key2: 'val2'\n    });\n    const prog = new App();\n    test.deepEqual(prog.getContext('key1'), 'val1');\n    test.deepEqual(prog.getContext('key2'), 'val2');\n    test.done();\n  },\n\n  'context from the command line can be used when creating the stack'(test: Test) {\n    const output = synthStack('stack2', false, { ctx1: 'HELLO' });\n\n    test.deepEqual(output.template, {\n      Resources: {\n        s2c1: {\n        Type: \"DummyResource\",\n        Properties: {\n          Prog2: \"Prog2\"\n        }\n        },\n        s1c2r1D1791C01: {\n        Type: \"ResourceType1\"\n        },\n        s1c2r25F685FFF: {\n        Type: \"ResourceType2\",\n        Properties: {\n          FromContext: \"HELLO\"\n        }\n        }\n      }\n    });\n    test.done();\n  },\n\n  'setContext(k,v) can be used to set context programmatically'(test: Test) {\n    const prog = new App();\n    prog.setContext('foo', 'bar');\n    test.deepEqual(prog.getContext('foo'), 'bar');\n    test.done();\n  },\n\n  'setContext(k,v) cannot be called after stacks have been added because stacks may use the context'(test: Test) {\n    const prog = new App();\n    new Stack(prog, 's1');\n    test.throws(() => prog.setContext('foo', 'bar'));\n    test.done();\n  },\n\n  'app.synthesizeStack(stack) performs validation first (app.validateAll()) and if there are errors, it returns the errors'(test: Test) {\n\n    class Child extends Construct {\n      public validate() {\n        return [ `Error from ${this.id}` ];\n      }\n    }\n\n    class Parent extends Stack {\n\n    }\n\n    const app = new App();\n\n    const parent = new Parent(app, 'Parent');\n    new Child(parent, 'C1');\n    new Child(parent, 'C2');\n\n    test.throws(() => {\n      app.synthesizeStacks(['Parent']);\n    }, /Stack validation failed with the following errors/);\n\n    test.done();\n  },\n\n  'app.synthesizeStack(stack) will return a list of missing contextual information'(test: Test) {\n    class MyStack extends Stack {\n      constructor(parent: App, name: string, props?: StackProps) {\n        super(parent, name, props);\n\n        this.reportMissingContext('missing-context-key', {\n          provider: 'fake',\n          props: {\n            account: '12345689012',\n            region: 'ab-north-1',\n          },\n        },\n        );\n\n        this.reportMissingContext('missing-context-key-2', {\n          provider: 'fake2',\n          props: {\n            foo: 'bar',\n            account: '12345689012',\n            region: 'ab-south-1',\n          },\n        },\n        );\n      }\n    }\n\n    const response = withApp(undefined, app => {\n      new MyStack(app, 'MyStack');\n    });\n\n    test.deepEqual(response.stacks[0].missing, {\n      \"missing-context-key\": {\n        provider: 'fake',\n        props: {\n          account: '12345689012',\n          region: 'ab-north-1',\n        },\n      },\n      \"missing-context-key-2\": {\n        provider: 'fake2',\n        props: {\n          account: '12345689012',\n          region: 'ab-south-1',\n          foo: 'bar',\n        },\n      },\n    });\n\n    test.done();\n  },\n};\n\nclass MyConstruct extends Construct {\n  constructor(parent: Construct, name: string) {\n    super(parent, name);\n\n    new Resource(this, 'r1', { type: 'ResourceType1' });\n    new Resource(this, 'r2', { type: 'ResourceType2', properties: { FromContext: this.getContext('ctx1') } });\n  }\n}\n\n/**\n * Strip stack traces from metadata\n */\nfunction stripStackTraces(meta: cxapi.StackMetadata) {\n  for (const key of Object.keys(meta)) {\n    meta[key] = meta[key].filter(entry => entry.type !== 'aws:cdk:logicalId');\n  }\n}\n"]}

@@ -6,6 +6,6 @@ import { Test } from 'nodeunit';

'AvailabilityZoneProvider will complain if not given a list'(test: Test): void;
'ContextProvider consistently generates a key'(test: Test): void;
'SSM parameter provider will return context values if available'(test: Test): void;
'SSM parameter provider has configurable default'(test: Test): void;
'Return default values if "env" is undefined to facilitate unit tests, but also expect metadata to include "error" messages'(test: Test): void;
};
export = _default;

@@ -39,20 +39,29 @@ "use strict";

},
'ContextProvider consistently generates a key'(test) {
const stack = new lib_1.Stack(undefined, 'TestStack', { env: { account: '12345', region: 'us-east-1' } });
const provider = new lib_1.ContextProvider(stack, 'ssm', {
parameterName: 'foo',
anyStringParam: 'bar',
});
const key = provider.key;
test.deepEqual(key, 'ssm:account=12345:anyStringParam=bar:parameterName=foo:region=us-east-1');
const complex = new lib_1.ContextProvider(stack, 'vpc', {
cidrBlock: '192.168.0.16',
tags: { Name: 'MyVPC', Env: 'Preprod' },
igw: false,
});
const complexKey = complex.key;
test.deepEqual(complexKey, 'vpc:account=12345:cidrBlock=192.168.0.16:igw=false:region=us-east-1:tags.Env=Preprod:tags.Name=MyVPC');
test.done();
},
'SSM parameter provider will return context values if available'(test) {
const stack = new lib_1.Stack(undefined, 'TestStack', { env: { account: '12345', region: 'us-east-1' } });
new lib_1.SSMParameterProvider(stack).getString('test');
new lib_1.SSMParameterProvider(stack, { parameterName: 'test' }).parameterValue();
const key = expectedContextKey(stack);
stack.setContext(key, 'abc');
const azs = lib_1.resolve(new lib_1.SSMParameterProvider(stack).getString('test'));
const ssmp = new lib_1.SSMParameterProvider(stack, { parameterName: 'test' });
const azs = lib_1.resolve(ssmp.parameterValue());
test.deepEqual(azs, 'abc');
test.done();
},
'SSM parameter provider has configurable default'(test) {
// GIVEN
const stack = new lib_1.Stack(undefined, 'TestStack', { env: { account: '12345', region: 'us-east-1' } });
// WHEN
const customizedDefault = new lib_1.SSMParameterProvider(stack).getString('test', 'some-value');
// THEN
test.equals(customizedDefault, 'some-value');
test.done();
},
'Return default values if "env" is undefined to facilitate unit tests, but also expect metadata to include "error" messages'(test) {

@@ -63,11 +72,11 @@ const app = new lib_1.App();

test.deepEqual(new lib_1.AvailabilityZoneProvider(stack).availabilityZones, ['dummy1a', 'dummy1b', 'dummy1c']);
test.deepEqual(new lib_1.SSMParameterProvider(child).getString('foo'), 'dummy');
test.deepEqual(new lib_1.SSMParameterProvider(child, { parameterName: 'foo' }).parameterValue(), 'dummy');
const output = app.synthesizeStack(stack.id);
const azError = output.metadata['/test-stack'].find(x => x.type === cxapi.ERROR_METADATA_KEY);
const ssmError = output.metadata['/test-stack/ChildConstruct'].find(x => x.type === cxapi.ERROR_METADATA_KEY);
test.ok(azError && azError.data.includes('Cannot determine scope for context provider availability-zones.'));
test.ok(ssmError && ssmError.data.includes('Cannot determine scope for context provider ssm["foo"].'));
test.ok(azError && azError.data.includes('Cannot determine scope for context provider availability-zones'));
test.ok(ssmError && ssmError.data.includes('Cannot determine scope for context provider ssm'));
test.done();
},
};
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"test.context.js","sourceRoot":"","sources":["test.context.ts"],"names":[],"mappings":"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","sourcesContent":["import cxapi = require('@aws-cdk/cx-api');\nimport { Test } from 'nodeunit';\nimport { App, AvailabilityZoneProvider, Construct, MetadataEntry, resolve, SSMParameterProvider, Stack } from '../lib';\n\nexport = {\n  'AvailabilityZoneProvider returns a list with dummy values if the context is not available'(test: Test) {\n    const stack = new Stack(undefined, 'TestStack', { env: { account: '12345', region: 'us-east-1' } });\n    const azs = new AvailabilityZoneProvider(stack).availabilityZones;\n\n    test.deepEqual(azs, ['dummy1a', 'dummy1b', 'dummy1c']);\n    test.done();\n  },\n\n  'AvailabilityZoneProvider will return context list if available'(test: Test) {\n    const stack = new Stack(undefined, 'TestStack', { env: { account: '12345', region: 'us-east-1' } });\n    const before = new AvailabilityZoneProvider(stack).availabilityZones;\n    test.deepEqual(before, [ 'dummy1a', 'dummy1b', 'dummy1c' ]);\n    const key = expectedContextKey(stack);\n\n    stack.setContext(key, ['us-east-1a', 'us-east-1b']);\n\n    const azs = new AvailabilityZoneProvider(stack).availabilityZones;\n    test.deepEqual(azs, ['us-east-1a', 'us-east-1b']);\n\n    test.done();\n  },\n\n  'AvailabilityZoneProvider will complain if not given a list'(test: Test) {\n    const stack = new Stack(undefined, 'TestStack', { env: { account: '12345', region: 'us-east-1' } });\n    const before = new AvailabilityZoneProvider(stack).availabilityZones;\n    test.deepEqual(before, [ 'dummy1a', 'dummy1b', 'dummy1c' ]);\n    const key = expectedContextKey(stack);\n\n    stack.setContext(key, 'not-a-list');\n\n    test.throws(\n      () => new AvailabilityZoneProvider(stack).availabilityZones\n    );\n\n    test.done();\n  },\n\n  'SSM parameter provider will return context values if available'(test: Test) {\n    const stack = new Stack(undefined, 'TestStack', { env: { account: '12345', region: 'us-east-1' } });\n    new SSMParameterProvider(stack).getString('test');\n    const key = expectedContextKey(stack);\n\n    stack.setContext(key, 'abc');\n\n    const azs = resolve(new SSMParameterProvider(stack).getString('test'));\n    test.deepEqual(azs, 'abc');\n\n    test.done();\n  },\n\n  'SSM parameter provider has configurable default'(test: Test) {\n    // GIVEN\n    const stack = new Stack(undefined, 'TestStack', { env: { account: '12345', region: 'us-east-1' } });\n\n    // WHEN\n    const customizedDefault = new SSMParameterProvider(stack).getString('test', 'some-value');\n\n    // THEN\n    test.equals(customizedDefault, 'some-value');\n\n    test.done();\n  },\n\n  'Return default values if \"env\" is undefined to facilitate unit tests, but also expect metadata to include \"error\" messages'(test: Test) {\n    const app = new App();\n    const stack = new Stack(app, 'test-stack');\n\n    const child = new Construct(stack, 'ChildConstruct');\n\n    test.deepEqual(new AvailabilityZoneProvider(stack).availabilityZones, [ 'dummy1a', 'dummy1b', 'dummy1c' ]);\n    test.deepEqual(new SSMParameterProvider(child).getString('foo'), 'dummy');\n\n    const output = app.synthesizeStack(stack.id);\n\n    const azError: MetadataEntry | undefined = output.metadata['/test-stack'].find(x => x.type === cxapi.ERROR_METADATA_KEY);\n    const ssmError: MetadataEntry | undefined = output.metadata['/test-stack/ChildConstruct'].find(x => x.type === cxapi.ERROR_METADATA_KEY);\n\n    test.ok(azError && (azError.data as string).includes('Cannot determine scope for context provider availability-zones.'));\n    test.ok(ssmError && (ssmError.data as string).includes('Cannot determine scope for context provider ssm[\"foo\"].'));\n\n    test.done();\n  },\n};\n\nfunction firstKey(obj: any): string {\n  return Object.keys(obj)[0];\n}\n\n/**\n * Get the expected context key from a stack with missing parameters\n */\nfunction expectedContextKey(stack: Stack): string {\n  return firstKey(stack.missingContext);\n}\n"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"test.context.js","sourceRoot":"","sources":["test.context.ts"],"names":[],"mappings":";AAAA,yCAA0C;AAE1C,gCACsE;AA6FtE,SAAS,QAAQ,CAAC,GAAQ;IACxB,OAAO,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;AAC7B,CAAC;AAED;;GAEG;AACH,SAAS,kBAAkB,CAAC,KAAY;IACtC,OAAO,QAAQ,CAAC,KAAK,CAAC,cAAc,CAAC,CAAC;AACxC,CAAC;AApGD,iBAAS;IACP,2FAA2F,CAAC,IAAU;QACpG,MAAM,KAAK,GAAG,IAAI,WAAK,CAAC,SAAS,EAAE,WAAW,EAAE,EAAE,GAAG,EAAE,EAAE,OAAO,EAAE,OAAO,EAAE,MAAM,EAAE,WAAW,EAAE,EAAE,CAAC,CAAC;QACpG,MAAM,GAAG,GAAG,IAAI,8BAAwB,CAAC,KAAK,CAAC,CAAC,iBAAiB,CAAC;QAElE,IAAI,CAAC,SAAS,CAAC,GAAG,EAAE,CAAC,SAAS,EAAE,SAAS,EAAE,SAAS,CAAC,CAAC,CAAC;QACvD,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED,gEAAgE,CAAC,IAAU;QACzE,MAAM,KAAK,GAAG,IAAI,WAAK,CAAC,SAAS,EAAE,WAAW,EAAE,EAAE,GAAG,EAAE,EAAE,OAAO,EAAE,OAAO,EAAE,MAAM,EAAE,WAAW,EAAE,EAAE,CAAC,CAAC;QACpG,MAAM,MAAM,GAAG,IAAI,8BAAwB,CAAC,KAAK,CAAC,CAAC,iBAAiB,CAAC;QACrE,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,CAAE,SAAS,EAAE,SAAS,EAAE,SAAS,CAAE,CAAC,CAAC;QAC5D,MAAM,GAAG,GAAG,kBAAkB,CAAC,KAAK,CAAC,CAAC;QAEtC,KAAK,CAAC,UAAU,CAAC,GAAG,EAAE,CAAC,YAAY,EAAE,YAAY,CAAC,CAAC,CAAC;QAEpD,MAAM,GAAG,GAAG,IAAI,8BAAwB,CAAC,KAAK,CAAC,CAAC,iBAAiB,CAAC;QAClE,IAAI,CAAC,SAAS,CAAC,GAAG,EAAE,CAAC,YAAY,EAAE,YAAY,CAAC,CAAC,CAAC;QAElD,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED,4DAA4D,CAAC,IAAU;QACrE,MAAM,KAAK,GAAG,IAAI,WAAK,CAAC,SAAS,EAAE,WAAW,EAAE,EAAE,GAAG,EAAE,EAAE,OAAO,EAAE,OAAO,EAAE,MAAM,EAAE,WAAW,EAAE,EAAE,CAAC,CAAC;QACpG,MAAM,MAAM,GAAG,IAAI,8BAAwB,CAAC,KAAK,CAAC,CAAC,iBAAiB,CAAC;QACrE,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,CAAE,SAAS,EAAE,SAAS,EAAE,SAAS,CAAE,CAAC,CAAC;QAC5D,MAAM,GAAG,GAAG,kBAAkB,CAAC,KAAK,CAAC,CAAC;QAEtC,KAAK,CAAC,UAAU,CAAC,GAAG,EAAE,YAAY,CAAC,CAAC;QAEpC,IAAI,CAAC,MAAM,CACT,GAAG,EAAE,CAAC,IAAI,8BAAwB,CAAC,KAAK,CAAC,CAAC,iBAAiB,CAC5D,CAAC;QAEF,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED,8CAA8C,CAAC,IAAU;QACvD,MAAM,KAAK,GAAG,IAAI,WAAK,CAAC,SAAS,EAAE,WAAW,EAAE,EAAE,GAAG,EAAE,EAAE,OAAO,EAAE,OAAO,EAAE,MAAM,EAAE,WAAW,EAAE,EAAE,CAAC,CAAC;QACpG,MAAM,QAAQ,GAAG,IAAI,qBAAe,CAAC,KAAK,EAAE,KAAK,EAAE;YACjD,aAAa,EAAE,KAAK;YACpB,cAAc,EAAE,KAAK;SACtB,CAAC,CAAC;QACH,MAAM,GAAG,GAAG,QAAQ,CAAC,GAAG,CAAC;QACzB,IAAI,CAAC,SAAS,CAAC,GAAG,EAAE,yEAAyE,CAAC,CAAC;QAC/F,MAAM,OAAO,GAAG,IAAI,qBAAe,CAAC,KAAK,EAAE,KAAK,EAAE;YAChD,SAAS,EAAE,cAAc;YACzB,IAAI,EAAE,EAAE,IAAI,EAAE,OAAO,EAAE,GAAG,EAAE,SAAS,EAAE;YACvC,GAAG,EAAE,KAAK;SACX,CAAC,CAAC;QACH,MAAM,UAAU,GAAG,OAAO,CAAC,GAAG,CAAC;QAC/B,IAAI,CAAC,SAAS,CAAC,UAAU,EACvB,sGAAsG,CAAC,CAAC;QAC1G,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IACD,gEAAgE,CAAC,IAAU;QACzE,MAAM,KAAK,GAAG,IAAI,WAAK,CAAC,SAAS,EAAE,WAAW,EAAE,EAAE,GAAG,EAAE,EAAE,OAAO,EAAE,OAAO,EAAE,MAAM,EAAE,WAAW,EAAE,EAAE,CAAC,CAAC;QACpG,IAAI,0BAAoB,CAAC,KAAK,EAAG,EAAC,aAAa,EAAE,MAAM,EAAC,CAAC,CAAC,cAAc,EAAE,CAAC;QAC3E,MAAM,GAAG,GAAG,kBAAkB,CAAC,KAAK,CAAC,CAAC;QAEtC,KAAK,CAAC,UAAU,CAAC,GAAG,EAAE,KAAK,CAAC,CAAC;QAE7B,MAAM,IAAI,GAAG,IAAI,0BAAoB,CAAC,KAAK,EAAG,EAAC,aAAa,EAAE,MAAM,EAAC,CAAC,CAAC;QACvE,MAAM,GAAG,GAAG,aAAO,CAAC,IAAI,CAAC,cAAc,EAAE,CAAC,CAAC;QAC3C,IAAI,CAAC,SAAS,CAAC,GAAG,EAAE,KAAK,CAAC,CAAC;QAE3B,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED,4HAA4H,CAAC,IAAU;QACrI,MAAM,GAAG,GAAG,IAAI,SAAG,EAAE,CAAC;QACtB,MAAM,KAAK,GAAG,IAAI,WAAK,CAAC,GAAG,EAAE,YAAY,CAAC,CAAC;QAE3C,MAAM,KAAK,GAAG,IAAI,eAAS,CAAC,KAAK,EAAE,gBAAgB,CAAC,CAAC;QAErD,IAAI,CAAC,SAAS,CAAC,IAAI,8BAAwB,CAAC,KAAK,CAAC,CAAC,iBAAiB,EAAE,CAAE,SAAS,EAAE,SAAS,EAAE,SAAS,CAAE,CAAC,CAAC;QAC3G,IAAI,CAAC,SAAS,CAAC,IAAI,0BAAoB,CAAC,KAAK,EAAE,EAAC,aAAa,EAAE,KAAK,EAAC,CAAC,CAAC,cAAc,EAAE,EAAE,OAAO,CAAC,CAAC;QAElG,MAAM,MAAM,GAAG,GAAG,CAAC,eAAe,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;QAE7C,MAAM,OAAO,GAA8B,MAAM,CAAC,QAAQ,CAAC,aAAa,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,KAAK,CAAC,kBAAkB,CAAC,CAAC;QACzH,MAAM,QAAQ,GAA8B,MAAM,CAAC,QAAQ,CAAC,4BAA4B,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,KAAK,CAAC,kBAAkB,CAAC,CAAC;QAEzI,IAAI,CAAC,EAAE,CAAC,OAAO,IAAK,OAAO,CAAC,IAAe,CAAC,QAAQ,CAAC,gEAAgE,CAAC,CAAC,CAAC;QACxH,IAAI,CAAC,EAAE,CAAC,QAAQ,IAAK,QAAQ,CAAC,IAAe,CAAC,QAAQ,CAAC,iDAAiD,CAAC,CAAC,CAAC;QAE3G,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;CACF,CAAC","sourcesContent":["import cxapi = require('@aws-cdk/cx-api');\nimport { Test } from 'nodeunit';\nimport { App, AvailabilityZoneProvider, Construct, ContextProvider,\n  MetadataEntry, resolve, SSMParameterProvider, Stack } from '../lib';\n\nexport = {\n  'AvailabilityZoneProvider returns a list with dummy values if the context is not available'(test: Test) {\n    const stack = new Stack(undefined, 'TestStack', { env: { account: '12345', region: 'us-east-1' } });\n    const azs = new AvailabilityZoneProvider(stack).availabilityZones;\n\n    test.deepEqual(azs, ['dummy1a', 'dummy1b', 'dummy1c']);\n    test.done();\n  },\n\n  'AvailabilityZoneProvider will return context list if available'(test: Test) {\n    const stack = new Stack(undefined, 'TestStack', { env: { account: '12345', region: 'us-east-1' } });\n    const before = new AvailabilityZoneProvider(stack).availabilityZones;\n    test.deepEqual(before, [ 'dummy1a', 'dummy1b', 'dummy1c' ]);\n    const key = expectedContextKey(stack);\n\n    stack.setContext(key, ['us-east-1a', 'us-east-1b']);\n\n    const azs = new AvailabilityZoneProvider(stack).availabilityZones;\n    test.deepEqual(azs, ['us-east-1a', 'us-east-1b']);\n\n    test.done();\n  },\n\n  'AvailabilityZoneProvider will complain if not given a list'(test: Test) {\n    const stack = new Stack(undefined, 'TestStack', { env: { account: '12345', region: 'us-east-1' } });\n    const before = new AvailabilityZoneProvider(stack).availabilityZones;\n    test.deepEqual(before, [ 'dummy1a', 'dummy1b', 'dummy1c' ]);\n    const key = expectedContextKey(stack);\n\n    stack.setContext(key, 'not-a-list');\n\n    test.throws(\n      () => new AvailabilityZoneProvider(stack).availabilityZones\n    );\n\n    test.done();\n  },\n\n  'ContextProvider consistently generates a key'(test: Test) {\n    const stack = new Stack(undefined, 'TestStack', { env: { account: '12345', region: 'us-east-1' } });\n    const provider = new ContextProvider(stack, 'ssm', {\n      parameterName: 'foo',\n      anyStringParam: 'bar',\n    });\n    const key = provider.key;\n    test.deepEqual(key, 'ssm:account=12345:anyStringParam=bar:parameterName=foo:region=us-east-1');\n    const complex = new ContextProvider(stack, 'vpc', {\n      cidrBlock: '192.168.0.16',\n      tags: { Name: 'MyVPC', Env: 'Preprod' },\n      igw: false,\n    });\n    const complexKey = complex.key;\n    test.deepEqual(complexKey,\n      'vpc:account=12345:cidrBlock=192.168.0.16:igw=false:region=us-east-1:tags.Env=Preprod:tags.Name=MyVPC');\n    test.done();\n  },\n  'SSM parameter provider will return context values if available'(test: Test) {\n    const stack = new Stack(undefined, 'TestStack', { env: { account: '12345', region: 'us-east-1' } });\n    new SSMParameterProvider(stack,  {parameterName: 'test'}).parameterValue();\n    const key = expectedContextKey(stack);\n\n    stack.setContext(key, 'abc');\n\n    const ssmp = new SSMParameterProvider(stack,  {parameterName: 'test'});\n    const azs = resolve(ssmp.parameterValue());\n    test.deepEqual(azs, 'abc');\n\n    test.done();\n  },\n\n  'Return default values if \"env\" is undefined to facilitate unit tests, but also expect metadata to include \"error\" messages'(test: Test) {\n    const app = new App();\n    const stack = new Stack(app, 'test-stack');\n\n    const child = new Construct(stack, 'ChildConstruct');\n\n    test.deepEqual(new AvailabilityZoneProvider(stack).availabilityZones, [ 'dummy1a', 'dummy1b', 'dummy1c' ]);\n    test.deepEqual(new SSMParameterProvider(child, {parameterName: 'foo'}).parameterValue(), 'dummy');\n\n    const output = app.synthesizeStack(stack.id);\n\n    const azError: MetadataEntry | undefined = output.metadata['/test-stack'].find(x => x.type === cxapi.ERROR_METADATA_KEY);\n    const ssmError: MetadataEntry | undefined = output.metadata['/test-stack/ChildConstruct'].find(x => x.type === cxapi.ERROR_METADATA_KEY);\n\n    test.ok(azError && (azError.data as string).includes('Cannot determine scope for context provider availability-zones'));\n    test.ok(ssmError && (ssmError.data as string).includes('Cannot determine scope for context provider ssm'));\n\n    test.done();\n  },\n};\n\nfunction firstKey(obj: any): string {\n  return Object.keys(obj)[0];\n}\n\n/**\n * Get the expected context key from a stack with missing parameters\n */\nfunction expectedContextKey(stack: Stack): string {\n  return firstKey(stack.missingContext);\n}\n"]}

Sorry, the diff of this file is not supported yet

Sorry, the diff of this file is too big to display

SocketSocket SOC 2 Logo

Product

  • Package Alerts
  • Integrations
  • Docs
  • Pricing
  • FAQ
  • Roadmap
  • Changelog

Packages

npm

Stay in touch

Get open source security insights delivered straight into your inbox.


  • Terms
  • Privacy
  • Security

Made with ⚡️ by Socket Inc