Socket
Socket
Sign inDemoInstall

constructs

Package Overview
Dependencies
Maintainers
3
Versions
1334
Alerts
File Explorer

Advanced tools

Socket logo

Install Socket

Detect and block malicious and high-risk dependencies

Install

constructs - npm Package Compare versions

Comparing version 1.0.0 to 1.0.1

.github/PULL_REQUEST_TEMPLATE.md

57

lib/construct.d.ts

@@ -11,3 +11,3 @@ import { IAspect } from './aspect';

*/
export declare class ConstructNode {
export declare class Node {
private readonly host;

@@ -22,22 +22,4 @@ /**

*/
static of(construct: IConstruct): ConstructNode;
static of(construct: IConstruct): Node;
/**
* Synthesizes a CloudAssembly from a construct tree.
* @param root The root of the construct tree.
* @param options Synthesis options.
*/
static synthesizeNode(root: ConstructNode, options: SynthesisOptions): void;
/**
* Invokes "prepare" on all constructs (depth-first, post-order) in the tree under `node`.
* @param node The root node
*/
static prepareNode(node: ConstructNode): void;
/**
* Invokes "validate" on all constructs in the tree (depth-first, pre-order) and returns
* the list of all errors. An empty list indicates that there are no errors.
*
* @param node The root node
*/
static validateNode(node: ConstructNode): ValidationError[];
/**
* Returns the scope in which this construct is defined.

@@ -211,2 +193,16 @@ *

/**
* Synthesizes a CloudAssembly from a construct tree.
* @param options Synthesis options.
*/
synthesize(options: SynthesisOptions): void;
/**
* Invokes "prepare" on all constructs (depth-first, post-order) in the tree under `node`.
*/
prepare(): void;
/**
* Invokes "validate" on all constructs in the tree (depth-first, pre-order) and returns
* the list of all errors. An empty list indicates that there are no errors.
*/
validate(): ValidationError[];
/**
* Adds a child construct to this node.

@@ -238,2 +234,3 @@ *

* dash `--`.
* @param options Options
*/

@@ -253,3 +250,3 @@ constructor(scope: Construct, id: string, options?: ConstructOptions);

*/
protected validateConstruct(): string[];
protected onValidate(): string[];
/**

@@ -265,3 +262,3 @@ * Perform final modifications before synthesis

*/
protected prepareConstruct(): void;
protected onPrepare(): void;
/**

@@ -275,3 +272,3 @@ * Allows this construct to emit artifacts into the cloud assembly during synthesis.

*/
protected synthesizeConstruct(session: ISynthesisSession): void;
protected onSynthesize(session: ISynthesisSession): void;
}

@@ -318,3 +315,3 @@ /**

/**
* Represents a single session of synthesis. Passed into `construct.synthesizeConstruct()` methods.
* Represents a single session of synthesis. Passed into `construct.onSynthesize()` methods.
*/

@@ -347,3 +344,3 @@ export interface ISynthesisSession {

* Additional context passed into the synthesis session object when `construct.synth` is called.
* @default - no additional context is passed to `synthesizeConstruct`
* @default - no additional context is passed to `onSynthesize`
*/

@@ -359,4 +356,4 @@ readonly sessionContext?: {

/**
* A factory for attaching `ConstructNode`s to the construct.
* @default - the default `ConstructNode` is associated
* A factory for attaching `Node`s to the construct.
* @default - the default `Node` is associated
*/

@@ -366,7 +363,7 @@ readonly nodeFactory?: INodeFactory;

/**
* A factory for attaching `ConstructNode`s to the construct.
* A factory for attaching `Node`s to the construct.
*/
export interface INodeFactory {
/**
* Returns a new `ConstructNode` associated with `host`.
* Returns a new `Node` associated with `host`.
* @param host the associated construct

@@ -376,3 +373,3 @@ * @param scope the construct's scope (parent)

*/
createNode(host: Construct, scope: IConstruct, id: string): ConstructNode;
createNode(host: Construct, scope: IConstruct, id: string): Node;
}

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

*/
class ConstructNode {
class Node {
constructor(host, scope, id) {

@@ -33,3 +33,3 @@ this.host = host;

// Has side effect so must be very last thing in constructor
ConstructNode.of(scope).addChild(host, this.id);
Node.of(scope).addChild(host, this.id);
}

@@ -51,3 +51,3 @@ else {

if (!node) {
throw new Error(`construct does not have an associated ConstructNode`);
throw new Error(`construct does not have an associated node`);
}

@@ -57,67 +57,2 @@ return node;

/**
* Synthesizes a CloudAssembly from a construct tree.
* @param root The root of the construct tree.
* @param options Synthesis options.
*/
static synthesizeNode(root, options) {
// the three holy phases of synthesis: prepare, validate and synthesize
// prepare
this.prepareNode(root);
// validate
const validate = options.skipValidation === undefined ? true : !options.skipValidation;
if (validate) {
const errors = this.validateNode(root);
if (errors.length > 0) {
const errorList = errors.map(e => `[${ConstructNode.of(e.source).path}] ${e.message}`).join('\n ');
throw new Error(`Validation failed with the following errors:\n ${errorList}`);
}
}
// synthesize (leaves first)
for (const construct of root.findAll(ConstructOrder.POSTORDER)) {
const node = this.of(construct);
try {
node._lock();
const ctx = {
...options.sessionContext,
outdir: options.outdir
};
construct.synthesizeConstruct(ctx); // "as any" is needed because we want to keep "synthesize" protected
}
finally {
node._unlock();
}
}
}
/**
* Invokes "prepare" on all constructs (depth-first, post-order) in the tree under `node`.
* @param node The root node
*/
static prepareNode(node) {
const constructs = node.findAll(ConstructOrder.PREORDER);
// Aspects are applied root to leaf
for (const construct of constructs) {
ConstructNode.of(construct).invokeAspects();
}
// Use .reverse() to achieve post-order traversal
for (const construct of constructs.reverse()) {
if (construct instanceof Construct) {
construct.prepareConstruct(); // "as any" is needed because we want to keep "prepare" protected
}
}
}
/**
* Invokes "validate" on all constructs in the tree (depth-first, pre-order) and returns
* the list of all errors. An empty list indicates that there are no errors.
*
* @param node The root node
*/
static validateNode(node) {
let errors = new Array();
for (const child of node.children) {
errors = errors.concat(this.validateNode(ConstructNode.of(child)));
}
const localErrors = node.host.validateConstruct(); // "as any" is needed because we want to keep "validate" protected
return errors.concat(localErrors.map(msg => ({ source: node.host, message: msg })));
}
/**
* The full, absolute path of this construct in the tree.

@@ -128,4 +63,4 @@ *

get path() {
const components = this.scopes.slice(1).map(c => ConstructNode.of(c).id);
return components.join(ConstructNode.PATH_SEP);
const components = this.scopes.slice(1).map(c => Node.of(c).id);
return components.join(Node.PATH_SEP);
}

@@ -137,3 +72,3 @@ /**

get uniqueId() {
const components = this.scopes.slice(1).map(c => ConstructNode.of(c).id);
const components = this.scopes.slice(1).map(c => Node.of(c).id);
return components.length > 0 ? uniqueid_1.makeUniqueId(components) : '';

@@ -214,3 +149,3 @@ }

}
for (const child of ConstructNode.of(c).children) {
for (const child of Node.of(c).children) {
visit(child);

@@ -235,3 +170,3 @@ }

if (this.children.length > 0) {
const names = this.children.map(c => ConstructNode.of(c).id);
const names = this.children.map(c => Node.of(c).id);
throw new Error('Cannot set context after children have been added: ' + names.join(','));

@@ -257,3 +192,3 @@ }

}
return this.scope && ConstructNode.of(this.scope).tryGetContext(key);
return this.scope && Node.of(this.scope).tryGetContext(key);
}

@@ -330,3 +265,3 @@ /**

ret.unshift(curr);
curr = ConstructNode.of(curr).scope;
curr = Node.of(curr).scope;
}

@@ -350,3 +285,3 @@ return ret;

}
if (this.scope && ConstructNode.of(this.scope).locked) {
if (this.scope && Node.of(this.scope).locked) {
return true;

@@ -374,3 +309,3 @@ }

for (const source of this.findAll()) {
for (const dependable of ConstructNode.of(source)._dependencies) {
for (const dependable of Node.of(source)._dependencies) {
for (const target of dependency_1.DependableTrait.get(dependable).dependencyRoots) {

@@ -404,2 +339,63 @@ let foundTargets = found.get(source);

/**
* Synthesizes a CloudAssembly from a construct tree.
* @param options Synthesis options.
*/
synthesize(options) {
// the three holy phases of synthesis: prepare, validate and synthesize
// prepare
this.prepare();
// validate
const validate = options.skipValidation === undefined ? true : !options.skipValidation;
if (validate) {
const errors = this.validate();
if (errors.length > 0) {
const errorList = errors.map(e => `[${Node.of(e.source).path}] ${e.message}`).join('\n ');
throw new Error(`Validation failed with the following errors:\n ${errorList}`);
}
}
// synthesize (leaves first)
for (const construct of this.findAll(ConstructOrder.POSTORDER)) {
const node = Node.of(construct);
try {
node._lock();
const ctx = {
...options.sessionContext,
outdir: options.outdir
};
construct.onSynthesize(ctx); // "as any" is needed because we want to keep "synthesize" protected
}
finally {
node._unlock();
}
}
}
/**
* Invokes "prepare" on all constructs (depth-first, post-order) in the tree under `node`.
*/
prepare() {
const constructs = this.findAll(ConstructOrder.PREORDER);
// Aspects are applied root to leaf
for (const construct of constructs) {
Node.of(construct).invokeAspects();
}
// Use .reverse() to achieve post-order traversal
for (const construct of constructs.reverse()) {
if (construct instanceof Construct) {
construct.onPrepare(); // "as any" is needed because we want to keep "prepare" protected
}
}
}
/**
* Invokes "validate" on all constructs in the tree (depth-first, pre-order) and returns
* the list of all errors. An empty list indicates that there are no errors.
*/
validate() {
let errors = new Array();
for (const child of this.children) {
errors = errors.concat(Node.of(child).validate());
}
const localErrors = this.host.onValidate(); // "as any" is needed because we want to keep "validate" protected
return errors.concat(localErrors.map(msg => ({ source: this.host, message: msg })));
}
/**
* Locks this construct from allowing more children to be added. After this

@@ -455,7 +451,7 @@ * call, no more children can be added to this construct or to any children.

}
exports.ConstructNode = ConstructNode;
exports.Node = Node;
/**
* Separator used to delimit construct path components.
*/
ConstructNode.PATH_SEP = '/';
Node.PATH_SEP = '/';
/**

@@ -475,5 +471,8 @@ * Represents the building block of the construct graph.

* dash `--`.
* @param options Options
*/
constructor(scope, id, options = {}) {
const nodeFactory = options.nodeFactory || new DefaultNodeFactory();
var _a;
// attach the construct to the construct tree by creating a node
const nodeFactory = (_a = options.nodeFactory, (_a !== null && _a !== void 0 ? _a : { createNode: () => new Node(this, scope, id) }));
Object.defineProperty(this, CONSTRUCT_NODE_PROPERTY_SYMBOL, {

@@ -493,3 +492,3 @@ value: nodeFactory.createNode(this, scope, id),

toString() {
return ConstructNode.of(this).path || '<root>';
return Node.of(this).path || '<root>';
}

@@ -504,3 +503,3 @@ /**

*/
validateConstruct() {
onValidate() {
return [];

@@ -518,3 +517,3 @@ }

*/
prepareConstruct() {
onPrepare() {
return;

@@ -530,3 +529,3 @@ }

*/
synthesizeConstruct(session) {
onSynthesize(session) {
ignore(session);

@@ -554,3 +553,3 @@ }

// Import this _after_ everything else to help node work the classes out in the correct order...
const PATH_SEP_REGEX = new RegExp(`${ConstructNode.PATH_SEP}`, 'g');
const PATH_SEP_REGEX = new RegExp(`${Node.PATH_SEP}`, 'g');
/**

@@ -563,7 +562,2 @@ * Return a sanitized version of an arbitrary string, so it can be used as an ID

}
class DefaultNodeFactory {
createNode(host, scope, id) {
return new ConstructNode(host, scope, id);
}
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"construct.js","sourceRoot":"","sources":["construct.ts"],"names":[],"mappings":"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","sourcesContent":["import { IAspect } from './aspect';\nimport { ConstructMetadata, MetadataEntry } from './metadata';\nimport { DependableTrait } from './private/dependency';\nimport { captureStackTrace } from './private/stack-trace';\nimport { makeUniqueId } from './private/uniqueid';\nimport { Token } from './token';\n\nconst CONSTRUCT_NODE_PROPERTY_SYMBOL = Symbol.for('constructs.Construct.node');\n\n/**\n * Represents a construct.\n */\nexport interface IConstruct { }\n\n/**\n * Represents the construct node in the scope tree.\n */\nexport class ConstructNode {\n  /**\n   * Separator used to delimit construct path components.\n   */\n  public static readonly PATH_SEP = '/';\n\n  /**\n   * Returns the node associated with a construct.\n   * @param construct the construct\n   */\n  public static of(construct: IConstruct): ConstructNode {\n    const node = (construct as any)[CONSTRUCT_NODE_PROPERTY_SYMBOL] as ConstructNode;\n    if (!node) {\n      throw new Error(`construct does not have an associated ConstructNode`);\n    }\n\n    return node;\n  }\n\n  /**\n   * Synthesizes a CloudAssembly from a construct tree.\n   * @param root The root of the construct tree.\n   * @param options Synthesis options.\n   */\n  public static synthesizeNode(root: ConstructNode, options: SynthesisOptions): void {\n    // the three holy phases of synthesis: prepare, validate and synthesize\n\n    // prepare\n    this.prepareNode(root);\n\n    // validate\n    const validate = options.skipValidation === undefined ? true : !options.skipValidation;\n    if (validate) {\n      const errors = this.validateNode(root);\n      if (errors.length > 0) {\n        const errorList = errors.map(e => `[${ConstructNode.of(e.source).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 construct of root.findAll(ConstructOrder.POSTORDER)) {\n      const node = this.of(construct);\n      try {\n        node._lock();\n        const ctx = {\n          ...options.sessionContext,\n          outdir: options.outdir\n        };\n        (construct as any).synthesizeConstruct(ctx); // \"as any\" is needed because we want to keep \"synthesize\" protected\n      } finally {\n        node._unlock();\n      }\n    }\n  }\n\n  /**\n   * Invokes \"prepare\" on all constructs (depth-first, post-order) in the tree under `node`.\n   * @param node The root node\n   */\n  public static prepareNode(node: ConstructNode) {\n    const constructs = node.findAll(ConstructOrder.PREORDER);\n\n    // Aspects are applied root to leaf\n    for (const construct of constructs) {\n      ConstructNode.of(construct).invokeAspects();\n    }\n\n    // Use .reverse() to achieve post-order traversal\n    for (const construct of constructs.reverse()) {\n      if (construct instanceof Construct) {\n        (construct as any).prepareConstruct(); // \"as any\" is needed because we want to keep \"prepare\" protected\n      }\n    }\n  }\n\n  /**\n   * Invokes \"validate\" on all constructs in the tree (depth-first, pre-order) and returns\n   * the list of all errors. An empty list indicates that there are no errors.\n   *\n   * @param node The root node\n   */\n  public static validateNode(node: ConstructNode) {\n    let errors = new Array<ValidationError>();\n\n    for (const child of node.children) {\n      errors = errors.concat(this.validateNode(ConstructNode.of(child)));\n    }\n\n    const localErrors: string[] = (node.host as any).validateConstruct(); // \"as any\" is needed because we want to keep \"validate\" protected\n    return errors.concat(localErrors.map(msg => ({ source: node.host, message: msg })));\n  }\n\n  /**\n   * Returns the scope in which this construct is defined.\n   *\n   * The value is `undefined` at the root of the construct scope tree.\n   */\n  public readonly scope?: IConstruct;\n\n  /**\n   * The id of this construct within the current scope.\n   *\n   * This is a a scope-unique id. To obtain an app-unique id for this construct, use `uniqueId`.\n   */\n  public readonly id: string;\n\n  private _locked = false; // if this is \"true\", addChild will fail\n  private readonly _aspects: IAspect[] = [];\n  private readonly _children: { [id: string]: IConstruct } = { };\n  private readonly _context: { [key: string]: any } = { };\n  private readonly _metadata = new Array<MetadataEntry>();\n  private readonly _dependencies = new Set<IConstruct>();\n  private readonly invokedAspects: IAspect[] = [];\n  private _defaultChild: IConstruct | undefined;\n\n  constructor(private readonly host: Construct, scope: IConstruct, id: string) {\n    id = id || ''; // if undefined, convert to empty string\n\n    this.id = sanitizeId(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      ConstructNode.of(scope).addChild(host, this.id);\n    } else {\n      // This is a root construct.\n      this.id = id;\n    }\n\n    if (Token.isUnresolved(id)) {\n      throw new Error(`Cannot use tokens in construct ID: ${id}`);\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.scopes.slice(1).map(c => ConstructNode.of(c).id);\n    return components.join(ConstructNode.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.scopes.slice(1).map(c => ConstructNode.of(c).id);\n    return components.length > 0 ? makeUniqueId(components) : '';\n  }\n\n  /**\n   * Return a direct child by id, or undefined\n   *\n   * @param id Identifier of direct child\n   * @returns the child if found, or undefined\n   */\n  public tryFindChild(id: string): IConstruct | undefined {\n    return this._children[sanitizeId(id)];\n  }\n\n  /**\n   * Return a direct child by id\n   *\n   * Throws an error if the child is not found.\n   *\n   * @param id Identifier of direct child\n   * @returns Child with the given id.\n   */\n  public findChild(id: string): IConstruct {\n    const ret = this.tryFindChild(id);\n    if (!ret) {\n      throw new Error(`No child with id: '${id}'`);\n    }\n    return ret;\n  }\n\n  /**\n   * Returns the child construct that has the id `Default` or `Resource\"`.\n   * This is usually the construct that provides the bulk of the underlying functionality.\n   * Useful for modifications of the underlying construct that are not available at the higher levels.\n   *\n   * @throws if there is more than one child\n   * @returns a construct or undefined if there is no default child\n   */\n  public get defaultChild(): IConstruct | undefined {\n    if (this._defaultChild !== undefined) {\n      return this._defaultChild;\n    }\n\n    const resourceChild = this.tryFindChild('Resource');\n    const defaultChild = this.tryFindChild('Default');\n    if (resourceChild && defaultChild) {\n      throw new Error(`Cannot determine default child for ${this.path}. There is both a child with id \"Resource\" and id \"Default\"`);\n    }\n\n    return defaultChild || resourceChild;\n  }\n\n  /**\n   * Override the defaultChild property.\n   *\n   * This should only be used in the cases where the correct\n   * default child is not named 'Resource' or 'Default' as it\n   * should be.\n   *\n   * If you set this to undefined, the default behavior of finding\n   * the child named 'Resource' or 'Default' will be used.\n   */\n  public set defaultChild(value: IConstruct | undefined) {\n    this._defaultChild = value;\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(c: IConstruct) {\n      if (order === ConstructOrder.PREORDER) {\n        ret.push(c);\n      }\n\n      for (const child of ConstructNode.of(c).children) {\n        visit(child);\n      }\n\n      if (order === ConstructOrder.POSTORDER) {\n        ret.push(c);\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 (Token.isUnresolved(key)) {\n      throw new Error(`Invalid context key \"${key}\". It contains unresolved tokens`);\n    }\n\n    if (this.children.length > 0) {\n      const names = this.children.map(c => ConstructNode.of(c).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` if there is no context value for thie key.\n   */\n  public tryGetContext(key: string): any {\n    if (Token.isUnresolved(key)) {\n      throw new Error(`Invalid context key \"${key}\". It contains unresolved tokens`);\n    }\n\n    const value = this._context[key];\n    if (value !== undefined) { return value; }\n\n    return this.scope && ConstructNode.of(this.scope).tryGetContext(key);\n  }\n\n  /**\n   * An immutable 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 fromFunction a function under which to restrict the metadata entry's stack trace (defaults to this.addMetadata)\n   */\n  public addMetadata(type: string, data: any, fromFunction?: any): void {\n    if (data == null) {\n      return;\n    }\n\n    const trace = this.tryGetContext(ConstructMetadata.DISABLE_STACK_TRACE_IN_METADATA)\n      ? undefined\n      : captureStackTrace(fromFunction || this.addMetadata);\n\n    this._metadata.push({ type, data, trace });\n  }\n\n  /**\n   * Adds a { \"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(ConstructMetadata.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(ConstructMetadata.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(ConstructMetadata.ERROR_METADATA_KEY, message);\n  }\n\n  /**\n   * Applies the aspect to this Constructs node\n   */\n  public applyAspect(aspect: IAspect): void {\n    this._aspects.push(aspect);\n    return;\n  }\n\n  /**\n   * All parent scopes of this construct.\n   *\n   * @returns a list of parent scopes. The last element in the list will always\n   * be the current construct and the first element will be the root of the\n   * tree.\n   */\n  public get scopes(): IConstruct[] {\n    const ret = new Array<IConstruct>();\n\n    let curr: IConstruct | undefined = this.host;\n    while (curr) {\n      ret.unshift(curr);\n      curr = ConstructNode.of(curr).scope;\n    }\n\n    return ret;\n  }\n\n  /**\n   * Returns the root of the construct tree.\n   * @returns The root of the construct tree.\n   */\n  public get root() {\n    return this.scopes[0];\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 && ConstructNode.of(this.scope).locked) {\n      return true;\n    }\n\n    return false;\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: IConstruct[]) {\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 get dependencies(): 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 ConstructNode.of(source)._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   * Remove the child with the given name, if present.\n   *\n   * @returns Whether a child with the given name was deleted.\n   * @experimental\n   */\n  public tryRemoveChild(childName: string): boolean {\n    if (!(childName in this._children)) { return false; }\n    delete this._children[childName];\n    return true;\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   * @internal\n   */\n  private _lock() {\n    this._locked = true;\n  }\n\n  /**\n   * Unlocks this costruct and allows mutations (adding children).\n   * @internal\n   */\n  private _unlock() {\n    this._locked = false;\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  private addChild(child: Construct, 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      const typeName = this.host.constructor.name;\n      throw new Error(`There is already a Construct with name '${childName}' in ${typeName}${name.length > 0 ? ' [' + name + ']' : ''}`);\n    }\n\n    this._children[childName] = child;\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/**\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   * 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, options: ConstructOptions = { }) {\n    const nodeFactory = options.nodeFactory || new DefaultNodeFactory();\n\n    Object.defineProperty(this, CONSTRUCT_NODE_PROPERTY_SYMBOL, {\n      value: nodeFactory.createNode(this, scope, id),\n      enumerable: false,\n      configurable: false\n    });\n\n    // implement IDependable privately\n    DependableTrait.implement(this, {\n      dependencyRoots: [ this ]\n    });\n  }\n\n  /**\n   * Returns a string representation of this construct.\n   */\n  public toString() {\n    return ConstructNode.of(this).path || '<root>';\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 validateConstruct(): 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 prepareConstruct(): void {\n    return;\n  }\n\n  /**\n   * Allows this construct to emit artifacts into the cloud assembly during synthesis.\n   *\n   * This method is usually implemented by framework-level constructs such as `Stack` and `Asset`\n   * as they participate in synthesizing the cloud assembly.\n   *\n   * @param session The synthesis session.\n   */\n  protected synthesizeConstruct(session: ISynthesisSession): void {\n    ignore(session);\n  }\n}\n\n/**\n * An error returned during the validation phase.\n */\nexport interface ValidationError {\n  /**\n   * The construct which emitted the error.\n   */\n  readonly source: Construct;\n\n  /**\n   * The error message.\n   */\n  readonly message: string;\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 * Represents a single session of synthesis. Passed into `construct.synthesizeConstruct()` methods.\n */\nexport interface ISynthesisSession {\n  /**\n   * The output directory for this synthesis session.\n   */\n  readonly outdir: string;\n\n  /**\n   * Additional context passed to synthesizeNode through `sessionContext`.\n   */\n  [key: string]: any;\n}\n\n/**\n * Options for synthesis.\n */\nexport interface SynthesisOptions {\n  /**\n   * The output directory into which to synthesize the cloud assembly.\n   * @default - creates a temporary directory\n   */\n  readonly outdir: string;\n\n  /**\n   * Whether synthesis should skip the validation phase.\n   * @default false\n   */\n  readonly skipValidation?: boolean;\n\n  /**\n   * Additional context passed into the synthesis session object when `construct.synth` is called.\n   * @default - no additional context is passed to `synthesizeConstruct`\n   */\n  readonly sessionContext?: { [key: string]: any };\n}\n\nfunction ignore(_x: any) {\n  return;\n}\n\n// Import this _after_ everything else to help node work the classes out in the correct order...\nconst PATH_SEP_REGEX = new RegExp(`${ConstructNode.PATH_SEP}`, 'g');\n\n/**\n * Return a sanitized version of an arbitrary string, so it can be used as an ID\n */\nfunction sanitizeId(id: string) {\n  // Escape path seps as double dashes\n  return id.replace(PATH_SEP_REGEX, '--');\n}\n\n/**\n * Options for creating constructs.\n */\nexport interface ConstructOptions {\n  /**\n   * A factory for attaching `ConstructNode`s to the construct.\n   * @default - the default `ConstructNode` is associated\n   */\n  readonly nodeFactory?: INodeFactory;\n}\n\n/**\n * A factory for attaching `ConstructNode`s to the construct.\n */\nexport interface INodeFactory {\n  /**\n   * Returns a new `ConstructNode` associated with `host`.\n   * @param host the associated construct\n   * @param scope the construct's scope (parent)\n   * @param id the construct id\n   */\n  createNode(host: Construct, scope: IConstruct, id: string): ConstructNode;\n}\n\nclass DefaultNodeFactory implements INodeFactory {\n  public createNode(host: Construct, scope: IConstruct, id: string) {\n    return new ConstructNode(host, scope, id);\n  }\n}\n"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"construct.js","sourceRoot":"","sources":["construct.ts"],"names":[],"mappings":"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","sourcesContent":["import { IAspect } from './aspect';\nimport { ConstructMetadata, MetadataEntry } from './metadata';\nimport { DependableTrait } from './private/dependency';\nimport { captureStackTrace } from './private/stack-trace';\nimport { makeUniqueId } from './private/uniqueid';\nimport { Token } from './token';\n\nconst CONSTRUCT_NODE_PROPERTY_SYMBOL = Symbol.for('constructs.Construct.node');\n\n/**\n * Represents a construct.\n */\nexport interface IConstruct { }\n\n/**\n * Represents the construct node in the scope tree.\n */\nexport class Node {\n  /**\n   * Separator used to delimit construct path components.\n   */\n  public static readonly PATH_SEP = '/';\n\n  /**\n   * Returns the node associated with a construct.\n   * @param construct the construct\n   */\n  public static of(construct: IConstruct): Node {\n    const node = (construct as any)[CONSTRUCT_NODE_PROPERTY_SYMBOL] as Node;\n    if (!node) {\n      throw new Error(`construct does not have an associated node`);\n    }\n\n    return node;\n  }\n\n  /**\n   * Returns the scope in which this construct is defined.\n   *\n   * The value is `undefined` at the root of the construct scope tree.\n   */\n  public readonly scope?: IConstruct;\n\n  /**\n   * The id of this construct within the current scope.\n   *\n   * This is a a scope-unique id. To obtain an app-unique id for this construct, use `uniqueId`.\n   */\n  public readonly id: string;\n\n  private _locked = false; // if this is \"true\", addChild will fail\n  private readonly _aspects: IAspect[] = [];\n  private readonly _children: { [id: string]: IConstruct } = { };\n  private readonly _context: { [key: string]: any } = { };\n  private readonly _metadata = new Array<MetadataEntry>();\n  private readonly _dependencies = new Set<IConstruct>();\n  private readonly invokedAspects: IAspect[] = [];\n  private _defaultChild: IConstruct | undefined;\n\n  constructor(private readonly host: Construct, scope: IConstruct, id: string) {\n    id = id || ''; // if undefined, convert to empty string\n\n    this.id = sanitizeId(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      Node.of(scope).addChild(host, this.id);\n    } else {\n      // This is a root construct.\n      this.id = id;\n    }\n\n    if (Token.isUnresolved(id)) {\n      throw new Error(`Cannot use tokens in construct ID: ${id}`);\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.scopes.slice(1).map(c => Node.of(c).id);\n    return components.join(Node.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.scopes.slice(1).map(c => Node.of(c).id);\n    return components.length > 0 ? makeUniqueId(components) : '';\n  }\n\n  /**\n   * Return a direct child by id, or undefined\n   *\n   * @param id Identifier of direct child\n   * @returns the child if found, or undefined\n   */\n  public tryFindChild(id: string): IConstruct | undefined {\n    return this._children[sanitizeId(id)];\n  }\n\n  /**\n   * Return a direct child by id\n   *\n   * Throws an error if the child is not found.\n   *\n   * @param id Identifier of direct child\n   * @returns Child with the given id.\n   */\n  public findChild(id: string): IConstruct {\n    const ret = this.tryFindChild(id);\n    if (!ret) {\n      throw new Error(`No child with id: '${id}'`);\n    }\n    return ret;\n  }\n\n  /**\n   * Returns the child construct that has the id `Default` or `Resource\"`.\n   * This is usually the construct that provides the bulk of the underlying functionality.\n   * Useful for modifications of the underlying construct that are not available at the higher levels.\n   *\n   * @throws if there is more than one child\n   * @returns a construct or undefined if there is no default child\n   */\n  public get defaultChild(): IConstruct | undefined {\n    if (this._defaultChild !== undefined) {\n      return this._defaultChild;\n    }\n\n    const resourceChild = this.tryFindChild('Resource');\n    const defaultChild = this.tryFindChild('Default');\n    if (resourceChild && defaultChild) {\n      throw new Error(`Cannot determine default child for ${this.path}. There is both a child with id \"Resource\" and id \"Default\"`);\n    }\n\n    return defaultChild || resourceChild;\n  }\n\n  /**\n   * Override the defaultChild property.\n   *\n   * This should only be used in the cases where the correct\n   * default child is not named 'Resource' or 'Default' as it\n   * should be.\n   *\n   * If you set this to undefined, the default behavior of finding\n   * the child named 'Resource' or 'Default' will be used.\n   */\n  public set defaultChild(value: IConstruct | undefined) {\n    this._defaultChild = value;\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(c: IConstruct) {\n      if (order === ConstructOrder.PREORDER) {\n        ret.push(c);\n      }\n\n      for (const child of Node.of(c).children) {\n        visit(child);\n      }\n\n      if (order === ConstructOrder.POSTORDER) {\n        ret.push(c);\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 (Token.isUnresolved(key)) {\n      throw new Error(`Invalid context key \"${key}\". It contains unresolved tokens`);\n    }\n\n    if (this.children.length > 0) {\n      const names = this.children.map(c => Node.of(c).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` if there is no context value for thie key.\n   */\n  public tryGetContext(key: string): any {\n    if (Token.isUnresolved(key)) {\n      throw new Error(`Invalid context key \"${key}\". It contains unresolved tokens`);\n    }\n\n    const value = this._context[key];\n    if (value !== undefined) { return value; }\n\n    return this.scope && Node.of(this.scope).tryGetContext(key);\n  }\n\n  /**\n   * An immutable 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 fromFunction a function under which to restrict the metadata entry's stack trace (defaults to this.addMetadata)\n   */\n  public addMetadata(type: string, data: any, fromFunction?: any): void {\n    if (data == null) {\n      return;\n    }\n\n    const trace = this.tryGetContext(ConstructMetadata.DISABLE_STACK_TRACE_IN_METADATA)\n      ? undefined\n      : captureStackTrace(fromFunction || this.addMetadata);\n\n    this._metadata.push({ type, data, trace });\n  }\n\n  /**\n   * Adds a { \"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(ConstructMetadata.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(ConstructMetadata.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(ConstructMetadata.ERROR_METADATA_KEY, message);\n  }\n\n  /**\n   * Applies the aspect to this Constructs node\n   */\n  public applyAspect(aspect: IAspect): void {\n    this._aspects.push(aspect);\n    return;\n  }\n\n  /**\n   * All parent scopes of this construct.\n   *\n   * @returns a list of parent scopes. The last element in the list will always\n   * be the current construct and the first element will be the root of the\n   * tree.\n   */\n  public get scopes(): IConstruct[] {\n    const ret = new Array<IConstruct>();\n\n    let curr: IConstruct | undefined = this.host;\n    while (curr) {\n      ret.unshift(curr);\n      curr = Node.of(curr).scope;\n    }\n\n    return ret;\n  }\n\n  /**\n   * Returns the root of the construct tree.\n   * @returns The root of the construct tree.\n   */\n  public get root() {\n    return this.scopes[0];\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 && Node.of(this.scope).locked) {\n      return true;\n    }\n\n    return false;\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: IConstruct[]) {\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 get dependencies(): 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 Node.of(source)._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   * Remove the child with the given name, if present.\n   *\n   * @returns Whether a child with the given name was deleted.\n   * @experimental\n   */\n  public tryRemoveChild(childName: string): boolean {\n    if (!(childName in this._children)) { return false; }\n    delete this._children[childName];\n    return true;\n  }\n\n    /**\n   * Synthesizes a CloudAssembly from a construct tree.\n   * @param options Synthesis options.\n   */\n  public synthesize(options: SynthesisOptions): void {\n    // the three holy phases of synthesis: prepare, validate and synthesize\n\n    // prepare\n    this.prepare();\n\n    // validate\n    const validate = options.skipValidation === undefined ? true : !options.skipValidation;\n    if (validate) {\n      const errors = this.validate();\n      if (errors.length > 0) {\n        const errorList = errors.map(e => `[${Node.of(e.source).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 construct of this.findAll(ConstructOrder.POSTORDER)) {\n      const node = Node.of(construct);\n      try {\n        node._lock();\n        const ctx = {\n          ...options.sessionContext,\n          outdir: options.outdir\n        };\n        (construct as any).onSynthesize(ctx); // \"as any\" is needed because we want to keep \"synthesize\" protected\n      } finally {\n        node._unlock();\n      }\n    }\n  }\n\n  /**\n   * Invokes \"prepare\" on all constructs (depth-first, post-order) in the tree under `node`.\n   */\n  public prepare() {\n    const constructs = this.findAll(ConstructOrder.PREORDER);\n\n    // Aspects are applied root to leaf\n    for (const construct of constructs) {\n      Node.of(construct).invokeAspects();\n    }\n\n    // Use .reverse() to achieve post-order traversal\n    for (const construct of constructs.reverse()) {\n      if (construct instanceof Construct) {\n        (construct as any).onPrepare(); // \"as any\" is needed because we want to keep \"prepare\" protected\n      }\n    }\n  }\n\n  /**\n   * Invokes \"validate\" on all constructs in the tree (depth-first, pre-order) and returns\n   * the list of all errors. An empty list indicates that there are no errors.\n   */\n  public validate() {\n    let errors = new Array<ValidationError>();\n\n    for (const child of this.children) {\n      errors = errors.concat(Node.of(child).validate());\n    }\n\n    const localErrors: string[] = (this.host as any).onValidate(); // \"as any\" is needed because we want to keep \"validate\" protected\n    return errors.concat(localErrors.map(msg => ({ source: this.host, message: msg })));\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   * @internal\n   */\n  private _lock() {\n    this._locked = true;\n  }\n\n  /**\n   * Unlocks this costruct and allows mutations (adding children).\n   * @internal\n   */\n  private _unlock() {\n    this._locked = false;\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  private addChild(child: Construct, 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      const typeName = this.host.constructor.name;\n      throw new Error(`There is already a Construct with name '${childName}' in ${typeName}${name.length > 0 ? ' [' + name + ']' : ''}`);\n    }\n\n    this._children[childName] = child;\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/**\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   * 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   * @param options Options\n   */\n  constructor(scope: Construct, id: string, options: ConstructOptions = { }) {\n\n    // attach the construct to the construct tree by creating a node\n    const nodeFactory = options.nodeFactory ?? { createNode: () => new Node(this, scope, id) };\n    Object.defineProperty(this, CONSTRUCT_NODE_PROPERTY_SYMBOL, {\n      value: nodeFactory.createNode(this, scope, id),\n      enumerable: false,\n      configurable: false\n    });\n\n    // implement IDependable privately\n    DependableTrait.implement(this, {\n      dependencyRoots: [ this ]\n    });\n  }\n\n  /**\n   * Returns a string representation of this construct.\n   */\n  public toString() {\n    return Node.of(this).path || '<root>';\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 onValidate(): 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 onPrepare(): void {\n    return;\n  }\n\n  /**\n   * Allows this construct to emit artifacts into the cloud assembly during synthesis.\n   *\n   * This method is usually implemented by framework-level constructs such as `Stack` and `Asset`\n   * as they participate in synthesizing the cloud assembly.\n   *\n   * @param session The synthesis session.\n   */\n  protected onSynthesize(session: ISynthesisSession): void {\n    ignore(session);\n  }\n}\n\n/**\n * An error returned during the validation phase.\n */\nexport interface ValidationError {\n  /**\n   * The construct which emitted the error.\n   */\n  readonly source: Construct;\n\n  /**\n   * The error message.\n   */\n  readonly message: string;\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 * Represents a single session of synthesis. Passed into `construct.onSynthesize()` methods.\n */\nexport interface ISynthesisSession {\n  /**\n   * The output directory for this synthesis session.\n   */\n  readonly outdir: string;\n\n  /**\n   * Additional context passed to synthesizeNode through `sessionContext`.\n   */\n  [key: string]: any;\n}\n\n/**\n * Options for synthesis.\n */\nexport interface SynthesisOptions {\n  /**\n   * The output directory into which to synthesize the cloud assembly.\n   * @default - creates a temporary directory\n   */\n  readonly outdir: string;\n\n  /**\n   * Whether synthesis should skip the validation phase.\n   * @default false\n   */\n  readonly skipValidation?: boolean;\n\n  /**\n   * Additional context passed into the synthesis session object when `construct.synth` is called.\n   * @default - no additional context is passed to `onSynthesize`\n   */\n  readonly sessionContext?: { [key: string]: any };\n}\n\nfunction ignore(_x: any) {\n  return;\n}\n\n// Import this _after_ everything else to help node work the classes out in the correct order...\nconst PATH_SEP_REGEX = new RegExp(`${Node.PATH_SEP}`, 'g');\n\n/**\n * Return a sanitized version of an arbitrary string, so it can be used as an ID\n */\nfunction sanitizeId(id: string) {\n  // Escape path seps as double dashes\n  return id.replace(PATH_SEP_REGEX, '--');\n}\n\n/**\n * Options for creating constructs.\n */\nexport interface ConstructOptions {\n  /**\n   * A factory for attaching `Node`s to the construct.\n   * @default - the default `Node` is associated\n   */\n  readonly nodeFactory?: INodeFactory;\n}\n\n/**\n * A factory for attaching `Node`s to the construct.\n */\nexport interface INodeFactory {\n  /**\n   * Returns a new `Node` associated with `host`.\n   * @param host the associated construct\n   * @param scope the construct's scope (parent)\n   * @param id the construct id\n   */\n  createNode(host: Construct, scope: IConstruct, id: string): Node;\n}\n"]}
{
"name": "constructs",
"version": "1.0.0",
"version": "1.0.1",
"description": "A programming model for composable configuration",

@@ -5,0 +5,0 @@ "stability": "stable",

@@ -24,7 +24,7 @@ "use strict";

const root = new MyConstruct(app, 'MyConstruct');
const node = construct_1.ConstructNode.of(root);
const node = construct_1.Node.of(root);
node.applyAspect(new VisitOnce());
construct_1.ConstructNode.prepareNode(node);
node.prepare();
test.deepEqual(root.visitCounter, 1);
construct_1.ConstructNode.prepareNode(node);
node.prepare();
test.deepEqual(root.visitCounter, 1);

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

};
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGVzdC5hc3BlY3QuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJ0ZXN0LmFzcGVjdC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBRUEsZ0RBQXdFO0FBQ3hFLGlDQUE2QjtBQUU3QixNQUFNLFdBQVksU0FBUSxxQkFBUztJQUFuQzs7UUFJUyxpQkFBWSxHQUFXLENBQUMsQ0FBQztJQUNsQyxDQUFDO0lBSlEsTUFBTSxDQUFDLGFBQWEsQ0FBQyxDQUFNO1FBQ2hDLE9BQU8sQ0FBQyxDQUFDLFlBQVksS0FBSyxTQUFTLENBQUM7SUFDdEMsQ0FBQztDQUVGO0FBRUQsTUFBTSxTQUFTO0lBQ04sS0FBSyxDQUFDLElBQWdCO1FBQzNCLElBQUksV0FBVyxDQUFDLGFBQWEsQ0FBQyxJQUFJLENBQUMsRUFBRTtZQUNuQyxJQUFJLENBQUMsWUFBWSxJQUFJLENBQUMsQ0FBQztTQUN4QjtJQUNILENBQUM7Q0FDRjtBQUNELGlCQUFTO0lBQ1AsK0JBQStCLENBQUMsSUFBVTtRQUN4QyxNQUFNLEdBQUcsR0FBRyxJQUFJLFVBQUcsRUFBRSxDQUFDO1FBQ3RCLE1BQU0sSUFBSSxHQUFHLElBQUksV0FBVyxDQUFDLEdBQUcsRUFBRSxhQUFhLENBQUMsQ0FBQztRQUNqRCxNQUFNLElBQUksR0FBRyx5QkFBYSxDQUFDLEVBQUUsQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUNwQyxJQUFJLENBQUMsV0FBVyxDQUFDLElBQUksU0FBUyxFQUFFLENBQUMsQ0FBQztRQUNsQyx5QkFBYSxDQUFDLFdBQVcsQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUNoQyxJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxZQUFZLEVBQUUsQ0FBQyxDQUFDLENBQUM7UUFDckMseUJBQWEsQ0FBQyxXQUFXLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDaEMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQUMsWUFBWSxFQUFFLENBQUMsQ0FBQyxDQUFDO1FBQ3JDLElBQUksQ0FBQyxJQUFJLEVBQUUsQ0FBQztJQUNkLENBQUM7Q0FDRixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgVGVzdCB9IGZyb20gJ25vZGV1bml0JztcbmltcG9ydCB7IElBc3BlY3QgfSBmcm9tICcuLi9saWIvYXNwZWN0JztcbmltcG9ydCB7IENvbnN0cnVjdCwgQ29uc3RydWN0Tm9kZSwgSUNvbnN0cnVjdCB9IGZyb20gJy4uL2xpYi9jb25zdHJ1Y3QnO1xuaW1wb3J0IHsgQXBwIH0gZnJvbSAnLi91dGlsJztcblxuY2xhc3MgTXlDb25zdHJ1Y3QgZXh0ZW5kcyBDb25zdHJ1Y3Qge1xuICBwdWJsaWMgc3RhdGljIElzTXlDb25zdHJ1Y3QoeDogYW55KTogeCBpcyBNeUNvbnN0cnVjdCB7XG4gICAgcmV0dXJuIHgudmlzaXRDb3VudGVyICE9PSB1bmRlZmluZWQ7XG4gIH1cbiAgcHVibGljIHZpc2l0Q291bnRlcjogbnVtYmVyID0gMDtcbn1cblxuY2xhc3MgVmlzaXRPbmNlIGltcGxlbWVudHMgSUFzcGVjdCB7XG4gIHB1YmxpYyB2aXNpdChub2RlOiBJQ29uc3RydWN0KTogdm9pZCB7XG4gICAgaWYgKE15Q29uc3RydWN0LklzTXlDb25zdHJ1Y3Qobm9kZSkpIHtcbiAgICAgIG5vZGUudmlzaXRDb3VudGVyICs9IDE7XG4gICAgfVxuICB9XG59XG5leHBvcnQgPSB7XG4gICdBc3BlY3RzIGFyZSBpbnZva2VkIG9ubHkgb25jZScodGVzdDogVGVzdCkge1xuICAgIGNvbnN0IGFwcCA9IG5ldyBBcHAoKTtcbiAgICBjb25zdCByb290ID0gbmV3IE15Q29uc3RydWN0KGFwcCwgJ015Q29uc3RydWN0Jyk7XG4gICAgY29uc3Qgbm9kZSA9IENvbnN0cnVjdE5vZGUub2Yocm9vdCk7XG4gICAgbm9kZS5hcHBseUFzcGVjdChuZXcgVmlzaXRPbmNlKCkpO1xuICAgIENvbnN0cnVjdE5vZGUucHJlcGFyZU5vZGUobm9kZSk7XG4gICAgdGVzdC5kZWVwRXF1YWwocm9vdC52aXNpdENvdW50ZXIsIDEpO1xuICAgIENvbnN0cnVjdE5vZGUucHJlcGFyZU5vZGUobm9kZSk7XG4gICAgdGVzdC5kZWVwRXF1YWwocm9vdC52aXNpdENvdW50ZXIsIDEpO1xuICAgIHRlc3QuZG9uZSgpO1xuICB9LFxufTtcbiJdfQ==
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGVzdC5hc3BlY3QuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJ0ZXN0LmFzcGVjdC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBRUEsZ0RBQStEO0FBQy9ELGlDQUE2QjtBQUU3QixNQUFNLFdBQVksU0FBUSxxQkFBUztJQUFuQzs7UUFJUyxpQkFBWSxHQUFXLENBQUMsQ0FBQztJQUNsQyxDQUFDO0lBSlEsTUFBTSxDQUFDLGFBQWEsQ0FBQyxDQUFNO1FBQ2hDLE9BQU8sQ0FBQyxDQUFDLFlBQVksS0FBSyxTQUFTLENBQUM7SUFDdEMsQ0FBQztDQUVGO0FBRUQsTUFBTSxTQUFTO0lBQ04sS0FBSyxDQUFDLElBQWdCO1FBQzNCLElBQUksV0FBVyxDQUFDLGFBQWEsQ0FBQyxJQUFJLENBQUMsRUFBRTtZQUNuQyxJQUFJLENBQUMsWUFBWSxJQUFJLENBQUMsQ0FBQztTQUN4QjtJQUNILENBQUM7Q0FDRjtBQUNELGlCQUFTO0lBQ1AsK0JBQStCLENBQUMsSUFBVTtRQUN4QyxNQUFNLEdBQUcsR0FBRyxJQUFJLFVBQUcsRUFBRSxDQUFDO1FBQ3RCLE1BQU0sSUFBSSxHQUFHLElBQUksV0FBVyxDQUFDLEdBQUcsRUFBRSxhQUFhLENBQUMsQ0FBQztRQUNqRCxNQUFNLElBQUksR0FBRyxnQkFBSSxDQUFDLEVBQUUsQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUMzQixJQUFJLENBQUMsV0FBVyxDQUFDLElBQUksU0FBUyxFQUFFLENBQUMsQ0FBQztRQUNsQyxJQUFJLENBQUMsT0FBTyxFQUFFLENBQUM7UUFDZixJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxZQUFZLEVBQUUsQ0FBQyxDQUFDLENBQUM7UUFDckMsSUFBSSxDQUFDLE9BQU8sRUFBRSxDQUFDO1FBQ2YsSUFBSSxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQUMsWUFBWSxFQUFFLENBQUMsQ0FBQyxDQUFDO1FBQ3JDLElBQUksQ0FBQyxJQUFJLEVBQUUsQ0FBQztJQUNkLENBQUM7Q0FDRixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgVGVzdCB9IGZyb20gJ25vZGV1bml0JztcbmltcG9ydCB7IElBc3BlY3QgfSBmcm9tICcuLi9saWIvYXNwZWN0JztcbmltcG9ydCB7IENvbnN0cnVjdCwgTm9kZSwgSUNvbnN0cnVjdCB9IGZyb20gJy4uL2xpYi9jb25zdHJ1Y3QnO1xuaW1wb3J0IHsgQXBwIH0gZnJvbSAnLi91dGlsJztcblxuY2xhc3MgTXlDb25zdHJ1Y3QgZXh0ZW5kcyBDb25zdHJ1Y3Qge1xuICBwdWJsaWMgc3RhdGljIElzTXlDb25zdHJ1Y3QoeDogYW55KTogeCBpcyBNeUNvbnN0cnVjdCB7XG4gICAgcmV0dXJuIHgudmlzaXRDb3VudGVyICE9PSB1bmRlZmluZWQ7XG4gIH1cbiAgcHVibGljIHZpc2l0Q291bnRlcjogbnVtYmVyID0gMDtcbn1cblxuY2xhc3MgVmlzaXRPbmNlIGltcGxlbWVudHMgSUFzcGVjdCB7XG4gIHB1YmxpYyB2aXNpdChub2RlOiBJQ29uc3RydWN0KTogdm9pZCB7XG4gICAgaWYgKE15Q29uc3RydWN0LklzTXlDb25zdHJ1Y3Qobm9kZSkpIHtcbiAgICAgIG5vZGUudmlzaXRDb3VudGVyICs9IDE7XG4gICAgfVxuICB9XG59XG5leHBvcnQgPSB7XG4gICdBc3BlY3RzIGFyZSBpbnZva2VkIG9ubHkgb25jZScodGVzdDogVGVzdCkge1xuICAgIGNvbnN0IGFwcCA9IG5ldyBBcHAoKTtcbiAgICBjb25zdCByb290ID0gbmV3IE15Q29uc3RydWN0KGFwcCwgJ015Q29uc3RydWN0Jyk7XG4gICAgY29uc3Qgbm9kZSA9IE5vZGUub2Yocm9vdCk7XG4gICAgbm9kZS5hcHBseUFzcGVjdChuZXcgVmlzaXRPbmNlKCkpO1xuICAgIG5vZGUucHJlcGFyZSgpO1xuICAgIHRlc3QuZGVlcEVxdWFsKHJvb3QudmlzaXRDb3VudGVyLCAxKTtcbiAgICBub2RlLnByZXBhcmUoKTtcbiAgICB0ZXN0LmRlZXBFcXVhbChyb290LnZpc2l0Q291bnRlciwgMSk7XG4gICAgdGVzdC5kb25lKCk7XG4gIH0sXG59O1xuIl19

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

'multiple children of the same type, with explicit names are welcome'(test: Test): void;
'construct.validateConstruct() can be implemented to perform validation, ConstructNode.validateNode(construct.node) will return all errors from the subtree (DFS)'(test: Test): void;
'construct.onValidate() can be implemented to perform validation, node.validate() will return all errors from the subtree (DFS)'(test: Test): void;
'construct.lock() protects against adding children anywhere under this construct (direct or indirect)'(test: Test): void;

@@ -31,0 +31,0 @@ 'findAll returns a list of all children in either DFS or BFS'(test: Test): void;

Sorry, the diff of this file is not supported yet

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

SocketSocket SOC 2 Logo

Product

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

Packages

npm

Stay in touch

Get open source security insights delivered straight into your inbox.


  • Terms
  • Privacy
  • Security

Made with ⚡️ by Socket Inc