parameters.py 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162
  1. import enum
  2. from typing import Any, Optional, Union
  3. from pydantic import BaseModel, Field, field_validator
  4. from core.entities.parameter_entities import CommonParameterType
  5. from core.tools.entities.common_entities import I18nObject
  6. class PluginParameterOption(BaseModel):
  7. value: str = Field(..., description="The value of the option")
  8. label: I18nObject = Field(..., description="The label of the option")
  9. @field_validator("value", mode="before")
  10. @classmethod
  11. def transform_id_to_str(cls, value) -> str:
  12. if not isinstance(value, str):
  13. return str(value)
  14. else:
  15. return value
  16. class PluginParameterType(enum.StrEnum):
  17. """
  18. all available parameter types
  19. """
  20. STRING = CommonParameterType.STRING.value
  21. NUMBER = CommonParameterType.NUMBER.value
  22. BOOLEAN = CommonParameterType.BOOLEAN.value
  23. SELECT = CommonParameterType.SELECT.value
  24. SECRET_INPUT = CommonParameterType.SECRET_INPUT.value
  25. FILE = CommonParameterType.FILE.value
  26. FILES = CommonParameterType.FILES.value
  27. APP_SELECTOR = CommonParameterType.APP_SELECTOR.value
  28. MODEL_SELECTOR = CommonParameterType.MODEL_SELECTOR.value
  29. TOOLS_SELECTOR = CommonParameterType.TOOLS_SELECTOR.value
  30. # deprecated, should not use.
  31. SYSTEM_FILES = CommonParameterType.SYSTEM_FILES.value
  32. class PluginParameterAutoGenerate(BaseModel):
  33. class Type(enum.StrEnum):
  34. PROMPT_INSTRUCTION = "prompt_instruction"
  35. type: Type
  36. class PluginParameterTemplate(BaseModel):
  37. enabled: bool = Field(default=False, description="Whether the parameter is jinja enabled")
  38. class PluginParameter(BaseModel):
  39. name: str = Field(..., description="The name of the parameter")
  40. label: I18nObject = Field(..., description="The label presented to the user")
  41. placeholder: Optional[I18nObject] = Field(default=None, description="The placeholder presented to the user")
  42. scope: str | None = None
  43. auto_generate: Optional[PluginParameterAutoGenerate] = None
  44. template: Optional[PluginParameterTemplate] = None
  45. required: bool = False
  46. default: Optional[Union[float, int, str]] = None
  47. min: Optional[Union[float, int]] = None
  48. max: Optional[Union[float, int]] = None
  49. precision: Optional[int] = None
  50. options: list[PluginParameterOption] = Field(default_factory=list)
  51. @field_validator("options", mode="before")
  52. @classmethod
  53. def transform_options(cls, v):
  54. if not isinstance(v, list):
  55. return []
  56. return v
  57. def as_normal_type(typ: enum.StrEnum):
  58. if typ.value in {
  59. PluginParameterType.SECRET_INPUT,
  60. PluginParameterType.SELECT,
  61. }:
  62. return "string"
  63. return typ.value
  64. def cast_parameter_value(typ: enum.StrEnum, value: Any, /):
  65. try:
  66. match typ.value:
  67. case PluginParameterType.STRING | PluginParameterType.SECRET_INPUT | PluginParameterType.SELECT:
  68. if value is None:
  69. return ""
  70. else:
  71. return value if isinstance(value, str) else str(value)
  72. case PluginParameterType.BOOLEAN:
  73. if value is None:
  74. return False
  75. elif isinstance(value, str):
  76. # Allowed YAML boolean value strings: https://yaml.org/type/bool.html
  77. # and also '0' for False and '1' for True
  78. match value.lower():
  79. case "true" | "yes" | "y" | "1":
  80. return True
  81. case "false" | "no" | "n" | "0":
  82. return False
  83. case _:
  84. return bool(value)
  85. else:
  86. return value if isinstance(value, bool) else bool(value)
  87. case PluginParameterType.NUMBER:
  88. if isinstance(value, int | float):
  89. return value
  90. elif isinstance(value, str) and value:
  91. if "." in value:
  92. return float(value)
  93. else:
  94. return int(value)
  95. case PluginParameterType.SYSTEM_FILES | PluginParameterType.FILES:
  96. if not isinstance(value, list):
  97. return [value]
  98. return value
  99. case PluginParameterType.FILE:
  100. if isinstance(value, list):
  101. if len(value) != 1:
  102. raise ValueError("This parameter only accepts one file but got multiple files while invoking.")
  103. else:
  104. return value[0]
  105. return value
  106. case PluginParameterType.MODEL_SELECTOR | PluginParameterType.APP_SELECTOR:
  107. if not isinstance(value, dict):
  108. raise ValueError("The selector must be a dictionary.")
  109. return value
  110. case PluginParameterType.TOOLS_SELECTOR:
  111. if not isinstance(value, list):
  112. raise ValueError("The tools selector must be a list.")
  113. return value
  114. case _:
  115. return str(value)
  116. except ValueError:
  117. raise
  118. except Exception:
  119. raise ValueError(f"The tool parameter value {value} is not in correct type of {as_normal_type(typ)}.")
  120. def init_frontend_parameter(rule: PluginParameter, type: enum.StrEnum, value: Any):
  121. """
  122. init frontend parameter by rule
  123. """
  124. parameter_value = value
  125. if not parameter_value and parameter_value != 0 and type != PluginParameterType.TOOLS_SELECTOR:
  126. # get default value
  127. parameter_value = rule.default
  128. if not parameter_value and rule.required:
  129. raise ValueError(f"tool parameter {rule.name} not found in tool config")
  130. if type == PluginParameterType.SELECT:
  131. # check if tool_parameter_config in options
  132. options = [x.value for x in rule.options]
  133. if parameter_value is not None and parameter_value not in options:
  134. raise ValueError(f"tool parameter {rule.name} value {parameter_value} not in options {options}")
  135. return cast_parameter_value(type, parameter_value)