rubocop-rails 2.8.1 → 2.9.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/README.md +16 -0
- data/config/default.yml +33 -1
- data/lib/rubocop/cop/mixin/active_record_helper.rb +4 -3
- data/lib/rubocop/cop/mixin/enforce_superclass.rb +40 -0
- data/lib/rubocop/cop/mixin/index_method.rb +8 -11
- data/lib/rubocop/cop/rails/action_filter.rb +10 -14
- data/lib/rubocop/cop/rails/active_record_aliases.rb +13 -17
- data/lib/rubocop/cop/rails/active_record_callbacks_order.rb +17 -12
- data/lib/rubocop/cop/rails/active_record_override.rb +1 -1
- data/lib/rubocop/cop/rails/active_support_aliases.rb +12 -21
- data/lib/rubocop/cop/rails/after_commit_override.rb +1 -1
- data/lib/rubocop/cop/rails/application_controller.rb +3 -7
- data/lib/rubocop/cop/rails/application_job.rb +2 -1
- data/lib/rubocop/cop/rails/application_mailer.rb +2 -7
- data/lib/rubocop/cop/rails/application_record.rb +2 -7
- data/lib/rubocop/cop/rails/arel_star.rb +41 -0
- data/lib/rubocop/cop/rails/assert_not.rb +8 -10
- data/lib/rubocop/cop/rails/attribute_default_block_value.rb +90 -0
- data/lib/rubocop/cop/rails/belongs_to.rb +9 -18
- data/lib/rubocop/cop/rails/blank.rb +27 -27
- data/lib/rubocop/cop/rails/bulk_change_table.rb +1 -1
- data/lib/rubocop/cop/rails/content_tag.rb +16 -16
- data/lib/rubocop/cop/rails/create_table_with_timestamps.rb +2 -1
- data/lib/rubocop/cop/rails/date.rb +10 -11
- data/lib/rubocop/cop/rails/default_scope.rb +11 -4
- data/lib/rubocop/cop/rails/delegate.rb +9 -9
- data/lib/rubocop/cop/rails/delegate_allow_blank.rb +7 -8
- data/lib/rubocop/cop/rails/dynamic_find_by.rb +12 -11
- data/lib/rubocop/cop/rails/enum_hash.rb +11 -10
- data/lib/rubocop/cop/rails/enum_uniqueness.rb +2 -1
- data/lib/rubocop/cop/rails/environment_comparison.rb +18 -14
- data/lib/rubocop/cop/rails/exit.rb +4 -10
- data/lib/rubocop/cop/rails/file_path.rb +4 -3
- data/lib/rubocop/cop/rails/find_by.rb +13 -13
- data/lib/rubocop/cop/rails/find_by_id.rb +12 -21
- data/lib/rubocop/cop/rails/find_each.rb +16 -14
- data/lib/rubocop/cop/rails/has_and_belongs_to_many.rb +3 -2
- data/lib/rubocop/cop/rails/has_many_or_has_one_dependent.rb +3 -2
- data/lib/rubocop/cop/rails/helper_instance_variable.rb +2 -2
- data/lib/rubocop/cop/rails/http_positional_arguments.rb +19 -21
- data/lib/rubocop/cop/rails/http_status.rb +7 -9
- data/lib/rubocop/cop/rails/ignored_skip_action_filter_option.rb +8 -6
- data/lib/rubocop/cop/rails/index_by.rb +2 -1
- data/lib/rubocop/cop/rails/index_with.rb +2 -1
- data/lib/rubocop/cop/rails/inquiry.rb +4 -3
- data/lib/rubocop/cop/rails/inverse_of.rb +3 -2
- data/lib/rubocop/cop/rails/lexically_scoped_action_filter.rb +17 -15
- data/lib/rubocop/cop/rails/link_to_blank.rb +20 -22
- data/lib/rubocop/cop/rails/mailer_name.rb +19 -13
- data/lib/rubocop/cop/rails/match_route.rb +14 -13
- data/lib/rubocop/cop/rails/negate_include.rb +10 -8
- data/lib/rubocop/cop/rails/not_null_column.rb +2 -1
- data/lib/rubocop/cop/rails/order_by_id.rb +1 -2
- data/lib/rubocop/cop/rails/output.rb +5 -2
- data/lib/rubocop/cop/rails/output_safety.rb +3 -2
- data/lib/rubocop/cop/rails/pick.rb +14 -12
- data/lib/rubocop/cop/rails/pluck.rb +6 -9
- data/lib/rubocop/cop/rails/pluck_id.rb +4 -6
- data/lib/rubocop/cop/rails/pluck_in_where.rb +7 -7
- data/lib/rubocop/cop/rails/pluralization_grammar.rb +10 -14
- data/lib/rubocop/cop/rails/presence.rb +12 -13
- data/lib/rubocop/cop/rails/present.rb +30 -24
- data/lib/rubocop/cop/rails/rake_environment.rb +8 -10
- data/lib/rubocop/cop/rails/read_write_attribute.rb +12 -11
- data/lib/rubocop/cop/rails/redundant_allow_nil.rb +29 -31
- data/lib/rubocop/cop/rails/redundant_foreign_key.rb +9 -12
- data/lib/rubocop/cop/rails/redundant_receiver_in_with_options.rb +11 -10
- data/lib/rubocop/cop/rails/reflection_class_name.rb +3 -2
- data/lib/rubocop/cop/rails/refute_methods.rb +9 -10
- data/lib/rubocop/cop/rails/relative_date_constant.rb +16 -8
- data/lib/rubocop/cop/rails/render_inline.rb +2 -1
- data/lib/rubocop/cop/rails/render_plain_text.rb +9 -14
- data/lib/rubocop/cop/rails/request_referer.rb +7 -7
- data/lib/rubocop/cop/rails/reversible_migration.rb +2 -6
- data/lib/rubocop/cop/rails/safe_navigation.rb +11 -10
- data/lib/rubocop/cop/rails/safe_navigation_with_blank.rb +5 -10
- data/lib/rubocop/cop/rails/save_bang.rb +17 -20
- data/lib/rubocop/cop/rails/scope_args.rb +2 -1
- data/lib/rubocop/cop/rails/short_i18n.rb +7 -9
- data/lib/rubocop/cop/rails/skips_model_validations.rb +4 -4
- data/lib/rubocop/cop/rails/squished_sql_heredocs.rb +5 -6
- data/lib/rubocop/cop/rails/time_zone.rb +22 -20
- data/lib/rubocop/cop/rails/uniq_before_pluck.rb +4 -6
- data/lib/rubocop/cop/rails/unique_validation_without_index.rb +2 -2
- data/lib/rubocop/cop/rails/unknown_env.rb +2 -2
- data/lib/rubocop/cop/rails/validation.rb +15 -14
- data/lib/rubocop/cop/rails/where_equals.rb +94 -0
- data/lib/rubocop/cop/rails/where_exists.rb +8 -13
- data/lib/rubocop/cop/rails/where_not.rb +5 -16
- data/lib/rubocop/cop/rails_cops.rb +4 -0
- data/lib/rubocop/rails/schema_loader.rb +4 -4
- data/lib/rubocop/rails/schema_loader/schema.rb +1 -1
- data/lib/rubocop/rails/version.rb +5 -1
- metadata +19 -9
@@ -24,8 +24,9 @@ module RuboCop
|
|
24
24
|
# end
|
25
25
|
# end
|
26
26
|
#
|
27
|
-
class RenderInline <
|
27
|
+
class RenderInline < Base
|
28
28
|
MSG = 'Prefer using a template over inline rendering.'
|
29
|
+
RESTRICT_ON_SEND = %i[render].freeze
|
29
30
|
|
30
31
|
def_node_matcher :render_with_inline_option?, <<~PATTERN
|
31
32
|
(send nil? :render (hash <(pair {(sym :inline) (str "inline")} _) ...>))
|
@@ -24,30 +24,25 @@ module RuboCop
|
|
24
24
|
# # bad - sets MIME type to `text/html`
|
25
25
|
# render text: 'Ruby!'
|
26
26
|
#
|
27
|
-
class RenderPlainText <
|
27
|
+
class RenderPlainText < Base
|
28
|
+
extend AutoCorrector
|
29
|
+
|
28
30
|
MSG = 'Prefer `render plain:` over `render text:`.'
|
31
|
+
RESTRICT_ON_SEND = %i[render].freeze
|
29
32
|
|
30
33
|
def_node_matcher :render_plain_text?, <<~PATTERN
|
31
34
|
(send nil? :render $(hash <$(pair (sym :text) $_) ...>))
|
32
35
|
PATTERN
|
33
36
|
|
34
37
|
def on_send(node)
|
35
|
-
render_plain_text?(node) do |options_node, _option_node, _option_value|
|
36
|
-
content_type_node = find_content_type(options_node)
|
37
|
-
add_offense(node) if compatible_content_type?(content_type_node)
|
38
|
-
end
|
39
|
-
end
|
40
|
-
|
41
|
-
def autocorrect(node)
|
42
38
|
render_plain_text?(node) do |options_node, option_node, option_value|
|
43
39
|
content_type_node = find_content_type(options_node)
|
44
|
-
|
40
|
+
return unless compatible_content_type?(content_type_node)
|
41
|
+
|
42
|
+
add_offense(node) do |corrector|
|
43
|
+
rest_options = options_node.pairs - [option_node, content_type_node].compact
|
45
44
|
|
46
|
-
|
47
|
-
corrector.replace(
|
48
|
-
node,
|
49
|
-
replacement(rest_options, option_value)
|
50
|
-
)
|
45
|
+
corrector.replace(node, replacement(rest_options, option_value))
|
51
46
|
end
|
52
47
|
end
|
53
48
|
end
|
@@ -19,11 +19,13 @@ module RuboCop
|
|
19
19
|
#
|
20
20
|
# # good
|
21
21
|
# request.referrer
|
22
|
-
class RequestReferer <
|
22
|
+
class RequestReferer < Base
|
23
23
|
include ConfigurableEnforcedStyle
|
24
|
+
extend AutoCorrector
|
24
25
|
|
25
26
|
MSG = 'Use `request.%<prefer>s` instead of ' \
|
26
27
|
'`request.%<current>s`.'
|
28
|
+
RESTRICT_ON_SEND = %i[referer referrer].freeze
|
27
29
|
|
28
30
|
def_node_matcher :referer?, <<~PATTERN
|
29
31
|
(send (send nil? :request) {:referer :referrer})
|
@@ -33,17 +35,15 @@ module RuboCop
|
|
33
35
|
referer?(node) do
|
34
36
|
return unless node.method?(wrong_method_name)
|
35
37
|
|
36
|
-
add_offense(node.source_range
|
38
|
+
add_offense(node.source_range) do |corrector|
|
39
|
+
corrector.replace(node, "request.#{style}")
|
40
|
+
end
|
37
41
|
end
|
38
42
|
end
|
39
43
|
|
40
|
-
def autocorrect(node)
|
41
|
-
->(corrector) { corrector.replace(node, "request.#{style}") }
|
42
|
-
end
|
43
|
-
|
44
44
|
private
|
45
45
|
|
46
|
-
def message(
|
46
|
+
def message(_range)
|
47
47
|
format(MSG, prefer: style, current: wrong_method_name)
|
48
48
|
end
|
49
49
|
|
@@ -175,7 +175,7 @@ module RuboCop
|
|
175
175
|
# end
|
176
176
|
#
|
177
177
|
# @see https://api.rubyonrails.org/classes/ActiveRecord/Migration/CommandRecorder.html
|
178
|
-
class ReversibleMigration <
|
178
|
+
class ReversibleMigration < Base
|
179
179
|
MSG = '%<action>s is not reversible.'
|
180
180
|
|
181
181
|
def_node_matcher :irreversible_schema_statement_call, <<~PATTERN
|
@@ -271,11 +271,7 @@ module RuboCop
|
|
271
271
|
def check_remove_foreign_key_node(node)
|
272
272
|
remove_foreign_key_call(node) do |arg|
|
273
273
|
if arg.hash_type? && !all_hash_key?(arg, :to_table)
|
274
|
-
add_offense(
|
275
|
-
node,
|
276
|
-
message: format(MSG,
|
277
|
-
action: 'remove_foreign_key(without table)')
|
278
|
-
)
|
274
|
+
add_offense(node, message: format(MSG, action: 'remove_foreign_key(without table)'))
|
279
275
|
end
|
280
276
|
end
|
281
277
|
end
|
@@ -36,10 +36,12 @@ module RuboCop
|
|
36
36
|
# foo&.bar
|
37
37
|
# foo&.bar(baz)
|
38
38
|
# foo&.bar { |e| e.baz }
|
39
|
-
class SafeNavigation <
|
39
|
+
class SafeNavigation < Base
|
40
40
|
include RangeHelp
|
41
|
+
extend AutoCorrector
|
41
42
|
|
42
43
|
MSG = 'Use safe navigation (`&.`) instead of `%<try>s`.'
|
44
|
+
RESTRICT_ON_SEND = %i[try try!].freeze
|
43
45
|
|
44
46
|
def_node_matcher :try_call, <<~PATTERN
|
45
47
|
(send !nil? ${:try :try!} $_ ...)
|
@@ -50,24 +52,23 @@ module RuboCop
|
|
50
52
|
return if try_method == :try && !cop_config['ConvertTry']
|
51
53
|
return unless dispatch.sym_type? && dispatch.value.match?(/\w+[=!?]?/)
|
52
54
|
|
53
|
-
add_offense(node, message: format(MSG, try: try_method))
|
55
|
+
add_offense(node, message: format(MSG, try: try_method)) do |corrector|
|
56
|
+
autocorrect(corrector, node)
|
57
|
+
end
|
54
58
|
end
|
55
59
|
end
|
56
60
|
|
57
|
-
|
61
|
+
private
|
62
|
+
|
63
|
+
def autocorrect(corrector, node)
|
58
64
|
method_node, *params = *node.arguments
|
59
65
|
method = method_node.source[1..-1]
|
60
66
|
|
61
|
-
range = range_between(node.loc.dot.begin_pos,
|
62
|
-
node.loc.expression.end_pos)
|
67
|
+
range = range_between(node.loc.dot.begin_pos, node.loc.expression.end_pos)
|
63
68
|
|
64
|
-
|
65
|
-
corrector.replace(range, replacement(method, params))
|
66
|
-
end
|
69
|
+
corrector.replace(range, replacement(method, params))
|
67
70
|
end
|
68
71
|
|
69
|
-
private
|
70
|
-
|
71
72
|
def replacement(method, params)
|
72
73
|
new_params = params.map(&:source).join(', ')
|
73
74
|
|
@@ -19,7 +19,9 @@ module RuboCop
|
|
19
19
|
# do_something if foo.blank?
|
20
20
|
# do_something unless foo.blank?
|
21
21
|
#
|
22
|
-
class SafeNavigationWithBlank <
|
22
|
+
class SafeNavigationWithBlank < Base
|
23
|
+
extend AutoCorrector
|
24
|
+
|
23
25
|
MSG =
|
24
26
|
'Avoid calling `blank?` with the safe navigation operator ' \
|
25
27
|
'in conditionals.'
|
@@ -31,15 +33,8 @@ module RuboCop
|
|
31
33
|
def on_if(node)
|
32
34
|
return unless safe_navigation_blank_in_conditional?(node)
|
33
35
|
|
34
|
-
add_offense(node)
|
35
|
-
|
36
|
-
|
37
|
-
def autocorrect(node)
|
38
|
-
lambda do |corrector|
|
39
|
-
corrector.replace(
|
40
|
-
safe_navigation_blank_in_conditional?(node).location.dot,
|
41
|
-
'.'
|
42
|
-
)
|
36
|
+
add_offense(node) do |corrector|
|
37
|
+
corrector.replace(safe_navigation_blank_in_conditional?(node).location.dot, '.')
|
43
38
|
end
|
44
39
|
end
|
45
40
|
end
|
@@ -98,8 +98,9 @@ module RuboCop
|
|
98
98
|
# Services::Service::Mailer.update(message: 'Message')
|
99
99
|
# Service::Mailer::update
|
100
100
|
#
|
101
|
-
class SaveBang <
|
101
|
+
class SaveBang < Base
|
102
102
|
include NegativeConditional
|
103
|
+
extend AutoCorrector
|
103
104
|
|
104
105
|
MSG = 'Use `%<prefer>s` instead of `%<current>s` if the return ' \
|
105
106
|
'value is not checked.'
|
@@ -113,11 +114,10 @@ module RuboCop
|
|
113
114
|
first_or_create find_or_create_by].freeze
|
114
115
|
MODIFY_PERSIST_METHODS = %i[save
|
115
116
|
update update_attributes destroy].freeze
|
116
|
-
|
117
|
-
MODIFY_PERSIST_METHODS).freeze
|
117
|
+
RESTRICT_ON_SEND = (CREATE_PERSIST_METHODS + MODIFY_PERSIST_METHODS).freeze
|
118
118
|
|
119
|
-
def
|
120
|
-
|
119
|
+
def self.joining_forces
|
120
|
+
VariableForce
|
121
121
|
end
|
122
122
|
|
123
123
|
def after_leaving_scope(scope, _variable_table)
|
@@ -135,7 +135,7 @@ module RuboCop
|
|
135
135
|
return unless persist_method?(node, CREATE_PERSIST_METHODS)
|
136
136
|
return if persisted_referenced?(assignment)
|
137
137
|
|
138
|
-
|
138
|
+
register_offense(node, CREATE_MSG)
|
139
139
|
end
|
140
140
|
|
141
141
|
# rubocop:disable Metrics/CyclomaticComplexity
|
@@ -148,25 +148,22 @@ module RuboCop
|
|
148
148
|
return if explicit_return?(node)
|
149
149
|
return if checked_immediately?(node)
|
150
150
|
|
151
|
-
|
151
|
+
register_offense(node, MSG)
|
152
152
|
end
|
153
153
|
# rubocop:enable Metrics/CyclomaticComplexity
|
154
154
|
alias on_csend on_send
|
155
155
|
|
156
|
-
def autocorrect(node)
|
157
|
-
save_loc = node.loc.selector
|
158
|
-
new_method = "#{node.method_name}!"
|
159
|
-
|
160
|
-
->(corrector) { corrector.replace(save_loc, new_method) }
|
161
|
-
end
|
162
|
-
|
163
156
|
private
|
164
157
|
|
165
|
-
def
|
166
|
-
|
167
|
-
|
158
|
+
def register_offense(node, msg)
|
159
|
+
current_method = node.method_name
|
160
|
+
bang_method = "#{current_method}!"
|
161
|
+
full_message = format(msg, prefer: bang_method, current: current_method)
|
168
162
|
|
169
|
-
|
163
|
+
range = node.loc.selector
|
164
|
+
add_offense(range, message: full_message) do |corrector|
|
165
|
+
corrector.replace(range, bang_method)
|
166
|
+
end
|
170
167
|
end
|
171
168
|
|
172
169
|
def right_assignment_node(assignment)
|
@@ -218,7 +215,7 @@ module RuboCop
|
|
218
215
|
def check_used_in_condition_or_compound_boolean(node)
|
219
216
|
return false unless in_condition_or_compound_boolean?(node)
|
220
217
|
|
221
|
-
|
218
|
+
register_offense(node, CREATE_CONDITIONAL_MSG) unless MODIFY_PERSIST_METHODS.include?(node.method_name)
|
222
219
|
|
223
220
|
true
|
224
221
|
end
|
@@ -318,7 +315,7 @@ module RuboCop
|
|
318
315
|
assignment&.lvasgn_type?
|
319
316
|
end
|
320
317
|
|
321
|
-
def persist_method?(node, methods =
|
318
|
+
def persist_method?(node, methods = RESTRICT_ON_SEND)
|
322
319
|
methods.include?(node.method_name) &&
|
323
320
|
expected_signature?(node) &&
|
324
321
|
!allowed_receiver?(node)
|
@@ -13,8 +13,9 @@ module RuboCop
|
|
13
13
|
#
|
14
14
|
# # good
|
15
15
|
# scope :something, -> { where(something: true) }
|
16
|
-
class ScopeArgs <
|
16
|
+
class ScopeArgs < Base
|
17
17
|
MSG = 'Use `lambda`/`proc` instead of a plain method call.'
|
18
|
+
RESTRICT_ON_SEND = %i[scope].freeze
|
18
19
|
|
19
20
|
def_node_matcher :scope?, '(send nil? :scope _ $send)'
|
20
21
|
|
@@ -38,8 +38,9 @@ module RuboCop
|
|
38
38
|
# t :key
|
39
39
|
# l Time.now
|
40
40
|
#
|
41
|
-
class ShortI18n <
|
41
|
+
class ShortI18n < Base
|
42
42
|
include ConfigurableEnforcedStyle
|
43
|
+
extend AutoCorrector
|
43
44
|
|
44
45
|
MSG = 'Use `%<good_method>s` instead of `%<bad_method>s`.'
|
45
46
|
|
@@ -48,6 +49,8 @@ module RuboCop
|
|
48
49
|
localize: :l
|
49
50
|
}.freeze
|
50
51
|
|
52
|
+
RESTRICT_ON_SEND = PREFERRED_METHODS.keys.freeze
|
53
|
+
|
51
54
|
def_node_matcher :long_i18n?, <<~PATTERN
|
52
55
|
(send {nil? (const nil? :I18n)} ${:translate :localize} ...)
|
53
56
|
PATTERN
|
@@ -58,15 +61,10 @@ module RuboCop
|
|
58
61
|
long_i18n?(node) do |method_name|
|
59
62
|
good_method = PREFERRED_METHODS[method_name]
|
60
63
|
message = format(MSG, good_method: good_method, bad_method: method_name)
|
64
|
+
range = node.loc.selector
|
61
65
|
|
62
|
-
add_offense(
|
63
|
-
|
64
|
-
end
|
65
|
-
|
66
|
-
def autocorrect(node)
|
67
|
-
long_i18n?(node) do |method_name|
|
68
|
-
lambda do |corrector|
|
69
|
-
corrector.replace(node.loc.selector, PREFERRED_METHODS[method_name])
|
66
|
+
add_offense(range, message: message) do |corrector|
|
67
|
+
corrector.replace(range, PREFERRED_METHODS[method_name])
|
70
68
|
end
|
71
69
|
end
|
72
70
|
end
|
@@ -7,7 +7,7 @@ module RuboCop
|
|
7
7
|
# validations which are listed in
|
8
8
|
# https://guides.rubyonrails.org/active_record_validations.html#skipping-validations
|
9
9
|
#
|
10
|
-
# Methods may be ignored from this rule by configuring a `
|
10
|
+
# Methods may be ignored from this rule by configuring a `AllowedMethods`.
|
11
11
|
#
|
12
12
|
# @example
|
13
13
|
# # bad
|
@@ -26,7 +26,7 @@ module RuboCop
|
|
26
26
|
# user.update(website: 'example.com')
|
27
27
|
# FileUtils.touch('file')
|
28
28
|
#
|
29
|
-
# @example
|
29
|
+
# @example AllowedMethods: ["touch"]
|
30
30
|
# # bad
|
31
31
|
# DiscussionBoard.decrement_counter(:post_count, 5)
|
32
32
|
# DiscussionBoard.increment_counter(:post_count, 5)
|
@@ -35,7 +35,7 @@ module RuboCop
|
|
35
35
|
# # good
|
36
36
|
# user.touch
|
37
37
|
#
|
38
|
-
class SkipsModelValidations <
|
38
|
+
class SkipsModelValidations < Base
|
39
39
|
MSG = 'Avoid using `%<method>s` because it skips validations.'
|
40
40
|
|
41
41
|
METHODS_WITH_ARGUMENTS = %w[decrement!
|
@@ -76,7 +76,7 @@ module RuboCop
|
|
76
76
|
return if good_touch?(node)
|
77
77
|
return if good_insert?(node)
|
78
78
|
|
79
|
-
add_offense(node,
|
79
|
+
add_offense(node.loc.selector, message: message(node))
|
80
80
|
end
|
81
81
|
alias on_csend on_send
|
82
82
|
|
@@ -5,6 +5,8 @@ module RuboCop
|
|
5
5
|
module Rails
|
6
6
|
#
|
7
7
|
# Checks SQL heredocs to use `.squish`.
|
8
|
+
# Some SQL syntax (e.g. PostgreSQL comments and functions) requires newlines
|
9
|
+
# to be preserved in order to work, thus auto-correction for this cop is not safe.
|
8
10
|
#
|
9
11
|
# @example
|
10
12
|
# # bad
|
@@ -37,8 +39,9 @@ module RuboCop
|
|
37
39
|
# WHERE post_id = 1
|
38
40
|
# SQL
|
39
41
|
#
|
40
|
-
class SquishedSQLHeredocs <
|
42
|
+
class SquishedSQLHeredocs < Base
|
41
43
|
include Heredoc
|
44
|
+
extend AutoCorrector
|
42
45
|
|
43
46
|
SQL = 'SQL'
|
44
47
|
SQUISH = '.squish'
|
@@ -47,11 +50,7 @@ module RuboCop
|
|
47
50
|
def on_heredoc(node)
|
48
51
|
return unless offense_detected?(node)
|
49
52
|
|
50
|
-
add_offense(node)
|
51
|
-
end
|
52
|
-
|
53
|
-
def autocorrect(node)
|
54
|
-
lambda do |corrector|
|
53
|
+
add_offense(node) do |corrector|
|
55
54
|
corrector.insert_after(node, SQUISH)
|
56
55
|
end
|
57
56
|
end
|
@@ -43,8 +43,9 @@ module RuboCop
|
|
43
43
|
# # good
|
44
44
|
# Time.current
|
45
45
|
# Time.at(timestamp).in_time_zone
|
46
|
-
class TimeZone <
|
46
|
+
class TimeZone < Base
|
47
47
|
include ConfigurableEnforcedStyle
|
48
|
+
extend AutoCorrector
|
48
49
|
|
49
50
|
MSG = 'Do not use `%<current>s` without zone. Use `%<prefer>s` ' \
|
50
51
|
'instead.'
|
@@ -71,26 +72,24 @@ module RuboCop
|
|
71
72
|
check_time_node(klass, node.parent) if klass == :Time
|
72
73
|
end
|
73
74
|
|
74
|
-
|
75
|
-
lambda do |corrector|
|
76
|
-
# add `.zone`: `Time.at` => `Time.zone.at`
|
77
|
-
corrector.insert_after(node.children[0].source_range, '.zone')
|
75
|
+
private
|
78
76
|
|
79
|
-
|
80
|
-
|
81
|
-
|
82
|
-
corrector.replace(node.loc.selector, 'now')
|
83
|
-
when :new
|
84
|
-
autocorrect_time_new(node, corrector)
|
85
|
-
end
|
77
|
+
def autocorrect(corrector, node)
|
78
|
+
# add `.zone`: `Time.at` => `Time.zone.at`
|
79
|
+
corrector.insert_after(node.children[0].source_range, '.zone')
|
86
80
|
|
87
|
-
|
88
|
-
|
89
|
-
|
81
|
+
case node.method_name
|
82
|
+
when :current
|
83
|
+
# replace `Time.zone.current` => `Time.zone.now`
|
84
|
+
corrector.replace(node.loc.selector, 'now')
|
85
|
+
when :new
|
86
|
+
autocorrect_time_new(node, corrector)
|
90
87
|
end
|
91
|
-
end
|
92
88
|
|
93
|
-
|
89
|
+
# prefer `Time` over `DateTime` class
|
90
|
+
corrector.replace(node.children.first.source_range, 'Time') if strict?
|
91
|
+
remove_redundant_in_time_zone(corrector, node)
|
92
|
+
end
|
94
93
|
|
95
94
|
def autocorrect_time_new(node, corrector)
|
96
95
|
if node.arguments?
|
@@ -128,7 +127,9 @@ module RuboCop
|
|
128
127
|
|
129
128
|
message = build_message(klass, method_name, node)
|
130
129
|
|
131
|
-
add_offense(node
|
130
|
+
add_offense(node.loc.selector, message: message) do |corrector|
|
131
|
+
autocorrect(corrector, node)
|
132
|
+
end
|
132
133
|
end
|
133
134
|
|
134
135
|
def build_message(klass, method_name, node)
|
@@ -193,8 +194,9 @@ module RuboCop
|
|
193
194
|
|
194
195
|
return if node.arguments?
|
195
196
|
|
196
|
-
add_offense(selector_node,
|
197
|
-
|
197
|
+
add_offense(selector_node.loc.selector, message: MSG_LOCALTIME) do |corrector|
|
198
|
+
autocorrect(corrector, selector_node)
|
199
|
+
end
|
198
200
|
end
|
199
201
|
|
200
202
|
def not_danger_chain?(chain)
|