Skip to content

Commit 0da18e8

Browse files
committedJan 13, 2025
gh-128308: pass **kwargs to asyncio task_factory
1 parent d0ecbdd commit 0da18e8

File tree

6 files changed

+38
-29
lines changed

6 files changed

+38
-29
lines changed
 

‎Doc/library/asyncio-eventloop.rst

+2-2
Original file line numberDiff line numberDiff line change
@@ -392,9 +392,9 @@ Creating Futures and Tasks
392392

393393
If *factory* is ``None`` the default task factory will be set.
394394
Otherwise, *factory* must be a *callable* with the signature matching
395-
``(loop, coro, context=None)``, where *loop* is a reference to the active
395+
``(loop, coro, **kwargs)``, where *loop* is a reference to the active
396396
event loop, and *coro* is a coroutine object. The callable
397-
must return a :class:`asyncio.Future`-compatible object.
397+
must pass on all *kwargs*, and return a :class:`asyncio.Task`-compatible object.
398398

399399
.. method:: loop.get_task_factory()
400400

‎Lib/asyncio/base_events.py

+7-13
Original file line numberDiff line numberDiff line change
@@ -458,25 +458,18 @@ def create_future(self):
458458
"""Create a Future object attached to the loop."""
459459
return futures.Future(loop=self)
460460

461-
def create_task(self, coro, *, name=None, context=None):
461+
def create_task(self, coro, **kwargs):
462462
"""Schedule a coroutine object.
463463
464464
Return a task object.
465465
"""
466466
self._check_closed()
467467
if self._task_factory is None:
468-
task = tasks.Task(coro, loop=self, name=name, context=context)
468+
task = tasks.Task(coro, loop=self, **kwargs)
469469
if task._source_traceback:
470470
del task._source_traceback[-1]
471471
else:
472-
if context is None:
473-
# Use legacy API if context is not needed
474-
task = self._task_factory(self, coro)
475-
else:
476-
task = self._task_factory(self, coro, context=context)
477-
478-
task.set_name(name)
479-
472+
task = self._task_factory(self, coro, **kwargs)
480473
try:
481474
return task
482475
finally:
@@ -490,9 +483,10 @@ def set_task_factory(self, factory):
490483
If factory is None the default task factory will be set.
491484
492485
If factory is a callable, it should have a signature matching
493-
'(loop, coro)', where 'loop' will be a reference to the active
494-
event loop, 'coro' will be a coroutine object. The callable
495-
must return a Future.
486+
'(loop, coro, **kwargs)', where 'loop' will be a reference to the active
487+
event loop, 'coro' will be a coroutine object, and **kwargs will be
488+
arbitrary keyword arguments that should be passed on to Task.
489+
The callable must return a Task.
496490
"""
497491
if factory is not None and not callable(factory):
498492
raise TypeError('task factory must be a callable or None')

‎Lib/asyncio/taskgroups.py

+3-5
Original file line numberDiff line numberDiff line change
@@ -178,7 +178,7 @@ async def _aexit(self, et, exc):
178178
exc = None
179179

180180

181-
def create_task(self, coro, *, name=None, context=None):
181+
def create_task(self, coro, **kwargs):
182182
"""Create a new task in this group and return it.
183183
184184
Similar to `asyncio.create_task`.
@@ -192,10 +192,8 @@ def create_task(self, coro, *, name=None, context=None):
192192
if self._aborting:
193193
coro.close()
194194
raise RuntimeError(f"TaskGroup {self!r} is shutting down")
195-
if context is None:
196-
task = self._loop.create_task(coro, name=name)
197-
else:
198-
task = self._loop.create_task(coro, name=name, context=context)
195+
196+
task = self._loop.create_task(coro, **kwargs)
199197

200198
# optimization: Immediately call the done callback if the task is
201199
# already done (e.g. if the coro was able to complete eagerly),

‎Lib/asyncio/tasks.py

+2-9
Original file line numberDiff line numberDiff line change
@@ -384,19 +384,12 @@ def __wakeup(self, future):
384384
Task = _CTask = _asyncio.Task
385385

386386

387-
def create_task(coro, *, name=None, context=None):
387+
def create_task(coro, **kwargs):
388388
"""Schedule the execution of a coroutine object in a spawn task.
389389
390390
Return a Task object.
391391
"""
392-
loop = events.get_running_loop()
393-
if context is None:
394-
# Use legacy API if context is not needed
395-
task = loop.create_task(coro, name=name)
396-
else:
397-
task = loop.create_task(coro, name=name, context=context)
398-
399-
return task
392+
return events.get_running_loop().create_task(coro, **kwargs)
400393

401394

402395
# wait() and as_completed() similar to those in PEP 3148.

‎Lib/test/test_asyncio/test_eager_task_factory.py

+12
Original file line numberDiff line numberDiff line change
@@ -302,6 +302,18 @@ async def run():
302302

303303
self.run_coro(run())
304304

305+
def test_name(self):
306+
name = None
307+
async def coro():
308+
nonlocal name
309+
name = asyncio.current_task().get_name()
310+
311+
async def main():
312+
task = self.loop.create_task(coro(), name="test name")
313+
self.assertEqual(name, "test name")
314+
await task
315+
316+
self.run_coro(coro())
305317

306318
class AsyncTaskCounter:
307319
def __init__(self, loop, *, task_class, eager):

‎Lib/test/test_asyncio/test_taskgroups.py

+12
Original file line numberDiff line numberDiff line change
@@ -1040,6 +1040,18 @@ class MyKeyboardInterrupt(KeyboardInterrupt):
10401040
self.assertIsNotNone(exc)
10411041
self.assertListEqual(gc.get_referrers(exc), no_other_refs())
10421042

1043+
async def test_name(self):
1044+
name = None
1045+
1046+
async def asyncfn():
1047+
nonlocal name
1048+
name = asyncio.current_task().get_name()
1049+
1050+
async with asyncio.TaskGroup() as tg:
1051+
tg.create_task(asyncfn(), name="example name")
1052+
1053+
self.assertEqual(name, "example name")
1054+
10431055

10441056
class TestTaskGroup(BaseTestTaskGroup, unittest.IsolatedAsyncioTestCase):
10451057
loop_factory = asyncio.EventLoop

0 commit comments

Comments
 (0)