cdk-rds-sql
Version:
A CDK construct that allows creating roles or users and databases on Aurora Serverless PostgreSQL or MySQL/MariaDB clusters, as well as AWS DSQL clusters.
240 lines • 37 kB
JavaScript
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.Provider = exports.DatabaseEngine = 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 dsql = require("aws-cdk-lib/aws-dsql");
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");
/**
* Helper function to determine if a cluster is a DSQL cluster
*/
function isDsqlCluster(cluster) {
return cluster instanceof dsql.CfnCluster;
}
/**
* Supported database engines
*/
var DatabaseEngine;
(function (DatabaseEngine) {
DatabaseEngine["POSTGRES"] = "postgres";
DatabaseEngine["MYSQL"] = "mysql";
DatabaseEngine["DSQL"] = "dsql";
})(DatabaseEngine || (exports.DatabaseEngine = DatabaseEngine = {}));
class Provider extends constructs_1.Construct {
/**
* Import an existing provider Lambda function
*/
static fromProviderAttributes(scope, id, attrs) {
return new ImportedProvider(scope, id, attrs);
}
constructor(scope, id, props) {
super(scope, id);
// Validate configuration
const isDsql = isDsqlCluster(props.cluster);
if (!isDsql && !props.secret) {
throw new Error("Either secret (for traditional RDS) or cluster with DSQL must be provided");
}
if (!isDsql && !props.vpc) {
throw new Error("VPC is required for traditional RDS databases");
}
if (isDsql && props.secret) {
throw new Error("secret should not be provided when using DSQL cluster (uses IAM authentication)");
}
this.secret = props.secret;
this.cluster = props.cluster;
// Determine engine from cluster/instance instead of hardcoding
if (isDsql) {
// DSQL is always PostgreSQL-compatible
this.engine = DatabaseEngine.DSQL;
}
else if ("clusterIdentifier" in props.cluster) {
// It's a DatabaseCluster
const clusterEngine = props.cluster.engine;
this.engine =
clusterEngine && clusterEngine.engineFamily === "MYSQL"
? DatabaseEngine.MYSQL
: DatabaseEngine.POSTGRES;
}
else if ("instanceIdentifier" in props.cluster) {
// It's a DatabaseInstance
const instanceEngine = props.cluster.engine;
this.engine =
instanceEngine && instanceEngine.engineFamily === "MYSQL"
? DatabaseEngine.MYSQL
: DatabaseEngine.POSTGRES;
}
else {
// Fallback to postgres if engine hasn't been provided
this.engine = DatabaseEngine.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;
// Handle database connection setup
if (isDsql) {
// For DSQL, grant IAM permissions instead of VPC security groups
const dsqlCluster = props.cluster;
this.handler.addToRolePolicy(new iam.PolicyStatement({
effect: iam.Effect.ALLOW,
actions: ["dsql:DbConnectAdmin"],
resources: [dsqlCluster.attrResourceArn],
}));
}
else {
// Traditional RDS setup with security groups and secrets
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 isDsql = isDsqlCluster(props.cluster);
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;
// Build environment variables
const environment = {
LOGGER: logger.toString(),
...ssl_options,
};
// Add DSQL-specific environment variables
if (isDsql) {
const dsqlCluster = props.cluster;
const clusterId = dsqlCluster.attrIdentifier;
const region = aws_cdk_lib_1.Stack.of(scope).region;
environment.DSQL_ENDPOINT = `${clusterId}.dsql.${region}.on.aws`;
environment.DSQL_PORT = "5432";
}
else if (props.secret) {
// Add secret ARN to environment for traditional RDS
environment.SECRET_ARN = props.secret.secretArn;
}
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,
// Only configure VPC for traditional RDS
...(isDsql
? {}
: {
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,
initialPolicy: [
deleteParameterPolicy,
...(props.functionProps?.initialPolicy ?? []),
],
});
// Only configure security groups for traditional RDS (not DSQL)
if (!isDsql &&
(!props.functionProps?.securityGroups ||
props.functionProps?.securityGroups.length === 0)) {
const rdsCluster = props.cluster;
rdsCluster.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: "7.3.2" };
class ImportedProvider extends constructs_1.Construct {
constructor(scope, id, attrs) {
super(scope, id);
// Validate that either functionArn or functionName is provided
if (!attrs.functionArn && !attrs.functionName) {
throw new Error("Either functionArn or functionName must be provided");
}
if (attrs.functionArn && attrs.functionName) {
throw new Error("Provide either functionArn or functionName, not both");
}
// Import the existing Lambda function
this.handler = attrs.functionArn
? aws_lambda_1.Function.fromFunctionArn(this, "Handler", attrs.functionArn)
: aws_lambda_1.Function.fromFunctionName(this, "Handler", attrs.functionName);
// Derive serviceToken by wrapping in custom resource provider
const provider = new customResources.Provider(this, "Provider", {
onEventHandler: this.handler,
});
this.serviceToken = provider.serviceToken;
this.engine = attrs.engine;
this.secret = undefined; // Imported providers get secret from environment
this.cluster = attrs.cluster; // Optional cluster for role creation
}
}
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 * as dsql from \"aws-cdk-lib/aws-dsql\"\nimport { IVpc, SubnetType, SubnetSelection } from \"aws-cdk-lib/aws-ec2\"\nimport * as iam from \"aws-cdk-lib/aws-iam\"\nimport { Function, 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\n/**\n * Helper function to determine if a cluster is a DSQL cluster\n */\nfunction isDsqlCluster(\n  cluster: IDatabaseCluster | IDatabaseInstance | dsql.CfnCluster\n): cluster is dsql.CfnCluster {\n  return cluster instanceof dsql.CfnCluster\n}\n\nexport interface RdsSqlProps {\n  /**\n   * VPC network to place the provider lambda.\n   *\n   * Normally this is the VPC of your database.\n   * Required when your database is only accessible in a VPC.\n   * Not required for DSQL as it uses public endpoints with IAM authentication.\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 cluster or instance.\n   * Supports both traditional RDS/Aurora clusters and DSQL clusters.\n   * - For RDS/Aurora: security groups will be configured to allow access\n   * - For DSQL: IAM authentication will be used instead of secrets\n   */\n  readonly cluster: IDatabaseCluster | IDatabaseInstance | dsql.CfnCluster\n\n  /**\n   * Secret that grants access to your database.\n   *\n   * Usually this is your cluster's master secret.\n   * Not required when relying on IAM authentication (such as DSQL).\n   *\n   * @default - undefined for DSQL clusters using IAM authentication\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 - true\n   */\n  readonly ssl?: boolean\n}\n\n/**\n * Supported database engines\n */\nexport enum DatabaseEngine {\n  POSTGRES = \"postgres\",\n  MYSQL = \"mysql\",\n  DSQL = \"dsql\",\n}\n\nexport interface IProvider {\n  readonly serviceToken: string\n  readonly handler: IFunction\n  readonly secret?: ISecret\n  readonly engine: string\n  readonly cluster?: IDatabaseCluster | IDatabaseInstance | dsql.CfnCluster\n}\n\nexport interface ProviderAttributes {\n  /**\n   * Either the ARN or name of the Lambda function.\n   * Use functionArn for cross-account or cross-region scenarios.\n   * Use functionName for same-account, same-region scenarios.\n   */\n  readonly functionArn?: string\n  readonly functionName?: string\n  readonly engine: DatabaseEngine\n  /**\n   * Optional cluster information for role creation.\n   *\n   * When importing a provider, cluster details are often not available.\n   * However, some operations like role creation require cluster endpoint\n   * information to build connection secrets.\n   *\n   * If you plan to create roles with the imported provider, you must\n   * provide the cluster reference. If you only plan to use existing\n   * roles, databases, schemas, or SQL operations, this can be omitted.\n   */\n  readonly cluster?: IDatabaseCluster | IDatabaseInstance | dsql.CfnCluster\n}\n\nexport class Provider extends Construct implements IProvider {\n  /**\n   * Import an existing provider Lambda function\n   */\n  static fromProviderAttributes(\n    scope: Construct,\n    id: string,\n    attrs: ProviderAttributes\n  ): IProvider {\n    return new ImportedProvider(scope, id, attrs)\n  }\n\n  public readonly serviceToken: string\n  public readonly secret?: ISecret\n  public readonly handler: IFunction\n  public readonly cluster?: IDatabaseCluster | IDatabaseInstance | dsql.CfnCluster\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\n    // Validate configuration\n    const isDsql = isDsqlCluster(props.cluster)\n    if (!isDsql && !props.secret) {\n      throw new Error(\n        \"Either secret (for traditional RDS) or cluster with DSQL must be provided\"\n      )\n    }\n    if (!isDsql && !props.vpc) {\n      throw new Error(\"VPC is required for traditional RDS databases\")\n    }\n    if (isDsql && props.secret) {\n      throw new Error(\n        \"secret should not be provided when using DSQL cluster (uses IAM authentication)\"\n      )\n    }\n\n    this.secret = props.secret\n    this.cluster = props.cluster\n\n    // Determine engine from cluster/instance instead of hardcoding\n    if (isDsql) {\n      // DSQL is always PostgreSQL-compatible\n      this.engine = DatabaseEngine.DSQL\n    } else 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\"\n          ? DatabaseEngine.MYSQL\n          : DatabaseEngine.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\"\n          ? DatabaseEngine.MYSQL\n          : DatabaseEngine.POSTGRES\n    } else {\n      // Fallback to postgres if engine hasn't been provided\n      this.engine = DatabaseEngine.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\n    // Handle database connection setup\n    if (isDsql) {\n      // For DSQL, grant IAM permissions instead of VPC security groups\n      const dsqlCluster = props.cluster as dsql.CfnCluster\n      this.handler.addToRolePolicy(\n        new iam.PolicyStatement({\n          effect: iam.Effect.ALLOW,\n          actions: [\"dsql:DbConnectAdmin\"],\n          resources: [dsqlCluster.attrResourceArn],\n        })\n      )\n    } else {\n      // Traditional RDS setup with security groups and secrets\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    }\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 isDsql = isDsqlCluster(props.cluster)\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    // Build environment variables\n    const environment: Record<string, string> = {\n      LOGGER: logger.toString(),\n      ...ssl_options,\n    }\n\n    // Add DSQL-specific environment variables\n    if (isDsql) {\n      const dsqlCluster = props.cluster as dsql.CfnCluster\n      const clusterId = dsqlCluster.attrIdentifier\n      const region = Stack.of(scope).region\n      environment.DSQL_ENDPOINT = `${clusterId}.dsql.${region}.on.aws`\n      environment.DSQL_PORT = \"5432\"\n    } else if (props.secret) {\n      // Add secret ARN to environment for traditional RDS\n      environment.SECRET_ARN = props.secret.secretArn\n    }\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      // Only configure VPC for traditional RDS\n      ...(isDsql\n        ? {}\n        : {\n            vpc: props.vpc,\n            vpcSubnets: props.vpcSubnets ?? {\n              subnetType: SubnetType.PRIVATE_ISOLATED,\n            },\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      initialPolicy: [\n        deleteParameterPolicy,\n        ...(props.functionProps?.initialPolicy ?? []),\n      ],\n    })\n\n    // Only configure security groups for traditional RDS (not DSQL)\n    if (\n      !isDsql &&\n      (!props.functionProps?.securityGroups ||\n        props.functionProps?.securityGroups.length === 0)\n    ) {\n      const rdsCluster = props.cluster as IDatabaseCluster | IDatabaseInstance\n      rdsCluster.connections.allowDefaultPortFrom(\n        fn,\n        \"Allow the rds sql handler to connect to db\"\n      )\n    }\n\n    return fn\n  }\n}\n\nclass ImportedProvider extends Construct implements IProvider {\n  public readonly serviceToken: string\n  public readonly handler: IFunction\n  public readonly secret?: ISecret\n  public readonly engine: string\n  public readonly cluster?: IDatabaseCluster | IDatabaseInstance | dsql.CfnCluster\n\n  constructor(scope: Construct, id: string, attrs: ProviderAttributes) {\n    super(scope, id)\n\n    // Validate that either functionArn or functionName is provided\n    if (!attrs.functionArn && !attrs.functionName) {\n      throw new Error(\"Either functionArn or functionName must be provided\")\n    }\n    if (attrs.functionArn && attrs.functionName) {\n      throw new Error(\"Provide either functionArn or functionName, not both\")\n    }\n\n    // Import the existing Lambda function\n    this.handler = attrs.functionArn\n      ? Function.fromFunctionArn(this, \"Handler\", attrs.functionArn)\n      : Function.fromFunctionName(this, \"Handler\", attrs.functionName!)\n\n    // Derive serviceToken by wrapping in custom resource provider\n    const provider = new customResources.Provider(this, \"Provider\", {\n      onEventHandler: this.handler,\n    })\n    this.serviceToken = provider.serviceToken\n\n    this.engine = attrs.engine\n    this.secret = undefined // Imported providers get secret from environment\n    this.cluster = attrs.cluster // Optional cluster for role creation\n  }\n}\n\nfunction slugify(x: string): string {\n  return x.replace(/[^a-zA-Z0-9]/g, \"\")\n}\n"]}
;