Skip to content

Commit

Permalink
feat: implement model_bool hook for controlling model truthiness
Browse files Browse the repository at this point in the history
Model truthiness is an important metric for the rdfproxy grouping mechanism.
Currently, a model is considered truthy if at least one of its fields is truthy.

This is a sane default, yet certain frontend demands require different model truth conditions.

The feature introduces a model_bool option in the model_config
that allows client code to specify conditions/predicates for determining model truthiness.

Closes #110, closes #112.
  • Loading branch information
lu-pl committed Nov 16, 2024
1 parent d8e307b commit 6544cac
Show file tree
Hide file tree
Showing 3 changed files with 61 additions and 4 deletions.
7 changes: 5 additions & 2 deletions rdfproxy/mapper.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,13 +4,14 @@
from typing import Any, Generic, get_args

from pydantic import BaseModel
from rdfproxy.utils._types import _TModelInstance
from rdfproxy.utils._types import ModelBoolPredicate, _TModelInstance
from rdfproxy.utils.utils import (
_collect_values_from_bindings,
_get_group_by,
_get_key_from_metadata,
_is_list_basemodel_type,
_is_list_type,
get_model_bool_predicate,
)


Expand All @@ -29,10 +30,12 @@ def get_models(self) -> list[_TModelInstance]:
def _get_unique_models(self, model, bindings):
"""Call the mapping logic and collect unique and non-empty models."""
models = []
model_bool_predicate: ModelBoolPredicate = get_model_bool_predicate(model)

for _bindings in bindings:
_model = model(**dict(self._generate_binding_pairs(model, **_bindings)))

if any(_model.model_dump().values()) and (_model not in models):
if model_bool_predicate(_model) and (_model not in models):
models.append(_model)

return models
Expand Down
13 changes: 12 additions & 1 deletion rdfproxy/utils/_types.py
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
"""Type definitions for rdfproxy."""

from typing import Protocol, TypeVar
from collections.abc import Iterable
from typing import Protocol, TypeAlias, TypeVar, runtime_checkable

from pydantic import BaseModel

Expand Down Expand Up @@ -31,3 +32,13 @@ class Person(BaseModel):
"""

...


@runtime_checkable
class ModelBoolPredicate(Protocol):
"""Type for model_bool predicate functions."""

def __call__(self, model: BaseModel) -> bool: ...


_TModelBoolValue: TypeAlias = ModelBoolPredicate | str | Iterable[str]
45 changes: 44 additions & 1 deletion rdfproxy/utils/utils.py
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@
MissingModelConfigException,
UnboundGroupingKeyException,
)
from rdfproxy.utils._types import SPARQLBinding
from rdfproxy.utils._types import ModelBoolPredicate, SPARQLBinding, _TModelBoolValue


def _is_type(obj: type | None, _type: type) -> bool:
Expand Down Expand Up @@ -70,3 +70,46 @@ def _get_group_by(model: type[BaseModel], kwargs: dict) -> str:
f"Applicable grouping keys: {', '.join(kwargs.keys())}."
)
return group_by


def default_model_bool_predicate(model: BaseModel) -> bool:
"""Default predicate for determining model truthiness.
Adheres to rdfproxy.utils._types.ModelBoolPredicate.
"""
return any(dict(model).values())


def _is_iterable_of_str(iterable: Iterable) -> bool:
return isinstance(iterable, Iterable) and all(
map(lambda i: isinstance(i, str), iterable)
)


def _get_model_bool_predicate_from_config_value(
model_bool_value: _TModelBoolValue,
) -> ModelBoolPredicate:
"""Get a model_bool predicate function given the value of the model_bool config setting."""
match model_bool_value:
case ModelBoolPredicate():
return model_bool_value
case str():
return lambda model: bool(dict(model)[model_bool_value])
case model_bool_value if _is_iterable_of_str(model_bool_value):
return lambda model: all(map(lambda k: dict(model)[k], model_bool_value))
case _:
raise TypeError(
"Argument for 'model_bool' must be of type ModelBoolPredicate | str | Iterable[str].\n"
f"Received {type(model_bool_value)}"
)


def get_model_bool_predicate(model: BaseModel) -> ModelBoolPredicate:
"""Get the applicable model_bool predicate function given a model."""
model_bool_predicate: ModelBoolPredicate = (
default_model_bool_predicate
if (model_bool_value := model.model_config.get("model_bool", None)) is None
else _get_model_bool_predicate_from_config_value(model_bool_value)
)

return model_bool_predicate

0 comments on commit 6544cac

Please sign in to comment.