Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Issues a warning when a function is not collected as a test case just because it uses @pytest.fixture (#12989) #13051

Closed
wants to merge 3 commits into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
8 changes: 8 additions & 0 deletions changelog/12989.improvement.rst
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
Issues a warning when a function is not collected as a test case just because it uses :py:func:`pytest.fixture`.
This helps beginners distinguish fixtures from tests; experienced users can ignore the warning via config.
Comment on lines +1 to +2
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
Issues a warning when a function is not collected as a test case just because it uses :py:func:`pytest.fixture`.
This helps beginners distinguish fixtures from tests; experienced users can ignore the warning via config.
A warning is now issued when a fixture uses the same name convention as a test function, for example a fixture named `test_user`.
This helps preventing beginners that expect anything named `test_` to be considered a test.

However I'm not convinced this warning is really necessary... besides being noisy for a common pattern (test_user is a good example of a fixture which provides a "test user", a "user for testing"), even if somebody confuses it is not the end of the world, because being a fixture, it will usually be called anyway by another test function (and if not then the user did not understand anything related to fixtures and functions and the warning won't help much anyway). It would make more sense if the situation was that the code were not being called at all (so silently being ignored), but this is not the case here.

I'm -0 on this:

  • Noisy for a common and valid pattern.
  • Even if a beginner confuses the two, the downside is not so severe because the fixture will be called anyway.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

silently-being-ignored sounds very tricky to catch, especially once you factor in partial runs with -k etc. That's for codecov to catch, although beginners probably wouldn't have that configured.

A lint rule does sound more suitable for this


.. code-block:: ini

[pytest]
filterwarnings =
ignore:.*becomes a fixture.*:pytest.PytestCollectionWarning
59 changes: 59 additions & 0 deletions doc/en/explanation/fixtures.rst
Original file line number Diff line number Diff line change
Expand Up @@ -88,6 +88,65 @@ prefer. You can also start out from existing :ref:`unittest.TestCase
style <unittest.TestCase>`.


Distinguishing fixtures and tests
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't think adding this section is net positive, if you think https://docs.pytest.org/en/stable/how-to/fixtures.html#how-to-fixtures doesn't explain well what a fixture is then the intro section should be improved.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I agree.

------------------------------------

Fixtures and tests may seem similar: both are functions (or methods) and both can utilize fixtures.
However, there is a fundamental difference:

- **Tests** are the leading role.
They can actively use :ref:`mark <mark>` and fixtures.

- **Fixtures** are supporting role.
They cannot use mark or tests and only be used directly or indirectly by test cases.


Two classic traps:


- Will not fail, because adding any tags to the fixture is invalid

.. code-block:: python

import warnings


@pytest.fixture
@pytest.mark.filterwarnings("error") # fixture cannot use mark
def server():
print("fixture is running!")


def test_foo(server):
warnings.warn(UserWarning("api v1, should use functions from v2"))



- No code will execute, because the ``test_foo`` becomes a fixture after using ``@pytest.fixture``


.. code-block:: python

@pytest.fixture
def server():
print("fixture is running!")


@pytest.fixture # turn test case into fixtures
def test_foo(server):
warnings.warn(UserWarning("api v1, should use functions from v2"))



.. versionadded:: 8.0

Applying a mark to a fixture function now issues a warning and will become an error in pytest 9.0.


.. versionadded:: 8.4

Issues a warning when a function is not collected as a test case just because it uses ``@pytest.fixture``


Fixture errors
--------------
Expand Down
1 change: 1 addition & 0 deletions pyproject.toml
Original file line number Diff line number Diff line change
Expand Up @@ -403,6 +403,7 @@ filterwarnings = [
"ignore:VendorImporter\\.find_spec\\(\\) not found; falling back to find_module\\(\\):ImportWarning",
# https://github.com/pytest-dev/execnet/pull/127
"ignore:isSet\\(\\) is deprecated, use is_set\\(\\) instead:DeprecationWarning",
"ignore:.*becomes a fixture.*:pytest.PytestCollectionWarning",
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

having to add an entry to filterwarnings is not great. What's the reason we need to add it?

]
pytester_example_dir = "testing/example_scripts"
markers = [
Expand Down
10 changes: 10 additions & 0 deletions src/_pytest/python.py
Original file line number Diff line number Diff line change
Expand Up @@ -354,6 +354,16 @@

def istestfunction(self, obj: object, name: str) -> bool:
if self.funcnamefilter(name) or self.isnosetest(obj):
if isinstance(obj, fixtures.FixtureFunctionDefinition):
self.warn(

Check warning on line 358 in src/_pytest/python.py

View check run for this annotation

Codecov / codecov/patch

src/_pytest/python.py#L358

Added line #L358 was not covered by tests
PytestCollectionWarning(
f"cannot collect test function {name!r},"
f"because it used the '@pytest.fixture' than becomes a fixture "
f"(from: {self.nodeid})"
Comment on lines +360 to +362
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
f"cannot collect test function {name!r},"
f"because it used the '@pytest.fixture' than becomes a fixture "
f"(from: {self.nodeid})"
f"failed to collect test function {name!r},"
f"because it uses '@pytest.fixture'. A function cannot simultaneously "
f"be both a test and a fixture.\n"
f"If you want to use this as a fixture, rename it to not start with "
f'"test_" to suppress this warning\n'
f"If you want to test the fixture, write a separate test that "
f"uses it.\n"
f"(from: {self.nodeid})"

Maybe something like that? Also I'm not sure we need both the name and the nodeid?

)
)
return False

Check warning on line 365 in src/_pytest/python.py

View check run for this annotation

Codecov / codecov/patch

src/_pytest/python.py#L365

Added line #L365 was not covered by tests

if isinstance(obj, (staticmethod, classmethod)):
# staticmethods and classmethods need to be unwrapped.
obj = safe_getattr(obj, "__func__", False)
Expand Down
20 changes: 20 additions & 0 deletions testing/python/collect.py
Original file line number Diff line number Diff line change
Expand Up @@ -783,6 +783,26 @@ def test_function_with_square_brackets(self, pytester: Pytester) -> None:
]
)

@pytest.mark.filterwarnings("default::UserWarning")
def test_function_used_fixture(self, pytester: Pytester):
pytester.makeini("[pytest]")
pytester.makepyfile(
"""
import pytest
@pytest.fixture
def test_function():
pass
"""
)
result = pytester.runpytest()
result.stdout.fnmatch_lines(
[
"collected 0 items",
"*== warnings summary ==*",
"*because it used the '@pytest.fixture' than becomes a fixture*",
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

when we're warning with an f-string I think it's good if this checks the entire warning string.
I also think it's good form to use result.assert_outcomes(warnings=1) here to make sure we're not outputting multiple warnings or something.

]
)


class TestSorting:
def test_check_equality(self, pytester: Pytester) -> None:
Expand Down
Loading