aws-ddk-core
Version:
The AWS DataOps Development Kit is an open source development framework for customers that build data workflows and modern data architecture on AWS.
79 lines • 13.3 kB
JavaScript
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.EMRServerlessJobStage = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
const cdk = require("aws-cdk-lib");
const aws_cdk_lib_1 = require("aws-cdk-lib");
const iam = require("aws-cdk-lib/aws-iam");
const sfn = require("aws-cdk-lib/aws-stepfunctions");
const tasks = require("aws-cdk-lib/aws-stepfunctions-tasks");
const pipelines_1 = require("../pipelines");
/**
* Stage that contains a step function that runs an EMR Job.
*/
class EMRServerlessJobStage extends pipelines_1.StateMachineStage {
/**
* Constructs EMRServerlessJobStage.
* @param scope Scope within which this construct is defined.
* @param id Identifier of the stage.
* @param props Properties for the stage.
*/
constructor(scope, id, props) {
super(scope, id, props);
const stack = cdk.Stack.of(this);
const emrApplicationArn = `arn:${stack.partition}:emr-serverless:${stack.region}:${stack.account}:/applications/${props.applicationId}`;
const jobExecutionStatusWait = new sfn.Wait(this, "Wait Before Checking Job Status", {
time: sfn.WaitTime.duration(props.jobExecutionStatusCheckPeriod ?? aws_cdk_lib_1.Duration.seconds(30)),
});
const runJobTask = new tasks.CallAwsService(this, "Start Job Run", {
service: "emrserverless",
action: "startJobRun",
iamResources: [emrApplicationArn],
parameters: {
ApplicationId: props.applicationId,
ExecutionRoleArn: props.executionRoleArn,
JobDriver: props.jobDriver,
ClientToken: Math.random().toString(36),
...props.startJobRunProps,
},
});
const getJobTask = new tasks.CallAwsService(scope, "Get Job Status", {
service: "emrserverless",
action: "getJobRun",
resultPath: "$.JobStatus",
iamResources: [emrApplicationArn],
parameters: {
"ApplicationId.$": "$.ApplicationId",
"JobRunId.$": "$.JobRunId",
},
});
const successState = new sfn.Succeed(this, "Success State");
const failState = new sfn.Fail(this, "Fail State");
const retryChain = jobExecutionStatusWait.next(getJobTask);
const jobStatusChoice = new sfn.Choice(scope, "Job Status Choice")
.when(sfn.Condition.stringEquals("$.JobStatus.JobRun.State", "SUCCESS"), successState)
.when(sfn.Condition.or(sfn.Condition.stringEquals("$.JobStatus.JobRun.State", "FAILED"), sfn.Condition.stringEquals("$.JobStatus.JobRun.State", "CANCELLED")), failState)
.otherwise(retryChain);
const definition = runJobTask.next(getJobTask).next(jobStatusChoice);
({
eventPattern: this.eventPattern,
targets: this.targets,
stateMachine: this.stateMachine,
} = this.createStateMachine({ definition: definition, ...props }));
this.stateMachine.addToRolePolicy(new iam.PolicyStatement({
effect: iam.Effect.ALLOW,
actions: ["emr-serverless:StartJobRun", "emr-serverless:GetJobRun"],
resources: [emrApplicationArn, `${emrApplicationArn}/jobruns/*`],
}));
this.stateMachine.addToRolePolicy(new iam.PolicyStatement({
effect: iam.Effect.ALLOW,
actions: ["iam:PassRole"],
resources: [props.executionRoleArn],
}));
}
}
exports.EMRServerlessJobStage = EMRServerlessJobStage;
_a = JSII_RTTI_SYMBOL_1;
EMRServerlessJobStage[_a] = { fqn: "aws-ddk-core.EMRServerlessJobStage", version: "1.4.1" };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"emr-serverless-job.js","sourceRoot":"","sources":["../../src/stages/emr-serverless-job.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from \"aws-cdk-lib\";\nimport { Duration } from \"aws-cdk-lib\";\nimport * as events from \"aws-cdk-lib/aws-events\";\nimport * as iam from \"aws-cdk-lib/aws-iam\";\nimport * as sfn from \"aws-cdk-lib/aws-stepfunctions\";\nimport * as tasks from \"aws-cdk-lib/aws-stepfunctions-tasks\";\nimport { Construct } from \"constructs\";\nimport { StateMachineStage, StateMachineStageProps } from \"../pipelines\";\n/**\n * Properties of the EMR Serverless Job stage.\n */\nexport interface EMRServerlessJobStageProps extends StateMachineStageProps {\n  /**\n   * EMR Serverless Application Id.\n   */\n  readonly applicationId: string;\n  /**\n   * EMR Execution Role Arn.\n   */\n  readonly executionRoleArn: string;\n  /**\n   * The job driver for the job run.\n   * This is a Tagged Union structure.\n   * Only one of the following top level\n   * keys can be set: 'sparkSubmit', 'hive'\n   */\n  readonly jobDriver: { [key: string]: any };\n  /**\n   * Duration to wait between polling job status.\n   * Defaults to 30 seconds.\n   */\n  readonly jobExecutionStatusCheckPeriod?: cdk.Duration;\n  /**\n   * Additional properties to pass to 'emrserverless:StartJobRun'.\n   * https://docs.aws.amazon.com/emr-serverless/latest/APIReference/API_StartJobRun.html\n   */\n  readonly startJobRunProps?: { [key: string]: any };\n}\n\n/**\n * Stage that contains a step function that runs an EMR Job.\n */\nexport class EMRServerlessJobStage extends StateMachineStage {\n  readonly targets?: events.IRuleTarget[];\n  readonly eventPattern?: events.EventPattern;\n  readonly stateMachine: sfn.StateMachine;\n\n  /**\n   * Constructs EMRServerlessJobStage.\n   * @param scope Scope within which this construct is defined.\n   * @param id Identifier of the stage.\n   * @param props Properties for the stage.\n   */\n  constructor(scope: Construct, id: string, props: EMRServerlessJobStageProps) {\n    super(scope, id, props);\n\n    const stack = cdk.Stack.of(this);\n    const emrApplicationArn = `arn:${stack.partition}:emr-serverless:${stack.region}:${stack.account}:/applications/${props.applicationId}`;\n    const jobExecutionStatusWait = new sfn.Wait(this, \"Wait Before Checking Job Status\", {\n      time: sfn.WaitTime.duration(props.jobExecutionStatusCheckPeriod ?? Duration.seconds(30)),\n    });\n\n    const runJobTask = new tasks.CallAwsService(this, \"Start Job Run\", {\n      service: \"emrserverless\",\n      action: \"startJobRun\",\n      iamResources: [emrApplicationArn],\n      parameters: {\n        ApplicationId: props.applicationId,\n        ExecutionRoleArn: props.executionRoleArn,\n        JobDriver: props.jobDriver,\n        ClientToken: Math.random().toString(36),\n        ...props.startJobRunProps,\n      },\n    });\n\n    const getJobTask = new tasks.CallAwsService(scope, \"Get Job Status\", {\n      service: \"emrserverless\",\n      action: \"getJobRun\",\n      resultPath: \"$.JobStatus\",\n      iamResources: [emrApplicationArn],\n      parameters: {\n        \"ApplicationId.$\": \"$.ApplicationId\",\n        \"JobRunId.$\": \"$.JobRunId\",\n      },\n    });\n\n    const successState = new sfn.Succeed(this, \"Success State\");\n    const failState = new sfn.Fail(this, \"Fail State\");\n    const retryChain = jobExecutionStatusWait.next(getJobTask);\n    const jobStatusChoice = new sfn.Choice(scope, \"Job Status Choice\")\n      .when(sfn.Condition.stringEquals(\"$.JobStatus.JobRun.State\", \"SUCCESS\"), successState)\n      .when(\n        sfn.Condition.or(\n          sfn.Condition.stringEquals(\"$.JobStatus.JobRun.State\", \"FAILED\"),\n          sfn.Condition.stringEquals(\"$.JobStatus.JobRun.State\", \"CANCELLED\"),\n        ),\n        failState,\n      )\n      .otherwise(retryChain);\n\n    const definition = runJobTask.next(getJobTask).next(jobStatusChoice);\n\n    ({\n      eventPattern: this.eventPattern,\n      targets: this.targets,\n      stateMachine: this.stateMachine,\n    } = this.createStateMachine({ definition: definition, ...props }));\n    this.stateMachine.addToRolePolicy(\n      new iam.PolicyStatement({\n        effect: iam.Effect.ALLOW,\n        actions: [\"emr-serverless:StartJobRun\", \"emr-serverless:GetJobRun\"],\n        resources: [emrApplicationArn, `${emrApplicationArn}/jobruns/*`],\n      }),\n    );\n    this.stateMachine.addToRolePolicy(\n      new iam.PolicyStatement({\n        effect: iam.Effect.ALLOW,\n        actions: [\"iam:PassRole\"],\n        resources: [props.executionRoleArn],\n      }),\n    );\n  }\n}\n"]}
;