diff options
author | Victor K <hellysmile@gmail.com> | 2017-10-05 16:04:39 (GMT) |
---|---|---|
committer | Yury Selivanov <yury@magic.io> | 2017-10-05 16:04:39 (GMT) |
commit | 4d071897880b7e84e1a217ebe19971c118316970 (patch) | |
tree | 51656b5e3e40004a2a8d4251ebdb013723bbce54 /Lib | |
parent | 11045c9d8a21dd9bd182a3939189db02815f9783 (diff) | |
download | cpython-4d071897880b7e84e1a217ebe19971c118316970.zip cpython-4d071897880b7e84e1a217ebe19971c118316970.tar.gz cpython-4d071897880b7e84e1a217ebe19971c118316970.tar.bz2 |
bpo-31556: asyncio.wait_for can cancel futures faster with timeout <= 0 (#3703)
Diffstat (limited to 'Lib')
-rw-r--r-- | Lib/asyncio/tasks.py | 9 | ||||
-rw-r--r-- | Lib/test/test_asyncio/test_tasks.py | 70 |
2 files changed, 79 insertions, 0 deletions
diff --git a/Lib/asyncio/tasks.py b/Lib/asyncio/tasks.py index 575d205..52fef18 100644 --- a/Lib/asyncio/tasks.py +++ b/Lib/asyncio/tasks.py @@ -334,6 +334,15 @@ def wait_for(fut, timeout, *, loop=None): if timeout is None: return (yield from fut) + if timeout <= 0: + fut = ensure_future(fut, loop=loop) + + if fut.done(): + return fut.result() + + fut.cancel() + raise futures.TimeoutError() + waiter = loop.create_future() timeout_handle = loop.call_later(timeout, _release_waiter, waiter) cb = functools.partial(_release_waiter, waiter) diff --git a/Lib/test/test_asyncio/test_tasks.py b/Lib/test/test_asyncio/test_tasks.py index 36082ec..7ff56b5 100644 --- a/Lib/test/test_asyncio/test_tasks.py +++ b/Lib/test/test_asyncio/test_tasks.py @@ -661,6 +661,76 @@ class BaseTaskTests: t.cancel() self.assertRaises(asyncio.CancelledError, loop.run_until_complete, t) + def test_wait_for_timeout_less_then_0_or_0_future_done(self): + def gen(): + when = yield + self.assertAlmostEqual(0, when) + + loop = self.new_test_loop(gen) + + fut = self.new_future(loop) + fut.set_result('done') + + ret = loop.run_until_complete(asyncio.wait_for(fut, 0, loop=loop)) + + self.assertEqual(ret, 'done') + self.assertTrue(fut.done()) + self.assertAlmostEqual(0, loop.time()) + + def test_wait_for_timeout_less_then_0_or_0_coroutine_do_not_started(self): + def gen(): + when = yield + self.assertAlmostEqual(0, when) + + loop = self.new_test_loop(gen) + + foo_started = False + + @asyncio.coroutine + def foo(): + nonlocal foo_started + foo_started = True + + with self.assertRaises(asyncio.TimeoutError): + loop.run_until_complete(asyncio.wait_for(foo(), 0, loop=loop)) + + self.assertAlmostEqual(0, loop.time()) + self.assertEqual(foo_started, False) + + def test_wait_for_timeout_less_then_0_or_0(self): + def gen(): + when = yield + self.assertAlmostEqual(0.2, when) + when = yield 0 + self.assertAlmostEqual(0, when) + + for timeout in [0, -1]: + with self.subTest(timeout=timeout): + loop = self.new_test_loop(gen) + + foo_running = None + + @asyncio.coroutine + def foo(): + nonlocal foo_running + foo_running = True + try: + yield from asyncio.sleep(0.2, loop=loop) + finally: + foo_running = False + return 'done' + + fut = self.new_task(loop, foo()) + + with self.assertRaises(asyncio.TimeoutError): + loop.run_until_complete(asyncio.wait_for( + fut, timeout, loop=loop)) + self.assertTrue(fut.done()) + # it should have been cancelled due to the timeout + self.assertTrue(fut.cancelled()) + self.assertAlmostEqual(0, loop.time()) + self.assertEqual(foo_running, False) + def test_wait_for(self): def gen(): |