UNPKG

32 kBJavaScriptView Raw
1"use strict";
2var _a;
3Object.defineProperty(exports, "__esModule", { value: true });
4exports.User = 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_1 = require("./policy");
10const principals_1 = require("./principals");
11const util_1 = require("./util");
12/**
13 * Define a new IAM user
14 */
15class User extends core_1.Resource {
16 constructor(scope, id, props = {}) {
17 super(scope, id, {
18 physicalName: props.userName,
19 });
20 this.grantPrincipal = this;
21 this.principalAccount = this.env.account;
22 this.assumeRoleAction = 'sts:AssumeRole';
23 this.groups = new Array();
24 this.managedPolicies = new Array();
25 this.attachedPolicies = new util_1.AttachedPolicies();
26 try {
27 jsiiDeprecationWarnings._aws_cdk_aws_iam_UserProps(props);
28 }
29 catch (error) {
30 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
31 Error.captureStackTrace(error, User);
32 }
33 throw error;
34 }
35 this.managedPolicies.push(...props.managedPolicies || []);
36 this.permissionsBoundary = props.permissionsBoundary;
37 const user = new iam_generated_1.CfnUser(this, 'Resource', {
38 userName: this.physicalName,
39 groups: util_1.undefinedIfEmpty(() => this.groups),
40 managedPolicyArns: core_1.Lazy.list({ produce: () => this.managedPolicies.map(p => p.managedPolicyArn) }, { omitEmpty: true }),
41 path: props.path,
42 permissionsBoundary: this.permissionsBoundary ? this.permissionsBoundary.managedPolicyArn : undefined,
43 loginProfile: this.parseLoginProfile(props),
44 });
45 this.userName = this.getResourceNameAttribute(user.ref);
46 this.userArn = this.getResourceArnAttribute(user.attrArn, {
47 region: '',
48 service: 'iam',
49 resource: 'user',
50 // Removes leading slash from path
51 resourceName: `${props.path ? props.path.substr(props.path.charAt(0) === '/' ? 1 : 0) : ''}${this.physicalName}`,
52 });
53 this.policyFragment = new principals_1.ArnPrincipal(this.userArn).policyFragment;
54 if (props.groups) {
55 props.groups.forEach(g => this.addToGroup(g));
56 }
57 }
58 /**
59 * Import an existing user given a username.
60 *
61 * @param scope construct scope
62 * @param id construct id
63 * @param userName the username of the existing user to import
64 */
65 static fromUserName(scope, id, userName) {
66 const userArn = core_1.Stack.of(scope).formatArn({
67 service: 'iam',
68 region: '',
69 resource: 'user',
70 resourceName: userName,
71 });
72 return User.fromUserAttributes(scope, id, { userArn });
73 }
74 /**
75 * Import an existing user given a user ARN.
76 *
77 * If the ARN comes from a Token, the User cannot have a path; if so, any attempt
78 * to reference its username will fail.
79 *
80 * @param scope construct scope
81 * @param id construct id
82 * @param userArn the ARN of an existing user to import
83 */
84 static fromUserArn(scope, id, userArn) {
85 return User.fromUserAttributes(scope, id, { userArn });
86 }
87 /**
88 * Import an existing user given user attributes.
89 *
90 * If the ARN comes from a Token, the User cannot have a path; if so, any attempt
91 * to reference its username will fail.
92 *
93 * @param scope construct scope
94 * @param id construct id
95 * @param attrs the attributes of the user to import
96 */
97 static fromUserAttributes(scope, id, attrs) {
98 try {
99 jsiiDeprecationWarnings._aws_cdk_aws_iam_UserAttributes(attrs);
100 }
101 catch (error) {
102 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
103 Error.captureStackTrace(error, this.fromUserAttributes);
104 }
105 throw error;
106 }
107 class Import extends core_1.Resource {
108 constructor() {
109 super(...arguments);
110 this.grantPrincipal = this;
111 this.principalAccount = core_1.Aws.ACCOUNT_ID;
112 // Resource name with path can have multiple elements separated by slash.
113 // Therefore, use element after last slash as userName. Happens to work for Tokens since
114 // they don't have a '/' in them.
115 this.userName = core_1.Arn.extractResourceName(attrs.userArn, 'user').split('/').pop();
116 this.userArn = attrs.userArn;
117 this.assumeRoleAction = 'sts:AssumeRole';
118 this.policyFragment = new principals_1.ArnPrincipal(attrs.userArn).policyFragment;
119 this.attachedPolicies = new util_1.AttachedPolicies();
120 this.groupId = 0;
121 }
122 addToPolicy(statement) {
123 return this.addToPrincipalPolicy(statement).statementAdded;
124 }
125 addToPrincipalPolicy(statement) {
126 if (!this.defaultPolicy) {
127 this.defaultPolicy = new policy_1.Policy(this, 'Policy');
128 this.defaultPolicy.attachToUser(this);
129 }
130 this.defaultPolicy.addStatements(statement);
131 return { statementAdded: true, policyDependable: this.defaultPolicy };
132 }
133 addToGroup(group) {
134 new iam_generated_1.CfnUserToGroupAddition(core_1.Stack.of(group), `${this.userName}Group${this.groupId}`, {
135 groupName: group.groupName,
136 users: [this.userName],
137 });
138 this.groupId += 1;
139 }
140 attachInlinePolicy(policy) {
141 this.attachedPolicies.attach(policy);
142 policy.attachToUser(this);
143 }
144 addManagedPolicy(_policy) {
145 throw new Error('Cannot add managed policy to imported User');
146 }
147 }
148 return new Import(scope, id);
149 }
150 /**
151 * Adds this user to a group.
152 */
153 addToGroup(group) {
154 try {
155 jsiiDeprecationWarnings._aws_cdk_aws_iam_IGroup(group);
156 }
157 catch (error) {
158 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
159 Error.captureStackTrace(error, this.addToGroup);
160 }
161 throw error;
162 }
163 this.groups.push(group.groupName);
164 }
165 /**
166 * Attaches a managed policy to the user.
167 * @param policy The managed policy to attach.
168 */
169 addManagedPolicy(policy) {
170 try {
171 jsiiDeprecationWarnings._aws_cdk_aws_iam_IManagedPolicy(policy);
172 }
173 catch (error) {
174 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
175 Error.captureStackTrace(error, this.addManagedPolicy);
176 }
177 throw error;
178 }
179 if (this.managedPolicies.find(mp => mp === policy)) {
180 return;
181 }
182 this.managedPolicies.push(policy);
183 }
184 /**
185 * Attaches a policy to this user.
186 */
187 attachInlinePolicy(policy) {
188 try {
189 jsiiDeprecationWarnings._aws_cdk_aws_iam_Policy(policy);
190 }
191 catch (error) {
192 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
193 Error.captureStackTrace(error, this.attachInlinePolicy);
194 }
195 throw error;
196 }
197 this.attachedPolicies.attach(policy);
198 policy.attachToUser(this);
199 }
200 /**
201 * Adds an IAM statement to the default policy.
202 *
203 * @returns true
204 */
205 addToPrincipalPolicy(statement) {
206 try {
207 jsiiDeprecationWarnings._aws_cdk_aws_iam_PolicyStatement(statement);
208 }
209 catch (error) {
210 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
211 Error.captureStackTrace(error, this.addToPrincipalPolicy);
212 }
213 throw error;
214 }
215 if (!this.defaultPolicy) {
216 this.defaultPolicy = new policy_1.Policy(this, 'DefaultPolicy');
217 this.defaultPolicy.attachToUser(this);
218 }
219 this.defaultPolicy.addStatements(statement);
220 return { statementAdded: true, policyDependable: this.defaultPolicy };
221 }
222 addToPolicy(statement) {
223 try {
224 jsiiDeprecationWarnings._aws_cdk_aws_iam_PolicyStatement(statement);
225 }
226 catch (error) {
227 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
228 Error.captureStackTrace(error, this.addToPolicy);
229 }
230 throw error;
231 }
232 return this.addToPrincipalPolicy(statement).statementAdded;
233 }
234 parseLoginProfile(props) {
235 if (props.password) {
236 return {
237 password: props.password.unsafeUnwrap(),
238 passwordResetRequired: props.passwordResetRequired,
239 };
240 }
241 if (props.passwordResetRequired) {
242 throw new Error('Cannot set "passwordResetRequired" without specifying "initialPassword"');
243 }
244 return undefined; // no console access
245 }
246}
247exports.User = User;
248_a = JSII_RTTI_SYMBOL_1;
249User[_a] = { fqn: "@aws-cdk/aws-iam.User", version: "1.190.0" };
250//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"user.js","sourceRoot":"","sources":["user.ts"],"names":[],"mappings":"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","sourcesContent":["import { Arn, Aws, Lazy, Resource, SecretValue, Stack } from '@aws-cdk/core';\nimport { Construct } from 'constructs';\nimport { IGroup } from './group';\nimport { CfnUser, CfnUserToGroupAddition } from './iam.generated';\nimport { IIdentity } from './identity-base';\nimport { IManagedPolicy } from './managed-policy';\nimport { Policy } from './policy';\nimport { PolicyStatement } from './policy-statement';\nimport { AddToPrincipalPolicyResult, ArnPrincipal, IPrincipal, PrincipalPolicyFragment } from './principals';\nimport { AttachedPolicies, undefinedIfEmpty } from './util';\n\n/**\n * Represents an IAM user\n *\n * @see https://docs.aws.amazon.com/IAM/latest/UserGuide/id_users.html\n */\nexport interface IUser extends IIdentity {\n  /**\n   * The user's name\n   * @attribute\n   */\n  readonly userName: string;\n\n  /**\n   * The user's ARN\n   * @attribute\n   */\n  readonly userArn: string;\n\n  /**\n   * Adds this user to a group.\n   */\n  addToGroup(group: IGroup): void;\n}\n\n/**\n * Properties for defining an IAM user\n */\nexport interface UserProps {\n  /**\n   * Groups to add this user to. You can also use `addToGroup` to add this\n   * user to a group.\n   *\n   * @default - No groups.\n   */\n  readonly groups?: IGroup[];\n\n  /**\n   * A list of managed policies associated with this role.\n   *\n   * You can add managed policies later using\n   * `addManagedPolicy(ManagedPolicy.fromAwsManagedPolicyName(policyName))`.\n   *\n   * @default - No managed policies.\n   */\n  readonly managedPolicies?: IManagedPolicy[];\n\n  /**\n   * The path for the user name. For more information about paths, see IAM\n   * Identifiers in the IAM User Guide.\n   *\n   * @default /\n   */\n  readonly path?: string;\n\n  /**\n   * AWS supports permissions boundaries for IAM entities (users or roles).\n   * A permissions boundary is an advanced feature for using a managed policy\n   * to set the maximum permissions that an identity-based policy can grant to\n   * an IAM entity. An entity's permissions boundary allows it to perform only\n   * the actions that are allowed by both its identity-based policies and its\n   * permissions boundaries.\n   *\n   * @link https://docs.aws.amazon.com/AWSCloudFormation/latest/UserGuide/aws-resource-iam-role.html#cfn-iam-role-permissionsboundary\n   * @link https://docs.aws.amazon.com/IAM/latest/UserGuide/access_policies_boundaries.html\n   *\n   * @default - No permissions boundary.\n   */\n  readonly permissionsBoundary?: IManagedPolicy;\n\n  /**\n   * A name for the IAM user. For valid values, see the UserName parameter for\n   * the CreateUser action in the IAM API Reference. If you don't specify a\n   * name, AWS CloudFormation generates a unique physical ID and uses that ID\n   * for the user name.\n   *\n   * If you specify a name, you cannot perform updates that require\n   * replacement of this resource. You can perform updates that require no or\n   * some interruption. If you must replace the resource, specify a new name.\n   *\n   * If you specify a name, you must specify the CAPABILITY_NAMED_IAM value to\n   * acknowledge your template's capabilities. For more information, see\n   * Acknowledging IAM Resources in AWS CloudFormation Templates.\n   *\n   * @default - Generated by CloudFormation (recommended)\n   */\n  readonly userName?: string;\n\n  /**\n   * The password for the user. This is required so the user can access the\n   * AWS Management Console.\n   *\n   * You can use `SecretValue.unsafePlainText` to specify a password in plain text or\n   * use `secretsmanager.Secret.fromSecretAttributes` to reference a secret in\n   * Secrets Manager.\n   *\n   * @default - User won't be able to access the management console without a password.\n   */\n  readonly password?: SecretValue;\n\n  /**\n   * Specifies whether the user is required to set a new password the next\n   * time the user logs in to the AWS Management Console.\n   *\n   * If this is set to 'true', you must also specify \"initialPassword\".\n   *\n   * @default false\n   */\n  readonly passwordResetRequired?: boolean;\n}\n\n/**\n * Represents a user defined outside of this stack.\n */\nexport interface UserAttributes {\n  /**\n   * The ARN of the user.\n   *\n   * Format: arn:<partition>:iam::<account-id>:user/<user-name-with-path>\n   */\n  readonly userArn: string;\n}\n\n/**\n * Define a new IAM user\n */\nexport class User extends Resource implements IIdentity, IUser {\n  /**\n   * Import an existing user given a username.\n   *\n   * @param scope construct scope\n   * @param id construct id\n   * @param userName the username of the existing user to import\n   */\n  public static fromUserName(scope: Construct, id: string, userName: string): IUser {\n    const userArn = Stack.of(scope).formatArn({\n      service: 'iam',\n      region: '',\n      resource: 'user',\n      resourceName: userName,\n    });\n\n    return User.fromUserAttributes(scope, id, { userArn });\n  }\n\n  /**\n   * Import an existing user given a user ARN.\n   *\n   * If the ARN comes from a Token, the User cannot have a path; if so, any attempt\n   * to reference its username will fail.\n   *\n   * @param scope construct scope\n   * @param id construct id\n   * @param userArn the ARN of an existing user to import\n   */\n  public static fromUserArn(scope: Construct, id: string, userArn: string): IUser {\n    return User.fromUserAttributes(scope, id, { userArn });\n  }\n\n  /**\n   * Import an existing user given user attributes.\n   *\n   * If the ARN comes from a Token, the User cannot have a path; if so, any attempt\n   * to reference its username will fail.\n   *\n   * @param scope construct scope\n   * @param id construct id\n   * @param attrs the attributes of the user to import\n   */\n  public static fromUserAttributes(scope: Construct, id: string, attrs: UserAttributes): IUser {\n    class Import extends Resource implements IUser {\n      public readonly grantPrincipal: IPrincipal = this;\n      public readonly principalAccount = Aws.ACCOUNT_ID;\n      // Resource name with path can have multiple elements separated by slash.\n      // Therefore, use element after last slash as userName. Happens to work for Tokens since\n      // they don't have a '/' in them.\n      public readonly userName: string = Arn.extractResourceName(attrs.userArn, 'user').split('/').pop()!;\n      public readonly userArn: string = attrs.userArn;\n      public readonly assumeRoleAction: string = 'sts:AssumeRole';\n      public readonly policyFragment: PrincipalPolicyFragment = new ArnPrincipal(attrs.userArn).policyFragment;\n      private readonly attachedPolicies = new AttachedPolicies();\n      private defaultPolicy?: Policy;\n      private groupId = 0;\n\n      public addToPolicy(statement: PolicyStatement): boolean {\n        return this.addToPrincipalPolicy(statement).statementAdded;\n      }\n\n      public addToPrincipalPolicy(statement: PolicyStatement): AddToPrincipalPolicyResult {\n        if (!this.defaultPolicy) {\n          this.defaultPolicy = new Policy(this, 'Policy');\n          this.defaultPolicy.attachToUser(this);\n        }\n        this.defaultPolicy.addStatements(statement);\n        return { statementAdded: true, policyDependable: this.defaultPolicy };\n      }\n\n      public addToGroup(group: IGroup): void {\n        new CfnUserToGroupAddition(Stack.of(group), `${this.userName}Group${this.groupId}`, {\n          groupName: group.groupName,\n          users: [this.userName],\n        });\n        this.groupId += 1;\n      }\n\n      public attachInlinePolicy(policy: Policy): void {\n        this.attachedPolicies.attach(policy);\n        policy.attachToUser(this);\n      }\n\n      public addManagedPolicy(_policy: IManagedPolicy): void {\n        throw new Error('Cannot add managed policy to imported User');\n      }\n    }\n\n    return new Import(scope, id);\n  }\n\n  public readonly grantPrincipal: IPrincipal = this;\n  public readonly principalAccount: string | undefined = this.env.account;\n  public readonly assumeRoleAction: string = 'sts:AssumeRole';\n\n  /**\n   * An attribute that represents the user name.\n   * @attribute\n   */\n  public readonly userName: string;\n\n  /**\n   * An attribute that represents the user's ARN.\n   * @attribute\n   */\n  public readonly userArn: string;\n\n  /**\n   * Returns the permissions boundary attached  to this user\n   */\n  public readonly permissionsBoundary?: IManagedPolicy;\n\n  public readonly policyFragment: PrincipalPolicyFragment;\n\n  private readonly groups = new Array<any>();\n  private readonly managedPolicies = new Array<IManagedPolicy>();\n  private readonly attachedPolicies = new AttachedPolicies();\n  private defaultPolicy?: Policy;\n\n  constructor(scope: Construct, id: string, props: UserProps = {}) {\n    super(scope, id, {\n      physicalName: props.userName,\n    });\n\n    this.managedPolicies.push(...props.managedPolicies || []);\n    this.permissionsBoundary = props.permissionsBoundary;\n\n    const user = new CfnUser(this, 'Resource', {\n      userName: this.physicalName,\n      groups: undefinedIfEmpty(() => this.groups),\n      managedPolicyArns: Lazy.list({ produce: () => this.managedPolicies.map(p => p.managedPolicyArn) }, { omitEmpty: true }),\n      path: props.path,\n      permissionsBoundary: this.permissionsBoundary ? this.permissionsBoundary.managedPolicyArn : undefined,\n      loginProfile: this.parseLoginProfile(props),\n    });\n\n    this.userName = this.getResourceNameAttribute(user.ref);\n    this.userArn = this.getResourceArnAttribute(user.attrArn, {\n      region: '', // IAM is global in each partition\n      service: 'iam',\n      resource: 'user',\n      // Removes leading slash from path\n      resourceName: `${props.path ? props.path.substr(props.path.charAt(0) === '/' ? 1 : 0) : ''}${this.physicalName}`,\n    });\n\n    this.policyFragment = new ArnPrincipal(this.userArn).policyFragment;\n\n    if (props.groups) {\n      props.groups.forEach(g => this.addToGroup(g));\n    }\n  }\n\n  /**\n   * Adds this user to a group.\n   */\n  public addToGroup(group: IGroup) {\n    this.groups.push(group.groupName);\n  }\n\n  /**\n   * Attaches a managed policy to the user.\n   * @param policy The managed policy to attach.\n   */\n  public addManagedPolicy(policy: IManagedPolicy) {\n    if (this.managedPolicies.find(mp => mp === policy)) { return; }\n    this.managedPolicies.push(policy);\n  }\n\n  /**\n   * Attaches a policy to this user.\n   */\n  public attachInlinePolicy(policy: Policy) {\n    this.attachedPolicies.attach(policy);\n    policy.attachToUser(this);\n  }\n\n  /**\n   * Adds an IAM statement to the default policy.\n   *\n   * @returns true\n   */\n  public addToPrincipalPolicy(statement: PolicyStatement): AddToPrincipalPolicyResult {\n    if (!this.defaultPolicy) {\n      this.defaultPolicy = new Policy(this, 'DefaultPolicy');\n      this.defaultPolicy.attachToUser(this);\n    }\n\n    this.defaultPolicy.addStatements(statement);\n    return { statementAdded: true, policyDependable: this.defaultPolicy };\n  }\n\n  public addToPolicy(statement: PolicyStatement): boolean {\n    return this.addToPrincipalPolicy(statement).statementAdded;\n  }\n\n  private parseLoginProfile(props: UserProps): CfnUser.LoginProfileProperty | undefined {\n    if (props.password) {\n      return {\n        password: props.password.unsafeUnwrap(), // Safe usage\n        passwordResetRequired: props.passwordResetRequired,\n      };\n    }\n\n    if (props.passwordResetRequired) {\n      throw new Error('Cannot set \"passwordResetRequired\" without specifying \"initialPassword\"');\n    }\n\n    return undefined; // no console access\n  }\n}\n"]}
\No newline at end of file