UNPKG

@aws-cdk/aws-iam

Version:

CDK routines for easily assigning correct and minimal IAM permissions

205 lines 23.7 kB
"use strict"; var _a; Object.defineProperty(exports, "__esModule", { value: true }); exports.Policy = void 0; const jsiiDeprecationWarnings = require("../.warnings.jsii.js"); const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); const core_1 = require("@aws-cdk/core"); const iam_generated_1 = require("./iam.generated"); const policy_document_1 = require("./policy-document"); const util_1 = require("./util"); /** * The AWS::IAM::Policy resource associates an IAM policy with IAM users, roles, * or groups. For more information about IAM policies, see [Overview of IAM * Policies](http://docs.aws.amazon.com/IAM/latest/UserGuide/policies_overview.html) * in the IAM User Guide guide. */ class Policy extends core_1.Resource { constructor(scope, id, props = {}) { super(scope, id, { physicalName: props.policyName || // generatePolicyName will take the last 128 characters of the logical id since // policy names are limited to 128. the last 8 chars are a stack-unique hash, so // that shouod be sufficient to ensure uniqueness within a principal. core_1.Lazy.string({ produce: () => util_1.generatePolicyName(scope, resource.logicalId) }), }); /** * The policy document. */ this.document = new policy_document_1.PolicyDocument(); this.roles = new Array(); this.users = new Array(); this.groups = new Array(); this.referenceTaken = false; try { jsiiDeprecationWarnings._aws_cdk_aws_iam_PolicyProps(props); } catch (error) { if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") { Error.captureStackTrace(error, Policy); } throw error; } const self = this; class CfnPolicyConditional extends iam_generated_1.CfnPolicy { /** * This function returns `true` if the CFN resource should be included in * the cloudformation template unless `force` is `true`, if the policy * document is empty, the resource will not be included. */ shouldSynthesize() { return self.force || self.referenceTaken || (!self.document.isEmpty && self.isAttached); } } if (props.document) { this.document = props.document; } const resource = new CfnPolicyConditional(this, 'Resource', { policyDocument: this.document, policyName: this.physicalName, roles: util_1.undefinedIfEmpty(() => this.roles.map(r => r.roleName)), users: util_1.undefinedIfEmpty(() => this.users.map(u => u.userName)), groups: util_1.undefinedIfEmpty(() => this.groups.map(g => g.groupName)), }); this._policyName = this.physicalName; this.force = props.force ?? false; if (props.users) { props.users.forEach(u => this.attachToUser(u)); } if (props.groups) { props.groups.forEach(g => this.attachToGroup(g)); } if (props.roles) { props.roles.forEach(r => this.attachToRole(r)); } if (props.statements) { props.statements.forEach(p => this.addStatements(p)); } } /** * Import a policy in this app based on its name */ static fromPolicyName(scope, id, policyName) { class Import extends core_1.Resource { constructor() { super(...arguments); this.policyName = policyName; } } return new Import(scope, id); } /** * Adds a statement to the policy document. */ addStatements(...statement) { try { jsiiDeprecationWarnings._aws_cdk_aws_iam_PolicyStatement(statement); } catch (error) { if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") { Error.captureStackTrace(error, this.addStatements); } throw error; } this.document.addStatements(...statement); } /** * Attaches this policy to a user. */ attachToUser(user) { try { jsiiDeprecationWarnings._aws_cdk_aws_iam_IUser(user); } catch (error) { if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") { Error.captureStackTrace(error, this.attachToUser); } throw error; } if (this.users.find(u => u === user)) { return; } this.users.push(user); user.attachInlinePolicy(this); } /** * Attaches this policy to a role. */ attachToRole(role) { try { jsiiDeprecationWarnings._aws_cdk_aws_iam_IRole(role); } catch (error) { if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") { Error.captureStackTrace(error, this.attachToRole); } throw error; } if (this.roles.find(r => r === role)) { return; } this.roles.push(role); role.attachInlinePolicy(this); } /** * Attaches this policy to a group. */ attachToGroup(group) { try { jsiiDeprecationWarnings._aws_cdk_aws_iam_IGroup(group); } catch (error) { if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") { Error.captureStackTrace(error, this.attachToGroup); } throw error; } if (this.groups.find(g => g === group)) { return; } this.groups.push(group); group.attachInlinePolicy(this); } /** * The name of this policy. * * @attribute */ get policyName() { this.referenceTaken = true; return this._policyName; } validate() { const result = new Array(); // validate that the policy document is not empty if (this.document.isEmpty) { if (this.force) { result.push('Policy created with force=true is empty. You must add statements to the policy'); } if (!this.force && this.referenceTaken) { result.push('This Policy has been referenced by a resource, so it must contain at least one statement.'); } } // validate that the policy is attached to at least one principal (role, user or group). if (!this.isAttached) { if (this.force) { result.push('Policy created with force=true must be attached to at least one principal: user, group or role'); } if (!this.force && this.referenceTaken) { result.push('This Policy has been referenced by a resource, so it must be attached to at least one user, group or role.'); } } result.push(...this.document.validateForIdentityPolicy()); return result; } /** * Whether the policy resource has been attached to any identity */ get isAttached() { return this.groups.length + this.users.length + this.roles.length > 0; } } exports.Policy = Policy; _a = JSII_RTTI_SYMBOL_1; Policy[_a] = { fqn: "@aws-cdk/aws-iam.Policy", version: "1.204.0" }; //# 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"]}