rubocop-rspec 1.38.1 → 1.43.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/CHANGELOG.md +57 -0
- data/CODE_OF_CONDUCT.md +17 -0
- data/README.md +1 -61
- data/config/default.yml +159 -19
- data/lib/rubocop-rspec.rb +5 -2
- data/lib/rubocop/cop/rspec/align_left_let_brace.rb +12 -19
- data/lib/rubocop/cop/rspec/align_right_let_brace.rb +12 -19
- data/lib/rubocop/cop/rspec/any_instance.rb +1 -1
- data/lib/rubocop/cop/rspec/around_block.rb +1 -1
- data/lib/rubocop/cop/rspec/base.rb +74 -0
- data/lib/rubocop/cop/rspec/be.rb +2 -2
- data/lib/rubocop/cop/rspec/be_eql.rb +6 -6
- data/lib/rubocop/cop/rspec/before_after_all.rb +1 -1
- data/lib/rubocop/cop/rspec/capybara/current_path_expectation.rb +19 -17
- data/lib/rubocop/cop/rspec/capybara/feature_methods.rb +14 -12
- data/lib/rubocop/cop/rspec/capybara/visibility_matcher.rb +69 -0
- data/lib/rubocop/cop/rspec/context_method.rb +7 -9
- data/lib/rubocop/cop/rspec/context_wording.rb +3 -3
- data/lib/rubocop/cop/rspec/cop.rb +3 -87
- data/lib/rubocop/cop/rspec/describe_class.rb +29 -23
- data/lib/rubocop/cop/rspec/describe_method.rb +14 -7
- data/lib/rubocop/cop/rspec/describe_symbol.rb +2 -2
- data/lib/rubocop/cop/rspec/described_class.rb +12 -9
- data/lib/rubocop/cop/rspec/described_class_module_wrapping.rb +1 -1
- data/lib/rubocop/cop/rspec/dialect.rb +5 -12
- data/lib/rubocop/cop/rspec/empty_example_group.rb +91 -7
- data/lib/rubocop/cop/rspec/empty_hook.rb +46 -0
- data/lib/rubocop/cop/rspec/empty_line_after_example.rb +5 -7
- data/lib/rubocop/cop/rspec/empty_line_after_example_group.rb +5 -9
- data/lib/rubocop/cop/rspec/empty_line_after_final_let.rb +8 -8
- data/lib/rubocop/cop/rspec/empty_line_after_hook.rb +5 -9
- data/lib/rubocop/cop/rspec/empty_line_after_subject.rb +6 -6
- data/lib/rubocop/cop/rspec/example_length.rb +1 -1
- data/lib/rubocop/cop/rspec/example_without_description.rb +1 -1
- data/lib/rubocop/cop/rspec/example_wording.rb +10 -11
- data/lib/rubocop/cop/rspec/expect_actual.rb +8 -11
- data/lib/rubocop/cop/rspec/expect_change.rb +10 -35
- data/lib/rubocop/cop/rspec/expect_in_hook.rb +3 -3
- data/lib/rubocop/cop/rspec/expect_output.rb +2 -2
- data/lib/rubocop/cop/rspec/factory_bot/attribute_defined_statically.rb +24 -21
- data/lib/rubocop/cop/rspec/factory_bot/create_list.rb +20 -22
- data/lib/rubocop/cop/rspec/factory_bot/factory_class_name.rb +7 -8
- data/lib/rubocop/cop/rspec/file_path.rb +57 -21
- data/lib/rubocop/cop/rspec/focus.rb +7 -11
- data/lib/rubocop/cop/rspec/hook_argument.rb +16 -23
- data/lib/rubocop/cop/rspec/hooks_before_examples.rb +10 -29
- data/lib/rubocop/cop/rspec/implicit_block_expectation.rb +1 -1
- data/lib/rubocop/cop/rspec/implicit_expect.rb +7 -15
- data/lib/rubocop/cop/rspec/implicit_subject.rb +16 -11
- data/lib/rubocop/cop/rspec/instance_spy.rb +18 -12
- data/lib/rubocop/cop/rspec/instance_variable.rb +4 -8
- data/lib/rubocop/cop/rspec/invalid_predicate_matcher.rb +3 -6
- data/lib/rubocop/cop/rspec/it_behaves_like.rb +5 -6
- data/lib/rubocop/cop/rspec/iterated_expectation.rb +1 -1
- data/lib/rubocop/cop/rspec/leading_subject.rb +22 -26
- data/lib/rubocop/cop/rspec/leaky_constant_declaration.rb +2 -5
- data/lib/rubocop/cop/rspec/let_before_examples.rb +10 -26
- data/lib/rubocop/cop/rspec/let_setup.rb +21 -6
- data/lib/rubocop/cop/rspec/message_chain.rb +7 -6
- data/lib/rubocop/cop/rspec/message_expectation.rb +2 -2
- data/lib/rubocop/cop/rspec/message_spies.rb +2 -3
- data/lib/rubocop/cop/rspec/missing_example_group_argument.rb +1 -1
- data/lib/rubocop/cop/rspec/multiple_describes.rb +11 -8
- data/lib/rubocop/cop/rspec/multiple_expectations.rb +7 -11
- data/lib/rubocop/cop/rspec/multiple_memoized_helpers.rb +148 -0
- data/lib/rubocop/cop/rspec/multiple_subjects.rb +18 -19
- data/lib/rubocop/cop/rspec/named_subject.rb +8 -8
- data/lib/rubocop/cop/rspec/nested_groups.rb +12 -13
- data/lib/rubocop/cop/rspec/not_to_not.rb +5 -6
- data/lib/rubocop/cop/rspec/overwriting_setup.rb +1 -1
- data/lib/rubocop/cop/rspec/pending.rb +1 -1
- data/lib/rubocop/cop/rspec/predicate_matcher.rb +32 -69
- data/lib/rubocop/cop/rspec/rails/http_status.rb +7 -9
- data/lib/rubocop/cop/rspec/receive_counts.rb +15 -17
- data/lib/rubocop/cop/rspec/receive_never.rb +12 -12
- data/lib/rubocop/cop/rspec/repeated_description.rb +1 -1
- data/lib/rubocop/cop/rspec/repeated_example.rb +2 -2
- data/lib/rubocop/cop/rspec/repeated_example_group_body.rb +12 -2
- data/lib/rubocop/cop/rspec/repeated_example_group_description.rb +1 -1
- data/lib/rubocop/cop/rspec/return_from_stub.rb +12 -22
- data/lib/rubocop/cop/rspec/scattered_let.rb +12 -2
- data/lib/rubocop/cop/rspec/scattered_setup.rb +1 -1
- data/lib/rubocop/cop/rspec/shared_context.rb +8 -21
- data/lib/rubocop/cop/rspec/shared_examples.rb +7 -9
- data/lib/rubocop/cop/rspec/single_argument_message_chain.rb +15 -18
- data/lib/rubocop/cop/rspec/subject_stub.rb +25 -53
- data/lib/rubocop/cop/rspec/unspecified_exception.rb +1 -1
- data/lib/rubocop/cop/rspec/variable_definition.rb +56 -0
- data/lib/rubocop/cop/rspec/variable_name.rb +66 -0
- data/lib/rubocop/cop/rspec/verified_doubles.rb +1 -1
- data/lib/rubocop/cop/rspec/void_expect.rb +1 -1
- data/lib/rubocop/cop/rspec/yield.rb +14 -11
- data/lib/rubocop/cop/rspec_cops.rb +6 -1
- data/lib/rubocop/rspec/corrector/move_node.rb +54 -0
- data/lib/rubocop/rspec/description_extractor.rb +2 -6
- data/lib/rubocop/rspec/{blank_line_separation.rb → empty_line_separation.rb} +13 -10
- data/lib/rubocop/rspec/example_group.rb +21 -49
- data/lib/rubocop/rspec/factory_bot.rb +7 -1
- data/lib/rubocop/rspec/language.rb +13 -3
- data/lib/rubocop/rspec/language/node_pattern.rb +11 -2
- data/lib/rubocop/rspec/top_level_describe.rb +2 -2
- data/lib/rubocop/rspec/top_level_group.rb +55 -0
- data/lib/rubocop/rspec/variable.rb +16 -0
- data/lib/rubocop/rspec/version.rb +1 -1
- metadata +36 -13
- data/lib/rubocop/rspec/util.rb +0 -19
| @@ -26,7 +26,8 @@ module RuboCop | |
| 26 26 | 
             
                  #   # good
         | 
