Skip to content

Commit 989d38f

Browse files
authored
gh-156: Support loop_factory parameter in bench_async_func (gh-157)
1 parent d5349bb commit 989d38f

File tree

5 files changed

+43
-11
lines changed

5 files changed

+43
-11
lines changed

doc/api.rst

+4-1
Original file line numberDiff line numberDiff line change
@@ -539,7 +539,7 @@ Runner class
539539

540540
See the :ref:`bench_func() example <bench_func_example>`.
541541

542-
.. method:: bench_async_func(name, func, \*args, inner_loops=None, metadata=None)
542+
.. method:: bench_async_func(name, func, \*args, inner_loops=None, metadata=None, loop_factory=None)
543543

544544
Benchmark the function ``await func(*args)`` in asyncio event loop.
545545

@@ -548,6 +548,9 @@ Runner class
548548
The *inner_loops* parameter is used to normalize timing per loop
549549
iteration.
550550

551+
The *loop_factory* parameter, if specified, will be used to create the
552+
event loop used by the benchmark.
553+
551554
To call ``func()`` with keyword arguments, use ``functools.partial``.
552555

553556
Return a :class:`Benchmark` instance.

doc/changelog.rst

+4-1
Original file line numberDiff line numberDiff line change
@@ -1,12 +1,15 @@
11
Changelog
22
=========
33

4-
Version 2.6.0 (2022-11-21)
4+
Version 2.6.0 (2023-03-22)
55
--------------------------
66

77
* Inherit ``PYTHONPATH`` environment variable by default.
88
Patch by Theodore Ni.
99

10+
* ``Runner.bench_async_func()`` takes an optional ``loop_factory`` to support custom loop construction.
11+
Patch by Itamar O.
12+
1013
Version 2.5.0 (2022-11-04)
1114
--------------------------
1215

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
#!/usr/bin/env python3
2+
import asyncio
3+
import pyperf
4+
5+
6+
def loop_factory():
7+
return asyncio.new_event_loop()
8+
9+
10+
async def func():
11+
await asyncio.sleep(0.001)
12+
13+
14+
runner = pyperf.Runner()
15+
runner.bench_async_func('async_sleep', func, loop_factory=loop_factory)

pyperf/_runner.py

+14-9
Original file line numberDiff line numberDiff line change
@@ -546,6 +546,7 @@ def bench_async_func(self, name, func, *args, **kwargs):
546546

547547
inner_loops = kwargs.pop('inner_loops', None)
548548
metadata = kwargs.pop('metadata', None)
549+
loop_factory = kwargs.pop('loop_factory', None)
549550
self._no_keyword_argument(kwargs)
550551

551552
if not self._check_worker_task():
@@ -582,16 +583,20 @@ async def main():
582583
return dt
583584

584585
import asyncio
585-
if hasattr(asyncio, 'run'): # Python 3.7+
586-
dt = asyncio.run(main())
587-
else: # Python 3.6
586+
# using the lower level loop API instead of asyncio.run because
587+
# asyncio.run gained the `loop_factory` arg only in Python 3.12.
588+
# we can go back to asyncio.run when Python 3.12 is the oldest
589+
# supported version for pyperf.
590+
if loop_factory is None:
588591
loop = asyncio.new_event_loop()
589-
asyncio.set_event_loop(loop)
590-
try:
591-
dt = loop.run_until_complete(main())
592-
finally:
593-
asyncio.set_event_loop(None)
594-
loop.close()
592+
else:
593+
loop = loop_factory()
594+
asyncio.set_event_loop(loop)
595+
try:
596+
dt = loop.run_until_complete(main())
597+
finally:
598+
asyncio.set_event_loop(None)
599+
loop.close()
595600

596601
return dt
597602

pyperf/tests/test_examples.py

+6
Original file line numberDiff line numberDiff line change
@@ -52,6 +52,12 @@ def test_bench_async_func(self):
5252
args = ['-p2', '-w1', '--min-time=0.001']
5353
self.check_command(script, args)
5454

55+
def test_bench_async_func_with_loop_factory(self):
56+
script = 'bench_async_func_with_loop_factory.py'
57+
# Use -w1 --min-time=0.001 to reduce the duration of the test
58+
args = ['-p2', '-w1', '--min-time=0.001']
59+
self.check_command(script, args)
60+
5561
def test_bench_time_func(self):
5662
script = 'bench_time_func.py'
5763
args = ['-p2', '-w1', '--min-time=0.001']

0 commit comments

Comments
 (0)