-
Notifications
You must be signed in to change notification settings - Fork 221
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
Reusable step functions regression: Specific step doesn't override generic one anymore #542
Comments
Thank you so much for testing pre-release versions! This is extremely helpful. |
I'm in early debugging of this, but it seems to be related to lines pytest-bdd/pytest_bdd/scenario.py Lines 41 to 43 in c0357d5
These lines are supposed to mimic what pytest does to determine the fixture to use. @The-Compiler do you think it would be possible for pytest to expose a public api to manipulate (or at least access) the fixture data structure? |
Thanks for having a look! Let me know if you need help somewhere, though I'm afraid I already have a lot on my plate...
In case you're curious: I run a GitHub workflow every night which (via tox) installs all requirements from their development repos (not transitive deps though, to keep things simple). That helped me find regressions in various projects. I should probably tweet or blog about it so that maybe more projects do this kind of thing...
Better exposing fixtures to plugins is something which has been discussed in the past, but from what I can gather, the fixture logic is unfortunately that part of pytest which nobody really dares to touch anymore, because even seemingly innocent changes caused regressions in some corner cases in the past. Somebody would need to step up to clean things up piece for piece, and unfortunately I don't think anybody is interested so far... |
@youtux @The-Compiler I've put some effort to provide such API for pytest-bdd https://github.com/elchupanebrej/pytest-bdd-ng/blob/default/pytest_bdd/plugin.py
|
My solution doesn't rely on the internal pytest fixtures mechanism, so it has to be pretty stable. It relies only on a cascade of fixture scopes, which, I believe, won't change in the future because it's one of the most fundamental parts of pytest itself. |
@elchupanebrej interesting approach. Does it work fine if you have step definitions defined somewhere and you import them in multiple modules multiple times? |
@youtux I don't think so (because of how registries are placed into module scope), but I'll check and return here. If it doesn't I'll prepare plugin-based solution |
I added a test that has many edge cases that should make this bug more visible: https://github.com/pytest-dev/pytest-bdd/pull/544/files#diff-79236ae04de04294d3972ec223e68c0268f0d14ab7bcda5e8ee9e9548661efc5R233. I run this test against version 6.0.1, and as I expected, it failed. It turned out that this was a bug for a very long time, it's just that with #534 we don't give higher priority to steps without parsers (they are not treated equally), so this bug is way more visible now. |
I've checked and from my perspective, it's not possible to just import steps without registration to make them work implicitly, and not use "hacky" ways based on internal pytest APIs (It is still possible to use the current pytest-bdd solution, but there should be context resolution issues or add pytest plugin patching on the fly. I don't like both approaches) |
Currently with pytest-bdd you can define your steps in a from steps.given import *
... I find this ability important, as it allows to make your step definitions modular. |
I propose another solution to re-use imported steps:
Usually, it's solved by placing step definitions at according conftest.py. The trade-off between comfortable step imports, overlapping steps hierarchy, re-implementing pytest internals, and monkey patching weights to overlapping steps hierarchy IMO(if there is a possibility to import steps in some way). Thanks for a good test for step hierarchy, it works perfectly with my solution |
To make my example more clear, in my generic
and my root level # conftest.py
from .steps.users.given import *
from .steps.users.when import *
from .steps.users.then import *
from .steps.order.given import *
from .steps.order.when import *
from .steps.order.then import *
# Steps for interaction with the browser
from .steps.browser.given import *
from .steps.browser.when import *
from .steps.browser.then import * This is to keep sanity and distinction between common steps for a specific part of the system. |
I like your project layout, will try to implement the idea from #310 about auto importing features. If the attempt would be successful there could be a good project layout which would have to be described at documentation as etalon |
Hey @The-Compiler, I think I found a fix (test suite is green), but I'd like to ask you to test your project against this fixed version. |
Sorry for the late answer here! I confirm things look good again: https://github.com/qutebrowser/qutebrowser/runs/7552888573?check_suite_focus=true#step:5:33 - thanks for the fix! |
After #534, with this
conftest.py
:this
test_specific.py
:and this
specific.feature
:I would expect that
specific
takes precedence overgeneric
, i.e., the value for the first test is"42"
, not"generic"
. This used to be the case, but isn't anymore after that commit:Note, however, it does work fine when
generic
is moved from theconftest.py
file intotest_specific.py
:The text was updated successfully, but these errors were encountered: