summaryrefslogtreecommitdiff
path: root/lib/irb/cmd/show_source.rb
diff options
context:
space:
mode:
authorStan Lo <stan001212@gmail.com>2023-07-31 20:57:29 +0100
committergit <svn-admin@ruby-lang.org>2023-07-31 19:57:36 +0000
commit5d78ec8a94b7d44b5b152f4fad7e419a2436fe70 (patch)
treec685f3b06492636a445b7f70830c3dc1b373754f /lib/irb/cmd/show_source.rb
parentc01b17f7fc74f005f23da0405eb8bad08a269a18 (diff)
[ruby/irb] Decouple `edit` and `show_source` commands
(https://github.com/ruby/irb/pull/658) * Decouple `edit` command from `show_source` 2 commands should not depend on each other. If `edit` command also needs to find a source, the source finding logic should be extracted into a separate class. * Return nil if is not an actual file path * Refactor SourceFinder https://github.com/ruby/irb/commit/9790517a0c
Diffstat (limited to 'lib/irb/cmd/show_source.rb')
-rw-r--r--lib/irb/cmd/show_source.rb65
1 files changed, 4 insertions, 61 deletions
diff --git a/lib/irb/cmd/show_source.rb b/lib/irb/cmd/show_source.rb
index f172123876..10463ebf08 100644
--- a/lib/irb/cmd/show_source.rb
+++ b/lib/irb/cmd/show_source.rb
@@ -1,12 +1,10 @@
# frozen_string_literal: true
require_relative "nop"
+require_relative "../source_finder"
require_relative "../color"
-require_relative "../ruby-lex"
module IRB
- # :stopdoc:
-
module ExtendCommand
class ShowSource < Nop
category "Context"
@@ -21,51 +19,6 @@ module IRB
args.strip.dump
end
end
-
- def find_source(str, irb_context)
- case str
- when /\A[A-Z]\w*(::[A-Z]\w*)*\z/ # Const::Name
- eval(str, irb_context.workspace.binding) # trigger autoload
- base = irb_context.workspace.binding.receiver.yield_self { |r| r.is_a?(Module) ? r : Object }
- file, line = base.const_source_location(str)
- when /\A(?<owner>[A-Z]\w*(::[A-Z]\w*)*)#(?<method>[^ :.]+)\z/ # Class#method
- owner = eval(Regexp.last_match[:owner], irb_context.workspace.binding)
- method = Regexp.last_match[:method]
- if owner.respond_to?(:instance_method)
- methods = owner.instance_methods + owner.private_instance_methods
- file, line = owner.instance_method(method).source_location if methods.include?(method.to_sym)
- end
- when /\A((?<receiver>.+)(\.|::))?(?<method>[^ :.]+)\z/ # method, receiver.method, receiver::method
- receiver = eval(Regexp.last_match[:receiver] || 'self', irb_context.workspace.binding)
- method = Regexp.last_match[:method]
- file, line = receiver.method(method).source_location if receiver.respond_to?(method, true)
- end
- if file && line
- Source.new(file: file, first_line: line, last_line: find_end(file, line, irb_context))
- end
- end
-
- private
-
- def find_end(file, first_line, irb_context)
- return first_line unless File.exist?(file)
- lex = RubyLex.new(irb_context)
- lines = File.read(file).lines[(first_line - 1)..-1]
- tokens = RubyLex.ripper_lex_without_warning(lines.join)
- prev_tokens = []
-
- # chunk with line number
- tokens.chunk { |tok| tok.pos[0] }.each do |lnum, chunk|
- code = lines[0..lnum].join
- prev_tokens.concat chunk
- continue = lex.should_continue?(prev_tokens)
- syntax = lex.check_code_syntax(code)
- if !continue && syntax == :valid
- return first_line + lnum
- end
- end
- first_line
- end
end
def execute(str = nil)
@@ -74,8 +27,9 @@ module IRB
return
end
- source = self.class.find_source(str, @irb_context)
- if source && File.exist?(source.file)
+ source = SourceFinder.new(@irb_context).find_source(str)
+
+ if source
show_source(source)
else
puts "Error: Couldn't locate a definition for #{str}"
@@ -85,7 +39,6 @@ module IRB
private
- # @param [IRB::ExtendCommand::ShowSource::Source] source
def show_source(source)
puts
puts "#{bold("From")}: #{source.file}:#{source.first_line}"
@@ -98,16 +51,6 @@ module IRB
def bold(str)
Color.colorize(str, [:BOLD])
end
-
- Source = Struct.new(
- :file, # @param [String] - file name
- :first_line, # @param [String] - first line
- :last_line, # @param [String] - last line
- keyword_init: true,
- )
- private_constant :Source
end
end
-
- # :startdoc:
end