Codebase list python-webargs / 02d1e391-ce1f-424d-b362-f605c9419fa5/upstream src / webargs / fields.py
02d1e391-ce1f-424d-b362-f605c9419fa5/upstream

Tree @02d1e391-ce1f-424d-b362-f605c9419fa5/upstream (Download .tar.gz)

fields.py @02d1e391-ce1f-424d-b362-f605c9419fa5/upstreamraw · history · blame

"""Field classes.

Includes all fields from `marshmallow.fields` in addition to a custom
`Nested` field and `DelimitedList`.

All fields can optionally take a special `location` keyword argument, which
tells webargs where to parse the request argument from.

.. code-block:: python

    args = {
        "active": fields.Bool(location="query"),
        "content_type": fields.Str(data_key="Content-Type", location="headers"),
    }

Note: `data_key` replaced `load_from` in marshmallow 3.
When using marshmallow 2, use `load_from`.
"""
import marshmallow as ma

# Expose all fields from marshmallow.fields.
from marshmallow.fields import *  # noqa: F40
from webargs.compat import MARSHMALLOW_VERSION_INFO
from webargs.dict2schema import dict2schema

__all__ = ["DelimitedList"] + ma.fields.__all__


class Nested(ma.fields.Nested):
    """Same as `marshmallow.fields.Nested`, except can be passed a dictionary as
    the first argument, which will be converted to a `marshmallow.Schema`.

    .. note::

        The schema class here will always be `marshmallow.Schema`, regardless
        of whether a custom schema class is set on the parser. Pass an explicit schema
        class if necessary.
    """

    def __init__(self, nested, *args, **kwargs):
        if isinstance(nested, dict):
            nested = dict2schema(nested)
        super().__init__(nested, *args, **kwargs)


class DelimitedFieldMixin:
    """
    This is a mixin class for subclasses of ma.fields.List and ma.fields.Tuple
    which split on a pre-specified delimiter. By default, the delimiter will be ","

    Because we want the MRO to reach this class before the List or Tuple class,
    it must be listed first in the superclasses

    For example, a DelimitedList-like type can be defined like so:

    >>> class MyDelimitedList(DelimitedFieldMixin, ma.fields.List):
    >>>     pass
    """

    delimiter = ","

    def _serialize(self, value, attr, obj):
        # serializing will start with parent-class serialization, so that we correctly
        # output lists of non-primitive types, e.g. DelimitedList(DateTime)
        return self.delimiter.join(
            format(each) for each in super()._serialize(value, attr, obj)
        )

    def _deserialize(self, value, attr, data, **kwargs):
        # attempting to deserialize from a non-string source is an error
        if not isinstance(value, (str, bytes)):
            if MARSHMALLOW_VERSION_INFO[0] < 3:
                self.fail("invalid")
            else:
                raise self.make_error("invalid")
        return super()._deserialize(value.split(self.delimiter), attr, data, **kwargs)


class DelimitedList(DelimitedFieldMixin, ma.fields.List):
    """A field which is similar to a List, but takes its input as a delimited
    string (e.g. "foo,bar,baz").

    Like List, it can be given a nested field type which it will use to
    de/serialize each element of the list.

    :param Field cls_or_instance: A field class or instance.
    :param str delimiter: Delimiter between values.
    """

    default_error_messages = {"invalid": "Not a valid delimited list."}
    delimiter = ","

    def __init__(self, cls_or_instance, *, delimiter=None, **kwargs):
        self.delimiter = delimiter or self.delimiter
        super().__init__(cls_or_instance, **kwargs)


# DelimitedTuple can only be defined when using marshmallow3, when Tuple was
# added
if MARSHMALLOW_VERSION_INFO[0] >= 3:

    class DelimitedTuple(DelimitedFieldMixin, ma.fields.Tuple):
        """A field which is similar to a Tuple, but takes its input as a delimited
        string (e.g. "foo,bar,baz").

        Like Tuple, it can be given a tuple of nested field types which it will use to
        de/serialize each element of the tuple.

        :param Iterable[Field] tuple_fields: An iterable of field classes or instances.
        :param str delimiter: Delimiter between values.
        """

        default_error_messages = {"invalid": "Not a valid delimited tuple."}
        delimiter = ","

        def __init__(self, tuple_fields, *, delimiter=None, **kwargs):
            self.delimiter = delimiter or self.delimiter
            super().__init__(tuple_fields, **kwargs)