UNPKG

23.7 kBJavaScriptView Raw
1"use strict";
2var _a;
3Object.defineProperty(exports, "__esModule", { value: true });
4exports.Policy = void 0;
5const jsiiDeprecationWarnings = require("../.warnings.jsii.js");
6const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
7const core_1 = require("@aws-cdk/core");
8const iam_generated_1 = require("./iam.generated");
9const policy_document_1 = require("./policy-document");
10const util_1 = require("./util");
11/**
12 * The AWS::IAM::Policy resource associates an IAM policy with IAM users, roles,
13 * or groups. For more information about IAM policies, see [Overview of IAM
14 * Policies](http://docs.aws.amazon.com/IAM/latest/UserGuide/policies_overview.html)
15 * in the IAM User Guide guide.
16 */
17class Policy extends core_1.Resource {
18 constructor(scope, id, props = {}) {
19 super(scope, id, {
20 physicalName: props.policyName ||
21 // generatePolicyName will take the last 128 characters of the logical id since
22 // policy names are limited to 128. the last 8 chars are a stack-unique hash, so
23 // that shouod be sufficient to ensure uniqueness within a principal.
24 core_1.Lazy.string({ produce: () => util_1.generatePolicyName(scope, resource.logicalId) }),
25 });
26 /**
27 * The policy document.
28 */
29 this.document = new policy_document_1.PolicyDocument();
30 this.roles = new Array();
31 this.users = new Array();
32 this.groups = new Array();
33 this.referenceTaken = false;
34 try {
35 jsiiDeprecationWarnings._aws_cdk_aws_iam_PolicyProps(props);
36 }
37 catch (error) {
38 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
39 Error.captureStackTrace(error, Policy);
40 }
41 throw error;
42 }
43 const self = this;
44 class CfnPolicyConditional extends iam_generated_1.CfnPolicy {
45 /**
46 * This function returns `true` if the CFN resource should be included in
47 * the cloudformation template unless `force` is `true`, if the policy
48 * document is empty, the resource will not be included.
49 */
50 shouldSynthesize() {
51 return self.force || self.referenceTaken || (!self.document.isEmpty && self.isAttached);
52 }
53 }
54 if (props.document) {
55 this.document = props.document;
56 }
57 const resource = new CfnPolicyConditional(this, 'Resource', {
58 policyDocument: this.document,
59 policyName: this.physicalName,
60 roles: util_1.undefinedIfEmpty(() => this.roles.map(r => r.roleName)),
61 users: util_1.undefinedIfEmpty(() => this.users.map(u => u.userName)),
62 groups: util_1.undefinedIfEmpty(() => this.groups.map(g => g.groupName)),
63 });
64 this._policyName = this.physicalName;
65 this.force = props.force ?? false;
66 if (props.users) {
67 props.users.forEach(u => this.attachToUser(u));
68 }
69 if (props.groups) {
70 props.groups.forEach(g => this.attachToGroup(g));
71 }
72 if (props.roles) {
73 props.roles.forEach(r => this.attachToRole(r));
74 }
75 if (props.statements) {
76 props.statements.forEach(p => this.addStatements(p));
77 }
78 }
79 /**
80 * Import a policy in this app based on its name
81 */
82 static fromPolicyName(scope, id, policyName) {
83 class Import extends core_1.Resource {
84 constructor() {
85 super(...arguments);
86 this.policyName = policyName;
87 }
88 }
89 return new Import(scope, id);
90 }
91 /**
92 * Adds a statement to the policy document.
93 */
94 addStatements(...statement) {
95 try {
96 jsiiDeprecationWarnings._aws_cdk_aws_iam_PolicyStatement(statement);
97 }
98 catch (error) {
99 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
100 Error.captureStackTrace(error, this.addStatements);
101 }
102 throw error;
103 }
104 this.document.addStatements(...statement);
105 }
106 /**
107 * Attaches this policy to a user.
108 */
109 attachToUser(user) {
110 try {
111 jsiiDeprecationWarnings._aws_cdk_aws_iam_IUser(user);
112 }
113 catch (error) {
114 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
115 Error.captureStackTrace(error, this.attachToUser);
116 }
117 throw error;
118 }
119 if (this.users.find(u => u === user)) {
120 return;
121 }
122 this.users.push(user);
123 user.attachInlinePolicy(this);
124 }
125 /**
126 * Attaches this policy to a role.
127 */
128 attachToRole(role) {
129 try {
130 jsiiDeprecationWarnings._aws_cdk_aws_iam_IRole(role);
131 }
132 catch (error) {
133 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
134 Error.captureStackTrace(error, this.attachToRole);
135 }
136 throw error;
137 }
138 if (this.roles.find(r => r === role)) {
139 return;
140 }
141 this.roles.push(role);
142 role.attachInlinePolicy(this);
143 }
144 /**
145 * Attaches this policy to a group.
146 */
147 attachToGroup(group) {
148 try {
149 jsiiDeprecationWarnings._aws_cdk_aws_iam_IGroup(group);
150 }
151 catch (error) {
152 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
153 Error.captureStackTrace(error, this.attachToGroup);
154 }
155 throw error;
156 }
157 if (this.groups.find(g => g === group)) {
158 return;
159 }
160 this.groups.push(group);
161 group.attachInlinePolicy(this);
162 }
163 /**
164 * The name of this policy.
165 *
166 * @attribute
167 */
168 get policyName() {
169 this.referenceTaken = true;
170 return this._policyName;
171 }
172 validate() {
173 const result = new Array();
174 // validate that the policy document is not empty
175 if (this.document.isEmpty) {
176 if (this.force) {
177 result.push('Policy created with force=true is empty. You must add statements to the policy');
178 }
179 if (!this.force && this.referenceTaken) {
180 result.push('This Policy has been referenced by a resource, so it must contain at least one statement.');
181 }
182 }
183 // validate that the policy is attached to at least one principal (role, user or group).
184 if (!this.isAttached) {
185 if (this.force) {
186 result.push('Policy created with force=true must be attached to at least one principal: user, group or role');
187 }
188 if (!this.force && this.referenceTaken) {
189 result.push('This Policy has been referenced by a resource, so it must be attached to at least one user, group or role.');
190 }
191 }
192 result.push(...this.document.validateForIdentityPolicy());
193 return result;
194 }
195 /**
196 * Whether the policy resource has been attached to any identity
197 */
198 get isAttached() {
199 return this.groups.length + this.users.length + this.roles.length > 0;
200 }
201}
202exports.Policy = Policy;
203_a = JSII_RTTI_SYMBOL_1;
204Policy[_a] = { fqn: "@aws-cdk/aws-iam.Policy", version: "1.161.0" };
205//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"policy.js","sourceRoot":"","sources":["policy.ts"],"names":[],"mappings":"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","sourcesContent":["import { IResource, Lazy, Resource } from '@aws-cdk/core';\nimport { Construct } from 'constructs';\nimport { IGroup } from './group';\nimport { CfnPolicy } from './iam.generated';\nimport { PolicyDocument } from './policy-document';\nimport { PolicyStatement } from './policy-statement';\nimport { IRole } from './role';\nimport { IUser } from './user';\nimport { generatePolicyName, undefinedIfEmpty } from './util';\n\n/**\n * Represents an IAM Policy\n *\n * @see https://docs.aws.amazon.com/IAM/latest/UserGuide/access_policies_manage.html\n */\nexport interface IPolicy extends IResource {\n  /**\n   * The name of this policy.\n   *\n   * @attribute\n   */\n  readonly policyName: string;\n}\n\n/**\n * Properties for defining an IAM inline policy document\n */\nexport interface PolicyProps {\n  /**\n   * The name of the policy. If you specify multiple policies for an entity,\n   * specify unique names. For example, if you specify a list of policies for\n   * an IAM role, each policy must have a unique name.\n   *\n   * @default - Uses the logical ID of the policy resource, which is ensured\n   * to be unique within the stack.\n   */\n  readonly policyName?: string;\n\n  /**\n   * Users to attach this policy to.\n   * You can also use `attachToUser(user)` to attach this policy to a user.\n   *\n   * @default - No users.\n   */\n  readonly users?: IUser[];\n\n  /**\n   * Roles to attach this policy to.\n   * You can also use `attachToRole(role)` to attach this policy to a role.\n   *\n   * @default - No roles.\n   */\n  readonly roles?: IRole[];\n\n  /**\n   * Groups to attach this policy to.\n   * You can also use `attachToGroup(group)` to attach this policy to a group.\n   *\n   * @default - No groups.\n   */\n  readonly groups?: IGroup[];\n\n  /**\n   * Initial set of permissions to add to this policy document.\n   * You can also use `addStatements(...statement)` to add permissions later.\n   *\n   * @default - No statements.\n   */\n  readonly statements?: PolicyStatement[];\n\n  /**\n   * Force creation of an `AWS::IAM::Policy`\n   *\n   * Unless set to `true`, this `Policy` construct will not materialize to an\n   * `AWS::IAM::Policy` CloudFormation resource in case it would have no effect\n   * (for example, if it remains unattached to an IAM identity or if it has no\n   * statements). This is generally desired behavior, since it prevents\n   * creating invalid--and hence undeployable--CloudFormation templates.\n   *\n   * In cases where you know the policy must be created and it is actually\n   * an error if no statements have been added to it, you can set this to `true`.\n   *\n   * @default false\n   */\n  readonly force?: boolean;\n\n  /**\n   * Initial PolicyDocument to use for this Policy. If omited, any\n   * `PolicyStatement` provided in the `statements` property will be applied\n   * against the empty default `PolicyDocument`.\n   *\n   * @default - An empty policy.\n   */\n  readonly document?: PolicyDocument;\n}\n\n/**\n * The AWS::IAM::Policy resource associates an IAM policy with IAM users, roles,\n * or groups. For more information about IAM policies, see [Overview of IAM\n * Policies](http://docs.aws.amazon.com/IAM/latest/UserGuide/policies_overview.html)\n * in the IAM User Guide guide.\n */\nexport class Policy extends Resource implements IPolicy {\n\n  /**\n   * Import a policy in this app based on its name\n   */\n  public static fromPolicyName(scope: Construct, id: string, policyName: string): IPolicy {\n    class Import extends Resource implements IPolicy {\n      public readonly policyName = policyName;\n    }\n\n    return new Import(scope, id);\n  }\n\n  /**\n   * The policy document.\n   */\n  public readonly document = new PolicyDocument();\n\n  private readonly _policyName: string;\n  private readonly roles = new Array<IRole>();\n  private readonly users = new Array<IUser>();\n  private readonly groups = new Array<IGroup>();\n  private readonly force: boolean;\n  private referenceTaken = false;\n\n  constructor(scope: Construct, id: string, props: PolicyProps = {}) {\n    super(scope, id, {\n      physicalName: props.policyName ||\n        // generatePolicyName will take the last 128 characters of the logical id since\n        // policy names are limited to 128. the last 8 chars are a stack-unique hash, so\n        // that shouod be sufficient to ensure uniqueness within a principal.\n        Lazy.string({ produce: () => generatePolicyName(scope, resource.logicalId) }),\n    });\n\n    const self = this;\n\n    class CfnPolicyConditional extends CfnPolicy {\n      /**\n       * This function returns `true` if the CFN resource should be included in\n       * the cloudformation template unless `force` is `true`, if the policy\n       * document is empty, the resource will not be included.\n       */\n      protected shouldSynthesize() {\n        return self.force || self.referenceTaken || (!self.document.isEmpty && self.isAttached);\n      }\n    }\n\n    if (props.document) {\n      this.document = props.document;\n    }\n\n    const resource = new CfnPolicyConditional(this, 'Resource', {\n      policyDocument: this.document,\n      policyName: this.physicalName,\n      roles: undefinedIfEmpty(() => this.roles.map(r => r.roleName)),\n      users: undefinedIfEmpty(() => this.users.map(u => u.userName)),\n      groups: undefinedIfEmpty(() => this.groups.map(g => g.groupName)),\n    });\n\n    this._policyName = this.physicalName!;\n    this.force = props.force ?? false;\n\n    if (props.users) {\n      props.users.forEach(u => this.attachToUser(u));\n    }\n\n    if (props.groups) {\n      props.groups.forEach(g => this.attachToGroup(g));\n    }\n\n    if (props.roles) {\n      props.roles.forEach(r => this.attachToRole(r));\n    }\n\n    if (props.statements) {\n      props.statements.forEach(p => this.addStatements(p));\n    }\n  }\n\n  /**\n   * Adds a statement to the policy document.\n   */\n  public addStatements(...statement: PolicyStatement[]) {\n    this.document.addStatements(...statement);\n  }\n\n  /**\n   * Attaches this policy to a user.\n   */\n  public attachToUser(user: IUser) {\n    if (this.users.find(u => u === user)) { return; }\n    this.users.push(user);\n    user.attachInlinePolicy(this);\n  }\n\n  /**\n   * Attaches this policy to a role.\n   */\n  public attachToRole(role: IRole) {\n    if (this.roles.find(r => r === role)) { return; }\n    this.roles.push(role);\n    role.attachInlinePolicy(this);\n  }\n\n  /**\n   * Attaches this policy to a group.\n   */\n  public attachToGroup(group: IGroup) {\n    if (this.groups.find(g => g === group)) { return; }\n    this.groups.push(group);\n    group.attachInlinePolicy(this);\n  }\n\n  /**\n   * The name of this policy.\n   *\n   * @attribute\n   */\n  public get policyName(): string {\n    this.referenceTaken = true;\n    return this._policyName;\n  }\n\n  protected validate(): string[] {\n    const result = new Array<string>();\n\n    // validate that the policy document is not empty\n    if (this.document.isEmpty) {\n      if (this.force) {\n        result.push('Policy created with force=true is empty. You must add statements to the policy');\n      }\n      if (!this.force && this.referenceTaken) {\n        result.push('This Policy has been referenced by a resource, so it must contain at least one statement.');\n      }\n    }\n\n    // validate that the policy is attached to at least one principal (role, user or group).\n    if (!this.isAttached) {\n      if (this.force) {\n        result.push('Policy created with force=true must be attached to at least one principal: user, group or role');\n      }\n      if (!this.force && this.referenceTaken) {\n        result.push('This Policy has been referenced by a resource, so it must be attached to at least one user, group or role.');\n      }\n    }\n\n    result.push(...this.document.validateForIdentityPolicy());\n\n    return result;\n  }\n\n  /**\n   * Whether the policy resource has been attached to any identity\n   */\n  private get isAttached() {\n    return this.groups.length + this.users.length + this.roles.length > 0;\n  }\n}\n"]}
\No newline at end of file