| 27 27 | 
             
                  #   it { expect(subject).to be_truthy }
         | 
| 28 28 | 
             
                  #
         | 
| 29 | 
            -
                  class ImplicitSubject <  | 
| 29 | 
            +
                  class ImplicitSubject < Base
         | 
| 30 | 
            +
                    extend AutoCorrector
         | 
| 30 31 | 
             
                    include ConfigurableEnforcedStyle
         | 
| 31 32 |  | 
| 32 33 | 
             
                    MSG = "Don't use implicit subject."
         | 
| @@ -39,33 +40,37 @@ module RuboCop | |
| 39 40 | 
             
                      return unless implicit_subject?(node)
         | 
| 40 41 | 
             
                      return if valid_usage?(node)
         | 
| 41 42 |  | 
| 42 | 
            -
                      add_offense(node)
         | 
| 43 | 
            +
                      add_offense(node) do |corrector|
         | 
| 44 | 
            +
                        autocorrect(corrector, node)
         | 
| 45 | 
            +
                      end
         | 
| 43 46 | 
             
                    end
         | 
| 44 47 |  | 
| 45 | 
            -
                     | 
| 48 | 
            +
                    private
         | 
| 49 | 
            +
             | 
| 50 | 
            +
                    def autocorrect(corrector, node)
         | 
