Commit 29fda8db authored by Ivan Levkivskyi's avatar Ivan Levkivskyi Committed by Guido van Rossum

bpo-28556: Updates to typing module (#2076)

This PR contains two updates to typing module:

- Support ContextManager on all versions (original PR by Jelle Zijlstra).
- Add generic AsyncContextManager.
parent ca816153
...@@ -1552,6 +1552,12 @@ class AsyncIteratorWrapper(typing.AsyncIterator[T_a]): ...@@ -1552,6 +1552,12 @@ class AsyncIteratorWrapper(typing.AsyncIterator[T_a]):
return data return data
else: else:
raise StopAsyncIteration raise StopAsyncIteration
class ACM:
async def __aenter__(self) -> int:
return 42
async def __aexit__(self, etype, eval, tb):
return None
""" """
if ASYNCIO: if ASYNCIO:
...@@ -1562,12 +1568,13 @@ if ASYNCIO: ...@@ -1562,12 +1568,13 @@ if ASYNCIO:
else: else:
# fake names for the sake of static analysis # fake names for the sake of static analysis
asyncio = None asyncio = None
AwaitableWrapper = AsyncIteratorWrapper = object AwaitableWrapper = AsyncIteratorWrapper = ACM = object
PY36 = sys.version_info[:2] >= (3, 6) PY36 = sys.version_info[:2] >= (3, 6)
PY36_TESTS = """ PY36_TESTS = """
from test import ann_module, ann_module2, ann_module3 from test import ann_module, ann_module2, ann_module3
from typing import AsyncContextManager
class A: class A:
y: float y: float
...@@ -1604,6 +1611,16 @@ class XRepr(NamedTuple): ...@@ -1604,6 +1611,16 @@ class XRepr(NamedTuple):
return f'{self.x} -> {self.y}' return f'{self.x} -> {self.y}'
def __add__(self, other): def __add__(self, other):
return 0 return 0
async def g_with(am: AsyncContextManager[int]):
x: int
async with am as x:
return x
try:
g_with(ACM()).send(None)
except StopIteration as e:
assert e.args[0] == 42
""" """
if PY36: if PY36:
...@@ -2156,8 +2173,6 @@ class CollectionsAbcTests(BaseTestCase): ...@@ -2156,8 +2173,6 @@ class CollectionsAbcTests(BaseTestCase):
class OtherABCTests(BaseTestCase): class OtherABCTests(BaseTestCase):
@skipUnless(hasattr(typing, 'ContextManager'),
'requires typing.ContextManager')
def test_contextmanager(self): def test_contextmanager(self):
@contextlib.contextmanager @contextlib.contextmanager
def manager(): def manager():
...@@ -2167,6 +2182,24 @@ class OtherABCTests(BaseTestCase): ...@@ -2167,6 +2182,24 @@ class OtherABCTests(BaseTestCase):
self.assertIsInstance(cm, typing.ContextManager) self.assertIsInstance(cm, typing.ContextManager)
self.assertNotIsInstance(42, typing.ContextManager) self.assertNotIsInstance(42, typing.ContextManager)
@skipUnless(ASYNCIO, 'Python 3.5 required')
def test_async_contextmanager(self):
class NotACM:
pass
self.assertIsInstance(ACM(), typing.AsyncContextManager)
self.assertNotIsInstance(NotACM(), typing.AsyncContextManager)
@contextlib.contextmanager
def manager():
yield 42
cm = manager()
self.assertNotIsInstance(cm, typing.AsyncContextManager)
self.assertEqual(typing.AsyncContextManager[int].__args__, (int,))
with self.assertRaises(TypeError):
isinstance(42, typing.AsyncContextManager[int])
with self.assertRaises(TypeError):
typing.AsyncContextManager[int, str]
class TypeTests(BaseTestCase): class TypeTests(BaseTestCase):
......
...@@ -10,6 +10,8 @@ try: ...@@ -10,6 +10,8 @@ try:
import collections.abc as collections_abc import collections.abc as collections_abc
except ImportError: except ImportError:
import collections as collections_abc # Fallback for PY3.2. import collections as collections_abc # Fallback for PY3.2.
if sys.version_info[:2] >= (3, 6):
import _collections_abc # Needed for private function _check_methods # noqa
try: try:
from types import WrapperDescriptorType, MethodWrapperType, MethodDescriptorType from types import WrapperDescriptorType, MethodWrapperType, MethodDescriptorType
except ImportError: except ImportError:
...@@ -37,6 +39,7 @@ __all__ = [ ...@@ -37,6 +39,7 @@ __all__ = [
# for 'Generic' and ABCs below. # for 'Generic' and ABCs below.
'ByteString', 'ByteString',
'Container', 'Container',
'ContextManager',
'Hashable', 'Hashable',
'ItemsView', 'ItemsView',
'Iterable', 'Iterable',
...@@ -57,8 +60,8 @@ __all__ = [ ...@@ -57,8 +60,8 @@ __all__ = [
# AsyncIterable, # AsyncIterable,
# Coroutine, # Coroutine,
# Collection, # Collection,
# ContextManager,
# AsyncGenerator, # AsyncGenerator,
# AsyncContextManager
# Structural checks, a.k.a. protocols. # Structural checks, a.k.a. protocols.
'Reversible', 'Reversible',
...@@ -1949,7 +1952,61 @@ class ValuesView(MappingView[VT_co], extra=collections_abc.ValuesView): ...@@ -1949,7 +1952,61 @@ class ValuesView(MappingView[VT_co], extra=collections_abc.ValuesView):
if hasattr(contextlib, 'AbstractContextManager'): if hasattr(contextlib, 'AbstractContextManager'):
class ContextManager(Generic[T_co], extra=contextlib.AbstractContextManager): class ContextManager(Generic[T_co], extra=contextlib.AbstractContextManager):
__slots__ = () __slots__ = ()
__all__.append('ContextManager') else:
class ContextManager(Generic[T_co]):
__slots__ = ()
def __enter__(self):
return self
@abc.abstractmethod
def __exit__(self, exc_type, exc_value, traceback):
return None
@classmethod
def __subclasshook__(cls, C):
if cls is ContextManager:
# In Python 3.6+, it is possible to set a method to None to
# explicitly indicate that the class does not implement an ABC
# (https://bugs.python.org/issue25958), but we do not support
# that pattern here because this fallback class is only used
# in Python 3.5 and earlier.
if (any("__enter__" in B.__dict__ for B in C.__mro__) and
any("__exit__" in B.__dict__ for B in C.__mro__)):
return True
return NotImplemented
if hasattr(contextlib, 'AbstractAsyncContextManager'):
class AsyncContextManager(Generic[T_co],
extra=contextlib.AbstractAsyncContextManager):
__slots__ = ()
__all__.append('AsyncContextManager')
elif sys.version_info[:2] >= (3, 5):
exec("""
class AsyncContextManager(Generic[T_co]):
__slots__ = ()
async def __aenter__(self):
return self
@abc.abstractmethod
async def __aexit__(self, exc_type, exc_value, traceback):
return None
@classmethod
def __subclasshook__(cls, C):
if cls is AsyncContextManager:
if sys.version_info[:2] >= (3, 6):
return _collections_abc._check_methods(C, "__aenter__", "__aexit__")
if (any("__aenter__" in B.__dict__ for B in C.__mro__) and
any("__aexit__" in B.__dict__ for B in C.__mro__)):
return True
return NotImplemented
__all__.append('AsyncContextManager')
""")
class Dict(dict, MutableMapping[KT, VT], extra=dict): class Dict(dict, MutableMapping[KT, VT], extra=dict):
......
...@@ -354,6 +354,10 @@ Library ...@@ -354,6 +354,10 @@ Library
non-blocking mode if it succeeded to aquire the lock but the acquire took non-blocking mode if it succeeded to aquire the lock but the acquire took
longer than the timeout. longer than the timeout.
- bpo-28556: Updates to typing module: Add generic AsyncContextManager, add
support for ContextManager on all versions. Original PRs by Jelle Zijlstra
and Ivan Levkivskyi
- bpo-30605: re.compile() no longer raises a BytesWarning when compiling a - bpo-30605: re.compile() no longer raises a BytesWarning when compiling a
bytes instance with misplaced inline modifier. Patch by Roy Williams. bytes instance with misplaced inline modifier. Patch by Roy Williams.
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment