1 | ;
|
2 | var _a;
|
3 | Object.defineProperty(exports, "__esModule", { value: true });
|
4 | exports.Role = 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 constructs_1 = require("constructs");
|
9 | const grant_1 = require("./grant");
|
10 | const iam_generated_1 = require("./iam.generated");
|
11 | const policy_1 = require("./policy");
|
12 | const policy_document_1 = require("./policy-document");
|
13 | const principals_1 = require("./principals");
|
14 | const assume_role_policy_1 = require("./private/assume-role-policy");
|
15 | const immutable_role_1 = require("./private/immutable-role");
|
16 | const policydoc_adapter_1 = require("./private/policydoc-adapter");
|
17 | const util_1 = require("./util");
|
18 | /**
|
19 | * IAM Role
|
20 | *
|
21 | * Defines an IAM role. The role is created with an assume policy document associated with
|
22 | * the specified AWS service principal defined in `serviceAssumeRole`.
|
23 | */
|
24 | class Role extends core_1.Resource {
|
25 | constructor(scope, id, props) {
|
26 | var _b;
|
27 | super(scope, id, {
|
28 | physicalName: props.roleName,
|
29 | });
|
30 | this.grantPrincipal = this;
|
31 | this.principalAccount = this.env.account;
|
32 | this.assumeRoleAction = 'sts:AssumeRole';
|
33 | this.managedPolicies = [];
|
34 | this.attachedPolicies = new util_1.AttachedPolicies();
|
35 | try {
|
36 | jsiiDeprecationWarnings._aws_cdk_aws_iam_RoleProps(props);
|
37 | }
|
38 | catch (error) {
|
39 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
40 | Error.captureStackTrace(error, this.constructor);
|
41 | }
|
42 | throw error;
|
43 | }
|
44 | const externalIds = props.externalIds || [];
|
45 | if (props.externalId) {
|
46 | externalIds.push(props.externalId);
|
47 | }
|
48 | this.assumeRolePolicy = createAssumeRolePolicy(props.assumedBy, externalIds);
|
49 | this.managedPolicies.push(...props.managedPolicies || []);
|
50 | this.inlinePolicies = props.inlinePolicies || {};
|
51 | this.permissionsBoundary = props.permissionsBoundary;
|
52 | const maxSessionDuration = props.maxSessionDuration && props.maxSessionDuration.toSeconds();
|
53 | validateMaxSessionDuration(maxSessionDuration);
|
54 | const description = (props.description && ((_b = props.description) === null || _b === void 0 ? void 0 : _b.length) > 0) ? props.description : undefined;
|
55 | if (description && description.length > 1000) {
|
56 | throw new Error('Role description must be no longer than 1000 characters.');
|
57 | }
|
58 | const role = new iam_generated_1.CfnRole(this, 'Resource', {
|
59 | assumeRolePolicyDocument: this.assumeRolePolicy,
|
60 | managedPolicyArns: util_1.UniqueStringSet.from(() => this.managedPolicies.map(p => p.managedPolicyArn)),
|
61 | policies: _flatten(this.inlinePolicies),
|
62 | path: props.path,
|
63 | permissionsBoundary: this.permissionsBoundary ? this.permissionsBoundary.managedPolicyArn : undefined,
|
64 | roleName: this.physicalName,
|
65 | maxSessionDuration,
|
66 | description,
|
67 | });
|
68 | this.roleId = role.attrRoleId;
|
69 | this.roleArn = this.getResourceArnAttribute(role.attrArn, {
|
70 | region: '',
|
71 | service: 'iam',
|
72 | resource: 'role',
|
73 | // Removes leading slash from path
|
74 | resourceName: `${props.path ? props.path.substr(props.path.charAt(0) === '/' ? 1 : 0) : ''}${this.physicalName}`,
|
75 | });
|
76 | this.roleName = this.getResourceNameAttribute(role.ref);
|
77 | this.policyFragment = new principals_1.ArnPrincipal(this.roleArn).policyFragment;
|
78 | function _flatten(policies) {
|
79 | if (policies == null || Object.keys(policies).length === 0) {
|
80 | return undefined;
|
81 | }
|
82 | const result = new Array();
|
83 | for (const policyName of Object.keys(policies)) {
|
84 | const policyDocument = policies[policyName];
|
85 | result.push({ policyName, policyDocument });
|
86 | }
|
87 | return result;
|
88 | }
|
89 | }
|
90 | /**
|
91 | * Import an external role by ARN.
|
92 | *
|
93 | * If the imported Role ARN is a Token (such as a
|
94 | * `CfnParameter.valueAsString` or a `Fn.importValue()`) *and* the referenced
|
95 | * role has a `path` (like `arn:...:role/AdminRoles/Alice`), the
|
96 | * `roleName` property will not resolve to the correct value. Instead it
|
97 | * will resolve to the first path component. We unfortunately cannot express
|
98 | * the correct calculation of the full path name as a CloudFormation
|
99 | * expression. In this scenario the Role ARN should be supplied without the
|
100 | * `path` in order to resolve the correct role resource.
|
101 | *
|
102 | * @param scope construct scope
|
103 | * @param id construct id
|
104 | * @param roleArn the ARN of the role to import
|
105 | * @param options allow customizing the behavior of the returned role
|
106 | */
|
107 | static fromRoleArn(scope, id, roleArn, options = {}) {
|
108 | var _b;
|
109 | try {
|
110 | jsiiDeprecationWarnings._aws_cdk_aws_iam_FromRoleArnOptions(options);
|
111 | }
|
112 | catch (error) {
|
113 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
114 | Error.captureStackTrace(error, this.fromRoleArn);
|
115 | }
|
116 | throw error;
|
117 | }
|
118 | const scopeStack = core_1.Stack.of(scope);
|
119 | const parsedArn = scopeStack.splitArn(roleArn, core_1.ArnFormat.SLASH_RESOURCE_NAME);
|
120 | const resourceName = parsedArn.resourceName;
|
121 | const roleAccount = parsedArn.account;
|
122 | // service roles have an ARN like 'arn:aws:iam::<account>:role/service-role/<roleName>'
|
123 | // or 'arn:aws:iam::<account>:role/service-role/servicename.amazonaws.com/service-role/<roleName>'
|
124 | // we want to support these as well, so we just use the element after the last slash as role name
|
125 | const roleName = resourceName.split('/').pop();
|
126 | class Import extends core_1.Resource {
|
127 | constructor(_scope, _id) {
|
128 | super(_scope, _id, {
|
129 | account: roleAccount,
|
130 | });
|
131 | this.grantPrincipal = this;
|
132 | this.principalAccount = roleAccount;
|
133 | this.assumeRoleAction = 'sts:AssumeRole';
|
134 | this.policyFragment = new principals_1.ArnPrincipal(roleArn).policyFragment;
|
135 | this.roleArn = roleArn;
|
136 | this.roleName = roleName;
|
137 | this.attachedPolicies = new util_1.AttachedPolicies();
|
138 | }
|
139 | addToPolicy(statement) {
|
140 | return this.addToPrincipalPolicy(statement).statementAdded;
|
141 | }
|
142 | addToPrincipalPolicy(statement) {
|
143 | if (!this.defaultPolicy) {
|
144 | this.defaultPolicy = new policy_1.Policy(this, 'Policy');
|
145 | this.attachInlinePolicy(this.defaultPolicy);
|
146 | }
|
147 | this.defaultPolicy.addStatements(statement);
|
148 | return { statementAdded: true, policyDependable: this.defaultPolicy };
|
149 | }
|
150 | attachInlinePolicy(policy) {
|
151 | const thisAndPolicyAccountComparison = core_1.Token.compareStrings(this.env.account, policy.env.account);
|
152 | const equalOrAnyUnresolved = thisAndPolicyAccountComparison === core_1.TokenComparison.SAME ||
|
153 | thisAndPolicyAccountComparison === core_1.TokenComparison.BOTH_UNRESOLVED ||
|
154 | thisAndPolicyAccountComparison === core_1.TokenComparison.ONE_UNRESOLVED;
|
155 | if (equalOrAnyUnresolved) {
|
156 | this.attachedPolicies.attach(policy);
|
157 | policy.attachToRole(this);
|
158 | }
|
159 | }
|
160 | addManagedPolicy(_policy) {
|
161 | // FIXME: Add warning that we're ignoring this
|
162 | }
|
163 | /**
|
164 | * Grant permissions to the given principal to pass this role.
|
165 | */
|
166 | grantPassRole(identity) {
|
167 | return this.grant(identity, 'iam:PassRole');
|
168 | }
|
169 | /**
|
170 | * Grant the actions defined in actions to the identity Principal on this resource.
|
171 | */
|
172 | grant(grantee, ...actions) {
|
173 | return grant_1.Grant.addToPrincipal({
|
174 | grantee,
|
175 | actions,
|
176 | resourceArns: [this.roleArn],
|
177 | scope: this,
|
178 | });
|
179 | }
|
180 | }
|
181 | if (options.addGrantsToResources !== undefined && options.mutable !== false) {
|
182 | throw new Error('\'addGrantsToResources\' can only be passed if \'mutable: false\'');
|
183 | }
|
184 | const importedRole = new Import(scope, id);
|
185 | const roleArnAndScopeStackAccountComparison = core_1.Token.compareStrings(importedRole.env.account, scopeStack.account);
|
186 | const equalOrAnyUnresolved = roleArnAndScopeStackAccountComparison === core_1.TokenComparison.SAME ||
|
187 | roleArnAndScopeStackAccountComparison === core_1.TokenComparison.BOTH_UNRESOLVED ||
|
188 | roleArnAndScopeStackAccountComparison === core_1.TokenComparison.ONE_UNRESOLVED;
|
189 | // we only return an immutable Role if both accounts were explicitly provided, and different
|
190 | return options.mutable !== false && equalOrAnyUnresolved
|
191 | ? importedRole
|
192 | : new immutable_role_1.ImmutableRole(scope, `ImmutableRole${id}`, importedRole, (_b = options.addGrantsToResources) !== null && _b !== void 0 ? _b : false);
|
193 | }
|
194 | /**
|
195 | * Import an external role by name.
|
196 | *
|
197 | * The imported role is assumed to exist in the same account as the account
|
198 | * the scope's containing Stack is being deployed to.
|
199 | */
|
200 | static fromRoleName(scope, id, roleName) {
|
201 | return Role.fromRoleArn(scope, id, core_1.Stack.of(scope).formatArn({
|
202 | region: '',
|
203 | service: 'iam',
|
204 | resource: 'role',
|
205 | resourceName: roleName,
|
206 | }));
|
207 | }
|
208 | /**
|
209 | * Adds a permission to the role's default policy document.
|
210 | * If there is no default policy attached to this role, it will be created.
|
211 | * @param statement The permission statement to add to the policy document
|
212 | */
|
213 | addToPrincipalPolicy(statement) {
|
214 | try {
|
215 | jsiiDeprecationWarnings._aws_cdk_aws_iam_PolicyStatement(statement);
|
216 | }
|
217 | catch (error) {
|
218 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
219 | Error.captureStackTrace(error, this.addToPrincipalPolicy);
|
220 | }
|
221 | throw error;
|
222 | }
|
223 | if (!this.defaultPolicy) {
|
224 | this.defaultPolicy = new policy_1.Policy(this, 'DefaultPolicy');
|
225 | this.attachInlinePolicy(this.defaultPolicy);
|
226 | }
|
227 | this.defaultPolicy.addStatements(statement);
|
228 | return { statementAdded: true, policyDependable: this.defaultPolicy };
|
229 | }
|
230 | addToPolicy(statement) {
|
231 | try {
|
232 | jsiiDeprecationWarnings._aws_cdk_aws_iam_PolicyStatement(statement);
|
233 | }
|
234 | catch (error) {
|
235 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
236 | Error.captureStackTrace(error, this.addToPolicy);
|
237 | }
|
238 | throw error;
|
239 | }
|
240 | return this.addToPrincipalPolicy(statement).statementAdded;
|
241 | }
|
242 | /**
|
243 | * Attaches a managed policy to this role.
|
244 | * @param policy The the managed policy to attach.
|
245 | */
|
246 | addManagedPolicy(policy) {
|
247 | try {
|
248 | jsiiDeprecationWarnings._aws_cdk_aws_iam_IManagedPolicy(policy);
|
249 | }
|
250 | catch (error) {
|
251 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
252 | Error.captureStackTrace(error, this.addManagedPolicy);
|
253 | }
|
254 | throw error;
|
255 | }
|
256 | if (this.managedPolicies.find(mp => mp === policy)) {
|
257 | return;
|
258 | }
|
259 | this.managedPolicies.push(policy);
|
260 | }
|
261 | /**
|
262 | * Attaches a policy to this role.
|
263 | * @param policy The policy to attach
|
264 | */
|
265 | attachInlinePolicy(policy) {
|
266 | try {
|
267 | jsiiDeprecationWarnings._aws_cdk_aws_iam_Policy(policy);
|
268 | }
|
269 | catch (error) {
|
270 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
271 | Error.captureStackTrace(error, this.attachInlinePolicy);
|
272 | }
|
273 | throw error;
|
274 | }
|
275 | this.attachedPolicies.attach(policy);
|
276 | policy.attachToRole(this);
|
277 | }
|
278 | /**
|
279 | * Grant the actions defined in actions to the identity Principal on this resource.
|
280 | */
|
281 | grant(grantee, ...actions) {
|
282 | try {
|
283 | jsiiDeprecationWarnings._aws_cdk_aws_iam_IPrincipal(grantee);
|
284 | }
|
285 | catch (error) {
|
286 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
287 | Error.captureStackTrace(error, this.grant);
|
288 | }
|
289 | throw error;
|
290 | }
|
291 | return grant_1.Grant.addToPrincipal({
|
292 | grantee,
|
293 | actions,
|
294 | resourceArns: [this.roleArn],
|
295 | scope: this,
|
296 | });
|
297 | }
|
298 | /**
|
299 | * Grant permissions to the given principal to pass this role.
|
300 | */
|
301 | grantPassRole(identity) {
|
302 | try {
|
303 | jsiiDeprecationWarnings._aws_cdk_aws_iam_IPrincipal(identity);
|
304 | }
|
305 | catch (error) {
|
306 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
307 | Error.captureStackTrace(error, this.grantPassRole);
|
308 | }
|
309 | throw error;
|
310 | }
|
311 | return this.grant(identity, 'iam:PassRole');
|
312 | }
|
313 | /**
|
314 | * Return a copy of this Role object whose Policies will not be updated
|
315 | *
|
316 | * Use the object returned by this method if you want this Role to be used by
|
317 | * a construct without it automatically updating the Role's Policies.
|
318 | *
|
319 | * If you do, you are responsible for adding the correct statements to the
|
320 | * Role's policies yourself.
|
321 | */
|
322 | withoutPolicyUpdates(options = {}) {
|
323 | var _b;
|
324 | try {
|
325 | jsiiDeprecationWarnings._aws_cdk_aws_iam_WithoutPolicyUpdatesOptions(options);
|
326 | }
|
327 | catch (error) {
|
328 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
329 | Error.captureStackTrace(error, this.withoutPolicyUpdates);
|
330 | }
|
331 | throw error;
|
332 | }
|
333 | if (!this.immutableRole) {
|
334 | this.immutableRole = new immutable_role_1.ImmutableRole(constructs_1.Node.of(this).scope, `ImmutableRole${this.node.id}`, this, (_b = options.addGrantsToResources) !== null && _b !== void 0 ? _b : false);
|
335 | }
|
336 | return this.immutableRole;
|
337 | }
|
338 | validate() {
|
339 | var _b;
|
340 | const errors = super.validate();
|
341 | errors.push(...((_b = this.assumeRolePolicy) === null || _b === void 0 ? void 0 : _b.validateForResourcePolicy()) || []);
|
342 | for (const policy of Object.values(this.inlinePolicies)) {
|
343 | errors.push(...policy.validateForIdentityPolicy());
|
344 | }
|
345 | return errors;
|
346 | }
|
347 | }
|
348 | exports.Role = Role;
|
349 | _a = JSII_RTTI_SYMBOL_1;
|
350 | Role[_a] = { fqn: "@aws-cdk/aws-iam.Role", version: "1.156.1" };
|
351 | function createAssumeRolePolicy(principal, externalIds) {
|
352 | const actualDoc = new policy_document_1.PolicyDocument();
|
353 | // If requested, add externalIds to every statement added to this doc
|
354 | const addDoc = externalIds.length === 0
|
355 | ? actualDoc
|
356 | : new policydoc_adapter_1.MutatingPolicyDocumentAdapter(actualDoc, (statement) => {
|
357 | statement.addCondition('StringEquals', {
|
358 | 'sts:ExternalId': externalIds.length === 1 ? externalIds[0] : externalIds,
|
359 | });
|
360 | return statement;
|
361 | });
|
362 | assume_role_policy_1.defaultAddPrincipalToAssumeRole(principal, addDoc);
|
363 | return actualDoc;
|
364 | }
|
365 | function validateMaxSessionDuration(duration) {
|
366 | if (duration === undefined) {
|
367 | return;
|
368 | }
|
369 | if (duration < 3600 || duration > 43200) {
|
370 | throw new Error(`maxSessionDuration is set to ${duration}, but must be >= 3600sec (1hr) and <= 43200sec (12hrs)`);
|
371 | }
|
372 | }
|
373 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"role.js","sourceRoot":"","sources":["role.ts"],"names":[],"mappings":"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","sourcesContent":["import { ArnFormat, Duration, Resource, Stack, Token, TokenComparison } from '@aws-cdk/core';\nimport { Construct, Node } from 'constructs';\nimport { Grant } from './grant';\nimport { CfnRole } from './iam.generated';\nimport { IIdentity } from './identity-base';\nimport { IManagedPolicy } from './managed-policy';\nimport { Policy } from './policy';\nimport { PolicyDocument } from './policy-document';\nimport { PolicyStatement } from './policy-statement';\nimport { AddToPrincipalPolicyResult, ArnPrincipal, IPrincipal, PrincipalPolicyFragment } from './principals';\nimport { defaultAddPrincipalToAssumeRole } from './private/assume-role-policy';\nimport { ImmutableRole } from './private/immutable-role';\nimport { MutatingPolicyDocumentAdapter } from './private/policydoc-adapter';\nimport { AttachedPolicies, UniqueStringSet } from './util';\n\n/**\n * Properties for defining an IAM Role\n */\nexport interface RoleProps {\n  /**\n   * The IAM principal (i.e. `new ServicePrincipal('sns.amazonaws.com')`)\n   * which can assume this role.\n   *\n   * You can later modify the assume role policy document by accessing it via\n   * the `assumeRolePolicy` property.\n   */\n  readonly assumedBy: IPrincipal;\n\n  /**\n   * ID that the role assumer needs to provide when assuming this role\n   *\n   * If the configured and provided external IDs do not match, the\n   * AssumeRole operation will fail.\n   *\n   * @deprecated see {@link externalIds}\n   *\n   * @default No external ID required\n   */\n  readonly externalId?: string;\n\n  /**\n   * List of IDs that the role assumer needs to provide one of when assuming this role\n   *\n   * If the configured and provided external IDs do not match, the\n   * AssumeRole operation will fail.\n   *\n   * @default No external ID required\n   */\n  readonly externalIds?: string[];\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   * A list of named policies to inline into this role. These policies will be\n   * created with the role, whereas those added by ``addToPolicy`` are added\n   * using a separate CloudFormation resource (allowing a way around circular\n   * dependencies that could otherwise be introduced).\n   *\n   * @default - No policy is inlined in the Role resource.\n   */\n  readonly inlinePolicies?: { [name: string]: PolicyDocument };\n\n  /**\n   * The path associated with this role. For information about IAM paths, see\n   * Friendly Names and Paths in 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 role. For valid values, see the RoleName parameter for\n   * the CreateRole action in the IAM API Reference.\n   *\n   * IMPORTANT: 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 - AWS CloudFormation generates a unique physical ID and uses that ID\n   * for the role name.\n   */\n  readonly roleName?: string;\n\n  /**\n   * The maximum session duration that you want to set for the specified role.\n   * This setting can have a value from 1 hour (3600sec) to 12 (43200sec) hours.\n   *\n   * Anyone who assumes the role from the AWS CLI or API can use the\n   * DurationSeconds API parameter or the duration-seconds CLI parameter to\n   * request a longer session. The MaxSessionDuration setting determines the\n   * maximum duration that can be requested using the DurationSeconds\n   * parameter.\n   *\n   * If users don't specify a value for the DurationSeconds parameter, their\n   * security credentials are valid for one hour by default. This applies when\n   * you use the AssumeRole* API operations or the assume-role* CLI operations\n   * but does not apply when you use those operations to create a console URL.\n   *\n   * @link https://docs.aws.amazon.com/IAM/latest/UserGuide/id_roles_use.html\n   *\n   * @default Duration.hours(1)\n   */\n  readonly maxSessionDuration?: Duration;\n\n  /**\n   * A description of the role. It can be up to 1000 characters long.\n   *\n   * @default - No description.\n   */\n  readonly description?: string;\n}\n\n/**\n * Options allowing customizing the behavior of {@link Role.fromRoleArn}.\n */\nexport interface FromRoleArnOptions {\n  /**\n   * Whether the imported role can be modified by attaching policy resources to it.\n   *\n   * @default true\n   */\n  readonly mutable?: boolean;\n\n  /**\n   * For immutable roles: add grants to resources instead of dropping them\n   *\n   * If this is `false` or not specified, grant permissions added to this role are ignored.\n   * It is your own responsibility to make sure the role has the required permissions.\n   *\n   * If this is `true`, any grant permissions will be added to the resource instead.\n   *\n   * @default false\n   */\n  readonly addGrantsToResources?: boolean;\n}\n\n/**\n * IAM Role\n *\n * Defines an IAM role. The role is created with an assume policy document associated with\n * the specified AWS service principal defined in `serviceAssumeRole`.\n */\nexport class Role extends Resource implements IRole {\n  /**\n   * Import an external role by ARN.\n   *\n   * If the imported Role ARN is a Token (such as a\n   * `CfnParameter.valueAsString` or a `Fn.importValue()`) *and* the referenced\n   * role has a `path` (like `arn:...:role/AdminRoles/Alice`), the\n   * `roleName` property will not resolve to the correct value. Instead it\n   * will resolve to the first path component. We unfortunately cannot express\n   * the correct calculation of the full path name as a CloudFormation\n   * expression. In this scenario the Role ARN should be supplied without the\n   * `path` in order to resolve the correct role resource.\n   *\n   * @param scope construct scope\n   * @param id construct id\n   * @param roleArn the ARN of the role to import\n   * @param options allow customizing the behavior of the returned role\n   */\n  public static fromRoleArn(scope: Construct, id: string, roleArn: string, options: FromRoleArnOptions = {}): IRole {\n    const scopeStack = Stack.of(scope);\n    const parsedArn = scopeStack.splitArn(roleArn, ArnFormat.SLASH_RESOURCE_NAME);\n    const resourceName = parsedArn.resourceName!;\n    const roleAccount = parsedArn.account;\n    // service roles have an ARN like 'arn:aws:iam::<account>:role/service-role/<roleName>'\n    // or 'arn:aws:iam::<account>:role/service-role/servicename.amazonaws.com/service-role/<roleName>'\n    // we want to support these as well, so we just use the element after the last slash as role name\n    const roleName = resourceName.split('/').pop()!;\n\n    class Import extends Resource implements IRole {\n      public readonly grantPrincipal: IPrincipal = this;\n      public readonly principalAccount = roleAccount;\n      public readonly assumeRoleAction: string = 'sts:AssumeRole';\n      public readonly policyFragment = new ArnPrincipal(roleArn).policyFragment;\n      public readonly roleArn = roleArn;\n      public readonly roleName = roleName;\n      private readonly attachedPolicies = new AttachedPolicies();\n      private defaultPolicy?: Policy;\n\n      constructor(_scope: Construct, _id: string) {\n        super(_scope, _id, {\n          account: roleAccount,\n        });\n      }\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.attachInlinePolicy(this.defaultPolicy);\n        }\n        this.defaultPolicy.addStatements(statement);\n        return { statementAdded: true, policyDependable: this.defaultPolicy };\n      }\n\n      public attachInlinePolicy(policy: Policy): void {\n        const thisAndPolicyAccountComparison = Token.compareStrings(this.env.account, policy.env.account);\n        const equalOrAnyUnresolved = thisAndPolicyAccountComparison === TokenComparison.SAME ||\n          thisAndPolicyAccountComparison === TokenComparison.BOTH_UNRESOLVED ||\n          thisAndPolicyAccountComparison === TokenComparison.ONE_UNRESOLVED;\n        if (equalOrAnyUnresolved) {\n          this.attachedPolicies.attach(policy);\n          policy.attachToRole(this);\n        }\n      }\n\n      public addManagedPolicy(_policy: IManagedPolicy): void {\n        // FIXME: Add warning that we're ignoring this\n      }\n\n      /**\n       * Grant permissions to the given principal to pass this role.\n       */\n      public grantPassRole(identity: IPrincipal): Grant {\n        return this.grant(identity, 'iam:PassRole');\n      }\n\n      /**\n       * Grant the actions defined in actions to the identity Principal on this resource.\n       */\n      public grant(grantee: IPrincipal, ...actions: string[]): Grant {\n        return Grant.addToPrincipal({\n          grantee,\n          actions,\n          resourceArns: [this.roleArn],\n          scope: this,\n        });\n      }\n    }\n\n    if (options.addGrantsToResources !== undefined && options.mutable !== false) {\n      throw new Error('\\'addGrantsToResources\\' can only be passed if \\'mutable: false\\'');\n    }\n\n    const importedRole = new Import(scope, id);\n    const roleArnAndScopeStackAccountComparison = Token.compareStrings(importedRole.env.account, scopeStack.account);\n    const equalOrAnyUnresolved = roleArnAndScopeStackAccountComparison === TokenComparison.SAME ||\n      roleArnAndScopeStackAccountComparison === TokenComparison.BOTH_UNRESOLVED ||\n      roleArnAndScopeStackAccountComparison === TokenComparison.ONE_UNRESOLVED;\n    // we only return an immutable Role if both accounts were explicitly provided, and different\n    return options.mutable !== false && equalOrAnyUnresolved\n      ? importedRole\n      : new ImmutableRole(scope, `ImmutableRole${id}`, importedRole, options.addGrantsToResources ?? false);\n  }\n\n  /**\n   * Import an external role by name.\n   *\n   * The imported role is assumed to exist in the same account as the account\n   * the scope's containing Stack is being deployed to.\n   */\n  public static fromRoleName(scope: Construct, id: string, roleName: string) {\n    return Role.fromRoleArn(scope, id, Stack.of(scope).formatArn({\n      region: '',\n      service: 'iam',\n      resource: 'role',\n      resourceName: roleName,\n    }));\n  }\n\n  public readonly grantPrincipal: IPrincipal = this;\n  public readonly principalAccount: string | undefined = this.env.account;\n\n  public readonly assumeRoleAction: string = 'sts:AssumeRole';\n\n  /**\n   * The assume role policy document associated with this role.\n   */\n  public readonly assumeRolePolicy?: PolicyDocument;\n\n  /**\n   * Returns the ARN of this role.\n   */\n  public readonly roleArn: string;\n\n  /**\n   * Returns the stable and unique string identifying the role. For example,\n   * AIDAJQABLZS4A3QDU576Q.\n   *\n   * @attribute\n   */\n  public readonly roleId: string;\n\n  /**\n   * Returns the name of the role.\n   */\n  public readonly roleName: string;\n\n  /**\n   * Returns the role.\n   */\n  public readonly policyFragment: PrincipalPolicyFragment;\n\n  /**\n   * Returns the permissions boundary attached to this role\n   */\n  public readonly permissionsBoundary?: IManagedPolicy;\n\n  private defaultPolicy?: Policy;\n  private readonly managedPolicies: IManagedPolicy[] = [];\n  private readonly attachedPolicies = new AttachedPolicies();\n  private readonly inlinePolicies: { [name: string]: PolicyDocument };\n  private immutableRole?: IRole;\n\n  constructor(scope: Construct, id: string, props: RoleProps) {\n    super(scope, id, {\n      physicalName: props.roleName,\n    });\n\n    const externalIds = props.externalIds || [];\n    if (props.externalId) {\n      externalIds.push(props.externalId);\n    }\n\n    this.assumeRolePolicy = createAssumeRolePolicy(props.assumedBy, externalIds);\n    this.managedPolicies.push(...props.managedPolicies || []);\n    this.inlinePolicies = props.inlinePolicies || {};\n    this.permissionsBoundary = props.permissionsBoundary;\n    const maxSessionDuration = props.maxSessionDuration && props.maxSessionDuration.toSeconds();\n    validateMaxSessionDuration(maxSessionDuration);\n    const description = (props.description && props.description?.length > 0) ? props.description : undefined;\n\n    if (description && description.length > 1000) {\n      throw new Error('Role description must be no longer than 1000 characters.');\n    }\n\n    const role = new CfnRole(this, 'Resource', {\n      assumeRolePolicyDocument: this.assumeRolePolicy as any,\n      managedPolicyArns: UniqueStringSet.from(() => this.managedPolicies.map(p => p.managedPolicyArn)),\n      policies: _flatten(this.inlinePolicies),\n      path: props.path,\n      permissionsBoundary: this.permissionsBoundary ? this.permissionsBoundary.managedPolicyArn : undefined,\n      roleName: this.physicalName,\n      maxSessionDuration,\n      description,\n    });\n\n    this.roleId = role.attrRoleId;\n    this.roleArn = this.getResourceArnAttribute(role.attrArn, {\n      region: '', // IAM is global in each partition\n      service: 'iam',\n      resource: 'role',\n      // Removes leading slash from path\n      resourceName: `${props.path ? props.path.substr(props.path.charAt(0) === '/' ? 1 : 0) : ''}${this.physicalName}`,\n    });\n    this.roleName = this.getResourceNameAttribute(role.ref);\n    this.policyFragment = new ArnPrincipal(this.roleArn).policyFragment;\n\n    function _flatten(policies?: { [name: string]: PolicyDocument }) {\n      if (policies == null || Object.keys(policies).length === 0) {\n        return undefined;\n      }\n      const result = new Array<CfnRole.PolicyProperty>();\n      for (const policyName of Object.keys(policies)) {\n        const policyDocument = policies[policyName];\n        result.push({ policyName, policyDocument });\n      }\n      return result;\n    }\n  }\n\n  /**\n   * Adds a permission to the role's default policy document.\n   * If there is no default policy attached to this role, it will be created.\n   * @param statement The permission statement to add to the policy document\n   */\n  public addToPrincipalPolicy(statement: PolicyStatement): AddToPrincipalPolicyResult {\n    if (!this.defaultPolicy) {\n      this.defaultPolicy = new Policy(this, 'DefaultPolicy');\n      this.attachInlinePolicy(this.defaultPolicy);\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  /**\n   * Attaches a managed policy to this role.\n   * @param policy The 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 role.\n   * @param policy The policy to attach\n   */\n  public attachInlinePolicy(policy: Policy) {\n    this.attachedPolicies.attach(policy);\n    policy.attachToRole(this);\n  }\n\n  /**\n   * Grant the actions defined in actions to the identity Principal on this resource.\n   */\n  public grant(grantee: IPrincipal, ...actions: string[]) {\n    return Grant.addToPrincipal({\n      grantee,\n      actions,\n      resourceArns: [this.roleArn],\n      scope: this,\n    });\n  }\n\n  /**\n   * Grant permissions to the given principal to pass this role.\n   */\n  public grantPassRole(identity: IPrincipal) {\n    return this.grant(identity, 'iam:PassRole');\n  }\n\n  /**\n   * Return a copy of this Role object whose Policies will not be updated\n   *\n   * Use the object returned by this method if you want this Role to be used by\n   * a construct without it automatically updating the Role's Policies.\n   *\n   * If you do, you are responsible for adding the correct statements to the\n   * Role's policies yourself.\n   */\n  public withoutPolicyUpdates(options: WithoutPolicyUpdatesOptions = {}): IRole {\n    if (!this.immutableRole) {\n      this.immutableRole = new ImmutableRole(Node.of(this).scope as Construct, `ImmutableRole${this.node.id}`, this, options.addGrantsToResources ?? false);\n    }\n\n    return this.immutableRole;\n  }\n\n  protected validate(): string[] {\n    const errors = super.validate();\n    errors.push(...this.assumeRolePolicy?.validateForResourcePolicy() || []);\n    for (const policy of Object.values(this.inlinePolicies)) {\n      errors.push(...policy.validateForIdentityPolicy());\n    }\n    return errors;\n  }\n}\n\n/**\n * A Role object\n */\nexport interface IRole extends IIdentity {\n  /**\n   * Returns the ARN of this role.\n   *\n   * @attribute\n   */\n  readonly roleArn: string;\n\n  /**\n   * Returns the name of this role.\n   *\n   * @attribute\n   */\n  readonly roleName: string;\n\n  /**\n   * Grant the actions defined in actions to the identity Principal on this resource.\n   */\n  grant(grantee: IPrincipal, ...actions: string[]): Grant;\n\n  /**\n   * Grant permissions to the given principal to pass this role.\n   */\n  grantPassRole(grantee: IPrincipal): Grant;\n}\n\nfunction createAssumeRolePolicy(principal: IPrincipal, externalIds: string[]) {\n  const actualDoc = new PolicyDocument();\n\n  // If requested, add externalIds to every statement added to this doc\n  const addDoc = externalIds.length === 0\n    ? actualDoc\n    : new MutatingPolicyDocumentAdapter(actualDoc, (statement) => {\n      statement.addCondition('StringEquals', {\n        'sts:ExternalId': externalIds.length === 1 ? externalIds[0] : externalIds,\n      });\n      return statement;\n    });\n\n  defaultAddPrincipalToAssumeRole(principal, addDoc);\n\n  return actualDoc;\n}\n\nfunction validateMaxSessionDuration(duration?: number) {\n  if (duration === undefined) {\n    return;\n  }\n\n  if (duration < 3600 || duration > 43200) {\n    throw new Error(`maxSessionDuration is set to ${duration}, but must be >= 3600sec (1hr) and <= 43200sec (12hrs)`);\n  }\n}\n\n/**\n * Options for the `withoutPolicyUpdates()` modifier of a Role\n */\nexport interface WithoutPolicyUpdatesOptions {\n  /**\n   * Add grants to resources instead of dropping them\n   *\n   * If this is `false` or not specified, grant permissions added to this role are ignored.\n   * It is your own responsibility to make sure the role has the required permissions.\n   *\n   * If this is `true`, any grant permissions will be added to the resource instead.\n   *\n   * @default false\n   */\n  readonly addGrantsToResources?: boolean;\n}"]} |
\ | No newline at end of file |