aws-delivlib
Version:
A fabulous library for defining continuous pipelines for building, testing and releasing code libraries.
142 lines • 23.4 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.CodeSigningCertificate = void 0;
const aws_cdk_lib_1 = require("aws-cdk-lib");
const constructs_1 = require("constructs");
const private_key_1 = require("./private-key");
const permissions = __importStar(require("../permissions"));
/**
* A Code-Signing certificate, that will use a private key that is generated by a Lambda function. The Certificate will
* not be usable until the ``pemCertificate`` value has been provided. A typical workflow to use this Construct would be:
*
* 1. Add an instance of the construct to your app, without providing the ``pemCertificate`` property
* 2. Deploy the stack to provision a Private Key and obtain the CSR (you can surface it using a Output, for example)
* 3. Submit the CSR to your Certificate Authority of choice.
* 4. Populate the ``pemCertificate`` property with the PEM-encoded certificate provided by your CA of coice.
* 5. Re-deploy the stack so make the certificate usable
*
* In order to renew the certificate, if you do not wish to retain the same private key (your clients do not rely on
* public key pinning), simply add a new instance of the construct to your app and follow the process listed above. If
* you wish to retain the private key, you can set ``forceCertificateSigningRequest`` to ``true`` in order to obtain a
* new CSR document.
*/
class CodeSigningCertificate extends constructs_1.Construct {
constructor(parent, id, props) {
super(parent, id);
// The construct path of this construct with respect to the containing stack, without any leading /
const stack = aws_cdk_lib_1.Stack.of(this);
const baseName = props.baseName ?? `${stack.stackName}${this.node.path.substr(stack.node.path.length)}`;
const privateKey = new private_key_1.RsaPrivateKeySecret(this, 'RSAPrivateKey', {
removalPolicy: props.retainPrivateKey === false ? aws_cdk_lib_1.RemovalPolicy.DESTROY : aws_cdk_lib_1.RemovalPolicy.RETAIN,
description: 'The PEM-encoded private key of the x509 Code-Signing Certificate',
keySize: props.rsaKeySize || 2048,
secretEncryptionKey: props.secretEncryptionKey,
// rename the secret name, as since this resource will be deleted and create a new resource,
// so the new resource will be created before the old one got deleted, and so we will not be able
// to create a new secrete with the same name, and even we could not reuse it, as it will be deleted once
// the old resource got deleted.
secretName: `${baseName}/RSAPrivateKeyV2`,
});
// this change to keep the permissions to access the old secret for the custom resource Lambda function role, so it can
// delete the old secret.
const oldSecretArnLike = 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: `${baseName}/RSAPrivateKey-??????`,
});
privateKey.customResource.addToRolePolicy(new aws_cdk_lib_1.aws_iam.PolicyStatement({
actions: [
'secretsmanager:CreateSecret',
'secretsmanager:DeleteSecret',
'secretsmanager:UpdateSecret',
],
resources: [oldSecretArnLike],
}));
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(privateKey.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': oldSecretArnLike,
},
},
}));
}
this.credential = aws_cdk_lib_1.aws_secretsmanager.Secret.fromSecretAttributes(this, 'Credential', {
encryptionKey: props.secretEncryptionKey,
secretCompleteArn: privateKey.secretArn,
});
let certificate = props.pemCertificate;
if (!certificate || props.forceCertificateSigningRequest) {
const csr = privateKey.newCertificateSigningRequest('CertificateSigningRequest', props.distinguishedName, 'critical,digitalSignature', 'critical,codeSigning');
this.certificateBucket = csr.outputBucket;
new aws_cdk_lib_1.CfnOutput(this, 'CSR', {
description: 'A PEM-encoded Certificate Signing Request for a Code-Signing Certificate',
value: csr.pemRequest,
});
if (!certificate) {
certificate = csr.selfSignedPemCertificate;
}
}
this.principal = new aws_cdk_lib_1.aws_ssm.StringParameter(this, 'Resource', {
description: `A PEM-encoded Code-Signing Certificate (private key in ${privateKey.secretArn})`,
parameterName: `/${baseName}/Certificate`,
stringValue: certificate,
});
}
/**
* Grant the IAM principal permissions to read the private key and
* certificate.
*/
grantDecrypt(principal) {
if (!principal) {
return;
}
permissions.grantSecretRead({
keyArn: this.credential.encryptionKey && this.credential.encryptionKey.keyArn,
secretArn: this.credential.secretArn,
}, principal);
principal.addToPrincipalPolicy(new aws_cdk_lib_1.aws_iam.PolicyStatement({
actions: ['ssm:GetParameter'],
resources: [aws_cdk_lib_1.Stack.of(this).formatArn({
// TODO: This is a workaround until https://github.com/awslabs/aws-cdk/pull/1726 is released
service: 'ssm',
resource: `parameter${this.principal.parameterName}`,
})],
}));
this.certificateBucket?.grantRead(principal);
}
}
exports.CodeSigningCertificate = CodeSigningCertificate;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"code-signing-certificate.js","sourceRoot":"","sources":["code-signing-certificate.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  CfnOutput, RemovalPolicy, Stack,\n  aws_iam as iam,\n  aws_kms as kms,\n  aws_s3 as s3,\n  aws_secretsmanager as secretsManager,\n  aws_ssm as ssm,\n  ArnFormat,\n} from 'aws-cdk-lib';\nimport { Construct, IConstruct } from 'constructs';\nimport { CertificateSigningRequest, DistinguishedName } from './certificate-signing-request';\nimport { RsaPrivateKeySecret } from './private-key';\nimport { ICredentialPair } from '../credential-pair';\nimport * as permissions from '../permissions';\n\n\nexport { DistinguishedName } from './certificate-signing-request';\n\ninterface CodeSigningCertificateProps {\n  /**\n   * The number of bits to compose the modulus of the generated private key for this certificate.\n   *\n   * @default 2048\n   */\n  rsaKeySize?: number;\n\n  /**\n   * The KMS CMK to use for encrypting the Private Key secret.\n   * @default A new KMS key will be allocated for you\n   */\n  secretEncryptionKey?: kms.IKey;\n\n  /**\n   * The PEM-encoded certificate that was signed by the relevant authority.\n   *\n   * @default If a certificate is not provided, a self-signed certificate will\n   * be generated and a CSR (certificate signing request) will by available in\n   * the stack output.\n   */\n  pemCertificate?: string;\n\n  /**\n   * Whether a CSR should be generated, even if the certificate is provided.\n   * This can be useful if one wants to renew a certificate that is close to\n   * expiry without generating a new private key (for example, to avoid breaking\n   * clients that make use of certificate pinning).\n   *\n   * @default false\n   */\n  forceCertificateSigningRequest?: boolean;\n\n  /**\n   * When enabled, the Private Key secret will have a DeletionPolicy of\n   * \"RETAIN\", making sure the Private Key is not inadvertently destroyed.\n   *\n   * @default true\n   */\n  retainPrivateKey?: boolean;\n\n  /**\n   * The Distinguished Name for this CSR.\n   */\n  distinguishedName: DistinguishedName;\n\n  /**\n   * Base names for the private key and output SSM parameter\n   *\n   * @default - Automatically generated\n   */\n  readonly baseName?: string;\n}\n\nexport interface ICodeSigningCertificate extends IConstruct, ICredentialPair {\n  /**\n   * The S3 bucket where the self-signed certificate is stored.\n   */\n  readonly certificateBucket?: s3.IBucket;\n\n  /**\n   * Grant the IAM principal permissions to read the private key and\n   * certificate.\n   */\n  grantDecrypt(principal?: iam.IPrincipal): void;\n}\n\n/**\n * A Code-Signing certificate, that will use a private key that is generated by a Lambda function. The Certificate will\n * not be usable until the ``pemCertificate`` value has been provided. A typical workflow to use this Construct would be:\n *\n * 1. Add an instance of the construct to your app, without providing the ``pemCertificate`` property\n * 2. Deploy the stack to provision a Private Key and obtain the CSR (you can surface it using a Output, for example)\n * 3. Submit the CSR to your Certificate Authority of choice.\n * 4. Populate the ``pemCertificate`` property with the PEM-encoded certificate provided by your CA of coice.\n * 5. Re-deploy the stack so make the certificate usable\n *\n * In order to renew the certificate, if you do not wish to retain the same private key (your clients do not rely on\n * public key pinning), simply add a new instance of the construct to your app and follow the process listed above. If\n * you wish to retain the private key, you can set ``forceCertificateSigningRequest`` to ``true`` in order to obtain a\n * new CSR document.\n */\nexport class CodeSigningCertificate extends Construct implements ICodeSigningCertificate {\n  /**\n   * The AWS Secrets Manager secret that holds the private key for this CSC\n   */\n  public readonly credential: secretsManager.ISecret;\n\n  /**\n   * The AWS SSM Parameter that holds the certificate for this CSC.\n   */\n  public readonly principal: ssm.IStringParameter;\n\n  /**\n   * The S3 bucket where the self-signed certificate is stored.\n   */\n  public readonly certificateBucket?: s3.IBucket;\n\n  constructor(parent: Construct, id: string, props: CodeSigningCertificateProps) {\n    super(parent, id);\n\n    // The construct path of this construct with respect to the containing stack, without any leading /\n    const stack = Stack.of(this);\n    const baseName = props.baseName ?? `${stack.stackName}${this.node.path.substr(stack.node.path.length)}`;\n\n    const privateKey = new RsaPrivateKeySecret(this, 'RSAPrivateKey', {\n      removalPolicy: props.retainPrivateKey === false ? RemovalPolicy.DESTROY : RemovalPolicy.RETAIN,\n      description: 'The PEM-encoded private key of the x509 Code-Signing Certificate',\n      keySize: props.rsaKeySize || 2048,\n      secretEncryptionKey: props.secretEncryptionKey,\n      // rename the secret name, as since this resource will be deleted and create a new resource,\n      // so the new resource will be created before the old one got deleted, and so we will not be able\n      // to create a new secrete with the same name, and even we could not reuse it, as it will be deleted once\n      // the old resource got deleted.\n      secretName: `${baseName}/RSAPrivateKeyV2`,\n    });\n\n    // this change to keep the permissions to access the old secret for the custom resource Lambda function role, so it can\n    // delete the old secret.\n    const oldSecretArnLike = 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: `${baseName}/RSAPrivateKey-??????`,\n    });\n    privateKey.customResource.addToRolePolicy(new iam.PolicyStatement({\n      actions: [\n        'secretsmanager:CreateSecret',\n        'secretsmanager:DeleteSecret',\n        'secretsmanager:UpdateSecret',\n      ],\n      resources: [oldSecretArnLike],\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(privateKey.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': oldSecretArnLike,\n          },\n        },\n      }));\n    }\n\n    this.credential = secretsManager.Secret.fromSecretAttributes(this, 'Credential', {\n      encryptionKey: props.secretEncryptionKey,\n      secretCompleteArn: privateKey.secretArn,\n    });\n\n    let certificate = props.pemCertificate;\n\n    if (!certificate || props.forceCertificateSigningRequest) {\n      const csr: CertificateSigningRequest = privateKey.newCertificateSigningRequest('CertificateSigningRequest',\n        props.distinguishedName,\n        'critical,digitalSignature',\n        'critical,codeSigning');\n\n      this.certificateBucket = csr.outputBucket;\n\n      new CfnOutput(this, 'CSR', {\n        description: 'A PEM-encoded Certificate Signing Request for a Code-Signing Certificate',\n        value: csr.pemRequest,\n      });\n\n      if (!certificate) {\n        certificate = csr.selfSignedPemCertificate;\n      }\n    }\n\n    this.principal = new ssm.StringParameter(this, 'Resource', {\n      description: `A PEM-encoded Code-Signing Certificate (private key in ${privateKey.secretArn})`,\n      parameterName: `/${baseName}/Certificate`,\n      stringValue: certificate!,\n    });\n  }\n\n  /**\n   * Grant the IAM principal permissions to read the private key and\n   * certificate.\n   */\n  public grantDecrypt(principal?: iam.IPrincipal) {\n    if (!principal) { return; }\n\n    permissions.grantSecretRead({\n      keyArn: this.credential.encryptionKey && this.credential.encryptionKey.keyArn,\n      secretArn: this.credential.secretArn,\n    }, principal);\n\n    principal.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['ssm:GetParameter'],\n      resources: [Stack.of(this).formatArn({\n        // TODO: This is a workaround until https://github.com/awslabs/aws-cdk/pull/1726 is released\n        service: 'ssm',\n        resource: `parameter${this.principal.parameterName}`,\n      })],\n    }));\n\n    this.certificateBucket?.grantRead(principal);\n  }\n}\n"]}