Skip to content

Commit 730d2f3

Browse files
committed
Initial custom attributes work
1 parent ecdb117 commit 730d2f3

File tree

3 files changed

+141
-0
lines changed

3 files changed

+141
-0
lines changed

src/attrs/__init__.py

+1
Original file line numberDiff line numberDiff line change
@@ -43,6 +43,7 @@
4343
"AttrsInstance",
4444
"cmp_using",
4545
"converters",
46+
"custom_fields",
4647
"define",
4748
"evolve",
4849
"exceptions",

src/attrs/custom_fields.py

+64
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,64 @@
1+
from __future__ import annotations
2+
3+
import typing
4+
5+
from typing_extensions import Protocol
6+
7+
from attr._make import _make_attr_tuple_class
8+
from attrs import Attribute, AttrsInstance, fields
9+
from attrs import resolve_types as _resolve_types
10+
11+
12+
__all__ = ["custom_fields"]
13+
14+
T = typing.TypeVar("T")
15+
16+
17+
class AttributeModel(Protocol[T]):
18+
"""Custom attributes must conform to this."""
19+
20+
@classmethod
21+
def _from_attrs_attribute(
22+
cls: type[AttributeModel],
23+
cl: type[AttrsInstance],
24+
attribute: Attribute[T],
25+
) -> AttributeModel[T]:
26+
"""Create a custom attribute model from an `attrs.Attribute`."""
27+
...
28+
29+
30+
def custom_fields(
31+
cls: type[AttrsInstance],
32+
attribute_model: type[AttributeModel],
33+
resolve_types: bool = False,
34+
):
35+
"""
36+
Return the attrs fields tuple for cls with the provided attribute model.
37+
38+
:param type cls: Class to introspect.
39+
:param attribute_model: The attribute model to use.
40+
:param resolve_types: Whether to resolve the class types first.
41+
42+
:raise TypeError: If *cls* is not a class.
43+
:raise attrs.exceptions.NotAnAttrsClassError: If *cls* is not an *attrs*
44+
class.
45+
46+
:rtype: tuple (with name accessors) of `attribute_model`.
47+
48+
.. versionadded:: 23.2.0
49+
"""
50+
attrs = getattr(cls, f"__attrs_{id(attribute_model)}__", None)
51+
52+
if attrs is None:
53+
if resolve_types:
54+
_resolve_types(cls)
55+
base_attrs = fields(cls)
56+
AttrsClass = _make_attr_tuple_class(
57+
cls.__name__, [a.name for a in base_attrs]
58+
)
59+
attrs = AttrsClass(
60+
attribute_model._from_attrs_attribute(cls, a) for a in base_attrs
61+
)
62+
setattr(cls, f"__attrs_{id(attribute_model)}__", attrs)
63+
64+
return attrs

tests/test_custom_fields.py

+76
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,76 @@
1+
"""Tests for the custom attributes functionality."""
2+
from __future__ import annotations
3+
4+
from functools import partial
5+
from typing import Generic, TypeVar
6+
7+
from attrs import Attribute, AttrsInstance, define
8+
from attrs.custom_fields import custom_fields
9+
10+
11+
T = TypeVar("T")
12+
13+
14+
@define
15+
class CustomAttribute(Generic[T]):
16+
"""A custom attribute, for tests."""
17+
18+
cl: type[AttrsInstance]
19+
name: str
20+
attribute_type: T
21+
22+
@classmethod
23+
def _from_attrs_attribute(
24+
cls, attrs_cls: type[AttrsInstance], attribute: Attribute[T]
25+
):
26+
return cls(attrs_cls, attribute.name, attribute.type)
27+
28+
29+
cust_fields = partial(custom_fields, attribute_model=CustomAttribute)
30+
cust_resolved_fields = partial(
31+
custom_fields, attribute_model=CustomAttribute, resolve_types=True
32+
)
33+
34+
35+
def test_simple_custom_fields():
36+
"""Simple custom attribute overriding works."""
37+
38+
@define
39+
class Test:
40+
a: int
41+
b: float
42+
43+
for _ in range(2):
44+
# Do it twice to test caching.
45+
f = cust_fields(Test)
46+
47+
assert isinstance(f.a, CustomAttribute)
48+
assert isinstance(f.b, CustomAttribute)
49+
50+
assert not hasattr(f, "c")
51+
52+
assert f.a.name == "a"
53+
assert f.a.cl is Test
54+
assert f.a.attribute_type == "int"
55+
56+
57+
def test_resolved_custom_fields():
58+
"""Resolved custom attributes work."""
59+
60+
@define
61+
class Test:
62+
a: int
63+
b: float
64+
65+
for _ in range(2):
66+
# Do it twice to test caching.
67+
f = cust_resolved_fields(Test)
68+
69+
assert isinstance(f.a, CustomAttribute)
70+
assert isinstance(f.b, CustomAttribute)
71+
72+
assert not hasattr(f, "c")
73+
74+
assert f.a.name == "a"
75+
assert f.a.cl is Test
76+
assert f.a.attribute_type is int

0 commit comments

Comments
 (0)