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

Add new RSpec/RepeatedSubjectCall cop #1722

Merged
merged 1 commit into from
Jan 10, 2024

Conversation

drcapulet
Copy link
Contributor

This adds a cop for catching uses of subject whereby you forget that it is memoized, often when trying to test idempotence:

it do
  expect { subject }.to change(Webhook.count).by(1)

  expect { subject }.to not_change(Webhook.count)
end

Had a hard time coming up with a succinct, descriptive name so open to suggestions.


Before submitting the PR make sure the following are checked:

  • Feature branch is up-to-date with master (if not - rebase it).
  • Squashed related commits together.
  • Added tests.
  • Updated documentation.
  • Added an entry to the CHANGELOG.md if the new code introduces user-observable changes.
  • The build (bundle exec rake) passes (be sure to run this locally, since it may produce updated documentation that you will need to commit).

If you have created a new cop:

  • Added the new cop to config/default.yml.
  • The cop is configured as Enabled: pending in config/default.yml.
  • The cop is configured as Enabled: true in .rubocop.yml.
  • The cop documents examples of good and bad code.
  • The tests assert both that bad code is reported and that good code is not reported.
  • Set VersionAdded: "<<next>>" in default/config.yml.

If you have modified an existing cop's configuration options:

  • Set VersionChanged: "<<next>>" in config/default.yml.

Copy link
Member

@pirj pirj left a comment

Choose a reason for hiding this comment

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

Thank you so much! This cop, indeed, is very helpful, especially with the ‘not to change’ case!
Let’s shape it up a bit.


# @!method subject_calls(node)
def_node_search :subject_calls, <<~PATTERN
(send nil? :subject)
Copy link
Member

Choose a reason for hiding this comment

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

I think we need to detect named subjects and memoized helpers in general, too.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Agreed - wasn't sure on the best way to achieve that especially given the complex searching we already do below with just subject.

Copy link
Member

Choose a reason for hiding this comment

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

This should work.

lib/rubocop/cop/rspec/repeated_subject_call.rb Outdated Show resolved Hide resolved
@drcapulet drcapulet force-pushed the alexc-repeated-subject-calls branch from 7bca2af to 2dde408 Compare October 2, 2023 13:31
Copy link
Member

@pirj pirj left a comment

Choose a reason for hiding this comment

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

Looks good already.
Leaving it up to you if you want to also cover named subjects, or leave it for a follow-up.
Thank you!

@pirj pirj requested review from bquorning, Darhazer and ydah October 10, 2023 23:14
@pirj
Copy link
Member

pirj commented Oct 11, 2023

Please rebase

@pirj
Copy link
Member

pirj commented Oct 11, 2023

I’m sorry, I broke something. I’m from my mobile. Can you please check?

@pirj
Copy link
Member

pirj commented Oct 11, 2023

It's strange, I've added:

  RSpec.describe RuboCop::Cop::RSpec::AlignRightLetBrace do
    # rubocop:disable RSpec/ExampleLength
    it 'registers offense for unaligned braces' do
+     expect { subject }.to change(:a)
+     expect { subject }.to change(:a)

and the cop didn't register that. 🤔

@drcapulet drcapulet force-pushed the alexc-repeated-subject-calls branch from ca12e41 to 00345a8 Compare December 11, 2023 21:44
@drcapulet
Copy link
Contributor Author

drcapulet commented Dec 11, 2023

@pirj Was a little trickier, but updated to support custom subjects!

@drcapulet drcapulet force-pushed the alexc-repeated-subject-calls branch from 00345a8 to d090985 Compare January 4, 2024 07:47
@drcapulet drcapulet requested a review from a team as a code owner January 4, 2024 07:47
Copy link
Member

@ydah ydah left a comment

Choose a reason for hiding this comment

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

Looks good! Thank you great work!

@bquorning bquorning mentioned this pull request Jan 4, 2024
6 tasks
@drcapulet drcapulet force-pushed the alexc-repeated-subject-calls branch from d090985 to cbe677d Compare January 10, 2024 07:41
@bquorning bquorning merged commit fd8c95e into rubocop:master Jan 10, 2024
24 checks passed
@drcapulet drcapulet deleted the alexc-repeated-subject-calls branch January 10, 2024 16:45
@timon
Copy link

timon commented Mar 1, 2024

This cop seems to bring in a lot of false positives, at least for me.

Here's the failing case:

describe LockManager do
  subject(:lock_manager) { described_class }

  describe '.lock!' do
    it 'requires a lock name and TTL in seconds' do # No offense raised
      expect { lock_manager.lock! }.to raise_error(ArgumentError, %r{wrong number of arguments.*required keyword:.*ttl})
    end

    it 'yields to a block if locking is successful' do # No offense raised
      expect { |block| lock_manager.lock!('some_lock', ttl: 1, &block) }.to yield_control
    end


    it 'raises an exception if locking is not successful', :aggregate_failures do
      lock_manager.lock!('some_lock', ttl: 3) do |_lock_held| # No offense raised
        expect do
          expect { |block| lock_manager.lock!('some_lock', ttl: 1, &block) }.not_to yield_control
          # C: RSpec/RepeatedSubjectCall: Calls to subject are memoized, this block is misleading
        end.to raise_error(Redlock::LockError, %r{failed to acquire.*some_lock})
      end
    end
  end
end

I find the actual error misleading:

  • there is no call to subject here
  • memoization is actually expected to happen

@pirj
Copy link
Member

pirj commented Mar 1, 2024

Doesn’t this PR fix this case for you, @timon ?

@timon
Copy link

timon commented Mar 1, 2024

No, dependabot PR with 2.27.0 just failed CI for me

No offence was registered for expect { |block| lock_manager.lock!(…) } line in previous version

@pirj
Copy link
Member

pirj commented Mar 1, 2024

You would have to use edge to get the fix from this PR, @timon

@timon
Copy link

timon commented Mar 1, 2024

@bquorning
Copy link
Collaborator

I think @pirj meant to ask if PR #1822 (and not this PR) fixes the issue. It was merged just about half an hour ago.

@timon
Copy link

timon commented Mar 1, 2024

Yes, I added github: 'rubocop/rubocop-rspec' to my Gemfile and no offenses were registered

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants