tool_manager.py 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694
  1. import json
  2. import logging
  3. import mimetypes
  4. from collections.abc import Generator
  5. from os import listdir, path
  6. from threading import Lock, Thread
  7. from typing import Any, Optional, Union, cast
  8. from configs import dify_config
  9. from core.agent.entities import AgentToolEntity
  10. from core.app.entities.app_invoke_entities import InvokeFrom
  11. from core.helper.module_import_helper import load_single_subclass_from_source
  12. from core.helper.position_helper import is_filtered
  13. from core.model_runtime.utils.encoders import jsonable_encoder
  14. from core.tools.entities.api_entities import UserToolProvider, UserToolProviderTypeLiteral
  15. from core.tools.entities.common_entities import I18nObject
  16. from core.tools.entities.tool_entities import ApiProviderAuthType, ToolInvokeFrom, ToolParameter
  17. from core.tools.errors import ToolNotFoundError, ToolProviderNotFoundError
  18. from core.tools.provider.api_tool_provider import ApiToolProviderController
  19. from core.tools.provider.builtin._positions import BuiltinToolProviderSort
  20. from core.tools.provider.builtin_tool_provider import BuiltinToolProviderController
  21. from core.tools.provider.tool_provider import ToolProviderController
  22. from core.tools.provider.workflow_tool_provider import WorkflowToolProviderController
  23. from core.tools.tool.api_tool import ApiTool
  24. from core.tools.tool.builtin_tool import BuiltinTool
  25. from core.tools.tool.tool import Tool
  26. from core.tools.tool_label_manager import ToolLabelManager
  27. from core.tools.utils.configuration import ToolConfigurationManager, ToolParameterConfigurationManager
  28. from core.workflow.nodes.tool.entities import ToolEntity
  29. from extensions.ext_database import db
  30. from models.tools import ApiToolProvider, BuiltinToolProvider, WorkflowToolProvider
  31. from services.tools.tools_transform_service import ToolTransformService
  32. logger = logging.getLogger(__name__)
  33. class ToolManager:
  34. _builtin_provider_lock = Lock()
  35. _builtin_providers: dict[str, BuiltinToolProviderController] = {}
  36. _builtin_providers_loaded = False
  37. _builtin_tools_labels: dict[str, Union[I18nObject, None]] = {}
  38. @classmethod
  39. def get_builtin_provider(cls, provider: str) -> BuiltinToolProviderController:
  40. """
  41. get the builtin provider
  42. :param provider: the name of the provider
  43. :return: the provider
  44. """
  45. if len(cls._builtin_providers) == 0:
  46. # init the builtin providers
  47. cls.load_builtin_providers_cache()
  48. if provider not in cls._builtin_providers:
  49. raise ToolProviderNotFoundError(f"builtin provider {provider} not found")
  50. return cls._builtin_providers[provider]
  51. @classmethod
  52. def get_builtin_tool(cls, provider: str, tool_name: str) -> Union[BuiltinTool, Tool]:
  53. """
  54. get the builtin tool
  55. :param provider: the name of the provider
  56. :param tool_name: the name of the tool
  57. :return: the provider, the tool
  58. """
  59. provider_controller = cls.get_builtin_provider(provider)
  60. tool = provider_controller.get_tool(tool_name)
  61. if tool is None:
  62. raise ToolNotFoundError(f"tool {tool_name} not found")
  63. return tool
  64. @classmethod
  65. def get_tool(
  66. cls, provider_type: str, provider_id: str, tool_name: str, tenant_id: Optional[str] = None
  67. ) -> Union[BuiltinTool, ApiTool, Tool]:
  68. """
  69. get the tool
  70. :param provider_type: the type of the provider
  71. :param provider_name: the name of the provider
  72. :param tool_name: the name of the tool
  73. :return: the tool
  74. """
  75. if provider_type == "builtin":
  76. return cls.get_builtin_tool(provider_id, tool_name)
  77. elif provider_type == "api":
  78. if tenant_id is None:
  79. raise ValueError("tenant id is required for api provider")
  80. api_provider, _ = cls.get_api_provider_controller(tenant_id, provider_id)
  81. return api_provider.get_tool(tool_name)
  82. elif provider_type == "app":
  83. raise NotImplementedError("app provider not implemented")
  84. else:
  85. raise ToolProviderNotFoundError(f"provider type {provider_type} not found")
  86. @classmethod
  87. def get_tool_runtime(
  88. cls,
  89. provider_type: str,
  90. provider_id: str,
  91. tool_name: str,
  92. tenant_id: str,
  93. invoke_from: InvokeFrom = InvokeFrom.DEBUGGER,
  94. tool_invoke_from: ToolInvokeFrom = ToolInvokeFrom.AGENT,
  95. ) -> Union[BuiltinTool, ApiTool, Tool]:
  96. """
  97. get the tool runtime
  98. :param provider_type: the type of the provider
  99. :param provider_name: the name of the provider
  100. :param tool_name: the name of the tool
  101. :return: the tool
  102. """
  103. controller: Union[BuiltinToolProviderController, ApiToolProviderController, WorkflowToolProviderController]
  104. if provider_type == "builtin":
  105. builtin_tool = cls.get_builtin_tool(provider_id, tool_name)
  106. # check if the builtin tool need credentials
  107. provider_controller = cls.get_builtin_provider(provider_id)
  108. if not provider_controller.need_credentials:
  109. return builtin_tool.fork_tool_runtime(
  110. runtime={
  111. "tenant_id": tenant_id,
  112. "credentials": {},
  113. "invoke_from": invoke_from,
  114. "tool_invoke_from": tool_invoke_from,
  115. }
  116. )
  117. # get credentials
  118. builtin_provider: Optional[BuiltinToolProvider] = (
  119. db.session.query(BuiltinToolProvider)
  120. .filter(
  121. BuiltinToolProvider.tenant_id == tenant_id,
  122. BuiltinToolProvider.provider == provider_id,
  123. )
  124. .first()
  125. )
  126. if builtin_provider is None:
  127. raise ToolProviderNotFoundError(f"builtin provider {provider_id} not found")
  128. # decrypt the credentials
  129. credentials = builtin_provider.credentials
  130. controller = cls.get_builtin_provider(provider_id)
  131. tool_configuration = ToolConfigurationManager(tenant_id=tenant_id, provider_controller=controller)
  132. decrypted_credentials = tool_configuration.decrypt_tool_credentials(credentials)
  133. return builtin_tool.fork_tool_runtime(
  134. runtime={
  135. "tenant_id": tenant_id,
  136. "credentials": decrypted_credentials,
  137. "runtime_parameters": {},
  138. "invoke_from": invoke_from,
  139. "tool_invoke_from": tool_invoke_from,
  140. }
  141. )
  142. elif provider_type == "api":
  143. if tenant_id is None:
  144. raise ValueError("tenant id is required for api provider")
  145. api_provider, credentials = cls.get_api_provider_controller(tenant_id, provider_id)
  146. # decrypt the credentials
  147. tool_configuration = ToolConfigurationManager(tenant_id=tenant_id, provider_controller=api_provider)
  148. decrypted_credentials = tool_configuration.decrypt_tool_credentials(credentials)
  149. return api_provider.get_tool(tool_name).fork_tool_runtime(
  150. runtime={
  151. "tenant_id": tenant_id,
  152. "credentials": decrypted_credentials,
  153. "invoke_from": invoke_from,
  154. "tool_invoke_from": tool_invoke_from,
  155. }
  156. )
  157. elif provider_type == "workflow":
  158. workflow_provider: Optional[WorkflowToolProvider] = (
  159. db.session.query(WorkflowToolProvider)
  160. .filter(WorkflowToolProvider.tenant_id == tenant_id, WorkflowToolProvider.id == provider_id)
  161. .first()
  162. )
  163. if workflow_provider is None:
  164. raise ToolProviderNotFoundError(f"workflow provider {provider_id} not found")
  165. controller = ToolTransformService.workflow_provider_to_controller(db_provider=workflow_provider)
  166. controller_tools: Optional[list[Tool]] = controller.get_tools(
  167. user_id="", tenant_id=workflow_provider.tenant_id
  168. )
  169. if controller_tools is None or len(controller_tools) == 0:
  170. raise ToolProviderNotFoundError(f"workflow provider {provider_id} not found")
  171. return controller_tools[0].fork_tool_runtime(
  172. runtime={
  173. "tenant_id": tenant_id,
  174. "credentials": {},
  175. "invoke_from": invoke_from,
  176. "tool_invoke_from": tool_invoke_from,
  177. }
  178. )
  179. elif provider_type == "app":
  180. raise NotImplementedError("app provider not implemented")
  181. else:
  182. raise ToolProviderNotFoundError(f"provider type {provider_type} not found")
  183. @classmethod
  184. def _init_runtime_parameter(cls, parameter_rule: ToolParameter, parameters: dict):
  185. """
  186. init runtime parameter
  187. """
  188. parameter_value = parameters.get(parameter_rule.name)
  189. if not parameter_value and parameter_value != 0:
  190. # get default value
  191. parameter_value = parameter_rule.default
  192. if not parameter_value and parameter_rule.required:
  193. raise ValueError(f"tool parameter {parameter_rule.name} not found in tool config")
  194. if parameter_rule.type == ToolParameter.ToolParameterType.SELECT:
  195. # check if tool_parameter_config in options
  196. options = [x.value for x in parameter_rule.options or []]
  197. if parameter_value is not None and parameter_value not in options:
  198. raise ValueError(
  199. f"tool parameter {parameter_rule.name} value {parameter_value} not in options {options}"
  200. )
  201. return parameter_rule.type.cast_value(parameter_value)
  202. @classmethod
  203. def get_agent_tool_runtime(
  204. cls, tenant_id: str, app_id: str, agent_tool: AgentToolEntity, invoke_from: InvokeFrom = InvokeFrom.DEBUGGER
  205. ) -> Tool:
  206. """
  207. get the agent tool runtime
  208. """
  209. tool_entity = cls.get_tool_runtime(
  210. provider_type=agent_tool.provider_type,
  211. provider_id=agent_tool.provider_id,
  212. tool_name=agent_tool.tool_name,
  213. tenant_id=tenant_id,
  214. invoke_from=invoke_from,
  215. tool_invoke_from=ToolInvokeFrom.AGENT,
  216. )
  217. runtime_parameters = {}
  218. parameters = tool_entity.get_all_runtime_parameters()
  219. for parameter in parameters:
  220. # check file types
  221. if (
  222. parameter.type
  223. in {
  224. ToolParameter.ToolParameterType.SYSTEM_FILES,
  225. ToolParameter.ToolParameterType.FILE,
  226. ToolParameter.ToolParameterType.FILES,
  227. }
  228. and parameter.required
  229. ):
  230. raise ValueError(f"file type parameter {parameter.name} not supported in agent")
  231. if parameter.form == ToolParameter.ToolParameterForm.FORM:
  232. # save tool parameter to tool entity memory
  233. value = cls._init_runtime_parameter(parameter, agent_tool.tool_parameters)
  234. runtime_parameters[parameter.name] = value
  235. # decrypt runtime parameters
  236. encryption_manager = ToolParameterConfigurationManager(
  237. tenant_id=tenant_id,
  238. tool_runtime=tool_entity,
  239. provider_name=agent_tool.provider_id,
  240. provider_type=agent_tool.provider_type,
  241. identity_id=f"AGENT.{app_id}",
  242. )
  243. runtime_parameters = encryption_manager.decrypt_tool_parameters(runtime_parameters)
  244. if tool_entity.runtime is None or tool_entity.runtime.runtime_parameters is None:
  245. raise ValueError("runtime not found or runtime parameters not found")
  246. tool_entity.runtime.runtime_parameters.update(runtime_parameters)
  247. return tool_entity
  248. @classmethod
  249. def get_workflow_tool_runtime(
  250. cls,
  251. tenant_id: str,
  252. app_id: str,
  253. node_id: str,
  254. workflow_tool: "ToolEntity",
  255. invoke_from: InvokeFrom = InvokeFrom.DEBUGGER,
  256. ) -> Tool:
  257. """
  258. get the workflow tool runtime
  259. """
  260. tool_entity = cls.get_tool_runtime(
  261. provider_type=workflow_tool.provider_type,
  262. provider_id=workflow_tool.provider_id,
  263. tool_name=workflow_tool.tool_name,
  264. tenant_id=tenant_id,
  265. invoke_from=invoke_from,
  266. tool_invoke_from=ToolInvokeFrom.WORKFLOW,
  267. )
  268. runtime_parameters = {}
  269. parameters = tool_entity.get_all_runtime_parameters()
  270. for parameter in parameters:
  271. # save tool parameter to tool entity memory
  272. if parameter.form == ToolParameter.ToolParameterForm.FORM:
  273. value = cls._init_runtime_parameter(parameter, workflow_tool.tool_configurations)
  274. runtime_parameters[parameter.name] = value
  275. # decrypt runtime parameters
  276. encryption_manager = ToolParameterConfigurationManager(
  277. tenant_id=tenant_id,
  278. tool_runtime=tool_entity,
  279. provider_name=workflow_tool.provider_id,
  280. provider_type=workflow_tool.provider_type,
  281. identity_id=f"WORKFLOW.{app_id}.{node_id}",
  282. )
  283. if runtime_parameters:
  284. runtime_parameters = encryption_manager.decrypt_tool_parameters(runtime_parameters)
  285. if tool_entity.runtime is None or tool_entity.runtime.runtime_parameters is None:
  286. raise ValueError("runtime not found or runtime parameters not found")
  287. tool_entity.runtime.runtime_parameters.update(runtime_parameters)
  288. return tool_entity
  289. @classmethod
  290. def get_builtin_provider_icon(cls, provider: str) -> tuple[str, str]:
  291. """
  292. get the absolute path of the icon of the builtin provider
  293. :param provider: the name of the provider
  294. :return: the absolute path of the icon, the mime type of the icon
  295. """
  296. # get provider
  297. provider_controller = cls.get_builtin_provider(provider)
  298. if provider_controller.identity is None:
  299. raise ToolProviderNotFoundError(f"builtin provider {provider} not found")
  300. absolute_path = path.join(
  301. path.dirname(path.realpath(__file__)),
  302. "provider",
  303. "builtin",
  304. provider,
  305. "_assets",
  306. provider_controller.identity.icon,
  307. )
  308. # check if the icon exists
  309. if not path.exists(absolute_path):
  310. raise ToolProviderNotFoundError(f"builtin provider {provider} icon not found")
  311. # get the mime type
  312. mime_type, _ = mimetypes.guess_type(absolute_path)
  313. mime_type = mime_type or "application/octet-stream"
  314. return absolute_path, mime_type
  315. @classmethod
  316. def list_builtin_providers(cls) -> Generator[BuiltinToolProviderController, None, None]:
  317. # use cache first
  318. if cls._builtin_providers_loaded:
  319. yield from list(cls._builtin_providers.values())
  320. return
  321. with cls._builtin_provider_lock:
  322. if cls._builtin_providers_loaded:
  323. yield from list(cls._builtin_providers.values())
  324. return
  325. yield from cls._list_builtin_providers()
  326. @classmethod
  327. def _list_builtin_providers(cls) -> Generator[BuiltinToolProviderController, None, None]:
  328. """
  329. list all the builtin providers
  330. """
  331. for provider in listdir(path.join(path.dirname(path.realpath(__file__)), "provider", "builtin")):
  332. if provider.startswith("__"):
  333. continue
  334. if path.isdir(path.join(path.dirname(path.realpath(__file__)), "provider", "builtin", provider)):
  335. if provider.startswith("__"):
  336. continue
  337. # init provider
  338. try:
  339. provider_class = load_single_subclass_from_source(
  340. module_name=f"core.tools.provider.builtin.{provider}.{provider}",
  341. script_path=path.join(
  342. path.dirname(path.realpath(__file__)), "provider", "builtin", provider, f"{provider}.py"
  343. ),
  344. parent_type=BuiltinToolProviderController,
  345. )
  346. provider_controller: BuiltinToolProviderController = provider_class()
  347. if provider_controller.identity is None:
  348. continue
  349. cls._builtin_providers[provider_controller.identity.name] = provider_controller
  350. for tool in provider_controller.get_tools() or []:
  351. if tool.identity is None:
  352. continue
  353. cls._builtin_tools_labels[tool.identity.name] = tool.identity.label
  354. yield provider_controller
  355. except Exception as e:
  356. logger.exception(f"load builtin provider {provider}")
  357. continue
  358. # set builtin providers loaded
  359. cls._builtin_providers_loaded = True
  360. @classmethod
  361. def load_builtin_providers_cache(cls):
  362. for _ in cls.list_builtin_providers():
  363. pass
  364. @classmethod
  365. def clear_builtin_providers_cache(cls):
  366. cls._builtin_providers = {}
  367. cls._builtin_providers_loaded = False
  368. @classmethod
  369. def get_tool_label(cls, tool_name: str) -> Union[I18nObject, None]:
  370. """
  371. get the tool label
  372. :param tool_name: the name of the tool
  373. :return: the label of the tool
  374. """
  375. if len(cls._builtin_tools_labels) == 0:
  376. # init the builtin providers
  377. cls.load_builtin_providers_cache()
  378. if tool_name not in cls._builtin_tools_labels:
  379. return None
  380. return cls._builtin_tools_labels[tool_name]
  381. @classmethod
  382. def user_list_providers(
  383. cls, user_id: str, tenant_id: str, typ: UserToolProviderTypeLiteral
  384. ) -> list[UserToolProvider]:
  385. result_providers: dict[str, UserToolProvider] = {}
  386. filters = []
  387. if not typ:
  388. filters.extend(["builtin", "api", "workflow"])
  389. else:
  390. filters.append(typ)
  391. if "builtin" in filters:
  392. # get builtin providers
  393. builtin_providers = cls.list_builtin_providers()
  394. # get db builtin providers
  395. db_builtin_providers: list[BuiltinToolProvider] = (
  396. db.session.query(BuiltinToolProvider).filter(BuiltinToolProvider.tenant_id == tenant_id).all()
  397. )
  398. find_db_builtin_provider = lambda provider: next(
  399. (x for x in db_builtin_providers if x.provider == provider), None
  400. )
  401. # append builtin providers
  402. for provider in builtin_providers:
  403. # handle include, exclude
  404. if provider.identity is None:
  405. continue
  406. if is_filtered(
  407. include_set=cast(set[str], dify_config.POSITION_TOOL_INCLUDES_SET),
  408. exclude_set=cast(set[str], dify_config.POSITION_TOOL_EXCLUDES_SET),
  409. data=provider,
  410. name_func=lambda x: x.identity.name,
  411. ):
  412. continue
  413. user_provider = ToolTransformService.builtin_provider_to_user_provider(
  414. provider_controller=provider,
  415. db_provider=find_db_builtin_provider(provider.identity.name),
  416. decrypt_credentials=False,
  417. )
  418. result_providers[provider.identity.name] = user_provider
  419. # get db api providers
  420. if "api" in filters:
  421. db_api_providers: list[ApiToolProvider] = (
  422. db.session.query(ApiToolProvider).filter(ApiToolProvider.tenant_id == tenant_id).all()
  423. )
  424. api_provider_controllers: list[dict[str, Any]] = [
  425. {"provider": provider, "controller": ToolTransformService.api_provider_to_controller(provider)}
  426. for provider in db_api_providers
  427. ]
  428. # get labels
  429. labels = ToolLabelManager.get_tools_labels([x["controller"] for x in api_provider_controllers])
  430. for api_provider_controller in api_provider_controllers:
  431. user_provider = ToolTransformService.api_provider_to_user_provider(
  432. provider_controller=api_provider_controller["controller"],
  433. db_provider=api_provider_controller["provider"],
  434. decrypt_credentials=False,
  435. labels=labels.get(api_provider_controller["controller"].provider_id, []),
  436. )
  437. result_providers[f"api_provider.{user_provider.name}"] = user_provider
  438. if "workflow" in filters:
  439. # get workflow providers
  440. workflow_providers: list[WorkflowToolProvider] = (
  441. db.session.query(WorkflowToolProvider).filter(WorkflowToolProvider.tenant_id == tenant_id).all()
  442. )
  443. workflow_provider_controllers: list[WorkflowToolProviderController] = []
  444. for provider in workflow_providers:
  445. try:
  446. workflow_provider_controllers.append(
  447. ToolTransformService.workflow_provider_to_controller(db_provider=provider)
  448. )
  449. except Exception as e:
  450. # app has been deleted
  451. pass
  452. labels = ToolLabelManager.get_tools_labels(
  453. [cast(ToolProviderController, controller) for controller in workflow_provider_controllers]
  454. )
  455. for provider_controller in workflow_provider_controllers:
  456. user_provider = ToolTransformService.workflow_provider_to_user_provider(
  457. provider_controller=provider_controller,
  458. labels=labels.get(provider_controller.provider_id, []),
  459. )
  460. result_providers[f"workflow_provider.{user_provider.name}"] = user_provider
  461. return BuiltinToolProviderSort.sort(list(result_providers.values()))
  462. @classmethod
  463. def get_api_provider_controller(
  464. cls, tenant_id: str, provider_id: str
  465. ) -> tuple[ApiToolProviderController, dict[str, Any]]:
  466. """
  467. get the api provider
  468. :param provider_name: the name of the provider
  469. :return: the provider controller, the credentials
  470. """
  471. provider: Optional[ApiToolProvider] = (
  472. db.session.query(ApiToolProvider)
  473. .filter(
  474. ApiToolProvider.id == provider_id,
  475. ApiToolProvider.tenant_id == tenant_id,
  476. )
  477. .first()
  478. )
  479. if provider is None:
  480. raise ToolProviderNotFoundError(f"api provider {provider_id} not found")
  481. controller = ApiToolProviderController.from_db(
  482. provider,
  483. ApiProviderAuthType.API_KEY if provider.credentials["auth_type"] == "api_key" else ApiProviderAuthType.NONE,
  484. )
  485. controller.load_bundled_tools(provider.tools)
  486. return controller, provider.credentials
  487. @classmethod
  488. def user_get_api_provider(cls, provider: str, tenant_id: str) -> dict:
  489. """
  490. get api provider
  491. """
  492. """
  493. get tool provider
  494. """
  495. provider_name = provider
  496. provider_tool: Optional[ApiToolProvider] = (
  497. db.session.query(ApiToolProvider)
  498. .filter(
  499. ApiToolProvider.tenant_id == tenant_id,
  500. ApiToolProvider.name == provider,
  501. )
  502. .first()
  503. )
  504. if provider_tool is None:
  505. raise ValueError(f"you have not added provider {provider_name}")
  506. try:
  507. credentials = json.loads(provider_tool.credentials_str) or {}
  508. except:
  509. credentials = {}
  510. # package tool provider controller
  511. controller = ApiToolProviderController.from_db(
  512. provider_tool,
  513. ApiProviderAuthType.API_KEY if credentials["auth_type"] == "api_key" else ApiProviderAuthType.NONE,
  514. )
  515. # init tool configuration
  516. tool_configuration = ToolConfigurationManager(tenant_id=tenant_id, provider_controller=controller)
  517. decrypted_credentials = tool_configuration.decrypt_tool_credentials(credentials)
  518. masked_credentials = tool_configuration.mask_tool_credentials(decrypted_credentials)
  519. try:
  520. icon = json.loads(provider_tool.icon)
  521. except:
  522. icon = {"background": "#252525", "content": "\ud83d\ude01"}
  523. # add tool labels
  524. labels = ToolLabelManager.get_tool_labels(controller)
  525. return cast(
  526. dict,
  527. jsonable_encoder(
  528. {
  529. "schema_type": provider_tool.schema_type,
  530. "schema": provider_tool.schema,
  531. "tools": provider_tool.tools,
  532. "icon": icon,
  533. "description": provider_tool.description,
  534. "credentials": masked_credentials,
  535. "privacy_policy": provider_tool.privacy_policy,
  536. "custom_disclaimer": provider_tool.custom_disclaimer,
  537. "labels": labels,
  538. }
  539. ),
  540. )
  541. @classmethod
  542. def get_tool_icon(cls, tenant_id: str, provider_type: str, provider_id: str) -> Union[str, dict]:
  543. """
  544. get the tool icon
  545. :param tenant_id: the id of the tenant
  546. :param provider_type: the type of the provider
  547. :param provider_id: the id of the provider
  548. :return:
  549. """
  550. provider_type = provider_type
  551. provider_id = provider_id
  552. provider: Optional[Union[BuiltinToolProvider, ApiToolProvider, WorkflowToolProvider]] = None
  553. if provider_type == "builtin":
  554. return (
  555. dify_config.CONSOLE_API_URL
  556. + "/console/api/workspaces/current/tool-provider/builtin/"
  557. + provider_id
  558. + "/icon"
  559. )
  560. elif provider_type == "api":
  561. try:
  562. provider = (
  563. db.session.query(ApiToolProvider)
  564. .filter(ApiToolProvider.tenant_id == tenant_id, ApiToolProvider.id == provider_id)
  565. .first()
  566. )
  567. if provider is None:
  568. raise ToolProviderNotFoundError(f"api provider {provider_id} not found")
  569. icon = json.loads(provider.icon)
  570. if isinstance(icon, (str, dict)):
  571. return icon
  572. return {"background": "#252525", "content": "\ud83d\ude01"}
  573. except:
  574. return {"background": "#252525", "content": "\ud83d\ude01"}
  575. elif provider_type == "workflow":
  576. provider = (
  577. db.session.query(WorkflowToolProvider)
  578. .filter(WorkflowToolProvider.tenant_id == tenant_id, WorkflowToolProvider.id == provider_id)
  579. .first()
  580. )
  581. if provider is None:
  582. raise ToolProviderNotFoundError(f"workflow provider {provider_id} not found")
  583. try:
  584. icon = json.loads(provider.icon)
  585. if isinstance(icon, (str, dict)):
  586. return icon
  587. return {"background": "#252525", "content": "\ud83d\ude01"}
  588. except:
  589. return {"background": "#252525", "content": "\ud83d\ude01"}
  590. else:
  591. raise ValueError(f"provider type {provider_type} not found")
  592. # preload builtin tool providers
  593. Thread(target=ToolManager.load_builtin_providers_cache, name="pre_load_builtin_providers_cache", daemon=True).start()