模块 ErrorHighlight
常量
- VERSION
公共类方法
formatter() 点击切换源代码
# File error_highlight/formatter.rb, line 67 def self.formatter Ractor.current[:__error_highlight_formatter__] || DefaultFormatter end
formatter=(formatter) 点击切换源代码
# File error_highlight/formatter.rb, line 71 def self.formatter=(formatter) Ractor.current[:__error_highlight_formatter__] = formatter end
spot(obj, **opts) 点击切换源代码
识别给定异常发生的代码片段。
选项
point_type: :name | :args
:name (default) points the method/variable name that the exception occurred. :args points the arguments of the method call that the exception occurred.
backtrace_location: Thread::Backtrace::Location
It locates the code fragment of the given backtrace_location. By default, it uses the first frame of backtrace_locations of the given exception.
返回值
{ first_lineno: Integer, first_column: Integer, last_lineno: Integer, last_column: Integer, snippet: String, script_lines: [String], } | nil
局限性
目前,ErrorHighlight.spot
仅支持单行代码片段。因此,如果返回值不为 nil,则 first_lineno 和 last_lineno 将具有相同的值。如果相关代码片段跨越多行(例如,+ary+ 的 Array#[]),则该方法将返回 nil。 此限制将来可能会被取消。
# File error_highlight/base.rb, line 33 def self.spot(obj, **opts) case obj when Exception exc = obj loc = opts[:backtrace_location] opts = { point_type: opts.fetch(:point_type, :name) } unless loc case exc when TypeError, ArgumentError opts[:point_type] = :args end locs = exc.backtrace_locations return nil unless locs loc = locs.first return nil unless loc opts[:name] = exc.name if NameError === obj end return nil unless Thread::Backtrace::Location === loc node = begin RubyVM::AbstractSyntaxTree.of(loc, keep_script_lines: true) rescue RuntimeError => error # RubyVM::AbstractSyntaxTree.of raises an error with a message that # includes "prism" when the ISEQ was compiled with the prism compiler. # In this case, we'll try to parse again with prism instead. raise unless error.message.include?("prism") prism_find(loc) end Spotter.new(node, **opts).spot when RubyVM::AbstractSyntaxTree::Node, Prism::Node Spotter.new(obj, **opts).spot else raise TypeError, "Exception is expected" end rescue SyntaxError, SystemCallError, # file not found or something ArgumentError # eval'ed code return nil end
私有类方法
prism_find(location) 点击切换源代码
接受一个 Thread::Backtrace::Location 对象,并返回源代码中与回溯位置对应的 Prism::Node。
# File error_highlight/base.rb, line 86 def self.prism_find(location) require "prism" return nil if Prism::VERSION < "1.0.0" absolute_path = location.absolute_path return unless absolute_path node_id = RubyVM::AbstractSyntaxTree.node_id_for_backtrace_location(location) Prism.parse_file(absolute_path).value.breadth_first_search { |node| node.node_id == node_id } end