1 | ;
|
2 | var _a;
|
3 | Object.defineProperty(exports, "__esModule", { value: true });
|
4 | exports.Policy = void 0;
|
5 | const jsiiDeprecationWarnings = require("../.warnings.jsii.js");
|
6 | const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
7 | const core_1 = require("@aws-cdk/core");
|
8 | const iam_generated_1 = require("./iam.generated");
|
9 | const policy_document_1 = require("./policy-document");
|
10 | const 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 | */
|
17 | class 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 | }
|
202 | exports.Policy = Policy;
|
203 | _a = JSII_RTTI_SYMBOL_1;
|
204 | Policy[_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 |