1 | ;
|
2 | var _a;
|
3 | Object.defineProperty(exports, "__esModule", { value: true });
|
4 | exports.Key = exports.KeyUsage = exports.KeySpec = void 0;
|
5 | const jsiiDeprecationWarnings = require("../.warnings.jsii.js");
|
6 | const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
7 | const iam = require("@aws-cdk/aws-iam");
|
8 | const cxschema = require("@aws-cdk/cloud-assembly-schema");
|
9 | const core_1 = require("@aws-cdk/core");
|
10 | const cxapi = require("@aws-cdk/cx-api");
|
11 | const constructs_1 = require("constructs");
|
12 | const alias_1 = require("./alias");
|
13 | const kms_generated_1 = require("./kms.generated");
|
14 | const perms = require("./private/perms");
|
15 | class 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 | */
|
170 | var 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 | */
|
227 | var 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 | */
|
243 | class 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 | }
|
515 | exports.Key = Key;
|
516 | _a = JSII_RTTI_SYMBOL_1;
|
517 | Key[_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 | */
|
527 | function 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 |