You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
158 lines
6.1 KiB
Python
158 lines
6.1 KiB
Python
"""
|
|
Celery tasks for asynchronous workflow execution storage operations.
|
|
|
|
These tasks provide asynchronous storage capabilities for workflow execution data,
|
|
improving performance by offloading storage operations to background workers.
|
|
"""
|
|
|
|
import json
|
|
import logging
|
|
|
|
from celery import shared_task
|
|
from sqlalchemy import select
|
|
from sqlalchemy.orm import sessionmaker
|
|
|
|
from core.workflow.entities.workflow_execution import WorkflowExecution, WorkflowExecutionStatus, WorkflowType
|
|
from core.workflow.workflow_type_encoder import WorkflowRuntimeTypeConverter
|
|
from extensions.ext_database import db
|
|
from models import CreatorUserRole, WorkflowRun
|
|
from models.enums import WorkflowRunTriggeredFrom
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
@shared_task(queue="workflow_storage", bind=True, max_retries=3, default_retry_delay=60)
|
|
def save_workflow_execution_task(
|
|
self,
|
|
execution_data: dict,
|
|
tenant_id: str,
|
|
app_id: str,
|
|
triggered_from: str,
|
|
creator_user_id: str,
|
|
creator_user_role: str,
|
|
) -> bool:
|
|
"""
|
|
Asynchronously save or update a workflow execution to the database.
|
|
|
|
Args:
|
|
execution_data: Serialized WorkflowExecution data
|
|
tenant_id: Tenant ID for multi-tenancy
|
|
app_id: Application ID
|
|
triggered_from: Source of the execution trigger
|
|
creator_user_id: ID of the user who created the execution
|
|
creator_user_role: Role of the user who created the execution
|
|
|
|
Returns:
|
|
True if successful, False otherwise
|
|
"""
|
|
try:
|
|
# Create a new session for this task
|
|
session_factory = sessionmaker(bind=db.engine, expire_on_commit=False)
|
|
|
|
with session_factory() as session:
|
|
# Deserialize execution data
|
|
execution = WorkflowExecution.model_validate(execution_data)
|
|
|
|
# Check if workflow run already exists
|
|
existing_run = session.scalar(select(WorkflowRun).where(WorkflowRun.id == execution.id_))
|
|
|
|
if existing_run:
|
|
# Update existing workflow run
|
|
_update_workflow_run_from_execution(existing_run, execution)
|
|
logger.debug(f"Updated existing workflow run: {execution.id_}")
|
|
else:
|
|
# Create new workflow run
|
|
workflow_run = _create_workflow_run_from_execution(
|
|
execution=execution,
|
|
tenant_id=tenant_id,
|
|
app_id=app_id,
|
|
triggered_from=WorkflowRunTriggeredFrom(triggered_from),
|
|
creator_user_id=creator_user_id,
|
|
creator_user_role=CreatorUserRole(creator_user_role),
|
|
)
|
|
session.add(workflow_run)
|
|
logger.debug(f"Created new workflow run: {execution.id_}")
|
|
|
|
session.commit()
|
|
return True
|
|
|
|
except Exception as e:
|
|
logger.exception(f"Failed to save workflow execution {execution_data.get('id_', 'unknown')}")
|
|
# Retry the task with exponential backoff
|
|
raise self.retry(exc=e, countdown=60 * (2**self.request.retries))
|
|
|
|
|
|
def _create_workflow_run_from_execution(
|
|
execution: WorkflowExecution,
|
|
tenant_id: str,
|
|
app_id: str,
|
|
triggered_from: WorkflowRunTriggeredFrom,
|
|
creator_user_id: str,
|
|
creator_user_role: CreatorUserRole,
|
|
) -> WorkflowRun:
|
|
"""
|
|
Create a WorkflowRun database model from a WorkflowExecution domain entity.
|
|
"""
|
|
workflow_run = WorkflowRun()
|
|
workflow_run.id = execution.id_
|
|
workflow_run.tenant_id = tenant_id
|
|
workflow_run.app_id = app_id
|
|
workflow_run.workflow_id = execution.workflow_id
|
|
workflow_run.type = execution.workflow_type.value
|
|
workflow_run.triggered_from = triggered_from.value
|
|
workflow_run.version = execution.workflow_version
|
|
json_converter = WorkflowRuntimeTypeConverter()
|
|
workflow_run.graph = json.dumps(json_converter.to_json_encodable(execution.graph))
|
|
workflow_run.inputs = json.dumps(json_converter.to_json_encodable(execution.inputs))
|
|
workflow_run.status = execution.status.value
|
|
workflow_run.outputs = (
|
|
json.dumps(json_converter.to_json_encodable(execution.outputs)) if execution.outputs else "{}"
|
|
)
|
|
workflow_run.error = execution.error_message
|
|
workflow_run.elapsed_time = execution.elapsed_time
|
|
workflow_run.total_tokens = execution.total_tokens
|
|
workflow_run.total_steps = execution.total_steps
|
|
workflow_run.created_by_role = creator_user_role.value
|
|
workflow_run.created_by = creator_user_id
|
|
workflow_run.created_at = execution.started_at
|
|
workflow_run.finished_at = execution.finished_at
|
|
|
|
return workflow_run
|
|
|
|
|
|
def _update_workflow_run_from_execution(workflow_run: WorkflowRun, execution: WorkflowExecution) -> None:
|
|
"""
|
|
Update a WorkflowRun database model from a WorkflowExecution domain entity.
|
|
"""
|
|
json_converter = WorkflowRuntimeTypeConverter()
|
|
workflow_run.status = execution.status.value
|
|
workflow_run.outputs = (
|
|
json.dumps(json_converter.to_json_encodable(execution.outputs)) if execution.outputs else "{}"
|
|
)
|
|
workflow_run.error = execution.error_message
|
|
workflow_run.elapsed_time = execution.elapsed_time
|
|
workflow_run.total_tokens = execution.total_tokens
|
|
workflow_run.total_steps = execution.total_steps
|
|
workflow_run.finished_at = execution.finished_at
|
|
|
|
|
|
def _create_execution_from_workflow_run(workflow_run: WorkflowRun) -> WorkflowExecution:
|
|
"""
|
|
Create a WorkflowExecution domain entity from a WorkflowRun database model.
|
|
"""
|
|
return WorkflowExecution(
|
|
id_=workflow_run.id,
|
|
workflow_id=workflow_run.workflow_id,
|
|
workflow_type=WorkflowType(workflow_run.type),
|
|
workflow_version=workflow_run.version,
|
|
graph=json.loads(workflow_run.graph or "{}"),
|
|
inputs=json.loads(workflow_run.inputs or "{}"),
|
|
outputs=json.loads(workflow_run.outputs or "{}"),
|
|
status=WorkflowExecutionStatus(workflow_run.status),
|
|
error_message=workflow_run.error or "",
|
|
total_tokens=workflow_run.total_tokens,
|
|
total_steps=workflow_run.total_steps,
|
|
started_at=workflow_run.created_at,
|
|
finished_at=workflow_run.finished_at,
|
|
)
|