-
-
Notifications
You must be signed in to change notification settings - Fork 2.8k
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
Changes from all commits
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
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. | ||
|
||
.. code-block:: ini | ||
|
||
[pytest] | ||
filterwarnings = | ||
ignore:.*becomes a fixture.*:pytest.PytestCollectionWarning |
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -88,6 +88,65 @@ prefer. You can also start out from existing :ref:`unittest.TestCase | |
style <unittest.TestCase>`. | ||
|
||
|
||
Distinguishing fixtures and tests | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe 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. There was a problem hiding this comment. Choose a reason for hiding this commentThe 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 | ||
-------------- | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -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", | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe 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 = [ | ||
|
Original file line number | Diff line number | Diff line change | ||||||||||||||||||||||
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
|
@@ -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( | ||||||||||||||||||||||||
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
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
Suggested change
Maybe something like that? Also I'm not sure we need both the name and the nodeid? |
||||||||||||||||||||||||
) | ||||||||||||||||||||||||
) | ||||||||||||||||||||||||
return False | ||||||||||||||||||||||||
|
||||||||||||||||||||||||
if isinstance(obj, (staticmethod, classmethod)): | ||||||||||||||||||||||||
# staticmethods and classmethods need to be unwrapped. | ||||||||||||||||||||||||
obj = safe_getattr(obj, "__func__", False) | ||||||||||||||||||||||||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -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*", | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe 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. |
||
] | ||
) | ||
|
||
|
||
class TestSorting: | ||
def test_check_equality(self, pytester: Pytester) -> None: | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
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:
There was a problem hiding this comment.
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 forcodecov
to catch, although beginners probably wouldn't have that configured.A lint rule does sound more suitable for this