@aws-cdk/core
Version:
AWS Cloud Development Kit Core Library
184 lines • 22.5 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", { value: true });
exports.synthesize = void 0;
const cxapi = require("@aws-cdk/cx-api");
const annotations_1 = require("../annotations");
const aspect_1 = require("../aspect");
const stack_1 = require("../stack");
const stage_1 = require("../stage");
const metadata_resource_1 = require("./metadata-resource");
const prepare_app_1 = require("./prepare-app");
const tree_metadata_1 = require("./tree-metadata");
function synthesize(root, options = {}) {
// we start by calling "synth" on all nested assemblies (which will take care of all their children)
synthNestedAssemblies(root, options);
invokeAspects(root);
injectMetadataResources(root);
// This is mostly here for legacy purposes as the framework itself does not use prepare anymore.
prepareTree(root);
// resolve references
prepare_app_1.prepareApp(root);
// give all children an opportunity to validate now that we've finished prepare
if (!options.skipValidation) {
validateTree(root);
}
// in unit tests, we support creating free-standing stacks, so we create the
// assembly builder here.
const builder = stage_1.Stage.isStage(root)
? root._assemblyBuilder
: new cxapi.CloudAssemblyBuilder(options.outdir);
// next, we invoke "onSynthesize" on all of our children. this will allow
// stacks to add themselves to the synthesized cloud assembly.
synthesizeTree(root, builder);
return builder.buildAssembly();
}
exports.synthesize = synthesize;
/**
* Find Assemblies inside the construct and call 'synth' on them
*
* (They will in turn recurse again)
*/
function synthNestedAssemblies(root, options) {
for (const child of root.node.children) {
if (stage_1.Stage.isStage(child)) {
child.synth(options);
}
else {
synthNestedAssemblies(child, options);
}
}
}
/**
* Invoke aspects on the given construct tree.
*
* Aspects are not propagated across Assembly boundaries. The same Aspect will not be invoked
* twice for the same construct.
*/
function invokeAspects(root) {
const invokedByPath = {};
let nestedAspectWarning = false;
recurse(root, []);
function recurse(construct, inheritedAspects) {
const node = construct.node;
const aspects = aspect_1.Aspects.of(construct);
const allAspectsHere = [...inheritedAspects !== null && inheritedAspects !== void 0 ? inheritedAspects : [], ...aspects.aspects];
const nodeAspectsCount = aspects.aspects.length;
for (const aspect of allAspectsHere) {
let invoked = invokedByPath[node.path];
if (!invoked) {
invoked = invokedByPath[node.path] = [];
}
if (invoked.includes(aspect)) {
continue;
}
aspect.visit(construct);
// if an aspect was added to the node while invoking another aspect it will not be invoked, emit a warning
// the `nestedAspectWarning` flag is used to prevent the warning from being emitted for every child
if (!nestedAspectWarning && nodeAspectsCount !== aspects.aspects.length) {
annotations_1.Annotations.of(construct).addWarning('We detected an Aspect was added via another Aspect, and will not be applied');
nestedAspectWarning = true;
}
// mark as invoked for this node
invoked.push(aspect);
}
for (const child of construct.node.children) {
if (!stage_1.Stage.isStage(child)) {
recurse(child, allAspectsHere);
}
}
}
}
/**
* Prepare all constructs in the given construct tree in post-order.
*
* Stop at Assembly boundaries.
*/
function prepareTree(root) {
visit(root, 'post', construct => construct.onPrepare());
}
/**
* Find all stacks and add Metadata Resources to all of them
*
* There is no good generic place to do this. Can't do it in the constructor
* (because adding a child construct makes it impossible to set context on the
* node), and the generic prepare phase is deprecated.
*
* Only do this on [parent] stacks (not nested stacks), don't do this when
* disabled by the user.
*
* Also, only when running via the CLI. If we do it unconditionally,
* all unit tests everywhere are going to break massively. I've spent a day
* fixing our own, but downstream users would be affected just as badly.
*
* Stop at Assembly boundaries.
*/
function injectMetadataResources(root) {
visit(root, 'post', construct => {
if (!stack_1.Stack.isStack(construct) || !construct._versionReportingEnabled) {
return;
}
// Because of https://github.com/aws/aws-cdk/blob/master/packages/@aws-cdk/assert/lib/synth-utils.ts#L74
// synthesize() may be called more than once on a stack in unit tests, and the below would break
// if we execute it a second time. Guard against the constructs already existing.
const CDKMetadata = 'CDKMetadata';
if (construct.node.tryFindChild(CDKMetadata)) {
return;
}
new metadata_resource_1.MetadataResource(construct, CDKMetadata);
});
}
/**
* Synthesize children in post-order into the given builder
*
* Stop at Assembly boundaries.
*/
function synthesizeTree(root, builder) {
visit(root, 'post', construct => {
const session = {
outdir: builder.outdir,
assembly: builder,
};
if (stack_1.Stack.isStack(construct)) {
construct.synthesizer.synthesize(session);
}
else if (construct instanceof tree_metadata_1.TreeMetadata) {
construct._synthesizeTree(session);
}
// this will soon be deprecated and removed in 2.x
// see https://github.com/aws/aws-cdk-rfcs/issues/192
construct.onSynthesize(session);
});
}
/**
* Validate all constructs in the given construct tree
*/
function validateTree(root) {
const errors = new Array();
visit(root, 'pre', construct => {
for (const message of construct.onValidate()) {
errors.push({ message, source: construct });
}
});
if (errors.length > 0) {
const errorList = errors.map(e => `[${e.source.node.path}] ${e.message}`).join('\n ');
throw new Error(`Validation failed with the following errors:\n ${errorList}`);
}
}
/**
* Visit the given construct tree in either pre or post order, stopping at Assemblies
*/
function visit(root, order, cb) {
if (order === 'pre') {
cb(root);
}
for (const child of root.node.children) {
if (stage_1.Stage.isStage(child)) {
continue;
}
visit(child, order, cb);
}
if (order === 'post') {
cb(root);
}
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"synthesis.js","sourceRoot":"","sources":["synthesis.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cxapi from '@aws-cdk/cx-api';\nimport * as constructs from 'constructs';\nimport { Annotations } from '../annotations';\nimport { Aspects, IAspect } from '../aspect';\nimport { Construct, IConstruct, SynthesisOptions, ValidationError } from '../construct-compat';\nimport { Stack } from '../stack';\nimport { Stage, StageSynthesisOptions } from '../stage';\nimport { MetadataResource } from './metadata-resource';\nimport { prepareApp } from './prepare-app';\nimport { TreeMetadata } from './tree-metadata';\n\nexport function synthesize(root: IConstruct, options: SynthesisOptions = { }): cxapi.CloudAssembly {\n  // we start by calling \"synth\" on all nested assemblies (which will take care of all their children)\n  synthNestedAssemblies(root, options);\n\n  invokeAspects(root);\n\n  injectMetadataResources(root);\n\n  // This is mostly here for legacy purposes as the framework itself does not use prepare anymore.\n  prepareTree(root);\n\n  // resolve references\n  prepareApp(root);\n\n  // give all children an opportunity to validate now that we've finished prepare\n  if (!options.skipValidation) {\n    validateTree(root);\n  }\n\n  // in unit tests, we support creating free-standing stacks, so we create the\n  // assembly builder here.\n  const builder = Stage.isStage(root)\n    ? root._assemblyBuilder\n    : new cxapi.CloudAssemblyBuilder(options.outdir);\n\n  // next, we invoke \"onSynthesize\" on all of our children. this will allow\n  // stacks to add themselves to the synthesized cloud assembly.\n  synthesizeTree(root, builder);\n\n  return builder.buildAssembly();\n}\n\n/**\n * Find Assemblies inside the construct and call 'synth' on them\n *\n * (They will in turn recurse again)\n */\nfunction synthNestedAssemblies(root: IConstruct, options: StageSynthesisOptions) {\n  for (const child of root.node.children) {\n    if (Stage.isStage(child)) {\n      child.synth(options);\n    } else {\n      synthNestedAssemblies(child, options);\n    }\n  }\n}\n\n/**\n * Invoke aspects on the given construct tree.\n *\n * Aspects are not propagated across Assembly boundaries. The same Aspect will not be invoked\n * twice for the same construct.\n */\nfunction invokeAspects(root: IConstruct) {\n  const invokedByPath: { [nodePath: string]: IAspect[] } = { };\n\n  let nestedAspectWarning = false;\n  recurse(root, []);\n\n  function recurse(construct: IConstruct, inheritedAspects: constructs.IAspect[]) {\n    const node = construct.node;\n    const aspects = Aspects.of(construct);\n    const allAspectsHere = [...inheritedAspects ?? [], ...aspects.aspects];\n    const nodeAspectsCount = aspects.aspects.length;\n    for (const aspect of allAspectsHere) {\n      let invoked = invokedByPath[node.path];\n      if (!invoked) {\n        invoked = invokedByPath[node.path] = [];\n      }\n\n      if (invoked.includes(aspect)) { continue; }\n\n      aspect.visit(construct);\n\n      // if an aspect was added to the node while invoking another aspect it will not be invoked, emit a warning\n      // the `nestedAspectWarning` flag is used to prevent the warning from being emitted for every child\n      if (!nestedAspectWarning && nodeAspectsCount !== aspects.aspects.length) {\n        Annotations.of(construct).addWarning('We detected an Aspect was added via another Aspect, and will not be applied');\n        nestedAspectWarning = true;\n      }\n\n      // mark as invoked for this node\n      invoked.push(aspect);\n    }\n\n    for (const child of construct.node.children) {\n      if (!Stage.isStage(child)) {\n        recurse(child, allAspectsHere);\n      }\n    }\n  }\n}\n\n/**\n * Prepare all constructs in the given construct tree in post-order.\n *\n * Stop at Assembly boundaries.\n */\nfunction prepareTree(root: IConstruct) {\n  visit(root, 'post', construct => construct.onPrepare());\n}\n\n/**\n * Find all stacks and add Metadata Resources to all of them\n *\n * There is no good generic place to do this. Can't do it in the constructor\n * (because adding a child construct makes it impossible to set context on the\n * node), and the generic prepare phase is deprecated.\n *\n * Only do this on [parent] stacks (not nested stacks), don't do this when\n * disabled by the user.\n *\n * Also, only when running via the CLI. If we do it unconditionally,\n * all unit tests everywhere are going to break massively. I've spent a day\n * fixing our own, but downstream users would be affected just as badly.\n *\n * Stop at Assembly boundaries.\n */\nfunction injectMetadataResources(root: IConstruct) {\n  visit(root, 'post', construct => {\n    if (!Stack.isStack(construct) || !construct._versionReportingEnabled) { return; }\n\n    // Because of https://github.com/aws/aws-cdk/blob/master/packages/@aws-cdk/assert/lib/synth-utils.ts#L74\n    // synthesize() may be called more than once on a stack in unit tests, and the below would break\n    // if we execute it a second time. Guard against the constructs already existing.\n    const CDKMetadata = 'CDKMetadata';\n    if (construct.node.tryFindChild(CDKMetadata)) { return; }\n\n    new MetadataResource(construct, CDKMetadata);\n  });\n}\n\n/**\n * Synthesize children in post-order into the given builder\n *\n * Stop at Assembly boundaries.\n */\nfunction synthesizeTree(root: IConstruct, builder: cxapi.CloudAssemblyBuilder) {\n  visit(root, 'post', construct => {\n    const session = {\n      outdir: builder.outdir,\n      assembly: builder,\n    };\n\n    if (Stack.isStack(construct)) {\n      construct.synthesizer.synthesize(session);\n    } else if (construct instanceof TreeMetadata) {\n      construct._synthesizeTree(session);\n    }\n\n    // this will soon be deprecated and removed in 2.x\n    // see https://github.com/aws/aws-cdk-rfcs/issues/192\n    construct.onSynthesize(session);\n  });\n}\n\n/**\n * Validate all constructs in the given construct tree\n */\nfunction validateTree(root: IConstruct) {\n  const errors = new Array<ValidationError>();\n\n  visit(root, 'pre', construct => {\n    for (const message of construct.onValidate()) {\n      errors.push({ message, source: construct as unknown as Construct });\n    }\n  });\n\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/**\n * Visit the given construct tree in either pre or post order, stopping at Assemblies\n */\nfunction visit(root: IConstruct, order: 'pre' | 'post', cb: (x: IProtectedConstructMethods) => void) {\n  if (order === 'pre') {\n    cb(root as IProtectedConstructMethods);\n  }\n\n  for (const child of root.node.children) {\n    if (Stage.isStage(child)) { continue; }\n    visit(child, order, cb);\n  }\n\n  if (order === 'post') {\n    cb(root as IProtectedConstructMethods);\n  }\n}\n\n/**\n * Interface which provides access to special methods of Construct\n *\n * @experimental\n */\ninterface IProtectedConstructMethods extends IConstruct {\n  /**\n   * Method that gets called when a construct should synthesize itself to an assembly\n   */\n  onSynthesize(session: constructs.ISynthesisSession): void;\n\n  /**\n   * Method that gets called to validate a construct\n   */\n  onValidate(): string[];\n\n  /**\n   * Method that gets called to prepare a construct\n   */\n  onPrepare(): void;\n}\n"]}