Sha256: dc1e7def0c58bed502dc91e904086c1711c98a191b5447f2d79fdd56a130cd6f

Contents?: true

Size: 1.65 KB

Versions: 38

Compression:

Stored size: 1.65 KB

Contents

# frozen_string_literal: true

module RuboCop
  module Cop
    module Lint
      # The safe navigation operator returns nil if the receiver is
      # nil. If you chain an ordinary method call after a safe
      # navigation operator, it raises NoMethodError. We should use a
      # safe navigation operator after a safe navigation operator.
      # This cop checks for the problem outlined above.
      #
      # @example
      #
      #   # bad
      #
      #   x&.foo.bar
      #   x&.foo + bar
      #   x&.foo[bar]
      #
      # @example
      #
      #   # good
      #
      #   x&.foo&.bar
      #   x&.foo || bar
      class SafeNavigationChain < Base
        include NilMethods

        MSG = 'Do not chain ordinary method call after safe navigation operator.'

        # @!method bad_method?(node)
        def_node_matcher :bad_method?, <<~PATTERN
          {
            (send $(csend ...) $_ ...)
            (send $({block numblock} (csend ...) ...) $_ ...)
          }
        PATTERN

        def on_send(node)
          bad_method?(node) do |safe_nav, method|
            return if nil_methods.include?(method)

            method_chain = method_chain(node)
            location =
              Parser::Source::Range.new(node.source_range.source_buffer,
                                        safe_nav.source_range.end_pos,
                                        method_chain.source_range.end_pos)
            add_offense(location)
          end
        end

        private

        def method_chain(node)
          chain = node
          chain = chain.parent if chain.send_type? && chain.parent&.call_type?
          chain
        end
      end
    end
  end
end

Version data entries

38 entries across 38 versions & 6 rubygems

Version Path
scrapbook-0.3.2 vendor/ruby/2.7.0/gems/rubocop-1.26.0/lib/rubocop/cop/lint/safe_navigation_chain.rb
scrapbook-0.3.1 vendor/ruby/2.7.0/gems/rubocop-1.26.0/lib/rubocop/cop/lint/safe_navigation_chain.rb
rubocop-1.29.1 lib/rubocop/cop/lint/safe_navigation_chain.rb
rubocop-1.29.0 lib/rubocop/cop/lint/safe_navigation_chain.rb
rubocop-1.28.2 lib/rubocop/cop/lint/safe_navigation_chain.rb
rubocop-1.28.1 lib/rubocop/cop/lint/safe_navigation_chain.rb
rubocop-1.28.0 lib/rubocop/cop/lint/safe_navigation_chain.rb
rubocop-1.27.0 lib/rubocop/cop/lint/safe_navigation_chain.rb
rubocop-1.26.1 lib/rubocop/cop/lint/safe_navigation_chain.rb
op_connect-0.1.2 vendor/bundle/ruby/3.1.0/gems/rubocop-1.26.0/lib/rubocop/cop/lint/safe_navigation_chain.rb
rubocop-1.26.0 lib/rubocop/cop/lint/safe_navigation_chain.rb
rubocop-1.25.1 lib/rubocop/cop/lint/safe_navigation_chain.rb
rubocop-1.25.0 lib/rubocop/cop/lint/safe_navigation_chain.rb
phillipug-foodie-0.1.0 .vendor/ruby/3.0.0/gems/rubocop-1.24.0/lib/rubocop/cop/lint/safe_navigation_chain.rb
rubocop-1.24.1 lib/rubocop/cop/lint/safe_navigation_chain.rb
rubocop-1.24.0 lib/rubocop/cop/lint/safe_navigation_chain.rb
rubocop-1.23.0 lib/rubocop/cop/lint/safe_navigation_chain.rb
rubocop-1.22.3 lib/rubocop/cop/lint/safe_navigation_chain.rb
rubocop-1.22.2 lib/rubocop/cop/lint/safe_navigation_chain.rb
rubocop-1.22.1 lib/rubocop/cop/lint/safe_navigation_chain.rb