UNPKG

@aws/pdk

Version:

All documentation is located at: https://aws.github.io/aws-pdk

195 lines 29.4 kB
"use strict"; 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"]}