@aws/pdk
Version:
All documentation is located at: https://aws.github.io/aws-pdk
195 lines • 29.4 kB
JavaScript
;
var _a, _b;
Object.defineProperty(exports, "__esModule", { value: true });
exports.CdkGraph = exports.CdkGraphContext = exports.CdkGraphArtifacts = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
/*! Copyright [Amazon.com](http://amazon.com/), Inc. or its affiliates. All Rights Reserved.
SPDX-License-Identifier: Apache-2.0 */
const path = require("path");
const monorepo_1 = require("../monorepo");
const aws_cdk_lib_1 = require("aws-cdk-lib");
const chalk = require("chalk"); // eslint-disable-line @typescript-eslint/no-require-imports
const constructs_1 = require("constructs");
const fs = require("fs-extra");
const cdk_internals_1 = require("./cdk-internals");
const config_1 = require("./config");
const core_1 = require("./core");
const GRAPH_ARTIFACT_ID = "GRAPH";
/** CdkGraph core artifacts */
var CdkGraphArtifacts;
(function (CdkGraphArtifacts) {
CdkGraphArtifacts["GRAPH_METADATA"] = "graph-metadata.json";
CdkGraphArtifacts["GRAPH"] = "graph.json";
})(CdkGraphArtifacts || (exports.CdkGraphArtifacts = CdkGraphArtifacts = {}));
/** CdkGraph context */
class CdkGraphContext {
constructor(store, outdir) {
this.store = store;
this.outdir = outdir;
/** @internal */
this._artifacts = {};
}
/**
* Get CdkGraph artifact by id
* @throws Error is artifact does not exist
*/
getArtifact(id) {
const artifact = this._artifacts[id];
if (artifact) {
return artifact;
}
throw new Error(`Graph artifact ${id} does not exist`);
}
/** Get CdkGraph core `graph.json` artifact */
get graphJson() {
return this.getArtifact(GRAPH_ARTIFACT_ID);
}
/** Indicates if context has an artifact with *filename* defined */
hasArtifactFile(filename) {
return !!Object.values(this._artifacts).find((artifact) => artifact.filename === filename);
}
/** Get record of all graph artifacts keyed by artifact id */
get artifacts() {
return this._artifacts;
}
/**
* Logs an artifact entry. In general this should not be called directly, as `writeArtifact` should be utilized
* to perform writing and logging artifacts. However some plugins utilize other tools that generate the artifacts,
* in which case the plugin would call this method to log the entry.
* @param source The source of the artifact, such as the name of plugin
* @param id Unique id of the artifact
* @param filepath Full path where the artifact is stored
* @param description Description of the artifact
* @returns
* @throws Error is artifact id or filename already exists
*/
logArtifact(source, id, filepath, description) {
if (id in this._artifacts) {
throw new Error(`Graph artifact ${id} already defined`);
}
if (this.hasArtifactFile(filepath)) {
throw new Error(`Graph artifact "${filepath}" already defined`);
}
const filename = path.relative(this.outdir, filepath);
if (!(source instanceof CdkGraph)) {
if (Object.keys(CdkGraphArtifacts).includes(id)) {
throw new Error(`Graph artifact id ${id} is reserved`);
}
if (Object.values(CdkGraphArtifacts).includes(filename)) {
throw new Error(`Graph artifact file ${filename} is reserved`);
}
}
const artifact = {
id,
filepath,
description,
filename,
source: source instanceof CdkGraph
? `${CdkGraph.ID}`
: `plugin:${source.id}@${source.version}`,
};
this._artifacts[id] = artifact;
console.info(chalk.cyanBright(`[CdkGraph] Artifact ${id} written to \x1B]8;;file://${artifact.filepath}\x1B\\${artifact.filename}\x1B]8;;\x1B\\ (${artifact.source})`));
return artifact;
}
/**
* Writes artifact data to outdir and logs the entry.
* @param source The source of the artifact, such as the name of plugin
* @param id Unique id of the artifact
* @param filename Relative name of the file
* @param description Description of the artifact
* @returns
*/
writeArtifact(source, id, filename, data, description) {
const filepath = path.join(this.outdir, filename);
const artifact = this.logArtifact(source, id, filepath, description);
fs.ensureDirSync(path.dirname(filepath));
fs.writeFileSync(filepath, data, { encoding: "utf-8" });
return artifact;
}
}
exports.CdkGraphContext = CdkGraphContext;
_a = JSII_RTTI_SYMBOL_1;
CdkGraphContext[_a] = { fqn: "@aws/pdk.cdk_graph.CdkGraphContext", version: "0.26.14" };
/**
* CdkGraph construct is the cdk-graph framework controller that is responsible for
* computing the graph, storing serialized graph, and instrumenting plugins per the
* plugin contract.
*/
class CdkGraph extends constructs_1.Construct {
/**
* Get the context for the graph instance.
*
* This will be `undefined` before construct synthesis has initiated.
*/
get graphContext() {
return this._context;
}
constructor(root, props = {}) {
super(root, CdkGraph.ID);
this.root = root;
(0, monorepo_1.addMetric)(root, "cdk-graph");
this.config = (0, config_1.resolveConfig)();
this.plugins = props.plugins || [];
// TODO: verify plugin deps via semver
// bind all plugins to this instance of the graph
this.plugins.forEach((plugin) => {
(0, monorepo_1.addMetric)(root, `cdk-graph-plugin-${plugin.id}`);
plugin.bind(this);
});
// Apply Aspect for each plugin that supports "inspect" phase
this.plugins.forEach((plugin) => {
if (plugin.inspect) {
aws_cdk_lib_1.Aspects.of(this.root).add({
visit: plugin.inspect,
});
}
});
(0, cdk_internals_1.addCustomSynthesis)(this, {
onSynthesize: (session) => {
this._synthesize(session);
},
});
}
/** @internal */
_synthesize(session) {
const store = (0, core_1.computeGraph)(this.root);
const outdir = (0, config_1.resolveOutdir)(session.outdir, this.config.outdir);
const context = new CdkGraphContext(store, outdir);
context.writeArtifact(this, GRAPH_ARTIFACT_ID, CdkGraphArtifacts.GRAPH, JSON.stringify(context.store.serialize(), null, 2), "Serialized graph");
this.plugins.forEach((plugin) => {
plugin.synthesize && plugin.synthesize(context);
});
fs.writeFileSync(path.join(outdir, CdkGraphArtifacts.GRAPH_METADATA), JSON.stringify({
version: CdkGraph.VERSION,
artifacts: context.artifacts,
}, null, 2), { encoding: "utf-8" });
// store context for reporting
this._context = context;
}
/**
* Asynchronous report generation. This operation enables running expensive and non-synchronous
* report generation by plugins post synthesis.
*
* If a given plugin requires performing asynchronous operations or is general expensive, it should
* utilize `report` rather than `synthesize`.
*/
async report() {
if (this._context == null) {
// TODO: support deserializing pdk-graph to generate store/context
console.warn(chalk.yellowBright("[CdkGraph] In the near future, reports will be runnable outside of cdk synth"));
throw new Error("CdkGraph report called outside of cdk synth process");
}
for (const plugin of this.plugins) {
plugin.report && (await plugin.report(this._context));
}
}
}
exports.CdkGraph = CdkGraph;
_b = JSII_RTTI_SYMBOL_1;
CdkGraph[_b] = { fqn: "@aws/pdk.cdk_graph.CdkGraph", version: "0.26.14" };
/** Fixed CdkGraph construct id */
CdkGraph.ID = core_1.GRAPH_ID;
/** Current CdkGraph semantic version */
CdkGraph.VERSION = "0.0.0"; // TODO: make dynamic from package
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cdk-graph.js","sourceRoot":"","sources":["cdk-graph.ts"],"names":[],"mappings":"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","sourcesContent":["/*! Copyright [Amazon.com](http://amazon.com/), Inc. or its affiliates. All Rights Reserved.\nSPDX-License-Identifier: Apache-2.0 */\nimport * as path from \"path\";\nimport { addMetric } from \"@aws/monorepo\";\nimport { Aspects, ISynthesisSession } from \"aws-cdk-lib\";\nimport chalk = require(\"chalk\"); // eslint-disable-line @typescript-eslint/no-require-imports\nimport { Construct, IConstruct } from \"constructs\";\nimport * as fs from \"fs-extra\";\nimport { addCustomSynthesis } from \"./cdk-internals\";\nimport { CdkGraphConfig, resolveConfig, resolveOutdir } from \"./config\";\nimport { computeGraph, Graph, Version, GRAPH_ID } from \"./core\";\n\nconst GRAPH_ARTIFACT_ID = \"GRAPH\";\n\n/** CdkGraph core artifacts */\nexport enum CdkGraphArtifacts {\n  GRAPH_METADATA = \"graph-metadata.json\",\n  GRAPH = \"graph.json\",\n}\n\n/**\n * CdkGraph artifact definition\n * @struct\n */\nexport interface CdkGraphArtifact {\n  /** The unique type of the artifact */\n  readonly id: string;\n  /** Filename of the artifact */\n  readonly filename: string;\n  /** Full path where artifact is stored */\n  readonly filepath: string;\n  /** Description of artifact */\n  readonly description?: string;\n  /** The source of the artifact (such as plugin, or core system, etc) */\n  readonly source: string;\n}\n\n/** Dictionary of graph artifacts by id */\nexport type CdkGraphArtifactDict = { [id: string]: CdkGraphArtifact };\n\n/** CdkGraph context */\nexport class CdkGraphContext {\n  /** @internal */\n  readonly _artifacts: CdkGraphArtifactDict = {};\n\n  constructor(\n    public readonly store: Graph.Store,\n    public readonly outdir: string\n  ) {}\n\n  /**\n   * Get CdkGraph artifact by id\n   * @throws Error is artifact does not exist\n   */\n  getArtifact(id: string): CdkGraphArtifact {\n    const artifact = this._artifacts[id];\n    if (artifact) {\n      return artifact;\n    }\n    throw new Error(`Graph artifact ${id} does not exist`);\n  }\n\n  /** Get CdkGraph core `graph.json` artifact */\n  get graphJson(): CdkGraphArtifact {\n    return this.getArtifact(GRAPH_ARTIFACT_ID);\n  }\n\n  /** Indicates if context has an artifact with *filename* defined */\n  hasArtifactFile(filename: string): boolean {\n    return !!Object.values(this._artifacts).find(\n      (artifact) => artifact.filename === filename\n    );\n  }\n\n  /** Get record of all graph artifacts keyed by artifact id */\n  get artifacts(): CdkGraphArtifactDict {\n    return this._artifacts;\n  }\n\n  /**\n   * Logs an artifact entry. In general this should not be called directly, as `writeArtifact` should be utilized\n   * to perform writing and logging artifacts. However some plugins utilize other tools that generate the artifacts,\n   * in which case the plugin would call this method to log the entry.\n   * @param source The source of the artifact, such as the name of plugin\n   * @param id Unique id of the artifact\n   * @param filepath Full path where the artifact is stored\n   * @param description Description of the artifact\n   * @returns\n   * @throws Error is artifact id or filename already exists\n   */\n  logArtifact(\n    source: CdkGraph | ICdkGraphPlugin,\n    id: string,\n    filepath: string,\n    description?: string\n  ): CdkGraphArtifact {\n    if (id in this._artifacts) {\n      throw new Error(`Graph artifact ${id} already defined`);\n    }\n    if (this.hasArtifactFile(filepath)) {\n      throw new Error(`Graph artifact \"${filepath}\" already defined`);\n    }\n\n    const filename = path.relative(this.outdir, filepath);\n\n    if (!(source instanceof CdkGraph)) {\n      if (Object.keys(CdkGraphArtifacts).includes(id)) {\n        throw new Error(`Graph artifact id ${id} is reserved`);\n      }\n      if (Object.values(CdkGraphArtifacts).includes(filename as any)) {\n        throw new Error(`Graph artifact file ${filename} is reserved`);\n      }\n    }\n\n    const artifact: CdkGraphArtifact = {\n      id,\n      filepath,\n      description,\n      filename,\n      source:\n        source instanceof CdkGraph\n          ? `${CdkGraph.ID}`\n          : `plugin:${source.id}@${source.version}`,\n    };\n\n    this._artifacts[id] = artifact;\n\n    console.info(\n      chalk.cyanBright(\n        `[CdkGraph] Artifact ${id} written to \\x1B]8;;file://${artifact.filepath}\\x1B\\\\${artifact.filename}\\x1B]8;;\\x1B\\\\ (${artifact.source})`\n      )\n    );\n\n    return artifact;\n  }\n\n  /**\n   * Writes artifact data to outdir and logs the entry.\n   * @param source The source of the artifact, such as the name of plugin\n   * @param id Unique id of the artifact\n   * @param filename Relative name of the file\n   * @param description Description of the artifact\n   * @returns\n   */\n  writeArtifact(\n    source: CdkGraph | ICdkGraphPlugin,\n    id: string,\n    filename: string,\n    data: string,\n    description?: string\n  ): CdkGraphArtifact {\n    const filepath = path.join(this.outdir, filename);\n    const artifact = this.logArtifact(source, id, filepath, description);\n\n    fs.ensureDirSync(path.dirname(filepath));\n\n    fs.writeFileSync(filepath, data, { encoding: \"utf-8\" });\n\n    return artifact;\n  }\n}\n\n/** Callback signature for graph `Plugin.bind` operation */\nexport interface IGraphPluginBindCallback {\n  (graph: CdkGraph): void;\n}\n\n/** Callback signature for graph `Plugin.inspect` operation */\nexport interface IGraphVisitorCallback {\n  (construct: IConstruct): void;\n}\n\n/** Callback signature for graph `Plugin.synthesize` operation */\nexport interface IGraphSynthesizeCallback {\n  (context: CdkGraphContext): void;\n}\n\n/** Callback signature for graph `Plugin.report` operation */\nexport interface IGraphReportCallback {\n  (context: CdkGraphContext): Promise<void>;\n}\n\n/** CdkGraph **Plugin** interface */\nexport interface ICdkGraphPlugin {\n  /** Unique identifier for this plugin */\n  readonly id: string;\n  /** Plugin version */\n  readonly version: Version;\n  /** List of plugins this plugin depends on, including optional semver version (eg: [\"foo\", \"bar@1.2\"]) */\n  readonly dependencies?: string[];\n\n  /**\n   * Binds the plugin to the CdkGraph instance. Enables plugins to receive base configs.\n   */\n  bind: IGraphPluginBindCallback;\n\n  /**\n   * Node visitor callback for construct tree traversal. This follows IAspect.visit pattern, but the order\n   * of visitor traversal in managed by the CdkGraph.\n   */\n  inspect?: IGraphVisitorCallback;\n  /**\n   * Called during CDK synthesize to generate synchronous artifacts based on the in-memory graph passed\n   * to the plugin. This is called in fifo order of plugins.\n   */\n  synthesize?: IGraphSynthesizeCallback;\n  /**\n   * Generate asynchronous reports based on the graph. This is not automatically called when synthesizing CDK.\n   * Developer must explicitly add `await graphInstance.report()` to the CDK bin or invoke this outside\n   * of the CDK synth. In either case, the plugin receives the in-memory graph interface when invoked, as the\n   * CdkGraph will deserialize the graph prior to invoking the plugin report.\n   */\n  report?: IGraphReportCallback;\n}\n\n/**\n *  {@link CdkGraph} props\n * @struct\n * */\nexport interface ICdkGraphProps {\n  /** List of plugins to extends the graph. Plugins are invoked at each phases in fifo order. */\n  readonly plugins?: ICdkGraphPlugin[];\n}\n\n/**\n * CdkGraph construct is the cdk-graph framework controller that is responsible for\n * computing the graph, storing serialized graph, and instrumenting plugins per the\n * plugin contract.\n */\nexport class CdkGraph extends Construct {\n  /** Fixed CdkGraph construct id */\n  static readonly ID = GRAPH_ID;\n  /** Current CdkGraph semantic version */\n  static readonly VERSION = \"0.0.0\"; // TODO: make dynamic from package\n\n  /** List of plugins registered with this instance */\n  readonly plugins: ICdkGraphPlugin[];\n\n  /** @internal */\n  private _context?: CdkGraphContext;\n\n  /** Config */\n  readonly config: CdkGraphConfig;\n\n  /**\n   * Get the context for the graph instance.\n   *\n   * This will be `undefined` before construct synthesis has initiated.\n   */\n  get graphContext(): CdkGraphContext | undefined {\n    return this._context;\n  }\n\n  constructor(public readonly root: Construct, props: ICdkGraphProps = {}) {\n    super(root, CdkGraph.ID);\n\n    addMetric(root, \"cdk-graph\");\n\n    this.config = resolveConfig();\n\n    this.plugins = props.plugins || [];\n    // TODO: verify plugin deps via semver\n\n    // bind all plugins to this instance of the graph\n    this.plugins.forEach((plugin) => {\n      addMetric(root, `cdk-graph-plugin-${plugin.id}` as any);\n      plugin.bind(this);\n    });\n\n    // Apply Aspect for each plugin that supports \"inspect\" phase\n    this.plugins.forEach((plugin) => {\n      if (plugin.inspect) {\n        Aspects.of(this.root).add({\n          visit: plugin.inspect,\n        });\n      }\n    });\n\n    addCustomSynthesis(this, {\n      onSynthesize: (session) => {\n        this._synthesize(session);\n      },\n    });\n  }\n\n  /** @internal */\n  protected _synthesize(session: ISynthesisSession): void {\n    const store = computeGraph(this.root);\n    const outdir = resolveOutdir(session.outdir, this.config.outdir);\n    const context = new CdkGraphContext(store, outdir);\n\n    context.writeArtifact(\n      this,\n      GRAPH_ARTIFACT_ID,\n      CdkGraphArtifacts.GRAPH,\n      JSON.stringify(context.store.serialize(), null, 2),\n      \"Serialized graph\"\n    );\n\n    this.plugins.forEach((plugin) => {\n      plugin.synthesize && plugin.synthesize(context);\n    });\n\n    fs.writeFileSync(\n      path.join(outdir, CdkGraphArtifacts.GRAPH_METADATA),\n      JSON.stringify(\n        {\n          version: CdkGraph.VERSION,\n          artifacts: context.artifacts,\n        },\n        null,\n        2\n      ),\n      { encoding: \"utf-8\" }\n    );\n\n    // store context for reporting\n    this._context = context;\n  }\n\n  /**\n   * Asynchronous report generation. This operation enables running expensive and non-synchronous\n   * report generation by plugins post synthesis.\n   *\n   * If a given plugin requires performing asynchronous operations or is general expensive, it should\n   * utilize `report` rather than `synthesize`.\n   */\n  public async report() {\n    if (this._context == null) {\n      // TODO: support deserializing pdk-graph to generate store/context\n      console.warn(\n        chalk.yellowBright(\n          \"[CdkGraph] In the near future, reports will be runnable outside of cdk synth\"\n        )\n      );\n      throw new Error(\"CdkGraph report called outside of cdk synth process\");\n    }\n    for (const plugin of this.plugins) {\n      plugin.report && (await plugin.report(this._context));\n    }\n  }\n}\n"]}