recommended_app_service.py 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251
  1. import json
  2. import logging
  3. from os import path
  4. from typing import Optional
  5. import requests
  6. from flask import current_app
  7. from constants.languages import languages
  8. from extensions.ext_database import db
  9. from models.model import App, RecommendedApp
  10. from services.app_service import AppService
  11. logger = logging.getLogger(__name__)
  12. class RecommendedAppService:
  13. builtin_data: Optional[dict] = None
  14. @classmethod
  15. def get_recommended_apps_and_categories(cls, language: str) -> dict:
  16. """
  17. Get recommended apps and categories.
  18. :param language: language
  19. :return:
  20. """
  21. mode = current_app.config.get('HOSTED_FETCH_APP_TEMPLATES_MODE', 'remote')
  22. if mode == 'remote':
  23. try:
  24. result = cls._fetch_recommended_apps_from_dify_official(language)
  25. except Exception as e:
  26. logger.warning(f'fetch recommended apps from dify official failed: {e}, switch to built-in.')
  27. result = cls._fetch_recommended_apps_from_builtin(language)
  28. elif mode == 'db':
  29. result = cls._fetch_recommended_apps_from_db(language)
  30. elif mode == 'builtin':
  31. result = cls._fetch_recommended_apps_from_builtin(language)
  32. else:
  33. raise ValueError(f'invalid fetch recommended apps mode: {mode}')
  34. if not result.get('recommended_apps') and language != 'en-US':
  35. result = cls._fetch_recommended_apps_from_builtin('en-US')
  36. return result
  37. @classmethod
  38. def _fetch_recommended_apps_from_db(cls, language: str) -> dict:
  39. """
  40. Fetch recommended apps from db.
  41. :param language: language
  42. :return:
  43. """
  44. recommended_apps = db.session.query(RecommendedApp).filter(
  45. RecommendedApp.is_listed == True,
  46. RecommendedApp.language == language
  47. ).all()
  48. if len(recommended_apps) == 0:
  49. recommended_apps = db.session.query(RecommendedApp).filter(
  50. RecommendedApp.is_listed == True,
  51. RecommendedApp.language == languages[0]
  52. ).all()
  53. categories = set()
  54. recommended_apps_result = []
  55. for recommended_app in recommended_apps:
  56. app = recommended_app.app
  57. if not app or not app.is_public:
  58. continue
  59. site = app.site
  60. if not site:
  61. continue
  62. recommended_app_result = {
  63. 'id': recommended_app.id,
  64. 'app': {
  65. 'id': app.id,
  66. 'name': app.name,
  67. 'mode': app.mode,
  68. 'icon': app.icon,
  69. 'icon_background': app.icon_background
  70. },
  71. 'app_id': recommended_app.app_id,
  72. 'description': site.description,
  73. 'copyright': site.copyright,
  74. 'privacy_policy': site.privacy_policy,
  75. 'category': recommended_app.category,
  76. 'position': recommended_app.position,
  77. 'is_listed': recommended_app.is_listed
  78. }
  79. recommended_apps_result.append(recommended_app_result)
  80. categories.add(recommended_app.category) # add category to categories
  81. return {'recommended_apps': recommended_apps_result, 'categories': list(categories)}
  82. @classmethod
  83. def _fetch_recommended_apps_from_dify_official(cls, language: str) -> dict:
  84. """
  85. Fetch recommended apps from dify official.
  86. :param language: language
  87. :return:
  88. """
  89. domain = current_app.config.get('HOSTED_FETCH_APP_TEMPLATES_REMOTE_DOMAIN', 'https://tmpl.dify.ai')
  90. url = f'{domain}/apps?language={language}'
  91. response = requests.get(url, timeout=(3, 10))
  92. if response.status_code != 200:
  93. raise ValueError(f'fetch recommended apps failed, status code: {response.status_code}')
  94. return response.json()
  95. @classmethod
  96. def _fetch_recommended_apps_from_builtin(cls, language: str) -> dict:
  97. """
  98. Fetch recommended apps from builtin.
  99. :param language: language
  100. :return:
  101. """
  102. builtin_data = cls._get_builtin_data()
  103. return builtin_data.get('recommended_apps', {}).get(language)
  104. @classmethod
  105. def get_recommend_app_detail(cls, app_id: str) -> Optional[dict]:
  106. """
  107. Get recommend app detail.
  108. :param app_id: app id
  109. :return:
  110. """
  111. mode = current_app.config.get('HOSTED_FETCH_APP_TEMPLATES_MODE', 'remote')
  112. if mode == 'remote':
  113. try:
  114. result = cls._fetch_recommended_app_detail_from_dify_official(app_id)
  115. except Exception as e:
  116. logger.warning(f'fetch recommended app detail from dify official failed: {e}, switch to built-in.')
  117. result = cls._fetch_recommended_app_detail_from_builtin(app_id)
  118. elif mode == 'db':
  119. result = cls._fetch_recommended_app_detail_from_db(app_id)
  120. elif mode == 'builtin':
  121. result = cls._fetch_recommended_app_detail_from_builtin(app_id)
  122. else:
  123. raise ValueError(f'invalid fetch recommended app detail mode: {mode}')
  124. return result
  125. @classmethod
  126. def _fetch_recommended_app_detail_from_dify_official(cls, app_id: str) -> Optional[dict]:
  127. """
  128. Fetch recommended app detail from dify official.
  129. :param app_id: App ID
  130. :return:
  131. """
  132. domain = current_app.config.get('HOSTED_FETCH_APP_TEMPLATES_REMOTE_DOMAIN', 'https://tmpl.dify.ai')
  133. url = f'{domain}/apps/{app_id}'
  134. response = requests.get(url, timeout=(3, 10))
  135. if response.status_code != 200:
  136. return None
  137. return response.json()
  138. @classmethod
  139. def _fetch_recommended_app_detail_from_db(cls, app_id: str) -> Optional[dict]:
  140. """
  141. Fetch recommended app detail from db.
  142. :param app_id: App ID
  143. :return:
  144. """
  145. # is in public recommended list
  146. recommended_app = db.session.query(RecommendedApp).filter(
  147. RecommendedApp.is_listed == True,
  148. RecommendedApp.app_id == app_id
  149. ).first()
  150. if not recommended_app:
  151. return None
  152. # get app detail
  153. app_model = db.session.query(App).filter(App.id == app_id).first()
  154. if not app_model or not app_model.is_public:
  155. return None
  156. app_service = AppService()
  157. export_str = app_service.export_app(app_model)
  158. return {
  159. 'id': app_model.id,
  160. 'name': app_model.name,
  161. 'icon': app_model.icon,
  162. 'icon_background': app_model.icon_background,
  163. 'mode': app_model.mode,
  164. 'export_data': export_str
  165. }
  166. @classmethod
  167. def _fetch_recommended_app_detail_from_builtin(cls, app_id: str) -> Optional[dict]:
  168. """
  169. Fetch recommended app detail from builtin.
  170. :param app_id: App ID
  171. :return:
  172. """
  173. builtin_data = cls._get_builtin_data()
  174. return builtin_data.get('app_details', {}).get(app_id)
  175. @classmethod
  176. def _get_builtin_data(cls) -> dict:
  177. """
  178. Get builtin data.
  179. :return:
  180. """
  181. if cls.builtin_data:
  182. return cls.builtin_data
  183. root_path = current_app.root_path
  184. with open(path.join(root_path, 'constants', 'recommended_apps.json'), encoding='utf-8') as f:
  185. json_data = f.read()
  186. data = json.loads(json_data)
  187. cls.builtin_data = data
  188. return cls.builtin_data
  189. @classmethod
  190. def fetch_all_recommended_apps_and_export_datas(cls):
  191. """
  192. Fetch all recommended apps and export datas
  193. :return:
  194. """
  195. templates = {
  196. "recommended_apps": {},
  197. "app_details": {}
  198. }
  199. for language in languages:
  200. try:
  201. result = cls._fetch_recommended_apps_from_dify_official(language)
  202. except Exception as e:
  203. logger.warning(f'fetch recommended apps from dify official failed: {e}, skip.')
  204. continue
  205. templates['recommended_apps'][language] = result
  206. for recommended_app in result.get('recommended_apps'):
  207. app_id = recommended_app.get('app_id')
  208. # get app detail
  209. app_detail = cls._fetch_recommended_app_detail_from_dify_official(app_id)
  210. if not app_detail:
  211. continue
  212. templates['app_details'][app_id] = app_detail
  213. return templates