UNPKG

@aws-solutions-constructs/core

Version:
210 lines 33.6 kB
"use strict"; /** * Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved. * * Licensed under the Apache License, Version 2.0 (the "License"). You may not use this file except in compliance * with the License. A copy of the License is located at * * http://www.apache.org/licenses/LICENSE-2.0 * * or in the 'license' file accompanying this file. This file is distributed on an 'AS IS' BASIS, WITHOUT WARRANTIES * OR CONDITIONS OF ANY KIND, express or implied. See the License for the specific language governing permissions * and limitations under the License. */ Object.defineProperty(exports, "__esModule", { value: true }); exports.SinkStoreType = void 0; exports.buildGlueJob = buildGlueJob; exports.deployGlueJob = deployGlueJob; exports.createGlueJobRole = createGlueJobRole; exports.createGlueTable = createGlueTable; exports.createGlueDatabase = createGlueDatabase; const glue = require("aws-cdk-lib/aws-glue"); const aws_iam_1 = require("aws-cdk-lib/aws-iam"); const aws_s3_1 = require("aws-cdk-lib/aws-s3"); const aws_cdk_lib_1 = require("aws-cdk-lib"); const defaults = require("../"); const utils_1 = require("./utils"); /** * Enumeration of data store types that could include S3, DynamoDB, DocumentDB, RDS or Redshift. Current * construct implementation only supports S3, but potential to add other output types in the future */ var SinkStoreType; (function (SinkStoreType) { SinkStoreType["S3"] = "S3"; })(SinkStoreType || (exports.SinkStoreType = SinkStoreType = {})); /** * @internal This is an internal core function and should not be called directly by Solutions Constructs clients. */ function buildGlueJob(scope, props) { if (!props.existingCfnJob) { if (props.glueJobProps) { if (props.glueJobProps.glueVersion === '2.0' && props.glueJobProps.maxCapacity) { throw Error('Cannot set "MaxCapacity" with GlueVersion 2.0 or higher. Use "NumberOfWorkers" and "WorkerType". ' + 'Refer the API documentation https://docs.aws.amazon.com/glue/latest/webapi/API_Job.html for more details'); } if (props.glueJobProps.maxCapacity && (props.glueJobProps.numberOfWorkers || props.glueJobProps.workerType)) { throw Error('Cannot set MaxCapacity and "WorkerType" or "NumberOfWorkers". If using glueVersion 2.0 or beyond, ' + 'it is recommended to use "WorkerType" or "NumberOfWorkers"'); } const deployGlueJobResponse = deployGlueJob(scope, props.glueJobProps, props.database, props.table, props.outputDataStore, props.etlCodeAsset); return { job: deployGlueJobResponse.job, role: deployGlueJobResponse.role, bucket: deployGlueJobResponse.bucket, loggingBucket: deployGlueJobResponse.loggingBucket }; } else { throw Error('Either glueJobProps or existingCfnJob is required'); } } else { return { job: props.existingCfnJob, role: aws_iam_1.Role.fromRoleArn(scope, 'ExistingRole', props.existingCfnJob.role) }; } } /** * @internal This is an internal core function and should not be called directly by Solutions Constructs clients. */ function deployGlueJob(scope, glueJobProps, database, table, outputDataStore, etlCodeAsset) { let glueSecurityConfigName; if (glueJobProps.securityConfiguration === undefined) { glueSecurityConfigName = `ETLJobSecurityConfig${aws_cdk_lib_1.Aws.STACK_ID}`; const glueKMSKey = `arn:${aws_cdk_lib_1.Aws.PARTITION}:kms:${aws_cdk_lib_1.Aws.REGION}:${aws_cdk_lib_1.Aws.ACCOUNT_ID}:alias/aws/glue`; const securityConfigurationProps = { name: glueSecurityConfigName, encryptionConfiguration: { jobBookmarksEncryption: { jobBookmarksEncryptionMode: 'CSE-KMS', kmsKeyArn: glueKMSKey }, s3Encryptions: [{ s3EncryptionMode: 'SSE-S3' }] } }; // Before turning off SonarQube for the line, reduce the line to it's minimum new glue.CfnSecurityConfiguration(scope, 'GlueSecurityConfig', securityConfigurationProps); // NOSONAR } else { glueSecurityConfigName = glueJobProps.securityConfiguration; } const glueJobPolicy = new aws_iam_1.Policy(scope, 'LogPolicy', { statements: [ new aws_iam_1.PolicyStatement({ effect: aws_iam_1.Effect.ALLOW, actions: ['logs:CreateLogGroup', 'logs:CreateLogStream', 'logs:PutLogEvents'], resources: [`arn:${aws_cdk_lib_1.Aws.PARTITION}:logs:${aws_cdk_lib_1.Aws.REGION}:${aws_cdk_lib_1.Aws.ACCOUNT_ID}:log-group:/aws-glue/*`] }) ] }); const jobRole = glueJobProps.role ? aws_iam_1.Role.fromRoleArn(scope, 'JobRole', glueJobProps.role) : defaults.createGlueJobRole(scope); glueJobPolicy.attachToRole(jobRole); let outputLocation; if (outputDataStore !== undefined && outputDataStore.datastoreType === SinkStoreType.S3) { if (outputDataStore.existingS3OutputBucket !== undefined) { outputLocation = { bucket: outputDataStore.existingS3OutputBucket }; } else { outputLocation = defaults.buildS3Bucket(scope, { bucketProps: outputDataStore.outputBucketProps }); } } else { outputLocation = defaults.buildS3Bucket(scope, {}); } outputLocation.bucket.grantReadWrite(jobRole); const jobArgumentsList = { "--enable-metrics": true, "--enable-continuous-cloudwatch-log": true, "--database_name": database.ref, "--table_name": table.ref, ...((outputDataStore === undefined || (outputDataStore && outputDataStore.datastoreType === SinkStoreType.S3)) && { '--output_path': `s3a://${outputLocation.bucket.bucketName}/output/` }), ...glueJobProps.defaultArguments }; const newGlueJobProps = (0, utils_1.overrideProps)(defaults.DefaultGlueJobProps(jobRole, glueJobProps, glueSecurityConfigName, jobArgumentsList, etlCodeAsset), glueJobProps); if (etlCodeAsset) { etlCodeAsset.grantRead(jobRole); } else { // create CDK Bucket instance from S3 url and grant read access to Glue Job's service principal if (isJobCommandProperty(newGlueJobProps.command)) { const scriptLocation = newGlueJobProps.command.scriptLocation; // Incoming Props, including scriptLocation, are checked upstream in CheckGlueProps() const scriptBucketLocation = aws_s3_1.Bucket.fromBucketArn(scope, 'ScriptLocation', getS3ArnfromS3Url(scriptLocation)); scriptBucketLocation.grantRead(jobRole); } } const glueJob = new glue.CfnJob(scope, 'KinesisETLJob', newGlueJobProps); return { job: glueJob, role: jobRole, bucket: outputLocation.bucket, loggingBucket: outputLocation.loggingBucket }; } /** * @internal This is an internal core function and should not be called directly by Solutions Constructs clients. * * This is a helper method to create the Role required for the Glue Job. If a role is already created then this * method is not required to be called. * * @param scope - The AWS Construct under which the role is to be created */ function createGlueJobRole(scope) { return new aws_iam_1.Role(scope, 'JobRole', { assumedBy: new aws_iam_1.ServicePrincipal('glue.amazonaws.com'), description: 'Service role that Glue custom ETL jobs will assume for execution', }); } /** * @internal This is an internal core function and should not be called directly by Solutions Constructs clients. * * This method creates an AWS Glue table. The method is called when an existing Glue table is not provided */ function createGlueTable(scope, database, tableProps, fieldSchema, sourceType, parameters) { return defaults.DefaultGlueTable(scope, tableProps !== undefined ? tableProps : defaults.DefaultGlueTableProps(database, fieldSchema, sourceType, parameters)); } /** * @internal This is an internal core function and should not be called directly by Solutions Constructs clients. * * This method creates an AWS Glue database. The method is only called with an existing Glue database type is not provided. * The method uses the user provided props to override the default props for the Glue database * * @param scope * @param databaseProps */ function createGlueDatabase(scope, databaseProps) { const mergedDBProps = (databaseProps !== undefined) ? (0, utils_1.overrideProps)(defaults.DefaultGlueDatabaseProps(), databaseProps) : defaults.DefaultGlueDatabaseProps(); return defaults.DefaultGlueDatabase(scope, mergedDBProps); } /** * A utility method to generate the S3 Arn from an S3 Url. * * @param s3Url */ function getS3ArnfromS3Url(s3Url) { if (s3Url && s3Url.startsWith('s3://')) { const splitString = s3Url.slice('s3://'.length); return `arn:${aws_cdk_lib_1.Aws.PARTITION}:s3:::${splitString}`; } else { throw Error(`Received S3URL as ${s3Url}. The S3 url string does not begin with s3://. This is not a standard S3 url`); } } /** * A utility method to type check CfnJob.JobCommandProperty type. For the construct to work for streaming ETL from Kinesis Data * Streams, all three attributes of the JobCommandProperty are required, even though they may be optional for other use cases. * * @param command */ function isJobCommandProperty(command) { if (command.name && command.pythonVersion && command.scriptLocation) { return true; } else { defaults.printWarning('command not of type JobCommandProperty type'); return false; } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"glue-job-helper.js","sourceRoot":"","sources":["glue-job-helper.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n *  Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.\n *\n *  Licensed under the Apache License, Version 2.0 (the \"License\"). You may not use this file except in compliance\n *  with the License. A copy of the License is located at\n *\n *      http://www.apache.org/licenses/LICENSE-2.0\n *\n *  or in the 'license' file accompanying this file. This file is distributed on an 'AS IS' BASIS, WITHOUT WARRANTIES\n *  OR CONDITIONS OF ANY KIND, express or implied. See the License for the specific language governing permissions\n *  and limitations under the License.\n */\n\n/*\n *  The functions found here in the core library are for internal use and can be changed\n *  or removed outside of a major release. We recommend against calling them directly from client code.\n */\n\nimport { Construct } from 'constructs';\nimport * as glue from 'aws-cdk-lib/aws-glue';\nimport { Effect, IRole, Policy, PolicyStatement, Role, ServicePrincipal } from 'aws-cdk-lib/aws-iam';\nimport { Bucket, BucketProps, IBucket } from 'aws-cdk-lib/aws-s3';\nimport { Aws, IResolvable } from 'aws-cdk-lib';\nimport * as s3assets from \"aws-cdk-lib/aws-s3-assets\";\nimport * as defaults from '../';\nimport { overrideProps } from './utils';\nimport { BuildS3BucketResponse } from '../';\n\n/**\n * Enumeration of data store types that could include S3, DynamoDB, DocumentDB, RDS or Redshift. Current\n * construct implementation only supports S3, but potential to add other output types in the future\n */\nexport enum SinkStoreType {\n  S3 = 'S3'\n}\n\n/**\n * Interface to define potential outputs to allow the construct define additional output destinations for ETL\n * transformation\n */\nexport interface SinkDataStoreProps {\n  /**\n   * Sink data store type\n   */\n  readonly datastoreType: SinkStoreType;\n  /**\n   * The output S3 location where the data should be written. The provided S3 bucket will be used to pass\n   * the output location to the etl script as an argument to the AWS Glue job.\n   *\n   * If no location is provided, it will check if @outputBucketProps are provided. If not it will create a new\n   * bucket if the @datastoreType is S3.\n   *\n   * The argument key is `output_path`. The value of the argument can be retrieve in the python script\n   * as follows:\n   *  getResolvedOptions(sys.argv, [\"JOB_NAME\", \"output_path\", <other arguments that are passed> ])\n   *  output_path = args[\"output_path\"]\n   */\n  readonly existingS3OutputBucket?: Bucket\n  /**\n   * If @existingS3OutputBUcket is provided, this parameter is ignored. If this parameter is not provided,\n   * the construct will create a new bucket if the @datastoreType is S3.\n   */\n  readonly outputBucketProps?: BucketProps;\n}\n\nexport interface BuildGlueJobProps {\n  /**\n   * Glue ETL job properties.\n   */\n  readonly glueJobProps?: glue.CfnJobProps | any\n  /**\n   * Existing instance of the S3 bucket object, if this is set then the script location is ignored.\n   */\n  readonly existingCfnJob?: glue.CfnJob;\n  /**\n   * AWS Glue table\n   */\n  readonly table: glue.CfnTable;\n  /**\n   * AWS Glue database\n   */\n  readonly database: glue.CfnDatabase;\n  /**\n   * Output storage options\n   */\n  readonly outputDataStore?: SinkDataStoreProps\n  /**\n   * Asset instance for the ETL code that performs Glue Job transformation\n   *\n   * @default - None\n   */\n   readonly etlCodeAsset?: s3assets.Asset;\n}\n\nexport interface BuildGlueJobResponse {\n  readonly job: glue.CfnJob,\n  readonly role: IRole,\n  readonly bucket?: Bucket,\n  readonly loggingBucket?: Bucket,\n}\n\n/**\n * @internal This is an internal core function and should not be called directly by Solutions Constructs clients.\n */\nexport function buildGlueJob(scope: Construct, props: BuildGlueJobProps): BuildGlueJobResponse {\n  if (!props.existingCfnJob) {\n    if (props.glueJobProps) {\n      if (props.glueJobProps.glueVersion === '2.0' && props.glueJobProps.maxCapacity) {\n        throw Error('Cannot set \"MaxCapacity\" with GlueVersion 2.0 or higher. Use \"NumberOfWorkers\" and \"WorkerType\". ' +\n        'Refer the API documentation https://docs.aws.amazon.com/glue/latest/webapi/API_Job.html for more details');\n      }\n\n      if (props.glueJobProps.maxCapacity && (props.glueJobProps.numberOfWorkers || props.glueJobProps.workerType)) {\n        throw Error('Cannot set MaxCapacity and \"WorkerType\" or  \"NumberOfWorkers\". If using glueVersion 2.0 or beyond, ' +\n        'it is recommended to use \"WorkerType\" or  \"NumberOfWorkers\"');\n      }\n\n      const deployGlueJobResponse =\n        deployGlueJob(scope, props.glueJobProps, props.database, props.table, props.outputDataStore!, props.etlCodeAsset);\n      return {\n        job: deployGlueJobResponse.job,\n        role: deployGlueJobResponse.role,\n        bucket: deployGlueJobResponse.bucket,\n        loggingBucket: deployGlueJobResponse.loggingBucket };\n    } else {\n      throw Error('Either glueJobProps or existingCfnJob is required');\n    }\n  } else {\n    return { job: props.existingCfnJob, role: Role.fromRoleArn(scope, 'ExistingRole', props.existingCfnJob.role)};\n  }\n}\n\nexport interface DeployGlueJobResponse {\n  readonly job: glue.CfnJob,\n  readonly role: IRole,\n  readonly bucket?: Bucket,\n  readonly loggingBucket?: Bucket,\n}\n\n/**\n * @internal This is an internal core function and should not be called directly by Solutions Constructs clients.\n */\nexport function deployGlueJob(scope: Construct, glueJobProps: glue.CfnJobProps, database: glue.CfnDatabase, table: glue.CfnTable,\n  outputDataStore: SinkDataStoreProps, etlCodeAsset?: s3assets.Asset): DeployGlueJobResponse {\n\n  let glueSecurityConfigName: string;\n\n  if (glueJobProps.securityConfiguration === undefined) {\n    glueSecurityConfigName = `ETLJobSecurityConfig${Aws.STACK_ID}`;\n    const glueKMSKey = `arn:${Aws.PARTITION}:kms:${Aws.REGION}:${Aws.ACCOUNT_ID}:alias/aws/glue`;\n\n    const securityConfigurationProps: glue.CfnSecurityConfigurationProps = {\n      name: glueSecurityConfigName,\n      encryptionConfiguration: {\n        jobBookmarksEncryption: {\n          jobBookmarksEncryptionMode: 'CSE-KMS',\n          kmsKeyArn: glueKMSKey\n        },\n        s3Encryptions: [{\n          s3EncryptionMode: 'SSE-S3'\n        }]\n      }\n    };\n\n    // Before turning off SonarQube for the line, reduce the line to it's minimum\n    new glue.CfnSecurityConfiguration(scope, 'GlueSecurityConfig', securityConfigurationProps);  // NOSONAR\n\n  } else {\n    glueSecurityConfigName = glueJobProps.securityConfiguration;\n  }\n\n  const glueJobPolicy = new Policy(scope, 'LogPolicy', {\n    statements: [\n      new PolicyStatement({\n        effect: Effect.ALLOW,\n        actions: [ 'logs:CreateLogGroup', 'logs:CreateLogStream', 'logs:PutLogEvents' ],\n        resources: [ `arn:${Aws.PARTITION}:logs:${Aws.REGION}:${Aws.ACCOUNT_ID}:log-group:/aws-glue/*` ]\n      })\n    ]\n  });\n\n  const jobRole = glueJobProps.role ?\n    Role.fromRoleArn(scope, 'JobRole', glueJobProps.role) :\n    defaults.createGlueJobRole(scope);\n\n  glueJobPolicy.attachToRole(jobRole);\n\n  let outputLocation: BuildS3BucketResponse;\n  if (outputDataStore !== undefined && outputDataStore.datastoreType === SinkStoreType.S3) {\n    if (outputDataStore.existingS3OutputBucket !== undefined) {\n      outputLocation = { bucket: outputDataStore.existingS3OutputBucket };\n    } else {\n      outputLocation = defaults.buildS3Bucket(scope, { bucketProps: outputDataStore.outputBucketProps } );\n    }\n  } else {\n    outputLocation = defaults.buildS3Bucket(scope, {});\n  }\n\n  outputLocation.bucket.grantReadWrite(jobRole);\n\n  const jobArgumentsList = {\n    \"--enable-metrics\" : true,\n    \"--enable-continuous-cloudwatch-log\" : true,\n    \"--database_name\": database.ref,\n    \"--table_name\": table.ref,\n    ...((outputDataStore === undefined || (outputDataStore && outputDataStore.datastoreType === SinkStoreType.S3)) &&\n      { '--output_path' : `s3a://${outputLocation.bucket.bucketName}/output/` }),\n    ...glueJobProps.defaultArguments\n  };\n\n  const newGlueJobProps: glue.CfnJobProps = overrideProps(defaults.DefaultGlueJobProps(jobRole, glueJobProps,\n    glueSecurityConfigName, jobArgumentsList, etlCodeAsset), glueJobProps);\n  if (etlCodeAsset) {\n    etlCodeAsset.grantRead(jobRole);\n  } else {\n    // create CDK Bucket instance from S3 url and grant read access to Glue Job's service principal\n    if (isJobCommandProperty(newGlueJobProps.command)) {\n      const scriptLocation = newGlueJobProps.command.scriptLocation;\n\n      // Incoming Props, including scriptLocation, are checked upstream in CheckGlueProps()\n      const scriptBucketLocation: IBucket = Bucket.fromBucketArn(scope, 'ScriptLocation', getS3ArnfromS3Url(scriptLocation!));\n      scriptBucketLocation.grantRead(jobRole);\n    }\n  }\n\n  const glueJob: glue.CfnJob = new glue.CfnJob(scope, 'KinesisETLJob', newGlueJobProps);\n  return  { job: glueJob, role: jobRole, bucket: outputLocation.bucket, loggingBucket: outputLocation.loggingBucket };\n}\n\n/**\n * @internal This is an internal core function and should not be called directly by Solutions Constructs clients.\n *\n * This is a helper method to create the Role required for the Glue Job. If a role is already created then this\n * method is not required to be called.\n *\n * @param scope - The AWS Construct under which the role is to be created\n */\nexport function createGlueJobRole(scope: Construct): Role {\n  return new Role(scope, 'JobRole', {\n    assumedBy: new ServicePrincipal('glue.amazonaws.com'),\n    description: 'Service role that Glue custom ETL jobs will assume for execution',\n  });\n}\n\n/**\n * @internal This is an internal core function and should not be called directly by Solutions Constructs clients.\n *\n * This method creates an AWS Glue table. The method is called when an existing Glue table is not provided\n */\nexport function createGlueTable(scope: Construct, database: glue.CfnDatabase, tableProps?: glue.CfnTableProps,\n  fieldSchema?: glue.CfnTable.ColumnProperty [], sourceType?: string, parameters?: any): glue.CfnTable {\n  return defaults.DefaultGlueTable(scope, tableProps !== undefined ? tableProps :\n    defaults.DefaultGlueTableProps(database, fieldSchema!, sourceType, parameters));\n}\n\n/**\n * @internal This is an internal core function and should not be called directly by Solutions Constructs clients.\n *\n * This method creates an AWS Glue database. The method is only called with an existing Glue database type is not provided.\n * The method uses the user provided props to override the default props for the Glue database\n *\n * @param scope\n * @param databaseProps\n */\nexport function createGlueDatabase(scope: Construct,  databaseProps?: glue.CfnDatabaseProps): glue.CfnDatabase {\n  const mergedDBProps: glue.CfnDatabaseProps = (databaseProps !== undefined) ? overrideProps(defaults.DefaultGlueDatabaseProps(), databaseProps) :\n    defaults.DefaultGlueDatabaseProps();\n  return defaults.DefaultGlueDatabase(scope, mergedDBProps);\n}\n\n/**\n * A utility method to generate the S3 Arn from an S3 Url.\n *\n * @param s3Url\n */\nfunction getS3ArnfromS3Url(s3Url: string): string {\n  if (s3Url && s3Url.startsWith('s3://')) {\n    const splitString: string = s3Url.slice('s3://'.length);\n    return `arn:${Aws.PARTITION}:s3:::${splitString}`;\n  } else {\n    throw Error(`Received S3URL as ${s3Url}. The S3 url string does not begin with s3://. This is not a standard S3 url`);\n  }\n}\n\n/**\n * A utility method to type check CfnJob.JobCommandProperty type. For the construct to work for streaming ETL from Kinesis Data\n * Streams, all three attributes of the JobCommandProperty are required, even though they may be optional for other use cases.\n *\n * @param command\n */\nfunction isJobCommandProperty(command: glue.CfnJob.JobCommandProperty | IResolvable): command is glue.CfnJob.JobCommandProperty {\n  if ((command as glue.CfnJob.JobCommandProperty).name &&\n    (command as glue.CfnJob.JobCommandProperty).pythonVersion &&\n    (command as glue.CfnJob.JobCommandProperty).scriptLocation) {\n    return true;\n  } else {\n    defaults.printWarning('command not of type JobCommandProperty type');\n    return false;\n  }\n}\n"]}