| 46 51 | 
             
                      replacement = 'expect(subject)'
         | 
| 47 | 
            -
                       | 
| 52 | 
            +
                      case node.method_name
         | 
| 53 | 
            +
                      when :should
         | 
| 48 54 | 
             
                        replacement += '.to'
         | 
| 49 | 
            -
                       | 
| 55 | 
            +
                      when :should_not
         | 
| 50 56 | 
             
                        replacement += '.not_to'
         | 
| 51 57 | 
             
                      end
         | 
| 52 58 |  | 
| 53 | 
            -
                       | 
| 59 | 
            +
                      corrector.replace(node.loc.selector, replacement)
         | 
| 54 60 | 
             
                    end
         | 
| 55 61 |  | 
| 56 | 
            -
                    private
         | 
| 57 | 
            -
             | 
| 58 62 | 
             
                    def valid_usage?(node)
         | 
| 59 63 | 
             
                      example = node.ancestors.find { |parent| example?(parent) }
         | 
| 60 64 | 
             
                      return false if example.nil?
         | 
| 61 65 |  | 
| 62 | 
            -
                      example. | 
| 66 | 
            +
                      example.method?(:its) || allowed_by_style?(example)
         | 
| 63 67 | 
             
                    end
         | 
| 64 68 |  | 
| 65 69 | 
             
                    def allowed_by_style?(example)
         | 
| 66 | 
            -
                       | 
| 70 | 
            +
                      case style
         | 
| 71 | 
            +
                      when :single_line_only
         | 
| 67 72 | 
             
                        example.single_line?
         | 
| 68 | 
            -
                       | 
| 73 | 
            +
                      when :single_statement_only
         | 
| 69 74 | 
             
                        !example.body.begin_type?
         | 
| 70 75 | 
             
                      else
         | 
| 71 76 | 
             
                        false
         | 
| @@ -18,7 +18,9 @@ module RuboCop | |
| 18 18 | 
             
                  #     expect(foo).to have_received(:bar)
         | 
| 19 19 | 
             
                  #   end
         | 
| 20 20 | 
             
                  #
         | 
| 21 | 
            -
                  class InstanceSpy <  | 
| 21 | 
            +
                  class InstanceSpy < Base
         | 
| 22 | 
            +
                    extend AutoCorrector
         | 
| 23 | 
            +
             | 
| 22 24 | 
             
                    MSG = 'Use `instance_spy` when you check your double '\
         | 
| 23 25 | 
             
                          'with `have_received`.'
         | 
| 24 26 |  | 
| @@ -43,22 +45,26 @@ module RuboCop | |
| 43 45 |  | 
| 44 46 | 
             
                      null_double(node) do |var, receiver|
         | 
| 45 47 | 
             
                        have_received_usage(node) do |expected|
         | 
| 46 | 
            -
                           | 
| 48 | 
            +
                          next if expected != var
         | 
| 49 | 
            +
             | 
| 50 | 
            +
                          add_offense(receiver) do |corrector|
         | 
| 51 | 
            +
                            autocorrect(corrector, receiver)
         | 
| 52 | 
            +
                          end
         | 
| 47 53 | 
             
                        end
         | 
| 48 54 | 
             
                      end
         | 
| 49 55 | 
             
                    end
         | 
| 50 56 |  | 
| 51 | 
            -
                     | 
| 52 | 
            -
                      lambda do |corrector|
         | 
| 53 | 
            -
                        replacement = 'instance_spy'
         | 
| 54 | 
            -
                        corrector.replace(node.loc.selector, replacement)
         | 
| 57 | 
            +
                    private
         | 
| 55 58 |  | 
| 56 | 
            -
             | 
| 57 | 
            -
             | 
| 58 | 
            -
             | 
