summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorYury Selivanov <yury@edgedb.com>2019-09-25 10:32:08 (GMT)
committerCarol Willing <carolcode@willingconsulting.com>2019-09-25 10:32:08 (GMT)
commitedad4d89e357c92f70c0324b937845d652b20afd (patch)
tree98cea6eaaf38962883784942aba23160b009360f
parentc64af8fad3c4f5751af624647fbb0ce023f525dc (diff)
downloadcpython-edad4d89e357c92f70c0324b937845d652b20afd.zip
cpython-edad4d89e357c92f70c0324b937845d652b20afd.tar.gz
cpython-edad4d89e357c92f70c0324b937845d652b20afd.tar.bz2
bpo-38248: Fix inconsistent immediate asyncio.Task cancellation (GH-16330)
-rw-r--r--Lib/asyncio/tasks.py2
-rw-r--r--Lib/test/test_asyncio/test_tasks.py4
-rw-r--r--Misc/NEWS.d/next/Library/2019-09-22-13-05-36.bpo-38248.Yo3N_1.rst1
-rw-r--r--Modules/_asynciomodule.c15
4 files changed, 14 insertions, 8 deletions
diff --git a/Lib/asyncio/tasks.py b/Lib/asyncio/tasks.py
index a0cb884..38d9827 100644
--- a/Lib/asyncio/tasks.py
+++ b/Lib/asyncio/tasks.py
@@ -284,7 +284,7 @@ class Task(futures._PyFuture): # Inherit Python Task implementation
if self._must_cancel:
# Task is cancelled right before coro stops.
self._must_cancel = False
- super().set_exception(exceptions.CancelledError())
+ super().cancel()
else:
super().set_result(exc.value)
except exceptions.CancelledError:
diff --git a/Lib/test/test_asyncio/test_tasks.py b/Lib/test/test_asyncio/test_tasks.py
index 576714f..dde84b8 100644
--- a/Lib/test/test_asyncio/test_tasks.py
+++ b/Lib/test/test_asyncio/test_tasks.py
@@ -604,9 +604,11 @@ class BaseTaskTests:
return 12
t = self.new_task(loop, task())
+ self.assertFalse(t.cancelled())
self.assertRaises(
asyncio.CancelledError, loop.run_until_complete, t)
self.assertTrue(t.done())
+ self.assertTrue(t.cancelled())
self.assertFalse(t._must_cancel) # White-box test.
self.assertFalse(t.cancel())
@@ -621,9 +623,11 @@ class BaseTaskTests:
return 12
t = self.new_task(loop, task())
+ self.assertFalse(t.cancelled())
self.assertRaises(
asyncio.CancelledError, loop.run_until_complete, t)
self.assertTrue(t.done())
+ self.assertTrue(t.cancelled())
self.assertFalse(t._must_cancel) # White-box test.
self.assertFalse(t.cancel())
diff --git a/Misc/NEWS.d/next/Library/2019-09-22-13-05-36.bpo-38248.Yo3N_1.rst b/Misc/NEWS.d/next/Library/2019-09-22-13-05-36.bpo-38248.Yo3N_1.rst
new file mode 100644
index 0000000..fc92209
--- /dev/null
+++ b/Misc/NEWS.d/next/Library/2019-09-22-13-05-36.bpo-38248.Yo3N_1.rst
@@ -0,0 +1 @@
+asyncio: Fix inconsistent immediate Task cancellation
diff --git a/Modules/_asynciomodule.c b/Modules/_asynciomodule.c
index 4d503a4..8ee0d7a 100644
--- a/Modules/_asynciomodule.c
+++ b/Modules/_asynciomodule.c
@@ -2628,18 +2628,19 @@ task_step_impl(TaskObj *task, PyObject *exc)
if (_PyGen_FetchStopIterationValue(&o) == 0) {
/* The error is StopIteration and that means that
the underlying coroutine has resolved */
+
+ PyObject *res;
if (task->task_must_cancel) {
// Task is cancelled right before coro stops.
- Py_DECREF(o);
task->task_must_cancel = 0;
- et = asyncio_CancelledError;
- Py_INCREF(et);
- ev = NULL;
- tb = NULL;
- goto set_exception;
+ res = future_cancel((FutureObj*)task);
+ }
+ else {
+ res = future_set_result((FutureObj*)task, o);
}
- PyObject *res = future_set_result((FutureObj*)task, o);
+
Py_DECREF(o);
+
if (res == NULL) {
return NULL;
}