cdk-pretty-diff
Version:
Formatting tool for CDK Diff output. Inspired by Terraform prettyplan (https://github.com/chrislewisdev/prettyplan)
147 lines • 19.7 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", { value: true });
exports.renderCustomDiffToHtmlString = exports.renderCustomDiffToHtmlNodeString = void 0;
const diff_1 = require("diff");
const diff2html = require("diff2html");
const pretty_diff_template_html_1 = require("./pretty-diff-template.html");
const prettify = (valueIn) => {
// fallback to empty string (eg. JSON.stringify of undefined is undefined)
const value = (typeof valueIn === "string" ? valueIn : JSON.stringify(valueIn, null, 2)) || '';
if (value === "<computed>") {
return `<em><computed></em>`;
}
if (value.startsWith("${") && value.endsWith("}")) {
return `<em>${value}</em>`;
}
if (value.indexOf("\\n") >= 0 || value.indexOf('\\"') >= 0) {
const sanitisedValue = value
.replace(new RegExp("\\\\n", "g"), "\n")
.replace(new RegExp('\\\\"', "g"), '"');
return `<pre>${prettifyJson(sanitisedValue)}</pre>`;
}
return value;
};
const prettifyJson = (maybeJson) => {
try {
return JSON.stringify(JSON.parse(maybeJson), null, 2);
}
catch (e) {
return maybeJson;
}
};
const components = {
badge: (label) => `
<span class="badge">${label}</span>
`,
id: (id) => `
<span class="id">
<span class="id-segment type">${id.resourceType}</span>
<span class="id-segment name">${id.resourceLabel}</span>
</span>
`,
warning: (warning) => `
<li>
${components.badge("warning")}
${components.id(warning.id)}
<span>${warning.detail}</span>
</li>
`,
changeCount: (count) => `
<span class="change-count">
${`${count} change${count > 1 ? "s" : ""}`}
</span>
`,
changeNoDiff: ({ action, to, label }) => `
<tr>
<td class="property">
${label}
${`<br /><span class="forces-new-resource">(${action})</span>`}
</td>
<td class="new-value">${prettify(to)}</td>
</tr>
`,
changeDiff: ({ from, to, label }) => `
<div>
${diff2html.html((0, diff_1.createTwoFilesPatch)(label, label, prettify(from), prettify(to)), {
outputFormat: 'line-by-line',
drawFileList: false,
matching: 'words',
matchWordsThreshold: 0.25,
matchingMaxComparisons: 200,
})}
</div>
`,
changes: (changes) => {
const diffChanges = changes.filter(({ from }) => !!from);
const noDiffChanges = changes.filter(({ from }) => !from);
return `
<div class="changes-breakdown">
<div class="no-diff-changes-breakdown">
${noDiffChanges.length ? (`
<table>
${noDiffChanges.map(components.changeNoDiff).join("")}
</table>
`) : ''}
</div>
${diffChanges.map(components.changeDiff).join("")}
</table>
`;
},
action: ({ cdkDiffRaw, nicerDiff, label }) => `
<li class="${(nicerDiff === null || nicerDiff === void 0 ? void 0 : nicerDiff.resourceAction.toLocaleLowerCase()) || "create"}">
<div class="summary" onclick="accordion(this)">
${components.badge((nicerDiff === null || nicerDiff === void 0 ? void 0 : nicerDiff.resourceAction) || "")}
${components.id(nicerDiff || { resourceType: "", resourceLabel: label })}
</div>
<div class="changes collapsed">
${(nicerDiff === null || nicerDiff === void 0 ? void 0 : nicerDiff.changes.length)
? components.changes(nicerDiff.changes)
: ""}
${components.rawDiff(cdkDiffRaw, "CDK Diff Output", {
collapsed: !(nicerDiff === null || nicerDiff === void 0 ? void 0 : nicerDiff.changes.length),
showButton: !!(nicerDiff === null || nicerDiff === void 0 ? void 0 : nicerDiff.changes.length),
})}
</div>
</li>
`,
modal: (content) => `
<div class="modal-pane" onclick="closeModal()"></div>
<div class="modal-content">
<div class="modal-close"><button class="text-button" onclick="closeModal()">close</button></div>
${content}
</div>
`,
rawDiff: (raw, toggleCaption, opts) => `
<div class="raw-diff">
${typeof (opts === null || opts === void 0 ? void 0 : opts.showButton) === "boolean" && (opts === null || opts === void 0 ? void 0 : opts.showButton) === false
? ""
: `<button onclick="accordion(this)">${toggleCaption}</button>`}
<div class="changes ${(opts === null || opts === void 0 ? void 0 : opts.collapsed) ? "collapsed" : ""}">
<pre>${raw}</pre>
</div>
</div>
`,
stackDiff: ({ stackName, raw, diff }) => `
<div class="stack">
<h2>${stackName}</h2>
${components.rawDiff(raw, "Orig CDK Diff", { collapsed: true })}
${!(diff === null || diff === void 0 ? void 0 : diff.length) ? `<div>No changes</div>` : ""}
<ul class="actions">
${diff === null || diff === void 0 ? void 0 : diff.filter(({ nicerDiff }) => !nicerDiff ||
!["parameters"].includes(nicerDiff === null || nicerDiff === void 0 ? void 0 : nicerDiff.cdkDiffCategory)).map(components.action).join("\n")}
</ul>
</div>
`,
};
const renderCustomDiffToHtmlNodeString = (diffs) => diffs.map(components.stackDiff).join(' ');
exports.renderCustomDiffToHtmlNodeString = renderCustomDiffToHtmlNodeString;
const renderCustomDiffToHtmlString = (diffs, title) => {
let html = pretty_diff_template_html_1.default;
html = html
.replace(`<h1>prettyplan</h1>`, `<h1>${title}</h1>`)
.replace(`<title>prettyplan</title>`, `<title>${title}</title>`);
html = html.replace(`<div id="stacks"></div>`, `<div id="stacks">${(0, exports.renderCustomDiffToHtmlNodeString)(diffs)}</div>`);
return html;
};
exports.renderCustomDiffToHtmlString = renderCustomDiffToHtmlString;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"render.js","sourceRoot":"","sources":["../src/render.ts"],"names":[],"mappings":"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","sourcesContent":["import { createTwoFilesPatch } from 'diff';\nimport * as diff2html from 'diff2html';\n\nimport { NicerDiff, NicerDiffChange, NicerStackDiff } from \"./types\";\nimport htmlTemplate from './pretty-diff-template.html';\n\nconst prettify = (valueIn: any): string => {\n  // fallback to empty string (eg. JSON.stringify of undefined is undefined)\n  const value =\n    (typeof valueIn === \"string\" ? valueIn : JSON.stringify(valueIn, null, 2)) || '';\n\n  if (value === \"<computed>\") {\n    return `<em>&lt;computed&gt;</em>`;\n  }\n\n  if (value.startsWith(\"${\") && value.endsWith(\"}\")) {\n    return `<em>${value}</em>`;\n  }\n\n  if (value.indexOf(\"\\\\n\") >= 0 || value.indexOf('\\\\\"') >= 0) {\n    const sanitisedValue = value\n      .replace(new RegExp(\"\\\\\\\\n\", \"g\"), \"\\n\")\n      .replace(new RegExp('\\\\\\\\\"', \"g\"), '\"');\n\n    return `<pre>${prettifyJson(sanitisedValue)}</pre>`;\n  }\n\n  return value;\n};\n\nconst prettifyJson = (maybeJson: string): string => {\n  try {\n    return JSON.stringify(JSON.parse(maybeJson), null, 2);\n  } catch (e) {\n    return maybeJson;\n  }\n};\n\nconst components = {\n  badge: (label: string): string => `\n    <span class=\"badge\">${label}</span>\n  `,\n\n  id: (id: any): string => `\n    <span class=\"id\">\n      <span class=\"id-segment type\">${id.resourceType}</span>\n      <span class=\"id-segment name\">${id.resourceLabel}</span>\n    </span>\n  `,\n\n  warning: (warning: any): string => `\n    <li>\n      ${components.badge(\"warning\")}\n      ${components.id(warning.id)}\n      <span>${warning.detail}</span>\n    </li>\n  `,\n\n  changeCount: (count: number): string => `\n    <span class=\"change-count\">\n      ${`${count} change${count > 1 ? \"s\" : \"\"}`}\n    </span>\n  `,\n\n  changeNoDiff: ({ action, to, label }: NicerDiffChange): string => `\n    <tr>\n      <td class=\"property\">\n        ${label}\n        ${`<br /><span class=\"forces-new-resource\">(${action})</span>`}\n      </td>\n      <td class=\"new-value\">${prettify(to)}</td>\n    </tr>\n  `,\n\n  changeDiff: ({ from, to, label }: NicerDiffChange): string => `\n    <div>\n      ${diff2html.html(\n        createTwoFilesPatch(label, label, prettify(from), prettify(to)),\n        {\n          outputFormat: 'line-by-line',\n          drawFileList: false,\n          matching: 'words',\n          matchWordsThreshold: 0.25,\n          matchingMaxComparisons: 200,\n        }\n      )}\n    </div>\n  `,\n\n  changes: (changes: NicerDiffChange[]) => {\n    const diffChanges = changes.filter(({ from }) => !!from);\n    const noDiffChanges = changes.filter(({ from }) => !from);\n\n    return `\n      <div class=\"changes-breakdown\">\n        <div class=\"no-diff-changes-breakdown\">\n          ${noDiffChanges.length ? (`\n            <table>\n              ${noDiffChanges.map(components.changeNoDiff).join(\"\")}\n            </table>\n          `) : ''}\n        </div>\n        ${diffChanges.map(components.changeDiff).join(\"\")}\n      </table>\n    `\n  },\n\n  action: ({ cdkDiffRaw, nicerDiff, label }: NicerDiff): string => `\n    <li class=\"${nicerDiff?.resourceAction.toLocaleLowerCase() || \"create\"}\">\n      <div class=\"summary\" onclick=\"accordion(this)\">\n        ${components.badge(nicerDiff?.resourceAction || \"\")}\n        ${components.id(\n          nicerDiff || { resourceType: \"\", resourceLabel: label }\n        )}\n      </div>\n      <div class=\"changes collapsed\">\n        ${\n          nicerDiff?.changes.length\n            ? components.changes(nicerDiff.changes)\n            : \"\"\n        }\n        ${components.rawDiff(cdkDiffRaw, \"CDK Diff Output\", {\n          collapsed: !nicerDiff?.changes.length,\n          showButton: !!nicerDiff?.changes.length,\n        })}\n      </div>\n    </li>\n  `,\n\n  modal: (content: string): string => `\n      <div class=\"modal-pane\" onclick=\"closeModal()\"></div>\n      <div class=\"modal-content\">\n          <div class=\"modal-close\"><button class=\"text-button\" onclick=\"closeModal()\">close</button></div>\n          ${content}\n      </div>\n  `,\n\n  rawDiff: (\n    raw: string,\n    toggleCaption: string,\n    opts?: { collapsed: boolean; showButton?: boolean }\n  ): string => `\n    <div class=\"raw-diff\">\n      ${\n        typeof opts?.showButton === \"boolean\" && opts?.showButton === false\n          ? \"\"\n          : `<button onclick=\"accordion(this)\">${toggleCaption}</button>`\n      }\n      <div class=\"changes ${opts?.collapsed ? \"collapsed\" : \"\"}\">\n          <pre>${raw}</pre>\n      </div>\n    </div>\n  `,\n\n  stackDiff: ({ stackName, raw, diff }: NicerStackDiff): string => `\n    <div class=\"stack\">\n      <h2>${stackName}</h2>\n      ${components.rawDiff(raw, \"Orig CDK Diff\", { collapsed: true })}\n      ${!diff?.length ? `<div>No changes</div>` : \"\"}\n      <ul class=\"actions\">\n        ${diff\n          ?.filter(\n            ({ nicerDiff }) =>\n              !nicerDiff ||\n              ![\"parameters\"].includes(nicerDiff?.cdkDiffCategory)\n          )\n          .map(components.action)\n          .join(\"\\n\")}\n      </ul>\n    </div>\n  `,\n};\n\nexport const renderCustomDiffToHtmlNodeString = (diffs: NicerStackDiff[]): string =>\n  diffs.map(components.stackDiff).join(' ');\n\nexport const renderCustomDiffToHtmlString = (\n  diffs: NicerStackDiff[],\n  title: string\n): string => {\n  let html = htmlTemplate;\n  html = html\n    .replace(`<h1>prettyplan</h1>`, `<h1>${title}</h1>`)\n    .replace(`<title>prettyplan</title>`, `<title>${title}</title>`);\n\n  html = html.replace(\n    `<div id=\"stacks\"></div>`,\n    `<div id=\"stacks\">${renderCustomDiffToHtmlNodeString(diffs)}</div>`\n  );\n\n  return html;\n};\n"]}