rubocop 1.73.1 → 1.75.1

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 (127) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +1 -1
  3. data/config/default.yml +64 -10
  4. data/config/internal_affairs.yml +4 -0
  5. data/config/obsoletion.yml +3 -1
  6. data/lib/rubocop/cli.rb +1 -1
  7. data/lib/rubocop/config.rb +35 -6
  8. data/lib/rubocop/config_loader.rb +4 -1
  9. data/lib/rubocop/config_loader_resolver.rb +2 -1
  10. data/lib/rubocop/config_obsoletion/extracted_cop.rb +4 -3
  11. data/lib/rubocop/config_obsoletion/renamed_cop.rb +18 -3
  12. data/lib/rubocop/config_obsoletion.rb +46 -2
  13. data/lib/rubocop/config_validator.rb +1 -0
  14. data/lib/rubocop/cop/internal_affairs/example_description.rb +3 -1
  15. data/lib/rubocop/cop/internal_affairs/node_pattern_groups.rb +1 -1
  16. data/lib/rubocop/cop/internal_affairs/node_type_group.rb +91 -0
  17. data/lib/rubocop/cop/internal_affairs/redundant_described_class_as_subject.rb +6 -5
  18. data/lib/rubocop/cop/internal_affairs.rb +1 -0
  19. data/lib/rubocop/cop/layout/block_alignment.rb +1 -0
  20. data/lib/rubocop/cop/layout/block_end_newline.rb +1 -0
  21. data/lib/rubocop/cop/layout/else_alignment.rb +1 -1
  22. data/lib/rubocop/cop/layout/empty_line_between_defs.rb +1 -1
  23. data/lib/rubocop/cop/layout/empty_lines_around_access_modifier.rb +1 -0
  24. data/lib/rubocop/cop/layout/empty_lines_around_block_body.rb +1 -0
  25. data/lib/rubocop/cop/layout/indentation_width.rb +1 -0
  26. data/lib/rubocop/cop/layout/line_length.rb +5 -1
  27. data/lib/rubocop/cop/layout/multiline_block_layout.rb +1 -0
  28. data/lib/rubocop/cop/layout/multiline_method_parameter_line_breaks.rb +1 -0
  29. data/lib/rubocop/cop/layout/redundant_line_break.rb +9 -5
  30. data/lib/rubocop/cop/layout/rescue_ensure_alignment.rb +1 -1
  31. data/lib/rubocop/cop/layout/space_around_operators.rb +4 -1
  32. data/lib/rubocop/cop/layout/space_before_block_braces.rb +1 -0
  33. data/lib/rubocop/cop/layout/space_inside_block_braces.rb +1 -0
  34. data/lib/rubocop/cop/lint/debugger.rb +2 -2
  35. data/lib/rubocop/cop/lint/empty_conditional_body.rb +15 -70
  36. data/lib/rubocop/cop/lint/erb_new_arguments.rb +0 -6
  37. data/lib/rubocop/cop/lint/literal_as_condition.rb +4 -0
  38. data/lib/rubocop/cop/lint/non_local_exit_from_iterator.rb +2 -2
  39. data/lib/rubocop/cop/lint/raise_exception.rb +29 -10
  40. data/lib/rubocop/cop/lint/redundant_type_conversion.rb +9 -3
  41. data/lib/rubocop/cop/lint/redundant_with_index.rb +3 -0
  42. data/lib/rubocop/cop/lint/redundant_with_object.rb +3 -0
  43. data/lib/rubocop/cop/lint/return_in_void_context.rb +4 -11
  44. data/lib/rubocop/cop/lint/shadowing_outer_local_variable.rb +8 -1
  45. data/lib/rubocop/cop/lint/shared_mutable_default.rb +12 -1
  46. data/lib/rubocop/cop/lint/unexpected_block_arity.rb +2 -0
  47. data/lib/rubocop/cop/lint/unreachable_code.rb +1 -0
  48. data/lib/rubocop/cop/lint/unreachable_loop.rb +5 -5
  49. data/lib/rubocop/cop/lint/useless_access_modifier.rb +1 -0
  50. data/lib/rubocop/cop/lint/useless_constant_scoping.rb +2 -11
  51. data/lib/rubocop/cop/lint/void.rb +1 -0
  52. data/lib/rubocop/cop/metrics/block_length.rb +1 -0
  53. data/lib/rubocop/cop/metrics/method_length.rb +1 -0
  54. data/lib/rubocop/cop/metrics/utils/code_length_calculator.rb +1 -1
  55. data/lib/rubocop/cop/mixin/check_line_breakable.rb +2 -2
  56. data/lib/rubocop/cop/mixin/check_single_line_suitability.rb +1 -1
  57. data/lib/rubocop/cop/mixin/forbidden_identifiers.rb +20 -0
  58. data/lib/rubocop/cop/mixin/forbidden_pattern.rb +16 -0
  59. data/lib/rubocop/cop/mixin/method_complexity.rb +1 -0
  60. data/lib/rubocop/cop/mixin/range_help.rb +12 -0
  61. data/lib/rubocop/cop/mixin/target_ruby_version.rb +1 -1
  62. data/lib/rubocop/cop/naming/method_name.rb +64 -8
  63. data/lib/rubocop/cop/naming/variable_name.rb +6 -19
  64. data/lib/rubocop/cop/registry.rb +9 -6
  65. data/lib/rubocop/cop/style/array_intersect.rb +39 -28
  66. data/lib/rubocop/cop/style/block_delimiters.rb +2 -1
  67. data/lib/rubocop/cop/style/class_and_module_children.rb +29 -7
  68. data/lib/rubocop/cop/style/collection_methods.rb +1 -0
  69. data/lib/rubocop/cop/style/combinable_loops.rb +1 -0
  70. data/lib/rubocop/cop/style/commented_keyword.rb +9 -2
  71. data/lib/rubocop/cop/style/comparable_between.rb +75 -0
  72. data/lib/rubocop/cop/style/double_negation.rb +1 -1
  73. data/lib/rubocop/cop/style/expand_path_arguments.rb +2 -7
  74. data/lib/rubocop/cop/style/exponential_notation.rb +2 -2
  75. data/lib/rubocop/cop/style/for.rb +1 -0
  76. data/lib/rubocop/cop/style/format_string_token.rb +38 -11
  77. data/lib/rubocop/cop/style/guard_clause.rb +2 -1
  78. data/lib/rubocop/cop/style/hash_each_methods.rb +3 -2
  79. data/lib/rubocop/cop/style/hash_fetch_chain.rb +105 -0
  80. data/lib/rubocop/cop/style/if_inside_else.rb +10 -13
  81. data/lib/rubocop/cop/style/if_unless_modifier.rb +2 -2
  82. data/lib/rubocop/cop/style/inverse_methods.rb +9 -5
  83. data/lib/rubocop/cop/style/invertible_unless_condition.rb +2 -2
  84. data/lib/rubocop/cop/style/ip_addresses.rb +2 -2
  85. data/lib/rubocop/cop/style/it_block_parameter.rb +100 -0
  86. data/lib/rubocop/cop/style/keyword_parameters_order.rb +13 -7
  87. data/lib/rubocop/cop/style/lambda.rb +1 -0
  88. data/lib/rubocop/cop/style/map_into_array.rb +1 -0
  89. data/lib/rubocop/cop/style/method_call_with_args_parentheses/omit_parentheses.rb +3 -3
  90. data/lib/rubocop/cop/style/method_called_on_do_end_block.rb +1 -0
  91. data/lib/rubocop/cop/style/multiline_block_chain.rb +2 -1
  92. data/lib/rubocop/cop/style/multiline_method_signature.rb +1 -9
  93. data/lib/rubocop/cop/style/next.rb +44 -0
  94. data/lib/rubocop/cop/style/object_then.rb +1 -0
  95. data/lib/rubocop/cop/style/proc.rb +1 -0
  96. data/lib/rubocop/cop/style/raise_args.rb +8 -8
  97. data/lib/rubocop/cop/style/redundant_begin.rb +1 -0
  98. data/lib/rubocop/cop/style/redundant_condition.rb +2 -3
  99. data/lib/rubocop/cop/style/redundant_current_directory_in_path.rb +14 -4
  100. data/lib/rubocop/cop/style/redundant_format.rb +10 -3
  101. data/lib/rubocop/cop/style/redundant_freeze.rb +1 -1
  102. data/lib/rubocop/cop/style/redundant_parentheses.rb +2 -1
  103. data/lib/rubocop/cop/style/redundant_self.rb +1 -0
  104. data/lib/rubocop/cop/style/redundant_sort_by.rb +17 -1
  105. data/lib/rubocop/cop/style/rescue_modifier.rb +3 -0
  106. data/lib/rubocop/cop/style/select_by_regexp.rb +4 -1
  107. data/lib/rubocop/cop/style/single_line_do_end_block.rb +3 -1
  108. data/lib/rubocop/cop/style/sole_nested_conditional.rb +41 -106
  109. data/lib/rubocop/cop/style/symbol_proc.rb +2 -0
  110. data/lib/rubocop/cop/style/top_level_method_definition.rb +1 -0
  111. data/lib/rubocop/cop/utils/format_string.rb +5 -2
  112. data/lib/rubocop/cop/variable_force/scope.rb +1 -1
  113. data/lib/rubocop/cop/variable_force/variable.rb +1 -6
  114. data/lib/rubocop/cop/variable_force.rb +1 -1
  115. data/lib/rubocop/directive_comment.rb +1 -1
  116. data/lib/rubocop/ext/regexp_node.rb +0 -1
  117. data/lib/rubocop/lsp/runtime.rb +4 -4
  118. data/lib/rubocop/lsp/stdin_runner.rb +3 -1
  119. data/lib/rubocop/rspec/cop_helper.rb +4 -1
  120. data/lib/rubocop/rspec/shared_contexts.rb +20 -0
  121. data/lib/rubocop/rspec/support.rb +2 -0
  122. data/lib/rubocop/runner.rb +5 -1
  123. data/lib/rubocop/target_ruby.rb +1 -1
  124. data/lib/rubocop/version.rb +14 -7
  125. data/lib/rubocop.rb +5 -0
  126. data/lib/ruby_lsp/rubocop/runtime_adapter.rb +20 -2
  127. metadata +12 -6
