Sha256: 50969d20353357518a991b747114f80ad6491d6bdac5c385c9472f0b3b3a8806

Contents?: true

Size: 1.64 KB

Versions: 13

Compression:

Stored size: 1.64 KB

Contents

# frozen_string_literal: true

module RuboCop
  module Cop
    module Rails
      # This cop enforces the use of `pick` over `pluck(...).first`.
      #
      # Using `pluck` followed by `first` creates an intermediate array, which
      # `pick` avoids. When called on an Active Record relation, `pick` adds a
      # limit to the query so that only one value is fetched from the database.
      #
      # @example
      #   # bad
      #   Model.pluck(:a).first
      #   [{ a: :b, c: :d }].pluck(:a, :b).first
      #
      #   # good
      #   Model.pick(:a)
      #   [{ a: :b, c: :d }].pick(:a, :b)
      class Pick < Base
        extend AutoCorrector
        extend TargetRailsVersion

        MSG = 'Prefer `pick(%<args>s)` over `pluck(%<args>s).first`.'
        RESTRICT_ON_SEND = %i[first].freeze

        minimum_target_rails_version 6.0

        def_node_matcher :pick_candidate?, <<~PATTERN
          (send (send _ :pluck ...) :first)
        PATTERN

        def on_send(node)
          pick_candidate?(node) do
            receiver = node.receiver
            receiver_selector = receiver.loc.selector
            node_selector = node.loc.selector
            range = receiver_selector.join(node_selector)

            add_offense(range, message: message(receiver)) do |corrector|
              first_range = receiver.source_range.end.join(node_selector)

              corrector.remove(first_range)
              corrector.replace(receiver_selector, 'pick')
            end
          end
        end

        private

        def message(receiver)
          format(MSG, args: receiver.arguments.map(&:source).join(', '))
        end
      end
    end
  end
end

Version data entries

13 entries across 13 versions & 1 rubygems

Version Path
rubocop-rails-2.12.4 lib/rubocop/cop/rails/pick.rb
rubocop-rails-2.12.3 lib/rubocop/cop/rails/pick.rb
rubocop-rails-2.12.2 lib/rubocop/cop/rails/pick.rb
rubocop-rails-2.12.1 lib/rubocop/cop/rails/pick.rb
rubocop-rails-2.12.0 lib/rubocop/cop/rails/pick.rb
rubocop-rails-2.11.3 lib/rubocop/cop/rails/pick.rb
rubocop-rails-2.11.2 lib/rubocop/cop/rails/pick.rb
rubocop-rails-2.11.1 lib/rubocop/cop/rails/pick.rb
rubocop-rails-2.11.0 lib/rubocop/cop/rails/pick.rb
rubocop-rails-2.10.1 lib/rubocop/cop/rails/pick.rb
rubocop-rails-2.10.0 lib/rubocop/cop/rails/pick.rb
rubocop-rails-2.9.1 lib/rubocop/cop/rails/pick.rb
rubocop-rails-2.9.0 lib/rubocop/cop/rails/pick.rb