Issue #24400: Resurrect inspect.isawaitable()

collections.abc.Awaitable and collections.abc.Coroutine no longer
use __instancecheck__ hook to detect generator-based coroutines.

inspect.isawaitable() can be used to detect generator-based coroutines
and to distinguish them from regular generator objects.
This commit is contained in:
Yury Selivanov 2015-07-03 13:11:35 -04:00
parent 2ab5b092e5
commit fdbeb2b4b6
10 changed files with 92 additions and 39 deletions

View File

@ -162,10 +162,11 @@ ABC Inherits from Abstract Methods Mixin
:class:`~collections.abc.Coroutine` ABC are all instances of this ABC. :class:`~collections.abc.Coroutine` ABC are all instances of this ABC.
.. note:: .. note::
In CPython, generator-based coroutines are *awaitables*, even though In CPython, generator-based coroutines (generators decorated with
they do not have an :meth:`__await__` method. This ABC :func:`types.coroutine` or :func:`asyncio.coroutine`) are
implements an :meth:`~class.__instancecheck__` method to make them *awaitables*, even though they do not have an :meth:`__await__` method.
instances of itself. Using ``isinstance(gencoro, Awaitable)`` for them will return ``False``.
Use :func:`inspect.isawaitable` to detect them.
.. versionadded:: 3.5 .. versionadded:: 3.5
@ -179,10 +180,11 @@ ABC Inherits from Abstract Methods Mixin
:class:`Awaitable`. See also the definition of :term:`coroutine`. :class:`Awaitable`. See also the definition of :term:`coroutine`.
.. note:: .. note::
In CPython, generator-based coroutines are *awaitables* and *coroutines*, In CPython, generator-based coroutines (generators decorated with
even though they do not have an :meth:`__await__` method. This ABC :func:`types.coroutine` or :func:`asyncio.coroutine`) are
implements an :meth:`~class.__instancecheck__` method to make them *awaitables*, even though they do not have an :meth:`__await__` method.
instances of itself. Using ``isinstance(gencoro, Coroutine)`` for them will return ``False``.
Use :func:`inspect.isawaitable` to detect them.
.. versionadded:: 3.5 .. versionadded:: 3.5

View File

@ -310,6 +310,25 @@ attributes:
.. versionadded:: 3.5 .. versionadded:: 3.5
.. function:: isawaitable(object)
Return true if the object can be used in :keyword:`await` expression.
Can also be used to distinguish generator-based coroutines from regular
generators::
def gen():
yield
@types.coroutine
def gen_coro():
yield
assert not isawaitable(gen())
assert isawaitable(gen_coro())
.. versionadded:: 3.5
.. function:: istraceback(object) .. function:: istraceback(object)
Return true if the object is a traceback. Return true if the object is a traceback.

View File

@ -532,8 +532,9 @@ inspect
* New argument ``follow_wrapped`` for :func:`inspect.signature`. * New argument ``follow_wrapped`` for :func:`inspect.signature`.
(Contributed by Yury Selivanov in :issue:`20691`.) (Contributed by Yury Selivanov in :issue:`20691`.)
* New :func:`~inspect.iscoroutine` and :func:`~inspect.iscoroutinefunction` * New :func:`~inspect.iscoroutine`, :func:`~inspect.iscoroutinefunction`
functions. (Contributed by Yury Selivanov in :issue:`24017`.) and :func:`~inspect.isawaitable` functions. (Contributed by
Yury Selivanov in :issue:`24017`.)
* New :func:`~inspect.getcoroutinelocals` and :func:`~inspect.getcoroutinestate` * New :func:`~inspect.getcoroutinelocals` and :func:`~inspect.getcoroutinestate`
functions. (Contributed by Yury Selivanov in :issue:`24400`.) functions. (Contributed by Yury Selivanov in :issue:`24400`.)

View File

@ -81,22 +81,7 @@ class Hashable(metaclass=ABCMeta):
return NotImplemented return NotImplemented
class _AwaitableMeta(ABCMeta): class Awaitable(metaclass=ABCMeta):
def __instancecheck__(cls, instance):
# This hook is needed because we can't add
# '__await__' method to generator objects, and
# we can't register GeneratorType on Awaitable.
# NB: 0x100 = CO_ITERABLE_COROUTINE
# (We don't want to import 'inspect' module, as
# a dependency for 'collections.abc')
if (instance.__class__ is generator and
instance.gi_code.co_flags & 0x100):
return True
return super().__instancecheck__(instance)
class Awaitable(metaclass=_AwaitableMeta):
__slots__ = () __slots__ = ()

View File

@ -207,6 +207,13 @@ def iscoroutine(object):
"""Return true if the object is a coroutine.""" """Return true if the object is a coroutine."""
return isinstance(object, types.CoroutineType) return isinstance(object, types.CoroutineType)
def isawaitable(object):
"""Return true is object can be passed to an ``await`` expression."""
return (isinstance(object, types.CoroutineType) or
isinstance(object, types.GeneratorType) and
object.gi_code.co_flags & CO_ITERABLE_COROUTINE or
isinstance(object, collections.abc.Awaitable))
def istraceback(object): def istraceback(object):
"""Return true if the object is a traceback. """Return true if the object is a traceback.

View File

