| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213 | 
							- import enum
 - from typing import Any, Optional, Union
 - 
 - from pydantic import BaseModel, Field, field_validator
 - 
 - from core.entities.parameter_entities import CommonParameterType
 - from core.tools.entities.common_entities import I18nObject
 - from core.workflow.nodes.base.entities import NumberType
 - 
 - 
 - class PluginParameterOption(BaseModel):
 -     value: str = Field(..., description="The value of the option")
 -     label: I18nObject = Field(..., description="The label of the option")
 -     icon: Optional[str] = Field(
 -         default=None, description="The icon of the option, can be a url or a base64 encoded image"
 -     )
 - 
 -     @field_validator("value", mode="before")
 -     @classmethod
 -     def transform_id_to_str(cls, value) -> str:
 -         if not isinstance(value, str):
 -             return str(value)
 -         else:
 -             return value
 - 
 - 
 - class PluginParameterType(enum.StrEnum):
 -     """
 -     all available parameter types
 -     """
 - 
 -     STRING = CommonParameterType.STRING.value
 -     NUMBER = CommonParameterType.NUMBER.value
 -     BOOLEAN = CommonParameterType.BOOLEAN.value
 -     SELECT = CommonParameterType.SELECT.value
 -     SECRET_INPUT = CommonParameterType.SECRET_INPUT.value
 -     FILE = CommonParameterType.FILE.value
 -     FILES = CommonParameterType.FILES.value
 -     APP_SELECTOR = CommonParameterType.APP_SELECTOR.value
 -     MODEL_SELECTOR = CommonParameterType.MODEL_SELECTOR.value
 -     TOOLS_SELECTOR = CommonParameterType.TOOLS_SELECTOR.value
 -     ANY = CommonParameterType.ANY.value
 -     DYNAMIC_SELECT = CommonParameterType.DYNAMIC_SELECT.value
 - 
 -     # deprecated, should not use.
 -     SYSTEM_FILES = CommonParameterType.SYSTEM_FILES.value
 - 
 -     # MCP object and array type parameters
 -     ARRAY = CommonParameterType.ARRAY.value
 -     OBJECT = CommonParameterType.OBJECT.value
 - 
 - 
 - class MCPServerParameterType(enum.StrEnum):
 -     """
 -     MCP server got complex parameter types
 -     """
 - 
 -     ARRAY = "array"
 -     OBJECT = "object"
 - 
 - 
 - class PluginParameterAutoGenerate(BaseModel):
 -     class Type(enum.StrEnum):
 -         PROMPT_INSTRUCTION = "prompt_instruction"
 - 
 -     type: Type
 - 
 - 
 - class PluginParameterTemplate(BaseModel):
 -     enabled: bool = Field(default=False, description="Whether the parameter is jinja enabled")
 - 
 - 
 - class PluginParameter(BaseModel):
 -     name: str = Field(..., description="The name of the parameter")
 -     label: I18nObject = Field(..., description="The label presented to the user")
 -     placeholder: Optional[I18nObject] = Field(default=None, description="The placeholder presented to the user")
 -     scope: str | None = None
 -     auto_generate: Optional[PluginParameterAutoGenerate] = None
 -     template: Optional[PluginParameterTemplate] = None
 -     required: bool = False
 -     default: Optional[Union[float, int, str]] = None
 -     min: Optional[Union[float, int]] = None
 -     max: Optional[Union[float, int]] = None
 -     precision: Optional[int] = None
 -     options: list[PluginParameterOption] = Field(default_factory=list)
 - 
 -     @field_validator("options", mode="before")
 -     @classmethod
 -     def transform_options(cls, v):
 -         if not isinstance(v, list):
 -             return []
 -         return v
 - 
 - 
 - def as_normal_type(typ: enum.StrEnum):
 -     if typ.value in {
 -         PluginParameterType.SECRET_INPUT,
 -         PluginParameterType.SELECT,
 -     }:
 -         return "string"
 -     return typ.value
 - 
 - 
 - def cast_parameter_value(typ: enum.StrEnum, value: Any, /):
 -     try:
 -         match typ.value:
 -             case PluginParameterType.STRING | PluginParameterType.SECRET_INPUT | PluginParameterType.SELECT:
 -                 if value is None:
 -                     return ""
 -                 else:
 -                     return value if isinstance(value, str) else str(value)
 - 
 -             case PluginParameterType.BOOLEAN:
 -                 if value is None:
 -                     return False
 -                 elif isinstance(value, str):
 -                     # Allowed YAML boolean value strings: https://yaml.org/type/bool.html
 -                     # and also '0' for False and '1' for True
 -                     match value.lower():
 -                         case "true" | "yes" | "y" | "1":
 -                             return True
 -                         case "false" | "no" | "n" | "0":
 -                             return False
 -                         case _:
 -                             return bool(value)
 -                 else:
 -                     return value if isinstance(value, bool) else bool(value)
 - 
 -             case PluginParameterType.NUMBER:
 -                 if isinstance(value, int | float):
 -                     return value
 -                 elif isinstance(value, str) and value:
 -                     if "." in value:
 -                         return float(value)
 -                     else:
 -                         return int(value)
 -             case PluginParameterType.SYSTEM_FILES | PluginParameterType.FILES:
 -                 if not isinstance(value, list):
 -                     return [value]
 -                 return value
 -             case PluginParameterType.FILE:
 -                 if isinstance(value, list):
 -                     if len(value) != 1:
 -                         raise ValueError("This parameter only accepts one file but got multiple files while invoking.")
 -                     else:
 -                         return value[0]
 -                 return value
 -             case PluginParameterType.MODEL_SELECTOR | PluginParameterType.APP_SELECTOR:
 -                 if not isinstance(value, dict):
 -                     raise ValueError("The selector must be a dictionary.")
 -                 return value
 -             case PluginParameterType.TOOLS_SELECTOR:
 -                 if value and not isinstance(value, list):
 -                     raise ValueError("The tools selector must be a list.")
 -                 return value
 -             case PluginParameterType.ANY:
 -                 if value and not isinstance(value, str | dict | list | NumberType):
 -                     raise ValueError("The var selector must be a string, dictionary, list or number.")
 -                 return value
 -             case PluginParameterType.ARRAY:
 -                 if not isinstance(value, list):
 -                     # Try to parse JSON string for arrays
 -                     if isinstance(value, str):
 -                         try:
 -                             import json
 - 
 -                             parsed_value = json.loads(value)
 -                             if isinstance(parsed_value, list):
 -                                 return parsed_value
 -                         except (json.JSONDecodeError, ValueError):
 -                             pass
 -                     return [value]
 -                 return value
 -             case PluginParameterType.OBJECT:
 -                 if not isinstance(value, dict):
 -                     # Try to parse JSON string for objects
 -                     if isinstance(value, str):
 -                         try:
 -                             import json
 - 
 -                             parsed_value = json.loads(value)
 -                             if isinstance(parsed_value, dict):
 -                                 return parsed_value
 -                         except (json.JSONDecodeError, ValueError):
 -                             pass
 -                     return {}
 -                 return value
 -             case _:
 -                 return str(value)
 -     except ValueError:
 -         raise
 -     except Exception:
 -         raise ValueError(f"The tool parameter value {value} is not in correct type of {as_normal_type(typ)}.")
 - 
 - 
 - def init_frontend_parameter(rule: PluginParameter, type: enum.StrEnum, value: Any):
 -     """
 -     init frontend parameter by rule
 -     """
 -     parameter_value = value
 -     if not parameter_value and parameter_value != 0:
 -         # get default value
 -         parameter_value = rule.default
 -         if not parameter_value and rule.required:
 -             raise ValueError(f"tool parameter {rule.name} not found in tool config")
 - 
 -     if type == PluginParameterType.SELECT:
 -         # check if tool_parameter_config in options
 -         options = [x.value for x in rule.options]
 -         if parameter_value is not None and parameter_value not in options:
 -             raise ValueError(f"tool parameter {rule.name} value {parameter_value} not in options {options}")
 - 
 -     return cast_parameter_value(type, parameter_value)
 
 
  |