UNPKG

@aws/pdk

Version:

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

272 lines 32.1 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.addGraphFontCssStyles = addGraphFontCssStyles; exports.resolveSvgAwsArchAssetImagesInline = resolveSvgAwsArchAssetImagesInline; exports.encodeDataUrl = encodeDataUrl; exports.encodeHtmlDataUrl = encodeHtmlDataUrl; exports.encodeSvgFileDataUrl = encodeSvgFileDataUrl; exports.reconcileViewBox = reconcileViewBox; exports.getSvgRootContainer = getSvgRootContainer; exports.parseSvgViewBox = parseSvgViewBox; exports.stringifySvgViewBox = stringifySvgViewBox; exports.parseSvgTransformScale = parseSvgTransformScale; exports.unescapeSvgTextValues = unescapeSvgTextValues; /*! 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 fs = require("fs-extra"); const he = require("he"); // eslint-disable-line @typescript-eslint/no-require-imports const svgson = require("svgson"); const traverse = require("traverse"); // eslint-disable-line @typescript-eslint/no-require-imports const fonts_1 = require("../fonts"); const XLINK_HREF = "xlink:href"; const DATAURL_SVG_BASE64 = "data:image/svg+xml;base64,"; /** * Add graph font css styles to svg * @internal */ function addGraphFontCssStyles(svg) { svg.children.unshift({ name: "style", type: "element", value: "", attributes: {}, children: [ { name: "", type: "text", attributes: {}, children: [], value: fonts_1.FONT_CSS_CLASSES, }, ], }); } /** * Resolve SVG image paths to inline base64 **Data URLs**. * @internal */ async function resolveSvgAwsArchAssetImagesInline(svgString) { let svg = await svgson.parse(svgString); const imageDefs = new Map(); svg = traverse(svg).forEach(function (x) { if (typeof x === "object" && x?.type === "element") { const node = x; if (node.name !== "image") { return; } const xlinkHref = node.attributes[XLINK_HREF]; const isAssetPath = xlinkHref && xlinkHref.length && !(xlinkHref.startsWith("http") || (xlinkHref.startsWith("/") && !xlinkHref.startsWith(aws_arch_1.AwsArchitecture.assetDirectory))); if (isAssetPath) { const { width, height, value, [XLINK_HREF]: assetPath, ...attributes } = node.attributes; const id = aws_arch_1.AwsArchitecture.parseAssetPath(assetPath).assetKey; if (!imageDefs.has(id)) { imageDefs.set(id, { type: "element", name: "image", value, children: [], attributes: { id, width, height, [XLINK_HREF]: aws_arch_1.AwsArchitecture.resolveAssetPath(assetPath), }, }); } const useDefNode = { type: "element", name: "use", value, children: [], attributes: { ...attributes, [XLINK_HREF]: `#${id}`, }, }; this.update(useDefNode, true); } } }); for (const [, imageDef] of imageDefs.entries()) { const href = imageDef.attributes[XLINK_HREF]; imageDef.attributes[XLINK_HREF] = await encodeSvgFileDataUrl(href); } svg.children.unshift({ type: "element", name: "defs", value: "", attributes: {}, children: Array.from(imageDefs.values()), }); addGraphFontCssStyles(svg); reconcileViewBox(svg); return svgson.stringify(svg); } /** * Encode buffer as base64 encoded **Data URL** * @internal */ function encodeDataUrl(buffer, prefix) { return prefix + buffer.toString("base64"); } /** * Encode string as html and base64 encoded **Data URL** * @internal */ function encodeHtmlDataUrl(data, prefix) { return encodeDataUrl(Buffer.from(unescape(encodeURIComponent(data)), "utf-8"), prefix); } /** * Encode SVG file as base64 encoded **Data URL** * @internal */ async function encodeSvgFileDataUrl(svgFile) { const svgXml = await fs.readFile(svgFile, { encoding: "utf-8" }); return encodeHtmlDataUrl(svgXml, DATAURL_SVG_BASE64); } /** * ==== VIEWBOX ==== */ const CSS_TRANSFORM_SCALE = /scale\((?<scale>[^)]+)\)/i; /** * Reconcile svg viewBox attribute based on root container ([g](https://developer.mozilla.org/en-US/docs/Web/SVG/Element/g)) scale. * * This will modify the viewbox and root container in place. * @param svg Svg to reconcile * @throws Error if svg does not define `viewBox` attribute * @internal */ function reconcileViewBox(svg) { const viewBox = parseSvgViewBox(svg); if (viewBox == null || !viewBox.width || !viewBox.height) { throw new Error("Svg `viewBox` undefined or does not define dimensions"); } // drop width/height to ensure only reconciled viewbox is used delete svg.attributes.width; delete svg.attributes.height; const container = getSvgRootContainer(svg); if (container == null) { return; } // let [x, y, width, height] = viewBox.split(" ").map((v) => parseFloat(v)) as [number, number, number, number]; let scale = parseSvgTransformScale(container); if (scale == null) { return; } let scaledViewBox = { ...viewBox, width: viewBox.width * (scale[0] || 1), height: viewBox.height * (scale[1] || scale[0] || 1), }; // Max allowed by sharp: https://github.com/lovell/sharp/blob/2c465282699432299c478ba00ab825e07d9bdab0/src/pipeline.cc#L288 const MAX_SVG = 10000; // 32767 is max width & height in sharp, but capping at 10k for perf and downscale huge diagrams if (scaledViewBox.width && scaledViewBox.width > MAX_SVG) { const downscale = MAX_SVG / scaledViewBox.width; scaledViewBox = { ...scaledViewBox, width: scaledViewBox.width * downscale, height: scaledViewBox.height ? scaledViewBox.height * downscale : undefined, }; if (scale[0]) scale[0] *= downscale; if (scale[1]) scale[1] *= downscale; } if (scaledViewBox.height && scaledViewBox.height > MAX_SVG) { const downscale = MAX_SVG / scaledViewBox.height; scaledViewBox = { ...scaledViewBox, height: scaledViewBox.height * downscale, width: scaledViewBox.width ? scaledViewBox.width * downscale : undefined, }; if (scale[0]) scale[0] *= downscale; if (scale[1]) scale[1] *= downscale; } svg.attributes.viewBox = stringifySvgViewBox(scaledViewBox); if (scale[0] || scale[1]) { const _scale = scale.filter((v) => v != null && !isNaN(v)).join(" "); container.attributes.transform = container.attributes.transform.replace(CSS_TRANSFORM_SCALE, `scale(${_scale})`); } } /** * Get SVG root container - the first "g" element * @internal */ function getSvgRootContainer(svg) { return svg.children.find((child) => child.name === "g"); } /** Parse SVG viewBox */ function parseSvgViewBox(svgOrViewBox) { let viewBox; if (typeof svgOrViewBox === "object") { viewBox = svgOrViewBox.attributes.viewBox; } else { viewBox = svgOrViewBox; } if (viewBox == null || viewBox === "") { return undefined; } let [x, y, width, height] = viewBox.split(" ").map((v) => parseFloat(v)); return { x: x || 0, y: y || 0, width, height, }; } /** Stringify SVG viewBox attribute */ function stringifySvgViewBox(viewBox) { return [viewBox.x, viewBox.y, viewBox.width, viewBox.height] .filter((v) => v != null) .join(" "); } /** Parse SVG transform attribute scale property */ function parseSvgTransformScale(elementOrTransform) { let transform; if (typeof elementOrTransform === "object") { transform = elementOrTransform.attributes.transform; } else { transform = elementOrTransform; } if (transform == null || transform === "") { return undefined; } const transformScale = transform?.match(CSS_TRANSFORM_SCALE)?.groups?.scale; if (transformScale == null) { return undefined; } return transformScale.split(" ").map((v) => parseFloat(v)); } /** * Unescape SVG **text** values. * * *dot-wasm* escapes svg text ("-" -> "&#45;") and [svgson](https://github.com/elrumordelaluz/svgson/blob/e7234b645b4e344f525d4d2fde2d3f2911d3a75a/src/stringify.js#L20) * escapes strings that contain *&...* by wrapping in `<![CDATA[...]]>` tag which causes the * resulting text value in SVG and PNG files to show raw escaped value (`&#45;`). * * We expect to have the original text values rendered rather than escaped version rendered. * * @example `Diagram (hyphenated&#45;value)` => `Diagram (hyphenated-value)` * @internal */ function unescapeSvgTextValues(svg) { traverse(svg).forEach(function (x) { if (this.key === "value" && typeof x === "string" && x !== "") { if (x.includes("&")) { this.update(he.decode(x), true); } } }); } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"svg.js","sourceRoot":"","sources":["svg.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 * as fs from \"fs-extra\";\nimport he = require(\"he\"); // eslint-disable-line @typescript-eslint/no-require-imports\nimport * as svgson from \"svgson\";\nimport traverse = require(\"traverse\"); // eslint-disable-line @typescript-eslint/no-require-imports\nimport { FONT_CSS_CLASSES } from \"../fonts\";\n\nconst XLINK_HREF = \"xlink:href\";\n\nconst DATAURL_SVG_BASE64 = \"data:image/svg+xml;base64,\";\n\n/**\n * Add graph font css styles to svg\n * @internal\n */\nexport function addGraphFontCssStyles(svg: svgson.INode): void {\n  svg.children.unshift({\n    name: \"style\",\n    type: \"element\",\n    value: \"\",\n    attributes: {},\n    children: [\n      {\n        name: \"\",\n        type: \"text\",\n        attributes: {},\n        children: [],\n        value: FONT_CSS_CLASSES,\n      },\n    ],\n  });\n}\n\n/**\n * Resolve SVG image paths to inline base64 **Data URLs**.\n * @internal\n */\nexport async function resolveSvgAwsArchAssetImagesInline(\n  svgString: string\n): Promise<string> {\n  let svg: svgson.INode = await svgson.parse(svgString);\n  const imageDefs = new Map<string, svgson.INode>();\n\n  svg = traverse(svg).forEach(function (this: traverse.TraverseContext, x) {\n    if (typeof x === \"object\" && x?.type === \"element\") {\n      const node = x as svgson.INode;\n      if (node.name !== \"image\") {\n        return;\n      }\n\n      const xlinkHref = node.attributes[XLINK_HREF];\n      const isAssetPath =\n        xlinkHref &&\n        xlinkHref.length &&\n        !(\n          xlinkHref.startsWith(\"http\") ||\n          (xlinkHref.startsWith(\"/\") &&\n            !xlinkHref.startsWith(AwsArchitecture.assetDirectory))\n        );\n\n      if (isAssetPath) {\n        const {\n          width,\n          height,\n          value,\n          [XLINK_HREF]: assetPath,\n          ...attributes\n        } = node.attributes;\n\n        const id = AwsArchitecture.parseAssetPath(assetPath!).assetKey;\n\n        if (!imageDefs.has(id)) {\n          imageDefs.set(id, {\n            type: \"element\",\n            name: \"image\",\n            value,\n            children: [],\n            attributes: {\n              id,\n              width,\n              height,\n              [XLINK_HREF]: AwsArchitecture.resolveAssetPath(assetPath),\n            },\n          });\n        }\n\n        const useDefNode: svgson.INode = {\n          type: \"element\",\n          name: \"use\",\n          value,\n          children: [],\n          attributes: {\n            ...attributes,\n            [XLINK_HREF]: `#${id}`,\n          },\n        };\n\n        this.update(useDefNode, true);\n      }\n    }\n  });\n\n  for (const [, imageDef] of imageDefs.entries()) {\n    const href = imageDef.attributes[XLINK_HREF];\n    imageDef.attributes[XLINK_HREF] = await encodeSvgFileDataUrl(href);\n  }\n\n  svg.children.unshift({\n    type: \"element\",\n    name: \"defs\",\n    value: \"\",\n    attributes: {},\n    children: Array.from(imageDefs.values()),\n  });\n\n  addGraphFontCssStyles(svg);\n  reconcileViewBox(svg);\n\n  return svgson.stringify(svg);\n}\n\n/**\n * Encode buffer as base64 encoded **Data URL**\n * @internal\n */\nexport function encodeDataUrl(buffer: Buffer, prefix: string): string {\n  return prefix + buffer.toString(\"base64\");\n}\n\n/**\n * Encode string as html and base64 encoded **Data URL**\n * @internal\n */\nexport function encodeHtmlDataUrl(data: string, prefix: string): string {\n  return encodeDataUrl(\n    Buffer.from(unescape(encodeURIComponent(data)), \"utf-8\"),\n    prefix\n  );\n}\n\n/**\n * Encode SVG file as base64 encoded **Data URL**\n * @internal\n */\nexport async function encodeSvgFileDataUrl(svgFile: string): Promise<string> {\n  const svgXml = await fs.readFile(svgFile, { encoding: \"utf-8\" });\n  return encodeHtmlDataUrl(svgXml, DATAURL_SVG_BASE64);\n}\n\n/**\n * ==== VIEWBOX ====\n */\n\nconst CSS_TRANSFORM_SCALE = /scale\\((?<scale>[^)]+)\\)/i;\n\n/**\n * SVG `viewBox` struct\n * @internal\n */\nexport interface SvgViewBox {\n  readonly x: number;\n  readonly y: number;\n  readonly width?: number;\n  readonly height?: number;\n}\n\n/**\n * Reconcile svg viewBox attribute based on root container ([g](https://developer.mozilla.org/en-US/docs/Web/SVG/Element/g)) scale.\n *\n * This will modify the viewbox and root container in place.\n * @param svg Svg to reconcile\n * @throws Error if svg does not define `viewBox` attribute\n * @internal\n */\nexport function reconcileViewBox(svg: svgson.INode): void {\n  const viewBox = parseSvgViewBox(svg);\n  if (viewBox == null || !viewBox.width || !viewBox.height) {\n    throw new Error(\"Svg `viewBox` undefined or does not define dimensions\");\n  }\n\n  // drop width/height to ensure only reconciled viewbox is used\n  delete svg.attributes.width;\n  delete svg.attributes.height;\n\n  const container = getSvgRootContainer(svg);\n  if (container == null) {\n    return;\n  }\n\n  // let [x, y, width, height] = viewBox.split(\" \").map((v) => parseFloat(v)) as [number, number, number, number];\n  let scale = parseSvgTransformScale(container);\n  if (scale == null) {\n    return;\n  }\n\n  let scaledViewBox: SvgViewBox = {\n    ...viewBox,\n    width: viewBox.width * (scale[0] || 1),\n    height: viewBox.height * (scale[1] || scale[0] || 1),\n  };\n\n  // Max allowed by sharp: https://github.com/lovell/sharp/blob/2c465282699432299c478ba00ab825e07d9bdab0/src/pipeline.cc#L288\n  const MAX_SVG = 10000; // 32767 is max width & height in sharp, but capping at 10k for perf and downscale huge diagrams\n\n  if (scaledViewBox.width && scaledViewBox.width > MAX_SVG) {\n    const downscale = MAX_SVG / scaledViewBox.width;\n    scaledViewBox = {\n      ...scaledViewBox,\n      width: scaledViewBox.width * downscale,\n      height: scaledViewBox.height\n        ? scaledViewBox.height * downscale\n        : undefined,\n    };\n    if (scale[0]) scale[0] *= downscale;\n    if (scale[1]) scale[1] *= downscale;\n  }\n\n  if (scaledViewBox.height && scaledViewBox.height > MAX_SVG) {\n    const downscale = MAX_SVG / scaledViewBox.height;\n    scaledViewBox = {\n      ...scaledViewBox,\n      height: scaledViewBox.height * downscale,\n      width: scaledViewBox.width ? scaledViewBox.width * downscale : undefined,\n    };\n    if (scale[0]) scale[0] *= downscale;\n    if (scale[1]) scale[1] *= downscale;\n  }\n\n  svg.attributes.viewBox = stringifySvgViewBox(scaledViewBox);\n  if (scale[0] || scale[1]) {\n    const _scale = scale.filter((v) => v != null && !isNaN(v)).join(\" \");\n    container.attributes.transform = container.attributes.transform.replace(\n      CSS_TRANSFORM_SCALE,\n      `scale(${_scale})`\n    );\n  }\n}\n\n/**\n * Get SVG root container - the first \"g\" element\n * @internal\n */\nexport function getSvgRootContainer(\n  svg: svgson.INode\n): svgson.INode | undefined {\n  return svg.children.find((child) => child.name === \"g\");\n}\n\n/** Parse SVG viewBox */\nexport function parseSvgViewBox(\n  svgOrViewBox: svgson.INode | string\n): SvgViewBox | undefined {\n  let viewBox: string;\n  if (typeof svgOrViewBox === \"object\") {\n    viewBox = svgOrViewBox.attributes.viewBox;\n  } else {\n    viewBox = svgOrViewBox;\n  }\n  if (viewBox == null || viewBox === \"\") {\n    return undefined;\n  }\n  let [x, y, width, height] = viewBox.split(\" \").map((v) => parseFloat(v)) as [\n    number,\n    number,\n    number,\n    number\n  ];\n\n  return {\n    x: x || 0,\n    y: y || 0,\n    width,\n    height,\n  };\n}\n\n/** Stringify SVG viewBox attribute */\nexport function stringifySvgViewBox(viewBox: SvgViewBox): string {\n  return [viewBox.x, viewBox.y, viewBox.width, viewBox.height]\n    .filter((v) => v != null)\n    .join(\" \");\n}\n\n/** Parse SVG transform attribute scale property */\nexport function parseSvgTransformScale(\n  elementOrTransform: svgson.INode | string\n): number[] | undefined {\n  let transform: string;\n  if (typeof elementOrTransform === \"object\") {\n    transform = elementOrTransform.attributes.transform;\n  } else {\n    transform = elementOrTransform;\n  }\n  if (transform == null || transform === \"\") {\n    return undefined;\n  }\n\n  const transformScale = transform?.match(CSS_TRANSFORM_SCALE)?.groups?.scale;\n  if (transformScale == null) {\n    return undefined;\n  }\n\n  return transformScale.split(\" \").map((v) => parseFloat(v));\n}\n\n/**\n * Unescape SVG **text** values.\n *\n * *dot-wasm* escapes svg text (\"-\" -> \"&#45;\") and [svgson](https://github.com/elrumordelaluz/svgson/blob/e7234b645b4e344f525d4d2fde2d3f2911d3a75a/src/stringify.js#L20)\n * escapes strings that contain *&...* by wrapping in `<![CDATA[...]]>` tag which causes the\n * resulting text value in SVG and PNG files to show raw escaped value (`&#45;`).\n *\n * We expect to have the original text values rendered rather than escaped version rendered.\n *\n * @example `Diagram (hyphenated&#45;value)` => `Diagram (hyphenated-value)`\n * @internal\n */\nexport function unescapeSvgTextValues(svg: svgson.INode): void {\n  traverse(svg).forEach(function (this: traverse.TraverseContext, x) {\n    if (this.key === \"value\" && typeof x === \"string\" && x !== \"\") {\n      if (x.includes(\"&\")) {\n        this.update(he.decode(x), true);\n      }\n    }\n  });\n}\n"]}