# frozen_string_literal: true module RuboCop module Cop module RSpec # Checks for usage of implicit subject (`is_expected` / `should`). # # This cop can be configured using the `EnforcedStyle` option # # @example `EnforcedStyle: single_line_only` # # bad # it do # is_expected.to be_truthy # end # # # good # it { is_expected.to be_truthy } # it do # expect(subject).to be_truthy # end # # @example `EnforcedStyle: disallow` # # bad # it { is_expected.to be_truthy } # # # good # it { expect(subject).to be_truthy } # class ImplicitSubject < Cop include ConfigurableEnforcedStyle MSG = "Don't use implicit subject." def_node_matcher :implicit_subject?, <<-PATTERN (send nil? {:should :should_not :is_expected} ...) PATTERN def on_send(node) return unless implicit_subject?(node) return if valid_usage?(node) add_offense(node) end def autocorrect(node) replacement = 'expect(subject)' if node.method_name == :should replacement += '.to' elsif node.method_name == :should_not replacement += '.not_to' end ->(corrector) { corrector.replace(node.loc.selector, replacement) } end private def valid_usage?(node) example = node.ancestors.find { |parent| example?(parent) } return false if example.nil? example.method_name == :its || allowed_by_style?(example) end def allowed_by_style?(example) if style == :single_line_only example.single_line? elsif style == :single_statement_only !example.body.begin_type? else false end end end end end end