rubocop 1.72.2 → 1.73.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (56) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +3 -3
  3. data/config/default.yml +25 -13
  4. data/config/internal_affairs.yml +20 -0
  5. data/lib/rubocop/config_loader.rb +0 -1
  6. data/lib/rubocop/config_loader_resolver.rb +2 -2
  7. data/lib/rubocop/config_validator.rb +1 -1
  8. data/lib/rubocop/cop/internal_affairs/example_description.rb +3 -1
  9. data/lib/rubocop/cop/internal_affairs/node_type_group.rb +91 -0
  10. data/lib/rubocop/cop/internal_affairs.rb +1 -0
  11. data/lib/rubocop/cop/layout/closing_parenthesis_indentation.rb +4 -4
  12. data/lib/rubocop/cop/layout/empty_lines_around_access_modifier.rb +26 -1
  13. data/lib/rubocop/cop/layout/line_length.rb +3 -3
  14. data/lib/rubocop/cop/lint/duplicate_methods.rb +0 -14
  15. data/lib/rubocop/cop/lint/empty_conditional_body.rb +14 -64
  16. data/lib/rubocop/cop/lint/erb_new_arguments.rb +0 -6
  17. data/lib/rubocop/cop/lint/float_comparison.rb +1 -6
  18. data/lib/rubocop/cop/lint/literal_as_condition.rb +99 -9
  19. data/lib/rubocop/cop/lint/mixed_case_range.rb +2 -2
  20. data/lib/rubocop/cop/lint/redundant_require_statement.rb +0 -21
  21. data/lib/rubocop/cop/lint/redundant_type_conversion.rb +23 -2
  22. data/lib/rubocop/cop/lint/void.rb +6 -0
  23. data/lib/rubocop/cop/mixin/hash_subset.rb +19 -4
  24. data/lib/rubocop/cop/mixin/range_help.rb +12 -0
  25. data/lib/rubocop/cop/mixin/trailing_comma.rb +12 -0
  26. data/lib/rubocop/cop/naming/variable_name.rb +64 -6
  27. data/lib/rubocop/cop/style/accessor_grouping.rb +19 -5
  28. data/lib/rubocop/cop/style/commented_keyword.rb +1 -1
  29. data/lib/rubocop/cop/style/endless_method.rb +163 -18
  30. data/lib/rubocop/cop/style/expand_path_arguments.rb +2 -7
  31. data/lib/rubocop/cop/style/inverse_methods.rb +8 -5
  32. data/lib/rubocop/cop/style/keyword_parameters_order.rb +13 -7
  33. data/lib/rubocop/cop/style/line_end_concatenation.rb +10 -4
  34. data/lib/rubocop/cop/style/method_called_on_do_end_block.rb +1 -1
  35. data/lib/rubocop/cop/style/multiline_block_chain.rb +1 -1
  36. data/lib/rubocop/cop/style/multiline_method_signature.rb +1 -9
  37. data/lib/rubocop/cop/style/redundant_condition.rb +45 -0
  38. data/lib/rubocop/cop/style/redundant_format.rb +23 -11
  39. data/lib/rubocop/cop/style/redundant_freeze.rb +1 -1
  40. data/lib/rubocop/cop/style/redundant_self_assignment.rb +1 -1
  41. data/lib/rubocop/cop/style/single_line_methods.rb +3 -3
  42. data/lib/rubocop/cop/style/sole_nested_conditional.rb +0 -6
  43. data/lib/rubocop/cop/style/string_concatenation.rb +1 -1
  44. data/lib/rubocop/cop/style/trailing_comma_in_array_literal.rb +47 -6
  45. data/lib/rubocop/cop/style/trailing_comma_in_hash_literal.rb +48 -6
  46. data/lib/rubocop/cop/style/trivial_accessors.rb +1 -1
  47. data/lib/rubocop/cop/utils/format_string.rb +5 -2
  48. data/lib/rubocop/cops_documentation_generator.rb +12 -1
  49. data/lib/rubocop/plugin/load_error.rb +1 -1
  50. data/lib/rubocop/plugin.rb +9 -2
  51. data/lib/rubocop/rspec/shared_contexts.rb +15 -0
  52. data/lib/rubocop/rspec/support.rb +1 -0
  53. data/lib/rubocop/version.rb +1 -1
  54. data/lib/rubocop.rb +0 -1
  55. metadata +5 -5
  56. data/lib/rubocop/cop/utils/regexp_ranges.rb +0 -113
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 4a6ff0f849d961da13d2131ff60e0ed49dce01a4c2d32caa8484a2344344609b
4
- data.tar.gz: 9da9fc43212c62cd6542649c90423e9365f50cb00e82b6337be1a241c6a17283
3
+ metadata.gz: 671bf05708f62eb3abe3b448168f7daa905cffe60e2652ad4ec21964594adf70
4
+ data.tar.gz: caef12b9b307560cda7125652fa0249fcc0908725e97cc963a10f4c1e0782faa
5
5
  SHA512:
