app_generate_service.py 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190
  1. from collections.abc import Generator, Mapping
  2. from typing import Any, Union
  3. from openai._exceptions import RateLimitError
  4. from configs import dify_config
  5. from core.app.apps.advanced_chat.app_generator import AdvancedChatAppGenerator
  6. from core.app.apps.agent_chat.app_generator import AgentChatAppGenerator
  7. from core.app.apps.chat.app_generator import ChatAppGenerator
  8. from core.app.apps.completion.app_generator import CompletionAppGenerator
  9. from core.app.apps.workflow.app_generator import WorkflowAppGenerator
  10. from core.app.entities.app_invoke_entities import InvokeFrom
  11. from core.app.features.rate_limiting import RateLimit
  12. from models.model import Account, App, AppMode, EndUser
  13. from models.workflow import Workflow
  14. from services.errors.llm import InvokeRateLimitError
  15. from services.workflow_service import WorkflowService
  16. class AppGenerateService:
  17. @classmethod
  18. def generate(
  19. cls,
  20. app_model: App,
  21. user: Union[Account, EndUser],
  22. args: Mapping[str, Any],
  23. invoke_from: InvokeFrom,
  24. streaming: bool = True,
  25. ):
  26. """
  27. App Content Generate
  28. :param app_model: app model
  29. :param user: user
  30. :param args: args
  31. :param invoke_from: invoke from
  32. :param streaming: streaming
  33. :return:
  34. """
  35. max_active_request = AppGenerateService._get_max_active_requests(app_model)
  36. rate_limit = RateLimit(app_model.id, max_active_request)
  37. request_id = RateLimit.gen_request_key()
  38. try:
  39. request_id = rate_limit.enter(request_id)
  40. if app_model.mode == AppMode.COMPLETION.value:
  41. return rate_limit.generate(
  42. generator=CompletionAppGenerator().generate(
  43. app_model=app_model,
  44. user=user,
  45. args=args,
  46. invoke_from=invoke_from,
  47. streaming=streaming,
  48. ),
  49. request_id=request_id,
  50. )
  51. elif app_model.mode == AppMode.AGENT_CHAT.value or app_model.is_agent:
  52. generator = AgentChatAppGenerator().generate(
  53. app_model=app_model,
  54. user=user,
  55. args=args,
  56. invoke_from=invoke_from,
  57. streaming=streaming,
  58. )
  59. return rate_limit.generate(
  60. generator=generator,
  61. request_id=request_id,
  62. )
  63. elif app_model.mode == AppMode.CHAT.value:
  64. return rate_limit.generate(
  65. generator=ChatAppGenerator().generate(
  66. app_model=app_model,
  67. user=user,
  68. args=args,
  69. invoke_from=invoke_from,
  70. streaming=streaming,
  71. ),
  72. request_id=request_id,
  73. )
  74. elif app_model.mode == AppMode.ADVANCED_CHAT.value:
  75. workflow = cls._get_workflow(app_model, invoke_from)
  76. return rate_limit.generate(
  77. generator=AdvancedChatAppGenerator().generate(
  78. app_model=app_model,
  79. workflow=workflow,
  80. user=user,
  81. args=args,
  82. invoke_from=invoke_from,
  83. streaming=streaming,
  84. ),
  85. request_id=request_id,
  86. )
  87. elif app_model.mode == AppMode.WORKFLOW.value:
  88. workflow = cls._get_workflow(app_model, invoke_from)
  89. generator = WorkflowAppGenerator().generate(
  90. app_model=app_model,
  91. workflow=workflow,
  92. user=user,
  93. args=args,
  94. invoke_from=invoke_from,
  95. streaming=streaming,
  96. call_depth=0,
  97. workflow_thread_pool_id=None,
  98. )
  99. return rate_limit.generate(
  100. generator=generator,
  101. request_id=request_id,
  102. )
  103. else:
  104. raise ValueError(f"Invalid app mode {app_model.mode}")
  105. except RateLimitError as e:
  106. raise InvokeRateLimitError(str(e))
  107. except Exception:
  108. rate_limit.exit(request_id)
  109. raise
  110. finally:
  111. if not streaming:
  112. rate_limit.exit(request_id)
  113. @staticmethod
  114. def _get_max_active_requests(app_model: App) -> int:
  115. max_active_requests = app_model.max_active_requests
  116. if max_active_requests is None:
  117. max_active_requests = int(dify_config.APP_MAX_ACTIVE_REQUESTS)
  118. return max_active_requests
  119. @classmethod
  120. def generate_single_iteration(cls, app_model: App, user: Account, node_id: str, args: Any, streaming: bool = True):
  121. if app_model.mode == AppMode.ADVANCED_CHAT.value:
  122. workflow = cls._get_workflow(app_model, InvokeFrom.DEBUGGER)
  123. return AdvancedChatAppGenerator().single_iteration_generate(
  124. app_model=app_model,
  125. workflow=workflow,
  126. node_id=node_id,
  127. user=user,
  128. args=args,
  129. streaming=streaming,
  130. )
  131. elif app_model.mode == AppMode.WORKFLOW.value:
  132. workflow = cls._get_workflow(app_model, InvokeFrom.DEBUGGER)
  133. return WorkflowAppGenerator().single_iteration_generate(
  134. app_model=app_model, workflow=workflow, node_id=node_id, user=user, args=args, streaming=streaming
  135. )
  136. else:
  137. raise ValueError(f"Invalid app mode {app_model.mode}")
  138. @classmethod
  139. def generate_more_like_this(
  140. cls,
  141. app_model: App,
  142. user: Union[Account, EndUser],
  143. message_id: str,
  144. invoke_from: InvokeFrom,
  145. streaming: bool = True,
  146. ) -> Union[Mapping, Generator]:
  147. """
  148. Generate more like this
  149. :param app_model: app model
  150. :param user: user
  151. :param message_id: message id
  152. :param invoke_from: invoke from
  153. :param streaming: streaming
  154. :return:
  155. """
  156. return CompletionAppGenerator().generate_more_like_this(
  157. app_model=app_model, message_id=message_id, user=user, invoke_from=invoke_from, stream=streaming
  158. )
  159. @classmethod
  160. def _get_workflow(cls, app_model: App, invoke_from: InvokeFrom) -> Workflow:
  161. """
  162. Get workflow
  163. :param app_model: app model
  164. :param invoke_from: invoke from
  165. :return:
  166. """
  167. workflow_service = WorkflowService()
  168. if invoke_from == InvokeFrom.DEBUGGER:
  169. # fetch draft workflow by app_model
  170. workflow = workflow_service.get_draft_workflow(app_model=app_model)
  171. if not workflow:
  172. raise ValueError("Workflow not initialized")
  173. else:
  174. # fetch published workflow by app_model
  175. workflow = workflow_service.get_published_workflow(app_model=app_model)
  176. if not workflow:
  177. raise ValueError("Workflow not published")
  178. return workflow