From b5295a5a62e926f38f24127f89d47ac823ada16d Mon Sep 17 00:00:00 2001 From: Tejas Bubane Date: Wed, 8 Jan 2025 15:16:59 +0400 Subject: [PATCH] [Fix #2001] Add new `RedundantContext` cop To check context with single example. Closes https://github.com/rubocop/rubocop-rspec/issues/2001 --- .rubocop.yml | 2 + CHANGELOG.md | 2 + config/default.yml | 6 +++ lib/rubocop/cop/rspec/redundant_context.rb | 39 +++++++++++++++++++ lib/rubocop/cop/rspec_cops.rb | 1 + .../cop/rspec/redundant_context_spec.rb | 25 ++++++++++++ 6 files changed, 75 insertions(+) create mode 100644 lib/rubocop/cop/rspec/redundant_context.rb create mode 100644 spec/rubocop/cop/rspec/redundant_context_spec.rb diff --git a/.rubocop.yml b/.rubocop.yml index 6fb427aef..9d16ea507 100644 --- a/.rubocop.yml +++ b/.rubocop.yml @@ -255,3 +255,5 @@ Style/YAMLFileRead: {Enabled: true} # RSpec/StringAsInstanceDoubleConstant: Enabled: true +RSpec/RedundantContext: + Enabled: true diff --git a/CHANGELOG.md b/CHANGELOG.md index aa199ab0c..769f25b2b 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -2,6 +2,8 @@ ## Master (Unreleased) +* Add new `RSpec/RedundantContext` cop. ([@tejasbubane]) + ## 3.3.0 (2024-12-12) - Deprecate `top_level_group?` method from `TopLevelGroup` mixin as all of its callers were intentionally removed from `Rubocop/RSpec`. ([@corsonknowles]) diff --git a/config/default.yml b/config/default.yml index 6c09b7d72..df3f32e6e 100644 --- a/config/default.yml +++ b/config/default.yml @@ -794,6 +794,12 @@ RSpec/RedundantAround: VersionAdded: '2.19' Reference: https://www.rubydoc.info/gems/rubocop-rspec/RuboCop/Cop/RSpec/RedundantAround +RSpec/RedundantContext: + Description: Detect redundant `context` hook. + Enabled: pending + VersionAdded: "<>" + Reference: https://www.rubydoc.info/gems/rubocop-rspec/RuboCop/Cop/RSpec/RedundantContext + RSpec/RedundantPredicateMatcher: Description: Checks for redundant predicate matcher. Enabled: true diff --git a/lib/rubocop/cop/rspec/redundant_context.rb b/lib/rubocop/cop/rspec/redundant_context.rb new file mode 100644 index 000000000..f4c4f82c9 --- /dev/null +++ b/lib/rubocop/cop/rspec/redundant_context.rb @@ -0,0 +1,39 @@ +# frozen_string_literal: true + +module RuboCop + module Cop + module RSpec + # Detect redundant `context` hook. + # + # @example + # # bad + # context 'when condition' do + # it 'tests something' do + # end + # end + # + # # good + # it 'tests something when condition' do + # end + # + class RedundantContext < Base + MSG = 'Redundant context with single example.' + + # @!method redundant_context?(node) + def_node_matcher :redundant_context?, <<~PATTERN + (block + (send #rspec? :context _) + _ + (block (send _ :it ...) ...)) + PATTERN + + def on_block(node) + return unless redundant_context?(node) + + add_offense(node) + end + alias on_numblock on_block + end + end + end +end diff --git a/lib/rubocop/cop/rspec_cops.rb b/lib/rubocop/cop/rspec_cops.rb index 85c177e8e..fa67cd921 100644 --- a/lib/rubocop/cop/rspec_cops.rb +++ b/lib/rubocop/cop/rspec_cops.rb @@ -81,6 +81,7 @@ require_relative 'rspec/receive_messages' require_relative 'rspec/receive_never' require_relative 'rspec/redundant_around' +require_relative 'rspec/redundant_context' require_relative 'rspec/redundant_predicate_matcher' require_relative 'rspec/remove_const' require_relative 'rspec/repeated_description' diff --git a/spec/rubocop/cop/rspec/redundant_context_spec.rb b/spec/rubocop/cop/rspec/redundant_context_spec.rb new file mode 100644 index 000000000..616d96dfe --- /dev/null +++ b/spec/rubocop/cop/rspec/redundant_context_spec.rb @@ -0,0 +1,25 @@ +# frozen_string_literal: true + +RSpec.describe RuboCop::Cop::RSpec::RedundantContext do + it 'registers an offense when single example inside context' do + expect_offense(<<~RUBY) + context 'when condition' do + ^^^^^^^^^^^^^^^^^^^^^^^^^^^ Redundant context with single example. + it 'does something' do + end + end + RUBY + end + + it 'does not register offense when multiple examples inside context' do + expect_no_offenses(<<~RUBY) + context 'when condition' do + it 'does something' do + end + + it 'does something else' do + end + end + RUBY + end +end