aws-delivlib
Version:
A fabulous library for defining continuous pipelines for building, testing and releasing code libraries.
174 lines • 23.4 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.OpenPGPKeyPair = exports.OpenPGPKeyPairRemovalPolicy = 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");
/**
* The type of the {@link OpenPGPKeyPairProps.removalPolicy} property.
*/
var OpenPGPKeyPairRemovalPolicy;
(function (OpenPGPKeyPairRemovalPolicy) {
/**
* Keep the secret when this resource is deleted from the stack.
* This is the default setting.
*/
OpenPGPKeyPairRemovalPolicy[OpenPGPKeyPairRemovalPolicy["RETAIN"] = 0] = "RETAIN";
/**
* Remove the secret when this resource is deleted from the stack,
* but leave a grace period of a few days that allows you to cancel the deletion from the AWS Console.
*/
OpenPGPKeyPairRemovalPolicy[OpenPGPKeyPairRemovalPolicy["DESTROY_SAFELY"] = 1] = "DESTROY_SAFELY";
/**
* Remove the secret when this resource is deleted from the stack immediately.
* Note that if you don't have a backup of this key somewhere,
* this means it will be gone forever!
*/
OpenPGPKeyPairRemovalPolicy[OpenPGPKeyPairRemovalPolicy["DESTROY_IMMEDIATELY"] = 2] = "DESTROY_IMMEDIATELY";
})(OpenPGPKeyPairRemovalPolicy = exports.OpenPGPKeyPairRemovalPolicy || (exports.OpenPGPKeyPairRemovalPolicy = {}));
/**
* A PGP key that is stored in Secrets Manager.
* The SecretsManager secret is by default retained when the resource is deleted,
* you can change that with the `removalPolicy` property.
*
* The string in secrets manager will be a JSON struct of
*
* { "PrivateKey": "... ASCII repr of key...", "Passphrase": "passphrase of the key" }
*/
class OpenPGPKeyPair extends constructs_1.Construct {
constructor(parent, name, props) {
super(parent, name);
const codeLocation = path.resolve(__dirname, 'custom-resource-handlers');
const fn = new aws_cdk_lib_1.aws_lambda.SingletonFunction(this, 'Lambda', {
// change the uuid to force deleting existing function, and create new one, as Package type change is not allowed
uuid: '2422BDC2-DBB0-47C1-B701-5599E0849C54',
description: 'Generates an OpenPGP Key and stores the private key in Secrets Manager and the public key in an SSM Parameter',
code: new aws_cdk_lib_1.aws_lambda.AssetImageCode(codeLocation, {
file: 'Dockerfile',
platform: aws_ecr_assets_1.Platform.LINUX_AMD64,
buildArgs: {
FUN_SRC_DIR: 'pgp-secret',
},
invalidation: {
buildArgs: true,
},
}),
handler: aws_cdk_lib_1.aws_lambda.Handler.FROM_IMAGE,
timeout: aws_cdk_lib_1.Duration.seconds(300),
runtime: aws_cdk_lib_1.aws_lambda.Runtime.FROM_IMAGE,
});
fn.addToRolePolicy(new aws_cdk_lib_1.aws_iam.PolicyStatement({
actions: [
'secretsmanager:CreateSecret',
'secretsmanager:GetSecretValue',
'secretsmanager:UpdateSecret',
'secretsmanager:DeleteSecret',
],
resources: [aws_cdk_lib_1.Stack.of(this).formatArn({
service: 'secretsmanager',
resource: 'secret',
arnFormat: aws_cdk_lib_1.ArnFormat.COLON_RESOURCE_NAME,
resourceName: `${props.secretName}-??????`,
})],
}));
// To allow easy migration from verison that handled the SSM parameter in the custom resource
fn.addToRolePolicy(new aws_cdk_lib_1.aws_iam.PolicyStatement({
actions: ['ssm:DeleteParameter'],
resources: ['*'],
}));
if (props.encryptionKey) {
props.encryptionKey.addToResourcePolicy(new aws_cdk_lib_1.aws_iam.PolicyStatement({
actions: ['kms:Decrypt', 'kms:GenerateDataKey'],
resources: ['*'],
principals: [fn.role.grantPrincipal],
conditions: {
StringEquals: {
'kms:ViaService': `secretsmanager.${aws_cdk_lib_1.Stack.of(this).region}.amazonaws.com`,
},
},
}));
}
//change the custom resource id to force recreating new one because the change of the underneath lambda function
const secret = new aws_cdk_lib_1.CustomResource(this, 'ResourceV2', {
serviceToken: fn.functionArn,
pascalCaseProperties: true,
properties: {
resourceVersion: (0, util_1.hashFileOrDirectory)(codeLocation),
identity: props.identity,
email: props.email,
expiry: props.expiry,
keySizeBits: props.keySizeBits,
secretName: props.secretName,
keyArn: props.encryptionKey && props.encryptionKey.keyArn,
version: props.version,
description: props.description,
deleteImmediately: props.removalPolicy === OpenPGPKeyPairRemovalPolicy.DESTROY_IMMEDIATELY,
},
removalPolicy: openPgpKeyPairRemovalPolicyToCoreRemovalPolicy(props.removalPolicy),
});
secret.node.addDependency(fn);
this.credential = aws_cdk_lib_1.aws_secretsmanager.Secret.fromSecretAttributes(this, 'Credential', {
encryptionKey: props.encryptionKey,
secretCompleteArn: secret.getAtt('SecretArn').toString(),
});
this.principal = new aws_cdk_lib_1.aws_ssm.StringParameter(this, 'Principal', {
description: `The public part of the OpenPGP key in ${this.credential.secretArn}`,
parameterName: props.pubKeyParameterName,
stringValue: secret.getAtt('PublicKey').toString(),
});
}
grantRead(grantee) {
// Secret grant, identity-based only
grantee.addToPrincipalPolicy(new aws_cdk_lib_1.aws_iam.PolicyStatement({
resources: [this.credential.secretArn],
actions: ['secretsmanager:ListSecrets', 'secretsmanager:DescribeSecret', 'secretsmanager:GetSecretValue'],
}));
// Key grant
if (this.credential.encryptionKey) {
grantee.addToPrincipalPolicy(new aws_cdk_lib_1.aws_iam.PolicyStatement({
resources: [this.credential.encryptionKey.keyArn],
actions: ['kms:Decrypt'],
}));
this.credential.encryptionKey.addToResourcePolicy(new aws_cdk_lib_1.aws_iam.PolicyStatement({
resources: ['*'],
principals: [grantee.grantPrincipal],
actions: ['kms:Decrypt'],
}));
}
}
}
exports.OpenPGPKeyPair = OpenPGPKeyPair;
function openPgpKeyPairRemovalPolicyToCoreRemovalPolicy(removalPolicy) {
if (removalPolicy === undefined) {
return aws_cdk_lib_1.RemovalPolicy.RETAIN;
}
return removalPolicy === OpenPGPKeyPairRemovalPolicy.RETAIN
? aws_cdk_lib_1.RemovalPolicy.RETAIN
: aws_cdk_lib_1.RemovalPolicy.DESTROY;
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"open-pgp-key-pair.js","sourceRoot":"","sources":["open-pgp-key-pair.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport {\n  Duration, Stack, RemovalPolicy,\n  CustomResource,\n  aws_iam as iam,\n  aws_kms as kms,\n  aws_lambda as lambda,\n  aws_secretsmanager as secretsManager,\n  aws_ssm as ssm,\n  ArnFormat,\n} from 'aws-cdk-lib';\nimport { Platform } from 'aws-cdk-lib/aws-ecr-assets';\nimport { Construct } from 'constructs';\nimport { ICredentialPair } from './credential-pair';\nimport { hashFileOrDirectory } from './util';\n\n/**\n * The type of the {@link OpenPGPKeyPairProps.removalPolicy} property.\n */\nexport enum OpenPGPKeyPairRemovalPolicy {\n  /**\n   * Keep the secret when this resource is deleted from the stack.\n   * This is the default setting.\n   */\n  RETAIN,\n\n  /**\n   * Remove the secret when this resource is deleted from the stack,\n   * but leave a grace period of a few days that allows you to cancel the deletion from the AWS Console.\n   */\n  DESTROY_SAFELY,\n\n  /**\n   * Remove the secret when this resource is deleted from the stack immediately.\n   * Note that if you don't have a backup of this key somewhere,\n   * this means it will be gone forever!\n   */\n  DESTROY_IMMEDIATELY,\n}\n\ninterface OpenPGPKeyPairProps {\n  /**\n   * Identity to put into the key\n   */\n  identity: string;\n\n  /**\n   * Email address to attach to the key\n   */\n  email: string;\n\n  /**\n   * Key size in bits (1024, 2048, 4096)\n   */\n  keySizeBits: number;\n\n  /**\n   * GPG expiry specifier\n   *\n   * Example: '1y'\n   */\n  expiry: string;\n\n  /**\n   * Name of secret to create in AWS Secrets Manager\n   */\n  secretName: string;\n\n  /**\n   * Name of SSM parameter to store public key\n   */\n  pubKeyParameterName: string;\n\n  /**\n   * KMS Key ARN to use to encrypt Secrets Manager Secret\n   */\n  encryptionKey?: kms.IKey;\n\n  /**\n   * Version of the key\n   *\n   * Bump this number to regenerate the key\n   */\n  version: number;\n\n  /**\n   * A description to attach to the AWS SecretsManager secret.\n   */\n  description?: string;\n\n  /**\n   * What happens to the SecretsManager secret when this resource is removed from the stack.\n   * The default is to keep the secret.\n   *\n   * @default OpenPGPKeyPairRemovalPolicy.RETAIN\n   */\n  removalPolicy?: OpenPGPKeyPairRemovalPolicy;\n}\n\n/**\n * A PGP key that is stored in Secrets Manager.\n * The SecretsManager secret is by default retained when the resource is deleted,\n * you can change that with the `removalPolicy` property.\n *\n * The string in secrets manager will be a JSON struct of\n *\n * { \"PrivateKey\": \"... ASCII repr of key...\", \"Passphrase\": \"passphrase of the key\" }\n */\nexport class OpenPGPKeyPair extends Construct implements ICredentialPair {\n  public readonly principal: ssm.IStringParameter;\n  public readonly credential: secretsManager.ISecret;\n\n  constructor(parent: Construct, name: string, props: OpenPGPKeyPairProps) {\n    super(parent, name);\n\n    const codeLocation = path.resolve(__dirname, 'custom-resource-handlers');\n\n    const fn = new lambda.SingletonFunction(this, 'Lambda', {\n      // change the uuid to force deleting existing function, and create new one, as Package type change is not allowed\n      uuid: '2422BDC2-DBB0-47C1-B701-5599E0849C54',\n      description: 'Generates an OpenPGP Key and stores the private key in Secrets Manager and the public key in an SSM Parameter',\n      code: new lambda.AssetImageCode(codeLocation, {\n        file: 'Dockerfile',\n        platform: Platform.LINUX_AMD64,\n        buildArgs: {\n          FUN_SRC_DIR: 'pgp-secret',\n        },\n        invalidation: {\n          buildArgs: true,\n        },\n      }),\n      handler: lambda.Handler.FROM_IMAGE,\n      timeout: Duration.seconds(300),\n      runtime: lambda.Runtime.FROM_IMAGE,\n    });\n\n    fn.addToRolePolicy(new iam.PolicyStatement({\n      actions: [\n        'secretsmanager:CreateSecret',\n        'secretsmanager:GetSecretValue',\n        'secretsmanager:UpdateSecret',\n        'secretsmanager:DeleteSecret',\n      ],\n      resources: [Stack.of(this).formatArn({\n        service: 'secretsmanager',\n        resource: 'secret',\n        arnFormat: ArnFormat.COLON_RESOURCE_NAME,\n        resourceName: `${props.secretName}-??????`,\n      })],\n    }));\n\n    // To allow easy migration from verison that handled the SSM parameter in the custom resource\n    fn.addToRolePolicy(new iam.PolicyStatement({\n      actions: ['ssm:DeleteParameter'],\n      resources: ['*'],\n    }));\n\n    if (props.encryptionKey) {\n      props.encryptionKey.addToResourcePolicy(new iam.PolicyStatement({\n        actions: ['kms:Decrypt', 'kms:GenerateDataKey'],\n        resources: ['*'],\n        principals: [fn.role!.grantPrincipal],\n        conditions: {\n          StringEquals: {\n            'kms:ViaService': `secretsmanager.${Stack.of(this).region}.amazonaws.com`,\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 secret = new CustomResource(this, 'ResourceV2', {\n      serviceToken: fn.functionArn,\n      pascalCaseProperties: true,\n      properties: {\n        resourceVersion: hashFileOrDirectory(codeLocation),\n        identity: props.identity,\n        email: props.email,\n        expiry: props.expiry,\n        keySizeBits: props.keySizeBits,\n        secretName: props.secretName,\n        keyArn: props.encryptionKey && props.encryptionKey.keyArn,\n        version: props.version,\n        description: props.description,\n        deleteImmediately: props.removalPolicy === OpenPGPKeyPairRemovalPolicy.DESTROY_IMMEDIATELY,\n      },\n      removalPolicy: openPgpKeyPairRemovalPolicyToCoreRemovalPolicy(props.removalPolicy),\n    });\n    secret.node.addDependency(fn);\n\n    this.credential = secretsManager.Secret.fromSecretAttributes(this, 'Credential', {\n      encryptionKey: props.encryptionKey,\n      secretCompleteArn: secret.getAtt('SecretArn').toString(),\n    });\n    this.principal = new ssm.StringParameter(this, 'Principal', {\n      description: `The public part of the OpenPGP key in ${this.credential.secretArn}`,\n      parameterName: props.pubKeyParameterName,\n      stringValue: secret.getAtt('PublicKey').toString(),\n    });\n  }\n\n  public grantRead(grantee: iam.IPrincipal): void {\n    // Secret grant, identity-based only\n    grantee.addToPrincipalPolicy(new iam.PolicyStatement({\n      resources: [this.credential.secretArn],\n      actions: ['secretsmanager:ListSecrets', 'secretsmanager:DescribeSecret', 'secretsmanager:GetSecretValue'],\n    }));\n\n    // Key grant\n    if (this.credential.encryptionKey) {\n      grantee.addToPrincipalPolicy(new iam.PolicyStatement({\n        resources: [this.credential.encryptionKey.keyArn],\n        actions: ['kms:Decrypt'],\n      }));\n\n      this.credential.encryptionKey.addToResourcePolicy(new iam.PolicyStatement({\n        resources: ['*'],\n        principals: [grantee.grantPrincipal],\n        actions: ['kms:Decrypt'],\n      }));\n    }\n  }\n}\n\nfunction openPgpKeyPairRemovalPolicyToCoreRemovalPolicy(removalPolicy?: OpenPGPKeyPairRemovalPolicy): RemovalPolicy {\n  if (removalPolicy === undefined) {\n    return RemovalPolicy.RETAIN;\n  }\n  return removalPolicy === OpenPGPKeyPairRemovalPolicy.RETAIN\n    ? RemovalPolicy.RETAIN\n    : RemovalPolicy.DESTROY;\n}\n"]}