class RuboCop::Cop::Style::SymbolProc
something.map(&:upcase)
# good
something.map { |s| s.upcase }
# bad
@example
Use symbols as procs when possible.
def self.autocorrect_incompatible_with
def self.autocorrect_incompatible_with [Layout::SpaceBeforeBlockBraces] end
def autocorrect(node)
def autocorrect(node) lambda do |corrector| block_send_or_super, _block_args, block_body = *node _receiver, method_name, _args = *block_body if super?(block_send_or_super) args = *block_send_or_super else _breceiver, _bmethod_name, *args = *block_send_or_super end autocorrect_method(corrector, node, args, method_name) end end
def autocorrect_method(corrector, node, args, method_name)
def autocorrect_method(corrector, node, args, method_name) if args.empty? autocorrect_no_args(corrector, node, method_name) else autocorrect_with_args(corrector, node, args, method_name) end end
def autocorrect_no_args(corrector, node, method_name)
def autocorrect_no_args(corrector, node, method_name) corrector.replace(block_range_with_space(node), "(&:#{method_name})") end
def autocorrect_with_args(corrector, node, args, method_name)
def autocorrect_with_args(corrector, node, args, method_name) arg_range = args.last.source_range arg_range = range_with_surrounding_comma(arg_range, :right) replacement = " &:#{method_name}" replacement = ',' + replacement unless arg_range.source.end_with?(',') corrector.insert_after(arg_range, replacement) corrector.remove(block_range_with_space(node)) end
def begin_pos_for_replacement(node)
def begin_pos_for_replacement(node) block_send_or_super, _block_args, _block_body = *node expr = block_send_or_super.source_range if (paren_pos = (expr.source =~ /\(\s*\)$/)) expr.begin_pos + paren_pos else node.loc.begin.begin_pos end end
def block_range_with_space(node)
def block_range_with_space(node) block_range = range_between(begin_pos_for_replacement(node), node.loc.end.end_pos) range_with_surrounding_space(block_range, :left) end
def ignored_method?(name)
def ignored_method?(name) ignored_methods.include?(name.to_s) end
def ignored_methods
def ignored_methods cop_config['IgnoredMethods'] end
def offense(node, method_name, block_method_name)
def offense(node, method_name, block_method_name) block_start = node.loc.begin.begin_pos block_end = node.loc.end.end_pos range = range_between(block_start, block_end) add_offense(node, range, format(MSG, method_name, block_method_name)) end
def on_block(node)
def on_block(node) symbol_proc?(node) do |send_or_super, method| block_method_name = resolve_block_method_name(send_or_super) # TODO: Rails-specific handling that we should probably make # configurable - https://github.com/bbatsov/rubocop/issues/1485 # we should ignore lambdas & procs return if proc_node?(send_or_super) return if %i[lambda proc].include?(block_method_name) return if ignored_method?(block_method_name) offense(node, method, block_method_name) end end
def resolve_block_method_name(block_send_or_super)
def resolve_block_method_name(block_send_or_super) return :super if super?(block_send_or_super) _receiver, method_name, _args = *block_send_or_super method_name end
def super?(node)
def super?(node) SUPER_TYPES.include?(node.type) end