@@ -4,8 +4,8 @@ module RuboCop
4
4
  module Cop
5
5
  module Lint
6
6
  # Checks for useless constant scoping. Private constants must be defined using
7
- # `private_constant` or `class << self`. Even if `private` access modifier is used,
8
- # it is public scope despite its appearance.
7
+ # `private_constant`. Even if `private` access modifier is used, it is public scope despite
8
+ # its appearance.
9
9
  #
10
10
  # It does not support autocorrection due to behavior change and multiple ways to fix it.
11
11
  # Or a public constant may be intended.
@@ -26,14 +26,6 @@ module RuboCop
26
26
  #
27
27
  # # good
28
28
  # class Foo
29
- # class << self
30
- # private
31
- # PRIVATE_CONST = 42
32
- # end
33
- # end
34
- #
35
- # # good
36
- # class Foo
37
29
  # PUBLIC_CONST = 42 # If private scope is not intended.
38
30
  # end
39
31
  #
@@ -46,7 +38,6 @@ module RuboCop
46
38
  PATTERN
47
39
 
48
40
  def on_casgn(node)
49
- return if node.each_ancestor(:sclass).any?
50
41
  return unless after_private_modifier?(node.left_siblings)
51
42
  return if private_constantize?(node.right_siblings, node.name)
