Skip to content

Commit

Permalink
Decouple edit and show_source commands (#658)
Browse files Browse the repository at this point in the history
* 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
  • Loading branch information
st0012 authored Jul 31, 2023
1 parent 82d1687 commit 9790517
Show file tree
Hide file tree
Showing 3 changed files with 71 additions and 65 deletions.
7 changes: 3 additions & 4 deletions lib/irb/cmd/edit.rb
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
require 'shellwords'
require_relative "nop"
require_relative "../source_finder"

module IRB
# :stopdoc:
Expand Down Expand Up @@ -28,17 +29,15 @@ def execute(*args)
end

if !File.exist?(path)
require_relative "show_source"

source =
begin
ShowSource.find_source(path, @irb_context)
SourceFinder.new(@irb_context).find_source(path)
rescue NameError
# if user enters a path that doesn't exist, it'll cause NameError when passed here because find_source would try to evaluate it as well
# in this case, we should just ignore the error
end

if source && File.exist?(source.file)
if source
path = source.file
else
puts "Can not find file: #{path}"
Expand Down
65 changes: 4 additions & 61 deletions lib/irb/cmd/show_source.rb
Original file line number Diff line number Diff line change
@@ -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"
Expand All @@ -21,51 +19,6 @@ def transform_args(args)
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)
Expand All @@ -74,8 +27,9 @@ def execute(str = nil)
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}"
Expand All @@ -85,7 +39,6 @@ def execute(str = nil)

private

# @param [IRB::ExtendCommand::ShowSource::Source] source
def show_source(source)
puts
puts "#{bold("From")}: #{source.file}:#{source.first_line}"
Expand All @@ -98,16 +51,6 @@ def show_source(source)
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
64 changes: 64 additions & 0 deletions lib/irb/source_finder.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,64 @@
# frozen_string_literal: true

require_relative "ruby-lex"

module IRB
class SourceFinder
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

def initialize(irb_context)
@irb_context = irb_context
end

def find_source(signature)
context_binding = @irb_context.workspace.binding
case signature
when /\A[A-Z]\w*(::[A-Z]\w*)*\z/ # Const::Name
eval(signature, context_binding) # trigger autoload
base = context_binding.receiver.yield_self { |r| r.is_a?(Module) ? r : Object }
file, line = base.const_source_location(signature)
when /\A(?<owner>[A-Z]\w*(::[A-Z]\w*)*)#(?<method>[^ :.]+)\z/ # Class#method
owner = eval(Regexp.last_match[:owner], context_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', context_binding)
method = Regexp.last_match[:method]
file, line = receiver.method(method).source_location if receiver.respond_to?(method, true)
end
if file && line && File.exist?(file)
Source.new(file: file, first_line: line, last_line: find_end(file, line))
end
end

private

def find_end(file, first_line)
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
end

0 comments on commit 9790517

Please sign in to comment.