| 59 | 
            -
             | 
| 60 | 
            -
             | 
| 61 | 
            -
                       | 
| 59 | 
            +
                    def autocorrect(corrector, node)
         | 
| 60 | 
            +
                      replacement = 'instance_spy'
         | 
| 61 | 
            +
                      corrector.replace(node.loc.selector, replacement)
         | 
| 62 | 
            +
             | 
| 63 | 
            +
                      double_source_map = node.parent.loc
         | 
| 64 | 
            +
                      as_null_object_range = double_source_map
         | 
| 65 | 
            +
                        .dot
         | 
| 66 | 
            +
                        .join(double_source_map.selector)
         | 
| 67 | 
            +
                      corrector.remove(as_null_object_range)
         | 
| 62 68 | 
             
                    end
         | 
| 63 69 | 
             
                  end
         | 
| 64 70 | 
             
                end
         | 
| @@ -46,14 +46,12 @@ module RuboCop | |
| 46 46 | 
             
                  #     it { expect(foo).to be_empty }
         | 
| 47 47 | 
             
                  #   end
         | 
| 48 48 | 
             
                  #
         | 
| 49 | 
            -
                  class InstanceVariable <  | 
| 49 | 
            +
                  class InstanceVariable < Base
         | 
| 50 | 
            +
                    include RuboCop::RSpec::TopLevelGroup
         | 
| 51 | 
            +
             | 
| 50 52 | 
             
                    MSG = 'Avoid instance variables – use let, ' \
         | 
| 51 53 | 
             
                          'a method call, or a local variable (if possible).'
         | 
| 52 54 |  | 
| 53 | 
            -
                    EXAMPLE_GROUP_METHODS = ExampleGroups::ALL + SharedGroups::ALL
         | 
| 54 | 
            -
             | 
| 55 | 
            -
                    def_node_matcher :spec_group?, EXAMPLE_GROUP_METHODS.block_pattern
         | 
| 56 | 
            -
             | 
| 57 55 | 
             
                    def_node_matcher :dynamic_class?, <<-PATTERN
         | 
| 58 56 | 
             
                      (block (send (const nil? :Class) :new ...) ...)
         | 
| 59 57 | 
             
                    PATTERN
         | 
| @@ -69,9 +67,7 @@ module RuboCop | |
| 69 67 |  | 
| 70 68 | 
             
                    def_node_search :ivar_assigned?, '(ivasgn % ...)'
         | 
| 71 69 |  | 
| 72 | 
            -
                    def  | 
| 73 | 
            -
                      return unless spec_group?(node)
         | 
| 74 | 
            -
             | 
| 70 | 
            +
                    def on_top_level_group(node)
         | 
| 75 71 | 
             
                      ivar_usage(node) do |ivar, name|
         | 
| 76 72 | 
             
                        next if valid_usage?(ivar)
         | 
| 77 73 | 
             
                        next if assignment_only? && !ivar_assigned?(node, name)
         | 
| @@ -15,7 +15,7 @@ module RuboCop | |
| 15 15 | 
             
                  #
         | 
| 16 16 | 
             
                  #   # good
         | 
| 17 17 | 
             
                  #   expect(foo).to be_something
         | 
| 18 | 
            -
                  class InvalidPredicateMatcher <  | 
| 18 | 
            +
                  class InvalidPredicateMatcher < Base
         | 
| 19 19 | 
             
                    MSG = 'Omit `?` from `%<matcher>s`.'
         | 
| 20 20 |  | 
| 21 21 | 
             
                    def_node_matcher :invalid_predicate_matcher?, <<-PATTERN
         | 
| @@ -24,7 +24,8 @@ module RuboCop | |
| 24 24 |  | 
| 25 25 | 
             
                    def on_send(node)
         | 
| 26 26 | 
             
                      invalid_predicate_matcher?(node) do |predicate|
         | 