6
- metadata.gz: 7ff34957554167b3b1e30645e3754094968fae93d0284a3882460b3e24f44fd64bcf358b23fd0e8154489206fd9c95fc6555b551f1af5ac810805fac2b6d27be
7
- data.tar.gz: 48e989049de37523554c740ba1368e32de38bc0f879c1fc717924a758a1ba465ef4413cba80a42be59ccaaf87616b52c0ad437256702bf5fd091c4a5818f428a
6
+ metadata.gz: fea1053c5a5f72ffc9eea331b438f1fc1d876eed21c0f251569ab541cf3f9724193e1c3bf72a0154e84608ac4346cb7a122fcc4a5266fe7b520109c56c5dea1a
7
+ data.tar.gz: e2b4c854a638b119a3b28e39e77dd05d86fe85b890180bc9a70ac10727f9e74e1ed64e659fe397042610e4bc87b3153d5783d472a596de110628395708da6177
data/README.md CHANGED
@@ -52,7 +52,7 @@ To prevent an unwanted RuboCop update you might want to use a conservative versi
52
52
  in your `Gemfile`:
53
53
 
54
54
  ```rb
55
- gem 'rubocop', '~> 1.72', require: false
55
+ gem 'rubocop', '~> 1.73', require: false
56
56
  ```
57
57
 
