-
-
Notifications
You must be signed in to change notification settings - Fork 0
/
bind_call.rb
77 lines (62 loc) · 2.18 KB
/
bind_call.rb
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
# frozen_string_literal: true
module RuboCop
module Cop
module Performance
# In Ruby 2.7, `UnboundMethod#bind_call` has been added.
#
# This cop identifies places where `bind(obj).call(args, ...)`
# can be replaced by `bind_call(obj, args, ...)`.
#
# The `bind_call(obj, args, ...)` method is faster than
# `bind(obj).call(args, ...)`.
#
# @example
# # bad
# umethod.bind(obj).call(foo, bar)
# umethod.bind(obj).(foo, bar)
#
# # good
# umethod.bind_call(obj, foo, bar)
#
class BindCall < Base
include RangeHelp
extend AutoCorrector
extend TargetRubyVersion
minimum_target_ruby_version 2.7
MSG = 'Use `bind_call(%<bind_arg>s%<comma>s%<call_args>s)` instead of `bind(%<bind_arg>s).call(%<call_args>s)`.'
RESTRICT_ON_SEND = %i[call].freeze
def_node_matcher :bind_with_call_method?, <<~PATTERN
(send
$(send
_ :bind
$(...)) :call
$...)
PATTERN
def on_send(node)
return unless (receiver, bind_arg, call_args_node = bind_with_call_method?(node))
range = correction_range(receiver, node)
call_args = build_call_args(call_args_node)
message = message(bind_arg.source, call_args)
add_offense(range, message: message) do |corrector|
call_args = ", #{call_args}" unless call_args.empty?
replacement_method = "bind_call(#{bind_arg.source}#{call_args})"
corrector.replace(range, replacement_method)
end
end
private
def message(bind_arg, call_args)
comma = call_args.empty? ? '' : ', '
format(MSG, bind_arg: bind_arg, comma: comma, call_args: call_args)
end
def correction_range(receiver, node)
location_of_bind = receiver.loc.selector.begin_pos
location_of_call = node.source_range.end.end_pos
range_between(location_of_bind, location_of_call)
end
def build_call_args(call_args_node)
call_args_node.map(&:source).join(', ')
end
end
end
end
end