@aws-cdk/cloudformation-diff
Version:
Utilities to diff CDK stacks against CloudFormation templates
96 lines • 12.9 kB
JavaScript
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
exports.SecurityGroupRule = void 0;
/**
* A single security group rule, either egress or ingress
*/
class SecurityGroupRule {
constructor(ruleObject, groupRef) {
this.ipProtocol = ruleObject.IpProtocol?.toString() || '*unknown*';
this.fromPort = ruleObject.FromPort;
this.toPort = ruleObject.ToPort;
this.groupId = ruleObject.GroupId || groupRef || '*unknown*'; // In case of an inline rule
this.peer =
findFirst(ruleObject, ['CidrIp', 'CidrIpv6'], (ip) => ({ kind: 'cidr-ip', ip }))
||
findFirst(ruleObject, ['DestinationSecurityGroupId', 'SourceSecurityGroupId'], (securityGroupId) => ({ kind: 'security-group', securityGroupId }))
||
findFirst(ruleObject, ['DestinationPrefixListId', 'SourcePrefixListId'], (prefixListId) => ({ kind: 'prefix-list', prefixListId }));
}
equal(other) {
return this.ipProtocol === other.ipProtocol
&& this.fromPort === other.fromPort
&& this.toPort === other.toPort
&& peerEqual(this.peer, other.peer);
}
describeProtocol() {
if (this.ipProtocol === '-1') {
return 'Everything';
}
const ipProtocol = this.ipProtocol.toUpperCase();
if (this.fromPort === -1) {
return `All ${ipProtocol}`;
}
if (this.fromPort === this.toPort) {
return `${ipProtocol} ${this.fromPort}`;
}
return `${ipProtocol} ${this.fromPort}-${this.toPort}`;
}
describePeer() {
if (this.peer) {
switch (this.peer.kind) {
case 'cidr-ip':
if (this.peer.ip === '0.0.0.0/0') {
return 'Everyone (IPv4)';
}
if (this.peer.ip === '::/0') {
return 'Everyone (IPv6)';
}
return `${this.peer.ip}`;
case 'prefix-list': return `${this.peer.prefixListId}`;
case 'security-group': return `${this.peer.securityGroupId}`;
}
}
return '?';
}
toJson() {
return {
groupId: this.groupId,
ipProtocol: this.ipProtocol,
fromPort: this.fromPort,
toPort: this.toPort,
peer: this.peer,
};
}
}
exports.SecurityGroupRule = SecurityGroupRule;
function peerEqual(a, b) {
if ((a === undefined) !== (b === undefined)) {
return false;
}
if (a === undefined) {
return true;
}
if (a.kind !== b.kind) {
return false;
}
switch (a.kind) {
case 'cidr-ip': return a.ip === b.ip;
case 'security-group': return a.securityGroupId === b.securityGroupId;
case 'prefix-list': return a.prefixListId === b.prefixListId;
}
}
function findFirst(obj, keys, fn) {
for (const key of keys) {
try {
if (key in obj) {
return fn(obj[key]);
}
}
catch (e) {
debugger;
}
}
return undefined;
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"security-group-rule.js","sourceRoot":"","sources":["security-group-rule.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * A single security group rule, either egress or ingress\n */\nexport class SecurityGroupRule {\n  /**\n   * Group ID of the group this rule applies to\n   */\n  public readonly groupId: string;\n\n  /**\n   * IP protocol this rule applies to\n   */\n  public readonly ipProtocol: string;\n\n  /**\n   * Start of port range this rule applies to, or ICMP type\n   */\n  public readonly fromPort?: number;\n\n  /**\n   * End of port range this rule applies to, or ICMP code\n   */\n  public readonly toPort?: number;\n\n  /**\n   * Peer of this rule\n   */\n  public readonly peer?: RulePeer;\n\n  constructor(ruleObject: any, groupRef?: string) {\n    this.ipProtocol = ruleObject.IpProtocol?.toString() || '*unknown*';\n    this.fromPort = ruleObject.FromPort;\n    this.toPort = ruleObject.ToPort;\n    this.groupId = ruleObject.GroupId || groupRef || '*unknown*'; // In case of an inline rule\n\n    this.peer =\n        findFirst(ruleObject,\n          ['CidrIp', 'CidrIpv6'],\n          (ip) => ({ kind: 'cidr-ip', ip } as CidrIpPeer))\n        ||\n        findFirst(ruleObject,\n          ['DestinationSecurityGroupId', 'SourceSecurityGroupId'],\n          (securityGroupId) => ({ kind: 'security-group', securityGroupId } as SecurityGroupPeer))\n        ||\n        findFirst(ruleObject,\n          ['DestinationPrefixListId', 'SourcePrefixListId'],\n          (prefixListId) => ({ kind: 'prefix-list', prefixListId } as PrefixListPeer));\n  }\n\n  public equal(other: SecurityGroupRule) {\n    return this.ipProtocol === other.ipProtocol\n        && this.fromPort === other.fromPort\n        && this.toPort === other.toPort\n        && peerEqual(this.peer, other.peer);\n  }\n\n  public describeProtocol() {\n    if (this.ipProtocol === '-1') {\n      return 'Everything';\n    }\n\n    const ipProtocol = this.ipProtocol.toUpperCase();\n\n    if (this.fromPort === -1) {\n      return `All ${ipProtocol}`;\n    }\n    if (this.fromPort === this.toPort) {\n      return `${ipProtocol} ${this.fromPort}`;\n    }\n    return `${ipProtocol} ${this.fromPort}-${this.toPort}`;\n  }\n\n  public describePeer() {\n    if (this.peer) {\n      switch (this.peer.kind) {\n        case 'cidr-ip':\n          if (this.peer.ip === '0.0.0.0/0') {\n            return 'Everyone (IPv4)';\n          }\n          if (this.peer.ip === '::/0') {\n            return 'Everyone (IPv6)';\n          }\n          return `${this.peer.ip}`;\n        case 'prefix-list': return `${this.peer.prefixListId}`;\n        case 'security-group': return `${this.peer.securityGroupId}`;\n      }\n    }\n\n    return '?';\n  }\n\n  public toJson(): RuleJson {\n    return {\n      groupId: this.groupId,\n      ipProtocol: this.ipProtocol,\n      fromPort: this.fromPort,\n      toPort: this.toPort,\n      peer: this.peer,\n    };\n  }\n}\n\nexport interface CidrIpPeer {\n  kind: 'cidr-ip';\n  ip: string;\n}\n\nexport interface SecurityGroupPeer {\n  kind: 'security-group';\n  securityGroupId: string;\n}\n\nexport interface PrefixListPeer {\n  kind: 'prefix-list';\n  prefixListId: string;\n}\n\nexport type RulePeer = CidrIpPeer | SecurityGroupPeer | PrefixListPeer;\n\nfunction peerEqual(a?: RulePeer, b?: RulePeer) {\n  if ((a === undefined) !== (b === undefined)) {\n    return false;\n  }\n  if (a === undefined) {\n    return true;\n  }\n\n  if (a.kind !== b!.kind) {\n    return false;\n  }\n  switch (a.kind) {\n    case 'cidr-ip': return a.ip === (b as typeof a).ip;\n    case 'security-group': return a.securityGroupId === (b as typeof a).securityGroupId;\n    case 'prefix-list': return a.prefixListId === (b as typeof a).prefixListId;\n  }\n}\n\nfunction findFirst<T>(obj: any, keys: string[], fn: (x: string) => T): T | undefined {\n  for (const key of keys) {\n    try {\n      if (key in obj) {\n        return fn(obj[key]);\n      }\n    } catch (e) {\n      debugger;\n    }\n  }\n  return undefined;\n}\n\nexport interface RuleJson {\n  groupId: string;\n  ipProtocol: string;\n  fromPort?: number;\n  toPort?: number;\n  peer?: RulePeer;\n}\n"]}