52
43
 
@@ -86,6 +86,7 @@ module RuboCop
86
86
  check_expression(node.body)
87
87
  end
88
88
  alias on_numblock on_block
89
+ alias on_itblock on_block
89
90
 
90
91
  def on_begin(node)
91
92
  check_begin(node)
@@ -57,6 +57,7 @@ module RuboCop
57
57
  check_code_length(node)
58
58
  end
59
59
  alias on_numblock on_block
60
+ alias on_itblock on_block
60
61
 
61
62
  private
62
63
 
@@ -63,6 +63,7 @@ module RuboCop
63
63
  check_code_length(node)
64
64
  end
65
65
  alias on_numblock on_block
66
+ alias on_itblock on_block
66
67
 
67
68
  private
68
69
 
@@ -145,7 +145,7 @@ module RuboCop
145
145
 
146
146
  def extract_body(node)
147
147
  case node.type
148
- when :class, :module, :sclass, :block, :numblock, :def, :defs
148
+ when :class, :module, :sclass, :block, :numblock, :itblock, :def, :defs
149
149
  node.body
150
150
  when :casgn
151
151
  extract_body(node.expression)
@@ -48,7 +48,7 @@ module RuboCop
48
48
 
49
49
  args = process_args(node.arguments)
50
50
  return extract_breakable_node_from_elements(node, args, max)
51
- elsif node.def_type?
51
+ elsif node.type?(:def, :defs)
52
52
  return extract_breakable_node_from_elements(node, node.arguments, max)
53
53
  elsif node.type?(:array, :hash)
54
54
  return extract_breakable_node_from_elements(node, node.children, max)
@@ -220,7 +220,7 @@ module RuboCop
220
220
 
221
221
  # @api private
222
222
  def already_on_multiple_lines?(node)
