@matthewbonig/rds-tools
Version:
A construct for working with RDS SQL servers
131 lines • 20.7 kB
JavaScript
;
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.DatabaseScript = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
const child_process_1 = require("child_process");
const os = require("os");
const path = require("path");
const aws_cdk_lib_1 = require("aws-cdk-lib");
const constructs_1 = require("constructs");
class DatabaseScript extends constructs_1.Construct {
constructor(scope, id, props) {
super(scope, id);
const secret = props.secret || props.databaseInstance?.secret;
if (!secret) {
throw new Error('You must either provide a secret or there must be one available on the databaseInstance');
}
const vpc = props.vpc || props.databaseInstance?.vpc;
if (!vpc) {
throw new Error('Please provide a VPC to use, either on the `vpc` prop or via the `databaseInstance` prop.');
}
const assetPath = path.join(__dirname, 'layer');
this.providerLayer = new aws_cdk_lib_1.aws_lambda.LayerVersion(this, `${id}-deps-layer`, {
code: aws_cdk_lib_1.aws_lambda.Code.fromAsset(assetPath, {
bundling: {
image: aws_cdk_lib_1.aws_lambda.Runtime.NODEJS_16_X.bundlingImage,
command: [
'bash', '-c',
'echo npm i && cp -r /asset-input/* /asset-output',
],
environment: {
npm_config_cache: 'npm-cache',
},
user: 'root',
workingDirectory: '/asset-input/nodejs',
local: {
tryBundle(outputDir) {
console.log('Going to try local bundling...');
if (os.platform() !== 'linux') {
console.warn('When using local bundling on another OS besides linux, you may end up building dependencies that will not run on AWS Lambda. Please build on a linux OS if you run into issues.');
}
const execOptions = { stdio: ['ignore', process.stderr, 'inherit'] };
try {
const layerDir = path.join(__dirname, 'layer');
(0, child_process_1.execSync)('npm install', {
...execOptions,
cwd: path.join(layerDir, 'nodejs'),
});
(0, child_process_1.execSync)(`mkdir -p ${outputDir}/nodejs/node_modules`, { ...execOptions });
(0, child_process_1.execSync)(`cp -r ${layerDir}/nodejs/node_modules/* ${outputDir}/nodejs/node_modules`, { ...execOptions });
}
catch {
return false;
}
return true;
},
},
},
}),
});
const handler = this.handler = this.createLambda('cr', props, vpc, secret);
new aws_cdk_lib_1.CustomResource(this, `${id}-customResource`, {
serviceToken: handler.functionArn,
properties: {
script: props.script,
databaseName: props.databaseName,
},
});
if (props.enableAdhoc) {
this.adhocHandler = this.createLambda('adhoc', props, vpc, secret, 'adhocHandler');
}
}
get connections() {
return this.handler.connections;
}
get adhocConnections() {
if (!this.adhocHandler) {
throw new Error('Please enable the adhoc handler using the enableAdhoc prop.');
}
return this.adhocHandler?.connections;
}
/**
* Grants access to the Lambda Function to the given SecurityGroup.
* Adds an ingress rule to the given security group and for the given port.
* @deprecated Do not use, pass this construct as an IConnectable
* @param securityGroup
* @param port
*/
bind(securityGroup, port) {
securityGroup.addIngressRule(this.handler.connections.securityGroups[0], port, 'access from Lambda ' + this.handler.node.id);
return this;
}
slugify(x) {
return x.replace(/[^a-zA-Z0-9]/g, '');
}
createLambda(id, props, vpc, secret, handler) {
const handlerFunction = this.ensureLambda(`${props.databaseInstance?.node.id ?? props.secret?.node.id}-${id}`, {
entry: path.join(__dirname, 'handlers', 'index.ts'),
depsLockFilePath: path.join(__dirname, 'handlers', 'package-lock.json'),
handler: handler ?? 'handler',
runtime: aws_cdk_lib_1.aws_lambda.Runtime.NODEJS_16_X,
vpc: vpc,
environment: {
SECRET_ARN: secret.secretArn,
},
bundling: {
externalModules: ['aws-sdk', 'mssql', 'promise-mysql', 'pg'],
},
timeout: aws_cdk_lib_1.Duration.seconds(15),
logRetention: aws_cdk_lib_1.aws_logs.RetentionDays.ONE_DAY,
});
handlerFunction.addLayers(this.providerLayer);
secret.grantRead(handlerFunction);
return handlerFunction;
}
ensureLambda(id, props) {
// TODO: Copy-pasted from CDK codebase until
// https://github.com/aws/aws-cdk/issues/6261 is fixed and we can
// use a proper SingletonFunction
const constructName = this.slugify(id) + 'singl';
const existing = aws_cdk_lib_1.Stack.of(this).node.tryFindChild(constructName);
if (existing) {
return existing;
}
return new aws_cdk_lib_1.aws_lambda_nodejs.NodejsFunction(aws_cdk_lib_1.Stack.of(this), constructName, props);
}
}
exports.DatabaseScript = DatabaseScript;
_a = JSII_RTTI_SYMBOL_1;
DatabaseScript[_a] = { fqn: "@matthewbonig/rds-tools.DatabaseScript", version: "2.0.19" };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"DatabaseScript.js","sourceRoot":"","sources":["../src/DatabaseScript.ts"],"names":[],"mappings":"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","sourcesContent":["import { execSync, ExecSyncOptions } from 'child_process';\nimport * as os from 'os';\nimport * as path from 'path';\nimport {\n  aws_ec2 as ec2,\n  aws_lambda,\n  aws_lambda_nodejs,\n  aws_logs,\n  aws_rds,\n  aws_secretsmanager,\n  CustomResource,\n  Duration,\n  Stack,\n} from 'aws-cdk-lib';\nimport { Connections, IConnectable, IVpc } from 'aws-cdk-lib/aws-ec2';\nimport { ISecret } from 'aws-cdk-lib/aws-secretsmanager';\nimport { Construct } from 'constructs';\n\nexport interface DatabaseScriptProps {\n\n  /**\n   * The VPC for the Lambda Function to attach to. If one is not provide, it's assumed from the database instance.\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * An optional databaseName. If none is provided then it will be the default for the rds instance, as defined by the AWS docs.\n   *\n   * mysql - mysql\n   * mssql - master\n   * postgres - postgres\n   *\n   */\n  readonly databaseName?: string;\n\n  /**\n   * The database instance to run the script against\n   */\n  readonly databaseInstance?: aws_rds.DatabaseInstance;\n\n  /**\n   * An optional secret that provides credentials for the database. Must have fields 'username' and 'password'\n   *\n   * @default the root secret from the database instance\n   */\n  readonly secret?: aws_secretsmanager.ISecret;\n\n  /**\n   * The script to execute.\n   */\n  readonly script: string;\n\n  /**\n   * Deploy a second Lambda function that allows for adhoc sql against the database?\n   *\n   * @default false\n   */\n  readonly enableAdhoc?: boolean;\n}\n\nexport class DatabaseScript extends Construct implements IConnectable {\n\n  /**\n   * The underlying Lambda handler function for making adhoc commands against the database.\n   * Undefined unless 'enableAdhoc' is true\n   */\n  adhocHandler?: aws_lambda.IFunction;\n  readonly handler: aws_lambda.IFunction;\n  private readonly providerLayer: aws_lambda.LayerVersion;\n\n  constructor(scope: Construct, id: string, props: DatabaseScriptProps) {\n    super(scope, id);\n\n    const secret = props.secret || props.databaseInstance?.secret;\n    if (!secret) {\n      throw new Error('You must either provide a secret or there must be one available on the databaseInstance');\n    }\n\n    const vpc = props.vpc || props.databaseInstance?.vpc;\n    if (!vpc) {\n      throw new Error('Please provide a VPC to use, either on the `vpc` prop or via the `databaseInstance` prop.');\n    }\n\n    const assetPath = path.join(__dirname, 'layer');\n    this.providerLayer = new aws_lambda.LayerVersion(this, `${id}-deps-layer`, {\n      code: aws_lambda.Code.fromAsset(assetPath, {\n        bundling: {\n          image: aws_lambda.Runtime.NODEJS_16_X.bundlingImage,\n          command: [\n            'bash', '-c',\n            'echo npm i && cp -r /asset-input/* /asset-output',\n          ],\n          environment: {\n            npm_config_cache: 'npm-cache',\n          },\n          user: 'root',\n          workingDirectory: '/asset-input/nodejs',\n          local: {\n            tryBundle(outputDir: string): boolean {\n              console.log('Going to try local bundling...');\n              if (os.platform() !== 'linux') {\n                console.warn('When using local bundling on another OS besides linux, you may end up building dependencies that will not run on AWS Lambda. Please build on a linux OS if you run into issues.');\n              }\n              const execOptions: ExecSyncOptions = { stdio: ['ignore', process.stderr, 'inherit'] };\n              try {\n                const layerDir = path.join(__dirname, 'layer');\n                execSync('npm install', {\n                  ...execOptions,\n                  cwd: path.join(layerDir, 'nodejs'),\n                });\n                execSync(`mkdir -p ${outputDir}/nodejs/node_modules`, { ...execOptions });\n                execSync(`cp -r ${layerDir}/nodejs/node_modules/* ${outputDir}/nodejs/node_modules`, { ...execOptions });\n\n              } catch {\n                return false;\n              }\n              return true;\n            },\n          },\n        },\n      }),\n    });\n\n    const handler = this.handler = this.createLambda('cr', props, vpc, secret);\n    new CustomResource(this, `${id}-customResource`, {\n      serviceToken: handler.functionArn,\n      properties: {\n        script: props.script,\n        databaseName: props.databaseName,\n      },\n    });\n\n    if (props.enableAdhoc) {\n      this.adhocHandler = this.createLambda('adhoc', props, vpc, secret, 'adhocHandler');\n    }\n\n  }\n\n  get connections(): Connections {\n    return this.handler.connections;\n  }\n\n  get adhocConnections(): Connections {\n    if (!this.adhocHandler) {\n      throw new Error('Please enable the adhoc handler using the enableAdhoc prop.');\n    }\n    return this.adhocHandler?.connections;\n  }\n\n  /**\n   * Grants access to the Lambda Function to the given SecurityGroup.\n   * Adds an ingress rule to the given security group and for the given port.\n   * @deprecated Do not use, pass this construct as an IConnectable\n   * @param securityGroup\n   * @param port\n   */\n  bind(securityGroup: ec2.SecurityGroup, port: ec2.Port): DatabaseScript {\n    securityGroup.addIngressRule(this.handler.connections.securityGroups[0], port, 'access from Lambda ' + this.handler.node.id);\n    return this;\n  }\n\n  slugify(x: string): string {\n    return x.replace(/[^a-zA-Z0-9]/g, '');\n  }\n\n  private createLambda(id: string, props: DatabaseScriptProps, vpc: IVpc, secret: ISecret, handler?: string) {\n    const handlerFunction = this.ensureLambda(`${props.databaseInstance?.node.id ?? props.secret?.node.id}-${id}`, {\n      entry: path.join(__dirname, 'handlers', 'index.ts'),\n      depsLockFilePath: path.join(__dirname, 'handlers', 'package-lock.json'),\n      handler: handler ?? 'handler',\n      runtime: aws_lambda.Runtime.NODEJS_16_X,\n      vpc: vpc,\n      environment: {\n        SECRET_ARN: secret.secretArn,\n      },\n      bundling: {\n        externalModules: ['aws-sdk', 'mssql', 'promise-mysql', 'pg'],\n      },\n      timeout: Duration.seconds(15), // TODO: should be overridable\n      logRetention: aws_logs.RetentionDays.ONE_DAY,\n    });\n\n    handlerFunction.addLayers(this.providerLayer);\n\n    secret.grantRead(handlerFunction);\n    return handlerFunction;\n  }\n\n  private ensureLambda(id: string, props: aws_lambda_nodejs.NodejsFunctionProps): aws_lambda_nodejs.NodejsFunction {\n    // TODO: Copy-pasted from CDK codebase until\n    //       https://github.com/aws/aws-cdk/issues/6261 is fixed and we can\n    //       use a proper SingletonFunction\n    const constructName = this.slugify(id) + 'singl';\n    const existing = Stack.of(this).node.tryFindChild(constructName);\n    if (existing) {\n      return existing as aws_lambda_nodejs.NodejsFunction;\n    }\n    return new aws_lambda_nodejs.NodejsFunction(Stack.of(this), constructName, props);\n  }\n}\n"]}