# frozen_string_literal: true module RuboCop module Cop module Rails # This cop looks for delegations that could have been created # automatically with the `delegate` method. # # Safe navigation `&.` is ignored because Rails' `allow_nil` # option checks not just for nil but also delegates if nil # responds to the delegated method. # # The `EnforceForPrefixed` option (defaulted to `true`) means that # using the target object as a prefix of the method name # without using the `delegate` method will be a violation. # When set to `false`, this case is legal. # # @example # # bad # def bar # foo.bar # end # # # good # delegate :bar, to: :foo # # # good # def bar # foo&.bar # end # # # good # private # def bar # foo.bar # end # # # EnforceForPrefixed: true # # bad # def foo_bar # foo.bar # end # # # good # delegate :bar, to: :foo, prefix: true # # # EnforceForPrefixed: false # # good # def foo_bar # foo.bar # end # # # good # delegate :bar, to: :foo, prefix: true class Delegate < Cop MSG = 'Use `delegate` to define delegations.'.freeze def_node_matcher :delegate?, <<-PATTERN (def _method_name _args (send (send nil? _) _ ...)) PATTERN def on_def(node) return unless trivial_delegate?(node) return if private_or_protected_delegation(node) add_offense(node, location: :keyword) end def autocorrect(node) method_name, _args, body = *node delegation = ["delegate :#{body.method_name}", "to: :#{body.receiver.method_name}"] if method_name == prefixed_method_name(body) delegation << ['prefix: true'] end lambda do |corrector| corrector.replace(node.source_range, delegation.join(', ')) end end private def trivial_delegate?(def_node) method_name, args, body = *def_node delegate?(def_node) && method_name_matches?(method_name, body) && arguments_match?(args, body) end def arguments_match?(arg_array, body) argument_array = body.arguments return false if arg_array.size != argument_array.size arg_array.zip(argument_array).all? do |arg, argument| arg.arg_type? && argument.lvar_type? && arg.children == argument.children end end def method_name_matches?(method_name, body) method_name == body.method_name || include_prefix_case? && method_name == prefixed_method_name(body) end def include_prefix_case? cop_config['EnforceForPrefixed'] end def prefixed_method_name(body) [body.receiver.method_name, body.method_name].join('_').to_sym end def private_or_protected_delegation(node) line = node.first_line private_or_protected_before(line) || private_or_protected_inline(line) end def private_or_protected_before(line) (processed_source[0..line].map(&:strip) & %w[private protected]).any? end def private_or_protected_inline(line) processed_source[line - 1].strip =~ /\A(private )|(protected )/ end end end end end