| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | import json | 
					
						
							|  |  |  | import logging | 
					
						
							| 
									
										
										
										
											2025-06-24 09:05:29 +08:00
										 |  |  | from collections.abc import Sequence | 
					
						
							| 
									
										
										
										
											2025-03-10 13:34:31 +08:00
										 |  |  | from typing import cast | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | from flask import abort, request | 
					
						
							| 
									
										
										
										
											2025-05-06 11:58:49 +08:00
										 |  |  | from flask_restful import Resource, inputs, marshal_with, reqparse | 
					
						
							| 
									
										
										
										
											2025-03-10 13:34:31 +08:00
										 |  |  | from sqlalchemy.orm import Session | 
					
						
							| 
									
										
										
										
											2024-06-17 03:06:32 -05:00
										 |  |  | from werkzeug.exceptions import Forbidden, InternalServerError, NotFound | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | import services | 
					
						
							| 
									
										
										
										
											2024-12-20 14:52:20 +08:00
										 |  |  | from configs import dify_config | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | from controllers.console import api | 
					
						
							| 
									
										
										
										
											2025-03-19 17:24:02 +08:00
										 |  |  | from controllers.console.app.error import ( | 
					
						
							|  |  |  |     ConversationCompletedError, | 
					
						
							|  |  |  |     DraftWorkflowNotExist, | 
					
						
							|  |  |  |     DraftWorkflowNotSync, | 
					
						
							|  |  |  | ) | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | from controllers.console.app.wraps import get_app_model | 
					
						
							| 
									
										
										
										
											2024-11-01 15:51:22 +08:00
										 |  |  | from controllers.console.wraps import account_initialization_required, setup_required | 
					
						
							| 
									
										
										
										
											2025-03-19 17:24:02 +08:00
										 |  |  | from controllers.web.error import InvokeRateLimitError as InvokeRateLimitHttpError | 
					
						
							| 
									
										
										
										
											2025-06-24 09:05:29 +08:00
										 |  |  | from core.app.app_config.features.file_upload.manager import FileUploadConfigManager | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | from core.app.apps.base_app_queue_manager import AppQueueManager | 
					
						
							|  |  |  | from core.app.entities.app_invoke_entities import InvokeFrom | 
					
						
							| 
									
										
										
										
											2025-06-24 09:05:29 +08:00
										 |  |  | from core.file.models import File | 
					
						
							| 
									
										
										
										
											2025-03-10 13:34:31 +08:00
										 |  |  | from extensions.ext_database import db | 
					
						
							| 
									
										
										
										
											2025-06-24 09:05:29 +08:00
										 |  |  | from factories import file_factory, variable_factory | 
					
						
							| 
									
										
										
										
											2024-12-27 21:05:06 +08:00
										 |  |  | from fields.workflow_fields import workflow_fields, workflow_pagination_fields | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | from fields.workflow_run_fields import workflow_run_node_execution_fields | 
					
						
							|  |  |  | from libs import helper | 
					
						
							|  |  |  | from libs.helper import TimestampField, uuid_value | 
					
						
							|  |  |  | from libs.login import current_user, login_required | 
					
						
							| 
									
										
										
										
											2024-10-21 10:43:49 +08:00
										 |  |  | from models import App | 
					
						
							| 
									
										
										
										
											2025-02-17 17:05:13 +08:00
										 |  |  | from models.account import Account | 
					
						
							| 
									
										
										
										
											2024-10-21 10:43:49 +08:00
										 |  |  | from models.model import AppMode | 
					
						
							| 
									
										
										
										
											2025-06-24 09:05:29 +08:00
										 |  |  | from models.workflow import Workflow | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | from services.app_generate_service import AppGenerateService | 
					
						
							| 
									
										
										
										
											2024-05-10 14:48:29 +08:00
										 |  |  | from services.errors.app import WorkflowHashNotEqualError | 
					
						
							| 
									
										
										
										
											2025-03-19 17:24:02 +08:00
										 |  |  | from services.errors.llm import InvokeRateLimitError | 
					
						
							| 
									
										
										
										
											2025-03-10 13:34:31 +08:00
										 |  |  | from services.workflow_service import DraftWorkflowDeletionError, WorkflowInUseError, WorkflowService | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | logger = logging.getLogger(__name__) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-24 09:05:29 +08:00
										 |  |  | # TODO(QuantumGhost): Refactor existing node run API to handle file parameter parsing | 
					
						
							|  |  |  | # at the controller level rather than in the workflow logic. This would improve separation | 
					
						
							|  |  |  | # of concerns and make the code more maintainable. | 
					
						
							|  |  |  | def _parse_file(workflow: Workflow, files: list[dict] | None = None) -> Sequence[File]: | 
					
						
							|  |  |  |     files = files or [] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     file_extra_config = FileUploadConfigManager.convert(workflow.features_dict, is_vision=False) | 
					
						
							|  |  |  |     file_objs: Sequence[File] = [] | 
					
						
							|  |  |  |     if file_extra_config is None: | 
					
						
							|  |  |  |         return file_objs | 
					
						
							|  |  |  |     file_objs = file_factory.build_from_mappings( | 
					
						
							|  |  |  |         mappings=files, | 
					
						
							|  |  |  |         tenant_id=workflow.tenant_id, | 
					
						
							|  |  |  |         config=file_extra_config, | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     return file_objs | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | class DraftWorkflowApi(Resource): | 
					
						
							|  |  |  |     @setup_required | 
					
						
							|  |  |  |     @login_required | 
					
						
							|  |  |  |     @account_initialization_required | 
					
						
							|  |  |  |     @get_app_model(mode=[AppMode.ADVANCED_CHAT, AppMode.WORKFLOW]) | 
					
						
							|  |  |  |     @marshal_with(workflow_fields) | 
					
						
							|  |  |  |     def get(self, app_model: App): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Get draft workflow | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2024-06-17 03:06:32 -05:00
										 |  |  |         # The role of the current user in the ta table must be admin, owner, or editor | 
					
						
							|  |  |  |         if not current_user.is_editor: | 
					
						
							|  |  |  |             raise Forbidden() | 
					
						
							| 
									
										
										
										
											2024-07-22 15:29:39 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |         # fetch draft workflow by app_model | 
					
						
							|  |  |  |         workflow_service = WorkflowService() | 
					
						
							|  |  |  |         workflow = workflow_service.get_draft_workflow(app_model=app_model) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if not workflow: | 
					
						
							|  |  |  |             raise DraftWorkflowNotExist() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # return workflow, if not found, return None (initiate graph by frontend) | 
					
						
							|  |  |  |         return workflow | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @setup_required | 
					
						
							|  |  |  |     @login_required | 
					
						
							|  |  |  |     @account_initialization_required | 
					
						
							|  |  |  |     @get_app_model(mode=[AppMode.ADVANCED_CHAT, AppMode.WORKFLOW]) | 
					
						
							|  |  |  |     def post(self, app_model: App): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Sync draft workflow | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2024-06-17 03:06:32 -05:00
										 |  |  |         # The role of the current user in the ta table must be admin, owner, or editor | 
					
						
							|  |  |  |         if not current_user.is_editor: | 
					
						
							|  |  |  |             raise Forbidden() | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  |         content_type = request.headers.get("Content-Type", "") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if "application/json" in content_type: | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |             parser = reqparse.RequestParser() | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  |             parser.add_argument("graph", type=dict, required=True, nullable=False, location="json") | 
					
						
							|  |  |  |             parser.add_argument("features", type=dict, required=True, nullable=False, location="json") | 
					
						
							|  |  |  |             parser.add_argument("hash", type=str, required=False, location="json") | 
					
						
							| 
									
										
										
										
											2025-05-17 13:45:00 +08:00
										 |  |  |             parser.add_argument("environment_variables", type=list, required=True, location="json") | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  |             parser.add_argument("conversation_variables", type=list, required=False, location="json") | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |             args = parser.parse_args() | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  |         elif "text/plain" in content_type: | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |             try: | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  |                 data = json.loads(request.data.decode("utf-8")) | 
					
						
							|  |  |  |                 if "graph" not in data or "features" not in data: | 
					
						
							|  |  |  |                     raise ValueError("graph or features not found in data") | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  |                 if not isinstance(data.get("graph"), dict) or not isinstance(data.get("features"), dict): | 
					
						
							|  |  |  |                     raise ValueError("graph or features is not a dict") | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 args = { | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  |                     "graph": data.get("graph"), | 
					
						
							|  |  |  |                     "features": data.get("features"), | 
					
						
							|  |  |  |                     "hash": data.get("hash"), | 
					
						
							|  |  |  |                     "environment_variables": data.get("environment_variables"), | 
					
						
							|  |  |  |                     "conversation_variables": data.get("conversation_variables"), | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |                 } | 
					
						
							|  |  |  |             except json.JSONDecodeError: | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  |                 return {"message": "Invalid JSON data"}, 400 | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |         else: | 
					
						
							|  |  |  |             abort(415) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-17 17:05:13 +08:00
										 |  |  |         if not isinstance(current_user, Account): | 
					
						
							|  |  |  |             raise Forbidden() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |         workflow_service = WorkflowService() | 
					
						
							| 
									
										
										
										
											2024-05-10 14:48:29 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         try: | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  |             environment_variables_list = args.get("environment_variables") or [] | 
					
						
							| 
									
										
										
										
											2024-10-21 10:43:49 +08:00
										 |  |  |             environment_variables = [ | 
					
						
							| 
									
										
										
										
											2024-12-03 13:56:40 +08:00
										 |  |  |                 variable_factory.build_environment_variable_from_mapping(obj) for obj in environment_variables_list | 
					
						
							| 
									
										
										
										
											2024-10-21 10:43:49 +08:00
										 |  |  |             ] | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  |             conversation_variables_list = args.get("conversation_variables") or [] | 
					
						
							| 
									
										
										
										
											2024-10-21 10:43:49 +08:00
										 |  |  |             conversation_variables = [ | 
					
						
							| 
									
										
										
										
											2024-12-03 13:56:40 +08:00
										 |  |  |                 variable_factory.build_conversation_variable_from_mapping(obj) for obj in conversation_variables_list | 
					
						
							| 
									
										
										
										
											2024-10-21 10:43:49 +08:00
										 |  |  |             ] | 
					
						
							| 
									
										
										
										
											2024-05-10 14:48:29 +08:00
										 |  |  |             workflow = workflow_service.sync_draft_workflow( | 
					
						
							|  |  |  |                 app_model=app_model, | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  |                 graph=args["graph"], | 
					
						
							|  |  |  |                 features=args["features"], | 
					
						
							|  |  |  |                 unique_hash=args.get("hash"), | 
					
						
							| 
									
										
										
										
											2024-07-22 15:29:39 +08:00
										 |  |  |                 account=current_user, | 
					
						
							|  |  |  |                 environment_variables=environment_variables, | 
					
						
							| 
									
										
										
										
											2024-08-13 14:44:10 +08:00
										 |  |  |                 conversation_variables=conversation_variables, | 
					
						
							| 
									
										
										
										
											2024-05-10 14:48:29 +08:00
										 |  |  |             ) | 
					
						
							|  |  |  |         except WorkflowHashNotEqualError: | 
					
						
							|  |  |  |             raise DraftWorkflowNotSync() | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return { | 
					
						
							|  |  |  |             "result": "success", | 
					
						
							| 
									
										
										
										
											2024-05-10 14:48:29 +08:00
										 |  |  |             "hash": workflow.unique_hash, | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  |             "updated_at": TimestampField().format(workflow.updated_at or workflow.created_at), | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class AdvancedChatDraftWorkflowRunApi(Resource): | 
					
						
							|  |  |  |     @setup_required | 
					
						
							|  |  |  |     @login_required | 
					
						
							|  |  |  |     @account_initialization_required | 
					
						
							|  |  |  |     @get_app_model(mode=[AppMode.ADVANCED_CHAT]) | 
					
						
							|  |  |  |     def post(self, app_model: App): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Run draft workflow | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2024-06-17 03:06:32 -05:00
										 |  |  |         # The role of the current user in the ta table must be admin, owner, or editor | 
					
						
							|  |  |  |         if not current_user.is_editor: | 
					
						
							|  |  |  |             raise Forbidden() | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-17 17:05:13 +08:00
										 |  |  |         if not isinstance(current_user, Account): | 
					
						
							|  |  |  |             raise Forbidden() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |         parser = reqparse.RequestParser() | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  |         parser.add_argument("inputs", type=dict, location="json") | 
					
						
							|  |  |  |         parser.add_argument("query", type=str, required=True, location="json", default="") | 
					
						
							|  |  |  |         parser.add_argument("files", type=list, location="json") | 
					
						
							|  |  |  |         parser.add_argument("conversation_id", type=uuid_value, location="json") | 
					
						
							| 
									
										
										
										
											2024-09-22 03:15:11 +08:00
										 |  |  |         parser.add_argument("parent_message_id", type=uuid_value, required=False, location="json") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |         args = parser.parse_args() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             response = AppGenerateService.generate( | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  |                 app_model=app_model, user=current_user, args=args, invoke_from=InvokeFrom.DEBUGGER, streaming=True | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |             ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return helper.compact_generate_response(response) | 
					
						
							|  |  |  |         except services.errors.conversation.ConversationNotExistsError: | 
					
						
							|  |  |  |             raise NotFound("Conversation Not Exists.") | 
					
						
							|  |  |  |         except services.errors.conversation.ConversationCompletedError: | 
					
						
							|  |  |  |             raise ConversationCompletedError() | 
					
						
							| 
									
										
										
										
											2025-03-19 17:24:02 +08:00
										 |  |  |         except InvokeRateLimitError as ex: | 
					
						
							|  |  |  |             raise InvokeRateLimitHttpError(ex.description) | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |         except ValueError as e: | 
					
						
							|  |  |  |             raise e | 
					
						
							| 
									
										
										
										
											2025-02-17 17:05:13 +08:00
										 |  |  |         except Exception: | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |             logging.exception("internal server error.") | 
					
						
							|  |  |  |             raise InternalServerError() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-27 22:01:11 +08:00
										 |  |  | class AdvancedChatDraftRunIterationNodeApi(Resource): | 
					
						
							|  |  |  |     @setup_required | 
					
						
							|  |  |  |     @login_required | 
					
						
							|  |  |  |     @account_initialization_required | 
					
						
							|  |  |  |     @get_app_model(mode=[AppMode.ADVANCED_CHAT]) | 
					
						
							|  |  |  |     def post(self, app_model: App, node_id: str): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Run draft workflow iteration node | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2024-06-17 03:06:32 -05:00
										 |  |  |         # The role of the current user in the ta table must be admin, owner, or editor | 
					
						
							|  |  |  |         if not current_user.is_editor: | 
					
						
							|  |  |  |             raise Forbidden() | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-17 17:05:13 +08:00
										 |  |  |         if not isinstance(current_user, Account): | 
					
						
							|  |  |  |             raise Forbidden() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-27 22:01:11 +08:00
										 |  |  |         parser = reqparse.RequestParser() | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  |         parser.add_argument("inputs", type=dict, location="json") | 
					
						
							| 
									
										
										
										
											2024-05-27 22:01:11 +08:00
										 |  |  |         args = parser.parse_args() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             response = AppGenerateService.generate_single_iteration( | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  |                 app_model=app_model, user=current_user, node_id=node_id, args=args, streaming=True | 
					
						
							| 
									
										
										
										
											2024-05-27 22:01:11 +08:00
										 |  |  |             ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return helper.compact_generate_response(response) | 
					
						
							|  |  |  |         except services.errors.conversation.ConversationNotExistsError: | 
					
						
							|  |  |  |             raise NotFound("Conversation Not Exists.") | 
					
						
							|  |  |  |         except services.errors.conversation.ConversationCompletedError: | 
					
						
							|  |  |  |             raise ConversationCompletedError() | 
					
						
							|  |  |  |         except ValueError as e: | 
					
						
							|  |  |  |             raise e | 
					
						
							| 
									
										
										
										
											2025-02-17 17:05:13 +08:00
										 |  |  |         except Exception: | 
					
						
							| 
									
										
										
										
											2024-05-27 22:01:11 +08:00
										 |  |  |             logging.exception("internal server error.") | 
					
						
							|  |  |  |             raise InternalServerError() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-27 22:01:11 +08:00
										 |  |  | class WorkflowDraftRunIterationNodeApi(Resource): | 
					
						
							|  |  |  |     @setup_required | 
					
						
							|  |  |  |     @login_required | 
					
						
							|  |  |  |     @account_initialization_required | 
					
						
							|  |  |  |     @get_app_model(mode=[AppMode.WORKFLOW]) | 
					
						
							|  |  |  |     def post(self, app_model: App, node_id: str): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Run draft workflow iteration node | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2024-06-17 03:06:32 -05:00
										 |  |  |         # The role of the current user in the ta table must be admin, owner, or editor | 
					
						
							|  |  |  |         if not current_user.is_editor: | 
					
						
							|  |  |  |             raise Forbidden() | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-17 17:05:13 +08:00
										 |  |  |         if not isinstance(current_user, Account): | 
					
						
							|  |  |  |             raise Forbidden() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-27 22:01:11 +08:00
										 |  |  |         parser = reqparse.RequestParser() | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  |         parser.add_argument("inputs", type=dict, location="json") | 
					
						
							| 
									
										
										
										
											2024-05-27 22:01:11 +08:00
										 |  |  |         args = parser.parse_args() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             response = AppGenerateService.generate_single_iteration( | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  |                 app_model=app_model, user=current_user, node_id=node_id, args=args, streaming=True | 
					
						
							| 
									
										
										
										
											2024-05-27 22:01:11 +08:00
										 |  |  |             ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return helper.compact_generate_response(response) | 
					
						
							|  |  |  |         except services.errors.conversation.ConversationNotExistsError: | 
					
						
							|  |  |  |             raise NotFound("Conversation Not Exists.") | 
					
						
							|  |  |  |         except services.errors.conversation.ConversationCompletedError: | 
					
						
							|  |  |  |             raise ConversationCompletedError() | 
					
						
							|  |  |  |         except ValueError as e: | 
					
						
							|  |  |  |             raise e | 
					
						
							| 
									
										
										
										
											2025-02-17 17:05:13 +08:00
										 |  |  |         except Exception: | 
					
						
							| 
									
										
										
										
											2024-05-27 22:01:11 +08:00
										 |  |  |             logging.exception("internal server error.") | 
					
						
							|  |  |  |             raise InternalServerError() | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-05 17:41:15 +08:00
										 |  |  | class AdvancedChatDraftRunLoopNodeApi(Resource): | 
					
						
							|  |  |  |     @setup_required | 
					
						
							|  |  |  |     @login_required | 
					
						
							|  |  |  |     @account_initialization_required | 
					
						
							|  |  |  |     @get_app_model(mode=[AppMode.ADVANCED_CHAT]) | 
					
						
							|  |  |  |     def post(self, app_model: App, node_id: str): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Run draft workflow loop node | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         # The role of the current user in the ta table must be admin, owner, or editor | 
					
						
							|  |  |  |         if not current_user.is_editor: | 
					
						
							|  |  |  |             raise Forbidden() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if not isinstance(current_user, Account): | 
					
						
							|  |  |  |             raise Forbidden() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         parser = reqparse.RequestParser() | 
					
						
							|  |  |  |         parser.add_argument("inputs", type=dict, location="json") | 
					
						
							|  |  |  |         args = parser.parse_args() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             response = AppGenerateService.generate_single_loop( | 
					
						
							|  |  |  |                 app_model=app_model, user=current_user, node_id=node_id, args=args, streaming=True | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return helper.compact_generate_response(response) | 
					
						
							|  |  |  |         except services.errors.conversation.ConversationNotExistsError: | 
					
						
							|  |  |  |             raise NotFound("Conversation Not Exists.") | 
					
						
							|  |  |  |         except services.errors.conversation.ConversationCompletedError: | 
					
						
							|  |  |  |             raise ConversationCompletedError() | 
					
						
							|  |  |  |         except ValueError as e: | 
					
						
							|  |  |  |             raise e | 
					
						
							|  |  |  |         except Exception: | 
					
						
							|  |  |  |             logging.exception("internal server error.") | 
					
						
							|  |  |  |             raise InternalServerError() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class WorkflowDraftRunLoopNodeApi(Resource): | 
					
						
							|  |  |  |     @setup_required | 
					
						
							|  |  |  |     @login_required | 
					
						
							|  |  |  |     @account_initialization_required | 
					
						
							|  |  |  |     @get_app_model(mode=[AppMode.WORKFLOW]) | 
					
						
							|  |  |  |     def post(self, app_model: App, node_id: str): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Run draft workflow loop node | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         # The role of the current user in the ta table must be admin, owner, or editor | 
					
						
							|  |  |  |         if not current_user.is_editor: | 
					
						
							|  |  |  |             raise Forbidden() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if not isinstance(current_user, Account): | 
					
						
							|  |  |  |             raise Forbidden() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         parser = reqparse.RequestParser() | 
					
						
							|  |  |  |         parser.add_argument("inputs", type=dict, location="json") | 
					
						
							|  |  |  |         args = parser.parse_args() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             response = AppGenerateService.generate_single_loop( | 
					
						
							|  |  |  |                 app_model=app_model, user=current_user, node_id=node_id, args=args, streaming=True | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return helper.compact_generate_response(response) | 
					
						
							|  |  |  |         except services.errors.conversation.ConversationNotExistsError: | 
					
						
							|  |  |  |             raise NotFound("Conversation Not Exists.") | 
					
						
							|  |  |  |         except services.errors.conversation.ConversationCompletedError: | 
					
						
							|  |  |  |             raise ConversationCompletedError() | 
					
						
							|  |  |  |         except ValueError as e: | 
					
						
							|  |  |  |             raise e | 
					
						
							|  |  |  |         except Exception: | 
					
						
							|  |  |  |             logging.exception("internal server error.") | 
					
						
							|  |  |  |             raise InternalServerError() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | class DraftWorkflowRunApi(Resource): | 
					
						
							|  |  |  |     @setup_required | 
					
						
							|  |  |  |     @login_required | 
					
						
							|  |  |  |     @account_initialization_required | 
					
						
							|  |  |  |     @get_app_model(mode=[AppMode.WORKFLOW]) | 
					
						
							|  |  |  |     def post(self, app_model: App): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Run draft workflow | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2024-06-17 03:06:32 -05:00
										 |  |  |         # The role of the current user in the ta table must be admin, owner, or editor | 
					
						
							|  |  |  |         if not current_user.is_editor: | 
					
						
							|  |  |  |             raise Forbidden() | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-17 17:05:13 +08:00
										 |  |  |         if not isinstance(current_user, Account): | 
					
						
							|  |  |  |             raise Forbidden() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |         parser = reqparse.RequestParser() | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  |         parser.add_argument("inputs", type=dict, required=True, nullable=False, location="json") | 
					
						
							|  |  |  |         parser.add_argument("files", type=list, required=False, location="json") | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |         args = parser.parse_args() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-19 17:24:02 +08:00
										 |  |  |         try: | 
					
						
							|  |  |  |             response = AppGenerateService.generate( | 
					
						
							|  |  |  |                 app_model=app_model, | 
					
						
							|  |  |  |                 user=current_user, | 
					
						
							|  |  |  |                 args=args, | 
					
						
							|  |  |  |                 invoke_from=InvokeFrom.DEBUGGER, | 
					
						
							|  |  |  |                 streaming=True, | 
					
						
							|  |  |  |             ) | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-19 17:24:02 +08:00
										 |  |  |             return helper.compact_generate_response(response) | 
					
						
							|  |  |  |         except InvokeRateLimitError as ex: | 
					
						
							|  |  |  |             raise InvokeRateLimitHttpError(ex.description) | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class WorkflowTaskStopApi(Resource): | 
					
						
							|  |  |  |     @setup_required | 
					
						
							|  |  |  |     @login_required | 
					
						
							|  |  |  |     @account_initialization_required | 
					
						
							|  |  |  |     @get_app_model(mode=[AppMode.ADVANCED_CHAT, AppMode.WORKFLOW]) | 
					
						
							|  |  |  |     def post(self, app_model: App, task_id: str): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Stop workflow task | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2024-06-17 03:06:32 -05:00
										 |  |  |         # The role of the current user in the ta table must be admin, owner, or editor | 
					
						
							|  |  |  |         if not current_user.is_editor: | 
					
						
							|  |  |  |             raise Forbidden() | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |         AppQueueManager.set_stop_flag(task_id, InvokeFrom.DEBUGGER, current_user.id) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  |         return {"result": "success"} | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class DraftWorkflowNodeRunApi(Resource): | 
					
						
							|  |  |  |     @setup_required | 
					
						
							|  |  |  |     @login_required | 
					
						
							|  |  |  |     @account_initialization_required | 
					
						
							|  |  |  |     @get_app_model(mode=[AppMode.ADVANCED_CHAT, AppMode.WORKFLOW]) | 
					
						
							|  |  |  |     @marshal_with(workflow_run_node_execution_fields) | 
					
						
							|  |  |  |     def post(self, app_model: App, node_id: str): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Run draft workflow node | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2024-06-17 03:06:32 -05:00
										 |  |  |         # The role of the current user in the ta table must be admin, owner, or editor | 
					
						
							|  |  |  |         if not current_user.is_editor: | 
					
						
							|  |  |  |             raise Forbidden() | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-17 17:05:13 +08:00
										 |  |  |         if not isinstance(current_user, Account): | 
					
						
							|  |  |  |             raise Forbidden() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |         parser = reqparse.RequestParser() | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  |         parser.add_argument("inputs", type=dict, required=True, nullable=False, location="json") | 
					
						
							| 
									
										
										
										
											2025-06-24 09:05:29 +08:00
										 |  |  |         parser.add_argument("query", type=str, required=False, location="json", default="") | 
					
						
							|  |  |  |         parser.add_argument("files", type=list, location="json", default=[]) | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |         args = parser.parse_args() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-24 09:05:29 +08:00
										 |  |  |         user_inputs = args.get("inputs") | 
					
						
							|  |  |  |         if user_inputs is None: | 
					
						
							| 
									
										
										
										
											2025-02-17 17:05:13 +08:00
										 |  |  |             raise ValueError("missing inputs") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-24 09:05:29 +08:00
										 |  |  |         workflow_srv = WorkflowService() | 
					
						
							|  |  |  |         # fetch draft workflow by app_model | 
					
						
							|  |  |  |         draft_workflow = workflow_srv.get_draft_workflow(app_model=app_model) | 
					
						
							|  |  |  |         if not draft_workflow: | 
					
						
							|  |  |  |             raise ValueError("Workflow not initialized") | 
					
						
							|  |  |  |         files = _parse_file(draft_workflow, args.get("files")) | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |         workflow_service = WorkflowService() | 
					
						
							| 
									
										
										
										
											2025-06-24 09:05:29 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |         workflow_node_execution = workflow_service.run_draft_workflow_node( | 
					
						
							| 
									
										
										
										
											2025-06-24 09:05:29 +08:00
										 |  |  |             app_model=app_model, | 
					
						
							|  |  |  |             draft_workflow=draft_workflow, | 
					
						
							|  |  |  |             node_id=node_id, | 
					
						
							|  |  |  |             user_inputs=user_inputs, | 
					
						
							|  |  |  |             account=current_user, | 
					
						
							|  |  |  |             query=args.get("query", ""), | 
					
						
							|  |  |  |             files=files, | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |         ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return workflow_node_execution | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class PublishedWorkflowApi(Resource): | 
					
						
							|  |  |  |     @setup_required | 
					
						
							|  |  |  |     @login_required | 
					
						
							|  |  |  |     @account_initialization_required | 
					
						
							|  |  |  |     @get_app_model(mode=[AppMode.ADVANCED_CHAT, AppMode.WORKFLOW]) | 
					
						
							|  |  |  |     @marshal_with(workflow_fields) | 
					
						
							|  |  |  |     def get(self, app_model: App): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Get published workflow | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2024-06-17 03:06:32 -05:00
										 |  |  |         # The role of the current user in the ta table must be admin, owner, or editor | 
					
						
							|  |  |  |         if not current_user.is_editor: | 
					
						
							|  |  |  |             raise Forbidden() | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |         # fetch published workflow by app_model | 
					
						
							|  |  |  |         workflow_service = WorkflowService() | 
					
						
							|  |  |  |         workflow = workflow_service.get_published_workflow(app_model=app_model) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # return workflow, if not found, return None | 
					
						
							|  |  |  |         return workflow | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @setup_required | 
					
						
							|  |  |  |     @login_required | 
					
						
							|  |  |  |     @account_initialization_required | 
					
						
							|  |  |  |     @get_app_model(mode=[AppMode.ADVANCED_CHAT, AppMode.WORKFLOW]) | 
					
						
							|  |  |  |     def post(self, app_model: App): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Publish workflow | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2024-06-17 03:06:32 -05:00
										 |  |  |         # The role of the current user in the ta table must be admin, owner, or editor | 
					
						
							|  |  |  |         if not current_user.is_editor: | 
					
						
							|  |  |  |             raise Forbidden() | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-17 17:05:13 +08:00
										 |  |  |         if not isinstance(current_user, Account): | 
					
						
							|  |  |  |             raise Forbidden() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-10 13:34:31 +08:00
										 |  |  |         parser = reqparse.RequestParser() | 
					
						
							|  |  |  |         parser.add_argument("marked_name", type=str, required=False, default="", location="json") | 
					
						
							|  |  |  |         parser.add_argument("marked_comment", type=str, required=False, default="", location="json") | 
					
						
							|  |  |  |         args = parser.parse_args() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # Validate name and comment length | 
					
						
							|  |  |  |         if args.marked_name and len(args.marked_name) > 20: | 
					
						
							|  |  |  |             raise ValueError("Marked name cannot exceed 20 characters") | 
					
						
							|  |  |  |         if args.marked_comment and len(args.marked_comment) > 100: | 
					
						
							|  |  |  |             raise ValueError("Marked comment cannot exceed 100 characters") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |         workflow_service = WorkflowService() | 
					
						
							| 
									
										
										
										
											2025-03-10 13:34:31 +08:00
										 |  |  |         with Session(db.engine) as session: | 
					
						
							|  |  |  |             workflow = workflow_service.publish_workflow( | 
					
						
							|  |  |  |                 session=session, | 
					
						
							|  |  |  |                 app_model=app_model, | 
					
						
							|  |  |  |                 account=current_user, | 
					
						
							|  |  |  |                 marked_name=args.marked_name or "", | 
					
						
							|  |  |  |                 marked_comment=args.marked_comment or "", | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             app_model.workflow_id = workflow.id | 
					
						
							|  |  |  |             db.session.commit() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             workflow_created_at = TimestampField().format(workflow.created_at) | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-10 13:34:31 +08:00
										 |  |  |             session.commit() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return { | 
					
						
							|  |  |  |             "result": "success", | 
					
						
							|  |  |  |             "created_at": workflow_created_at, | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class DefaultBlockConfigsApi(Resource): | 
					
						
							|  |  |  |     @setup_required | 
					
						
							|  |  |  |     @login_required | 
					
						
							|  |  |  |     @account_initialization_required | 
					
						
							|  |  |  |     @get_app_model(mode=[AppMode.ADVANCED_CHAT, AppMode.WORKFLOW]) | 
					
						
							|  |  |  |     def get(self, app_model: App): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Get default block config | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2024-06-17 03:06:32 -05:00
										 |  |  |         # The role of the current user in the ta table must be admin, owner, or editor | 
					
						
							|  |  |  |         if not current_user.is_editor: | 
					
						
							|  |  |  |             raise Forbidden() | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |         # Get default block configs | 
					
						
							|  |  |  |         workflow_service = WorkflowService() | 
					
						
							|  |  |  |         return workflow_service.get_default_block_configs() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class DefaultBlockConfigApi(Resource): | 
					
						
							|  |  |  |     @setup_required | 
					
						
							|  |  |  |     @login_required | 
					
						
							|  |  |  |     @account_initialization_required | 
					
						
							|  |  |  |     @get_app_model(mode=[AppMode.ADVANCED_CHAT, AppMode.WORKFLOW]) | 
					
						
							|  |  |  |     def get(self, app_model: App, block_type: str): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Get default block config | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2024-06-17 03:06:32 -05:00
										 |  |  |         # The role of the current user in the ta table must be admin, owner, or editor | 
					
						
							|  |  |  |         if not current_user.is_editor: | 
					
						
							|  |  |  |             raise Forbidden() | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-17 17:05:13 +08:00
										 |  |  |         if not isinstance(current_user, Account): | 
					
						
							|  |  |  |             raise Forbidden() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |         parser = reqparse.RequestParser() | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  |         parser.add_argument("q", type=str, location="args") | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |         args = parser.parse_args() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-17 17:05:13 +08:00
										 |  |  |         q = args.get("q") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |         filters = None | 
					
						
							| 
									
										
										
										
											2025-02-17 17:05:13 +08:00
										 |  |  |         if q: | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |             try: | 
					
						
							| 
									
										
										
										
											2024-12-03 13:56:40 +08:00
										 |  |  |                 filters = json.loads(args.get("q", "")) | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |             except json.JSONDecodeError: | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  |                 raise ValueError("Invalid filters") | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # Get default block configs | 
					
						
							|  |  |  |         workflow_service = WorkflowService() | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  |         return workflow_service.get_default_block_config(node_type=block_type, filters=filters) | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class ConvertToWorkflowApi(Resource): | 
					
						
							|  |  |  |     @setup_required | 
					
						
							|  |  |  |     @login_required | 
					
						
							|  |  |  |     @account_initialization_required | 
					
						
							|  |  |  |     @get_app_model(mode=[AppMode.CHAT, AppMode.COMPLETION]) | 
					
						
							|  |  |  |     def post(self, app_model: App): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Convert basic mode of chatbot app to workflow mode | 
					
						
							|  |  |  |         Convert expert mode of chatbot app to workflow mode | 
					
						
							|  |  |  |         Convert Completion App to Workflow App | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2024-06-17 03:06:32 -05:00
										 |  |  |         # The role of the current user in the ta table must be admin, owner, or editor | 
					
						
							|  |  |  |         if not current_user.is_editor: | 
					
						
							|  |  |  |             raise Forbidden() | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-17 17:05:13 +08:00
										 |  |  |         if not isinstance(current_user, Account): | 
					
						
							|  |  |  |             raise Forbidden() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |         if request.data: | 
					
						
							|  |  |  |             parser = reqparse.RequestParser() | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  |             parser.add_argument("name", type=str, required=False, nullable=True, location="json") | 
					
						
							|  |  |  |             parser.add_argument("icon_type", type=str, required=False, nullable=True, location="json") | 
					
						
							|  |  |  |             parser.add_argument("icon", type=str, required=False, nullable=True, location="json") | 
					
						
							|  |  |  |             parser.add_argument("icon_background", type=str, required=False, nullable=True, location="json") | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |             args = parser.parse_args() | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             args = {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # convert to workflow mode | 
					
						
							|  |  |  |         workflow_service = WorkflowService() | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  |         new_app_model = workflow_service.convert_to_workflow(app_model=app_model, account=current_user, args=args) | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # return app id | 
					
						
							|  |  |  |         return { | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  |             "new_app_id": new_app_model.id, | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-12-20 14:52:20 +08:00
										 |  |  | class WorkflowConfigApi(Resource): | 
					
						
							|  |  |  |     """Resource for workflow configuration.""" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @setup_required | 
					
						
							|  |  |  |     @login_required | 
					
						
							|  |  |  |     @account_initialization_required | 
					
						
							|  |  |  |     @get_app_model(mode=[AppMode.ADVANCED_CHAT, AppMode.WORKFLOW]) | 
					
						
							|  |  |  |     def get(self, app_model: App): | 
					
						
							|  |  |  |         return { | 
					
						
							|  |  |  |             "parallel_depth_limit": dify_config.WORKFLOW_PARALLEL_DEPTH_LIMIT, | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-12-27 21:05:06 +08:00
										 |  |  | class PublishedAllWorkflowApi(Resource): | 
					
						
							|  |  |  |     @setup_required | 
					
						
							|  |  |  |     @login_required | 
					
						
							|  |  |  |     @account_initialization_required | 
					
						
							|  |  |  |     @get_app_model(mode=[AppMode.ADVANCED_CHAT, AppMode.WORKFLOW]) | 
					
						
							|  |  |  |     @marshal_with(workflow_pagination_fields) | 
					
						
							|  |  |  |     def get(self, app_model: App): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Get published workflows | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         if not current_user.is_editor: | 
					
						
							|  |  |  |             raise Forbidden() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         parser = reqparse.RequestParser() | 
					
						
							|  |  |  |         parser.add_argument("page", type=inputs.int_range(1, 99999), required=False, default=1, location="args") | 
					
						
							|  |  |  |         parser.add_argument("limit", type=inputs.int_range(1, 100), required=False, default=20, location="args") | 
					
						
							| 
									
										
										
										
											2025-03-10 13:34:31 +08:00
										 |  |  |         parser.add_argument("user_id", type=str, required=False, location="args") | 
					
						
							|  |  |  |         parser.add_argument("named_only", type=inputs.boolean, required=False, default=False, location="args") | 
					
						
							|  |  |  |         args = parser.parse_args() | 
					
						
							|  |  |  |         page = int(args.get("page", 1)) | 
					
						
							|  |  |  |         limit = int(args.get("limit", 10)) | 
					
						
							|  |  |  |         user_id = args.get("user_id") | 
					
						
							|  |  |  |         named_only = args.get("named_only", False) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if user_id: | 
					
						
							|  |  |  |             if user_id != current_user.id: | 
					
						
							|  |  |  |                 raise Forbidden() | 
					
						
							|  |  |  |             user_id = cast(str, user_id) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         workflow_service = WorkflowService() | 
					
						
							|  |  |  |         with Session(db.engine) as session: | 
					
						
							|  |  |  |             workflows, has_more = workflow_service.get_all_published_workflow( | 
					
						
							|  |  |  |                 session=session, | 
					
						
							|  |  |  |                 app_model=app_model, | 
					
						
							|  |  |  |                 page=page, | 
					
						
							|  |  |  |                 limit=limit, | 
					
						
							|  |  |  |                 user_id=user_id, | 
					
						
							|  |  |  |                 named_only=named_only, | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return { | 
					
						
							|  |  |  |                 "items": workflows, | 
					
						
							|  |  |  |                 "page": page, | 
					
						
							|  |  |  |                 "limit": limit, | 
					
						
							|  |  |  |                 "has_more": has_more, | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class WorkflowByIdApi(Resource): | 
					
						
							|  |  |  |     @setup_required | 
					
						
							|  |  |  |     @login_required | 
					
						
							|  |  |  |     @account_initialization_required | 
					
						
							|  |  |  |     @get_app_model(mode=[AppMode.ADVANCED_CHAT, AppMode.WORKFLOW]) | 
					
						
							|  |  |  |     @marshal_with(workflow_fields) | 
					
						
							|  |  |  |     def patch(self, app_model: App, workflow_id: str): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Update workflow attributes | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         # Check permission | 
					
						
							|  |  |  |         if not current_user.is_editor: | 
					
						
							|  |  |  |             raise Forbidden() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if not isinstance(current_user, Account): | 
					
						
							|  |  |  |             raise Forbidden() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         parser = reqparse.RequestParser() | 
					
						
							|  |  |  |         parser.add_argument("marked_name", type=str, required=False, location="json") | 
					
						
							|  |  |  |         parser.add_argument("marked_comment", type=str, required=False, location="json") | 
					
						
							| 
									
										
										
										
											2024-12-27 21:05:06 +08:00
										 |  |  |         args = parser.parse_args() | 
					
						
							| 
									
										
										
										
											2025-03-10 13:34:31 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # Validate name and comment length | 
					
						
							|  |  |  |         if args.marked_name and len(args.marked_name) > 20: | 
					
						
							|  |  |  |             raise ValueError("Marked name cannot exceed 20 characters") | 
					
						
							|  |  |  |         if args.marked_comment and len(args.marked_comment) > 100: | 
					
						
							|  |  |  |             raise ValueError("Marked comment cannot exceed 100 characters") | 
					
						
							|  |  |  |         args = parser.parse_args() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # Prepare update data | 
					
						
							|  |  |  |         update_data = {} | 
					
						
							|  |  |  |         if args.get("marked_name") is not None: | 
					
						
							|  |  |  |             update_data["marked_name"] = args["marked_name"] | 
					
						
							|  |  |  |         if args.get("marked_comment") is not None: | 
					
						
							|  |  |  |             update_data["marked_comment"] = args["marked_comment"] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if not update_data: | 
					
						
							|  |  |  |             return {"message": "No valid fields to update"}, 400 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-12-27 21:05:06 +08:00
										 |  |  |         workflow_service = WorkflowService() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-10 13:34:31 +08:00
										 |  |  |         # Create a session and manage the transaction | 
					
						
							|  |  |  |         with Session(db.engine, expire_on_commit=False) as session: | 
					
						
							|  |  |  |             workflow = workflow_service.update_workflow( | 
					
						
							|  |  |  |                 session=session, | 
					
						
							|  |  |  |                 workflow_id=workflow_id, | 
					
						
							|  |  |  |                 tenant_id=app_model.tenant_id, | 
					
						
							|  |  |  |                 account_id=current_user.id, | 
					
						
							|  |  |  |                 data=update_data, | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if not workflow: | 
					
						
							|  |  |  |                 raise NotFound("Workflow not found") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # Commit the transaction in the controller | 
					
						
							|  |  |  |             session.commit() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return workflow | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @setup_required | 
					
						
							|  |  |  |     @login_required | 
					
						
							|  |  |  |     @account_initialization_required | 
					
						
							|  |  |  |     @get_app_model(mode=[AppMode.ADVANCED_CHAT, AppMode.WORKFLOW]) | 
					
						
							|  |  |  |     def delete(self, app_model: App, workflow_id: str): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Delete workflow | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         # Check permission | 
					
						
							|  |  |  |         if not current_user.is_editor: | 
					
						
							|  |  |  |             raise Forbidden() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if not isinstance(current_user, Account): | 
					
						
							|  |  |  |             raise Forbidden() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         workflow_service = WorkflowService() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # Create a session and manage the transaction | 
					
						
							|  |  |  |         with Session(db.engine) as session: | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 workflow_service.delete_workflow( | 
					
						
							|  |  |  |                     session=session, workflow_id=workflow_id, tenant_id=app_model.tenant_id | 
					
						
							|  |  |  |                 ) | 
					
						
							|  |  |  |                 # Commit the transaction in the controller | 
					
						
							|  |  |  |                 session.commit() | 
					
						
							|  |  |  |             except WorkflowInUseError as e: | 
					
						
							|  |  |  |                 abort(400, description=str(e)) | 
					
						
							|  |  |  |             except DraftWorkflowDeletionError as e: | 
					
						
							|  |  |  |                 abort(400, description=str(e)) | 
					
						
							|  |  |  |             except ValueError as e: | 
					
						
							|  |  |  |                 raise NotFound(str(e)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return None, 204 | 
					
						
							| 
									
										
										
										
											2024-12-27 21:05:06 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-24 09:05:29 +08:00
										 |  |  | class DraftWorkflowNodeLastRunApi(Resource): | 
					
						
							|  |  |  |     @setup_required | 
					
						
							|  |  |  |     @login_required | 
					
						
							|  |  |  |     @account_initialization_required | 
					
						
							|  |  |  |     @get_app_model(mode=[AppMode.ADVANCED_CHAT, AppMode.WORKFLOW]) | 
					
						
							|  |  |  |     @marshal_with(workflow_run_node_execution_fields) | 
					
						
							|  |  |  |     def get(self, app_model: App, node_id: str): | 
					
						
							|  |  |  |         srv = WorkflowService() | 
					
						
							|  |  |  |         workflow = srv.get_draft_workflow(app_model) | 
					
						
							|  |  |  |         if not workflow: | 
					
						
							|  |  |  |             raise NotFound("Workflow not found") | 
					
						
							|  |  |  |         node_exec = srv.get_node_last_run( | 
					
						
							|  |  |  |             app_model=app_model, | 
					
						
							|  |  |  |             workflow=workflow, | 
					
						
							|  |  |  |             node_id=node_id, | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         if node_exec is None: | 
					
						
							|  |  |  |             raise NotFound("last run not found") | 
					
						
							|  |  |  |         return node_exec | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-10 13:34:31 +08:00
										 |  |  | api.add_resource( | 
					
						
							|  |  |  |     DraftWorkflowApi, | 
					
						
							|  |  |  |     "/apps/<uuid:app_id>/workflows/draft", | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | api.add_resource( | 
					
						
							|  |  |  |     WorkflowConfigApi, | 
					
						
							|  |  |  |     "/apps/<uuid:app_id>/workflows/draft/config", | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | api.add_resource( | 
					
						
							|  |  |  |     AdvancedChatDraftWorkflowRunApi, | 
					
						
							|  |  |  |     "/apps/<uuid:app_id>/advanced-chat/workflows/draft/run", | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | api.add_resource( | 
					
						
							|  |  |  |     DraftWorkflowRunApi, | 
					
						
							|  |  |  |     "/apps/<uuid:app_id>/workflows/draft/run", | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | api.add_resource( | 
					
						
							|  |  |  |     WorkflowTaskStopApi, | 
					
						
							|  |  |  |     "/apps/<uuid:app_id>/workflow-runs/tasks/<string:task_id>/stop", | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | api.add_resource( | 
					
						
							|  |  |  |     DraftWorkflowNodeRunApi, | 
					
						
							|  |  |  |     "/apps/<uuid:app_id>/workflows/draft/nodes/<string:node_id>/run", | 
					
						
							|  |  |  | ) | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  | api.add_resource( | 
					
						
							|  |  |  |     AdvancedChatDraftRunIterationNodeApi, | 
					
						
							|  |  |  |     "/apps/<uuid:app_id>/advanced-chat/workflows/draft/iteration/nodes/<string:node_id>/run", | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | api.add_resource( | 
					
						
							| 
									
										
										
										
											2025-03-10 13:34:31 +08:00
										 |  |  |     WorkflowDraftRunIterationNodeApi, | 
					
						
							|  |  |  |     "/apps/<uuid:app_id>/workflows/draft/iteration/nodes/<string:node_id>/run", | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  | ) | 
					
						
							| 
									
										
										
										
											2025-03-05 17:41:15 +08:00
										 |  |  | api.add_resource( | 
					
						
							|  |  |  |     AdvancedChatDraftRunLoopNodeApi, | 
					
						
							|  |  |  |     "/apps/<uuid:app_id>/advanced-chat/workflows/draft/loop/nodes/<string:node_id>/run", | 
					
						
							|  |  |  | ) | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  | api.add_resource( | 
					
						
							| 
									
										
										
										
											2025-03-10 13:34:31 +08:00
										 |  |  |     WorkflowDraftRunLoopNodeApi, | 
					
						
							|  |  |  |     "/apps/<uuid:app_id>/workflows/draft/loop/nodes/<string:node_id>/run", | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | api.add_resource( | 
					
						
							|  |  |  |     PublishedWorkflowApi, | 
					
						
							|  |  |  |     "/apps/<uuid:app_id>/workflows/publish", | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | api.add_resource( | 
					
						
							|  |  |  |     PublishedAllWorkflowApi, | 
					
						
							|  |  |  |     "/apps/<uuid:app_id>/workflows", | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | api.add_resource( | 
					
						
							|  |  |  |     DefaultBlockConfigsApi, | 
					
						
							|  |  |  |     "/apps/<uuid:app_id>/workflows/default-workflow-block-configs", | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | api.add_resource( | 
					
						
							|  |  |  |     DefaultBlockConfigApi, | 
					
						
							|  |  |  |     "/apps/<uuid:app_id>/workflows/default-workflow-block-configs/<string:block_type>", | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | api.add_resource( | 
					
						
							|  |  |  |     ConvertToWorkflowApi, | 
					
						
							|  |  |  |     "/apps/<uuid:app_id>/convert-to-workflow", | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | api.add_resource( | 
					
						
							|  |  |  |     WorkflowByIdApi, | 
					
						
							|  |  |  |     "/apps/<uuid:app_id>/workflows/<string:workflow_id>", | 
					
						
							| 
									
										
										
										
											2024-08-26 15:29:10 +08:00
										 |  |  | ) | 
					
						
							| 
									
										
										
										
											2025-06-24 09:05:29 +08:00
										 |  |  | api.add_resource( | 
					
						
							|  |  |  |     DraftWorkflowNodeLastRunApi, | 
					
						
							|  |  |  |     "/apps/<uuid:app_id>/workflows/draft/nodes/<string:node_id>/last-run", | 
					
						
							|  |  |  | ) |