The first time any of the tasks belonging to the group fails
with an exception other than :exc:`asyncio.CancelledError`,
the remaining tasks in the group are cancelled.
+No further tasks can then be added to the group.
At this point, if the body of the ``async with`` statement is still active
(i.e., :meth:`~object.__aexit__` hasn't been called yet),
the task directly containing the ``async with`` statement is also cancelled.
raise RuntimeError(f"TaskGroup {self!r} has not been entered")
if self._exiting and not self._tasks:
raise RuntimeError(f"TaskGroup {self!r} is finished")
+ if self._aborting:
+ raise RuntimeError(f"TaskGroup {self!r} is shutting down")
if context is None:
task = self._loop.create_task(coro)
else:
self.assertTrue(t2.cancelled())
async def test_cancel_children_on_child_error(self):
- """
- When a child task raises an error, the rest of the children
- are cancelled and the errors are gathered into an EG.
- """
+ # When a child task raises an error, the rest of the children
+ # are cancelled and the errors are gathered into an EG.
NUM = 0
t2_cancel = False
await t2
self.assertEqual(2, ctx.get(cvar))
+ async def test_taskgroup_no_create_task_after_failure(self):
+ async def coro1():
+ await asyncio.sleep(0.001)
+ 1 / 0
+ async def coro2(g):
+ try:
+ await asyncio.sleep(1)
+ except asyncio.CancelledError:
+ with self.assertRaises(RuntimeError):
+ g.create_task(c1 := coro1())
+ # We still have to await c1 to avoid a warning
+ with self.assertRaises(ZeroDivisionError):
+ await c1
+
+ with self.assertRaises(ExceptionGroup) as cm:
+ async with taskgroups.TaskGroup() as g:
+ g.create_task(coro1())
+ g.create_task(coro2(g))
+
+ self.assertEqual(get_error_types(cm.exception), {ZeroDivisionError})
+
if __name__ == "__main__":
unittest.main()