UNPKG

aws-delivlib

Version:

A fabulous library for defining continuous pipelines for building, testing and releasing code libraries.

174 lines • 23.4 kB
"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"]}