UNPKG

cdk-ec2-key-pair

Version:

CDK Construct for managing EC2 key pairs

243 lines 40.3 kB
"use strict"; var _a; Object.defineProperty(exports, "__esModule", { value: true }); exports.KeyPair = exports.PublicKeyFormat = exports.LogLevel = exports.KeyType = void 0; const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); const aws_cdk_lib_1 = require("aws-cdk-lib"); const aws_ec2_1 = require("aws-cdk-lib/aws-ec2"); const path = require("path"); const types_1 = require("./types"); var types_2 = require("./types"); Object.defineProperty(exports, "KeyType", { enumerable: true, get: function () { return types_2.KeyType; } }); Object.defineProperty(exports, "LogLevel", { enumerable: true, get: function () { return types_2.LogLevel; } }); Object.defineProperty(exports, "PublicKeyFormat", { enumerable: true, get: function () { return types_2.PublicKeyFormat; } }); const resourceType = 'Custom::EC2-Key-Pair'; const ID = `CFN::Resource::${resourceType}`; const createdByTag = 'CreatedByCfnCustomResource'; const cleanID = ID.replace(/:+/g, '-'); const lambdaTimeout = 3; // minutes /** * An EC2 Key Pair */ class KeyPair extends aws_cdk_lib_1.Resource { get keyPairRef() { return { keyName: this.keyPairName, }; } /** * Defines a new EC2 Key Pair. The private key will be stored in AWS Secrets Manager */ constructor(scope, id, props) { super(scope, id); /** * The public key. * * Only filled, when `exposePublicKey = true` */ this.publicKeyValue = ''; /** * Name of the Key Pair */ this.keyPairName = ''; /** * ID of the Key Pair */ this.keyPairID = ''; /** * Fingerprint of the Key Pair */ this.keyPairFingerprint = ''; this.prefix = ''; if (props.removeKeySecretsAfterDays && (props.removeKeySecretsAfterDays < 0 || (props.removeKeySecretsAfterDays > 0 && props.removeKeySecretsAfterDays < 7) || props.removeKeySecretsAfterDays > 30)) { aws_cdk_lib_1.Annotations.of(this).addError(`Parameter removeKeySecretsAfterDays must be 0 or between 7 and 30. Got ${props.removeKeySecretsAfterDays}`); } if (props.publicKey?.length && props.publicKeyFormat !== undefined && props.publicKeyFormat !== types_1.PublicKeyFormat.OPENSSH) { aws_cdk_lib_1.Annotations.of(this).addError('When importing a key, the format has to be of type OpenSSH'); } if (props.keyType == types_1.KeyType.ED25519 && props.publicKeyFormat == types_1.PublicKeyFormat.PKCS1) { aws_cdk_lib_1.Annotations.of(this).addError('The public key format PKCS1 is not supported for key type ED25519'); } const stack = aws_cdk_lib_1.Stack.of(this).stackName; if (props.legacyLambdaName) { this.prefix = props.resourcePrefix ?? stack; if (this.prefix.length + cleanID.length > 62) { // Cloudformation limits names to 63 characters. aws_cdk_lib_1.Annotations.of(this).addError(`Cloudformation limits names to 63 characters. Prefix ${this.prefix} is too long to be used as a prefix for your roleName. Define parameter resourcePrefix?:`); } } this.lambdaFunction = this.ensureLambda(props.legacyLambdaName ?? false); this.tags = new aws_cdk_lib_1.TagManager(aws_cdk_lib_1.TagType.MAP, 'Custom::EC2-Key-Pair'); this.tags.setTag(createdByTag, ID); this.keyType = props.keyType ?? types_1.KeyType.RSA; this.publicKeyFormat = props.publicKeyFormat ?? types_1.PublicKeyFormat.SSH; const kmsPrivate = props.kmsPrivateKey ?? props.kms; const kmsPublic = props.kmsPublicKey ?? props.kms; const lambdaProperties = { /* eslint-disable @typescript-eslint/naming-convention */ Name: props.keyPairName, Description: props.description ?? '', KmsPrivate: kmsPrivate?.keyArn ?? 'alias/aws/secretsmanager', KmsPublic: kmsPublic?.keyArn ?? 'alias/aws/secretsmanager', PublicKey: props.publicKey ?? '', StorePublicKey: props.storePublicKey ? 'true' : 'false', ExposePublicKey: props.exposePublicKey ? 'true' : 'false', KeyType: this.keyType, PublicKeyFormat: props.publicKeyFormat ?? types_1.PublicKeyFormat.SSH, RemoveKeySecretsAfterDays: props.removeKeySecretsAfterDays ?? 0, SecretPrefix: props.secretPrefix ?? 'ec2-ssh-key/', StackName: stack, Tags: aws_cdk_lib_1.Lazy.any({ produce: () => this.tags.renderTags(), }), LogLevel: props.logLevel, /* eslint-enable @typescript-eslint/naming-convention */ }; const key = new aws_cdk_lib_1.CustomResource(this, `EC2-Key-Pair-${props.keyPairName}`, { serviceToken: this.lambdaFunction.functionArn, resourceType: resourceType, properties: lambdaProperties, }); this.privateKeySecret = aws_cdk_lib_1.aws_secretsmanager.Secret.fromSecretCompleteArn(this, 'PrivateKeySecret', key.getAttString('PrivateKeyARN')); if (props.storePublicKey) { this.publicKeySecret = aws_cdk_lib_1.aws_secretsmanager.Secret.fromSecretCompleteArn(this, 'PublicKeySecret', key.getAttString('PublicKeyARN')); } if (typeof props.kms !== 'undefined') { props.kms.grantEncryptDecrypt(this.lambdaFunction.role); key.node.addDependency(props.kms); key.node.addDependency(this.lambdaFunction.role); } if (typeof props.kmsPrivateKey !== 'undefined') { props.kmsPrivateKey.grantEncryptDecrypt(this.lambdaFunction.role); key.node.addDependency(props.kmsPrivateKey); key.node.addDependency(this.lambdaFunction.role); } if (typeof props.kmsPublicKey !== 'undefined') { props.kmsPublicKey.grantEncryptDecrypt(this.lambdaFunction.role); key.node.addDependency(props.kmsPublicKey); key.node.addDependency(this.lambdaFunction.role); } this.publicKeyValue = key.getAttString('PublicKeyValue'); this.keyPairName = key.getAttString('KeyPairName'); this.keyPairID = key.getAttString('KeyPairID'); this.keyPairFingerprint = key.getAttString('KeyPairFingerprint'); } ensureLambda(legacyLambdaName) { const stack = aws_cdk_lib_1.Stack.of(this); const constructName = legacyLambdaName ? 'EC2-Key-Name-Manager-Lambda' // this name was not intentional but we keep it for legacy resources : 'EC2-Key-Pair-Manager-Lambda'; const existing = stack.node.tryFindChild(constructName); if (existing) { return existing; } const resources = [`arn:${stack.partition}:ec2:*:*:key-pair/*`]; const statements = [ new aws_cdk_lib_1.aws_iam.PolicyStatement({ actions: ['ec2:DescribeKeyPairs'], resources: ['*'], }), new aws_cdk_lib_1.aws_iam.PolicyStatement({ actions: ['ec2:CreateKeyPair', 'ec2:CreateTags', 'ec2:ImportKeyPair'], conditions: { /* eslint-disable @typescript-eslint/naming-convention */ StringLike: { 'aws:RequestTag/CreatedByCfnCustomResource': ID, }, /* eslint-enable @typescript-eslint/naming-convention */ }, resources, }), new aws_cdk_lib_1.aws_iam.PolicyStatement({ // allow delete/update, only if createdByTag is set actions: ['ec2:CreateTags', 'ec2:DeleteKeyPair', 'ec2:DeleteTags'], conditions: { /* eslint-disable @typescript-eslint/naming-convention */ StringLike: { 'ec2:ResourceTag/CreatedByCfnCustomResource': ID, }, /* eslint-enable @typescript-eslint/naming-convention */ }, resources, }), new aws_cdk_lib_1.aws_iam.PolicyStatement({ // we need this to check if a secret exists before attempting to delete it actions: ['secretsmanager:ListSecrets'], resources: ['*'], }), new aws_cdk_lib_1.aws_iam.PolicyStatement({ actions: ['secretsmanager:CreateSecret', 'secretsmanager:TagResource'], conditions: { /* eslint-disable @typescript-eslint/naming-convention */ StringLike: { 'aws:RequestTag/CreatedByCfnCustomResource': ID, }, /* eslint-enable @typescript-eslint/naming-convention */ }, resources: ['*'], }), new aws_cdk_lib_1.aws_iam.PolicyStatement({ // allow delete/update, only if createdByTag is set actions: [ 'secretsmanager:DeleteResourcePolicy', 'secretsmanager:DeleteSecret', 'secretsmanager:DescribeSecret', 'secretsmanager:GetResourcePolicy', 'secretsmanager:GetSecretValue', 'secretsmanager:ListSecretVersionIds', 'secretsmanager:PutResourcePolicy', 'secretsmanager:PutSecretValue', 'secretsmanager:RestoreSecret', 'secretsmanager:UntagResource', 'secretsmanager:UpdateSecret', 'secretsmanager:UpdateSecretVersionStage', ], conditions: { /* eslint-disable @typescript-eslint/naming-convention */ StringLike: { 'secretsmanager:ResourceTag/CreatedByCfnCustomResource': ID, }, /* eslint-enable @typescript-eslint/naming-convention */ }, resources: ['*'], }), ]; const fn = new aws_cdk_lib_1.aws_lambda.Function(stack, constructName, { functionName: legacyLambdaName ? `${this.prefix}-${cleanID}` : undefined, description: 'Custom CFN resource: Manage EC2 Key Pairs', runtime: aws_cdk_lib_1.aws_lambda.Runtime.NODEJS_22_X, handler: 'index.handler', code: aws_cdk_lib_1.aws_lambda.Code.fromAsset(path.join(__dirname, '../lambda/code.zip')), timeout: aws_cdk_lib_1.Duration.minutes(lambdaTimeout), }); statements.forEach((statement) => { fn.role?.addToPrincipalPolicy(statement); }); return fn; } /** * Used internally to determine whether the key pair is compatible with an OS type. * * @internal */ _isOsCompatible(osType) { switch (this.keyType) { case types_1.KeyType.ED25519: return osType !== aws_ec2_1.OperatingSystemType.WINDOWS; default: return true; } } } exports.KeyPair = KeyPair; _a = JSII_RTTI_SYMBOL_1; KeyPair[_a] = { fqn: "cdk-ec2-key-pair.KeyPair", version: "5.0.0" }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Annotations,\n  CustomResource,\n  Duration,\n  ITaggable,\n  Lazy,\n  Resource,\n  ResourceProps,\n  Stack,\n  TagManager,\n  TagType,\n  aws_iam,\n  aws_kms,\n  aws_lambda,\n  aws_secretsmanager,\n} from 'aws-cdk-lib';\nimport {\n  IKeyPair,\n  KeyPairReference,\n  OperatingSystemType,\n} from 'aws-cdk-lib/aws-ec2';\nimport { Construct } from 'constructs';\nimport * as path from 'path';\nimport {\n  KeyType,\n  LogLevel,\n  PublicKeyFormat,\n  ResourceProperties,\n} from './types';\nexport { KeyType, LogLevel, PublicKeyFormat } from './types';\n\nconst resourceType = 'Custom::EC2-Key-Pair';\nconst ID = `CFN::Resource::${resourceType}`;\nconst createdByTag = 'CreatedByCfnCustomResource';\nconst cleanID = ID.replace(/:+/g, '-');\nconst lambdaTimeout = 3; // minutes\n\n/**\n * Definition of EC2 Key Pair\n */\nexport interface KeyPairProps extends ResourceProps {\n  /**\n   * Name of the Key Pair\n   *\n   * In AWS Secrets Manager the key will be prefixed with `ec2-ssh-key/`.\n   *\n   * The name can be up to 255 characters long. Valid characters include _, -, a-z, A-Z, and 0-9.\n   */\n  readonly keyPairName: string;\n\n  /**\n   * The description for the key in AWS Secrets Manager\n   * @default - ''\n   */\n  readonly description?: string;\n\n  /**\n   * The KMS key used to encrypt the Secrets Manager secrets with\n   *\n   * This needs to be a key created in the same stack. You cannot use a key imported via ARN, because the keys access policy will need to be modified.\n   *\n   * @default - `alias/aws/secretsmanager`\n   */\n  readonly kms?: aws_kms.Key;\n\n  /**\n   * The KMS key to use to encrypt the private key with\n   *\n   * This needs to be a key created in the same stack. You cannot use a key imported via ARN, because the keys access policy will need to be modified.\n   *\n   * If no value is provided, the property `kms` will be used instead.\n   *\n   * @default - `this.kms`\n   */\n  readonly kmsPrivateKey?: aws_kms.Key;\n\n  /**\n   * The KMS key to use to encrypt the public key with\n   *\n   * This needs to be a key created in the same stack. You cannot use a key imported via ARN, because the keys access policy will need to be modified.\n   *\n   * If no value is provided, the property `kms` will be used instead.\n   *\n   * @default - `this.kms`\n   */\n  readonly kmsPublicKey?: aws_kms.Key;\n\n  /**\n   * Import a public key instead of creating it\n   *\n   * If no public key is provided, a new key pair will be created.\n   */\n  readonly publicKey?: string;\n\n  /**\n   * Store the public key as a secret\n   *\n   * @default - false\n   */\n  readonly storePublicKey?: boolean;\n\n  /**\n   * Expose the public key as property `publicKeyValue`\n   *\n   * @default - false\n   */\n  readonly exposePublicKey?: boolean;\n\n  /**\n   * The type of key pair\n   *\n   * @default - RSA\n   */\n  readonly keyType?: KeyType;\n\n  /**\n   * Format for public key.\n   *\n   * Relevant only if the public key is stored and/or exposed.\n   *\n   * @default - SSH\n   */\n  readonly publicKeyFormat?: PublicKeyFormat;\n\n  /**\n   * When the resource is destroyed, after how many days the private and public key in the AWS Secrets Manager should be deleted.\n   *\n   * Valid values are 0 and 7 to 30\n   *\n   * @default 0\n   */\n  readonly removeKeySecretsAfterDays?: number;\n\n  /**\n   * Prefix for the secret in AWS Secrets Manager.\n   *\n   * @default `ec2-ssh-key/`\n   */\n  readonly secretPrefix?: string;\n\n  /**\n   * A prefix for all resource names.\n   *\n   * By default all resources are prefixed with the stack name to avoid collisions with other stacks. This might cause problems when you work with long stack names and can be overridden through this parameter.\n   *\n   * @default Name of the stack\n   */\n  readonly resourcePrefix?: string;\n\n  /**\n   * Whether to use the legacy name for the Lambda function, which backs the custom resource.\n   *\n   * Starting with v4 of this package, the Lambda function by default has no longer a fixed name.\n   *\n   * If you migrate from v3 to v4, you need to set this to `true` as CloudFormation does not allow to change the name of the Lambda function used by custom resource.\n   *\n   * @default false\n   */\n  readonly legacyLambdaName?: boolean;\n\n  /**\n   * The log level of the Lambda function\n   *\n   * @default LogLevel.warn\n   */\n  readonly logLevel?: LogLevel;\n}\n\n/**\n * An EC2 Key Pair\n */\nexport class KeyPair extends Resource implements ITaggable, IKeyPair {\n  /**\n   * The lambda function that is created\n   */\n  public readonly lambdaFunction: aws_lambda.IFunction;\n\n  /**\n   * The public key.\n   *\n   * Only filled, when `exposePublicKey = true`\n   */\n  public readonly publicKeyValue: string = '';\n\n  /**\n   * Name of the Key Pair\n   */\n  public readonly keyPairName: string = '';\n\n  /**\n   * ID of the Key Pair\n   */\n  public readonly keyPairID: string = '';\n\n  /**\n   * Fingerprint of the Key Pair\n   */\n  public readonly keyPairFingerprint: string = '';\n\n  /**\n   * Format of the public key\n   */\n  public readonly publicKeyFormat: PublicKeyFormat;\n\n  /**\n   * The private key secret in AWS Secrets Manager\n   */\n  public readonly privateKeySecret: aws_secretsmanager.ISecret;\n  /**\n   * The public key secret in AWS Secrets Manager\n   *\n   * Only available when `storePublicKey` is set to `true`.\n   */\n  public readonly publicKeySecret?: aws_secretsmanager.ISecret;\n\n  /**\n   * Type of the Key Pair\n   */\n  public readonly keyType: KeyType;\n\n  /**\n   * Resource tags\n   */\n  public readonly tags: TagManager;\n\n  public readonly prefix: string = '';\n\n  public get keyPairRef(): KeyPairReference {\n    return {\n      keyName: this.keyPairName,\n    };\n  }\n\n  /**\n   * Defines a new EC2 Key Pair. The private key will be stored in AWS Secrets Manager\n   */\n  constructor(scope: Construct, id: string, props: KeyPairProps) {\n    super(scope, id);\n\n    if (\n      props.removeKeySecretsAfterDays &&\n      (props.removeKeySecretsAfterDays < 0 ||\n        (props.removeKeySecretsAfterDays > 0 &&\n          props.removeKeySecretsAfterDays < 7) ||\n        props.removeKeySecretsAfterDays > 30)\n    ) {\n      Annotations.of(this).addError(\n        `Parameter removeKeySecretsAfterDays must be 0 or between 7 and 30. Got ${props.removeKeySecretsAfterDays}`,\n      );\n    }\n\n    if (\n      props.publicKey?.length &&\n      props.publicKeyFormat !== undefined &&\n      props.publicKeyFormat !== PublicKeyFormat.OPENSSH\n    ) {\n      Annotations.of(this).addError(\n        'When importing a key, the format has to be of type OpenSSH',\n      );\n    }\n\n    if (\n      props.keyType == KeyType.ED25519 &&\n      props.publicKeyFormat == PublicKeyFormat.PKCS1\n    ) {\n      Annotations.of(this).addError(\n        'The public key format PKCS1 is not supported for key type ED25519',\n      );\n    }\n\n    const stack = Stack.of(this).stackName;\n\n    if (props.legacyLambdaName) {\n      this.prefix = props.resourcePrefix ?? stack;\n      if (this.prefix.length + cleanID.length > 62) {\n        // Cloudformation limits names to 63 characters.\n        Annotations.of(this).addError(\n          `Cloudformation limits names to 63 characters.\n           Prefix ${this.prefix} is too long to be used as a prefix for your roleName. Define parameter resourcePrefix?:`,\n        );\n      }\n    }\n    this.lambdaFunction = this.ensureLambda(props.legacyLambdaName ?? false);\n\n    this.tags = new TagManager(TagType.MAP, 'Custom::EC2-Key-Pair');\n    this.tags.setTag(createdByTag, ID);\n\n    this.keyType = props.keyType ?? KeyType.RSA;\n    this.publicKeyFormat = props.publicKeyFormat ?? PublicKeyFormat.SSH;\n\n    const kmsPrivate = props.kmsPrivateKey ?? props.kms;\n    const kmsPublic = props.kmsPublicKey ?? props.kms;\n\n    const lambdaProperties: ResourceProperties = {\n      /* eslint-disable @typescript-eslint/naming-convention */\n      Name: props.keyPairName,\n      Description: props.description ?? '',\n      KmsPrivate: kmsPrivate?.keyArn ?? 'alias/aws/secretsmanager',\n      KmsPublic: kmsPublic?.keyArn ?? 'alias/aws/secretsmanager',\n      PublicKey: props.publicKey ?? '',\n      StorePublicKey: props.storePublicKey ? 'true' : 'false',\n      ExposePublicKey: props.exposePublicKey ? 'true' : 'false',\n      KeyType: this.keyType,\n      PublicKeyFormat: props.publicKeyFormat ?? PublicKeyFormat.SSH,\n      RemoveKeySecretsAfterDays: props.removeKeySecretsAfterDays ?? 0,\n      SecretPrefix: props.secretPrefix ?? 'ec2-ssh-key/',\n      StackName: stack,\n      Tags: Lazy.any({\n        produce: () => this.tags.renderTags() as Record<string, string>,\n      }) as unknown as Record<string, string>,\n      LogLevel: props.logLevel,\n      /* eslint-enable @typescript-eslint/naming-convention */\n    };\n\n    const key = new CustomResource(this, `EC2-Key-Pair-${props.keyPairName}`, {\n      serviceToken: this.lambdaFunction.functionArn,\n      resourceType: resourceType,\n      properties: lambdaProperties,\n    });\n\n    this.privateKeySecret = aws_secretsmanager.Secret.fromSecretCompleteArn(\n      this,\n      'PrivateKeySecret',\n      key.getAttString('PrivateKeyARN'),\n    );\n\n    if (props.storePublicKey) {\n      this.publicKeySecret = aws_secretsmanager.Secret.fromSecretCompleteArn(\n        this,\n        'PublicKeySecret',\n        key.getAttString('PublicKeyARN'),\n      );\n    }\n\n    if (typeof props.kms !== 'undefined') {\n      props.kms.grantEncryptDecrypt(this.lambdaFunction.role!);\n      key.node.addDependency(props.kms);\n      key.node.addDependency(this.lambdaFunction.role!);\n    }\n\n    if (typeof props.kmsPrivateKey !== 'undefined') {\n      props.kmsPrivateKey.grantEncryptDecrypt(this.lambdaFunction.role!);\n      key.node.addDependency(props.kmsPrivateKey);\n      key.node.addDependency(this.lambdaFunction.role!);\n    }\n\n    if (typeof props.kmsPublicKey !== 'undefined') {\n      props.kmsPublicKey.grantEncryptDecrypt(this.lambdaFunction.role!);\n      key.node.addDependency(props.kmsPublicKey);\n      key.node.addDependency(this.lambdaFunction.role!);\n    }\n\n    this.publicKeyValue = key.getAttString('PublicKeyValue');\n    this.keyPairName = key.getAttString('KeyPairName');\n    this.keyPairID = key.getAttString('KeyPairID');\n    this.keyPairFingerprint = key.getAttString('KeyPairFingerprint');\n  }\n\n  private ensureLambda(legacyLambdaName: boolean): aws_lambda.Function {\n    const stack = Stack.of(this);\n    const constructName = legacyLambdaName\n      ? 'EC2-Key-Name-Manager-Lambda' // this name was not intentional but we keep it for legacy resources\n      : 'EC2-Key-Pair-Manager-Lambda';\n    const existing = stack.node.tryFindChild(constructName);\n    if (existing) {\n      return existing as aws_lambda.Function;\n    }\n\n    const resources = [`arn:${stack.partition}:ec2:*:*:key-pair/*`];\n\n    const statements = [\n      new aws_iam.PolicyStatement({\n        actions: ['ec2:DescribeKeyPairs'],\n        resources: ['*'],\n      }),\n      new aws_iam.PolicyStatement({\n        actions: ['ec2:CreateKeyPair', 'ec2:CreateTags', 'ec2:ImportKeyPair'],\n        conditions: {\n          /* eslint-disable @typescript-eslint/naming-convention */\n          StringLike: {\n            'aws:RequestTag/CreatedByCfnCustomResource': ID,\n          },\n          /* eslint-enable @typescript-eslint/naming-convention */\n        },\n        resources,\n      }),\n      new aws_iam.PolicyStatement({\n        // allow delete/update, only if createdByTag is set\n        actions: ['ec2:CreateTags', 'ec2:DeleteKeyPair', 'ec2:DeleteTags'],\n        conditions: {\n          /* eslint-disable @typescript-eslint/naming-convention */\n          StringLike: {\n            'ec2:ResourceTag/CreatedByCfnCustomResource': ID,\n          },\n          /* eslint-enable @typescript-eslint/naming-convention */\n        },\n        resources,\n      }),\n\n      new aws_iam.PolicyStatement({\n        // we need this to check if a secret exists before attempting to delete it\n        actions: ['secretsmanager:ListSecrets'],\n        resources: ['*'],\n      }),\n      new aws_iam.PolicyStatement({\n        actions: ['secretsmanager:CreateSecret', 'secretsmanager:TagResource'],\n        conditions: {\n          /* eslint-disable @typescript-eslint/naming-convention */\n          StringLike: {\n            'aws:RequestTag/CreatedByCfnCustomResource': ID,\n          },\n          /* eslint-enable @typescript-eslint/naming-convention */\n        },\n        resources: ['*'],\n      }),\n      new aws_iam.PolicyStatement({\n        // allow delete/update, only if createdByTag is set\n        actions: [\n          'secretsmanager:DeleteResourcePolicy',\n          'secretsmanager:DeleteSecret',\n          'secretsmanager:DescribeSecret',\n          'secretsmanager:GetResourcePolicy',\n          'secretsmanager:GetSecretValue',\n          'secretsmanager:ListSecretVersionIds',\n          'secretsmanager:PutResourcePolicy',\n          'secretsmanager:PutSecretValue',\n          'secretsmanager:RestoreSecret',\n          'secretsmanager:UntagResource',\n          'secretsmanager:UpdateSecret',\n          'secretsmanager:UpdateSecretVersionStage',\n        ],\n        conditions: {\n          /* eslint-disable @typescript-eslint/naming-convention */\n          StringLike: {\n            'secretsmanager:ResourceTag/CreatedByCfnCustomResource': ID,\n          },\n          /* eslint-enable @typescript-eslint/naming-convention */\n        },\n        resources: ['*'],\n      }),\n    ];\n\n    const fn = new aws_lambda.Function(stack, constructName, {\n      functionName: legacyLambdaName ? `${this.prefix}-${cleanID}` : undefined,\n      description: 'Custom CFN resource: Manage EC2 Key Pairs',\n      runtime: aws_lambda.Runtime.NODEJS_22_X,\n      handler: 'index.handler',\n      code: aws_lambda.Code.fromAsset(\n        path.join(__dirname, '../lambda/code.zip'),\n      ),\n      timeout: Duration.minutes(lambdaTimeout),\n    });\n    statements.forEach((statement) => {\n      fn.role?.addToPrincipalPolicy(statement);\n    });\n\n    return fn;\n  }\n\n  /**\n   * Used internally to determine whether the key pair is compatible with an OS type.\n   *\n   * @internal\n   */\n  public _isOsCompatible(osType: OperatingSystemType): boolean {\n    switch (this.keyType) {\n      case KeyType.ED25519:\n        return osType !== OperatingSystemType.WINDOWS;\n      default:\n        return true;\n    }\n  }\n}\n"]}