Merge pull request #14330 from Rylan12/fix-method-source-gems
Vendor `method_source` gem
This commit is contained in:
commit
918b47e24b
1
.gitignore
vendored
1
.gitignore
vendored
@ -111,7 +111,6 @@
|
|||||||
**/vendor/bundle/ruby/*/gems/jaro_winkler-*/
|
**/vendor/bundle/ruby/*/gems/jaro_winkler-*/
|
||||||
**/vendor/bundle/ruby/*/gems/json-*/
|
**/vendor/bundle/ruby/*/gems/json-*/
|
||||||
**/vendor/bundle/ruby/*/gems/json_schemer-*/
|
**/vendor/bundle/ruby/*/gems/json_schemer-*/
|
||||||
**/vendor/bundle/ruby/*/gems/method_source-*/
|
|
||||||
**/vendor/bundle/ruby/*/gems/mime-types-data-*/
|
**/vendor/bundle/ruby/*/gems/mime-types-data-*/
|
||||||
**/vendor/bundle/ruby/*/gems/mime-types-*/
|
**/vendor/bundle/ruby/*/gems/mime-types-*/
|
||||||
**/vendor/bundle/ruby/*/gems/mini_portile2-*/
|
**/vendor/bundle/ruby/*/gems/mini_portile2-*/
|
||||||
|
|||||||
141
Library/Homebrew/vendor/bundle/ruby/2.6.0/gems/method_source-1.0.0/lib/method_source.rb
vendored
Normal file
141
Library/Homebrew/vendor/bundle/ruby/2.6.0/gems/method_source-1.0.0/lib/method_source.rb
vendored
Normal file
@ -0,0 +1,141 @@
|
|||||||
|
# (C) John Mair (banisterfiend) 2011
|
||||||
|
# MIT License
|
||||||
|
|
||||||
|
direc = File.dirname(__FILE__)
|
||||||
|
|
||||||
|
require "#{direc}/method_source/version"
|
||||||
|
require "#{direc}/method_source/source_location"
|
||||||
|
require "#{direc}/method_source/code_helpers"
|
||||||
|
|
||||||
|
module MethodSource
|
||||||
|
extend MethodSource::CodeHelpers
|
||||||
|
|
||||||
|
# An Exception to mark errors that were raised trying to find the source from
|
||||||
|
# a given source_location.
|
||||||
|
#
|
||||||
|
class SourceNotFoundError < StandardError; end
|
||||||
|
|
||||||
|
# Helper method responsible for extracting method body.
|
||||||
|
# Defined here to avoid polluting `Method` class.
|
||||||
|
# @param [Array] source_location The array returned by Method#source_location
|
||||||
|
# @param [String] method_name
|
||||||
|
# @return [String] The method body
|
||||||
|
def self.source_helper(source_location, name=nil)
|
||||||
|
raise SourceNotFoundError, "Could not locate source for #{name}!" unless source_location
|
||||||
|
file, line = *source_location
|
||||||
|
|
||||||
|
expression_at(lines_for(file), line)
|
||||||
|
rescue SyntaxError => e
|
||||||
|
raise SourceNotFoundError, "Could not parse source for #{name}: #{e.message}"
|
||||||
|
end
|
||||||
|
|
||||||
|
# Helper method responsible for opening source file and buffering up
|
||||||
|
# the comments for a specified method. Defined here to avoid polluting
|
||||||
|
# `Method` class.
|
||||||
|
# @param [Array] source_location The array returned by Method#source_location
|
||||||
|
# @param [String] method_name
|
||||||
|
# @return [String] The comments up to the point of the method.
|
||||||
|
def self.comment_helper(source_location, name=nil)
|
||||||
|
raise SourceNotFoundError, "Could not locate source for #{name}!" unless source_location
|
||||||
|
file, line = *source_location
|
||||||
|
|
||||||
|
comment_describing(lines_for(file), line)
|
||||||
|
end
|
||||||
|
|
||||||
|
# Load a memoized copy of the lines in a file.
|
||||||
|
#
|
||||||
|
# @param [String] file_name
|
||||||
|
# @param [String] method_name
|
||||||
|
# @return [Array<String>] the contents of the file
|
||||||
|
# @raise [SourceNotFoundError]
|
||||||
|
def self.lines_for(file_name, name=nil)
|
||||||
|
@lines_for_file ||= {}
|
||||||
|
@lines_for_file[file_name] ||= File.readlines(file_name)
|
||||||
|
rescue Errno::ENOENT => e
|
||||||
|
raise SourceNotFoundError, "Could not load source for #{name}: #{e.message}"
|
||||||
|
end
|
||||||
|
|
||||||
|
# @deprecated — use MethodSource::CodeHelpers#complete_expression?
|
||||||
|
def self.valid_expression?(str)
|
||||||
|
complete_expression?(str)
|
||||||
|
rescue SyntaxError
|
||||||
|
false
|
||||||
|
end
|
||||||
|
|
||||||
|
# @deprecated — use MethodSource::CodeHelpers#expression_at
|
||||||
|
def self.extract_code(source_location)
|
||||||
|
source_helper(source_location)
|
||||||
|
end
|
||||||
|
|
||||||
|
# This module is to be included by `Method` and `UnboundMethod` and
|
||||||
|
# provides the `#source` functionality
|
||||||
|
module MethodExtensions
|
||||||
|
|
||||||
|
# We use the included hook to patch Method#source on rubinius.
|
||||||
|
# We need to use the included hook as Rubinius defines a `source`
|
||||||
|
# on Method so including a module will have no effect (as it's
|
||||||
|
# higher up the MRO).
|
||||||
|
# @param [Class] klass The class that includes the module.
|
||||||
|
def self.included(klass)
|
||||||
|
if klass.method_defined?(:source) && Object.const_defined?(:RUBY_ENGINE) &&
|
||||||
|
RUBY_ENGINE =~ /rbx/
|
||||||
|
|
||||||
|
klass.class_eval do
|
||||||
|
orig_source = instance_method(:source)
|
||||||
|
|
||||||
|
define_method(:source) do
|
||||||
|
begin
|
||||||
|
super
|
||||||
|
rescue
|
||||||
|
orig_source.bind(self).call
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
# Return the sourcecode for the method as a string
|
||||||
|
# @return [String] The method sourcecode as a string
|
||||||
|
# @raise SourceNotFoundException
|
||||||
|
#
|
||||||
|
# @example
|
||||||
|
# Set.instance_method(:clear).source.display
|
||||||
|
# =>
|
||||||
|
# def clear
|
||||||
|
# @hash.clear
|
||||||
|
# self
|
||||||
|
# end
|
||||||
|
def source
|
||||||
|
MethodSource.source_helper(source_location, defined?(name) ? name : inspect)
|
||||||
|
end
|
||||||
|
|
||||||
|
# Return the comments associated with the method as a string.
|
||||||
|
# @return [String] The method's comments as a string
|
||||||
|
# @raise SourceNotFoundException
|
||||||
|
#
|
||||||
|
# @example
|
||||||
|
# Set.instance_method(:clear).comment.display
|
||||||
|
# =>
|
||||||
|
# # Removes all elements and returns self.
|
||||||
|
def comment
|
||||||
|
MethodSource.comment_helper(source_location, defined?(name) ? name : inspect)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
class Method
|
||||||
|
include MethodSource::SourceLocation::MethodExtensions
|
||||||
|
include MethodSource::MethodExtensions
|
||||||
|
end
|
||||||
|
|
||||||
|
class UnboundMethod
|
||||||
|
include MethodSource::SourceLocation::UnboundMethodExtensions
|
||||||
|
include MethodSource::MethodExtensions
|
||||||
|
end
|
||||||
|
|
||||||
|
class Proc
|
||||||
|
include MethodSource::SourceLocation::ProcExtensions
|
||||||
|
include MethodSource::MethodExtensions
|
||||||
|
end
|
||||||
|
|
||||||
154
Library/Homebrew/vendor/bundle/ruby/2.6.0/gems/method_source-1.0.0/lib/method_source/code_helpers.rb
vendored
Normal file
154
Library/Homebrew/vendor/bundle/ruby/2.6.0/gems/method_source-1.0.0/lib/method_source/code_helpers.rb
vendored
Normal file
@ -0,0 +1,154 @@
|
|||||||
|
module MethodSource
|
||||||
|
|
||||||
|
module CodeHelpers
|
||||||
|
# Retrieve the first expression starting on the given line of the given file.
|
||||||
|
#
|
||||||
|
# This is useful to get module or method source code.
|
||||||
|
#
|
||||||
|
# @param [Array<String>, File, String] file The file to parse, either as a File or as
|
||||||
|
# @param [Integer] line_number The line number at which to look.
|
||||||
|
# NOTE: The first line in a file is
|
||||||
|
# line 1!
|
||||||
|
# @param [Hash] options The optional configuration parameters.
|
||||||
|
# @option options [Boolean] :strict If set to true, then only completely
|
||||||
|
# valid expressions are returned. Otherwise heuristics are used to extract
|
||||||
|
# expressions that may have been valid inside an eval.
|
||||||
|
# @option options [Integer] :consume A number of lines to automatically
|
||||||
|
# consume (add to the expression buffer) without checking for validity.
|
||||||
|
# @return [String] The first complete expression
|
||||||
|
# @raise [SyntaxError] If the first complete expression can't be identified
|
||||||
|
def expression_at(file, line_number, options={})
|
||||||
|
options = {
|
||||||
|
:strict => false,
|
||||||
|
:consume => 0
|
||||||
|
}.merge!(options)
|
||||||
|
|
||||||
|
lines = file.is_a?(Array) ? file : file.each_line.to_a
|
||||||
|
|
||||||
|
relevant_lines = lines[(line_number - 1)..-1] || []
|
||||||
|
|
||||||
|
extract_first_expression(relevant_lines, options[:consume])
|
||||||
|
rescue SyntaxError => e
|
||||||
|
raise if options[:strict]
|
||||||
|
|
||||||
|
begin
|
||||||
|
extract_first_expression(relevant_lines) do |code|
|
||||||
|
code.gsub(/\#\{.*?\}/, "temp")
|
||||||
|
end
|
||||||
|
rescue SyntaxError
|
||||||
|
raise e
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
# Retrieve the comment describing the expression on the given line of the given file.
|
||||||
|
#
|
||||||
|
# This is useful to get module or method documentation.
|
||||||
|
#
|
||||||
|
# @param [Array<String>, File, String] file The file to parse, either as a File or as
|
||||||
|
# a String or an Array of lines.
|
||||||
|
# @param [Integer] line_number The line number at which to look.
|
||||||
|
# NOTE: The first line in a file is line 1!
|
||||||
|
# @return [String] The comment
|
||||||
|
def comment_describing(file, line_number)
|
||||||
|
lines = file.is_a?(Array) ? file : file.each_line.to_a
|
||||||
|
|
||||||
|
extract_last_comment(lines[0..(line_number - 2)])
|
||||||
|
end
|
||||||
|
|
||||||
|
# Determine if a string of code is a complete Ruby expression.
|
||||||
|
# @param [String] code The code to validate.
|
||||||
|
# @return [Boolean] Whether or not the code is a complete Ruby expression.
|
||||||
|
# @raise [SyntaxError] Any SyntaxError that does not represent incompleteness.
|
||||||
|
# @example
|
||||||
|
# complete_expression?("class Hello") #=> false
|
||||||
|
# complete_expression?("class Hello; end") #=> true
|
||||||
|
# complete_expression?("class 123") #=> SyntaxError: unexpected tINTEGER
|
||||||
|
def complete_expression?(str)
|
||||||
|
old_verbose = $VERBOSE
|
||||||
|
$VERBOSE = nil
|
||||||
|
|
||||||
|
catch(:valid) do
|
||||||
|
eval("BEGIN{throw :valid}\n#{str}")
|
||||||
|
end
|
||||||
|
|
||||||
|
# Assert that a line which ends with a , or \ is incomplete.
|
||||||
|
str !~ /[,\\]\s*\z/
|
||||||
|
rescue IncompleteExpression
|
||||||
|
false
|
||||||
|
ensure
|
||||||
|
$VERBOSE = old_verbose
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
# Get the first expression from the input.
|
||||||
|
#
|
||||||
|
# @param [Array<String>] lines
|
||||||
|
# @param [Integer] consume A number of lines to automatically
|
||||||
|
# consume (add to the expression buffer) without checking for validity.
|
||||||
|
# @yield a clean-up function to run before checking for complete_expression
|
||||||
|
# @return [String] a valid ruby expression
|
||||||
|
# @raise [SyntaxError]
|
||||||
|
def extract_first_expression(lines, consume=0, &block)
|
||||||
|
code = consume.zero? ? "" : lines.slice!(0..(consume - 1)).join
|
||||||
|
|
||||||
|
lines.each do |v|
|
||||||
|
code << v
|
||||||
|
return code if complete_expression?(block ? block.call(code) : code)
|
||||||
|
end
|
||||||
|
raise SyntaxError, "unexpected $end"
|
||||||
|
end
|
||||||
|
|
||||||
|
# Get the last comment from the input.
|
||||||
|
#
|
||||||
|
# @param [Array<String>] lines
|
||||||
|
# @return [String]
|
||||||
|
def extract_last_comment(lines)
|
||||||
|
buffer = ""
|
||||||
|
|
||||||
|
lines.each do |line|
|
||||||
|
# Add any line that is a valid ruby comment,
|
||||||
|
# but clear as soon as we hit a non comment line.
|
||||||
|
if (line =~ /^\s*#/) || (line =~ /^\s*$/)
|
||||||
|
buffer << line.lstrip
|
||||||
|
else
|
||||||
|
buffer.replace("")
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
buffer
|
||||||
|
end
|
||||||
|
|
||||||
|
# An exception matcher that matches only subsets of SyntaxErrors that can be
|
||||||
|
# fixed by adding more input to the buffer.
|
||||||
|
module IncompleteExpression
|
||||||
|
GENERIC_REGEXPS = [
|
||||||
|
/unexpected (\$end|end-of-file|end-of-input|END_OF_FILE)/, # mri, jruby, ruby-2.0, ironruby
|
||||||
|
/embedded document meets end of file/, # =begin
|
||||||
|
/unterminated (quoted string|string|regexp|list) meets end of file/, # "quoted string" is ironruby
|
||||||
|
/can't find string ".*" anywhere before EOF/, # rbx and jruby
|
||||||
|
/missing 'end' for/, /expecting kWHEN/ # rbx
|
||||||
|
]
|
||||||
|
|
||||||
|
RBX_ONLY_REGEXPS = [
|
||||||
|
/expecting '[})\]]'(?:$|:)/, /expecting keyword_end/
|
||||||
|
]
|
||||||
|
|
||||||
|
def self.===(ex)
|
||||||
|
return false unless SyntaxError === ex
|
||||||
|
case ex.message
|
||||||
|
when *GENERIC_REGEXPS
|
||||||
|
true
|
||||||
|
when *RBX_ONLY_REGEXPS
|
||||||
|
rbx?
|
||||||
|
else
|
||||||
|
false
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def self.rbx?
|
||||||
|
RbConfig::CONFIG['ruby_install_name'] == 'rbx'
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
@ -0,0 +1,138 @@
|
|||||||
|
module MethodSource
|
||||||
|
module ReeSourceLocation
|
||||||
|
# Ruby enterprise edition provides all the information that's
|
||||||
|
# needed, in a slightly different way.
|
||||||
|
def source_location
|
||||||
|
[__file__, __line__] rescue nil
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
module SourceLocation
|
||||||
|
module MethodExtensions
|
||||||
|
if Proc.method_defined? :__file__
|
||||||
|
include ReeSourceLocation
|
||||||
|
|
||||||
|
elsif defined?(RUBY_ENGINE) && RUBY_ENGINE =~ /jruby/
|
||||||
|
require 'java'
|
||||||
|
|
||||||
|
# JRuby version source_location hack
|
||||||
|
# @return [Array] A two element array containing the source location of the method
|
||||||
|
def source_location
|
||||||
|
to_java.source_location(Thread.current.to_java.getContext())
|
||||||
|
end
|
||||||
|
else
|
||||||
|
|
||||||
|
|
||||||
|
def trace_func(event, file, line, id, binding, classname)
|
||||||
|
return unless event == 'call'
|
||||||
|
set_trace_func nil
|
||||||
|
|
||||||
|
@file, @line = file, line
|
||||||
|
raise :found
|
||||||
|
end
|
||||||
|
|
||||||
|
private :trace_func
|
||||||
|
|
||||||
|
# Return the source location of a method for Ruby 1.8.
|
||||||
|
# @return [Array] A two element array. First element is the
|
||||||
|
# file, second element is the line in the file where the
|
||||||
|
# method definition is found.
|
||||||
|
def source_location
|
||||||
|
if @file.nil?
|
||||||
|
args =[*(1..(arity<-1 ? -arity-1 : arity ))]
|
||||||
|
|
||||||
|
set_trace_func method(:trace_func).to_proc
|
||||||
|
call(*args) rescue nil
|
||||||
|
set_trace_func nil
|
||||||
|
@file = File.expand_path(@file) if @file && File.exist?(File.expand_path(@file))
|
||||||
|
end
|
||||||
|
[@file, @line] if @file
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
module ProcExtensions
|
||||||
|
if Proc.method_defined? :__file__
|
||||||
|
include ReeSourceLocation
|
||||||
|
|
||||||
|
elsif defined?(RUBY_ENGINE) && RUBY_ENGINE =~ /rbx/
|
||||||
|
|
||||||
|
# Return the source location for a Proc (Rubinius only)
|
||||||
|
# @return [Array] A two element array. First element is the
|
||||||
|
# file, second element is the line in the file where the
|
||||||
|
# proc definition is found.
|
||||||
|
def source_location
|
||||||
|
[block.file.to_s, block.line]
|
||||||
|
end
|
||||||
|
else
|
||||||
|
|
||||||
|
# Return the source location for a Proc (in implementations
|
||||||
|
# without Proc#source_location)
|
||||||
|
# @return [Array] A two element array. First element is the
|
||||||
|
# file, second element is the line in the file where the
|
||||||
|
# proc definition is found.
|
||||||
|
def source_location
|
||||||
|
self.to_s =~ /@(.*):(\d+)/
|
||||||
|
[$1, $2.to_i]
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
module UnboundMethodExtensions
|
||||||
|
if Proc.method_defined? :__file__
|
||||||
|
include ReeSourceLocation
|
||||||
|
|
||||||
|
elsif defined?(RUBY_ENGINE) && RUBY_ENGINE =~ /jruby/
|
||||||
|
require 'java'
|
||||||
|
|
||||||
|
# JRuby version source_location hack
|
||||||
|
# @return [Array] A two element array containing the source location of the method
|
||||||
|
def source_location
|
||||||
|
to_java.source_location(Thread.current.to_java.getContext())
|
||||||
|
end
|
||||||
|
|
||||||
|
else
|
||||||
|
|
||||||
|
|
||||||
|
# Return the source location of an instance method for Ruby 1.8.
|
||||||
|
# @return [Array] A two element array. First element is the
|
||||||
|
# file, second element is the line in the file where the
|
||||||
|
# method definition is found.
|
||||||
|
def source_location
|
||||||
|
klass = case owner
|
||||||
|
when Class
|
||||||
|
owner
|
||||||
|
when Module
|
||||||
|
method_owner = owner
|
||||||
|
Class.new { include(method_owner) }
|
||||||
|
end
|
||||||
|
|
||||||
|
# deal with immediate values
|
||||||
|
case
|
||||||
|
when klass == Symbol
|
||||||
|
return :a.method(name).source_location
|
||||||
|
when klass == Integer
|
||||||
|
return 0.method(name).source_location
|
||||||
|
when klass == TrueClass
|
||||||
|
return true.method(name).source_location
|
||||||
|
when klass == FalseClass
|
||||||
|
return false.method(name).source_location
|
||||||
|
when klass == NilClass
|
||||||
|
return nil.method(name).source_location
|
||||||
|
end
|
||||||
|
|
||||||
|
begin
|
||||||
|
Object.instance_method(:method).bind(klass.allocate).call(name).source_location
|
||||||
|
rescue TypeError
|
||||||
|
|
||||||
|
# Assume we are dealing with a Singleton Class:
|
||||||
|
# 1. Get the instance object
|
||||||
|
# 2. Forward the source_location lookup to the instance
|
||||||
|
instance ||= ObjectSpace.each_object(owner).first
|
||||||
|
Object.instance_method(:method).bind(instance).call(name).source_location
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
@ -0,0 +1,3 @@
|
|||||||
|
module MethodSource
|
||||||
|
VERSION = '1.0.0'.freeze
|
||||||
|
end
|
||||||
Loading…
x
Reference in New Issue
Block a user