================================ What you can unstructure and how ================================ Unstructuring is intended to convert high-level, structured Python data (like instances of complex classes) into simple, unstructured data (like dictionaries). Unstructuring is simpler than structuring in that no target types are required. Simply provide an argument to ``unstructure`` and ``cattrs`` will produce a result based on the registered unstructuring hooks. A number of default unstructuring hooks are documented here. Unstructuring is primarily done using :py:attr:`.Converter.unstructure`. Primitive types and collections ------------------------------- Primitive types (integers, floats, strings...) are simply passed through. Collections are copied. There's relatively little value in unstructuring these types directly as they are already unstructured and third-party libraries tend to support them directly. A useful use case for unstructuring collections is to create a deep copy of a complex or recursive collection. .. doctest:: >>> # A dictionary of strings to lists of tuples of floats. >>> data = {'a': [[1.0, 2.0], [3.0, 4.0]]} >>> >>> copy = cattr.unstructure(data) >>> data == copy True >>> data is copy False Customizing collection unstructuring ------------------------------------ .. important:: This feature is supported for Python 3.9 and later. Sometimes it's useful to be able to override collection unstructuring in a generic way. A common example is using a JSON library that doesn't support sets, but expects lists and tuples instead. Using ordinary unstructuring hooks for this is unwieldy due to the semantics of ``singledispatch``; in other words, you'd need to register hooks for all specific types of set you're using (``set[int]``, ``set[float]``, ``set[str]``...), which is not useful. Function-based hooks can be used instead, but come with their own set of challenges - they're complicated to write efficiently. The ``GenConverter`` supports easy customizations of collection unstructuring using its ``unstruct_collection_overrides`` parameter. For example, to unstructure all sets into lists, try the following: .. doctest:: >>> from collections.abc import Set >>> converter = cattr.GenConverter(unstruct_collection_overrides={Set: list}) >>> >>> converter.unstructure({1, 2, 3}) [1, 2, 3] Going even further, the ``GenConverter`` contains heuristics to support the following Python types, in order of decreasing generality: * ``Sequence``, ``MutableSequence``, ``list``, ``tuple`` * ``Set``, ``frozenset``, ``MutableSet``, ``set`` * ``Mapping``, ``MutableMapping``, ``dict``, ``Counter`` For example, if you override the unstructure type for ``Sequence``, but not for ``MutableSequence``, ``list`` or ``tuple``, the override will also affect those types. An easy way to remember the rule: * all ``MutableSequence`` s are ``Sequence`` s, so the override will apply * all ``list`` s are ``MutableSequence`` s, so the override will apply * all ``tuple`` s are ``Sequence`` s, so the override will apply If, however, you override only ``MutableSequence``, fields annotated as ``Sequence`` will not be affected (since not all sequences are mutable sequences), and fields annotated as tuples will not be affected (since tuples are not mutable sequences in the first place). Similar logic applies to the set and mapping hierarchies. Make sure you're using the types from ``collections.abc`` on Python 3.9+, and from ``typing`` on older Python versions. ``typing.Annotated`` -------------------- Fields marked as ``typing.Annotated[type, ...]`` are supported and are matched using the first type present in the annotated type. ``attrs`` classes and dataclasses --------------------------------- ``attrs`` classes and dataclasses are supported out of the box. :class:`.Converter` s support two unstructuring strategies: * ``UnstructureStrategy.AS_DICT`` - similar to ``attr.asdict``, unstructures ``attrs`` and dataclass instances into dictionaries. This is the default. * ``UnstructureStrategy.AS_TUPLE`` - similar to ``attr.astuple``, unstructures ``attrs`` and dataclass instances into tuples. .. doctest:: >>> @define ... class C: ... a = field() ... b = field() ... >>> inst = C(1, 'a') >>> >>> converter = cattr.Converter(unstruct_strat=cattr.UnstructureStrategy.AS_TUPLE) >>> >>> converter.unstructure(inst) (1, 'a') Mixing and matching strategies ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ Converters publicly expose two helper metods, :meth:`.Converter.unstructure_attrs_asdict` and :meth:`.Converter.unstructure_attrs_astuple`. These methods can be used with custom unstructuring hooks to selectively apply one strategy to instances of particular classes. Assume two nested ``attrs`` classes, ``Inner`` and ``Outer``; instances of ``Outer`` contain instances of ``Inner``. Instances of ``Outer`` should be unstructured as dictionaries, and instances of ``Inner`` as tuples. Here's how to do this. .. doctest:: >>> @define ... class Inner: ... a: int ... >>> @define ... class Outer: ... i: Inner ... >>> inst = Outer(i=Inner(a=1)) >>> >>> converter = cattr.Converter() >>> converter.register_unstructure_hook(Inner, converter.unstructure_attrs_astuple) >>> >>> converter.unstructure(inst) {'i': (1,)} Of course, these methods can be used directly as well, without changing the converter strategy. .. doctest:: >>> @define ... class C: ... a: int ... b: str ... >>> inst = C(1, 'a') >>> >>> converter = cattr.Converter() >>> >>> converter.unstructure_attrs_astuple(inst) # Default is AS_DICT. (1, 'a') Unstructuring hook factories ---------------------------- Hook factories operate one level higher than unstructuring hooks; unstructuring hooks are functions registered to a class or predicate, and hook factories are functions (registered via a predicate) that produce unstructuring hooks. Unstructuring hooks factories are registered using :py:attr:`cattr.Converter.register_unstructure_hook_factory`. Here's a small example showing how to use factory hooks to skip unstructuring `init=False` attributes on all `attrs` classes. .. doctest:: >>> from attr import define, has, field, fields >>> from cattr import override >>> from cattr.gen import make_dict_unstructure_fn >>> c = cattr.GenConverter() >>> c.register_unstructure_hook_factory(has, lambda cl: make_dict_unstructure_fn(cl, c, **{a.name: override(omit=True) for a in fields(cl) if not a.init})) >>> @define ... class E: ... an_int: int ... another_int: int = field(init=False) >>> inst = E(1) >>> inst.another_int = 5 >>> c.unstructure(inst) {'an_int': 1} A complex use case for hook factories is described over at :ref:`Using factory hooks`.