# encoding: utf-8 require 'byebug/helpers/file' module Byebug # # Represents a frame in the stack trace # class Frame include Helpers::FileHelper attr_reader :pos def initialize(context, pos) @context = context @pos = pos end def file @context.frame_file(pos) end def line @context.frame_line(pos) end def _self @context.frame_self(pos) end def _binding @context.frame_binding(pos) end def _class @context.frame_class(pos) end def _method @context.frame_method(pos) end def current? @context.frame.pos == pos end # # Gets local variables for the frame. # # TODO: Use brand new local_variable_{get,set,defined?} for rubies >= 2.1 # def locals return [] unless _binding _binding.eval('local_variables.inject({}){|h, v| h[v] = eval(v.to_s); h}') end # # Gets current method arguments for the frame. # def args return c_args unless _binding ruby_args end # # Returns the current class in the frame or an empty string if the current # +callstyle+ setting is 'short' # def deco_class Setting[:callstyle] == 'short' || _class.to_s.empty? ? '' : "#{_class}." end def deco_block _method[/(?:block(?: \(\d+ levels\))?|rescue) in /] || '' end def deco_method _method[/((?:block(?: \(\d+ levels\))?|rescue) in )?(.*)/] end # # Builds a string containing all available args in the frame number, in a # verbose or non verbose way according to the value of the +callstyle+ # setting # def deco_args return '' if args.empty? my_args = args.map do |arg| prefix, default = prefix_and_default(arg[0]) kls = use_short_style?(arg) ? '' : "##{locals[arg[1]].class}" "#{prefix}#{arg[1] || default}#{kls}" end "(#{my_args.join(', ')})" end # # Builds a formatted string containing information about current method call # def deco_call deco_block + deco_class + deco_method + deco_args end # # Formatted filename in frame # def deco_file Setting[:fullpath] ? File.expand_path(file) : shortpath(file) end # # Properly formatted frame number of frame # def deco_pos format('%-2d', pos) end # # Formatted mark for the frame. # # --> marks the current frame # ͱ-- marks c-frames # marks regular frames # def mark return '-->' if current? return ' ͱ--' if c_frame? ' ' end # # Checks whether the frame is a c-frame # def c_frame? _binding.nil? end def to_hash { mark: mark, pos: deco_pos, call: deco_call, file: deco_file, line: line, full_path: File.expand_path(deco_file) } end private def c_args return [] unless _self.to_s != 'main' _self.method(_method).parameters end def ruby_args return [] unless _binding.eval('__method__') return [] unless _binding.eval('method(__method__)') _binding.eval('method(__method__).parameters') end def use_short_style?(arg) Setting[:callstyle] == 'short' || arg[1].nil? || locals.empty? end def prefix_and_default(arg_type) return ['&', 'block'] if arg_type == :block return ['*', 'args'] if arg_type == :rest ['', nil] end end end