1 | ;
|
2 | Object.defineProperty(exports, "__esModule", { value: true });
|
3 | exports.normalizeStatement = exports.PostProcessPolicyDocument = void 0;
|
4 | const cdk = require("@aws-cdk/core");
|
5 | const util_1 = require("../util");
|
6 | const 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 | */
|
16 | class 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 | }
|
54 | exports.PostProcessPolicyDocument = PostProcessPolicyDocument;
|
55 | function 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 | }
|
113 | exports.normalizeStatement = normalizeStatement;
|
114 | function 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 |