223
- return node.first_line != node.last_argument.last_line if node.def_type?
223
+ return node.first_line != node.last_argument.last_line if node.type?(:def, :defs)
224
224
 
225
225
  !node.single_line?
226
226
  end
@@ -35,7 +35,7 @@ module RuboCop
35
35
  comment_line_numbers = processed_source.comments.map { |comment| comment.loc.line }
36
36
 
37
37
  comment_line_numbers.any? do |comment_line_number|
38
- comment_line_number >= node.first_line && comment_line_number <= node.last_line
38
+ comment_line_number.between?(node.first_line, node.last_line)
39
39
  end
40
40
  end
41
41
 
@@ -0,0 +1,20 @@
1
+ # frozen_string_literal: true
2
+
3
+ module RuboCop
4
+ module Cop
5
+ # This module encapsulates the ability to forbid certain identifiers in a cop.
6
+ module ForbiddenIdentifiers
7
+ SIGILS = '@$' # if a variable starts with a sigil it will be removed
8
+
9
+ def forbidden_identifier?(name)
10
+ name = name.to_s.delete(SIGILS)
11
+
12
+ forbidden_identifiers.any? && forbidden_identifiers.include?(name)
13
+ end
14
+
15
+ def forbidden_identifiers
16
+ cop_config.fetch('ForbiddenIdentifiers', [])
17
+ end
18
+ end
19
+ end
20
+ end
@@ -0,0 +1,16 @@
1
+ # frozen_string_literal: true
2
+
3
+ module RuboCop
4
+ module Cop
5
+ # This module encapsulates the ability to forbid certain patterns in a cop.
6
+ module ForbiddenPattern
7
+ def forbidden_pattern?(name)
8
+ forbidden_patterns.any? { |pattern| Regexp.new(pattern).match?(name) }
9
+ end
10
+
11
+ def forbidden_patterns
12
+ cop_config.fetch('ForbiddenPatterns', [])
13
+ end
14
+ end
15
+ end
16
+ end
@@ -30,6 +30,7 @@ module RuboCop
30
30
  end
31
31
 
32
32
  alias on_numblock on_block
33
+ alias on_itblock on_block
33
34
 
34
35
  private
35
36
 
@@ -34,6 +34,18 @@ module RuboCop
34
34
  range_between(node.loc.begin.end_pos, node.loc.end.begin_pos)
35
35
  end
36
36
 
37
+ # A range containing the first to the last argument
38
+ # of a method call or method definition.
39
+ # def foo(a, b:)
40
+ # ^^^^^
41
+ # bar(1, 2, 3, &blk)
42
+ # ^^^^^^^^^^^^^
43
+ # baz { |x, y:, z:| }
44
+ # ^^^^^^^^^
45
+ def arguments_range(node)
46
+ node.first_argument.source_range.join(node.last_argument.source_range)
47
+ end
48
+
37
49
  def range_between(start_pos, end_pos)
38
50
  Parser::Source::Range.new(processed_source.buffer, start_pos, end_pos)
39
51
  end
@@ -29,7 +29,7 @@ module RuboCop
29
29
  min = required_minimum_ruby_version || 0
30
30
  max = required_maximum_ruby_version || Float::INFINITY
31
31
 
32
- min <= version && max >= version
32
+ version.between?(min, max)
33
33
  end
34
34
  end
35
35
  end
@@ -6,14 +6,31 @@ module RuboCop
6
6
  # Makes sure that all methods use the configured style,
7
7
  # snake_case or camelCase, for their names.
8
8
  #
9
- # This cop has `AllowedPatterns` configuration option.
9
+ # Method names matching patterns are always allowed.
10
10
  #
11
- # Naming/MethodName:
12
- # AllowedPatterns:
13
- # - '\AonSelectionBulkChange\z'
14
- # - '\AonSelectionCleared\z'
11
+ # The cop can be configured with `AllowedPatterns` to allow certain regexp patterns:
15
12
  #
16
- # Method names matching patterns are always allowed.
13
+ # [source,yaml]
14
+ # ----
15
+ # Naming/MethodName:
16
+ # AllowedPatterns:
17
+ # - '\AonSelectionBulkChange\z'
18
+ # - '\AonSelectionCleared\z'
19
+ # ----
20
+ #
21
+ # As well, you can also forbid specific method names or regexp patterns
22
+ # using `ForbiddenIdentifiers` or `ForbiddenPatterns`:
23
+ #
24
+ # [source,yaml]
25
+ # ----
26
+ # Naming/MethodName:
27
+ # ForbiddenIdentifiers:
28
+ # - 'def'
29
+ # - 'super'
30
+ # ForbiddenPatterns:
31
+ # - '_v1\z'
32
+ # - '_gen1\z'
33
+ # ----
17
34
  #