| 27 | 
            -
                        add_offense(predicate | 
| 27 | 
            +
                        add_offense(predicate,
         | 
| 28 | 
            +
                                    message: format(MSG, matcher: predicate.method_name))
         | 
| 28 29 | 
             
                      end
         | 
| 29 30 | 
             
                    end
         | 
| 30 31 |  | 
| @@ -34,10 +35,6 @@ module RuboCop | |
| 34 35 | 
             
                      name = name.to_s
         | 
| 35 36 | 
             
                      name.start_with?('be_', 'have_') && name.end_with?('?')
         | 
| 36 37 | 
             
                    end
         | 
| 37 | 
            -
             | 
| 38 | 
            -
                    def message(predicate)
         | 
| 39 | 
            -
                      format(MSG, matcher: predicate.method_name)
         | 
| 40 | 
            -
                    end
         | 
| 41 38 | 
             
                  end
         | 
| 42 39 | 
             
                end
         | 
| 43 40 | 
             
              end
         | 
| @@ -18,7 +18,8 @@ module RuboCop | |
| 18 18 | 
             
                  #
         | 
| 19 19 | 
             
                  #   # good
         | 
| 20 20 | 
             
                  #   it_should_behave_like 'a foo'
         | 
| 21 | 
            -
                  class ItBehavesLike <  | 
| 21 | 
            +
                  class ItBehavesLike < Base
         | 
| 22 | 
            +
                    extend AutoCorrector
         | 
| 22 23 | 
             
                    include ConfigurableEnforcedStyle
         | 
| 23 24 |  | 
| 24 25 | 
             
                    MSG = 'Prefer `%<replacement>s` over `%<original>s` when including '\
         | 
| @@ -28,14 +29,12 @@ module RuboCop | |
| 28 29 |  | 
| 29 30 | 
             
                    def on_send(node)
         | 
| 30 31 | 
             
                      example_inclusion_offense(node, alternative_style) do
         | 
| 31 | 
            -
                        add_offense(node)
         | 
| 32 | 
            +
                        add_offense(node) do |corrector|
         | 
| 33 | 
            +
                          corrector.replace(node.loc.selector, style.to_s)
         | 
| 34 | 
            +
                        end
         | 
| 32 35 | 
             
                      end
         | 
| 33 36 | 
             
                    end
         | 
| 34 37 |  | 
| 35 | 
            -
                    def autocorrect(node)
         | 
| 36 | 
            -
                      ->(corrector) { corrector.replace(node.loc.selector, style.to_s) }
         | 
| 37 | 
            -
                    end
         | 
| 38 | 
            -
             | 
| 39 38 | 
             
                    private
         | 
| 40 39 |  | 
| 41 40 | 
             
                    def message(_node)
         | 
| @@ -15,7 +15,7 @@ module RuboCop | |
| 15 15 | 
             
                  #   it 'validates users' do
         | 
| 16 16 | 
             
                  #     expect([user1, user2, user3]).to all(be_valid)
         | 
| 17 17 | 
             
                  #   end
         | 
| 18 | 
            -
                  class IteratedExpectation <  | 
| 18 | 
            +
                  class IteratedExpectation < Base
         | 
| 19 19 | 
             
                    MSG = 'Prefer using the `all` matcher instead ' \
         | 
| 20 20 | 
             
                              'of iterating over an array.'
         | 
| 21 21 |  | 
| @@ -31,8 +31,8 @@ module RuboCop | |
| 31 31 | 
             
                  #     it { expect_something }
         | 
| 32 32 | 
             
                  #     it { expect_something_else }
         | 
| 33 33 | 
             
                  #
         | 
| 34 | 
            -
                  class LeadingSubject <  | 
| 35 | 
            -
                     | 
| 34 | 
            +
                  class LeadingSubject < Base
         | 
| 35 | 
            +
                    extend AutoCorrector
         | 
| 36 36 |  | 
| 37 37 | 
             
                    MSG = 'Declare `subject` above any other `%<offending>s` declarations.'
         | 
| 38 38 |  | 
| @@ -43,40 +43,36 @@ module RuboCop | |
| 43 43 | 
             
                    end
         | 
| 44 44 |  | 
| 45 45 | 
             
                    def check_previous_nodes(node)
         | 
| 46 | 
            -
                      node | 
| 47 | 
            -
                         | 
| 48 | 
            -
             | 
| 49 | 
            -
             | 
| 50 | 
            -
                            message: format(MSG, offending: sibling.method_name)
         | 
| 51 | 
            -
                          )
         | 
| 46 | 
            +
                      offending_node(node) do |offender|
         | 
| 47 | 
            +
                        msg = format(MSG, offending: offender.method_name)
         | 
| 48 | 
            +
                        add_offense(node, message: msg) do |corrector|
         | 
| 49 | 
            +
                          autocorrect(corrector, node, offender)
         | 
| 52 50 | 
             
                        end
         | 
| 53 | 
            -
             | 
| 54 | 
            -
                        break if offending?(sibling) || sibling.equal?(node)
         | 
| 55 | 
            -
                      end
         | 
| 56 | 
            -
                    end
         | 
| 57 | 
            -
             | 
| 58 | 
            -
                    def autocorrect(node)
         | 
