UNPKG

cdk-pretty-diff

Version:

Formatting tool for CDK Diff output. Inspired by Terraform prettyplan (https://github.com/chrislewisdev/prettyplan)

166 lines 23.3 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.transformDiff = void 0; const cfnDiff = require("@aws-cdk/cloudformation-diff"); const through2 = require("through2"); const util_1 = require("./util"); const types_1 = require("./types"); const cdk_reverse_engineered_1 = require("./cdk-reverse-engineered"); // unable to emulate the --no-colors option, (tried passing no-colors option to cdk Configuration class to no avail) // this is workaround to remove the colors tty elements const fixRemoveColors = (input) => JSON.parse(JSON.stringify(input).replace(/\\u001b\[[^m]+m/g, '')); const buildRaw = async (diff) => { const strm = through2(); cfnDiff.formatDifferences(strm, diff.rawDiff, diff.logicalToPathMap); strm.end(); return fixRemoveColors(await (0, util_1.streamToString)(strm)); }; const buildChangeAction = (oldValue, newValue) => { if (oldValue !== undefined && newValue !== undefined) { return "UPDATE"; } else if (oldValue !== undefined) { return "REMOVAL"; } else { return "ADDITION"; } }; const transformIamChanges = async (diff) => { if (!diff.rawDiff.iamChanges.hasChanges) { return []; } const result = []; if (diff.rawDiff.iamChanges.statements.hasChanges) { const statementsSummarized = diff.rawDiff.iamChanges.summarizeStatements(); result.push({ label: "IAM Statement Changes", cdkDiffRaw: fixRemoveColors(cfnDiff.formatTable((0, cdk_reverse_engineered_1.deepSubstituteBracedLogicalIds)(diff.logicalToPathMap)(statementsSummarized), undefined)), }); } if (diff.rawDiff.iamChanges.managedPolicies.hasChanges) { const managedPoliciesSummarized = diff.rawDiff.iamChanges.summarizeManagedPolicies(); result.push({ label: "IAM Policy Changes", cdkDiffRaw: fixRemoveColors(cfnDiff.formatTable((0, cdk_reverse_engineered_1.deepSubstituteBracedLogicalIds)(diff.logicalToPathMap)(managedPoliciesSummarized), undefined)), }); } return result; }; const transformSecurityGroupChanges = async (diff) => { if (!diff.rawDiff.securityGroupChanges.hasChanges) { return []; } const summarized = diff.rawDiff.securityGroupChanges.summarize(); return [ { label: "Security Group Changes", cdkDiffRaw: fixRemoveColors(cfnDiff.formatTable((0, cdk_reverse_engineered_1.deepSubstituteBracedLogicalIds)(diff.logicalToPathMap)(summarized), undefined)), }, ]; }; const processIndividualDiff = (result, cdkDiffCategory) => (id, rdiff) => { var _a, _b, _c, _d; if (rdiff.isDifferent) { const resourceType = (0, types_1.guardResourceDiff)(rdiff) ? (rdiff.isRemoval ? (_a = rdiff.oldValue) === null || _a === void 0 ? void 0 : _a.Type : (_b = rdiff.newValue) === null || _b === void 0 ? void 0 : _b.Type) || cdkDiffCategory : (((_c = rdiff.oldValue) === null || _c === void 0 ? void 0 : _c.Type) || ((_d = rdiff.newValue) === null || _d === void 0 ? void 0 : _d.Type) || cdkDiffCategory); const changes = []; if ((0, types_1.guardResourceDiff)(rdiff) && rdiff.isUpdate) { rdiff.forEachDifference((_, label, values) => { changes.push({ label, action: buildChangeAction(values.oldValue, values.newValue), from: values.oldValue, to: values.newValue, }); }); } result.push({ label: cdkDiffCategory, cdkDiffRaw: JSON.stringify({ id, diff: rdiff }, null, 2), nicerDiff: { resourceType, changes, cdkDiffCategory, resourceAction: rdiff.isAddition ? "ADDITION" : rdiff.isRemoval ? "REMOVAL" : "UPDATE", resourceLabel: id, }, }); } }; const transformDiffForResourceTypes = async (diff) => { const result = []; for (const d of Object.entries(diff.rawDiff).filter(([k]) => !["iamChanges", "securityGroupChanges"].includes(k))) { const validatedDiff = (0, types_1.diffValidator)(d); if ('diffCollection' in validatedDiff) { const { diffCollectionKey, diffCollection } = validatedDiff; if (diffCollection.differenceCount > 0) { diffCollection.forEachDifference(processIndividualDiff(result, diffCollectionKey)); } } else if ('diffKey' in validatedDiff) { const { diffKey, diff } = validatedDiff; if (diff.isDifferent) { result.push({ label: diffKey, cdkDiffRaw: JSON.stringify({ id: diffKey, diff }, null, 2), }); } } } return result; }; const transformDescriptionChanges = (diff) => { var _a, _b, _c, _d, _e, _f, _g; if ((_a = diff.rawDiff.description) === null || _a === void 0 ? void 0 : _a.isDifferent) { return { label: 'Description', cdkDiffRaw: JSON.stringify({ description: diff.rawDiff.description }, null, 2), nicerDiff: { resourceType: 'Description', changes: [{ label: 'Description', action: buildChangeAction((_b = diff.rawDiff.description) === null || _b === void 0 ? void 0 : _b.oldValue, (_c = diff.rawDiff.description) === null || _c === void 0 ? void 0 : _c.newValue), from: (_d = diff.rawDiff.description) === null || _d === void 0 ? void 0 : _d.oldValue, to: (_e = diff.rawDiff.description) === null || _e === void 0 ? void 0 : _e.newValue }], cdkDiffCategory: 'description', resourceAction: ((_f = diff.rawDiff.description) === null || _f === void 0 ? void 0 : _f.isAddition) ? "ADDITION" : ((_g = diff.rawDiff.description) === null || _g === void 0 ? void 0 : _g.isRemoval) ? "REMOVAL" : "UPDATE", resourceLabel: 'Description', }, }; } return null; }; const transformDiff = async (diff) => { if (diff.rawDiff.isEmpty) { return { stackName: diff.stackName, raw: "There were no differences", diff: [], }; } const descriptionDiff = transformDescriptionChanges(diff); return { stackName: diff.stackName, raw: await buildRaw(diff), diff: [ ...(await transformIamChanges(diff)), ...(await transformSecurityGroupChanges(diff)), ...(await transformDiffForResourceTypes(diff)), ...(descriptionDiff ? [descriptionDiff] : []), ], }; }; exports.transformDiff = transformDiff; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transform.js","sourceRoot":"","sources":["../src/transform.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cfnDiff from \"@aws-cdk/cloudformation-diff\";\nimport * as through2 from \"through2\";\n\nimport { streamToString } from \"./util\";\nimport {\n  CdkDiffCategory,\n  diffValidator,\n  guardResourceDiff,\n  NicerDiff,\n  NicerDiffChange,\n  NicerStackDiff,\n  StackRawDiff,\n} from \"./types\";\nimport { deepSubstituteBracedLogicalIds } from \"./cdk-reverse-engineered\";\n\n// unable to emulate the --no-colors option, (tried passing no-colors option to cdk Configuration class to no avail)\n// this is workaround to remove the colors tty elements\nconst fixRemoveColors = (input: string): string => JSON.parse(JSON.stringify(input).replace(/\\\\u001b\\[[^m]+m/g, ''))\n\nconst buildRaw = async (diff: StackRawDiff): Promise<string> => {\n  const strm = through2();\n  cfnDiff.formatDifferences(strm, diff.rawDiff, diff.logicalToPathMap);\n  strm.end();\n  return fixRemoveColors(await streamToString(strm));\n};\n\nconst buildChangeAction = (oldValue: any, newValue: any) => {\n  if (oldValue !== undefined && newValue !== undefined) {\n    return \"UPDATE\";\n  } else if (oldValue !== undefined) {\n    return \"REMOVAL\";\n  } else {\n    return \"ADDITION\";\n  }\n};\n\nconst transformIamChanges = async (\n  diff: StackRawDiff\n): Promise<NicerDiff[]> => {\n  if (!diff.rawDiff.iamChanges.hasChanges) {\n    return [];\n  }\n\n  const result: NicerDiff[] = [];\n  if (diff.rawDiff.iamChanges.statements.hasChanges) {\n    const statementsSummarized = diff.rawDiff.iamChanges.summarizeStatements();\n    result.push({\n      label: \"IAM Statement Changes\",\n      cdkDiffRaw: fixRemoveColors(\n        cfnDiff.formatTable(\n          deepSubstituteBracedLogicalIds(diff.logicalToPathMap)(\n            statementsSummarized\n          ),\n          undefined\n        )\n      ),\n    });\n  }\n\n  if (diff.rawDiff.iamChanges.managedPolicies.hasChanges) {\n    const managedPoliciesSummarized =\n      diff.rawDiff.iamChanges.summarizeManagedPolicies();\n    result.push({\n      label: \"IAM Policy Changes\",\n      cdkDiffRaw: fixRemoveColors(\n        cfnDiff.formatTable(\n          deepSubstituteBracedLogicalIds(diff.logicalToPathMap)(\n            managedPoliciesSummarized\n          ),\n          undefined\n        )\n      ),\n    });\n  }\n\n  return result;\n};\n\nconst transformSecurityGroupChanges = async (\n  diff: StackRawDiff\n): Promise<NicerDiff[]> => {\n  if (!diff.rawDiff.securityGroupChanges.hasChanges) {\n    return [];\n  }\n\n  const summarized = diff.rawDiff.securityGroupChanges.summarize();\n\n  return [\n    {\n      label: \"Security Group Changes\",\n      cdkDiffRaw: fixRemoveColors(\n        cfnDiff.formatTable(\n          deepSubstituteBracedLogicalIds(diff.logicalToPathMap)(summarized),\n          undefined\n        )\n      ),\n    },\n  ];\n};\n\nconst processIndividualDiff =\n  (result: NicerDiff[], cdkDiffCategory: CdkDiffCategory) =>\n    (id: string, rdiff: cfnDiff.Difference<any>) => {\n      if (rdiff.isDifferent) {\n        const resourceType: string = guardResourceDiff(rdiff)\n          ? (rdiff.isRemoval ? rdiff.oldValue?.Type : rdiff.newValue?.Type) ||\n          cdkDiffCategory\n          : (rdiff.oldValue?.Type || rdiff.newValue?.Type || cdkDiffCategory);\n        const changes: NicerDiffChange[] = [];\n        if (guardResourceDiff(rdiff) && rdiff.isUpdate) {\n          rdiff.forEachDifference((_, label, values) => {\n            changes.push({\n              label,\n              action: buildChangeAction(values.oldValue, values.newValue),\n              from: values.oldValue,\n              to: values.newValue,\n            });\n          });\n        }\n\n        result.push({\n          label: cdkDiffCategory,\n          cdkDiffRaw: JSON.stringify({ id, diff: rdiff }, null, 2),\n          nicerDiff: {\n            resourceType,\n            changes,\n            cdkDiffCategory,\n            resourceAction: rdiff.isAddition\n              ? \"ADDITION\"\n              : rdiff.isRemoval\n                ? \"REMOVAL\"\n                : \"UPDATE\",\n            resourceLabel: id,\n          },\n        });\n      }\n    };\n\nconst transformDiffForResourceTypes = async (diff: StackRawDiff): Promise<NicerDiff[]> => {\n  const result: NicerDiff[] = [];\n  for (const d of Object.entries(diff.rawDiff).filter(([k]) => ![\"iamChanges\", \"securityGroupChanges\"].includes(k))) {\n    const validatedDiff = diffValidator(d);\n    if ('diffCollection' in validatedDiff) {\n      const { diffCollectionKey, diffCollection } = validatedDiff;\n      if (diffCollection.differenceCount > 0) {\n        diffCollection.forEachDifference(\n          processIndividualDiff(result, diffCollectionKey)\n        );\n      }\n    } else if ('diffKey' in validatedDiff) {\n      const { diffKey, diff } = validatedDiff;\n      if (diff.isDifferent) {\n        result.push({\n          label: diffKey,\n          cdkDiffRaw: JSON.stringify({ id: diffKey, diff }, null, 2),\n        });\n      }\n    }\n  }\n\n  return result;\n};\n\nconst transformDescriptionChanges = (diff: StackRawDiff): NicerDiff | null => {\n  if (diff.rawDiff.description?.isDifferent) {\n    return {\n      label: 'Description',\n      cdkDiffRaw: JSON.stringify({ description: diff.rawDiff.description }, null, 2),\n      nicerDiff: {\n        resourceType: 'Description',\n        changes: [{\n          label: 'Description',\n          action: buildChangeAction(diff.rawDiff.description?.oldValue, diff.rawDiff.description?.newValue),\n          from: diff.rawDiff.description?.oldValue,\n          to: diff.rawDiff.description?.newValue\n        }],\n        cdkDiffCategory: 'description',\n        resourceAction: diff.rawDiff.description?.isAddition\n          ? \"ADDITION\"\n          : diff.rawDiff.description?.isRemoval\n            ? \"REMOVAL\"\n            : \"UPDATE\",\n        resourceLabel: 'Description',\n      },\n    };\n  }\n  return null;\n};\n\nexport const transformDiff = async (\n  diff: StackRawDiff\n): Promise<NicerStackDiff> => {\n  if (diff.rawDiff.isEmpty) {\n    return {\n      stackName: diff.stackName,\n      raw: \"There were no differences\",\n      diff: [],\n    };\n  }\n\n  const descriptionDiff = transformDescriptionChanges(diff);\n  return {\n    stackName: diff.stackName,\n    raw: await buildRaw(diff),\n    diff: [\n      ...(await transformIamChanges(diff)),\n      ...(await transformSecurityGroupChanges(diff)),\n      ...(await transformDiffForResourceTypes(diff)),\n      ...(descriptionDiff ? [descriptionDiff] : []),\n    ],\n  };\n};\n\n"]}