Skip to content

Commit

Permalink
Issue python#28790: Fix error when using Generic and __slots__ (Ivan L)
Browse files Browse the repository at this point in the history
  • Loading branch information
gvanrossum committed Nov 29, 2016
1 parent 47a9a4b commit 61f0a02
Show file tree
Hide file tree
Showing 2 changed files with 53 additions and 3 deletions.
38 changes: 38 additions & 0 deletions Lib/test/test_typing.py
Original file line number Diff line number Diff line change
Expand Up @@ -896,6 +896,44 @@ class Node(Generic[T]): ...
self.assertEqual(t, copy(t))
self.assertEqual(t, deepcopy(t))

def test_parameterized_slots(self):
T = TypeVar('T')
class C(Generic[T]):
__slots__ = ('potato',)

c = C()
c_int = C[int]()
self.assertEqual(C.__slots__, C[str].__slots__)

c.potato = 0
c_int.potato = 0
with self.assertRaises(AttributeError):
c.tomato = 0
with self.assertRaises(AttributeError):
c_int.tomato = 0

def foo(x: C['C']): ...
self.assertEqual(get_type_hints(foo, globals(), locals())['x'], C[C])
self.assertEqual(get_type_hints(foo, globals(), locals())['x'].__slots__,
C.__slots__)
self.assertEqual(copy(C[int]), deepcopy(C[int]))

def test_parameterized_slots_dict(self):
T = TypeVar('T')
class D(Generic[T]):
__slots__ = {'banana': 42}

d = D()
d_int = D[int]()
self.assertEqual(D.__slots__, D[str].__slots__)

d.banana = 'yes'
d_int.banana = 'yes'
with self.assertRaises(AttributeError):
d.foobar = 'no'
with self.assertRaises(AttributeError):
d_int.foobar = 'no'

def test_errors(self):
with self.assertRaises(TypeError):
B = SimpleMapping[XK, Any]
Expand Down
18 changes: 15 additions & 3 deletions Lib/typing.py
Original file line number Diff line number Diff line change
Expand Up @@ -870,6 +870,17 @@ def __extrahook__(subclass):
return __extrahook__


def _no_slots_copy(dct):
"""Internal helper: copy class __dict__ and clean slots class variables.
(They will be re-created if necessary by normal class machinery.)
"""
dict_copy = dict(dct)
if '__slots__' in dict_copy:
for slot in dict_copy['__slots__']:
dict_copy.pop(slot, None)
return dict_copy


class GenericMeta(TypingMeta, abc.ABCMeta):
"""Metaclass for generic types."""

Expand Down Expand Up @@ -967,7 +978,7 @@ def _eval_type(self, globalns, localns):
return self
return self.__class__(self.__name__,
self.__bases__,
dict(self.__dict__),
_no_slots_copy(self.__dict__),
tvars=_type_vars(ev_args) if ev_args else None,
args=ev_args,
origin=ev_origin,
Expand Down Expand Up @@ -1043,7 +1054,7 @@ def __getitem__(self, params):
args = params
return self.__class__(self.__name__,
self.__bases__,
dict(self.__dict__),
_no_slots_copy(self.__dict__),
tvars=tvars,
args=args,
origin=self,
Expand All @@ -1059,7 +1070,8 @@ def __instancecheck__(self, instance):
return issubclass(instance.__class__, self)

def __copy__(self):
return self.__class__(self.__name__, self.__bases__, dict(self.__dict__),
return self.__class__(self.__name__, self.__bases__,
_no_slots_copy(self.__dict__),
self.__parameters__, self.__args__, self.__origin__,
self.__extra__, self.__orig_bases__)

Expand Down

0 comments on commit 61f0a02

Please sign in to comment.