UNPKG

@aws-cdk/cloudformation-diff

Version:

Utilities to diff CDK stacks against CloudFormation templates

270 lines 31.7 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.Effect = exports.Targets = exports.Statement = void 0; exports.parseStatements = parseStatements; exports.parseLambdaPermission = parseLambdaPermission; exports.renderCondition = renderCondition; const maybe_parsed_1 = require("../diff/maybe-parsed"); const util_1 = require("../util"); // namespace object imports won't work in the bundle for function exports // eslint-disable-next-line @typescript-eslint/no-require-imports const deepEqual = require('fast-deep-equal'); class Statement { constructor(statement) { if (typeof statement === 'string') { this.sid = undefined; this.effect = Effect.Unknown; this.resources = new Targets({}, '', ''); this.actions = new Targets({}, '', ''); this.principals = new Targets({}, '', ''); this.condition = undefined; this.serializedIntrinsic = statement; } else { this.sid = expectString(statement.Sid); this.effect = expectEffect(statement.Effect); this.resources = new Targets(statement, 'Resource', 'NotResource'); this.actions = new Targets(statement, 'Action', 'NotAction'); this.principals = new Targets(statement, 'Principal', 'NotPrincipal'); this.condition = statement.Condition; this.serializedIntrinsic = undefined; } } /** * Whether this statement is equal to the other statement */ equal(other) { return (this.sid === other.sid && this.effect === other.effect && this.serializedIntrinsic === other.serializedIntrinsic && this.resources.equal(other.resources) && this.actions.equal(other.actions) && this.principals.equal(other.principals) && deepEqual(this.condition, other.condition)); } render() { return this.serializedIntrinsic ? { resource: this.serializedIntrinsic, effect: '', action: '', principal: this.principals.render(), // these will be replaced by the call to replaceEmpty() from IamChanges condition: '', } : { resource: this.resources.render(), effect: this.effect, action: this.actions.render(), principal: this.principals.render(), condition: renderCondition(this.condition), }; } /** * Return a machine-readable version of the changes. * This is only used in tests. * * @internal */ _toJson() { return this.serializedIntrinsic ? (0, maybe_parsed_1.mkUnparseable)(this.serializedIntrinsic) : (0, maybe_parsed_1.mkParsed)((0, util_1.deepRemoveUndefined)({ sid: this.sid, effect: this.effect, resources: this.resources._toJson(), principals: this.principals._toJson(), actions: this.actions._toJson(), condition: this.condition, })); } /** * Whether this is a negative statement * * A statement is negative if any of its targets are negative, inverted * if the Effect is Deny. */ get isNegativeStatement() { const notTarget = this.actions.not || this.principals.not || this.resources.not; return this.effect === Effect.Allow ? notTarget : !notTarget; } } exports.Statement = Statement; /** * Parse a list of statements from undefined, a Statement, or a list of statements */ function parseStatements(x) { if (x === undefined) { x = []; } if (!Array.isArray(x)) { x = [x]; } return x.map((s) => new Statement(s)); } /** * Parse a Statement from a Lambda::Permission object * * This is actually what Lambda adds to the policy document if you call AddPermission. */ function parseLambdaPermission(x) { // Construct a statement from const statement = { Effect: 'Allow', Action: x.Action, Resource: x.FunctionName, }; if (x.Principal !== undefined) { if (x.Principal === '*') { // * statement.Principal = '*'; } else if (/^\d{12}$/.test(x.Principal)) { // Account number // eslint-disable-next-line @cdklabs/no-literal-partition statement.Principal = { AWS: `arn:aws:iam::${x.Principal}:root` }; } else { // Assume it's a service principal // We might get this wrong vs. the previous one for tokens. Nothing to be done // about that. It's only for human readable consumption after all. statement.Principal = { Service: x.Principal }; } } if (x.SourceArn !== undefined) { if (statement.Condition === undefined) { statement.Condition = {}; } statement.Condition.ArnLike = { 'AWS:SourceArn': x.SourceArn }; } if (x.SourceAccount !== undefined) { if (statement.Condition === undefined) { statement.Condition = {}; } statement.Condition.StringEquals = { 'AWS:SourceAccount': x.SourceAccount }; } if (x.EventSourceToken !== undefined) { if (statement.Condition === undefined) { statement.Condition = {}; } statement.Condition.StringEquals = { 'lambda:EventSourceToken': x.EventSourceToken }; } return new Statement(statement); } /** * Targets for a field */ class Targets { constructor(statement, positiveKey, negativeKey) { if (negativeKey in statement) { this.values = forceListOfStrings(statement[negativeKey]); this.not = true; } else { this.values = forceListOfStrings(statement[positiveKey]); this.not = false; } this.values.sort(); } get empty() { return this.values.length === 0; } /** * Whether this set of targets is equal to the other set of targets */ equal(other) { return this.not === other.not && deepEqual(this.values.sort(), other.values.sort()); } /** * If the current value set is empty, put this in it */ replaceEmpty(replacement) { if (this.empty) { this.values.push(replacement); } } /** * If the actions contains a '*', replace with this string. */ replaceStar(replacement) { for (let i = 0; i < this.values.length; i++) { if (this.values[i] === '*') { this.values[i] = replacement; } } this.values.sort(); } /** * Render into a summary table cell */ render() { return this.not ? this.values.map(s => `NOT ${s}`).join('\n') : this.values.join('\n'); } /** * Return a machine-readable version of the changes. * This is only used in tests. * * @internal */ _toJson() { return { not: this.not, values: this.values }; } } exports.Targets = Targets; var Effect; (function (Effect) { Effect["Unknown"] = "Unknown"; Effect["Allow"] = "Allow"; Effect["Deny"] = "Deny"; })(Effect || (exports.Effect = Effect = {})); function expectString(x) { return typeof x === 'string' ? x : undefined; } function expectEffect(x) { if (x === Effect.Allow || x === Effect.Deny) { return x; } return Effect.Unknown; } function forceListOfStrings(x) { if (typeof x === 'string') { return [x]; } if (typeof x === 'undefined' || x === null) { return []; } if (Array.isArray(x)) { return x.map(e => forceListOfStrings(e).join(',')); } if (typeof x === 'object' && x !== null) { const ret = []; for (const [key, value] of Object.entries(x)) { ret.push(...forceListOfStrings(value).map(s => `${key}:${s}`)); } return ret; } return [`${x}`]; } /** * Render the Condition column */ function renderCondition(condition) { if (!condition || Object.keys(condition).length === 0) { return ''; } const jsonRepresentation = JSON.stringify(condition, undefined, 2); // The JSON representation looks like this: // // { // "ArnLike": { // "AWS:SourceArn": "${MyTopic86869434}" // } // } // // We can make it more compact without losing information by getting rid of the outermost braces // and the indentation. const lines = jsonRepresentation.split('\n'); return lines.slice(1, lines.length - 1).map(s => s.slice(2)).join('\n'); } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"statement.js","sourceRoot":"","sources":["statement.ts"],"names":[],"mappings":"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","sourcesContent":["import type { MaybeParsed } from '../diff/maybe-parsed';\nimport { mkParsed, mkUnparseable } from '../diff/maybe-parsed';\nimport { deepRemoveUndefined } from '../util';\n\n// namespace object imports won't work in the bundle for function exports\n// eslint-disable-next-line @typescript-eslint/no-require-imports\nconst deepEqual = require('fast-deep-equal');\n\nexport class Statement {\n  /**\n   * Statement ID\n   */\n  public readonly sid: string | undefined;\n\n  /**\n   * Statement effect\n   */\n  public readonly effect: Effect;\n\n  /**\n   * Resources\n   */\n  public readonly resources: Targets;\n\n  /**\n   * Principals\n   */\n  public readonly principals: Targets;\n\n  /**\n   * Actions\n   */\n  public readonly actions: Targets;\n\n  /**\n   * Object with conditions\n   */\n  public readonly condition?: any;\n\n  private readonly serializedIntrinsic: string | undefined;\n\n  constructor(statement: UnknownMap | string) {\n    if (typeof statement === 'string') {\n      this.sid = undefined;\n      this.effect = Effect.Unknown;\n      this.resources = new Targets({}, '', '');\n      this.actions = new Targets({}, '', '');\n      this.principals = new Targets({}, '', '');\n      this.condition = undefined;\n      this.serializedIntrinsic = statement;\n    } else {\n      this.sid = expectString(statement.Sid);\n      this.effect = expectEffect(statement.Effect);\n      this.resources = new Targets(statement, 'Resource', 'NotResource');\n      this.actions = new Targets(statement, 'Action', 'NotAction');\n      this.principals = new Targets(statement, 'Principal', 'NotPrincipal');\n      this.condition = statement.Condition;\n      this.serializedIntrinsic = undefined;\n    }\n  }\n\n  /**\n   * Whether this statement is equal to the other statement\n   */\n  public equal(other: Statement): boolean {\n    return (this.sid === other.sid\n      && this.effect === other.effect\n      && this.serializedIntrinsic === other.serializedIntrinsic\n      && this.resources.equal(other.resources)\n      && this.actions.equal(other.actions)\n      && this.principals.equal(other.principals)\n      && deepEqual(this.condition, other.condition));\n  }\n\n  public render(): RenderedStatement {\n    return this.serializedIntrinsic\n      ? {\n        resource: this.serializedIntrinsic,\n        effect: '',\n        action: '',\n        principal: this.principals.render(), // these will be replaced by the call to replaceEmpty() from IamChanges\n        condition: '',\n      }\n      : {\n        resource: this.resources.render(),\n        effect: this.effect,\n        action: this.actions.render(),\n        principal: this.principals.render(),\n        condition: renderCondition(this.condition),\n      };\n  }\n\n  /**\n   * Return a machine-readable version of the changes.\n   * This is only used in tests.\n   *\n   * @internal\n   */\n  public _toJson(): MaybeParsed<StatementJson> {\n    return this.serializedIntrinsic\n      ? mkUnparseable(this.serializedIntrinsic)\n      : mkParsed(deepRemoveUndefined({\n        sid: this.sid,\n        effect: this.effect,\n        resources: this.resources._toJson(),\n        principals: this.principals._toJson(),\n        actions: this.actions._toJson(),\n        condition: this.condition,\n      }));\n  }\n\n  /**\n   * Whether this is a negative statement\n   *\n   * A statement is negative if any of its targets are negative, inverted\n   * if the Effect is Deny.\n   */\n  public get isNegativeStatement(): boolean {\n    const notTarget = this.actions.not || this.principals.not || this.resources.not;\n    return this.effect === Effect.Allow ? notTarget : !notTarget;\n  }\n}\n\nexport interface RenderedStatement {\n  readonly resource: string;\n  readonly effect: string;\n  readonly action: string;\n  readonly principal: string;\n  readonly condition: string;\n}\n\nexport interface StatementJson {\n  sid?: string;\n  effect: string;\n  resources: TargetsJson;\n  actions: TargetsJson;\n  principals: TargetsJson;\n  condition?: any;\n}\n\nexport interface TargetsJson {\n  not: boolean;\n  values: string[];\n}\n\n/**\n * Parse a list of statements from undefined, a Statement, or a list of statements\n */\nexport function parseStatements(x: any): Statement[] {\n  if (x === undefined) {\n    x = [];\n  }\n  if (!Array.isArray(x)) {\n    x = [x];\n  }\n  return x.map((s: any) => new Statement(s));\n}\n\n/**\n * Parse a Statement from a Lambda::Permission object\n *\n * This is actually what Lambda adds to the policy document if you call AddPermission.\n */\nexport function parseLambdaPermission(x: any): Statement {\n  // Construct a statement from\n  const statement: any = {\n    Effect: 'Allow',\n    Action: x.Action,\n    Resource: x.FunctionName,\n  };\n\n  if (x.Principal !== undefined) {\n    if (x.Principal === '*') {\n      // *\n      statement.Principal = '*';\n    } else if (/^\\d{12}$/.test(x.Principal)) {\n      // Account number\n      // eslint-disable-next-line @cdklabs/no-literal-partition\n      statement.Principal = { AWS: `arn:aws:iam::${x.Principal}:root` };\n    } else {\n      // Assume it's a service principal\n      // We might get this wrong vs. the previous one for tokens. Nothing to be done\n      // about that. It's only for human readable consumption after all.\n      statement.Principal = { Service: x.Principal };\n    }\n  }\n  if (x.SourceArn !== undefined) {\n    if (statement.Condition === undefined) {\n      statement.Condition = {};\n    }\n    statement.Condition.ArnLike = { 'AWS:SourceArn': x.SourceArn };\n  }\n  if (x.SourceAccount !== undefined) {\n    if (statement.Condition === undefined) {\n      statement.Condition = {};\n    }\n    statement.Condition.StringEquals = { 'AWS:SourceAccount': x.SourceAccount };\n  }\n  if (x.EventSourceToken !== undefined) {\n    if (statement.Condition === undefined) {\n      statement.Condition = {};\n    }\n    statement.Condition.StringEquals = { 'lambda:EventSourceToken': x.EventSourceToken };\n  }\n\n  return new Statement(statement);\n}\n\n/**\n * Targets for a field\n */\nexport class Targets {\n  /**\n   * The values of the targets\n   */\n  public readonly values: string[];\n\n  /**\n   * Whether positive or negative matchers\n   */\n  public readonly not: boolean;\n\n  constructor(statement: UnknownMap, positiveKey: string, negativeKey: string) {\n    if (negativeKey in statement) {\n      this.values = forceListOfStrings(statement[negativeKey]);\n      this.not = true;\n    } else {\n      this.values = forceListOfStrings(statement[positiveKey]);\n      this.not = false;\n    }\n    this.values.sort();\n  }\n\n  public get empty() {\n    return this.values.length === 0;\n  }\n\n  /**\n   * Whether this set of targets is equal to the other set of targets\n   */\n  public equal(other: Targets) {\n    return this.not === other.not && deepEqual(this.values.sort(), other.values.sort());\n  }\n\n  /**\n   * If the current value set is empty, put this in it\n   */\n  public replaceEmpty(replacement: string) {\n    if (this.empty) {\n      this.values.push(replacement);\n    }\n  }\n\n  /**\n   * If the actions contains a '*', replace with this string.\n   */\n  public replaceStar(replacement: string) {\n    for (let i = 0; i < this.values.length; i++) {\n      if (this.values[i] === '*') {\n        this.values[i] = replacement;\n      }\n    }\n    this.values.sort();\n  }\n\n  /**\n   * Render into a summary table cell\n   */\n  public render(): string {\n    return this.not\n      ? this.values.map(s => `NOT ${s}`).join('\\n')\n      : this.values.join('\\n');\n  }\n\n  /**\n   * Return a machine-readable version of the changes.\n   * This is only used in tests.\n   *\n   * @internal\n   */\n  public _toJson(): TargetsJson {\n    return { not: this.not, values: this.values };\n  }\n}\n\ntype UnknownMap = { [key: string]: unknown };\n\nexport enum Effect {\n  Unknown = 'Unknown',\n  Allow = 'Allow',\n  Deny = 'Deny',\n}\n\nfunction expectString(x: unknown): string | undefined {\n  return typeof x === 'string' ? x : undefined;\n}\n\nfunction expectEffect(x: unknown): Effect {\n  if (x === Effect.Allow || x === Effect.Deny) {\n    return x as Effect;\n  }\n  return Effect.Unknown;\n}\n\nfunction forceListOfStrings(x: unknown): string[] {\n  if (typeof x === 'string') {\n    return [x];\n  }\n  if (typeof x === 'undefined' || x === null) {\n    return [];\n  }\n\n  if (Array.isArray(x)) {\n    return x.map(e => forceListOfStrings(e).join(','));\n  }\n\n  if (typeof x === 'object' && x !== null) {\n    const ret: string[] = [];\n    for (const [key, value] of Object.entries(x)) {\n      ret.push(...forceListOfStrings(value).map(s => `${key}:${s}`));\n    }\n    return ret;\n  }\n\n  return [`${x}`];\n}\n\n/**\n * Render the Condition column\n */\nexport function renderCondition(condition: any): string {\n  if (!condition || Object.keys(condition).length === 0) {\n    return '';\n  }\n  const jsonRepresentation = JSON.stringify(condition, undefined, 2);\n\n  // The JSON representation looks like this:\n  //\n  //  {\n  //    \"ArnLike\": {\n  //      \"AWS:SourceArn\": \"${MyTopic86869434}\"\n  //    }\n  //  }\n  //\n  // We can make it more compact without losing information by getting rid of the outermost braces\n  // and the indentation.\n  const lines = jsonRepresentation.split('\\n');\n  return lines.slice(1, lines.length - 1).map(s => s.slice(2)).join('\\n');\n}\n"]}