lib/pry/commands/wtf.rb



# frozen_string_literal: true

class Pry
  class Command
    class Wtf < Pry::ClassCommand
      match(/wtf([?!]*)/)
      group 'Context'
      description 'Show the backtrace of the most recent exception.'
      options listing: 'wtf?'

      banner <<-'BANNER'
        Usage: wtf[?|!]

        Shows a few lines of the backtrace of the most recent exception (also available
        as `_ex_.backtrace`). If you want to see more lines, add more question marks or
        exclamation marks.

        wtf?
        wtf?!???!?!?

        # To see the entire backtrace, pass the `-v` or `--verbose` flag.
        wtf -v
      BANNER

      RUBY_FRAME_PATTERN = /\A(?<file>(.+)):(?<line>(\d+))/.freeze

      def options(opt)
        opt.on :v, :verbose, "Show the full backtrace"
        opt.on :c, :code, "Show code corresponding to the backtrace frame"
      end

      def process
        unless pry_instance.last_exception
          raise Pry::CommandError, "No most-recent exception"
        end

        text = ''.dup
        unwind_exceptions.each_with_index do |exception, i|
          title = (i == 0 ? 'Exception' : 'Caused by')
          text << format_header(title, exception)
          text << format_backtrace(exception.backtrace)
        end
        output.puts(text)
      end

      private

      def unwind_exceptions
        exception_list = []
        exception = pry_instance.last_exception

        while exception
          exception_list << exception
          exception = (exception.cause if exception.respond_to?(:cause))
        end

        exception_list
      end

      def format_header(title, exception)
        "#{bold(title + ':')} #{exception.class}: #{exception}\n--\n"
      end

      def format_backtrace(backtrace)
        lines = trim_backtrace(backtrace).map do |frame|
          next frame unless opts.code?

          match = frame.match(RUBY_FRAME_PATTERN)
          code = read_line(match[:file], match[:line].to_i)
          [bold(frame), code].join("\n")
        end

        Pry::Code.new(lines.compact, 0, :text).with_line_numbers.to_s
      end

      def trim_backtrace(backtrace)
        return backtrace if opts.verbose?

        size_of_backtrace = [captures[0].size, 0.5].max * 10
        backtrace.first(size_of_backtrace)
      end

      def read_line(file, line)
        File.open(file, 'r') do |f|
          (line - 1).times { f.gets }
          f.gets
        end
      rescue Errno::ENOENT
        nil
      end
    end

    Pry::Commands.add_command(Pry::Command::Wtf)
  end
end