| 59 | 
            -
                      lambda do |corrector|
         | 
| 60 | 
            -
                        first_node = find_first_offending_node(node)
         | 
| 61 | 
            -
                        first_node_position = first_node.loc.expression
         | 
| 62 | 
            -
                        indent = "\n" + ' ' * first_node.loc.column
         | 
| 63 | 
            -
                        corrector.insert_before(first_node_position, node.source + indent)
         | 
| 64 | 
            -
                        corrector.remove(node_range(node))
         | 
| 65 51 | 
             
                      end
         | 
| 66 52 | 
             
                    end
         | 
| 67 53 |  | 
| 68 54 | 
             
                    private
         | 
| 69 55 |  | 
| 70 | 
            -
                    def  | 
| 71 | 
            -
                       | 
| 56 | 
            +
                    def offending_node(node)
         | 
| 57 | 
            +
                      node.parent.each_child_node.find do |sibling|
         | 
| 58 | 
            +
                        break if sibling.equal?(node)
         | 
| 59 | 
            +
             | 
| 60 | 
            +
                        yield sibling if offending?(sibling)
         | 
| 61 | 
            +
                      end
         | 
| 72 62 | 
             
                    end
         | 
| 73 63 |  | 
| 74 | 
            -
                    def  | 
| 75 | 
            -
                       | 
| 64 | 
            +
                    def autocorrect(corrector, node, sibling)
         | 
| 65 | 
            +
                      RuboCop::RSpec::Corrector::MoveNode.new(
         | 
| 66 | 
            +
                        node, corrector, processed_source
         | 
| 67 | 
            +
                      ).move_before(sibling)
         | 
| 76 68 | 
             
                    end
         | 
| 77 69 |  | 
| 78 | 
            -
                    def  | 
| 79 | 
            -
                       | 
| 70 | 
            +
                    def offending?(node)
         | 
| 71 | 
            +
                      let?(node) ||
         | 
| 72 | 
            +
                        hook?(node) ||
         | 
| 73 | 
            +
                        example?(node) ||
         | 
| 74 | 
            +
                        spec_group?(node) ||
         | 
| 75 | 
            +
                        include?(node)
         | 
| 80 76 | 
             
                    end
         | 
| 81 77 |  | 
| 82 78 | 
             
                    def in_spec_block?(node)
         | 
| @@ -93,7 +93,7 @@ module RuboCop | |
| 93 93 | 
             
                  #       stub_const('SomeModule::SomeClass', foo_class)
         | 
| 94 94 | 
             
                  #     end
         | 
| 95 95 | 
             
                  #   end
         | 
| 96 | 
            -
                  class LeakyConstantDeclaration <  | 
| 96 | 
            +
                  class LeakyConstantDeclaration < Base
         | 
| 97 97 | 
             
                    MSG_CONST = 'Stub constant instead of declaring explicitly.'
         | 
| 98 98 | 
             
                    MSG_CLASS = 'Stub class constant instead of declaring explicitly.'
         | 
| 99 99 | 
             
                    MSG_MODULE = 'Stub module constant instead of declaring explicitly.'
         | 
| @@ -119,11 +119,8 @@ module RuboCop | |
| 119 119 | 
             
                    private
         | 
| 120 120 |  | 
| 121 121 | 
             
                    def inside_describe_block?(node)
         | 
