aws-delivlib
Version:
A fabulous library for defining continuous pipelines for building, testing and releasing code libraries.
192 lines • 26.5 kB
JavaScript
"use strict";
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
var desc = Object.getOwnPropertyDescriptor(m, k);
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
desc = { enumerable: true, get: function() { return m[k]; } };
}
Object.defineProperty(o, k2, desc);
}) : (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
o[k2] = m[k];
}));
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
Object.defineProperty(o, "default", { enumerable: true, value: v });
}) : function(o, v) {
o["default"] = v;
});
var __importStar = (this && this.__importStar) || function (mod) {
if (mod && mod.__esModule) return mod;
var result = {};
if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
__setModuleDefault(result, mod);
return result;
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.RsaPrivateKeySecret = void 0;
const path = __importStar(require("path"));
const aws_cdk_lib_1 = require("aws-cdk-lib");
const aws_ecr_assets_1 = require("aws-cdk-lib/aws-ecr-assets");
const constructs_1 = require("constructs");
const certificate_signing_request_1 = require("./certificate-signing-request");
const util_1 = require("../util");
/**
* An OpenSSL-generated RSA Private Key. It can for example be used to obtain a Certificate signed by a Certificate
* Authority through the use of the ``CertificateSigningRequest`` construct (or via the
* ``#newCertificateSigningRequest``) method.
*/
class RsaPrivateKeySecret extends constructs_1.Construct {
constructor(parent, id, props) {
super(parent, id);
const codeLocation = path.resolve(__dirname, '..', 'custom-resource-handlers');
// change the resource id to force deleting existing function, and create new one, as Package type change is not allowed
this.customResource = new aws_cdk_lib_1.aws_lambda.SingletonFunction(this, 'ResourceHandlerV2', {
lambdaPurpose: 'RSAPrivate-Key',
// change the uuid to force deleting existing function, and create new one, as Package type change is not allowed
uuid: '517D342F-A590-447B-B525-5D06E403A406',
description: 'Generates an RSA Private Key and stores it in AWS Secrets Manager',
runtime: aws_cdk_lib_1.aws_lambda.Runtime.FROM_IMAGE,
handler: aws_cdk_lib_1.aws_lambda.Handler.FROM_IMAGE,
code: new aws_cdk_lib_1.aws_lambda.AssetImageCode(codeLocation, {
file: 'Dockerfile',
platform: aws_ecr_assets_1.Platform.LINUX_AMD64,
buildArgs: {
FUN_SRC_DIR: 'private-key',
},
invalidation: {
buildArgs: true,
},
}),
timeout: aws_cdk_lib_1.Duration.seconds(300),
});
this.secretArnLike = aws_cdk_lib_1.Stack.of(this).formatArn({
service: 'secretsmanager',
resource: 'secret',
arnFormat: aws_cdk_lib_1.ArnFormat.COLON_RESOURCE_NAME,
// The ARN of a secret has "-" followed by 6 random characters appended at the end
resourceName: `${props.secretName}-??????`,
});
this.customResource.addToRolePolicy(new aws_cdk_lib_1.aws_iam.PolicyStatement({
actions: [
'secretsmanager:CreateSecret',
'secretsmanager:DeleteSecret',
'secretsmanager:UpdateSecret',
],
resources: [this.secretArnLike],
}));
if (props.secretEncryptionKey) {
props.secretEncryptionKey.addToResourcePolicy(new aws_cdk_lib_1.aws_iam.PolicyStatement({
// description: `Allow use via AWS Secrets Manager by CustomResource handler ${customResource.functionName}`,
principals: [new aws_cdk_lib_1.aws_iam.ArnPrincipal(this.customResource.role.roleArn)],
actions: ['kms:Decrypt', 'kms:GenerateDataKey'],
resources: ['*'],
conditions: {
StringEquals: {
'kms:ViaService': `secretsmanager.${aws_cdk_lib_1.Stack.of(this).region}.amazonaws.com`,
},
ArnLike: {
'kms:EncryptionContext:SecretARN': this.secretArnLike,
},
},
}));
}
//change the custom resource id to force recreating new one because the change of the underneath lambda function
const privateKey = new aws_cdk_lib_1.CustomResource(this, 'ResourceV2', {
serviceToken: this.customResource.functionArn,
resourceType: 'Custom::RsaPrivateKeySecret',
pascalCaseProperties: true,
properties: {
resourceVersion: (0, util_1.hashFileOrDirectory)(codeLocation),
description: props.description,
keySize: props.keySize,
secretName: props.secretName,
kmsKeyId: props.secretEncryptionKey && props.secretEncryptionKey.keyArn,
},
removalPolicy: props.removalPolicy || aws_cdk_lib_1.RemovalPolicy.RETAIN,
});
if (this.customResource.role) {
privateKey.node.addDependency(this.customResource.role);
if (props.secretEncryptionKey) {
// Modeling as a separate Policy to evade a dependency cycle (Role -> Key -> Role), as the Key refers to the
// role in it's resource policy.
privateKey.node.addDependency(new aws_cdk_lib_1.aws_iam.Policy(this, 'GrantLambdaRoleKeyAccess', {
roles: [this.customResource.role],
statements: [
new aws_cdk_lib_1.aws_iam.PolicyStatement({
// description: `AWSSecretsManager${props.secretName.replace(/[^0-9A-Za-z]/g, '')}CMK`,
actions: ['kms:Decrypt', 'kms:GenerateDataKey'],
resources: [props.secretEncryptionKey.keyArn],
conditions: {
StringEquals: {
'kms:ViaService': `secretsmanager.${aws_cdk_lib_1.Stack.of(this).region}.amazonaws.com`,
},
StringLike: { 'kms:EncryptionContext:SecretARN': [this.secretArnLike, 'RequestToValidateKeyAccess'] },
},
}),
],
}));
}
}
this.masterKey = props.secretEncryptionKey;
this.secretArn = privateKey.getAtt('SecretArn').toString();
}
/**
* Creates a new CSR resource using this private key.
*
* @param id the ID of the construct in the construct tree.
* @param dn the distinguished name to record on the CSR.
* @param keyUsage the intended key usage (for example: "critical,digitalSignature")
* @param extendedKeyUsage the indended extended key usage, if any (for example: "critical,digitalSignature")
*
* @returns a new ``CertificateSigningRequest`` instance that can be used to access the actual CSR document.
*/
newCertificateSigningRequest(id, dn, keyUsage, extendedKeyUsage) {
return new certificate_signing_request_1.CertificateSigningRequest(this, id, {
privateKey: this,
dn,
keyUsage,
extendedKeyUsage,
});
}
/**
* Allows a given IAM Role to read the secret value.
*
* @param grantee the principal to which permissions should be granted.
*/
grantGetSecretValue(grantee) {
grantee.addToPrincipalPolicy(new aws_cdk_lib_1.aws_iam.PolicyStatement({
actions: ['secretsmanager:GetSecretValue'],
resources: [this.secretArn],
}));
if (this.masterKey) {
// Add a key grant since we're using a CMK
this.masterKey.addToResourcePolicy(new aws_cdk_lib_1.aws_iam.PolicyStatement({
actions: ['kms:Decrypt'],
resources: ['*'],
principals: [grantee.grantPrincipal],
conditions: {
StringEquals: {
'kms:ViaService': `secretsmanager.${aws_cdk_lib_1.Stack.of(this).region}.amazonaws.com`,
},
ArnLike: {
'kms:EncryptionContext:SecretARN': this.secretArnLike,
},
},
}));
grantee.addToPrincipalPolicy(new aws_cdk_lib_1.aws_iam.PolicyStatement({
actions: ['kms:Decrypt'],
resources: [this.masterKey.keyArn],
conditions: {
StringEquals: {
'kms:ViaService': `secretsmanager.${aws_cdk_lib_1.Stack.of(this).region}.amazonaws.com`,
},
ArnEquals: {
'kms:EncryptionContext:SecretARN': this.secretArn,
},
},
}));
}
}
}
exports.RsaPrivateKeySecret = RsaPrivateKeySecret;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"private-key.js","sourceRoot":"","sources":["private-key.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport {\n  Duration, RemovalPolicy, Stack,\n  ArnFormat, CustomResource,\n  aws_iam as iam,\n  aws_kms as kms,\n  aws_lambda as lambda,\n} from 'aws-cdk-lib';\nimport { Platform } from 'aws-cdk-lib/aws-ecr-assets';\nimport { Construct } from 'constructs';\nimport { CertificateSigningRequest, DistinguishedName } from './certificate-signing-request';\nimport { hashFileOrDirectory } from '../util';\n\n\nexport interface RsaPrivateKeySecretProps {\n  /**\n   * The modulus size of the RSA key that will be generated.\n   *\n   * The NIST publishes a document that provides guidance on how to select an appropriate key size:\n   * @see https://csrc.nist.gov/publications/detail/sp/800-57-part-1/rev-4/final\n   */\n  keySize: number;\n\n  /**\n   * The name of the AWS Secrets Manager entity that will be created to hold the private key.\n   */\n  secretName: string;\n\n  /**\n   * The description to attach to the AWS Secrets Manager entity that will hold the private key.\n   */\n  description?: string;\n\n  /**\n   * The KMS key to be used for encrypting the AWS Secrets Manager entity.\n   *\n   * @default the default KMS key will be used in accordance with AWS Secrets Manager default behavior.\n   */\n  secretEncryptionKey?: kms.IKey;\n\n  /**\n   * The deletion policy to apply on the Private Key secret.\n   *\n   * @default Retain\n   */\n  removalPolicy?: RemovalPolicy;\n}\n\n/**\n * An OpenSSL-generated RSA Private Key. It can for example be used to obtain a Certificate signed by a Certificate\n * Authority through the use of the ``CertificateSigningRequest`` construct (or via the\n * ``#newCertificateSigningRequest``) method.\n */\nexport class RsaPrivateKeySecret extends Construct {\n  /**\n   * The ARN of the secret that holds the private key.\n   */\n  public secretArn: string;\n  public customResource: lambda.SingletonFunction;\n\n  private secretArnLike: string;\n  private masterKey?: kms.IKey;\n\n  constructor(parent: Construct, id: string, props: RsaPrivateKeySecretProps) {\n    super(parent, id);\n\n    const codeLocation = path.resolve(__dirname, '..', 'custom-resource-handlers');\n    // change the resource id to force deleting existing function, and create new one, as Package type change is not allowed\n    this.customResource = new lambda.SingletonFunction(this, 'ResourceHandlerV2', {\n      lambdaPurpose: 'RSAPrivate-Key',\n      // change the uuid to force deleting existing function, and create new one, as Package type change is not allowed\n      uuid: '517D342F-A590-447B-B525-5D06E403A406',\n      description: 'Generates an RSA Private Key and stores it in AWS Secrets Manager',\n      runtime: lambda.Runtime.FROM_IMAGE,\n      handler: lambda.Handler.FROM_IMAGE,\n      code: new lambda.AssetImageCode(codeLocation, {\n        file: 'Dockerfile',\n        platform: Platform.LINUX_AMD64,\n        buildArgs: {\n          FUN_SRC_DIR: 'private-key',\n        },\n        invalidation: {\n          buildArgs: true,\n        },\n      }),\n      timeout: Duration.seconds(300),\n    });\n\n    this.secretArnLike = Stack.of(this).formatArn({\n      service: 'secretsmanager',\n      resource: 'secret',\n      arnFormat: ArnFormat.COLON_RESOURCE_NAME,\n      // The ARN of a secret has \"-\" followed by 6 random characters appended at the end\n      resourceName: `${props.secretName}-??????`,\n    });\n    this.customResource.addToRolePolicy(new iam.PolicyStatement({\n      actions: [\n        'secretsmanager:CreateSecret',\n        'secretsmanager:DeleteSecret',\n        'secretsmanager:UpdateSecret',\n      ],\n      resources: [this.secretArnLike],\n    }));\n\n    if (props.secretEncryptionKey) {\n      props.secretEncryptionKey.addToResourcePolicy(new iam.PolicyStatement({\n        // description: `Allow use via AWS Secrets Manager by CustomResource handler ${customResource.functionName}`,\n        principals: [new iam.ArnPrincipal(this.customResource.role!.roleArn)],\n        actions: ['kms:Decrypt', 'kms:GenerateDataKey'],\n        resources: ['*'],\n        conditions: {\n          StringEquals: {\n            'kms:ViaService': `secretsmanager.${Stack.of(this).region}.amazonaws.com`,\n          },\n          ArnLike: {\n            'kms:EncryptionContext:SecretARN': this.secretArnLike,\n          },\n        },\n      }));\n    }\n\n    //change the custom resource id to force recreating new one because the change of the underneath lambda function\n    const privateKey = new CustomResource(this, 'ResourceV2', {\n      serviceToken: this.customResource.functionArn,\n      resourceType: 'Custom::RsaPrivateKeySecret',\n      pascalCaseProperties: true,\n      properties: {\n        resourceVersion: hashFileOrDirectory(codeLocation),\n        description: props.description,\n        keySize: props.keySize,\n        secretName: props.secretName,\n        kmsKeyId: props.secretEncryptionKey && props.secretEncryptionKey.keyArn,\n      },\n      removalPolicy: props.removalPolicy || RemovalPolicy.RETAIN,\n    });\n    if (this.customResource.role) {\n      privateKey.node.addDependency(this.customResource.role);\n      if (props.secretEncryptionKey) {\n        // Modeling as a separate Policy to evade a dependency cycle (Role -> Key -> Role), as the Key refers to the\n        // role in it's resource policy.\n        privateKey.node.addDependency(new iam.Policy(this, 'GrantLambdaRoleKeyAccess', {\n          roles: [this.customResource.role],\n          statements: [\n            new iam.PolicyStatement({\n              // description: `AWSSecretsManager${props.secretName.replace(/[^0-9A-Za-z]/g, '')}CMK`,\n              actions: ['kms:Decrypt', 'kms:GenerateDataKey'],\n              resources: [props.secretEncryptionKey.keyArn],\n              conditions: {\n                StringEquals: {\n                  'kms:ViaService': `secretsmanager.${Stack.of(this).region}.amazonaws.com`,\n                },\n                StringLike: { 'kms:EncryptionContext:SecretARN': [this.secretArnLike, 'RequestToValidateKeyAccess'] },\n              },\n            }),\n          ],\n        }));\n      }\n    }\n\n    this.masterKey = props.secretEncryptionKey;\n    this.secretArn = privateKey.getAtt('SecretArn').toString();\n  }\n\n  /**\n   * Creates a new CSR resource using this private key.\n   *\n   * @param id               the ID of the construct in the construct tree.\n   * @param dn               the distinguished name to record on the CSR.\n   * @param keyUsage         the intended key usage (for example: \"critical,digitalSignature\")\n   * @param extendedKeyUsage the indended extended key usage, if any (for example: \"critical,digitalSignature\")\n   *\n   * @returns a new ``CertificateSigningRequest`` instance that can be used to access the actual CSR document.\n   */\n  public newCertificateSigningRequest(id: string, dn: DistinguishedName, keyUsage: string, extendedKeyUsage?: string) {\n    return new CertificateSigningRequest(this, id, {\n      privateKey: this,\n      dn,\n      keyUsage,\n      extendedKeyUsage,\n    });\n  }\n\n  /**\n   * Allows a given IAM Role to read the secret value.\n   *\n   * @param grantee the principal to which permissions should be granted.\n   */\n  public grantGetSecretValue(grantee: iam.IPrincipal): void {\n    grantee.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['secretsmanager:GetSecretValue'],\n      resources: [this.secretArn],\n    }));\n    if (this.masterKey) {\n      // Add a key grant since we're using a CMK\n      this.masterKey.addToResourcePolicy(new iam.PolicyStatement({\n        actions: ['kms:Decrypt'],\n        resources: ['*'],\n        principals: [grantee.grantPrincipal],\n        conditions: {\n          StringEquals: {\n            'kms:ViaService': `secretsmanager.${Stack.of(this).region}.amazonaws.com`,\n          },\n          ArnLike: {\n            'kms:EncryptionContext:SecretARN': this.secretArnLike,\n          },\n        },\n      }));\n      grantee.addToPrincipalPolicy(new iam.PolicyStatement({\n        actions: ['kms:Decrypt'],\n        resources: [this.masterKey.keyArn],\n        conditions: {\n          StringEquals: {\n            'kms:ViaService': `secretsmanager.${Stack.of(this).region}.amazonaws.com`,\n          },\n          ArnEquals: {\n            'kms:EncryptionContext:SecretARN': this.secretArn,\n          },\n        },\n      }));\n    }\n  }\n}\n"]}