Skip to content

Commit

Permalink
Merge branch 'master' into pytest-ini
Browse files Browse the repository at this point in the history
  • Loading branch information
njsmith authored Jun 23, 2019
2 parents 7ab61ef + 23fc351 commit e050933
Show file tree
Hide file tree
Showing 9 changed files with 278 additions and 14 deletions.
12 changes: 12 additions & 0 deletions docs/source/history.rst
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,18 @@ Release history

.. towncrier release notes start
pytest-trio 0.5.2 (2019-02-13)
------------------------------

Features
~~~~~~~~

- pytest-trio now makes the Trio scheduler deterministic while running
inside a Hypothesis test. Hopefully you won't see any change, but if
you had scheduler-dependent bugs Hypothesis will be more effective now. (`#73 <https://github.com/python-trio/pytest-trio/issues/73>`__)

- Updated for compatibility with trio v0.11.0.

pytest-trio 0.5.1 (2018-09-28)
------------------------------

Expand Down
108 changes: 108 additions & 0 deletions docs/source/reference.rst
Original file line number Diff line number Diff line change
Expand Up @@ -106,6 +106,114 @@ but Trio fixtures **must be test scoped**. Class, module, and session
scope are not supported.


.. _cancel-yield:

An important note about ``yield`` fixtures
------------------------------------------

Like any pytest fixture, Trio fixtures can contain both setup and
teardown code separated by a ``yield``::

@pytest.fixture
async def my_fixture():
... setup code ...
yield
... teardown code ...

When pytest-trio executes this fixture, it creates a new task, and
runs the setup code until it reaches the ``yield``. Then the fixture's
task goes to sleep. Once the test has finished, the fixture task wakes
up again and resumes at the ``yield``, so it can execute the teardown
code.

So the ``yield`` in a fixture is sort of like calling ``await
wait_for_test_to_finish()``. And in Trio, any ``await``\-able
operation can be cancelled. For example, we could put a timeout on the
``yield``::

@pytest.fixture
async def my_fixture():
... setup code ...
with trio.move_on_after(5):
yield # this yield gets cancelled after 5 seconds
... teardown code ...

Now if the test takes more than 5 seconds to execute, this fixture
will cancel the ``yield``.

That's kind of a strange thing to do, but there's another version of
this that's extremely common. Suppose your fixture spawns a background
task, and then the background task raises an exception. Whenever a
background task raises an exception, it automatically cancels
everything inside the nursery's scope – which includes our ``yield``::

@pytest.fixture
async def my_fixture(nursery):
nursery.start_soon(function_that_raises_exception)
yield # this yield gets cancelled after the background task crashes
... teardown code ...

If you use fixtures with background tasks, you'll probably end up
cancelling one of these ``yield``\s sooner or later. So what happens
if the ``yield`` gets cancelled?

First, pytest-trio assumes that something has gone wrong and there's
no point in continuing the test. If the top-level test function is
running, then it cancels it.

Then, pytest-trio waits for the test function to finish, and
then begins tearing down fixtures as normal.

During this teardown process, it will eventually reach the fixture
that cancelled its ``yield``. This fixture gets resumed to execute its
teardown logic, but with a special twist: since the ``yield`` was
cancelled, the ``yield`` raises :exc:`trio.Cancelled`.

Now, here's the punchline: this means that in our examples above, the
teardown code might not be executed at all! **This is different from
how pytest fixtures normally work.** Normally, the ``yield`` in a
pytest fixture never raises an exception, so you can be certain that
any code you put after it will execute as normal. But if you have a
fixture with background tasks, and they crash, then your ``yield``
might raise an exception, and Python will skip executing the code
after the ``yield``.

In our experience, most fixtures are fine with this, and it prevents
some `weird problems
<https://github.com/python-trio/pytest-trio/issues/75>`__ that can
happen otherwise. But it's something to be aware of.

If you have a fixture where the ``yield`` might be cancelled but you
still need to run teardown code, then you can use a ``finally``
block::

@pytest.fixture
async def my_fixture(nursery):
nursery.start_soon(function_that_crashes)
try:
# This yield could be cancelled...
yield
finally:
# But this code will run anyway
... teardown code ...

(But, watch out: the teardown code is still running in a cancelled
context, so if it has any ``await``\s it could raise
:exc:`trio.Cancelled` again.)

Or if you use ``with`` to handle teardown, then you don't have to
worry about this because ``with`` blocks always perform cleanup even
if there's an exception::

@pytest.fixture
async def my_fixture(nursery):
with get_obj_that_must_be_torn_down() as obj:
nursery.start_soon(function_that_crashes, obj)
# This could raise trio.Cancelled...
# ...but that's OK, the 'with' block will still tear down 'obj'
yield obj


Concurrent setup/teardown
-------------------------

Expand Down
6 changes: 6 additions & 0 deletions newsfragments/75.feature.rst
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
Incompatible change: if you use ``yield`` inside a Trio fixture, and
the ``yield`` gets cancelled (for example, due to a background task
crashing), then the ``yield`` will now raise :exc:`trio.Cancelled`.
See :ref:`cancel-yield` for details. Also, in this same case,
pytest-trio will now reliably mark the test as failed, even if the
fixture doesn't go on to raise an exception.
29 changes: 29 additions & 0 deletions pytest_trio/_tests/test_fixture_mistakes.py
Original file line number Diff line number Diff line change
Expand Up @@ -117,3 +117,32 @@ def test_whatever(async_fixture):
result.stdout.fnmatch_lines(
["*: Trio fixtures can only be used by Trio tests*"]
)


