workflow_converter.py 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626
  1. import json
  2. from typing import Optional
  3. from core.app.app_config.entities import (
  4. DatasetEntity,
  5. DatasetRetrieveConfigEntity,
  6. EasyUIBasedAppConfig,
  7. ExternalDataVariableEntity,
  8. ModelConfigEntity,
  9. PromptTemplateEntity,
  10. VariableEntity,
  11. )
  12. from core.app.apps.agent_chat.app_config_manager import AgentChatAppConfigManager
  13. from core.app.apps.chat.app_config_manager import ChatAppConfigManager
  14. from core.app.apps.completion.app_config_manager import CompletionAppConfigManager
  15. from core.file.file_obj import FileExtraConfig
  16. from core.helper import encrypter
  17. from core.model_runtime.entities.llm_entities import LLMMode
  18. from core.model_runtime.utils.encoders import jsonable_encoder
  19. from core.prompt.simple_prompt_transform import SimplePromptTransform
  20. from core.workflow.entities.node_entities import NodeType
  21. from events.app_event import app_was_created
  22. from extensions.ext_database import db
  23. from models.account import Account
  24. from models.api_based_extension import APIBasedExtension, APIBasedExtensionPoint
  25. from models.model import App, AppMode, AppModelConfig
  26. from models.workflow import Workflow, WorkflowType
  27. class WorkflowConverter:
  28. """
  29. App Convert to Workflow Mode
  30. """
  31. def convert_to_workflow(
  32. self, app_model: App, account: Account, name: str, icon_type: str, icon: str, icon_background: str
  33. ):
  34. """
  35. Convert app to workflow
  36. - basic mode of chatbot app
  37. - expert mode of chatbot app
  38. - completion app
  39. :param app_model: App instance
  40. :param account: Account
  41. :param name: new app name
  42. :param icon: new app icon
  43. :param icon_type: new app icon type
  44. :param icon_background: new app icon background
  45. :return: new App instance
  46. """
  47. # convert app model config
  48. if not app_model.app_model_config:
  49. raise ValueError("App model config is required")
  50. workflow = self.convert_app_model_config_to_workflow(
  51. app_model=app_model, app_model_config=app_model.app_model_config, account_id=account.id
  52. )
  53. # create new app
  54. new_app = App()
  55. new_app.tenant_id = app_model.tenant_id
  56. new_app.name = name if name else app_model.name + "(workflow)"
  57. new_app.mode = AppMode.ADVANCED_CHAT.value if app_model.mode == AppMode.CHAT.value else AppMode.WORKFLOW.value
  58. new_app.icon_type = icon_type if icon_type else app_model.icon_type
  59. new_app.icon = icon if icon else app_model.icon
  60. new_app.icon_background = icon_background if icon_background else app_model.icon_background
  61. new_app.enable_site = app_model.enable_site
  62. new_app.enable_api = app_model.enable_api
  63. new_app.api_rpm = app_model.api_rpm
  64. new_app.api_rph = app_model.api_rph
  65. new_app.is_demo = False
  66. new_app.is_public = app_model.is_public
  67. db.session.add(new_app)
  68. db.session.flush()
  69. db.session.commit()
  70. workflow.app_id = new_app.id
  71. db.session.commit()
  72. app_was_created.send(new_app, account=account)
  73. return new_app
  74. def convert_app_model_config_to_workflow(self, app_model: App, app_model_config: AppModelConfig, account_id: str):
  75. """
  76. Convert app model config to workflow mode
  77. :param app_model: App instance
  78. :param app_model_config: AppModelConfig instance
  79. :param account_id: Account ID
  80. """
  81. # get new app mode
  82. new_app_mode = self._get_new_app_mode(app_model)
  83. # convert app model config
  84. app_config = self._convert_to_app_config(app_model=app_model, app_model_config=app_model_config)
  85. # init workflow graph
  86. graph = {"nodes": [], "edges": []}
  87. # Convert list:
  88. # - variables -> start
  89. # - model_config -> llm
  90. # - prompt_template -> llm
  91. # - file_upload -> llm
  92. # - external_data_variables -> http-request
  93. # - dataset -> knowledge-retrieval
  94. # - show_retrieve_source -> knowledge-retrieval
  95. # convert to start node
  96. start_node = self._convert_to_start_node(variables=app_config.variables)
  97. graph["nodes"].append(start_node)
  98. # convert to http request node
  99. external_data_variable_node_mapping = {}
  100. if app_config.external_data_variables:
  101. http_request_nodes, external_data_variable_node_mapping = self._convert_to_http_request_node(
  102. app_model=app_model,
  103. variables=app_config.variables,
  104. external_data_variables=app_config.external_data_variables,
  105. )
  106. for http_request_node in http_request_nodes:
  107. graph = self._append_node(graph, http_request_node)
  108. # convert to knowledge retrieval node
  109. if app_config.dataset:
  110. knowledge_retrieval_node = self._convert_to_knowledge_retrieval_node(
  111. new_app_mode=new_app_mode, dataset_config=app_config.dataset, model_config=app_config.model
  112. )
  113. if knowledge_retrieval_node:
  114. graph = self._append_node(graph, knowledge_retrieval_node)
  115. # convert to llm node
  116. llm_node = self._convert_to_llm_node(
  117. original_app_mode=AppMode.value_of(app_model.mode),
  118. new_app_mode=new_app_mode,
  119. graph=graph,
  120. model_config=app_config.model,
  121. prompt_template=app_config.prompt_template,
  122. file_upload=app_config.additional_features.file_upload,
  123. external_data_variable_node_mapping=external_data_variable_node_mapping,
  124. )
  125. graph = self._append_node(graph, llm_node)
  126. if new_app_mode == AppMode.WORKFLOW:
  127. # convert to end node by app mode
  128. end_node = self._convert_to_end_node()
  129. graph = self._append_node(graph, end_node)
  130. else:
  131. answer_node = self._convert_to_answer_node()
  132. graph = self._append_node(graph, answer_node)
  133. app_model_config_dict = app_config.app_model_config_dict
  134. # features
  135. if new_app_mode == AppMode.ADVANCED_CHAT:
  136. features = {
  137. "opening_statement": app_model_config_dict.get("opening_statement"),
  138. "suggested_questions": app_model_config_dict.get("suggested_questions"),
  139. "suggested_questions_after_answer": app_model_config_dict.get("suggested_questions_after_answer"),
  140. "speech_to_text": app_model_config_dict.get("speech_to_text"),
  141. "text_to_speech": app_model_config_dict.get("text_to_speech"),
  142. "file_upload": app_model_config_dict.get("file_upload"),
  143. "sensitive_word_avoidance": app_model_config_dict.get("sensitive_word_avoidance"),
  144. "retriever_resource": app_model_config_dict.get("retriever_resource"),
  145. }
  146. else:
  147. features = {
  148. "text_to_speech": app_model_config_dict.get("text_to_speech"),
  149. "file_upload": app_model_config_dict.get("file_upload"),
  150. "sensitive_word_avoidance": app_model_config_dict.get("sensitive_word_avoidance"),
  151. }
  152. # create workflow record
  153. workflow = Workflow(
  154. tenant_id=app_model.tenant_id,
  155. app_id=app_model.id,
  156. type=WorkflowType.from_app_mode(new_app_mode).value,
  157. version="draft",
  158. graph=json.dumps(graph),
  159. features=json.dumps(features),
  160. created_by=account_id,
  161. environment_variables=[],
  162. conversation_variables=[],
  163. )
  164. db.session.add(workflow)
  165. db.session.commit()
  166. return workflow
  167. def _convert_to_app_config(self, app_model: App, app_model_config: AppModelConfig) -> EasyUIBasedAppConfig:
  168. app_mode = AppMode.value_of(app_model.mode)
  169. if app_mode == AppMode.AGENT_CHAT or app_model.is_agent:
  170. app_model.mode = AppMode.AGENT_CHAT.value
  171. app_config = AgentChatAppConfigManager.get_app_config(
  172. app_model=app_model, app_model_config=app_model_config
  173. )
  174. elif app_mode == AppMode.CHAT:
  175. app_config = ChatAppConfigManager.get_app_config(app_model=app_model, app_model_config=app_model_config)
  176. elif app_mode == AppMode.COMPLETION:
  177. app_config = CompletionAppConfigManager.get_app_config(
  178. app_model=app_model, app_model_config=app_model_config
  179. )
  180. else:
  181. raise ValueError("Invalid app mode")
  182. return app_config
  183. def _convert_to_start_node(self, variables: list[VariableEntity]) -> dict:
  184. """
  185. Convert to Start Node
  186. :param variables: list of variables
  187. :return:
  188. """
  189. return {
  190. "id": "start",
  191. "position": None,
  192. "data": {
  193. "title": "START",
  194. "type": NodeType.START.value,
  195. "variables": [jsonable_encoder(v) for v in variables],
  196. },
  197. }
  198. def _convert_to_http_request_node(
  199. self, app_model: App, variables: list[VariableEntity], external_data_variables: list[ExternalDataVariableEntity]
  200. ) -> tuple[list[dict], dict[str, str]]:
  201. """
  202. Convert API Based Extension to HTTP Request Node
  203. :param app_model: App instance
  204. :param variables: list of variables
  205. :param external_data_variables: list of external data variables
  206. :return:
  207. """
  208. index = 1
  209. nodes = []
  210. external_data_variable_node_mapping = {}
  211. tenant_id = app_model.tenant_id
  212. for external_data_variable in external_data_variables:
  213. tool_type = external_data_variable.type
  214. if tool_type != "api":
  215. continue
  216. tool_variable = external_data_variable.variable
  217. tool_config = external_data_variable.config
  218. # get params from config
  219. api_based_extension_id = tool_config.get("api_based_extension_id")
  220. if not api_based_extension_id:
  221. continue
  222. # get api_based_extension
  223. api_based_extension = self._get_api_based_extension(
  224. tenant_id=tenant_id, api_based_extension_id=api_based_extension_id
  225. )
  226. # decrypt api_key
  227. api_key = encrypter.decrypt_token(tenant_id=tenant_id, token=api_based_extension.api_key)
  228. inputs = {}
  229. for v in variables:
  230. inputs[v.variable] = "{{#start." + v.variable + "#}}"
  231. request_body = {
  232. "point": APIBasedExtensionPoint.APP_EXTERNAL_DATA_TOOL_QUERY.value,
  233. "params": {
  234. "app_id": app_model.id,
  235. "tool_variable": tool_variable,
  236. "inputs": inputs,
  237. "query": "{{#sys.query#}}" if app_model.mode == AppMode.CHAT.value else "",
  238. },
  239. }
  240. request_body_json = json.dumps(request_body)
  241. request_body_json = request_body_json.replace(r"\{\{", "{{").replace(r"\}\}", "}}")
  242. http_request_node = {
  243. "id": f"http_request_{index}",
  244. "position": None,
  245. "data": {
  246. "title": f"HTTP REQUEST {api_based_extension.name}",
  247. "type": NodeType.HTTP_REQUEST.value,
  248. "method": "post",
  249. "url": api_based_extension.api_endpoint,
  250. "authorization": {"type": "api-key", "config": {"type": "bearer", "api_key": api_key}},
  251. "headers": "",
  252. "params": "",
  253. "body": {"type": "json", "data": request_body_json},
  254. },
  255. }
  256. nodes.append(http_request_node)
  257. # append code node for response body parsing
  258. code_node = {
  259. "id": f"code_{index}",
  260. "position": None,
  261. "data": {
  262. "title": f"Parse {api_based_extension.name} Response",
  263. "type": NodeType.CODE.value,
  264. "variables": [{"variable": "response_json", "value_selector": [http_request_node["id"], "body"]}],
  265. "code_language": "python3",
  266. "code": "import json\n\ndef main(response_json: str) -> str:\n response_body = json.loads("
  267. 'response_json)\n return {\n "result": response_body["result"]\n }',
  268. "outputs": {"result": {"type": "string"}},
  269. },
  270. }
  271. nodes.append(code_node)
  272. external_data_variable_node_mapping[external_data_variable.variable] = code_node["id"]
  273. index += 1
  274. return nodes, external_data_variable_node_mapping
  275. def _convert_to_knowledge_retrieval_node(
  276. self, new_app_mode: AppMode, dataset_config: DatasetEntity, model_config: ModelConfigEntity
  277. ) -> Optional[dict]:
  278. """
  279. Convert datasets to Knowledge Retrieval Node
  280. :param new_app_mode: new app mode
  281. :param dataset_config: dataset
  282. :param model_config: model config
  283. :return:
  284. """
  285. retrieve_config = dataset_config.retrieve_config
  286. if new_app_mode == AppMode.ADVANCED_CHAT:
  287. query_variable_selector = ["sys", "query"]
  288. elif retrieve_config.query_variable:
  289. # fetch query variable
  290. query_variable_selector = ["start", retrieve_config.query_variable]
  291. else:
  292. return None
  293. return {
  294. "id": "knowledge_retrieval",
  295. "position": None,
  296. "data": {
  297. "title": "KNOWLEDGE RETRIEVAL",
  298. "type": NodeType.KNOWLEDGE_RETRIEVAL.value,
  299. "query_variable_selector": query_variable_selector,
  300. "dataset_ids": dataset_config.dataset_ids,
  301. "retrieval_mode": retrieve_config.retrieve_strategy.value,
  302. "single_retrieval_config": {
  303. "model": {
  304. "provider": model_config.provider,
  305. "name": model_config.model,
  306. "mode": model_config.mode,
  307. "completion_params": {
  308. **model_config.parameters,
  309. "stop": model_config.stop,
  310. },
  311. }
  312. }
  313. if retrieve_config.retrieve_strategy == DatasetRetrieveConfigEntity.RetrieveStrategy.SINGLE
  314. else None,
  315. "multiple_retrieval_config": {
  316. "top_k": retrieve_config.top_k,
  317. "score_threshold": retrieve_config.score_threshold,
  318. "reranking_model": retrieve_config.reranking_model,
  319. }
  320. if retrieve_config.retrieve_strategy == DatasetRetrieveConfigEntity.RetrieveStrategy.MULTIPLE
  321. else None,
  322. },
  323. }
  324. def _convert_to_llm_node(
  325. self,
  326. original_app_mode: AppMode,
  327. new_app_mode: AppMode,
  328. graph: dict,
  329. model_config: ModelConfigEntity,
  330. prompt_template: PromptTemplateEntity,
  331. file_upload: Optional[FileExtraConfig] = None,
  332. external_data_variable_node_mapping: dict[str, str] | None = None,
  333. ) -> dict:
  334. """
  335. Convert to LLM Node
  336. :param original_app_mode: original app mode
  337. :param new_app_mode: new app mode
  338. :param graph: graph
  339. :param model_config: model config
  340. :param prompt_template: prompt template
  341. :param file_upload: file upload config (optional)
  342. :param external_data_variable_node_mapping: external data variable node mapping
  343. """
  344. # fetch start and knowledge retrieval node
  345. start_node = next(filter(lambda n: n["data"]["type"] == NodeType.START.value, graph["nodes"]))
  346. knowledge_retrieval_node = next(
  347. filter(lambda n: n["data"]["type"] == NodeType.KNOWLEDGE_RETRIEVAL.value, graph["nodes"]), None
  348. )
  349. role_prefix = None
  350. # Chat Model
  351. if model_config.mode == LLMMode.CHAT.value:
  352. if prompt_template.prompt_type == PromptTemplateEntity.PromptType.SIMPLE:
  353. if not prompt_template.simple_prompt_template:
  354. raise ValueError("Simple prompt template is required")
  355. # get prompt template
  356. prompt_transform = SimplePromptTransform()
  357. prompt_template_config = prompt_transform.get_prompt_template(
  358. app_mode=original_app_mode,
  359. provider=model_config.provider,
  360. model=model_config.model,
  361. pre_prompt=prompt_template.simple_prompt_template,
  362. has_context=knowledge_retrieval_node is not None,
  363. query_in_prompt=False,
  364. )
  365. template = prompt_template_config["prompt_template"].template
  366. if not template:
  367. prompts = []
  368. else:
  369. template = self._replace_template_variables(
  370. template, start_node["data"]["variables"], external_data_variable_node_mapping
  371. )
  372. prompts = [{"role": "user", "text": template}]
  373. else:
  374. advanced_chat_prompt_template = prompt_template.advanced_chat_prompt_template
  375. prompts = []
  376. if advanced_chat_prompt_template:
  377. for m in advanced_chat_prompt_template.messages:
  378. text = m.text
  379. text = self._replace_template_variables(
  380. text, start_node["data"]["variables"], external_data_variable_node_mapping
  381. )
  382. prompts.append({"role": m.role.value, "text": text})
  383. # Completion Model
  384. else:
  385. if prompt_template.prompt_type == PromptTemplateEntity.PromptType.SIMPLE:
  386. if not prompt_template.simple_prompt_template:
  387. raise ValueError("Simple prompt template is required")
  388. # get prompt template
  389. prompt_transform = SimplePromptTransform()
  390. prompt_template_config = prompt_transform.get_prompt_template(
  391. app_mode=original_app_mode,
  392. provider=model_config.provider,
  393. model=model_config.model,
  394. pre_prompt=prompt_template.simple_prompt_template,
  395. has_context=knowledge_retrieval_node is not None,
  396. query_in_prompt=False,
  397. )
  398. template = prompt_template_config["prompt_template"].template
  399. template = self._replace_template_variables(
  400. template=template,
  401. variables=start_node["data"]["variables"],
  402. external_data_variable_node_mapping=external_data_variable_node_mapping,
  403. )
  404. prompts = {"text": template}
  405. prompt_rules = prompt_template_config["prompt_rules"]
  406. role_prefix = {
  407. "user": prompt_rules.get("human_prefix", "Human"),
  408. "assistant": prompt_rules.get("assistant_prefix", "Assistant"),
  409. }
  410. else:
  411. advanced_completion_prompt_template = prompt_template.advanced_completion_prompt_template
  412. if advanced_completion_prompt_template:
  413. text = advanced_completion_prompt_template.prompt
  414. text = self._replace_template_variables(
  415. template=text,
  416. variables=start_node["data"]["variables"],
  417. external_data_variable_node_mapping=external_data_variable_node_mapping,
  418. )
  419. else:
  420. text = ""
  421. text = text.replace("{{#query#}}", "{{#sys.query#}}")
  422. prompts = {
  423. "text": text,
  424. }
  425. if advanced_completion_prompt_template and advanced_completion_prompt_template.role_prefix:
  426. role_prefix = {
  427. "user": advanced_completion_prompt_template.role_prefix.user,
  428. "assistant": advanced_completion_prompt_template.role_prefix.assistant,
  429. }
  430. memory = None
  431. if new_app_mode == AppMode.ADVANCED_CHAT:
  432. memory = {"role_prefix": role_prefix, "window": {"enabled": False}}
  433. completion_params = model_config.parameters
  434. completion_params.update({"stop": model_config.stop})
  435. return {
  436. "id": "llm",
  437. "position": None,
  438. "data": {
  439. "title": "LLM",
  440. "type": NodeType.LLM.value,
  441. "model": {
  442. "provider": model_config.provider,
  443. "name": model_config.model,
  444. "mode": model_config.mode,
  445. "completion_params": completion_params,
  446. },
  447. "prompt_template": prompts,
  448. "memory": memory,
  449. "context": {
  450. "enabled": knowledge_retrieval_node is not None,
  451. "variable_selector": ["knowledge_retrieval", "result"]
  452. if knowledge_retrieval_node is not None
  453. else None,
  454. },
  455. "vision": {
  456. "enabled": file_upload is not None,
  457. "variable_selector": ["sys", "files"] if file_upload is not None else None,
  458. "configs": {"detail": file_upload.image_config["detail"]}
  459. if file_upload is not None and file_upload.image_config is not None
  460. else None,
  461. },
  462. },
  463. }
  464. def _replace_template_variables(
  465. self, template: str, variables: list[dict], external_data_variable_node_mapping: dict[str, str] | None = None
  466. ) -> str:
  467. """
  468. Replace Template Variables
  469. :param template: template
  470. :param variables: list of variables
  471. :param external_data_variable_node_mapping: external data variable node mapping
  472. :return:
  473. """
  474. for v in variables:
  475. template = template.replace("{{" + v["variable"] + "}}", "{{#start." + v["variable"] + "#}}")
  476. if external_data_variable_node_mapping:
  477. for variable, code_node_id in external_data_variable_node_mapping.items():
  478. template = template.replace("{{" + variable + "}}", "{{#" + code_node_id + ".result#}}")
  479. return template
  480. def _convert_to_end_node(self) -> dict:
  481. """
  482. Convert to End Node
  483. :return:
  484. """
  485. # for original completion app
  486. return {
  487. "id": "end",
  488. "position": None,
  489. "data": {
  490. "title": "END",
  491. "type": NodeType.END.value,
  492. "outputs": [{"variable": "result", "value_selector": ["llm", "text"]}],
  493. },
  494. }
  495. def _convert_to_answer_node(self) -> dict:
  496. """
  497. Convert to Answer Node
  498. :return:
  499. """
  500. # for original chat app
  501. return {
  502. "id": "answer",
  503. "position": None,
  504. "data": {"title": "ANSWER", "type": NodeType.ANSWER.value, "answer": "{{#llm.text#}}"},
  505. }
  506. def _create_edge(self, source: str, target: str) -> dict:
  507. """
  508. Create Edge
  509. :param source: source node id
  510. :param target: target node id
  511. :return:
  512. """
  513. return {"id": f"{source}-{target}", "source": source, "target": target}
  514. def _append_node(self, graph: dict, node: dict) -> dict:
  515. """
  516. Append Node to Graph
  517. :param graph: Graph, include: nodes, edges
  518. :param node: Node to append
  519. :return:
  520. """
  521. previous_node = graph["nodes"][-1]
  522. graph["nodes"].append(node)
  523. graph["edges"].append(self._create_edge(previous_node["id"], node["id"]))
  524. return graph
  525. def _get_new_app_mode(self, app_model: App) -> AppMode:
  526. """
  527. Get new app mode
  528. :param app_model: App instance
  529. :return: AppMode
  530. """
  531. if app_model.mode == AppMode.COMPLETION.value:
  532. return AppMode.WORKFLOW
  533. else:
  534. return AppMode.ADVANCED_CHAT
  535. def _get_api_based_extension(self, tenant_id: str, api_based_extension_id: str):
  536. """
  537. Get API Based Extension
  538. :param tenant_id: tenant id
  539. :param api_based_extension_id: api based extension id
  540. :return:
  541. """
  542. api_based_extension = (
  543. db.session.query(APIBasedExtension)
  544. .filter(APIBasedExtension.tenant_id == tenant_id, APIBasedExtension.id == api_based_extension_id)
  545. .first()
  546. )
  547. if not api_based_extension:
  548. raise ValueError(f"API Based Extension not found, id: {api_based_extension_id}")
  549. return api_based_extension