# frozen_string_literal: true module RuboCop module Cop module Style # This cop checks for accessing the first element of `String#unpack` # which can be replaced with the shorter method `unpack1`. # # @example # # # bad # 'foo'.unpack('h*').first # 'foo'.unpack('h*')[0] # 'foo'.unpack('h*').slice(0) # 'foo'.unpack('h*').at(0) # # # good # 'foo'.unpack1('h*') # class UnpackFirst < Base extend AutoCorrector MSG = 'Use `%s.unpack1(%s)` instead of '\ '`%s.unpack(%s)%s`.' RESTRICT_ON_SEND = %i[first [] slice at].freeze # @!method unpack_and_first_element?(node) def_node_matcher :unpack_and_first_element?, <<~PATTERN { (send $(send (...) :unpack $(...)) :first) (send $(send (...) :unpack $(...)) {:[] :slice :at} (int 0)) } PATTERN def on_send(node) unpack_and_first_element?(node) do |unpack_call, unpack_arg| range = first_element_range(node, unpack_call) message = format(MSG, receiver: unpack_call.receiver.source, format: unpack_arg.source, method: range.source) add_offense(node, message: message) do |corrector| corrector.remove(first_element_range(node, unpack_call)) corrector.replace(unpack_call.loc.selector, 'unpack1') end end end private def first_element_range(node, unpack_call) Parser::Source::Range.new(node.loc.expression.source_buffer, unpack_call.loc.expression.end_pos, node.loc.expression.end_pos) end end end end end