workflow_converter.py 26 KB

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