# frozen_string_literal: true module RuboCop module Cop # Common functionality shared by Uncommunicative cops module UncommunicativeName CASE_MSG = 'Only use lowercase characters for %s.' NUM_MSG = 'Do not end %s with a number.' LENGTH_MSG = '%s must be at least %s ' \ 'characters long.' FORBIDDEN_MSG = 'Do not use %s as a name for a ' \ '%s.' def check(node, args) args.each do |arg| # Argument names might be "_" or prefixed with "_" to indicate they # are unused. Trim away this prefix and only analyse the basename. full_name = arg.children.first.to_s next if full_name == '_' name = full_name.gsub(/\A([_]+)/, '') next if (arg.restarg_type? || arg.kwrestarg_type?) && name.empty? next if allowed_names.include?(name) range = arg_range(arg, name.size) issue_offenses(node, range, name) end end private def issue_offenses(node, range, name) forbidden_offense(node, range, name) if forbidden_names.include?(name) case_offense(node, range) if uppercase?(name) length_offense(node, range) unless long_enough?(name) return if allow_nums num_offense(node, range) if ends_with_num?(name) end def case_offense(node, range) add_offense(node, location: range, message: format(CASE_MSG, name_type: name_type(node))) end def uppercase?(name) name =~ /[[:upper:]]/ end def name_type(node) @name_type ||= begin case node.type when :block then 'block parameter' when :def, :defs then 'method parameter' end end end def num_offense(node, range) add_offense(node, location: range, message: format(NUM_MSG, name_type: name_type(node))) end def ends_with_num?(name) name[-1] =~ /\d/ end def length_offense(node, range) add_offense(node, location: range, message: format(LENGTH_MSG, name_type: name_type(node).capitalize, min: min_length)) end def long_enough?(name) name.size >= min_length end def arg_range(arg, length) begin_pos = arg.source_range.begin_pos Parser::Source::Range.new(processed_source.buffer, begin_pos, begin_pos + length) end def forbidden_offense(node, range, name) add_offense( node, location: range, message: format(FORBIDDEN_MSG, name: name, name_type: name_type(node)) ) end def allowed_names cop_config['AllowedNames'] end def forbidden_names cop_config['ForbiddenNames'] end def allow_nums cop_config['AllowNamesEndingInNumbers'] end def min_length cop_config['MinNameLength'] end end end end