UNPKG

@aws-cdk/aws-redshift-alpha

Version:

The CDK Construct Library for AWS::Redshift

104 lines 16.3 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.DatabaseQuery = void 0; const path = require("path"); const iam = require("aws-cdk-lib/aws-iam"); const lambda = require("aws-cdk-lib/aws-lambda"); const cdk = require("aws-cdk-lib/core"); const customresources = require("aws-cdk-lib/custom-resources"); const constructs_1 = require("constructs"); const cluster_1 = require("../cluster"); class DatabaseQuery extends constructs_1.Construct { grantPrincipal; ref; resource; constructor(scope, id, props) { super(scope, id); if (props.timeout && !cdk.Token.isUnresolved(props.timeout)) { if (props.timeout.toMilliseconds() < cdk.Duration.seconds(1).toMilliseconds()) { throw new cdk.ValidationError(`The timeout for the handler must be BETWEEN 1 second and 15 minutes, got ${props.timeout.toMilliseconds()} milliseconds.`, this); } if (props.timeout.toSeconds() > cdk.Duration.minutes(15).toSeconds()) { throw new cdk.ValidationError(`The timeout for the handler must be between 1 second and 15 minutes, got ${props.timeout.toSeconds()} seconds.`, this); } } const adminUser = this.getAdminUser(props); const handler = new lambda.SingletonFunction(this, 'Handler', { code: lambda.Code.fromAsset(path.join(__dirname, 'database-query-provider'), { exclude: ['*.ts'], }), runtime: lambda.determineLatestNodeRuntime(this), handler: 'index.handler', timeout: props.timeout ?? cdk.Duration.minutes(1), uuid: '3de5bea7-27da-4796-8662-5efb56431b5f', lambdaPurpose: 'Query Redshift Database', }); handler.addToRolePolicy(new iam.PolicyStatement({ actions: ['redshift-data:DescribeStatement', 'redshift-data:ExecuteStatement'], resources: ['*'], })); adminUser.grantRead(handler); const provider = new customresources.Provider(this, 'Provider', { onEventHandler: handler, role: this.getOrCreateInvokerRole(handler), }); const queryHandlerProps = { handler: props.handler, clusterName: props.cluster.clusterName, adminUserArn: adminUser.secretArn, databaseName: props.databaseName, ...props.properties, }; this.resource = new cdk.CustomResource(this, 'Resource', { resourceType: 'Custom::RedshiftDatabaseQuery', serviceToken: provider.serviceToken, removalPolicy: props.removalPolicy, properties: queryHandlerProps, }); this.grantPrincipal = handler.grantPrincipal; this.ref = this.resource.ref; } applyRemovalPolicy(policy) { this.resource.applyRemovalPolicy(policy); } getAtt(attributeName) { return this.resource.getAtt(attributeName); } getAttString(attributeName) { return this.resource.getAttString(attributeName); } getAdminUser(props) { const cluster = props.cluster; let adminUser = props.adminUser; if (!adminUser) { if (cluster instanceof cluster_1.Cluster) { if (cluster.secret) { adminUser = cluster.secret; } else { throw new cdk.ValidationError('Administrative access to the Redshift cluster is required but an admin user secret was not provided and the cluster did not generate admin user credentials (they were provided explicitly)', this); } } else { throw new cdk.ValidationError('Administrative access to the Redshift cluster is required but an admin user secret was not provided and the cluster was imported', this); } } return adminUser; } /** * Get or create the IAM role for the singleton lambda function. * We only need one function since it's just acting as an invoker. * */ getOrCreateInvokerRole(handler) { const id = handler.constructName + 'InvokerRole'; const existing = cdk.Stack.of(this).node.tryFindChild(id); return existing != null ? existing : new iam.Role(cdk.Stack.of(this), id, { assumedBy: new iam.ServicePrincipal('lambda.amazonaws.com'), managedPolicies: [iam.ManagedPolicy.fromAwsManagedPolicyName('service-role/AWSLambdaBasicExecutionRole')], }); } } exports.DatabaseQuery = DatabaseQuery; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"database-query.js","sourceRoot":"","sources":["database-query.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport * as iam from 'aws-cdk-lib/aws-iam';\nimport * as lambda from 'aws-cdk-lib/aws-lambda';\nimport * as secretsmanager from 'aws-cdk-lib/aws-secretsmanager';\nimport * as cdk from 'aws-cdk-lib/core';\nimport * as customresources from 'aws-cdk-lib/custom-resources';\nimport { Construct } from 'constructs';\nimport { DatabaseQueryHandlerProps } from './handler-props';\nimport { Cluster } from '../cluster';\nimport { DatabaseOptions } from '../database-options';\n\nexport interface DatabaseQueryProps<HandlerProps> extends DatabaseOptions {\n  readonly handler: string;\n  readonly properties: HandlerProps;\n  /**\n   * The policy to apply when this resource is removed from the application.\n   *\n   * @default cdk.RemovalPolicy.Destroy\n   */\n  readonly removalPolicy?: cdk.RemovalPolicy;\n\n  /**\n   * The handler timeout duration\n   *\n   * @default cdk.Duration.minutes(1)\n   */\n  readonly timeout?: cdk.Duration;\n}\n\nexport class DatabaseQuery<HandlerProps> extends Construct implements iam.IGrantable {\n  readonly grantPrincipal: iam.IPrincipal;\n  readonly ref: string;\n\n  private readonly resource: cdk.CustomResource;\n\n  constructor(scope: Construct, id: string, props: DatabaseQueryProps<HandlerProps>) {\n    super(scope, id);\n\n    if (props.timeout && !cdk.Token.isUnresolved(props.timeout)) {\n      if (props.timeout.toMilliseconds() < cdk.Duration.seconds(1).toMilliseconds()) {\n        throw new cdk.ValidationError(`The timeout for the handler must be BETWEEN 1 second and 15 minutes, got ${props.timeout.toMilliseconds()} milliseconds.`, this);\n      }\n      if (props.timeout.toSeconds() > cdk.Duration.minutes(15).toSeconds()) {\n        throw new cdk.ValidationError(`The timeout for the handler must be between 1 second and 15 minutes, got ${props.timeout.toSeconds()} seconds.`, this);\n      }\n    }\n\n    const adminUser = this.getAdminUser(props);\n    const handler = new lambda.SingletonFunction(this, 'Handler', {\n      code: lambda.Code.fromAsset(path.join(__dirname, 'database-query-provider'), {\n        exclude: ['*.ts'],\n      }),\n      runtime: lambda.determineLatestNodeRuntime(this),\n      handler: 'index.handler',\n      timeout: props.timeout ?? cdk.Duration.minutes(1),\n      uuid: '3de5bea7-27da-4796-8662-5efb56431b5f',\n      lambdaPurpose: 'Query Redshift Database',\n    });\n    handler.addToRolePolicy(new iam.PolicyStatement({\n      actions: ['redshift-data:DescribeStatement', 'redshift-data:ExecuteStatement'],\n      resources: ['*'],\n    }));\n    adminUser.grantRead(handler);\n\n    const provider = new customresources.Provider(this, 'Provider', {\n      onEventHandler: handler,\n      role: this.getOrCreateInvokerRole(handler),\n    });\n\n    const queryHandlerProps: DatabaseQueryHandlerProps & HandlerProps = {\n      handler: props.handler,\n      clusterName: props.cluster.clusterName,\n      adminUserArn: adminUser.secretArn,\n      databaseName: props.databaseName,\n      ...props.properties,\n    };\n    this.resource = new cdk.CustomResource(this, 'Resource', {\n      resourceType: 'Custom::RedshiftDatabaseQuery',\n      serviceToken: provider.serviceToken,\n      removalPolicy: props.removalPolicy,\n      properties: queryHandlerProps,\n    });\n\n    this.grantPrincipal = handler.grantPrincipal;\n    this.ref = this.resource.ref;\n  }\n\n  public applyRemovalPolicy(policy: cdk.RemovalPolicy): void {\n    this.resource.applyRemovalPolicy(policy);\n  }\n\n  public getAtt(attributeName: string): cdk.Reference {\n    return this.resource.getAtt(attributeName);\n  }\n\n  public getAttString(attributeName: string): string {\n    return this.resource.getAttString(attributeName);\n  }\n\n  private getAdminUser(props: DatabaseOptions): secretsmanager.ISecret {\n    const cluster = props.cluster;\n    let adminUser = props.adminUser;\n    if (!adminUser) {\n      if (cluster instanceof Cluster) {\n        if (cluster.secret) {\n          adminUser = cluster.secret;\n        } else {\n          throw new cdk.ValidationError(\n            'Administrative access to the Redshift cluster is required but an admin user secret was not provided and the cluster did not generate admin user credentials (they were provided explicitly)',\n            this,\n          );\n        }\n      } else {\n        throw new cdk.ValidationError(\n          'Administrative access to the Redshift cluster is required but an admin user secret was not provided and the cluster was imported',\n          this,\n        );\n      }\n    }\n    return adminUser;\n  }\n\n  /**\n   * Get or create the IAM role for the singleton lambda function.\n   * We only need one function since it's just acting as an invoker.\n   * */\n  private getOrCreateInvokerRole(handler: lambda.SingletonFunction): iam.IRole {\n    const id = handler.constructName + 'InvokerRole';\n    const existing = cdk.Stack.of(this).node.tryFindChild(id);\n    return existing != null\n      ? existing as iam.Role\n      : new iam.Role(cdk.Stack.of(this), id, {\n        assumedBy: new iam.ServicePrincipal('lambda.amazonaws.com'),\n        managedPolicies: [iam.ManagedPolicy.fromAwsManagedPolicyName('service-role/AWSLambdaBasicExecutionRole')],\n      });\n  }\n}\n"]}