Skip to content

Commit 9bce8aa

Browse files
authored
Expose mapping_unstructure_factory (#587)
* Expose `mapping_unstructure_factory` * Optimize a little * Use new names * Fix syntax * Reformat * Optimize and tests * Cleanup * Fix `unstructure_to`
1 parent ae80674 commit 9bce8aa

File tree

6 files changed

+64
-21
lines changed

6 files changed

+64
-21
lines changed

HISTORY.md

+2-1
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,7 @@ Our backwards-compatibility policy can be found [here](https://github.com/python
1818
([#577](https://github.com/python-attrs/cattrs/pull/577))
1919
- Add a [Migrations](https://catt.rs/latest/migrations.html) page, with instructions on migrating changed behavior for each version.
2020
([#577](https://github.com/python-attrs/cattrs/pull/577))
21+
- Expose {func}`cattrs.cols.mapping_unstructure_factory` through {mod}`cattrs.cols`.
2122
- Python 3.13 is now supported.
2223
([#543](https://github.com/python-attrs/cattrs/pull/543) [#547](https://github.com/python-attrs/cattrs/issues/547))
2324

@@ -39,7 +40,7 @@ Our backwards-compatibility policy can be found [here](https://github.com/python
3940
([#473](https://github.com/python-attrs/cattrs/pull/473))
4041
- **Minor change**: Heterogeneous tuples are now unstructured into tuples instead of lists by default; this is significantly faster and widely supported by serialization libraries.
4142
([#486](https://github.com/python-attrs/cattrs/pull/486))
42-
- **Minor change**: {py:func}`cattrs.gen.make_dict_structure_fn` will use the value for the `prefer_attrib_converters` parameter from the given converter by default now.
43+
- **Minor change**: {func}`cattrs.gen.make_dict_structure_fn` will use the value for the `prefer_attrib_converters` parameter from the given converter by default now.
4344
If you're using this function directly, the old behavior can be restored by passing in the desired values explicitly.
4445
([#527](https://github.com/python-attrs/cattrs/issues/527) [#528](https://github.com/python-attrs/cattrs/pull/528))
4546
- Introduce {meth}`BaseConverter.get_structure_hook` and {meth}`BaseConverter.get_unstructure_hook` methods.

docs/customizing.md

+1
Original file line numberDiff line numberDiff line change
@@ -189,6 +189,7 @@ Available hook factories are:
189189
* {meth}`namedtuple_dict_structure_factory <cattrs.cols.namedtuple_dict_structure_factory>`
190190
* {meth}`namedtuple_dict_unstructure_factory <cattrs.cols.namedtuple_dict_unstructure_factory>`
191191
* {meth}`mapping_structure_factory <cattrs.cols.mapping_structure_factory>`
192+
* {meth}`mapping_unstructure_factory <cattrs.cols.mapping_unstructure_factory>`
192193

193194
Additional predicates and hook factories will be added as requested.
194195

src/cattrs/cols.py

+2
Original file line numberDiff line numberDiff line change
@@ -28,6 +28,7 @@
2828
make_dict_unstructure_fn_from_attrs,
2929
make_hetero_tuple_unstructure_fn,
3030
mapping_structure_factory,
31+
mapping_unstructure_factory,
3132
)
3233
from .gen import make_iterable_unstructure_fn as iterable_unstructure_factory
3334

@@ -48,6 +49,7 @@
4849
"namedtuple_dict_structure_factory",
4950
"namedtuple_dict_unstructure_factory",
5051
"mapping_structure_factory",
52+
"mapping_unstructure_factory",
5153
]
5254

5355

src/cattrs/converters.py

+5-5
Original file line numberDiff line numberDiff line change
@@ -57,6 +57,8 @@
5757
is_namedtuple,
5858
iterable_unstructure_factory,
5959
list_structure_factory,
60+
mapping_structure_factory,
61+
mapping_unstructure_factory,
6062
namedtuple_structure_factory,
6163
namedtuple_unstructure_factory,
6264
)
@@ -86,8 +88,6 @@
8688
make_dict_structure_fn,
8789
make_dict_unstructure_fn,
8890
make_hetero_tuple_unstructure_fn,
89-
make_mapping_structure_fn,
90-
make_mapping_unstructure_fn,
9191
)
9292
from .gen.typeddicts import make_dict_structure_fn as make_typeddict_dict_struct_fn
9393
from .gen.typeddicts import make_dict_unstructure_fn as make_typeddict_dict_unstruct_fn
@@ -1335,14 +1335,14 @@ def gen_unstructure_mapping(
13351335
unstructure_to = self._unstruct_collection_overrides.get(
13361336
get_origin(cl) or cl, unstructure_to or dict
13371337
)
1338-
h = make_mapping_unstructure_fn(
1338+
h = mapping_unstructure_factory(
13391339
cl, self, unstructure_to=unstructure_to, key_handler=key_handler
13401340
)
13411341
self._unstructure_func.register_cls_list([(cl, h)], direct=True)
13421342
return h
13431343

13441344
def gen_structure_counter(self, cl: Any) -> MappingStructureFn[T]:
1345-
h = make_mapping_structure_fn(
1345+
h = mapping_structure_factory(
13461346
cl,
13471347
self,
13481348
structure_to=Counter,
@@ -1361,7 +1361,7 @@ def gen_structure_mapping(self, cl: Any) -> MappingStructureFn[T]:
13611361
AbcMapping,
13621362
): # These default to dicts
13631363
structure_to = dict
1364-
h = make_mapping_structure_fn(
1364+
h = mapping_structure_factory(
13651365
cl, self, structure_to, detailed_validation=self.detailed_validation
13661366
)
13671367
self._structure_func.register_cls_list([(cl, h)], direct=True)

src/cattrs/gen/__init__.py

+27-14
Original file line numberDiff line numberDiff line change
@@ -844,17 +844,23 @@ def make_hetero_tuple_unstructure_fn(
844844
MappingUnstructureFn = Callable[[Mapping[Any, Any]], Any]
845845

846846

847-
def make_mapping_unstructure_fn(
847+
# This factory is here for backwards compatibility and circular imports.
848+
def mapping_unstructure_factory(
848849
cl: Any,
849850
converter: BaseConverter,
850851
unstructure_to: Any = None,
851852
key_handler: Callable[[Any, Any | None], Any] | None = None,
852853
) -> MappingUnstructureFn:
853-
"""Generate a specialized unstructure function for a mapping."""
854+
"""Generate a specialized unstructure function for a mapping.
855+
856+
:param unstructure_to: The class to unstructure to; defaults to the
857+
same class as the mapping being unstructured.
858+
"""
854859
kh = key_handler or converter.unstructure
855860
val_handler = converter.unstructure
856861

857862
fn_name = "unstructure_mapping"
863+
origin = cl
858864

859865
# Let's try fishing out the type args.
860866
if getattr(cl, "__args__", None) is not None:
@@ -873,29 +879,36 @@ def make_mapping_unstructure_fn(
873879
if val_handler == identity:
874880
val_handler = None
875881

876-
globs = {
877-
"__cattr_mapping_cl": unstructure_to or cl,
878-
"__cattr_k_u": kh,
879-
"__cattr_v_u": val_handler,
880-
}
882+
origin = get_origin(cl)
883+
884+
globs = {"__cattr_k_u": kh, "__cattr_v_u": val_handler}
881885

882886
k_u = "__cattr_k_u(k)" if kh is not None else "k"
883887
v_u = "__cattr_v_u(v)" if val_handler is not None else "v"
884888

885-
lines = []
889+
lines = [f"def {fn_name}(mapping):"]
886890

887-
lines.append(f"def {fn_name}(mapping):")
888-
lines.append(
889-
f" res = __cattr_mapping_cl(({k_u}, {v_u}) for k, v in mapping.items())"
890-
)
891+
if unstructure_to is dict or unstructure_to is None and origin is dict:
892+
if kh is None and val_handler is None:
893+
# Simplest path.
894+
return dict
891895

892-
total_lines = [*lines, " return res"]
896+
lines.append(f" return {{{k_u}: {v_u} for k, v in mapping.items()}}")
897+
else:
898+
globs["__cattr_mapping_cl"] = unstructure_to or cl
899+
lines.append(
900+
f" res = __cattr_mapping_cl(({k_u}, {v_u}) for k, v in mapping.items())"
901+
)
893902

894-
eval(compile("\n".join(total_lines), "", "exec"), globs)
903+
lines = [*lines, " return res"]
904+
905+
eval(compile("\n".join(lines), "", "exec"), globs)
895906

896907
return globs[fn_name]
897908

898909

910+
make_mapping_unstructure_fn: Final = mapping_unstructure_factory
911+
899912
MappingStructureFn = Callable[[Mapping[Any, Any], Any], T]
900913

901914

tests/test_cols.py

+27-1
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,18 @@
11
"""Tests for the `cattrs.cols` module."""
22

3+
from typing import Dict
4+
35
from immutables import Map
46

57
from cattrs import BaseConverter, Converter
68
from cattrs._compat import AbstractSet, FrozenSet
7-
from cattrs.cols import is_any_set, iterable_unstructure_factory
9+
from cattrs.cols import (
10+
is_any_set,
11+
iterable_unstructure_factory,
12+
mapping_unstructure_factory,
13+
)
14+
15+
from ._compat import is_py310_plus
816

917

1018
def test_set_overriding(converter: BaseConverter):
@@ -26,3 +34,21 @@ def test_set_overriding(converter: BaseConverter):
2634
def test_structuring_immutables_map(genconverter: Converter):
2735
"""This should work due to our new is_mapping predicate."""
2836
assert genconverter.structure({"a": 1}, Map[str, int]) == Map(a=1)
37+
38+
39+
def test_mapping_unstructure_direct(genconverter: Converter):
40+
"""Some cases reduce to just `dict`."""
41+
assert genconverter.get_unstructure_hook(Dict[str, int]) is dict
42+
43+
# `dict` is equivalent to `dict[Any, Any]`, which should not reduce to
44+
# just `dict`.
45+
assert genconverter.get_unstructure_hook(dict) is not dict
46+
47+
if is_py310_plus:
48+
assert genconverter.get_unstructure_hook(dict[str, int]) is dict
49+
50+
51+
def test_mapping_unstructure_to(genconverter: Converter):
52+
"""`unstructure_to` works."""
53+
hook = mapping_unstructure_factory(Dict[str, str], genconverter, unstructure_to=Map)
54+
assert hook({"a": "a"}).__class__ is Map

0 commit comments

Comments
 (0)