UNPKG

@aws-cdk/cloudformation-diff

Version:

Utilities to diff CDK stacks against CloudFormation templates

99 lines 16.8 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.SecurityGroupChanges = void 0; const chalk = require("chalk"); const security_group_rule_1 = require("./security-group-rule"); const diffable_1 = require("../diffable"); const render_intrinsics_1 = require("../render-intrinsics"); const util_1 = require("../util"); /** * Changes to IAM statements */ class SecurityGroupChanges { constructor(props) { this.ingress = new diffable_1.DiffableCollection(); this.egress = new diffable_1.DiffableCollection(); // Group rules for (const ingressProp of props.ingressRulePropertyChanges) { this.ingress.addOld(...this.readInlineRules(ingressProp.oldValue, ingressProp.resourceLogicalId)); this.ingress.addNew(...this.readInlineRules(ingressProp.newValue, ingressProp.resourceLogicalId)); } for (const egressProp of props.egressRulePropertyChanges) { this.egress.addOld(...this.readInlineRules(egressProp.oldValue, egressProp.resourceLogicalId)); this.egress.addNew(...this.readInlineRules(egressProp.newValue, egressProp.resourceLogicalId)); } // Rule resources for (const ingressRes of props.ingressRuleResourceChanges) { this.ingress.addOld(...this.readRuleResource(ingressRes.oldProperties)); this.ingress.addNew(...this.readRuleResource(ingressRes.newProperties)); } for (const egressRes of props.egressRuleResourceChanges) { this.egress.addOld(...this.readRuleResource(egressRes.oldProperties)); this.egress.addNew(...this.readRuleResource(egressRes.newProperties)); } this.ingress.calculateDiff(); this.egress.calculateDiff(); } get hasChanges() { return this.ingress.hasChanges || this.egress.hasChanges; } /** * Return a summary table of changes */ summarize() { const ret = []; const header = ['', 'Group', 'Dir', 'Protocol', 'Peer']; const inWord = 'In'; const outWord = 'Out'; // Render a single rule to the table (curried function so we can map it across rules easily--thank you JavaScript!) const renderRule = (plusMin, inOut) => (rule) => [ plusMin, rule.groupId, inOut, rule.describeProtocol(), rule.describePeer(), ].map(s => plusMin === '+' ? chalk.green(s) : chalk.red(s)); // First generate all lines, sort later ret.push(...this.ingress.additions.map(renderRule('+', inWord))); ret.push(...this.egress.additions.map(renderRule('+', outWord))); ret.push(...this.ingress.removals.map(renderRule('-', inWord))); ret.push(...this.egress.removals.map(renderRule('-', outWord))); // Sort by group name then ingress/egress (ingress first) ret.sort((0, util_1.makeComparator)((row) => [row[1], row[2].indexOf(inWord) > -1 ? 0 : 1])); ret.splice(0, 0, header); return ret; } toJson() { return (0, util_1.deepRemoveUndefined)({ ingressRuleAdditions: (0, util_1.dropIfEmpty)(this.ingress.additions.map(s => s.toJson())), ingressRuleRemovals: (0, util_1.dropIfEmpty)(this.ingress.removals.map(s => s.toJson())), egressRuleAdditions: (0, util_1.dropIfEmpty)(this.egress.additions.map(s => s.toJson())), egressRuleRemovals: (0, util_1.dropIfEmpty)(this.egress.removals.map(s => s.toJson())), }); } get rulesAdded() { return this.ingress.hasAdditions || this.egress.hasAdditions; } readInlineRules(rules, logicalId) { if (!rules || !Array.isArray(rules)) { return []; } // UnCloudFormation so the parser works in an easier domain const ref = '${' + logicalId + '.GroupId}'; return rules.flatMap((r) => { const rendered = (0, render_intrinsics_1.renderIntrinsics)(r); // SecurityGroupRule is not robust against unparsed objects return typeof rendered === 'object' ? [new security_group_rule_1.SecurityGroupRule(rendered, ref)] : []; }); } readRuleResource(resource) { if (!resource) { return []; } // UnCloudFormation so the parser works in an easier domain return [new security_group_rule_1.SecurityGroupRule((0, render_intrinsics_1.renderIntrinsics)(resource))]; } } exports.SecurityGroupChanges = SecurityGroupChanges; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"security-group-changes.js","sourceRoot":"","sources":["security-group-changes.ts"],"names":[],"mappings":"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","sourcesContent":["import * as chalk from 'chalk';\nimport type { RuleJson } from './security-group-rule';\nimport { SecurityGroupRule } from './security-group-rule';\nimport type { PropertyChange, ResourceChange } from '../diff/types';\nimport { DiffableCollection } from '../diffable';\nimport { renderIntrinsics } from '../render-intrinsics';\nimport { deepRemoveUndefined, dropIfEmpty, makeComparator } from '../util';\n\nexport interface SecurityGroupChangesProps {\n  ingressRulePropertyChanges: PropertyChange[];\n  ingressRuleResourceChanges: ResourceChange[];\n  egressRuleResourceChanges: ResourceChange[];\n  egressRulePropertyChanges: PropertyChange[];\n}\n\n/**\n * Changes to IAM statements\n */\nexport class SecurityGroupChanges {\n  public readonly ingress = new DiffableCollection<SecurityGroupRule>();\n  public readonly egress = new DiffableCollection<SecurityGroupRule>();\n\n  constructor(props: SecurityGroupChangesProps) {\n    // Group rules\n    for (const ingressProp of props.ingressRulePropertyChanges) {\n      this.ingress.addOld(...this.readInlineRules(ingressProp.oldValue, ingressProp.resourceLogicalId));\n      this.ingress.addNew(...this.readInlineRules(ingressProp.newValue, ingressProp.resourceLogicalId));\n    }\n    for (const egressProp of props.egressRulePropertyChanges) {\n      this.egress.addOld(...this.readInlineRules(egressProp.oldValue, egressProp.resourceLogicalId));\n      this.egress.addNew(...this.readInlineRules(egressProp.newValue, egressProp.resourceLogicalId));\n    }\n\n    // Rule resources\n    for (const ingressRes of props.ingressRuleResourceChanges) {\n      this.ingress.addOld(...this.readRuleResource(ingressRes.oldProperties));\n      this.ingress.addNew(...this.readRuleResource(ingressRes.newProperties));\n    }\n    for (const egressRes of props.egressRuleResourceChanges) {\n      this.egress.addOld(...this.readRuleResource(egressRes.oldProperties));\n      this.egress.addNew(...this.readRuleResource(egressRes.newProperties));\n    }\n\n    this.ingress.calculateDiff();\n    this.egress.calculateDiff();\n  }\n\n  public get hasChanges() {\n    return this.ingress.hasChanges || this.egress.hasChanges;\n  }\n\n  /**\n   * Return a summary table of changes\n   */\n  public summarize(): string[][] {\n    const ret: string[][] = [];\n\n    const header = ['', 'Group', 'Dir', 'Protocol', 'Peer'];\n\n    const inWord = 'In';\n    const outWord = 'Out';\n\n    // Render a single rule to the table (curried function so we can map it across rules easily--thank you JavaScript!)\n    const renderRule = (plusMin: string, inOut: string) => (rule: SecurityGroupRule) => [\n      plusMin,\n      rule.groupId,\n      inOut,\n      rule.describeProtocol(),\n      rule.describePeer(),\n    ].map(s => plusMin === '+' ? chalk.green(s) : chalk.red(s));\n\n    // First generate all lines, sort later\n    ret.push(...this.ingress.additions.map(renderRule('+', inWord)));\n    ret.push(...this.egress.additions.map(renderRule('+', outWord)));\n    ret.push(...this.ingress.removals.map(renderRule('-', inWord)));\n    ret.push(...this.egress.removals.map(renderRule('-', outWord)));\n\n    // Sort by group name then ingress/egress (ingress first)\n    ret.sort(makeComparator((row: string[]) => [row[1], row[2].indexOf(inWord) > -1 ? 0 : 1]));\n\n    ret.splice(0, 0, header);\n\n    return ret;\n  }\n\n  public toJson(): SecurityGroupChangesJson {\n    return deepRemoveUndefined({\n      ingressRuleAdditions: dropIfEmpty(this.ingress.additions.map(s => s.toJson())),\n      ingressRuleRemovals: dropIfEmpty(this.ingress.removals.map(s => s.toJson())),\n      egressRuleAdditions: dropIfEmpty(this.egress.additions.map(s => s.toJson())),\n      egressRuleRemovals: dropIfEmpty(this.egress.removals.map(s => s.toJson())),\n    });\n  }\n\n  public get rulesAdded(): boolean {\n    return this.ingress.hasAdditions\n        || this.egress.hasAdditions;\n  }\n\n  private readInlineRules(rules: any, logicalId: string): SecurityGroupRule[] {\n    if (!rules || !Array.isArray(rules)) {\n      return [];\n    }\n\n    // UnCloudFormation so the parser works in an easier domain\n\n    const ref = '${' + logicalId + '.GroupId}';\n    return rules.flatMap((r: any) => {\n      const rendered = renderIntrinsics(r);\n      // SecurityGroupRule is not robust against unparsed objects\n      return typeof rendered === 'object' ? [new SecurityGroupRule(rendered, ref)] : [];\n    });\n  }\n\n  private readRuleResource(resource: any): SecurityGroupRule[] {\n    if (!resource) {\n      return [];\n    }\n\n    // UnCloudFormation so the parser works in an easier domain\n\n    return [new SecurityGroupRule(renderIntrinsics(resource))];\n  }\n}\n\nexport interface SecurityGroupChangesJson {\n  ingressRuleAdditions?: RuleJson[];\n  ingressRuleRemovals?: RuleJson[];\n  egressRuleAdditions?: RuleJson[];\n  egressRuleRemovals?: RuleJson[];\n}\n"]}