@enable_trio_mode
def test_fixture_cancels_test_but_doesnt_raise(testdir, enable_trio_mode):
enable_trio_mode(testdir)

testdir.makepyfile(
"""
import pytest
import trio
from async_generator import async_generator, yield_
@pytest.fixture
@async_generator
async def async_fixture():
with trio.CancelScope() as cscope:
cscope.cancel()
await yield_()
async def test_whatever(async_fixture):
pass
"""
)

result = testdir.runpytest()

result.assert_outcomes(failed=1)
result.stdout.fnmatch_lines(["*async_fixture*cancelled the test*"])
52 changes: 49 additions & 3 deletions pytest_trio/_tests/test_fixture_ordering.py
Original file line number Diff line number Diff line change
Expand Up @@ -171,7 +171,7 @@ async def crash_late_agen():
raise RuntimeError("crash_late_agen".upper())
async def crash(when, token):
with trio.open_cancel_scope(shield=True):
with trio.CancelScope(shield=True):
await trio.sleep(when)
raise RuntimeError(token.upper())
Expand Down Expand Up @@ -213,7 +213,8 @@ def test_background_crash_cancellation_propagation(bgmode, testdir):
@trio_fixture
def crashyfix(nursery):
nursery.start_soon(crashy)
yield
with pytest.raises(trio.Cancelled):
yield
# We should be cancelled here
teardown_deadlines["crashyfix"] = trio.current_effective_deadline()
"""
Expand All @@ -224,7 +225,8 @@ def crashyfix(nursery):
async def crashyfix():
async with trio.open_nursery() as nursery:
nursery.start_soon(crashy)
await yield_()
with pytest.raises(trio.Cancelled):
await yield_()
# We should be cancelled here
teardown_deadlines["crashyfix"] = trio.current_effective_deadline()
"""
Expand Down Expand Up @@ -284,3 +286,47 @@ def test_post():

result = testdir.runpytest()
result.assert_outcomes(passed=1, failed=1)


# See the thread starting at
# https://github.com/python-trio/pytest-trio/pull/77#issuecomment-499979536
# for details on the real case that this was minimized from
def test_complex_cancel_interaction_regression(testdir):
testdir.makepyfile(
"""
import pytest
import trio
from async_generator import asynccontextmanager, async_generator, yield_
async def die_soon():
raise RuntimeError('oops'.upper())
@asynccontextmanager
@async_generator
async def async_finalizer():
try:
await yield_()
finally:
await trio.sleep(0)
@pytest.fixture
@async_generator
async def fixture(nursery):
async with trio.open_nursery() as nursery1:
async with async_finalizer():
async with trio.open_nursery() as nursery2:
nursery2.start_soon(die_soon)
await yield_()
nursery1.cancel_scope.cancel()
@pytest.mark.trio
async def test_try(fixture):
await trio.sleep_forever()
"""
)

result = testdir.runpytest()
result.assert_outcomes(passed=0, failed=1)
result.stdout.fnmatch_lines_random([
"*OOPS*",
])
27 changes: 27 additions & 0 deletions pytest_trio/_tests/test_hypothesis_interaction.py
Original file line number Diff line number Diff line change
@@ -1,6 +1,13 @@
import pytest
import trio
from trio.tests.test_scheduler_determinism import (
scheduler_trace, test_the_trio_scheduler_is_not_deterministic,
test_the_trio_scheduler_is_deterministic_if_seeded
)
from hypothesis import given, settings, strategies as st

from pytest_trio.plugin import _trio_test_runner_factory

# deadline=None avoids unpredictable warnings/errors when CI happens to be
# slow (example: https://travis-ci.org/python-trio/pytest-trio/jobs/406738296)
# max_examples=5 speeds things up a bit
Expand Down Expand Up @@ -28,3 +35,23 @@ async def test_mark_outer(n):
async def test_mark_and_parametrize(x, y):
assert x is None
assert y in (1, 2)


def test_the_trio_scheduler_is_deterministic_under_hypothesis():
traces = []

@our_settings
@given(st.integers())
@pytest.mark.trio
async def inner(_):
traces.append(await scheduler_trace())

# The pytest.mark.trio doesn't do it's magic thing to
# inner functions, so we invoke it explicitly here.
inner.hypothesis.inner_test = _trio_test_runner_factory(
None, inner.hypothesis.inner_test
)
inner() # Tada, now it's a sync function!

assert len(traces) >= 5
assert len(set(traces)) == 1
2 changes: 1 addition & 1 deletion pytest_trio/_version.py
Original file line number Diff line number Diff line change
@@ -1,3 +1,3 @@
# This file is imported from __init__.py and exec'd from setup.py

__version__ = "0.5.1"
__version__ = "0.5.2+dev"
Loading

0 comments on commit e050933

Please sign in to comment.