mirror of
https://github.com/ruby/ruby.git
synced 2022-11-09 12:17:21 -05:00
f91b1ab33d
Previously, the entire method was not run for Readline 4.3, probably
because it was known to fail. Commit
c754e979d3
removed the check for
Readline 4.3. Other than this one assertion, which also doesn't
work when using Reline, the method runs correctly when using
Readline 4.3.
Fixes [Bug #15853].
724 lines
21 KiB
Ruby
724 lines
21 KiB
Ruby
# frozen_string_literal: false
|
|
require_relative "helper"
|
|
require "test/unit"
|
|
require "tempfile"
|
|
require "timeout"
|
|
|
|
module BasetestReadline
|
|
INPUTRC = "INPUTRC"
|
|
SAVED_ENV = %w[COLUMNS LINES]
|
|
|
|
def setup
|
|
@saved_env = ENV.values_at(*SAVED_ENV)
|
|
@inputrc, ENV[INPUTRC] = ENV[INPUTRC], IO::NULL
|
|
end
|
|
|
|
def teardown
|
|
ENV[INPUTRC] = @inputrc
|
|
Readline.instance_variable_set("@completion_proc", nil)
|
|
begin
|
|
Readline.delete_text
|
|
Readline.point = 0
|
|
rescue NotImplementedError
|
|
end
|
|
Readline.input = nil
|
|
Readline.output = nil
|
|
SAVED_ENV.each_with_index {|k, i| ENV[k] = @saved_env[i] }
|
|
end
|
|
|
|
def test_readline
|
|
skip "Skip Editline" if /EditLine/n.match(Readline::VERSION)
|
|
with_temp_stdio do |stdin, stdout|
|
|
stdin.write("hello\n")
|
|
stdin.close
|
|
stdout.flush
|
|
line = replace_stdio(stdin.path, stdout.path) {
|
|
Readline.readline("> ", true)
|
|
}
|
|
assert_equal("hello", line)
|
|
assert_equal(true, line.tainted?)
|
|
stdout.rewind
|
|
assert_equal("> ", stdout.read(2))
|
|
assert_equal(1, Readline::HISTORY.length)
|
|
assert_equal("hello", Readline::HISTORY[0])
|
|
Thread.start {
|
|
$SAFE = 1
|
|
assert_raise(SecurityError) do
|
|
replace_stdio(stdin.path, stdout.path) do
|
|
Readline.readline("> ".taint)
|
|
end
|
|
end
|
|
}.join
|
|
ensure
|
|
$SAFE = 0
|
|
end
|
|
end
|
|
|
|
# line_buffer
|
|
# point
|
|
def test_line_buffer__point
|
|
skip "Skip Editline" if /EditLine/n.match(Readline::VERSION)
|
|
skip "GNU Readline has special behaviors" if defined?(Reline) and Readline == Reline
|
|
begin
|
|
Readline.line_buffer
|
|
Readline.point
|
|
rescue NotImplementedError
|
|
return
|
|
end
|
|
|
|
with_temp_stdio do |stdin, stdout|
|
|
actual_text = nil
|
|
actual_line_buffer = nil
|
|
actual_point = nil
|
|
Readline.completion_proc = ->(text) {
|
|
actual_text = text
|
|
actual_point = Readline.point
|
|
actual_line_buffer = Readline.line_buffer
|
|
stdin.write(" finish\n")
|
|
stdin.flush
|
|
stdout.flush
|
|
return ["complete"]
|
|
}
|
|
|
|
stdin.write("first second\t")
|
|
stdin.flush
|
|
Readline.completion_append_character = " "
|
|
replace_stdio(stdin.path, stdout.path) {
|
|
Readline.readline("> ", false)
|
|
}
|
|
assert_equal("second", actual_text)
|
|
assert_equal("first second", actual_line_buffer)
|
|
assert_equal(12, actual_point)
|
|
assert_equal("first complete finish", Readline.line_buffer)
|
|
assert_equal(Encoding.find("locale"), Readline.line_buffer.encoding)
|
|
assert_equal(true, Readline.line_buffer.tainted?)
|
|
assert_equal(22, Readline.point)
|
|
|
|
stdin.rewind
|
|
stdout.rewind
|
|
|
|
stdin.write("first second\t")
|
|
stdin.flush
|
|
Readline.completion_append_character = nil
|
|
replace_stdio(stdin.path, stdout.path) {
|
|
Readline.readline("> ", false)
|
|
}
|
|
assert_equal("second", actual_text)
|
|
assert_equal("first second", actual_line_buffer)
|
|
assert_equal(12, actual_point)
|
|
assert_equal("first complete finish", Readline.line_buffer)
|
|
assert_equal(Encoding.find("locale"), Readline.line_buffer.encoding)
|
|
assert_equal(true, Readline.line_buffer.tainted?)
|
|
assert_equal(21, Readline.point)
|
|
end
|
|
end
|
|
|
|
def test_input=
|
|
assert_raise(TypeError) do
|
|
Readline.input = "This is not a file."
|
|
end
|
|
end
|
|
|
|
def test_output=
|
|
assert_raise(TypeError) do
|
|
Readline.output = "This is not a file."
|
|
end
|
|
end
|
|
|
|
def test_completion_proc
|
|
expected = proc { |input| input }
|
|
Readline.completion_proc = expected
|
|
assert_equal(expected, Readline.completion_proc)
|
|
|
|
assert_raise(ArgumentError) do
|
|
Readline.completion_proc = "This does not have call method."
|
|
end
|
|
end
|
|
|
|
def test_completion_case_fold
|
|
expected = [true, false, "string", {"a" => "b"}]
|
|
expected.each do |e|
|
|
Readline.completion_case_fold = e
|
|
assert_equal(e, Readline.completion_case_fold)
|
|
end
|
|
end
|
|
|
|
def test_completion_proc_empty_result
|
|
skip "Skip Editline" if /EditLine/n.match(Readline::VERSION)
|
|
with_temp_stdio do |stdin, stdout|
|
|
stdin.write("first\t")
|
|
stdin.flush
|
|
Readline.completion_proc = ->(text) {[]}
|
|
line1 = line2 = nil
|
|
replace_stdio(stdin.path, stdout.path) {
|
|
assert_nothing_raised(NoMemoryError) {line1 = Readline.readline("> ")}
|
|
stdin.write("\n")
|
|
stdin.flush
|
|
assert_nothing_raised(NoMemoryError) {line2 = Readline.readline("> ")}
|
|
}
|
|
assert_equal("first", line1)
|
|
assert_equal("", line2)
|
|
begin
|
|
assert_equal("", Readline.line_buffer)
|
|
rescue NotimplementedError
|
|
end
|
|
end
|
|
end
|
|
|
|
def test_get_screen_size
|
|
begin
|
|
res = Readline.get_screen_size
|
|
assert(res.is_a?(Array))
|
|
rows, columns = *res
|
|
assert(rows.is_a?(Integer))
|
|
assert(rows >= 0)
|
|
assert(columns.is_a?(Integer))
|
|
assert(columns >= 0)
|
|
rescue NotImplementedError
|
|
end
|
|
end
|
|
|
|
# vi_editing_mode
|
|
# emacs_editing_mode
|
|
def test_editing_mode
|
|
begin
|
|
assert_equal(false, Readline.vi_editing_mode?)
|
|
assert_equal(true, Readline.emacs_editing_mode?)
|
|
|
|
assert_equal(nil, Readline.vi_editing_mode)
|
|
assert_equal(true, Readline.vi_editing_mode?)
|
|
assert_equal(false, Readline.emacs_editing_mode?)
|
|
assert_equal(nil, Readline.vi_editing_mode)
|
|
assert_equal(true, Readline.vi_editing_mode?)
|
|
assert_equal(false, Readline.emacs_editing_mode?)
|
|
|
|
assert_equal(nil, Readline.emacs_editing_mode)
|
|
assert_equal(false, Readline.vi_editing_mode?)
|
|
assert_equal(true, Readline.emacs_editing_mode?)
|
|
assert_equal(nil, Readline.emacs_editing_mode)
|
|
assert_equal(false, Readline.vi_editing_mode?)
|
|
assert_equal(true, Readline.emacs_editing_mode?)
|
|
rescue NotImplementedError
|
|
end
|
|
end
|
|
|
|
def test_completion_append_character
|
|
begin
|
|
enc = get_default_internal_encoding
|
|
data_expected = [
|
|
["x", "x"],
|
|
["xyx", "x"],
|
|
[" ", " "],
|
|
["\t", "\t"],
|
|
]
|
|
data_expected.each do |(data, expected)|
|
|
Readline.completion_append_character = data
|
|
assert_equal(expected, Readline.completion_append_character)
|
|
assert_equal(enc, Readline.completion_append_character.encoding)
|
|
end
|
|
Readline.completion_append_character = ""
|
|
assert_equal(nil, Readline.completion_append_character)
|
|
rescue NotImplementedError
|
|
end
|
|
end
|
|
|
|
def test_completion_encoding
|
|
skip "Skip Editline" if /EditLine/n.match(Readline::VERSION)
|
|
bug5941 = '[Bug #5941]'
|
|
append_character = Readline.completion_append_character
|
|
Readline.completion_append_character = ""
|
|
completion_case_fold = Readline.completion_case_fold
|
|
locale = Encoding.find("locale")
|
|
if locale == Encoding::UTF_8
|
|
enc1 = Encoding::EUC_JP
|
|
else
|
|
enc1 = Encoding::UTF_8
|
|
end
|
|
results = nil
|
|
Readline.completion_proc = ->(text) {results}
|
|
|
|
[%W"\u{3042 3042} \u{3042 3044}", %W"\u{fe5b fe5b} \u{fe5b fe5c}"].any? do |w|
|
|
begin
|
|
results = w.map {|s| s.encode(locale)}
|
|
rescue Encoding::UndefinedConversionError
|
|
end
|
|
end or
|
|
begin
|
|
"\xa1\xa2".encode(Encoding::UTF_8, locale)
|
|
rescue
|
|
else
|
|
results = %W"\xa1\xa1 \xa1\xa2".map {|s| s.force_encoding(locale)}
|
|
end or
|
|
begin
|
|
return if assert_under_utf8
|
|
skip("missing test for locale #{locale.name}")
|
|
end
|
|
expected = results[0][0...1]
|
|
Readline.completion_case_fold = false
|
|
assert_equal(expected, with_pipe {|r, w| w << "\t"}, bug5941)
|
|
Readline.completion_case_fold = true
|
|
assert_equal(expected, with_pipe {|r, w| w << "\t"}, bug5941)
|
|
results.map! {|s| s.encode(enc1)}
|
|
assert_raise(Encoding::CompatibilityError, bug5941) do
|
|
with_pipe {|r, w| w << "\t"}
|
|
end
|
|
ensure
|
|
return if /EditLine/n.match(Readline::VERSION)
|
|
Readline.completion_case_fold = completion_case_fold
|
|
Readline.completion_append_character = append_character
|
|
end
|
|
|
|
# basic_word_break_characters
|
|
# completer_word_break_characters
|
|
# basic_quote_characters
|
|
# completer_quote_characters
|
|
# filename_quote_characters
|
|
# special_prefixes
|
|
def test_some_characters_methods
|
|
method_names = [
|
|
"basic_word_break_characters",
|
|
"completer_word_break_characters",
|
|
"basic_quote_characters",
|
|
"completer_quote_characters",
|
|
"filename_quote_characters",
|
|
"special_prefixes",
|
|
]
|
|
method_names.each do |method_name|
|
|
begin
|
|
begin
|
|
enc = get_default_internal_encoding
|
|
saved = Readline.send(method_name.to_sym)
|
|
expecteds = [" ", " .,|\t", ""]
|
|
expecteds.each do |e|
|
|
Readline.send((method_name + "=").to_sym, e)
|
|
res = Readline.send(method_name.to_sym)
|
|
assert_equal(e, res)
|
|
assert_equal(enc, res.encoding, "Readline.#{method_name} should be #{enc.name}")
|
|
end
|
|
ensure
|
|
Readline.send((method_name + "=").to_sym, saved) if saved
|
|
end
|
|
rescue NotImplementedError
|
|
end
|
|
end
|
|
end
|
|
|
|
def test_closed_outstream
|
|
bug5803 = '[ruby-dev:45043]'
|
|
IO.pipe do |r, w|
|
|
Readline.input = r
|
|
Readline.output = w
|
|
(w << "##\t").close
|
|
assert_raise(IOError, bug5803) {Readline.readline}
|
|
end
|
|
end
|
|
|
|
def test_pre_input_hook
|
|
begin
|
|
pr = proc {}
|
|
Readline.pre_input_hook = pr
|
|
assert_equal(pr, Readline.pre_input_hook)
|
|
Readline.pre_input_hook = nil
|
|
assert_nil(Readline.pre_input_hook)
|
|
rescue NotImplementedError
|
|
end
|
|
end
|
|
|
|
def test_point
|
|
skip "Skip Editline" if /EditLine/n.match(Readline::VERSION)
|
|
assert_equal(0, Readline.point)
|
|
Readline.insert_text('12345')
|
|
assert_equal(5, Readline.point)
|
|
|
|
assert_equal(4, Readline.point=(4))
|
|
|
|
Readline.insert_text('abc')
|
|
assert_equal(7, Readline.point)
|
|
|
|
assert_equal('1234abc5', Readline.line_buffer)
|
|
rescue NotImplementedError
|
|
end
|
|
|
|
def test_insert_text
|
|
skip "Skip Editline" if /EditLine/n.match(Readline::VERSION)
|
|
str = "test_insert_text"
|
|
assert_equal(0, Readline.point)
|
|
assert_equal(Readline, Readline.insert_text(str))
|
|
assert_equal(str, Readline.line_buffer)
|
|
assert_equal(16, Readline.point)
|
|
assert_equal(get_default_internal_encoding,
|
|
Readline.line_buffer.encoding)
|
|
|
|
Readline.delete_text(1, 3)
|
|
assert_equal("t_insert_text", Readline.line_buffer)
|
|
Readline.delete_text(11)
|
|
assert_equal("t_insert_te", Readline.line_buffer)
|
|
Readline.delete_text(-3...-1)
|
|
assert_equal("t_inserte", Readline.line_buffer)
|
|
Readline.delete_text(-3..-1)
|
|
assert_equal("t_inse", Readline.line_buffer)
|
|
Readline.delete_text(3..-3)
|
|
assert_equal("t_ise", Readline.line_buffer)
|
|
Readline.delete_text(3, 1)
|
|
assert_equal("t_ie", Readline.line_buffer)
|
|
Readline.delete_text(1..1)
|
|
assert_equal("tie", Readline.line_buffer)
|
|
Readline.delete_text(1...2)
|
|
assert_equal("te", Readline.line_buffer)
|
|
Readline.delete_text
|
|
assert_equal("", Readline.line_buffer)
|
|
rescue NotImplementedError
|
|
end
|
|
|
|
def test_delete_text
|
|
skip "Skip Editline" if /EditLine/n.match(Readline::VERSION)
|
|
str = "test_insert_text"
|
|
assert_equal(0, Readline.point)
|
|
assert_equal(Readline, Readline.insert_text(str))
|
|
assert_equal(16, Readline.point)
|
|
assert_equal(str, Readline.line_buffer)
|
|
Readline.delete_text
|
|
|
|
if !defined?(Reline) or Readline != Reline
|
|
# NOTE: unexpected but GNU Readline's spec
|
|
assert_equal(16, Readline.point)
|
|
assert_equal("", Readline.line_buffer)
|
|
assert_equal(Readline, Readline.insert_text(str))
|
|
assert_equal(32, Readline.point)
|
|
assert_equal("", Readline.line_buffer)
|
|
end
|
|
rescue NotImplementedError
|
|
end
|
|
|
|
def test_modify_text_in_pre_input_hook
|
|
skip "Skip Editline" if /EditLine/n.match(Readline::VERSION)
|
|
with_temp_stdio {|stdin, stdout|
|
|
begin
|
|
stdin.write("world\n")
|
|
stdin.close
|
|
Readline.pre_input_hook = proc do
|
|
assert_equal("", Readline.line_buffer)
|
|
Readline.insert_text("hello ")
|
|
Readline.redisplay
|
|
end
|
|
replace_stdio(stdin.path, stdout.path) do
|
|
line = Readline.readline("> ")
|
|
assert_equal("hello world", line)
|
|
end
|
|
# Readline 4.3 doesn't include inserted text or input
|
|
# Reline's rendering logic is tricky
|
|
if Readline::VERSION != '4.3' and (!defined?(Reline) or Readline != Reline)
|
|
assert_equal("> hello world\n", stdout.read)
|
|
end
|
|
stdout.close
|
|
rescue NotImplementedError
|
|
ensure
|
|
begin
|
|
Readline.pre_input_hook = nil
|
|
rescue NotImplementedError
|
|
end
|
|
end
|
|
}
|
|
end
|
|
|
|
def test_input_metachar
|
|
skip "Skip Editline" if /EditLine/n.match(Readline::VERSION)
|
|
skip("Won't pass on mingw w/readline 7.0.005 [ruby-core:45682]") if mingw?
|
|
bug6601 = '[ruby-core:45682]'
|
|
Readline::HISTORY << "hello"
|
|
wo = nil
|
|
line = with_pipe do |r, w|
|
|
wo = w.dup
|
|
wo.write("\C-re\ef\n")
|
|
end
|
|
assert_equal("hello", line, bug6601)
|
|
ensure
|
|
wo&.close
|
|
return if /EditLine/n.match(Readline::VERSION)
|
|
Readline.delete_text
|
|
Readline::HISTORY.clear
|
|
end
|
|
|
|
def test_input_metachar_multibyte
|
|
skip "Skip Editline" if /EditLine/n.match(Readline::VERSION)
|
|
unless Encoding.find("locale") == Encoding::UTF_8
|
|
return if assert_under_utf8
|
|
skip 'this test needs UTF-8 locale'
|
|
end
|
|
bug6602 = '[ruby-core:45683]'
|
|
Readline::HISTORY << "\u3042\u3093"
|
|
Readline::HISTORY << "\u3044\u3093"
|
|
Readline::HISTORY << "\u3046\u3093"
|
|
open(IO::NULL, 'w') do |null|
|
|
IO.pipe do |r, w|
|
|
Readline.input = r
|
|
Readline.output = null
|
|
w << "\cr\u3093\n\n"
|
|
w << "\cr\u3042\u3093"
|
|
w.reopen(IO::NULL)
|
|
assert_equal("\u3046\u3093", Readline.readline("", true), bug6602)
|
|
Timeout.timeout(2) do
|
|
assert_equal("\u3042\u3093", Readline.readline("", true), bug6602)
|
|
end
|
|
assert_equal(nil, Readline.readline("", true), bug6602)
|
|
end
|
|
end
|
|
ensure
|
|
return if /EditLine/n.match(Readline::VERSION)
|
|
Readline.delete_text
|
|
Readline::HISTORY.clear
|
|
end
|
|
|
|
def test_refresh_line
|
|
skip "Only when refresh_line exists" unless Readline.respond_to?(:refresh_line)
|
|
bug6232 = '[ruby-core:43957] [Bug #6232] refresh_line after set_screen_size'
|
|
with_temp_stdio do |stdin, stdout|
|
|
replace_stdio(stdin.path, stdout.path) do
|
|
assert_ruby_status(%w[-rreadline -], <<-'end;', bug6232)
|
|
Readline.set_screen_size(40, 80)
|
|
Readline.refresh_line
|
|
end;
|
|
end
|
|
end
|
|
end
|
|
|
|
def test_setting_quoting_detection_proc
|
|
return unless Readline.respond_to?(:quoting_detection_proc=)
|
|
|
|
expected = proc { |text, index| false }
|
|
Readline.quoting_detection_proc = expected
|
|
assert_equal(expected, Readline.quoting_detection_proc)
|
|
|
|
assert_raise(ArgumentError) do
|
|
Readline.quoting_detection_proc = "This does not have call method."
|
|
end
|
|
end
|
|
|
|
def test_using_quoting_detection_proc
|
|
saved_completer_quote_characters = Readline.completer_quote_characters
|
|
saved_completer_word_break_characters = Readline.completer_word_break_characters
|
|
return unless Readline.respond_to?(:quoting_detection_proc=)
|
|
|
|
passed_text = nil
|
|
line = nil
|
|
|
|
with_temp_stdio do |stdin, stdout|
|
|
replace_stdio(stdin.path, stdout.path) do
|
|
Readline.completion_proc = ->(text) do
|
|
passed_text = text
|
|
['completion']
|
|
end
|
|
Readline.completer_quote_characters = '\'"'
|
|
Readline.completer_word_break_characters = ' '
|
|
Readline.quoting_detection_proc = ->(text, index) do
|
|
index > 0 && text[index-1] == '\\'
|
|
end
|
|
|
|
stdin.write("first second\\ third\t")
|
|
stdin.flush
|
|
line = Readline.readline('> ', false)
|
|
end
|
|
end
|
|
|
|
assert_equal('second\\ third', passed_text)
|
|
assert_equal('first completion', line)
|
|
ensure
|
|
Readline.completer_quote_characters = saved_completer_quote_characters
|
|
Readline.completer_word_break_characters = saved_completer_word_break_characters
|
|
end
|
|
|
|
def test_using_quoting_detection_proc_with_multibyte_input
|
|
saved_completer_quote_characters = Readline.completer_quote_characters
|
|
saved_completer_word_break_characters = Readline.completer_word_break_characters
|
|
return unless Readline.respond_to?(:quoting_detection_proc=)
|
|
unless Encoding.find("locale") == Encoding::UTF_8
|
|
return if assert_under_utf8
|
|
skip 'this test needs UTF-8 locale'
|
|
end
|
|
|
|
passed_text = nil
|
|
escaped_char_indexes = []
|
|
line = nil
|
|
|
|
with_temp_stdio do |stdin, stdout|
|
|
replace_stdio(stdin.path, stdout.path) do
|
|
Readline.completion_proc = ->(text) do
|
|
passed_text = text
|
|
['completion']
|
|
end
|
|
Readline.completer_quote_characters = '\'"'
|
|
Readline.completer_word_break_characters = ' '
|
|
Readline.quoting_detection_proc = ->(text, index) do
|
|
escaped = index > 0 && text[index-1] == '\\'
|
|
escaped_char_indexes << index if escaped
|
|
escaped
|
|
end
|
|
|
|
stdin.write("\u3042\u3093 second\\ third\t")
|
|
stdin.flush
|
|
line = Readline.readline('> ', false)
|
|
end
|
|
end
|
|
|
|
assert_equal([10], escaped_char_indexes)
|
|
assert_equal('second\\ third', passed_text)
|
|
assert_equal("\u3042\u3093 completion", line)
|
|
ensure
|
|
Readline.completer_quote_characters = saved_completer_quote_characters
|
|
Readline.completer_word_break_characters = saved_completer_word_break_characters
|
|
end
|
|
|
|
def test_completion_quote_character_completing_unquoted_argument
|
|
return unless Readline.respond_to?(:completion_quote_character)
|
|
|
|
quote_character = "original value"
|
|
Readline.completion_proc = -> (_) do
|
|
quote_character = Readline.completion_quote_character
|
|
[]
|
|
end
|
|
Readline.completer_quote_characters = "'\""
|
|
|
|
with_temp_stdio do |stdin, stdout|
|
|
replace_stdio(stdin.path, stdout.path) do
|
|
stdin.write("input\t")
|
|
stdin.flush
|
|
Readline.readline("> ", false)
|
|
end
|
|
end
|
|
|
|
assert_nil(quote_character)
|
|
end
|
|
|
|
def test_completion_quote_character_completing_quoted_argument
|
|
return unless Readline.respond_to?(:completion_quote_character)
|
|
|
|
quote_character = "original value"
|
|
Readline.completion_proc = -> (_) do
|
|
quote_character = Readline.completion_quote_character
|
|
[]
|
|
end
|
|
Readline.completer_quote_characters = "'\""
|
|
|
|
with_temp_stdio do |stdin, stdout|
|
|
replace_stdio(stdin.path, stdout.path) do
|
|
stdin.write("'input\t")
|
|
stdin.flush
|
|
Readline.readline("> ", false)
|
|
end
|
|
end
|
|
|
|
assert_equal("'", quote_character)
|
|
end
|
|
|
|
def test_completion_quote_character_after_completion
|
|
return unless Readline.respond_to?(:completion_quote_character)
|
|
if /solaris/i =~ RUBY_PLATFORM
|
|
# http://rubyci.s3.amazonaws.com/solaris11s-sunc/ruby-trunk/log/20181228T102505Z.fail.html.gz
|
|
skip 'This test does not succeed on Oracle Developer Studio for now'
|
|
end
|
|
|
|
Readline.completion_proc = -> (_) { [] }
|
|
Readline.completer_quote_characters = "'\""
|
|
|
|
with_temp_stdio do |stdin, stdout|
|
|
replace_stdio(stdin.path, stdout.path) do
|
|
stdin.write("'input\t")
|
|
stdin.flush
|
|
Readline.readline("> ", false)
|
|
end
|
|
end
|
|
|
|
assert_nil(Readline.completion_quote_character)
|
|
end
|
|
|
|
private
|
|
|
|
def replace_stdio(stdin_path, stdout_path)
|
|
open(stdin_path, "r"){|stdin|
|
|
open(stdout_path, "w"){|stdout|
|
|
orig_stdin = STDIN.dup
|
|
orig_stdout = STDOUT.dup
|
|
orig_stderr = STDERR.dup
|
|
STDIN.reopen(stdin)
|
|
STDOUT.reopen(stdout)
|
|
STDERR.reopen(stdout)
|
|
begin
|
|
Readline.input = STDIN
|
|
Readline.output = STDOUT
|
|
yield
|
|
ensure
|
|
STDERR.reopen(orig_stderr)
|
|
STDIN.reopen(orig_stdin)
|
|
STDOUT.reopen(orig_stdout)
|
|
orig_stdin.close
|
|
orig_stdout.close
|
|
orig_stderr.close
|
|
end
|
|
}
|
|
}
|
|
end
|
|
|
|
def with_temp_stdio
|
|
Tempfile.create("test_readline_stdin") {|stdin|
|
|
Tempfile.create("test_readline_stdout") {|stdout|
|
|
yield stdin, stdout
|
|
if windows?
|
|
# needed since readline holds refs to tempfiles, can't delete on Windows
|
|
Readline.input = STDIN
|
|
Readline.output = STDOUT
|
|
end
|
|
}
|
|
}
|
|
end
|
|
|
|
def with_pipe
|
|
stderr = nil
|
|
IO.pipe do |r, w|
|
|
yield(r, w)
|
|
Readline.input = r
|
|
Readline.output = w.reopen(IO::NULL)
|
|
stderr = STDERR.dup
|
|
STDERR.reopen(w)
|
|
Readline.readline
|
|
end
|
|
ensure
|
|
if stderr
|
|
STDERR.reopen(stderr)
|
|
stderr.close
|
|
end
|
|
Readline.input = STDIN
|
|
Readline.output = STDOUT
|
|
end
|
|
|
|
def get_default_internal_encoding
|
|
return Encoding.default_internal || Encoding.find("locale")
|
|
end
|
|
|
|
def assert_under_utf8
|
|
return false if ENV['LC_ALL'] == 'UTF-8'
|
|
loc = caller_locations(1, 1)[0].base_label.to_s
|
|
assert_separately([{"LC_ALL"=>"UTF-8"}, "-r", __FILE__], <<SRC)
|
|
#skip "test \#{ENV['LC_ALL']}"
|
|
#{self.class.name}.new(#{loc.dump}).run(Test::Unit::Runner.new)
|
|
SRC
|
|
return true
|
|
end
|
|
end
|
|
|
|
class TestReadline < Test::Unit::TestCase
|
|
include BasetestReadline
|
|
|
|
def setup
|
|
use_ext_readline
|
|
super
|
|
end
|
|
end if defined?(ReadlineSo)
|
|
|
|
class TestRelineAsReadline < Test::Unit::TestCase
|
|
include BasetestReadline
|
|
|
|
def setup
|
|
use_lib_reline
|
|
super
|
|
end
|
|
end
|