UNPKG

29.9 kBJavaScriptView Raw
1"use strict";
2var _a, _b;
3Object.defineProperty(exports, "__esModule", { value: true });
4exports.CompositeDependable = exports.Grant = void 0;
5const jsiiDeprecationWarnings = require("../.warnings.jsii.js");
6const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
7const cdk = require("@aws-cdk/core");
8const policy_statement_1 = require("./policy-statement");
9/**
10 * Result of a grant() operation
11 *
12 * This class is not instantiable by consumers on purpose, so that they will be
13 * required to call the Grant factory functions.
14 */
15class Grant {
16 constructor(props) {
17 this.options = props.options;
18 this.principalStatement = props.principalStatement;
19 this.resourceStatement = props.resourceStatement;
20 cdk.DependableTrait.implement(this, {
21 get dependencyRoots() {
22 return props.policyDependable ? cdk.DependableTrait.get(props.policyDependable).dependencyRoots : [];
23 },
24 });
25 }
26 /**
27 * Grant the given permissions to the principal
28 *
29 * The permissions will be added to the principal policy primarily, falling
30 * back to the resource policy if necessary. The permissions must be granted
31 * somewhere.
32 *
33 * - Trying to grant permissions to a principal that does not admit adding to
34 * the principal policy while not providing a resource with a resource policy
35 * is an error.
36 * - Trying to grant permissions to an absent principal (possible in the
37 * case of imported resources) leads to a warning being added to the
38 * resource construct.
39 */
40 static addToPrincipalOrResource(options) {
41 var _c;
42 try {
43 jsiiDeprecationWarnings._aws_cdk_aws_iam_GrantWithResourceOptions(options);
44 }
45 catch (error) {
46 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
47 Error.captureStackTrace(error, this.addToPrincipalOrResource);
48 }
49 throw error;
50 }
51 const result = Grant.addToPrincipal({
52 ...options,
53 scope: options.resource,
54 });
55 const resourceAndPrincipalAccountComparison = options.grantee.grantPrincipal.principalAccount
56 ? cdk.Token.compareStrings(options.resource.env.account, options.grantee.grantPrincipal.principalAccount)
57 : undefined;
58 // if both accounts are tokens, we assume here they are the same
59 const equalOrBothUnresolved = resourceAndPrincipalAccountComparison === cdk.TokenComparison.SAME
60 || resourceAndPrincipalAccountComparison == cdk.TokenComparison.BOTH_UNRESOLVED;
61 const sameAccount = resourceAndPrincipalAccountComparison
62 ? equalOrBothUnresolved
63 // if the principal doesn't have an account (for example, a service principal),
64 // we should modify the resource's trust policy
65 : false;
66 // If we added to the principal AND we're in the same account, then we're done.
67 // If not, it's a different account and we must also add a trust policy on the resource.
68 if (result.success && sameAccount) {
69 return result;
70 }
71 const statement = new policy_statement_1.PolicyStatement({
72 actions: options.actions,
73 resources: (options.resourceSelfArns || options.resourceArns),
74 principals: [options.grantee.grantPrincipal],
75 });
76 const resourceResult = options.resource.addToResourcePolicy(statement);
77 return new Grant({
78 resourceStatement: statement,
79 options,
80 policyDependable: resourceResult.statementAdded ? (_c = resourceResult.policyDependable) !== null && _c !== void 0 ? _c : options.resource : undefined,
81 });
82 }
83 /**
84 * Try to grant the given permissions to the given principal
85 *
86 * Absence of a principal leads to a warning, but failing to add
87 * the permissions to a present principal is not an error.
88 */
89 static addToPrincipal(options) {
90 try {
91 jsiiDeprecationWarnings._aws_cdk_aws_iam_GrantOnPrincipalOptions(options);
92 }
93 catch (error) {
94 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
95 Error.captureStackTrace(error, this.addToPrincipal);
96 }
97 throw error;
98 }
99 const statement = new policy_statement_1.PolicyStatement({
100 actions: options.actions,
101 resources: options.resourceArns,
102 });
103 const addedToPrincipal = options.grantee.grantPrincipal.addToPrincipalPolicy(statement);
104 if (!addedToPrincipal.statementAdded) {
105 return new Grant({ principalStatement: undefined, options });
106 }
107 if (!addedToPrincipal.policyDependable) {
108 throw new Error('Contract violation: when Principal returns statementAdded=true, it should return a dependable');
109 }
110 return new Grant({ principalStatement: statement, options, policyDependable: addedToPrincipal.policyDependable });
111 }
112 /**
113 * Add a grant both on the principal and on the resource
114 *
115 * As long as any principal is given, granting on the principal may fail (in
116 * case of a non-identity principal), but granting on the resource will
117 * never fail.
118 *
119 * Statement will be the resource statement.
120 */
121 static addToPrincipalAndResource(options) {
122 var _c;
123 try {
124 jsiiDeprecationWarnings._aws_cdk_aws_iam_GrantOnPrincipalAndResourceOptions(options);
125 }
126 catch (error) {
127 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
128 Error.captureStackTrace(error, this.addToPrincipalAndResource);
129 }
130 throw error;
131 }
132 const result = Grant.addToPrincipal({
133 ...options,
134 scope: options.resource,
135 });
136 const statement = new policy_statement_1.PolicyStatement({
137 actions: options.actions,
138 resources: (options.resourceSelfArns || options.resourceArns),
139 principals: [options.resourcePolicyPrincipal || options.grantee.grantPrincipal],
140 });
141 const resourceResult = options.resource.addToResourcePolicy(statement);
142 const resourceDependable = resourceResult.statementAdded ? (_c = resourceResult.policyDependable) !== null && _c !== void 0 ? _c : options.resource : undefined;
143 return new Grant({
144 principalStatement: statement,
145 resourceStatement: result.resourceStatement,
146 options,
147 policyDependable: resourceDependable ? new CompositeDependable(result, resourceDependable) : result,
148 });
149 }
150 /**
151 * Returns a "no-op" `Grant` object which represents a "dropped grant".
152 *
153 * This can be used for e.g. imported resources where you may not be able to modify
154 * the resource's policy or some underlying policy which you don't know about.
155 *
156 * @param grantee The intended grantee
157 * @param _intent The user's intent (will be ignored at the moment)
158 */
159 static drop(grantee, _intent) {
160 try {
161 jsiiDeprecationWarnings._aws_cdk_aws_iam_IGrantable(grantee);
162 }
163 catch (error) {
164 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
165 Error.captureStackTrace(error, this.drop);
166 }
167 throw error;
168 }
169 return new Grant({
170 options: { grantee, actions: [], resourceArns: [] },
171 });
172 }
173 /**
174 * Whether the grant operation was successful
175 */
176 get success() {
177 return this.principalStatement !== undefined || this.resourceStatement !== undefined;
178 }
179 /**
180 * Throw an error if this grant wasn't successful
181 */
182 assertSuccess() {
183 if (!this.success) {
184 // eslint-disable-next-line max-len
185 throw new Error(`${describeGrant(this.options)} could not be added on either identity or resource policy.`);
186 }
187 }
188 /**
189 * Make sure this grant is applied before the given constructs are deployed
190 *
191 * The same as construct.node.addDependency(grant), but slightly nicer to read.
192 */
193 applyBefore(...constructs) {
194 for (const construct of constructs) {
195 construct.node.addDependency(this);
196 }
197 }
198}
199exports.Grant = Grant;
200_a = JSII_RTTI_SYMBOL_1;
201Grant[_a] = { fqn: "@aws-cdk/aws-iam.Grant", version: "1.156.1" };
202function describeGrant(options) {
203 return `Permissions for '${options.grantee}' to call '${options.actions}' on '${options.resourceArns}'`;
204}
205/**
206 * Composite dependable
207 *
208 * Not as simple as eagerly getting the dependency roots from the
209 * inner dependables, as they may be mutable so we need to defer
210 * the query.
211 */
212class CompositeDependable {
213 constructor(...dependables) {
214 cdk.DependableTrait.implement(this, {
215 get dependencyRoots() {
216 return Array.prototype.concat.apply([], dependables.map(d => cdk.DependableTrait.get(d).dependencyRoots));
217 },
218 });
219 }
220}
221exports.CompositeDependable = CompositeDependable;
222_b = JSII_RTTI_SYMBOL_1;
223CompositeDependable[_b] = { fqn: "@aws-cdk/aws-iam.CompositeDependable", version: "1.156.1" };
224//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"grant.js","sourceRoot":"","sources":["grant.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from '@aws-cdk/core';\nimport { PolicyStatement } from './policy-statement';\nimport { IGrantable, IPrincipal } from './principals';\n\n/**\n * Basic options for a grant operation\n *\n */\nexport interface CommonGrantOptions {\n  /**\n   * The principal to grant to\n   *\n   * @default if principal is undefined, no work is done.\n   */\n  readonly grantee: IGrantable;\n\n  /**\n   * The actions to grant\n   */\n  readonly actions: string[];\n\n  /**\n   * The resource ARNs to grant to\n   */\n  readonly resourceArns: string[];\n}\n\n/**\n * Options for a grant operation\n *\n */\nexport interface GrantWithResourceOptions extends CommonGrantOptions {\n  /**\n   * The resource with a resource policy\n   *\n   * The statement will be added to the resource policy if it couldn't be\n   * added to the principal policy.\n   */\n  readonly resource: IResourceWithPolicy;\n\n  /**\n   * When referring to the resource in a resource policy, use this as ARN.\n   *\n   * (Depending on the resource type, this needs to be '*' in a resource policy).\n   *\n   * @default Same as regular resource ARNs\n   */\n  readonly resourceSelfArns?: string[];\n}\n\n/**\n * Options for a grant operation that only applies to principals\n *\n */\nexport interface GrantOnPrincipalOptions extends CommonGrantOptions {\n  /**\n   * Construct to report warnings on in case grant could not be registered\n   *\n   * @default - the construct in which this construct is defined\n   */\n  readonly scope?: cdk.IConstruct;\n}\n\n/**\n * Options for a grant operation to both identity and resource\n *\n */\nexport interface GrantOnPrincipalAndResourceOptions extends CommonGrantOptions {\n  /**\n   * The resource with a resource policy\n   *\n   * The statement will always be added to the resource policy.\n   */\n  readonly resource: IResourceWithPolicy;\n\n  /**\n   * When referring to the resource in a resource policy, use this as ARN.\n   *\n   * (Depending on the resource type, this needs to be '*' in a resource policy).\n   *\n   * @default Same as regular resource ARNs\n   */\n  readonly resourceSelfArns?: string[];\n\n  /**\n   * The principal to use in the statement for the resource policy.\n   *\n   * @default - the principal of the grantee will be used\n   */\n  readonly resourcePolicyPrincipal?: IPrincipal;\n}\n\n/**\n * Result of a grant() operation\n *\n * This class is not instantiable by consumers on purpose, so that they will be\n * required to call the Grant factory functions.\n */\nexport class Grant implements cdk.IDependable {\n  /**\n   * Grant the given permissions to the principal\n   *\n   * The permissions will be added to the principal policy primarily, falling\n   * back to the resource policy if necessary. The permissions must be granted\n   * somewhere.\n   *\n   * - Trying to grant permissions to a principal that does not admit adding to\n   *   the principal policy while not providing a resource with a resource policy\n   *   is an error.\n   * - Trying to grant permissions to an absent principal (possible in the\n   *   case of imported resources) leads to a warning being added to the\n   *   resource construct.\n   */\n  public static addToPrincipalOrResource(options: GrantWithResourceOptions): Grant {\n    const result = Grant.addToPrincipal({\n      ...options,\n      scope: options.resource,\n    });\n\n    const resourceAndPrincipalAccountComparison = options.grantee.grantPrincipal.principalAccount\n      ? cdk.Token.compareStrings(options.resource.env.account, options.grantee.grantPrincipal.principalAccount)\n      : undefined;\n    // if both accounts are tokens, we assume here they are the same\n    const equalOrBothUnresolved = resourceAndPrincipalAccountComparison === cdk.TokenComparison.SAME\n      || resourceAndPrincipalAccountComparison == cdk.TokenComparison.BOTH_UNRESOLVED;\n    const sameAccount: boolean = resourceAndPrincipalAccountComparison\n      ? equalOrBothUnresolved\n      // if the principal doesn't have an account (for example, a service principal),\n      // we should modify the resource's trust policy\n      : false;\n    // If we added to the principal AND we're in the same account, then we're done.\n    // If not, it's a different account and we must also add a trust policy on the resource.\n    if (result.success && sameAccount) {\n      return result;\n    }\n\n    const statement = new PolicyStatement({\n      actions: options.actions,\n      resources: (options.resourceSelfArns || options.resourceArns),\n      principals: [options.grantee!.grantPrincipal],\n    });\n\n    const resourceResult = options.resource.addToResourcePolicy(statement);\n\n    return new Grant({\n      resourceStatement: statement,\n      options,\n      policyDependable: resourceResult.statementAdded ? resourceResult.policyDependable ?? options.resource : undefined,\n    });\n  }\n\n  /**\n   * Try to grant the given permissions to the given principal\n   *\n   * Absence of a principal leads to a warning, but failing to add\n   * the permissions to a present principal is not an error.\n   */\n  public static addToPrincipal(options: GrantOnPrincipalOptions): Grant {\n    const statement = new PolicyStatement({\n      actions: options.actions,\n      resources: options.resourceArns,\n    });\n\n    const addedToPrincipal = options.grantee.grantPrincipal.addToPrincipalPolicy(statement);\n    if (!addedToPrincipal.statementAdded) {\n      return new Grant({ principalStatement: undefined, options });\n    }\n\n    if (!addedToPrincipal.policyDependable) {\n      throw new Error('Contract violation: when Principal returns statementAdded=true, it should return a dependable');\n    }\n\n    return new Grant({ principalStatement: statement, options, policyDependable: addedToPrincipal.policyDependable });\n  }\n\n  /**\n   * Add a grant both on the principal and on the resource\n   *\n   * As long as any principal is given, granting on the principal may fail (in\n   * case of a non-identity principal), but granting on the resource will\n   * never fail.\n   *\n   * Statement will be the resource statement.\n   */\n  public static addToPrincipalAndResource(options: GrantOnPrincipalAndResourceOptions): Grant {\n    const result = Grant.addToPrincipal({\n      ...options,\n      scope: options.resource,\n    });\n\n    const statement = new PolicyStatement({\n      actions: options.actions,\n      resources: (options.resourceSelfArns || options.resourceArns),\n      principals: [options.resourcePolicyPrincipal || options.grantee!.grantPrincipal],\n    });\n\n    const resourceResult = options.resource.addToResourcePolicy(statement);\n    const resourceDependable = resourceResult.statementAdded ? resourceResult.policyDependable ?? options.resource : undefined;\n\n    return new Grant({\n      principalStatement: statement,\n      resourceStatement: result.resourceStatement,\n      options,\n      policyDependable: resourceDependable ? new CompositeDependable(result, resourceDependable) : result,\n    });\n  }\n\n  /**\n   * Returns a \"no-op\" `Grant` object which represents a \"dropped grant\".\n   *\n   * This can be used for e.g. imported resources where you may not be able to modify\n   * the resource's policy or some underlying policy which you don't know about.\n   *\n   * @param grantee The intended grantee\n   * @param _intent The user's intent (will be ignored at the moment)\n   */\n  public static drop(grantee: IGrantable, _intent: string): Grant {\n    return new Grant({\n      options: { grantee, actions: [], resourceArns: [] },\n    });\n  }\n\n  /**\n   * The statement that was added to the principal's policy\n   *\n   * Can be accessed to (e.g.) add additional conditions to the statement.\n   */\n  public readonly principalStatement?: PolicyStatement;\n\n  /**\n   * The statement that was added to the resource policy\n   *\n   * Can be accessed to (e.g.) add additional conditions to the statement.\n   */\n  public readonly resourceStatement?: PolicyStatement;\n\n  /**\n   * The options originally used to set this result\n   *\n   * Private member doubles as a way to make it impossible for an object literal to\n   * be structurally the same as this class.\n   */\n  private readonly options: CommonGrantOptions;\n\n  private constructor(props: GrantProps) {\n    this.options = props.options;\n    this.principalStatement = props.principalStatement;\n    this.resourceStatement = props.resourceStatement;\n\n    cdk.DependableTrait.implement(this, {\n      get dependencyRoots() {\n        return props.policyDependable ? cdk.DependableTrait.get(props.policyDependable).dependencyRoots : [];\n      },\n    });\n  }\n\n  /**\n   * Whether the grant operation was successful\n   */\n  public get success(): boolean {\n    return this.principalStatement !== undefined || this.resourceStatement !== undefined;\n  }\n\n  /**\n   * Throw an error if this grant wasn't successful\n   */\n  public assertSuccess(): void {\n    if (!this.success) {\n      // eslint-disable-next-line max-len\n      throw new Error(`${describeGrant(this.options)} could not be added on either identity or resource policy.`);\n    }\n  }\n\n  /**\n   * Make sure this grant is applied before the given constructs are deployed\n   *\n   * The same as construct.node.addDependency(grant), but slightly nicer to read.\n   */\n  public applyBefore(...constructs: cdk.IConstruct[]) {\n    for (const construct of constructs) {\n      construct.node.addDependency(this);\n    }\n  }\n}\n\nfunction describeGrant(options: CommonGrantOptions) {\n  return `Permissions for '${options.grantee}' to call '${options.actions}' on '${options.resourceArns}'`;\n}\n\ninterface GrantProps {\n  readonly options: CommonGrantOptions;\n  readonly principalStatement?: PolicyStatement;\n  readonly resourceStatement?: PolicyStatement;\n\n  /**\n   * Constructs whose deployment applies the grant\n   *\n   * Used to add dependencies on grants\n   */\n  readonly policyDependable?: cdk.IDependable;\n}\n\n/**\n * A resource with a resource policy that can be added to\n */\nexport interface IResourceWithPolicy extends cdk.IResource {\n  /**\n   * Add a statement to the resource's resource policy\n   */\n  addToResourcePolicy(statement: PolicyStatement): AddToResourcePolicyResult;\n}\n\n/**\n * Result of calling addToResourcePolicy\n */\nexport interface AddToResourcePolicyResult {\n  /**\n   * Whether the statement was added\n   */\n  readonly statementAdded: boolean;\n\n  /**\n   * Dependable which allows depending on the policy change being applied\n   *\n   * @default - If `statementAdded` is true, the resource object itself.\n   * Otherwise, no dependable.\n   */\n  readonly policyDependable?: cdk.IDependable;\n}\n\n/**\n * Composite dependable\n *\n * Not as simple as eagerly getting the dependency roots from the\n * inner dependables, as they may be mutable so we need to defer\n * the query.\n */\nexport class CompositeDependable implements cdk.IDependable {\n  constructor(...dependables: cdk.IDependable[]) {\n    cdk.DependableTrait.implement(this, {\n      get dependencyRoots(): cdk.IConstruct[] {\n        return Array.prototype.concat.apply([], dependables.map(d => cdk.DependableTrait.get(d).dependencyRoots));\n      },\n    });\n  }\n}\n"]}
\No newline at end of file