@ -511,8 +511,10 @@ class TestOneTrickPonyABCs(ABCTestCase):
self.assertTrue(issubclass(type(x), Awaitable)) self.assertTrue(issubclass(type(x), Awaitable))
c = coro() c = coro()
self.assertIsInstance(c, Awaitable) # Iterable coroutines (generators with CO_ITERABLE_COROUTINE
c.close() # awoid RuntimeWarning that coro() was not awaited # flag don't have '__await__' method, hence can't be instances
# of Awaitable. Use inspect.isawaitable to detect them.
self.assertNotIsInstance(c, Awaitable)
c = new_coro() c = new_coro()
self.assertIsInstance(c, Awaitable) self.assertIsInstance(c, Awaitable)
@ -559,8 +561,10 @@ class TestOneTrickPonyABCs(ABCTestCase):
self.assertTrue(issubclass(type(x), Awaitable)) self.assertTrue(issubclass(type(x), Awaitable))
c = coro() c = coro()
self.assertIsInstance(c, Coroutine) # Iterable coroutines (generators with CO_ITERABLE_COROUTINE
c.close() # awoid RuntimeWarning that coro() was not awaited # flag don't have '__await__' method, hence can't be instances
# of Coroutine. Use inspect.isawaitable to detect them.
self.assertNotIsInstance(c, Coroutine)
c = new_coro() c = new_coro()
self.assertIsInstance(c, Coroutine) self.assertIsInstance(c, Coroutine)

View File

@ -151,6 +151,29 @@ class TestPredicates(IsTestBase):
coro.close(); gen_coro.close() # silence warnings coro.close(); gen_coro.close() # silence warnings
def test_isawaitable(self):
def gen(): yield
self.assertFalse(inspect.isawaitable(gen()))
coro = coroutine_function_example(1)
gen_coro = gen_coroutine_function_example(1)
self.assertTrue(inspect.isawaitable(coro))
self.assertTrue(inspect.isawaitable(gen_coro))
class Future:
def __await__():
pass
self.assertTrue(inspect.isawaitable(Future()))
self.assertFalse(inspect.isawaitable(Future))
class NotFuture: pass
not_fut = NotFuture()
not_fut.__await__ = lambda: None
self.assertFalse(inspect.isawaitable(not_fut))
coro.close(); gen_coro.close() # silence warnings
def test_isroutine(self): def test_isroutine(self):
self.assertTrue(inspect.isroutine(mod.spam)) self.assertTrue(inspect.isroutine(mod.spam))
self.assertTrue(inspect.isroutine([].count)) self.assertTrue(inspect.isroutine([].count))

View File

@ -1447,6 +1447,19 @@ class CoroutineTests(unittest.TestCase):
with self.assertRaisesRegex(Exception, 'ham'): with self.assertRaisesRegex(Exception, 'ham'):
wrapper.throw(Exception, Exception('ham')) wrapper.throw(Exception, Exception('ham'))
def test_returning_itercoro(self):
@types.coroutine
def gen():
yield
gencoro = gen()
@types.coroutine
def foo():
return gencoro
self.assertIs(foo(), gencoro)
def test_genfunc(self): def test_genfunc(self):
def gen(): yield def gen(): yield
self.assertIs(types.coroutine(gen), gen) self.assertIs(types.coroutine(gen), gen)
@ -1457,9 +1470,6 @@ class CoroutineTests(unittest.TestCase):
g = gen() g = gen()
self.assertTrue(g.gi_code.co_flags & inspect.CO_ITERABLE_COROUTINE) self.assertTrue(g.gi_code.co_flags & inspect.CO_ITERABLE_COROUTINE)
self.assertFalse(g.gi_code.co_flags & inspect.CO_COROUTINE) self.assertFalse(g.gi_code.co_flags & inspect.CO_COROUTINE)
self.assertIsInstance(g, collections.abc.Coroutine)
self.assertIsInstance(g, collections.abc.Awaitable)
g.close() # silence warning
self.assertIs(types.coroutine(gen), gen) self.assertIs(types.coroutine(gen), gen)

View File

@ -241,12 +241,12 @@ def coroutine(func):
@_functools.wraps(func) @_functools.wraps(func)
def wrapped(*args, **kwargs): def wrapped(*args, **kwargs):
coro = func(*args, **kwargs) coro = func(*args, **kwargs)
if coro.__class__ is CoroutineType: if (coro.__class__ is CoroutineType or
# 'coro' is a native coroutine object. coro.__class__ is GeneratorType and coro.gi_code.co_flags & 0x100):
# 'coro' is a native coroutine object or an iterable coroutine
return coro return coro
if (coro.__class__ is GeneratorType or if (isinstance(coro, _collections_abc.Generator) and
(isinstance(coro, _collections_abc.Generator) and not isinstance(coro, _collections_abc.Coroutine)):
not isinstance(coro, _collections_abc.Coroutine))):
# 'coro' is either a pure Python generator iterator, or it # 'coro' is either a pure Python generator iterator, or it
# implements collections.abc.Generator (and does not implement # implements collections.abc.Generator (and does not implement
# collections.abc.Coroutine). # collections.abc.Coroutine).

View File

@ -25,7 +25,9 @@ Core and Builtins
uses collections.abc.Coroutine, it's intended to test for pure 'async def' uses collections.abc.Coroutine, it's intended to test for pure 'async def'
coroutines only; add new opcode: GET_YIELD_FROM_ITER; fix generators wrapper coroutines only; add new opcode: GET_YIELD_FROM_ITER; fix generators wrapper
used in types.coroutine to be instance of collections.abc.Generator; used in types.coroutine to be instance of collections.abc.Generator;
inspect.isawaitable was removed (use collections.abc.Awaitable). collections.abc.Awaitable and collections.abc.Coroutine can no longer
be used to detect generator-based coroutines--use inspect.isawaitable
instead.
- Issue #24450: Add gi_yieldfrom to generators and cr_await to coroutines. - Issue #24450: Add gi_yieldfrom to generators and cr_await to coroutines.
Contributed by Benno Leslie and Yury Selivanov. Contributed by Benno Leslie and Yury Selivanov.