Skip to content

Commit

Permalink
Add timeout support via pytest-timeout
Browse files Browse the repository at this point in the history
Whenever the trio_timeout option is enabled, this plugin will hook into
requests from pytest-timeout to set a timeout. It will then start a
thread in the background that, after the timeout has reached, will
inject a system task in the test loop. This system task will collect
stacktraces for all tasks and raise an exception that will terminate the
test. The timeout thread is reused for other tests as well to not incur
a startup cost for every test.

Since this feature integrates with pytest-timeout, it also honors things
like whether a debugger is attached or not.

Drawbacks:

- Ideally, whether trio does timeouts should not be a global option, but
  would be better suited for the timeout-method in pytest-timeout. This
  would require a change in pytest-timeout to let plugins register other
  timeout methods.
- This method requires a functioning loop.

Fixes python-trio#53
  • Loading branch information
VincentVanlaer committed Sep 3, 2023
1 parent e930e6f commit ea74f06
Show file tree
Hide file tree
Showing 9 changed files with 223 additions and 6 deletions.
1 change: 1 addition & 0 deletions docs-requirements.in
Original file line number Diff line number Diff line change
Expand Up @@ -12,3 +12,4 @@ towncrier != 19.9.0,!= 21.3.0
trio >= 0.22.0
outcome >= 1.1.0
pytest >= 7.2.0
pytest_timeout
12 changes: 6 additions & 6 deletions docs-requirements.txt
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
#
# This file is autogenerated by pip-compile with python 3.8
# To update, run:
# This file is autogenerated by pip-compile with Python 3.10
# by the following command:
#
# pip-compile docs-requirements.in
#
Expand Down Expand Up @@ -40,8 +40,6 @@ idna==3.4
# trio
imagesize==1.4.1
# via sphinx
importlib-metadata==5.0.0
# via sphinx
incremental==22.10.0
# via towncrier
iniconfig==1.1.1
Expand All @@ -67,6 +65,10 @@ pygments==2.13.0
pyparsing==3.0.9
# via packaging
pytest==7.2.0
# via
# -r docs-requirements.in
# pytest-timeout
pytest-timeout==2.1.0
# via -r docs-requirements.in
pytz==2022.5
# via babel
Expand Down Expand Up @@ -109,8 +111,6 @@ trio==0.22.0
# via -r docs-requirements.in
urllib3==1.26.12
# via requests
zipp==3.10.0
# via importlib-metadata

# The following packages are considered to be unsafe in a requirements file:
# setuptools
2 changes: 2 additions & 0 deletions docs/source/index.rst
Original file line number Diff line number Diff line change
Expand Up @@ -33,6 +33,8 @@ and async I/O in Python. Features include:
<https://hypothesis.works/>`__ library, so your async tests can use
property-based testing: just use ``@given`` like you're used to.

* Integration with `pytest-timeout <https://github.com/pytest-dev/pytest-timeout>`

* Support for testing projects that use Trio exclusively and want to
use pytest-trio everywhere, and also for testing projects that
support multiple async libraries and only want to enable
Expand Down
25 changes: 25 additions & 0 deletions docs/source/reference.rst
Original file line number Diff line number Diff line change
Expand Up @@ -420,3 +420,28 @@ it can be passed directly to the marker.
@pytest.mark.trio(run=qtrio.run)
async def test():
assert True
Configuring timeouts with pytest-timeout
----------------------------------------

Timeouts can be configured using the ``@pytest.mark.timeout`` decorator.

.. code-block:: python
import pytest
import trio
@pytest.mark.timeout(10)
async def test():
await trio.sleep_forever() # will error after 10 seconds
To get clean stacktraces that cover all tasks running when the timeout was triggered, enable the ``trio_timeout`` option.

.. code-block:: ini
# pytest.ini
[pytest]
trio_timeout = true
This timeout method requires a functioning loop, and hence will not be triggered if your test doesn't yield to the loop. This typically occurs when the test is stuck on some non-async piece of code.
1 change: 1 addition & 0 deletions newsfragments/53.feature.rst
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
Add support for pytest-timeout using our own timeout method. This timeout method can be enable via the option ``trio_timeout`` in ``pytest.ini`` and will print structured tracebacks of all tasks running when the timeout happened.
12 changes: 12 additions & 0 deletions pytest_trio/plugin.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
"""pytest-trio implementation."""
from __future__ import annotations
import sys
from functools import wraps, partial
from collections.abc import Coroutine, Generator
Expand All @@ -11,6 +12,8 @@
from trio.abc import Clock, Instrument
from trio.testing import MockClock
from _pytest.outcomes import Skipped, XFailed
# pytest_timeout_set_timer needs to be imported here for pluggy
from .timeout import set_timeout, pytest_timeout_set_timer as pytest_timeout_set_timer

