1 | ;
|
2 | var _a;
|
3 | Object.defineProperty(exports, "__esModule", { value: true });
|
4 | exports.User = 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_1 = require("./policy");
|
10 | const principals_1 = require("./principals");
|
11 | const util_1 = require("./util");
|
12 | /**
|
13 | * Define a new IAM user
|
14 | */
|
15 | class 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 | }
|
247 | exports.User = User;
|
248 | _a = JSII_RTTI_SYMBOL_1;
|
249 | User[_a] = { fqn: "@aws-cdk/aws-iam.User", version: "1.161.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 |