58
58
  See [our versioning policy](https://docs.rubocop.org/rubocop/versioning.html) for further details.
@@ -241,9 +241,9 @@ Become a sponsor and get your logo on our README on GitHub with a link to your s
241
241
  <a href="https://opencollective.com/rubocop/organization/28/website" target="_blank"><img src="https://opencollective.com/rubocop/organization/28/avatar.svg"></a>
242
242
  <a href="https://opencollective.com/rubocop/organization/29/website" target="_blank"><img src="https://opencollective.com/rubocop/organization/29/avatar.svg"></a>
243
243
 
244
- ## Changelog
244
+ ## Release Notes
245
245
 
246
- RuboCop's changelog is available [here](CHANGELOG.md).
246
+ RuboCop's release notes are available [here](https://github.com/rubocop/rubocop/releases).
247
247
 
248
248
  ## Copyright
249
249
 
data/config/default.yml CHANGED
@@ -1887,10 +1887,9 @@ Lint/EmptyConditionalBody:
1887
1887
  Description: 'Checks for the presence of `if`, `elsif` and `unless` branches without a body.'
1888
1888
  Enabled: true
1889
1889
  AutoCorrect: contextual
1890
- SafeAutoCorrect: false
1891
1890
  AllowComments: true
1892
1891
  VersionAdded: '0.89'
1893
- VersionChanged: '1.61'
1892
+ VersionChanged: '1.73'
1894
1893
 
1895
1894
  Lint/EmptyEnsure:
1896
1895
  Description: 'Checks for empty ensure block.'
@@ -2046,6 +2045,7 @@ Lint/LambdaWithoutLiteralBlock:
2046
2045
  Lint/LiteralAsCondition:
2047
2046
  Description: 'Checks of literals used in conditions.'
2048
2047
  Enabled: true
2048
+ AutoCorrect: contextual
2049
2049
  VersionAdded: '0.51'
2050
2050
 
2051
2051
  Lint/LiteralAssignmentInCondition:
@@ -2259,9 +2259,8 @@ Lint/RedundantRegexpQuantifiers:
2259
2259
  Lint/RedundantRequireStatement:
2260
2260
  Description: 'Checks for unnecessary `require` statement.'
2261
2261
  Enabled: true
2262
- SafeAutoCorrect: false
2263
2262
  VersionAdded: '0.76'
2264
- VersionChanged: '1.57'
2263
+ VersionChanged: '1.73'
2265
2264
 
2266
2265
  Lint/RedundantSafeNavigation:
2267
2266
  Description: 'Checks for redundant safe navigation calls.'
@@ -3081,13 +3080,15 @@ Naming/VariableName:
3081
3080
  StyleGuide: '#snake-case-symbols-methods-vars'
3082
3081
  Enabled: true
3083
3082
  VersionAdded: '0.50'
3084
- VersionChanged: '1.8'
3083
+ VersionChanged: '1.73'
3085
3084
  EnforcedStyle: snake_case
3086
3085
  SupportedStyles:
3087
3086
  - snake_case
3088
3087
  - camelCase
3089
3088
  AllowedIdentifiers: []
3090
3089
  AllowedPatterns: []
3090
+ ForbiddenIdentifiers: []
3091
+ ForbiddenPatterns: []
3091
3092
 
3092
3093
  Naming/VariableNumber:
3093
3094
  Description: 'Use the configured style when numbering symbols, methods and variables.'
@@ -3922,6 +3923,8 @@ Style/EndlessMethod:
3922
3923
  - allow_single_line
3923
3924
  - allow_always
3924
3925
  - disallow
3926
+ - require_single_line
3927
+ - require_always
3925
3928
 
3926
3929
  Style/EnvHome:
3927
3930
  Description: "Checks for consistent usage of `ENV['HOME']`."
@@ -5104,6 +5107,9 @@ Style/RedundantCondition:
5104
5107
  Description: 'Checks for unnecessary conditional expressions.'
5105
5108
  Enabled: true
5106
5109
  VersionAdded: '0.76'
5110
+ VersionChanged: '1.73'
5111
+ AllowedMethods:
5112
+ - nonzero?
5107
5113
 
5108
5114
  Style/RedundantConditional:
5109
5115
  Description: "Don't return true/false from a conditional."
@@ -5701,14 +5707,17 @@ Style/TrailingCommaInArrayLiteral:
5701
5707
  StyleGuide: '#no-trailing-array-commas'
5702
5708
  Enabled: true
5703
5709
  VersionAdded: '0.53'
5704
- # If `comma`, the cop requires a comma after the last item in an array,
5705
- # but only when each item is on its own line.
5706
- # If `consistent_comma`, the cop requires a comma after the last item of all
5707
- # non-empty, multiline array literals.
5710
+ # If `comma`, the cop requires a comma after the last item in an array, but only when each item is
5711
+ # on its own line.
5712
+ # If `consistent_comma`, the cop requires a comma after the last item of all non-empty, multiline
5713
+ # array literals.
5714
+ # If `diff_comma`, the cop requires a comma after the last item of all non-empty, multiline array
5715
+ # literals, but only when that last item immediately precedes a newline.
5708
5716
  EnforcedStyleForMultiline: no_comma
5709
5717
  SupportedStylesForMultiline:
5710
5718
  - comma
5711
5719
  - consistent_comma
5720
+ - diff_comma
5712
5721
  - no_comma
5713
5722
 
5714
5723
  Style/TrailingCommaInBlockArgs:
@@ -5720,14 +5729,17 @@ Style/TrailingCommaInBlockArgs:
5720
5729
  Style/TrailingCommaInHashLiteral:
5721
5730
  Description: 'Checks for trailing comma in hash literals.'
5722
5731
  Enabled: true
5723
- # If `comma`, the cop requires a comma after the last item in a hash,
5724
- # but only when each item is on its own line.
5725
- # If `consistent_comma`, the cop requires a comma after the last item of all
5726
- # non-empty, multiline hash literals.
5732
+ # If `comma`, the cop requires a comma after the last item in a hash, but only when each item is
5733
+ # on its own line.
5734
+ # If `consistent_comma`, the cop requires a comma after the last item of all non-empty, multiline
5735
+ # hash literals.
5736
+ # If `diff_comma`, the cop requires a comma after the last item of all non-empty, multiline hash
5737
+ # literals, but only when that last item immediately precedes a newline.
5727
5738
  EnforcedStyleForMultiline: no_comma
5728
5739
  SupportedStylesForMultiline:
5729
5740
  - comma
5730
5741
  - consistent_comma
5742
+ - diff_comma
5731
5743
  - no_comma
5732
5744
  VersionAdded: '0.53'
5733
5745
 
@@ -6,6 +6,26 @@ InternalAffairs/CopDescription:
6
6
  Include:
7
7
  - 'lib/rubocop/cop/**/*.rb'
8
8
 
9
+ InternalAffairs/ExampleHeredocDelimiter:
10
+ Include:
11
+ - 'spec/rubocop/cop/**/*.rb'
12
+
13
+ InternalAffairs/ExampleDescription:
14
+ Include:
15
+ - 'spec/rubocop/cop/**/*.rb'
16
+
17
+ InternalAffairs/NodeTypeGroup:
18
+ Include:
19
+ - 'lib/rubocop/cop/**/*.rb'
20
+
21
+ InternalAffairs/OnSendWithoutOnCSend:
22
+ Include:
23
+ - 'lib/rubocop/cop/**/*.rb'
24
+
25
+ InternalAffairs/UndefinedConfig:
26
+ Include:
27
+ - 'lib/rubocop/cop/**/*.rb'
28
+
9
29
  InternalAffairs/UselessMessageAssertion:
10
30
  Include:
11
31
  - '**/*_spec.rb'
@@ -63,7 +63,6 @@ module RuboCop
63
63
  loaded_features = resolver.resolve_requires(path, hash)
64
64
  add_loaded_features(loaded_features)
65
65
 
66
- resolver.override_department_setting_for_cops({}, hash)
67
66
  resolver.resolve_inheritance_from_gems(hash)
68
67
  resolver.resolve_inheritance(path, hash, file, debug?)
69
68
  hash.delete('inherit_from')
@@ -123,7 +123,7 @@ module RuboCop
123
123
  elsif merge_hashes?(base_hash, derived_hash, key)
124
124
  result[key] = merge(base_hash[key], derived_hash[key], **opts)
125
125
  elsif should_union?(derived_hash, base_hash, opts[:inherit_mode], key)
126
- result[key] = base_hash[key] | derived_hash[key]
126
+ result[key] = Array(base_hash[key]) | Array(derived_hash[key])
127
127
  elsif opts[:debug]
128
128
  warn_on_duplicate_setting(base_hash, derived_hash, key, **opts)
129
129
  end
@@ -205,7 +205,7 @@ module RuboCop
205
205
  end
206
206
 
207
207
  def should_union?(derived_hash, base_hash, root_mode, key)
208
- return false unless base_hash[key].is_a?(Array)
208
+ return false unless base_hash[key].is_a?(Array) || derived_hash[key].is_a?(Array)
209
209
 
210
210
  derived_mode = derived_hash['inherit_mode']
211
211
  return false if should_override?(derived_mode, key)
@@ -9,7 +9,7 @@ module RuboCop
9
9
 
10
10
  # @api private
11
11
  COMMON_PARAMS = %w[Exclude Include Severity inherit_mode AutoCorrect StyleGuide Details
12
- Enabled].freeze
12
+ Enabled Reference].freeze
13
13
  # @api private
