@aws-cdk/aws-glue-alpha
Version:
The CDK Construct Library for AWS::Glue
422 lines • 47.6 kB
JavaScript
"use strict";
var __esDecorate = (this && this.__esDecorate) || function (ctor, descriptorIn, decorators, contextIn, initializers, extraInitializers) {
function accept(f) { if (f !== void 0 && typeof f !== "function") throw new TypeError("Function expected"); return f; }
var kind = contextIn.kind, key = kind === "getter" ? "get" : kind === "setter" ? "set" : "value";
var target = !descriptorIn && ctor ? contextIn["static"] ? ctor : ctor.prototype : null;
var descriptor = descriptorIn || (target ? Object.getOwnPropertyDescriptor(target, contextIn.name) : {});
var _, done = false;
for (var i = decorators.length - 1; i >= 0; i--) {
var context = {};
for (var p in contextIn) context[p] = p === "access" ? {} : contextIn[p];
for (var p in contextIn.access) context.access[p] = contextIn.access[p];
context.addInitializer = function (f) { if (done) throw new TypeError("Cannot add initializers after decoration has completed"); extraInitializers.push(accept(f || null)); };
var result = (0, decorators[i])(kind === "accessor" ? { get: descriptor.get, set: descriptor.set } : descriptor[key], context);
if (kind === "accessor") {
if (result === void 0) continue;
if (result === null || typeof result !== "object") throw new TypeError("Object expected");
if (_ = accept(result.get)) descriptor.get = _;
if (_ = accept(result.set)) descriptor.set = _;
if (_ = accept(result.init)) initializers.unshift(_);
}
else if (_ = accept(result)) {
if (kind === "field") initializers.unshift(_);
else descriptor[key] = _;
}
}
if (target) Object.defineProperty(target, contextIn.name, descriptor);
done = true;
};
var __runInitializers = (this && this.__runInitializers) || function (thisArg, initializers, value) {
var useValue = arguments.length > 2;
for (var i = 0; i < initializers.length; i++) {
value = useValue ? initializers[i].call(thisArg, value) : initializers[i].call(thisArg);
}
return useValue ? value : void 0;
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.Workflow = exports.WorkflowBase = void 0;
const jsiiDeprecationWarnings = require("../../.warnings.jsii.js");
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
const cdk = require("aws-cdk-lib/core");
const aws_glue_1 = require("aws-cdk-lib/aws-glue");
const constants_1 = require("../constants");
const trigger_options_1 = require("./trigger-options");
const metadata_resource_1 = require("aws-cdk-lib/core/lib/metadata-resource");
const prop_injectable_1 = require("aws-cdk-lib/core/lib/prop-injectable");
/**
* Base abstract class for Workflow
*
* @see https://docs.aws.amazon.com/glue/latest/dg/about-triggers.html
*/
class WorkflowBase extends cdk.Resource {
static [JSII_RTTI_SYMBOL_1] = { fqn: "@aws-cdk/aws-glue-alpha.WorkflowBase", version: "2.225.0-alpha.0" };
/**
* Extract workflowName from arn
*/
static extractNameFromArn(scope, workflowArn) {
return cdk.Stack.of(scope).splitArn(workflowArn, cdk.ArnFormat.SLASH_RESOURCE_NAME).resourceName;
}
/**
* Add an on-demand trigger to the workflow.
*
* @param id The id of the trigger.
* @param options Additional options for the trigger.
* @throws If both job and crawler are provided, or if neither job nor crawler is provided.
* @returns The created CfnTrigger resource.
*/
addOnDemandTrigger(id, options) {
try {
jsiiDeprecationWarnings._aws_cdk_aws_glue_alpha_OnDemandTriggerOptions(options);
}
catch (error) {
if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
Error.captureStackTrace(error, this.addOnDemandTrigger);
}
throw error;
}
const trigger = new aws_glue_1.CfnTrigger(this, id, {
...options,
workflowName: this.workflowName,
type: 'ON_DEMAND',
actions: options.actions?.map(this.renderAction.bind(this)),
description: options.description || undefined,
});
return trigger;
}
/**
* Add a daily-scheduled trigger to the workflow.
*
* @param id The id of the trigger.
* @param options Additional options for the trigger.
* @throws If both job and crawler are provided, or if neither job nor crawler is provided.
* @returns The created CfnTrigger resource.
*/
addDailyScheduledTrigger(id, options) {
try {
jsiiDeprecationWarnings._aws_cdk_aws_glue_alpha_DailyScheduleTriggerOptions(options);
}
catch (error) {
if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
Error.captureStackTrace(error, this.addDailyScheduledTrigger);
}
throw error;
}
const dailySchedule = trigger_options_1.TriggerSchedule.cron({
minute: '0',
hour: '0',
});
const trigger = new aws_glue_1.CfnTrigger(this, id, {
...options,
workflowName: this.workflowName,
type: 'SCHEDULED',
actions: options.actions?.map(this.renderAction.bind(this)),
schedule: dailySchedule.expressionString,
startOnCreation: options.startOnCreation ?? false,
});
return trigger;
}
/**
* Add a weekly-scheduled trigger to the workflow.
*
* @param id The id of the trigger.
* @param options Additional options for the trigger.
* @throws If both job and crawler are provided, or if neither job nor crawler is provided.
* @returns The created CfnTrigger resource.
*/
addWeeklyScheduledTrigger(id, options) {
try {
jsiiDeprecationWarnings._aws_cdk_aws_glue_alpha_WeeklyScheduleTriggerOptions(options);
}
catch (error) {
if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
Error.captureStackTrace(error, this.addWeeklyScheduledTrigger);
}
throw error;
}
const weeklySchedule = trigger_options_1.TriggerSchedule.cron({
minute: '0',
hour: '0',
weekDay: 'SUN',
});
const trigger = new aws_glue_1.CfnTrigger(this, id, {
...options,
workflowName: this.workflowName,
type: 'SCHEDULED',
actions: options.actions?.map(this.renderAction.bind(this)),
schedule: weeklySchedule.expressionString,
startOnCreation: options.startOnCreation ?? false,
});
return trigger;
}
/**
* Add a custom-scheduled trigger to the workflow.
*
* @param id The id of the trigger.
* @param options Additional options for the trigger.
* @throws If both job and crawler are provided, or if neither job nor crawler is provided.
* @returns The created CfnTrigger resource.
*/
addCustomScheduledTrigger(id, options) {
try {
jsiiDeprecationWarnings._aws_cdk_aws_glue_alpha_CustomScheduledTriggerOptions(options);
}
catch (error) {
if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
Error.captureStackTrace(error, this.addCustomScheduledTrigger);
}
throw error;
}
const trigger = new aws_glue_1.CfnTrigger(this, id, {
...options,
workflowName: this.workflowName,
type: 'SCHEDULED',
actions: options.actions?.map(this.renderAction.bind(this)),
schedule: options.schedule.expressionString,
startOnCreation: options.startOnCreation ?? false,
});
return trigger;
}
/**
* Add an Event Bridge based trigger to the workflow.
*
* @param id The id of the trigger.
* @param options Additional options for the trigger.
* @throws If both job and crawler are provided, or if neither job nor crawler is provided.
* @returns The created CfnTrigger resource.
*/
addNotifyEventTrigger(id, options) {
try {
jsiiDeprecationWarnings._aws_cdk_aws_glue_alpha_NotifyEventTriggerOptions(options);
}
catch (error) {
if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
Error.captureStackTrace(error, this.addNotifyEventTrigger);
}
throw error;
}
const trigger = new aws_glue_1.CfnTrigger(this, id, {
...options,
workflowName: this.workflowName,
type: 'EVENT',
actions: options.actions?.map(this.renderAction.bind(this)),
eventBatchingCondition: this.renderEventBatchingCondition(options),
description: options.description ?? undefined,
});
return trigger;
}
/**
* Add a Condition (Predicate) based trigger to the workflow.
*
* @param id The id of the trigger.
* @param options Additional options for the trigger.
* @throws If both job and crawler are provided, or if neither job nor crawler is provided for any condition.
* @throws If a job is provided without a job state, or if a crawler is provided without a crawler state for any condition.
* @returns The created CfnTrigger resource.
*/
addConditionalTrigger(id, options) {
try {
jsiiDeprecationWarnings._aws_cdk_aws_glue_alpha_ConditionalTriggerOptions(options);
}
catch (error) {
if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
Error.captureStackTrace(error, this.addConditionalTrigger);
}
throw error;
}
const trigger = new aws_glue_1.CfnTrigger(this, id, {
...options,
workflowName: this.workflowName,
type: 'CONDITIONAL',
actions: options.actions?.map(this.renderAction.bind(this)),
predicate: this.renderPredicate(options),
eventBatchingCondition: this.renderEventBatchingCondition(options),
description: options.description ?? undefined,
});
return trigger;
}
renderAction(action) {
// Validate that either job or crawler is provided, but not both
if (!action.job && !action.crawler) {
throw new cdk.ValidationError('You must provide either a job or a crawler for the action.', this);
}
else if (action.job && action.crawler) {
throw new cdk.ValidationError('You cannot provide both a job and a crawler for the action.', this);
}
return {
jobName: action.job?.jobName,
arguments: action.arguments,
timeout: action.timeout?.toMinutes(),
securityConfiguration: action.securityConfiguration?.securityConfigurationName,
crawlerName: action.crawler?.name,
};
}
renderPredicate(props) {
const conditions = props.predicate.conditions?.map(condition => {
// Validate that either job or crawler is provided, but not both
if (!condition.job && !condition.crawlerName) {
throw new cdk.ValidationError('You must provide either a job or a crawler for the condition.', this);
}
else if (condition.job && condition.crawlerName) {
throw new cdk.ValidationError('You cannot provide both a job and a crawler for the condition.', this);
}
// Validate that if job is provided, job state is also provided
if (condition.job && !condition.state) {
throw new cdk.ValidationError('If you provide a job for the condition, you must also provide a job state.', this);
}
// Validate that if crawler is provided, crawler state is also provided
if (condition.crawlerName && !condition.crawlState) {
throw new cdk.ValidationError('If you provide a crawler for the condition, you must also provide a crawler state.', this);
}
return {
logicalOperator: condition.logicalOperator ?? constants_1.ConditionLogicalOperator.EQUALS,
jobName: condition.job?.jobName ?? undefined,
state: condition.state ?? undefined,
crawlerName: condition.crawlerName ?? undefined,
crawlState: condition.crawlState ?? undefined,
};
});
return {
logical: props.predicate.conditions?.length === 1 ? undefined : props.predicate.logical ?? constants_1.PredicateLogical.AND,
conditions: conditions,
};
}
renderEventBatchingCondition(props) {
const defaultBatchSize = 1;
const defaultBatchWindow = cdk.Duration.seconds(900).toSeconds();
if (!props.eventBatchingCondition) {
return {
batchSize: defaultBatchSize,
batchWindow: defaultBatchWindow,
};
}
return {
batchSize: props.eventBatchingCondition.batchSize || defaultBatchSize,
batchWindow: props.eventBatchingCondition.batchWindow?.toSeconds() || defaultBatchWindow,
};
}
buildWorkflowArn(scope, workflowName) {
return cdk.Stack.of(scope).formatArn({
service: 'glue',
resource: 'workflow',
resourceName: workflowName,
});
}
}
exports.WorkflowBase = WorkflowBase;
/**
* This module defines a construct for creating and managing AWS Glue Workflows and Triggers.
*
* AWS Glue Workflows are orchestration services that allow you to create, manage, and monitor complex extract, transform, and load (ETL) activities involving multiple crawlers, jobs, and triggers. Workflows are designed to allow you to manage interdependent jobs and crawlers as a single unit, making it easier to orchestrate and monitor complex ETL pipelines.
*
* Triggers are used to initiate an AWS Glue Workflow. You can configure different types of triggers, such as on-demand, scheduled, event-based, or conditional triggers, to start your Workflow based on specific conditions or events.
*
* @see https://docs.aws.amazon.com/glue/latest/dg/workflows_overview.html
* @see https://docs.aws.amazon.com/glue/latest/dg/about-triggers.html
*
* ## Usage Example
*
* ```ts
* const app = new App();
* const stack = new Stack(app, 'TestStack');
*
* // Create a Glue Job
* declare const role: iam.IRole;
* declare const script: glue.Code;
* const job = new glue.PySparkStreamingJob(stack, 'ImportedJob', { role, script });
*
* // Create a Glue Workflow
* const workflow = new glue.Workflow(stack, 'TestWorkflow');
*
* // Add an on-demand trigger to the Workflow
* workflow.addOnDemandTrigger('OnDemandTrigger', {
* actions: [{ job: job }],
* });
* ```
*/
let Workflow = (() => {
let _classDecorators = [prop_injectable_1.propertyInjectable];
let _classDescriptor;
let _classExtraInitializers = [];
let _classThis;
let _classSuper = WorkflowBase;
var Workflow = class extends _classSuper {
static { _classThis = this; }
static {
const _metadata = typeof Symbol === "function" && Symbol.metadata ? Object.create(_classSuper[Symbol.metadata] ?? null) : void 0;
__esDecorate(null, _classDescriptor = { value: _classThis }, _classDecorators, { kind: "class", name: _classThis.name, metadata: _metadata }, null, _classExtraInitializers);
Workflow = _classThis = _classDescriptor.value;
if (_metadata) Object.defineProperty(_classThis, Symbol.metadata, { enumerable: true, configurable: true, writable: true, value: _metadata });
}
static [JSII_RTTI_SYMBOL_1] = { fqn: "@aws-cdk/aws-glue-alpha.Workflow", version: "2.225.0-alpha.0" };
/** Uniquely identifies this class. */
static PROPERTY_INJECTION_ID = '@aws-cdk.aws-glue-alpha.Workflow';
/**
* Import a workflow from its name
*/
static fromWorkflowName(scope, id, workflowName) {
return this.fromWorkflowAttributes(scope, id, {
workflowName,
});
}
/**
* Import an workflow from it's name
*/
static fromWorkflowArn(scope, id, workflowArn) {
return this.fromWorkflowAttributes(scope, id, {
workflowName: this.extractNameFromArn(scope, workflowArn),
workflowArn,
});
}
/**
* Import an existing workflow
*/
static fromWorkflowAttributes(scope, id, attrs) {
try {
jsiiDeprecationWarnings._aws_cdk_aws_glue_alpha_WorkflowAttributes(attrs);
}
catch (error) {
if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
Error.captureStackTrace(error, this.fromWorkflowAttributes);
}
throw error;
}
class Import extends WorkflowBase {
workflowName = attrs.workflowName;
workflowArn = this.buildWorkflowArn(scope, this.workflowName);
}
return new Import(scope, id);
}
workflowName;
workflowArn;
constructor(scope, id, props) {
super(scope, id, {
physicalName: props?.workflowName,
});
try {
jsiiDeprecationWarnings._aws_cdk_aws_glue_alpha_WorkflowProps(props);
}
catch (error) {
if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
Error.captureStackTrace(error, Workflow);
}
throw error;
}
// Enhanced CDK Analytics Telemetry
(0, metadata_resource_1.addConstructMetadata)(this, props);
const resource = new aws_glue_1.CfnWorkflow(this, 'Resource', {
name: this.physicalName,
description: props?.description,
defaultRunProperties: props?.defaultRunProperties,
maxConcurrentRuns: props?.maxConcurrentRuns,
});
this.workflowName = this.getResourceNameAttribute(resource.ref);
this.workflowArn = this.buildWorkflowArn(this, this.workflowName);
}
static {
__runInitializers(_classThis, _classExtraInitializers);
}
};
return Workflow = _classThis;
})();
exports.Workflow = Workflow;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"workflow.js","sourceRoot":"","sources":["workflow.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from 'aws-cdk-lib/core';\nimport * as constructs from 'constructs';\nimport { CfnWorkflow, CfnTrigger } from 'aws-cdk-lib/aws-glue';\nimport {\n  ConditionLogicalOperator,\n  PredicateLogical,\n} from '../constants';\nimport {\n  Action,\n  TriggerSchedule,\n  OnDemandTriggerOptions,\n  WeeklyScheduleTriggerOptions,\n  DailyScheduleTriggerOptions,\n  CustomScheduledTriggerOptions,\n  NotifyEventTriggerOptions,\n  ConditionalTriggerOptions,\n} from './trigger-options';\nimport { addConstructMetadata } from 'aws-cdk-lib/core/lib/metadata-resource';\nimport { propertyInjectable } from 'aws-cdk-lib/core/lib/prop-injectable';\n\n/**\n * The base interface for Glue Workflow\n *\n * @see {@link Workflow}\n * @see https://docs.aws.amazon.com/glue/latest/dg/workflows_overview.html\n */\nexport interface IWorkflow extends cdk.IResource {\n  /**\n   * The name of the workflow\n   * @attribute\n   */\n  readonly workflowName: string;\n\n  /**\n   * The ARN of the workflow\n   * @attribute\n   */\n  readonly workflowArn: string;\n\n  /**\n   * Add an on-demand trigger to the workflow\n   */\n  addOnDemandTrigger(id: string, options: OnDemandTriggerOptions): CfnTrigger;\n\n  /**\n   * Add an daily-scheduled trigger to the workflow\n   */\n  addDailyScheduledTrigger(id: string, options: DailyScheduleTriggerOptions): CfnTrigger;\n\n  /**\n   * Add an weekly-scheduled trigger to the workflow\n   */\n  addWeeklyScheduledTrigger(id: string, options: WeeklyScheduleTriggerOptions): CfnTrigger;\n\n  /**\n   * Add an custom-scheduled trigger to the workflow\n   */\n  addCustomScheduledTrigger(id: string, options: CustomScheduledTriggerOptions): CfnTrigger;\n}\n\n/**\n * Properties for importing a Workflow using its attributes\n */\nexport interface WorkflowAttributes {\n  /**\n   * The name of the workflow to import\n   */\n  readonly workflowName: string;\n  /**\n   * The ARN of the workflow to import\n   *\n   * @default - derived from the workflow name\n   */\n  readonly workflowArn?: string;\n}\n\n/**\n * Properties for defining a Workflow\n */\nexport interface WorkflowProps {\n  /**\n   * Name of the workflow\n   *\n   * @default - a name will be generated\n   */\n  readonly workflowName?: string;\n\n  /**\n   * A description of the workflow\n   *\n   * @default - no description\n   */\n  readonly description?: string;\n\n  /**\n   * A map of properties to use when this workflow is executed\n   *\n   * @default - no default run properties\n   */\n  readonly defaultRunProperties?: { [key: string]: string };\n\n  /**\n   * The maximum number of concurrent runs allowed for the workflow\n   *\n   * @default - no limit\n   */\n  readonly maxConcurrentRuns?: number;\n}\n\n/**\n * Base abstract class for Workflow\n *\n * @see https://docs.aws.amazon.com/glue/latest/dg/about-triggers.html\n */\nexport abstract class WorkflowBase extends cdk.Resource implements IWorkflow {\n  /**\n   * Extract workflowName from arn\n   */\n  protected static extractNameFromArn(scope: constructs.Construct, workflowArn: string): string {\n    return cdk.Stack.of(scope).splitArn(\n      workflowArn,\n      cdk.ArnFormat.SLASH_RESOURCE_NAME).resourceName!;\n  }\n\n  public abstract readonly workflowName: string;\n  public abstract readonly workflowArn: string;\n\n  /**\n   * Add an on-demand trigger to the workflow.\n   *\n   * @param id The id of the trigger.\n   * @param options Additional options for the trigger.\n   * @throws If both job and crawler are provided, or if neither job nor crawler is provided.\n   * @returns The created CfnTrigger resource.\n   */\n  public addOnDemandTrigger(id: string, options: OnDemandTriggerOptions): CfnTrigger {\n    const trigger = new CfnTrigger(this, id, {\n      ...options,\n      workflowName: this.workflowName,\n      type: 'ON_DEMAND',\n      actions: options.actions?.map(this.renderAction.bind(this)),\n      description: options.description || undefined,\n    });\n\n    return trigger;\n  }\n\n  /**\n   * Add a daily-scheduled trigger to the workflow.\n   *\n   * @param id The id of the trigger.\n   * @param options Additional options for the trigger.\n   * @throws If both job and crawler are provided, or if neither job nor crawler is provided.\n   * @returns The created CfnTrigger resource.\n   */\n  public addDailyScheduledTrigger(id: string, options: DailyScheduleTriggerOptions): CfnTrigger {\n    const dailySchedule = TriggerSchedule.cron({\n      minute: '0',\n      hour: '0',\n    });\n\n    const trigger = new CfnTrigger(this, id, {\n      ...options,\n      workflowName: this.workflowName,\n      type: 'SCHEDULED',\n      actions: options.actions?.map(this.renderAction.bind(this)),\n      schedule: dailySchedule.expressionString,\n      startOnCreation: options.startOnCreation ?? false,\n    });\n\n    return trigger;\n  }\n\n  /**\n   * Add a weekly-scheduled trigger to the workflow.\n   *\n   * @param id The id of the trigger.\n   * @param options Additional options for the trigger.\n   * @throws If both job and crawler are provided, or if neither job nor crawler is provided.\n   * @returns The created CfnTrigger resource.\n   */\n  public addWeeklyScheduledTrigger(id: string, options: WeeklyScheduleTriggerOptions): CfnTrigger {\n    const weeklySchedule = TriggerSchedule.cron({\n      minute: '0',\n      hour: '0',\n      weekDay: 'SUN',\n    });\n\n    const trigger = new CfnTrigger(this, id, {\n      ...options,\n      workflowName: this.workflowName,\n      type: 'SCHEDULED',\n      actions: options.actions?.map(this.renderAction.bind(this)),\n      schedule: weeklySchedule.expressionString,\n      startOnCreation: options.startOnCreation ?? false,\n    });\n\n    return trigger;\n  }\n\n  /**\n   * Add a custom-scheduled trigger to the workflow.\n   *\n   * @param id The id of the trigger.\n   * @param options Additional options for the trigger.\n   * @throws If both job and crawler are provided, or if neither job nor crawler is provided.\n   * @returns The created CfnTrigger resource.\n   */\n  public addCustomScheduledTrigger(id: string, options: CustomScheduledTriggerOptions): CfnTrigger {\n    const trigger = new CfnTrigger(this, id, {\n      ...options,\n      workflowName: this.workflowName,\n      type: 'SCHEDULED',\n      actions: options.actions?.map(this.renderAction.bind(this)),\n      schedule: options.schedule.expressionString,\n      startOnCreation: options.startOnCreation ?? false,\n    });\n\n    return trigger;\n  }\n\n  /**\n   * Add an Event Bridge based trigger to the workflow.\n   *\n   * @param id The id of the trigger.\n   * @param options Additional options for the trigger.\n   * @throws If both job and crawler are provided, or if neither job nor crawler is provided.\n   * @returns The created CfnTrigger resource.\n   */\n  public addNotifyEventTrigger(id: string, options: NotifyEventTriggerOptions): CfnTrigger {\n    const trigger = new CfnTrigger(this, id, {\n      ...options,\n      workflowName: this.workflowName,\n      type: 'EVENT',\n      actions: options.actions?.map(this.renderAction.bind(this)),\n      eventBatchingCondition: this.renderEventBatchingCondition(options),\n      description: options.description ?? undefined,\n    });\n\n    return trigger;\n  }\n\n  /**\n   * Add a Condition (Predicate) based trigger to the workflow.\n   *\n   * @param id The id of the trigger.\n   * @param options Additional options for the trigger.\n   * @throws If both job and crawler are provided, or if neither job nor crawler is provided for any condition.\n   * @throws If a job is provided without a job state, or if a crawler is provided without a crawler state for any condition.\n   * @returns The created CfnTrigger resource.\n   */\n  public addConditionalTrigger(id: string, options: ConditionalTriggerOptions): CfnTrigger {\n    const trigger = new CfnTrigger(this, id, {\n      ...options,\n      workflowName: this.workflowName,\n      type: 'CONDITIONAL',\n      actions: options.actions?.map(this.renderAction.bind(this)),\n      predicate: this.renderPredicate(options),\n      eventBatchingCondition: this.renderEventBatchingCondition(options),\n      description: options.description ?? undefined,\n    });\n\n    return trigger;\n  }\n\n  private renderAction(action: Action): CfnTrigger.ActionProperty {\n    // Validate that either job or crawler is provided, but not both\n    if (!action.job && !action.crawler) {\n      throw new cdk.ValidationError('You must provide either a job or a crawler for the action.', this);\n    } else if (action.job && action.crawler) {\n      throw new cdk.ValidationError('You cannot provide both a job and a crawler for the action.', this);\n    }\n\n    return {\n      jobName: action.job?.jobName,\n      arguments: action.arguments,\n      timeout: action.timeout?.toMinutes(),\n      securityConfiguration: action.securityConfiguration?.securityConfigurationName,\n      crawlerName: action.crawler?.name,\n    };\n  }\n\n  private renderPredicate(props: ConditionalTriggerOptions): CfnTrigger.PredicateProperty {\n    const conditions = props.predicate.conditions?.map(condition => {\n      // Validate that either job or crawler is provided, but not both\n      if (!condition.job && !condition.crawlerName) {\n        throw new cdk.ValidationError('You must provide either a job or a crawler for the condition.', this);\n      } else if (condition.job && condition.crawlerName) {\n        throw new cdk.ValidationError('You cannot provide both a job and a crawler for the condition.', this);\n      }\n\n      // Validate that if job is provided, job state is also provided\n      if (condition.job && !condition.state) {\n        throw new cdk.ValidationError('If you provide a job for the condition, you must also provide a job state.', this);\n      }\n\n      // Validate that if crawler is provided, crawler state is also provided\n      if (condition.crawlerName && !condition.crawlState) {\n        throw new cdk.ValidationError('If you provide a crawler for the condition, you must also provide a crawler state.', this);\n      }\n\n      return {\n        logicalOperator: condition.logicalOperator ?? ConditionLogicalOperator.EQUALS,\n        jobName: condition.job?.jobName ?? undefined,\n        state: condition.state ?? undefined,\n        crawlerName: condition.crawlerName ?? undefined,\n        crawlState: condition.crawlState ?? undefined,\n      };\n    });\n\n    return {\n      logical: props.predicate.conditions?.length === 1 ? undefined : props.predicate.logical ?? PredicateLogical.AND,\n      conditions: conditions,\n    };\n  }\n\n  private renderEventBatchingCondition(props: NotifyEventTriggerOptions): CfnTrigger.EventBatchingConditionProperty {\n    const defaultBatchSize = 1;\n    const defaultBatchWindow = cdk.Duration.seconds(900).toSeconds();\n\n    if (!props.eventBatchingCondition) {\n      return {\n        batchSize: defaultBatchSize,\n        batchWindow: defaultBatchWindow,\n      };\n    }\n\n    return {\n      batchSize: props.eventBatchingCondition.batchSize || defaultBatchSize,\n      batchWindow: props.eventBatchingCondition.batchWindow?.toSeconds() || defaultBatchWindow,\n    };\n  }\n\n  protected buildWorkflowArn(scope: constructs.Construct, workflowName: string): string {\n    return cdk.Stack.of(scope).formatArn({\n      service: 'glue',\n      resource: 'workflow',\n      resourceName: workflowName,\n    });\n  }\n}\n\n/**\n * This module defines a construct for creating and managing AWS Glue Workflows and Triggers.\n *\n * AWS Glue Workflows are orchestration services that allow you to create, manage, and monitor complex extract, transform, and load (ETL) activities involving multiple crawlers, jobs, and triggers. Workflows are designed to allow you to manage interdependent jobs and crawlers as a single unit, making it easier to orchestrate and monitor complex ETL pipelines.\n *\n * Triggers are used to initiate an AWS Glue Workflow. You can configure different types of triggers, such as on-demand, scheduled, event-based, or conditional triggers, to start your Workflow based on specific conditions or events.\n *\n * @see https://docs.aws.amazon.com/glue/latest/dg/workflows_overview.html\n * @see https://docs.aws.amazon.com/glue/latest/dg/about-triggers.html\n *\n * ## Usage Example\n *\n * ```ts\n * const app = new App();\n * const stack = new Stack(app, 'TestStack');\n *\n * // Create a Glue Job\n * declare const role: iam.IRole;\n * declare const script: glue.Code;\n * const job = new glue.PySparkStreamingJob(stack, 'ImportedJob', { role, script });\n *\n * // Create a Glue Workflow\n * const workflow = new glue.Workflow(stack, 'TestWorkflow');\n *\n * // Add an on-demand trigger to the Workflow\n * workflow.addOnDemandTrigger('OnDemandTrigger', {\n *   actions: [{ job: job }],\n * });\n * ```\n */\n@propertyInjectable\nexport class Workflow extends WorkflowBase {\n  /** Uniquely identifies this class. */\n  public static readonly PROPERTY_INJECTION_ID: string = '@aws-cdk.aws-glue-alpha.Workflow';\n\n  /**\n   * Import a workflow from its name\n   */\n  public static fromWorkflowName(scope: constructs.Construct, id: string, workflowName: string): IWorkflow {\n    return this.fromWorkflowAttributes(scope, id, {\n      workflowName,\n    });\n  }\n\n  /**\n   * Import an workflow from it's name\n   */\n  public static fromWorkflowArn(scope: constructs.Construct, id: string, workflowArn: string): IWorkflow {\n    return this.fromWorkflowAttributes(scope, id, {\n      workflowName: this.extractNameFromArn(scope, workflowArn),\n      workflowArn,\n    });\n  }\n\n  /**\n   * Import an existing workflow\n   */\n  public static fromWorkflowAttributes(scope: constructs.Construct, id: string, attrs: WorkflowAttributes): IWorkflow {\n    class Import extends WorkflowBase {\n      public readonly workflowName = attrs.workflowName;\n      public readonly workflowArn = this.buildWorkflowArn(scope, this.workflowName);\n    }\n\n    return new Import(scope, id);\n  }\n\n  public readonly workflowName: string;\n  public readonly workflowArn: string;\n\n  constructor(scope: constructs.Construct, id: string, props?: WorkflowProps) {\n    super(scope, id, {\n      physicalName: props?.workflowName,\n    });\n    // Enhanced CDK Analytics Telemetry\n    addConstructMetadata(this, props);\n\n    const resource = new CfnWorkflow(this, 'Resource', {\n      name: this.physicalName,\n      description: props?.description,\n      defaultRunProperties: props?.defaultRunProperties,\n      maxConcurrentRuns: props?.maxConcurrentRuns,\n    });\n\n    this.workflowName = this.getResourceNameAttribute(resource.ref);\n    this.workflowArn = this.buildWorkflowArn(this, this.workflowName);\n  }\n}\n"]}