UNPKG

53.7 kBJavaScriptView Raw
1"use strict";
2var _a;
3Object.defineProperty(exports, "__esModule", { value: true });
4exports.Role = 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 constructs_1 = require("constructs");
9const grant_1 = require("./grant");
10const iam_generated_1 = require("./iam.generated");
11const policy_1 = require("./policy");
12const policy_document_1 = require("./policy-document");
13const principals_1 = require("./principals");
14const assume_role_policy_1 = require("./private/assume-role-policy");
15const immutable_role_1 = require("./private/immutable-role");
16const policydoc_adapter_1 = require("./private/policydoc-adapter");
17const 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 */
24class 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}
348exports.Role = Role;
349_a = JSII_RTTI_SYMBOL_1;
350Role[_a] = { fqn: "@aws-cdk/aws-iam.Role", version: "1.156.1" };
351function 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}
365function 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