2019-04-26 05:28:54 -04:00
|
|
|
# frozen_string_literal: false
|
2019-11-28 01:32:21 -05:00
|
|
|
require "test/unit"
|
2022-09-21 09:53:06 -04:00
|
|
|
require "pathname"
|
2019-11-28 01:32:21 -05:00
|
|
|
require "irb"
|
2019-04-26 05:28:54 -04:00
|
|
|
|
|
|
|
module TestIRB
|
|
|
|
class TestCompletion < Test::Unit::TestCase
|
2022-09-19 11:14:03 -04:00
|
|
|
def setup
|
|
|
|
# make sure require completion candidates are not cached
|
|
|
|
IRB::InputCompletor.class_variable_set(:@@files_from_load_path, nil)
|
|
|
|
end
|
|
|
|
|
2019-04-26 05:28:54 -04:00
|
|
|
def test_nonstring_module_name
|
|
|
|
begin
|
|
|
|
require "irb/completion"
|
|
|
|
bug5938 = '[ruby-core:42244]'
|
2019-06-12 11:29:45 -04:00
|
|
|
bundle_exec = ENV.key?('BUNDLE_GEMFILE') ? ['-rbundler/setup'] : []
|
|
|
|
cmds = bundle_exec + %W[-W0 -rirb -rirb/completion -e IRB.setup(__FILE__)
|
2019-04-26 05:28:54 -04:00
|
|
|
-e IRB.conf[:MAIN_CONTEXT]=IRB::Irb.new.context
|
|
|
|
-e module\sFoo;def\sself.name;//;end;end
|
|
|
|
-e IRB::InputCompletor::CompletionProc.call("[1].first.")
|
|
|
|
-- -f --]
|
|
|
|
status = assert_in_out_err(cmds, "", //, [], bug5938)
|
|
|
|
assert(status.success?, bug5938)
|
|
|
|
rescue LoadError
|
2021-05-31 01:56:50 -04:00
|
|
|
pend "cannot load irb/completion"
|
2019-04-26 05:28:54 -04:00
|
|
|
end
|
|
|
|
end
|
2019-11-28 01:15:41 -05:00
|
|
|
|
2022-10-06 06:49:51 -04:00
|
|
|
class TestMethodCompletion < TestCompletion
|
|
|
|
def test_complete_string
|
|
|
|
assert_include(IRB::InputCompletor.retrieve_completion_data("'foo'.up", bind: binding), "'foo'.upcase")
|
2022-10-06 07:30:18 -04:00
|
|
|
assert_equal("String.upcase", IRB::InputCompletor.retrieve_completion_data("'foo'.upcase", bind: binding, doc_namespace: true))
|
2022-10-06 06:49:51 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def test_complete_regexp
|
|
|
|
assert_include(IRB::InputCompletor.retrieve_completion_data("/foo/.ma", bind: binding), "/foo/.match")
|
2022-10-06 07:30:18 -04:00
|
|
|
assert_equal("Regexp.match", IRB::InputCompletor.retrieve_completion_data("/foo/.match", bind: binding, doc_namespace: true))
|
2022-10-06 06:49:51 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def test_complete_array
|
|
|
|
assert_include(IRB::InputCompletor.retrieve_completion_data("[].an", bind: binding), "[].any?")
|
2022-10-06 07:30:18 -04:00
|
|
|
assert_equal("Array.any?", IRB::InputCompletor.retrieve_completion_data("[].any?", bind: binding, doc_namespace: true))
|
2022-10-06 06:49:51 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def test_complete_hash_and_proc
|
|
|
|
# hash
|
|
|
|
assert_include(IRB::InputCompletor.retrieve_completion_data("{}.an", bind: binding), "{}.any?")
|
2022-10-06 07:30:18 -04:00
|
|
|
assert_equal(["Proc.any?", "Hash.any?"], IRB::InputCompletor.retrieve_completion_data("{}.any?", bind: binding, doc_namespace: true))
|
2022-10-06 06:49:51 -04:00
|
|
|
|
|
|
|
# proc
|
|
|
|
assert_include(IRB::InputCompletor.retrieve_completion_data("{}.bin", bind: binding), "{}.binding")
|
2022-10-06 07:30:18 -04:00
|
|
|
assert_equal(["Proc.binding", "Hash.binding"], IRB::InputCompletor.retrieve_completion_data("{}.binding", bind: binding, doc_namespace: true))
|
2022-10-06 06:49:51 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def test_complete_numeric
|
|
|
|
assert_include(IRB::InputCompletor.retrieve_completion_data("1.positi", bind: binding), "1.positive?")
|
2022-10-06 07:30:18 -04:00
|
|
|
assert_equal("Integer.positive?", IRB::InputCompletor.retrieve_completion_data("1.positive?", bind: binding, doc_namespace: true))
|
2022-10-06 06:49:51 -04:00
|
|
|
|
|
|
|
assert_include(IRB::InputCompletor.retrieve_completion_data("1r.positi", bind: binding), "1r.positive?")
|
2022-10-06 07:30:18 -04:00
|
|
|
assert_equal("Rational.positive?", IRB::InputCompletor.retrieve_completion_data("1r.positive?", bind: binding, doc_namespace: true))
|
2022-10-06 06:49:51 -04:00
|
|
|
|
|
|
|
assert_include(IRB::InputCompletor.retrieve_completion_data("0xFFFF.positi", bind: binding), "0xFFFF.positive?")
|
2022-10-06 07:30:18 -04:00
|
|
|
assert_equal("Integer.positive?", IRB::InputCompletor.retrieve_completion_data("0xFFFF.positive?", bind: binding, doc_namespace: true))
|
2022-10-06 06:49:51 -04:00
|
|
|
|
|
|
|
assert_empty(IRB::InputCompletor.retrieve_completion_data("1i.positi", bind: binding))
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_complete_symbol
|
|
|
|
assert_include(IRB::InputCompletor.retrieve_completion_data(":foo.to_p", bind: binding), ":foo.to_proc")
|
2022-10-06 07:30:18 -04:00
|
|
|
assert_equal("Symbol.to_proc", IRB::InputCompletor.retrieve_completion_data(":foo.to_proc", bind: binding, doc_namespace: true))
|
2022-10-06 06:49:51 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def test_complete_class
|
|
|
|
assert_include(IRB::InputCompletor.retrieve_completion_data("String.ne", bind: binding), "String.new")
|
2022-10-06 07:30:18 -04:00
|
|
|
assert_equal("String.new", IRB::InputCompletor.retrieve_completion_data("String.new", bind: binding, doc_namespace: true))
|
2022-10-06 06:49:51 -04:00
|
|
|
end
|
2019-11-28 01:15:41 -05:00
|
|
|
end
|
2020-02-10 09:15:13 -05:00
|
|
|
|
2022-10-06 06:59:49 -04:00
|
|
|
class TestRequireComepletion < TestCompletion
|
|
|
|
def test_complete_require
|
|
|
|
candidates = IRB::InputCompletor::CompletionProc.("'irb", "require ", "")
|
|
|
|
%w['irb/init 'irb/ruby-lex].each do |word|
|
|
|
|
assert_include candidates, word
|
|
|
|
end
|
|
|
|
# Test cache
|
|
|
|
candidates = IRB::InputCompletor::CompletionProc.("'irb", "require ", "")
|
|
|
|
%w['irb/init 'irb/ruby-lex].each do |word|
|
|
|
|
assert_include candidates, word
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_complete_require_with_pathname_in_load_path
|
|
|
|
temp_dir = Dir.mktmpdir
|
|
|
|
File.write(File.join(temp_dir, "foo.rb"), "test")
|
|
|
|
test_path = Pathname.new(temp_dir)
|
|
|
|
$LOAD_PATH << test_path
|
|
|
|
|
|
|
|
candidates = IRB::InputCompletor::CompletionProc.("'foo", "require ", "")
|
|
|
|
assert_include candidates, "'foo"
|
|
|
|
ensure
|
|
|
|
$LOAD_PATH.pop if test_path
|
|
|
|
FileUtils.remove_entry(temp_dir) if temp_dir
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_complete_require_with_string_convertable_in_load_path
|
|
|
|
temp_dir = Dir.mktmpdir
|
|
|
|
File.write(File.join(temp_dir, "foo.rb"), "test")
|
|
|
|
object = Object.new
|
|
|
|
object.define_singleton_method(:to_s) { temp_dir }
|
|
|
|
$LOAD_PATH << object
|
|
|
|
|
|
|
|
candidates = IRB::InputCompletor::CompletionProc.("'foo", "require ", "")
|
|
|
|
assert_include candidates, "'foo"
|
|
|
|
ensure
|
|
|
|
$LOAD_PATH.pop if object
|
|
|
|
FileUtils.remove_entry(temp_dir) if temp_dir
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_complete_require_with_malformed_object_in_load_path
|
|
|
|
object = Object.new
|
|
|
|
def object.to_s; raise; end
|
|
|
|
$LOAD_PATH << object
|
|
|
|
|
|
|
|
assert_nothing_raised do
|
|
|
|
IRB::InputCompletor::CompletionProc.("'foo", "require ", "")
|
|
|
|
end
|
|
|
|
ensure
|
|
|
|
$LOAD_PATH.pop if object
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_complete_require_library_name_first
|
|
|
|
pend 'Need to use virtual library paths'
|
|
|
|
candidates = IRB::InputCompletor::CompletionProc.("'csv", "require ", "")
|
|
|
|
assert_equal "'csv", candidates.first
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_complete_require_relative
|
|
|
|
candidates = Dir.chdir(__dir__ + "/../..") do
|
|
|
|
IRB::InputCompletor::CompletionProc.("'lib/irb", "require_relative ", "")
|
|
|
|
end
|
|
|
|
%w['lib/irb/init 'lib/irb/ruby-lex].each do |word|
|
|
|
|
assert_include candidates, word
|
|
|
|
end
|
|
|
|
# Test cache
|
|
|
|
candidates = Dir.chdir(__dir__ + "/../..") do
|
|
|
|
IRB::InputCompletor::CompletionProc.("'lib/irb", "require_relative ", "")
|
|
|
|
end
|
|
|
|
%w['lib/irb/init 'lib/irb/ruby-lex].each do |word|
|
|
|
|
assert_include candidates, word
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
class TestVariableCompletion < TestCompletion
|
|
|
|
def test_complete_variable
|
|
|
|
# Bug fix issues https://github.com/ruby/irb/issues/368
|
|
|
|
# Variables other than `str_example` and `@str_example` are defined to ensure that irb completion does not cause unintended behavior
|
|
|
|
str_example = ''
|
|
|
|
@str_example = ''
|
|
|
|
private_methods = ''
|
|
|
|
methods = ''
|
|
|
|
global_variables = ''
|
|
|
|
local_variables = ''
|
|
|
|
instance_variables = ''
|
|
|
|
|
|
|
|
# suppress "assigned but unused variable" warning
|
|
|
|
str_example.clear
|
|
|
|
@str_example.clear
|
|
|
|
private_methods.clear
|
|
|
|
methods.clear
|
|
|
|
global_variables.clear
|
|
|
|
local_variables.clear
|
|
|
|
instance_variables.clear
|
|
|
|
|
|
|
|
assert_include(IRB::InputCompletor.retrieve_completion_data("str_examp", bind: binding), "str_example")
|
2022-10-06 07:30:18 -04:00
|
|
|
assert_equal("String", IRB::InputCompletor.retrieve_completion_data("str_example", bind: binding, doc_namespace: true))
|
|
|
|
assert_equal("String.to_s", IRB::InputCompletor.retrieve_completion_data("str_example.to_s", bind: binding, doc_namespace: true))
|
2022-10-06 06:59:49 -04:00
|
|
|
|
|
|
|
assert_include(IRB::InputCompletor.retrieve_completion_data("@str_examp", bind: binding), "@str_example")
|
2022-10-06 07:30:18 -04:00
|
|
|
assert_equal("String", IRB::InputCompletor.retrieve_completion_data("@str_example", bind: binding, doc_namespace: true))
|
|
|
|
assert_equal("String.to_s", IRB::InputCompletor.retrieve_completion_data("@str_example.to_s", bind: binding, doc_namespace: true))
|
2022-10-06 06:59:49 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def test_complete_sort_variables
|
|
|
|
xzy, xzy_1, xzy2 = '', '', ''
|
|
|
|
|
|
|
|
xzy.clear
|
|
|
|
xzy_1.clear
|
|
|
|
xzy2.clear
|
|
|
|
|
|
|
|
candidates = IRB::InputCompletor.retrieve_completion_data("xz", bind: binding, doc_namespace: false)
|
2022-10-06 07:30:18 -04:00
|
|
|
assert_equal(%w[xzy xzy2 xzy_1], candidates)
|
2022-10-06 06:59:49 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2022-10-06 07:27:16 -04:00
|
|
|
class TestConstantCompletion < TestCompletion
|
|
|
|
class Foo
|
|
|
|
B1 = 1
|
|
|
|
B2 = 2
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_complete_constants
|
|
|
|
assert_equal(["Foo"], IRB::InputCompletor.retrieve_completion_data("Fo", bind: binding))
|
|
|
|
assert_equal(["Foo::B1", "Foo::B2"], IRB::InputCompletor.retrieve_completion_data("Foo::B", bind: binding))
|
|
|
|
assert_equal(["Foo::B1.positive?"], IRB::InputCompletor.retrieve_completion_data("Foo::B1.pos", bind: binding))
|
|
|
|
|
|
|
|
assert_equal(["::Forwardable"], IRB::InputCompletor.retrieve_completion_data("::Fo", bind: binding))
|
|
|
|
assert_equal("Forwardable", IRB::InputCompletor.retrieve_completion_data("::Forwardable", bind: binding, doc_namespace: true))
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-02-12 05:40:10 -05:00
|
|
|
def test_complete_symbol
|
2021-09-23 03:28:28 -04:00
|
|
|
%w"UTF-16LE UTF-7".each do |enc|
|
|
|
|
"K".force_encoding(enc).to_sym
|
|
|
|
rescue
|
|
|
|
end
|
2020-02-13 10:07:28 -05:00
|
|
|
_ = :aiueo
|
2020-02-12 05:40:10 -05:00
|
|
|
assert_include(IRB::InputCompletor.retrieve_completion_data(":a", bind: binding), ":aiueo")
|
|
|
|
assert_empty(IRB::InputCompletor.retrieve_completion_data(":irb_unknown_symbol_abcdefg", bind: binding))
|
|
|
|
end
|
|
|
|
|
2021-09-10 15:30:29 -04:00
|
|
|
def test_complete_invalid_three_colons
|
|
|
|
assert_empty(IRB::InputCompletor.retrieve_completion_data(":::A", bind: binding))
|
|
|
|
assert_empty(IRB::InputCompletor.retrieve_completion_data(":::", bind: binding))
|
|
|
|
end
|
|
|
|
|
2021-10-10 15:30:42 -04:00
|
|
|
def test_complete_absolute_constants_with_special_characters
|
|
|
|
assert_empty(IRB::InputCompletor.retrieve_completion_data("::A:", bind: binding))
|
|
|
|
assert_empty(IRB::InputCompletor.retrieve_completion_data("::A.", bind: binding))
|
|
|
|
assert_empty(IRB::InputCompletor.retrieve_completion_data("::A(", bind: binding))
|
|
|
|
assert_empty(IRB::InputCompletor.retrieve_completion_data("::A)", bind: binding))
|
|
|
|
end
|
|
|
|
|
2020-02-10 09:22:24 -05:00
|
|
|
def test_complete_symbol_failure
|
|
|
|
assert_nil(IRB::InputCompletor::PerfectMatchedProc.(":aiueo", bind: binding))
|
|
|
|
end
|
2020-02-12 05:16:12 -05:00
|
|
|
|
|
|
|
def test_complete_reserved_words
|
|
|
|
candidates = IRB::InputCompletor.retrieve_completion_data("de", bind: binding)
|
|
|
|
%w[def defined?].each do |word|
|
|
|
|
assert_include candidates, word
|
|
|
|
end
|
|
|
|
|
|
|
|
candidates = IRB::InputCompletor.retrieve_completion_data("__", bind: binding)
|
|
|
|
%w[__ENCODING__ __LINE__ __FILE__].each do |word|
|
|
|
|
assert_include candidates, word
|
|
|
|
end
|
|
|
|
end
|
2021-01-06 05:05:46 -05:00
|
|
|
|
2022-10-02 17:55:50 -04:00
|
|
|
def test_complete_methods
|
|
|
|
obj = Object.new
|
|
|
|
obj.singleton_class.class_eval {
|
|
|
|
def public_hoge; end
|
|
|
|
private def private_hoge; end
|
|
|
|
|
|
|
|
# Support for overriding #methods etc.
|
|
|
|
def methods; end
|
|
|
|
def private_methods; end
|
|
|
|
def global_variables; end
|
|
|
|
def local_variables; end
|
|
|
|
def instance_variables; end
|
|
|
|
}
|
|
|
|
bind = obj.instance_exec { binding }
|
|
|
|
|
|
|
|
assert_include(IRB::InputCompletor.retrieve_completion_data("public_hog", bind: bind), "public_hoge")
|
|
|
|
assert_include(IRB::InputCompletor.retrieve_completion_data("public_hoge.to_s", bind: bind), "public_hoge.to_s")
|
|
|
|
assert_include(IRB::InputCompletor.retrieve_completion_data("public_hoge", bind: bind, doc_namespace: true), "public_hoge")
|
|
|
|
|
|
|
|
assert_include(IRB::InputCompletor.retrieve_completion_data("private_hog", bind: bind), "private_hoge")
|
|
|
|
assert_include(IRB::InputCompletor.retrieve_completion_data("private_hoge.to_s", bind: bind), "private_hoge.to_s")
|
|
|
|
assert_include(IRB::InputCompletor.retrieve_completion_data("private_hoge", bind: bind, doc_namespace: true), "private_hoge")
|
2021-09-02 08:34:53 -04:00
|
|
|
end
|
2019-04-26 05:28:54 -04:00
|
|
|
end
|
|
|
|
end
|