Sha256: 512aaf79aff45d2ecc5c96c1b98037d3aeb8bd6c1409f39c773bcb0c03877761

Contents?: true

Size: 1.97 KB

Versions: 21

Compression:

Stored size: 1.97 KB

Contents

# frozen_string_literal: true

module RuboCop
  module Cop
    module Rails
      # This cop enforces that `exit` calls are not used within a rails app.
      # Valid options are instead to raise an error, break, return, or some
      # other form of stopping execution of current request.
      #
      # There are two obvious cases where `exit` is particularly harmful:
      #
      # * Usage in library code for your application. Even though Rails will
      # rescue from a `SystemExit` and continue on, unit testing that library
      # code will result in specs exiting (potentially silently if `exit(0)`
      # is used.)
      # * Usage in application code outside of the web process could result in
      # the program exiting, which could result in the code failing to run and
      # do its job.
      #
      # @example
      #   # bad
      #   exit(0)
      #
      #   # good
      #   raise 'a bad error has happened'
      class Exit < Base
        include ConfigurableEnforcedStyle

        MSG = 'Do not use `exit` in Rails applications.'
        RESTRICT_ON_SEND = %i[exit exit!].freeze
        EXPLICIT_RECEIVERS = %i[Kernel Process].freeze

        def on_send(node)
          add_offense(node.loc.selector) if offending_node?(node)
        end

        private

        def offending_node?(node)
          right_argument_count?(node.arguments) && right_receiver?(node.receiver)
        end

        # More than 1 argument likely means it is a different
        # `exit` implementation than the one we are preventing.
        def right_argument_count?(arg_nodes)
          arg_nodes.size <= 1
        end

        # Only register if exit is being called explicitly on `Kernel`,
        # `Process`, or if receiver node is nil for plain `exit` calls.
        def right_receiver?(receiver_node)
          return true unless receiver_node

          _a, receiver_node_class, _c = *receiver_node

          EXPLICIT_RECEIVERS.include?(receiver_node_class)
        end
      end
    end
  end
end

Version data entries

21 entries across 21 versions & 2 rubygems

Version Path
scrapbook-0.3.2 vendor/ruby/2.7.0/gems/rubocop-rails-2.14.2/lib/rubocop/cop/rails/exit.rb
scrapbook-0.3.1 vendor/ruby/2.7.0/gems/rubocop-rails-2.14.2/lib/rubocop/cop/rails/exit.rb
rubocop-rails-2.14.2 lib/rubocop/cop/rails/exit.rb
rubocop-rails-2.14.1 lib/rubocop/cop/rails/exit.rb
rubocop-rails-2.14.0 lib/rubocop/cop/rails/exit.rb
rubocop-rails-2.13.2 lib/rubocop/cop/rails/exit.rb
rubocop-rails-2.13.1 lib/rubocop/cop/rails/exit.rb
rubocop-rails-2.13.0 lib/rubocop/cop/rails/exit.rb
rubocop-rails-2.12.4 lib/rubocop/cop/rails/exit.rb
rubocop-rails-2.12.3 lib/rubocop/cop/rails/exit.rb
rubocop-rails-2.12.2 lib/rubocop/cop/rails/exit.rb
rubocop-rails-2.12.1 lib/rubocop/cop/rails/exit.rb
rubocop-rails-2.12.0 lib/rubocop/cop/rails/exit.rb
rubocop-rails-2.11.3 lib/rubocop/cop/rails/exit.rb
rubocop-rails-2.11.2 lib/rubocop/cop/rails/exit.rb
rubocop-rails-2.11.1 lib/rubocop/cop/rails/exit.rb
rubocop-rails-2.11.0 lib/rubocop/cop/rails/exit.rb
rubocop-rails-2.10.1 lib/rubocop/cop/rails/exit.rb
rubocop-rails-2.10.0 lib/rubocop/cop/rails/exit.rb
rubocop-rails-2.9.1 lib/rubocop/cop/rails/exit.rb