Source code for geoh5py.ui_json.forms
# ''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''
# Copyright (c) 2025 Mira Geoscience Ltd. '
# '
# This file is part of geoh5py. '
# '
# geoh5py is free software: you can redistribute it and/or modify '
# it under the terms of the GNU Lesser General Public License as published by '
# the Free Software Foundation, either version 3 of the License, or '
# (at your option) any later version. '
# '
# geoh5py is distributed in the hope that it will be useful, '
# but WITHOUT ANY WARRANTY; without even the implied warranty of '
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the '
# GNU Lesser General Public License for more details. '
# '
# You should have received a copy of the GNU Lesser General Public License '
# along with geoh5py. If not, see <https://www.gnu.org/licenses/>. '
# ''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''
from __future__ import annotations
from enum import Enum
from pathlib import Path
from typing import Annotated, Any
from uuid import UUID
import numpy as np
from pydantic import (
BaseModel,
ConfigDict,
field_serializer,
field_validator,
model_validator,
)
from pydantic.alias_generators import to_camel
from pydantic.functional_validators import BeforeValidator
from geoh5py.groups import Group
from geoh5py.objects import ObjectBase
from geoh5py.shared.validators import (
empty_string_to_uid,
to_class,
to_list,
to_path,
to_uuid,
)
from geoh5py.ui_json.validation import UIJsonError
[docs]
class BaseForm(BaseModel):
"""
Base class for uijson forms
:param label: Label for ui element.
:param value: The parameter's value.
:param optional: If True, ui element is rendered with a checkbox to
control the enabled state.
:param enabled: If False, ui element is rendered grey and value will
be written to file as None.
:param main: Controls whether ui element will render in the general
parameters tab (True) or optional parameters (False).
:param tooltip: String rendered on hover over ui element.
:param group: Grouped ui elements will be rendered within a box labelled
with the group name.
:param group_optional: If True, ui group is rendered with a checkbox that
controls the enabled state of all of the groups members
:param dependency: Name of parameter that controls the enabled or
visible state of the ui element.
:param dependency_type: Controls whether the ui element is enabled
or visible when the dependency is enabled if optional or True
if a bool type.
:param group_dependency: Name of parameter that controls the enabled
or visible state of the ui group.
:param group_dependency_type: Controls whether the ui group is
enabled or visible when the group dependency is enabled if
optional or True if a bool type.
"""
model_config = ConfigDict(
extra="allow",
frozen=True,
populate_by_name=True,
loc_by_alias=True,
alias_generator=to_camel,
)
label: str
value: Any
optional: bool = False
enabled: bool = True
main: bool = True
tooltip: str = ""
group: str = ""
group_optional: bool = False
dependency: str = ""
dependency_type: DependencyType = DependencyType.ENABLED
group_dependency: str = ""
group_dependency_type: DependencyType = DependencyType.ENABLED
@property
def json_string(self):
return self.model_dump_json(exclude_unset=True, by_alias=True)
[docs]
class IntegerForm(BaseForm):
"""
Integer valued uijson form.
"""
value: int = 1
min: float = -np.inf
max: float = np.inf
[docs]
class FloatForm(BaseForm):
"""
Float valued uijson form.
"""
value: float = 1.0
min: float = -np.inf
max: float = np.inf
precision: int = 2
line_edit: bool = True
[docs]
class ChoiceForm(BaseForm):
"""
Choice list uijson form.
"""
value: list[str]
choice_list: list[str]
multi_select: bool = False
[docs]
@field_validator("value", mode="before")
@classmethod
def to_list(cls, value):
if not isinstance(value, list):
value = [value]
return value
[docs]
@field_serializer("value", when_used="json")
def string_if_single(self, value):
if len(value) == 1:
value = value[0]
return value
[docs]
@model_validator(mode="after")
def valid_choice(self):
bad_choices = []
for val in self.value:
if val not in self.choice_list:
bad_choices.append(val)
if bad_choices:
raise ValueError(f"Provided value(s): {bad_choices} are not valid choices.")
return self
PathList = Annotated[
list[Path],
BeforeValidator(to_path),
BeforeValidator(to_list),
]
[docs]
class FileForm(BaseForm):
"""
File path uijson form
"""
value: PathList
file_description: list[str]
file_type: list[str]
file_multi: bool = False
[docs]
@field_serializer("value", when_used="json")
def to_string(self, value):
return ";".join([str(path) for path in value])
[docs]
@field_validator("value")
@classmethod
def valid_file(cls, value):
bad_paths = []
for path in value:
if not path.exists():
bad_paths.append(path)
if any(bad_paths):
raise ValueError(f"Provided path(s) {bad_paths} does not exist.")
return value
[docs]
@model_validator(mode="after")
def same_length(self):
if len(self.file_description) != len(self.file_type):
raise ValueError("File description and type lists must be the same length.")
return self
[docs]
@model_validator(mode="before")
@classmethod
def value_file_type(cls, data):
bad_paths = []
for path in data["value"].split(";"):
if Path(path).suffix[1:] not in data["file_type"]:
bad_paths.append(path)
if any(bad_paths):
raise ValueError(f"Provided paths {bad_paths} have invalid extensions.")
return data
MeshTypes = Annotated[
list[type[ObjectBase] | type[Group]],
BeforeValidator(to_class),
BeforeValidator(to_uuid),
BeforeValidator(to_list),
]
[docs]
class ObjectForm(BaseForm):
"""
Geoh5py object uijson form.
"""
value: UUID = UUID("00000000-0000-0000-0000-000000000000")
mesh_type: MeshTypes
_empty_string_to_uid = field_validator("value", mode="before")(empty_string_to_uid)
[docs]
class Association(str, Enum):
"""
Geoh5py object association types.
"""
VERTEX = "Vertex"
CELL = "Cell"
FACE = "Face"
[docs]
class DataType(str, Enum):
"""
Geoh5py data types.
"""
INTEGER = "Integer"
FLOAT = "Float"
BOOLEAN = "Boolean"
REFERENCED = "Referenced"
VECTOR = "Vector"
DATETIME = "DateTime"
GEOMETRIC = "Geometric"
TEXT = "Text"
[docs]
class DataForm(BaseForm):
"""
Geoh5py data uijson form.
"""
value: UUID | float | int
parent: str
association: Association | list[Association]
data_type: DataType | list[DataType]
is_value: bool = False
property: UUID = UUID("00000000-0000-0000-0000-000000000000")
min: float = -np.inf
max: float = np.inf
precision: int = 2
[docs]
@field_validator("property", mode="before")
@classmethod
def empty_string_to_uid(cls, val):
if val == "":
return UUID("00000000-0000-0000-0000-000000000000")
return val
[docs]
@model_validator(mode="after")
def value_if_is_value(self):
if (
"is_value" in self.model_fields_set # pylint: disable=unsupported-membership-test
and self.is_value
):
if isinstance(self.value, UUID):
raise ValueError("Value must be numeric if is_value is True.")
return self
[docs]
@model_validator(mode="after")
def property_if_not_is_value(self):
if (
"is_value" in self.model_fields_set # pylint: disable=unsupported-membership-test
and "property" not in self.model_fields_set # pylint: disable=unsupported-membership-test
):
raise ValueError("A property must be provided if is_value is used.")
return self
def _validate_parent(self, params: dict[str, Any]):
"""Validate form uid is a child of the parent object."""
child = None
if isinstance(self.value, UUID):
child = self.value
elif "property" in list(self.model_fields_set) and not self.is_value:
child = self.property
if child is not None:
if (
not isinstance(params[self.parent], ObjectBase)
or params[self.parent].get_entity(child)[0] is None
):
raise UIJsonError(f"{child} data is not a child of {self.parent}.")
[docs]
def validate_data(self, params: dict[str, Any]):
"""Validate the form data."""
self._validate_parent(params)
[docs]
def flatten(self):
"""Returns the data for the form."""
if (
"is_value" in self.model_fields_set # pylint: disable=unsupported-membership-test
and not self.is_value
):
return self.property
return self.value