18
35
  # @example EnforcedStyle: snake_case (default)
19
36
  # # bad
@@ -28,12 +45,26 @@ module RuboCop
28
45
  #
29
46
  # # good
30
47
  # def fooBar; end
48
+ #
49
+ # @example ForbiddenIdentifiers: ['def', 'super']
50
+ # # bad
51
+ # def def; end
52
+ # def super; end
53
+ #
54
+ # @example ForbiddenPatterns: ['_v1\z', '_gen1\z']
55
+ # # bad
56
+ # def release_v1; end
57
+ # def api_gen1; end
58
+ #
31
59
  class MethodName < Base
32
60
  include ConfigurableNaming
33
61
  include AllowedPattern
34
62
  include RangeHelp
63
+ include ForbiddenIdentifiers
64
+ include ForbiddenPattern
35
65
 
36
66
  MSG = 'Use %<style>s for method names.'
67
+ MSG_FORBIDDEN = '`%<identifier>s` is forbidden, use another method name instead.'
37
68
 
38
69
  # @!method sym_name(node)
39
70
  def_node_matcher :sym_name, '(sym $_name)'
@@ -48,19 +79,44 @@ module RuboCop
48
79
  name = attr_name(name_item)
49
80
  next if !name || matches_allowed_pattern?(name)
50
81
 
51
- check_name(node, name, range_position(node))
82
+ if forbidden_name?(name.to_s)
83
+ register_forbidden_name(node)
84
+ else
85
+ check_name(node, name, range_position(node))
86
+ end
52
87
  end
53
88
  end
54
89
 
55
90
  def on_def(node)
56
91
  return if node.operator_method? || matches_allowed_pattern?(node.method_name)
57
92
 
58
- check_name(node, node.method_name, node.loc.name)
93
+ if forbidden_name?(node.method_name.to_s)
94
+ register_forbidden_name(node)
95
+ else
96
+ check_name(node, node.method_name, node.loc.name)
97
+ end
59
98
  end
60
99
  alias on_defs on_def
61
100
 
62
101
  private
63
102
 
103
+ def forbidden_name?(name)
104
+ forbidden_identifier?(name) || forbidden_pattern?(name)
105
+ end
106
+
107
+ def register_forbidden_name(node)
108
+ if node.type?(:def, :defs)
109
+ name_node = node.loc.name
110
+ method_name = node.method_name
111
+ else
112
+ attrs = node.attribute_accessor?
113
+ name_node = attrs.last.last
114
+ method_name = attr_name(name_node)
115
+ end
116
+ message = format(MSG_FORBIDDEN, identifier: method_name)
117
+ add_offense(name_node, message: message)
118
+ end
119
+
64
120
  def attr_name(name_item)
65
121
  sym_name(name_item) || str_name(name_item)
66
122
  end
@@ -53,6 +53,8 @@ module RuboCop
53
53
  include AllowedIdentifiers
54
54
  include ConfigurableNaming
55
55
  include AllowedPattern
56
+ include ForbiddenIdentifiers
57
+ include ForbiddenPattern
56
58
 
57
59
  MSG = 'Use %<style>s for variable names.'
58
60
  MSG_FORBIDDEN = '`%<identifier>s` is forbidden, use another name instead.'
@@ -92,27 +94,12 @@ module RuboCop
92
94
 
93
95
  private
94
96
 
95
- def message(style)
96
- format(MSG, style: style)
97
- end
98
-
99
- def forbidden_identifiers
100
- cop_config.fetch('ForbiddenIdentifiers', [])
101
- end
102
-
103
- def forbidden_patterns
104
- cop_config.fetch('ForbiddenPatterns', [])
105
- end
106
-
107
- def matches_forbidden_pattern?(name)
108
- forbidden_patterns.any? { |pattern| Regexp.new(pattern).match?(name) }
109
- end
110
-
111
97
  def forbidden_name?(name)
112
- name = name.to_s.delete(SIGILS)
98
+ forbidden_identifier?(name) || forbidden_pattern?(name)
99
+ end
113
100
 
114
- (forbidden_identifiers.any? && forbidden_identifiers.include?(name)) ||
115
- (forbidden_patterns.any? && matches_forbidden_pattern?(name))
101
+ def message(style)
102
+ format(MSG, style: style)
116
103
  end
117
104
 
118
105
  def register_forbidden_name(node)
