| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428 | import base64import enumfrom collections.abc import Mappingfrom enum import Enumfrom typing import Any, Optional, Unionfrom pydantic import BaseModel, ConfigDict, Field, ValidationInfo, field_serializer, field_validator, model_validatorfrom core.entities.provider_entities import ProviderConfigfrom core.plugin.entities.parameters import (    PluginParameter,    PluginParameterOption,    PluginParameterType,    as_normal_type,    cast_parameter_value,    init_frontend_parameter,)from core.tools.entities.common_entities import I18nObjectfrom core.tools.entities.constants import TOOL_SELECTOR_MODEL_IDENTITYclass ToolLabelEnum(Enum):    SEARCH = "search"    IMAGE = "image"    VIDEOS = "videos"    WEATHER = "weather"    FINANCE = "finance"    DESIGN = "design"    TRAVEL = "travel"    SOCIAL = "social"    NEWS = "news"    MEDICAL = "medical"    PRODUCTIVITY = "productivity"    EDUCATION = "education"    BUSINESS = "business"    ENTERTAINMENT = "entertainment"    UTILITIES = "utilities"    OTHER = "other"class ToolProviderType(enum.StrEnum):    """    Enum class for tool provider    """    PLUGIN = "plugin"    BUILT_IN = "builtin"    WORKFLOW = "workflow"    API = "api"    APP = "app"    DATASET_RETRIEVAL = "dataset-retrieval"    @classmethod    def value_of(cls, value: str) -> "ToolProviderType":        """        Get value of given mode.        :param value: mode value        :return: mode        """        for mode in cls:            if mode.value == value:                return mode        raise ValueError(f"invalid mode value {value}")class ApiProviderSchemaType(Enum):    """    Enum class for api provider schema type.    """    OPENAPI = "openapi"    SWAGGER = "swagger"    OPENAI_PLUGIN = "openai_plugin"    OPENAI_ACTIONS = "openai_actions"    @classmethod    def value_of(cls, value: str) -> "ApiProviderSchemaType":        """        Get value of given mode.        :param value: mode value        :return: mode        """        for mode in cls:            if mode.value == value:                return mode        raise ValueError(f"invalid mode value {value}")class ApiProviderAuthType(Enum):    """    Enum class for api provider auth type.    """    NONE = "none"    API_KEY = "api_key"    @classmethod    def value_of(cls, value: str) -> "ApiProviderAuthType":        """        Get value of given mode.        :param value: mode value        :return: mode        """        for mode in cls:            if mode.value == value:                return mode        raise ValueError(f"invalid mode value {value}")class ToolInvokeMessage(BaseModel):    class TextMessage(BaseModel):        text: str    class JsonMessage(BaseModel):        json_object: dict    class BlobMessage(BaseModel):        blob: bytes    class FileMessage(BaseModel):        pass    class VariableMessage(BaseModel):        variable_name: str = Field(..., description="The name of the variable")        variable_value: Any = Field(..., description="The value of the variable")        stream: bool = Field(default=False, description="Whether the variable is streamed")        @model_validator(mode="before")        @classmethod        def transform_variable_value(cls, values) -> Any:            """            Only basic types and lists are allowed.            """            value = values.get("variable_value")            if not isinstance(value, dict | list | str | int | float | bool):                raise ValueError("Only basic types and lists are allowed.")            # if stream is true, the value must be a string            if values.get("stream"):                if not isinstance(value, str):                    raise ValueError("When 'stream' is True, 'variable_value' must be a string.")            return values        @field_validator("variable_name", mode="before")        @classmethod        def transform_variable_name(cls, value: str) -> str:            """            The variable name must be a string.            """            if value in {"json", "text", "files"}:                raise ValueError(f"The variable name '{value}' is reserved.")            return value    class LogMessage(BaseModel):        class LogStatus(Enum):            START = "start"            ERROR = "error"            SUCCESS = "success"        id: str        label: str = Field(..., description="The label of the log")        parent_id: Optional[str] = Field(default=None, description="Leave empty for root log")        error: Optional[str] = Field(default=None, description="The error message")        status: LogStatus = Field(..., description="The status of the log")        data: Mapping[str, Any] = Field(..., description="Detailed log data")        metadata: Optional[Mapping[str, Any]] = Field(default=None, description="The metadata of the log")    class MessageType(Enum):        TEXT = "text"        IMAGE = "image"        LINK = "link"        BLOB = "blob"        JSON = "json"        IMAGE_LINK = "image_link"        BINARY_LINK = "binary_link"        VARIABLE = "variable"        FILE = "file"        LOG = "log"    type: MessageType = MessageType.TEXT    """        plain text, image url or link url    """    message: JsonMessage | TextMessage | BlobMessage | LogMessage | FileMessage | None | VariableMessage    meta: dict[str, Any] | None = None    @field_validator("message", mode="before")    @classmethod    def decode_blob_message(cls, v):        if isinstance(v, dict) and "blob" in v:            try:                v["blob"] = base64.b64decode(v["blob"])            except Exception:                pass        return v    @field_serializer("message")    def serialize_message(self, v):        if isinstance(v, self.BlobMessage):            return {"blob": base64.b64encode(v.blob).decode("utf-8")}        return vclass ToolInvokeMessageBinary(BaseModel):    mimetype: str = Field(..., description="The mimetype of the binary")    url: str = Field(..., description="The url of the binary")    file_var: Optional[dict[str, Any]] = Noneclass ToolParameter(PluginParameter):    """    Overrides type    """    class ToolParameterType(enum.StrEnum):        """        removes TOOLS_SELECTOR from PluginParameterType        """        STRING = PluginParameterType.STRING.value        NUMBER = PluginParameterType.NUMBER.value        BOOLEAN = PluginParameterType.BOOLEAN.value        SELECT = PluginParameterType.SELECT.value        SECRET_INPUT = PluginParameterType.SECRET_INPUT.value        FILE = PluginParameterType.FILE.value        FILES = PluginParameterType.FILES.value        APP_SELECTOR = PluginParameterType.APP_SELECTOR.value        MODEL_SELECTOR = PluginParameterType.MODEL_SELECTOR.value        # deprecated, should not use.        SYSTEM_FILES = PluginParameterType.SYSTEM_FILES.value        def as_normal_type(self):            return as_normal_type(self)        def cast_value(self, value: Any):            return cast_parameter_value(self, value)    class ToolParameterForm(Enum):        SCHEMA = "schema"  # should be set while adding tool        FORM = "form"  # should be set before invoking tool        LLM = "llm"  # will be set by LLM    type: ToolParameterType = Field(..., description="The type of the parameter")    human_description: Optional[I18nObject] = Field(default=None, description="The description presented to the user")    form: ToolParameterForm = Field(..., description="The form of the parameter, schema/form/llm")    llm_description: Optional[str] = None    @classmethod    def get_simple_instance(        cls,        name: str,        llm_description: str,        typ: ToolParameterType,        required: bool,        options: Optional[list[str]] = None,    ) -> "ToolParameter":        """        get a simple tool parameter        :param name: the name of the parameter        :param llm_description: the description presented to the LLM        :param type: the type of the parameter        :param required: if the parameter is required        :param options: the options of the parameter        """        # convert options to ToolParameterOption        # FIXME fix the type error        if options:            option_objs = [                PluginParameterOption(value=option, label=I18nObject(en_US=option, zh_Hans=option))                for option in options            ]        else:            option_objs = []        return cls(            name=name,            label=I18nObject(en_US="", zh_Hans=""),            placeholder=None,            human_description=I18nObject(en_US="", zh_Hans=""),            type=typ,            form=cls.ToolParameterForm.LLM,            llm_description=llm_description,            required=required,            options=option_objs,        )    def init_frontend_parameter(self, value: Any):        return init_frontend_parameter(self, self.type, value)class ToolProviderIdentity(BaseModel):    author: str = Field(..., description="The author of the tool")    name: str = Field(..., description="The name of the tool")    description: I18nObject = Field(..., description="The description of the tool")    icon: str = Field(..., description="The icon of the tool")    label: I18nObject = Field(..., description="The label of the tool")    tags: Optional[list[ToolLabelEnum]] = Field(        default=[],        description="The tags of the tool",    )class ToolIdentity(BaseModel):    author: str = Field(..., description="The author of the tool")    name: str = Field(..., description="The name of the tool")    label: I18nObject = Field(..., description="The label of the tool")    provider: str = Field(..., description="The provider of the tool")    icon: Optional[str] = Noneclass ToolDescription(BaseModel):    human: I18nObject = Field(..., description="The description presented to the user")    llm: str = Field(..., description="The description presented to the LLM")class ToolEntity(BaseModel):    identity: ToolIdentity    parameters: list[ToolParameter] = Field(default_factory=list)    description: Optional[ToolDescription] = None    output_schema: Optional[dict] = None    has_runtime_parameters: bool = Field(default=False, description="Whether the tool has runtime parameters")    # pydantic configs    model_config = ConfigDict(protected_namespaces=())    @field_validator("parameters", mode="before")    @classmethod    def set_parameters(cls, v, validation_info: ValidationInfo) -> list[ToolParameter]:        return v or []class ToolProviderEntity(BaseModel):    identity: ToolProviderIdentity    plugin_id: Optional[str] = None    credentials_schema: list[ProviderConfig] = Field(default_factory=list)class ToolProviderEntityWithPlugin(ToolProviderEntity):    tools: list[ToolEntity] = Field(default_factory=list)class WorkflowToolParameterConfiguration(BaseModel):    """    Workflow tool configuration    """    name: str = Field(..., description="The name of the parameter")    description: str = Field(..., description="The description of the parameter")    form: ToolParameter.ToolParameterForm = Field(..., description="The form of the parameter")class ToolInvokeMeta(BaseModel):    """    Tool invoke meta    """    time_cost: float = Field(..., description="The time cost of the tool invoke")    error: Optional[str] = None    tool_config: Optional[dict] = None    @classmethod    def empty(cls) -> "ToolInvokeMeta":        """        Get an empty instance of ToolInvokeMeta        """        return cls(time_cost=0.0, error=None, tool_config={})    @classmethod    def error_instance(cls, error: str) -> "ToolInvokeMeta":        """        Get an instance of ToolInvokeMeta with error        """        return cls(time_cost=0.0, error=error, tool_config={})    def to_dict(self) -> dict:        return {            "time_cost": self.time_cost,            "error": self.error,            "tool_config": self.tool_config,        }class ToolLabel(BaseModel):    """    Tool label    """    name: str = Field(..., description="The name of the tool")    label: I18nObject = Field(..., description="The label of the tool")    icon: str = Field(..., description="The icon of the tool")class ToolInvokeFrom(Enum):    """    Enum class for tool invoke    """    WORKFLOW = "workflow"    AGENT = "agent"    PLUGIN = "plugin"class ToolSelector(BaseModel):    dify_model_identity: str = TOOL_SELECTOR_MODEL_IDENTITY    class Parameter(BaseModel):        name: str = Field(..., description="The name of the parameter")        type: ToolParameter.ToolParameterType = Field(..., description="The type of the parameter")        required: bool = Field(..., description="Whether the parameter is required")        description: str = Field(..., description="The description of the parameter")        default: Optional[Union[int, float, str]] = None        options: Optional[list[PluginParameterOption]] = None    provider_id: str = Field(..., description="The id of the provider")    tool_name: str = Field(..., description="The name of the tool")    tool_description: str = Field(..., description="The description of the tool")    tool_configuration: Mapping[str, Any] = Field(..., description="Configuration, type form")    tool_parameters: Mapping[str, Parameter] = Field(..., description="Parameters, type llm")    def to_plugin_parameter(self) -> dict[str, Any]:        return self.model_dump()
 |