UNPKG

cdk-rds-sql

Version:

A CDK construct that allows creating roles or users and databases an on Aurora Serverless Postgresql or Mysql/MariaDB cluster.

139 lines 21 kB
"use strict"; var _a; Object.defineProperty(exports, "__esModule", { value: true }); exports.Provider = void 0; const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); const fs_1 = require("fs"); const path = require("path"); const aws_cdk_lib_1 = require("aws-cdk-lib"); const aws_ec2_1 = require("aws-cdk-lib/aws-ec2"); const iam = require("aws-cdk-lib/aws-iam"); const aws_lambda_1 = require("aws-cdk-lib/aws-lambda"); const lambda = require("aws-cdk-lib/aws-lambda-nodejs"); const customResources = require("aws-cdk-lib/custom-resources"); const constructs_1 = require("constructs"); class Provider extends constructs_1.Construct { constructor(scope, id, props) { super(scope, id); this.secret = props.secret; this.cluster = props.cluster; // Determine engine from cluster/instance instead of hardcoding if ("clusterIdentifier" in props.cluster) { // It's a DatabaseCluster const clusterEngine = props.cluster.engine; this.engine = clusterEngine && clusterEngine.engineFamily === "MYSQL" ? "mysql" : "postgres"; } else if ("instanceIdentifier" in props.cluster) { // It's a DatabaseInstance const instanceEngine = props.cluster.engine; this.engine = instanceEngine && instanceEngine.engineFamily === "MYSQL" ? "mysql" : "postgres"; } else { // Fallback to postgres if engine hasn't been provided this.engine = "postgres"; } const functionName = "RdsSql" + slugify("28b9e791-af60-4a33-bca8-ffb6f30ef8c5"); this.handler = aws_cdk_lib_1.Stack.of(this).node.tryFindChild(functionName) ?? this.newCustomResourceHandler(scope, functionName, props); const provider = new customResources.Provider(this, "RdsSql", { onEventHandler: this.handler, }); this.serviceToken = provider.serviceToken; this.secret.grantRead(this.handler); if (this.secret.encryptionKey) { // It seems we need to grant explicit permission this.secret.encryptionKey.grantDecrypt(this.handler); } if (props.cluster.connections.securityGroups.length === 0) { throw new Error("Cluster does not have a security group."); } else { const securityGroup = props.cluster.connections.securityGroups[0]; this.handler.node.defaultChild?.node.addDependency(securityGroup); } this.node.addDependency(props.cluster); } newCustomResourceHandler(scope, id, props) { const handlerDir = path.join(__dirname, "handler"); const index_ts = path.join(handlerDir, "index.ts"); const index_js = path.join(handlerDir, "index.js"); let entry; if ((0, fs_1.existsSync)(index_ts)) { entry = index_ts; } else if ((0, fs_1.existsSync)(index_js)) { entry = index_js; } else { // Ugly hack to support SST (possibly caused by my hack to make SST work with CommonJS libraries) entry = path.join(path.dirname(process.env.npm_package_json || process.cwd()), "node_modules/cdk-rds-sql/lib/handler/index.js"); } let ssl_options; if (props.ssl !== undefined && !props.ssl) { ssl_options = { SSL: JSON.stringify(props.ssl), }; } const logger = props.logger ?? false; const deleteParameterPolicy = new iam.PolicyStatement({ actions: ["ssm:DeleteParameter"], resources: [ `arn:aws:ssm:${aws_cdk_lib_1.Stack.of(scope).region}:${aws_cdk_lib_1.Stack.of(scope).account}:parameter/*`, ], conditions: { StringEquals: { "ssm:ResourceTag/created-by": "cdk-rds-sql", }, }, }); const fn = new lambda.NodejsFunction(scope, id, { ...props.functionProps, vpc: props.vpc, vpcSubnets: props.vpcSubnets ?? { subnetType: aws_ec2_1.SubnetType.PRIVATE_ISOLATED, }, entry: entry, runtime: aws_lambda_1.Runtime.NODEJS_22_X, timeout: props.timeout ?? props.functionProps?.timeout ?? aws_cdk_lib_1.Duration.seconds(300), bundling: { // Include the migrations directory in the bundle commandHooks: { beforeBundling(_, outputDir) { return [ `curl --silent -fL https://truststore.pki.rds.amazonaws.com/global/global-bundle.pem -o ${path.join(outputDir, "global-bundle.pem")}`, ]; }, afterBundling() { return []; }, beforeInstall() { return []; }, }, }, environment: { LOGGER: logger.toString(), ...ssl_options, }, initialPolicy: [ deleteParameterPolicy, ...(props.functionProps?.initialPolicy ?? []), ], }); if (!props.functionProps?.securityGroups || props.functionProps?.securityGroups.length === 0) { props.cluster.connections.allowDefaultPortFrom(fn, "Allow the rds sql handler to connect to db"); } return fn; } } exports.Provider = Provider; _a = JSII_RTTI_SYMBOL_1; Provider[_a] = { fqn: "cdk-rds-sql.Provider", version: "6.1.4" }; function slugify(x) { return x.replace(/[^a-zA-Z0-9]/g, ""); } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"provider.js","sourceRoot":"","sources":["../src/provider.ts"],"names":[],"mappings":"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","sourcesContent":["import { existsSync } from \"fs\"\nimport * as path from \"path\"\nimport { Duration, Stack } from \"aws-cdk-lib\"\nimport { IVpc, SubnetType, SubnetSelection } from \"aws-cdk-lib/aws-ec2\"\nimport * as iam from \"aws-cdk-lib/aws-iam\"\nimport { IFunction, Runtime } from \"aws-cdk-lib/aws-lambda\"\nimport * as lambda from \"aws-cdk-lib/aws-lambda-nodejs\"\nimport { NodejsFunctionProps } from \"aws-cdk-lib/aws-lambda-nodejs\"\nimport { IDatabaseCluster, IDatabaseInstance } from \"aws-cdk-lib/aws-rds\"\nimport { ISecret } from \"aws-cdk-lib/aws-secretsmanager\"\nimport * as customResources from \"aws-cdk-lib/custom-resources\"\nimport { Construct } from \"constructs\"\n\nexport interface RdsSqlProps {\n  /**\n   * VPC network to place the provider lambda.\n   *\n   * Normally this is the VPC of your database.\n   *\n   * @default - Function is not placed within a VPC.\n   */\n  readonly vpc: IVpc\n\n  /**\n   * Where to place the network provider lambda within the VPC.\n   *\n   * @default - the isolated subnet if not specified\n   */\n  readonly vpcSubnets?: SubnetSelection\n\n  /**\n   * Your database.\n   */\n  readonly cluster: IDatabaseCluster | IDatabaseInstance\n\n  /**\n   * Secret that grants access to your database.\n   *\n   * Usually this is your cluster's master secret.\n   */\n  readonly secret: ISecret\n\n  /**\n   * Timeout for lambda to do its work.\n   *\n   * @default - 5 minutes\n   */\n  readonly timeout?: Duration\n\n  /**\n   * Log SQL statements. This includes passwords. Use only for debugging.\n   *\n   * @default - false\n   */\n  readonly logger?: boolean\n\n  /**\n   * Additional function customization.\n   *\n   * This enables additional function customization such as the log group. However,\n   * lambda function properties controlled by other {RdsSqlProps} parameters will trump\n   * opions set via this parameter.\n   *\n   * @default - empty\n   */\n  readonly functionProps?: NodejsFunctionProps\n\n  /**\n   * Use SSL?\n   *\n   * @default - false\n   */\n  readonly ssl?: boolean\n}\n\nexport class Provider extends Construct {\n  public readonly serviceToken: string\n  public readonly secret: ISecret\n  public readonly handler: IFunction\n  public readonly cluster: IDatabaseCluster | IDatabaseInstance\n\n  /**\n   * The engine like \"postgres\" or \"mysql\"\n   *\n   * @default - if we cannot determine this \"postgres\"\n   */\n  public readonly engine: string\n\n  constructor(scope: Construct, id: string, props: RdsSqlProps) {\n    super(scope, id)\n    this.secret = props.secret\n    this.cluster = props.cluster\n\n    // Determine engine from cluster/instance instead of hardcoding\n    if (\"clusterIdentifier\" in props.cluster) {\n      // It's a DatabaseCluster\n      const clusterEngine = (props.cluster as IDatabaseCluster).engine\n      this.engine =\n        clusterEngine && clusterEngine.engineFamily === \"MYSQL\" ? \"mysql\" : \"postgres\"\n    } else if (\"instanceIdentifier\" in props.cluster) {\n      // It's a DatabaseInstance\n      const instanceEngine = (props.cluster as IDatabaseInstance).engine\n      this.engine =\n        instanceEngine && instanceEngine.engineFamily === \"MYSQL\" ? \"mysql\" : \"postgres\"\n    } else {\n      // Fallback to postgres if engine hasn't been provided\n      this.engine = \"postgres\"\n    }\n\n    const functionName = \"RdsSql\" + slugify(\"28b9e791-af60-4a33-bca8-ffb6f30ef8c5\")\n    this.handler =\n      (Stack.of(this).node.tryFindChild(functionName) as IFunction) ??\n      this.newCustomResourceHandler(scope, functionName, props)\n\n    const provider = new customResources.Provider(this, \"RdsSql\", {\n      onEventHandler: this.handler,\n    })\n    this.serviceToken = provider.serviceToken\n    this.secret.grantRead(this.handler)\n    if (this.secret.encryptionKey) {\n      // It seems we need to grant explicit permission\n      this.secret.encryptionKey.grantDecrypt(this.handler)\n    }\n    if (props.cluster.connections.securityGroups.length === 0) {\n      throw new Error(\"Cluster does not have a security group.\")\n    } else {\n      const securityGroup = props.cluster.connections.securityGroups[0]!\n      this.handler.node.defaultChild?.node.addDependency(securityGroup)\n    }\n    this.node.addDependency(props.cluster)\n  }\n\n  protected newCustomResourceHandler(\n    scope: Construct,\n    id: string,\n    props: RdsSqlProps\n  ): lambda.NodejsFunction {\n    const handlerDir = path.join(__dirname, \"handler\")\n    const index_ts = path.join(handlerDir, \"index.ts\")\n    const index_js = path.join(handlerDir, \"index.js\")\n    let entry: string\n\n    if (existsSync(index_ts)) {\n      entry = index_ts\n    } else if (existsSync(index_js)) {\n      entry = index_js\n    } else {\n      // Ugly hack to support SST (possibly caused by my hack to make SST work with CommonJS libraries)\n      entry = path.join(\n        path.dirname(process.env.npm_package_json || process.cwd()),\n        \"node_modules/cdk-rds-sql/lib/handler/index.js\"\n      )\n    }\n    let ssl_options: Record<string, string> | undefined\n    if (props.ssl !== undefined && !props.ssl) {\n      ssl_options = {\n        SSL: JSON.stringify(props.ssl),\n      }\n    }\n    const logger = props.logger ?? false\n\n    const deleteParameterPolicy = new iam.PolicyStatement({\n      actions: [\"ssm:DeleteParameter\"],\n      resources: [\n        `arn:aws:ssm:${Stack.of(scope).region}:${Stack.of(scope).account}:parameter/*`,\n      ],\n      conditions: {\n        StringEquals: {\n          \"ssm:ResourceTag/created-by\": \"cdk-rds-sql\",\n        },\n      },\n    })\n\n    const fn = new lambda.NodejsFunction(scope, id, {\n      ...props.functionProps,\n      vpc: props.vpc,\n      vpcSubnets: props.vpcSubnets ?? {\n        subnetType: SubnetType.PRIVATE_ISOLATED,\n      },\n      entry: entry,\n      runtime: Runtime.NODEJS_22_X,\n      timeout: props.timeout ?? props.functionProps?.timeout ?? Duration.seconds(300),\n      bundling: {\n        // Include the migrations directory in the bundle\n        commandHooks: {\n          beforeBundling(_: string, outputDir: string): string[] {\n            return [\n              `curl --silent -fL https://truststore.pki.rds.amazonaws.com/global/global-bundle.pem -o ${path.join(\n                outputDir,\n                \"global-bundle.pem\"\n              )}`,\n            ]\n          },\n          afterBundling(): string[] {\n            return []\n          },\n          beforeInstall(): string[] {\n            return []\n          },\n        },\n      },\n      environment: {\n        LOGGER: logger.toString(),\n        ...ssl_options,\n      },\n      initialPolicy: [\n        deleteParameterPolicy,\n        ...(props.functionProps?.initialPolicy ?? []),\n      ],\n    })\n\n    if (\n      !props.functionProps?.securityGroups ||\n      props.functionProps?.securityGroups.length === 0\n    ) {\n      props.cluster.connections.allowDefaultPortFrom(\n        fn,\n        \"Allow the rds sql handler to connect to db\"\n      )\n    }\n\n    return fn\n  }\n}\n\nfunction slugify(x: string): string {\n  return x.replace(/[^a-zA-Z0-9]/g, \"\")\n}\n"]}