-
-
Notifications
You must be signed in to change notification settings - Fork 277
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
Add new RSpec/ExpectInHook
cop
#445
Conversation
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.
This is a good cop to have. I've seen many times people writing expectations in a before or after hook
# | ||
# # good | ||
# it do | ||
# expect_any_instance_of(Something).to receive(:foo) |
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.
Not connected to the cop itself, but since rubocop-rspec discourages the usage of expect_any_instance_of
, I would not add it in the good
examples
format(MSG, expect: expect, hook: hook) | ||
end | ||
|
||
def_node_matcher :hook, <<-PATTERN |
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.
In most cops the node_matchers are at the top
PATTERN | ||
|
||
def_node_search :expect, <<-PATTERN | ||
(send nil {:expect :expect_any_instance_of} ...) |
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.
You should also detect expects with block. Maybe this can be moved to language, as many cops would be interested in detecting an expectation. @backus what do you think?
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.
Yeah it would be nice to centralize as much RSpec DSL traversal as possible but I would only ask @pocke to do so if it is in this PR if is an easy change.
I updated this pull-request.
done, but the language helper is used only by |
Actually I've mislead you - you can check only for the send pattern and get directly the send node. |
Very nice! Will merge once the build is green 😄 |
@pocke I invited you to my custom CI setup on buildkite. You should be able to view the builds there now. They are slower for the first build of the day but if you push frequently like me then you will benefit from the ~20 second build times |
Could you provide some reasoning behind this statement?
|
The documentation says “Use I can’t remember having seen any documentation or guides recommending having expectations in the before/after hooks. |
Moreover, if the expectation in the before hook fails, all examples will fail. This also breaks the setup/run/verify arrangement, adding a verification in the setup phase |
In my understanding of the word "arbitrary", the phrase "arbitrary code" means "expectations and anything else" 🙂 As for the reason why you might want to use an expectation in a Also, current implementation produces a false positive for this code: before do
allow(User).to receive(:where) do |opts|
expect(opts[:signed_up_at]).to be_a(Date)
users
end
end |
IMO it's not a false positive, you have an expectation in the hook, although a conditional one |
I don't agree that it's "in the hook". The declaration is in the hook, but the execution is going to happen somewhere else. |
There are plenty of scenarios where you want to make expectations about code in a before block, and have that apply for each test. I'm disappointed to see this rule included, as it makes writing idiomatic tests for complex systems much more difficult. |
I think I agree with @seanlinsley. This PR description provides no justification for introducing the cop, it just says "expect should be in |
@aspiers Just like our parent project, RuboCop,
We provide different styles, and could even potentially provide a style that would make the cop insist on putting all expectations to hooks. We strive to align the default configuration with the Community RSpec style guide that says:
No doubt there are different approaches to covering the same code with tests. It's up to you to adjust the configuration to meet your established code style. If you feel that the default is off, please feel free to perform an analysis of specs of popular open-source projects that use RSpec. We're open to adjust the default if it turns out that expectations are quite often used in hooks. So does the community RSpec style guide. I hope this provides you with an answer to "why". |
@pirj Many thanks for taking the time to share this excellent explanation. It would be great if that was more easily discoverable for other users who walk into this particular cop 🙏 |
It's a long-term misconception that our default configuration is a de-facto standard for writing specs: The Ruby Style guide says:
Unfortunately, the Community RSpec Style guide is missing such a disclaimer. RSpec Core team is even more cautious in introducing an official style guide:
The gist is that you can write the code the way you see fit, the way that's best for you. While guidelines are a common ground for the majority of Ruby programmers, and tell the way of writing things in a certain way that doesn't come as surprise to anyone. It does not mean other ways are strictly prohibited if you know what you're doing and if you are comfortable to document the style guide amendments, train your colleagues to understand such style deviations, and be consistent about it. @aspiers Would you like to send a pull request with an update to
|
This cop adds an offense for usage of
expect
in hooks such asbefore
.expect
should be init
, not inbefore
. But RSpec says nothing about this code.