123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204 |
- import json
- import logging
- from collections.abc import Callable, Generator
- from typing import Optional, TypeVar
- import requests
- from pydantic import BaseModel
- from yarl import URL
- from configs import dify_config
- from core.model_runtime.errors.invoke import (
- InvokeAuthorizationError,
- InvokeBadRequestError,
- InvokeConnectionError,
- InvokeRateLimitError,
- InvokeServerUnavailableError,
- )
- from core.plugin.entities.plugin_daemon import PluginDaemonBasicResponse, PluginDaemonError, PluginDaemonInnerError
- plugin_daemon_inner_api_baseurl = dify_config.PLUGIN_API_URL
- plugin_daemon_inner_api_key = dify_config.PLUGIN_API_KEY
- T = TypeVar("T", bound=(BaseModel | dict | list | bool | str))
- logger = logging.getLogger(__name__)
- class BasePluginManager:
- def _request(
- self,
- method: str,
- path: str,
- headers: dict | None = None,
- data: bytes | dict | str | None = None,
- params: dict | None = None,
- files: dict | None = None,
- stream: bool = False,
- ) -> requests.Response:
- """
- Make a request to the plugin daemon inner API.
- """
- url = URL(str(plugin_daemon_inner_api_baseurl)) / path
- headers = headers or {}
- headers["X-Api-Key"] = plugin_daemon_inner_api_key
- headers["Accept-Encoding"] = "gzip, deflate, br"
- if headers.get("Content-Type") == "application/json" and isinstance(data, dict):
- data = json.dumps(data)
- try:
- response = requests.request(
- method=method, url=str(url), headers=headers, data=data, params=params, stream=stream, files=files
- )
- except requests.exceptions.ConnectionError as e:
- logger.exception(f"Request to Plugin Daemon Service failed: {e}")
- raise PluginDaemonInnerError(code=-500, message="Request to Plugin Daemon Service failed")
- return response
- def _stream_request(
- self,
- method: str,
- path: str,
- params: dict | None = None,
- headers: dict | None = None,
- data: bytes | dict | None = None,
- files: dict | None = None,
- ) -> Generator[bytes, None, None]:
- """
- Make a stream request to the plugin daemon inner API
- """
- response = self._request(method, path, headers, data, params, files, stream=True)
- for line in response.iter_lines():
- line = line.decode("utf-8").strip()
- if line.startswith("data:"):
- line = line[5:].strip()
- if line:
- yield line
- def _stream_request_with_model(
- self,
- method: str,
- path: str,
- type: type[T],
- headers: dict | None = None,
- data: bytes | dict | None = None,
- params: dict | None = None,
- files: dict | None = None,
- ) -> Generator[T, None, None]:
- """
- Make a stream request to the plugin daemon inner API and yield the response as a model.
- """
- for line in self._stream_request(method, path, params, headers, data, files):
- yield type(**json.loads(line))
- def _request_with_model(
- self,
- method: str,
- path: str,
- type: type[T],
- headers: dict | None = None,
- data: bytes | None = None,
- params: dict | None = None,
- files: dict | None = None,
- ) -> T:
- """
- Make a request to the plugin daemon inner API and return the response as a model.
- """
- response = self._request(method, path, headers, data, params, files)
- return type(**response.json())
- def _request_with_plugin_daemon_response(
- self,
- method: str,
- path: str,
- type: type[T],
- headers: dict | None = None,
- data: bytes | dict | None = None,
- params: dict | None = None,
- files: dict | None = None,
- transformer: Callable[[dict], dict] | None = None,
- ) -> T:
- """
- Make a request to the plugin daemon inner API and return the response as a model.
- """
- response = self._request(method, path, headers, data, params, files)
- json_response = response.json()
- if transformer:
- json_response = transformer(json_response)
- rep = PluginDaemonBasicResponse[type](**json_response)
- if rep.code != 0:
- if rep.code == -500:
- try:
- error = PluginDaemonError(**json.loads(rep.message))
- except Exception as e:
- raise ValueError(f"got error from plugin daemon: {rep.message}, code: {rep.code}")
- self._handle_plugin_daemon_error(error.error_type, error.message, error.args)
- raise ValueError(f"got error from plugin daemon: {rep.message}, code: {rep.code}")
- if rep.data is None:
- raise ValueError("got empty data from plugin daemon")
- return rep.data
- def _request_with_plugin_daemon_response_stream(
- self,
- method: str,
- path: str,
- type: type[T],
- headers: dict | None = None,
- data: bytes | dict | None = None,
- params: dict | None = None,
- files: dict | None = None,
- ) -> Generator[T, None, None]:
- """
- Make a stream request to the plugin daemon inner API and yield the response as a model.
- """
- for line in self._stream_request(method, path, params, headers, data, files):
- line_data = None
- try:
- line_data = json.loads(line)
- rep = PluginDaemonBasicResponse[type](**line_data)
- except Exception as e:
- # TODO modify this when line_data has code and message
- if line_data and "error" in line_data:
- raise ValueError(line_data["error"])
- else:
- raise ValueError(line)
- if rep.code != 0:
- if rep.code == -500:
- try:
- error = PluginDaemonError(**json.loads(rep.message))
- except Exception as e:
- raise PluginDaemonInnerError(code=rep.code, message=rep.message)
- self._handle_plugin_daemon_error(error.error_type, error.message, error.args)
- raise ValueError(f"got error from plugin daemon: {rep.message}, code: {rep.code}")
- if rep.data is None:
- raise ValueError("got empty data from plugin daemon")
- yield rep.data
- def _handle_plugin_daemon_error(self, error_type: str, message: str, args: Optional[dict] = None):
- """
- handle the error from plugin daemon
- """
- args = args or {}
- if error_type == PluginDaemonInnerError.__name__:
- raise PluginDaemonInnerError(code=-500, message=message)
- elif error_type == InvokeRateLimitError.__name__:
- raise InvokeRateLimitError(description=args.get("description"))
- elif error_type == InvokeAuthorizationError.__name__:
- raise InvokeAuthorizationError(description=args.get("description"))
- elif error_type == InvokeBadRequestError.__name__:
- raise InvokeBadRequestError(description=args.get("description"))
- elif error_type == InvokeConnectionError.__name__:
- raise InvokeConnectionError(description=args.get("description"))
- elif error_type == InvokeServerUnavailableError.__name__:
- raise InvokeServerUnavailableError(description=args.get("description"))
- else:
- raise ValueError(f"got unknown error from plugin daemon: {error_type}, message: {message}, args: {args}")
|