@@ -23,8 +23,8 @@ module RuboCop
23
23
  global.without_department(:Test).cops
24
24
  end
25
25
 
26
- def self.qualified_cop_name(name, origin)
27
- global.qualified_cop_name(name, origin)
26
+ def self.qualified_cop_name(name, origin, warn: true)
27
+ global.qualified_cop_name(name, origin, warn: warn)
28
28
  end
29
29
 
30
30
  # Changes momentarily the global registry
@@ -139,7 +139,7 @@ module RuboCop
139
139
 
140
140
  case potential_badges.size
141
141
  when 0 then name # No namespace found. Deal with it later in caller.
142
- when 1 then resolve_badge(badge, potential_badges.first, path)
142
+ when 1 then resolve_badge(badge, potential_badges.first, path, warn: warn)
143
143
  else raise AmbiguousCopName.new(badge, path, potential_badges)
144
144
  end
145
145
  end
@@ -296,11 +296,14 @@ module RuboCop
296
296
  self.class.new(cops)
297
297
  end
298
298
 
299
- def resolve_badge(given_badge, real_badge, source_path)
299
+ def resolve_badge(given_badge, real_badge, source_path, warn: true)
300
300
  unless given_badge.match?(real_badge)
301
301
  path = PathUtil.smart_path(source_path)
302
- warn "#{path}: #{given_badge} has the wrong namespace - " \
303
- "replace it with #{given_badge.with_department(real_badge.department)}"
302
+
303
+ if warn
304
+ warn("#{path}: #{given_badge} has the wrong namespace - " \
305
+ "replace it with #{given_badge.with_department(real_badge.department)}")
306
+ end
304
307
  end
305
308
 
306
309
  real_badge.to_s
@@ -6,7 +6,8 @@ module RuboCop
6
6
  # In Ruby 3.1, `Array#intersect?` has been added.
7
7
  #
8
8
  # This cop identifies places where `(array1 & array2).any?`
9
- # can be replaced by `array1.intersect?(array2)`.
9
+ # or `(array1.intersection(array2)).any?` can be replaced by
10
+ # `array1.intersect?(array2)`.
10
11
  #
11
12
  # The `array1.intersect?(array2)` method is faster than
12
13
  # `(array1 & array2).any?` and is more readable.
@@ -20,6 +21,10 @@ module RuboCop
20
21
  # [1].intersect?([1,2]) { |x| false } # => true
21
22
  # ----
22
23
  #
24
+ # NOTE: Although `Array#intersection` can take zero or multiple arguments,
25
+ # only cases where exactly one argument is provided can be replaced with
26
+ # `Array#intersect?` and are handled by this cop.
27
+ #
23
28
  # @safety
24
29
  # This cop cannot guarantee that `array1` and `array2` are
25
30
  # actually arrays while method `intersect?` is for arrays only.
@@ -30,6 +35,11 @@ module RuboCop
30
35
  # (array1 & array2).empty?
31
36
  # (array1 & array2).none?
32
37
  #
38
+ # # bad
39
+ # array1.intersection(array2).any?
40
+ # array1.intersection(array2).empty?
41
+ # array1.intersection(array2).none?
42
+ #
33
43
  # # good
34
44
  # array1.intersect?(array2)
35
45
  # !array1.intersect?(array2)
@@ -53,65 +63,66 @@ module RuboCop
53
63
 
54
64
  minimum_target_ruby_version 3.1
55
65
 
56
- # @!method regular_bad_intersection_check?(node)
57
- def_node_matcher :regular_bad_intersection_check?, <<~PATTERN
58
- (send
59
- (begin
60
- (send $(...) :& $(...))
61
- ) ${:any? :empty? :none?}
62
- )
63
- PATTERN
66
+ PREDICATES = %i[any? empty? none?].to_set.freeze
67
+ ACTIVE_SUPPORT_PREDICATES = (PREDICATES + %i[present? blank?]).freeze
64
68
 
