mirror of
https://github.com/langgenius/dify.git
synced 2026-01-14 06:07:33 +08:00
Signed-off-by: -LAN- <laipz8200@outlook.com> Signed-off-by: kenwoodjw <blackxin55+@gmail.com> Signed-off-by: Yongtao Huang <yongtaoh2022@gmail.com> Signed-off-by: yihong0618 <zouzou0208@gmail.com> Signed-off-by: zhanluxianshen <zhanluxianshen@163.com> Co-authored-by: -LAN- <laipz8200@outlook.com> Co-authored-by: GuanMu <ballmanjq@gmail.com> Co-authored-by: Davide Delbianco <davide.delbianco@outlook.com> Co-authored-by: NeatGuyCoding <15627489+NeatGuyCoding@users.noreply.github.com> Co-authored-by: kenwoodjw <blackxin55+@gmail.com> Co-authored-by: Yongtao Huang <yongtaoh2022@gmail.com> Co-authored-by: Yongtao Huang <99629139+hyongtao-db@users.noreply.github.com> Co-authored-by: Qiang Lee <18018968632@163.com> Co-authored-by: 李强04 <liqiang04@gaotu.cn> Co-authored-by: autofix-ci[bot] <114827586+autofix-ci[bot]@users.noreply.github.com> Co-authored-by: Asuka Minato <i@asukaminato.eu.org> Co-authored-by: Matri Qi <matrixdom@126.com> Co-authored-by: huayaoyue6 <huayaoyue@163.com> Co-authored-by: Bowen Liang <liangbowen@gf.com.cn> Co-authored-by: znn <jubinkumarsoni@gmail.com> Co-authored-by: crazywoola <427733928@qq.com> Co-authored-by: crazywoola <100913391+crazywoola@users.noreply.github.com> Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> Co-authored-by: yihong <zouzou0208@gmail.com> Co-authored-by: Muke Wang <shaodwaaron@gmail.com> Co-authored-by: wangmuke <wangmuke@kingsware.cn> Co-authored-by: Wu Tianwei <30284043+WTW0313@users.noreply.github.com> Co-authored-by: quicksand <quicksandzn@gmail.com> Co-authored-by: 非法操作 <hjlarry@163.com> Co-authored-by: zxhlyh <jasonapring2015@outlook.com> Co-authored-by: Eric Guo <eric.guocz@gmail.com> Co-authored-by: Zhedong Cen <cenzhedong2@126.com> Co-authored-by: jiangbo721 <jiangbo721@163.com> Co-authored-by: github-actions[bot] <41898282+github-actions[bot]@users.noreply.github.com> Co-authored-by: hjlarry <25834719+hjlarry@users.noreply.github.com> Co-authored-by: lxsummer <35754229+lxjustdoit@users.noreply.github.com> Co-authored-by: 湛露先生 <zhanluxianshen@163.com> Co-authored-by: Guangdong Liu <liugddx@gmail.com> Co-authored-by: QuantumGhost <obelisk.reg+git@gmail.com> Co-authored-by: Claude <noreply@anthropic.com> Co-authored-by: Yessenia-d <yessenia.contact@gmail.com> Co-authored-by: huangzhuo1949 <167434202+huangzhuo1949@users.noreply.github.com> Co-authored-by: huangzhuo <huangzhuo1@xiaomi.com> Co-authored-by: 17hz <0x149527@gmail.com> Co-authored-by: Amy <1530140574@qq.com> Co-authored-by: Joel <iamjoel007@gmail.com> Co-authored-by: Nite Knite <nkCoding@gmail.com> Co-authored-by: Yeuoly <45712896+Yeuoly@users.noreply.github.com> Co-authored-by: Petrus Han <petrus.hanks@gmail.com> Co-authored-by: iamjoel <2120155+iamjoel@users.noreply.github.com> Co-authored-by: Kalo Chin <frog.beepers.0n@icloud.com> Co-authored-by: Ujjwal Maurya <ujjwalsbx@gmail.com> Co-authored-by: Maries <xh001x@hotmail.com>
137 lines
5.2 KiB
Python
137 lines
5.2 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
|
|
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("Updated existing workflow run: %s", 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("Created new workflow run: %s", execution.id_)
|
|
|
|
session.commit()
|
|
return True
|
|
|
|
except Exception as e:
|
|
logger.exception("Failed to save workflow execution %s", 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
|