lib/rubocop/cop/performance/caller.rb
# frozen_string_literal: true module RuboCop module Cop module Performance # Identifies places where `caller[n]` can be replaced by `caller(n..n).first`. # # @example # # bad # caller[1] # caller.first # caller_locations[1] # caller_locations.first # # # good # caller(2..2).first # caller(1..1).first # caller_locations(2..2).first # caller_locations(1..1).first class Caller < Base extend AutoCorrector MSG = 'Use `%<preferred_method>s` instead of `%<current_method>s`.' RESTRICT_ON_SEND = %i[first []].freeze def_node_matcher :slow_caller?, <<~PATTERN { (send nil? {:caller :caller_locations}) (send nil? {:caller :caller_locations} int) } PATTERN def_node_matcher :caller_with_scope_method?, <<~PATTERN { (send #slow_caller? :first) (send #slow_caller? :[] int) } PATTERN def on_send(node) return unless caller_with_scope_method?(node) method_name = node.receiver.method_name caller_arg = node.receiver.first_argument n = caller_arg ? int_value(caller_arg) : 1 if node.method?(:[]) m = int_value(node.first_argument) n += m end preferred_method = "#{method_name}(#{n}..#{n}).first" message = format(MSG, preferred_method: preferred_method, current_method: node.source) add_offense(node, message: message) do |corrector| corrector.replace(node, preferred_method) end end private def int_value(node) node.children[0] end end end end end