65
- # @!method active_support_bad_intersection_check?(node)
66
- def_node_matcher :active_support_bad_intersection_check?, <<~PATTERN
67
- (send
68
- (begin
69
- (send $(...) :& $(...))
70
- ) ${:present? :any? :blank? :empty? :none?}
69
+ # @!method bad_intersection_check?(node, predicates)
70
+ def_node_matcher :bad_intersection_check?, <<~PATTERN
71
+ (call
72
+ {
73
+ (begin (send $_ :& $_))
74
+ (call $_ :intersection $_)
75
+ }
76
+ $%1
71
77
  )
72
78
  PATTERN
73
79
 
74
- MSG = 'Use `%<negated>s%<receiver>s.intersect?(%<argument>s)` ' \
75
- 'instead of `(%<receiver>s & %<argument>s).%<method_name>s`.'
80
+ MSG = 'Use `%<negated>s%<receiver>s%<dot>sintersect?(%<argument>s)` ' \
81
+ 'instead of `%<existing>s`.'
76
82
  STRAIGHT_METHODS = %i[present? any?].freeze
77
83
  NEGATED_METHODS = %i[blank? empty? none?].freeze
78
84
  RESTRICT_ON_SEND = (STRAIGHT_METHODS + NEGATED_METHODS).freeze
79
85
 
80
86
  def on_send(node)
81
87
  return if node.block_literal?
82
- return unless (receiver, argument, method_name = bad_intersection_check?(node))
88
+ return unless (receiver, argument, method_name = bad_intersection?(node))
83
89
 
84
- message = message(receiver.source, argument.source, method_name)
90
+ dot = node.loc.dot.source
91
+ message = message(receiver.source, argument.source, method_name, dot, node.source)
85
92
 
86
93
  add_offense(node, message: message) do |corrector|
87
94
  bang = straight?(method_name) ? '' : '!'
88
95
 
89
- corrector.replace(node, "#{bang}#{receiver.source}.intersect?(#{argument.source})")
96
+ corrector.replace(node, "#{bang}#{receiver.source}#{dot}intersect?(#{argument.source})")
90
97
  end
91
98
  end
99
+ alias on_csend on_send
92
100
 
93
101
  private
94
102
 
95
- def bad_intersection_check?(node)
96
- if active_support_extensions_enabled?
97
- active_support_bad_intersection_check?(node)
98
- else
99
- regular_bad_intersection_check?(node)
100
- end
103
+ def bad_intersection?(node)
104
+ predicates = if active_support_extensions_enabled?
105
+ ACTIVE_SUPPORT_PREDICATES
106
+ else
107
+ PREDICATES
108
+ end
109
+
110
+ bad_intersection_check?(node, predicates)
101
111
  end
102
112
 
103
113
  def straight?(method_name)
104
114
  STRAIGHT_METHODS.include?(method_name.to_sym)
105
115
  end
106
116
 
107
- def message(receiver, argument, method_name)
117
+ def message(receiver, argument, method_name, dot, existing)
108
118
  negated = straight?(method_name) ? '' : '!'
109
119
  format(
110
120
  MSG,
111
121
  negated: negated,
112
122
  receiver: receiver,
113
123
  argument: argument,
114
- method_name: method_name
124
+ dot: dot,
125
+ existing: existing
115
126
  )
116
127
  end
117
128
  end
@@ -208,6 +208,7 @@ module RuboCop
208
208
  end
209
209
 
210
210
  alias on_numblock on_block
211
+ alias on_itblock on_block
211
212
 
212
213
  private
213
214
 
@@ -347,7 +348,7 @@ module RuboCop
347
348
  # rubocop:disable Metrics/CyclomaticComplexity
348
349
  def get_blocks(node, &block)
349
350
  case node.type
350
- when :block, :numblock
351
+ when :block, :numblock, :itblock
351
352
  yield node
352
353
  when :send, :csend
353
354
  # When a method has an argument which is another method with a block,
@@ -3,14 +3,26 @@
3
3
  module RuboCop
4
4
  module Cop
5
5
  module Style
6
- # Checks the style of children definitions at classes and
7
- # modules. Basically there are two different styles:
6
+ # Checks that namespaced classes and modules are defined with a consistent style.
7
+ #
8
+ # With `nested` style, classes and modules should be defined separately (one constant
9
+ # on each line, without `::`). With `compact` style, classes and modules should be
10
+ # defined with fully qualified names (using `::` for namespaces).
11
+ #
12
+ # NOTE: The style chosen will affect `Module.nesting` for the class or module. Using
13
+ # `nested` style will result in each level being added, whereas `compact` style will
14
+ # only include the fully qualified class or module name.
15
+ #
16
+ # By default, `EnforcedStyle` applies to both classes and modules. If desired, separate
17
+ # styles can be defined for classes and modules by using `EnforcedStyleForClasses` and
18
+ # `EnforcedStyleForModules` respectively. If not set, or set to nil, the `EnforcedStyle`
19
+ # value will be used.
8
20
  #
9
21
  # @safety
10
22
  # Autocorrection is unsafe.
11
23
  #
12
- # Moving from compact to nested children requires knowledge of whether the
13
- # outer parent is a module or a class. Moving from nested to compact requires
24
+ # Moving from `compact` to `nested` children requires knowledge of whether the
25
+ # outer parent is a module or a class. Moving from `nested` to `compact` requires
14
26
  # verification that the outer parent is defined elsewhere. RuboCop does not
15
27
  # have the knowledge to perform either operation safely and thus requires
16
28
  # manual oversight.
@@ -42,16 +54,18 @@ module RuboCop
42
54
  def on_class(node)
43
55
  return if node.parent_class && style != :nested
44
56
 
45
- check_style(node, node.body)
57
+ check_style(node, node.body, style_for_classes)
46
58
  end
47
59
 
48
60
  def on_module(node)
49
- check_style(node, node.body)
61
+ check_style(node, node.body, style_for_modules)
50
62
  end
51
63
 
52
64
  private
53
65
 
54
66
  def nest_or_compact(corrector, node)
67
+ style = node.class_type? ? style_for_classes : style_for_modules
68
+
55
69
  if style == :nested
56
70
  nest_definition(corrector, node)
57
71
  else
@@ -141,7 +155,7 @@ module RuboCop
141
155
  node.source_range.source_line[/\A\s*/]
142
156
  end
143
157
 
144
- def check_style(node, body)
158
+ def check_style(node, body, style)
145
159
  return if node.identifier.namespace&.cbase_type?
146
160
 
147
161
  if style == :nested
@@ -183,6 +197,14 @@ module RuboCop
183
197
  def compact_node_name?(node)
184
198
  node.identifier.source.include?('::')
185
199
  end
200
+
201
+ def style_for_classes
202
+ cop_config['EnforcedStyleForClasses'] || style
203
+ end
204
+
205
+ def style_for_modules
206
+ cop_config['EnforcedStyleForModules'] || style
207
+ end
186
208
  end
187
209
  end
188
210
  end
@@ -50,6 +50,7 @@ module RuboCop
50
50
  check_method_node(node.send_node)
51
51
  end
52
52
  alias on_numblock on_block
53
+ alias on_itblock on_block
53
54
 
54
55
  def on_send(node)
55
56
  return unless implicit_block?(node)
@@ -80,6 +80,7 @@ module RuboCop
80
80
  # rubocop:enable Metrics/CyclomaticComplexity
81
81
 
82
82
  alias on_numblock on_block
83
+ alias on_itblock on_block
83
84
 
84
85
  def on_for(node)
85
86
  return unless node.parent&.begin_type?
@@ -9,8 +9,8 @@ module RuboCop
9
9
  # These keywords are: `class`, `module`, `def`, `begin`, `end`.
10
10
  #
11
11
  # Note that some comments
12
- # (`:nodoc:`, `:yields:`, `rubocop:disable` and `rubocop:todo`)
13
- # and RBS::Inline annotation comments are allowed.
12
+ # (`:nodoc:`, `:yields:`, `rubocop:disable` and `rubocop:todo`),
13
+ # RBS::Inline annotation, and Steep annotation (`steep:ignore`) are allowed.
14
14
  #
15
15
  # Autocorrection removes comments from `end` keyword and keeps comments
16
16
  # for `class`, `module`, `def` and `begin` above the keyword.
@@ -60,6 +60,8 @@ module RuboCop
60
60
  SUBCLASS_DEFINITION = /\A\s*class\s+(\w|::)+\s*<\s*(\w|::)+/.freeze
61
61
  METHOD_DEFINITION = /\A\s*def\s/.freeze
62
62
 
63
+ STEEP_REGEXP = /#\ssteep:ignore(\s|\z)/.freeze
64
+
63
65
  def on_new_investigation
64
66
  processed_source.comments.each do |comment|
65
67
  next unless offensive?(comment) && (match = source_line(comment).match(REGEXP))
@@ -86,6 +88,7 @@ module RuboCop
86
88
  def offensive?(comment)
87
89
  line = source_line(comment)
88
90
  return false if rbs_inline_annotation?(line, comment)
91
+ return false if steep_annotation?(comment)
89
92
 
90
93
  KEYWORD_REGEXES.any? { |r| r.match?(line) } &&
91
94
  ALLOWED_COMMENT_REGEXES.none? { |r| r.match?(line) }
@@ -105,6 +108,10 @@ module RuboCop
105
108
  false
106
109
  end
107
110
  end
111
+
112
+ def steep_annotation?(comment)
113
+ comment.text.match?(STEEP_REGEXP)
114
+ end
108
115
  end
109
116
  end
110
117
  end