Commit bc565e3b authored by Yury Selivanov's avatar Yury Selivanov

Issue #28372: Fix asyncio to support formatting of non-python coroutines

parent 2356964d
...@@ -261,6 +261,25 @@ def iscoroutine(obj): ...@@ -261,6 +261,25 @@ def iscoroutine(obj):
def _format_coroutine(coro): def _format_coroutine(coro):
assert iscoroutine(coro) assert iscoroutine(coro)
if not hasattr(coro, 'cr_code') and not hasattr(coro, 'gi_code'):
# Most likely a Cython coroutine.
coro_name = getattr(coro, '__qualname__', coro.__name__)
coro_name = '{}()'.format(coro_name)
running = False
try:
running = coro.cr_running
except AttributeError:
try:
running = coro.gi_running
except AttributeError:
pass
if running:
return '{} running'.format(coro_name)
else:
return coro_name
coro_name = None coro_name = None
if isinstance(coro, CoroWrapper): if isinstance(coro, CoroWrapper):
func = coro.func func = coro.func
......
"""Tests for events.py.""" """Tests for events.py."""
import collections.abc
import functools import functools
import gc import gc
import io import io
...@@ -25,6 +26,7 @@ if sys.platform != 'win32': ...@@ -25,6 +26,7 @@ if sys.platform != 'win32':
import tty import tty
import asyncio import asyncio
from asyncio import coroutines
from asyncio import proactor_events from asyncio import proactor_events
from asyncio import selector_events from asyncio import selector_events
from asyncio import sslproto from asyncio import sslproto
...@@ -2380,6 +2382,38 @@ class HandleTests(test_utils.TestCase): ...@@ -2380,6 +2382,38 @@ class HandleTests(test_utils.TestCase):
h = loop.call_later(0, noop) h = loop.call_later(0, noop)
check_source_traceback(h) check_source_traceback(h)
@unittest.skipUnless(hasattr(collections.abc, 'Coroutine'),
'No collections.abc.Coroutine')
def test_coroutine_like_object_debug_formatting(self):
# Test that asyncio can format coroutines that are instances of
# collections.abc.Coroutine, but lack cr_core or gi_code attributes
# (such as ones compiled with Cython).
class Coro:
__name__ = 'AAA'
def send(self, v):
pass
def throw(self, *exc):
pass
def close(self):
pass
def __await__(self):
pass
coro = Coro()
self.assertTrue(asyncio.iscoroutine(coro))
self.assertEqual(coroutines._format_coroutine(coro), 'AAA()')
coro.__qualname__ = 'BBB'
self.assertEqual(coroutines._format_coroutine(coro), 'BBB()')
coro.cr_running = True
self.assertEqual(coroutines._format_coroutine(coro), 'BBB() running')
class TimerTests(unittest.TestCase): class TimerTests(unittest.TestCase):
......
...@@ -358,6 +358,8 @@ Library ...@@ -358,6 +358,8 @@ Library
- Issue #28371: Deprecate passing asyncio.Handles to run_in_executor. - Issue #28371: Deprecate passing asyncio.Handles to run_in_executor.
- Issue #28372: Fix asyncio to support formatting of non-python coroutines.
IDLE IDLE
---- ----
......
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