Skip to content

Commit

Permalink
Pydantic models for parameter validators.
Browse files Browse the repository at this point in the history
  • Loading branch information
jmchilton committed Oct 3, 2024
1 parent 7aca4db commit 78e5379
Show file tree
Hide file tree
Showing 17 changed files with 1,084 additions and 238 deletions.
11 changes: 11 additions & 0 deletions lib/galaxy/tool_util/parameters/_types.py
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@
"""

from typing import (
Any,
cast,
List,
Optional,
Expand All @@ -15,6 +16,7 @@

# https://stackoverflow.com/questions/56832881/check-if-a-field-is-typing-optional
from typing_extensions import (
Annotated,
get_args,
get_origin,
)
Expand Down Expand Up @@ -46,3 +48,12 @@ def cast_as_type(arg) -> Type:

def is_optional(field) -> bool:
return get_origin(field) is Union and type(None) in get_args(field)


def expand_annotation(field: Type, new_annotations: List[Any]) -> Type:
is_annotation = get_origin(field) is Annotated
if is_annotation:
args = get_args(field) # noqa: F841
return Annotated[tuple([args[0], *args[1:], *new_annotations])] # type: ignore[return-value]
else:
return Annotated[tuple([field, *new_annotations])] # type: ignore[return-value]
34 changes: 33 additions & 1 deletion lib/galaxy/tool_util/parameters/factory.py
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,13 @@
PagesSource,
ToolSource,
)
from galaxy.tool_util.parser.parameter_validators import (
ExpressionParameterValidatorModel,
InRangeParameterValidatorModel,
LengthParameterValidatorModel,
RegexParameterValidatorModel,
static_validators,
)
from galaxy.tool_util.parser.util import parse_profile_version
from galaxy.util import string_as_bool
from .models import (
Expand Down Expand Up @@ -42,10 +49,12 @@
HiddenParameterModel,
IntegerParameterModel,
LabelValue,
NumberCompatiableValidators,
RepeatParameterModel,
RulesParameterModel,
SectionParameterModel,
SelectParameterModel,
TextCompatiableValidators,
TextParameterModel,
ToolParameterBundle,
ToolParameterBundleModel,
Expand Down Expand Up @@ -82,7 +91,14 @@ def _from_input_source_galaxy(input_source: InputSource, profile: float) -> Tool
int_value = None
else:
raise ParameterDefinitionError()
return IntegerParameterModel(name=input_source.parse_name(), optional=optional, value=int_value)
static_validator_models = static_validators(input_source.parse_validators())
validators: List[NumberCompatiableValidators] = []
for static_validator in static_validator_models:
if static_validator.type == "in_range":
validators.append(cast(InRangeParameterValidatorModel, static_validator))
return IntegerParameterModel(
name=input_source.parse_name(), optional=optional, value=int_value, validators=validators
)
elif param_type == "boolean":
nullable = input_source.parse_optional()
value = input_source.get_bool_or_none("checked", None if nullable else False)
Expand All @@ -93,9 +109,19 @@ def _from_input_source_galaxy(input_source: InputSource, profile: float) -> Tool
)
elif param_type == "text":
optional = input_source.parse_optional()
static_validator_models = static_validators(input_source.parse_validators())
validators: List[TextCompatiableValidators] = []
for static_validator in static_validator_models:
if static_validator.type == "length":
validators.append(cast(LengthParameterValidatorModel, static_validator))
elif static_validator.type == "regex":
validators.append(cast(RegexParameterValidatorModel, static_validator))
elif static_validator.type == "expression":
validators.append(cast(ExpressionParameterValidatorModel, static_validator))
return TextParameterModel(
name=input_source.parse_name(),
optional=optional,
validators=validators,
)
elif param_type == "float":
optional = input_source.parse_optional()
Expand All @@ -107,10 +133,16 @@ def _from_input_source_galaxy(input_source: InputSource, profile: float) -> Tool
float_value = None
else:
raise ParameterDefinitionError()
static_validator_models = static_validators(input_source.parse_validators())
validators: List[NumberCompatiableValidators] = []
for static_validator in static_validator_models:
if static_validator.type == "in_range":
validators.append(cast(InRangeParameterValidatorModel, static_validator))
return FloatParameterModel(
name=input_source.parse_name(),
optional=optional,
value=float_value,
validators=validators,
)
elif param_type == "hidden":
optional = input_source.parse_optional()
Expand Down
48 changes: 48 additions & 0 deletions lib/galaxy/tool_util/parameters/models.py
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@
Field,
field_validator,
HttpUrl,
PlainValidator,
RootModel,
StrictBool,
StrictFloat,
Expand All @@ -44,8 +45,16 @@
JsonTestCollectionDefDict,
JsonTestDatasetDefDict,
)
from galaxy.tool_util.parser.parameter_validators import (
ExpressionParameterValidatorModel,
InRangeParameterValidatorModel,
LengthParameterValidatorModel,
RegexParameterValidatorModel,
StaticValidatorModel,
)
from ._types import (
cast_as_type,
expand_annotation,
is_optional as is_python_type_optional,
list_type,
optional,
Expand Down Expand Up @@ -179,11 +188,36 @@ class LabelValue(BaseModel):
selected: bool


TextCompatiableValidators = Union[
LengthParameterValidatorModel,
RegexParameterValidatorModel,
ExpressionParameterValidatorModel,
]


def pydantic_validator_for(validator_model: StaticValidatorModel):

def validator(v: Any) -> Any:
validator_model.statically_validate(v)
return v

return PlainValidator(validator)


# actual typing will require generics and a typevar I think...
def static_tool_validators_to_pydantic(static_tool_param_validators: List[StaticValidatorModel]):
pydantic_validators = []
for static_validator in static_tool_param_validators:
pydantic_validators.append(pydantic_validator_for(static_validator))
return pydantic_validators


class TextParameterModel(BaseGalaxyToolParameterModelDefinition):
parameter_type: Literal["gx_text"] = "gx_text"
area: bool = False
default_value: Optional[str] = Field(default=None, alias="value")
default_options: List[LabelValue] = []
validators: List[TextCompatiableValidators] = []

@property
def py_type(self) -> Type:
Expand All @@ -196,19 +230,26 @@ def pydantic_template(self, state_representation: StateRepresentationT) -> Dynam
requires_value = self.request_requires_value
if state_representation == "job_internal":
requires_value = True
validators = static_tool_validators_to_pydantic(self.validators)
if validators:
py_type = expand_annotation(py_type, validators)
return dynamic_model_information_from_py_type(self, py_type, requires_value=requires_value)

@property
def request_requires_value(self) -> bool:
return False


NumberCompatiableValidators = Union[InRangeParameterValidatorModel,]


class IntegerParameterModel(BaseGalaxyToolParameterModelDefinition):
parameter_type: Literal["gx_integer"] = "gx_integer"
optional: bool
value: Optional[int] = None
min: Optional[int] = None
max: Optional[int] = None
validators: List[NumberCompatiableValidators] = None

@property
def py_type(self) -> Type:
Expand All @@ -223,6 +264,9 @@ def pydantic_template(self, state_representation: StateRepresentationT) -> Dynam
requires_value = True
elif _is_landing_request(state_representation):
requires_value = False
validators = static_tool_validators_to_pydantic(self.validators)
if validators:
py_type = expand_annotation(py_type, validators)
return dynamic_model_information_from_py_type(self, py_type, requires_value=requires_value)

@property
Expand All @@ -235,6 +279,7 @@ class FloatParameterModel(BaseGalaxyToolParameterModelDefinition):
value: Optional[float] = None
min: Optional[float] = None
max: Optional[float] = None
validators: List[NumberCompatiableValidators] = None

@property
def py_type(self) -> Type:
Expand All @@ -244,6 +289,9 @@ def pydantic_template(self, state_representation: StateRepresentationT) -> Dynam
py_type = self.py_type
if state_representation == "workflow_step_linked":
py_type = allow_connected_value(py_type)
validators = static_tool_validators_to_pydantic(self.validators)
if validators:
py_type = expand_annotation(py_type, validators)
return dynamic_model_information_from_py_type(self, py_type)

@property
Expand Down
3 changes: 2 additions & 1 deletion lib/galaxy/tool_util/parser/interface.py
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,7 @@

from galaxy.util import Element
from galaxy.util.path import safe_walk
from .parameter_validators import AnyValidatorModel
from .util import _parse_name

if TYPE_CHECKING:
Expand Down Expand Up @@ -502,7 +503,7 @@ def parse_sanitizer_elem(self):
"""
return None

def parse_validator_elems(self):
def parse_validators(self) -> List[AnyValidatorModel]:
"""Return an XML description of sanitizers. This is a stop gap
until we can rework galaxy.tools.parameters.validation to not
explicitly depend on XML.
Expand Down
Loading

0 comments on commit 78e5379

Please sign in to comment.