UNPKG

14.5 kBJavaScriptView Raw
1"use strict";
2Object.defineProperty(exports, "__esModule", { value: true });
3exports.normalizeStatement = exports.PostProcessPolicyDocument = void 0;
4const cdk = require("@aws-cdk/core");
5const util_1 = require("../util");
6const merge_statements_1 = require("./merge-statements");
7/**
8 * A Token postprocesser for policy documents
9 *
10 * Removes duplicate statements, merges statements, and assign Sids if necessary
11 *
12 * Because policy documents can contain all kinds of crazy things,
13 * we do all the necessary work here after the document has been mostly resolved
14 * into a predictable CloudFormation form.
15 */
16class PostProcessPolicyDocument {
17 constructor(autoAssignSids, minimize) {
18 this.autoAssignSids = autoAssignSids;
19 this.minimize = minimize;
20 }
21 postProcess(input, _context) {
22 if (!input || !input.Statement) {
23 return input;
24 }
25 if (this.minimize) {
26 input.Statement = merge_statements_1.mergeStatements(input.Statement);
27 }
28 // Also remove full-on duplicates (this will not be necessary if
29 // we minimized, but it might still dedupe statements we didn't
30 // minimize like 'Deny' statements, and definitely is still necessary
31 // if we didn't minimize)
32 const jsonStatements = new Set();
33 const uniqueStatements = [];
34 for (const statement of input.Statement) {
35 const jsonStatement = JSON.stringify(statement);
36 if (!jsonStatements.has(jsonStatement)) {
37 uniqueStatements.push(statement);
38 jsonStatements.add(jsonStatement);
39 }
40 }
41 // assign unique SIDs (the statement index) if `autoAssignSids` is enabled
42 const statements = uniqueStatements.map((s, i) => {
43 if (this.autoAssignSids && !s.Sid) {
44 s.Sid = i.toString();
45 }
46 return s;
47 });
48 return {
49 ...input,
50 Statement: statements,
51 };
52 }
53}
54exports.PostProcessPolicyDocument = PostProcessPolicyDocument;
55function normalizeStatement(s) {
56 return noUndef({
57 Action: _norm(s.Action, { unique: true }),
58 NotAction: _norm(s.NotAction, { unique: true }),
59 Condition: _norm(s.Condition),
60 Effect: _norm(s.Effect),
61 Principal: _normPrincipal(s.Principal),
62 NotPrincipal: _normPrincipal(s.NotPrincipal),
63 Resource: _norm(s.Resource, { unique: true }),
64 NotResource: _norm(s.NotResource, { unique: true }),
65 Sid: _norm(s.Sid),
66 });
67 function _norm(values, { unique = false } = { unique: false }) {
68 if (values == null) {
69 return undefined;
70 }
71 if (cdk.Token.isUnresolved(values)) {
72 return values;
73 }
74 if (Array.isArray(values)) {
75 if (!values || values.length === 0) {
76 return undefined;
77 }
78 if (values.length === 1) {
79 return values[0];
80 }
81 return unique ? Array.from(new Set(values)) : values;
82 }
83 if (values && typeof (values) === 'object') {
84 if (Object.keys(values).length === 0) {
85 return undefined;
86 }
87 }
88 return values;
89 }
90 function _normPrincipal(principal) {
91 if (!principal) {
92 return undefined;
93 }
94 const keys = Object.keys(principal);
95 if (keys.length === 0) {
96 return undefined;
97 }
98 // This is handling a special case for round-tripping a literal
99 // string principal loaded from JSON.
100 if (util_1.LITERAL_STRING_KEY in principal) {
101 return principal[util_1.LITERAL_STRING_KEY][0];
102 }
103 const result = {};
104 for (const key of keys) {
105 const normVal = _norm(principal[key]);
106 if (normVal) {
107 result[key] = normVal;
108 }
109 }
110 return result;
111 }
112}
113exports.normalizeStatement = normalizeStatement;
114function noUndef(x) {
115 const ret = {};
116 for (const [key, value] of Object.entries(x)) {
117 if (value !== undefined) {
118 ret[key] = value;
119 }
120 }
121 return ret;
122}
123//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"postprocess-policy-document.js","sourceRoot":"","sources":["postprocess-policy-document.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from '@aws-cdk/core';\nimport { LITERAL_STRING_KEY } from '../util';\nimport { mergeStatements } from './merge-statements';\n\n/**\n * A Token postprocesser for policy documents\n *\n * Removes duplicate statements, merges statements, and assign Sids if necessary\n *\n * Because policy documents can contain all kinds of crazy things,\n * we do all the necessary work here after the document has been mostly resolved\n * into a predictable CloudFormation form.\n */\nexport class PostProcessPolicyDocument implements cdk.IPostProcessor {\n  constructor(private readonly autoAssignSids: boolean, private readonly minimize: boolean) {\n  }\n\n  public postProcess(input: any, _context: cdk.IResolveContext): any {\n    if (!input || !input.Statement) {\n      return input;\n    }\n\n    if (this.minimize) {\n      input.Statement = mergeStatements(input.Statement);\n    }\n\n    // Also remove full-on duplicates (this will not be necessary if\n    // we minimized, but it might still dedupe statements we didn't\n    // minimize like 'Deny' statements, and definitely is still necessary\n    // if we didn't minimize)\n    const jsonStatements = new Set<string>();\n    const uniqueStatements: any[] = [];\n\n    for (const statement of input.Statement) {\n      const jsonStatement = JSON.stringify(statement);\n      if (!jsonStatements.has(jsonStatement)) {\n        uniqueStatements.push(statement);\n        jsonStatements.add(jsonStatement);\n      }\n    }\n\n    // assign unique SIDs (the statement index) if `autoAssignSids` is enabled\n    const statements = uniqueStatements.map((s, i) => {\n      if (this.autoAssignSids && !s.Sid) {\n        s.Sid = i.toString();\n      }\n\n      return s;\n    });\n\n    return {\n      ...input,\n      Statement: statements,\n    };\n  }\n}\n\n// An IAM value is a string or a CloudFormation intrinsic\nexport type IamValue = string | Record<string, any> | Array<string | Record<string, any>>;\n\nexport interface StatementSchema {\n  readonly Sid?: string;\n  readonly Effect?: string;\n  readonly Principal?: Record<string, IamValue>;\n  readonly NotPrincipal?: Record<string, IamValue>;\n  readonly Resource?: IamValue;\n  readonly NotResource?: IamValue;\n  readonly Action?: IamValue;\n  readonly NotAction?: IamValue;\n  readonly Condition?: unknown;\n}\n\n\nexport function normalizeStatement(s: StatementSchema) {\n  return noUndef({\n    Action: _norm(s.Action, { unique: true }),\n    NotAction: _norm(s.NotAction, { unique: true }),\n    Condition: _norm(s.Condition),\n    Effect: _norm(s.Effect),\n    Principal: _normPrincipal(s.Principal),\n    NotPrincipal: _normPrincipal(s.NotPrincipal),\n    Resource: _norm(s.Resource, { unique: true }),\n    NotResource: _norm(s.NotResource, { unique: true }),\n    Sid: _norm(s.Sid),\n  });\n\n  function _norm(values: any, { unique = false }: { unique: boolean } = { unique: false }) {\n\n    if (values == null) {\n      return undefined;\n    }\n\n    if (cdk.Token.isUnresolved(values)) {\n      return values;\n    }\n\n    if (Array.isArray(values)) {\n      if (!values || values.length === 0) {\n        return undefined;\n      }\n\n      if (values.length === 1) {\n        return values[0];\n      }\n\n      return unique ? Array.from(new Set(values)) : values;\n    }\n\n    if (values && typeof(values) === 'object') {\n      if (Object.keys(values).length === 0) {\n        return undefined;\n      }\n    }\n\n    return values;\n  }\n\n  function _normPrincipal(principal?: { [key: string]: any }) {\n    if (!principal) { return undefined; }\n\n    const keys = Object.keys(principal);\n    if (keys.length === 0) { return undefined; }\n\n    // This is handling a special case for round-tripping a literal\n    // string principal loaded from JSON.\n    if (LITERAL_STRING_KEY in principal) {\n      return principal[LITERAL_STRING_KEY][0];\n    }\n\n    const result: any = {};\n    for (const key of keys) {\n      const normVal = _norm(principal[key]);\n      if (normVal) {\n        result[key] = normVal;\n      }\n    }\n    return result;\n  }\n}\n\nfunction noUndef(x: any): any {\n  const ret: any = {};\n  for (const [key, value] of Object.entries(x)) {\n    if (value !== undefined) {\n      ret[key] = value;\n    }\n  }\n  return ret;\n}\n"]}
\No newline at end of file