UNPKG

@aws-cdk/aws-redshift-alpha

Version:

The CDK Construct Library for AWS::Redshift

105 lines 17 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 helpers_internal_1 = require("aws-cdk-lib/core/lib/helpers-internal"); 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((0, helpers_internal_1.lit) `TimeoutTooShort`, `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((0, helpers_internal_1.lit) `TimeoutTooLong`, `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((0, helpers_internal_1.lit) `AdminUserSecretNotAvailable`, '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((0, helpers_internal_1.lit) `AdminUserSecretNotProvided`, '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 type * as secretsmanager from 'aws-cdk-lib/aws-secretsmanager';\nimport * as cdk from 'aws-cdk-lib/core';\nimport { lit } from 'aws-cdk-lib/core/lib/helpers-internal';\nimport * as customresources from 'aws-cdk-lib/custom-resources';\nimport { Construct } from 'constructs';\nimport type { DatabaseQueryHandlerProps } from './handler-props';\nimport { Cluster } from '../cluster';\nimport type { 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(lit`TimeoutTooShort`, `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(lit`TimeoutTooLong`, `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            lit`AdminUserSecretNotAvailable`,\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          lit`AdminUserSecretNotProvided`,\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"]}