Module pydantic_partials.meta

Classes

class PartialMeta (*args, **kwargs)
Expand source code
class PartialMeta(ModelMetaclass):
    """ Metaclass of `pydantic_partials.partial.PartialModel`, used to support partial fields,
        including the ability ot automatically apply `pydantic_partials.partial.Partial`[...] to fields
        via `pydantic_partials.config.PartialConfigDict.auto_partials` configuration option.
    """
    model_partial_fields: typing.ClassVar[set[str]]
    """ Set of strings representing field names that can be missing from validation/serialization.
        If they are missing, they will return the `Missing` sentinel value and the field will be entirely
        skipped when serializing the model.

        If not missing, they will return their normal value, be included in serializing the model
        and otherwise act normally.

        If any type has their type-hint unioned with `MissingType` it will be a partial field.
        If `auto_add_missing_type` is True (default is True), then all fields on model will automatically
        be unioned with MissingType.
    """

    config_dict: typing.ClassVar[PartialConfigDict]
    """ Adding some extra/new config options on-top of the already pre-existing Pydantic ones.
        Seemed like a good place to put them, as it deals with configuration of the class model behavior.
    """

    def __new__(
            cls,
            name: str,
            bases: tuple[type],
            namespaces: dict[str, Any],
            *,

            auto_partials: bool | DefaultType = Default,
            auto_partials_exclude: Iterable[str] | DefaultType = Default,

            # A private/internal detail for generic base subclasses that want to also change the fields,
            # this prevents having to rebuild the class a second time; if this is True then the subclass
            # is responsible for calling rebuild when finished.
            ___PartialMeta__delay_rebuild: bool = False,

            **kwargs
    ):
        """

        Args:
            auto_partials: For more details see `pydantic_partials.config.PartialConfigDict.auto_partials`.
                If `Default`: Inherit behavior from parent/model_config; otherwise defaults to `False`.
                If `True`: Will automatically make all fields on the model `Partial`.
                If `False` (default): User needs to mark individual fields as `Partial` where they want.

            auto_partials_exclude: If `auto_partials` is True, or you inherit from `AutoPartialModel`
                (which simply has `auto_partials` set to `True` by default for you) then this field will be consulted.

                If auto partials is `False` (default) then `auto_partials_exclude` is ignored,
                but we will still keep track of any provided `auto_partials_exclude` in case there is a
                subclass that does enable auto_partials.

                This is a set of strings of field names to exclude from automatic partials.
                If you explicitly mark a field as a Partial, this won't affect that. THis only effects
                automatically applied partial fields.

                You can also use `pydantic_partials.partial.AutoPartialExclude` to more easily mark fields as excluded.
                For more details see `pydantic_partials.config.PartialConfigDict.auto_partials_exclude`.

            **kwargs: Passed along other class arguments to Pydantic and any __init_subclass__ methods.
        """
        # Create the class first...
        cls: 'Type[PartialModel]' = super().__new__(cls, name, bases, namespaces, **kwargs)  # type: ignore

        model_config: PartialConfigDict = cls.model_config  # type: ignore

        # We now have the fields Pydantic found and can now easily add our MissingType's as needed.
        # Then a model_rebuild is forced to update the class schema to include the MissingType/LazyType annotations,
        # and any needed changes to the field default value.

        if auto_partials is not Default:
            model_config['auto_partials'] = bool(auto_partials)

        if auto_partials_exclude:
            model_config['auto_partials_exclude'] = set(auto_partials_exclude)  # type: ignore

        # Inherit any pre-existing `auto_partials_exclude` items.
        # For now if someone wants to override this, they can simply manually do this on subclass:
        #     `field_name: Partial[...]`
        # I might consider a simple way to inform us to not inherit excludes, but for now keeping things simple.
        final_auto_exclude = model_config.get('auto_partials_exclude', set())
        for c in cls.__mro__:  # type: ignore
            parent_config: PartialConfigDict | None = getattr(c, 'model_config', None)
            if not parent_config:
                continue

            final_auto_exclude.update(parent_config.get('auto_partials_exclude', set()))

        need_rebuild = False

        partial_fields = set()
        for k, v in cls.model_fields.items():  # type: ignore
            field_type = v.annotation
            origin = get_origin(field_type)
            if origin is None:
                if field_type is MissingType:
                    partial_fields.add(k)

            # TODO: Check that `field_type` is a union?
            for arg_type in get_args(field_type):
                if arg_type is MissingType:
                    partial_fields.add(k)

            for mdv in v.metadata:
                # If we find the marker AND we are not already marked as a partial_field,
                # add field to auto-exclude list.
                if mdv is AutoPartialExcludeMarker and k not in partial_fields:
                    final_auto_exclude.add(k)

        if final_auto_exclude:
            model_config['auto_partials_exclude'] = final_auto_exclude

        # Default to `False` for auto-partials.
        final_partial_auto = model_config.get('auto_partials', False)
        if final_partial_auto is not False:
            # I'll be putting in more options for `final_partial_auto` in the near future,
            # so just check for `is True` for now.
            if final_partial_auto is not True:
                raise ValueError(f'Invalid/Unknown `partial_auto` config value ({final_partial_auto}), use bool value.')

            for k, v in cls.model_fields.items():  # type: ignore
                if k in partial_fields:
                    # The field is already a Partial
                    continue

                if k in final_auto_exclude:
                    # The field is excluded.
                    continue

                if v.default is PydanticUndefined and v.default_factory is None:
                    v.annotation = v.annotation | MissingType
                    partial_fields.add(k)
                    need_rebuild = True

        fields: dict[str, FieldInfo] = cls.model_fields  # type: ignore
        for k in partial_fields:
            v = fields[k]
            if v.default is PydanticUndefined and v.default_factory is None:
                v.default = Missing
                need_rebuild = True

        cls.model_partial_fields = partial_fields

        if need_rebuild and not ___PartialMeta__delay_rebuild:
            cls.model_rebuild(force=True)  # type: ignore
        return cls

Metaclass of PartialModel, used to support partial fields, including the ability ot automatically apply Partial[…] to fields via PartialConfigDict.auto_partials configuration option.

Ancestors

  • pydantic._internal._model_construction.ModelMetaclass
  • abc.ABCMeta
  • builtins.type

Class variables

var config_dict : ClassVar[PartialConfigDict]

Adding some extra/new config options on-top of the already pre-existing Pydantic ones. Seemed like a good place to put them, as it deals with configuration of the class model behavior.

var model_partial_fields : ClassVar[set[str]]

Set of strings representing field names that can be missing from validation/serialization. If they are missing, they will return the Missing sentinel value and the field will be entirely skipped when serializing the model.

If not missing, they will return their normal value, be included in serializing the model and otherwise act normally.

If any type has their type-hint unioned with MissingType it will be a partial field. If auto_add_missing_type is True (default is True), then all fields on model will automatically be unioned with MissingType.