Skip to content

Commit 56fa438

Browse files
authored
Decouple types.DynamicClassAttribute from property (#13276)
1 parent 7cb3eef commit 56fa438

File tree

6 files changed

+40
-10
lines changed

6 files changed

+40
-10
lines changed

stdlib/@tests/stubtest_allowlists/py310.txt

-2
Original file line numberDiff line numberDiff line change
@@ -25,8 +25,6 @@ posixpath.join
2525
ntpath.join
2626
os.path.join
2727

28-
types.DynamicClassAttribute..* # In the stub we pretend it's an alias for property, but it has positional-only differences
29-
3028
# typing.IO uses positional-or-keyword arguments, but in the stubs we prefer
3129
# to mark these as positional-only for compatibility with existing sub-classes.
3230
typing(_extensions)?\.BinaryIO\.write

stdlib/@tests/stubtest_allowlists/py311.txt

-2
Original file line numberDiff line numberDiff line change
@@ -46,8 +46,6 @@ posixpath.join
4646
ntpath.join
4747
os.path.join
4848

49-
types.DynamicClassAttribute..* # In the stub we pretend it's an alias for property, but it has positional-only differences
50-
5149
# typing.IO uses positional-or-keyword arguments, but in the stubs we prefer
5250
# to mark these as positional-only for compatibility with existing sub-classes.
5351
typing(_extensions)?\.BinaryIO\.write

stdlib/@tests/stubtest_allowlists/py312.txt

-2
Original file line numberDiff line numberDiff line change
@@ -45,8 +45,6 @@ posixpath.join
4545
ntpath.join
4646
os.path.join
4747

48-
types.DynamicClassAttribute..* # In the stub we pretend it's an alias for property, but it has positional-only differences
49-
5048
# typing.IO uses positional-or-keyword arguments, but in the stubs we prefer
5149
# to mark these as positional-only for compatibility with existing sub-classes.
5250
typing(_extensions)?\.BinaryIO\.write

stdlib/@tests/stubtest_allowlists/py313.txt

-2
Original file line numberDiff line numberDiff line change
@@ -45,8 +45,6 @@ posixpath.join
4545
ntpath.join
4646
os.path.join
4747

48-
types.DynamicClassAttribute..* # In the stub we pretend it's an alias for property, but it has positional-only differences
49-
5048
# typing.IO uses positional-or-keyword arguments, but in the stubs we prefer
5149
# to mark these as positional-only for compatibility with existing sub-classes.
5250
typing(_extensions)?\.BinaryIO\.write

stdlib/@tests/test_cases/check_types.py

+19
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,5 @@
1+
from __future__ import annotations
2+
13
import sys
24
import types
35
from collections import UserDict
@@ -39,3 +41,20 @@
3941
assert_type(item_3, Union[int, str])
4042
# Default isn't accepted as a keyword argument.
4143
mp.get(4, default="default") # type: ignore
44+
45+
46+
# test: `types.DynamicClassAttribute`
47+
class DCAtest:
48+
_value: int | None = None
49+
50+
@types.DynamicClassAttribute
51+
def foo(self) -> int | None:
52+
return self._value
53+
54+
@foo.setter
55+
def foo(self, value: int) -> None:
56+
self._value = value
57+
58+
@foo.deleter
59+
def foo(self) -> None:
60+
self._value = None

stdlib/types.pyi

+21-2
Original file line numberDiff line numberDiff line change
@@ -615,8 +615,27 @@ def prepare_class(
615615
if sys.version_info >= (3, 12):
616616
def get_original_bases(cls: type, /) -> tuple[Any, ...]: ...
617617

618-
# Actually a different type, but `property` is special and we want that too.
619-
DynamicClassAttribute = property
618+
# Does not actually inherit from property, but saying it does makes sure that
619+
# pyright handles this class correctly.
620+
class DynamicClassAttribute(property):
621+
fget: Callable[[Any], Any] | None
622+
fset: Callable[[Any, Any], object] | None # type: ignore[assignment]
623+
fdel: Callable[[Any], object] | None # type: ignore[assignment]
624+
overwrite_doc: bool
625+
__isabstractmethod__: bool
626+
def __init__(
627+
self,
628+
fget: Callable[[Any], Any] | None = None,
629+
fset: Callable[[Any, Any], object] | None = None,
630+
fdel: Callable[[Any], object] | None = None,
631+
doc: str | None = None,
632+
) -> None: ...
633+
def __get__(self, instance: Any, ownerclass: type | None = None) -> Any: ...
634+
def __set__(self, instance: Any, value: Any) -> None: ...
635+
def __delete__(self, instance: Any) -> None: ...
636+
def getter(self, fget: Callable[[Any], Any]) -> DynamicClassAttribute: ...
637+
def setter(self, fset: Callable[[Any, Any], object]) -> DynamicClassAttribute: ...
638+
def deleter(self, fdel: Callable[[Any], object]) -> DynamicClassAttribute: ...
620639

621640
_Fn = TypeVar("_Fn", bound=Callable[..., object])
622641
_R = TypeVar("_R")

0 commit comments

Comments
 (0)