UNPKG

@aws/pdk

Version:

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

268 lines 35.1 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.buildDiagram = buildDiagram; /*! Copyright [Amazon.com](http://amazon.com/), Inc. or its affiliates. All Rights Reserved. SPDX-License-Identifier: Apache-2.0 */ const aws_arch_1 = require("../../../aws-arch"); const cdk_graph_1 = require("../../../cdk-graph"); const uniqBy = require("lodash.uniqby"); // eslint-disable-line @typescript-eslint/no-require-imports const Diagram = require("./entities"); const entities_1 = require("./entities"); const theme_1 = require("./theme"); const debug_1 = require("../debug"); /** * EdgeResolver class resolves edges within a diagram for rendering * @internal */ class EdgeResolver { constructor() { /** @internal */ this._edges = new Map(); } /** Adds diagram edge to the resolver to be resolved */ trackEdge(edge) { let fromId; let toId; if (edge instanceof Diagram.Edge) { fromId = edge.graphEdge.source.uuid; toId = edge.graphEdge.target.uuid; } else { fromId = edge.from.graphNode.uuid; toId = edge.to.graphNode.uuid; } const key = fromId < toId ? `${fromId}:${toId}` : `${toId}:${fromId}`; const edgeSet = this._edges.get(key) || []; this._edges.set(key, edgeSet.concat([edge])); } /** Resolve all edges based on diagram options */ resolveEdges(options) { const compact = options.preset === cdk_graph_1.FilterPreset.COMPACT; const resolvedEdges = []; for (let edges of this._edges.values()) { if (compact) { edges = edges.filter((edge) => !edge.isVerbose); } if (edges.length === 0) { continue; } edges.sort((a, b) => { const _a = getEdgeRank(a); const _b = getEdgeRank(b); if (_a === _b) return 0; if (_a < _b) return -1; return 1; }); edges = uniqBy(edges, getEdgeRank); // only return highest ranked edge unless verbose if (compact) { resolvedEdges.push(edges[0]); continue; } for (const _edge1 of edges) { for (const _edge2 of edges) { if (_edge1 === _edge2) continue; const _id1 = _edge1.attributes.get("id"); const _id2 = _edge2.attributes.get("id"); let _sameHead1 = _edge1.attributes.get("samehead"); _edge1.attributes.set("samehead", !_sameHead1 ? _id2 : `${_sameHead1},${_id2}`); let _sameTail1 = _edge1.attributes.get("sametail"); _edge1.attributes.set("sametail", !_sameTail1 ? _id2 : `${_sameTail1},${_id2}`); let _sameHead2 = _edge2.attributes.get("samehead"); _edge2.attributes.set("samehead", !_sameHead2 ? _id1 : `${_sameHead2},${_id1}`); let _sameTail2 = _edge2.attributes.get("sametail"); _edge2.attributes.set("sametail", !_sameTail2 ? _id1 : `${_sameTail2},${_id1}`); } } resolvedEdges.push(...edges); } return resolvedEdges; } } /** * Build a {@link Diagram.Diagram Diagram} for a given {@link Graph.Store} based on {@link DiagramOptions Options} * @internal */ function buildDiagram(store, options) { const { title, nodePositions } = options; const edgeResolve = new EdgeResolver(); theme_1.GraphTheme.init(options.theme); const entities = new Map(); const diagram = new Diagram.Diagram(title, aws_arch_1.AwsArchitecture.assetDirectory); function visit(gNode, parent) { if (gNode.isDestroyed) return; let entity; switch (gNode.nodeType) { case cdk_graph_1.NodeTypeEnum.RESOURCE: { entity = new Diagram.ResourceNode(gNode); break; } case cdk_graph_1.NodeTypeEnum.CFN_RESOURCE: { entity = new Diagram.CfnResourceNode(gNode); break; } case cdk_graph_1.NodeTypeEnum.NESTED_STACK: { entity = new Diagram.NestedStackCluster(gNode); break; } case cdk_graph_1.NodeTypeEnum.STACK: { if (theme_1.GraphTheme.instance.rendering.stack && new RegExp(theme_1.GraphTheme.instance.rendering.stack).test(gNode.id) === false) { // Ignore non-matching root stacks return; } entity = new Diagram.StackCluster(gNode); break; } default: { if (gNode.isLeaf) { if (gNode.hasFlag(cdk_graph_1.FlagEnum.CUSTOM_RESOURCE)) { entity = new Diagram.CustomResourceNode(gNode); } else { entity = new Diagram.Node(gNode); } } else { entity = new Diagram.Cluster(gNode); gNode.addFlag(cdk_graph_1.FlagEnum.CLUSTER); } break; } } if (nodePositions && entity instanceof Diagram.Node) { if (entity.graphNode.id in nodePositions) { entity.position = nodePositions[entity.graphNode.id]; } } if (entity instanceof entities_1.ImageNode && entity.image) { diagram.trackImage(entity.image); } if (parent instanceof Diagram.Container && parent.linkChildren) { edgeResolve.trackEdge(new entities_1.ChildLink(parent, entity)); } if (gNode.isLeaf) { entities.set(gNode.uuid, entity); parent.addNode(entity); } else { if (entity instanceof Diagram.Node) { entity = asContainer(entity); } parent.addSubgraph(entity); entities.set(gNode.uuid, entity); gNode.children.forEach((child) => visit(child, entity)); } } if (store.stages.length > 1) { const stageRendering = theme_1.GraphTheme.instance.rendering.stage || "last"; let stages; switch (stageRendering) { case "all": { stages = store.stages; break; } case "first": { stages = store.stages.slice(0, 1); break; } case "last": { stages = store.stages.slice(-1); break; } default: { stages = store.stages.filter((stage) => stage.id.match(stageRendering)); } } // traverse all stages stages.forEach((gStage) => { const dStage = new Diagram.StageCluster(gStage); diagram.addSubgraph(dStage); entities.set(gStage.uuid, dStage); gStage.children.forEach((child) => visit(child, dStage)); }); } else if (store.rootStacks.length) { // traverse all root stack store.rootStacks.forEach((gStack) => { const dStack = new Diagram.StackCluster(gStack); diagram.addSubgraph(dStack); entities.set(gStack.uuid, dStack); gStack.children.forEach((child) => visit(child, dStack)); }); } else { store.root.children.forEach((gChild) => { if (gChild.isGraphContainer) { gChild.children.forEach((_gChild) => { visit(_gChild, diagram); }); } else { visit(gChild, diagram); } }); } // apply all edges store.edges.forEach((gEdge) => { if (gEdge.isDestroyed) return; const dSource = entities.get(gEdge.source.uuid); const dTarget = entities.get(gEdge.target.uuid); if (!dSource || !dTarget) { debug_1.IS_DEBUG && console.warn("Diagram.Edge unresolved source and/or target:", `source(${gEdge.source} => ${dSource})`, `target(${gEdge.target} => ${dTarget})`); return; } let edge = undefined; switch (gEdge.edgeType) { case cdk_graph_1.EdgeTypeEnum.REFERENCE: { edge = new Diagram.ReferenceLink(gEdge, dSource, dTarget); break; } case cdk_graph_1.EdgeTypeEnum.DEPENDENCY: { edge = new Diagram.DependencyLink(gEdge, dSource, dTarget); break; } } if (edge) { entities.set(gEdge.uuid, edge); edgeResolve.trackEdge(edge); } }); edgeResolve.resolveEdges(options).forEach((edge) => { diagram.addEdge(edge); }); return diagram; } /** * Wrap a {@link Diagram.Node} with {@link Diagram.Container} to support adding child {@link Diagram.Node}s * @internal */ function asContainer(node) { const container = new Diagram.Container(node.graphNode); container.addNode(node); return container; } /** * Get the rank score of an {@link Diagram.BaseEdge Edge} used to sort and prioritize edges * @internal */ function getEdgeRank(edge) { if (edge instanceof Diagram.ChildLink) { return 0; } if (edge instanceof Diagram.ReferenceLink) { return 1; } if (edge instanceof Diagram.DependencyLink) { return 2; } return 3; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"diagram.js","sourceRoot":"","sources":["diagram.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 { AwsArchitecture } from \"@aws/aws-arch\";\nimport {\n  EdgeTypeEnum,\n  FilterPreset,\n  FlagEnum,\n  Graph,\n  NodeTypeEnum,\n  UUID,\n} from \"@aws/cdk-graph\";\nimport uniqBy = require(\"lodash.uniqby\"); // eslint-disable-line @typescript-eslint/no-require-imports\nimport * as Dot from \"ts-graphviz\";\nimport * as Diagram from \"./entities\";\nimport { ChildLink, ImageNode } from \"./entities\";\nimport { GraphTheme, GraphThemeConfigProp } from \"./theme\";\nimport { INodePosition } from \"../../config\";\nimport { IS_DEBUG } from \"../debug\";\n\n/**\n * Union of entities support by diagram\n * @internal\n */\ntype TDiagramEntity = Diagram.Subgraph | Diagram.Node | Diagram.Edge;\n\n/** Options for diagrams */\nexport interface DiagramOptions {\n  readonly title: string;\n  readonly preset?: FilterPreset;\n  readonly theme?: GraphThemeConfigProp;\n  readonly nodePositions?: { [cdkConstructId: string]: INodePosition };\n}\n\n/**\n * EdgeResolver class resolves edges within a diagram for rendering\n * @internal\n */\nclass EdgeResolver {\n  /** @internal */\n  private readonly _edges: Map<string, Diagram.BaseEdge[]> = new Map();\n\n  /** Adds diagram edge to the resolver to be resolved */\n  trackEdge(edge: Diagram.BaseEdge): void {\n    let fromId: string;\n    let toId: string;\n    if (edge instanceof Diagram.Edge) {\n      fromId = edge.graphEdge.source.uuid;\n      toId = edge.graphEdge.target.uuid;\n    } else {\n      fromId = edge.from.graphNode.uuid;\n      toId = edge.to.graphNode.uuid;\n    }\n    const key = fromId < toId ? `${fromId}:${toId}` : `${toId}:${fromId}`;\n    const edgeSet = this._edges.get(key) || [];\n    this._edges.set(key, edgeSet.concat([edge]));\n  }\n\n  /** Resolve all edges based on diagram options */\n  resolveEdges(options: DiagramOptions): Dot.Edge[] {\n    const compact = options.preset === FilterPreset.COMPACT;\n\n    const resolvedEdges: Dot.Edge[] = [];\n    for (let edges of this._edges.values()) {\n      if (compact) {\n        edges = edges.filter((edge) => !edge.isVerbose);\n      }\n\n      if (edges.length === 0) {\n        continue;\n      }\n\n      edges.sort((a, b) => {\n        const _a = getEdgeRank(a);\n        const _b = getEdgeRank(b);\n        if (_a === _b) return 0;\n        if (_a < _b) return -1;\n        return 1;\n      });\n\n      edges = uniqBy(edges, getEdgeRank);\n\n      // only return highest ranked edge unless verbose\n      if (compact) {\n        resolvedEdges.push(edges[0]);\n        continue;\n      }\n\n      for (const _edge1 of edges) {\n        for (const _edge2 of edges) {\n          if (_edge1 === _edge2) continue;\n\n          const _id1 = _edge1.attributes.get(\"id\")!;\n          const _id2 = _edge2.attributes.get(\"id\")!;\n\n          let _sameHead1 = _edge1.attributes.get(\"samehead\") as string;\n          _edge1.attributes.set(\n            \"samehead\",\n            !_sameHead1 ? _id2 : `${_sameHead1},${_id2}`\n          );\n          let _sameTail1 = _edge1.attributes.get(\"sametail\") as string;\n          _edge1.attributes.set(\n            \"sametail\",\n            !_sameTail1 ? _id2 : `${_sameTail1},${_id2}`\n          );\n\n          let _sameHead2 = _edge2.attributes.get(\"samehead\") as string;\n          _edge2.attributes.set(\n            \"samehead\",\n            !_sameHead2 ? _id1 : `${_sameHead2},${_id1}`\n          );\n          let _sameTail2 = _edge2.attributes.get(\"sametail\") as string;\n          _edge2.attributes.set(\n            \"sametail\",\n            !_sameTail2 ? _id1 : `${_sameTail2},${_id1}`\n          );\n        }\n      }\n\n      resolvedEdges.push(...edges);\n    }\n\n    return resolvedEdges;\n  }\n}\n\n/**\n * Build a {@link Diagram.Diagram Diagram} for a given {@link Graph.Store} based on {@link DiagramOptions Options}\n * @internal\n */\nexport function buildDiagram(\n  store: Graph.Store,\n  options: DiagramOptions\n): Diagram.Diagram {\n  const { title, nodePositions } = options;\n\n  const edgeResolve = new EdgeResolver();\n\n  GraphTheme.init(options.theme);\n\n  const entities: Map<UUID, TDiagramEntity> = new Map();\n  const diagram = new Diagram.Diagram(title, AwsArchitecture.assetDirectory);\n\n  function visit(\n    gNode: Graph.Node,\n    parent: Diagram.Subgraph | Diagram.Diagram\n  ) {\n    if (gNode.isDestroyed) return;\n\n    let entity: Diagram.Container | Diagram.Node;\n\n    switch (gNode.nodeType) {\n      case NodeTypeEnum.RESOURCE: {\n        entity = new Diagram.ResourceNode(gNode as Graph.ResourceNode);\n        break;\n      }\n      case NodeTypeEnum.CFN_RESOURCE: {\n        entity = new Diagram.CfnResourceNode(gNode as Graph.CfnResourceNode);\n        break;\n      }\n      case NodeTypeEnum.NESTED_STACK: {\n        entity = new Diagram.NestedStackCluster(gNode as Graph.NestedStackNode);\n        break;\n      }\n      case NodeTypeEnum.STACK: {\n        if (\n          GraphTheme.instance.rendering.stack &&\n          new RegExp(GraphTheme.instance.rendering.stack).test(gNode.id) ===\n            false\n        ) {\n          // Ignore non-matching root stacks\n          return;\n        }\n        entity = new Diagram.StackCluster(gNode as Graph.StackNode);\n        break;\n      }\n      default: {\n        if (gNode.isLeaf) {\n          if (gNode.hasFlag(FlagEnum.CUSTOM_RESOURCE)) {\n            entity = new Diagram.CustomResourceNode(gNode);\n          } else {\n            entity = new Diagram.Node(gNode);\n          }\n        } else {\n          entity = new Diagram.Cluster(gNode);\n          gNode.addFlag(FlagEnum.CLUSTER);\n        }\n        break;\n      }\n    }\n\n    if (nodePositions && entity instanceof Diagram.Node) {\n      if (entity.graphNode.id in nodePositions) {\n        entity.position = nodePositions[entity.graphNode.id]!;\n      }\n    }\n\n    if (entity instanceof ImageNode && entity.image) {\n      diagram.trackImage(entity.image);\n    }\n\n    if (parent instanceof Diagram.Container && parent.linkChildren) {\n      edgeResolve.trackEdge(new ChildLink(parent, entity));\n    }\n\n    if (gNode.isLeaf) {\n      entities.set(gNode.uuid, entity);\n      parent.addNode(entity as Diagram.Node);\n    } else {\n      if (entity instanceof Diagram.Node) {\n        entity = asContainer(entity);\n      }\n\n      parent.addSubgraph(entity as Diagram.Container);\n\n      entities.set(gNode.uuid, entity);\n\n      gNode.children.forEach((child) =>\n        visit(child, entity as Diagram.Container)\n      );\n    }\n  }\n\n  if (store.stages.length > 1) {\n    const stageRendering = GraphTheme.instance.rendering.stage || \"last\";\n    let stages: Graph.StageNode[];\n    switch (stageRendering) {\n      case \"all\": {\n        stages = store.stages;\n        break;\n      }\n      case \"first\": {\n        stages = store.stages.slice(0, 1);\n        break;\n      }\n      case \"last\": {\n        stages = store.stages.slice(-1);\n        break;\n      }\n      default: {\n        stages = store.stages.filter((stage) => stage.id.match(stageRendering));\n      }\n    }\n    // traverse all stages\n    stages.forEach((gStage) => {\n      const dStage = new Diagram.StageCluster(gStage);\n      diagram.addSubgraph(dStage);\n      entities.set(gStage.uuid, dStage);\n      gStage.children.forEach((child) => visit(child, dStage));\n    });\n  } else if (store.rootStacks.length) {\n    // traverse all root stack\n    store.rootStacks.forEach((gStack) => {\n      const dStack = new Diagram.StackCluster(gStack);\n      diagram.addSubgraph(dStack);\n      entities.set(gStack.uuid, dStack);\n      gStack.children.forEach((child) => visit(child, dStack));\n    });\n  } else {\n    store.root.children.forEach((gChild) => {\n      if (gChild.isGraphContainer) {\n        gChild.children.forEach((_gChild) => {\n          visit(_gChild, diagram);\n        });\n      } else {\n        visit(gChild, diagram);\n      }\n    });\n  }\n\n  // apply all edges\n  store.edges.forEach((gEdge) => {\n    if (gEdge.isDestroyed) return;\n\n    const dSource = entities.get(gEdge.source.uuid) as\n      | Diagram.Container\n      | Diagram.Node;\n    const dTarget = entities.get(gEdge.target.uuid) as\n      | Diagram.Container\n      | Diagram.Node;\n\n    if (!dSource || !dTarget) {\n      IS_DEBUG &&\n        console.warn(\n          \"Diagram.Edge unresolved source and/or target:\",\n          `source(${gEdge.source} => ${dSource})`,\n          `target(${gEdge.target} => ${dTarget})`\n        );\n      return;\n    }\n\n    let edge: Diagram.Edge | undefined = undefined;\n\n    switch (gEdge.edgeType) {\n      case EdgeTypeEnum.REFERENCE: {\n        edge = new Diagram.ReferenceLink(gEdge, dSource, dTarget);\n        break;\n      }\n      case EdgeTypeEnum.DEPENDENCY: {\n        edge = new Diagram.DependencyLink(gEdge, dSource, dTarget);\n        break;\n      }\n    }\n\n    if (edge) {\n      entities.set(gEdge.uuid, edge);\n      edgeResolve.trackEdge(edge);\n    }\n  });\n\n  edgeResolve.resolveEdges(options).forEach((edge) => {\n    diagram.addEdge(edge);\n  });\n\n  return diagram;\n}\n\n/**\n * Wrap a {@link Diagram.Node} with {@link Diagram.Container} to support adding child {@link Diagram.Node}s\n * @internal\n */\nfunction asContainer(node: Diagram.Node): Diagram.Container {\n  const container = new Diagram.Container(node.graphNode);\n  container.addNode(node);\n  return container;\n}\n\n/**\n * Get the rank score of an {@link Diagram.BaseEdge Edge} used to sort and prioritize edges\n * @internal\n */\nfunction getEdgeRank(edge: Diagram.BaseEdge): number {\n  if (edge instanceof Diagram.ChildLink) {\n    return 0;\n  }\n  if (edge instanceof Diagram.ReferenceLink) {\n    return 1;\n  }\n  if (edge instanceof Diagram.DependencyLink) {\n    return 2;\n  }\n  return 3;\n}\n"]}