| 122 | 
            -
                      node.each_ancestor(:block).any?(&method(: | 
| 122 | 
            +
                      node.each_ancestor(:block).any?(&method(:spec_group?))
         | 
| 123 123 | 
             
                    end
         | 
| 124 | 
            -
             | 
| 125 | 
            -
                    def_node_matcher :in_example_or_shared_group?,
         | 
| 126 | 
            -
                                     (ExampleGroups::ALL + SharedGroups::ALL).block_pattern
         | 
| 127 124 | 
             
                  end
         | 
| 128 125 | 
             
                end
         | 
| 129 126 | 
             
              end
         | 
| @@ -30,9 +30,8 @@ module RuboCop | |
| 30 30 | 
             
                  #   it 'checks what some does' do
         | 
| 31 31 | 
             
                  #     expect(some).to be
         | 
| 32 32 | 
             
                  #   end
         | 
| 33 | 
            -
                  class LetBeforeExamples <  | 
| 34 | 
            -
                     | 
| 35 | 
            -
                    include RuboCop::RSpec::FinalEndLocation
         | 
| 33 | 
            +
                  class LetBeforeExamples < Base
         | 
| 34 | 
            +
                    extend AutoCorrector
         | 
| 36 35 |  | 
| 37 36 | 
             
                    MSG = 'Move `let` before the examples in the group.'
         | 
| 38 37 |  | 
| @@ -49,17 +48,6 @@ module RuboCop | |
| 49 48 | 
             
                      check_let_declarations(node.body) if multiline_block?(node.body)
         | 
| 50 49 | 
             
                    end
         | 
| 51 50 |  | 
| 52 | 
            -
                    def autocorrect(node)
         | 
| 53 | 
            -
                      lambda do |corrector|
         | 
| 54 | 
            -
                        first_example = find_first_example(node.parent)
         | 
| 55 | 
            -
                        first_example_pos = first_example.loc.expression
         | 
| 56 | 
            -
                        indent = "\n" + ' ' * first_example.loc.column
         | 
| 57 | 
            -
             | 
| 58 | 
            -
                        corrector.insert_before(first_example_pos, source(node) + indent)
         | 
| 59 | 
            -
                        corrector.remove(node_range_with_surrounding_space(node))
         | 
| 60 | 
            -
                      end
         | 
| 61 | 
            -
                    end
         | 
| 62 | 
            -
             | 
| 63 51 | 
             
                    private
         | 
| 64 52 |  | 
| 65 53 | 
             
                    def multiline_block?(block)
         | 
| @@ -72,8 +60,11 @@ module RuboCop | |
| 72 60 |  | 
| 73 61 | 
             
                      node.each_child_node do |child|
         | 
| 74 62 | 
             
                        next if child.sibling_index < first_example.sibling_index
         | 
| 63 | 
            +
                        next unless let?(child)
         | 
| 75 64 |  | 
| 76 | 
            -
                        add_offense(child)  | 
| 65 | 
            +
                        add_offense(child) do |corrector|
         | 
| 66 | 
            +
                          autocorrect(corrector, child, first_example)
         | 
| 67 | 
            +
                        end
         | 
| 77 68 | 
             
                      end
         | 
| 78 69 | 
             
                    end
         | 
| 79 70 |  | 
| @@ -81,17 +72,10 @@ module RuboCop | |
| 81 72 | 
             
                      node.children.find { |sibling| example_or_group?(sibling) }
         | 
| 82 73 | 
             
                    end
         | 
| 83 74 |  | 
| 84 | 
            -
                    def  | 
| 85 | 
            -
                       | 
| 86 | 
            -
             | 
| 87 | 
            -
             | 
| 88 | 
            -
             | 
| 89 | 
            -
                    def source(node)
         | 
| 90 | 
            -
                      node_range(node).source
         | 
| 91 | 
            -
                    end
         | 
| 92 | 
            -
             | 
| 93 | 
            -
                    def node_range(node)
         | 
| 94 | 
            -
                      node.loc.expression.with(end_pos: final_end_location(node).end_pos)
         | 
| 75 | 
            +
                    def autocorrect(corrector, node, first_example)
         | 
| 76 | 
            +
                      RuboCop::RSpec::Corrector::MoveNode.new(
         | 
| 77 | 
            +
                        node, corrector, processed_source
         | 
| 78 | 
            +
                      ).move_before(first_example)
         | 
| 95 79 | 
             
                    end
         | 
| 96 80 | 
             
                  end
         | 
| 97 81 | 
             
                end
         | 
| @@ -25,17 +25,26 @@ module RuboCop | |
| 25 25 | 
             
                  #   it 'counts widgets' do
         | 
| 26 26 | 
             
                  #     expect(Widget.count).to eq(1)
         | 
| 27 27 | 
             
                  #   end
         | 
| 28 | 
            -
                  class LetSetup <  | 
| 28 | 
            +
                  class LetSetup < Base
         | 
| 29 29 | 
             
                    MSG = 'Do not use `let!` to setup objects not referenced in tests.'
         | 
| 30 30 |  | 
| 31 | 
            -
                     | 
| 32 | 
            -
             | 
| 31 | 
            +
                    def_node_matcher :example_or_shared_group_or_including?,
         | 
| 32 | 
            +
                                     (
         | 
| 33 | 
            +
                                       ExampleGroups::ALL + SharedGroups::ALL +
         | 
| 34 | 
            +
                                       Includes::ALL
         | 
| 35 | 
            +
                                     ).block_pattern
         | 
| 36 | 
            +
             | 
| 37 | 
            +
                    def_node_matcher :let_bang, <<-PATTERN
         | 
| 38 | 
            +
                      {
         | 
| 39 | 
            +
                        (block $(send nil? :let! {(sym $_) (str $_)}) ...)
         | 
| 40 | 
            +
                        $(send nil? :let! {(sym $_) (str $_)} block_pass)
         | 
| 41 | 
            +
                      }
         | 
| 33 42 | 
             
                    PATTERN
         | 
| 34 43 |  | 
| 35 44 | 
             
                    def_node_search :method_called?, '(send nil? %)'
         | 
| 36 45 |  | 
| 37 46 | 
             
                    def on_block(node)
         | 
| 38 | 
            -
                      return unless  | 
| 47 | 
            +
                      return unless example_or_shared_group_or_including?(node)
         | 
| 39 48 |  | 
| 40 49 | 
             
                      unused_let_bang(node) do |let|
         | 
| 41 50 | 
             
                        add_offense(let)
         | 
| @@ -45,8 +54,14 @@ module RuboCop | |
| 45 54 | 
             
                    private
         | 
| 46 55 |  | 
| 47 56 | 
             
                    def unused_let_bang(node)
         | 
| 48 | 
            -
                       | 
| 49 | 
            -
                        yield(method_send) unless method_called?(node, method_name)
         | 
| 57 | 
            +
                      child_let_bang(node) do |method_send, method_name|
         | 
| 58 | 
            +
                        yield(method_send) unless method_called?(node, method_name.to_sym)
         | 
| 59 | 
            +
                      end
         | 
| 60 | 
            +
                    end
         | 
| 61 | 
            +
             | 
| 62 | 
            +
                    def child_let_bang(node, &block)
         | 
| 63 | 
            +
                      RuboCop::RSpec::ExampleGroup.new(node).lets.each do |let|
         | 
| 64 | 
            +
                        let_bang(let, &block)
         | 
| 50 65 | 
             
                      end
         | 
| 51 66 | 
             
                    end
         | 
| 52 67 | 
             
                  end
         | 
| @@ -13,7 +13,7 @@ module RuboCop | |
| 13 13 | 
             
                  #   thing = Thing.new(baz: 42)
         | 
| 14 14 | 
             
                  #   allow(foo).to receive(:bar).and_return(thing)
         | 
| 15 15 | 
             
                  #
         | 
| 16 | 
            -
                  class MessageChain <  | 
| 16 | 
            +
                  class MessageChain < Base
         | 
| 17 17 | 
             
                    MSG = 'Avoid stubbing using `%<method>s`.'
         | 
| 18 18 |  | 
| 19 19 | 
             
                    def_node_matcher :message_chain, <<-PATTERN
         | 
| @@ -21,11 +21,12 @@ module RuboCop | |
| 21 21 | 
             
                    PATTERN
         | 
| 22 22 |  | 
| 23 23 | 
             
                    def on_send(node)
         | 
| 24 | 
            -
                      message_chain(node)  | 
| 25 | 
            -
             | 
| 26 | 
            -
             | 
| 27 | 
            -
             | 
| 28 | 
            -
             | 
| 24 | 
            +
                      message_chain(node) do
         | 
| 25 | 
            +
                        add_offense(
         | 
| 26 | 
            +
                          node.loc.selector,
         | 
| 27 | 
            +
                          message: format(MSG, method: node.method_name)
         | 
| 28 | 
            +
                        )
         | 
| 29 | 
            +
                      end
         | 
| 29 30 | 
             
                    end
         | 
| 30 31 | 
             
                  end
         | 
| 31 32 | 
             
                end
         | 
| @@ -24,7 +24,7 @@ module RuboCop | |
| 24 24 | 
             
                  #   # good
         | 
| 25 25 | 
             
                  #   expect(foo).to receive(:bar)
         | 
| 26 26 | 
             
                  #
         | 
| 27 | 
            -
                  class MessageExpectation <  | 
| 27 | 
            +
                  class MessageExpectation < Base
         | 
| 28 28 | 
             
                    include ConfigurableEnforcedStyle
         | 
| 29 29 |  | 
| 30 30 | 
             
                    MSG = 'Prefer `%<style>s` for setting message expectations.'
         | 
| @@ -42,7 +42,7 @@ module RuboCop | |
| 42 42 | 
             
                        return correct_style_detected if preferred_style?(match)
         | 
| 43 43 |  | 
| 44 44 | 
             
                        message = format(MSG, style: style)
         | 
| 45 | 
            -
                        add_offense(match | 
| 45 | 
            +
                        add_offense(match.loc.selector, message: message) do
         | 
| 46 46 | 
             
                          opposite_style_detected
         | 
| 47 47 | 
             
                        end
         | 
| 48 48 | 
             
                      end
         |