if sys.version_info[:2] < (3, 11):
from exceptiongroup import BaseExceptionGroup
Expand Down Expand Up @@ -41,6 +44,12 @@ def pytest_addoption(parser):
type="bool",
default=False,
)
parser.addini(
"trio_timeout",
"should pytest-trio handle timeouts on async functions?",
type="bool",
default=False,
)
parser.addini(
"trio_run",
"what runner should pytest-trio use? [trio, qtrio]",
Expand Down Expand Up @@ -404,6 +413,9 @@ async def _bootstrap_fixtures_and_run_test(**kwargs):
contextvars_ctx = contextvars.copy_context()
contextvars_ctx.run(canary.set, "in correct context")

if item is not None:
set_timeout(item)

async with trio.open_nursery() as nursery:
for fixture in test.register_and_collect_dependencies():
nursery.start_soon(
Expand Down
105 changes: 105 additions & 0 deletions pytest_trio/timeout.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,105 @@
from __future__ import annotations
from typing import Optional
import warnings
import threading
import trio
import pytest
import pytest_timeout
from .traceback_format import format_recursive_nursery_stack


pytest_timeout_settings = pytest.StashKey[pytest_timeout.Settings]()
send_timeout_callable = None
send_timeout_callable_ready_event = threading.Event()


def set_timeout(item: pytest.Item) -> None:
try:
settings = item.stash[pytest_timeout_settings]
except KeyError:
# No timeout or not our timeout
return

if settings.func_only:
warnings.warn(
"Function only timeouts are not supported for trio based timeouts"
)

global send_timeout_callable

# Shouldn't be racy, as xdist uses different processes
if send_timeout_callable is None:
threading.Thread(target=trio_timeout_thread, daemon=True).start()

send_timeout_callable_ready_event.wait()

send_timeout_callable(settings.timeout)


@pytest.hookimpl()
def pytest_timeout_set_timer(
item: pytest.Item, settings: pytest_timeout.Settings
) -> Optional[bool]:
if item.get_closest_marker("trio") is not None and item.config.getini(
"trio_timeout"
):
item.stash[pytest_timeout_settings] = settings
return True


# No need for pytest_timeout_cancel_timer as we detect that the test loop has exited


def trio_timeout_thread():
async def run_timeouts():
async with trio.open_nursery() as nursery:
token = trio.lowlevel.current_trio_token()

async def wait_timeout(token: trio.TrioToken, timeout: float) -> None:
await trio.sleep(timeout)

try:
token.run_sync_soon(
lambda: trio.lowlevel.spawn_system_task(execute_timeout)
)
except RuntimeError:
# test has finished
pass

def send_timeout(timeout: float):
test_token = trio.lowlevel.current_trio_token()
token.run_sync_soon(
lambda: nursery.start_soon(wait_timeout, test_token, timeout)
)

global send_timeout_callable
send_timeout_callable = send_timeout
send_timeout_callable_ready_event.set()

await trio.sleep_forever()

trio.run(run_timeouts)


async def execute_timeout() -> None:
if pytest_timeout.is_debugging():
return

nursery = get_test_nursery()
stack = "\n".join(format_recursive_nursery_stack(nursery) + ["Timeout reached"])

async def report():
pytest.fail(stack, pytrace=False)

nursery.start_soon(report)


def get_test_nursery() -> trio.Nursery:
task = trio.lowlevel.current_task().parent_nursery.parent_task

for nursery in task.child_nurseries:
for task in nursery.child_tasks:
if task.name.startswith("pytest_trio.plugin._trio_test_runner_factory"):
return task.child_nurseries[0]

raise Exception("Could not find test nursery")
70 changes: 70 additions & 0 deletions pytest_trio/traceback_format.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,70 @@
from __future__ import annotations
from trio.lowlevel import Task
from itertools import chain
import traceback


def format_stack_for_task(task: Task, prefix: str) -> list[str]:
stack = list(task.iter_await_frames())

nursery_waiting_children = False

for i, (frame, lineno) in enumerate(stack):
if frame.f_code.co_name == "_nested_child_finished":
stack = stack[: i - 1]
nursery_waiting_children = True
break
if frame.f_code.co_name == "wait_task_rescheduled":
stack = stack[:i]
break
if frame.f_code.co_name == "checkpoint":
stack = stack[:i]
break

stack = (frame for frame in stack if "__tracebackhide__" not in frame[0].f_locals)

ss = traceback.StackSummary.extract(stack)
formated_traceback = list(
map(lambda x: prefix + x[2:], "".join(ss.format()).splitlines())
)

if nursery_waiting_children:
formated_traceback.append(prefix + "Awaiting completion of children")
formated_traceback.append(prefix)

return formated_traceback


def format_task(task: Task, prefix: str = "") -> list[str]:
lines = []

subtasks = list(
chain(*(child_nursery.child_tasks for child_nursery in task.child_nurseries))
)

if subtasks:
trace_prefix = prefix + "│"
else:
trace_prefix = prefix + " "

lines.extend(format_stack_for_task(task, trace_prefix))

for i, subtask in enumerate(subtasks):
if (i + 1) != len(subtasks):
lines.append(f"{prefix}{subtask.name}")
lines.extend(format_task(subtask, prefix=f"{prefix}│ "))
else:
lines.append(f"{prefix}{subtask.name}")
lines.extend(format_task(subtask, prefix=f"{prefix} "))

return lines


def format_recursive_nursery_stack(nursery) -> list[str]:
stack = []

for task in nursery.child_tasks:
stack.append(task.name)
stack.extend(format_task(task))

return stack
1 change: 1 addition & 0 deletions setup.py
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@
"trio >= 0.22.0", # for ExceptionGroup support
"outcome >= 1.1.0",
"pytest >= 7.2.0", # for ExceptionGroup support
"pytest_timeout",
],
keywords=[
"async",
Expand Down

0 comments on commit ea74f06

Please sign in to comment.