diff options
author | Guido van Rossum <guido@python.org> | 2017-01-23 01:47:20 (GMT) |
---|---|---|
committer | Guido van Rossum <guido@python.org> | 2017-01-23 01:47:20 (GMT) |
commit | 95919c096ca74b6a28000193e1c502408a9f3e6c (patch) | |
tree | 446025c2377497ec33ad583ed23a5c326a94229e | |
parent | d7adfe129cadbd35916152dc014da9278e472760 (diff) | |
download | cpython-95919c096ca74b6a28000193e1c502408a9f3e6c.zip cpython-95919c096ca74b6a28000193e1c502408a9f3e6c.tar.gz cpython-95919c096ca74b6a28000193e1c502408a9f3e6c.tar.bz2 |
Issue #28556: Allow defining methods in NamedTuple class syntax (#362)
-rw-r--r-- | Lib/test/test_typing.py | 32 | ||||
-rw-r--r-- | Lib/typing.py | 4 |
2 files changed, 31 insertions, 5 deletions
diff --git a/Lib/test/test_typing.py b/Lib/test/test_typing.py index e0a9b3e..fce6b5a 100644 --- a/Lib/test/test_typing.py +++ b/Lib/test/test_typing.py @@ -612,8 +612,10 @@ class GenericTests(BaseTestCase): self.assertEqual(repr(typing.Mapping[T, TS][TS, T]), 'typing.Mapping[~TS, ~T]') self.assertEqual(repr(List[Tuple[T, TS]][int, T]), 'typing.List[typing.Tuple[int, ~T]]') - self.assertEqual(repr(List[Tuple[T, T]][List[int]]), - 'typing.List[typing.Tuple[typing.List[int], typing.List[int]]]') + self.assertEqual( + repr(List[Tuple[T, T]][List[int]]), + 'typing.List[typing.Tuple[typing.List[int], typing.List[int]]]' + ) def test_new_repr_bare(self): T = TypeVar('T') @@ -684,8 +686,10 @@ class GenericTests(BaseTestCase): raise NotImplementedError if tp.__args__: KT, VT = tp.__args__ - return all(isinstance(k, KT) and isinstance(v, VT) - for k, v in obj.items()) + return all( + isinstance(k, KT) and isinstance(v, VT) + for k, v in obj.items() + ) self.assertTrue(naive_dict_check({'x': 1}, typing.Dict[str, int])) self.assertFalse(naive_dict_check({1: 'x'}, typing.Dict[str, int])) with self.assertRaises(NotImplementedError): @@ -1409,6 +1413,16 @@ class CoolEmployee(NamedTuple): class CoolEmployeeWithDefault(NamedTuple): name: str cool: int = 0 + +class XMeth(NamedTuple): + x: int + def double(self): + return 2 * self.x + +class XMethBad(NamedTuple): + x: int + def _fields(self): + return 'no chance for this' """ if PY36: @@ -1417,6 +1431,7 @@ else: # fake names for the sake of static analysis ann_module = ann_module2 = ann_module3 = None A = B = CSub = G = CoolEmployee = CoolEmployeeWithDefault = object + XMeth = XMethBad = object gth = get_type_hints @@ -1750,7 +1765,7 @@ class CollectionsAbcTests(BaseTestCase): def test_async_generator(self): ns = {} exec("async def f():\n" - " yield 42\n", globals(), ns) + " yield 42\n", globals(), ns) g = ns['f']() self.assertIsSubclass(type(g), typing.AsyncGenerator) @@ -2039,6 +2054,13 @@ class NonDefaultAfterDefault(NamedTuple): """) @skipUnless(PY36, 'Python 3.6 required') + def test_annotation_usage_with_methods(self): + self.assertEquals(XMeth(1).double(), 2) + self.assertEquals(XMeth(42).x, XMeth(42)[0]) + self.assertEquals(XMethBad(1)._fields, ('x',)) + self.assertEquals(XMethBad(1).__annotations__, {'x': int}) + + @skipUnless(PY36, 'Python 3.6 required') def test_namedtuple_keyword_usage(self): LocalEmployee = NamedTuple("LocalEmployee", name=str, age=int) nick = LocalEmployee('Nick', 25) diff --git a/Lib/typing.py b/Lib/typing.py index eb42c19..c9e3417 100644 --- a/Lib/typing.py +++ b/Lib/typing.py @@ -2000,6 +2000,10 @@ class NamedTupleMeta(type): default_names=', '.join(defaults_dict.keys()))) nm_tpl.__new__.__defaults__ = tuple(defaults) nm_tpl._field_defaults = defaults_dict + # update from user namespace without overriding special namedtuple attributes + for key in ns: + if not hasattr(nm_tpl, key): + setattr(nm_tpl, key, ns[key]) return nm_tpl |