UNPKG

31.1 kBJavaScriptView Raw
1"use strict";
2var _a;
3Object.defineProperty(exports, "__esModule", { value: true });
4exports.PolicyDocument = void 0;
5const jsiiDeprecationWarnings = require("../.warnings.jsii.js");
6const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
7const cdk = require("@aws-cdk/core");
8const cxapi = require("@aws-cdk/cx-api");
9const policy_statement_1 = require("./policy-statement");
10const merge_statements_1 = require("./private/merge-statements");
11const postprocess_policy_document_1 = require("./private/postprocess-policy-document");
12/**
13 * A PolicyDocument is a collection of statements
14 */
15class PolicyDocument {
16 constructor(props = {}) {
17 this.statements = new Array();
18 try {
19 jsiiDeprecationWarnings._aws_cdk_aws_iam_PolicyDocumentProps(props);
20 }
21 catch (error) {
22 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
23 Error.captureStackTrace(error, PolicyDocument);
24 }
25 throw error;
26 }
27 this.creationStack = cdk.captureStackTrace();
28 this.autoAssignSids = !!props.assignSids;
29 this.minimize = props.minimize;
30 this.addStatements(...props.statements || []);
31 }
32 /**
33 * Creates a new PolicyDocument based on the object provided.
34 * This will accept an object created from the `.toJSON()` call
35 * @param obj the PolicyDocument in object form.
36 */
37 static fromJson(obj) {
38 const newPolicyDocument = new PolicyDocument();
39 const statement = obj.Statement ?? [];
40 if (statement && !Array.isArray(statement)) {
41 throw new Error('Statement must be an array');
42 }
43 newPolicyDocument.addStatements(...obj.Statement.map((s) => policy_statement_1.PolicyStatement.fromJson(s)));
44 return newPolicyDocument;
45 }
46 resolve(context) {
47 this._maybeMergeStatements(context.scope);
48 // In the previous implementation of 'merge', sorting of actions/resources on
49 // a statement always happened, even on singular statements. In the new
50 // implementation of 'merge', sorting only happens when actually combining 2
51 // statements. This affects all test snapshots, so we need to put in mechanisms
52 // to avoid having to update all snapshots.
53 //
54 // To do sorting in a way compatible with the previous implementation of merging,
55 // (so we don't have to update snapshots) do it after rendering, but only when
56 // merging is enabled.
57 const sort = this.shouldMerge(context.scope);
58 context.registerPostProcessor(new postprocess_policy_document_1.PostProcessPolicyDocument(this.autoAssignSids, sort));
59 return this.render();
60 }
61 /**
62 * Whether the policy document contains any statements.
63 */
64 get isEmpty() {
65 return this.statements.length === 0;
66 }
67 /**
68 * The number of statements already added to this policy.
69 * Can be used, for example, to generate unique "sid"s within the policy.
70 */
71 get statementCount() {
72 return this.statements.length;
73 }
74 /**
75 * Adds a statement to the policy document.
76 *
77 * @param statement the statement to add.
78 */
79 addStatements(...statement) {
80 try {
81 jsiiDeprecationWarnings._aws_cdk_aws_iam_PolicyStatement(statement);
82 }
83 catch (error) {
84 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
85 Error.captureStackTrace(error, this.addStatements);
86 }
87 throw error;
88 }
89 this.statements.push(...statement);
90 }
91 /**
92 * Encode the policy document as a string
93 */
94 toString() {
95 return cdk.Token.asString(this, {
96 displayHint: 'PolicyDocument',
97 });
98 }
99 /**
100 * JSON-ify the document
101 *
102 * Used when JSON.stringify() is called
103 */
104 toJSON() {
105 return this.render();
106 }
107 /**
108 * Validate that all policy statements in the policy document satisfies the
109 * requirements for any policy.
110 *
111 * @see https://docs.aws.amazon.com/IAM/latest/UserGuide/access_policies.html#access_policies-json
112 *
113 * @returns An array of validation error messages, or an empty array if the document is valid.
114 */
115 validateForAnyPolicy() {
116 const errors = new Array();
117 for (const statement of this.statements) {
118 errors.push(...statement.validateForAnyPolicy());
119 }
120 return errors;
121 }
122 /**
123 * Validate that all policy statements in the policy document satisfies the
124 * requirements for a resource-based policy.
125 *
126 * @see https://docs.aws.amazon.com/IAM/latest/UserGuide/access_policies.html#access_policies-json
127 *
128 * @returns An array of validation error messages, or an empty array if the document is valid.
129 */
130 validateForResourcePolicy() {
131 const errors = new Array();
132 for (const statement of this.statements) {
133 errors.push(...statement.validateForResourcePolicy());
134 }
135 return errors;
136 }
137 /**
138 * Validate that all policy statements in the policy document satisfies the
139 * requirements for an identity-based policy.
140 *
141 * @see https://docs.aws.amazon.com/IAM/latest/UserGuide/access_policies.html#access_policies-json
142 *
143 * @returns An array of validation error messages, or an empty array if the document is valid.
144 */
145 validateForIdentityPolicy() {
146 const errors = new Array();
147 for (const statement of this.statements) {
148 errors.push(...statement.validateForIdentityPolicy());
149 }
150 return errors;
151 }
152 /**
153 * Perform statement merging (if enabled and not done yet)
154 *
155 * @internal
156 */
157 _maybeMergeStatements(scope) {
158 if (this.shouldMerge(scope)) {
159 const result = merge_statements_1.mergeStatements(scope, this.statements, false);
160 this.statements.splice(0, this.statements.length, ...result.mergedStatements);
161 }
162 }
163 /**
164 * Split the statements of the PolicyDocument into multiple groups, limited by their size
165 *
166 * We do a round of size-limited merging first (making sure to not produce statements too
167 * large to fit into standalone policies), so that we can most accurately estimate total
168 * policy size. Another final round of minimization will be done just before rendering to
169 * end up with minimal policies that look nice to humans.
170 *
171 * Return a map of the final set of policy documents, mapped to the ORIGINAL (pre-merge)
172 * PolicyStatements that ended up in the given PolicyDocument.
173 *
174 * @internal
175 */
176 _splitDocument(scope, selfMaximumSize, splitMaximumSize) {
177 const self = this;
178 const newDocs = [];
179 // Maps final statements to original statements
180 let statementsToOriginals = new Map(this.statements.map(s => [s, [s]]));
181 if (this.shouldMerge(scope)) {
182 const result = merge_statements_1.mergeStatements(scope, this.statements, true);
183 this.statements.splice(0, this.statements.length, ...result.mergedStatements);
184 statementsToOriginals = result.originsMap;
185 }
186 const sizeOptions = policy_statement_1.deriveEstimateSizeOptions(scope);
187 // Cache statement sizes to avoid recomputing them based on the fields
188 const statementSizes = new Map(this.statements.map(s => [s, s._estimateSize(sizeOptions)]));
189 // Keep some size counters so we can avoid recomputing them based on the statements in each
190 let selfSize = 0;
191 const polSizes = new Map();
192 // Getter with a default to save some syntactic noise
193 const polSize = (x) => polSizes.get(x) ?? 0;
194 let i = 0;
195 while (i < this.statements.length) {
196 const statement = this.statements[i];
197 const statementSize = statementSizes.get(statement) ?? 0;
198 if (selfSize + statementSize < selfMaximumSize) {
199 // Fits in self
200 selfSize += statementSize;
201 i++;
202 continue;
203 }
204 // Split off to new PolicyDocument. Find the PolicyDocument we can add this to,
205 // or add a fresh one.
206 const addToDoc = findDocWithSpace(statementSize);
207 addToDoc.addStatements(statement);
208 polSizes.set(addToDoc, polSize(addToDoc) + statementSize);
209 this.statements.splice(i, 1);
210 }
211 // Return the set of all policy document and original statements
212 const ret = new Map();
213 ret.set(this, this.statements.flatMap(s => statementsToOriginals.get(s) ?? [s]));
214 for (const newDoc of newDocs) {
215 ret.set(newDoc, newDoc.statements.flatMap(s => statementsToOriginals.get(s) ?? [s]));
216 }
217 return ret;
218 function findDocWithSpace(size) {
219 let j = 0;
220 while (j < newDocs.length && polSize(newDocs[j]) + size > splitMaximumSize) {
221 j++;
222 }
223 if (j < newDocs.length) {
224 return newDocs[j];
225 }
226 const newDoc = new PolicyDocument({
227 assignSids: self.autoAssignSids,
228 minimize: self.minimize,
229 });
230 newDocs.push(newDoc);
231 return newDoc;
232 }
233 }
234 render() {
235 if (this.isEmpty) {
236 return undefined;
237 }
238 const doc = {
239 Statement: this.statements.map(s => s.toStatementJson()),
240 Version: '2012-10-17',
241 };
242 return doc;
243 }
244 shouldMerge(scope) {
245 return this.minimize ?? cdk.FeatureFlags.of(scope).isEnabled(cxapi.IAM_MINIMIZE_POLICIES) ?? false;
246 }
247}
248exports.PolicyDocument = PolicyDocument;
249_a = JSII_RTTI_SYMBOL_1;
250PolicyDocument[_a] = { fqn: "@aws-cdk/aws-iam.PolicyDocument", version: "1.161.0" };
251//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"policy-document.js","sourceRoot":"","sources":["policy-document.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from '@aws-cdk/core';\nimport { IConstruct } from '@aws-cdk/core';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport { PolicyStatement, deriveEstimateSizeOptions } from './policy-statement';\nimport { mergeStatements } from './private/merge-statements';\nimport { PostProcessPolicyDocument } from './private/postprocess-policy-document';\n\n/**\n * Properties for a new PolicyDocument\n */\nexport interface PolicyDocumentProps {\n  /**\n   * Automatically assign Statement Ids to all statements\n   *\n   * @default false\n   */\n  readonly assignSids?: boolean;\n\n  /**\n   * Initial statements to add to the policy document\n   *\n   * @default - No statements\n   */\n  readonly statements?: PolicyStatement[];\n\n  /**\n   * Try to minimize the policy by merging statements\n   *\n   * To avoid overrunning the maximum policy size, combine statements if they produce\n   * the same result. Merging happens according to the following rules:\n   *\n   * - The Effect of both statements is the same\n   * - Neither of the statements have a 'Sid'\n   * - Combine Principals if the rest of the statement is exactly the same.\n   * - Combine Resources if the rest of the statement is exactly the same.\n   * - Combine Actions if the rest of the statement is exactly the same.\n   * - We will never combine NotPrincipals, NotResources or NotActions, because doing\n   *   so would change the meaning of the policy document.\n   *\n   * @default - false, unless the feature flag `@aws-cdk/aws-iam:minimizePolicies` is set\n   */\n  readonly minimize?: boolean;\n}\n\n/**\n * A PolicyDocument is a collection of statements\n */\nexport class PolicyDocument implements cdk.IResolvable {\n\n  /**\n   * Creates a new PolicyDocument based on the object provided.\n   * This will accept an object created from the `.toJSON()` call\n   * @param obj the PolicyDocument in object form.\n   */\n  public static fromJson(obj: any): PolicyDocument {\n    const newPolicyDocument = new PolicyDocument();\n    const statement = obj.Statement ?? [];\n    if (statement && !Array.isArray(statement)) {\n      throw new Error('Statement must be an array');\n    }\n    newPolicyDocument.addStatements(...obj.Statement.map((s: any) => PolicyStatement.fromJson(s)));\n    return newPolicyDocument;\n  }\n\n  public readonly creationStack: string[];\n  private readonly statements = new Array<PolicyStatement>();\n  private readonly autoAssignSids: boolean;\n  private readonly minimize?: boolean;\n\n  constructor(props: PolicyDocumentProps = {}) {\n    this.creationStack = cdk.captureStackTrace();\n    this.autoAssignSids = !!props.assignSids;\n    this.minimize = props.minimize;\n\n    this.addStatements(...props.statements || []);\n  }\n\n  public resolve(context: cdk.IResolveContext): any {\n    this._maybeMergeStatements(context.scope);\n\n    // In the previous implementation of 'merge', sorting of actions/resources on\n    // a statement always happened, even  on singular statements. In the new\n    // implementation of 'merge', sorting only happens when actually combining 2\n    // statements. This affects all test snapshots, so we need to put in mechanisms\n    // to avoid having to update all snapshots.\n    //\n    // To do sorting in a way compatible with the previous implementation of merging,\n    // (so we don't have to update snapshots) do it after rendering, but only when\n    // merging is enabled.\n    const sort = this.shouldMerge(context.scope);\n    context.registerPostProcessor(new PostProcessPolicyDocument(this.autoAssignSids, sort));\n    return this.render();\n  }\n\n  /**\n   * Whether the policy document contains any statements.\n   */\n  public get isEmpty(): boolean {\n    return this.statements.length === 0;\n  }\n\n  /**\n   * The number of statements already added to this policy.\n   * Can be used, for example, to generate unique \"sid\"s within the policy.\n   */\n  public get statementCount(): number {\n    return this.statements.length;\n  }\n\n  /**\n   * Adds a statement to the policy document.\n   *\n   * @param statement the statement to add.\n   */\n  public addStatements(...statement: PolicyStatement[]) {\n    this.statements.push(...statement);\n  }\n\n  /**\n   * Encode the policy document as a string\n   */\n  public toString() {\n    return cdk.Token.asString(this, {\n      displayHint: 'PolicyDocument',\n    });\n  }\n\n  /**\n   * JSON-ify the document\n   *\n   * Used when JSON.stringify() is called\n   */\n  public toJSON() {\n    return this.render();\n  }\n\n  /**\n   * Validate that all policy statements in the policy document satisfies the\n   * requirements for any policy.\n   *\n   * @see https://docs.aws.amazon.com/IAM/latest/UserGuide/access_policies.html#access_policies-json\n   *\n   * @returns An array of validation error messages, or an empty array if the document is valid.\n   */\n  public validateForAnyPolicy(): string[] {\n    const errors = new Array<string>();\n    for (const statement of this.statements) {\n      errors.push(...statement.validateForAnyPolicy());\n    }\n    return errors;\n  }\n\n  /**\n   * Validate that all policy statements in the policy document satisfies the\n   * requirements for a resource-based policy.\n   *\n   * @see https://docs.aws.amazon.com/IAM/latest/UserGuide/access_policies.html#access_policies-json\n   *\n   * @returns An array of validation error messages, or an empty array if the document is valid.\n   */\n  public validateForResourcePolicy(): string[] {\n    const errors = new Array<string>();\n    for (const statement of this.statements) {\n      errors.push(...statement.validateForResourcePolicy());\n    }\n    return errors;\n  }\n\n  /**\n   * Validate that all policy statements in the policy document satisfies the\n   * requirements for an identity-based policy.\n   *\n   * @see https://docs.aws.amazon.com/IAM/latest/UserGuide/access_policies.html#access_policies-json\n   *\n   * @returns An array of validation error messages, or an empty array if the document is valid.\n   */\n  public validateForIdentityPolicy(): string[] {\n    const errors = new Array<string>();\n    for (const statement of this.statements) {\n      errors.push(...statement.validateForIdentityPolicy());\n    }\n    return errors;\n  }\n\n  /**\n   * Perform statement merging (if enabled and not done yet)\n   *\n   * @internal\n   */\n  public _maybeMergeStatements(scope: cdk.IConstruct): void {\n    if (this.shouldMerge(scope)) {\n      const result = mergeStatements(scope, this.statements, false);\n      this.statements.splice(0, this.statements.length, ...result.mergedStatements);\n    }\n  }\n\n  /**\n   * Split the statements of the PolicyDocument into multiple groups, limited by their size\n   *\n   * We do a round of size-limited merging first (making sure to not produce statements too\n   * large to fit into standalone policies), so that we can most accurately estimate total\n   * policy size. Another final round of minimization will be done just before rendering to\n   * end up with minimal policies that look nice to humans.\n   *\n   * Return a map of the final set of policy documents, mapped to the ORIGINAL (pre-merge)\n   * PolicyStatements that ended up in the given PolicyDocument.\n   *\n   * @internal\n   */\n  public _splitDocument(scope: IConstruct, selfMaximumSize: number, splitMaximumSize: number): Map<PolicyDocument, PolicyStatement[]> {\n    const self = this;\n    const newDocs: PolicyDocument[] = [];\n\n    // Maps final statements to original statements\n    let statementsToOriginals = new Map(this.statements.map(s => [s, [s]]));\n    if (this.shouldMerge(scope)) {\n      const result = mergeStatements(scope, this.statements, true);\n      this.statements.splice(0, this.statements.length, ...result.mergedStatements);\n      statementsToOriginals = result.originsMap;\n    }\n\n    const sizeOptions = deriveEstimateSizeOptions(scope);\n\n    // Cache statement sizes to avoid recomputing them based on the fields\n    const statementSizes = new Map<PolicyStatement, number>(this.statements.map(s => [s, s._estimateSize(sizeOptions)]));\n\n    // Keep some size counters so we can avoid recomputing them based on the statements in each\n    let selfSize = 0;\n    const polSizes = new Map<PolicyDocument, number>();\n    // Getter with a default to save some syntactic noise\n    const polSize = (x: PolicyDocument) => polSizes.get(x) ?? 0;\n\n    let i = 0;\n    while (i < this.statements.length) {\n      const statement = this.statements[i];\n\n      const statementSize = statementSizes.get(statement) ?? 0;\n      if (selfSize + statementSize < selfMaximumSize) {\n        // Fits in self\n        selfSize += statementSize;\n        i++;\n        continue;\n      }\n\n      // Split off to new PolicyDocument. Find the PolicyDocument we can add this to,\n      // or add a fresh one.\n      const addToDoc = findDocWithSpace(statementSize);\n      addToDoc.addStatements(statement);\n      polSizes.set(addToDoc, polSize(addToDoc) + statementSize);\n      this.statements.splice(i, 1);\n    }\n\n    // Return the set of all policy document and original statements\n    const ret = new Map<PolicyDocument, PolicyStatement[]>();\n    ret.set(this, this.statements.flatMap(s => statementsToOriginals.get(s) ?? [s]));\n    for (const newDoc of newDocs) {\n      ret.set(newDoc, newDoc.statements.flatMap(s => statementsToOriginals.get(s) ?? [s]));\n    }\n    return ret;\n\n    function findDocWithSpace(size: number) {\n      let j = 0;\n      while (j < newDocs.length && polSize(newDocs[j]) + size > splitMaximumSize) {\n        j++;\n      }\n      if (j < newDocs.length) {\n        return newDocs[j];\n      }\n\n      const newDoc = new PolicyDocument({\n        assignSids: self.autoAssignSids,\n        minimize: self.minimize,\n      });\n      newDocs.push(newDoc);\n      return newDoc;\n    }\n  }\n\n  private render(): any {\n    if (this.isEmpty) {\n      return undefined;\n    }\n\n    const doc = {\n      Statement: this.statements.map(s => s.toStatementJson()),\n      Version: '2012-10-17',\n    };\n\n    return doc;\n  }\n\n  private shouldMerge(scope: IConstruct) {\n    return this.minimize ?? cdk.FeatureFlags.of(scope).isEnabled(cxapi.IAM_MINIMIZE_POLICIES) ?? false;\n  }\n}\n"]}
\No newline at end of file