summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorGuido van Rossum <guido@python.org>2017-01-23 01:47:20 (GMT)
committerGuido van Rossum <guido@python.org>2017-01-23 01:47:20 (GMT)
commit95919c096ca74b6a28000193e1c502408a9f3e6c (patch)
tree446025c2377497ec33ad583ed23a5c326a94229e
parentd7adfe129cadbd35916152dc014da9278e472760 (diff)
downloadcpython-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.py32
-rw-r--r--Lib/typing.py4
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