Socket
Socket
Sign inDemoInstall

@aws-cdk/cdk

Package Overview
Dependencies
Maintainers
5
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.32.0 to 0.33.0

lib/stack-trace.d.ts

46

lib/app.d.ts

@@ -1,4 +0,3 @@

import cxapi = require('@aws-cdk/cx-api');
import { CloudAssembly } from '@aws-cdk/cx-api';
import { Construct } from './construct';
import { ISynthesisSession } from './synthesis';
/**

@@ -17,2 +16,19 @@ * Custom construction properties for a CDK program

/**
* The output directory into which to emit synthesized artifacts.
*
* @default - If this value is _not_ set, considers the environment variable `CDK_OUTDIR`.
* If `CDK_OUTDIR` is not defined, uses a temp directory.
*/
readonly outdir?: string;
/**
* Include stack traces in construct metadata entries.
* @default true stack traces are included
*/
readonly stackTraces?: boolean;
/**
* Include runtime versioning information in cloud assembly manifest
* @default true runtime info is included
*/
readonly runtimeInfo?: boolean;
/**
* Additional context values for the application

@@ -30,5 +46,6 @@ *

export declare class App extends Construct {
private _session?;
private readonly legacyManifest;
private readonly runtimeInformation;
static isApp(obj: any): obj is App;
private _assembly?;
private readonly runtimeInfo;
private readonly outdir?;
/**

@@ -40,17 +57,10 @@ * Initializes a CDK application.

/**
* Runs the program. Output is written to output directory as specified in the request.
* Runs the program. Output is written to output directory as specified in the
* request.
*
* @returns a `CloudAssembly` which includes all the synthesized artifacts
* such as CloudFormation templates and assets.
*/
run(): ISynthesisSession;
/**
* Synthesize and validate a single stack.
* @param stackName The name of the stack to synthesize
* @deprecated This method is going to be deprecated in a future version of the CDK
*/
synthesizeStack(stackName: string): cxapi.SynthesizedStack;
/**
* Synthesizes multiple stacks
* @deprecated This method is going to be deprecated in a future version of the CDK
*/
synthesizeStacks(stackNames: string[]): cxapi.SynthesizedStack[];
run(): CloudAssembly;
private loadContext;
}

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

const construct_1 = require("./construct");
const runtime_info_1 = require("./runtime-info");
const synthesis_1 = require("./synthesis");
const APP_SYMBOL = Symbol.for('@aws-cdk/cdk.App');
/**

@@ -11,2 +13,5 @@ * Represents a CDK program.

class App extends construct_1.Construct {
static isApp(obj) {
return APP_SYMBOL in obj;
}
/**

@@ -18,6 +23,13 @@ * Initializes a CDK application.

super(undefined, '');
Object.defineProperty(this, APP_SYMBOL, { value: true });
this.loadContext(props.context);
if (props.stackTraces === false) {
this.node.setContext(cxapi.DISABLE_METADATA_STACK_TRACE, true);
}
if (props.runtimeInfo === false) {
this.node.setContext(cxapi.DISABLE_VERSION_REPORTING, true);
}
// both are reverse logic
this.legacyManifest = this.node.getContext(cxapi.DISABLE_LEGACY_MANIFEST_CONTEXT) ? false : true;
this.runtimeInformation = this.node.getContext(cxapi.DISABLE_VERSION_REPORTING) ? false : true;
this.runtimeInfo = this.node.getContext(cxapi.DISABLE_VERSION_REPORTING) ? false : true;
this.outdir = props.outdir || process.env[cxapi.OUTDIR_ENV];
const autoRun = props.autoRun !== undefined ? props.autoRun : cxapi.OUTDIR_ENV in process.env;

@@ -31,53 +43,21 @@ if (autoRun) {

/**
* Runs the program. Output is written to output directory as specified in the request.
* Runs the program. Output is written to output directory as specified in the
* request.
*
* @returns a `CloudAssembly` which includes all the synthesized artifacts
* such as CloudFormation templates and assets.
*/
run() {
// this app has already been executed, no-op for you
if (this._session) {
return this._session;
if (this._assembly) {
return this._assembly;
}
const outdir = process.env[cxapi.OUTDIR_ENV];
let store;
if (outdir) {
store = new synthesis_1.FileSystemStore({ outdir });
}
else {
store = new synthesis_1.InMemoryStore();
}
const synth = new synthesis_1.Synthesizer();
this._session = synth.synthesize(this, {
store,
legacyManifest: this.legacyManifest,
runtimeInformation: this.runtimeInformation
const assembly = synth.synthesize(this, {
outdir: this.outdir,
runtimeInfo: this.runtimeInfo ? runtime_info_1.collectRuntimeInformation() : undefined
});
return this._session;
this._assembly = assembly;
return assembly;
}
/**
* Synthesize and validate a single stack.
* @param stackName The name of the stack to synthesize
* @deprecated This method is going to be deprecated in a future version of the CDK
*/
synthesizeStack(stackName) {
if (!this.legacyManifest) {
throw new Error('No legacy manifest available, return an old-style stack output');
}
const session = this.run();
const legacy = session.store.readJson(cxapi.OUTFILE_NAME);
const res = legacy.stacks.find(s => s.name === stackName);
if (!res) {
throw new Error(`Stack "${stackName}" not found`);
}
return res;
}
/**
* Synthesizes multiple stacks
* @deprecated This method is going to be deprecated in a future version of the CDK
*/
synthesizeStacks(stackNames) {
const ret = [];
for (const stackName of stackNames) {
ret.push(this.synthesizeStack(stackName));
}
return ret;
}
loadContext(defaults = {}) {

@@ -99,2 +79,2 @@ // prime with defaults passed through constructor

exports.App = App;
//# 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 { Construct } from './construct';\nimport { FileSystemStore, InMemoryStore, ISynthesisSession, Synthesizer } from './synthesis';\n\n/**\n * Custom construction properties for a CDK program\n */\nexport interface AppProps {\n  /**\n   * Automatically call run before the application exits\n   *\n   * If you set this, you don't have to call `run()` anymore.\n   *\n   * @default true if running via CDK toolkit (CDK_OUTDIR is set), false otherwise\n   */\n  readonly autoRun?: boolean;\n\n  /**\n   * Additional context values for the application\n   *\n   * @default No additional context\n   */\n  readonly context?: { [key: string]: string };\n}\n\n/**\n * Represents a CDK program.\n */\nexport class App extends Construct {\n  private _session?: ISynthesisSession;\n  private readonly legacyManifest: boolean;\n  private readonly runtimeInformation: boolean;\n\n  /**\n   * Initializes a CDK application.\n   * @param request Optional toolkit request (e.g. for tests)\n   */\n  constructor(props: AppProps = {}) {\n    super(undefined as any, '');\n\n    this.loadContext(props.context);\n\n    // both are reverse logic\n    this.legacyManifest = this.node.getContext(cxapi.DISABLE_LEGACY_MANIFEST_CONTEXT) ? false : true;\n    this.runtimeInformation = this.node.getContext(cxapi.DISABLE_VERSION_REPORTING) ? false : true;\n\n    const autoRun = props.autoRun !== undefined ? props.autoRun : cxapi.OUTDIR_ENV in process.env;\n\n    if (autoRun) {\n      // run() guarantuees it will only execute once, so a default of 'true' doesn't bite manual calling\n      // of the function.\n      process.once('beforeExit', () => this.run());\n    }\n  }\n\n  /**\n   * Runs the program. Output is written to output directory as specified in the request.\n   */\n  public run(): ISynthesisSession {\n    // this app has already been executed, no-op for you\n    if (this._session) {\n      return this._session;\n    }\n\n    const outdir = process.env[cxapi.OUTDIR_ENV];\n    let store;\n    if (outdir) {\n      store = new FileSystemStore({ outdir });\n    } else {\n      store = new InMemoryStore();\n    }\n\n    const synth = new Synthesizer();\n\n    this._session = synth.synthesize(this, {\n      store,\n      legacyManifest: this.legacyManifest,\n      runtimeInformation: this.runtimeInformation\n    });\n\n    return this._session;\n  }\n\n  /**\n   * Synthesize and validate a single stack.\n   * @param stackName The name of the stack to synthesize\n   * @deprecated This method is going to be deprecated in a future version of the CDK\n   */\n  public synthesizeStack(stackName: string): cxapi.SynthesizedStack {\n    if (!this.legacyManifest) {\n      throw new Error('No legacy manifest available, return an old-style stack output');\n    }\n\n    const session = this.run();\n    const legacy: cxapi.SynthesizeResponse = session.store.readJson(cxapi.OUTFILE_NAME);\n\n    const res = legacy.stacks.find(s => s.name === stackName);\n    if (!res) {\n      throw new Error(`Stack \"${stackName}\" not found`);\n    }\n\n    return res;\n  }\n\n  /**\n   * Synthesizes multiple stacks\n   * @deprecated This method is going to be deprecated in a future version of the CDK\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  private loadContext(defaults: { [key: string]: string } = { }) {\n    // prime with defaults passed through constructor\n    for (const [ k, v ] of Object.entries(defaults)) {\n      this.node.setContext(k, v);\n    }\n\n    // read from environment\n    const contextJson = process.env[cxapi.CONTEXT_ENV];\n    const contextFromEnvironment = contextJson\n      ? JSON.parse(contextJson)\n      : { };\n\n    for (const [ k, v ] of Object.entries(contextFromEnvironment)) {\n      this.node.setContext(k, v);\n    }\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 { CloudAssembly } from '@aws-cdk/cx-api';\nimport { Construct } from './construct';\nimport { collectRuntimeInformation } from './runtime-info';\nimport { Synthesizer } from './synthesis';\n\nconst APP_SYMBOL = Symbol.for('@aws-cdk/cdk.App');\n\n/**\n * Custom construction properties for a CDK program\n */\nexport interface AppProps {\n  /**\n   * Automatically call run before the application exits\n   *\n   * If you set this, you don't have to call `run()` anymore.\n   *\n   * @default true if running via CDK toolkit (CDK_OUTDIR is set), false otherwise\n   */\n  readonly autoRun?: boolean;\n\n  /**\n   * The output directory into which to emit synthesized artifacts.\n   *\n   * @default - If this value is _not_ set, considers the environment variable `CDK_OUTDIR`.\n   *            If `CDK_OUTDIR` is not defined, uses a temp directory.\n   */\n  readonly outdir?: string;\n\n  /**\n   * Include stack traces in construct metadata entries.\n   * @default true stack traces are included\n   */\n  readonly stackTraces?: boolean;\n\n  /**\n   * Include runtime versioning information in cloud assembly manifest\n   * @default true runtime info is included\n   */\n  readonly runtimeInfo?: boolean;\n\n  /**\n   * Additional context values for the application\n   *\n   * @default No additional context\n   */\n  readonly context?: { [key: string]: string };\n}\n\n/**\n * Represents a CDK program.\n */\nexport class App extends Construct {\n\n  public static isApp(obj: any): obj is App {\n    return APP_SYMBOL in obj;\n  }\n\n  private _assembly?: CloudAssembly;\n  private readonly runtimeInfo: boolean;\n  private readonly outdir?: string;\n\n  /**\n   * Initializes a CDK application.\n   * @param request Optional toolkit request (e.g. for tests)\n   */\n  constructor(props: AppProps = {}) {\n    super(undefined as any, '');\n\n    Object.defineProperty(this, APP_SYMBOL, { value: true });\n\n    this.loadContext(props.context);\n\n    if (props.stackTraces === false) {\n      this.node.setContext(cxapi.DISABLE_METADATA_STACK_TRACE, true);\n    }\n\n    if (props.runtimeInfo === false) {\n      this.node.setContext(cxapi.DISABLE_VERSION_REPORTING, true);\n    }\n\n    // both are reverse logic\n    this.runtimeInfo = this.node.getContext(cxapi.DISABLE_VERSION_REPORTING) ? false : true;\n    this.outdir = props.outdir || process.env[cxapi.OUTDIR_ENV];\n\n    const autoRun = props.autoRun !== undefined ? props.autoRun : cxapi.OUTDIR_ENV in process.env;\n\n    if (autoRun) {\n      // run() guarantuees it will only execute once, so a default of 'true' doesn't bite manual calling\n      // of the function.\n      process.once('beforeExit', () => this.run());\n    }\n  }\n\n  /**\n   * Runs the program. Output is written to output directory as specified in the\n   * request.\n   *\n   * @returns a `CloudAssembly` which includes all the synthesized artifacts\n   * such as CloudFormation templates and assets.\n   */\n  public run(): CloudAssembly {\n    // this app has already been executed, no-op for you\n    if (this._assembly) {\n      return this._assembly;\n    }\n\n    const synth = new Synthesizer();\n\n    const assembly = synth.synthesize(this, {\n      outdir: this.outdir,\n      runtimeInfo: this.runtimeInfo ? collectRuntimeInformation() : undefined\n    });\n\n    this._assembly = assembly;\n    return assembly;\n  }\n\n  private loadContext(defaults: { [key: string]: string } = { }) {\n    // prime with defaults passed through constructor\n    for (const [ k, v ] of Object.entries(defaults)) {\n      this.node.setContext(k, v);\n    }\n\n    // read from environment\n    const contextJson = process.env[cxapi.CONTEXT_ENV];\n    const contextFromEnvironment = contextJson\n      ? JSON.parse(contextJson)\n      : { };\n\n    for (const [ k, v ] of Object.entries(contextFromEnvironment)) {\n      this.node.setContext(k, v);\n    }\n  }\n}\n"]}

@@ -46,3 +46,3 @@ import { Construct, IConstruct } from "./construct";

*/
readonly creationStackTrace: string[];
readonly creationStackTrace: string[] | undefined;
/**

@@ -49,0 +49,0 @@ * Return the path with respect to the stack

@@ -48,3 +48,7 @@ "use strict";

get creationStackTrace() {
return filterStackTrace(this.node.metadata.find(md => md.type === LOGICAL_ID_MD).trace);
const trace = this.node.metadata.find(md => md.type === LOGICAL_ID_MD).trace;
if (!trace) {
return undefined;
}
return filterStackTrace(trace);
function filterStackTrace(stack) {

@@ -125,2 +129,2 @@ const result = Array.of(...stack);

const resolve_1 = require("./resolve");
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cfn-element.js","sourceRoot":"","sources":["cfn-element.ts"],"names":[],"mappings":"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","sourcesContent":["import { Construct, IConstruct, PATH_SEP } from \"./construct\";\nimport { Token } from './token';\n\nconst LOGICAL_ID_MD = 'aws:cdk:logicalId';\n\n/**\n * An element of a CloudFormation stack.\n */\nexport abstract class CfnElement extends Construct {\n  /**\n   * Returns `true` if a construct is a stack element (i.e. part of the\n   * synthesized cloudformation template).\n   *\n   * Uses duck-typing instead of `instanceof` to allow stack elements from different\n   * versions of this library to be included in the same stack.\n   *\n   * @returns The construct as a stack element or undefined if it is not a stack element.\n   */\n  public static isCfnElement(construct: IConstruct): construct is CfnElement {\n    return ('logicalId' in construct && '_toCloudFormation' in construct);\n  }\n\n  /**\n   * The logical ID for this CloudFormation stack element. The logical ID of the element\n   * is calculated from the path of the resource node in the construct tree.\n   *\n   * To override this value, use `overrideLogicalId(newLogicalId)`.\n   *\n   * @returns the logical ID as a stringified token. This value will only get\n   * resolved during synthesis.\n   */\n  public readonly logicalId: string;\n\n  private _logicalId: string;\n\n  /**\n   * Creates an entity and binds it to a tree.\n   * Note that the root of the tree must be a Stack object (not just any Root).\n   *\n   * @param scope The parent construct\n   * @param props Construct properties\n   */\n  constructor(scope: Construct, id: string) {\n    super(scope, id);\n\n    this.node.addMetadata(LOGICAL_ID_MD, new (require(\"./token\").Token)(() => this.logicalId), this.constructor);\n\n    this._logicalId = this.node.stack.logicalIds.getLogicalId(this);\n    this.logicalId = new Token(() => this._logicalId, `${notTooLong(this.node.path)}.LogicalID`).toString();\n  }\n\n  /**\n   * Overrides the auto-generated logical ID with a specific ID.\n   * @param newLogicalId The new logical ID to use for this stack element.\n   */\n  public overrideLogicalId(newLogicalId: string) {\n    this._logicalId = newLogicalId;\n  }\n\n  /**\n   * @returns the stack trace of the point where this Resource was created from, sourced\n   *      from the +metadata+ entry typed +aws:cdk:logicalId+, and with the bottom-most\n   *      node +internal+ entries filtered.\n   */\n  public get creationStackTrace(): string[] {\n    return filterStackTrace(this.node.metadata.find(md => md.type === LOGICAL_ID_MD)!.trace);\n\n    function filterStackTrace(stack: string[]): string[] {\n      const result = Array.of(...stack);\n      while (result.length > 0 && shouldFilter(result[result.length - 1])) {\n        result.pop();\n      }\n      // It's weird if we filtered everything, so return the whole stack...\n      return result.length === 0 ? stack : result;\n    }\n\n    function shouldFilter(str: string): boolean {\n      return str.match(/[^(]+\\(internal\\/.*/) !== null;\n    }\n  }\n\n  /**\n   * Return the path with respect to the stack\n   */\n  public get stackPath(): string {\n    return this.node.ancestors(this.node.stack).map(c => c.node.id).join(PATH_SEP);\n  }\n\n  /**\n   * Returns the CloudFormation 'snippet' for this entity. The snippet will only be merged\n   * at the root level to ensure there are no identity conflicts.\n   *\n   * For example, a Resource class will return something like:\n   * {\n   *   Resources: {\n   *     [this.logicalId]: {\n   *       Type: this.resourceType,\n   *       Properties: this.props,\n   *       Condition: this.condition\n   *     }\n   *   }\n   * }\n   *\n   * @internal\n   */\n  public abstract _toCloudFormation(): object;\n\n  /**\n   * Automatically detect references in this CfnElement\n   */\n  protected prepare() {\n    try {\n      // Note: it might be that the properties of the CFN object aren't valid.\n      // This will usually be preventatively caught in a construct's validate()\n      // and turned into a nicely descriptive error, but we're running prepare()\n      // before validate(). Swallow errors that occur because the CFN layer\n      // doesn't validate completely.\n      //\n      // This does make the assumption that the error will not be rectified,\n      // but the error will be thrown later on anyway. If the error doesn't\n      // get thrown down the line, we may miss references.\n      this.node.recordReference(...findTokens(this, () => this._toCloudFormation()));\n    } catch (e) {\n      if (e.type !== 'CfnSynthesisError') { throw e; }\n    }\n  }\n}\n\n/**\n * Base class for referenceable CloudFormation constructs which are not Resources\n *\n * These constructs are things like Conditions and Parameters, can be\n * referenced by taking the `.ref` attribute.\n *\n * Resource constructs do not inherit from CfnRefElement because they have their\n * own, more specific types returned from the .ref attribute. Also, some\n * resources aren't referenceable at all (such as BucketPolicies or GatewayAttachments).\n */\nexport abstract class CfnRefElement extends CfnElement {\n  /**\n   * Returns a token to a CloudFormation { Ref } that references this entity based on it's logical ID.\n   */\n  public get ref(): string {\n    return this.referenceToken.toString();\n  }\n\n  /**\n   * Return a token that will CloudFormation { Ref } this stack element\n   */\n  public get referenceToken(): Token {\n    return CfnReference.for(this, 'Ref');\n  }\n}\n\nfunction notTooLong(x: string) {\n  if (x.length < 100) { return x; }\n  return x.substr(0, 47) + '...' + x.substr(x.length - 47);\n}\n\nimport { CfnReference } from \"./cfn-reference\";\nimport { findTokens } from \"./resolve\";\n"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cfn-element.js","sourceRoot":"","sources":["cfn-element.ts"],"names":[],"mappings":"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","sourcesContent":["import { Construct, IConstruct, PATH_SEP } from \"./construct\";\nimport { Token } from './token';\n\nconst LOGICAL_ID_MD = 'aws:cdk:logicalId';\n\n/**\n * An element of a CloudFormation stack.\n */\nexport abstract class CfnElement extends Construct {\n  /**\n   * Returns `true` if a construct is a stack element (i.e. part of the\n   * synthesized cloudformation template).\n   *\n   * Uses duck-typing instead of `instanceof` to allow stack elements from different\n   * versions of this library to be included in the same stack.\n   *\n   * @returns The construct as a stack element or undefined if it is not a stack element.\n   */\n  public static isCfnElement(construct: IConstruct): construct is CfnElement {\n    return ('logicalId' in construct && '_toCloudFormation' in construct);\n  }\n\n  /**\n   * The logical ID for this CloudFormation stack element. The logical ID of the element\n   * is calculated from the path of the resource node in the construct tree.\n   *\n   * To override this value, use `overrideLogicalId(newLogicalId)`.\n   *\n   * @returns the logical ID as a stringified token. This value will only get\n   * resolved during synthesis.\n   */\n  public readonly logicalId: string;\n\n  private _logicalId: string;\n\n  /**\n   * Creates an entity and binds it to a tree.\n   * Note that the root of the tree must be a Stack object (not just any Root).\n   *\n   * @param scope The parent construct\n   * @param props Construct properties\n   */\n  constructor(scope: Construct, id: string) {\n    super(scope, id);\n\n    this.node.addMetadata(LOGICAL_ID_MD, new (require(\"./token\").Token)(() => this.logicalId), this.constructor);\n\n    this._logicalId = this.node.stack.logicalIds.getLogicalId(this);\n    this.logicalId = new Token(() => this._logicalId, `${notTooLong(this.node.path)}.LogicalID`).toString();\n  }\n\n  /**\n   * Overrides the auto-generated logical ID with a specific ID.\n   * @param newLogicalId The new logical ID to use for this stack element.\n   */\n  public overrideLogicalId(newLogicalId: string) {\n    this._logicalId = newLogicalId;\n  }\n\n  /**\n   * @returns the stack trace of the point where this Resource was created from, sourced\n   *      from the +metadata+ entry typed +aws:cdk:logicalId+, and with the bottom-most\n   *      node +internal+ entries filtered.\n   */\n  public get creationStackTrace(): string[] | undefined {\n    const trace = this.node.metadata.find(md => md.type === LOGICAL_ID_MD)!.trace;\n    if (!trace) {\n      return undefined;\n    }\n\n    return filterStackTrace(trace);\n\n    function filterStackTrace(stack: string[]): string[] {\n      const result = Array.of(...stack);\n      while (result.length > 0 && shouldFilter(result[result.length - 1])) {\n        result.pop();\n      }\n      // It's weird if we filtered everything, so return the whole stack...\n      return result.length === 0 ? stack : result;\n    }\n\n    function shouldFilter(str: string): boolean {\n      return str.match(/[^(]+\\(internal\\/.*/) !== null;\n    }\n  }\n\n  /**\n   * Return the path with respect to the stack\n   */\n  public get stackPath(): string {\n    return this.node.ancestors(this.node.stack).map(c => c.node.id).join(PATH_SEP);\n  }\n\n  /**\n   * Returns the CloudFormation 'snippet' for this entity. The snippet will only be merged\n   * at the root level to ensure there are no identity conflicts.\n   *\n   * For example, a Resource class will return something like:\n   * {\n   *   Resources: {\n   *     [this.logicalId]: {\n   *       Type: this.resourceType,\n   *       Properties: this.props,\n   *       Condition: this.condition\n   *     }\n   *   }\n   * }\n   *\n   * @internal\n   */\n  public abstract _toCloudFormation(): object;\n\n  /**\n   * Automatically detect references in this CfnElement\n   */\n  protected prepare() {\n    try {\n      // Note: it might be that the properties of the CFN object aren't valid.\n      // This will usually be preventatively caught in a construct's validate()\n      // and turned into a nicely descriptive error, but we're running prepare()\n      // before validate(). Swallow errors that occur because the CFN layer\n      // doesn't validate completely.\n      //\n      // This does make the assumption that the error will not be rectified,\n      // but the error will be thrown later on anyway. If the error doesn't\n      // get thrown down the line, we may miss references.\n      this.node.recordReference(...findTokens(this, () => this._toCloudFormation()));\n    } catch (e) {\n      if (e.type !== 'CfnSynthesisError') { throw e; }\n    }\n  }\n}\n\n/**\n * Base class for referenceable CloudFormation constructs which are not Resources\n *\n * These constructs are things like Conditions and Parameters, can be\n * referenced by taking the `.ref` attribute.\n *\n * Resource constructs do not inherit from CfnRefElement because they have their\n * own, more specific types returned from the .ref attribute. Also, some\n * resources aren't referenceable at all (such as BucketPolicies or GatewayAttachments).\n */\nexport abstract class CfnRefElement extends CfnElement {\n  /**\n   * Returns a token to a CloudFormation { Ref } that references this entity based on it's logical ID.\n   */\n  public get ref(): string {\n    return this.referenceToken.toString();\n  }\n\n  /**\n   * Return a token that will CloudFormation { Ref } this stack element\n   */\n  public get referenceToken(): Token {\n    return CfnReference.for(this, 'Ref');\n  }\n}\n\nfunction notTooLong(x: string) {\n  if (x.length < 100) { return x; }\n  return x.substr(0, 47) + '...' + x.substr(x.length - 47);\n}\n\nimport { CfnReference } from \"./cfn-reference\";\nimport { findTokens } from \"./resolve\";\n"]}

@@ -93,2 +93,6 @@ "use strict";

consumeFromStack(consumingStack, consumingConstruct) {
if (this.producingStack && consumingStack.node.root !== this.producingStack.node.root) {
throw this.newError(`Cannot reference across apps. ` +
`Consuming and producing stacks must be defined within the same CDK app.`);
}
// tslint:disable-next-line:max-line-length

@@ -109,3 +113,3 @@ if (this.producingStack && this.producingStack !== consumingStack && !this.replacementTokens.has(consumingStack)) {

if (producingStack.env.account !== consumingStack.env.account || producingStack.env.region !== consumingStack.env.region) {
throw new Error('Can only reference cross stacks in the same region and account.');
throw this.newError('Can only reference cross stacks in the same region and account.');
}

@@ -141,2 +145,2 @@ // Ensure a singleton "Exports" scoping Construct

const token_1 = require("./token");
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cfn-reference.js","sourceRoot":"","sources":["cfn-reference.ts"],"names":[],"mappings":"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","sourcesContent":["import { Reference } from \"./reference\";\n\nconst CFN_REFERENCE_SYMBOL = Symbol.for('@aws-cdk/cdk.CfnReference');\n\n/**\n * A Token that represents a CloudFormation reference to another resource\n *\n * If these references are used in a different stack from where they are\n * defined, appropriate CloudFormation `Export`s and `Fn::ImportValue`s will be\n * synthesized automatically instead of the regular CloudFormation references.\n *\n * Additionally, the dependency between the stacks will be recorded, and the toolkit\n * will make sure to deploy producing stack before the consuming stack.\n *\n * This magic happens in the prepare() phase, where consuming stacks will call\n * `consumeFromStack` on these Tokens and if they happen to be exported by a different\n * Stack, we'll register the dependency.\n */\nexport class CfnReference extends Reference {\n  /**\n   * Check whether this is actually a Reference\n   */\n  public static isCfnReference(x: Token): x is CfnReference {\n    return (x as any)[CFN_REFERENCE_SYMBOL] === true;\n  }\n\n  /**\n   * Return the CfnReference for the indicated target\n   *\n   * Will make sure that multiple invocations for the same target and intrinsic\n   * return the same CfnReference. Because CfnReferences accumulate state in\n   * the prepare() phase (for the purpose of cross-stack references), it's\n   * important that the state isn't lost if it's lazily created, like so:\n   *\n   *     new Token(() => new CfnReference(...))\n   */\n  public static for(target: CfnRefElement, attribute: string) {\n    return CfnReference.singletonReference(target, attribute, () => {\n      const cfnInstrinsic = attribute === 'Ref' ? { Ref: target.logicalId } : { 'Fn::GetAtt': [ target.logicalId, attribute ]};\n      return new CfnReference(cfnInstrinsic, attribute, target);\n    });\n  }\n\n  /**\n   * Return a CfnReference that references a pseudo referencd\n   */\n  public static forPseudo(pseudoName: string, scope: Construct) {\n    return CfnReference.singletonReference(scope, `Pseudo:${pseudoName}`, () => {\n      const cfnInstrinsic = { Ref: pseudoName };\n      return new CfnReference(cfnInstrinsic, pseudoName, scope);\n    });\n  }\n\n  /**\n   * Static table where we keep singleton CfnReference instances\n   */\n  private static referenceTable = new Map<Construct, Map<string, CfnReference>>();\n\n  /**\n   * Get or create the table\n   */\n  private static singletonReference(target: Construct, attribKey: string, fresh: () => CfnReference) {\n    let attribs = CfnReference.referenceTable.get(target);\n    if (!attribs) {\n      attribs = new Map();\n      CfnReference.referenceTable.set(target, attribs);\n    }\n    let ref = attribs.get(attribKey);\n    if (!ref) {\n      ref = fresh();\n      attribs.set(attribKey, ref);\n    }\n    return ref;\n  }\n\n  /**\n   * What stack this Token is pointing to\n   */\n  private readonly producingStack?: Stack;\n\n  /**\n   * The Tokens that should be returned for each consuming stack (as decided by the producing Stack)\n   */\n  private readonly replacementTokens: Map<Stack, Token>;\n\n  private readonly originalDisplayName: string;\n\n  private constructor(value: any, displayName: string, target: Construct) {\n    if (typeof(value) === 'function') {\n      throw new Error('Reference can only hold CloudFormation intrinsics (not a function)');\n    }\n    // prepend scope path to display name\n    super(value, `${target.node.id}.${displayName}`, target);\n    this.originalDisplayName = displayName;\n    this.replacementTokens = new Map<Stack, Token>();\n\n    this.producingStack = target.node.stack;\n    Object.defineProperty(this, CFN_REFERENCE_SYMBOL, { value: true });\n  }\n\n  public resolve(context: IResolveContext): any {\n    // If we have a special token for this consuming stack, resolve that. Otherwise resolve as if\n    // we are in the same stack.\n    const token = this.replacementTokens.get(context.scope.node.stack);\n    if (token) {\n      return token.resolve(context);\n    } else {\n      return super.resolve(context);\n    }\n  }\n\n  /**\n   * Register a stack this references is being consumed from.\n   */\n  public consumeFromStack(consumingStack: Stack, consumingConstruct: IConstruct) {\n    // tslint:disable-next-line:max-line-length\n    if (this.producingStack && this.producingStack !== consumingStack && !this.replacementTokens.has(consumingStack)) {\n      // We're trying to resolve a cross-stack reference\n      consumingStack.addDependency(this.producingStack, `${consumingConstruct.node.path} -> ${this.target.node.path}.${this.originalDisplayName}`);\n      this.replacementTokens.set(consumingStack, this.exportValue(this, consumingStack));\n    }\n  }\n\n  /**\n   * Export a Token value for use in another stack\n   *\n   * Works by mutating the producing stack in-place.\n   */\n  private exportValue(tokenValue: Token, consumingStack: Stack): Token {\n    const producingStack = this.producingStack!;\n\n    if (producingStack.env.account !== consumingStack.env.account || producingStack.env.region !== consumingStack.env.region) {\n      throw new Error('Can only reference cross stacks in the same region and account.');\n    }\n\n    // Ensure a singleton \"Exports\" scoping Construct\n    // This mostly exists to trigger LogicalID munging, which would be\n    // disabled if we parented constructs directly under Stack.\n    // Also it nicely prevents likely construct name clashes\n\n    const exportsName = 'Exports';\n    let stackExports = producingStack.node.tryFindChild(exportsName) as Construct;\n    if (stackExports === undefined) {\n      stackExports = new Construct(producingStack, exportsName);\n    }\n\n    // Ensure a singleton CfnOutput for this value\n    const resolved = producingStack.node.resolve(tokenValue);\n    const id = 'Output' + JSON.stringify(resolved);\n    let output = stackExports.node.tryFindChild(id) as CfnOutput;\n    if (!output) {\n      output = new CfnOutput(stackExports, id, { value: tokenValue });\n    }\n\n    // We want to return an actual FnImportValue Token here, but Fn.importValue() returns a 'string',\n    // so construct one in-place.\n    return new Token({ 'Fn::ImportValue': output.obtainExportName() });\n  }\n}\n\nimport { CfnRefElement } from \"./cfn-element\";\nimport { CfnOutput } from \"./cfn-output\";\nimport { Construct, IConstruct } from \"./construct\";\nimport { Stack } from \"./stack\";\nimport { IResolveContext, Token } from \"./token\";\n"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cfn-reference.js","sourceRoot":"","sources":["cfn-reference.ts"],"names":[],"mappings":"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","sourcesContent":["import { Reference } from \"./reference\";\n\nconst CFN_REFERENCE_SYMBOL = Symbol.for('@aws-cdk/cdk.CfnReference');\n\n/**\n * A Token that represents a CloudFormation reference to another resource\n *\n * If these references are used in a different stack from where they are\n * defined, appropriate CloudFormation `Export`s and `Fn::ImportValue`s will be\n * synthesized automatically instead of the regular CloudFormation references.\n *\n * Additionally, the dependency between the stacks will be recorded, and the toolkit\n * will make sure to deploy producing stack before the consuming stack.\n *\n * This magic happens in the prepare() phase, where consuming stacks will call\n * `consumeFromStack` on these Tokens and if they happen to be exported by a different\n * Stack, we'll register the dependency.\n */\nexport class CfnReference extends Reference {\n  /**\n   * Check whether this is actually a Reference\n   */\n  public static isCfnReference(x: Token): x is CfnReference {\n    return (x as any)[CFN_REFERENCE_SYMBOL] === true;\n  }\n\n  /**\n   * Return the CfnReference for the indicated target\n   *\n   * Will make sure that multiple invocations for the same target and intrinsic\n   * return the same CfnReference. Because CfnReferences accumulate state in\n   * the prepare() phase (for the purpose of cross-stack references), it's\n   * important that the state isn't lost if it's lazily created, like so:\n   *\n   *     new Token(() => new CfnReference(...))\n   */\n  public static for(target: CfnRefElement, attribute: string) {\n    return CfnReference.singletonReference(target, attribute, () => {\n      const cfnInstrinsic = attribute === 'Ref' ? { Ref: target.logicalId } : { 'Fn::GetAtt': [ target.logicalId, attribute ]};\n      return new CfnReference(cfnInstrinsic, attribute, target);\n    });\n  }\n\n  /**\n   * Return a CfnReference that references a pseudo referencd\n   */\n  public static forPseudo(pseudoName: string, scope: Construct) {\n    return CfnReference.singletonReference(scope, `Pseudo:${pseudoName}`, () => {\n      const cfnInstrinsic = { Ref: pseudoName };\n      return new CfnReference(cfnInstrinsic, pseudoName, scope);\n    });\n  }\n\n  /**\n   * Static table where we keep singleton CfnReference instances\n   */\n  private static referenceTable = new Map<Construct, Map<string, CfnReference>>();\n\n  /**\n   * Get or create the table\n   */\n  private static singletonReference(target: Construct, attribKey: string, fresh: () => CfnReference) {\n    let attribs = CfnReference.referenceTable.get(target);\n    if (!attribs) {\n      attribs = new Map();\n      CfnReference.referenceTable.set(target, attribs);\n    }\n    let ref = attribs.get(attribKey);\n    if (!ref) {\n      ref = fresh();\n      attribs.set(attribKey, ref);\n    }\n    return ref;\n  }\n\n  /**\n   * What stack this Token is pointing to\n   */\n  private readonly producingStack?: Stack;\n\n  /**\n   * The Tokens that should be returned for each consuming stack (as decided by the producing Stack)\n   */\n  private readonly replacementTokens: Map<Stack, Token>;\n\n  private readonly originalDisplayName: string;\n\n  private constructor(value: any, displayName: string, target: Construct) {\n    if (typeof(value) === 'function') {\n      throw new Error('Reference can only hold CloudFormation intrinsics (not a function)');\n    }\n\n    // prepend scope path to display name\n    super(value, `${target.node.id}.${displayName}`, target);\n    this.originalDisplayName = displayName;\n    this.replacementTokens = new Map<Stack, Token>();\n\n    this.producingStack = target.node.stack;\n    Object.defineProperty(this, CFN_REFERENCE_SYMBOL, { value: true });\n  }\n\n  public resolve(context: IResolveContext): any {\n    // If we have a special token for this consuming stack, resolve that. Otherwise resolve as if\n    // we are in the same stack.\n    const token = this.replacementTokens.get(context.scope.node.stack);\n    if (token) {\n      return token.resolve(context);\n    } else {\n      return super.resolve(context);\n    }\n  }\n\n  /**\n   * Register a stack this references is being consumed from.\n   */\n  public consumeFromStack(consumingStack: Stack, consumingConstruct: IConstruct) {\n    if (this.producingStack && consumingStack.node.root !== this.producingStack.node.root) {\n      throw this.newError(\n        `Cannot reference across apps. ` +\n        `Consuming and producing stacks must be defined within the same CDK app.`);\n    }\n\n    // tslint:disable-next-line:max-line-length\n    if (this.producingStack && this.producingStack !== consumingStack && !this.replacementTokens.has(consumingStack)) {\n      // We're trying to resolve a cross-stack reference\n      consumingStack.addDependency(this.producingStack, `${consumingConstruct.node.path} -> ${this.target.node.path}.${this.originalDisplayName}`);\n      this.replacementTokens.set(consumingStack, this.exportValue(this, consumingStack));\n    }\n  }\n\n  /**\n   * Export a Token value for use in another stack\n   *\n   * Works by mutating the producing stack in-place.\n   */\n  private exportValue(tokenValue: Token, consumingStack: Stack): Token {\n    const producingStack = this.producingStack!;\n\n    if (producingStack.env.account !== consumingStack.env.account || producingStack.env.region !== consumingStack.env.region) {\n      throw this.newError('Can only reference cross stacks in the same region and account.');\n    }\n\n    // Ensure a singleton \"Exports\" scoping Construct\n    // This mostly exists to trigger LogicalID munging, which would be\n    // disabled if we parented constructs directly under Stack.\n    // Also it nicely prevents likely construct name clashes\n\n    const exportsName = 'Exports';\n    let stackExports = producingStack.node.tryFindChild(exportsName) as Construct;\n    if (stackExports === undefined) {\n      stackExports = new Construct(producingStack, exportsName);\n    }\n\n    // Ensure a singleton CfnOutput for this value\n    const resolved = producingStack.node.resolve(tokenValue);\n    const id = 'Output' + JSON.stringify(resolved);\n    let output = stackExports.node.tryFindChild(id) as CfnOutput;\n    if (!output) {\n      output = new CfnOutput(stackExports, id, { value: tokenValue });\n    }\n\n    // We want to return an actual FnImportValue Token here, but Fn.importValue() returns a 'string',\n    // so construct one in-place.\n    return new Token({ 'Fn::ImportValue': output.obtainExportName() });\n  }\n}\n\nimport { CfnRefElement } from \"./cfn-element\";\nimport { CfnOutput } from \"./cfn-output\";\nimport { Construct, IConstruct } from \"./construct\";\nimport { Stack } from \"./stack\";\nimport { IResolveContext, Token } from \"./token\";\n"]}

@@ -188,5 +188,8 @@ "use strict";

// Adjust stack trace (make it look like node built it, too...)
const creationStack = ['--- resource created at ---', ...this.creationStackTrace].join('\n at ');
const problemTrace = e.stack.substr(e.stack.indexOf(e.message) + e.message.length);
e.stack = `${e.message}\n ${creationStack}\n --- problem discovered at ---${problemTrace}`;
const trace = this.creationStackTrace;
if (trace) {
const creationStack = ['--- resource created at ---', ...trace].join('\n at ');
const problemTrace = e.stack.substr(e.stack.indexOf(e.message) + e.message.length);
e.stack = `${e.message}\n ${creationStack}\n --- problem discovered at ---${problemTrace}`;
}
// Re-throw

@@ -256,2 +259,2 @@ throw e;

exports.deepMerge = deepMerge;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cfn-resource.js","sourceRoot":"","sources":["cfn-resource.ts"],"names":[],"mappings":";;AAAA,yCAA0C;AAK1C,iCAAgF;AAChF,6FAA6F;AAC7F,2CAA2C;AAC3C,+CAA8C;AAC9C,mDAA+C;AAsB/C;;GAEG;AACH,MAAa,WAAY,SAAQ,2BAAa;IAwE5C;;;OAGG;IACH,YAAY,KAAgB,EAAE,EAAU,EAAE,KAAuB;QAC/D,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QA9CnB;;WAEG;QACa,YAAO,GAAqB,EAAE,CAAC;QAc/C;;;;;;;;;WASG;QACgB,6BAAwB,GAAQ,EAAG,CAAC;QAEvD;;WAEG;QACc,iBAAY,GAAQ,EAAG,CAAC;QAEzC;;;;WAIG;QACc,cAAS,GAAG,IAAI,GAAG,EAAe,CAAC;QASlD,IAAI,CAAC,KAAK,CAAC,IAAI,EAAE;YACf,MAAM,IAAI,KAAK,CAAC,iCAAiC,CAAC,CAAC;SACpD;QAED,IAAI,CAAC,YAAY,GAAG,KAAK,CAAC,IAAI,CAAC;QAC/B,IAAI,CAAC,UAAU,GAAG,KAAK,CAAC,UAAU,IAAI,EAAG,CAAC;QAE1C,wEAAwE;QACxE,uEAAuE;QACvE,qCAAqC;QACrC,IAAI,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,4BAA4B,CAAC,EAAE;YAC5D,IAAI,CAAC,OAAO,CAAC,QAAQ,GAAG;gBACtB,CAAC,KAAK,CAAC,iBAAiB,CAAC,EAAE,IAAI,CAAC,IAAI,CAAC,IAAI;aAC1C,CAAC;SACH;IACH,CAAC;IA7FD;;;;OAIG;IACI,MAAM,CAAC,SAAS,CAAC,UAAmB;QACzC,OAAO,CAAC,SAAc,EAAE,GAAW,EAAE,EAAE;YACrC,MAAM,IAAI,GAAG,UAAU,IAAI,GAAG,CAAC;YAC/B,MAAM,CAAC,cAAc,CAAC,SAAS,EAAE,GAAG,EAAE;gBACpC,GAAG;oBACD,OAAQ,IAAY,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBACpC,CAAC;aACF,CAAC,CAAC;QACL,CAAC,CAAC;IACJ,CAAC;IAED;;OAEG;IACI,MAAM,CAAC,aAAa,CAAC,SAAqB;QAC/C,OAAQ,SAAiB,CAAC,YAAY,KAAK,SAAS,CAAC;IACvD,CAAC;IAED;;OAEG;IACI,MAAM,CAAC,UAAU,CAAC,SAAc;QACrC,OAAQ,SAAiB,CAAC,IAAI,KAAK,SAAS,CAAC;IAC/C,CAAC;IAmED;;;;;OAKG;IACI,MAAM,CAAC,aAAqB;QACjC,OAAO,4BAAY,CAAC,GAAG,CAAC,IAAI,EAAE,aAAa,CAAC,CAAC;IAC/C,CAAC;IAED;;;;;;;;;OASG;IACI,WAAW,CAAC,IAAY,EAAE,KAAU;QACzC,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QAC9B,IAAI,IAAI,GAAQ,IAAI,CAAC,YAAY,CAAC;QAElC,OAAO,KAAK,CAAC,MAAM,GAAG,CAAC,EAAE;YACvB,MAAM,GAAG,GAAG,KAAK,CAAC,KAAK,EAAG,CAAC;YAE3B,8DAA8D;YAC9D,sCAAsC;YACtC,MAAM,QAAQ,GAAG,IAAI,CAAC,GAAG,CAAC,IAAI,IAAI,IAAI,OAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,KAAK,QAAQ,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC;YAClG,IAAI,CAAC,QAAQ,EAAE;gBACb,IAAI,CAAC,GAAG,CAAC,GAAG,EAAG,CAAC;aACjB;YAED,IAAI,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC;SAClB;QAED,MAAM,OAAO,GAAG,KAAK,CAAC,KAAK,EAAG,CAAC;QAC/B,IAAI,CAAC,OAAO,CAAC,GAAG,KAAK,CAAC;IACxB,CAAC;IAED;;;OAGG;IACI,mBAAmB,CAAC,IAAY;QACrC,IAAI,CAAC,WAAW,CAAC,IAAI,EAAE,SAAS,CAAC,CAAC;IACpC,CAAC;IAED;;;;;;;OAOG;IACI,mBAAmB,CAAC,YAAoB,EAAE,KAAU;QACzD,IAAI,CAAC,WAAW,CAAC,cAAc,YAAY,EAAE,EAAE,KAAK,CAAC,CAAC;IACxD,CAAC;IAED;;;OAGG;IACI,2BAA2B,CAAC,YAAoB;QACrD,IAAI,CAAC,mBAAmB,CAAC,YAAY,EAAE,SAAS,CAAC,CAAC;IACpD,CAAC;IAED;;;OAGG;IACI,YAAY,CAAC,QAAqB;QACvC,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;IAC/B,CAAC;IAED;;;OAGG;IACI,iBAAiB;QACtB,IAAI;YACF,2EAA2E;YAC3E,MAAM,IAAI,GAAG,WAAW,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC;YAC/E,MAAM,UAAU,GAAG,SAAS,CAC1B,IAAI,CAAC,UAAU,IAAI,EAAE,EACrB,EAAE,IAAI,EAAE,EACR,IAAI,CAAC,wBAAwB,CAC9B,CAAC;YAEF,MAAM,GAAG,GAAG;gBACV,SAAS,EAAE;oBACT,+EAA+E;oBAC/E,6CAA6C;oBAC7C,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,IAAI,uBAAgB,CAAC;wBACrC,IAAI,EAAE,IAAI,CAAC,YAAY;wBACvB,UAAU,EAAE,kBAAW,CAAC,UAAU,CAAC;wBACnC,SAAS,EAAE,kBAAW,CAAC,eAAe,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;wBACvD,cAAc,EAAG,8BAAuB,CAAC,IAAI,EAAE,IAAI,CAAC,OAAO,CAAC,cAAc,CAAC;wBAC3E,YAAY,EAAE,8BAAuB,CAAC,IAAI,EAAE,IAAI,CAAC,OAAO,CAAC,YAAY,CAAC;wBACtE,mBAAmB,EAAE,8BAAuB,CAAC,IAAI,EAAE,IAAI,CAAC,OAAO,CAAC,mBAAmB,CAAC;wBACpF,cAAc,EAAE,8BAAuB,CAAC,IAAI,EAAE,IAAI,CAAC,OAAO,CAAC,cAAc,CAAC;wBAC1E,QAAQ,EAAE,kBAAW,CAAC,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC;wBAC5C,SAAS,EAAE,IAAI,CAAC,OAAO,CAAC,SAAS,IAAI,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,SAAS;qBACtE,EAAE,KAAK,CAAC,EAAE;wBACT,MAAM,CAAC,GAAG,SAAS,CAAC,KAAK,EAAE,IAAI,CAAC,YAAY,CAAC,CAAC;wBAC9C,CAAC,CAAC,UAAU,GAAG,IAAI,CAAC,gBAAgB,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC;wBACnD,OAAO,CAAC,CAAC;oBACX,CAAC,CAAC;iBACH;aACF,CAAC;YACF,OAAO,GAAG,CAAC;SACZ;QAAC,OAAO,CAAC,EAAE;YACV,iBAAiB;YACjB,CAAC,CAAC,OAAO,GAAG,sBAAsB,IAAI,CAAC,IAAI,CAAC,IAAI,KAAK,CAAC,CAAC,OAAO,EAAE,CAAC;YACjE,+DAA+D;YAC/D,MAAM,aAAa,GAAG,CAAC,6BAA6B,EAAE,GAAG,IAAI,CAAC,kBAAkB,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;YAClG,MAAM,YAAY,GAAG,CAAC,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;YACnF,CAAC,CAAC,KAAK,GAAG,GAAG,CAAC,CAAC,OAAO,OAAO,aAAa,oCAAoC,YAAY,EAAE,CAAC;YAC7F,WAAW;YACX,MAAM,CAAC,CAAC;SACT;QAED,kEAAkE;QAClE,uDAAuD;QACvD,SAAS,eAAe,CAAC,SAA2B;YAClD,OAAO,KAAK;iBACT,IAAI,CAAC,SAAS,CAAC;iBACf,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;iBACtD,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC;QAC3B,CAAC;IACH,CAAC;IAES,gBAAgB,CAAC,UAAe;QACxC,OAAO,UAAU,CAAC;IACpB,CAAC;IAES,kBAAkB,CAAC,WAAgB;QAC3C,UAAU;IACZ,CAAC;CACF;AA7OD,kCA6OC;AAED,IAAY,OAKX;AALD,WAAY,OAAO;IACjB,mCAAwB,CAAA;IACxB,mDAAwC,CAAA;IACxC,oCAAyB,CAAA;IACzB,sCAA2B,CAAA;AAC7B,CAAC,EALW,OAAO,GAAP,eAAO,KAAP,eAAO,QAKlB;AA8CD;;;GAGG;AACH,SAAgB,SAAS,CAAC,MAAW,EAAE,GAAG,OAAc;IACtD,KAAK,MAAM,MAAM,IAAI,OAAO,EAAE;QAC5B,IAAI,OAAM,CAAC,MAAM,CAAC,KAAK,QAAQ,IAAI,OAAM,CAAC,MAAM,CAAC,KAAK,QAAQ,EAAE;YAC9D,MAAM,IAAI,KAAK,CAAC,+BAA+B,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,iBAAiB,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,mBAAmB,CAAC,CAAC;SAClI;QAED,KAAK,MAAM,GAAG,IAAI,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE;YACrC,MAAM,KAAK,GAAG,MAAM,CAAC,GAAG,CAAC,CAAC;YAC1B,IAAI,OAAM,CAAC,KAAK,CAAC,KAAK,QAAQ,IAAI,KAAK,IAAI,IAAI,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE;gBACxE,mEAAmE;gBACnE,0CAA0C;gBAC1C,IAAI,OAAM,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,KAAK,QAAQ,EAAE;oBACpC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAG,CAAC;iBACnB;gBAED,SAAS,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,KAAK,CAAC,CAAC;gBAE9B,kEAAkE;gBAClE,8DAA8D;gBAC9D,iEAAiE;gBACjE,MAAM,MAAM,GAAG,MAAM,CAAC,GAAG,CAAC,CAAC;gBAC3B,IAAI,OAAM,CAAC,MAAM,CAAC,KAAK,QAAQ,IAAI,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,MAAM,KAAK,CAAC,EAAE;oBACnE,OAAO,MAAM,CAAC,GAAG,CAAC,CAAC;iBACpB;aACF;iBAAM,IAAI,KAAK,KAAK,SAAS,EAAE;gBAC9B,OAAO,MAAM,CAAC,GAAG,CAAC,CAAC;aACpB;iBAAM;gBACL,MAAM,CAAC,GAAG,CAAC,GAAG,KAAK,CAAC;aACrB;SACF;KACF;IAED,OAAO,MAAM,CAAC;AAChB,CAAC;AAjCD,8BAiCC","sourcesContent":["import cxapi = require('@aws-cdk/cx-api');\nimport { CfnCondition } from './cfn-condition';\nimport { Construct, IConstruct } from './construct';\nimport { CreationPolicy, DeletionPolicy, UpdatePolicy } from './resource-policy';\nimport { TagManager } from './tag-manager';\nimport { capitalizePropertyNames, ignoreEmpty, PostResolveToken } from './util';\n// import required to be here, otherwise causes a cycle when running the generated JavaScript\n// tslint:disable-next-line:ordered-imports\nimport { CfnRefElement } from './cfn-element';\nimport { CfnReference } from './cfn-reference';\n\nexport interface CfnResourceProps {\n  /**\n   * CloudFormation resource type.\n   */\n  readonly type: string;\n\n  /**\n   * CloudFormation properties.\n   *\n   * @default - No resource properties.\n   */\n  readonly properties?: any;\n}\n\nexport interface ITaggable {\n  /**\n   * TagManager to set, remove and format tags\n   */\n  readonly tags: TagManager;\n}\n/**\n * Represents a CloudFormation resource.\n */\nexport class CfnResource extends CfnRefElement {\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   * Check whether the given construct is a CfnResource\n   */\n  public static isCfnResource(construct: IConstruct): construct is CfnResource {\n    return (construct as any).resourceType !== undefined;\n  }\n\n  /**\n   * Check whether the given construct is Taggable\n   */\n  public static isTaggable(construct: any): construct is ITaggable {\n    return (construct as any).tags !== undefined;\n  }\n\n  /**\n   * Options for this resource, such as condition, update policy etc.\n   */\n  public readonly options: IResourceOptions = {};\n\n  /**\n   * AWS resource type.\n   */\n  public readonly resourceType: string;\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   * 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   * An object to be merged on top of the entire resource definition.\n   */\n  private readonly rawOverrides: any = { };\n\n  /**\n   * Logical IDs of dependencies.\n   *\n   * Is filled during prepare().\n   */\n  private readonly dependsOn = new Set<CfnResource>();\n\n  /**\n   * Creates a resource construct.\n   * @param resourceType The CloudFormation type of this resource (e.g. AWS::DynamoDB::Table)\n   */\n  constructor(scope: Construct, id: string, props: CfnResourceProps) {\n    super(scope, id);\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    // if aws:cdk:enable-path-metadata is set, embed the current construct's\n    // path in the CloudFormation template, so it will be possible to trace\n    // back to the actual construct path.\n    if (this.node.getContext(cxapi.PATH_METADATA_ENABLE_CONTEXT)) {\n      this.options.metadata = {\n        [cxapi.PATH_METADATA_KEY]: this.node.path\n      };\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 CfnReference.for(this, attributeName);\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   * Indicates that this resource depends on another resource and cannot be provisioned\n   * unless the other resource has been successfully provisioned.\n   */\n  public addDependsOn(resource: CfnResource) {\n    this.dependsOn.add(resource);\n  }\n\n  /**\n   * Emits CloudFormation for this resource.\n   * @internal\n   */\n  public _toCloudFormation(): object {\n    try {\n      // merge property overrides onto properties and then render (and validate).\n      const tags = CfnResource.isTaggable(this) ? this.tags.renderTags() : undefined;\n      const properties = deepMerge(\n        this.properties || {},\n        { tags },\n        this.untypedPropertyOverrides\n      );\n\n      const ret = {\n        Resources: {\n          // Post-Resolve operation since otherwise deepMerge is going to mix values into\n          // the Token objects returned by ignoreEmpty.\n          [this.logicalId]: new PostResolveToken({\n            Type: this.resourceType,\n            Properties: ignoreEmpty(properties),\n            DependsOn: ignoreEmpty(renderDependsOn(this.dependsOn)),\n            CreationPolicy:  capitalizePropertyNames(this, this.options.creationPolicy),\n            UpdatePolicy: capitalizePropertyNames(this, this.options.updatePolicy),\n            UpdateReplacePolicy: capitalizePropertyNames(this, this.options.updateReplacePolicy),\n            DeletionPolicy: capitalizePropertyNames(this, this.options.deletionPolicy),\n            Metadata: ignoreEmpty(this.options.metadata),\n            Condition: this.options.condition && this.options.condition.logicalId\n          }, props => {\n            const r = deepMerge(props, this.rawOverrides);\n            r.Properties = this.renderProperties(r.Properties);\n            return r;\n          })\n        }\n      };\n      return ret;\n    } catch (e) {\n      // Change message\n      e.message = `While synthesizing ${this.node.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    // returns the set of logical ID (tokens) this resource depends on\n    // sorted by construct paths to ensure test determinism\n    function renderDependsOn(dependsOn: Set<CfnResource>) {\n      return Array\n        .from(dependsOn)\n        .sort((x, y) => x.node.path.localeCompare(y.node.path))\n        .map(r => r.logicalId);\n    }\n  }\n\n  protected renderProperties(properties: any): { [key: string]: any } {\n    return properties;\n  }\n\n  protected validateProperties(_properties: any) {\n    // Nothing\n  }\n}\n\nexport enum TagType {\n  Standard = 'StandardTag',\n  AutoScalingGroup = 'AutoScalingGroupTag',\n  Map = 'StringToStringMap',\n  NotTaggable = 'NotTaggable',\n}\n\nexport interface IResourceOptions {\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?: CfnCondition;\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   * Use the UpdateReplacePolicy attribute to retain or (in some cases) backup the existing physical instance of a resource\n   * when it is replaced during a stack update operation.\n   */\n  updateReplacePolicy?: DeletionPolicy;\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, ...sources: any[]) {\n  for (const source of sources) {\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\n  return target;\n}\n"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cfn-resource.js","sourceRoot":"","sources":["cfn-resource.ts"],"names":[],"mappings":"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","sourcesContent":["import cxapi = require('@aws-cdk/cx-api');\nimport { CfnCondition } from './cfn-condition';\nimport { Construct, IConstruct } from './construct';\nimport { CreationPolicy, DeletionPolicy, UpdatePolicy } from './resource-policy';\nimport { TagManager } from './tag-manager';\nimport { capitalizePropertyNames, ignoreEmpty, PostResolveToken } from './util';\n// import required to be here, otherwise causes a cycle when running the generated JavaScript\n// tslint:disable-next-line:ordered-imports\nimport { CfnRefElement } from './cfn-element';\nimport { CfnReference } from './cfn-reference';\n\nexport interface CfnResourceProps {\n  /**\n   * CloudFormation resource type.\n   */\n  readonly type: string;\n\n  /**\n   * CloudFormation properties.\n   *\n   * @default - No resource properties.\n   */\n  readonly properties?: any;\n}\n\nexport interface ITaggable {\n  /**\n   * TagManager to set, remove and format tags\n   */\n  readonly tags: TagManager;\n}\n/**\n * Represents a CloudFormation resource.\n */\nexport class CfnResource extends CfnRefElement {\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   * Check whether the given construct is a CfnResource\n   */\n  public static isCfnResource(construct: IConstruct): construct is CfnResource {\n    return (construct as any).resourceType !== undefined;\n  }\n\n  /**\n   * Check whether the given construct is Taggable\n   */\n  public static isTaggable(construct: any): construct is ITaggable {\n    return (construct as any).tags !== undefined;\n  }\n\n  /**\n   * Options for this resource, such as condition, update policy etc.\n   */\n  public readonly options: IResourceOptions = {};\n\n  /**\n   * AWS resource type.\n   */\n  public readonly resourceType: string;\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   * 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   * An object to be merged on top of the entire resource definition.\n   */\n  private readonly rawOverrides: any = { };\n\n  /**\n   * Logical IDs of dependencies.\n   *\n   * Is filled during prepare().\n   */\n  private readonly dependsOn = new Set<CfnResource>();\n\n  /**\n   * Creates a resource construct.\n   * @param resourceType The CloudFormation type of this resource (e.g. AWS::DynamoDB::Table)\n   */\n  constructor(scope: Construct, id: string, props: CfnResourceProps) {\n    super(scope, id);\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    // if aws:cdk:enable-path-metadata is set, embed the current construct's\n    // path in the CloudFormation template, so it will be possible to trace\n    // back to the actual construct path.\n    if (this.node.getContext(cxapi.PATH_METADATA_ENABLE_CONTEXT)) {\n      this.options.metadata = {\n        [cxapi.PATH_METADATA_KEY]: this.node.path\n      };\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 CfnReference.for(this, attributeName);\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   * Indicates that this resource depends on another resource and cannot be provisioned\n   * unless the other resource has been successfully provisioned.\n   */\n  public addDependsOn(resource: CfnResource) {\n    this.dependsOn.add(resource);\n  }\n\n  /**\n   * Emits CloudFormation for this resource.\n   * @internal\n   */\n  public _toCloudFormation(): object {\n    try {\n      // merge property overrides onto properties and then render (and validate).\n      const tags = CfnResource.isTaggable(this) ? this.tags.renderTags() : undefined;\n      const properties = deepMerge(\n        this.properties || {},\n        { tags },\n        this.untypedPropertyOverrides\n      );\n\n      const ret = {\n        Resources: {\n          // Post-Resolve operation since otherwise deepMerge is going to mix values into\n          // the Token objects returned by ignoreEmpty.\n          [this.logicalId]: new PostResolveToken({\n            Type: this.resourceType,\n            Properties: ignoreEmpty(properties),\n            DependsOn: ignoreEmpty(renderDependsOn(this.dependsOn)),\n            CreationPolicy:  capitalizePropertyNames(this, this.options.creationPolicy),\n            UpdatePolicy: capitalizePropertyNames(this, this.options.updatePolicy),\n            UpdateReplacePolicy: capitalizePropertyNames(this, this.options.updateReplacePolicy),\n            DeletionPolicy: capitalizePropertyNames(this, this.options.deletionPolicy),\n            Metadata: ignoreEmpty(this.options.metadata),\n            Condition: this.options.condition && this.options.condition.logicalId\n          }, props => {\n            const r = deepMerge(props, this.rawOverrides);\n            r.Properties = this.renderProperties(r.Properties);\n            return r;\n          })\n        }\n      };\n      return ret;\n    } catch (e) {\n      // Change message\n      e.message = `While synthesizing ${this.node.path}: ${e.message}`;\n      // Adjust stack trace (make it look like node built it, too...)\n      const trace = this.creationStackTrace;\n      if (trace) {\n        const creationStack = ['--- resource created at ---', ...trace].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      }\n\n      // Re-throw\n      throw e;\n    }\n\n    // returns the set of logical ID (tokens) this resource depends on\n    // sorted by construct paths to ensure test determinism\n    function renderDependsOn(dependsOn: Set<CfnResource>) {\n      return Array\n        .from(dependsOn)\n        .sort((x, y) => x.node.path.localeCompare(y.node.path))\n        .map(r => r.logicalId);\n    }\n  }\n\n  protected renderProperties(properties: any): { [key: string]: any } {\n    return properties;\n  }\n\n  protected validateProperties(_properties: any) {\n    // Nothing\n  }\n}\n\nexport enum TagType {\n  Standard = 'StandardTag',\n  AutoScalingGroup = 'AutoScalingGroupTag',\n  Map = 'StringToStringMap',\n  NotTaggable = 'NotTaggable',\n}\n\nexport interface IResourceOptions {\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?: CfnCondition;\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   * Use the UpdateReplacePolicy attribute to retain or (in some cases) backup the existing physical instance of a resource\n   * when it is replaced during a stack update operation.\n   */\n  updateReplacePolicy?: DeletionPolicy;\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, ...sources: any[]) {\n  for (const source of sources) {\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\n  return target;\n}\n"]}

@@ -0,1 +1,2 @@

import cxapi = require('@aws-cdk/cx-api');
import { IAspect } from './aspect';

@@ -126,3 +127,3 @@ import { IDependable } from './dependency';

*/
readonly metadata: MetadataEntry[];
readonly metadata: cxapi.MetadataEntry[];
/**

@@ -172,9 +173,19 @@ * Adds a metadata entry to this construct.

/**
* Return the ancestors (including self) of this Construct up until and excluding the indicated component
* Return the ancestors (including self) of this Construct up until and
* excluding the indicated component
*
* @param upTo The construct to return the path components relative to, or
* the entire list of ancestors (including root) if omitted.
* @param upTo The construct to return the path components relative to, or the
* entire list of ancestors (including root) if omitted. This construct will
* not be included in the returned list.
*
* @returns a list of parent scopes. The last element in the list will always
* be `this` and the first element is the oldest scope (if `upTo` is not set,
* it will be the root of the construct tree).
*/
ancestors(upTo?: Construct): IConstruct[];
/**
* @returns The root of the construct tree.
*/
readonly root: IConstruct;
/**
* Throws if the `props` bag doesn't include the property `name`.

@@ -266,9 +277,2 @@ * In the future we can add some type-checking here, maybe even auto-generate during compilation.

/**
* The set of constructs that form the root of this dependable
*
* All resources under all returned constructs are included in the ordering
* dependency.
*/
readonly dependencyRoots: IConstruct[];
/**
* Creates a new construct node.

@@ -307,19 +311,2 @@ *

}
/**
* An metadata entry in the construct.
*/
export interface MetadataEntry {
/**
* The type of the metadata entry.
*/
readonly type: string;
/**
* The data.
*/
readonly data?: any;
/**
* A stack trace for when the entry was created.
*/
readonly trace: string[];
}
export declare class ValidationError {

@@ -326,0 +313,0 @@ readonly source: IConstruct;

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

const cloudformation_lang_1 = require("./cloudformation-lang");
const dependency_1 = require("./dependency");
const resolve_1 = require("./resolve");
const stack_trace_1 = require("./stack-trace");
const token_1 = require("./token");

@@ -230,3 +232,3 @@ const uniqueid_1 = require("./uniqueid");

}
const trace = createStackTrace(from || this.addMetadata);
const trace = this.getContext(cxapi.DISABLE_METADATA_STACK_TRACE) ? undefined : stack_trace_1.createStackTrace(from || this.addMetadata);
this._metadata.push({ type, data, trace });

@@ -295,6 +297,12 @@ }

/**
* Return the ancestors (including self) of this Construct up until and excluding the indicated component
* Return the ancestors (including self) of this Construct up until and
* excluding the indicated component
*
* @param upTo The construct to return the path components relative to, or
* the entire list of ancestors (including root) if omitted.
* @param upTo The construct to return the path components relative to, or the
* entire list of ancestors (including root) if omitted. This construct will
* not be included in the returned list.
*
* @returns a list of parent scopes. The last element in the list will always
* be `this` and the first element is the oldest scope (if `upTo` is not set,
* it will be the root of the construct tree).
*/

@@ -311,2 +319,8 @@ ancestors(upTo) {

/**
* @returns The root of the construct tree.
*/
get root() {
return this.ancestors()[0];
}
/**
* Throws if the `props` bag doesn't include the property `name`.

@@ -441,3 +455,3 @@ * In the future we can add some type-checking here, maybe even auto-generate during compilation.

for (const dependable of source.node.dependencies) {
for (const target of dependable.dependencyRoots) {
for (const target of dependency_1.DependableTrait.get(dependable).dependencyRoots) {
let foundTargets = found.get(source);

@@ -488,2 +502,8 @@ if (!foundTargets) {

/**
* Return whether the given object is a Construct
*/
static isConstruct(x) {
return x.prepare !== undefined && x.validate !== undefined;
}
/**
* Creates a new construct node.

@@ -497,18 +517,10 @@ *

constructor(scope, id) {
/**
* The set of constructs that form the root of this dependable
*
* All resources under all returned constructs are included in the ordering
* dependency.
*/
this.dependencyRoots = [this];
this.node = new ConstructNode(this, scope, id);
// Implement IDependable privately
const self = this;
dependency_1.DependableTrait.implement(this, {
get dependencyRoots() { return [self]; },
});
}
/**
* Return whether the given object is a Construct
*/
static isConstruct(x) {
return x.prepare !== undefined && x.validate !== undefined;
}
/**
* Returns a string representation of this construct.

@@ -553,18 +565,2 @@ */

exports.ValidationError = ValidationError;
// tslint:disable-next-line:ban-types
function createStackTrace(below) {
const object = { stack: '' };
const previousLimit = Error.stackTraceLimit;
try {
Error.stackTraceLimit = Number.MAX_SAFE_INTEGER;
Error.captureStackTrace(object, below);
}
finally {
Error.stackTraceLimit = previousLimit;
}
if (!object.stack) {
return [];
}
return object.stack.split('\n').slice(1).map(s => s.replace(/^\s*at\s+/, ''));
}
/**

@@ -586,2 +582,2 @@ * In what order to return constructs

const reference_1 = require("./reference");
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"construct.js","sourceRoot":"","sources":["construct.ts"],"names":[],"mappings":"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","sourcesContent":["import cxapi = require('@aws-cdk/cx-api');\nimport { IAspect } from './aspect';\nimport { CLOUDFORMATION_TOKEN_RESOLVER, CloudFormationLang } from './cloudformation-lang';\nimport { IDependable } from './dependency';\nimport { resolve } from './resolve';\nimport { Token } from './token';\nimport { makeUniqueId } from './uniqueid';\n\nexport const PATH_SEP = '/';\n\n/**\n * Represents a construct.\n */\nexport interface IConstruct extends IDependable {\n  /**\n   * The construct node in the scope tree.\n   */\n  readonly node: ConstructNode;\n}\n\n/**\n * Represents the construct node in the scope tree.\n */\nexport class ConstructNode {\n  /**\n   * Returns the scope in which this construct is defined.\n   */\n  public readonly scope?: IConstruct;\n\n  /**\n   * The scoped construct ID\n   * This ID is unique amongst all constructs defined in the same scope.\n   * To obtain a global unique id for this construct, use `uniqueId`.\n   */\n  public readonly id: string;\n\n  /**\n   * An array of aspects applied to this node\n   */\n  public readonly aspects: IAspect[] = [];\n\n  /**\n   * List of children and their names\n   */\n  private readonly _children: { [name: string]: IConstruct } = { };\n  private readonly context: { [key: string]: any } = { };\n  private readonly _metadata = new Array<MetadataEntry>();\n  private readonly references = new Set<Reference>();\n  private readonly dependencies = new Set<IDependable>();\n\n  /** Will be used to cache the value of ``this.stack``. */\n  private _stack?: import('./stack').Stack;\n\n  /**\n   * If this is set to 'true'. addChild() calls for this construct and any child\n   * will fail. This is used to prevent tree mutations during synthesis.\n   */\n  private _locked = false;\n\n  private invokedAspects: IAspect[] = [];\n\n  constructor(private readonly host: Construct, scope: IConstruct, id: string) {\n    id = id || ''; // if undefined, convert to empty string\n\n    this.id = id;\n    this.scope = scope;\n\n    // We say that scope is required, but root scopes will bypass the type\n    // checks and actually pass in 'undefined'.\n    if (scope != null) {\n      if (id === '') {\n        throw new Error('Only root constructs may have an empty name');\n      }\n\n      // Has side effect so must be very last thing in constructor\n      scope.node.addChild(host, this.id);\n    } else {\n      // This is a root construct.\n      this.id = id;\n    }\n\n    // escape any path separators so they don't wreck havoc\n    this.id = this._escapePathSeparator(this.id);\n\n    if (Token.isToken(id)) {\n      throw new Error(`Cannot use tokens in construct ID: ${id}`);\n    }\n  }\n\n  /**\n   * The stack the construct is a part of.\n   */\n  public get stack(): import('./stack').Stack {\n    // Lazy import to break cyclic import\n    const stack: typeof import('./stack') = require('./stack');\n    return this._stack || (this._stack = _lookStackUp(this));\n\n    function _lookStackUp(_this: ConstructNode): import('./stack').Stack  {\n      if (stack.Stack.isStack(_this.host)) {\n        return _this.host;\n      }\n      if (!_this.scope) {\n        throw new Error(`No stack could be identified for the construct at path ${_this.path}`);\n      }\n      return _this.scope.node.stack;\n    }\n  }\n\n  /**\n   * The full, absolute path of this construct in the tree.\n   *\n   * Components are separated by '/'.\n   */\n  public get path(): string {\n    const components = this.ancestors().slice(1).map(c => c.node.id);\n    return components.join(PATH_SEP);\n  }\n\n  /**\n   * A tree-global unique alphanumeric identifier for this construct.\n   * Includes all components of the tree.\n   */\n  public get uniqueId(): string {\n    const components = this.ancestors().slice(1).map(c => c.node.id);\n    return components.length > 0 ? makeUniqueId(components) : '';\n  }\n\n  /**\n   * Returns a string with a tree representation of this construct and it's children.\n   */\n  public toTreeString(depth = 0) {\n    let out = '';\n    for (let i = 0; i < depth; ++i) {\n      out += '  ';\n    }\n    const name = this.id || '';\n    out += `${this.typename}${name.length > 0 ? ' [' + name + ']' : ''}\\n`;\n    for (const child of this.children) {\n      out += child.node.toTreeString(depth + 1);\n    }\n    return out;\n  }\n\n  /**\n   * Return a descendant by path, or undefined\n   *\n   * Note that if the original ID of the construct you are looking for contained\n   * a '/', then it would have been replaced by '--'.\n   *\n   * @param path Relative path of a direct or indirect child\n   * @returns a child by path or undefined if not found.\n   */\n  public tryFindChild(path: string): IConstruct | undefined {\n    if (path.startsWith(PATH_SEP)) {\n      throw new Error('Path must be relative');\n    }\n    const parts = path.split(PATH_SEP);\n\n    let curr: IConstruct|undefined = this.host;\n    while (curr != null && parts.length > 0) {\n      curr = curr.node._children[parts.shift()!];\n    }\n    return curr;\n  }\n\n  /**\n   * Return a descendant by path\n   *\n   * Throws an exception if the descendant is not found.\n   *\n   * Note that if the original ID of the construct you are looking for contained\n   * a '/', then it would have been replaced by '--'.\n   *\n   * @param path Relative path of a direct or indirect child\n   * @returns Child with the given path.\n   */\n  public findChild(path: string): IConstruct {\n    const ret = this.tryFindChild(path);\n    if (!ret) {\n      throw new Error(`No child with path: '${path}'`);\n    }\n    return ret;\n  }\n\n  /**\n   * All direct children of this construct.\n   */\n  public get children() {\n    return Object.values(this._children);\n  }\n\n  /**\n   * Return this construct and all of its children in the given order\n   */\n  public findAll(order: ConstructOrder = ConstructOrder.PreOrder): IConstruct[] {\n    const ret = new Array<IConstruct>();\n    visit(this.host);\n    return ret;\n\n    function visit(node: IConstruct) {\n      if (order === ConstructOrder.PreOrder) {\n        ret.push(node);\n      }\n\n      for (const child of node.node.children) {\n        visit(child);\n      }\n\n      if (order === ConstructOrder.PostOrder) {\n        ret.push(node);\n      }\n    }\n  }\n\n  /**\n   * This can be used to set contextual values.\n   * Context must be set before any children are added, since children may consult context info during construction.\n   * If the key already exists, it will be overridden.\n   * @param key The context key\n   * @param value The context value\n   */\n  public setContext(key: string, value: any) {\n    if (this.children.length > 0) {\n      const names = this.children.map(c => c.node.id);\n      throw new Error('Cannot set context after children have been added: ' + names.join(','));\n    }\n    this.context[key] = value;\n  }\n\n  /**\n   * Retrieves a value from tree context.\n   *\n   * Context is usually initialized at the root, but can be overridden at any point in the tree.\n   *\n   * @param key The context key\n   * @returns The context value or undefined\n   */\n  public getContext(key: string): any {\n    const value = this.context[key];\n    if (value !== undefined) { return value; }\n\n    return this.scope && this.scope.node.getContext(key);\n  }\n\n  /**\n   * Retrieve a value from tree-global context\n   *\n   * It is an error if the context object is not available.\n   */\n  public requireContext(key: string): any {\n    const value = this.getContext(key);\n\n    if (value == null) {\n      throw new Error(`You must supply a context value named '${key}'`);\n    }\n\n    return value;\n  }\n\n  /**\n   * An array of metadata objects associated with this construct.\n   * This can be used, for example, to implement support for deprecation notices, source mapping, etc.\n   */\n  public get metadata() {\n    return this._metadata;\n  }\n\n  /**\n   * Adds a metadata entry to this construct.\n   * Entries are arbitrary values and will also include a stack trace to allow tracing back to\n   * the code location for when the entry was added. It can be used, for example, to include source\n   * mapping in CloudFormation templates to improve diagnostics.\n   *\n   * @param type a string denoting the type of metadata\n   * @param data the value of the metadata (can be a Token). If null/undefined, metadata will not be added.\n   * @param from a function under which to restrict the metadata entry's stack trace (defaults to this.addMetadata)\n   */\n  public addMetadata(type: string, data: any, from?: any): void {\n    if (data == null) {\n      return;\n    }\n    const trace = createStackTrace(from || this.addMetadata);\n    this._metadata.push({ type, data, trace });\n  }\n\n  /**\n   * Adds a { \"aws:cdk:info\": <message> } metadata entry to this construct.\n   * The toolkit will display the info message when apps are synthesized.\n   * @param message The info message.\n   */\n  public addInfo(message: string): void {\n    this.addMetadata(cxapi.INFO_METADATA_KEY, message);\n  }\n\n  /**\n   * Adds a { warning: <message> } metadata entry to this construct.\n   * The toolkit will display the warning when an app is synthesized, or fail\n   * if run in --strict mode.\n   * @param message The warning message.\n   */\n  public addWarning(message: string): void {\n    this.addMetadata(cxapi.WARNING_METADATA_KEY, message);\n  }\n\n  /**\n   * Adds an { error: <message> } metadata entry to this construct.\n   * The toolkit will fail synthesis when errors are reported.\n   * @param message The error message.\n   */\n  public addError(message: string) {\n    this.addMetadata(cxapi.ERROR_METADATA_KEY, message);\n  }\n\n  /**\n   * Invokes 'validate' on all child constructs and then on this construct (depth-first).\n   * @returns A list of validation errors. If the list is empty, all constructs are valid.\n   */\n  public validateTree(): ValidationError[] {\n    let errors = new Array<ValidationError>();\n\n    for (const child of this.children) {\n      errors = errors.concat(child.node.validateTree());\n    }\n\n    const localErrors: string[] = (this.host as any).validate();\n    return errors.concat(localErrors.map(msg => new ValidationError(this.host, msg)));\n  }\n\n  /**\n   * Run 'prepare()' on all constructs in the tree\n   */\n  public prepareTree() {\n    const constructs = this.host.node.findAll(ConstructOrder.PreOrder);\n    // Aspects are applied root to leaf\n    for (const construct of constructs) {\n      construct.node.invokeAspects();\n    }\n    // Use .reverse() to achieve post-order traversal\n    for (const construct of constructs.reverse()) {\n      if (Construct.isConstruct(construct)) {\n        (construct as any).prepare();\n      }\n    }\n  }\n\n  /**\n   * Applies the aspect to this Constructs node\n   */\n  public apply(aspect: IAspect): void {\n    this.aspects.push(aspect);\n    return;\n  }\n  /**\n   * Return the ancestors (including self) of this Construct up until and excluding the indicated component\n   *\n   * @param upTo The construct to return the path components relative to, or\n   * the entire list of ancestors (including root) if omitted.\n   */\n  public ancestors(upTo?: Construct): IConstruct[] {\n    const ret = new Array<IConstruct>();\n\n    let curr: IConstruct | undefined = this.host;\n    while (curr && curr !== upTo) {\n      ret.unshift(curr);\n      curr = curr.node && curr.node.scope;\n    }\n\n    return ret;\n  }\n\n  /**\n   * Throws if the `props` bag doesn't include the property `name`.\n   * In the future we can add some type-checking here, maybe even auto-generate during compilation.\n   * @param props The props bag.\n   * @param name The name of the required property.\n   *\n   * @deprecated use ``requireProperty`` from ``@aws-cdk/runtime`` instead.\n   */\n  public required(props: any, name: string): any {\n    if (!(name in props)) {\n      throw new Error(`Construct of type ${this.typename} is missing required property: ${name}`);\n    }\n\n    const value = props[name];\n    return value;\n  }\n\n  /**\n   * @returns The type name of this node.\n   */\n  public get typename(): string {\n    const ctor: any = this.host.constructor;\n    return ctor.name || 'Construct';\n  }\n\n  /**\n   * Adds a child construct to this node.\n   *\n   * @param child The child construct\n   * @param childName The type name of the child construct.\n   * @returns The resolved path part name of the child\n   */\n  public addChild(child: IConstruct, childName: string) {\n    if (this.locked) {\n\n      // special error if root is locked\n      if (!this.path) {\n        throw new Error('Cannot add children during synthesis');\n      }\n\n      throw new Error(`Cannot add children to \"${this.path}\" during synthesis`);\n    }\n\n    if (childName in this._children) {\n      const name = this.id || '';\n      throw new Error(`There is already a Construct with name '${childName}' in ${this.typename}${name.length > 0 ? ' [' + name + ']' : ''}`);\n    }\n\n    this._children[childName] = child;\n  }\n\n  /**\n   * Locks this construct from allowing more children to be added. After this\n   * call, no more children can be added to this construct or to any children.\n   */\n  public lock() {\n    this._locked = true;\n  }\n\n  /**\n   * Unlocks this costruct and allows mutations (adding children).\n   */\n  public unlock() {\n    this._locked = false;\n  }\n\n  /**\n   * Returns true if this construct or the scopes in which it is defined are\n   * locked.\n   */\n  public get locked() {\n    if (this._locked) {\n      return true;\n    }\n\n    if (this.scope && this.scope.node.locked) {\n      return true;\n    }\n\n    return false;\n  }\n\n  /**\n   * Resolve a tokenized value in the context of the current Construct\n   */\n  public resolve(obj: any): any {\n    return resolve(obj, {\n      scope: this.host,\n      prefix: [],\n      resolver: CLOUDFORMATION_TOKEN_RESOLVER,\n    });\n  }\n\n  /**\n   * Convert an object, potentially containing tokens, to a JSON string\n   */\n  public stringifyJson(obj: any): string {\n    return CloudFormationLang.toJSON(obj).toString();\n  }\n\n  /**\n   * Record a reference originating from this construct node\n   */\n  public recordReference(...refs: Token[]) {\n    for (const ref of refs) {\n      if (Reference.isReference(ref)) {\n        this.references.add(ref);\n      }\n    }\n  }\n\n  /**\n   * Return all references of the given type originating from this node or any of its children\n   */\n  public findReferences(): OutgoingReference[] {\n    const ret = new Set<OutgoingReference>();\n\n    function recurse(node: ConstructNode) {\n      for (const reference of node.references) {\n        ret.add({ source: node.host, reference });\n      }\n\n      for (const child of node.children) {\n        recurse(child.node);\n      }\n    }\n\n    recurse(this);\n\n    return Array.from(ret);\n  }\n\n  /**\n   * Add an ordering dependency on another Construct.\n   *\n   * All constructs in the dependency's scope will be deployed before any\n   * construct in this construct's scope.\n   */\n  public addDependency(...dependencies: IDependable[]) {\n    for (const dependency of dependencies) {\n      this.dependencies.add(dependency);\n    }\n  }\n\n  /**\n   * Return all dependencies registered on this node or any of its children\n   */\n  public findDependencies(): Dependency[] {\n    const found = new Map<IConstruct, Set<IConstruct>>(); // Deduplication map\n    const ret = new Array<Dependency>();\n\n    for (const source of this.findAll()) {\n      for (const dependable of source.node.dependencies) {\n        for (const target of dependable.dependencyRoots) {\n          let foundTargets = found.get(source);\n          if (!foundTargets) { found.set(source, foundTargets = new Set()); }\n\n          if (!foundTargets.has(target)) {\n            ret.push({ source, target });\n            foundTargets.add(target);\n          }\n        }\n      }\n    }\n\n    return ret;\n  }\n\n  /**\n   * Triggers each aspect to invoke visit\n   */\n  private invokeAspects(): void {\n    const descendants = this.findAll();\n    for (const aspect of this.aspects) {\n      if (this.invokedAspects.includes(aspect)) {\n        continue;\n      }\n      descendants.forEach( member => aspect.visit(member));\n      this.invokedAspects.push(aspect);\n    }\n  }\n\n  /**\n   * If the construct ID contains a path separator, it is replaced by double dash (`--`).\n   */\n  private _escapePathSeparator(id: string) {\n    if (!id) { return id; }\n    return id.split(PATH_SEP).join('--');\n  }\n}\n\n/**\n * Represents the building block of the construct graph.\n *\n * All constructs besides the root construct must be created within the scope of\n * another construct.\n */\nexport class Construct implements IConstruct {\n  /**\n   * Return whether the given object is a Construct\n   */\n  public static isConstruct(x: IConstruct): x is Construct {\n    return (x as any).prepare !== undefined && (x as any).validate !== undefined;\n  }\n\n  /**\n   * Construct node.\n   */\n  public readonly node: ConstructNode;\n\n  /**\n   * The set of constructs that form the root of this dependable\n   *\n   * All resources under all returned constructs are included in the ordering\n   * dependency.\n   */\n  public readonly dependencyRoots: IConstruct[] = [this];\n\n  /**\n   * Creates a new construct node.\n   *\n   * @param scope The scope in which to define this construct\n   * @param id The scoped construct ID. Must be unique amongst siblings. If\n   * the ID includes a path separator (`/`), then it will be replaced by double\n   * dash `--`.\n   */\n  constructor(scope: Construct, id: string) {\n    this.node = new ConstructNode(this, scope, id);\n  }\n\n  /**\n   * Returns a string representation of this construct.\n   */\n  public toString() {\n    const path = this.node.path;\n    return this.node.typename + (path.length > 0 ? ` [${path}]` : '');\n  }\n\n  /**\n   * Validate the current construct.\n   *\n   * This method can be implemented by derived constructs in order to perform\n   * validation logic. It is called on all constructs before synthesis.\n   *\n   * @returns An array of validation error messages, or an empty array if there the construct is valid.\n   */\n  protected validate(): string[] {\n    return [];\n  }\n\n  /**\n   * Perform final modifications before synthesis\n   *\n   * This method can be implemented by derived constructs in order to perform\n   * final changes before synthesis. prepare() will be called after child\n   * constructs have been prepared.\n   *\n   * This is an advanced framework feature. Only use this if you\n   * understand the implications.\n   */\n  protected prepare(): void {\n    return;\n  }\n}\n\n/**\n * An metadata entry in the construct.\n */\nexport interface MetadataEntry {\n  /**\n   * The type of the metadata entry.\n   */\n  readonly type: string;\n\n  /**\n   * The data.\n   */\n  readonly data?: any;\n\n  /**\n   * A stack trace for when the entry was created.\n   */\n  readonly trace: string[];\n}\n\nexport class ValidationError {\n  constructor(public readonly source: IConstruct, public readonly message: string) {\n\n  }\n}\n\n// tslint:disable-next-line:ban-types\nfunction createStackTrace(below: Function): string[] {\n  const object = { stack: '' };\n  const previousLimit = Error.stackTraceLimit;\n  try {\n    Error.stackTraceLimit = Number.MAX_SAFE_INTEGER;\n    Error.captureStackTrace(object, below);\n  } finally {\n    Error.stackTraceLimit = previousLimit;\n  }\n  if (!object.stack) {\n    return [];\n  }\n  return object.stack.split('\\n').slice(1).map(s => s.replace(/^\\s*at\\s+/, ''));\n}\n\n/**\n * In what order to return constructs\n */\nexport enum ConstructOrder {\n  /**\n   * Depth-first, pre-order\n   */\n  PreOrder,\n\n  /**\n   * Depth-first, post-order (leaf nodes first)\n   */\n  PostOrder\n}\n\n/**\n * A single dependency\n */\nexport interface Dependency {\n  /**\n   * Source the dependency\n   */\n  readonly source: IConstruct;\n\n  /**\n   * Target of the dependency\n   */\n  readonly target: IConstruct;\n}\n\n/**\n * A single dependency\n */\nexport interface Dependency {\n  /**\n   * Source the dependency\n   */\n  readonly source: IConstruct;\n\n  /**\n   * Target of the dependency\n   */\n  readonly target: IConstruct;\n}\n\nexport interface OutgoingReference {\n  readonly source: IConstruct;\n  readonly reference: Reference;\n}\n\n// Import this _after_ everything else to help node work the classes out in the correct order...\nimport { Reference } from './reference';"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"construct.js","sourceRoot":"","sources":["construct.ts"],"names":[],"mappings":"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","sourcesContent":["import cxapi = require('@aws-cdk/cx-api');\nimport { IAspect } from './aspect';\nimport { CLOUDFORMATION_TOKEN_RESOLVER, CloudFormationLang } from './cloudformation-lang';\nimport { DependableTrait, IDependable } from './dependency';\nimport { resolve } from './resolve';\nimport { createStackTrace } from './stack-trace';\nimport { Token } from './token';\nimport { makeUniqueId } from './uniqueid';\n\nexport const PATH_SEP = '/';\n\n/**\n * Represents a construct.\n */\nexport interface IConstruct extends IDependable {\n  /**\n   * The construct node in the scope tree.\n   */\n  readonly node: ConstructNode;\n}\n\n/**\n * Represents the construct node in the scope tree.\n */\nexport class ConstructNode {\n  /**\n   * Returns the scope in which this construct is defined.\n   */\n  public readonly scope?: IConstruct;\n\n  /**\n   * The scoped construct ID\n   * This ID is unique amongst all constructs defined in the same scope.\n   * To obtain a global unique id for this construct, use `uniqueId`.\n   */\n  public readonly id: string;\n\n  /**\n   * An array of aspects applied to this node\n   */\n  public readonly aspects: IAspect[] = [];\n\n  /**\n   * List of children and their names\n   */\n  private readonly _children: { [name: string]: IConstruct } = { };\n  private readonly context: { [key: string]: any } = { };\n  private readonly _metadata = new Array<cxapi.MetadataEntry>();\n  private readonly references = new Set<Reference>();\n  private readonly dependencies = new Set<IDependable>();\n\n  /** Will be used to cache the value of ``this.stack``. */\n  private _stack?: import('./stack').Stack;\n\n  /**\n   * If this is set to 'true'. addChild() calls for this construct and any child\n   * will fail. This is used to prevent tree mutations during synthesis.\n   */\n  private _locked = false;\n\n  private invokedAspects: IAspect[] = [];\n\n  constructor(private readonly host: Construct, scope: IConstruct, id: string) {\n    id = id || ''; // if undefined, convert to empty string\n\n    this.id = id;\n    this.scope = scope;\n\n    // We say that scope is required, but root scopes will bypass the type\n    // checks and actually pass in 'undefined'.\n    if (scope != null) {\n      if (id === '') {\n        throw new Error('Only root constructs may have an empty name');\n      }\n\n      // Has side effect so must be very last thing in constructor\n      scope.node.addChild(host, this.id);\n    } else {\n      // This is a root construct.\n      this.id = id;\n    }\n\n    // escape any path separators so they don't wreck havoc\n    this.id = this._escapePathSeparator(this.id);\n\n    if (Token.isToken(id)) {\n      throw new Error(`Cannot use tokens in construct ID: ${id}`);\n    }\n  }\n\n  /**\n   * The stack the construct is a part of.\n   */\n  public get stack(): import('./stack').Stack {\n    // Lazy import to break cyclic import\n    const stack: typeof import('./stack') = require('./stack');\n    return this._stack || (this._stack = _lookStackUp(this));\n\n    function _lookStackUp(_this: ConstructNode): import('./stack').Stack  {\n      if (stack.Stack.isStack(_this.host)) {\n        return _this.host;\n      }\n      if (!_this.scope) {\n        throw new Error(`No stack could be identified for the construct at path ${_this.path}`);\n      }\n      return _this.scope.node.stack;\n    }\n  }\n\n  /**\n   * The full, absolute path of this construct in the tree.\n   *\n   * Components are separated by '/'.\n   */\n  public get path(): string {\n    const components = this.ancestors().slice(1).map(c => c.node.id);\n    return components.join(PATH_SEP);\n  }\n\n  /**\n   * A tree-global unique alphanumeric identifier for this construct.\n   * Includes all components of the tree.\n   */\n  public get uniqueId(): string {\n    const components = this.ancestors().slice(1).map(c => c.node.id);\n    return components.length > 0 ? makeUniqueId(components) : '';\n  }\n\n  /**\n   * Returns a string with a tree representation of this construct and it's children.\n   */\n  public toTreeString(depth = 0) {\n    let out = '';\n    for (let i = 0; i < depth; ++i) {\n      out += '  ';\n    }\n    const name = this.id || '';\n    out += `${this.typename}${name.length > 0 ? ' [' + name + ']' : ''}\\n`;\n    for (const child of this.children) {\n      out += child.node.toTreeString(depth + 1);\n    }\n    return out;\n  }\n\n  /**\n   * Return a descendant by path, or undefined\n   *\n   * Note that if the original ID of the construct you are looking for contained\n   * a '/', then it would have been replaced by '--'.\n   *\n   * @param path Relative path of a direct or indirect child\n   * @returns a child by path or undefined if not found.\n   */\n  public tryFindChild(path: string): IConstruct | undefined {\n    if (path.startsWith(PATH_SEP)) {\n      throw new Error('Path must be relative');\n    }\n    const parts = path.split(PATH_SEP);\n\n    let curr: IConstruct|undefined = this.host;\n    while (curr != null && parts.length > 0) {\n      curr = curr.node._children[parts.shift()!];\n    }\n    return curr;\n  }\n\n  /**\n   * Return a descendant by path\n   *\n   * Throws an exception if the descendant is not found.\n   *\n   * Note that if the original ID of the construct you are looking for contained\n   * a '/', then it would have been replaced by '--'.\n   *\n   * @param path Relative path of a direct or indirect child\n   * @returns Child with the given path.\n   */\n  public findChild(path: string): IConstruct {\n    const ret = this.tryFindChild(path);\n    if (!ret) {\n      throw new Error(`No child with path: '${path}'`);\n    }\n    return ret;\n  }\n\n  /**\n   * All direct children of this construct.\n   */\n  public get children() {\n    return Object.values(this._children);\n  }\n\n  /**\n   * Return this construct and all of its children in the given order\n   */\n  public findAll(order: ConstructOrder = ConstructOrder.PreOrder): IConstruct[] {\n    const ret = new Array<IConstruct>();\n    visit(this.host);\n    return ret;\n\n    function visit(node: IConstruct) {\n      if (order === ConstructOrder.PreOrder) {\n        ret.push(node);\n      }\n\n      for (const child of node.node.children) {\n        visit(child);\n      }\n\n      if (order === ConstructOrder.PostOrder) {\n        ret.push(node);\n      }\n    }\n  }\n\n  /**\n   * This can be used to set contextual values.\n   * Context must be set before any children are added, since children may consult context info during construction.\n   * If the key already exists, it will be overridden.\n   * @param key The context key\n   * @param value The context value\n   */\n  public setContext(key: string, value: any) {\n    if (this.children.length > 0) {\n      const names = this.children.map(c => c.node.id);\n      throw new Error('Cannot set context after children have been added: ' + names.join(','));\n    }\n    this.context[key] = value;\n  }\n\n  /**\n   * Retrieves a value from tree context.\n   *\n   * Context is usually initialized at the root, but can be overridden at any point in the tree.\n   *\n   * @param key The context key\n   * @returns The context value or undefined\n   */\n  public getContext(key: string): any {\n    const value = this.context[key];\n    if (value !== undefined) { return value; }\n\n    return this.scope && this.scope.node.getContext(key);\n  }\n\n  /**\n   * Retrieve a value from tree-global context\n   *\n   * It is an error if the context object is not available.\n   */\n  public requireContext(key: string): any {\n    const value = this.getContext(key);\n\n    if (value == null) {\n      throw new Error(`You must supply a context value named '${key}'`);\n    }\n\n    return value;\n  }\n\n  /**\n   * An array of metadata objects associated with this construct.\n   * This can be used, for example, to implement support for deprecation notices, source mapping, etc.\n   */\n  public get metadata() {\n    return this._metadata;\n  }\n\n  /**\n   * Adds a metadata entry to this construct.\n   * Entries are arbitrary values and will also include a stack trace to allow tracing back to\n   * the code location for when the entry was added. It can be used, for example, to include source\n   * mapping in CloudFormation templates to improve diagnostics.\n   *\n   * @param type a string denoting the type of metadata\n   * @param data the value of the metadata (can be a Token). If null/undefined, metadata will not be added.\n   * @param from a function under which to restrict the metadata entry's stack trace (defaults to this.addMetadata)\n   */\n  public addMetadata(type: string, data: any, from?: any): void {\n    if (data == null) {\n      return;\n    }\n\n    const trace = this.getContext(cxapi.DISABLE_METADATA_STACK_TRACE) ? undefined : createStackTrace(from || this.addMetadata);\n    this._metadata.push({ type, data, trace });\n  }\n\n  /**\n   * Adds a { \"aws:cdk:info\": <message> } metadata entry to this construct.\n   * The toolkit will display the info message when apps are synthesized.\n   * @param message The info message.\n   */\n  public addInfo(message: string): void {\n    this.addMetadata(cxapi.INFO_METADATA_KEY, message);\n  }\n\n  /**\n   * Adds a { warning: <message> } metadata entry to this construct.\n   * The toolkit will display the warning when an app is synthesized, or fail\n   * if run in --strict mode.\n   * @param message The warning message.\n   */\n  public addWarning(message: string): void {\n    this.addMetadata(cxapi.WARNING_METADATA_KEY, message);\n  }\n\n  /**\n   * Adds an { error: <message> } metadata entry to this construct.\n   * The toolkit will fail synthesis when errors are reported.\n   * @param message The error message.\n   */\n  public addError(message: string) {\n    this.addMetadata(cxapi.ERROR_METADATA_KEY, message);\n  }\n\n  /**\n   * Invokes 'validate' on all child constructs and then on this construct (depth-first).\n   * @returns A list of validation errors. If the list is empty, all constructs are valid.\n   */\n  public validateTree(): ValidationError[] {\n    let errors = new Array<ValidationError>();\n\n    for (const child of this.children) {\n      errors = errors.concat(child.node.validateTree());\n    }\n\n    const localErrors: string[] = (this.host as any).validate();\n    return errors.concat(localErrors.map(msg => new ValidationError(this.host, msg)));\n  }\n\n  /**\n   * Run 'prepare()' on all constructs in the tree\n   */\n  public prepareTree() {\n    const constructs = this.host.node.findAll(ConstructOrder.PreOrder);\n    // Aspects are applied root to leaf\n    for (const construct of constructs) {\n      construct.node.invokeAspects();\n    }\n    // Use .reverse() to achieve post-order traversal\n    for (const construct of constructs.reverse()) {\n      if (Construct.isConstruct(construct)) {\n        (construct as any).prepare();\n      }\n    }\n  }\n\n  /**\n   * Applies the aspect to this Constructs node\n   */\n  public apply(aspect: IAspect): void {\n    this.aspects.push(aspect);\n    return;\n  }\n\n  /**\n   * Return the ancestors (including self) of this Construct up until and\n   * excluding the indicated component\n   *\n   * @param upTo The construct to return the path components relative to, or the\n   * entire list of ancestors (including root) if omitted. This construct will\n   * not be included in the returned list.\n   *\n   * @returns a list of parent scopes. The last element in the list will always\n   * be `this` and the first element is the oldest scope (if `upTo` is not set,\n   * it will be the root of the construct tree).\n   */\n  public ancestors(upTo?: Construct): IConstruct[] {\n    const ret = new Array<IConstruct>();\n\n    let curr: IConstruct | undefined = this.host;\n    while (curr && curr !== upTo) {\n      ret.unshift(curr);\n      curr = curr.node && curr.node.scope;\n    }\n\n    return ret;\n  }\n\n  /**\n   * @returns The root of the construct tree.\n   */\n  public get root() {\n    return this.ancestors()[0];\n  }\n\n  /**\n   * Throws if the `props` bag doesn't include the property `name`.\n   * In the future we can add some type-checking here, maybe even auto-generate during compilation.\n   * @param props The props bag.\n   * @param name The name of the required property.\n   *\n   * @deprecated use ``requireProperty`` from ``@aws-cdk/runtime`` instead.\n   */\n  public required(props: any, name: string): any {\n    if (!(name in props)) {\n      throw new Error(`Construct of type ${this.typename} is missing required property: ${name}`);\n    }\n\n    const value = props[name];\n    return value;\n  }\n\n  /**\n   * @returns The type name of this node.\n   */\n  public get typename(): string {\n    const ctor: any = this.host.constructor;\n    return ctor.name || 'Construct';\n  }\n\n  /**\n   * Adds a child construct to this node.\n   *\n   * @param child The child construct\n   * @param childName The type name of the child construct.\n   * @returns The resolved path part name of the child\n   */\n  public addChild(child: IConstruct, childName: string) {\n    if (this.locked) {\n\n      // special error if root is locked\n      if (!this.path) {\n        throw new Error('Cannot add children during synthesis');\n      }\n\n      throw new Error(`Cannot add children to \"${this.path}\" during synthesis`);\n    }\n\n    if (childName in this._children) {\n      const name = this.id || '';\n      throw new Error(`There is already a Construct with name '${childName}' in ${this.typename}${name.length > 0 ? ' [' + name + ']' : ''}`);\n    }\n\n    this._children[childName] = child;\n  }\n\n  /**\n   * Locks this construct from allowing more children to be added. After this\n   * call, no more children can be added to this construct or to any children.\n   */\n  public lock() {\n    this._locked = true;\n  }\n\n  /**\n   * Unlocks this costruct and allows mutations (adding children).\n   */\n  public unlock() {\n    this._locked = false;\n  }\n\n  /**\n   * Returns true if this construct or the scopes in which it is defined are\n   * locked.\n   */\n  public get locked() {\n    if (this._locked) {\n      return true;\n    }\n\n    if (this.scope && this.scope.node.locked) {\n      return true;\n    }\n\n    return false;\n  }\n\n  /**\n   * Resolve a tokenized value in the context of the current Construct\n   */\n  public resolve(obj: any): any {\n    return resolve(obj, {\n      scope: this.host,\n      prefix: [],\n      resolver: CLOUDFORMATION_TOKEN_RESOLVER,\n    });\n  }\n\n  /**\n   * Convert an object, potentially containing tokens, to a JSON string\n   */\n  public stringifyJson(obj: any): string {\n    return CloudFormationLang.toJSON(obj).toString();\n  }\n\n  /**\n   * Record a reference originating from this construct node\n   */\n  public recordReference(...refs: Token[]) {\n    for (const ref of refs) {\n      if (Reference.isReference(ref)) {\n        this.references.add(ref);\n      }\n    }\n  }\n\n  /**\n   * Return all references of the given type originating from this node or any of its children\n   */\n  public findReferences(): OutgoingReference[] {\n    const ret = new Set<OutgoingReference>();\n\n    function recurse(node: ConstructNode) {\n      for (const reference of node.references) {\n        ret.add({ source: node.host, reference });\n      }\n\n      for (const child of node.children) {\n        recurse(child.node);\n      }\n    }\n\n    recurse(this);\n\n    return Array.from(ret);\n  }\n\n  /**\n   * Add an ordering dependency on another Construct.\n   *\n   * All constructs in the dependency's scope will be deployed before any\n   * construct in this construct's scope.\n   */\n  public addDependency(...dependencies: IDependable[]) {\n    for (const dependency of dependencies) {\n      this.dependencies.add(dependency);\n    }\n  }\n\n  /**\n   * Return all dependencies registered on this node or any of its children\n   */\n  public findDependencies(): Dependency[] {\n    const found = new Map<IConstruct, Set<IConstruct>>(); // Deduplication map\n    const ret = new Array<Dependency>();\n\n    for (const source of this.findAll()) {\n      for (const dependable of source.node.dependencies) {\n        for (const target of DependableTrait.get(dependable).dependencyRoots) {\n          let foundTargets = found.get(source);\n          if (!foundTargets) { found.set(source, foundTargets = new Set()); }\n\n          if (!foundTargets.has(target)) {\n            ret.push({ source, target });\n            foundTargets.add(target);\n          }\n        }\n      }\n    }\n\n    return ret;\n  }\n\n  /**\n   * Triggers each aspect to invoke visit\n   */\n  private invokeAspects(): void {\n    const descendants = this.findAll();\n    for (const aspect of this.aspects) {\n      if (this.invokedAspects.includes(aspect)) {\n        continue;\n      }\n      descendants.forEach( member => aspect.visit(member));\n      this.invokedAspects.push(aspect);\n    }\n  }\n\n  /**\n   * If the construct ID contains a path separator, it is replaced by double dash (`--`).\n   */\n  private _escapePathSeparator(id: string) {\n    if (!id) { return id; }\n    return id.split(PATH_SEP).join('--');\n  }\n}\n\n/**\n * Represents the building block of the construct graph.\n *\n * All constructs besides the root construct must be created within the scope of\n * another construct.\n */\nexport class Construct implements IConstruct {\n  /**\n   * Return whether the given object is a Construct\n   */\n  public static isConstruct(x: IConstruct): x is Construct {\n    return (x as any).prepare !== undefined && (x as any).validate !== undefined;\n  }\n\n  /**\n   * Construct node.\n   */\n  public readonly node: ConstructNode;\n\n  /**\n   * Creates a new construct node.\n   *\n   * @param scope The scope in which to define this construct\n   * @param id The scoped construct ID. Must be unique amongst siblings. If\n   * the ID includes a path separator (`/`), then it will be replaced by double\n   * dash `--`.\n   */\n  constructor(scope: Construct, id: string) {\n    this.node = new ConstructNode(this, scope, id);\n\n    // Implement IDependable privately\n    const self = this;\n    DependableTrait.implement(this, {\n      get dependencyRoots() { return [self]; },\n    });\n  }\n\n  /**\n   * Returns a string representation of this construct.\n   */\n  public toString() {\n    const path = this.node.path;\n    return this.node.typename + (path.length > 0 ? ` [${path}]` : '');\n  }\n\n  /**\n   * Validate the current construct.\n   *\n   * This method can be implemented by derived constructs in order to perform\n   * validation logic. It is called on all constructs before synthesis.\n   *\n   * @returns An array of validation error messages, or an empty array if there the construct is valid.\n   */\n  protected validate(): string[] {\n    return [];\n  }\n\n  /**\n   * Perform final modifications before synthesis\n   *\n   * This method can be implemented by derived constructs in order to perform\n   * final changes before synthesis. prepare() will be called after child\n   * constructs have been prepared.\n   *\n   * This is an advanced framework feature. Only use this if you\n   * understand the implications.\n   */\n  protected prepare(): void {\n    return;\n  }\n}\n\nexport class ValidationError {\n  constructor(public readonly source: IConstruct, public readonly message: string) {\n\n  }\n}\n\n/**\n * In what order to return constructs\n */\nexport enum ConstructOrder {\n  /**\n   * Depth-first, pre-order\n   */\n  PreOrder,\n\n  /**\n   * Depth-first, post-order (leaf nodes first)\n   */\n  PostOrder\n}\n\n/**\n * A single dependency\n */\nexport interface Dependency {\n  /**\n   * Source the dependency\n   */\n  readonly source: IConstruct;\n\n  /**\n   * Target of the dependency\n   */\n  readonly target: IConstruct;\n}\n\n/**\n * A single dependency\n */\nexport interface Dependency {\n  /**\n   * Source the dependency\n   */\n  readonly source: IConstruct;\n\n  /**\n   * Target of the dependency\n   */\n  readonly target: IConstruct;\n}\n\nexport interface OutgoingReference {\n  readonly source: IConstruct;\n  readonly reference: Reference;\n}\n\n// Import this _after_ everything else to help node work the classes out in the correct order...\nimport { Reference } from './reference';\n"]}
import { IConstruct } from "./construct";
/**
* A set of constructs that can be depended upon
* Trait marker for classes that can be depended upon
*
* The presence of this interface indicates that an object has
* an `IDependableTrait` implementation.
*
* This interface can be used to take an (ordering) dependency on a set of

@@ -11,9 +14,2 @@ * constructs. An ordering dependency implies that the resources represented by

export interface IDependable {
/**
* The set of constructs that form the root of this dependable
*
* All resources under all returned constructs are included in the ordering
* dependency.
*/
readonly dependencyRoots: IConstruct[];
}

@@ -28,2 +24,3 @@ /**

private readonly _dependencyRoots;
constructor();
/**

@@ -33,6 +30,41 @@ * Add a construct to the dependency roots

add(construct: IConstruct): void;
}
/**
* Trait for IDependable
*
* Traits are interfaces that are privately implemented by objects. Instead of
* showing up in the public interface of a class, they need to be queried
* explicitly. This is used to implement certain framework features that are
* not intended to be used by Construct consumers, and so should be hidden
* from accidental use.
*
* @example
*
* // Usage
* const roots = DependableTrait.get(construct).dependencyRoots;
*
* // Definition
* DependableTrait.implement(construct, {
* get dependencyRoots() { return []; }
* });
*/
export declare abstract class DependableTrait {
/**
* Retrieve the current set of dependency roots
* Register `instance` to have the given DependableTrait
*
* Should be called in the class constructor.
*/
readonly dependencyRoots: IConstruct[];
static implement(instance: IDependable, trait: DependableTrait): void;
/**
* Return the matching DependableTrait for the given class instance.
*/
static get(instance: IDependable): DependableTrait;
private static traitMap;
/**
* The set of constructs that form the root of this dependable
*
* All resources under all returned constructs are included in the ordering
* dependency.
*/
abstract readonly dependencyRoots: IConstruct[];
}

@@ -12,2 +12,6 @@ "use strict";

this._dependencyRoots = new Array();
const self = this;
DependableTrait.implement(this, {
get dependencyRoots() { return self._dependencyRoots; },
});
}

@@ -20,10 +24,48 @@ /**

}
}
exports.ConcreteDependable = ConcreteDependable;
/**
* Trait for IDependable
*
* Traits are interfaces that are privately implemented by objects. Instead of
* showing up in the public interface of a class, they need to be queried
* explicitly. This is used to implement certain framework features that are
* not intended to be used by Construct consumers, and so should be hidden
* from accidental use.
*
* @example
*
* // Usage
* const roots = DependableTrait.get(construct).dependencyRoots;
*
* // Definition
* DependableTrait.implement(construct, {
* get dependencyRoots() { return []; }
* });
*/
class DependableTrait {
/**
* Retrieve the current set of dependency roots
* Register `instance` to have the given DependableTrait
*
* Should be called in the class constructor.
*/
get dependencyRoots() {
return this._dependencyRoots;
static implement(instance, trait) {
// I would also like to reference classes (to cut down on the list of objects
// we need to manage), but we can't do that either since jsii doesn't have the
// concept of a class reference.
DependableTrait.traitMap.set(instance, trait);
}
/**
* Return the matching DependableTrait for the given class instance.
*/
static get(instance) {
const ret = DependableTrait.traitMap.get(instance);
if (!ret) {
throw new Error(`${instance} does not implement DependableTrait`);
}
return ret;
}
}
exports.ConcreteDependable = ConcreteDependable;
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGVwZW5kZW5jeS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbImRlcGVuZGVuY3kudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFvQkE7Ozs7O0dBS0c7QUFDSCxNQUFhLGtCQUFrQjtJQUEvQjtRQUNtQixxQkFBZ0IsR0FBRyxJQUFJLEtBQUssRUFBYyxDQUFDO0lBZTlELENBQUM7SUFiQzs7T0FFRztJQUNJLEdBQUcsQ0FBQyxTQUFxQjtRQUM5QixJQUFJLENBQUMsZ0JBQWdCLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxDQUFDO0lBQ3hDLENBQUM7SUFFRDs7T0FFRztJQUNILElBQVcsZUFBZTtRQUN4QixPQUFPLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQztJQUMvQixDQUFDO0NBQ0Y7QUFoQkQsZ0RBZ0JDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSUNvbnN0cnVjdCB9IGZyb20gXCIuL2NvbnN0cnVjdFwiO1xuXG4vKipcbiAqIEEgc2V0IG9mIGNvbnN0cnVjdHMgdGhhdCBjYW4gYmUgZGVwZW5kZWQgdXBvblxuICpcbiAqIFRoaXMgaW50ZXJmYWNlIGNhbiBiZSB1c2VkIHRvIHRha2UgYW4gKG9yZGVyaW5nKSBkZXBlbmRlbmN5IG9uIGEgc2V0IG9mXG4gKiBjb25zdHJ1Y3RzLiBBbiBvcmRlcmluZyBkZXBlbmRlbmN5IGltcGxpZXMgdGhhdCB0aGUgcmVzb3VyY2VzIHJlcHJlc2VudGVkIGJ5XG4gKiB0aG9zZSBjb25zdHJ1Y3RzIGFyZSBkZXBsb3llZCBiZWZvcmUgdGhlIHJlc291cmNlcyBkZXBlbmRpbmcgT04gdGhlbSBhcmVcbiAqIGRlcGxveWVkLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIElEZXBlbmRhYmxlIHtcbiAgLyoqXG4gICAqIFRoZSBzZXQgb2YgY29uc3RydWN0cyB0aGF0IGZvcm0gdGhlIHJvb3Qgb2YgdGhpcyBkZXBlbmRhYmxlXG4gICAqXG4gICAqIEFsbCByZXNvdXJjZXMgdW5kZXIgYWxsIHJldHVybmVkIGNvbnN0cnVjdHMgYXJlIGluY2x1ZGVkIGluIHRoZSBvcmRlcmluZ1xuICAgKiBkZXBlbmRlbmN5LlxuICAgKi9cbiAgcmVhZG9ubHkgZGVwZW5kZW5jeVJvb3RzOiBJQ29uc3RydWN0W107XG59XG5cbi8qKlxuICogQSBzZXQgb2YgY29uc3RydWN0cyB0byBiZSB1c2VkIGFzIGEgZGVwZW5kYWJsZVxuICpcbiAqIFRoaXMgY2xhc3MgY2FuIGJlIHVzZWQgd2hlbiBhIHNldCBvZiBjb25zdHJ1Y3RzIHdoaWNoIGFyZSBkaXNqb2ludCBpbiB0aGVcbiAqIGNvbnN0cnVjdCB0cmVlIG5lZWRzIHRvIGJlIGNvbWJpbmVkIHRvIGJlIHVzZWQgYXMgYSBzaW5nbGUgZGVwZW5kYWJsZS5cbiAqL1xuZXhwb3J0IGNsYXNzIENvbmNyZXRlRGVwZW5kYWJsZSBpbXBsZW1lbnRzIElEZXBlbmRhYmxlIHtcbiAgcHJpdmF0ZSByZWFkb25seSBfZGVwZW5kZW5jeVJvb3RzID0gbmV3IEFycmF5PElDb25zdHJ1Y3Q+KCk7XG5cbiAgLyoqXG4gICAqIEFkZCBhIGNvbnN0cnVjdCB0byB0aGUgZGVwZW5kZW5jeSByb290c1xuICAgKi9cbiAgcHVibGljIGFkZChjb25zdHJ1Y3Q6IElDb25zdHJ1Y3QpIHtcbiAgICB0aGlzLl9kZXBlbmRlbmN5Um9vdHMucHVzaChjb25zdHJ1Y3QpO1xuICB9XG5cbiAgLyoqXG4gICAqIFJldHJpZXZlIHRoZSBjdXJyZW50IHNldCBvZiBkZXBlbmRlbmN5IHJvb3RzXG4gICAqL1xuICBwdWJsaWMgZ2V0IGRlcGVuZGVuY3lSb290cygpOiBJQ29uc3RydWN0W10ge1xuICAgIHJldHVybiB0aGlzLl9kZXBlbmRlbmN5Um9vdHM7XG4gIH1cbn0iXX0=
DependableTrait.traitMap = new WeakMap();
exports.DependableTrait = DependableTrait;
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGVwZW5kZW5jeS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbImRlcGVuZGVuY3kudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFpQkE7Ozs7O0dBS0c7QUFDSCxNQUFhLGtCQUFrQjtJQUc3QjtRQUZpQixxQkFBZ0IsR0FBRyxJQUFJLEtBQUssRUFBYyxDQUFDO1FBRzFELE1BQU0sSUFBSSxHQUFHLElBQUksQ0FBQztRQUNsQixlQUFlLENBQUMsU0FBUyxDQUFDLElBQUksRUFBRTtZQUM5QixJQUFJLGVBQWUsS0FBSyxPQUFPLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxDQUFDLENBQUM7U0FDeEQsQ0FBQyxDQUFDO0lBQ0wsQ0FBQztJQUVEOztPQUVHO0lBQ0ksR0FBRyxDQUFDLFNBQXFCO1FBQzlCLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLENBQUM7SUFDeEMsQ0FBQztDQUNGO0FBaEJELGdEQWdCQztBQUVEOzs7Ozs7Ozs7Ozs7Ozs7Ozs7R0FrQkc7QUFDSCxNQUFzQixlQUFlO0lBQ25DOzs7O09BSUc7SUFDSSxNQUFNLENBQUMsU0FBUyxDQUFDLFFBQXFCLEVBQUUsS0FBc0I7UUFDbkUsNkVBQTZFO1FBQzdFLDhFQUE4RTtRQUM5RSxnQ0FBZ0M7UUFDaEMsZUFBZSxDQUFDLFFBQVEsQ0FBQyxHQUFHLENBQUMsUUFBUSxFQUFFLEtBQUssQ0FBQyxDQUFDO0lBQ2hELENBQUM7SUFFRDs7T0FFRztJQUNJLE1BQU0sQ0FBQyxHQUFHLENBQUMsUUFBcUI7UUFDckMsTUFBTSxHQUFHLEdBQUcsZUFBZSxDQUFDLFFBQVEsQ0FBQyxHQUFHLENBQUMsUUFBUSxDQUFDLENBQUM7UUFDbkQsSUFBSSxDQUFDLEdBQUcsRUFBRTtZQUNSLE1BQU0sSUFBSSxLQUFLLENBQUMsR0FBRyxRQUFRLHFDQUFxQyxDQUFDLENBQUM7U0FDbkU7UUFDRCxPQUFPLEdBQUcsQ0FBQztJQUNiLENBQUM7O0FBRWMsd0JBQVEsR0FBRyxJQUFJLE9BQU8sRUFBZ0MsQ0FBQztBQXhCeEUsMENBaUNDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSUNvbnN0cnVjdCB9IGZyb20gXCIuL2NvbnN0cnVjdFwiO1xuXG4vKipcbiAqIFRyYWl0IG1hcmtlciBmb3IgY2xhc3NlcyB0aGF0IGNhbiBiZSBkZXBlbmRlZCB1cG9uXG4gKlxuICogVGhlIHByZXNlbmNlIG9mIHRoaXMgaW50ZXJmYWNlIGluZGljYXRlcyB0aGF0IGFuIG9iamVjdCBoYXNcbiAqIGFuIGBJRGVwZW5kYWJsZVRyYWl0YCBpbXBsZW1lbnRhdGlvbi5cbiAqXG4gKiBUaGlzIGludGVyZmFjZSBjYW4gYmUgdXNlZCB0byB0YWtlIGFuIChvcmRlcmluZykgZGVwZW5kZW5jeSBvbiBhIHNldCBvZlxuICogY29uc3RydWN0cy4gQW4gb3JkZXJpbmcgZGVwZW5kZW5jeSBpbXBsaWVzIHRoYXQgdGhlIHJlc291cmNlcyByZXByZXNlbnRlZCBieVxuICogdGhvc2UgY29uc3RydWN0cyBhcmUgZGVwbG95ZWQgYmVmb3JlIHRoZSByZXNvdXJjZXMgZGVwZW5kaW5nIE9OIHRoZW0gYXJlXG4gKiBkZXBsb3llZC5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBJRGVwZW5kYWJsZSB7XG4gIC8vIEVtcHR5LCB0aGlzIGludGVyZmFjZSBpcyBhIHRyYWl0IG1hcmtlclxufVxuXG4vKipcbiAqIEEgc2V0IG9mIGNvbnN0cnVjdHMgdG8gYmUgdXNlZCBhcyBhIGRlcGVuZGFibGVcbiAqXG4gKiBUaGlzIGNsYXNzIGNhbiBiZSB1c2VkIHdoZW4gYSBzZXQgb2YgY29uc3RydWN0cyB3aGljaCBhcmUgZGlzam9pbnQgaW4gdGhlXG4gKiBjb25zdHJ1Y3QgdHJlZSBuZWVkcyB0byBiZSBjb21iaW5lZCB0byBiZSB1c2VkIGFzIGEgc2luZ2xlIGRlcGVuZGFibGUuXG4gKi9cbmV4cG9ydCBjbGFzcyBDb25jcmV0ZURlcGVuZGFibGUgaW1wbGVtZW50cyBJRGVwZW5kYWJsZSB7XG4gIHByaXZhdGUgcmVhZG9ubHkgX2RlcGVuZGVuY3lSb290cyA9IG5ldyBBcnJheTxJQ29uc3RydWN0PigpO1xuXG4gIGNvbnN0cnVjdG9yKCkge1xuICAgIGNvbnN0IHNlbGYgPSB0aGlzO1xuICAgIERlcGVuZGFibGVUcmFpdC5pbXBsZW1lbnQodGhpcywge1xuICAgICAgZ2V0IGRlcGVuZGVuY3lSb290cygpIHsgcmV0dXJuIHNlbGYuX2RlcGVuZGVuY3lSb290czsgfSxcbiAgICB9KTtcbiAgfVxuXG4gIC8qKlxuICAgKiBBZGQgYSBjb25zdHJ1Y3QgdG8gdGhlIGRlcGVuZGVuY3kgcm9vdHNcbiAgICovXG4gIHB1YmxpYyBhZGQoY29uc3RydWN0OiBJQ29uc3RydWN0KSB7XG4gICAgdGhpcy5fZGVwZW5kZW5jeVJvb3RzLnB1c2goY29uc3RydWN0KTtcbiAgfVxufVxuXG4vKipcbiAqIFRyYWl0IGZvciBJRGVwZW5kYWJsZVxuICpcbiAqIFRyYWl0cyBhcmUgaW50ZXJmYWNlcyB0aGF0IGFyZSBwcml2YXRlbHkgaW1wbGVtZW50ZWQgYnkgb2JqZWN0cy4gSW5zdGVhZCBvZlxuICogc2hvd2luZyB1cCBpbiB0aGUgcHVibGljIGludGVyZmFjZSBvZiBhIGNsYXNzLCB0aGV5IG5lZWQgdG8gYmUgcXVlcmllZFxuICogZXhwbGljaXRseS4gVGhpcyBpcyB1c2VkIHRvIGltcGxlbWVudCBjZXJ0YWluIGZyYW1ld29yayBmZWF0dXJlcyB0aGF0IGFyZVxuICogbm90IGludGVuZGVkIHRvIGJlIHVzZWQgYnkgQ29uc3RydWN0IGNvbnN1bWVycywgYW5kIHNvIHNob3VsZCBiZSBoaWRkZW5cbiAqIGZyb20gYWNjaWRlbnRhbCB1c2UuXG4gKlxuICogQGV4YW1wbGVcbiAqXG4gKiAvLyBVc2FnZVxuICogY29uc3Qgcm9vdHMgPSBEZXBlbmRhYmxlVHJhaXQuZ2V0KGNvbnN0cnVjdCkuZGVwZW5kZW5jeVJvb3RzO1xuICpcbiAqIC8vIERlZmluaXRpb25cbiAqIERlcGVuZGFibGVUcmFpdC5pbXBsZW1lbnQoY29uc3RydWN0LCB7XG4gKiAgIGdldCBkZXBlbmRlbmN5Um9vdHMoKSB7IHJldHVybiBbXTsgfVxuICogfSk7XG4gKi9cbmV4cG9ydCBhYnN0cmFjdCBjbGFzcyBEZXBlbmRhYmxlVHJhaXQge1xuICAvKipcbiAgICogUmVnaXN0ZXIgYGluc3RhbmNlYCB0byBoYXZlIHRoZSBnaXZlbiBEZXBlbmRhYmxlVHJhaXRcbiAgICpcbiAgICogU2hvdWxkIGJlIGNhbGxlZCBpbiB0aGUgY2xhc3MgY29uc3RydWN0b3IuXG4gICAqL1xuICBwdWJsaWMgc3RhdGljIGltcGxlbWVudChpbnN0YW5jZTogSURlcGVuZGFibGUsIHRyYWl0OiBEZXBlbmRhYmxlVHJhaXQpIHtcbiAgICAvLyBJIHdvdWxkIGFsc28gbGlrZSB0byByZWZlcmVuY2UgY2xhc3NlcyAodG8gY3V0IGRvd24gb24gdGhlIGxpc3Qgb2Ygb2JqZWN0c1xuICAgIC8vIHdlIG5lZWQgdG8gbWFuYWdlKSwgYnV0IHdlIGNhbid0IGRvIHRoYXQgZWl0aGVyIHNpbmNlIGpzaWkgZG9lc24ndCBoYXZlIHRoZVxuICAgIC8vIGNvbmNlcHQgb2YgYSBjbGFzcyByZWZlcmVuY2UuXG4gICAgRGVwZW5kYWJsZVRyYWl0LnRyYWl0TWFwLnNldChpbnN0YW5jZSwgdHJhaXQpO1xuICB9XG5cbiAgLyoqXG4gICAqIFJldHVybiB0aGUgbWF0Y2hpbmcgRGVwZW5kYWJsZVRyYWl0IGZvciB0aGUgZ2l2ZW4gY2xhc3MgaW5zdGFuY2UuXG4gICAqL1xuICBwdWJsaWMgc3RhdGljIGdldChpbnN0YW5jZTogSURlcGVuZGFibGUpOiBEZXBlbmRhYmxlVHJhaXQge1xuICAgIGNvbnN0IHJldCA9IERlcGVuZGFibGVUcmFpdC50cmFpdE1hcC5nZXQoaW5zdGFuY2UpO1xuICAgIGlmICghcmV0KSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoYCR7aW5zdGFuY2V9IGRvZXMgbm90IGltcGxlbWVudCBEZXBlbmRhYmxlVHJhaXRgKTtcbiAgICB9XG4gICAgcmV0dXJuIHJldDtcbiAgfVxuXG4gIHByaXZhdGUgc3RhdGljIHRyYWl0TWFwID0gbmV3IFdlYWtNYXA8SURlcGVuZGFibGUsIERlcGVuZGFibGVUcmFpdD4oKTtcblxuICAvKipcbiAgICogVGhlIHNldCBvZiBjb25zdHJ1Y3RzIHRoYXQgZm9ybSB0aGUgcm9vdCBvZiB0aGlzIGRlcGVuZGFibGVcbiAgICpcbiAgICogQWxsIHJlc291cmNlcyB1bmRlciBhbGwgcmV0dXJuZWQgY29uc3RydWN0cyBhcmUgaW5jbHVkZWQgaW4gdGhlIG9yZGVyaW5nXG4gICAqIGRlcGVuZGVuY3kuXG4gICAqL1xuICBwdWJsaWMgYWJzdHJhY3QgcmVhZG9ubHkgZGVwZW5kZW5jeVJvb3RzOiBJQ29uc3RydWN0W107XG59Il19

@@ -81,3 +81,3 @@ import { ICfnConditionExpression } from './cfn-condition';

*/
static cidr(ipBlock: string, count: number, sizeMask?: string): string;
static cidr(ipBlock: string, count: number, sizeMask?: string): string[];
/**

@@ -84,0 +84,0 @@ * The intrinsic function ``Fn::GetAZs`` returns an array that lists

@@ -5,2 +5,2 @@ import cxapi = require('@aws-cdk/cx-api');

*/
export declare function collectRuntimeInformation(): cxapi.AppRuntime;
export declare function collectRuntimeInformation(): cxapi.RuntimeInfo;

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

}
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicnVudGltZS1pbmZvLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsicnVudGltZS1pbmZvLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7O0FBQ0EsaURBQTJEO0FBRTNEOztHQUVHO0FBQ0gsU0FBZ0IseUJBQXlCO0lBQ3ZDLE1BQU0sU0FBUyxHQUErQixFQUFFLENBQUM7SUFFakQsS0FBSyxNQUFNLFFBQVEsSUFBSSxNQUFNLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxLQUFLLENBQUMsRUFBRTtRQUNqRCxNQUFNLEdBQUcsR0FBRyxjQUFjLENBQUMsUUFBUSxDQUFDLENBQUM7UUFDckMsSUFBSSxHQUFHLElBQUksQ0FBQyxHQUFHLENBQUMsT0FBTyxFQUFFO1lBQ3ZCLFNBQVMsQ0FBQyxHQUFHLENBQUMsSUFBSSxDQUFDLEdBQUcsR0FBRyxDQUFDLE9BQU8sQ0FBQztTQUNuQztLQUNGO0lBRUQsNERBQTREO0lBQzVELEtBQUssTUFBTSxJQUFJLElBQUksTUFBTSxDQUFDLElBQUksQ0FBQyxTQUFTLENBQUMsRUFBRTtRQUN6QyxJQUFJLENBQUMsSUFBSSxDQUFDLFVBQVUsQ0FBQyxXQUFXLENBQUMsRUFBRTtZQUNqQyxPQUFPLFNBQVMsQ0FBQyxJQUFJLENBQUMsQ0FBQztTQUN4QjtLQUNGO0lBRUQsMkJBQTJCO0lBQzNCLFNBQVMsQ0FBQyxjQUFjLENBQUMsR0FBRyxtQkFBbUIsRUFBRSxDQUFDO0lBRWxELE9BQU8sRUFBRSxTQUFTLEVBQUUsQ0FBQztBQUN2QixDQUFDO0FBckJELDhEQXFCQztBQUVEOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQXVCRztBQUNILFNBQVMsY0FBYyxDQUFDLFFBQWdCO0lBQ3RDLE1BQU0sR0FBRyxHQUFHLE9BQU8sQ0FBQyxLQUFLLENBQUMsUUFBUSxDQUFDLENBQUM7SUFDcEMsTUFBTSxLQUFLLEdBQUcsR0FBRyxDQUFDLEtBQUssQ0FBQyxHQUFHLENBQUMsZ0JBQWdCLENBQUMsQ0FBQztJQUU5QyxJQUFJO1FBQ0YsTUFBTSxXQUFXLEdBQUcsT0FBTyxDQUFDLE9BQU87UUFDakMsMkZBQTJGO1FBQzNGLG9CQUFnQixJQUFJLEVBQUUsQ0FBQyxDQUFDLENBQUMsZ0JBQWdCLENBQUMsQ0FBQyxDQUFDLGNBQWMsRUFDMUQsRUFBRSxLQUFLLEVBQUUsQ0FDVixDQUFDO1FBQ0YsT0FBTyxPQUFPLENBQUMsV0FBVyxDQUFDLENBQUM7S0FDN0I7SUFBQyxPQUFPLENBQUMsRUFBRTtRQUNWLE9BQU8sU0FBUyxDQUFDO0tBQ2xCO0lBRUQ7Ozs7T0FJRztJQUNILFNBQVMsZ0JBQWdCLENBQUMsQ0FBUztRQUNqQyxJQUFJLENBQUMsQ0FBQyxRQUFRLENBQUMsZUFBZSxDQUFDLElBQUksQ0FBQyxDQUFDLFFBQVEsQ0FBQyxnQkFBZ0IsQ0FBQyxFQUFFO1lBQy9ELGlDQUFpQztZQUNqQyxPQUFPLENBQUMsQ0FBQyxNQUFNLENBQUMsQ0FBQyxFQUFFLENBQUMsQ0FBQyxNQUFNLEdBQUcsRUFBRSxDQUFDLENBQUM7U0FDbkM7UUFDRCxPQUFPLENBQUMsQ0FBQztJQUNYLENBQUM7QUFDSCxDQUFDO0FBRUQsU0FBUyxtQkFBbUI7SUFDMUIsSUFBSSxTQUFTLEdBQUcsT0FBTyxDQUFDLEdBQUcsQ0FBQyxVQUFVLENBQUM7SUFFdkMsMkVBQTJFO0lBQzNFLGtDQUFrQztJQUNsQyxJQUFJLENBQUMsU0FBUyxFQUFFO1FBQ2QsU0FBUyxHQUFHLFdBQVcsT0FBTyxDQUFDLE9BQU8sRUFBRSxDQUFDO0tBQzFDO0lBRUQsT0FBTyxTQUFTLENBQUM7QUFDbkIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBjeGFwaSA9IHJlcXVpcmUoJ0Bhd3MtY2RrL2N4LWFwaScpO1xuaW1wb3J0IHsgbWFqb3IgYXMgbm9kZU1ham9yVmVyc2lvbiB9IGZyb20gJy4vbm9kZS12ZXJzaW9uJztcblxuLyoqXG4gKiBSZXR1cm5zIGEgbGlzdCBvZiBsb2FkZWQgbW9kdWxlcyBhbmQgdGhlaXIgdmVyc2lvbnMuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBjb2xsZWN0UnVudGltZUluZm9ybWF0aW9uKCk6IGN4YXBpLkFwcFJ1bnRpbWUge1xuICBjb25zdCBsaWJyYXJpZXM6IHsgW25hbWU6IHN0cmluZ106IHN0cmluZyB9ID0ge307XG5cbiAgZm9yIChjb25zdCBmaWxlTmFtZSBvZiBPYmplY3Qua2V5cyhyZXF1aXJlLmNhY2hlKSkge1xuICAgIGNvbnN0IHBrZyA9IGZpbmROcG1QYWNrYWdlKGZpbGVOYW1lKTtcbiAgICBpZiAocGtnICYmICFwa2cucHJpdmF0ZSkge1xuICAgICAgbGlicmFyaWVzW3BrZy5uYW1lXSA9IHBrZy52ZXJzaW9uO1xuICAgIH1cbiAgfVxuXG4gIC8vIGluY2x1ZGUgb25seSBsaWJyYXJpZXMgdGhhdCBhcmUgaW4gdGhlIEBhd3MtY2RrIG5wbSBzY29wZVxuICBmb3IgKGNvbnN0IG5hbWUgb2YgT2JqZWN0LmtleXMobGlicmFyaWVzKSkge1xuICAgIGlmICghbmFtZS5zdGFydHNXaXRoKCdAYXdzLWNkay8nKSkge1xuICAgICAgZGVsZXRlIGxpYnJhcmllc1tuYW1lXTtcbiAgICB9XG4gIH1cblxuICAvLyBhZGQganNpaSBydW50aW1lIHZlcnNpb25cbiAgbGlicmFyaWVzWydqc2lpLXJ1bnRpbWUnXSA9IGdldEpzaWlBZ2VudFZlcnNpb24oKTtcblxuICByZXR1cm4geyBsaWJyYXJpZXMgfTtcbn1cblxuLyoqXG4gKiBEZXRlcm1pbmVzIHdoaWNoIE5QTSBtb2R1bGUgYSBnaXZlbiBsb2FkZWQgamF2YXNjcmlwdCBmaWxlIGlzIGZyb20uXG4gKlxuICogVGhlIG9ubHkgaW5mcm9tYXRpb24gdGhhdCBpcyBhdmFpbGFibGUgbG9jYWxseSBpcyBhIGxpc3Qgb2YgSmF2YXNjcmlwdCBmaWxlcyxcbiAqIGFuZCBldmVyeSBzb3VyY2UgZmlsZSBpcyBhc3NvY2lhdGVkIHdpdGggYSBzZWFyY2ggcGF0aCB0byByZXNvbHZlIHRoZSBmdXJ0aGVyXG4gKiBgYHJlcXVpcmVgYCBjYWxscyBtYWRlIGZyb20gdGhlcmUsIHdoaWNoIGluY2x1ZGVzIGl0cyBvd24gZGlyZWN0b3J5IG9uIGRpc2ssXG4gKiBhbmQgcGFyZW50IGRpcmVjdG9yaWVzIC0gZm9yIGV4YW1wbGU6XG4gKlxuICogWyAnLi4ucmVwby9wYWNrYWdlcy9hd3MtY2RrLXJlc291cmNlcy9saWIvY2ZuL25vZGVfbW9kdWxlcycsXG4gKiAgICcuLi5yZXBvL3BhY2thZ2VzL2F3cy1jZGstcmVzb3VyY2VzL2xpYi9ub2RlX21vZHVsZXMnLFxuICogICAnLi4ucmVwby9wYWNrYWdlcy9hd3MtY2RrLXJlc291cmNlcy9ub2RlX21vZHVsZXMnLFxuICogICAnLi4ucmVwby9wYWNrYWdlcy9ub2RlX21vZHVsZXMnLFxuICogICAvLyBldGMuLi5cbiAqIF1cbiAqXG4gKiBXZSBhcmUgbG9va2luZyBmb3IgYGBwYWNrYWdlLmpzb25gYCB0aGF0IGlzIGFueXdoZXJlIGluIHRoZSB0cmVlLCBleGNlcHQgaXQnc1xuICogaW4gdGhlIHBhcmVudCBkaXJlY3RvcnksIG5vdCBpbiB0aGUgYGBub2RlX21vZHVsZXNgYCBkaXJlY3RvcnkuIEZvciB0aGlzXG4gKiByZWFzb24sIHdlIHN0cmlwIHRoZSBgYC9ub2RlX21vZHVsZXNgYCBzdWZmaXggb2ZmIGVhY2ggcGF0aCBhbmQgdXNlIHJlZ3VsYXJcbiAqIG1vZHVsZSByZXNvbHV0aW9uIHRvIG9idGFpbiBhIHJlZmVyZW5jZSB0byBgYHBhY2thZ2UuanNvbmBgLlxuICpcbiAqIEBwYXJhbSBmaWxlTmFtZSBhIGphdmFzY3JpcHQgZmlsZSBuYW1lLlxuICogQHJldHVybnMgdGhlIE5QTSBtb2R1bGUgaW5mb3MgKGFrYSBgYHBhY2thZ2UuanNvbmBgIGNvbnRlbnRzKSwgb3JcbiAqICAgICAgYGB1bmRlZmluZWRgYCBpZiB0aGUgbG9va3VwIHdhcyB1bnN1Y2Nlc3NmdWwuXG4gKi9cbmZ1bmN0aW9uIGZpbmROcG1QYWNrYWdlKGZpbGVOYW1lOiBzdHJpbmcpOiB7IG5hbWU6IHN0cmluZywgdmVyc2lvbjogc3RyaW5nLCBwcml2YXRlPzogYm9vbGVhbiB9IHwgdW5kZWZpbmVkIHtcbiAgY29uc3QgbW9kID0gcmVxdWlyZS5jYWNoZVtmaWxlTmFtZV07XG4gIGNvbnN0IHBhdGhzID0gbW9kLnBhdGhzLm1hcChzdHJpcE5vZGVNb2R1bGVzKTtcblxuICB0cnkge1xuICAgIGNvbnN0IHBhY2thZ2VQYXRoID0gcmVxdWlyZS5yZXNvbHZlKFxuICAgICAgLy8gUmVzb2x1dGlvbiBiZWhhdmlvciBjaGFuZ2VkIGluIG5vZGUgMTIuMC4wIC0gaHR0cHM6Ly9naXRodWIuY29tL25vZGVqcy9ub2RlL2lzc3Vlcy8yNzU4M1xuICAgICAgbm9kZU1ham9yVmVyc2lvbiA+PSAxMiA/ICcuL3BhY2thZ2UuanNvbicgOiAncGFja2FnZS5qc29uJyxcbiAgICAgIHsgcGF0aHMgfVxuICAgICk7XG4gICAgcmV0dXJuIHJlcXVpcmUocGFja2FnZVBhdGgpO1xuICB9IGNhdGNoIChlKSB7XG4gICAgcmV0dXJuIHVuZGVmaW5lZDtcbiAgfVxuXG4gIC8qKlxuICAgKiBAcGFyYW0gcyBhIHBhdGguXG4gICAqIEByZXR1cm5zIGBgc2BgIHdpdGggYW55IHRlcm1pbmF0aW5nIGBgL25vZGVfbW9kdWxlc2BgXG4gICAqICAgICAgKG9yIGBgXFxcXG5vZGVfbW9kdWxlc2BgKSBzdHJpcHBlZCBvZmYuKVxuICAgKi9cbiAgZnVuY3Rpb24gc3RyaXBOb2RlTW9kdWxlcyhzOiBzdHJpbmcpOiBzdHJpbmcge1xuICAgIGlmIChzLmVuZHNXaXRoKCcvbm9kZV9tb2R1bGVzJykgfHwgcy5lbmRzV2l0aCgnXFxcXG5vZGVfbW9kdWxlcycpKSB7XG4gICAgICAvLyAvbm9kZV9tb2R1bGVzIGlzIDEzIGNoYXJhY3RlcnNcbiAgICAgIHJldHVybiBzLnN1YnN0cigwLCBzLmxlbmd0aCAtIDEzKTtcbiAgICB9XG4gICAgcmV0dXJuIHM7XG4gIH1cbn1cblxuZnVuY3Rpb24gZ2V0SnNpaUFnZW50VmVyc2lvbigpIHtcbiAgbGV0IGpzaWlBZ2VudCA9IHByb2Nlc3MuZW52LkpTSUlfQUdFTlQ7XG5cbiAgLy8gaWYgSlNJSV9BR0VOVCBpcyBub3Qgc3BlY2lmaWVkLCB3ZSB3aWxsIGFzc3VtZSB0aGlzIGlzIGEgbm9kZS5qcyBydW50aW1lXG4gIC8vIGFuZCBwbHVnIGluIG91ciBub2RlLmpzIHZlcnNpb25cbiAgaWYgKCFqc2lpQWdlbnQpIHtcbiAgICBqc2lpQWdlbnQgPSBgbm9kZS5qcy8ke3Byb2Nlc3MudmVyc2lvbn1gO1xuICB9XG5cbiAgcmV0dXJuIGpzaWlBZ2VudDtcbn1cbiJdfQ==
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicnVudGltZS1pbmZvLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsicnVudGltZS1pbmZvLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7O0FBQ0EsaURBQTJEO0FBRTNEOztHQUVHO0FBQ0gsU0FBZ0IseUJBQXlCO0lBQ3ZDLE1BQU0sU0FBUyxHQUErQixFQUFFLENBQUM7SUFFakQsS0FBSyxNQUFNLFFBQVEsSUFBSSxNQUFNLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxLQUFLLENBQUMsRUFBRTtRQUNqRCxNQUFNLEdBQUcsR0FBRyxjQUFjLENBQUMsUUFBUSxDQUFDLENBQUM7UUFDckMsSUFBSSxHQUFHLElBQUksQ0FBQyxHQUFHLENBQUMsT0FBTyxFQUFFO1lBQ3ZCLFNBQVMsQ0FBQyxHQUFHLENBQUMsSUFBSSxDQUFDLEdBQUcsR0FBRyxDQUFDLE9BQU8sQ0FBQztTQUNuQztLQUNGO0lBRUQsNERBQTREO0lBQzVELEtBQUssTUFBTSxJQUFJLElBQUksTUFBTSxDQUFDLElBQUksQ0FBQyxTQUFTLENBQUMsRUFBRTtRQUN6QyxJQUFJLENBQUMsSUFBSSxDQUFDLFVBQVUsQ0FBQyxXQUFXLENBQUMsRUFBRTtZQUNqQyxPQUFPLFNBQVMsQ0FBQyxJQUFJLENBQUMsQ0FBQztTQUN4QjtLQUNGO0lBRUQsMkJBQTJCO0lBQzNCLFNBQVMsQ0FBQyxjQUFjLENBQUMsR0FBRyxtQkFBbUIsRUFBRSxDQUFDO0lBRWxELE9BQU8sRUFBRSxTQUFTLEVBQUUsQ0FBQztBQUN2QixDQUFDO0FBckJELDhEQXFCQztBQUVEOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQXVCRztBQUNILFNBQVMsY0FBYyxDQUFDLFFBQWdCO0lBQ3RDLE1BQU0sR0FBRyxHQUFHLE9BQU8sQ0FBQyxLQUFLLENBQUMsUUFBUSxDQUFDLENBQUM7SUFDcEMsTUFBTSxLQUFLLEdBQUcsR0FBRyxDQUFDLEtBQUssQ0FBQyxHQUFHLENBQUMsZ0JBQWdCLENBQUMsQ0FBQztJQUU5QyxJQUFJO1FBQ0YsTUFBTSxXQUFXLEdBQUcsT0FBTyxDQUFDLE9BQU87UUFDakMsMkZBQTJGO1FBQzNGLG9CQUFnQixJQUFJLEVBQUUsQ0FBQyxDQUFDLENBQUMsZ0JBQWdCLENBQUMsQ0FBQyxDQUFDLGNBQWMsRUFDMUQsRUFBRSxLQUFLLEVBQUUsQ0FDVixDQUFDO1FBQ0YsT0FBTyxPQUFPLENBQUMsV0FBVyxDQUFDLENBQUM7S0FDN0I7SUFBQyxPQUFPLENBQUMsRUFBRTtRQUNWLE9BQU8sU0FBUyxDQUFDO0tBQ2xCO0lBRUQ7Ozs7T0FJRztJQUNILFNBQVMsZ0JBQWdCLENBQUMsQ0FBUztRQUNqQyxJQUFJLENBQUMsQ0FBQyxRQUFRLENBQUMsZUFBZSxDQUFDLElBQUksQ0FBQyxDQUFDLFFBQVEsQ0FBQyxnQkFBZ0IsQ0FBQyxFQUFFO1lBQy9ELGlDQUFpQztZQUNqQyxPQUFPLENBQUMsQ0FBQyxNQUFNLENBQUMsQ0FBQyxFQUFFLENBQUMsQ0FBQyxNQUFNLEdBQUcsRUFBRSxDQUFDLENBQUM7U0FDbkM7UUFDRCxPQUFPLENBQUMsQ0FBQztJQUNYLENBQUM7QUFDSCxDQUFDO0FBRUQsU0FBUyxtQkFBbUI7SUFDMUIsSUFBSSxTQUFTLEdBQUcsT0FBTyxDQUFDLEdBQUcsQ0FBQyxVQUFVLENBQUM7SUFFdkMsMkVBQTJFO0lBQzNFLGtDQUFrQztJQUNsQyxJQUFJLENBQUMsU0FBUyxFQUFFO1FBQ2QsU0FBUyxHQUFHLFdBQVcsT0FBTyxDQUFDLE9BQU8sRUFBRSxDQUFDO0tBQzFDO0lBRUQsT0FBTyxTQUFTLENBQUM7QUFDbkIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBjeGFwaSA9IHJlcXVpcmUoJ0Bhd3MtY2RrL2N4LWFwaScpO1xuaW1wb3J0IHsgbWFqb3IgYXMgbm9kZU1ham9yVmVyc2lvbiB9IGZyb20gJy4vbm9kZS12ZXJzaW9uJztcblxuLyoqXG4gKiBSZXR1cm5zIGEgbGlzdCBvZiBsb2FkZWQgbW9kdWxlcyBhbmQgdGhlaXIgdmVyc2lvbnMuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBjb2xsZWN0UnVudGltZUluZm9ybWF0aW9uKCk6IGN4YXBpLlJ1bnRpbWVJbmZvIHtcbiAgY29uc3QgbGlicmFyaWVzOiB7IFtuYW1lOiBzdHJpbmddOiBzdHJpbmcgfSA9IHt9O1xuXG4gIGZvciAoY29uc3QgZmlsZU5hbWUgb2YgT2JqZWN0LmtleXMocmVxdWlyZS5jYWNoZSkpIHtcbiAgICBjb25zdCBwa2cgPSBmaW5kTnBtUGFja2FnZShmaWxlTmFtZSk7XG4gICAgaWYgKHBrZyAmJiAhcGtnLnByaXZhdGUpIHtcbiAgICAgIGxpYnJhcmllc1twa2cubmFtZV0gPSBwa2cudmVyc2lvbjtcbiAgICB9XG4gIH1cblxuICAvLyBpbmNsdWRlIG9ubHkgbGlicmFyaWVzIHRoYXQgYXJlIGluIHRoZSBAYXdzLWNkayBucG0gc2NvcGVcbiAgZm9yIChjb25zdCBuYW1lIG9mIE9iamVjdC5rZXlzKGxpYnJhcmllcykpIHtcbiAgICBpZiAoIW5hbWUuc3RhcnRzV2l0aCgnQGF3cy1jZGsvJykpIHtcbiAgICAgIGRlbGV0ZSBsaWJyYXJpZXNbbmFtZV07XG4gICAgfVxuICB9XG5cbiAgLy8gYWRkIGpzaWkgcnVudGltZSB2ZXJzaW9uXG4gIGxpYnJhcmllc1snanNpaS1ydW50aW1lJ10gPSBnZXRKc2lpQWdlbnRWZXJzaW9uKCk7XG5cbiAgcmV0dXJuIHsgbGlicmFyaWVzIH07XG59XG5cbi8qKlxuICogRGV0ZXJtaW5lcyB3aGljaCBOUE0gbW9kdWxlIGEgZ2l2ZW4gbG9hZGVkIGphdmFzY3JpcHQgZmlsZSBpcyBmcm9tLlxuICpcbiAqIFRoZSBvbmx5IGluZnJvbWF0aW9uIHRoYXQgaXMgYXZhaWxhYmxlIGxvY2FsbHkgaXMgYSBsaXN0IG9mIEphdmFzY3JpcHQgZmlsZXMsXG4gKiBhbmQgZXZlcnkgc291cmNlIGZpbGUgaXMgYXNzb2NpYXRlZCB3aXRoIGEgc2VhcmNoIHBhdGggdG8gcmVzb2x2ZSB0aGUgZnVydGhlclxuICogYGByZXF1aXJlYGAgY2FsbHMgbWFkZSBmcm9tIHRoZXJlLCB3aGljaCBpbmNsdWRlcyBpdHMgb3duIGRpcmVjdG9yeSBvbiBkaXNrLFxuICogYW5kIHBhcmVudCBkaXJlY3RvcmllcyAtIGZvciBleGFtcGxlOlxuICpcbiAqIFsgJy4uLnJlcG8vcGFja2FnZXMvYXdzLWNkay1yZXNvdXJjZXMvbGliL2Nmbi9ub2RlX21vZHVsZXMnLFxuICogICAnLi4ucmVwby9wYWNrYWdlcy9hd3MtY2RrLXJlc291cmNlcy9saWIvbm9kZV9tb2R1bGVzJyxcbiAqICAgJy4uLnJlcG8vcGFja2FnZXMvYXdzLWNkay1yZXNvdXJjZXMvbm9kZV9tb2R1bGVzJyxcbiAqICAgJy4uLnJlcG8vcGFja2FnZXMvbm9kZV9tb2R1bGVzJyxcbiAqICAgLy8gZXRjLi4uXG4gKiBdXG4gKlxuICogV2UgYXJlIGxvb2tpbmcgZm9yIGBgcGFja2FnZS5qc29uYGAgdGhhdCBpcyBhbnl3aGVyZSBpbiB0aGUgdHJlZSwgZXhjZXB0IGl0J3NcbiAqIGluIHRoZSBwYXJlbnQgZGlyZWN0b3J5LCBub3QgaW4gdGhlIGBgbm9kZV9tb2R1bGVzYGAgZGlyZWN0b3J5LiBGb3IgdGhpc1xuICogcmVhc29uLCB3ZSBzdHJpcCB0aGUgYGAvbm9kZV9tb2R1bGVzYGAgc3VmZml4IG9mZiBlYWNoIHBhdGggYW5kIHVzZSByZWd1bGFyXG4gKiBtb2R1bGUgcmVzb2x1dGlvbiB0byBvYnRhaW4gYSByZWZlcmVuY2UgdG8gYGBwYWNrYWdlLmpzb25gYC5cbiAqXG4gKiBAcGFyYW0gZmlsZU5hbWUgYSBqYXZhc2NyaXB0IGZpbGUgbmFtZS5cbiAqIEByZXR1cm5zIHRoZSBOUE0gbW9kdWxlIGluZm9zIChha2EgYGBwYWNrYWdlLmpzb25gYCBjb250ZW50cyksIG9yXG4gKiAgICAgIGBgdW5kZWZpbmVkYGAgaWYgdGhlIGxvb2t1cCB3YXMgdW5zdWNjZXNzZnVsLlxuICovXG5mdW5jdGlvbiBmaW5kTnBtUGFja2FnZShmaWxlTmFtZTogc3RyaW5nKTogeyBuYW1lOiBzdHJpbmcsIHZlcnNpb246IHN0cmluZywgcHJpdmF0ZT86IGJvb2xlYW4gfSB8IHVuZGVmaW5lZCB7XG4gIGNvbnN0IG1vZCA9IHJlcXVpcmUuY2FjaGVbZmlsZU5hbWVdO1xuICBjb25zdCBwYXRocyA9IG1vZC5wYXRocy5tYXAoc3RyaXBOb2RlTW9kdWxlcyk7XG5cbiAgdHJ5IHtcbiAgICBjb25zdCBwYWNrYWdlUGF0aCA9IHJlcXVpcmUucmVzb2x2ZShcbiAgICAgIC8vIFJlc29sdXRpb24gYmVoYXZpb3IgY2hhbmdlZCBpbiBub2RlIDEyLjAuMCAtIGh0dHBzOi8vZ2l0aHViLmNvbS9ub2RlanMvbm9kZS9pc3N1ZXMvMjc1ODNcbiAgICAgIG5vZGVNYWpvclZlcnNpb24gPj0gMTIgPyAnLi9wYWNrYWdlLmpzb24nIDogJ3BhY2thZ2UuanNvbicsXG4gICAgICB7IHBhdGhzIH1cbiAgICApO1xuICAgIHJldHVybiByZXF1aXJlKHBhY2thZ2VQYXRoKTtcbiAgfSBjYXRjaCAoZSkge1xuICAgIHJldHVybiB1bmRlZmluZWQ7XG4gIH1cblxuICAvKipcbiAgICogQHBhcmFtIHMgYSBwYXRoLlxuICAgKiBAcmV0dXJucyBgYHNgYCB3aXRoIGFueSB0ZXJtaW5hdGluZyBgYC9ub2RlX21vZHVsZXNgYFxuICAgKiAgICAgIChvciBgYFxcXFxub2RlX21vZHVsZXNgYCkgc3RyaXBwZWQgb2ZmLilcbiAgICovXG4gIGZ1bmN0aW9uIHN0cmlwTm9kZU1vZHVsZXMoczogc3RyaW5nKTogc3RyaW5nIHtcbiAgICBpZiAocy5lbmRzV2l0aCgnL25vZGVfbW9kdWxlcycpIHx8IHMuZW5kc1dpdGgoJ1xcXFxub2RlX21vZHVsZXMnKSkge1xuICAgICAgLy8gL25vZGVfbW9kdWxlcyBpcyAxMyBjaGFyYWN0ZXJzXG4gICAgICByZXR1cm4gcy5zdWJzdHIoMCwgcy5sZW5ndGggLSAxMyk7XG4gICAgfVxuICAgIHJldHVybiBzO1xuICB9XG59XG5cbmZ1bmN0aW9uIGdldEpzaWlBZ2VudFZlcnNpb24oKSB7XG4gIGxldCBqc2lpQWdlbnQgPSBwcm9jZXNzLmVudi5KU0lJX0FHRU5UO1xuXG4gIC8vIGlmIEpTSUlfQUdFTlQgaXMgbm90IHNwZWNpZmllZCwgd2Ugd2lsbCBhc3N1bWUgdGhpcyBpcyBhIG5vZGUuanMgcnVudGltZVxuICAvLyBhbmQgcGx1ZyBpbiBvdXIgbm9kZS5qcyB2ZXJzaW9uXG4gIGlmICghanNpaUFnZW50KSB7XG4gICAganNpaUFnZW50ID0gYG5vZGUuanMvJHtwcm9jZXNzLnZlcnNpb259YDtcbiAgfVxuXG4gIHJldHVybiBqc2lpQWdlbnQ7XG59XG4iXX0=

@@ -7,3 +7,2 @@ import cxapi = require('@aws-cdk/cx-api');

import { IAddressingScheme, LogicalIDs } from './logical-id';
import { ISynthesisSession } from './synthesis';
export interface StackProps {

@@ -125,3 +124,3 @@ /**

*/
findResource(path: string): CfnResource | undefined;
findResource(constructPath: string): CfnResource | undefined;
/**

@@ -272,3 +271,3 @@ * @param why more information about why region is required.

protected prepare(): void;
protected synthesize(session: ISynthesisSession): void;
protected synthesize(builder: cxapi.CloudAssemblyBuilder): void;
/**

@@ -275,0 +274,0 @@ * Applied defaults to environment attributes.

@@ -1,20 +0,12 @@

import cxapi = require('@aws-cdk/cx-api');
import { BuildOptions, CloudAssembly, CloudAssemblyBuilder } from '@aws-cdk/cx-api';
import { IConstruct } from './construct';
export interface ISynthesizable {
synthesize(session: ISynthesisSession): void;
synthesize(session: CloudAssemblyBuilder): void;
}
export interface ISynthesisSession {
readonly store: ISessionStore;
readonly manifest: cxapi.AssemblyManifest;
addArtifact(id: string, droplet: cxapi.Artifact): void;
addBuildStep(id: string, step: cxapi.BuildStep): void;
tryGetArtifact(id: string): cxapi.Artifact | undefined;
getArtifact(id: string): cxapi.Artifact;
}
export interface SynthesisOptions extends ManifestOptions {
export interface SynthesisOptions extends BuildOptions {
/**
* The file store used for this session.
* @default InMemoryStore
* @default - creates a temporary directory
*/
readonly store?: ISessionStore;
readonly outdir?: string;
/**

@@ -27,122 +19,3 @@ * Whether synthesis should skip the validation phase.

export declare class Synthesizer {
synthesize(root: IConstruct, options?: SynthesisOptions): ISynthesisSession;
synthesize(root: IConstruct, options?: SynthesisOptions): CloudAssembly;
}
export declare class SynthesisSession implements ISynthesisSession {
/**
* @returns true if `obj` implements `ISynthesizable`.
*/
static isSynthesizable(obj: any): obj is ISynthesizable;
readonly store: ISessionStore;
private readonly artifacts;
private readonly buildSteps;
private _manifest?;
constructor(options: SynthesisOptions);
readonly manifest: cxapi.AssemblyManifest;
addArtifact(id: string, artifact: cxapi.Artifact): void;
tryGetArtifact(id: string): cxapi.Artifact | undefined;
getArtifact(id: string): cxapi.Artifact;
addBuildStep(id: string, step: cxapi.BuildStep): void;
close(options?: ManifestOptions): cxapi.AssemblyManifest;
}
export interface ManifestOptions {
/**
* Emit the legacy manifest (`cdk.out`) when the session is closed (alongside `manifest.json`).
* @default false
*/
readonly legacyManifest?: boolean;
/**
* Include runtime information (module versions) in manifest.
* @default true
*/
readonly runtimeInformation?: boolean;
}
export interface ISessionStore {
/**
* Creates a directory and returns it's full path.
* @param directoryName The name of the directory to create.
* @throws if a directory by that name already exists in the session or if the session has already been finalized.
*/
mkdir(directoryName: string): string;
/**
* Returns the list of files in a directory.
* @param directoryName The name of the artifact
* @throws if there is no directory artifact under this name
*/
readdir(directoryName: string): string[];
/**
* Writes a file into the store.
* @param artifactName The name of the file.
* @param data The contents of the file.
*/
writeFile(artifactName: string, data: any): void;
/**
* Writes a formatted JSON output file to the store
* @param artifactName the name of the artifact
* @param json the JSON object
*/
writeJson(artifactName: string, json: any): void;
/**
* Reads a file from the store.
* @param fileName The name of the file.
* @throws if the file is not found
*/
readFile(fileName: string): any;
/**
* Reads a JSON object from the store.
*/
readJson(fileName: string): any;
/**
* @returns true if the file `fileName` exists in the store.
* @param name The name of the file or directory to look up.
*/
exists(name: string): boolean;
/**
* List all top-level files that were emitted to the store.
*/
list(): string[];
/**
* Do not allow further writes into the store.
*/
lock(): void;
}
export interface FileSystemStoreOptions {
/**
* The output directory for synthesis artifacts
*/
readonly outdir: string;
}
/**
* Can be used to prepare and emit synthesis artifacts into an output directory.
*/
export declare class FileSystemStore implements ISessionStore {
private readonly outdir;
private locked;
constructor(options: FileSystemStoreOptions);
writeFile(fileName: string, data: any): void;
writeJson(fileName: string, json: any): void;
readFile(fileName: string): any;
readJson(fileName: string): any;
exists(name: string): boolean;
mkdir(directoryName: string): string;
readdir(directoryName: string): string[];
list(): string[];
lock(): void;
private pathForArtifact;
private canWrite;
}
export declare class InMemoryStore implements ISessionStore {
private files;
private dirs;
private locked;
writeFile(fileName: string, data: any): void;
writeJson(fileName: string, json: any): void;
readFile(fileName: string): any;
readJson(fileName: string): any;
exists(name: string): boolean;
mkdir(directoryName: string): string;
readdir(directoryName: string): string[];
list(): string[];
lock(): void;
private canWrite;
}
export declare function renderLegacyStacks(manifest: cxapi.AssemblyManifest, store: ISessionStore): cxapi.SynthesizedStack[];
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
const cxapi = require("@aws-cdk/cx-api");
const fs = require("fs");
const os = require("os");
const path = require("path");
const cx_api_1 = require("@aws-cdk/cx-api");
const construct_1 = require("./construct");
const runtime_info_1 = require("./runtime-info");
const util_1 = require("./util");
class Synthesizer {
synthesize(root, options = {}) {
const session = new SynthesisSession(options);
const session = new cx_api_1.CloudAssemblyBuilder(options.outdir);
// the three holy phases of synthesis: prepare, validate and synthesize

@@ -27,3 +22,3 @@ // prepare

for (const c of root.node.findAll(construct_1.ConstructOrder.PostOrder)) {
if (SynthesisSession.isSynthesizable(c)) {
if (isSynthesizable(c)) {
c.synthesize(session);

@@ -33,220 +28,12 @@ }

// write session manifest and lock store
session.close(options);
return session;
return session.build(options);
}
}
exports.Synthesizer = Synthesizer;
class SynthesisSession {
constructor(options) {
this.artifacts = {};
this.buildSteps = {};
this.store = options.store || new InMemoryStore();
}
/**
* @returns true if `obj` implements `ISynthesizable`.
*/
static isSynthesizable(obj) {
return 'synthesize' in obj;
}
get manifest() {
if (!this._manifest) {
throw new Error(`Cannot read assembly manifest before the session has been finalized`);
}
return this._manifest;
}
addArtifact(id, artifact) {
cxapi.validateArtifact(artifact);
this.artifacts[id] = util_1.filterUndefined(artifact);
}
tryGetArtifact(id) {
return this.artifacts[id];
}
getArtifact(id) {
const artifact = this.tryGetArtifact(id);
if (!artifact) {
throw new Error(`Cannot find artifact ${id}`);
}
return artifact;
}
addBuildStep(id, step) {
if (id in this.buildSteps) {
throw new Error(`Build step ${id} already exists`);
}
this.buildSteps[id] = util_1.filterUndefined(step);
}
close(options = {}) {
const legacyManifest = options.legacyManifest !== undefined ? options.legacyManifest : false;
const runtimeInfo = options.runtimeInformation !== undefined ? options.runtimeInformation : true;
const manifest = this._manifest = util_1.filterUndefined({
version: cxapi.PROTO_RESPONSE_VERSION,
artifacts: this.artifacts,
runtime: runtimeInfo ? runtime_info_1.collectRuntimeInformation() : undefined
});
this.store.writeFile(cxapi.MANIFEST_FILE, JSON.stringify(manifest, undefined, 2));
// write build manifest if we have build steps
if (Object.keys(this.buildSteps).length > 0) {
const buildManifest = {
steps: this.buildSteps
};
this.store.writeFile(cxapi.BUILD_FILE, JSON.stringify(buildManifest, undefined, 2));
}
if (legacyManifest) {
const legacy = {
...manifest,
stacks: renderLegacyStacks(manifest, this.store)
};
// render the legacy manifest (cdk.out) which also contains a "stacks" attribute with all the rendered stacks.
this.store.writeFile(cxapi.OUTFILE_NAME, JSON.stringify(legacy, undefined, 2));
}
this.store.lock();
return manifest;
}
}
exports.SynthesisSession = SynthesisSession;
/**
* Can be used to prepare and emit synthesis artifacts into an output directory.
* @returns true if `obj` implements `ISynthesizable`.
*/
class FileSystemStore {
constructor(options) {
this.locked = false;
this.outdir = options.outdir;
return;
}
writeFile(fileName, data) {
this.canWrite(fileName);
const p = this.pathForArtifact(fileName);
fs.writeFileSync(p, data);
}
writeJson(fileName, json) {
this.writeFile(fileName, JSON.stringify(json, undefined, 2));
}
readFile(fileName) {
const p = this.pathForArtifact(fileName);
if (!fs.existsSync(p)) {
throw new Error(`File not found: ${p}`);
}
return fs.readFileSync(p);
}
readJson(fileName) {
return JSON.parse(this.readFile(fileName).toString());
}
exists(name) {
const p = this.pathForArtifact(name);
return fs.existsSync(p);
}
mkdir(directoryName) {
this.canWrite(directoryName);
const p = this.pathForArtifact(directoryName);
fs.mkdirSync(p);
return p;
}
readdir(directoryName) {
if (!this.exists(directoryName)) {
throw new Error(`${directoryName} not found`);
}
const p = this.pathForArtifact(directoryName);
return fs.readdirSync(p);
}
list() {
return fs.readdirSync(this.outdir).sort();
}
lock() {
this.locked = true;
}
pathForArtifact(id) {
return path.join(this.outdir, id);
}
canWrite(artifactName) {
if (this.exists(artifactName)) {
throw new Error(`An artifact named ${artifactName} was already written to this session`);
}
if (this.locked) {
throw new Error('Session has already been finalized');
}
}
function isSynthesizable(obj) {
return 'synthesize' in obj;
}
exports.FileSystemStore = FileSystemStore;
class InMemoryStore {
constructor() {
this.files = {};
this.dirs = {}; // value is path to a temporary directory
this.locked = false;
}
writeFile(fileName, data) {
this.canWrite(fileName);
this.files[fileName] = data;
}
writeJson(fileName, json) {
this.writeFile(fileName, JSON.stringify(json, undefined, 2));
}
readFile(fileName) {
if (!(fileName in this.files)) {
throw new Error(`${fileName} not found`);
}
return this.files[fileName];
}
readJson(fileName) {
return JSON.parse(this.readFile(fileName).toString());
}
exists(name) {
return name in this.files || name in this.dirs;
}
mkdir(directoryName) {
this.canWrite(directoryName);
const p = fs.mkdtempSync(path.join(os.tmpdir(), directoryName));
this.dirs[directoryName] = p;
return p;
}
readdir(directoryName) {
if (!this.exists(directoryName)) {
throw new Error(`${directoryName} not found`);
}
const p = this.dirs[directoryName];
return fs.readdirSync(p);
}
list() {
return [...Object.keys(this.files), ...Object.keys(this.dirs)].sort();
}
lock() {
this.locked = true;
}
canWrite(artifactName) {
if (this.exists(artifactName)) {
throw new Error(`An artifact named ${artifactName} was already written to this session`);
}
if (this.locked) {
throw new Error('Session has already been finalized');
}
}
}
exports.InMemoryStore = InMemoryStore;
function renderLegacyStacks(manifest, store) {
// special case for backwards compat. build a list of stacks for the manifest
const stacks = new Array();
const artifacts = manifest.artifacts || {};
for (const [id, artifact] of Object.entries(artifacts)) {
if (artifact.type === cxapi.ArtifactType.AwsCloudFormationStack) {
const templateFile = artifact.properties && artifact.properties.templateFile;
if (!templateFile) {
throw new Error(`Invalid cloudformation artifact. Missing "template" property`);
}
const template = store.readJson(templateFile);
const match = cxapi.AWS_ENV_REGEX.exec(artifact.environment);
if (!match) {
throw new Error(`"environment" must match regex: ${cxapi.AWS_ENV_REGEX}`);
}
stacks.push(util_1.filterUndefined({
name: id,
environment: { name: artifact.environment.substr('aws://'.length), account: match[1], region: match[2] },
template,
metadata: artifact.metadata || {},
autoDeploy: artifact.autoDeploy,
dependsOn: artifact.dependencies && artifact.dependencies.length > 0 ? artifact.dependencies : undefined,
missing: artifact.missing
}));
}
}
return stacks;
}
exports.renderLegacyStacks = renderLegacyStacks;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"synthesis.js","sourceRoot":"","sources":["synthesis.ts"],"names":[],"mappings":"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","sourcesContent":["import cxapi = require('@aws-cdk/cx-api');\nimport fs = require('fs');\nimport os = require('os');\nimport path = require('path');\nimport { ConstructOrder, IConstruct } from './construct';\nimport { collectRuntimeInformation } from './runtime-info';\nimport { filterUndefined } from './util';\n\nexport interface ISynthesizable {\n  synthesize(session: ISynthesisSession): void;\n}\n\nexport interface ISynthesisSession {\n  readonly store: ISessionStore;\n  readonly manifest: cxapi.AssemblyManifest;\n  addArtifact(id: string, droplet: cxapi.Artifact): void;\n  addBuildStep(id: string, step: cxapi.BuildStep): void;\n  tryGetArtifact(id: string): cxapi.Artifact | undefined;\n  getArtifact(id: string): cxapi.Artifact;\n}\n\nexport interface SynthesisOptions extends ManifestOptions {\n  /**\n   * The file store used for this session.\n   * @default InMemoryStore\n   */\n  readonly store?: ISessionStore;\n\n  /**\n   * Whether synthesis should skip the validation phase.\n   * @default false\n   */\n  readonly skipValidation?: boolean;\n}\n\nexport class Synthesizer {\n  public synthesize(root: IConstruct, options: SynthesisOptions = { }): ISynthesisSession {\n    const session = new SynthesisSession(options);\n\n    // the three holy phases of synthesis: prepare, validate and synthesize\n\n    // prepare\n    root.node.prepareTree();\n\n    // validate\n    const validate = options.skipValidation === undefined ? true : !options.skipValidation;\n    if (validate) {\n      const errors = root.node.validateTree();\n      if (errors.length > 0) {\n        const errorList = errors.map(e => `[${e.source.node.path}] ${e.message}`).join('\\n  ');\n        throw new Error(`Validation failed with the following errors:\\n  ${errorList}`);\n      }\n    }\n\n    // synthesize (leaves first)\n    for (const c of root.node.findAll(ConstructOrder.PostOrder)) {\n      if (SynthesisSession.isSynthesizable(c)) {\n        c.synthesize(session);\n      }\n    }\n\n    // write session manifest and lock store\n    session.close(options);\n\n    return session;\n  }\n}\n\nexport class SynthesisSession implements ISynthesisSession {\n  /**\n   * @returns true if `obj` implements `ISynthesizable`.\n   */\n  public static isSynthesizable(obj: any): obj is ISynthesizable {\n    return 'synthesize' in obj;\n  }\n\n  public readonly store: ISessionStore;\n\n  private readonly artifacts: { [id: string]: cxapi.Artifact } = { };\n  private readonly buildSteps: { [id: string]: cxapi.BuildStep } = { };\n  private _manifest?: cxapi.AssemblyManifest;\n\n  constructor(options: SynthesisOptions) {\n    this.store = options.store || new InMemoryStore();\n  }\n\n  public get manifest() {\n    if (!this._manifest) {\n      throw new Error(`Cannot read assembly manifest before the session has been finalized`);\n    }\n\n    return this._manifest;\n  }\n\n  public addArtifact(id: string, artifact: cxapi.Artifact): void {\n    cxapi.validateArtifact(artifact);\n    this.artifacts[id] = filterUndefined(artifact);\n  }\n\n  public tryGetArtifact(id: string): cxapi.Artifact | undefined {\n    return this.artifacts[id];\n  }\n\n  public getArtifact(id: string): cxapi.Artifact {\n    const artifact = this.tryGetArtifact(id);\n    if (!artifact) {\n      throw new Error(`Cannot find artifact ${id}`);\n    }\n    return artifact;\n  }\n\n  public addBuildStep(id: string, step: cxapi.BuildStep) {\n    if (id in this.buildSteps) {\n      throw new Error(`Build step ${id} already exists`);\n    }\n    this.buildSteps[id] = filterUndefined(step);\n  }\n\n  public close(options: ManifestOptions = { }): cxapi.AssemblyManifest {\n    const legacyManifest = options.legacyManifest !== undefined ? options.legacyManifest : false;\n    const runtimeInfo = options.runtimeInformation !== undefined ? options.runtimeInformation : true;\n\n    const manifest: cxapi.AssemblyManifest = this._manifest = filterUndefined({\n      version: cxapi.PROTO_RESPONSE_VERSION,\n      artifacts: this.artifacts,\n      runtime: runtimeInfo ? collectRuntimeInformation() : undefined\n    });\n\n    this.store.writeFile(cxapi.MANIFEST_FILE, JSON.stringify(manifest, undefined, 2));\n\n    // write build manifest if we have build steps\n    if (Object.keys(this.buildSteps).length > 0) {\n      const buildManifest: cxapi.BuildManifest = {\n        steps: this.buildSteps\n      };\n\n      this.store.writeFile(cxapi.BUILD_FILE, JSON.stringify(buildManifest, undefined, 2));\n    }\n\n    if (legacyManifest) {\n      const legacy: cxapi.SynthesizeResponse = {\n        ...manifest,\n        stacks: renderLegacyStacks(manifest, this.store)\n      };\n\n      // render the legacy manifest (cdk.out) which also contains a \"stacks\" attribute with all the rendered stacks.\n      this.store.writeFile(cxapi.OUTFILE_NAME, JSON.stringify(legacy, undefined, 2));\n    }\n\n    this.store.lock();\n\n    return manifest;\n  }\n}\n\nexport interface ManifestOptions {\n  /**\n   * Emit the legacy manifest (`cdk.out`) when the session is closed (alongside `manifest.json`).\n   * @default false\n   */\n  readonly legacyManifest?: boolean;\n\n  /**\n   * Include runtime information (module versions) in manifest.\n   * @default true\n   */\n  readonly runtimeInformation?: boolean;\n}\n\nexport interface ISessionStore {\n  /**\n   * Creates a directory and returns it's full path.\n   * @param directoryName The name of the directory to create.\n   * @throws if a directory by that name already exists in the session or if the session has already been finalized.\n   */\n  mkdir(directoryName: string): string;\n\n  /**\n   * Returns the list of files in a directory.\n   * @param directoryName The name of the artifact\n   * @throws if there is no directory artifact under this name\n   */\n  readdir(directoryName: string): string[];\n\n  /**\n   * Writes a file into the store.\n   * @param artifactName The name of the file.\n   * @param data The contents of the file.\n   */\n  writeFile(artifactName: string, data: any): void;\n\n  /**\n   * Writes a formatted JSON output file to the store\n   * @param artifactName the name of the artifact\n   * @param json the JSON object\n   */\n  writeJson(artifactName: string, json: any): void;\n\n  /**\n   * Reads a file from the store.\n   * @param fileName The name of the file.\n   * @throws if the file is not found\n   */\n  readFile(fileName: string): any;\n\n  /**\n   * Reads a JSON object from the store.\n   */\n  readJson(fileName: string): any;\n\n  /**\n   * @returns true if the file `fileName` exists in the store.\n   * @param name The name of the file or directory to look up.\n   */\n  exists(name: string): boolean;\n\n  /**\n   * List all top-level files that were emitted to the store.\n   */\n  list(): string[];\n\n  /**\n   * Do not allow further writes into the store.\n   */\n  lock(): void;\n}\n\nexport interface FileSystemStoreOptions {\n  /**\n   * The output directory for synthesis artifacts\n   */\n  readonly outdir: string;\n}\n\n/**\n * Can be used to prepare and emit synthesis artifacts into an output directory.\n */\nexport class FileSystemStore implements ISessionStore {\n  private readonly outdir: string;\n  private locked = false;\n\n  constructor(options: FileSystemStoreOptions) {\n    this.outdir = options.outdir;\n    return;\n  }\n\n  public writeFile(fileName: string, data: any) {\n    this.canWrite(fileName);\n\n    const p = this.pathForArtifact(fileName);\n    fs.writeFileSync(p, data);\n  }\n\n  public writeJson(fileName: string, json: any) {\n    this.writeFile(fileName, JSON.stringify(json, undefined, 2));\n  }\n\n  public readFile(fileName: string): any {\n    const p = this.pathForArtifact(fileName);\n    if (!fs.existsSync(p)) {\n      throw new Error(`File not found: ${p}`);\n    }\n\n    return fs.readFileSync(p);\n  }\n\n  public readJson(fileName: string): any {\n    return JSON.parse(this.readFile(fileName).toString());\n  }\n\n  public exists(name: string): boolean {\n    const p = this.pathForArtifact(name);\n    return fs.existsSync(p);\n  }\n\n  public mkdir(directoryName: string): string {\n    this.canWrite(directoryName);\n    const p = this.pathForArtifact(directoryName);\n    fs.mkdirSync(p);\n    return p;\n  }\n\n  public readdir(directoryName: string): string[] {\n    if (!this.exists(directoryName)) {\n      throw new Error(`${directoryName} not found`);\n    }\n\n    const p = this.pathForArtifact(directoryName);\n    return fs.readdirSync(p);\n  }\n\n  public list(): string[] {\n    return fs.readdirSync(this.outdir).sort();\n  }\n\n  public lock() {\n    this.locked = true;\n  }\n\n  private pathForArtifact(id: string) {\n    return path.join(this.outdir, id);\n  }\n\n  private canWrite(artifactName: string) {\n    if (this.exists(artifactName)) {\n      throw new Error(`An artifact named ${artifactName} was already written to this session`);\n    }\n    if (this.locked) {\n      throw new Error('Session has already been finalized');\n    }\n  }\n}\n\nexport class InMemoryStore implements ISessionStore {\n  private files: { [fileName: string]: any } = { };\n  private dirs: { [dirName: string]: string } = { }; // value is path to a temporary directory\n\n  private locked = false;\n\n  public writeFile(fileName: string, data: any): void {\n    this.canWrite(fileName);\n    this.files[fileName] = data;\n  }\n\n  public writeJson(fileName: string, json: any): void {\n    this.writeFile(fileName, JSON.stringify(json, undefined, 2));\n  }\n\n  public readFile(fileName: string) {\n    if (!(fileName in this.files)) {\n      throw new Error(`${fileName} not found`);\n    }\n    return this.files[fileName];\n  }\n\n  public readJson(fileName: string): any {\n    return JSON.parse(this.readFile(fileName).toString());\n  }\n\n  public exists(name: string) {\n    return name in this.files || name in this.dirs;\n  }\n\n  public mkdir(directoryName: string): string {\n    this.canWrite(directoryName);\n\n    const p = fs.mkdtempSync(path.join(os.tmpdir(), directoryName));\n    this.dirs[directoryName] = p;\n    return p;\n  }\n\n  public readdir(directoryName: string): string[] {\n    if (!this.exists(directoryName)) {\n      throw new Error(`${directoryName} not found`);\n    }\n\n    const p = this.dirs[directoryName];\n    return fs.readdirSync(p);\n  }\n\n  public list(): string[] {\n    return [ ...Object.keys(this.files), ...Object.keys(this.dirs) ].sort();\n  }\n\n  public lock() {\n    this.locked = true;\n  }\n\n  private canWrite(artifactName: string) {\n    if (this.exists(artifactName)) {\n      throw new Error(`An artifact named ${artifactName} was already written to this session`);\n    }\n    if (this.locked) {\n      throw new Error('Session has already been finalized');\n    }\n  }\n}\n\nexport function renderLegacyStacks(manifest: cxapi.AssemblyManifest, store: ISessionStore) {\n  // special case for backwards compat. build a list of stacks for the manifest\n  const stacks = new Array<cxapi.SynthesizedStack>();\n  const artifacts = manifest.artifacts || { };\n\n  for (const [ id, artifact ] of Object.entries(artifacts)) {\n    if (artifact.type === cxapi.ArtifactType.AwsCloudFormationStack) {\n      const templateFile = artifact.properties && artifact.properties.templateFile;\n      if (!templateFile) {\n        throw new Error(`Invalid cloudformation artifact. Missing \"template\" property`);\n      }\n      const template = store.readJson(templateFile);\n\n      const match = cxapi.AWS_ENV_REGEX.exec(artifact.environment);\n      if (!match) {\n        throw new Error(`\"environment\" must match regex: ${cxapi.AWS_ENV_REGEX}`);\n      }\n\n      stacks.push(filterUndefined({\n        name: id,\n        environment: { name: artifact.environment.substr('aws://'.length), account: match[1], region: match[2] },\n        template,\n        metadata: artifact.metadata || {},\n        autoDeploy: artifact.autoDeploy,\n        dependsOn: artifact.dependencies && artifact.dependencies.length > 0 ? artifact.dependencies : undefined,\n        missing: artifact.missing\n      }));\n    }\n  }\n\n  return stacks;\n}\n"]}
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3ludGhlc2lzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsic3ludGhlc2lzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7O0FBQUEsNENBQW9GO0FBQ3BGLDJDQUF5RDtBQW9CekQsTUFBYSxXQUFXO0lBQ2YsVUFBVSxDQUFDLElBQWdCLEVBQUUsVUFBNEIsRUFBRztRQUNqRSxNQUFNLE9BQU8sR0FBRyxJQUFJLDZCQUFvQixDQUFDLE9BQU8sQ0FBQyxNQUFNLENBQUMsQ0FBQztRQUV6RCx1RUFBdUU7UUFFdkUsVUFBVTtRQUNWLElBQUksQ0FBQyxJQUFJLENBQUMsV0FBVyxFQUFFLENBQUM7UUFFeEIsV0FBVztRQUNYLE1BQU0sUUFBUSxHQUFHLE9BQU8sQ0FBQyxjQUFjLEtBQUssU0FBUyxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLENBQUMsT0FBTyxDQUFDLGNBQWMsQ0FBQztRQUN2RixJQUFJLFFBQVEsRUFBRTtZQUNaLE1BQU0sTUFBTSxHQUFHLElBQUksQ0FBQyxJQUFJLENBQUMsWUFBWSxFQUFFLENBQUM7WUFDeEMsSUFBSSxNQUFNLENBQUMsTUFBTSxHQUFHLENBQUMsRUFBRTtnQkFDckIsTUFBTSxTQUFTLEdBQUcsTUFBTSxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDLElBQUksQ0FBQyxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsSUFBSSxLQUFLLENBQUMsQ0FBQyxPQUFPLEVBQUUsQ0FBQyxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsQ0FBQztnQkFDdkYsTUFBTSxJQUFJLEtBQUssQ0FBQyxtREFBbUQsU0FBUyxFQUFFLENBQUMsQ0FBQzthQUNqRjtTQUNGO1FBRUQsNEJBQTRCO1FBQzVCLEtBQUssTUFBTSxDQUFDLElBQUksSUFBSSxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsMEJBQWMsQ0FBQyxTQUFTLENBQUMsRUFBRTtZQUMzRCxJQUFJLGVBQWUsQ0FBQyxDQUFDLENBQUMsRUFBRTtnQkFDdEIsQ0FBQyxDQUFDLFVBQVUsQ0FBQyxPQUFPLENBQUMsQ0FBQzthQUN2QjtTQUNGO1FBRUQsd0NBQXdDO1FBQ3hDLE9BQU8sT0FBTyxDQUFDLEtBQUssQ0FBQyxPQUFPLENBQUMsQ0FBQztJQUNoQyxDQUFDO0NBQ0Y7QUE3QkQsa0NBNkJDO0FBRUQ7O0dBRUc7QUFDSCxTQUFTLGVBQWUsQ0FBQyxHQUFRO0lBQy9CLE9BQU8sWUFBWSxJQUFJLEdBQUcsQ0FBQztBQUM3QixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQnVpbGRPcHRpb25zLCBDbG91ZEFzc2VtYmx5LCBDbG91ZEFzc2VtYmx5QnVpbGRlciB9IGZyb20gJ0Bhd3MtY2RrL2N4LWFwaSc7XG5pbXBvcnQgeyBDb25zdHJ1Y3RPcmRlciwgSUNvbnN0cnVjdCB9IGZyb20gJy4vY29uc3RydWN0JztcblxuZXhwb3J0IGludGVyZmFjZSBJU3ludGhlc2l6YWJsZSB7XG4gIHN5bnRoZXNpemUoc2Vzc2lvbjogQ2xvdWRBc3NlbWJseUJ1aWxkZXIpOiB2b2lkO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFN5bnRoZXNpc09wdGlvbnMgZXh0ZW5kcyBCdWlsZE9wdGlvbnMge1xuICAvKipcbiAgICogVGhlIGZpbGUgc3RvcmUgdXNlZCBmb3IgdGhpcyBzZXNzaW9uLlxuICAgKiBAZGVmYXVsdCAtIGNyZWF0ZXMgYSB0ZW1wb3JhcnkgZGlyZWN0b3J5XG4gICAqL1xuICByZWFkb25seSBvdXRkaXI/OiBzdHJpbmc7XG5cbiAgLyoqXG4gICAqIFdoZXRoZXIgc3ludGhlc2lzIHNob3VsZCBza2lwIHRoZSB2YWxpZGF0aW9uIHBoYXNlLlxuICAgKiBAZGVmYXVsdCBmYWxzZVxuICAgKi9cbiAgcmVhZG9ubHkgc2tpcFZhbGlkYXRpb24/OiBib29sZWFuO1xufVxuXG5leHBvcnQgY2xhc3MgU3ludGhlc2l6ZXIge1xuICBwdWJsaWMgc3ludGhlc2l6ZShyb290OiBJQ29uc3RydWN0LCBvcHRpb25zOiBTeW50aGVzaXNPcHRpb25zID0geyB9KTogQ2xvdWRBc3NlbWJseSB7XG4gICAgY29uc3Qgc2Vzc2lvbiA9IG5ldyBDbG91ZEFzc2VtYmx5QnVpbGRlcihvcHRpb25zLm91dGRpcik7XG5cbiAgICAvLyB0aGUgdGhyZWUgaG9seSBwaGFzZXMgb2Ygc3ludGhlc2lzOiBwcmVwYXJlLCB2YWxpZGF0ZSBhbmQgc3ludGhlc2l6ZVxuXG4gICAgLy8gcHJlcGFyZVxuICAgIHJvb3Qubm9kZS5wcmVwYXJlVHJlZSgpO1xuXG4gICAgLy8gdmFsaWRhdGVcbiAgICBjb25zdCB2YWxpZGF0ZSA9IG9wdGlvbnMuc2tpcFZhbGlkYXRpb24gPT09IHVuZGVmaW5lZCA/IHRydWUgOiAhb3B0aW9ucy5za2lwVmFsaWRhdGlvbjtcbiAgICBpZiAodmFsaWRhdGUpIHtcbiAgICAgIGNvbnN0IGVycm9ycyA9IHJvb3Qubm9kZS52YWxpZGF0ZVRyZWUoKTtcbiAgICAgIGlmIChlcnJvcnMubGVuZ3RoID4gMCkge1xuICAgICAgICBjb25zdCBlcnJvckxpc3QgPSBlcnJvcnMubWFwKGUgPT4gYFske2Uuc291cmNlLm5vZGUucGF0aH1dICR7ZS5tZXNzYWdlfWApLmpvaW4oJ1xcbiAgJyk7XG4gICAgICAgIHRocm93IG5ldyBFcnJvcihgVmFsaWRhdGlvbiBmYWlsZWQgd2l0aCB0aGUgZm9sbG93aW5nIGVycm9yczpcXG4gICR7ZXJyb3JMaXN0fWApO1xuICAgICAgfVxuICAgIH1cblxuICAgIC8vIHN5bnRoZXNpemUgKGxlYXZlcyBmaXJzdClcbiAgICBmb3IgKGNvbnN0IGMgb2Ygcm9vdC5ub2RlLmZpbmRBbGwoQ29uc3RydWN0T3JkZXIuUG9zdE9yZGVyKSkge1xuICAgICAgaWYgKGlzU3ludGhlc2l6YWJsZShjKSkge1xuICAgICAgICBjLnN5bnRoZXNpemUoc2Vzc2lvbik7XG4gICAgICB9XG4gICAgfVxuXG4gICAgLy8gd3JpdGUgc2Vzc2lvbiBtYW5pZmVzdCBhbmQgbG9jayBzdG9yZVxuICAgIHJldHVybiBzZXNzaW9uLmJ1aWxkKG9wdGlvbnMpO1xuICB9XG59XG5cbi8qKlxuICogQHJldHVybnMgdHJ1ZSBpZiBgb2JqYCBpbXBsZW1lbnRzIGBJU3ludGhlc2l6YWJsZWAuXG4gKi9cbmZ1bmN0aW9uIGlzU3ludGhlc2l6YWJsZShvYmo6IGFueSk6IG9iaiBpcyBJU3ludGhlc2l6YWJsZSB7XG4gIHJldHVybiAnc3ludGhlc2l6ZScgaW4gb2JqO1xufVxuIl19

@@ -31,2 +31,6 @@ import { IConstruct } from "./construct";

static isToken(obj: any): boolean;
/**
* The captured stack trace which represents the location in which this token was created.
*/
protected readonly trace: string[];
private tokenStringification?;

@@ -101,2 +105,7 @@ private tokenListification?;

toNumber(): number;
/**
* Creates a throwable Error object that contains the token creation stack trace.
* @param message Error message
*/
protected newError(message: string): any;
}

@@ -103,0 +112,0 @@ /**

"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
const encoding_1 = require("./encoding");
const stack_trace_1 = require("./stack-trace");
const token_map_1 = require("./token-map");

@@ -43,2 +44,3 @@ /**

this.displayName = displayName;
this.trace = stack_trace_1.createStackTrace();
}

@@ -126,3 +128,3 @@ /**

if (valueType === 'string' || valueType === 'number' || valueType === 'boolean') {
throw new Error('Got a literal Token value; only intrinsics can ever evaluate to lists.');
throw this.newError('Got a literal Token value; only intrinsics can ever evaluate to lists.');
}

@@ -152,3 +154,3 @@ if (this.tokenListification === undefined) {

if (valueType !== 'function') {
throw new Error(`Token value is not number or lazy, can't represent as number: ${this.valueOrFunction}`);
throw this.newError(`Token value is not number or lazy, can't represent as number: ${this.valueOrFunction}`);
}

@@ -159,2 +161,9 @@ this.tokenNumberification = token_map_1.TokenMap.instance().registerNumber(this);

}
/**
* Creates a throwable Error object that contains the token creation stack trace.
* @param message Error message
*/
newError(message) {
return new Error(`${message}\nToken created:\n at ${this.trace.join('\n at ')}\nError thrown:`);
}
}

@@ -169,2 +178,2 @@ exports.Token = Token;

exports.isResolvedValuePostProcessor = isResolvedValuePostProcessor;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"token.js","sourceRoot":"","sources":["token.ts"],"names":[],"mappings":"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","sourcesContent":["import { IConstruct } from \"./construct\";\nimport { unresolved } from \"./encoding\";\nimport { TokenMap } from \"./token-map\";\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  /**\n   * @deprecated use `Token.isToken`\n   */\n  public static unresolved(obj: any): boolean {\n    return unresolved(obj);\n  }\n\n  /**\n   * Returns true if obj is a token (i.e. has the resolve() method or is a\n   * string or array which includes token markers).\n   *\n   * @param obj The object to test.\n   */\n  public static isToken(obj: any): boolean {\n    return unresolved(obj);\n  }\n\n  private tokenStringification?: string;\n  private tokenListification?: string[];\n  private tokenNumberification?: number;\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(context: IResolveContext): any {\n    let value = this.valueOrFunction;\n    if (typeof(value) === 'function') {\n      value = value(context);\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.tokenStringification === undefined) {\n      this.tokenStringification = TokenMap.instance().registerString(this, this.displayName);\n    }\n    return this.tokenStringification;\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    // We can't do the right work here because in case we contain a function, we\n    // won't know the type of value that function represents (in the simplest\n    // case, string or number), and we can't know that without an\n    // IResolveContext to actually do the resolution, which we don't have.\n\n    // We used to throw an error, but since JSON.stringify() is often used in\n    // error messages to produce a readable representation of an object, if we\n    // throw here we'll obfuscate that descriptive error with something worse.\n    // So return a string representation that indicates this thing is a token\n    // and needs resolving.\n    return JSON.stringify(`<unresolved-token:${this.displayName || 'TOKEN'}>`);\n  }\n\n  /**\n   * Return a string list representation of this token\n   *\n   * Call this if the Token intrinsically evaluates to a list of strings.\n   * If so, you can represent the Token in a similar way in the type\n   * system.\n   *\n   * Note that even though the Token is represented as a list of strings, you\n   * still cannot do any operations on it such as concatenation, indexing,\n   * or taking its length. The only useful operations you can do to these lists\n   * is constructing a `FnJoin` or a `FnSelect` on it.\n   */\n  public toList(): string[] {\n    const valueType = typeof this.valueOrFunction;\n    if (valueType === 'string' || valueType === 'number' || valueType === 'boolean') {\n      throw new Error('Got a literal Token value; only intrinsics can ever evaluate to lists.');\n    }\n\n    if (this.tokenListification === undefined) {\n      this.tokenListification = TokenMap.instance().registerList(this, this.displayName);\n    }\n    return this.tokenListification;\n  }\n\n  /**\n   * Return a floating point representation of this Token\n   *\n   * Call this if the Token intrinsically resolves to something that represents\n   * a number, and you need to pass it into an API that expects a number.\n   *\n   * You may not do any operations on the returned value; any arithmetic or\n   * other operations can and probably will destroy the token-ness of the value.\n   */\n  public toNumber(): number {\n    if (this.tokenNumberification === undefined) {\n      const valueType = typeof this.valueOrFunction;\n      // Optimization: if we can immediately resolve this, don't bother\n      // registering a Token.\n      if (valueType === 'number') { return this.valueOrFunction; }\n      if (valueType !== 'function') {\n        throw new Error(`Token value is not number or lazy, can't represent as number: ${this.valueOrFunction}`);\n      }\n      this.tokenNumberification = TokenMap.instance().registerNumber(this);\n    }\n\n    return this.tokenNumberification;\n  }\n}\n\n/**\n * Current resolution context for tokens\n */\nexport interface IResolveContext {\n  /**\n   * The scope from which resolution has been initiated\n   */\n  readonly scope: IConstruct;\n\n  /**\n   * Resolve an inner object\n   */\n  resolve(x: any): any;\n}\n\n/**\n * A Token that can post-process the complete resolved value, after resolve() has recursed over it\n */\nexport interface IResolvedValuePostProcessor {\n  /**\n   * Process the completely resolved value, after full recursion/resolution has happened\n   */\n  postProcess(input: any, context: IResolveContext): any;\n}\n\n/**\n * Whether the given object is an `IResolvedValuePostProcessor`\n */\nexport function isResolvedValuePostProcessor(x: any): x is IResolvedValuePostProcessor {\n  return x.postProcess !== undefined;\n}"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"token.js","sourceRoot":"","sources":["token.ts"],"names":[],"mappings":";;AACA,yCAAwC;AACxC,+CAAiD;AACjD,2CAAuC;AAEvC;;;GAGG;AACU,QAAA,cAAc,GAAG,SAAS,CAAC;AAExC;;;;;;;;;GASG;AACH,MAAa,KAAK;IA2BhB;;;;;;;;;;;;;;;;;;OAkBG;IACH,YAA6B,eAAqB,EAAmB,WAAoB;QAA5D,oBAAe,GAAf,eAAe,CAAM;QAAmB,gBAAW,GAAX,WAAW,CAAS;QACvF,IAAI,CAAC,KAAK,GAAG,8BAAgB,EAAE,CAAC;IAClC,CAAC;IA/CD;;OAEG;IACI,MAAM,CAAC,UAAU,CAAC,GAAQ;QAC/B,OAAO,qBAAU,CAAC,GAAG,CAAC,CAAC;IACzB,CAAC;IAED;;;;;OAKG;IACI,MAAM,CAAC,OAAO,CAAC,GAAQ;QAC5B,OAAO,qBAAU,CAAC,GAAG,CAAC,CAAC;IACzB,CAAC;IAkCD;;OAEG;IACI,OAAO,CAAC,OAAwB;QACrC,IAAI,KAAK,GAAG,IAAI,CAAC,eAAe,CAAC;QACjC,IAAI,OAAM,CAAC,KAAK,CAAC,KAAK,UAAU,EAAE;YAChC,KAAK,GAAG,KAAK,CAAC,OAAO,CAAC,CAAC;SACxB;QAED,OAAO,KAAK,CAAC;IACf,CAAC;IAED;;;;;;;;;;OAUG;IACI,QAAQ;QACb,MAAM,SAAS,GAAG,OAAO,IAAI,CAAC,eAAe,CAAC;QAC9C,iEAAiE;QACjE,uBAAuB;QACvB,IAAI,SAAS,KAAK,QAAQ,IAAI,SAAS,KAAK,QAAQ,IAAI,SAAS,KAAK,SAAS,EAAE;YAC/E,OAAO,IAAI,CAAC,eAAe,CAAC,QAAQ,EAAE,CAAC;SACxC;QAED,IAAI,IAAI,CAAC,oBAAoB,KAAK,SAAS,EAAE;YAC3C,IAAI,CAAC,oBAAoB,GAAG,oBAAQ,CAAC,QAAQ,EAAE,CAAC,cAAc,CAAC,IAAI,EAAE,IAAI,CAAC,WAAW,CAAC,CAAC;SACxF;QACD,OAAO,IAAI,CAAC,oBAAoB,CAAC;IACnC,CAAC;IAED;;;;;OAKG;IACI,MAAM;QACX,4EAA4E;QAC5E,yEAAyE;QACzE,6DAA6D;QAC7D,sEAAsE;QAEtE,yEAAyE;QACzE,0EAA0E;QAC1E,0EAA0E;QAC1E,yEAAyE;QACzE,uBAAuB;QACvB,OAAO,IAAI,CAAC,SAAS,CAAC,qBAAqB,IAAI,CAAC,WAAW,IAAI,OAAO,GAAG,CAAC,CAAC;IAC7E,CAAC;IAED;;;;;;;;;;;OAWG;IACI,MAAM;QACX,MAAM,SAAS,GAAG,OAAO,IAAI,CAAC,eAAe,CAAC;QAC9C,IAAI,SAAS,KAAK,QAAQ,IAAI,SAAS,KAAK,QAAQ,IAAI,SAAS,KAAK,SAAS,EAAE;YAC/E,MAAM,IAAI,CAAC,QAAQ,CAAC,wEAAwE,CAAC,CAAC;SAC/F;QAED,IAAI,IAAI,CAAC,kBAAkB,KAAK,SAAS,EAAE;YACzC,IAAI,CAAC,kBAAkB,GAAG,oBAAQ,CAAC,QAAQ,EAAE,CAAC,YAAY,CAAC,IAAI,EAAE,IAAI,CAAC,WAAW,CAAC,CAAC;SACpF;QACD,OAAO,IAAI,CAAC,kBAAkB,CAAC;IACjC,CAAC;IAED;;;;;;;;OAQG;IACI,QAAQ;QACb,IAAI,IAAI,CAAC,oBAAoB,KAAK,SAAS,EAAE;YAC3C,MAAM,SAAS,GAAG,OAAO,IAAI,CAAC,eAAe,CAAC;YAC9C,iEAAiE;YACjE,uBAAuB;YACvB,IAAI,SAAS,KAAK,QAAQ,EAAE;gBAAE,OAAO,IAAI,CAAC,eAAe,CAAC;aAAE;YAC5D,IAAI,SAAS,KAAK,UAAU,EAAE;gBAC5B,MAAM,IAAI,CAAC,QAAQ,CAAC,iEAAiE,IAAI,CAAC,eAAe,EAAE,CAAC,CAAC;aAC9G;YACD,IAAI,CAAC,oBAAoB,GAAG,oBAAQ,CAAC,QAAQ,EAAE,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC;SACtE;QAED,OAAO,IAAI,CAAC,oBAAoB,CAAC;IACnC,CAAC;IAED;;;OAGG;IACO,QAAQ,CAAC,OAAe;QAChC,OAAO,IAAI,KAAK,CAAC,GAAG,OAAO,4BAA4B,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,WAAW,CAAC,iBAAiB,CAAC,CAAC;IACxG,CAAC;CACF;AAlKD,sBAkKC;AA2BD;;GAEG;AACH,SAAgB,4BAA4B,CAAC,CAAM;IACjD,OAAO,CAAC,CAAC,WAAW,KAAK,SAAS,CAAC;AACrC,CAAC;AAFD,oEAEC","sourcesContent":["import { IConstruct } from \"./construct\";\nimport { unresolved } from \"./encoding\";\nimport { createStackTrace } from './stack-trace';\nimport { TokenMap } from \"./token-map\";\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  /**\n   * @deprecated use `Token.isToken`\n   */\n  public static unresolved(obj: any): boolean {\n    return unresolved(obj);\n  }\n\n  /**\n   * Returns true if obj is a token (i.e. has the resolve() method or is a\n   * string or array which includes token markers).\n   *\n   * @param obj The object to test.\n   */\n  public static isToken(obj: any): boolean {\n    return unresolved(obj);\n  }\n\n  /**\n   * The captured stack trace which represents the location in which this token was created.\n   */\n  protected readonly trace: string[];\n\n  private tokenStringification?: string;\n  private tokenListification?: string[];\n  private tokenNumberification?: number;\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    this.trace = createStackTrace();\n  }\n\n  /**\n   * @returns The resolved value for this token.\n   */\n  public resolve(context: IResolveContext): any {\n    let value = this.valueOrFunction;\n    if (typeof(value) === 'function') {\n      value = value(context);\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.tokenStringification === undefined) {\n      this.tokenStringification = TokenMap.instance().registerString(this, this.displayName);\n    }\n    return this.tokenStringification;\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    // We can't do the right work here because in case we contain a function, we\n    // won't know the type of value that function represents (in the simplest\n    // case, string or number), and we can't know that without an\n    // IResolveContext to actually do the resolution, which we don't have.\n\n    // We used to throw an error, but since JSON.stringify() is often used in\n    // error messages to produce a readable representation of an object, if we\n    // throw here we'll obfuscate that descriptive error with something worse.\n    // So return a string representation that indicates this thing is a token\n    // and needs resolving.\n    return JSON.stringify(`<unresolved-token:${this.displayName || 'TOKEN'}>`);\n  }\n\n  /**\n   * Return a string list representation of this token\n   *\n   * Call this if the Token intrinsically evaluates to a list of strings.\n   * If so, you can represent the Token in a similar way in the type\n   * system.\n   *\n   * Note that even though the Token is represented as a list of strings, you\n   * still cannot do any operations on it such as concatenation, indexing,\n   * or taking its length. The only useful operations you can do to these lists\n   * is constructing a `FnJoin` or a `FnSelect` on it.\n   */\n  public toList(): string[] {\n    const valueType = typeof this.valueOrFunction;\n    if (valueType === 'string' || valueType === 'number' || valueType === 'boolean') {\n      throw this.newError('Got a literal Token value; only intrinsics can ever evaluate to lists.');\n    }\n\n    if (this.tokenListification === undefined) {\n      this.tokenListification = TokenMap.instance().registerList(this, this.displayName);\n    }\n    return this.tokenListification;\n  }\n\n  /**\n   * Return a floating point representation of this Token\n   *\n   * Call this if the Token intrinsically resolves to something that represents\n   * a number, and you need to pass it into an API that expects a number.\n   *\n   * You may not do any operations on the returned value; any arithmetic or\n   * other operations can and probably will destroy the token-ness of the value.\n   */\n  public toNumber(): number {\n    if (this.tokenNumberification === undefined) {\n      const valueType = typeof this.valueOrFunction;\n      // Optimization: if we can immediately resolve this, don't bother\n      // registering a Token.\n      if (valueType === 'number') { return this.valueOrFunction; }\n      if (valueType !== 'function') {\n        throw this.newError(`Token value is not number or lazy, can't represent as number: ${this.valueOrFunction}`);\n      }\n      this.tokenNumberification = TokenMap.instance().registerNumber(this);\n    }\n\n    return this.tokenNumberification;\n  }\n\n  /**\n   * Creates a throwable Error object that contains the token creation stack trace.\n   * @param message Error message\n   */\n  protected newError(message: string): any {\n    return new Error(`${message}\\nToken created:\\n    at ${this.trace.join('\\n    at ')}\\nError thrown:`);\n  }\n}\n\n/**\n * Current resolution context for tokens\n */\nexport interface IResolveContext {\n  /**\n   * The scope from which resolution has been initiated\n   */\n  readonly scope: IConstruct;\n\n  /**\n   * Resolve an inner object\n   */\n  resolve(x: any): any;\n}\n\n/**\n * A Token that can post-process the complete resolved value, after resolve() has recursed over it\n */\nexport interface IResolvedValuePostProcessor {\n  /**\n   * Process the completely resolved value, after full recursion/resolution has happened\n   */\n  postProcess(input: any, context: IResolveContext): any;\n}\n\n/**\n * Whether the given object is an `IResolvedValuePostProcessor`\n */\nexport function isResolvedValuePostProcessor(x: any): x is IResolvedValuePostProcessor {\n  return x.postProcess !== undefined;\n}"]}
{
"name": "@aws-cdk/cdk",
"version": "0.32.0",
"version": "0.33.0",
"description": "AWS Cloud Development Kit Core Library",

@@ -73,14 +73,14 @@ "main": "lib/index.js",

"@types/lodash": "^4.14.130",
"cdk-build-tools": "^0.32.0",
"cfn2ts": "^0.32.0",
"cdk-build-tools": "^0.33.0",
"cfn2ts": "^0.33.0",
"fast-check": "^1.14.0",
"lodash": "^4.17.11",
"pkglint": "^0.32.0"
"pkglint": "^0.33.0"
},
"dependencies": {
"@aws-cdk/cx-api": "^0.32.0"
"@aws-cdk/cx-api": "^0.33.0"
},
"homepage": "https://github.com/awslabs/aws-cdk",
"peerDependencies": {
"@aws-cdk/cx-api": "^0.32.0"
"@aws-cdk/cx-api": "^0.33.0"
},

@@ -90,2 +90,2 @@ "engines": {

}
}
}
import { Test } from 'nodeunit';
declare const _default: {
'synthesizes all stacks and returns synthesis result'(test: Test): void;
'synth(name) can be used programmatically'(test: Test): void;
'synth(name) also collects metadata from all constructs in the stack'(test: Test): void;
'context can be passed through CDK_CONTEXT'(test: Test): void;

@@ -10,3 +8,3 @@ 'context from the command line can be used when creating the stack'(test: Test): void;

'setContext(k,v) cannot be called after stacks have been added because stacks may use the context'(test: Test): void;
'app.synthesizeStack(stack) performs validation first (app.validateAll()) and if there are errors, it returns the errors'(test: Test): void;
'app.run() performs validation first and if there are errors, it returns the errors'(test: Test): void;
'app.synthesizeStack(stack) will return a list of missing contextual information'(test: Test): void;

@@ -13,0 +11,0 @@ 'runtime library versions disabled'(test: Test): void;

@@ -5,11 +5,13 @@ "use strict";

const app_1 = require("../lib/app");
function withApp(context, block) {
const app = new app_1.App({ context });
function withApp(props, block) {
const app = new app_1.App({
runtimeInfo: false,
stackTraces: false,
...props,
});
block(app);
const session = app.run();
// return the legacy manifest
return session.store.readJson(cxapi.OUTFILE_NAME);
return app.run();
}
function synth(context) {
return withApp(context, app => {
return withApp({ context }, app => {
const stack1 = new lib_1.Stack(app, 'stack1', { env: { account: '12345', region: 'us-east-1' } });

@@ -31,6 +33,3 @@ new lib_1.CfnResource(stack1, 's1c1', { type: 'DummyResource', properties: { Prop1: 'Prop1' } });

const response = synth(context);
const stack = response.stacks.find(s => s.name === name);
if (!stack) {
throw new Error(`Stack ${name} not found`);
}
const stack = response.getStack(name);
if (!includeMetadata) {

@@ -48,64 +47,37 @@ delete stack.metadata;

}
/**
* Strip stack traces from metadata
*/
function stripStackTraces(meta) {
for (const key of Object.keys(meta)) {
meta[key] = meta[key].filter(entry => entry.type !== 'aws:cdk:logicalId');
}
}
module.exports = {
'synthesizes all stacks and returns synthesis result'(test) {
const response = synth();
// clean up metadata so assertion will be sane
response.stacks.forEach(s => delete s.metadata);
delete response.runtime;
delete response.artifacts;
test.deepEqual(response, {
version: '0.19.0',
stacks: [{ name: 'stack1',
environment: { name: '12345/us-east-1',
account: '12345',
region: 'us-east-1' },
template: { Resources: { s1c1: { Type: 'DummyResource', Properties: { Prop1: 'Prop1' } },
s1c2: { Type: 'DummyResource', Properties: { Foo: 123 } } } } },
{ name: 'stack2',
environment: { name: 'unknown-account/unknown-region',
account: 'unknown-account',
region: 'unknown-region' },
template: { Resources: { s2c1: { Type: 'DummyResource', Properties: { Prog2: 'Prog2' } },
s1c2r1D1791C01: { Type: 'ResourceType1' },
s1c2r25F685FFF: { Type: 'ResourceType2' } } } }]
delete response.dir;
test.deepEqual(response.stacks.length, 2);
const stack1 = response.stacks[0];
test.deepEqual(stack1.name, 'stack1');
test.deepEqual(stack1.environment.account, 12345);
test.deepEqual(stack1.environment.region, 'us-east-1');
test.deepEqual(stack1.environment.name, 'aws://12345/us-east-1');
test.deepEqual(stack1.template, { Resources: { s1c1: { Type: 'DummyResource', Properties: { Prop1: 'Prop1' } },
s1c2: { Type: 'DummyResource', Properties: { Foo: 123 } } } });
test.deepEqual(stack1.metadata, {
'/stack1': [{ type: 'meta', data: 111 }],
'/stack1/s1c1': [{ type: 'aws:cdk:logicalId', data: 's1c1' }],
'/stack1/s1c2': [{ type: 'aws:cdk:logicalId', data: 's1c2' },
{ type: 'aws:cdk:warning', data: 'warning1' },
{ type: 'aws:cdk:warning', data: 'warning2' }],
'/': [{ type: 'applevel', data: 123 }]
});
const stack2 = response.stacks[1];
test.deepEqual(stack2.name, 'stack2');
test.deepEqual(stack2.environment.name, 'aws://unknown-account/unknown-region');
test.deepEqual(stack2.template, { Resources: { s2c1: { Type: 'DummyResource', Properties: { Prog2: 'Prog2' } },
s1c2r1D1791C01: { Type: 'ResourceType1' },
s1c2r25F685FFF: { Type: 'ResourceType2' } } });
test.deepEqual(stack2.metadata, {
'/stack2/s2c1': [{ type: 'aws:cdk:logicalId', data: 's2c1' }],
'/stack2/s1c2': [{ type: 'meta', data: { key: 'value' } }],
'/stack2/s1c2/r1': [{ type: 'aws:cdk:logicalId', data: 's1c2r1D1791C01' }],
'/stack2/s1c2/r2': [{ type: 'aws:cdk:logicalId', data: 's1c2r25F685FFF' }],
'/': [{ type: 'applevel', data: 123 }]
});
test.done();
},
'synth(name) can be used programmatically'(test) {
const prog = new app_1.App();
const stack = new lib_1.Stack(prog, 'MyStack');
new lib_1.CfnResource(stack, 'MyResource', { type: 'MyResourceType' });
test.throws(() => prog.synthesizeStacks(['foo']), /foo/);
test.deepEqual(prog.synthesizeStack('MyStack').template, { Resources: { MyResource: { Type: 'MyResourceType' } } });
test.done();
},
'synth(name) also collects metadata from all constructs in the stack'(test) {
const stack = synthStack('stack1', true);
const output = stack.metadata;
stripStackTraces(output);
test.ok(output['/'], 'app-level metadata is included under "."');
test.equal(output['/'][0].type, 'applevel');
test.equal(output['/'][0].data, 123);
test.ok(output['/stack1'], 'the construct "stack1" has metadata');
test.equal(output['/stack1'][0].type, 'meta');
test.equal(output['/stack1'][0].data, 111);
test.ok(output['/stack1'][0].trace.length > 0, 'trace contains multiple entries');
test.ok(output['/stack1/s1c2']);
test.equal(output['/stack1/s1c2'].length, 2, 'two entries');
test.equal(output['/stack1/s1c2'][0].data, 'warning1');
const stack2 = synthStack('stack2', true);
const output2 = stack2.metadata;
test.ok(output2['/stack2/s1c2']);
test.equal(output2['/stack2/s1c2'][0].type, 'meta');
test.deepEqual(output2['/stack2/s1c2'][0].data, { key: 'value' });
test.done();
},
'context can be passed through CDK_CONTEXT'(test) {

@@ -156,3 +128,3 @@ process.env[cxapi.CONTEXT_ENV] = JSON.stringify({

},
'app.synthesizeStack(stack) performs validation first (app.validateAll()) and if there are errors, it returns the errors'(test) {
'app.run() performs validation first and if there are errors, it returns the errors'(test) {
class Child extends lib_1.Construct {

@@ -169,5 +141,3 @@ validate() {

new Child(parent, 'C2');
test.throws(() => {
app.synthesizeStacks(['Parent']);
}, /Validation failed with the following errors/);
test.throws(() => app.run(), /Validation failed with the following errors/);
test.done();

@@ -196,3 +166,3 @@ },

}
const response = withApp(undefined, app => {
const response = withApp({}, app => {
new MyStack(app, 'MyStack');

@@ -226,7 +196,7 @@ });

});
test.equals(response.runtime, undefined);
test.deepEqual(response.runtime, { libraries: {} });
test.done();
},
'runtime library versions'(test) {
const response = withApp({}, app => {
const response = withApp({ runtimeInfo: true }, app => {
const stack = new lib_1.Stack(app, 'stack1');

@@ -244,3 +214,3 @@ new lib_1.CfnResource(stack, 'MyResource', { type: 'Resource::Type' });

process.env.JSII_AGENT = 'Java/1.2.3.4';
const response = withApp({}, app => {
const response = withApp({ runtimeInfo: true }, app => {
const stack = new lib_1.Stack(app, 'stack1');

@@ -255,3 +225,3 @@ new lib_1.CfnResource(stack, 'MyResource', { type: 'Resource::Type' });

'version reporting includes only @aws-cdk, aws-cdk and jsii libraries'(test) {
const response = withApp({}, app => {
const response = withApp({ runtimeInfo: true }, app => {
const stack = new lib_1.Stack(app, 'stack1');

@@ -271,3 +241,3 @@ new lib_1.CfnResource(stack, 'MyResource', { type: 'Resource::Type' });

// WHEN
const response = withApp(undefined, (app) => {
const response = withApp({}, (app) => {
const topStack = new lib_1.Stack(app, 'Stack');

@@ -281,9 +251,9 @@ const topResource = new lib_1.CfnResource(topStack, 'Res', { type: 'CDK::TopStack::Resource' });

{
name: 'Stack',
template: { Resources: { Res: { Type: 'CDK::TopStack::Resource' } } },
},
{
name: 'StackResStack7E4AFA86',
template: { Resources: { Res: { Type: 'CDK::BottomStack::Resource' } } },
},
{
name: 'Stack',
template: { Resources: { Res: { Type: 'CDK::TopStack::Resource' } } },
}
]);

@@ -293,2 +263,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 { Test } from 'nodeunit';\nimport { CfnResource, Construct, Stack, StackProps } from '../lib';\nimport { App } from '../lib/app';\n\nfunction withApp(context: { [key: string]: any } | undefined, block: (app: App) => void): cxapi.SynthesizeResponse {\n  const app = new App({ context });\n\n  block(app);\n\n  const session = app.run();\n\n  // return the legacy manifest\n  return session.store.readJson(cxapi.OUTFILE_NAME);\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 CfnResource(stack1, 's1c1', { type: 'DummyResource', properties: { Prop1: 'Prop1' } });\n    const r2 = new CfnResource(stack1, 's1c2', { type: 'DummyResource', properties: { Foo: 123 } });\n\n    const stack2 = new Stack(app, 'stack2');\n    new CfnResource(stack2, 's2c1', { type: 'DummyResource', properties: { Prog2: 'Prog2' } });\n    const c1 = new MyConstruct(stack2, 's1c2');\n\n    // add some metadata\n    stack1.node.addMetadata('meta', 111);\n    r2.node.addWarning('warning1');\n    r2.node.addWarning('warning2');\n    c1.node.addMetadata('meta', { key: 'value' });\n    app.node.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 as any).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 as any).metadata);\n    delete (response as any).runtime;\n    delete (response as any).artifacts;\n\n    test.deepEqual(response, {\n      version: '0.19.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 CfnResource(stack, 'MyResource', { type: 'MyResourceType' });\n\n    test.throws(() => prog.synthesizeStacks(['foo']), /foo/);\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.node.getContext('key1'), 'val1');\n    test.deepEqual(prog.node.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.node.setContext('foo', 'bar');\n    test.deepEqual(prog.node.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.node.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      protected validate() {\n        return [ `Error from ${this.node.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    }, /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(scope: App, id: string, props?: StackProps) {\n        super(scope, id, 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  'runtime library versions disabled'(test: Test) {\n    const context: any = {};\n    context[cxapi.DISABLE_VERSION_REPORTING] = true;\n\n    const response = withApp(context, app => {\n      const stack = new Stack(app, 'stack1');\n      new CfnResource(stack, 'MyResource', { type: 'Resource::Type' });\n    });\n\n    test.equals(response.runtime, undefined);\n    test.done();\n  },\n\n  'runtime library versions'(test: Test) {\n    const response = withApp({}, app => {\n      const stack = new Stack(app, 'stack1');\n      new CfnResource(stack, 'MyResource', { type: 'Resource::Type' });\n    });\n\n    const libs = (response.runtime && response.runtime.libraries) || { };\n\n    const version = require('../package.json').version;\n    test.deepEqual(libs['@aws-cdk/cdk'], version);\n    test.deepEqual(libs['@aws-cdk/cx-api'], version);\n    test.deepEqual(libs['jsii-runtime'], `node.js/${process.version}`);\n    test.done();\n  },\n\n  'jsii-runtime version loaded from JSII_AGENT'(test: Test) {\n    process.env.JSII_AGENT = 'Java/1.2.3.4';\n\n    const response = withApp({}, app => {\n      const stack = new Stack(app, 'stack1');\n      new CfnResource(stack, 'MyResource', { type: 'Resource::Type' });\n    });\n\n    const libs = (response.runtime && response.runtime.libraries) || { };\n    test.deepEqual(libs['jsii-runtime'], `Java/1.2.3.4`);\n\n    delete process.env.JSII_AGENT;\n    test.done();\n  },\n\n  'version reporting includes only @aws-cdk, aws-cdk and jsii libraries'(test: Test) {\n    const response = withApp({}, app => {\n      const stack = new Stack(app, 'stack1');\n      new CfnResource(stack, 'MyResource', { type: 'Resource::Type' });\n    });\n\n    const libs = (response.runtime && response.runtime.libraries) || { };\n\n    const version = require('../package.json').version;\n    test.deepEqual(libs, {\n      '@aws-cdk/cdk': version,\n      '@aws-cdk/cx-api': version,\n      'jsii-runtime': `node.js/${process.version}`\n    });\n\n    test.done();\n  },\n\n  'deep stack is shown and synthesized properly'(test: Test) {\n    // WHEN\n    const response = withApp(undefined, (app) => {\n      const topStack = new Stack(app, 'Stack');\n      const topResource = new CfnResource(topStack, 'Res', { type: 'CDK::TopStack::Resource' });\n\n      const bottomStack = new Stack(topResource, 'Stack');\n      new CfnResource(bottomStack, 'Res', { type: 'CDK::BottomStack::Resource' });\n    });\n\n    // THEN\n    test.deepEqual(response.stacks.map(s => ({ name: s.name, template: s.template })), [\n      {\n        name: 'StackResStack7E4AFA86',\n        template: { Resources: { Res: { Type: 'CDK::BottomStack::Resource' } } },\n      },\n      {\n        name: 'Stack',\n        template: { Resources: { Res: { Type: 'CDK::TopStack::Resource' } } },\n      }\n    ]);\n\n    test.done();\n  },\n};\n\nclass MyConstruct extends Construct {\n  constructor(scope: Construct, id: string) {\n    super(scope, id);\n\n    new CfnResource(this, 'r1', { type: 'ResourceType1' });\n    new CfnResource(this, 'r2', { type: 'ResourceType2', properties: { FromContext: this.node.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 { Test } from 'nodeunit';\nimport { CfnResource, Construct, Stack, StackProps } from '../lib';\nimport { App, AppProps } from '../lib/app';\n\nfunction withApp(props: AppProps, block: (app: App) => void): cxapi.CloudAssembly {\n  const app = new App({\n    runtimeInfo: false,\n    stackTraces: false,\n    ...props,\n  });\n\n  block(app);\n\n  return app.run();\n}\n\nfunction synth(context?: { [key: string]: any }): cxapi.CloudAssembly {\n  return withApp({ context }, app => {\n    const stack1 = new Stack(app, 'stack1', { env: { account: '12345', region: 'us-east-1' } });\n    new CfnResource(stack1, 's1c1', { type: 'DummyResource', properties: { Prop1: 'Prop1' } });\n    const r2 = new CfnResource(stack1, 's1c2', { type: 'DummyResource', properties: { Foo: 123 } });\n\n    const stack2 = new Stack(app, 'stack2');\n    new CfnResource(stack2, 's2c1', { type: 'DummyResource', properties: { Prog2: 'Prog2' } });\n    const c1 = new MyConstruct(stack2, 's1c2');\n\n    // add some metadata\n    stack1.node.addMetadata('meta', 111);\n    r2.node.addWarning('warning1');\n    r2.node.addWarning('warning2');\n    c1.node.addMetadata('meta', { key: 'value' });\n    app.node.addMetadata('applevel', 123); // apps can also have metadata\n  });\n}\n\nfunction synthStack(name: string, includeMetadata: boolean = false, context?: any): cxapi.CloudFormationStackArtifact {\n  const response = synth(context);\n  const stack = response.getStack(name);\n\n  if (!includeMetadata) {\n    delete (stack as any).metadata;\n  }\n\n  return stack;\n}\n\nexport = {\n  'synthesizes all stacks and returns synthesis result'(test: Test) {\n    const response = synth();\n    delete (response as any).dir;\n\n    test.deepEqual(response.stacks.length, 2);\n\n    const stack1 = response.stacks[0];\n    test.deepEqual(stack1.name, 'stack1');\n    test.deepEqual(stack1.environment.account, 12345);\n    test.deepEqual(stack1.environment.region, 'us-east-1');\n    test.deepEqual(stack1.environment.name, 'aws://12345/us-east-1');\n    test.deepEqual(stack1.template, { Resources:\n      { s1c1: { Type: 'DummyResource', Properties: { Prop1: 'Prop1' } },\n        s1c2: { Type: 'DummyResource', Properties: { Foo: 123 } } } });\n    test.deepEqual(stack1.metadata, {\n      '/stack1': [{ type: 'meta', data: 111 }],\n      '/stack1/s1c1': [{ type: 'aws:cdk:logicalId', data: 's1c1' }],\n      '/stack1/s1c2':\n        [{ type: 'aws:cdk:logicalId', data: 's1c2' },\n        { type: 'aws:cdk:warning', data: 'warning1' },\n        { type: 'aws:cdk:warning', data: 'warning2' }],\n      '/': [{ type: 'applevel', data: 123 }]\n    });\n\n    const stack2 = response.stacks[1];\n    test.deepEqual(stack2.name, 'stack2');\n    test.deepEqual(stack2.environment.name, 'aws://unknown-account/unknown-region');\n    test.deepEqual(stack2.template, { Resources:\n      { s2c1: { Type: 'DummyResource', Properties: { Prog2: 'Prog2' } },\n        s1c2r1D1791C01: { Type: 'ResourceType1' },\n        s1c2r25F685FFF: { Type: 'ResourceType2' } } });\n    test.deepEqual(stack2.metadata, {\n      '/stack2/s2c1': [{ type: 'aws:cdk:logicalId', data: 's2c1' }],\n      '/stack2/s1c2': [{ type: 'meta', data: { key: 'value' } }],\n      '/stack2/s1c2/r1':\n        [{ type: 'aws:cdk:logicalId', data: 's1c2r1D1791C01' }],\n      '/stack2/s1c2/r2':\n        [{ type: 'aws:cdk:logicalId', data: 's1c2r25F685FFF' }],\n      '/': [{ type: 'applevel', data: 123 }]\n    });\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.node.getContext('key1'), 'val1');\n  test.deepEqual(prog.node.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.node.setContext('foo', 'bar');\n  test.deepEqual(prog.node.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.node.setContext('foo', 'bar'));\n  test.done();\n},\n\n'app.run() performs validation first and if there are errors, it returns the errors'(test: Test) {\n\n  class Child extends Construct {\n    protected validate() {\n      return [`Error from ${this.node.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(() => app.run(), /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(scope: App, id: string, props?: StackProps) {\n      super(scope, id, 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({}, 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'runtime library versions disabled'(test: Test) {\n  const context: any = {};\n  context[cxapi.DISABLE_VERSION_REPORTING] = true;\n\n  const response = withApp(context, app => {\n    const stack = new Stack(app, 'stack1');\n    new CfnResource(stack, 'MyResource', { type: 'Resource::Type' });\n  });\n\n  test.deepEqual(response.runtime, { libraries: {} });\n  test.done();\n},\n\n'runtime library versions'(test: Test) {\n  const response = withApp({ runtimeInfo: true }, app => {\n    const stack = new Stack(app, 'stack1');\n    new CfnResource(stack, 'MyResource', { type: 'Resource::Type' });\n  });\n\n  const libs = (response.runtime && response.runtime.libraries) || {};\n\n  const version = require('../package.json').version;\n  test.deepEqual(libs['@aws-cdk/cdk'], version);\n  test.deepEqual(libs['@aws-cdk/cx-api'], version);\n  test.deepEqual(libs['jsii-runtime'], `node.js/${process.version}`);\n  test.done();\n},\n\n'jsii-runtime version loaded from JSII_AGENT'(test: Test) {\n  process.env.JSII_AGENT = 'Java/1.2.3.4';\n\n  const response = withApp({ runtimeInfo: true }, app => {\n    const stack = new Stack(app, 'stack1');\n    new CfnResource(stack, 'MyResource', { type: 'Resource::Type' });\n  });\n\n  const libs = (response.runtime && response.runtime.libraries) || {};\n  test.deepEqual(libs['jsii-runtime'], `Java/1.2.3.4`);\n\n  delete process.env.JSII_AGENT;\n  test.done();\n},\n\n'version reporting includes only @aws-cdk, aws-cdk and jsii libraries'(test: Test) {\n  const response = withApp({ runtimeInfo: true }, app => {\n    const stack = new Stack(app, 'stack1');\n    new CfnResource(stack, 'MyResource', { type: 'Resource::Type' });\n  });\n\n  const libs = (response.runtime && response.runtime.libraries) || {};\n\n  const version = require('../package.json').version;\n  test.deepEqual(libs, {\n    '@aws-cdk/cdk': version,\n    '@aws-cdk/cx-api': version,\n    'jsii-runtime': `node.js/${process.version}`\n  });\n\n  test.done();\n},\n\n'deep stack is shown and synthesized properly'(test: Test) {\n  // WHEN\n  const response = withApp({}, (app) => {\n    const topStack = new Stack(app, 'Stack');\n    const topResource = new CfnResource(topStack, 'Res', { type: 'CDK::TopStack::Resource' });\n\n    const bottomStack = new Stack(topResource, 'Stack');\n    new CfnResource(bottomStack, 'Res', { type: 'CDK::BottomStack::Resource' });\n  });\n\n  // THEN\n  test.deepEqual(response.stacks.map(s => ({ name: s.name, template: s.template })), [\n    {\n      name: 'Stack',\n      template: { Resources: { Res: { Type: 'CDK::TopStack::Resource' } } },\n    },\n    {\n      name: 'StackResStack7E4AFA86',\n      template: { Resources: { Res: { Type: 'CDK::BottomStack::Resource' } } },\n    },\n  ]);\n\n  test.done();\n},\n};\n\nclass MyConstruct extends Construct {\n  constructor(scope: Construct, id: string) {\n    super(scope, id);\n\n    new CfnResource(this, 'r1', { type: 'ResourceType1' });\n    new CfnResource(this, 'r2', { type: 'ResourceType2', properties: { FromContext: this.node.getContext('ctx1') } });\n  }\n}\n"]}

@@ -36,3 +36,5 @@ import { Test } from 'nodeunit';

'findAll returns a list of all children in either DFS or BFS'(test: Test): void;
'ancestors returns a list of parents up to root'(test: Test): void;
'"root" returns the root construct'(test: Test): void;
};
export = _default;

@@ -86,5 +86,7 @@ "use strict";

test.deepEqual(new lib_1.SSMParameterProvider(child, { parameterName: 'foo' }).parameterValue(), 'dummy');
const output = app.synthesizeStack(stack.name);
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);
const assembly = app.run();
const output = assembly.getStack('test-stack');
const metadata = output.metadata;
const azError = metadata['/test-stack'].find(x => x.type === cxapi.ERROR_METADATA_KEY);
const ssmError = 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'));

@@ -95,2 +97,2 @@ test.ok(ssmError && ssmError.data.includes('Cannot determine scope for context provider ssm'));

};
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"test.context.js","sourceRoot":"","sources":["test.context.ts"],"names":[],"mappings":";AAAA,yCAA0C;AAE1C,gCAC6D;AAgH7D,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;AAvHD,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,IAAI,CAAC,UAAU,CAAC,GAAG,EAAE,CAAC,YAAY,EAAE,YAAY,CAAC,CAAC,CAAC;QAEzD,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,IAAI,CAAC,UAAU,CAAC,GAAG,EAAE,YAAY,CAAC,CAAC;QAEzC,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;IAED,wDAAwD,CAAC,IAAU;QACjE,QAAQ;QACR,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;QAEpG,OAAO;QACP,MAAM,QAAQ,GAAG,IAAI,qBAAe,CAAC,KAAK,EAAE,UAAU,EAAE;YACtD,IAAI,EAAE;gBACJ,EAAE,GAAG,EAAE,MAAM,EAAE,KAAK,EAAE,QAAQ,EAAE;gBAChC,EAAE,GAAG,EAAE,MAAM,EAAE,KAAK,EAAE,QAAQ,EAAE;aACjC;SACF,CAAC,CAAC;QAEH,OAAO;QACP,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,GAAG,EAAE,iHAAiH,CAAC,CAAC;QAE7I,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,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,IAAI,CAAC,UAAU,CAAC,GAAG,EAAE,KAAK,CAAC,CAAC;QAElC,MAAM,IAAI,GAAG,IAAI,0BAAoB,CAAC,KAAK,EAAG,EAAC,aAAa,EAAE,MAAM,EAAC,CAAC,CAAC;QACvE,MAAM,GAAG,GAAG,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,cAAc,EAAE,CAAC,CAAC;QACtD,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,IAAI,CAAC,CAAC;QAE/C,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, 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.node.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.node.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\n  'Key generation can contain arbitrarily deep structures'(test: Test) {\n    // GIVEN\n    const stack = new Stack(undefined, 'TestStack', { env: { account: '12345', region: 'us-east-1' } });\n\n    // WHEN\n    const provider = new ContextProvider(stack, 'provider', {\n      list: [\n        { key: 'key1', value: 'value1' },\n        { key: 'key2', value: 'value2' },\n      ],\n    });\n\n    // THEN\n    test.equals(provider.key, 'provider:account=12345:list.0.key=key1:list.0.value=value1:list.1.key=key2:list.1.value=value2:region=us-east-1');\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,  {parameterName: 'test'}).parameterValue();\n    const key = expectedContextKey(stack);\n\n    stack.node.setContext(key, 'abc');\n\n    const ssmp = new SSMParameterProvider(stack,  {parameterName: 'test'});\n    const azs = stack.node.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.name);\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"]}
//# 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, ContextProvider, 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.node.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.node.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\n  'Key generation can contain arbitrarily deep structures'(test: Test) {\n    // GIVEN\n    const stack = new Stack(undefined, 'TestStack', { env: { account: '12345', region: 'us-east-1' } });\n\n    // WHEN\n    const provider = new ContextProvider(stack, 'provider', {\n      list: [\n        { key: 'key1', value: 'value1' },\n        { key: 'key2', value: 'value2' },\n      ],\n    });\n\n    // THEN\n    test.equals(provider.key, 'provider:account=12345:list.0.key=key1:list.0.value=value1:list.1.key=key2:list.1.value=value2:region=us-east-1');\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,  {parameterName: 'test'}).parameterValue();\n    const key = expectedContextKey(stack);\n\n    stack.node.setContext(key, 'abc');\n\n    const ssmp = new SSMParameterProvider(stack,  {parameterName: 'test'});\n    const azs = stack.node.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 assembly = app.run();\n    const output = assembly.getStack('test-stack');\n    const metadata = output.metadata;\n    const azError: cxapi.MetadataEntry | undefined = metadata['/test-stack'].find(x => x.type === cxapi.ERROR_METADATA_KEY);\n    const ssmError: cxapi.MetadataEntry | undefined = 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"]}

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

test.deepEqual(stack.name, 'valid-stack-name');
test.ok('valid-stack-name' in (session.manifest.artifacts || {}));
test.ok(session.tryGetArtifact('valid-stack-name'));
test.done();

@@ -364,2 +364,2 @@ },

};
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"test.stack.js","sourceRoot":"","sources":["test.stack.ts"],"names":[],"mappings":"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","sourcesContent":["import cxapi = require('@aws-cdk/cx-api');\nimport { Test } from 'nodeunit';\nimport { App, CfnCondition, CfnOutput, CfnParameter, CfnResource, Construct, Include, ScopedAws, Stack, Token } from '../lib';\n\nexport = {\n  'a stack can be serialized into a CloudFormation template, initially it\\'s empty'(test: Test) {\n    const stack = new Stack();\n    test.deepEqual(stack._toCloudFormation(), { });\n    test.done();\n  },\n\n  'stack objects have some template-level propeties, such as Description, Version, Transform'(test: Test) {\n    const stack = new Stack();\n    stack.templateOptions.templateFormatVersion = 'MyTemplateVersion';\n    stack.templateOptions.description = 'This is my description';\n    stack.templateOptions.transform = 'SAMy';\n    test.deepEqual(stack._toCloudFormation(), {\n      Description: 'This is my description',\n      AWSTemplateFormatVersion: 'MyTemplateVersion',\n      Transform: 'SAMy'\n    });\n    test.done();\n  },\n\n  'Stack.isStack indicates that a construct is a stack'(test: Test) {\n    const stack = new Stack();\n    const c = new Construct(stack, 'Construct');\n    test.ok(Stack.isStack(stack));\n    test.ok(!Stack.isStack(c));\n    test.done();\n  },\n\n  'stack.id is not included in the logical identities of resources within it'(test: Test) {\n    const stack = new Stack(undefined, 'MyStack');\n    new CfnResource(stack, 'MyResource', { type: 'MyResourceType' });\n\n    test.deepEqual(stack._toCloudFormation(), { Resources: { MyResource: { Type: 'MyResourceType' } } });\n    test.done();\n  },\n\n  'stack.templateOptions can be used to set template-level options'(test: Test) {\n    const stack = new Stack();\n\n    stack.templateOptions.description = 'StackDescription';\n    stack.templateOptions.templateFormatVersion = 'TemplateVersion';\n    stack.templateOptions.transform = 'Transform';\n    stack.templateOptions.metadata = {\n      MetadataKey: 'MetadataValue'\n    };\n\n    test.deepEqual(stack._toCloudFormation(), {\n      Description: 'StackDescription',\n      Transform: 'Transform',\n      AWSTemplateFormatVersion: 'TemplateVersion',\n      Metadata: { MetadataKey: 'MetadataValue' }\n    });\n\n    test.done();\n  },\n\n  // This approach will only apply to TypeScript code, but at least it's a temporary\n  // workaround for people running into issues caused by SDK-3003.\n  // We should come up with a proper solution that involved jsii callbacks (when they exist)\n  // so this can be implemented by jsii languages as well.\n  'Overriding `Stack._toCloudFormation` allows arbitrary post-processing of the generated template during synthesis'(test: Test) {\n\n    const stack = new StackWithPostProcessor();\n\n    new CfnResource(stack, 'myResource', {\n      type: 'AWS::MyResource',\n      properties: {\n        MyProp1: 'hello',\n        MyProp2: 'howdy',\n        Environment: {\n          Key: 'value'\n        }\n      }\n    });\n\n    test.deepEqual(stack._toCloudFormation(), { Resources:\n      { myResource:\n         { Type: 'AWS::MyResource',\n         Properties:\n          { MyProp1: 'hello',\n          MyProp2: 'howdy',\n          Environment: { key: 'value' } } } } });\n\n    test.done();\n  },\n\n  'Construct.findResource(logicalId) can be used to retrieve a resource by its path'(test: Test) {\n    const stack = new Stack();\n\n    test.ok(!stack.node.tryFindChild('foo'), 'empty stack');\n\n    const r1 = new CfnResource(stack, 'Hello', { type: 'MyResource' });\n    test.equal(stack.findResource(r1.stackPath), r1, 'look up top-level');\n\n    const child = new Construct(stack, 'Child');\n    const r2 = new CfnResource(child, 'Hello', { type: 'MyResource' });\n\n    test.equal(stack.findResource(r2.stackPath), r2, 'look up child');\n\n    test.done();\n  },\n\n  'Stack.findResource will fail if the element is not a resource'(test: Test) {\n    const stack = new Stack();\n\n    const p = new CfnParameter(stack, 'MyParam', { type: 'String' });\n\n    test.throws(() => stack.findResource(p.node.path));\n    test.done();\n  },\n\n  'Stack.getByPath can be used to find any CloudFormation element (Parameter, Output, etc)'(test: Test) {\n\n    const stack = new Stack();\n\n    const p = new CfnParameter(stack, 'MyParam', { type: 'String' });\n    const o = new CfnOutput(stack, 'MyOutput', { value: 'boom' });\n    const c = new CfnCondition(stack, 'MyCondition');\n\n    test.equal(stack.node.findChild(p.node.path), p);\n    test.equal(stack.node.findChild(o.node.path), o);\n    test.equal(stack.node.findChild(c.node.path), c);\n\n    test.done();\n  },\n\n  'Stack names can have hyphens in them'(test: Test) {\n    const root = new App();\n\n    new Stack(root, 'Hello-World');\n    // Did not throw\n\n    test.done();\n  },\n\n  'Include should support non-hash top-level template elements like \"Description\"'(test: Test) {\n    const stack = new Stack();\n\n    const template = {\n      Description: 'hello, world'\n    };\n\n    new Include(stack, 'Include', { template });\n\n    const output = stack._toCloudFormation();\n\n    test.equal(typeof output.Description, 'string');\n    test.done();\n  },\n\n  'Pseudo values attached to one stack can be referenced in another stack'(test: Test) {\n    // GIVEN\n    const app = new App();\n    const stack1 = new Stack(app, 'Stack1');\n    const account1 = new ScopedAws(stack1).accountId;\n    const stack2 = new Stack(app, 'Stack2');\n\n    // WHEN - used in another stack\n    new CfnParameter(stack2, 'SomeParameter', { type: 'String', default: account1 });\n\n    // THEN\n    // Need to do this manually now, since we're in testing mode. In a normal CDK app,\n    // this happens as part of app.run().\n    app.node.prepareTree();\n\n    test.deepEqual(stack1._toCloudFormation(), {\n      Outputs: {\n        ExportsOutputRefAWSAccountIdAD568057: {\n          Value: { Ref: 'AWS::AccountId' },\n          Export: { Name: 'Stack1:ExportsOutputRefAWSAccountIdAD568057' }\n        }\n      }\n    });\n\n    test.deepEqual(stack2._toCloudFormation(), {\n      Parameters: {\n        SomeParameter: {\n          Type: 'String',\n          Default: { 'Fn::ImportValue': 'Stack1:ExportsOutputRefAWSAccountIdAD568057' }\n        }\n      }\n    });\n\n    test.done();\n  },\n\n  'Cross-stack references are detected in resource properties'(test: Test) {\n    // GIVEN\n    const app = new App();\n    const stack1 = new Stack(app, 'Stack1');\n    const resource1 = new CfnResource(stack1, 'Resource', { type: 'BLA' });\n    const stack2 = new Stack(app, 'Stack2');\n\n    // WHEN - used in another resource\n    new CfnResource(stack2, 'SomeResource', { type: 'AWS::Some::Resource', properties: {\n      someProperty: new Token(() => resource1.ref),\n    }});\n\n    // THEN\n    // Need to do this manually now, since we're in testing mode. In a normal CDK app,\n    // this happens as part of app.run().\n    app.node.prepareTree();\n\n    test.deepEqual(stack2._toCloudFormation(), {\n      Resources: {\n        SomeResource: {\n          Type: 'AWS::Some::Resource',\n          Properties: {\n            someProperty: { 'Fn::ImportValue': 'Stack1:ExportsOutputRefResource1D5D905A' }\n          }\n        }\n      }\n    });\n    test.done();\n  },\n\n  'cross-stack references in lazy tokens work'(test: Test) {\n    // GIVEN\n    const app = new App();\n    const stack1 = new Stack(app, 'Stack1');\n    const account1 = new ScopedAws(stack1).accountId;\n    const stack2 = new Stack(app, 'Stack2');\n\n    // WHEN - used in another stack\n    new CfnParameter(stack2, 'SomeParameter', { type: 'String', default: new Token(() => account1) });\n\n    app.node.prepareTree();\n\n    // THEN\n    test.deepEqual(stack1._toCloudFormation(), {\n      Outputs: {\n        ExportsOutputRefAWSAccountIdAD568057: {\n          Value: { Ref: 'AWS::AccountId' },\n          Export: { Name: 'Stack1:ExportsOutputRefAWSAccountIdAD568057' }\n        }\n      }\n    });\n\n    test.deepEqual(stack2._toCloudFormation(), {\n      Parameters: {\n        SomeParameter: {\n          Type: 'String',\n          Default: { 'Fn::ImportValue': 'Stack1:ExportsOutputRefAWSAccountIdAD568057' }\n        }\n      }\n    });\n\n    test.done();\n  },\n\n  'Cross-stack use of Region returns nonscoped intrinsic'(test: Test) {\n    // GIVEN\n    const app = new App();\n    const stack1 = new Stack(app, 'Stack1');\n    const stack2 = new Stack(app, 'Stack2');\n\n    // WHEN - used in another stack\n    new CfnOutput(stack2, 'DemOutput', { value: stack1.region });\n\n    // THEN\n    // Need to do this manually now, since we're in testing mode. In a normal CDK app,\n    // this happens as part of app.run().\n    app.node.prepareTree();\n\n    test.deepEqual(stack2._toCloudFormation(), {\n      Outputs: {\n        DemOutput: {\n          Value: { Ref: 'AWS::Region' },\n        }\n      }\n    });\n\n    test.done();\n  },\n\n  'cross-stack references in strings work'(test: Test) {\n    // GIVEN\n    const app = new App();\n    const stack1 = new Stack(app, 'Stack1');\n    const account1 = new ScopedAws(stack1).accountId;\n    const stack2 = new Stack(app, 'Stack2');\n\n    // WHEN - used in another stack\n    new CfnParameter(stack2, 'SomeParameter', { type: 'String', default: `TheAccountIs${account1}` });\n\n    app.node.prepareTree();\n\n    // THEN\n    test.deepEqual(stack2._toCloudFormation(), {\n      Parameters: {\n        SomeParameter: {\n          Type: 'String',\n          Default: { 'Fn::Join': [ '', [ 'TheAccountIs', { 'Fn::ImportValue': 'Stack1:ExportsOutputRefAWSAccountIdAD568057' } ]] }\n        }\n      }\n    });\n\n    test.done();\n  },\n\n  'cannot create cyclic reference between stacks'(test: Test) {\n    // GIVEN\n    const app = new App();\n    const stack1 = new Stack(app, 'Stack1');\n    const account1 = new ScopedAws(stack1).accountId;\n    const stack2 = new Stack(app, 'Stack2');\n    const account2 = new ScopedAws(stack2).accountId;\n\n    // WHEN\n    new CfnParameter(stack2, 'SomeParameter', { type: 'String', default: account1 });\n    new CfnParameter(stack1, 'SomeParameter', { type: 'String', default: account2 });\n\n    test.throws(() => {\n      app.node.prepareTree();\n      // tslint:disable-next-line:max-line-length\n    }, \"'Stack2' depends on 'Stack1' (Stack2/SomeParameter -> Stack1.AWS::AccountId). Adding this dependency (Stack1/SomeParameter -> Stack2.AWS::AccountId) would create a cyclic reference.\");\n\n    test.done();\n  },\n\n  'stacks know about their dependencies'(test: Test) {\n    // GIVEN\n    const app = new App();\n    const stack1 = new Stack(app, 'Stack1');\n    const account1 = new ScopedAws(stack1).accountId;\n    const stack2 = new Stack(app, 'Stack2');\n\n    // WHEN\n    new CfnParameter(stack2, 'SomeParameter', { type: 'String', default: account1 });\n\n    app.node.prepareTree();\n\n    // THEN\n    test.deepEqual(stack2.dependencies().map(s => s.node.id), ['Stack1']);\n\n    test.done();\n  },\n\n  'cannot create references to stacks in other regions/accounts'(test: Test) {\n    // GIVEN\n    const app = new App();\n    const stack1 = new Stack(app, 'Stack1', { env: { account: '123456789012', region: 'es-norst-1' }});\n    const account1 = new ScopedAws(stack1).accountId;\n    const stack2 = new Stack(app, 'Stack2', { env: { account: '123456789012', region: 'es-norst-2' }});\n\n    // WHEN\n    new CfnParameter(stack2, 'SomeParameter', { type: 'String', default: account1 });\n\n    test.throws(() => {\n      app.node.prepareTree();\n    }, /Can only reference cross stacks in the same region and account/);\n\n    test.done();\n  },\n\n  'stack with region supplied via props returns literal value'(test: Test) {\n    // GIVEN\n    const app = new App();\n    const stack = new Stack(app, 'Stack1', { env: { account: '123456789012', region: 'es-norst-1' }});\n\n    // THEN\n    test.equal(stack.node.resolve(stack.region), 'es-norst-1');\n\n    test.done();\n  },\n\n  'stack with region supplied via context returns symbolic value'(test: Test) {\n    // GIVEN\n    const app = new App();\n\n    app.node.setContext(cxapi.DEFAULT_REGION_CONTEXT_KEY, 'es-norst-1');\n    const stack = new Stack(app, 'Stack1');\n\n    // THEN\n    test.deepEqual(stack.node.resolve(stack.region), { Ref: 'AWS::Region' });\n\n    test.done();\n  },\n\n  'overrideLogicalId(id) can be used to override the logical ID of a resource'(test: Test) {\n    // GIVEN\n    const stack = new Stack();\n    const bonjour = new CfnResource(stack, 'BonjourResource', { type: 'Resource::Type' });\n\n    // { Ref } and { GetAtt }\n    new CfnResource(stack, 'RefToBonjour', { type: 'Other::Resource', properties: {\n      RefToBonjour: bonjour.ref.toString(),\n      GetAttBonjour: bonjour.getAtt('TheAtt').toString()\n    }});\n\n    bonjour.overrideLogicalId('BOOM');\n\n    // THEN\n    test.deepEqual(stack._toCloudFormation(), { Resources:\n      { BOOM: { Type: 'Resource::Type' },\n        RefToBonjour:\n         { Type: 'Other::Resource',\n           Properties:\n            { RefToBonjour: { Ref: 'BOOM' },\n              GetAttBonjour: { 'Fn::GetAtt': [ 'BOOM', 'TheAtt' ] } } } } });\n    test.done();\n  },\n\n  'Stack name can be overridden via properties'(test: Test) {\n    // WHEN\n    const stack = new Stack(undefined, 'Stack', { stackName: 'otherName' });\n\n    // THEN\n    test.deepEqual(stack.name, 'otherName');\n\n    test.done();\n  },\n\n  'Stack name is inherited from App name if available'(test: Test) {\n    // WHEN\n    const root = new App();\n    const app = new Construct(root, 'Prod');\n    const stack = new Stack(app, 'Stack');\n\n    // THEN\n    test.deepEqual(stack.name, 'ProdStackD5279B22');\n\n    test.done();\n  },\n\n  'stack construct id does not go through stack name validation if there is an explicit stack name'(test: Test) {\n    // GIVEN\n    const app = new App();\n\n    // WHEN\n    const stack = new Stack(app, 'invalid as : stack name, but thats fine', {\n      stackName: 'valid-stack-name'\n    });\n\n    // THEN\n    const session = app.run();\n    test.deepEqual(stack.name, 'valid-stack-name');\n    test.ok('valid-stack-name' in (session.manifest.artifacts || {}));\n    test.done();\n  },\n\n  'stack validation is performed on explicit stack name'(test: Test) {\n    // GIVEN\n    const app = new App();\n\n    // THEN\n    test.throws(() => new Stack(app, 'boom', { stackName: 'invalid:stack:name' }),\n      /Stack name must match the regular expression/);\n\n    test.done();\n  }\n};\n\nclass StackWithPostProcessor extends Stack {\n\n  // ...\n\n  public _toCloudFormation() {\n    const template = super._toCloudFormation();\n\n    // manipulate template (e.g. rename \"Key\" to \"key\")\n    template.Resources.myResource.Properties.Environment.key =\n      template.Resources.myResource.Properties.Environment.Key;\n    delete template.Resources.myResource.Properties.Environment.Key;\n\n    return template;\n  }\n}\n"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"test.stack.js","sourceRoot":"","sources":["test.stack.ts"],"names":[],"mappings":"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","sourcesContent":["import cxapi = require('@aws-cdk/cx-api');\nimport { Test } from 'nodeunit';\nimport { App, CfnCondition, CfnOutput, CfnParameter, CfnResource, Construct, Include, ScopedAws, Stack, Token } from '../lib';\n\nexport = {\n  'a stack can be serialized into a CloudFormation template, initially it\\'s empty'(test: Test) {\n    const stack = new Stack();\n    test.deepEqual(stack._toCloudFormation(), { });\n    test.done();\n  },\n\n  'stack objects have some template-level propeties, such as Description, Version, Transform'(test: Test) {\n    const stack = new Stack();\n    stack.templateOptions.templateFormatVersion = 'MyTemplateVersion';\n    stack.templateOptions.description = 'This is my description';\n    stack.templateOptions.transform = 'SAMy';\n    test.deepEqual(stack._toCloudFormation(), {\n      Description: 'This is my description',\n      AWSTemplateFormatVersion: 'MyTemplateVersion',\n      Transform: 'SAMy'\n    });\n    test.done();\n  },\n\n  'Stack.isStack indicates that a construct is a stack'(test: Test) {\n    const stack = new Stack();\n    const c = new Construct(stack, 'Construct');\n    test.ok(Stack.isStack(stack));\n    test.ok(!Stack.isStack(c));\n    test.done();\n  },\n\n  'stack.id is not included in the logical identities of resources within it'(test: Test) {\n    const stack = new Stack(undefined, 'MyStack');\n    new CfnResource(stack, 'MyResource', { type: 'MyResourceType' });\n\n    test.deepEqual(stack._toCloudFormation(), { Resources: { MyResource: { Type: 'MyResourceType' } } });\n    test.done();\n  },\n\n  'stack.templateOptions can be used to set template-level options'(test: Test) {\n    const stack = new Stack();\n\n    stack.templateOptions.description = 'StackDescription';\n    stack.templateOptions.templateFormatVersion = 'TemplateVersion';\n    stack.templateOptions.transform = 'Transform';\n    stack.templateOptions.metadata = {\n      MetadataKey: 'MetadataValue'\n    };\n\n    test.deepEqual(stack._toCloudFormation(), {\n      Description: 'StackDescription',\n      Transform: 'Transform',\n      AWSTemplateFormatVersion: 'TemplateVersion',\n      Metadata: { MetadataKey: 'MetadataValue' }\n    });\n\n    test.done();\n  },\n\n  // This approach will only apply to TypeScript code, but at least it's a temporary\n  // workaround for people running into issues caused by SDK-3003.\n  // We should come up with a proper solution that involved jsii callbacks (when they exist)\n  // so this can be implemented by jsii languages as well.\n  'Overriding `Stack._toCloudFormation` allows arbitrary post-processing of the generated template during synthesis'(test: Test) {\n\n    const stack = new StackWithPostProcessor();\n\n    new CfnResource(stack, 'myResource', {\n      type: 'AWS::MyResource',\n      properties: {\n        MyProp1: 'hello',\n        MyProp2: 'howdy',\n        Environment: {\n          Key: 'value'\n        }\n      }\n    });\n\n    test.deepEqual(stack._toCloudFormation(), { Resources:\n      { myResource:\n         { Type: 'AWS::MyResource',\n         Properties:\n          { MyProp1: 'hello',\n          MyProp2: 'howdy',\n          Environment: { key: 'value' } } } } });\n\n    test.done();\n  },\n\n  'Construct.findResource(logicalId) can be used to retrieve a resource by its path'(test: Test) {\n    const stack = new Stack();\n\n    test.ok(!stack.node.tryFindChild('foo'), 'empty stack');\n\n    const r1 = new CfnResource(stack, 'Hello', { type: 'MyResource' });\n    test.equal(stack.findResource(r1.stackPath), r1, 'look up top-level');\n\n    const child = new Construct(stack, 'Child');\n    const r2 = new CfnResource(child, 'Hello', { type: 'MyResource' });\n\n    test.equal(stack.findResource(r2.stackPath), r2, 'look up child');\n\n    test.done();\n  },\n\n  'Stack.findResource will fail if the element is not a resource'(test: Test) {\n    const stack = new Stack();\n\n    const p = new CfnParameter(stack, 'MyParam', { type: 'String' });\n\n    test.throws(() => stack.findResource(p.node.path));\n    test.done();\n  },\n\n  'Stack.getByPath can be used to find any CloudFormation element (Parameter, Output, etc)'(test: Test) {\n\n    const stack = new Stack();\n\n    const p = new CfnParameter(stack, 'MyParam', { type: 'String' });\n    const o = new CfnOutput(stack, 'MyOutput', { value: 'boom' });\n    const c = new CfnCondition(stack, 'MyCondition');\n\n    test.equal(stack.node.findChild(p.node.path), p);\n    test.equal(stack.node.findChild(o.node.path), o);\n    test.equal(stack.node.findChild(c.node.path), c);\n\n    test.done();\n  },\n\n  'Stack names can have hyphens in them'(test: Test) {\n    const root = new App();\n\n    new Stack(root, 'Hello-World');\n    // Did not throw\n\n    test.done();\n  },\n\n  'Include should support non-hash top-level template elements like \"Description\"'(test: Test) {\n    const stack = new Stack();\n\n    const template = {\n      Description: 'hello, world'\n    };\n\n    new Include(stack, 'Include', { template });\n\n    const output = stack._toCloudFormation();\n\n    test.equal(typeof output.Description, 'string');\n    test.done();\n  },\n\n  'Pseudo values attached to one stack can be referenced in another stack'(test: Test) {\n    // GIVEN\n    const app = new App();\n    const stack1 = new Stack(app, 'Stack1');\n    const account1 = new ScopedAws(stack1).accountId;\n    const stack2 = new Stack(app, 'Stack2');\n\n    // WHEN - used in another stack\n    new CfnParameter(stack2, 'SomeParameter', { type: 'String', default: account1 });\n\n    // THEN\n    // Need to do this manually now, since we're in testing mode. In a normal CDK app,\n    // this happens as part of app.run().\n    app.node.prepareTree();\n\n    test.deepEqual(stack1._toCloudFormation(), {\n      Outputs: {\n        ExportsOutputRefAWSAccountIdAD568057: {\n          Value: { Ref: 'AWS::AccountId' },\n          Export: { Name: 'Stack1:ExportsOutputRefAWSAccountIdAD568057' }\n        }\n      }\n    });\n\n    test.deepEqual(stack2._toCloudFormation(), {\n      Parameters: {\n        SomeParameter: {\n          Type: 'String',\n          Default: { 'Fn::ImportValue': 'Stack1:ExportsOutputRefAWSAccountIdAD568057' }\n        }\n      }\n    });\n\n    test.done();\n  },\n\n  'Cross-stack references are detected in resource properties'(test: Test) {\n    // GIVEN\n    const app = new App();\n    const stack1 = new Stack(app, 'Stack1');\n    const resource1 = new CfnResource(stack1, 'Resource', { type: 'BLA' });\n    const stack2 = new Stack(app, 'Stack2');\n\n    // WHEN - used in another resource\n    new CfnResource(stack2, 'SomeResource', { type: 'AWS::Some::Resource', properties: {\n      someProperty: new Token(() => resource1.ref),\n    }});\n\n    // THEN\n    // Need to do this manually now, since we're in testing mode. In a normal CDK app,\n    // this happens as part of app.run().\n    app.node.prepareTree();\n\n    test.deepEqual(stack2._toCloudFormation(), {\n      Resources: {\n        SomeResource: {\n          Type: 'AWS::Some::Resource',\n          Properties: {\n            someProperty: { 'Fn::ImportValue': 'Stack1:ExportsOutputRefResource1D5D905A' }\n          }\n        }\n      }\n    });\n    test.done();\n  },\n\n  'cross-stack references in lazy tokens work'(test: Test) {\n    // GIVEN\n    const app = new App();\n    const stack1 = new Stack(app, 'Stack1');\n    const account1 = new ScopedAws(stack1).accountId;\n    const stack2 = new Stack(app, 'Stack2');\n\n    // WHEN - used in another stack\n    new CfnParameter(stack2, 'SomeParameter', { type: 'String', default: new Token(() => account1) });\n\n    app.node.prepareTree();\n\n    // THEN\n    test.deepEqual(stack1._toCloudFormation(), {\n      Outputs: {\n        ExportsOutputRefAWSAccountIdAD568057: {\n          Value: { Ref: 'AWS::AccountId' },\n          Export: { Name: 'Stack1:ExportsOutputRefAWSAccountIdAD568057' }\n        }\n      }\n    });\n\n    test.deepEqual(stack2._toCloudFormation(), {\n      Parameters: {\n        SomeParameter: {\n          Type: 'String',\n          Default: { 'Fn::ImportValue': 'Stack1:ExportsOutputRefAWSAccountIdAD568057' }\n        }\n      }\n    });\n\n    test.done();\n  },\n\n  'Cross-stack use of Region returns nonscoped intrinsic'(test: Test) {\n    // GIVEN\n    const app = new App();\n    const stack1 = new Stack(app, 'Stack1');\n    const stack2 = new Stack(app, 'Stack2');\n\n    // WHEN - used in another stack\n    new CfnOutput(stack2, 'DemOutput', { value: stack1.region });\n\n    // THEN\n    // Need to do this manually now, since we're in testing mode. In a normal CDK app,\n    // this happens as part of app.run().\n    app.node.prepareTree();\n\n    test.deepEqual(stack2._toCloudFormation(), {\n      Outputs: {\n        DemOutput: {\n          Value: { Ref: 'AWS::Region' },\n        }\n      }\n    });\n\n    test.done();\n  },\n\n  'cross-stack references in strings work'(test: Test) {\n    // GIVEN\n    const app = new App();\n    const stack1 = new Stack(app, 'Stack1');\n    const account1 = new ScopedAws(stack1).accountId;\n    const stack2 = new Stack(app, 'Stack2');\n\n    // WHEN - used in another stack\n    new CfnParameter(stack2, 'SomeParameter', { type: 'String', default: `TheAccountIs${account1}` });\n\n    app.node.prepareTree();\n\n    // THEN\n    test.deepEqual(stack2._toCloudFormation(), {\n      Parameters: {\n        SomeParameter: {\n          Type: 'String',\n          Default: { 'Fn::Join': [ '', [ 'TheAccountIs', { 'Fn::ImportValue': 'Stack1:ExportsOutputRefAWSAccountIdAD568057' } ]] }\n        }\n      }\n    });\n\n    test.done();\n  },\n\n  'cannot create cyclic reference between stacks'(test: Test) {\n    // GIVEN\n    const app = new App();\n    const stack1 = new Stack(app, 'Stack1');\n    const account1 = new ScopedAws(stack1).accountId;\n    const stack2 = new Stack(app, 'Stack2');\n    const account2 = new ScopedAws(stack2).accountId;\n\n    // WHEN\n    new CfnParameter(stack2, 'SomeParameter', { type: 'String', default: account1 });\n    new CfnParameter(stack1, 'SomeParameter', { type: 'String', default: account2 });\n\n    test.throws(() => {\n      app.node.prepareTree();\n      // tslint:disable-next-line:max-line-length\n    }, \"'Stack2' depends on 'Stack1' (Stack2/SomeParameter -> Stack1.AWS::AccountId). Adding this dependency (Stack1/SomeParameter -> Stack2.AWS::AccountId) would create a cyclic reference.\");\n\n    test.done();\n  },\n\n  'stacks know about their dependencies'(test: Test) {\n    // GIVEN\n    const app = new App();\n    const stack1 = new Stack(app, 'Stack1');\n    const account1 = new ScopedAws(stack1).accountId;\n    const stack2 = new Stack(app, 'Stack2');\n\n    // WHEN\n    new CfnParameter(stack2, 'SomeParameter', { type: 'String', default: account1 });\n\n    app.node.prepareTree();\n\n    // THEN\n    test.deepEqual(stack2.dependencies().map(s => s.node.id), ['Stack1']);\n\n    test.done();\n  },\n\n  'cannot create references to stacks in other regions/accounts'(test: Test) {\n    // GIVEN\n    const app = new App();\n    const stack1 = new Stack(app, 'Stack1', { env: { account: '123456789012', region: 'es-norst-1' }});\n    const account1 = new ScopedAws(stack1).accountId;\n    const stack2 = new Stack(app, 'Stack2', { env: { account: '123456789012', region: 'es-norst-2' }});\n\n    // WHEN\n    new CfnParameter(stack2, 'SomeParameter', { type: 'String', default: account1 });\n\n    test.throws(() => {\n      app.node.prepareTree();\n    }, /Can only reference cross stacks in the same region and account/);\n\n    test.done();\n  },\n\n  'stack with region supplied via props returns literal value'(test: Test) {\n    // GIVEN\n    const app = new App();\n    const stack = new Stack(app, 'Stack1', { env: { account: '123456789012', region: 'es-norst-1' }});\n\n    // THEN\n    test.equal(stack.node.resolve(stack.region), 'es-norst-1');\n\n    test.done();\n  },\n\n  'stack with region supplied via context returns symbolic value'(test: Test) {\n    // GIVEN\n    const app = new App();\n\n    app.node.setContext(cxapi.DEFAULT_REGION_CONTEXT_KEY, 'es-norst-1');\n    const stack = new Stack(app, 'Stack1');\n\n    // THEN\n    test.deepEqual(stack.node.resolve(stack.region), { Ref: 'AWS::Region' });\n\n    test.done();\n  },\n\n  'overrideLogicalId(id) can be used to override the logical ID of a resource'(test: Test) {\n    // GIVEN\n    const stack = new Stack();\n    const bonjour = new CfnResource(stack, 'BonjourResource', { type: 'Resource::Type' });\n\n    // { Ref } and { GetAtt }\n    new CfnResource(stack, 'RefToBonjour', { type: 'Other::Resource', properties: {\n      RefToBonjour: bonjour.ref.toString(),\n      GetAttBonjour: bonjour.getAtt('TheAtt').toString()\n    }});\n\n    bonjour.overrideLogicalId('BOOM');\n\n    // THEN\n    test.deepEqual(stack._toCloudFormation(), { Resources:\n      { BOOM: { Type: 'Resource::Type' },\n        RefToBonjour:\n         { Type: 'Other::Resource',\n           Properties:\n            { RefToBonjour: { Ref: 'BOOM' },\n              GetAttBonjour: { 'Fn::GetAtt': [ 'BOOM', 'TheAtt' ] } } } } });\n    test.done();\n  },\n\n  'Stack name can be overridden via properties'(test: Test) {\n    // WHEN\n    const stack = new Stack(undefined, 'Stack', { stackName: 'otherName' });\n\n    // THEN\n    test.deepEqual(stack.name, 'otherName');\n\n    test.done();\n  },\n\n  'Stack name is inherited from App name if available'(test: Test) {\n    // WHEN\n    const root = new App();\n    const app = new Construct(root, 'Prod');\n    const stack = new Stack(app, 'Stack');\n\n    // THEN\n    test.deepEqual(stack.name, 'ProdStackD5279B22');\n\n    test.done();\n  },\n\n  'stack construct id does not go through stack name validation if there is an explicit stack name'(test: Test) {\n    // GIVEN\n    const app = new App();\n\n    // WHEN\n    const stack = new Stack(app, 'invalid as : stack name, but thats fine', {\n      stackName: 'valid-stack-name'\n    });\n\n    // THEN\n    const session = app.run();\n    test.deepEqual(stack.name, 'valid-stack-name');\n    test.ok(session.tryGetArtifact('valid-stack-name'));\n    test.done();\n  },\n\n  'stack validation is performed on explicit stack name'(test: Test) {\n    // GIVEN\n    const app = new App();\n\n    // THEN\n    test.throws(() => new Stack(app, 'boom', { stackName: 'invalid:stack:name' }),\n      /Stack name must match the regular expression/);\n\n    test.done();\n  }\n};\n\nclass StackWithPostProcessor extends Stack {\n\n  // ...\n\n  public _toCloudFormation() {\n    const template = super._toCloudFormation();\n\n    // manipulate template (e.g. rename \"Key\" to \"key\")\n    template.Resources.myResource.Properties.Environment.key =\n      template.Resources.myResource.Properties.Environment.Key;\n    delete template.Resources.myResource.Properties.Environment.Key;\n\n    return template;\n  }\n}\n"]}

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

'some random construct implements "synthesize"'(test: Test): void;
'backwards compatibility: cdk.out contains all synthesized stacks'(test: Test): void;
'it should be possible to synthesize without an app'(test: Test): void;
'store': any;
'stack.setParameterValue can be used to assign parameters'(test: Test): void;
};
export = _default;

@@ -8,7 +8,5 @@ "use strict";

const lib_1 = require("../lib");
const storeTestMatrix = {};
function createModernApp() {
return new cdk.App({
context: {
[cxapi.DISABLE_LEGACY_MANIFEST_CONTEXT]: 'true',
[cxapi.DISABLE_VERSION_REPORTING]: 'true',

@@ -18,181 +16,11 @@ }

}
//
// all these tests will be executed for each type of store
//
const storeTests = {
'writeFile()/readFile()'(test, store) {
// WHEN
store.writeFile('bla.txt', 'hello');
store.writeFile('hey.txt', '1234');
// THEN
test.deepEqual(store.readFile('bla.txt').toString(), 'hello');
test.deepEqual(store.readFile('hey.txt').toString(), '1234');
test.throws(() => store.writeFile('bla.txt', 'override is forbidden'));
// WHEN
store.lock();
// THEN
test.throws(() => store.writeFile('another.txt', 'locked!'));
test.done();
},
'exists() for files'(test, store) {
// WHEN
store.writeFile('A.txt', 'aaa');
// THEN
test.ok(store.exists('A.txt'));
test.ok(!store.exists('B.txt'));
test.done();
},
'mkdir'(test, store) {
// WHEN
const dir1 = store.mkdir('dir1');
const dir2 = store.mkdir('dir2');
// THEN
test.ok(fs.statSync(dir1).isDirectory());
test.ok(fs.statSync(dir2).isDirectory());
test.throws(() => store.mkdir('dir1'));
// WHEN
store.lock();
test.throws(() => store.mkdir('dir3'));
test.done();
},
'list'(test, store) {
// WHEN
store.mkdir('dir1');
store.writeFile('file1.txt', 'boom1');
// THEN
test.deepEqual(store.list(), ['dir1', 'file1.txt']);
test.done();
},
'SynthesisSession'(test, store) {
// GIVEN
const session = new lib_1.SynthesisSession({ store });
const templateFile = 'foo.template.json';
// WHEN
session.addArtifact('my-first-artifact', {
type: cxapi.ArtifactType.AwsCloudFormationStack,
environment: 'aws://1222344/us-east-1',
dependencies: ['a', 'b'],
metadata: {
foo: { bar: 123 }
},
properties: {
templateFile,
parameters: {
prop1: '1234',
prop2: '555'
}
},
missing: {
foo: {
provider: 'context-provider',
props: {
a: 'A',
b: 2
}
}
}
});
session.addArtifact('minimal-artifact', {
type: cxapi.ArtifactType.AwsCloudFormationStack,
environment: 'aws://111/helo-world',
properties: {
templateFile
}
});
session.store.writeJson(templateFile, {
Resources: {
MyTopic: {
Type: 'AWS::S3::Topic'
}
}
});
session.close();
const manifest = session.store.readJson(cxapi.MANIFEST_FILE);
// THEN
delete manifest.runtime; // deterministic tests
// verify the manifest looks right
test.deepEqual(manifest, {
version: cxapi.PROTO_RESPONSE_VERSION,
artifacts: {
'my-first-artifact': {
type: 'aws:cloudformation:stack',
environment: 'aws://1222344/us-east-1',
dependencies: ['a', 'b'],
metadata: { foo: { bar: 123 } },
properties: {
templateFile: 'foo.template.json',
parameters: {
prop1: '1234',
prop2: '555'
},
},
missing: {
foo: { provider: 'context-provider', props: { a: 'A', b: 2 } }
}
},
'minimal-artifact': {
type: 'aws:cloudformation:stack',
environment: 'aws://111/helo-world',
properties: { templateFile: 'foo.template.json' }
}
}
});
// verify we have a template file
test.deepEqual(session.store.readJson(templateFile), {
Resources: {
MyTopic: {
Type: 'AWS::S3::Topic'
}
}
});
test.done();
},
'stack.setParameterValue can be used to assign parameters'(test) {
// GIVEN
const app = createModernApp();
const stack = new cdk.Stack(app, 'my-stack');
const param = new cdk.CfnParameter(stack, 'MyParam', { type: 'string' });
// WHEN
stack.setParameterValue(param, 'Foo');
// THEN
const session = app.run();
const props = (session.manifest.artifacts && session.manifest.artifacts['my-stack'].properties) || {};
test.deepEqual(props.parameters, {
MyParam: 'Foo'
});
test.done();
},
'addBuildStep can be used to produce build.json'(test) {
// GIVEN
const app = createModernApp();
// WHEN
class BuildMe extends cdk.Construct {
synthesize(s) {
s.addBuildStep('step_id', {
type: 'build-step-type',
parameters: {
boom: 123
}
});
}
}
new BuildMe(app, 'hey');
// THEN
const session = app.run();
test.deepEqual(session.store.list(), ['build.json', 'manifest.json']);
test.deepEqual(session.store.readJson('build.json'), {
steps: {
step_id: { type: 'build-step-type', parameters: { boom: 123 } }
}
});
test.done();
}
};
for (const [name, fn] of Object.entries(storeTests)) {
const outdir = fs.mkdtempSync(path.join(os.tmpdir(), 'synthesis-tests'));
const fsStore = new lib_1.FileSystemStore({ outdir });
const memoryStore = new lib_1.InMemoryStore();
storeTestMatrix[`FileSystemStore - ${name}`] = (test) => fn(test, fsStore);
storeTestMatrix[`InMemoryStore - ${name}`] = (test) => fn(test, memoryStore);
function list(outdir) {
return fs.readdirSync(outdir).sort();
}
function readJson(outdir, file) {
return JSON.parse(fs.readFileSync(path.join(outdir, file), 'utf-8'));
}
function writeJson(outdir, file, data) {
fs.writeFileSync(path.join(outdir, file), JSON.stringify(data, undefined, 2));
}
module.exports = {

@@ -206,4 +34,4 @@ 'synthesis with an empty app'(test) {

test.same(app.run(), session); // same session if we run() again
test.deepEqual(session.store.list(), ['manifest.json']);
test.deepEqual(session.store.readJson('manifest.json').artifacts, {});
test.deepEqual(list(session.directory), ['cdk.out', 'manifest.json']);
test.deepEqual(readJson(session.directory, 'manifest.json').artifacts, {});
test.done();

@@ -218,3 +46,4 @@ },

// THEN
test.deepEqual(session.store.list(), [
test.deepEqual(list(session.directory), [
'cdk.out',
'manifest.json',

@@ -231,3 +60,3 @@ 'one-stack.template.json'

synthesize(s) {
s.store.writeJson('foo.json', { bar: 123 });
writeJson(s.outdir, 'foo.json', { bar: 123 });
s.addArtifact('my-random-construct', {

@@ -237,3 +66,3 @@ type: cxapi.ArtifactType.AwsCloudFormationStack,

properties: {
templateFile: 'file://boom'
templateFile: 'foo.json'
}

@@ -247,3 +76,4 @@ });

// THEN
test.deepEqual(session.store.list(), [
test.deepEqual(list(session.directory), [
'cdk.out',
'foo.json',

@@ -253,5 +83,5 @@ 'manifest.json',

]);
test.deepEqual(session.store.readJson('foo.json'), { bar: 123 });
test.deepEqual(readJson(session.directory, 'foo.json'), { bar: 123 });
test.deepEqual(session.manifest, {
version: '0.19.0',
version: '0.33.0',
artifacts: {

@@ -261,3 +91,3 @@ 'my-random-construct': {

environment: 'aws://12345/bar',
properties: { templateFile: 'file://boom' }
properties: { templateFile: 'foo.json' }
},

@@ -273,29 +103,2 @@ 'one-stack': {

},
'backwards compatibility: cdk.out contains all synthesized stacks'(test) {
// GIVEN
const app = new cdk.App();
const stack1 = new cdk.Stack(app, 'stack1');
new cdk.CfnResource(stack1, 'Resource1', { type: 'AWS::CDK::Resource' });
new cdk.CfnResource(stack1, 'Resource2', { type: 'AWS::CDK::Resource' });
const stack2 = new cdk.Stack(app, 'stack2');
new cdk.CfnResource(stack2, 'ResourceA', { type: 'AWS::CDK::Resource' });
// WHEN
const session = app.run();
const legacy = session.store.readJson(cxapi.OUTFILE_NAME);
// THEN
const t1 = legacy.stacks.find(s => s.name === 'stack1').template;
const t2 = legacy.stacks.find(s => s.name === 'stack2').template;
test.deepEqual(t1, {
Resources: {
Resource1: { Type: 'AWS::CDK::Resource' },
Resource2: { Type: 'AWS::CDK::Resource' }
}
});
test.deepEqual(t2, {
Resources: {
ResourceA: { Type: 'AWS::CDK::Resource' }
}
});
test.done();
},
'it should be possible to synthesize without an app'(test) {

@@ -317,6 +120,7 @@ const calls = new Array();

session.addArtifact('art', {
type: cxapi.ArtifactType.AwsEcrDockerImage,
type: cxapi.ArtifactType.AwsCloudFormationStack,
properties: { templateFile: 'hey.json' },
environment: 'aws://unknown-account/us-east-1'
});
session.store.writeJson('hey.json', { hello: 123 });
writeJson(session.outdir, 'hey.json', { hello: 123 });
}

@@ -326,13 +130,26 @@ }

const synth = new lib_1.Synthesizer();
const result = synth.synthesize(root);
const assembly = synth.synthesize(root, { outdir: fs.mkdtempSync(path.join(os.tmpdir(), 'outdir')) });
test.deepEqual(calls, ['prepare', 'validate', 'synthesize']);
test.deepEqual(result.store.readJson('hey.json'), { hello: 123 });
test.deepEqual(result.manifest.artifacts.art, {
type: 'aws:ecr:image',
environment: 'aws://unknown-account/us-east-1'
const stack = assembly.getStack('art');
test.deepEqual(stack.template, { hello: 123 });
test.deepEqual(stack.properties, { templateFile: 'hey.json' });
test.deepEqual(stack.environment, { region: 'us-east-1', account: 'unknown-account', name: 'aws://unknown-account/us-east-1' });
test.done();
},
'stack.setParameterValue can be used to assign parameters'(test) {
// GIVEN
const app = createModernApp();
const stack = new cdk.Stack(app, 'my-stack');
const param = new cdk.CfnParameter(stack, 'MyParam', { type: 'string' });
// WHEN
stack.setParameterValue(param, 'Foo');
// THEN
const session = app.run();
const props = session.getStack('my-stack').properties;
test.deepEqual(props.parameters, {
MyParam: 'Foo'
});
test.done();
},
'store': storeTestMatrix
};
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"test.synthesis.js","sourceRoot":"","sources":["test.synthesis.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 cdk = require('../lib');\nimport { Construct, FileSystemStore, InMemoryStore, ISynthesisSession, SynthesisSession, Synthesizer } from '../lib';\n\nconst storeTestMatrix: any = {};\n\nfunction createModernApp() {\n  return new cdk.App({\n    context: {\n      [cxapi.DISABLE_LEGACY_MANIFEST_CONTEXT]: 'true',\n      [cxapi.DISABLE_VERSION_REPORTING]: 'true', // for test reproducibility\n    }\n  });\n}\n\nexport = {\n  'synthesis with an empty app'(test: Test) {\n    // GIVEN\n    const app = createModernApp();\n\n    // WHEN\n    const session = app.run();\n\n    // THEN\n    test.same(app.run(), session); // same session if we run() again\n    test.deepEqual(session.store.list(), [ 'manifest.json' ]);\n    test.deepEqual(session.store.readJson('manifest.json').artifacts, {});\n    test.done();\n  },\n\n  'single empty stack'(test: Test) {\n    // GIVEN\n    const app = createModernApp();\n    new cdk.Stack(app, 'one-stack');\n\n    // WHEN\n    const session = app.run();\n\n    // THEN\n    test.deepEqual(session.store.list(), [\n      'manifest.json',\n      'one-stack.template.json'\n    ]);\n    test.done();\n  },\n\n  'some random construct implements \"synthesize\"'(test: Test) {\n    // GIVEN\n    const app = createModernApp();\n    const stack = new cdk.Stack(app, 'one-stack');\n\n    class MyConstruct extends cdk.Construct implements cdk.ISynthesizable {\n      public synthesize(s: cdk.ISynthesisSession) {\n        s.store.writeJson('foo.json', { bar: 123 });\n        s.addArtifact('my-random-construct', {\n          type: cxapi.ArtifactType.AwsCloudFormationStack,\n          environment: 'aws://12345/bar',\n          properties: {\n            templateFile: 'file://boom'\n          }\n        });\n      }\n    }\n\n    new MyConstruct(stack, 'MyConstruct');\n\n    // WHEN\n    const session = app.run();\n\n    // THEN\n    test.deepEqual(session.store.list(), [\n      'foo.json',\n      'manifest.json',\n      'one-stack.template.json'\n    ]);\n    test.deepEqual(session.store.readJson('foo.json'), { bar: 123 });\n    test.deepEqual(session.manifest, {\n      version: '0.19.0',\n      artifacts: {\n        'my-random-construct': {\n          type: 'aws:cloudformation:stack',\n          environment: 'aws://12345/bar',\n          properties: { templateFile: 'file://boom' }\n        },\n        'one-stack': {\n          type: 'aws:cloudformation:stack',\n          environment: 'aws://unknown-account/unknown-region',\n          properties: { templateFile: 'one-stack.template.json' },\n        }\n      },\n    });\n    test.done();\n  },\n\n  'backwards compatibility: cdk.out contains all synthesized stacks'(test: Test) {\n    // GIVEN\n    const app = new cdk.App();\n    const stack1 = new cdk.Stack(app, 'stack1');\n    new cdk.CfnResource(stack1, 'Resource1', { type: 'AWS::CDK::Resource' });\n    new cdk.CfnResource(stack1, 'Resource2', { type: 'AWS::CDK::Resource' });\n    const stack2 = new cdk.Stack(app, 'stack2');\n    new cdk.CfnResource(stack2, 'ResourceA', { type: 'AWS::CDK::Resource' });\n\n    // WHEN\n    const session = app.run();\n    const legacy: cxapi.SynthesizeResponse = session.store.readJson(cxapi.OUTFILE_NAME);\n\n    // THEN\n    const t1 = legacy.stacks.find(s => s.name === 'stack1')!.template;\n    const t2 = legacy.stacks.find(s => s.name === 'stack2')!.template;\n\n    test.deepEqual(t1, {\n      Resources: {\n        Resource1: { Type: 'AWS::CDK::Resource' },\n        Resource2: { Type: 'AWS::CDK::Resource' }\n      }\n    });\n    test.deepEqual(t2, {\n      Resources: {\n        ResourceA: { Type: 'AWS::CDK::Resource' }\n      }\n    });\n    test.done();\n  },\n\n  'it should be possible to synthesize without an app'(test: Test) {\n    const calls = new Array<string>();\n\n    class SynthesizeMe extends Construct {\n      constructor() {\n        super(undefined as any, 'id');\n      }\n\n      protected validate(): string[] {\n        calls.push('validate');\n        return [];\n      }\n\n      protected prepare(): void {\n        calls.push('prepare');\n      }\n\n      protected synthesize(session: ISynthesisSession) {\n        calls.push('synthesize');\n\n        session.addArtifact('art', {\n          type: cxapi.ArtifactType.AwsEcrDockerImage,\n          environment: 'aws://unknown-account/us-east-1'\n        });\n\n        session.store.writeJson('hey.json', { hello: 123 });\n      }\n    }\n\n    const root = new SynthesizeMe();\n\n    const synth = new Synthesizer();\n    const result = synth.synthesize(root);\n\n    test.deepEqual(calls, [ 'prepare', 'validate', 'synthesize' ]);\n    test.deepEqual(result.store.readJson('hey.json'), { hello: 123 });\n    test.deepEqual(result.manifest.artifacts!.art, {\n      type: 'aws:ecr:image',\n      environment: 'aws://unknown-account/us-east-1'\n    });\n    test.done();\n  },\n\n  'store': storeTestMatrix\n};\n\n//\n// all these tests will be executed for each type of store\n//\nconst storeTests = {\n  'writeFile()/readFile()'(test: Test, store: cdk.ISessionStore) {\n    // WHEN\n    store.writeFile('bla.txt', 'hello');\n    store.writeFile('hey.txt', '1234');\n\n    // THEN\n    test.deepEqual(store.readFile('bla.txt').toString(), 'hello');\n    test.deepEqual(store.readFile('hey.txt').toString(), '1234');\n    test.throws(() => store.writeFile('bla.txt', 'override is forbidden'));\n\n    // WHEN\n    store.lock();\n\n    // THEN\n    test.throws(() => store.writeFile('another.txt', 'locked!'));\n    test.done();\n  },\n\n  'exists() for files'(test: Test, store: cdk.ISessionStore) {\n    // WHEN\n    store.writeFile('A.txt', 'aaa');\n\n    // THEN\n    test.ok(store.exists('A.txt'));\n    test.ok(!store.exists('B.txt'));\n    test.done();\n  },\n\n  'mkdir'(test: Test, store: cdk.ISessionStore) {\n    // WHEN\n    const dir1 = store.mkdir('dir1');\n    const dir2 = store.mkdir('dir2');\n\n    // THEN\n    test.ok(fs.statSync(dir1).isDirectory());\n    test.ok(fs.statSync(dir2).isDirectory());\n    test.throws(() => store.mkdir('dir1'));\n\n    // WHEN\n    store.lock();\n    test.throws(() => store.mkdir('dir3'));\n    test.done();\n  },\n\n  'list'(test: Test, store: cdk.ISessionStore) {\n    // WHEN\n    store.mkdir('dir1');\n    store.writeFile('file1.txt', 'boom1');\n\n    // THEN\n    test.deepEqual(store.list(), ['dir1', 'file1.txt']);\n    test.done();\n  },\n\n  'SynthesisSession'(test: Test, store: cdk.ISessionStore) {\n    // GIVEN\n    const session = new SynthesisSession({ store });\n    const templateFile = 'foo.template.json';\n\n    // WHEN\n    session.addArtifact('my-first-artifact', {\n      type: cxapi.ArtifactType.AwsCloudFormationStack,\n      environment: 'aws://1222344/us-east-1',\n      dependencies: ['a', 'b'],\n      metadata: {\n        foo: { bar: 123 }\n      },\n      properties: {\n        templateFile,\n        parameters: {\n          prop1: '1234',\n          prop2: '555'\n        }\n      },\n      missing: {\n        foo: {\n          provider: 'context-provider',\n          props: {\n            a: 'A',\n            b: 2\n          }\n        }\n      }\n    });\n\n    session.addArtifact('minimal-artifact', {\n      type: cxapi.ArtifactType.AwsCloudFormationStack,\n      environment: 'aws://111/helo-world',\n      properties: {\n        templateFile\n      }\n    });\n\n    session.store.writeJson(templateFile, {\n      Resources: {\n        MyTopic: {\n          Type: 'AWS::S3::Topic'\n        }\n      }\n    });\n\n    session.close();\n\n    const manifest = session.store.readJson(cxapi.MANIFEST_FILE);\n\n    // THEN\n    delete manifest.runtime; // deterministic tests\n\n    // verify the manifest looks right\n    test.deepEqual(manifest, {\n      version: cxapi.PROTO_RESPONSE_VERSION,\n      artifacts: {\n        'my-first-artifact': {\n          type: 'aws:cloudformation:stack',\n          environment: 'aws://1222344/us-east-1',\n          dependencies: ['a', 'b'],\n          metadata: { foo: { bar: 123 } },\n          properties: {\n            templateFile: 'foo.template.json',\n            parameters: {\n              prop1: '1234',\n              prop2: '555'\n            },\n          },\n          missing: {\n            foo: { provider: 'context-provider', props: { a: 'A', b: 2 } }\n          }\n        },\n        'minimal-artifact': {\n          type: 'aws:cloudformation:stack',\n          environment: 'aws://111/helo-world',\n          properties: { templateFile: 'foo.template.json' }\n        }\n      }\n    });\n\n    // verify we have a template file\n    test.deepEqual(session.store.readJson(templateFile), {\n      Resources: {\n        MyTopic: {\n          Type: 'AWS::S3::Topic'\n        }\n      }\n    });\n\n    test.done();\n  },\n\n  'stack.setParameterValue can be used to assign parameters'(test: Test) {\n    // GIVEN\n    const app = createModernApp();\n    const stack = new cdk.Stack(app, 'my-stack');\n    const param = new cdk.CfnParameter(stack, 'MyParam', { type: 'string' });\n\n    // WHEN\n    stack.setParameterValue(param, 'Foo');\n\n    // THEN\n    const session = app.run();\n    const props = (session.manifest.artifacts && session.manifest.artifacts['my-stack'].properties) || {};\n    test.deepEqual(props.parameters, {\n      MyParam: 'Foo'\n    });\n    test.done();\n  },\n\n  'addBuildStep can be used to produce build.json'(test: Test) {\n    // GIVEN\n    const app = createModernApp();\n\n    // WHEN\n    class BuildMe extends cdk.Construct implements cdk.ISynthesizable {\n      public synthesize(s: cdk.ISynthesisSession) {\n        s.addBuildStep('step_id', {\n          type: 'build-step-type',\n          parameters: {\n            boom: 123\n          }\n        });\n      }\n    }\n\n    new BuildMe(app, 'hey');\n\n    // THEN\n    const session  = app.run();\n    test.deepEqual(session.store.list(), [ 'build.json', 'manifest.json' ]);\n    test.deepEqual(session.store.readJson('build.json'), {\n      steps: {\n        step_id: { type: 'build-step-type', parameters: { boom: 123 } }\n      }\n    });\n    test.done();\n  }\n};\n\nfor (const [name, fn] of Object.entries(storeTests)) {\n  const outdir = fs.mkdtempSync(path.join(os.tmpdir(), 'synthesis-tests'));\n  const fsStore = new FileSystemStore({ outdir });\n  const memoryStore = new InMemoryStore();\n  storeTestMatrix[`FileSystemStore - ${name}`] = (test: Test) => fn(test, fsStore);\n  storeTestMatrix[`InMemoryStore - ${name}`] = (test: Test) => fn(test, memoryStore);\n}\n"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"test.synthesis.js","sourceRoot":"","sources":["test.synthesis.ts"],"names":[],"mappings":";AAAA,yCAA0C;AAE1C,yBAA0B;AAE1B,yBAA0B;AAC1B,6BAA8B;AAC9B,8BAA+B;AAC/B,gCAAgD;AAEhD,SAAS,eAAe;IACtB,OAAO,IAAI,GAAG,CAAC,GAAG,CAAC;QACjB,OAAO,EAAE;YACP,CAAC,KAAK,CAAC,yBAAyB,CAAC,EAAE,MAAM;SAC1C;KACF,CAAC,CAAC;AACL,CAAC;AAiJD,SAAS,IAAI,CAAC,MAAc;IAC1B,OAAO,EAAE,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC,IAAI,EAAE,CAAC;AACvC,CAAC;AAED,SAAS,QAAQ,CAAC,MAAc,EAAE,IAAY;IAC5C,OAAO,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,IAAI,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC;AACvE,CAAC;AAED,SAAS,SAAS,CAAC,MAAc,EAAE,IAAY,EAAE,IAAS;IACxD,EAAE,CAAC,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,IAAI,CAAC,EAAE,IAAI,CAAC,SAAS,CAAC,IAAI,EAAE,SAAS,EAAE,CAAC,CAAC,CAAC,CAAC;AAChF,CAAC;AAzJD,iBAAS;IACP,6BAA6B,CAAC,IAAU;QACtC,QAAQ;QACR,MAAM,GAAG,GAAG,eAAe,EAAE,CAAC;QAE9B,OAAO;QACP,MAAM,OAAO,GAAG,GAAG,CAAC,GAAG,EAAE,CAAC;QAE1B,OAAO;QACP,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,EAAE,OAAO,CAAC,CAAC,CAAC,iCAAiC;QAChE,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,EAAE,CAAE,SAAS,EAAE,eAAe,CAAE,CAAC,CAAC;QACxE,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,OAAO,CAAC,SAAS,EAAE,eAAe,CAAC,CAAC,SAAS,EAAE,EAAE,CAAC,CAAC;QAC3E,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED,oBAAoB,CAAC,IAAU;QAC7B,QAAQ;QACR,MAAM,GAAG,GAAG,eAAe,EAAE,CAAC;QAC9B,IAAI,GAAG,CAAC,KAAK,CAAC,GAAG,EAAE,WAAW,CAAC,CAAC;QAEhC,OAAO;QACP,MAAM,OAAO,GAAG,GAAG,CAAC,GAAG,EAAE,CAAC;QAE1B,OAAO;QACP,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,EAAE;YACtC,SAAS;YACT,eAAe;YACf,yBAAyB;SAC1B,CAAC,CAAC;QACH,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED,+CAA+C,CAAC,IAAU;QACxD,QAAQ;QACR,MAAM,GAAG,GAAG,eAAe,EAAE,CAAC;QAC9B,MAAM,KAAK,GAAG,IAAI,GAAG,CAAC,KAAK,CAAC,GAAG,EAAE,WAAW,CAAC,CAAC;QAE9C,MAAM,WAAY,SAAQ,GAAG,CAAC,SAAS;YAC9B,UAAU,CAAC,CAA6B;gBAC7C,SAAS,CAAC,CAAC,CAAC,MAAM,EAAE,UAAU,EAAE,EAAE,GAAG,EAAE,GAAG,EAAE,CAAC,CAAC;gBAC9C,CAAC,CAAC,WAAW,CAAC,qBAAqB,EAAE;oBACnC,IAAI,EAAE,KAAK,CAAC,YAAY,CAAC,sBAAsB;oBAC/C,WAAW,EAAE,iBAAiB;oBAC9B,UAAU,EAAE;wBACV,YAAY,EAAE,UAAU;qBACzB;iBACF,CAAC,CAAC;YACL,CAAC;SACF;QAED,IAAI,WAAW,CAAC,KAAK,EAAE,aAAa,CAAC,CAAC;QAEtC,OAAO;QACP,MAAM,OAAO,GAAG,GAAG,CAAC,GAAG,EAAE,CAAC;QAE1B,OAAO;QACP,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,EAAE;YACtC,SAAS;YACT,UAAU;YACV,eAAe;YACf,yBAAyB;SAC1B,CAAC,CAAC;QACH,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,OAAO,CAAC,SAAS,EAAE,UAAU,CAAC,EAAE,EAAE,GAAG,EAAE,GAAG,EAAE,CAAC,CAAC;QACtE,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,QAAQ,EAAE;YAC/B,OAAO,EAAE,QAAQ;YACjB,SAAS,EAAE;gBACT,qBAAqB,EAAE;oBACrB,IAAI,EAAE,0BAA0B;oBAChC,WAAW,EAAE,iBAAiB;oBAC9B,UAAU,EAAE,EAAE,YAAY,EAAE,UAAU,EAAE;iBACzC;gBACD,WAAW,EAAE;oBACX,IAAI,EAAE,0BAA0B;oBAChC,WAAW,EAAE,sCAAsC;oBACnD,UAAU,EAAE,EAAE,YAAY,EAAE,yBAAyB,EAAE;iBACxD;aACF;SACF,CAAC,CAAC;QACH,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED,oDAAoD,CAAC,IAAU;QAC7D,MAAM,KAAK,GAAG,IAAI,KAAK,EAAU,CAAC;QAElC,MAAM,YAAa,SAAQ,eAAS;YAClC;gBACE,KAAK,CAAC,SAAgB,EAAE,IAAI,CAAC,CAAC;YAChC,CAAC;YAES,QAAQ;gBAChB,KAAK,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;gBACvB,OAAO,EAAE,CAAC;YACZ,CAAC;YAES,OAAO;gBACf,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;YACxB,CAAC;YAES,UAAU,CAAC,OAA6B;gBAChD,KAAK,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;gBAEzB,OAAO,CAAC,WAAW,CAAC,KAAK,EAAE;oBACzB,IAAI,EAAE,KAAK,CAAC,YAAY,CAAC,sBAAsB;oBAC/C,UAAU,EAAE,EAAE,YAAY,EAAE,UAAU,EAAE;oBACxC,WAAW,EAAE,iCAAiC;iBAC/C,CAAC,CAAC;gBAEH,SAAS,CAAC,OAAO,CAAC,MAAM,EAAE,UAAU,EAAE,EAAE,KAAK,EAAE,GAAG,EAAE,CAAC,CAAC;YACxD,CAAC;SACF;QAED,MAAM,IAAI,GAAG,IAAI,YAAY,EAAE,CAAC;QAEhC,MAAM,KAAK,GAAG,IAAI,iBAAW,EAAE,CAAC;QAChC,MAAM,QAAQ,GAAG,KAAK,CAAC,UAAU,CAAC,IAAI,EAAE,EAAE,MAAM,EAAE,EAAE,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,MAAM,EAAE,EAAE,QAAQ,CAAC,CAAC,EAAE,CAAC,CAAC;QAEtG,IAAI,CAAC,SAAS,CAAC,KAAK,EAAE,CAAE,SAAS,EAAE,UAAU,EAAE,YAAY,CAAE,CAAC,CAAC;QAC/D,MAAM,KAAK,GAAG,QAAQ,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;QACvC,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,QAAQ,EAAE,EAAE,KAAK,EAAE,GAAG,EAAE,CAAC,CAAC;QAC/C,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,UAAU,EAAE,EAAE,YAAY,EAAE,UAAU,EAAE,CAAC,CAAC;QAC/D,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,WAAW,EAAE,EAAE,MAAM,EAAE,WAAW,EAAE,OAAO,EAAE,iBAAiB,EAAE,IAAI,EAAE,iCAAiC,EAAE,CAAC,CAAC;QAChI,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED,0DAA0D,CAAC,IAAU;QACnE,QAAQ;QACR,MAAM,GAAG,GAAG,eAAe,EAAE,CAAC;QAC9B,MAAM,KAAK,GAAG,IAAI,GAAG,CAAC,KAAK,CAAC,GAAG,EAAE,UAAU,CAAC,CAAC;QAC7C,MAAM,KAAK,GAAG,IAAI,GAAG,CAAC,YAAY,CAAC,KAAK,EAAE,SAAS,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE,CAAC,CAAC;QAEzE,OAAO;QACP,KAAK,CAAC,iBAAiB,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC;QAEtC,OAAO;QACP,MAAM,OAAO,GAAG,GAAG,CAAC,GAAG,EAAE,CAAC;QAC1B,MAAM,KAAK,GAAG,OAAO,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC,UAAU,CAAC;QACtD,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,UAAU,EAAE;YAC/B,OAAO,EAAE,KAAK;SACf,CAAC,CAAC;QACH,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;CACF,CAAC","sourcesContent":["import cxapi = require('@aws-cdk/cx-api');\nimport { CloudAssemblyBuilder } from '@aws-cdk/cx-api';\nimport fs = require('fs');\nimport { Test } from 'nodeunit';\nimport os = require('os');\nimport path = require('path');\nimport cdk = require('../lib');\nimport { Construct, Synthesizer } from '../lib';\n\nfunction createModernApp() {\n  return new cdk.App({\n    context: {\n      [cxapi.DISABLE_VERSION_REPORTING]: 'true', // for test reproducibility\n    }\n  });\n}\n\nexport = {\n  'synthesis with an empty app'(test: Test) {\n    // GIVEN\n    const app = createModernApp();\n\n    // WHEN\n    const session = app.run();\n\n    // THEN\n    test.same(app.run(), session); // same session if we run() again\n    test.deepEqual(list(session.directory), [ 'cdk.out', 'manifest.json' ]);\n    test.deepEqual(readJson(session.directory, 'manifest.json').artifacts, {});\n    test.done();\n  },\n\n  'single empty stack'(test: Test) {\n    // GIVEN\n    const app = createModernApp();\n    new cdk.Stack(app, 'one-stack');\n\n    // WHEN\n    const session = app.run();\n\n    // THEN\n    test.deepEqual(list(session.directory), [\n      'cdk.out',\n      'manifest.json',\n      'one-stack.template.json'\n    ]);\n    test.done();\n  },\n\n  'some random construct implements \"synthesize\"'(test: Test) {\n    // GIVEN\n    const app = createModernApp();\n    const stack = new cdk.Stack(app, 'one-stack');\n\n    class MyConstruct extends cdk.Construct implements cdk.ISynthesizable {\n      public synthesize(s: cxapi.CloudAssemblyBuilder) {\n        writeJson(s.outdir, 'foo.json', { bar: 123 });\n        s.addArtifact('my-random-construct', {\n          type: cxapi.ArtifactType.AwsCloudFormationStack,\n          environment: 'aws://12345/bar',\n          properties: {\n            templateFile: 'foo.json'\n          }\n        });\n      }\n    }\n\n    new MyConstruct(stack, 'MyConstruct');\n\n    // WHEN\n    const session = app.run();\n\n    // THEN\n    test.deepEqual(list(session.directory), [\n      'cdk.out',\n      'foo.json',\n      'manifest.json',\n      'one-stack.template.json'\n    ]);\n    test.deepEqual(readJson(session.directory, 'foo.json'), { bar: 123 });\n    test.deepEqual(session.manifest, {\n      version: '0.33.0',\n      artifacts: {\n        'my-random-construct': {\n          type: 'aws:cloudformation:stack',\n          environment: 'aws://12345/bar',\n          properties: { templateFile: 'foo.json' }\n        },\n        'one-stack': {\n          type: 'aws:cloudformation:stack',\n          environment: 'aws://unknown-account/unknown-region',\n          properties: { templateFile: 'one-stack.template.json' },\n        }\n      },\n    });\n    test.done();\n  },\n\n  'it should be possible to synthesize without an app'(test: Test) {\n    const calls = new Array<string>();\n\n    class SynthesizeMe extends Construct {\n      constructor() {\n        super(undefined as any, 'id');\n      }\n\n      protected validate(): string[] {\n        calls.push('validate');\n        return [];\n      }\n\n      protected prepare(): void {\n        calls.push('prepare');\n      }\n\n      protected synthesize(session: CloudAssemblyBuilder) {\n        calls.push('synthesize');\n\n        session.addArtifact('art', {\n          type: cxapi.ArtifactType.AwsCloudFormationStack,\n          properties: { templateFile: 'hey.json' },\n          environment: 'aws://unknown-account/us-east-1'\n        });\n\n        writeJson(session.outdir, 'hey.json', { hello: 123 });\n      }\n    }\n\n    const root = new SynthesizeMe();\n\n    const synth = new Synthesizer();\n    const assembly = synth.synthesize(root, { outdir: fs.mkdtempSync(path.join(os.tmpdir(), 'outdir')) });\n\n    test.deepEqual(calls, [ 'prepare', 'validate', 'synthesize' ]);\n    const stack = assembly.getStack('art');\n    test.deepEqual(stack.template, { hello: 123 });\n    test.deepEqual(stack.properties, { templateFile: 'hey.json' });\n    test.deepEqual(stack.environment, { region: 'us-east-1', account: 'unknown-account', name: 'aws://unknown-account/us-east-1' });\n    test.done();\n  },\n\n  'stack.setParameterValue can be used to assign parameters'(test: Test) {\n    // GIVEN\n    const app = createModernApp();\n    const stack = new cdk.Stack(app, 'my-stack');\n    const param = new cdk.CfnParameter(stack, 'MyParam', { type: 'string' });\n\n    // WHEN\n    stack.setParameterValue(param, 'Foo');\n\n    // THEN\n    const session = app.run();\n    const props = session.getStack('my-stack').properties;\n    test.deepEqual(props.parameters, {\n      MyParam: 'Foo'\n    });\n    test.done();\n  },\n};\n\nfunction list(outdir: string) {\n  return fs.readdirSync(outdir).sort();\n}\n\nfunction readJson(outdir: string, file: string) {\n  return JSON.parse(fs.readFileSync(path.join(outdir, file), 'utf-8'));\n}\n\nfunction writeJson(outdir: string, file: string, data: any) {\n  fs.writeFileSync(path.join(outdir, file), JSON.stringify(data, undefined, 2));\n}"]}

@@ -41,3 +41,5 @@ import { Test } from 'nodeunit';

};
'stack trace is captured at token creation'(test: Test): void;
'newError returns an error with the creation stack trace'(test: Test): void;
};
export = _default;

@@ -452,3 +452,39 @@ "use strict";

},
'stack trace is captured at token creation'(test) {
function fn1() {
function fn2() {
class ExposeTrace extends lib_1.Token {
get creationTrace() {
return this.trace;
}
}
return new ExposeTrace(() => 'hello');
}
return fn2();
}
const token = fn1();
test.ok(token.creationTrace.find(x => x.includes('fn1')));
test.ok(token.creationTrace.find(x => x.includes('fn2')));
test.done();
},
'newError returns an error with the creation stack trace'(test) {
function fn1() {
function fn2() {
function fn3() {
class ThrowingToken extends lib_1.Token {
throwError(message) {
throw this.newError(message);
}
}
return new ThrowingToken('boom');
}
return fn3();
}
return fn2();
}
const token = fn1();
test.throws(() => token.throwError('message!'), /Token created:/);
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 { App as Root, findTokens, Fn, Stack, Token } from '../lib';\nimport { createTokenDouble, extractTokenDouble } from '../lib/encoding';\nimport { TokenMap } from '../lib/token-map';\nimport { evaluateCFN } from './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(Token.isToken({ resolve: () => 123 }));\n    test.ok(Token.isToken({ a: 1, b: 2, resolve: () => 'hello' }));\n    test.ok(!Token.isToken({ 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 can be reversed'(test: Test) {\n    // GIVEN\n    const token = new Token(() => 'woof woof');\n\n    // THEN\n    test.equal(token, TokenMap.instance().lookupString(`${token}`));\n    test.done();\n  },\n\n  'concatenated tokens are undefined'(test: Test) {\n    // GIVEN\n    const token = new Token(() => 'woof woof');\n\n    // WHEN\n    test.equal(undefined, TokenMap.instance().lookupString(`${token}bla`));\n    test.equal(undefined, TokenMap.instance().lookupString(`bla${token}`));\n    test.equal(undefined, TokenMap.instance().lookupString(`bla`));\n\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 Token(() => ({ 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  'tokens can be nested in hash keys'(test: Test) {\n    // GIVEN\n    const token = new Token(() => new 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  'tokens can be nested and concatenated in hash keys'(test: Test) {\n    // GIVEN\n    const innerToken = new Token(() => 'toot');\n    const token = new Token(() => `${innerToken} the woot`);\n\n    // WHEN\n    const s = {\n      [token.toString()]: `boom chicago`\n    };\n\n    // THEN\n    test.deepEqual(resolve(s), { 'toot the woot': 'boom chicago' });\n    test.done();\n  },\n\n  'can find nested tokens in hash keys'(test: Test) {\n    // GIVEN\n    const innerToken = new Token(() => 'toot');\n    const token = new Token(() => `${innerToken} the woot`);\n\n    // WHEN\n    const s = {\n      [token.toString()]: `boom chicago`\n    };\n\n    // THEN\n    const tokens = findTokens(new Stack(), () => s);\n    test.ok(tokens.some(t => t === innerToken), 'Cannot find innerToken');\n    test.ok(tokens.some(t => t === token), 'Cannot find token');\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 Token({ 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  'list encoding': {\n    'can encode Token to string and resolve the encoding'(test: Test) {\n      // GIVEN\n      const token = new Token({ Ref: 'Other' });\n\n      // WHEN\n      const struct = {\n        XYZ: token.toList()\n      };\n\n      // THEN\n      test.deepEqual(resolve(struct), {\n        XYZ: { Ref: 'Other'}\n      });\n\n      test.done();\n    },\n\n    'cannot add to encoded list'(test: Test) {\n      // GIVEN\n      const token = new Token({ Ref: 'Other' });\n\n      // WHEN\n      const encoded: string[] = token.toList();\n      encoded.push('hello');\n\n      // THEN\n      test.throws(() => {\n        resolve(encoded);\n      }, /Cannot add elements to list token/);\n\n      test.done();\n    },\n\n    'cannot add to strings in encoded list'(test: Test) {\n      // GIVEN\n      const token = new Token({ Ref: 'Other' });\n\n      // WHEN\n      const encoded: string[] = token.toList();\n      encoded[0] += 'hello';\n\n      // THEN\n      test.throws(() => {\n        resolve(encoded);\n      }, /concatenate strings in/);\n\n      test.done();\n    },\n\n    'can pass encoded lists to FnSelect'(test: Test) {\n      // GIVEN\n      const encoded: string[] = new Token({ Ref: 'Other' }).toList();\n\n      // WHEN\n      const struct = Fn.select(1, encoded);\n\n      // THEN\n      test.deepEqual(resolve(struct), {\n        'Fn::Select': [1, { Ref: 'Other'}]\n      });\n\n      test.done();\n    },\n\n    'can pass encoded lists to FnJoin'(test: Test) {\n      // GIVEN\n      const encoded: string[] = new Token({ Ref: 'Other' }).toList();\n\n      // WHEN\n      const struct = Fn.join('/', encoded);\n\n      // THEN\n      test.deepEqual(resolve(struct), {\n        'Fn::Join': ['/', { Ref: 'Other'}]\n      });\n\n      test.done();\n    },\n\n    'can pass encoded lists to FnJoin, even if join is stringified'(test: Test) {\n      // GIVEN\n      const encoded: string[] = new Token({ Ref: 'Other' }).toList();\n\n      // WHEN\n      const struct = Fn.join('/', encoded).toString();\n\n      // THEN\n      test.deepEqual(resolve(struct), {\n        'Fn::Join': ['/', { Ref: 'Other'}]\n      });\n\n      test.done();\n    },\n  },\n\n  'number encoding': {\n    'basic integer encoding works'(test: Test) {\n      test.equal(16, extractTokenDouble(createTokenDouble(16)));\n      test.done();\n    },\n\n    'arbitrary integers can be encoded, stringified, and recovered'(test: Test) {\n      for (let i = 0; i < 100; i++) {\n        // We can encode all numbers up to 2^48-1\n        const x = Math.floor(Math.random() * (Math.pow(2, 48) - 1));\n\n        const encoded = createTokenDouble(x);\n        // Roundtrip through JSONification\n        const roundtripped = JSON.parse(JSON.stringify({ theNumber: encoded })).theNumber;\n        const decoded = extractTokenDouble(roundtripped);\n        test.equal(decoded, x, `Fail roundtrip encoding of ${x}`);\n      }\n\n      test.done();\n    },\n\n    'arbitrary numbers are correctly detected as not being tokens'(test: Test) {\n      test.equal(undefined, extractTokenDouble(0));\n      test.equal(undefined, extractTokenDouble(1243));\n      test.equal(undefined, extractTokenDouble(4835e+532));\n\n      test.done();\n    },\n\n    'can number-encode and resolve Token objects'(test: Test) {\n      // GIVEN\n      const stack = new Stack();\n      const x = new Token(() => 123);\n\n      // THEN\n      const encoded = x.toNumber();\n      test.equal(true, Token.isToken(encoded), 'encoded number does not test as token');\n\n      // THEN\n      const resolved = stack.node.resolve({ value: encoded });\n      test.deepEqual(resolved, { value: 123 });\n\n      test.done();\n    },\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 Token(value),\n    new Token(() => 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\n/**\n * Wrapper for resolve that creates an throwaway Construct to call it on\n *\n * So I don't have to change all call sites in this file.\n */\nfunction resolve(x: any) {\n  return new Root().node.resolve(x);\n}\n"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"test.tokens.js","sourceRoot":"","sources":["test.tokens.ts"],"names":[],"mappings":";AACA,gCAAmE;AACnE,8CAAwE;AACxE,gDAA4C;AAC5C,iDAA6C;AA2gB7C,MAAM,QAAS,SAAQ,WAAK;IACnB,OAAO;QACZ,OAAO;YACL,IAAI,EAAE;gBACJ,UAAU,EAAE,OAAO;gBACnB,UAAU,EAAE,IAAI;aACjB;YACD,OAAO,EAAE,IAAI,WAAK,CAAC,GAAG,EAAE,CAAC,EAAE,CAAC;SAC7B,CAAC;IACJ,CAAC;CACF;AAED,MAAM,QAAS,SAAQ,WAAK;IAA5B;;QACS,OAAE,GAAG,CAAE,IAAI,QAAQ,EAAE,EAAE,IAAI,QAAQ,EAAE,CAAE,CAAC;IAKjD,CAAC;IAHQ,OAAO;QACZ,OAAO,IAAI,CAAC,EAAE,CAAC;IACjB,CAAC;CACF;AAED,MAAM,YAAY;IAChB,YAAqB,GAAW;QAAX,QAAG,GAAH,GAAG,CAAQ;IAChC,CAAC;CACF;AAED,MAAM,QAAS,SAAQ,YAAY;IAGjC;QACE,KAAK,CAAC,EAAE,CAAC,CAAC;QAHL,QAAG,GAAG,OAAO,CAAC;IAIrB,CAAC;CACF;AAED;;GAEG;AACH,SAAS,0BAA0B,CAAC,KAAU;IAC5C,OAAO;QACL,IAAI,WAAK,CAAC,KAAK,CAAC;QAChB,IAAI,WAAK,CAAC,GAAG,EAAE,CAAC,KAAK,CAAC;KACvB,CAAC;AACJ,CAAC;AAED;;GAEG;AACH,SAAS,iCAAiC,CAAC,KAAU;IACnD,OAAO;QACL,IAAI,WAAK,CAAC,KAAK,CAAC;QAChB,IAAI,WAAK,CAAC,GAAG,EAAE,CAAC,KAAK,CAAC;KACvB,CAAC;AACJ,CAAC;AAED;;GAEG;AACH,SAAS,mBAAmB,CAAC,KAAa;IACxC,OAAO,0BAA0B,CAAC,KAAK,CAAC,CAAC,MAAM,CAAC,iCAAiC,CAAC,KAAK,CAAC,CAAC,CAAC;AAC5F,CAAC;AAED;;;;GAIG;AACH,SAAS,OAAO,CAAC,CAAM;IACrB,OAAO,IAAI,SAAI,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC;AACpC,CAAC;AA5kBD,iBAAS;IACP,0DAA0D,CAAC,IAAU;QACnE,MAAM,GAAG,GAAG,EAAE,cAAc,EAAE,GAAG,EAAE,KAAK,EAAE,CAAE,CAAC,EAAE,CAAC,EAAE,CAAC,CAAE,EAAE,CAAC;QACxD,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE,GAAG,CAAC,CAAC;QAClC,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED,kEAAkE,CAAC,IAAU;QAC3E,MAAM,GAAG,GAAG;YACV,YAAY,EAAE,OAAO;YACrB,SAAS,EAAE,IAAI,WAAK,CAAC,OAAO,CAAC;SAC9B,CAAC;QAEF,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE;YAC3B,YAAY,EAAE,OAAO;YACrB,SAAS,EAAE,OAAO;SACnB,CAAC,CAAC;QAEH,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED,kDAAkD,CAAC,IAAU;QAC3D,MAAM,GAAG,GAAG,IAAI,QAAQ,EAAE,CAAC;QAC3B,MAAM,MAAM,GAAG,OAAO,CAAC,EAAE,GAAG,EAAE,GAAG,EAAE,CAAC,CAAC;QAErC,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE;YACrB,GAAG,EAAE;gBACH;oBACA,IAAI,EAAE;wBACJ,UAAU,EAAE,OAAO;wBACnB,UAAU,EAAE,IAAI;qBACjB;oBACD,OAAO,EAAE,EAAE;iBACV;gBACD;oBACA,IAAI,EAAE;wBACJ,UAAU,EAAE,OAAO;wBACnB,UAAU,EAAE,IAAI;qBACjB;oBACD,OAAO,EAAE,EAAE;iBACV;aACF;SACF,CAAC,CAAC;QAEH,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED,kCAAkC,CAAC,IAAU;QAC3C,MAAM,GAAG,GAAG,IAAI,QAAQ,EAAE,CAAC;QAC3B,MAAM,MAAM,GAAG,OAAO,CAAC,IAAI,WAAK,CAAC,GAAG,EAAE,CAAC,CAAC,EAAE,GAAG,EAAE,GAAG,EAAE,CAAC,CAAC,CAAC,CAAC;QAExD,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE;YACrB,GAAG,EAAE;gBACH;oBACA,IAAI,EAAE;wBACJ,UAAU,EAAE,OAAO;wBACnB,UAAU,EAAE,IAAI;qBACjB;oBACD,OAAO,EAAE,EAAE;iBACV;gBACD;oBACA,IAAI,EAAE;wBACJ,UAAU,EAAE,OAAO;wBACnB,UAAU,EAAE,IAAI;qBACjB;oBACD,OAAO,EAAE,EAAE;iBACV;aACF;SACF,CAAC,CAAC;QAEH,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED,kCAAkC,CAAC,IAAU;QAC3C,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,EAAG,CAAC,EAAE,EAAG,CAAC,CAAC;QAClC,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,EAAG,CAAC,EAAE,EAAG,CAAC,CAAC;QAElC,MAAM,GAAG,GAAG;YACV,KAAK,EAAE,IAAI;YACX,KAAK,EAAE,EAAG;YACV,KAAK,EAAE,EAAG;YACV,KAAK,EAAE,OAAO;YACd,KAAK,EAAE;gBACL,KAAK,EAAE,EAAG;gBACV,KAAK,EAAE;oBACL,KAAK,EAAE,CAAE,SAAS,EAAE,SAAS,CAAE;oBAC/B,KAAK,EAAE,QAAQ;iBAChB;aACF;SACF,CAAC;QAEF,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE;YAC3B,KAAK,EAAE,IAAI;YACX,KAAK,EAAE,EAAG;YACV,KAAK,EAAE,EAAG;YACV,KAAK,EAAE,OAAO;YACd,KAAK,EAAE;gBACL,KAAK,EAAE,EAAG;gBACV,KAAK,EAAE;oBACL,KAAK,EAAE,EAAG;oBACV,KAAK,EAAE,QAAQ;iBAChB;aACF;SACF,CAAC,CAAC;QAEH,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED,4FAA4F,CAAC,IAAU;QACrG,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,EAAE,OAAO,EAAE,EAAE,OAAO,EAAE,GAAG,EAAE,CAAC,KAAK,EAAE,EAAC,CAAC,EAAE,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC,CAAC;QAClF,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,EAAE,WAAW,EAAE,EAAE,OAAO,EAAE,EAAE,EAAE,EAAE,CAAC,EAAI,EAAE,WAAW,EAAE,EAAE,OAAO,EAAE,EAAE,EAAE,EAAE,CAAC,CAAC;QAC9F,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED,2CAA2C;IAC3C,iKAAiK,CAAC,IAAU;QAC1K,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,EAAE,IAAI,EAAE,IAAI,QAAQ,EAAE,EAAE,CAAC,EAAE,EAAE,IAAI,EAAE,EAAE,GAAG,EAAE,EAAE,EAAE,GAAG,EAAE,OAAO,EAAE,EAAE,CAAC,CAAC;QACvF,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED,+DAA+D,CAAC,IAAU;QACxE,IAAI,CAAC,EAAE,CAAC,WAAK,CAAC,OAAO,CAAC,EAAE,OAAO,EAAE,GAAG,EAAE,CAAC,GAAG,EAAE,CAAC,CAAC,CAAC;QAC/C,IAAI,CAAC,EAAE,CAAC,WAAK,CAAC,OAAO,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,OAAO,EAAE,GAAG,EAAE,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC;QAC/D,IAAI,CAAC,EAAE,CAAC,CAAC,WAAK,CAAC,OAAO,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,OAAO,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC;QACpD,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED,kEAAkE,CAAC,IAAU;QAC3E,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,IAAI,WAAK,CAAC,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;QACvC,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,IAAI,WAAK,CAAC,OAAO,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC;QACjD,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,IAAI,WAAK,CAAC,CAAE,IAAI,EAAE,OAAO,CAAE,CAAC,CAAC,EAAE,CAAE,IAAI,EAAE,OAAO,CAAE,CAAC,CAAC;QACzE,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED,iDAAiD,CAAC,IAAU;QAC1D,MAAM,IAAI,GAAG,IAAI,IAAI,CAAC,YAAY,CAAC,CAAC;QACpC,MAAM,QAAQ,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;QAE/B,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,QAAQ,EAAE,EAAE,QAAQ,CAAC,QAAQ,EAAE,CAAC,CAAC;QACjD,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED,6DAA6D,CAAC,IAAU;QACtE,QAAQ;QACR,MAAM,KAAK,GAAG,IAAI,WAAK,CAAC,GAAG,EAAE,CAAC,WAAW,CAAC,CAAC;QAE3C,OAAO;QACP,MAAM,WAAW,GAAG,iBAAiB,KAAK,EAAE,CAAC;QAC7C,MAAM,QAAQ,GAAG,OAAO,CAAC,WAAW,CAAC,CAAC;QAEtC,OAAO;QACP,IAAI,CAAC,SAAS,CAAC,0BAAW,CAAC,QAAQ,CAAC,EAAE,yBAAyB,CAAC,CAAC;QACjE,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED,wCAAwC,CAAC,IAAU;QACjD,QAAQ;QACR,MAAM,KAAK,GAAG,IAAI,WAAK,CAAC,GAAG,EAAE,CAAC,WAAW,CAAC,CAAC;QAE3C,OAAO;QACP,IAAI,CAAC,KAAK,CAAC,KAAK,EAAE,oBAAQ,CAAC,QAAQ,EAAE,CAAC,YAAY,CAAC,GAAG,KAAK,EAAE,CAAC,CAAC,CAAC;QAChE,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED,mCAAmC,CAAC,IAAU;QAC5C,QAAQ;QACR,MAAM,KAAK,GAAG,IAAI,WAAK,CAAC,GAAG,EAAE,CAAC,WAAW,CAAC,CAAC;QAE3C,OAAO;QACP,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,oBAAQ,CAAC,QAAQ,EAAE,CAAC,YAAY,CAAC,GAAG,KAAK,KAAK,CAAC,CAAC,CAAC;QACvE,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,oBAAQ,CAAC,QAAQ,EAAE,CAAC,YAAY,CAAC,MAAM,KAAK,EAAE,CAAC,CAAC,CAAC;QACvE,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,oBAAQ,CAAC,QAAQ,EAAE,CAAC,YAAY,CAAC,KAAK,CAAC,CAAC,CAAC;QAE/D,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED,6FAA6F,CAAC,IAAU;QACtG,QAAQ;QACR,MAAM,KAAK,GAAG,IAAI,WAAK,CAAC,GAAG,EAAE,CAAC,CAAC,EAAE,IAAI,EAAE,MAAM,EAAE,CAAC,CAAC,CAAC;QAElD,OAAO;QACP,MAAM,WAAW,GAAG,iBAAiB,KAAK,EAAE,CAAC;QAC7C,MAAM,QAAQ,GAAG,OAAO,CAAC,WAAW,CAAC,CAAC;QAEtC,OAAO;QACP,IAAI,CAAC,SAAS,CAAC,QAAQ,EAAE;YACvB,UAAU,EAAE,CAAC,EAAE,EAAE,CAAC,gBAAgB,EAAE,EAAE,IAAI,EAAE,MAAM,EAAE,CAAC,CAAC;SACvD,CAAC,CAAC;QACH,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED,4DAA4D,CAAC,IAAU;QACrE,QAAQ;QACR,MAAM,MAAM,GAAG,IAAI,WAAK,CAAC,GAAG,EAAE,CAAC,OAAO,CAAC,CAAC;QACxC,MAAM,MAAM,GAAG,IAAI,WAAK,CAAC,GAAG,EAAE,CAAC,SAAS,MAAM,EAAE,CAAC,CAAC;QAElD,OAAO;QACP,MAAM,SAAS,GAAG,OAAO,CAAC,MAAM,CAAC,QAAQ,EAAE,CAAC,CAAC;QAC7C,MAAM,SAAS,GAAG,OAAO,CAAC,MAAM,CAAC,CAAC;QAElC,OAAO;QACP,IAAI,CAAC,SAAS,CAAC,0BAAW,CAAC,SAAS,CAAC,EAAE,aAAa,CAAC,CAAC;QACtD,IAAI,CAAC,SAAS,CAAC,0BAAW,CAAC,SAAS,CAAC,EAAE,aAAa,CAAC,CAAC;QAEtD,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED,oEAAoE,CAAC,IAAU;QAC7E,QAAQ;QACR,KAAK,MAAM,KAAK,IAAI,0BAA0B,CAAC,CAAC,CAAC,EAAE;YACjD,OAAO;YACP,MAAM,WAAW,GAAG,iBAAiB,KAAK,EAAE,CAAC;YAC7C,MAAM,QAAQ,GAAG,OAAO,CAAC,WAAW,CAAC,CAAC;YAEtC,OAAO;YACP,IAAI,CAAC,SAAS,CAAC,0BAAW,CAAC,QAAQ,CAAC,EAAE,iBAAiB,CAAC,CAAC;SAC1D;QACD,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED,sEAAsE,CAAC,IAAU;QAC/E,QAAQ;QACR,KAAK,MAAM,UAAU,IAAI,iCAAiC,CAAC,EAAE,GAAG,EAAE,UAAU,EAAE,CAAC,EAAE;YAC/E,OAAO;YACP,MAAM,QAAQ,GAAG,OAAO,CAAC,sBAAsB,UAAU,EAAE,CAAC,CAAC;YAE7D,OAAO;YACP,MAAM,OAAO,GAAG,EAAC,QAAQ,EAAE,SAAS,EAAC,CAAC;YACtC,IAAI,CAAC,KAAK,CAAC,0BAAW,CAAC,QAAQ,EAAE,OAAO,CAAC,EAAE,4BAA4B,CAAC,CAAC;SAC1E;QAED,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED,6CAA6C,CAAC,IAAU;QACtD,QAAQ;QACR,KAAK,MAAM,KAAK,IAAI,mBAAmB,CAAC,OAAO,CAAC,EAAE;YAChD,OAAO;YACP,MAAM,QAAQ,GAAG,OAAO,CAAC,GAAG,KAAK,QAAQ,CAAC,CAAC;YAE3C,OAAO;YACP,IAAI,CAAC,KAAK,CAAC,0BAAW,CAAC,QAAQ,CAAC,EAAE,aAAa,CAAC,CAAC;SAClD;QAED,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED,uCAAuC,CAAC,IAAU;QAChD,QAAQ;QACR,KAAK,MAAM,MAAM,IAAI,mBAAmB,CAAC,QAAQ,CAAC,EAAE;YAClD,KAAK,MAAM,MAAM,IAAI,mBAAmB,CAAC,OAAO,CAAC,EAAE;gBACjD,OAAO;gBACP,MAAM,QAAQ,GAAG,OAAO,CAAC,GAAG,MAAM,GAAG,MAAM,EAAE,CAAC,CAAC;gBAE/C,OAAO;gBACP,IAAI,CAAC,KAAK,CAAC,0BAAW,CAAC,QAAQ,CAAC,EAAE,aAAa,CAAC,CAAC;aAClD;SACF;QAED,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED,8DAA8D,CAAC,IAAU;QACvE,QAAQ;QACR,MAAM,KAAK,GAAG,IAAI,WAAK,CAAC,GAAG,EAAE,CAAC,eAAe,CAAC,CAAC;QAE/C,OAAO;QACP,MAAM,CAAC,GAAG;YACR,CAAC,KAAK,CAAC,QAAQ,EAAE,CAAC,EAAE,QAAQ,KAAK,EAAE;SACpC,CAAC;QAEF,OAAO;QACP,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE,eAAe,EAAE,oBAAoB,EAAE,CAAC,CAAC;QACtE,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED,mCAAmC,CAAC,IAAU;QAC5C,QAAQ;QACR,MAAM,KAAK,GAAG,IAAI,WAAK,CAAC,GAAG,EAAE,CAAC,IAAI,WAAK,CAAC,GAAG,EAAE,CAAC,IAAI,WAAK,CAAC,GAAG,EAAE,CAAC,eAAe,CAAC,CAAC,CAAC,CAAC;QAEjF,OAAO;QACP,MAAM,CAAC,GAAG;YACR,CAAC,KAAK,CAAC,QAAQ,EAAE,CAAC,EAAE,QAAQ,KAAK,EAAE;SACpC,CAAC;QAEF,OAAO;QACP,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE,eAAe,EAAE,oBAAoB,EAAE,CAAC,CAAC;QACtE,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED,oDAAoD,CAAC,IAAU;QAC7D,QAAQ;QACR,MAAM,UAAU,GAAG,IAAI,WAAK,CAAC,GAAG,EAAE,CAAC,MAAM,CAAC,CAAC;QAC3C,MAAM,KAAK,GAAG,IAAI,WAAK,CAAC,GAAG,EAAE,CAAC,GAAG,UAAU,WAAW,CAAC,CAAC;QAExD,OAAO;QACP,MAAM,CAAC,GAAG;YACR,CAAC,KAAK,CAAC,QAAQ,EAAE,CAAC,EAAE,cAAc;SACnC,CAAC;QAEF,OAAO;QACP,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE,eAAe,EAAE,cAAc,EAAE,CAAC,CAAC;QAChE,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED,qCAAqC,CAAC,IAAU;QAC9C,QAAQ;QACR,MAAM,UAAU,GAAG,IAAI,WAAK,CAAC,GAAG,EAAE,CAAC,MAAM,CAAC,CAAC;QAC3C,MAAM,KAAK,GAAG,IAAI,WAAK,CAAC,GAAG,EAAE,CAAC,GAAG,UAAU,WAAW,CAAC,CAAC;QAExD,OAAO;QACP,MAAM,CAAC,GAAG;YACR,CAAC,KAAK,CAAC,QAAQ,EAAE,CAAC,EAAE,cAAc;SACnC,CAAC;QAEF,OAAO;QACP,MAAM,MAAM,GAAG,gBAAU,CAAC,IAAI,WAAK,EAAE,EAAE,GAAG,EAAE,CAAC,CAAC,CAAC,CAAC;QAChD,IAAI,CAAC,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,KAAK,UAAU,CAAC,EAAE,wBAAwB,CAAC,CAAC;QACtE,IAAI,CAAC,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,KAAK,KAAK,CAAC,EAAE,mBAAmB,CAAC,CAAC;QAC5D,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED,uDAAuD,CAAC,IAAU;QAChE,QAAQ;QACR,MAAM,KAAK,GAAG,IAAI,WAAK,CAAC,EAAE,GAAG,EAAE,OAAO,EAAE,CAAC,CAAC;QAE1C,OAAO;QACP,MAAM,CAAC,GAAG;YACR,CAAC,KAAK,CAAC,QAAQ,EAAE,CAAC,EAAE,QAAQ,KAAK,EAAE;SACpC,CAAC;QAEF,OAAO;QACP,IAAI,CAAC,MAAM,CAAC,GAAG,EAAE,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,sGAAsG,CAAC,CAAC;QACtI,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED,eAAe,EAAE;QACf,qDAAqD,CAAC,IAAU;YAC9D,QAAQ;YACR,MAAM,KAAK,GAAG,IAAI,WAAK,CAAC,EAAE,GAAG,EAAE,OAAO,EAAE,CAAC,CAAC;YAE1C,OAAO;YACP,MAAM,MAAM,GAAG;gBACb,GAAG,EAAE,KAAK,CAAC,MAAM,EAAE;aACpB,CAAC;YAEF,OAAO;YACP,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE;gBAC9B,GAAG,EAAE,EAAE,GAAG,EAAE,OAAO,EAAC;aACrB,CAAC,CAAC;YAEH,IAAI,CAAC,IAAI,EAAE,CAAC;QACd,CAAC;QAED,4BAA4B,CAAC,IAAU;YACrC,QAAQ;YACR,MAAM,KAAK,GAAG,IAAI,WAAK,CAAC,EAAE,GAAG,EAAE,OAAO,EAAE,CAAC,CAAC;YAE1C,OAAO;YACP,MAAM,OAAO,GAAa,KAAK,CAAC,MAAM,EAAE,CAAC;YACzC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;YAEtB,OAAO;YACP,IAAI,CAAC,MAAM,CAAC,GAAG,EAAE;gBACf,OAAO,CAAC,OAAO,CAAC,CAAC;YACnB,CAAC,EAAE,mCAAmC,CAAC,CAAC;YAExC,IAAI,CAAC,IAAI,EAAE,CAAC;QACd,CAAC;QAED,uCAAuC,CAAC,IAAU;YAChD,QAAQ;YACR,MAAM,KAAK,GAAG,IAAI,WAAK,CAAC,EAAE,GAAG,EAAE,OAAO,EAAE,CAAC,CAAC;YAE1C,OAAO;YACP,MAAM,OAAO,GAAa,KAAK,CAAC,MAAM,EAAE,CAAC;YACzC,OAAO,CAAC,CAAC,CAAC,IAAI,OAAO,CAAC;YAEtB,OAAO;YACP,IAAI,CAAC,MAAM,CAAC,GAAG,EAAE;gBACf,OAAO,CAAC,OAAO,CAAC,CAAC;YACnB,CAAC,EAAE,wBAAwB,CAAC,CAAC;YAE7B,IAAI,CAAC,IAAI,EAAE,CAAC;QACd,CAAC;QAED,oCAAoC,CAAC,IAAU;YAC7C,QAAQ;YACR,MAAM,OAAO,GAAa,IAAI,WAAK,CAAC,EAAE,GAAG,EAAE,OAAO,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC;YAE/D,OAAO;YACP,MAAM,MAAM,GAAG,QAAE,CAAC,MAAM,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC;YAErC,OAAO;YACP,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE;gBAC9B,YAAY,EAAE,CAAC,CAAC,EAAE,EAAE,GAAG,EAAE,OAAO,EAAC,CAAC;aACnC,CAAC,CAAC;YAEH,IAAI,CAAC,IAAI,EAAE,CAAC;QACd,CAAC;QAED,kCAAkC,CAAC,IAAU;YAC3C,QAAQ;YACR,MAAM,OAAO,GAAa,IAAI,WAAK,CAAC,EAAE,GAAG,EAAE,OAAO,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC;YAE/D,OAAO;YACP,MAAM,MAAM,GAAG,QAAE,CAAC,IAAI,CAAC,GAAG,EAAE,OAAO,CAAC,CAAC;YAErC,OAAO;YACP,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE;gBAC9B,UAAU,EAAE,CAAC,GAAG,EAAE,EAAE,GAAG,EAAE,OAAO,EAAC,CAAC;aACnC,CAAC,CAAC;YAEH,IAAI,CAAC,IAAI,EAAE,CAAC;QACd,CAAC;QAED,+DAA+D,CAAC,IAAU;YACxE,QAAQ;YACR,MAAM,OAAO,GAAa,IAAI,WAAK,CAAC,EAAE,GAAG,EAAE,OAAO,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC;YAE/D,OAAO;YACP,MAAM,MAAM,GAAG,QAAE,CAAC,IAAI,CAAC,GAAG,EAAE,OAAO,CAAC,CAAC,QAAQ,EAAE,CAAC;YAEhD,OAAO;YACP,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE;gBAC9B,UAAU,EAAE,CAAC,GAAG,EAAE,EAAE,GAAG,EAAE,OAAO,EAAC,CAAC;aACnC,CAAC,CAAC;YAEH,IAAI,CAAC,IAAI,EAAE,CAAC;QACd,CAAC;KACF;IAED,iBAAiB,EAAE;QACjB,8BAA8B,CAAC,IAAU;YACvC,IAAI,CAAC,KAAK,CAAC,EAAE,EAAE,6BAAkB,CAAC,4BAAiB,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;YAC1D,IAAI,CAAC,IAAI,EAAE,CAAC;QACd,CAAC;QAED,+DAA+D,CAAC,IAAU;YACxE,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,GAAG,EAAE,CAAC,EAAE,EAAE;gBAC5B,yCAAyC;gBACzC,MAAM,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,MAAM,EAAE,GAAG,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;gBAE5D,MAAM,OAAO,GAAG,4BAAiB,CAAC,CAAC,CAAC,CAAC;gBACrC,kCAAkC;gBAClC,MAAM,YAAY,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,SAAS,EAAE,OAAO,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC;gBAClF,MAAM,OAAO,GAAG,6BAAkB,CAAC,YAAY,CAAC,CAAC;gBACjD,IAAI,CAAC,KAAK,CAAC,OAAO,EAAE,CAAC,EAAE,8BAA8B,CAAC,EAAE,CAAC,CAAC;aAC3D;YAED,IAAI,CAAC,IAAI,EAAE,CAAC;QACd,CAAC;QAED,8DAA8D,CAAC,IAAU;YACvE,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,6BAAkB,CAAC,CAAC,CAAC,CAAC,CAAC;YAC7C,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,6BAAkB,CAAC,IAAI,CAAC,CAAC,CAAC;YAChD,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,6BAAkB,CAAC,SAAS,CAAC,CAAC,CAAC;YAErD,IAAI,CAAC,IAAI,EAAE,CAAC;QACd,CAAC;QAED,6CAA6C,CAAC,IAAU;YACtD,QAAQ;YACR,MAAM,KAAK,GAAG,IAAI,WAAK,EAAE,CAAC;YAC1B,MAAM,CAAC,GAAG,IAAI,WAAK,CAAC,GAAG,EAAE,CAAC,GAAG,CAAC,CAAC;YAE/B,OAAO;YACP,MAAM,OAAO,GAAG,CAAC,CAAC,QAAQ,EAAE,CAAC;YAC7B,IAAI,CAAC,KAAK,CAAC,IAAI,EAAE,WAAK,CAAC,OAAO,CAAC,OAAO,CAAC,EAAE,uCAAuC,CAAC,CAAC;YAElF,OAAO;YACP,MAAM,QAAQ,GAAG,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,EAAE,KAAK,EAAE,OAAO,EAAE,CAAC,CAAC;YACxD,IAAI,CAAC,SAAS,CAAC,QAAQ,EAAE,EAAE,KAAK,EAAE,GAAG,EAAE,CAAC,CAAC;YAEzC,IAAI,CAAC,IAAI,EAAE,CAAC;QACd,CAAC;KACF;IAED,2CAA2C,CAAC,IAAU;QACpD,SAAS,GAAG;YACV,SAAS,GAAG;gBACV,MAAM,WAAY,SAAQ,WAAK;oBAC7B,IAAW,aAAa;wBACtB,OAAO,IAAI,CAAC,KAAK,CAAC;oBACpB,CAAC;iBACF;gBAED,OAAO,IAAI,WAAW,CAAC,GAAG,EAAE,CAAC,OAAO,CAAC,CAAC;YACxC,CAAC;YAED,OAAO,GAAG,EAAE,CAAC;QACf,CAAC;QAED,MAAM,KAAK,GAAG,GAAG,EAAE,CAAC;QACpB,IAAI,CAAC,EAAE,CAAC,KAAK,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;QAC1D,IAAI,CAAC,EAAE,CAAC,KAAK,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;QAC1D,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED,yDAAyD,CAAC,IAAU;QAClE,SAAS,GAAG;YACV,SAAS,GAAG;gBACV,SAAS,GAAG;oBACV,MAAM,aAAc,SAAQ,WAAK;wBACxB,UAAU,CAAC,OAAe;4BAC/B,MAAM,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;wBAC/B,CAAC;qBACF;oBACD,OAAO,IAAI,aAAa,CAAC,MAAM,CAAC,CAAC;gBACnC,CAAC;gBAED,OAAO,GAAG,EAAE,CAAC;YACf,CAAC;YACD,OAAO,GAAG,EAAE,CAAC;QACf,CAAC;QACD,MAAM,KAAK,GAAG,GAAG,EAAE,CAAC;QACpB,IAAI,CAAC,MAAM,CAAC,GAAG,EAAE,CAAC,KAAK,CAAC,UAAU,CAAC,UAAU,CAAC,EAAE,gBAAgB,CAAC,CAAC;QAClE,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;CACF,CAAC","sourcesContent":["import { Test } from 'nodeunit';\nimport { App as Root, findTokens, Fn, Stack, Token } from '../lib';\nimport { createTokenDouble, extractTokenDouble } from '../lib/encoding';\nimport { TokenMap } from '../lib/token-map';\nimport { evaluateCFN } from './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(Token.isToken({ resolve: () => 123 }));\n    test.ok(Token.isToken({ a: 1, b: 2, resolve: () => 'hello' }));\n    test.ok(!Token.isToken({ 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 can be reversed'(test: Test) {\n    // GIVEN\n    const token = new Token(() => 'woof woof');\n\n    // THEN\n    test.equal(token, TokenMap.instance().lookupString(`${token}`));\n    test.done();\n  },\n\n  'concatenated tokens are undefined'(test: Test) {\n    // GIVEN\n    const token = new Token(() => 'woof woof');\n\n    // WHEN\n    test.equal(undefined, TokenMap.instance().lookupString(`${token}bla`));\n    test.equal(undefined, TokenMap.instance().lookupString(`bla${token}`));\n    test.equal(undefined, TokenMap.instance().lookupString(`bla`));\n\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 Token(() => ({ 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  'tokens can be nested in hash keys'(test: Test) {\n    // GIVEN\n    const token = new Token(() => new 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  'tokens can be nested and concatenated in hash keys'(test: Test) {\n    // GIVEN\n    const innerToken = new Token(() => 'toot');\n    const token = new Token(() => `${innerToken} the woot`);\n\n    // WHEN\n    const s = {\n      [token.toString()]: `boom chicago`\n    };\n\n    // THEN\n    test.deepEqual(resolve(s), { 'toot the woot': 'boom chicago' });\n    test.done();\n  },\n\n  'can find nested tokens in hash keys'(test: Test) {\n    // GIVEN\n    const innerToken = new Token(() => 'toot');\n    const token = new Token(() => `${innerToken} the woot`);\n\n    // WHEN\n    const s = {\n      [token.toString()]: `boom chicago`\n    };\n\n    // THEN\n    const tokens = findTokens(new Stack(), () => s);\n    test.ok(tokens.some(t => t === innerToken), 'Cannot find innerToken');\n    test.ok(tokens.some(t => t === token), 'Cannot find token');\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 Token({ 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  'list encoding': {\n    'can encode Token to string and resolve the encoding'(test: Test) {\n      // GIVEN\n      const token = new Token({ Ref: 'Other' });\n\n      // WHEN\n      const struct = {\n        XYZ: token.toList()\n      };\n\n      // THEN\n      test.deepEqual(resolve(struct), {\n        XYZ: { Ref: 'Other'}\n      });\n\n      test.done();\n    },\n\n    'cannot add to encoded list'(test: Test) {\n      // GIVEN\n      const token = new Token({ Ref: 'Other' });\n\n      // WHEN\n      const encoded: string[] = token.toList();\n      encoded.push('hello');\n\n      // THEN\n      test.throws(() => {\n        resolve(encoded);\n      }, /Cannot add elements to list token/);\n\n      test.done();\n    },\n\n    'cannot add to strings in encoded list'(test: Test) {\n      // GIVEN\n      const token = new Token({ Ref: 'Other' });\n\n      // WHEN\n      const encoded: string[] = token.toList();\n      encoded[0] += 'hello';\n\n      // THEN\n      test.throws(() => {\n        resolve(encoded);\n      }, /concatenate strings in/);\n\n      test.done();\n    },\n\n    'can pass encoded lists to FnSelect'(test: Test) {\n      // GIVEN\n      const encoded: string[] = new Token({ Ref: 'Other' }).toList();\n\n      // WHEN\n      const struct = Fn.select(1, encoded);\n\n      // THEN\n      test.deepEqual(resolve(struct), {\n        'Fn::Select': [1, { Ref: 'Other'}]\n      });\n\n      test.done();\n    },\n\n    'can pass encoded lists to FnJoin'(test: Test) {\n      // GIVEN\n      const encoded: string[] = new Token({ Ref: 'Other' }).toList();\n\n      // WHEN\n      const struct = Fn.join('/', encoded);\n\n      // THEN\n      test.deepEqual(resolve(struct), {\n        'Fn::Join': ['/', { Ref: 'Other'}]\n      });\n\n      test.done();\n    },\n\n    'can pass encoded lists to FnJoin, even if join is stringified'(test: Test) {\n      // GIVEN\n      const encoded: string[] = new Token({ Ref: 'Other' }).toList();\n\n      // WHEN\n      const struct = Fn.join('/', encoded).toString();\n\n      // THEN\n      test.deepEqual(resolve(struct), {\n        'Fn::Join': ['/', { Ref: 'Other'}]\n      });\n\n      test.done();\n    },\n  },\n\n  'number encoding': {\n    'basic integer encoding works'(test: Test) {\n      test.equal(16, extractTokenDouble(createTokenDouble(16)));\n      test.done();\n    },\n\n    'arbitrary integers can be encoded, stringified, and recovered'(test: Test) {\n      for (let i = 0; i < 100; i++) {\n        // We can encode all numbers up to 2^48-1\n        const x = Math.floor(Math.random() * (Math.pow(2, 48) - 1));\n\n        const encoded = createTokenDouble(x);\n        // Roundtrip through JSONification\n        const roundtripped = JSON.parse(JSON.stringify({ theNumber: encoded })).theNumber;\n        const decoded = extractTokenDouble(roundtripped);\n        test.equal(decoded, x, `Fail roundtrip encoding of ${x}`);\n      }\n\n      test.done();\n    },\n\n    'arbitrary numbers are correctly detected as not being tokens'(test: Test) {\n      test.equal(undefined, extractTokenDouble(0));\n      test.equal(undefined, extractTokenDouble(1243));\n      test.equal(undefined, extractTokenDouble(4835e+532));\n\n      test.done();\n    },\n\n    'can number-encode and resolve Token objects'(test: Test) {\n      // GIVEN\n      const stack = new Stack();\n      const x = new Token(() => 123);\n\n      // THEN\n      const encoded = x.toNumber();\n      test.equal(true, Token.isToken(encoded), 'encoded number does not test as token');\n\n      // THEN\n      const resolved = stack.node.resolve({ value: encoded });\n      test.deepEqual(resolved, { value: 123 });\n\n      test.done();\n    },\n  },\n\n  'stack trace is captured at token creation'(test: Test) {\n    function fn1() {\n      function fn2() {\n        class ExposeTrace extends Token {\n          public get creationTrace() {\n            return this.trace;\n          }\n        }\n\n        return new ExposeTrace(() => 'hello');\n      }\n\n      return fn2();\n    }\n\n    const token = fn1();\n    test.ok(token.creationTrace.find(x => x.includes('fn1')));\n    test.ok(token.creationTrace.find(x => x.includes('fn2')));\n    test.done();\n  },\n\n  'newError returns an error with the creation stack trace'(test: Test) {\n    function fn1() {\n      function fn2() {\n        function fn3() {\n          class ThrowingToken extends Token {\n            public throwError(message: string) {\n              throw this.newError(message);\n            }\n          }\n          return new ThrowingToken('boom');\n        }\n\n        return fn3();\n      }\n      return fn2();\n    }\n    const token = fn1();\n    test.throws(() => token.throwError('message!'), /Token created:/);\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 Token(value),\n    new Token(() => 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\n/**\n * Wrapper for resolve that creates an throwaway Construct to call it on\n *\n * So I don't have to change all call sites in this file.\n */\nfunction resolve(x: any) {\n  return new Root().node.resolve(x);\n}\n"]}

Sorry, the diff of this file is not supported yet

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

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

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