service.py 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488
  1. import logging
  2. from collections.abc import Mapping
  3. from typing import Any
  4. import yaml
  5. from packaging import version
  6. from core.helper import ssrf_proxy
  7. from events.app_event import app_model_config_was_updated, app_was_created
  8. from extensions.ext_database import db
  9. from factories import variable_factory
  10. from models.account import Account
  11. from models.model import App, AppMode, AppModelConfig
  12. from models.workflow import Workflow
  13. from services.workflow_service import WorkflowService
  14. from .exc import (
  15. ContentDecodingError,
  16. EmptyContentError,
  17. FileSizeLimitExceededError,
  18. InvalidAppModeError,
  19. InvalidYAMLFormatError,
  20. MissingAppDataError,
  21. MissingModelConfigError,
  22. MissingWorkflowDataError,
  23. )
  24. logger = logging.getLogger(__name__)
  25. current_dsl_version = "0.1.2"
  26. dsl_to_dify_version_mapping: dict[str, str] = {
  27. "0.1.2": "0.8.0",
  28. "0.1.1": "0.6.0", # dsl version -> from dify version
  29. }
  30. class AppDslService:
  31. @classmethod
  32. def import_and_create_new_app_from_url(cls, tenant_id: str, url: str, args: dict, account: Account) -> App:
  33. """
  34. Import app dsl from url and create new app
  35. :param tenant_id: tenant id
  36. :param url: import url
  37. :param args: request args
  38. :param account: Account instance
  39. """
  40. max_size = 10 * 1024 * 1024 # 10MB
  41. response = ssrf_proxy.get(url.strip(), follow_redirects=True, timeout=(10, 10))
  42. response.raise_for_status()
  43. content = response.content
  44. if len(content) > max_size:
  45. raise FileSizeLimitExceededError("File size exceeds the limit of 10MB")
  46. if not content:
  47. raise EmptyContentError("Empty content from url")
  48. try:
  49. data = content.decode("utf-8")
  50. except UnicodeDecodeError as e:
  51. raise ContentDecodingError(f"Error decoding content: {e}")
  52. return cls.import_and_create_new_app(tenant_id, data, args, account)
  53. @classmethod
  54. def import_and_create_new_app(cls, tenant_id: str, data: str, args: dict, account: Account) -> App:
  55. """
  56. Import app dsl and create new app
  57. :param tenant_id: tenant id
  58. :param data: import data
  59. :param args: request args
  60. :param account: Account instance
  61. """
  62. try:
  63. import_data = yaml.safe_load(data)
  64. except yaml.YAMLError:
  65. raise InvalidYAMLFormatError("Invalid YAML format in data argument.")
  66. # check or repair dsl version
  67. import_data = _check_or_fix_dsl(import_data)
  68. app_data = import_data.get("app")
  69. if not app_data:
  70. raise MissingAppDataError("Missing app in data argument")
  71. # get app basic info
  72. name = args.get("name") or app_data.get("name")
  73. description = args.get("description") or app_data.get("description", "")
  74. icon_type = args.get("icon_type") or app_data.get("icon_type")
  75. icon = args.get("icon") or app_data.get("icon")
  76. icon_background = args.get("icon_background") or app_data.get("icon_background")
  77. use_icon_as_answer_icon = app_data.get("use_icon_as_answer_icon", False)
  78. # import dsl and create app
  79. app_mode = AppMode.value_of(app_data.get("mode"))
  80. if app_mode in {AppMode.ADVANCED_CHAT, AppMode.WORKFLOW}:
  81. workflow_data = import_data.get("workflow")
  82. if not workflow_data or not isinstance(workflow_data, dict):
  83. raise MissingWorkflowDataError(
  84. "Missing workflow in data argument when app mode is advanced-chat or workflow"
  85. )
  86. app = cls._import_and_create_new_workflow_based_app(
  87. tenant_id=tenant_id,
  88. app_mode=app_mode,
  89. workflow_data=workflow_data,
  90. account=account,
  91. name=name,
  92. description=description,
  93. icon_type=icon_type,
  94. icon=icon,
  95. icon_background=icon_background,
  96. use_icon_as_answer_icon=use_icon_as_answer_icon,
  97. )
  98. elif app_mode in {AppMode.CHAT, AppMode.AGENT_CHAT, AppMode.COMPLETION}:
  99. model_config = import_data.get("model_config")
  100. if not model_config or not isinstance(model_config, dict):
  101. raise MissingModelConfigError(
  102. "Missing model_config in data argument when app mode is chat, agent-chat or completion"
  103. )
  104. app = cls._import_and_create_new_model_config_based_app(
  105. tenant_id=tenant_id,
  106. app_mode=app_mode,
  107. model_config_data=model_config,
  108. account=account,
  109. name=name,
  110. description=description,
  111. icon_type=icon_type,
  112. icon=icon,
  113. icon_background=icon_background,
  114. use_icon_as_answer_icon=use_icon_as_answer_icon,
  115. )
  116. else:
  117. raise InvalidAppModeError("Invalid app mode")
  118. return app
  119. @classmethod
  120. def import_and_overwrite_workflow(cls, app_model: App, data: str, account: Account) -> Workflow:
  121. """
  122. Import app dsl and overwrite workflow
  123. :param app_model: App instance
  124. :param data: import data
  125. :param account: Account instance
  126. """
  127. try:
  128. import_data = yaml.safe_load(data)
  129. except yaml.YAMLError:
  130. raise InvalidYAMLFormatError("Invalid YAML format in data argument.")
  131. # check or repair dsl version
  132. import_data = _check_or_fix_dsl(import_data)
  133. app_data = import_data.get("app")
  134. if not app_data:
  135. raise MissingAppDataError("Missing app in data argument")
  136. # import dsl and overwrite app
  137. app_mode = AppMode.value_of(app_data.get("mode"))
  138. if app_mode not in {AppMode.ADVANCED_CHAT, AppMode.WORKFLOW}:
  139. raise InvalidAppModeError("Only support import workflow in advanced-chat or workflow app.")
  140. if app_data.get("mode") != app_model.mode:
  141. raise ValueError(f"App mode {app_data.get('mode')} is not matched with current app mode {app_mode.value}")
  142. workflow_data = import_data.get("workflow")
  143. if not workflow_data or not isinstance(workflow_data, dict):
  144. raise MissingWorkflowDataError(
  145. "Missing workflow in data argument when app mode is advanced-chat or workflow"
  146. )
  147. return cls._import_and_overwrite_workflow_based_app(
  148. app_model=app_model,
  149. workflow_data=workflow_data,
  150. account=account,
  151. )
  152. @classmethod
  153. def export_dsl(cls, app_model: App, include_secret: bool = False) -> str:
  154. """
  155. Export app
  156. :param app_model: App instance
  157. :return:
  158. """
  159. app_mode = AppMode.value_of(app_model.mode)
  160. export_data = {
  161. "version": current_dsl_version,
  162. "kind": "app",
  163. "app": {
  164. "name": app_model.name,
  165. "mode": app_model.mode,
  166. "icon": "🤖" if app_model.icon_type == "image" else app_model.icon,
  167. "icon_background": "#FFEAD5" if app_model.icon_type == "image" else app_model.icon_background,
  168. "description": app_model.description,
  169. "use_icon_as_answer_icon": app_model.use_icon_as_answer_icon,
  170. },
  171. }
  172. if app_mode in {AppMode.ADVANCED_CHAT, AppMode.WORKFLOW}:
  173. cls._append_workflow_export_data(
  174. export_data=export_data, app_model=app_model, include_secret=include_secret
  175. )
  176. else:
  177. cls._append_model_config_export_data(export_data, app_model)
  178. return yaml.dump(export_data, allow_unicode=True)
  179. @classmethod
  180. def _import_and_create_new_workflow_based_app(
  181. cls,
  182. tenant_id: str,
  183. app_mode: AppMode,
  184. workflow_data: Mapping[str, Any],
  185. account: Account,
  186. name: str,
  187. description: str,
  188. icon_type: str,
  189. icon: str,
  190. icon_background: str,
  191. use_icon_as_answer_icon: bool,
  192. ) -> App:
  193. """
  194. Import app dsl and create new workflow based app
  195. :param tenant_id: tenant id
  196. :param app_mode: app mode
  197. :param workflow_data: workflow data
  198. :param account: Account instance
  199. :param name: app name
  200. :param description: app description
  201. :param icon_type: app icon type, "emoji" or "image"
  202. :param icon: app icon
  203. :param icon_background: app icon background
  204. :param use_icon_as_answer_icon: use app icon as answer icon
  205. """
  206. if not workflow_data:
  207. raise MissingWorkflowDataError(
  208. "Missing workflow in data argument when app mode is advanced-chat or workflow"
  209. )
  210. app = cls._create_app(
  211. tenant_id=tenant_id,
  212. app_mode=app_mode,
  213. account=account,
  214. name=name,
  215. description=description,
  216. icon_type=icon_type,
  217. icon=icon,
  218. icon_background=icon_background,
  219. use_icon_as_answer_icon=use_icon_as_answer_icon,
  220. )
  221. # init draft workflow
  222. environment_variables_list = workflow_data.get("environment_variables") or []
  223. environment_variables = [
  224. variable_factory.build_variable_from_mapping(obj) for obj in environment_variables_list
  225. ]
  226. conversation_variables_list = workflow_data.get("conversation_variables") or []
  227. conversation_variables = [
  228. variable_factory.build_variable_from_mapping(obj) for obj in conversation_variables_list
  229. ]
  230. workflow_service = WorkflowService()
  231. draft_workflow = workflow_service.sync_draft_workflow(
  232. app_model=app,
  233. graph=workflow_data.get("graph", {}),
  234. features=workflow_data.get("features", {}),
  235. unique_hash=None,
  236. account=account,
  237. environment_variables=environment_variables,
  238. conversation_variables=conversation_variables,
  239. )
  240. workflow_service.publish_workflow(app_model=app, account=account, draft_workflow=draft_workflow)
  241. return app
  242. @classmethod
  243. def _import_and_overwrite_workflow_based_app(
  244. cls, app_model: App, workflow_data: Mapping[str, Any], account: Account
  245. ) -> Workflow:
  246. """
  247. Import app dsl and overwrite workflow based app
  248. :param app_model: App instance
  249. :param workflow_data: workflow data
  250. :param account: Account instance
  251. """
  252. if not workflow_data:
  253. raise MissingWorkflowDataError(
  254. "Missing workflow in data argument when app mode is advanced-chat or workflow"
  255. )
  256. # fetch draft workflow by app_model
  257. workflow_service = WorkflowService()
  258. current_draft_workflow = workflow_service.get_draft_workflow(app_model=app_model)
  259. if current_draft_workflow:
  260. unique_hash = current_draft_workflow.unique_hash
  261. else:
  262. unique_hash = None
  263. # sync draft workflow
  264. environment_variables_list = workflow_data.get("environment_variables") or []
  265. environment_variables = [
  266. variable_factory.build_variable_from_mapping(obj) for obj in environment_variables_list
  267. ]
  268. conversation_variables_list = workflow_data.get("conversation_variables") or []
  269. conversation_variables = [
  270. variable_factory.build_variable_from_mapping(obj) for obj in conversation_variables_list
  271. ]
  272. draft_workflow = workflow_service.sync_draft_workflow(
  273. app_model=app_model,
  274. graph=workflow_data.get("graph", {}),
  275. features=workflow_data.get("features", {}),
  276. unique_hash=unique_hash,
  277. account=account,
  278. environment_variables=environment_variables,
  279. conversation_variables=conversation_variables,
  280. )
  281. return draft_workflow
  282. @classmethod
  283. def _import_and_create_new_model_config_based_app(
  284. cls,
  285. tenant_id: str,
  286. app_mode: AppMode,
  287. model_config_data: Mapping[str, Any],
  288. account: Account,
  289. name: str,
  290. description: str,
  291. icon_type: str,
  292. icon: str,
  293. icon_background: str,
  294. use_icon_as_answer_icon: bool,
  295. ) -> App:
  296. """
  297. Import app dsl and create new model config based app
  298. :param tenant_id: tenant id
  299. :param app_mode: app mode
  300. :param model_config_data: model config data
  301. :param account: Account instance
  302. :param name: app name
  303. :param description: app description
  304. :param icon: app icon
  305. :param icon_background: app icon background
  306. """
  307. if not model_config_data:
  308. raise MissingModelConfigError(
  309. "Missing model_config in data argument when app mode is chat, agent-chat or completion"
  310. )
  311. app = cls._create_app(
  312. tenant_id=tenant_id,
  313. app_mode=app_mode,
  314. account=account,
  315. name=name,
  316. description=description,
  317. icon_type=icon_type,
  318. icon=icon,
  319. icon_background=icon_background,
  320. use_icon_as_answer_icon=use_icon_as_answer_icon,
  321. )
  322. app_model_config = AppModelConfig()
  323. app_model_config = app_model_config.from_model_config_dict(model_config_data)
  324. app_model_config.app_id = app.id
  325. app_model_config.created_by = account.id
  326. app_model_config.updated_by = account.id
  327. db.session.add(app_model_config)
  328. db.session.commit()
  329. app.app_model_config_id = app_model_config.id
  330. app_model_config_was_updated.send(app, app_model_config=app_model_config)
  331. return app
  332. @classmethod
  333. def _create_app(
  334. cls,
  335. tenant_id: str,
  336. app_mode: AppMode,
  337. account: Account,
  338. name: str,
  339. description: str,
  340. icon_type: str,
  341. icon: str,
  342. icon_background: str,
  343. use_icon_as_answer_icon: bool,
  344. ) -> App:
  345. """
  346. Create new app
  347. :param tenant_id: tenant id
  348. :param app_mode: app mode
  349. :param account: Account instance
  350. :param name: app name
  351. :param description: app description
  352. :param icon_type: app icon type, "emoji" or "image"
  353. :param icon: app icon
  354. :param icon_background: app icon background
  355. :param use_icon_as_answer_icon: use app icon as answer icon
  356. """
  357. app = App(
  358. tenant_id=tenant_id,
  359. mode=app_mode.value,
  360. name=name,
  361. description=description,
  362. icon_type=icon_type,
  363. icon=icon,
  364. icon_background=icon_background,
  365. enable_site=True,
  366. enable_api=True,
  367. use_icon_as_answer_icon=use_icon_as_answer_icon,
  368. created_by=account.id,
  369. updated_by=account.id,
  370. )
  371. db.session.add(app)
  372. db.session.commit()
  373. app_was_created.send(app, account=account)
  374. return app
  375. @classmethod
  376. def _append_workflow_export_data(cls, *, export_data: dict, app_model: App, include_secret: bool) -> None:
  377. """
  378. Append workflow export data
  379. :param export_data: export data
  380. :param app_model: App instance
  381. """
  382. workflow_service = WorkflowService()
  383. workflow = workflow_service.get_draft_workflow(app_model)
  384. if not workflow:
  385. raise ValueError("Missing draft workflow configuration, please check.")
  386. export_data["workflow"] = workflow.to_dict(include_secret=include_secret)
  387. @classmethod
  388. def _append_model_config_export_data(cls, export_data: dict, app_model: App) -> None:
  389. """
  390. Append model config export data
  391. :param export_data: export data
  392. :param app_model: App instance
  393. """
  394. app_model_config = app_model.app_model_config
  395. if not app_model_config:
  396. raise ValueError("Missing app configuration, please check.")
  397. export_data["model_config"] = app_model_config.to_dict()
  398. def _check_or_fix_dsl(import_data: dict[str, Any]) -> Mapping[str, Any]:
  399. """
  400. Check or fix dsl
  401. :param import_data: import data
  402. :raises DSLVersionNotSupportedError: if the imported DSL version is newer than the current version
  403. """
  404. if not import_data.get("version"):
  405. import_data["version"] = "0.1.0"
  406. if not import_data.get("kind") or import_data.get("kind") != "app":
  407. import_data["kind"] = "app"
  408. imported_version = import_data.get("version")
  409. if imported_version != current_dsl_version:
  410. if imported_version and version.parse(imported_version) > version.parse(current_dsl_version):
  411. errmsg = (
  412. f"The imported DSL version {imported_version} is newer than "
  413. f"the current supported version {current_dsl_version}. "
  414. f"Please upgrade your Dify instance to import this configuration."
  415. )
  416. logger.warning(errmsg)
  417. # raise DSLVersionNotSupportedError(errmsg)
  418. else:
  419. logger.warning(
  420. f"DSL version {imported_version} is older than "
  421. f"the current version {current_dsl_version}. "
  422. f"This may cause compatibility issues."
  423. )
  424. return import_data