aws-delivlib
Version:
A fabulous library for defining continuous pipelines for building, testing and releasing code libraries.
108 lines • 15.9 kB
JavaScript
;
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.CertificateSigningRequest = 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 util_1 = require("../util");
/**
* Creates a Certificate Signing Request (CSR), which will allow a Certificate Authority to provide a signed certificate
* that uses the specified RSA Private Key. A CSR document can usually be shared publicly, however it must be noted that
* the information provided in the ``dn`` fields, information about the public key and the intended ley usage will be
* readable by anyone who can access the CSR.
*
* @see https://www.openssl.org/docs/manmaster/man1/req.html
*/
class CertificateSigningRequest 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
const customResource = new aws_cdk_lib_1.aws_lambda.SingletonFunction(this, 'ResourceHandlerV2', {
// change the uuid to force deleting existing function, and create new one, as Package type change is not allowed
uuid: 'F0641C15-2BC0-481E-94BA-7BF43F8BBDE3',
lambdaPurpose: 'CreateCSR',
description: 'Creates a Certificate Signing Request document for an x509 certificate',
architecture: aws_cdk_lib_1.aws_lambda.Architecture.X86_64,
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: 'certificate-signing-request',
},
invalidation: {
buildArgs: true,
},
}),
timeout: aws_cdk_lib_1.Duration.seconds(300),
});
const outputBucket = new aws_cdk_lib_1.aws_s3.Bucket(this, 'Bucket', {
// CSRs can be easily re-created if lost or corrupt, so we can let those get to a black hole, no worries.
autoDeleteObjects: true,
removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY,
encryption: aws_cdk_lib_1.aws_s3.BucketEncryption.S3_MANAGED,
enforceSSL: true,
});
outputBucket.grantReadWrite(customResource);
this.outputBucket = outputBucket;
//change the custom resource id to force recreating new one because the change of the underneath lambda function
const csr = new aws_cdk_lib_1.CustomResource(this, 'ResourceV2', {
serviceToken: customResource.functionArn,
resourceType: 'Custom::CertificateSigningRequest',
pascalCaseProperties: true,
properties: {
resourceVersion: (0, util_1.hashFileOrDirectory)(codeLocation),
// Private key
privateKeySecretId: props.privateKey.secretArn,
// Distinguished name
dnCommonName: props.dn.commonName,
dnCountry: props.dn.country,
dnStateOrProvince: props.dn.stateOrProvince,
dnLocality: props.dn.locality,
dnOrganizationName: props.dn.organizationName,
dnOrganizationalUnitName: props.dn.organizationalUnitName,
dnEmailAddress: props.dn.emailAddress,
// Key Usage
extendedKeyUsage: props.extendedKeyUsage || '',
keyUsage: props.keyUsage,
// Ouput location
outputBucket: outputBucket.bucketName,
},
});
if (customResource.role) {
// Make sure the permissions are all good before proceeding
csr.node.addDependency(customResource.role);
props.privateKey.grantGetSecretValue(customResource.role);
}
this.pemRequest = csr.getAtt('CSR').toString();
this.selfSignedPemCertificate = csr.getAtt('SelfSignedCertificate').toString();
}
}
exports.CertificateSigningRequest = CertificateSigningRequest;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"certificate-signing-request.js","sourceRoot":"","sources":["certificate-signing-request.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport {\n  Duration,\n  CustomResource,\n  aws_lambda as lambda,\n  aws_s3 as s3,\n  RemovalPolicy,\n} from 'aws-cdk-lib';\nimport { Platform } from 'aws-cdk-lib/aws-ecr-assets';\nimport { Construct } from 'constructs';\nimport { RsaPrivateKeySecret } from './private-key';\nimport { hashFileOrDirectory } from '../util';\n\n\nexport interface CertificateSigningRequestProps {\n  /**\n   * The RSA Private Key to use for this CSR.\n   */\n  privateKey: RsaPrivateKeySecret;\n  /**\n   * The Distinguished Name for this CSR.\n   */\n  dn: DistinguishedName;\n  /**\n   * The key usage requests for this CSR.\n   *\n   * @example critical,digitalSignature\n   */\n  keyUsage: string;\n  /**\n   * The extended key usage requests for this CSR.\n   *\n   * @example critical,codeSigning\n   */\n  extendedKeyUsage?: string;\n}\n\n/**\n * Creates a Certificate Signing Request (CSR), which will allow a Certificate Authority to provide a signed certificate\n * that uses the specified RSA Private Key. A CSR document can usually be shared publicly, however it must be noted that\n * the information provided in the ``dn`` fields, information about the public key and the intended ley usage will be\n * readable by anyone who can access the CSR.\n *\n * @see https://www.openssl.org/docs/manmaster/man1/req.html\n */\nexport class CertificateSigningRequest extends Construct {\n  /**\n   * The S3 URL to the CSR document.\n   */\n  public readonly pemRequest: string;\n\n  /**\n   * The S3 URL to a self-signed certificate that corresponds with this CSR.\n   */\n  public readonly selfSignedPemCertificate: string;\n\n  /**\n   * The S3 bucket where the self-signed certificate is stored.\n   */\n  public readonly outputBucket: s3.IBucket;\n\n  constructor(parent: Construct, id: string, props: CertificateSigningRequestProps) {\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    const customResource = new lambda.SingletonFunction(this, 'ResourceHandlerV2', {\n      // change the uuid to force deleting existing function, and create new one, as Package type change is not allowed\n      uuid: 'F0641C15-2BC0-481E-94BA-7BF43F8BBDE3',\n      lambdaPurpose: 'CreateCSR',\n      description: 'Creates a Certificate Signing Request document for an x509 certificate',\n      architecture: lambda.Architecture.X86_64,\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: 'certificate-signing-request',\n        },\n        invalidation: {\n          buildArgs: true,\n        },\n      }),\n      timeout: Duration.seconds(300),\n    });\n\n    const outputBucket = new s3.Bucket(this, 'Bucket', {\n      // CSRs can be easily re-created if lost or corrupt, so we can let those get to a black hole, no worries.\n      autoDeleteObjects: true,\n      removalPolicy: RemovalPolicy.DESTROY,\n      encryption: s3.BucketEncryption.S3_MANAGED,\n      enforceSSL: true,\n    });\n    outputBucket.grantReadWrite(customResource);\n    this.outputBucket = outputBucket;\n\n    //change the custom resource id to force recreating new one because the change of the underneath lambda function\n    const csr = new CustomResource(this, 'ResourceV2', {\n      serviceToken: customResource.functionArn,\n      resourceType: 'Custom::CertificateSigningRequest',\n      pascalCaseProperties: true,\n      properties: {\n        resourceVersion: hashFileOrDirectory(codeLocation),\n        // Private key\n        privateKeySecretId: props.privateKey.secretArn,\n        // Distinguished name\n        dnCommonName: props.dn.commonName,\n        dnCountry: props.dn.country,\n        dnStateOrProvince: props.dn.stateOrProvince,\n        dnLocality: props.dn.locality,\n        dnOrganizationName: props.dn.organizationName,\n        dnOrganizationalUnitName: props.dn.organizationalUnitName,\n        dnEmailAddress: props.dn.emailAddress,\n        // Key Usage\n        extendedKeyUsage: props.extendedKeyUsage || '',\n        keyUsage: props.keyUsage,\n        // Ouput location\n        outputBucket: outputBucket.bucketName,\n      },\n    });\n    if (customResource.role) {\n      // Make sure the permissions are all good before proceeding\n      csr.node.addDependency(customResource.role);\n      props.privateKey.grantGetSecretValue(customResource.role);\n    }\n\n    this.pemRequest = csr.getAtt('CSR').toString();\n    this.selfSignedPemCertificate = csr.getAtt('SelfSignedCertificate').toString();\n  }\n}\n\n/**\n * Fields that compose the distinguished name of a certificate\n */\nexport interface DistinguishedName {\n  /** The Common Name (CN) */\n  commonName: string;\n  /** The email address (emailAddress) */\n  emailAddress: string;\n\n  /** The Country (C) */\n  country: string;\n  /** The State or Province (ST) */\n  stateOrProvince: string;\n  /** The locality (L) */\n  locality: string;\n\n  /** The organization name (O) */\n  organizationName: string;\n  /** The organizational unit name (OU) */\n  organizationalUnitName: string;\n}\n"]}