UNPKG

71.6 kBJavaScriptView Raw
1"use strict";
2var _a;
3Object.defineProperty(exports, "__esModule", { value: true });
4exports.Key = exports.KeyUsage = exports.KeySpec = void 0;
5const jsiiDeprecationWarnings = require("../.warnings.jsii.js");
6const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
7const iam = require("@aws-cdk/aws-iam");
8const cxschema = require("@aws-cdk/cloud-assembly-schema");
9const core_1 = require("@aws-cdk/core");
10const cxapi = require("@aws-cdk/cx-api");
11const constructs_1 = require("constructs");
12const alias_1 = require("./alias");
13const kms_generated_1 = require("./kms.generated");
14const perms = require("./private/perms");
15class KeyBase extends core_1.Resource {
16 constructor() {
17 super(...arguments);
18 /**
19 * Collection of aliases added to the key
20 *
21 * Tracked to determine whether or not the aliasName should be added to the end of its ID
22 */
23 this.aliases = [];
24 }
25 /**
26 * Defines a new alias for the key.
27 */
28 addAlias(aliasName) {
29 const aliasId = this.aliases.length > 0 ? `Alias${aliasName}` : 'Alias';
30 const alias = new alias_1.Alias(this, aliasId, { aliasName, targetKey: this });
31 this.aliases.push(alias);
32 return alias;
33 }
34 /**
35 * Adds a statement to the KMS key resource policy.
36 * @param statement The policy statement to add
37 * @param allowNoOp If this is set to `false` and there is no policy
38 * defined (i.e. external key), the operation will fail. Otherwise, it will
39 * no-op.
40 */
41 addToResourcePolicy(statement, allowNoOp = true) {
42 const stack = core_1.Stack.of(this);
43 if (!this.policy) {
44 if (allowNoOp) {
45 return { statementAdded: false };
46 }
47 throw new Error(`Unable to add statement to IAM resource policy for KMS key: ${JSON.stringify(stack.resolve(this.keyArn))}`);
48 }
49 this.policy.addStatements(statement);
50 return { statementAdded: true, policyDependable: this.policy };
51 }
52 validate() {
53 const errors = super.validate();
54 errors.push(...this.policy?.validateForResourcePolicy() || []);
55 return errors;
56 }
57 /**
58 * Grant the indicated permissions on this key to the given principal
59 *
60 * This modifies both the principal's policy as well as the resource policy,
61 * since the default CloudFormation setup for KMS keys is that the policy
62 * must not be empty and so default grants won't work.
63 */
64 grant(grantee, ...actions) {
65 // KMS verifies whether the principals included in its key policy actually exist.
66 // This is a problem if the stack the grantee is part of depends on the key stack
67 // (as it won't exist before the key policy is attempted to be created).
68 // In that case, make the account the resource policy principal
69 const granteeStackDependsOnKeyStack = this.granteeStackDependsOnKeyStack(grantee);
70 const principal = granteeStackDependsOnKeyStack
71 ? new iam.AccountPrincipal(granteeStackDependsOnKeyStack)
72 : grantee.grantPrincipal;
73 const crossAccountAccess = this.isGranteeFromAnotherAccount(grantee);
74 const crossRegionAccess = this.isGranteeFromAnotherRegion(grantee);
75 const crossEnvironment = crossAccountAccess || crossRegionAccess;
76 const grantOptions = {
77 grantee,
78 actions,
79 resource: this,
80 resourceArns: [this.keyArn],
81 resourceSelfArns: crossEnvironment ? undefined : ['*'],
82 };
83 if (this.trustAccountIdentities && !crossEnvironment) {
84 return iam.Grant.addToPrincipalOrResource(grantOptions);
85 }
86 else {
87 return iam.Grant.addToPrincipalAndResource({
88 ...grantOptions,
89 // if the key is used in a cross-environment matter,
90 // we can't access the Key ARN (they don't have physical names),
91 // so fall back to using '*'. ToDo we need to make this better... somehow
92 resourceArns: crossEnvironment ? ['*'] : [this.keyArn],
93 resourcePolicyPrincipal: principal,
94 });
95 }
96 }
97 /**
98 * Grant decryption permissions using this key to the given principal
99 */
100 grantDecrypt(grantee) {
101 return this.grant(grantee, ...perms.DECRYPT_ACTIONS);
102 }
103 /**
104 * Grant encryption permissions using this key to the given principal
105 */
106 grantEncrypt(grantee) {
107 return this.grant(grantee, ...perms.ENCRYPT_ACTIONS);
108 }
109 /**
110 * Grant encryption and decryption permissions using this key to the given principal
111 */
112 grantEncryptDecrypt(grantee) {
113 return this.grant(grantee, ...[...perms.DECRYPT_ACTIONS, ...perms.ENCRYPT_ACTIONS]);
114 }
115 /**
116 * Checks whether the grantee belongs to a stack that will be deployed
117 * after the stack containing this key.
118 *
119 * @param grantee the grantee to give permissions to
120 * @returns the account ID of the grantee stack if its stack does depend on this stack,
121 * undefined otherwise
122 */
123 granteeStackDependsOnKeyStack(grantee) {
124 const grantPrincipal = grantee.grantPrincipal;
125 if (!isConstruct(grantPrincipal)) {
126 return undefined;
127 }
128 // this logic should only apply to newly created
129 // (= not imported) resources
130 if (!this.principalIsANewlyCreatedResource(grantPrincipal)) {
131 return undefined;
132 }
133 // return undefined;
134 const keyStack = core_1.Stack.of(this);
135 const granteeStack = core_1.Stack.of(grantPrincipal);
136 if (keyStack === granteeStack) {
137 return undefined;
138 }
139 return granteeStack.dependencies.includes(keyStack)
140 ? granteeStack.account
141 : undefined;
142 }
143 principalIsANewlyCreatedResource(principal) {
144 // yes, this sucks
145 // this is just a temporary stopgap to stem the bleeding while we work on a proper fix
146 return principal instanceof iam.Role ||
147 principal instanceof iam.User ||
148 principal instanceof iam.Group;
149 }
150 isGranteeFromAnotherRegion(grantee) {
151 if (!isConstruct(grantee)) {
152 return false;
153 }
154 const bucketStack = core_1.Stack.of(this);
155 const identityStack = core_1.Stack.of(grantee);
156 return bucketStack.region !== identityStack.region;
157 }
158 isGranteeFromAnotherAccount(grantee) {
159 if (!isConstruct(grantee)) {
160 return false;
161 }
162 const bucketStack = core_1.Stack.of(this);
163 const identityStack = core_1.Stack.of(grantee);
164 return bucketStack.account !== identityStack.account;
165 }
166}
167/**
168 * The key spec, represents the cryptographic configuration of keys.
169 */
170var KeySpec;
171(function (KeySpec) {
172 /**
173 * The default key spec.
174 *
175 * Valid usage: ENCRYPT_DECRYPT
176 */
177 KeySpec["SYMMETRIC_DEFAULT"] = "SYMMETRIC_DEFAULT";
178 /**
179 * RSA with 2048 bits of key.
180 *
181 * Valid usage: ENCRYPT_DECRYPT and SIGN_VERIFY
182 */
183 KeySpec["RSA_2048"] = "RSA_2048";
184 /**
185 * RSA with 3072 bits of key.
186 *
187 * Valid usage: ENCRYPT_DECRYPT and SIGN_VERIFY
188 */
189 KeySpec["RSA_3072"] = "RSA_3072";
190 /**
191 * RSA with 4096 bits of key.
192 *
193 * Valid usage: ENCRYPT_DECRYPT and SIGN_VERIFY
194 */
195 KeySpec["RSA_4096"] = "RSA_4096";
196 /**
197 * NIST FIPS 186-4, Section 6.4, ECDSA signature using the curve specified by the key and
198 * SHA-256 for the message digest.
199 *
200 * Valid usage: SIGN_VERIFY
201 */
202 KeySpec["ECC_NIST_P256"] = "ECC_NIST_P256";
203 /**
204 * NIST FIPS 186-4, Section 6.4, ECDSA signature using the curve specified by the key and
205 * SHA-384 for the message digest.
206 *
207 * Valid usage: SIGN_VERIFY
208 */
209 KeySpec["ECC_NIST_P384"] = "ECC_NIST_P384";
210 /**
211 * NIST FIPS 186-4, Section 6.4, ECDSA signature using the curve specified by the key and
212 * SHA-512 for the message digest.
213 *
214 * Valid usage: SIGN_VERIFY
215 */
216 KeySpec["ECC_NIST_P521"] = "ECC_NIST_P521";
217 /**
218 * Standards for Efficient Cryptography 2, Section 2.4.1, ECDSA signature on the Koblitz curve.
219 *
220 * Valid usage: SIGN_VERIFY
221 */
222 KeySpec["ECC_SECG_P256K1"] = "ECC_SECG_P256K1";
223})(KeySpec = exports.KeySpec || (exports.KeySpec = {}));
224/**
225 * The key usage, represents the cryptographic operations of keys.
226 */
227var KeyUsage;
228(function (KeyUsage) {
229 /**
230 * Encryption and decryption.
231 */
232 KeyUsage["ENCRYPT_DECRYPT"] = "ENCRYPT_DECRYPT";
233 /**
234 * Signing and verification
235 */
236 KeyUsage["SIGN_VERIFY"] = "SIGN_VERIFY";
237})(KeyUsage = exports.KeyUsage || (exports.KeyUsage = {}));
238/**
239 * Defines a KMS key.
240 *
241 * @resource AWS::KMS::Key
242 */
243class Key extends KeyBase {
244 constructor(scope, id, props = {}) {
245 super(scope, id);
246 try {
247 jsiiDeprecationWarnings._aws_cdk_aws_kms_KeyProps(props);
248 }
249 catch (error) {
250 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
251 Error.captureStackTrace(error, Key);
252 }
253 throw error;
254 }
255 const denyLists = {
256 [KeyUsage.ENCRYPT_DECRYPT]: [
257 KeySpec.ECC_NIST_P256,
258 KeySpec.ECC_NIST_P384,
259 KeySpec.ECC_NIST_P521,
260 KeySpec.ECC_SECG_P256K1,
261 ],
262 [KeyUsage.SIGN_VERIFY]: [
263 KeySpec.SYMMETRIC_DEFAULT,
264 ],
265 };
266 const keySpec = props.keySpec ?? KeySpec.SYMMETRIC_DEFAULT;
267 const keyUsage = props.keyUsage ?? KeyUsage.ENCRYPT_DECRYPT;
268 if (denyLists[keyUsage].includes(keySpec)) {
269 throw new Error(`key spec '${keySpec}' is not valid with usage '${keyUsage}'`);
270 }
271 if (keySpec !== KeySpec.SYMMETRIC_DEFAULT && props.enableKeyRotation) {
272 throw new Error('key rotation cannot be enabled on asymmetric keys');
273 }
274 const defaultKeyPoliciesFeatureEnabled = core_1.FeatureFlags.of(this).isEnabled(cxapi.KMS_DEFAULT_KEY_POLICIES);
275 this.policy = props.policy ?? new iam.PolicyDocument();
276 if (defaultKeyPoliciesFeatureEnabled) {
277 if (props.trustAccountIdentities === false) {
278 throw new Error('`trustAccountIdentities` cannot be false if the @aws-cdk/aws-kms:defaultKeyPolicies feature flag is set');
279 }
280 this.trustAccountIdentities = true;
281 // Set the default key policy if one hasn't been provided by the user.
282 if (!props.policy) {
283 this.addDefaultAdminPolicy();
284 }
285 }
286 else {
287 this.trustAccountIdentities = props.trustAccountIdentities ?? false;
288 if (this.trustAccountIdentities) {
289 this.addDefaultAdminPolicy();
290 }
291 else {
292 this.addLegacyAdminPolicy();
293 }
294 }
295 let pendingWindowInDays;
296 if (props.pendingWindow) {
297 pendingWindowInDays = props.pendingWindow.toDays();
298 if (pendingWindowInDays < 7 || pendingWindowInDays > 30) {
299 throw new Error(`'pendingWindow' value must between 7 and 30 days. Received: ${pendingWindowInDays}`);
300 }
301 }
302 const resource = new kms_generated_1.CfnKey(this, 'Resource', {
303 description: props.description,
304 enableKeyRotation: props.enableKeyRotation,
305 enabled: props.enabled,
306 keySpec: props.keySpec,
307 keyUsage: props.keyUsage,
308 keyPolicy: this.policy,
309 pendingWindowInDays: pendingWindowInDays,
310 });
311 this.keyArn = resource.attrArn;
312 this.keyId = resource.ref;
313 resource.applyRemovalPolicy(props.removalPolicy);
314 (props.admins ?? []).forEach((p) => this.grantAdmin(p));
315 if (props.alias !== undefined) {
316 this.addAlias(props.alias);
317 }
318 }
319 /**
320 * Import an externally defined KMS Key using its ARN.
321 *
322 * @param scope the construct that will "own" the imported key.
323 * @param id the id of the imported key in the construct tree.
324 * @param keyArn the ARN of an existing KMS key.
325 */
326 static fromKeyArn(scope, id, keyArn) {
327 class Import extends KeyBase {
328 constructor(keyId) {
329 super(scope, id);
330 this.keyArn = keyArn;
331 this.policy = undefined;
332 // defaulting true: if we are importing the key the key policy is
333 // undefined and impossible to change here; this means updating identity
334 // policies is really the only option
335 this.trustAccountIdentities = true;
336 this.keyId = keyId;
337 }
338 }
339 const keyResourceName = core_1.Stack.of(scope).splitArn(keyArn, core_1.ArnFormat.SLASH_RESOURCE_NAME).resourceName;
340 if (!keyResourceName) {
341 throw new Error(`KMS key ARN must be in the format 'arn:aws:kms:<region>:<account>:key/<keyId>', got: '${keyArn}'`);
342 }
343 return new Import(keyResourceName);
344 }
345 /**
346 * Create a mutable {@link IKey} based on a low-level {@link CfnKey}.
347 * This is most useful when combined with the cloudformation-include module.
348 * This method is different than {@link fromKeyArn()} because the {@link IKey}
349 * returned from this method is mutable;
350 * meaning, calling any mutating methods on it,
351 * like {@link IKey.addToResourcePolicy()},
352 * will actually be reflected in the resulting template,
353 * as opposed to the object returned from {@link fromKeyArn()},
354 * on which calling those methods would have no effect.
355 */
356 static fromCfnKey(cfnKey) {
357 try {
358 jsiiDeprecationWarnings._aws_cdk_aws_kms_CfnKey(cfnKey);
359 }
360 catch (error) {
361 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
362 Error.captureStackTrace(error, this.fromCfnKey);
363 }
364 throw error;
365 }
366 // use a "weird" id that has a higher chance of being unique
367 const id = '@FromCfnKey';
368 // if fromCfnKey() was already called on this cfnKey,
369 // return the same L2
370 // (as different L2s would conflict, because of the mutation of the keyPolicy property of the L1 below)
371 const existing = cfnKey.node.tryFindChild(id);
372 if (existing) {
373 return existing;
374 }
375 let keyPolicy;
376 try {
377 keyPolicy = iam.PolicyDocument.fromJson(cfnKey.keyPolicy);
378 }
379 catch (e) {
380 // If the KeyPolicy contains any CloudFormation functions,
381 // PolicyDocument.fromJson() throws an exception.
382 // In that case, because we would have to effectively make the returned IKey immutable,
383 // throw an exception suggesting to use the other importing methods instead.
384 // We might make this parsing logic smarter later,
385 // but let's start by erroring out.
386 throw new Error('Could not parse the PolicyDocument of the passed AWS::KMS::Key resource because it contains CloudFormation functions. ' +
387 'This makes it impossible to create a mutable IKey from that Policy. ' +
388 'You have to use fromKeyArn instead, passing it the ARN attribute property of the low-level CfnKey');
389 }
390 // change the key policy of the L1, so that all changes done in the L2 are reflected in the resulting template
391 cfnKey.keyPolicy = core_1.Lazy.any({ produce: () => keyPolicy.toJSON() });
392 return new class extends KeyBase {
393 constructor() {
394 super(...arguments);
395 this.keyArn = cfnKey.attrArn;
396 this.keyId = cfnKey.ref;
397 this.policy = keyPolicy;
398 this.trustAccountIdentities = false;
399 }
400 }(cfnKey, id);
401 }
402 /**
403 * Import an existing Key by querying the AWS environment this stack is deployed to.
404 *
405 * This function only needs to be used to use Keys not defined in your CDK
406 * application. If you are looking to share a Key between stacks, you can
407 * pass the `Key` object between stacks and use it as normal. In addition,
408 * it's not necessary to use this method if an interface accepts an `IKey`.
409 * In this case, `Alias.fromAliasName()` can be used which returns an alias
410 * that extends `IKey`.
411 *
412 * Calling this method will lead to a lookup when the CDK CLI is executed.
413 * You can therefore not use any values that will only be available at
414 * CloudFormation execution time (i.e., Tokens).
415 *
416 * The Key information will be cached in `cdk.context.json` and the same Key
417 * will be used on future runs. To refresh the lookup, you will have to
418 * evict the value from the cache using the `cdk context` command. See
419 * https://docs.aws.amazon.com/cdk/latest/guide/context.html for more information.
420 */
421 static fromLookup(scope, id, options) {
422 try {
423 jsiiDeprecationWarnings._aws_cdk_aws_kms_KeyLookupOptions(options);
424 }
425 catch (error) {
426 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
427 Error.captureStackTrace(error, this.fromLookup);
428 }
429 throw error;
430 }
431 class Import extends KeyBase {
432 constructor(keyId, keyArn) {
433 super(scope, id);
434 this.policy = undefined;
435 // defaulting true: if we are importing the key the key policy is
436 // undefined and impossible to change here; this means updating identity
437 // policies is really the only option
438 this.trustAccountIdentities = true;
439 this.keyId = keyId;
440 this.keyArn = keyArn;
441 }
442 }
443 if (core_1.Token.isUnresolved(options.aliasName)) {
444 throw new Error('All arguments to Key.fromLookup() must be concrete (no Tokens)');
445 }
446 const attributes = core_1.ContextProvider.getValue(scope, {
447 provider: cxschema.ContextProvider.KEY_PROVIDER,
448 props: {
449 aliasName: options.aliasName,
450 },
451 dummyValue: {
452 keyId: '1234abcd-12ab-34cd-56ef-1234567890ab',
453 },
454 }).value;
455 return new Import(attributes.keyId, core_1.Arn.format({ resource: 'key', service: 'kms', resourceName: attributes.keyId }, core_1.Stack.of(scope)));
456 }
457 /**
458 * Grant admins permissions using this key to the given principal
459 *
460 * Key administrators have permissions to manage the key (e.g., change permissions, revoke), but do not have permissions
461 * to use the key in cryptographic operations (e.g., encrypt, decrypt).
462 */
463 grantAdmin(grantee) {
464 return this.grant(grantee, ...perms.ADMIN_ACTIONS);
465 }
466 /**
467 * Adds the default key policy to the key. This policy gives the AWS account (root user) full access to the CMK,
468 * which reduces the risk of the CMK becoming unmanageable and enables IAM policies to allow access to the CMK.
469 * This is the same policy that is default when creating a Key via the KMS API or Console.
470 * @see https://docs.aws.amazon.com/kms/latest/developerguide/key-policies.html#key-policy-default
471 */
472 addDefaultAdminPolicy() {
473 this.addToResourcePolicy(new iam.PolicyStatement({
474 resources: ['*'],
475 actions: ['kms:*'],
476 principals: [new iam.AccountRootPrincipal()],
477 }));
478 }
479 /**
480 * Grants the account admin privileges -- not full account access -- plus the GenerateDataKey action.
481 * The GenerateDataKey action was added for interop with S3 in https://github.com/aws/aws-cdk/issues/3458.
482 *
483 * This policy is discouraged and deprecated by the `@aws-cdk/aws-kms:defaultKeyPolicies` feature flag.
484 *
485 * @link https://docs.aws.amazon.com/kms/latest/developerguide/key-policies.html#key-policy-default
486 * @deprecated
487 */
488 addLegacyAdminPolicy() {
489 // This is equivalent to `[...perms.ADMIN_ACTIONS, 'kms:GenerateDataKey']`,
490 // but keeping this explicit ordering for backwards-compatibility (changing the ordering causes resource updates)
491 const actions = [
492 'kms:Create*',
493 'kms:Describe*',
494 'kms:Enable*',
495 'kms:List*',
496 'kms:Put*',
497 'kms:Update*',
498 'kms:Revoke*',
499 'kms:Disable*',
500 'kms:Get*',
501 'kms:Delete*',
502 'kms:ScheduleKeyDeletion',
503 'kms:CancelKeyDeletion',
504 'kms:GenerateDataKey',
505 'kms:TagResource',
506 'kms:UntagResource',
507 ];
508 this.addToResourcePolicy(new iam.PolicyStatement({
509 resources: ['*'],
510 actions,
511 principals: [new iam.AccountRootPrincipal()],
512 }));
513 }
514}
515exports.Key = Key;
516_a = JSII_RTTI_SYMBOL_1;
517Key[_a] = { fqn: "@aws-cdk/aws-kms.Key", version: "1.191.0" };
518/**
519 * Whether the given object is a Construct
520 *
521 * Normally we'd do `x instanceof Construct`, but that is not robust against
522 * multiple copies of the `constructs` library on disk. This can happen
523 * when upgrading and downgrading between v2 and v1, and in the use of CDK
524 * Pipelines is going to an error that says "Can't use Pipeline/Pipeline/Role in
525 * a cross-environment fashion", which is very confusing.
526 */
527function isConstruct(x) {
528 const sym = Symbol.for('constructs.Construct.node');
529 return (typeof x === 'object' && x &&
530 (x instanceof constructs_1.Construct // happy fast case
531 || !!x.node // constructs v10
532 || !!x[sym])); // constructs v3
533}
534//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"key.js","sourceRoot":"","sources":["key.ts"],"names":[],"mappings":"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","sourcesContent":["import * as iam from '@aws-cdk/aws-iam';\nimport * as cxschema from '@aws-cdk/cloud-assembly-schema';\nimport { FeatureFlags, IResource, Lazy, RemovalPolicy, Resource, Stack, Duration, Token, ContextProvider, Arn, ArnFormat } from '@aws-cdk/core';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport { IConstruct, Construct } from 'constructs';\nimport { Alias } from './alias';\nimport { KeyLookupOptions } from './key-lookup';\nimport { CfnKey } from './kms.generated';\nimport * as perms from './private/perms';\n\n/**\n * A KMS Key, either managed by this CDK app, or imported.\n */\nexport interface IKey extends IResource {\n  /**\n   * The ARN of the key.\n   *\n   * @attribute\n   */\n  readonly keyArn: string;\n\n  /**\n   * The ID of the key\n   * (the part that looks something like: 1234abcd-12ab-34cd-56ef-1234567890ab).\n   *\n   * @attribute\n   */\n  readonly keyId: string;\n\n  /**\n   * Defines a new alias for the key.\n   */\n  addAlias(alias: string): Alias;\n\n  /**\n   * Adds a statement to the KMS key resource policy.\n   * @param statement The policy statement to add\n   * @param allowNoOp If this is set to `false` and there is no policy\n   * defined (i.e. external key), the operation will fail. Otherwise, it will\n   * no-op.\n   */\n  addToResourcePolicy(statement: iam.PolicyStatement, allowNoOp?: boolean): iam.AddToResourcePolicyResult;\n\n  /**\n   * Grant the indicated permissions on this key to the given principal\n   */\n  grant(grantee: iam.IGrantable, ...actions: string[]): iam.Grant;\n\n  /**\n   * Grant decryption permissions using this key to the given principal\n   */\n  grantDecrypt(grantee: iam.IGrantable): iam.Grant;\n\n  /**\n   * Grant encryption permissions using this key to the given principal\n   */\n  grantEncrypt(grantee: iam.IGrantable): iam.Grant;\n\n  /**\n   * Grant encryption and decryption permissions using this key to the given principal\n   */\n  grantEncryptDecrypt(grantee: iam.IGrantable): iam.Grant;\n}\n\nabstract class KeyBase extends Resource implements IKey {\n  /**\n   * The ARN of the key.\n   */\n  public abstract readonly keyArn: string;\n\n  public abstract readonly keyId: string;\n\n  /**\n   * Optional policy document that represents the resource policy of this key.\n   *\n   * If specified, addToResourcePolicy can be used to edit this policy.\n   * Otherwise this method will no-op.\n   */\n  protected abstract readonly policy?: iam.PolicyDocument;\n\n  /**\n   * Optional property to control trusting account identities.\n   *\n   * If specified, grants will default identity policies instead of to both\n   * resource and identity policies. This matches the default behavior when creating\n   * KMS keys via the API or console.\n   */\n  protected abstract readonly trustAccountIdentities: boolean;\n\n  /**\n   * Collection of aliases added to the key\n   *\n   * Tracked to determine whether or not the aliasName should be added to the end of its ID\n   */\n  private readonly aliases: Alias[] = [];\n\n  /**\n   * Defines a new alias for the key.\n   */\n  public addAlias(aliasName: string): Alias {\n    const aliasId = this.aliases.length > 0 ? `Alias${aliasName}` : 'Alias';\n\n    const alias = new Alias(this, aliasId, { aliasName, targetKey: this });\n    this.aliases.push(alias);\n\n    return alias;\n  }\n\n  /**\n   * Adds a statement to the KMS key resource policy.\n   * @param statement The policy statement to add\n   * @param allowNoOp If this is set to `false` and there is no policy\n   * defined (i.e. external key), the operation will fail. Otherwise, it will\n   * no-op.\n   */\n  public addToResourcePolicy(statement: iam.PolicyStatement, allowNoOp = true): iam.AddToResourcePolicyResult {\n    const stack = Stack.of(this);\n\n    if (!this.policy) {\n      if (allowNoOp) { return { statementAdded: false }; }\n      throw new Error(`Unable to add statement to IAM resource policy for KMS key: ${JSON.stringify(stack.resolve(this.keyArn))}`);\n    }\n\n    this.policy.addStatements(statement);\n    return { statementAdded: true, policyDependable: this.policy };\n  }\n\n  protected validate(): string[] {\n    const errors = super.validate();\n    errors.push(...this.policy?.validateForResourcePolicy() || []);\n    return errors;\n  }\n\n  /**\n   * Grant the indicated permissions on this key to the given principal\n   *\n   * This modifies both the principal's policy as well as the resource policy,\n   * since the default CloudFormation setup for KMS keys is that the policy\n   * must not be empty and so default grants won't work.\n   */\n  public grant(grantee: iam.IGrantable, ...actions: string[]): iam.Grant {\n    // KMS verifies whether the principals included in its key policy actually exist.\n    // This is a problem if the stack the grantee is part of depends on the key stack\n    // (as it won't exist before the key policy is attempted to be created).\n    // In that case, make the account the resource policy principal\n    const granteeStackDependsOnKeyStack = this.granteeStackDependsOnKeyStack(grantee);\n    const principal = granteeStackDependsOnKeyStack\n      ? new iam.AccountPrincipal(granteeStackDependsOnKeyStack)\n      : grantee.grantPrincipal;\n\n    const crossAccountAccess = this.isGranteeFromAnotherAccount(grantee);\n    const crossRegionAccess = this.isGranteeFromAnotherRegion(grantee);\n    const crossEnvironment = crossAccountAccess || crossRegionAccess;\n    const grantOptions: iam.GrantWithResourceOptions = {\n      grantee,\n      actions,\n      resource: this,\n      resourceArns: [this.keyArn],\n      resourceSelfArns: crossEnvironment ? undefined : ['*'],\n    };\n    if (this.trustAccountIdentities && !crossEnvironment) {\n      return iam.Grant.addToPrincipalOrResource(grantOptions);\n    } else {\n      return iam.Grant.addToPrincipalAndResource({\n        ...grantOptions,\n        // if the key is used in a cross-environment matter,\n        // we can't access the Key ARN (they don't have physical names),\n        // so fall back to using '*'. ToDo we need to make this better... somehow\n        resourceArns: crossEnvironment ? ['*'] : [this.keyArn],\n        resourcePolicyPrincipal: principal,\n      });\n    }\n  }\n\n  /**\n   * Grant decryption permissions using this key to the given principal\n   */\n  public grantDecrypt(grantee: iam.IGrantable): iam.Grant {\n    return this.grant(grantee, ...perms.DECRYPT_ACTIONS);\n  }\n\n  /**\n   * Grant encryption permissions using this key to the given principal\n   */\n  public grantEncrypt(grantee: iam.IGrantable): iam.Grant {\n    return this.grant(grantee, ...perms.ENCRYPT_ACTIONS);\n  }\n\n  /**\n   * Grant encryption and decryption permissions using this key to the given principal\n   */\n  public grantEncryptDecrypt(grantee: iam.IGrantable): iam.Grant {\n    return this.grant(grantee, ...[...perms.DECRYPT_ACTIONS, ...perms.ENCRYPT_ACTIONS]);\n  }\n\n  /**\n   * Checks whether the grantee belongs to a stack that will be deployed\n   * after the stack containing this key.\n   *\n   * @param grantee the grantee to give permissions to\n   * @returns the account ID of the grantee stack if its stack does depend on this stack,\n   *   undefined otherwise\n   */\n  private granteeStackDependsOnKeyStack(grantee: iam.IGrantable): string | undefined {\n    const grantPrincipal = grantee.grantPrincipal;\n    if (!isConstruct(grantPrincipal)) {\n      return undefined;\n    }\n    // this logic should only apply to newly created\n    // (= not imported) resources\n    if (!this.principalIsANewlyCreatedResource(grantPrincipal)) {\n      return undefined;\n    }\n    // return undefined;\n    const keyStack = Stack.of(this);\n    const granteeStack = Stack.of(grantPrincipal);\n    if (keyStack === granteeStack) {\n      return undefined;\n    }\n    return granteeStack.dependencies.includes(keyStack)\n      ? granteeStack.account\n      : undefined;\n  }\n\n  private principalIsANewlyCreatedResource(principal: IConstruct): boolean {\n    // yes, this sucks\n    // this is just a temporary stopgap to stem the bleeding while we work on a proper fix\n    return principal instanceof iam.Role ||\n      principal instanceof iam.User ||\n      principal instanceof iam.Group;\n  }\n\n  private isGranteeFromAnotherRegion(grantee: iam.IGrantable): boolean {\n    if (!isConstruct(grantee)) {\n      return false;\n    }\n    const bucketStack = Stack.of(this);\n    const identityStack = Stack.of(grantee);\n    return bucketStack.region !== identityStack.region;\n  }\n\n  private isGranteeFromAnotherAccount(grantee: iam.IGrantable): boolean {\n    if (!isConstruct(grantee)) {\n      return false;\n    }\n    const bucketStack = Stack.of(this);\n    const identityStack = Stack.of(grantee);\n    return bucketStack.account !== identityStack.account;\n  }\n}\n\n/**\n * The key spec, represents the cryptographic configuration of keys.\n */\nexport enum KeySpec {\n  /**\n   * The default key spec.\n   *\n   * Valid usage: ENCRYPT_DECRYPT\n   */\n  SYMMETRIC_DEFAULT = 'SYMMETRIC_DEFAULT',\n\n  /**\n   * RSA with 2048 bits of key.\n   *\n   * Valid usage: ENCRYPT_DECRYPT and SIGN_VERIFY\n   */\n  RSA_2048 = 'RSA_2048',\n\n  /**\n   * RSA with 3072 bits of key.\n   *\n   * Valid usage: ENCRYPT_DECRYPT and SIGN_VERIFY\n   */\n  RSA_3072 = 'RSA_3072',\n\n  /**\n   * RSA with 4096 bits of key.\n   *\n   * Valid usage: ENCRYPT_DECRYPT and SIGN_VERIFY\n   */\n  RSA_4096 = 'RSA_4096',\n\n  /**\n   * NIST FIPS 186-4, Section 6.4, ECDSA signature using the curve specified by the key and\n   * SHA-256 for the message digest.\n   *\n   * Valid usage: SIGN_VERIFY\n   */\n  ECC_NIST_P256 = 'ECC_NIST_P256',\n\n  /**\n   * NIST FIPS 186-4, Section 6.4, ECDSA signature using the curve specified by the key and\n   * SHA-384 for the message digest.\n   *\n   * Valid usage: SIGN_VERIFY\n   */\n  ECC_NIST_P384 = 'ECC_NIST_P384',\n\n  /**\n   * NIST FIPS 186-4, Section 6.4, ECDSA signature using the curve specified by the key and\n   * SHA-512 for the message digest.\n   *\n   * Valid usage: SIGN_VERIFY\n   */\n  ECC_NIST_P521 = 'ECC_NIST_P521',\n\n  /**\n   * Standards for Efficient Cryptography 2, Section 2.4.1, ECDSA signature on the Koblitz curve.\n   *\n   * Valid usage: SIGN_VERIFY\n   */\n  ECC_SECG_P256K1 = 'ECC_SECG_P256K1',\n}\n\n/**\n * The key usage, represents the cryptographic operations of keys.\n */\nexport enum KeyUsage {\n  /**\n   * Encryption and decryption.\n   */\n  ENCRYPT_DECRYPT = 'ENCRYPT_DECRYPT',\n\n  /**\n   * Signing and verification\n   */\n  SIGN_VERIFY = 'SIGN_VERIFY',\n}\n\n/**\n * Construction properties for a KMS Key object\n */\nexport interface KeyProps {\n  /**\n   * A description of the key. Use a description that helps your users decide\n   * whether the key is appropriate for a particular task.\n   *\n   * @default - No description.\n   */\n  readonly description?: string;\n\n  /**\n   * Initial alias to add to the key\n   *\n   * More aliases can be added later by calling `addAlias`.\n   *\n   * @default - No alias is added for the key.\n   */\n  readonly alias?: string;\n\n  /**\n   * Indicates whether AWS KMS rotates the key.\n   *\n   * @default false\n   */\n  readonly enableKeyRotation?: boolean;\n\n  /**\n   * Indicates whether the key is available for use.\n   *\n   * @default - Key is enabled.\n   */\n  readonly enabled?: boolean;\n\n  /**\n   * The cryptographic configuration of the key. The valid value depends on usage of the key.\n   *\n   * IMPORTANT: If you change this property of an existing key, the existing key is scheduled for deletion\n   * and a new key is created with the specified value.\n   *\n   * @default KeySpec.SYMMETRIC_DEFAULT\n   */\n  readonly keySpec?: KeySpec;\n\n  /**\n   * The cryptographic operations for which the key can be used.\n   *\n   * IMPORTANT: If you change this property of an existing key, the existing key is scheduled for deletion\n   * and a new key is created with the specified value.\n   *\n   * @default KeyUsage.ENCRYPT_DECRYPT\n   */\n  readonly keyUsage?: KeyUsage;\n\n  /**\n   * Custom policy document to attach to the KMS key.\n   *\n   * NOTE - If the `@aws-cdk/aws-kms:defaultKeyPolicies` feature flag is set (the default for new projects),\n   * this policy will *override* the default key policy and become the only key policy for the key. If the\n   * feature flag is not set, this policy will be appended to the default key policy.\n   *\n   * @default - A policy document with permissions for the account root to\n   * administer the key will be created.\n   */\n  readonly policy?: iam.PolicyDocument;\n\n  /**\n   * A list of principals to add as key administrators to the key policy.\n   *\n   * Key administrators have permissions to manage the key (e.g., change permissions, revoke), but do not have permissions\n   * to use the key in cryptographic operations (e.g., encrypt, decrypt).\n   *\n   * These principals will be added to the default key policy (if none specified), or to the specified policy (if provided).\n   *\n   * @default []\n   */\n  readonly admins?: iam.IPrincipal[];\n\n  /**\n   * Whether the encryption key should be retained when it is removed from the Stack. This is useful when one wants to\n   * retain access to data that was encrypted with a key that is being retired.\n   *\n   * @default RemovalPolicy.Retain\n   */\n  readonly removalPolicy?: RemovalPolicy;\n\n  /**\n   * Whether the key usage can be granted by IAM policies\n   *\n   * Setting this to true adds a default statement which delegates key\n   * access control completely to the identity's IAM policy (similar\n   * to how it works for other AWS resources). This matches the default behavior\n   * when creating KMS keys via the API or console.\n   *\n   * If the `@aws-cdk/aws-kms:defaultKeyPolicies` feature flag is set (the default for new projects),\n   * this flag will always be treated as 'true' and does not need to be explicitly set.\n   *\n   * @default - false, unless the `@aws-cdk/aws-kms:defaultKeyPolicies` feature flag is set.\n   * @see https://docs.aws.amazon.com/kms/latest/developerguide/key-policies.html#key-policy-default-allow-root-enable-iam\n   * @deprecated redundant with the `@aws-cdk/aws-kms:defaultKeyPolicies` feature flag\n   */\n  readonly trustAccountIdentities?: boolean;\n\n  /**\n   * Specifies the number of days in the waiting period before\n   * AWS KMS deletes a CMK that has been removed from a CloudFormation stack.\n   *\n   * When you remove a customer master key (CMK) from a CloudFormation stack, AWS KMS schedules the CMK for deletion\n   * and starts the mandatory waiting period. The PendingWindowInDays property determines the length of waiting period.\n   * During the waiting period, the key state of CMK is Pending Deletion, which prevents the CMK from being used in\n   * cryptographic operations. When the waiting period expires, AWS KMS permanently deletes the CMK.\n   *\n   * Enter a value between 7 and 30 days.\n   *\n   * @see https://docs.aws.amazon.com/AWSCloudFormation/latest/UserGuide/aws-resource-kms-key.html#cfn-kms-key-pendingwindowindays\n   * @default - 30 days\n   */\n  readonly pendingWindow?: Duration;\n}\n\n/**\n * Defines a KMS key.\n *\n * @resource AWS::KMS::Key\n */\nexport class Key extends KeyBase {\n  /**\n   * Import an externally defined KMS Key using its ARN.\n   *\n   * @param scope  the construct that will \"own\" the imported key.\n   * @param id     the id of the imported key in the construct tree.\n   * @param keyArn the ARN of an existing KMS key.\n   */\n  public static fromKeyArn(scope: Construct, id: string, keyArn: string): IKey {\n    class Import extends KeyBase {\n      public readonly keyArn = keyArn;\n      public readonly keyId: string;\n      protected readonly policy?: iam.PolicyDocument | undefined = undefined;\n      // defaulting true: if we are importing the key the key policy is\n      // undefined and impossible to change here; this means updating identity\n      // policies is really the only option\n      protected readonly trustAccountIdentities: boolean = true;\n\n      constructor(keyId: string) {\n        super(scope, id);\n\n        this.keyId = keyId;\n      }\n    }\n\n    const keyResourceName = Stack.of(scope).splitArn(keyArn, ArnFormat.SLASH_RESOURCE_NAME).resourceName;\n    if (!keyResourceName) {\n      throw new Error(`KMS key ARN must be in the format 'arn:aws:kms:<region>:<account>:key/<keyId>', got: '${keyArn}'`);\n    }\n\n    return new Import(keyResourceName);\n  }\n\n  /**\n   * Create a mutable {@link IKey} based on a low-level {@link CfnKey}.\n   * This is most useful when combined with the cloudformation-include module.\n   * This method is different than {@link fromKeyArn()} because the {@link IKey}\n   * returned from this method is mutable;\n   * meaning, calling any mutating methods on it,\n   * like {@link IKey.addToResourcePolicy()},\n   * will actually be reflected in the resulting template,\n   * as opposed to the object returned from {@link fromKeyArn()},\n   * on which calling those methods would have no effect.\n   */\n  public static fromCfnKey(cfnKey: CfnKey): IKey {\n    // use a \"weird\" id that has a higher chance of being unique\n    const id = '@FromCfnKey';\n\n    // if fromCfnKey() was already called on this cfnKey,\n    // return the same L2\n    // (as different L2s would conflict, because of the mutation of the keyPolicy property of the L1 below)\n    const existing = cfnKey.node.tryFindChild(id);\n    if (existing) {\n      return <IKey>existing;\n    }\n\n    let keyPolicy: iam.PolicyDocument;\n    try {\n      keyPolicy = iam.PolicyDocument.fromJson(cfnKey.keyPolicy);\n    } catch (e) {\n      // If the KeyPolicy contains any CloudFormation functions,\n      // PolicyDocument.fromJson() throws an exception.\n      // In that case, because we would have to effectively make the returned IKey immutable,\n      // throw an exception suggesting to use the other importing methods instead.\n      // We might make this parsing logic smarter later,\n      // but let's start by erroring out.\n      throw new Error('Could not parse the PolicyDocument of the passed AWS::KMS::Key resource because it contains CloudFormation functions. ' +\n        'This makes it impossible to create a mutable IKey from that Policy. ' +\n        'You have to use fromKeyArn instead, passing it the ARN attribute property of the low-level CfnKey');\n    }\n\n    // change the key policy of the L1, so that all changes done in the L2 are reflected in the resulting template\n    cfnKey.keyPolicy = Lazy.any({ produce: () => keyPolicy.toJSON() });\n\n    return new class extends KeyBase {\n      public readonly keyArn = cfnKey.attrArn;\n      public readonly keyId = cfnKey.ref;\n      protected readonly policy = keyPolicy;\n      protected readonly trustAccountIdentities = false;\n    }(cfnKey, id);\n  }\n\n  /**\n   * Import an existing Key by querying the AWS environment this stack is deployed to.\n   *\n   * This function only needs to be used to use Keys not defined in your CDK\n   * application. If you are looking to share a Key between stacks, you can\n   * pass the `Key` object between stacks and use it as normal. In addition,\n   * it's not necessary to use this method if an interface accepts an `IKey`.\n   * In this case, `Alias.fromAliasName()` can be used which returns an alias\n   * that extends `IKey`.\n   *\n   * Calling this method will lead to a lookup when the CDK CLI is executed.\n   * You can therefore not use any values that will only be available at\n   * CloudFormation execution time (i.e., Tokens).\n   *\n   * The Key information will be cached in `cdk.context.json` and the same Key\n   * will be used on future runs. To refresh the lookup, you will have to\n   * evict the value from the cache using the `cdk context` command. See\n   * https://docs.aws.amazon.com/cdk/latest/guide/context.html for more information.\n   */\n  public static fromLookup(scope: Construct, id: string, options: KeyLookupOptions): IKey {\n    class Import extends KeyBase {\n      public readonly keyArn: string;\n      public readonly keyId: string;\n      protected readonly policy?: iam.PolicyDocument | undefined = undefined;\n      // defaulting true: if we are importing the key the key policy is\n      // undefined and impossible to change here; this means updating identity\n      // policies is really the only option\n      protected readonly trustAccountIdentities: boolean = true;\n\n      constructor(keyId: string, keyArn: string) {\n        super(scope, id);\n\n        this.keyId = keyId;\n        this.keyArn = keyArn;\n      }\n    }\n    if (Token.isUnresolved(options.aliasName)) {\n      throw new Error('All arguments to Key.fromLookup() must be concrete (no Tokens)');\n    }\n\n    const attributes: cxapi.KeyContextResponse = ContextProvider.getValue(scope, {\n      provider: cxschema.ContextProvider.KEY_PROVIDER,\n      props: {\n        aliasName: options.aliasName,\n      } as cxschema.KeyContextQuery,\n      dummyValue: {\n        keyId: '1234abcd-12ab-34cd-56ef-1234567890ab',\n      },\n    }).value;\n\n    return new Import(attributes.keyId,\n      Arn.format({ resource: 'key', service: 'kms', resourceName: attributes.keyId }, Stack.of(scope)));\n  }\n\n  public readonly keyArn: string;\n  public readonly keyId: string;\n  protected readonly policy?: iam.PolicyDocument;\n  protected readonly trustAccountIdentities: boolean;\n\n  constructor(scope: Construct, id: string, props: KeyProps = {}) {\n    super(scope, id);\n\n    const denyLists = {\n      [KeyUsage.ENCRYPT_DECRYPT]: [\n        KeySpec.ECC_NIST_P256,\n        KeySpec.ECC_NIST_P384,\n        KeySpec.ECC_NIST_P521,\n        KeySpec.ECC_SECG_P256K1,\n      ],\n      [KeyUsage.SIGN_VERIFY]: [\n        KeySpec.SYMMETRIC_DEFAULT,\n      ],\n    };\n    const keySpec = props.keySpec ?? KeySpec.SYMMETRIC_DEFAULT;\n    const keyUsage = props.keyUsage ?? KeyUsage.ENCRYPT_DECRYPT;\n    if (denyLists[keyUsage].includes(keySpec)) {\n      throw new Error(`key spec '${keySpec}' is not valid with usage '${keyUsage}'`);\n    }\n\n    if (keySpec !== KeySpec.SYMMETRIC_DEFAULT && props.enableKeyRotation) {\n      throw new Error('key rotation cannot be enabled on asymmetric keys');\n    }\n\n    const defaultKeyPoliciesFeatureEnabled = FeatureFlags.of(this).isEnabled(cxapi.KMS_DEFAULT_KEY_POLICIES);\n\n    this.policy = props.policy ?? new iam.PolicyDocument();\n    if (defaultKeyPoliciesFeatureEnabled) {\n      if (props.trustAccountIdentities === false) {\n        throw new Error('`trustAccountIdentities` cannot be false if the @aws-cdk/aws-kms:defaultKeyPolicies feature flag is set');\n      }\n\n      this.trustAccountIdentities = true;\n      // Set the default key policy if one hasn't been provided by the user.\n      if (!props.policy) {\n        this.addDefaultAdminPolicy();\n      }\n    } else {\n      this.trustAccountIdentities = props.trustAccountIdentities ?? false;\n      if (this.trustAccountIdentities) {\n        this.addDefaultAdminPolicy();\n      } else {\n        this.addLegacyAdminPolicy();\n      }\n    }\n\n    let pendingWindowInDays;\n    if (props.pendingWindow) {\n      pendingWindowInDays = props.pendingWindow.toDays();\n      if (pendingWindowInDays < 7 || pendingWindowInDays > 30) {\n        throw new Error(`'pendingWindow' value must between 7 and 30 days. Received: ${pendingWindowInDays}`);\n      }\n    }\n\n    const resource = new CfnKey(this, 'Resource', {\n      description: props.description,\n      enableKeyRotation: props.enableKeyRotation,\n      enabled: props.enabled,\n      keySpec: props.keySpec,\n      keyUsage: props.keyUsage,\n      keyPolicy: this.policy,\n      pendingWindowInDays: pendingWindowInDays,\n    });\n\n    this.keyArn = resource.attrArn;\n    this.keyId = resource.ref;\n    resource.applyRemovalPolicy(props.removalPolicy);\n\n    (props.admins ?? []).forEach((p) => this.grantAdmin(p));\n\n    if (props.alias !== undefined) {\n      this.addAlias(props.alias);\n    }\n  }\n\n  /**\n   * Grant admins permissions using this key to the given principal\n   *\n   * Key administrators have permissions to manage the key (e.g., change permissions, revoke), but do not have permissions\n   * to use the key in cryptographic operations (e.g., encrypt, decrypt).\n   */\n  public grantAdmin(grantee: iam.IGrantable): iam.Grant {\n    return this.grant(grantee, ...perms.ADMIN_ACTIONS);\n  }\n\n  /**\n   * Adds the default key policy to the key. This policy gives the AWS account (root user) full access to the CMK,\n   * which reduces the risk of the CMK becoming unmanageable and enables IAM policies to allow access to the CMK.\n   * This is the same policy that is default when creating a Key via the KMS API or Console.\n   * @see https://docs.aws.amazon.com/kms/latest/developerguide/key-policies.html#key-policy-default\n   */\n  private addDefaultAdminPolicy() {\n    this.addToResourcePolicy(new iam.PolicyStatement({\n      resources: ['*'],\n      actions: ['kms:*'],\n      principals: [new iam.AccountRootPrincipal()],\n    }));\n  }\n\n  /**\n   * Grants the account admin privileges -- not full account access -- plus the GenerateDataKey action.\n   * The GenerateDataKey action was added for interop with S3 in https://github.com/aws/aws-cdk/issues/3458.\n   *\n   * This policy is discouraged and deprecated by the `@aws-cdk/aws-kms:defaultKeyPolicies` feature flag.\n   *\n   * @link https://docs.aws.amazon.com/kms/latest/developerguide/key-policies.html#key-policy-default\n   * @deprecated\n   */\n  private addLegacyAdminPolicy() {\n    // This is equivalent to `[...perms.ADMIN_ACTIONS, 'kms:GenerateDataKey']`,\n    // but keeping this explicit ordering for backwards-compatibility (changing the ordering causes resource updates)\n    const actions = [\n      'kms:Create*',\n      'kms:Describe*',\n      'kms:Enable*',\n      'kms:List*',\n      'kms:Put*',\n      'kms:Update*',\n      'kms:Revoke*',\n      'kms:Disable*',\n      'kms:Get*',\n      'kms:Delete*',\n      'kms:ScheduleKeyDeletion',\n      'kms:CancelKeyDeletion',\n      'kms:GenerateDataKey',\n      'kms:TagResource',\n      'kms:UntagResource',\n    ];\n\n    this.addToResourcePolicy(new iam.PolicyStatement({\n      resources: ['*'],\n      actions,\n      principals: [new iam.AccountRootPrincipal()],\n    }));\n  }\n}\n\n/**\n * Whether the given object is a Construct\n *\n * Normally we'd do `x instanceof Construct`, but that is not robust against\n * multiple copies of the `constructs` library on disk. This can happen\n * when upgrading and downgrading between v2 and v1, and in the use of CDK\n * Pipelines is going to an error that says \"Can't use Pipeline/Pipeline/Role in\n * a cross-environment fashion\", which is very confusing.\n */\nfunction isConstruct(x: any): x is Construct {\n  const sym = Symbol.for('constructs.Construct.node');\n  return (typeof x === 'object' && x &&\n    (x instanceof Construct // happy fast case\n    || !!(x as any).node // constructs v10\n    || !!(x as any)[sym])); // constructs v3\n}"]}
\No newline at end of file