rubocop 1.84.2 → 1.88.0
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.
- checksums.yaml +4 -4
- data/config/default.yml +180 -86
- data/config/obsoletion.yml +26 -1
- data/lib/rubocop/cache_config.rb +1 -1
- data/lib/rubocop/cli/command/auto_generate_config.rb +34 -2
- data/lib/rubocop/cli/command/list_enabled_cops_for.rb +40 -0
- data/lib/rubocop/cli/command/mcp.rb +19 -0
- data/lib/rubocop/cli/command/show_cops.rb +2 -2
- data/lib/rubocop/cli/command/show_docs_url.rb +4 -8
- data/lib/rubocop/cli/command/suggest_extensions.rb +1 -1
- data/lib/rubocop/cli.rb +9 -7
- data/lib/rubocop/comment_config.rb +12 -15
- data/lib/rubocop/config.rb +14 -10
- data/lib/rubocop/config_finder.rb +1 -1
- data/lib/rubocop/config_loader.rb +17 -2
- data/lib/rubocop/config_loader_resolver.rb +13 -4
- data/lib/rubocop/config_obsoletion/extracted_cop.rb +4 -2
- data/lib/rubocop/config_store.rb +2 -2
- data/lib/rubocop/config_validator.rb +1 -1
- data/lib/rubocop/cop/autocorrect_logic.rb +2 -1
- data/lib/rubocop/cop/base.rb +25 -4
- data/lib/rubocop/cop/bundler/gem_comment.rb +2 -2
- data/lib/rubocop/cop/correctors/condition_corrector.rb +1 -1
- data/lib/rubocop/cop/correctors/multiline_literal_brace_corrector.rb +1 -5
- data/lib/rubocop/cop/correctors/parentheses_corrector.rb +33 -2
- data/lib/rubocop/cop/correctors/percent_literal_corrector.rb +2 -2
- data/lib/rubocop/cop/correctors.rb +28 -0
- data/lib/rubocop/cop/documentation.rb +2 -3
- data/lib/rubocop/cop/exclude_limit.rb +31 -5
- data/lib/rubocop/cop/gemspec/duplicated_assignment.rb +2 -2
- data/lib/rubocop/cop/gemspec/require_mfa.rb +5 -5
- data/lib/rubocop/cop/gemspec/ruby_version_globals_usage.rb +3 -3
- data/lib/rubocop/cop/internal_affairs/itblock_handler.rb +69 -0
- data/lib/rubocop/cop/internal_affairs/location_line_equality_comparison.rb +1 -0
- data/lib/rubocop/cop/internal_affairs/redundant_let_rubocop_config_new.rb +5 -3
- data/lib/rubocop/cop/internal_affairs.rb +1 -0
- data/lib/rubocop/cop/layout/argument_alignment.rb +2 -2
- data/lib/rubocop/cop/layout/array_alignment.rb +1 -1
- data/lib/rubocop/cop/layout/begin_end_alignment.rb +1 -1
- data/lib/rubocop/cop/layout/block_alignment.rb +41 -4
- data/lib/rubocop/cop/layout/class_structure.rb +1 -1
- data/lib/rubocop/cop/layout/dot_position.rb +1 -1
- data/lib/rubocop/cop/layout/empty_line_after_guard_clause.rb +23 -7
- data/lib/rubocop/cop/layout/empty_line_between_defs.rb +1 -1
- data/lib/rubocop/cop/layout/empty_lines_around_attribute_accessor.rb +1 -0
- data/lib/rubocop/cop/layout/empty_lines_around_block_body.rb +12 -2
- data/lib/rubocop/cop/layout/empty_lines_around_class_body.rb +16 -2
- data/lib/rubocop/cop/layout/empty_lines_around_module_body.rb +16 -2
- data/lib/rubocop/cop/layout/end_alignment.rb +8 -5
- data/lib/rubocop/cop/layout/first_hash_element_indentation.rb +7 -1
- data/lib/rubocop/cop/layout/hash_alignment.rb +1 -1
- data/lib/rubocop/cop/layout/indentation_width.rb +12 -0
- data/lib/rubocop/cop/layout/line_length.rb +5 -3
- data/lib/rubocop/cop/layout/multiline_assignment_layout.rb +9 -2
- data/lib/rubocop/cop/layout/multiline_method_call_brace_layout.rb +1 -1
- data/lib/rubocop/cop/layout/multiline_method_call_indentation.rb +53 -3
- data/lib/rubocop/cop/layout/parameter_alignment.rb +1 -1
- data/lib/rubocop/cop/layout/redundant_line_break.rb +3 -1
- data/lib/rubocop/cop/layout/space_around_block_parameters.rb +1 -1
- data/lib/rubocop/cop/layout/space_around_keyword.rb +3 -1
- data/lib/rubocop/cop/layout/space_before_brackets.rb +1 -1
- data/lib/rubocop/cop/layout/space_in_lambda_literal.rb +1 -0
- data/lib/rubocop/cop/lint/ambiguous_assignment.rb +1 -11
- data/lib/rubocop/cop/lint/ambiguous_block_association.rb +1 -1
- data/lib/rubocop/cop/lint/ambiguous_operator_precedence.rb +1 -10
- data/lib/rubocop/cop/lint/circular_argument_reference.rb +1 -3
- data/lib/rubocop/cop/lint/constant_overwritten_in_rescue.rb +1 -1
- data/lib/rubocop/cop/lint/constant_reassignment.rb +93 -11
- data/lib/rubocop/cop/lint/constant_resolution.rb +6 -6
- data/lib/rubocop/cop/lint/data_define_override.rb +63 -0
- data/lib/rubocop/cop/lint/debugger.rb +0 -1
- data/lib/rubocop/cop/lint/deprecated_constants.rb +2 -8
- data/lib/rubocop/cop/lint/duplicate_methods.rb +55 -8
- data/lib/rubocop/cop/lint/empty_block.rb +4 -4
- data/lib/rubocop/cop/lint/empty_conditional_body.rb +6 -1
- data/lib/rubocop/cop/lint/empty_in_pattern.rb +8 -1
- data/lib/rubocop/cop/lint/empty_when.rb +8 -1
- data/lib/rubocop/cop/lint/ensure_return.rb +19 -1
- data/lib/rubocop/cop/lint/erb_new_arguments.rb +4 -2
- data/lib/rubocop/cop/lint/float_comparison.rb +1 -0
- data/lib/rubocop/cop/lint/incompatible_io_select_with_fiber_scheduler.rb +5 -1
- data/lib/rubocop/cop/lint/interpolation_check.rb +25 -5
- data/lib/rubocop/cop/lint/lambda_without_literal_block.rb +1 -1
- data/lib/rubocop/cop/lint/literal_assignment_in_condition.rb +11 -1
- data/lib/rubocop/cop/lint/literal_in_interpolation.rb +8 -11
- data/lib/rubocop/cop/lint/missing_cop_enable_directive.rb +5 -5
- data/lib/rubocop/cop/lint/multiple_comparison.rb +2 -2
- data/lib/rubocop/cop/lint/next_without_accumulator.rb +2 -0
- data/lib/rubocop/cop/lint/no_return_in_begin_end_blocks.rb +16 -0
- data/lib/rubocop/cop/lint/non_deterministic_require_order.rb +4 -2
- data/lib/rubocop/cop/lint/non_local_exit_from_iterator.rb +1 -1
- data/lib/rubocop/cop/lint/number_conversion.rb +19 -10
- data/lib/rubocop/cop/lint/numbered_parameter_assignment.rb +1 -1
- data/lib/rubocop/cop/lint/numeric_operation_with_constant_result.rb +3 -0
- data/lib/rubocop/cop/lint/ordered_magic_comments.rb +7 -7
- data/lib/rubocop/cop/lint/parentheses_as_grouped_expression.rb +3 -13
- data/lib/rubocop/cop/lint/raise_exception.rb +1 -1
- data/lib/rubocop/cop/lint/rand_one.rb +1 -1
- data/lib/rubocop/cop/lint/redundant_cop_disable_directive.rb +4 -1
- data/lib/rubocop/cop/lint/redundant_cop_enable_directive.rb +6 -12
- data/lib/rubocop/cop/lint/redundant_dir_glob_sort.rb +15 -4
- data/lib/rubocop/cop/lint/redundant_safe_navigation.rb +36 -12
- data/lib/rubocop/cop/lint/redundant_splat_expansion.rb +4 -0
- data/lib/rubocop/cop/lint/redundant_type_conversion.rb +10 -3
- data/lib/rubocop/cop/lint/redundant_with_index.rb +1 -1
- data/lib/rubocop/cop/lint/redundant_with_object.rb +5 -0
- data/lib/rubocop/cop/lint/refinement_import_methods.rb +8 -1
- data/lib/rubocop/cop/lint/regexp_as_condition.rb +9 -1
- data/lib/rubocop/cop/lint/require_parentheses.rb +13 -4
- data/lib/rubocop/cop/lint/require_range_parentheses.rb +2 -1
- data/lib/rubocop/cop/lint/require_relative_self_path.rb +7 -5
- data/lib/rubocop/cop/lint/rescue_type.rb +1 -1
- data/lib/rubocop/cop/lint/safe_navigation_chain.rb +18 -0
- data/lib/rubocop/cop/lint/safe_navigation_consistency.rb +7 -1
- data/lib/rubocop/cop/lint/safe_navigation_with_empty.rb +1 -1
- data/lib/rubocop/cop/lint/script_permission.rb +5 -1
- data/lib/rubocop/cop/lint/self_assignment.rb +24 -1
- data/lib/rubocop/cop/lint/send_with_mixin_argument.rb +1 -1
- data/lib/rubocop/cop/lint/shadowed_exception.rb +1 -1
- data/lib/rubocop/cop/lint/shadowing_outer_local_variable.rb +14 -0
- data/lib/rubocop/cop/lint/shared_mutable_default.rb +3 -1
- data/lib/rubocop/cop/lint/suppressed_exception_in_number_conversion.rb +12 -0
- data/lib/rubocop/cop/lint/symbol_conversion.rb +21 -4
- data/lib/rubocop/cop/lint/syntax.rb +25 -1
- data/lib/rubocop/cop/lint/to_enum_arguments.rb +28 -1
- data/lib/rubocop/cop/lint/top_level_return_with_argument.rb +1 -1
- data/lib/rubocop/cop/lint/trailing_comma_in_attribute_declaration.rb +5 -1
- data/lib/rubocop/cop/lint/unescaped_bracket_in_regexp.rb +4 -2
- data/lib/rubocop/cop/lint/unmodified_reduce_accumulator.rb +1 -0
- data/lib/rubocop/cop/lint/unreachable_code.rb +2 -2
- data/lib/rubocop/cop/lint/unreachable_pattern_branch.rb +113 -0
- data/lib/rubocop/cop/lint/unused_method_argument.rb +10 -0
- data/lib/rubocop/cop/lint/useless_assignment.rb +14 -14
- data/lib/rubocop/cop/lint/useless_constant_scoping.rb +4 -4
- data/lib/rubocop/cop/lint/useless_default_value_argument.rb +2 -0
- data/lib/rubocop/cop/lint/useless_ruby2_keywords.rb +8 -4
- data/lib/rubocop/cop/lint/useless_setter_call.rb +4 -1
- data/lib/rubocop/cop/lint/useless_times.rb +22 -1
- data/lib/rubocop/cop/lint/utils/nil_receiver_checker.rb +35 -9
- data/lib/rubocop/cop/lint/void.rb +32 -12
- data/lib/rubocop/cop/metrics/block_length.rb +1 -1
- data/lib/rubocop/cop/metrics/block_nesting.rb +23 -0
- data/lib/rubocop/cop/metrics/collection_literal_length.rb +1 -1
- data/lib/rubocop/cop/metrics/method_length.rb +1 -1
- data/lib/rubocop/cop/metrics/utils/iterating_block.rb +1 -1
- data/lib/rubocop/cop/migration/department_name.rb +12 -1
- data/lib/rubocop/cop/mixin/check_line_breakable.rb +1 -1
- data/lib/rubocop/cop/mixin/check_single_line_suitability.rb +2 -2
- data/lib/rubocop/cop/mixin/configurable_max.rb +6 -5
- data/lib/rubocop/cop/mixin/hash_transform_method/autocorrection.rb +63 -0
- data/lib/rubocop/cop/mixin/hash_transform_method.rb +10 -60
- data/lib/rubocop/cop/mixin/project_index_help.rb +48 -0
- data/lib/rubocop/cop/mixin.rb +86 -0
- data/lib/rubocop/cop/naming/binary_operator_parameter_name.rb +1 -1
- data/lib/rubocop/cop/naming/block_parameter_name.rb +1 -1
- data/lib/rubocop/cop/naming/memoized_instance_variable_name.rb +1 -1
- data/lib/rubocop/cop/naming/predicate_method.rb +2 -2
- data/lib/rubocop/cop/naming/predicate_prefix.rb +1 -1
- data/lib/rubocop/cop/naming/rescued_exceptions_variable_name.rb +1 -1
- data/lib/rubocop/cop/offense.rb +8 -0
- data/lib/rubocop/cop/registry.rb +62 -38
- data/lib/rubocop/cop/security/eval.rb +15 -2
- data/lib/rubocop/cop/security/io_methods.rb +1 -1
- data/lib/rubocop/cop/style/access_modifier_declarations.rb +14 -2
- data/lib/rubocop/cop/style/accessor_grouping.rb +4 -2
- data/lib/rubocop/cop/style/alias.rb +15 -3
- data/lib/rubocop/cop/style/and_or.rb +2 -1
- data/lib/rubocop/cop/style/arguments_forwarding.rb +25 -7
- data/lib/rubocop/cop/style/array_first_last.rb +12 -1
- data/lib/rubocop/cop/style/array_intersect.rb +4 -0
- data/lib/rubocop/cop/style/array_intersect_with_single_element.rb +3 -0
- data/lib/rubocop/cop/style/array_join.rb +4 -2
- data/lib/rubocop/cop/style/ascii_comments.rb +6 -3
- data/lib/rubocop/cop/style/attr.rb +5 -2
- data/lib/rubocop/cop/style/bare_percent_literals.rb +3 -1
- data/lib/rubocop/cop/style/begin_block.rb +3 -1
- data/lib/rubocop/cop/style/block_delimiters.rb +37 -31
- data/lib/rubocop/cop/style/case_equality.rb +18 -2
- data/lib/rubocop/cop/style/character_literal.rb +2 -2
- data/lib/rubocop/cop/style/class_and_module_children.rb +18 -2
- data/lib/rubocop/cop/style/class_equality_comparison.rb +21 -13
- data/lib/rubocop/cop/style/class_methods_definitions.rb +11 -5
- data/lib/rubocop/cop/style/collection_compact.rb +36 -16
- data/lib/rubocop/cop/style/colon_method_call.rb +16 -7
- data/lib/rubocop/cop/style/combinable_loops.rb +5 -0
- data/lib/rubocop/cop/style/comparable_clamp.rb +12 -1
- data/lib/rubocop/cop/style/concat_array_literals.rb +7 -1
- data/lib/rubocop/cop/style/conditional_assignment.rb +6 -5
- data/lib/rubocop/cop/style/constant_visibility.rb +4 -1
- data/lib/rubocop/cop/style/copyright.rb +22 -11
- data/lib/rubocop/cop/style/date_time.rb +4 -4
- data/lib/rubocop/cop/style/dig_chain.rb +5 -0
- data/lib/rubocop/cop/style/disable_cops_within_source_code_directive.rb +1 -1
- data/lib/rubocop/cop/style/document_dynamic_eval_definition.rb +6 -1
- data/lib/rubocop/cop/style/each_for_simple_loop.rb +1 -1
- data/lib/rubocop/cop/style/each_with_object.rb +2 -0
- data/lib/rubocop/cop/style/empty_block_parameter.rb +1 -1
- data/lib/rubocop/cop/style/empty_class_definition.rb +43 -20
- data/lib/rubocop/cop/style/empty_lambda_parameter.rb +1 -1
- data/lib/rubocop/cop/style/encoding.rb +7 -1
- data/lib/rubocop/cop/style/end_block.rb +3 -1
- data/lib/rubocop/cop/style/endless_method.rb +8 -3
- data/lib/rubocop/cop/style/fetch_env_var.rb +1 -1
- data/lib/rubocop/cop/style/file_open.rb +84 -0
- data/lib/rubocop/cop/style/file_write.rb +21 -16
- data/lib/rubocop/cop/style/for.rb +3 -0
- data/lib/rubocop/cop/style/format_string.rb +4 -3
- data/lib/rubocop/cop/style/format_string_token.rb +29 -2
- data/lib/rubocop/cop/style/global_vars.rb +5 -2
- data/lib/rubocop/cop/style/guard_clause.rb +9 -6
- data/lib/rubocop/cop/style/hash_as_last_array_item.rb +21 -5
- data/lib/rubocop/cop/style/hash_conversion.rb +1 -1
- data/lib/rubocop/cop/style/hash_lookup_method.rb +19 -7
- data/lib/rubocop/cop/style/hash_slice.rb +16 -0
- data/lib/rubocop/cop/style/hash_transform_keys.rb +17 -7
- data/lib/rubocop/cop/style/hash_transform_values.rb +17 -7
- data/lib/rubocop/cop/style/if_inside_else.rb +16 -7
- data/lib/rubocop/cop/style/if_unless_modifier.rb +15 -4
- data/lib/rubocop/cop/style/if_with_semicolon.rb +7 -5
- data/lib/rubocop/cop/style/inline_comment.rb +4 -1
- data/lib/rubocop/cop/style/ip_addresses.rb +1 -2
- data/lib/rubocop/cop/style/magic_comment_format.rb +3 -3
- data/lib/rubocop/cop/style/map_join.rb +123 -0
- data/lib/rubocop/cop/style/method_call_with_args_parentheses/require_parentheses.rb +5 -3
- data/lib/rubocop/cop/style/min_max_comparison.rb +1 -1
- data/lib/rubocop/cop/style/module_member_existence_check.rb +7 -14
- data/lib/rubocop/cop/style/multiline_if_then.rb +3 -1
- data/lib/rubocop/cop/style/mutable_constant.rb +106 -12
- data/lib/rubocop/cop/style/nil_comparison.rb +2 -3
- data/lib/rubocop/cop/style/nil_lambda.rb +1 -1
- data/lib/rubocop/cop/style/non_nil_check.rb +5 -11
- data/lib/rubocop/cop/style/not.rb +2 -0
- data/lib/rubocop/cop/style/numeric_literals.rb +3 -2
- data/lib/rubocop/cop/style/one_class_per_file.rb +115 -0
- data/lib/rubocop/cop/style/one_line_conditional.rb +4 -3
- data/lib/rubocop/cop/style/parallel_assignment.rb +12 -1
- data/lib/rubocop/cop/style/partition_instead_of_double_select.rb +270 -0
- data/lib/rubocop/cop/style/percent_literal_delimiters.rb +2 -0
- data/lib/rubocop/cop/style/predicate_with_kind.rb +84 -0
- data/lib/rubocop/cop/style/proc.rb +3 -2
- data/lib/rubocop/cop/style/raise_args.rb +1 -1
- data/lib/rubocop/cop/style/reduce_to_hash.rb +200 -0
- data/lib/rubocop/cop/style/redundant_array_constructor.rb +2 -2
- data/lib/rubocop/cop/style/redundant_begin.rb +3 -3
- data/lib/rubocop/cop/style/redundant_constant_base.rb +5 -5
- data/lib/rubocop/cop/style/redundant_each.rb +3 -3
- data/lib/rubocop/cop/style/redundant_fetch_block.rb +1 -1
- data/lib/rubocop/cop/style/redundant_format.rb +1 -0
- data/lib/rubocop/cop/style/redundant_interpolation_unfreeze.rb +26 -10
- data/lib/rubocop/cop/style/redundant_line_continuation.rb +16 -0
- data/lib/rubocop/cop/style/redundant_min_max_by.rb +93 -0
- data/lib/rubocop/cop/style/redundant_parentheses.rb +25 -22
- data/lib/rubocop/cop/style/redundant_percent_q.rb +4 -1
- data/lib/rubocop/cop/style/redundant_regexp_constructor.rb +2 -2
- data/lib/rubocop/cop/style/redundant_return.rb +3 -1
- data/lib/rubocop/cop/style/redundant_self.rb +2 -2
- data/lib/rubocop/cop/style/redundant_self_assignment_branch.rb +0 -5
- data/lib/rubocop/cop/style/redundant_struct_keyword_init.rb +114 -0
- data/lib/rubocop/cop/style/regexp_literal.rb +31 -2
- data/lib/rubocop/cop/style/rescue_modifier.rb +3 -3
- data/lib/rubocop/cop/style/safe_navigation.rb +7 -7
- data/lib/rubocop/cop/style/select_by_kind.rb +158 -0
- data/lib/rubocop/cop/style/select_by_range.rb +197 -0
- data/lib/rubocop/cop/style/select_by_regexp.rb +51 -21
- data/lib/rubocop/cop/style/self_assignment.rb +1 -1
- data/lib/rubocop/cop/style/semicolon.rb +18 -1
- data/lib/rubocop/cop/style/single_line_block_params.rb +2 -2
- data/lib/rubocop/cop/style/single_line_do_end_block.rb +1 -1
- data/lib/rubocop/cop/style/single_line_methods.rb +3 -1
- data/lib/rubocop/cop/style/sole_nested_conditional.rb +4 -2
- data/lib/rubocop/cop/style/special_global_vars.rb +6 -1
- data/lib/rubocop/cop/style/struct_inheritance.rb +13 -0
- data/lib/rubocop/cop/style/symbol_proc.rb +7 -6
- data/lib/rubocop/cop/style/tally_method.rb +181 -0
- data/lib/rubocop/cop/style/top_level_method_definition.rb +2 -2
- data/lib/rubocop/cop/style/trailing_comma_in_block_args.rb +1 -1
- data/lib/rubocop/cop/style/trailing_method_end_statement.rb +1 -0
- data/lib/rubocop/cop/style/unless_logical_operators.rb +3 -3
- data/lib/rubocop/cop/style/while_until_do.rb +7 -0
- data/lib/rubocop/cop/style/while_until_modifier.rb +16 -0
- data/lib/rubocop/cop/style/word_array.rb +1 -0
- data/lib/rubocop/cop/style/yoda_condition.rb +1 -1
- data/lib/rubocop/cop/style/yoda_expression.rb +1 -1
- data/lib/rubocop/cop/style/zero_length_predicate.rb +6 -3
- data/lib/rubocop/cop/team.rb +86 -35
- data/lib/rubocop/cop/variable_force/branch.rb +2 -2
- data/lib/rubocop/directive_comment.rb +2 -1
- data/lib/rubocop/file_patterns.rb +9 -1
- data/lib/rubocop/formatter/disabled_config_formatter.rb +19 -9
- data/lib/rubocop/formatter/formatter_set.rb +1 -1
- data/lib/rubocop/formatter/junit_formatter.rb +1 -1
- data/lib/rubocop/formatter/simple_text_formatter.rb +0 -2
- data/lib/rubocop/formatter/worst_offenders_formatter.rb +1 -1
- data/lib/rubocop/formatter.rb +22 -21
- data/lib/rubocop/lsp/diagnostic.rb +1 -0
- data/lib/rubocop/lsp/routes.rb +10 -3
- data/lib/rubocop/lsp/runtime.rb +1 -2
- data/lib/rubocop/mcp/server.rb +200 -0
- data/lib/rubocop/options.rb +35 -4
- data/lib/rubocop/path_util.rb +14 -2
- data/lib/rubocop/plugin/loader.rb +1 -1
- data/lib/rubocop/project_index_loader.rb +66 -0
- data/lib/rubocop/result_cache.rb +22 -10
- data/lib/rubocop/rspec/cop_helper.rb +8 -0
- data/lib/rubocop/rspec/shared_contexts.rb +32 -2
- data/lib/rubocop/runner.rb +124 -53
- data/lib/rubocop/server/cache.rb +5 -7
- data/lib/rubocop/server/core.rb +8 -0
- data/lib/rubocop/target_finder.rb +14 -7
- data/lib/rubocop/target_ruby.rb +18 -12
- data/lib/rubocop/version.rb +21 -3
- data/lib/rubocop.rb +22 -96
- metadata +27 -5
|
@@ -3,9 +3,9 @@
|
|
|
3
3
|
module RuboCop
|
|
4
4
|
module Cop
|
|
5
5
|
module Lint
|
|
6
|
-
# Warns the usage of unsafe number conversions. Unsafe
|
|
7
|
-
# number conversion can cause unexpected error if auto type conversion
|
|
8
|
-
# fails.
|
|
6
|
+
# Warns against the usage of unsafe number conversions. Unsafe
|
|
7
|
+
# number conversion can cause an unexpected error if auto type conversion
|
|
8
|
+
# fails. The cop prefers parsing with a number class instead.
|
|
9
9
|
#
|
|
10
10
|
# Conversion with `Integer`, `Float`, etc. will raise an `ArgumentError`
|
|
11
11
|
# if given input that is not numeric (eg. an empty string), whereas
|
|
@@ -14,10 +14,10 @@ module RuboCop
|
|
|
14
14
|
# always correct to raise if a value is not numeric.
|
|
15
15
|
#
|
|
16
16
|
# NOTE: Some values cannot be converted properly using one of the `Kernel`
|
|
17
|
-
#
|
|
17
|
+
# methods (for instance, `Time` and `DateTime` values are allowed by this
|
|
18
18
|
# cop by default). Similarly, Rails' duration methods do not work well
|
|
19
19
|
# with `Integer()` and can be allowed with `AllowedMethods`. By default,
|
|
20
|
-
# there are no methods
|
|
20
|
+
# there are no allowed methods.
|
|
21
21
|
#
|
|
22
22
|
# @safety
|
|
23
23
|
# Autocorrection is unsafe because it is not guaranteed that the
|
|
@@ -66,7 +66,7 @@ module RuboCop
|
|
|
66
66
|
# # good
|
|
67
67
|
# 10.minutes.to_i
|
|
68
68
|
#
|
|
69
|
-
# @example
|
|
69
|
+
# @example AllowedClasses: [Time, DateTime] (default)
|
|
70
70
|
#
|
|
71
71
|
# # good
|
|
72
72
|
# Time.now.to_datetime.to_i
|
|
@@ -117,11 +117,11 @@ module RuboCop
|
|
|
117
117
|
|
|
118
118
|
message = format(
|
|
119
119
|
MSG,
|
|
120
|
-
current:
|
|
120
|
+
current: current_method(node, receiver, to_method),
|
|
121
121
|
corrected_method: correct_method(node, receiver)
|
|
122
122
|
)
|
|
123
123
|
add_offense(node, message: message) do |corrector|
|
|
124
|
-
next if part_of_ignored_node?(node)
|
|
124
|
+
next if safe_navigation?(node) || part_of_ignored_node?(node)
|
|
125
125
|
|
|
126
126
|
corrector.replace(node, correct_method(node, node.receiver))
|
|
127
127
|
|
|
@@ -156,13 +156,22 @@ module RuboCop
|
|
|
156
156
|
"{ |i| #{body} }"
|
|
157
157
|
end
|
|
158
158
|
|
|
159
|
+
def current_method(node, receiver, to_method)
|
|
160
|
+
operator = node.csend_type? ? '&.' : '.'
|
|
161
|
+
"#{receiver.source}#{operator}#{to_method}"
|
|
162
|
+
end
|
|
163
|
+
|
|
159
164
|
def remove_parentheses(corrector, node)
|
|
160
165
|
corrector.replace(node.loc.begin, ' ')
|
|
161
166
|
corrector.remove(node.loc.end)
|
|
162
167
|
end
|
|
163
168
|
|
|
169
|
+
def safe_navigation?(node)
|
|
170
|
+
node.csend_type? || node.each_descendant(:csend).any?
|
|
171
|
+
end
|
|
172
|
+
|
|
164
173
|
def allow_receiver?(receiver)
|
|
165
|
-
if receiver.numeric_type? || (receiver.
|
|
174
|
+
if receiver.numeric_type? || (receiver.call_type? &&
|
|
166
175
|
(conversion_method?(receiver.method_name) ||
|
|
167
176
|
allowed_method_name?(receiver.method_name)))
|
|
168
177
|
true
|
|
@@ -188,7 +197,7 @@ module RuboCop
|
|
|
188
197
|
end
|
|
189
198
|
|
|
190
199
|
def ignored_classes
|
|
191
|
-
cop_config.fetch('
|
|
200
|
+
cop_config.fetch('AllowedClasses', [])
|
|
192
201
|
end
|
|
193
202
|
|
|
194
203
|
def ignored_class?(name)
|
|
@@ -16,7 +16,7 @@ module RuboCop
|
|
|
16
16
|
# ruby 3.0.0p0 (2020-12-25 revision 95aff21468) [x86_64-darwin19]
|
|
17
17
|
# -e:1: _1 is reserved for numbered parameter
|
|
18
18
|
#
|
|
19
|
-
# NOTE: The
|
|
19
|
+
# NOTE: The numbered parameters are from `_1` to `_9`. This cop checks `_0`, and over `_10`
|
|
20
20
|
# as well to prevent confusion.
|
|
21
21
|
#
|
|
22
22
|
# @example
|
|
@@ -58,6 +58,9 @@ module RuboCop
|
|
|
58
58
|
'(op-asgn (lvasgn $_lhs) $_operation ({int lvar} $_rhs))'
|
|
59
59
|
|
|
60
60
|
def on_send(node)
|
|
61
|
+
# Safe navigation short-circuits to `nil` when the receiver is `nil`, so the
|
|
62
|
+
# result is not constant and replacing it with `0`/`1` would change behavior.
|
|
63
|
+
return if node.csend_type?
|
|
61
64
|
return unless (lhs, operation, rhs = operation_with_constant_result?(node))
|
|
62
65
|
return unless (result = constant_result?(lhs, operation, rhs))
|
|
63
66
|
|
|
@@ -38,23 +38,23 @@ module RuboCop
|
|
|
38
38
|
def on_new_investigation
|
|
39
39
|
return if processed_source.buffer.source.empty?
|
|
40
40
|
|
|
41
|
-
encoding_line,
|
|
41
|
+
encoding_line, other_magic_comment_line = magic_comment_lines
|
|
42
42
|
|
|
43
|
-
return unless encoding_line &&
|
|
44
|
-
return if encoding_line <
|
|
43
|
+
return unless encoding_line && other_magic_comment_line
|
|
44
|
+
return if encoding_line < other_magic_comment_line
|
|
45
45
|
|
|
46
46
|
range = processed_source.buffer.line_range(encoding_line + 1)
|
|
47
47
|
|
|
48
48
|
add_offense(range) do |corrector|
|
|
49
|
-
autocorrect(corrector, encoding_line,
|
|
49
|
+
autocorrect(corrector, encoding_line, other_magic_comment_line)
|
|
50
50
|
end
|
|
51
51
|
end
|
|
52
52
|
|
|
53
53
|
private
|
|
54
54
|
|
|
55
|
-
def autocorrect(corrector, encoding_line,
|
|
55
|
+
def autocorrect(corrector, encoding_line, other_magic_comment_line)
|
|
56
56
|
range1 = processed_source.buffer.line_range(encoding_line + 1)
|
|
57
|
-
range2 = processed_source.buffer.line_range(
|
|
57
|
+
range2 = processed_source.buffer.line_range(other_magic_comment_line + 1)
|
|
58
58
|
|
|
59
59
|
corrector.replace(range1, range2.source)
|
|
60
60
|
corrector.replace(range2, range1.source)
|
|
@@ -66,7 +66,7 @@ module RuboCop
|
|
|
66
66
|
leading_magic_comments.each.with_index do |comment, index|
|
|
67
67
|
if comment.encoding_specified?
|
|
68
68
|
lines[0] = index
|
|
69
|
-
elsif comment.
|
|
69
|
+
elsif comment.valid?
|
|
70
70
|
lines[1] = index
|
|
71
71
|
end
|
|
72
72
|
|
|
@@ -63,19 +63,9 @@ module RuboCop
|
|
|
63
63
|
end
|
|
64
64
|
|
|
65
65
|
def spaces_before_left_parenthesis(node)
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
else
|
|
70
|
-
0
|
|
71
|
-
end
|
|
72
|
-
without_receiver = node.source[receiver_length..]
|
|
73
|
-
|
|
74
|
-
# Escape question mark if any.
|
|
75
|
-
method_regexp = Regexp.escape(node.method_name)
|
|
76
|
-
|
|
77
|
-
match = without_receiver.match(/^\s*&?\.?\s*#{method_regexp}(\s+)\(/)
|
|
78
|
-
match ? match.captures[0].length : 0
|
|
66
|
+
return 0 if node.parenthesized? || !node.first_argument.source.start_with?('(')
|
|
67
|
+
|
|
68
|
+
node.first_argument.source_range.begin_pos - node.loc.selector.end_pos
|
|
79
69
|
end
|
|
80
70
|
|
|
81
71
|
def space_range(expr, space_length)
|
|
@@ -283,7 +283,10 @@ module RuboCop
|
|
|
283
283
|
end
|
|
284
284
|
|
|
285
285
|
def matching_range(haystack, needle)
|
|
286
|
-
|
|
286
|
+
# Match the cop name as a whole token so a shorter name is not found inside a
|
|
287
|
+
# longer one that shares its prefix (e.g. `Lint/AmbiguousOperator` in
|
|
288
|
+
# `Lint/AmbiguousOperatorPrecedence`).
|
|
289
|
+
offset = haystack.source.index(/#{Regexp.escape(needle)}(?!\w)/)
|
|
287
290
|
return unless offset
|
|
288
291
|
|
|
289
292
|
offset += haystack.begin_pos
|
|
@@ -1,19 +1,13 @@
|
|
|
1
1
|
# frozen_string_literal: true
|
|
2
2
|
|
|
3
|
-
# The Lint/RedundantCopEnableDirective and Lint/RedundantCopDisableDirective
|
|
4
|
-
# cops need to be disabled so as to be able to provide a (bad) example of an
|
|
5
|
-
# unneeded enable.
|
|
6
|
-
|
|
7
|
-
# rubocop:disable Lint/RedundantCopEnableDirective
|
|
8
|
-
# rubocop:disable Lint/RedundantCopDisableDirective
|
|
9
3
|
module RuboCop
|
|
10
4
|
module Cop
|
|
11
5
|
module Lint
|
|
12
6
|
# Detects instances of rubocop:enable comments that can be
|
|
13
7
|
# removed.
|
|
14
8
|
#
|
|
15
|
-
# When comment enables all cops at once `rubocop:enable all`
|
|
16
|
-
#
|
|
9
|
+
# When a comment enables all cops at once `rubocop:enable all`
|
|
10
|
+
# the cop checks whether any cop was actually enabled.
|
|
17
11
|
#
|
|
18
12
|
# @example
|
|
19
13
|
#
|
|
@@ -80,7 +74,10 @@ module RuboCop
|
|
|
80
74
|
end
|
|
81
75
|
|
|
82
76
|
def cop_name_indention(comment, name)
|
|
83
|
-
|
|
77
|
+
# Match the cop name as a whole token so a shorter name is not found inside a
|
|
78
|
+
# longer one that shares its prefix (e.g. `Layout/EmptyLines` in
|
|
79
|
+
# `Layout/EmptyLinesAfterModuleInclusion`).
|
|
80
|
+
comment.text.index(/#{Regexp.escape(name)}(?!\w)/)
|
|
84
81
|
end
|
|
85
82
|
|
|
86
83
|
def range_with_comma(comment, name)
|
|
@@ -130,6 +127,3 @@ module RuboCop
|
|
|
130
127
|
end
|
|
131
128
|
end
|
|
132
129
|
end
|
|
133
|
-
|
|
134
|
-
# rubocop:enable Lint/RedundantCopDisableDirective
|
|
135
|
-
# rubocop:enable Lint/RedundantCopEnableDirective
|
|
@@ -38,10 +38,10 @@ module RuboCop
|
|
|
38
38
|
GLOB_METHODS = %i[glob []].freeze
|
|
39
39
|
|
|
40
40
|
def on_send(node)
|
|
41
|
-
return unless (
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
return if multiple_argument?(receiver)
|
|
41
|
+
return unless dir_glob?(node.receiver)
|
|
42
|
+
# `sort` with a comparator block or block-pass changes the order, so it is
|
|
43
|
+
# not redundant with the default sorting performed by `Dir.glob`/`Dir[]`.
|
|
44
|
+
return if sort_with_comparator?(node) || multiple_argument?(node.receiver)
|
|
45
45
|
|
|
46
46
|
selector = node.loc.selector
|
|
47
47
|
|
|
@@ -53,9 +53,20 @@ module RuboCop
|
|
|
53
53
|
|
|
54
54
|
private
|
|
55
55
|
|
|
56
|
+
def dir_glob?(receiver)
|
|
57
|
+
return false unless receiver&.receiver&.const_type?
|
|
58
|
+
return false unless receiver.receiver.short_name == :Dir
|
|
59
|
+
|
|
60
|
+
GLOB_METHODS.include?(receiver.method_name)
|
|
61
|
+
end
|
|
62
|
+
|
|
56
63
|
def multiple_argument?(glob_method)
|
|
57
64
|
glob_method.arguments.count >= 2 || glob_method.first_argument&.splat_type?
|
|
58
65
|
end
|
|
66
|
+
|
|
67
|
+
def sort_with_comparator?(node)
|
|
68
|
+
node.parent&.any_block_type? || node.last_argument&.block_pass_type?
|
|
69
|
+
end
|
|
59
70
|
end
|
|
60
71
|
end
|
|
61
72
|
end
|
|
@@ -62,6 +62,22 @@ module RuboCop
|
|
|
62
62
|
# do_something if attrs.respond_to?(:[])
|
|
63
63
|
#
|
|
64
64
|
# # bad
|
|
65
|
+
# foo&.bar ? foo&.bar.baz : qux
|
|
66
|
+
#
|
|
67
|
+
# # good
|
|
68
|
+
# foo&.bar ? foo.bar.baz : qux
|
|
69
|
+
#
|
|
70
|
+
# # bad
|
|
71
|
+
# if foo&.bar
|
|
72
|
+
# foo&.bar.baz
|
|
73
|
+
# end
|
|
74
|
+
#
|
|
75
|
+
# # good
|
|
76
|
+
# if foo&.bar
|
|
77
|
+
# foo.bar.baz
|
|
78
|
+
# end
|
|
79
|
+
#
|
|
80
|
+
# # bad
|
|
65
81
|
# while node&.is_a?(BeginNode)
|
|
66
82
|
# node = node.parent
|
|
67
83
|
# end
|
|
@@ -71,8 +87,9 @@ module RuboCop
|
|
|
71
87
|
# node = node.parent
|
|
72
88
|
# end
|
|
73
89
|
#
|
|
74
|
-
# # good - without `&.` this
|
|
90
|
+
# # good - without `&.` this changes the return value for `nil`
|
|
75
91
|
# foo&.respond_to?(:to_a)
|
|
92
|
+
# foo&.respond_to?(:class)
|
|
76
93
|
#
|
|
77
94
|
# # bad - for `nil`s conversion methods return default values for the type
|
|
78
95
|
# foo&.to_h || {}
|
|
@@ -151,22 +168,22 @@ module RuboCop
|
|
|
151
168
|
MSG_NON_NIL = 'Redundant safe navigation on non-nil receiver (detected by analyzing ' \
|
|
152
169
|
'previous code/method invocations).'
|
|
153
170
|
|
|
154
|
-
|
|
171
|
+
NIL_METHODS = nil.methods.to_set.freeze
|
|
155
172
|
|
|
156
173
|
SNAKE_CASE = /\A[[:digit:][:upper:]_]+\z/.freeze
|
|
157
174
|
|
|
158
175
|
GUARANTEED_INSTANCE_METHODS = %i[to_s to_i to_f to_a to_h].freeze
|
|
159
176
|
|
|
160
|
-
# @!method
|
|
161
|
-
def_node_matcher :
|
|
162
|
-
(csend _ :respond_to? (sym %
|
|
177
|
+
# @!method respond_to_nil_method?(node)
|
|
178
|
+
def_node_matcher :respond_to_nil_method?, <<~PATTERN
|
|
179
|
+
(csend _ :respond_to? (sym %NIL_METHODS))
|
|
163
180
|
PATTERN
|
|
164
181
|
|
|
165
182
|
# @!method conversion_with_default?(node)
|
|
166
183
|
def_node_matcher :conversion_with_default?, <<~PATTERN
|
|
167
184
|
{
|
|
168
185
|
(or $(csend _ :to_h) (hash))
|
|
169
|
-
(or (
|
|
186
|
+
(or (any_block $(csend _ :to_h) ...) (hash))
|
|
170
187
|
(or $(csend _ :to_a) (array))
|
|
171
188
|
(or $(csend _ :to_i) (int 0))
|
|
172
189
|
(or $(csend _ :to_f) (float 0.0))
|
|
@@ -174,7 +191,6 @@ module RuboCop
|
|
|
174
191
|
}
|
|
175
192
|
PATTERN
|
|
176
193
|
|
|
177
|
-
# rubocop:disable Metrics/AbcSize
|
|
178
194
|
def on_csend(node)
|
|
179
195
|
range = node.loc.dot
|
|
180
196
|
|
|
@@ -187,14 +203,10 @@ module RuboCop
|
|
|
187
203
|
end
|
|
188
204
|
end
|
|
189
205
|
|
|
190
|
-
|
|
191
|
-
return if !guaranteed_instance?(node.receiver) && !check?(node)
|
|
192
|
-
return if respond_to_nil_specific_method?(node)
|
|
193
|
-
end
|
|
206
|
+
return if guarded_by_nil_receiver?(node)
|
|
194
207
|
|
|
195
208
|
add_offense(range) { |corrector| corrector.replace(range, '.') }
|
|
196
209
|
end
|
|
197
|
-
# rubocop:enable Metrics/AbcSize
|
|
198
210
|
|
|
199
211
|
# rubocop:disable Metrics/AbcSize
|
|
200
212
|
def on_or(node)
|
|
@@ -213,6 +225,18 @@ module RuboCop
|
|
|
213
225
|
|
|
214
226
|
private
|
|
215
227
|
|
|
228
|
+
# Returns true when the `&.` is meaningful because the receiver may actually be nil.
|
|
229
|
+
def guarded_by_nil_receiver?(node)
|
|
230
|
+
return false if assume_receiver_instance_exists?(node.receiver)
|
|
231
|
+
|
|
232
|
+
guaranteed_instance = guaranteed_instance?(node.receiver)
|
|
233
|
+
return true if !guaranteed_instance && !check?(node)
|
|
234
|
+
|
|
235
|
+
# `nil.respond_to?(<nil method>)` is `true`, so `&.` is meaningful when the receiver
|
|
236
|
+
# may be nil. A guaranteed instance can never be nil, so `&.` is still redundant there.
|
|
237
|
+
respond_to_nil_method?(node) && !guaranteed_instance
|
|
238
|
+
end
|
|
239
|
+
|
|
216
240
|
def assume_receiver_instance_exists?(receiver)
|
|
217
241
|
return true if receiver.const_type? && !receiver.short_name.match?(SNAKE_CASE)
|
|
218
242
|
|
|
@@ -122,6 +122,10 @@ module RuboCop
|
|
|
122
122
|
|
|
123
123
|
grandparent = node.parent.parent
|
|
124
124
|
return if grandparent && !ASSIGNMENT_TYPES.include?(grandparent.type)
|
|
125
|
+
# An empty array/percent literal (`*[]`, `*%w()`, ...) expands to nothing, so
|
|
126
|
+
# removing the splat would produce invalid or semantically different code.
|
|
127
|
+
elsif expanded_item.array_type? && expanded_item.children.empty?
|
|
128
|
+
return
|
|
125
129
|
end
|
|
126
130
|
|
|
127
131
|
yield
|
|
@@ -17,14 +17,15 @@ module RuboCop
|
|
|
17
17
|
# or with `String.new` or `String()`.
|
|
18
18
|
# * `to_sym` when called on a symbol literal or interpolated symbol.
|
|
19
19
|
# * `to_i` when called on an integer literal or with `Integer()`.
|
|
20
|
-
# * `to_f` when called on a float literal
|
|
20
|
+
# * `to_f` when called on a float literal or with `Float()`.
|
|
21
|
+
# * `to_d` when called with `BigDecimal()`.
|
|
21
22
|
# * `to_r` when called on a rational literal or with `Rational()`.
|
|
22
|
-
# * `to_c` when called on a complex literal
|
|
23
|
+
# * `to_c` when called on a complex literal or with `Complex()`.
|
|
23
24
|
# * `to_a` when called on an array literal, or with `Array.new`, `Array()` or `Array[]`.
|
|
24
25
|
# * `to_h` when called on a hash literal, or with `Hash.new`, `Hash()` or `Hash[]`.
|
|
25
26
|
# * `to_set` when called on `Set.new` or `Set[]`.
|
|
26
27
|
#
|
|
27
|
-
# In all cases, chaining one same `to_*` conversion methods listed above is redundant.
|
|
28
|
+
# In all cases, chaining one of the same `to_*` conversion methods listed above is redundant.
|
|
28
29
|
#
|
|
29
30
|
# The cop can also register an offense for chaining conversion methods on methods that are
|
|
30
31
|
# expected to return a specific type regardless of receiver (eg. `foo.inspect.to_s` and
|
|
@@ -63,6 +64,12 @@ module RuboCop
|
|
|
63
64
|
# # in this case, `Integer()` could return `nil`
|
|
64
65
|
# Integer(var, exception: false).to_i
|
|
65
66
|
#
|
|
67
|
+
# # bad
|
|
68
|
+
# BigDecimal(var).to_d
|
|
69
|
+
#
|
|
70
|
+
# # good
|
|
71
|
+
# BigDecimal(var)
|
|
72
|
+
#
|
|
66
73
|
# # bad - chaining the same conversion
|
|
67
74
|
# foo.to_s.to_s
|
|
68
75
|
#
|
|
@@ -5,6 +5,11 @@ module RuboCop
|
|
|
5
5
|
module Lint
|
|
6
6
|
# Checks for redundant `with_object`.
|
|
7
7
|
#
|
|
8
|
+
# @safety
|
|
9
|
+
# This cop's autocorrection is unsafe because the return value changes:
|
|
10
|
+
# `each_with_object` returns the memo object, while the corrected `each` returns
|
|
11
|
+
# the receiver. This matters when the result of the expression is used.
|
|
12
|
+
#
|
|
8
13
|
# @example
|
|
9
14
|
# # bad
|
|
10
15
|
# ary.each_with_object([]) do |v|
|
|
@@ -32,9 +32,12 @@ module RuboCop
|
|
|
32
32
|
# end
|
|
33
33
|
#
|
|
34
34
|
class RefinementImportMethods < Base
|
|
35
|
+
extend AutoCorrector
|
|
35
36
|
extend TargetRubyVersion
|
|
36
37
|
|
|
37
38
|
MSG = 'Use `import_methods` instead of `%<current>s` because it is deprecated in Ruby 3.1.'
|
|
39
|
+
MSG_REMOVED = 'Use `import_methods` instead of `%<current>s` ' \
|
|
40
|
+
'because it was removed in Ruby 3.2.'
|
|
38
41
|
RESTRICT_ON_SEND = %i[include prepend].freeze
|
|
39
42
|
|
|
40
43
|
minimum_target_ruby_version 3.1
|
|
@@ -44,7 +47,11 @@ module RuboCop
|
|
|
44
47
|
return unless (parent = node.parent)
|
|
45
48
|
return unless parent.block_type? && parent.method?(:refine)
|
|
46
49
|
|
|
47
|
-
|
|
50
|
+
template = target_ruby_version >= 3.2 ? MSG_REMOVED : MSG
|
|
51
|
+
message = format(template, current: node.method_name)
|
|
52
|
+
add_offense(node.loc.selector, message: message) do |corrector|
|
|
53
|
+
corrector.replace(node.loc.selector, 'import_methods')
|
|
54
|
+
end
|
|
48
55
|
end
|
|
49
56
|
end
|
|
50
57
|
end
|
|
@@ -26,7 +26,15 @@ module RuboCop
|
|
|
26
26
|
return if node.ancestors.none?(&:conditional?)
|
|
27
27
|
return if part_of_ignored_node?(node)
|
|
28
28
|
|
|
29
|
-
add_offense(node)
|
|
29
|
+
add_offense(node) do |corrector|
|
|
30
|
+
# `!` binds tighter than `=~`, so `!/foo/ =~ $_` would parse as
|
|
31
|
+
# `(!/foo/) =~ $_`. Wrap the match in parentheses to preserve the meaning.
|
|
32
|
+
if node.parent&.send_type? && node.parent.method?(:!)
|
|
33
|
+
corrector.replace(node.parent, "!(#{node.source} =~ $_)")
|
|
34
|
+
else
|
|
35
|
+
corrector.replace(node, "#{node.source} =~ $_")
|
|
36
|
+
end
|
|
37
|
+
end
|
|
30
38
|
|
|
31
39
|
ignore_node(node)
|
|
32
40
|
end
|
|
@@ -3,10 +3,13 @@
|
|
|
3
3
|
module RuboCop
|
|
4
4
|
module Cop
|
|
5
5
|
module Lint
|
|
6
|
-
# Checks for
|
|
7
|
-
#
|
|
8
|
-
#
|
|
9
|
-
#
|
|
6
|
+
# Checks for method calls with at least one argument where no parentheses
|
|
7
|
+
# are used around the parameter list, and the call could be misread as an
|
|
8
|
+
# operand of a boolean operator (`&&` or `||`). Two forms are flagged:
|
|
9
|
+
#
|
|
10
|
+
# * a predicate method whose last argument is a `&&`/`||` expression, and
|
|
11
|
+
# * any method whose first argument is a ternary expression with a
|
|
12
|
+
# `&&`/`||` condition.
|
|
10
13
|
#
|
|
11
14
|
# The idea behind warning for these constructs is that the user might
|
|
12
15
|
# be under the impression that the return value from the method call is
|
|
@@ -23,6 +26,12 @@ module RuboCop
|
|
|
23
26
|
# if day.is?(:tuesday) && month == :jan
|
|
24
27
|
# # ...
|
|
25
28
|
# end
|
|
29
|
+
#
|
|
30
|
+
# # bad
|
|
31
|
+
# foo a && b ? c : d
|
|
32
|
+
#
|
|
33
|
+
# # good
|
|
34
|
+
# foo(a && b ? c : d)
|
|
26
35
|
class RequireParentheses < Base
|
|
27
36
|
include RangeHelp
|
|
28
37
|
|
|
@@ -38,7 +38,8 @@ module RuboCop
|
|
|
38
38
|
# 42)
|
|
39
39
|
#
|
|
40
40
|
class RequireRangeParentheses < Base
|
|
41
|
-
MSG = 'Wrap the
|
|
41
|
+
MSG = 'Wrap the range literal `%<range>s` in parentheses ' \
|
|
42
|
+
'to avoid confusion with an endless range.'
|
|
42
43
|
|
|
43
44
|
def on_irange(node)
|
|
44
45
|
return if node.parent&.begin_type?
|
|
@@ -3,7 +3,7 @@
|
|
|
3
3
|
module RuboCop
|
|
4
4
|
module Cop
|
|
5
5
|
module Lint
|
|
6
|
-
# Checks for
|
|
6
|
+
# Checks for a file requiring itself with `require_relative`.
|
|
7
7
|
#
|
|
8
8
|
# @example
|
|
9
9
|
#
|
|
@@ -38,11 +38,13 @@ module RuboCop
|
|
|
38
38
|
private
|
|
39
39
|
|
|
40
40
|
def same_file?(file_path, required_feature)
|
|
41
|
-
|
|
42
|
-
end
|
|
41
|
+
return false unless File.extname(file_path) == '.rb'
|
|
43
42
|
|
|
44
|
-
|
|
45
|
-
|
|
43
|
+
# `require_relative` is resolved relative to the current file's directory, so a
|
|
44
|
+
# bare `foo`/`foo.rb` (no path separator) requires the current file itself. Compare
|
|
45
|
+
# against the basename so this works whether `file_path` is relative or absolute.
|
|
46
|
+
basename = File.basename(file_path, '.rb')
|
|
47
|
+
required_feature == basename || required_feature == "#{basename}.rb"
|
|
46
48
|
end
|
|
47
49
|
end
|
|
48
50
|
end
|
|
@@ -39,7 +39,7 @@ module RuboCop
|
|
|
39
39
|
|
|
40
40
|
MSG = 'Rescuing from `%<invalid_exceptions>s` will raise a ' \
|
|
41
41
|
'`TypeError` instead of catching the actual exception.'
|
|
42
|
-
INVALID_TYPES = %i[array dstr float hash nil int str sym].freeze
|
|
42
|
+
INVALID_TYPES = %i[array complex dstr false float hash nil int rational str sym true].freeze
|
|
43
43
|
|
|
44
44
|
def on_resbody(node)
|
|
45
45
|
invalid_exceptions = invalid_exceptions(node.exceptions)
|
|
@@ -34,23 +34,29 @@ module RuboCop
|
|
|
34
34
|
{
|
|
35
35
|
(send $(csend ...) $_ ...)
|
|
36
36
|
(send $(any_block (csend ...) ...) $_ ...)
|
|
37
|
+
(send $(begin (csend ...)) $_ ...)
|
|
37
38
|
}
|
|
38
39
|
PATTERN
|
|
39
40
|
|
|
41
|
+
# rubocop:disable Metrics/AbcSize
|
|
40
42
|
def on_send(node)
|
|
41
43
|
return unless require_safe_navigation?(node)
|
|
42
44
|
|
|
43
45
|
bad_method?(node) do |safe_nav, method|
|
|
44
46
|
return if nil_methods.include?(method) || PLUS_MINUS_METHODS.include?(node.method_name)
|
|
47
|
+
return if ternary_safe_navigation?(node, safe_nav)
|
|
45
48
|
|
|
46
49
|
begin_range = node.loc.dot || safe_nav.source_range.end
|
|
47
50
|
location = begin_range.join(node.source_range.end)
|
|
48
51
|
|
|
49
52
|
add_offense(location) do |corrector|
|
|
53
|
+
next if ternary_else_branch?(node, safe_nav)
|
|
54
|
+
|
|
50
55
|
autocorrect(corrector, offense_range: location, send_node: node)
|
|
51
56
|
end
|
|
52
57
|
end
|
|
53
58
|
end
|
|
59
|
+
# rubocop:enable Metrics/AbcSize
|
|
54
60
|
|
|
55
61
|
private
|
|
56
62
|
|
|
@@ -61,6 +67,18 @@ module RuboCop
|
|
|
61
67
|
parent.rhs != node || parent.lhs.receiver != parent.rhs.receiver
|
|
62
68
|
end
|
|
63
69
|
|
|
70
|
+
def ternary_safe_navigation?(node, safe_nav)
|
|
71
|
+
return false unless (parent = node.parent)
|
|
72
|
+
|
|
73
|
+
parent.if_type? && node.equal?(parent.if_branch) && parent.condition == safe_nav
|
|
74
|
+
end
|
|
75
|
+
|
|
76
|
+
def ternary_else_branch?(node, safe_nav)
|
|
77
|
+
return false unless (parent = node.parent)
|
|
78
|
+
|
|
79
|
+
parent.if_type? && node.equal?(parent.else_branch) && parent.condition == safe_nav
|
|
80
|
+
end
|
|
81
|
+
|
|
64
82
|
# @param [Parser::Source::Range] offense_range
|
|
65
83
|
# @param [RuboCop::AST::SendNode] send_node
|
|
66
84
|
# @return [String]
|
|
@@ -3,10 +3,16 @@
|
|
|
3
3
|
module RuboCop
|
|
4
4
|
module Cop
|
|
5
5
|
module Lint
|
|
6
|
-
#
|
|
6
|
+
# Checks that if safe navigation is used in an `&&` or `||` condition,
|
|
7
7
|
# consistent and appropriate safe navigation, without excess or deficiency,
|
|
8
8
|
# is used for all method calls on the same object.
|
|
9
9
|
#
|
|
10
|
+
# @safety
|
|
11
|
+
# Autocorrection is unsafe because if the receiver is not a local variable
|
|
12
|
+
# but a method call, it may not be idempotent. For example, replacing
|
|
13
|
+
# `foo&.bar` with `foo.bar` could raise `NoMethodError` if `foo` returns
|
|
14
|
+
# `nil` on a subsequent call.
|
|
15
|
+
#
|
|
10
16
|
# @example
|
|
11
17
|
# # bad
|
|
12
18
|
# foo&.bar && foo&.baz
|