# frozen_string_literal: true module RuboCop module Cop module Naming # This cop makes sure that rescued exceptions variables are named as # expected. # # The `PreferredName` config option takes a `String`. It represents # the required name of the variable. Its default is `e`. # # @example PreferredName: e (default) # # bad # begin # # do something # rescue MyException => exception # # do something # end # # # good # begin # # do something # rescue MyException => e # # do something # end # # # good # begin # # do something # rescue MyException => _e # # do something # end # # @example PreferredName: exception # # bad # begin # # do something # rescue MyException => e # # do something # end # # # good # begin # # do something # rescue MyException => exception # # do something # end # # # good # begin # # do something # rescue MyException => _exception # # do something # end # class RescuedExceptionsVariableName < Cop MSG = 'Use `%s` instead of `%s`.' def on_resbody(node) name = variable_name(node) return unless name return if preferred_name(name).to_sym == name add_offense(node, location: offense_range(node)) end def autocorrect(node) lambda do |corrector| offending_name = variable_name(node) preferred_name = preferred_name(offending_name) corrector.replace(offense_range(node), preferred_name) node.body&.each_descendant(:lvar) do |var| next unless var.children.first == offending_name corrector.replace(var.loc.expression, preferred_name) end end end private def offense_range(resbody) variable = resbody.exception_variable variable.loc.expression end def preferred_name(variable_name) preferred_name = cop_config.fetch('PreferredName', 'e') if variable_name.to_s.start_with?('_') "_#{preferred_name}" else preferred_name end end def variable_name(node) asgn_node = node.exception_variable return unless asgn_node asgn_node.children.last end def message(node) offending_name = variable_name(node) preferred_name = preferred_name(offending_name) format(MSG, preferred: preferred_name, bad: offending_name) end end end end end