14
14
  INTERNAL_PARAMS = %w[Description StyleGuide
15
15
  VersionAdded VersionChanged VersionRemoved
@@ -50,10 +50,12 @@ module RuboCop
50
50
  }.freeze
51
51
 
52
52
  EXPECT_NO_CORRECTIONS_DESCRIPTION_MAPPING = {
53
- /\A(auto[- ]?)?correct/ => 'does not correct'
53
+ /\A(auto[- ]?)?corrects?/ => 'does not correct',
54
+ /\band (auto[- ]?)?corrects/ => 'but does not correct'
54
55
  }.freeze
55
56
 
56
57
  EXPECT_CORRECTION_DESCRIPTION_MAPPING = {
58
+ /\bbut (does not|doesn't) (auto[- ]?)?correct/ => 'and autocorrects',
57
59
  /\b(does not|doesn't) (auto[- ]?)?correct/ => 'autocorrects'
58
60
  }.freeze
59
61
 
@@ -0,0 +1,91 @@
1
+ # frozen_string_literal: true
2
+
3
+ module RuboCop
4
+ module Cop
5
+ module InternalAffairs
6
+ # Checks that node types are checked against their group when all types of a
7
+ # group are checked.
8
+ #
9
+ # @example
10
+ # # bad
11
+ # node.type?(:irange, :erange)
12
+ #
13
+ # # good
14
+ # node.range_type?
15
+ #
16
+ # # bad
17
+ # node.type?(:irange, :erange, :send, :csend)
18
+ #
19
+ # # good
20
+ # node.type?(:range, :call)
21
+ #
22
+ class NodeTypeGroup < Base
23
+ extend AutoCorrector
24
+ include RangeHelp
25
+
26
+ MSG = 'Use `:%<group>s` instead of individually listing group types.'
27
+
28
+ RESTRICT_ON_SEND = %i[type? each_ancestor each_child_node each_descendant each_node].freeze
29
+
30
+ def on_send(node)
31
+ return unless node.receiver
32
+
33
+ symbol_args = node.arguments.select(&:sym_type?)
34
+ return if symbol_args.none?
35
+
36
+ NodePatternGroups::NODE_GROUPS.each do |group_name, group_types|
37
+ next unless group_satisfied?(group_types, symbol_args)
38
+
39
+ offense_range = arguments_range(node)
40
+ add_offense(offense_range, message: format(MSG, group: group_name)) do |corrector|
41
+ autocorrect(corrector, node, symbol_args, group_name, group_types)
42
+ end
43
+ end
44
+ end
45
+ alias on_csend on_send
46
+
47
+ private
48
+
49
+ def arguments_range(node)
50
+ range_between(
51
+ node.first_argument.source_range.begin_pos,
52
+ node.last_argument.source_range.end_pos
53
+ )
54
+ end
55
+
56
+ def group_satisfied?(group_types, symbol_args)
57
+ group_types.all? { |type| symbol_args.any? { |arg| arg.value == type } }
58
+ end
59
+
60
+ def autocorrect(corrector, node, symbol_args, group_name, group_types)
61
+ if node.method?(:type?) && node.arguments.count == group_types.count
62
+ autocorrect_to_explicit_predicate(corrector, node, group_name)
63
+ else
64
+ autocorrect_keep_method(corrector, symbol_args, group_name, group_types)
65
+ end
66
+ end
67
+
68
+ def autocorrect_to_explicit_predicate(corrector, node, group_name)
69
+ corrector.replace(node.selector, "#{group_name}_type?")
70
+ corrector.remove(arguments_range(node))
71
+ end
72
+
73
+ def autocorrect_keep_method(corrector, symbol_args, group_name, group_types)
74
+ first_replaced = false
75
+ symbol_args.each do |arg|
76
+ next unless group_types.include?(arg.value)
77
+
78
+ if first_replaced
79
+ range = range_with_surrounding_space(arg.source_range)
80
+ range = range_with_surrounding_comma(range, :left)
81
+ corrector.remove(range)
82
+ else
83
+ first_replaced = true
84
+ corrector.replace(arg, ":#{group_name}")
85
+ end
86
+ end
87
+ end
88
+ end
89
+ end
90
+ end
91
+ end
@@ -17,6 +17,7 @@ require_relative 'internal_affairs/node_destructuring'
17
17
  require_relative 'internal_affairs/node_first_or_last_argument'
18
18
  require_relative 'internal_affairs/node_matcher_directive'
19
19
  require_relative 'internal_affairs/node_pattern_groups'
20
+ require_relative 'internal_affairs/node_type_group'
20
21
  require_relative 'internal_affairs/node_type_multiple_predicates'
21
22
  require_relative 'internal_affairs/node_type_predicate'
22
23
  require_relative 'internal_affairs/numblock_handler'
@@ -155,10 +155,10 @@ module RuboCop
155
155
  end
156
156
 
157
157
  def all_elements_aligned?(elements)
158
- elements.flat_map do |e|
159
- if e.hash_type?
160
- e.each_child_node.map { |child| child.loc.column }
161
- else
158
+ if elements.first.hash_type?
159
+ elements.first.each_child_node.map { |child| child.loc.column }
160
+ else
161
+ elements.flat_map do |e|
162
162
  e.loc.column
163
163
  end
164
164
  end.uniq.count == 1
@@ -93,7 +93,7 @@ module RuboCop
93
93
  add_offense(node, message: message) do |corrector|
94
94
  line = range_by_whole_lines(node.source_range)
95
95
 
96
- corrector.insert_before(line, "\n") unless previous_line_empty?(node.first_line)
96
+ corrector.insert_before(line, "\n") if should_insert_line_before?(node)
97
97
 
98
98
  correct_next_line_if_denied_style(corrector, node, line)
99
99
  end
@@ -122,6 +122,8 @@ module RuboCop
122
122
  end
123
123
 
124
124
  def correct_next_line_if_denied_style(corrector, node, line)
125
+ return unless should_insert_line_after?(node)
126
+
125
127
  case style
126
128
  when :around
127
129
  corrector.insert_after(line, "\n") unless next_line_empty?(node.last_line)
@@ -205,6 +207,29 @@ module RuboCop
205
207
  format(MSG_BEFORE_FOR_ONLY_BEFORE, modifier: modifier)
206
208
  end
207
209
  end
210
+
211
+ def should_insert_line_before?(node)
212
+ return false if previous_line_empty?(node.first_line)
213
+ return true unless inside_block?(node) && no_empty_lines_around_block_body?
214
+ return true unless node.parent.begin_type?
215
+
216
+ node.parent.children.first != node
217
+ end
218
+
219
+ def should_insert_line_after?(node)
220
+ return true unless inside_block?(node) && no_empty_lines_around_block_body?
221
+
222
+ node.parent.children.last != node
223
+ end
224
+
225
+ def inside_block?(node)
226
+ node.parent.block_type? || (node.parent.begin_type? && node.parent.parent&.block_type?)
227
+ end
228
+
229
+ def no_empty_lines_around_block_body?
230
+ config.for_enabled_cop('Layout/EmptyLinesAroundBlockBody')['EnforcedStyle'] ==
231
+ 'no_empty_lines'
232
+ end
208
233
  end
209
234
  end
210
235
  end
@@ -209,7 +209,7 @@ module RuboCop
209
209
  # are not bisected.
210
210
  # If the string contains spaces, use them to determine a place for a clean break;
211
211
  # otherwise, the string will be broken at the line length limit.
212
- def breakable_string_range(node) # rubocop:disable Metrics/AbcSize
212
+ def breakable_string_range(node)
213
213
  source_range = node.source_range
214
214
  relevant_substr = largest_possible_string(node)
215
215
 
@@ -221,13 +221,13 @@ module RuboCop
221
221
  adjustment = max - source_range.last_column - 3
222
222
  return if adjustment.abs > source_range.size
223
223
 
224
- source_range.adjust(end_pos: max - source_range.last_column - 3)
224
+ source_range.adjust(end_pos: adjustment)
225
225
  end
226
226
  end
227
227
 
228
228
  def breakable_dstr_begin_position(node)
229
229
  source_range = node.source_range
230
- source_range.begin_pos if source_range.begin_pos < max && source_range.end_pos >= max
230
+ source_range.begin_pos if source_range.column < max && source_range.last_column >= max
231
231
  end
232
232
 
233
233
  def breakable_range_by_line_index
@@ -54,14 +54,12 @@ module RuboCop
54
54
  # if a method definition is inside an if, it is very likely
55
55
  # that a different definition is used depending on platform, etc.
56
56
  return if node.each_ancestor.any?(&:if_type?)
57
- return if possible_dsl?(node)
58
57
 
59
58
  found_instance_method(node, node.method_name)
60
59
  end
61
60
 
62
61
  def on_defs(node)
63
62
  return if node.each_ancestor.any?(&:if_type?)
64
- return if possible_dsl?(node)
65
63
 
66
64
  if node.receiver.const_type?
67
65
  _, const_name = *node.receiver
@@ -79,7 +77,6 @@ module RuboCop
79
77
  def on_alias(node)
80
78
  return unless (name = method_alias?(node))
81
79
  return if node.ancestors.any?(&:if_type?)
82
- return if possible_dsl?(node)
83
80
 
84
81
  found_instance_method(node, name)
85
82
  end
@@ -94,7 +91,6 @@ module RuboCop
94
91
  def on_send(node)
95
92
  if (name = alias_method?(node))
96
93
  return if node.ancestors.any?(&:if_type?)
97
- return if possible_dsl?(node)
98
94
 
99
95
  found_instance_method(node, name)
100
96
  elsif (attr = node.attribute_accessor?)
@@ -237,16 +233,6 @@ module RuboCop
237
233
  end
238
234
  end
239
235
 
240
- def possible_dsl?(node)
241
- # DSL methods may evaluate a block in the context of a newly created
242
- # class or module
243
- # Assume that if a method definition is inside any block call which
244
- # we can't identify, it could be a DSL
245
- node.each_ancestor(:block).any? do |ancestor|
246
- !ancestor.method?(:class_eval) && !ancestor.class_constructor?
247
- end
248
- end
249
-
250
236
  def source_location(node)
251
237
  range = node.source_range
252
238
  path = smart_path(range.source_buffer.name)
@@ -7,11 +7,6 @@ module RuboCop
7
7
  #
8
8
  # NOTE: empty `else` branches are handled by `Style/EmptyElse`.
9
9
  #
10
- # @safety
11
- # Autocorrection for this cop is not safe. The conditions for empty branches that
12
- # the autocorrection removes may have side effects, or the logic in subsequent
13
- # branches may change due to the removal of a previous condition.
14
- #
15
10
  # @example
16
11
  # # bad
17
12
  # if condition
@@ -41,6 +36,13 @@ module RuboCop
41
36
  # if condition
42
37
  # do_something
43
38
  # elsif other_condition
39
+ # nil
40
+ # end
41
+ #
42
+ # # good
43
+ # if condition
44
+ # do_something
45
+ # elsif other_condition
44
46
  # do_something_else
45
47
  # end
46
48
  #
@@ -63,11 +65,9 @@ module RuboCop
63
65
  class EmptyConditionalBody < Base
64
66
  extend AutoCorrector
65
67
  include CommentsHelp
66
- include RangeHelp
67
68
 
68
69
  MSG = 'Avoid `%<keyword>s` branches without a body.'
69
70
 
70
- # rubocop:disable Metrics/AbcSize
71
71
  def on_if(node)
72
72
  return if node.body || same_line?(node.loc.begin, node.loc.end)
73
73
  return if cop_config['AllowComments'] && contains_comments?(node)
@@ -75,12 +75,11 @@ module RuboCop
75
75
  range = offense_range(node)
76
76
 
77
77
  add_offense(range, message: format(MSG, keyword: node.keyword)) do |corrector|
78
- next if node.parent&.call_type?
78
+ next unless can_simplify_conditional?(node)
79
79
 
80
- autocorrect(corrector, node)
80
+ flip_orphaned_else(corrector, node)
81
81
  end
82
82
  end
83
- # rubocop:enable Metrics/AbcSize
84
83
 
85
84
  private
86
85
 
@@ -92,53 +91,23 @@ module RuboCop
92
91
  end
93
92
  end
94
93
 
95
- def autocorrect(corrector, node)
96
- remove_comments(corrector, node)
97
- remove_empty_branch(corrector, node)
98
- correct_other_branches(corrector, node)
99
- end
100
-
101
- def remove_comments(corrector, node)
102
- comments_in_range(node).each do |comment|
103
- range = range_by_whole_lines(comment.source_range, include_final_newline: true)
104
- corrector.remove(range)
105
- end
94
+ def can_simplify_conditional?(node)
95
+ node.else_branch && node.loc.else.source == 'else'
106
96
  end
107
97
 
108
- # rubocop:disable Metrics/AbcSize
109
98
  def remove_empty_branch(corrector, node)
110
99
  range = if empty_if_branch?(node) && else_branch?(node)
111
100
  branch_range(node)
112
- elsif same_line?(node, else_kw_loc = node.loc.else)
113
- node.source_range.begin.join(else_kw_loc.begin)
114
- elsif node.parent&.loc.respond_to?(:end) &&
115
- same_line?(node, end_loc = node.parent.loc.end)
116
- node.source_range.begin.join(end_loc.begin)
117
101
  else
118
102
  deletion_range(branch_range(node))
119
103
  end
120
104
 
121
105
  corrector.remove(range)
122
106
  end
123
- # rubocop:enable Metrics/AbcSize
124
-
125
- def correct_other_branches(corrector, node)
126
- return unless require_other_branches_correction?(node)
127
-
128
- if node.else_branch&.if_type? && !node.else_branch.modifier_form?
129
- # Replace an orphaned `elsif` with `if`
130
- corrector.replace(node.else_branch.loc.keyword, 'if')
131
- else
132
- # Flip orphaned `else`
133
- corrector.replace(node.loc.else, "#{node.inverse_keyword} #{node.condition.source}")
134
- end
135
- end
136
-
137
- def require_other_branches_correction?(node)
138
- return false unless node.if_type? && node.else?
139
- return false if !empty_if_branch?(node) && node.elsif?
140
107
 
141
- !empty_elsif_branch?(node)
108
+ def flip_orphaned_else(corrector, node)
109
+ corrector.replace(node.loc.else, "#{node.inverse_keyword} #{node.condition.source}")
110
+ remove_empty_branch(corrector, node)
142
111
  end
143
112
 
144
113
  def empty_if_branch?(node)
@@ -149,36 +118,17 @@ module RuboCop
149
118
  if_branch.if_type? && !if_branch.body
150
119
  end
151
120
 
152
- def empty_elsif_branch?(node)
153
- return false unless (else_branch = node.else_branch)
154
-
155
- else_branch.if_type? && !else_branch.body
156
- end
157
-
158
121
  def else_branch?(node)
159
122
  node.else_branch && !node.else_branch.if_type?
160
123
  end
161
124
 
162
- # rubocop:disable Metrics/AbcSize
163
125
  def branch_range(node)
164
126
  if empty_if_branch?(node) && else_branch?(node)
165
127
  node.source_range.with(end_pos: node.loc.else.begin_pos)
166
128
  elsif node.loc.else
167
129
  node.source_range.with(end_pos: node.condition.source_range.end_pos)
168
- elsif all_branches_body_missing?(node)
169
- if_node = node.ancestors.detect(&:if?)
170
- node.source_range.join(if_node.loc.end.end)
171
- else
172
- node.source_range
173
130
  end
174
131
  end
175
- # rubocop:enable Metrics/AbcSize
176
-
177
- def all_branches_body_missing?(node)
178
- return false unless node.parent&.if_type?
179
-
180
- node.parent.branches.compact.empty?
181
- end
182
132
 
183
133
  def deletion_range(range)
184
134
  # Collect a range between the start of the `if` node and the next relevant node,
@@ -156,12 +156,6 @@ module RuboCop
156
156
 
157
157
  overridden_kwargs
158
158
  end
159
-
160
- def arguments_range(node)
161
- arguments = node.arguments
162
-
163
- range_between(arguments.first.source_range.begin_pos, arguments.last.source_range.end_pos)
164
- end
165
159
  end
166
160
  end
167
161
  end
@@ -81,21 +81,16 @@ module RuboCop
81
81
  (node.numeric_type? && node.value.zero?) || node.nil_type?
82
82
  end
83
83
 
84
- # rubocop:disable Metrics/PerceivedComplexity
85
84
  def check_send(node)
86
85
  if node.arithmetic_operation?
87
86
  float?(node.receiver) || float?(node.first_argument)
88
87
  elsif FLOAT_RETURNING_METHODS.include?(node.method_name)
89
88
  true
90
89
  elsif node.receiver&.float_type?
91
- if FLOAT_INSTANCE_METHODS.include?(node.method_name)
92
- true
93
- else
90
+ FLOAT_INSTANCE_METHODS.include?(node.method_name) ||
94
91
  check_numeric_returning_method(node)
95
- end
96
92
  end
97
93
  end
98
- # rubocop:enable Metrics/PerceivedComplexity
99
94
 
100
95
  def check_numeric_returning_method(node)
101
96
  return false unless node.receiver