2018-04-14 15:49:30 +03:00
|
|
|
# frozen_string_literal: false
|
|
|
|
require 'test/unit'
|
|
|
|
require 'tempfile'
|
|
|
|
require 'irb'
|
|
|
|
require 'rubygems' if defined?(Gem)
|
|
|
|
|
|
|
|
module TestIRB
|
|
|
|
class TestContext < Test::Unit::TestCase
|
|
|
|
class TestInputMethod < ::IRB::InputMethod
|
2018-05-12 12:09:23 +03:00
|
|
|
attr_reader :list, :line_no
|
2018-04-14 15:49:30 +03:00
|
|
|
|
|
|
|
def initialize(list = [])
|
|
|
|
super("test")
|
|
|
|
@line_no = 0
|
2018-05-12 12:09:23 +03:00
|
|
|
@list = list
|
2018-04-14 15:49:30 +03:00
|
|
|
end
|
|
|
|
|
|
|
|
def gets
|
2018-05-12 12:09:23 +03:00
|
|
|
@list[@line_no]&.tap {@line_no += 1}
|
2018-04-14 15:49:30 +03:00
|
|
|
end
|
|
|
|
|
|
|
|
def eof?
|
|
|
|
@line_no >= @list.size
|
|
|
|
end
|
2018-05-12 12:13:45 +03:00
|
|
|
|
|
|
|
def encoding
|
|
|
|
Encoding.default_external
|
|
|
|
end
|
2019-03-20 21:50:05 +03:00
|
|
|
|
|
|
|
def reset
|
|
|
|
@line_no = 0
|
|
|
|
end
|
2018-04-14 15:49:30 +03:00
|
|
|
end
|
|
|
|
|
|
|
|
def setup
|
|
|
|
IRB.init_config(nil)
|
2019-10-30 03:32:16 +03:00
|
|
|
IRB.conf[:USE_SINGLELINE] = false
|
2018-04-14 15:49:30 +03:00
|
|
|
IRB.conf[:VERBOSE] = false
|
|
|
|
workspace = IRB::WorkSpace.new(Object.new)
|
|
|
|
@context = IRB::Context.new(nil, workspace, TestInputMethod.new)
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_last_value
|
|
|
|
assert_nil(@context.last_value)
|
|
|
|
assert_nil(@context.evaluate('_', 1))
|
|
|
|
obj = Object.new
|
|
|
|
@context.set_last_value(obj)
|
|
|
|
assert_same(obj, @context.last_value)
|
|
|
|
assert_same(obj, @context.evaluate('_', 1))
|
|
|
|
end
|
2018-04-14 16:05:52 +03:00
|
|
|
|
|
|
|
def test_evaluate_with_exception
|
|
|
|
assert_nil(@context.evaluate("$!", 1))
|
|
|
|
e = assert_raise_with_message(RuntimeError, 'foo') {
|
|
|
|
@context.evaluate("raise 'foo'", 1)
|
|
|
|
}
|
|
|
|
assert_equal('foo', e.message)
|
|
|
|
assert_same(e, @context.evaluate('$!', 1, exception: e))
|
2019-05-27 05:05:51 +03:00
|
|
|
e = assert_raise(SyntaxError) {
|
|
|
|
@context.evaluate("1,2,3", 1, exception: e)
|
|
|
|
}
|
|
|
|
assert_match(/\A\(irb\):1:/, e.message)
|
|
|
|
assert_not_match(/rescue _\.class/, e.message)
|
2018-04-14 16:05:52 +03:00
|
|
|
end
|
2018-05-12 12:13:45 +03:00
|
|
|
|
2020-03-26 10:43:16 +03:00
|
|
|
def test_evaluate_with_encoding_error_without_lineno
|
|
|
|
assert_raise_with_message(EncodingError, /invalid symbol/) {
|
|
|
|
@context.evaluate(%q[{"\xAE": 1}], 1)
|
|
|
|
# The backtrace of this invalid encoding hash doesn't contain lineno.
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
2019-11-13 09:10:05 +03:00
|
|
|
def test_evaluate_with_onigmo_warning
|
|
|
|
assert_warning("(irb):1: warning: character class has duplicated range: /[aa]/\n") do
|
|
|
|
@context.evaluate('/[aa]/', 1)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-05-12 12:13:45 +03:00
|
|
|
def test_eval_input
|
2019-04-21 08:48:35 +03:00
|
|
|
verbose, $VERBOSE = $VERBOSE, nil
|
2018-05-12 12:13:45 +03:00
|
|
|
input = TestInputMethod.new([
|
|
|
|
"raise 'Foo'\n",
|
|
|
|
"_\n",
|
|
|
|
"0\n",
|
|
|
|
"_\n",
|
|
|
|
])
|
|
|
|
irb = IRB::Irb.new(IRB::WorkSpace.new(Object.new), input)
|
2019-08-08 11:19:23 +03:00
|
|
|
out, err = capture_output do
|
2018-05-12 12:13:45 +03:00
|
|
|
irb.eval_input
|
|
|
|
end
|
|
|
|
assert_empty err
|
|
|
|
assert_pattern_list([:*, /RuntimeError \(.*Foo.*\).*\n/,
|
|
|
|
:*, /#<RuntimeError: Foo>\n/,
|
|
|
|
:*, /0$/,
|
|
|
|
:*, /0$/,
|
|
|
|
/\s*/], out)
|
2019-04-21 08:48:35 +03:00
|
|
|
ensure
|
|
|
|
$VERBOSE = verbose
|
2018-05-12 12:13:45 +03:00
|
|
|
end
|
2019-05-21 14:32:15 +03:00
|
|
|
|
2020-06-24 12:41:10 +03:00
|
|
|
def test_eval_object_without_inspect_method
|
|
|
|
verbose, $VERBOSE = $VERBOSE, nil
|
|
|
|
input = TestInputMethod.new([
|
|
|
|
"BasicObject.new\n",
|
|
|
|
])
|
|
|
|
irb = IRB::Irb.new(IRB::WorkSpace.new(Object.new), input)
|
|
|
|
out, err = capture_output do
|
|
|
|
irb.eval_input
|
|
|
|
end
|
|
|
|
assert_empty err
|
2020-07-21 21:13:10 +03:00
|
|
|
assert_equal("(Object doesn't support #inspect)\n=> \n", out)
|
2020-06-24 12:41:10 +03:00
|
|
|
ensure
|
|
|
|
$VERBOSE = verbose
|
|
|
|
end
|
|
|
|
|
2019-05-21 14:32:15 +03:00
|
|
|
def test_default_config
|
|
|
|
assert_equal(true, @context.use_colorize?)
|
|
|
|
end
|
2019-03-20 21:50:05 +03:00
|
|
|
|
|
|
|
def test_assignment_expression
|
|
|
|
input = TestInputMethod.new
|
|
|
|
irb = IRB::Irb.new(IRB::WorkSpace.new(Object.new), input)
|
|
|
|
[
|
|
|
|
"foo = bar",
|
|
|
|
"@foo = bar",
|
|
|
|
"$foo = bar",
|
|
|
|
"@@foo = bar",
|
|
|
|
"::Foo = bar",
|
|
|
|
"a::Foo = bar",
|
|
|
|
"Foo = bar",
|
|
|
|
"foo.bar = 1",
|
|
|
|
"foo[1] = bar",
|
|
|
|
"foo += bar",
|
|
|
|
"foo -= bar",
|
|
|
|
"foo ||= bar",
|
|
|
|
"foo &&= bar",
|
|
|
|
"foo, bar = 1, 2",
|
|
|
|
"foo.bar=(1)",
|
|
|
|
"foo; foo = bar",
|
|
|
|
"foo; foo = bar; ;\n ;",
|
|
|
|
"foo\nfoo = bar",
|
|
|
|
].each do |exp|
|
|
|
|
assert(
|
|
|
|
irb.assignment_expression?(exp),
|
|
|
|
"#{exp.inspect}: should be an assignment expression"
|
|
|
|
)
|
|
|
|
end
|
|
|
|
|
|
|
|
[
|
|
|
|
"foo",
|
|
|
|
"foo.bar",
|
|
|
|
"foo[0]",
|
|
|
|
"foo = bar; foo",
|
|
|
|
"foo = bar\nfoo",
|
|
|
|
].each do |exp|
|
|
|
|
refute(
|
|
|
|
irb.assignment_expression?(exp),
|
|
|
|
"#{exp.inspect}: should not be an assignment expression"
|
|
|
|
)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_echo_on_assignment
|
|
|
|
input = TestInputMethod.new([
|
|
|
|
"a = 1\n",
|
|
|
|
"a\n",
|
|
|
|
"a, b = 2, 3\n",
|
|
|
|
"a\n",
|
|
|
|
"b\n",
|
|
|
|
"b = 4\n",
|
|
|
|
"_\n"
|
|
|
|
])
|
|
|
|
irb = IRB::Irb.new(IRB::WorkSpace.new(Object.new), input)
|
2019-08-16 01:10:45 +03:00
|
|
|
irb.context.return_format = "=> %s\n"
|
2019-03-20 21:50:05 +03:00
|
|
|
|
|
|
|
# The default
|
|
|
|
irb.context.echo = true
|
|
|
|
irb.context.echo_on_assignment = false
|
|
|
|
out, err = capture_io do
|
|
|
|
irb.eval_input
|
|
|
|
end
|
|
|
|
assert_empty err
|
|
|
|
assert_equal("=> 1\n=> 2\n=> 3\n=> 4\n", out)
|
|
|
|
|
|
|
|
# Everything is output, like before echo_on_assignment was introduced
|
|
|
|
input.reset
|
|
|
|
irb.context.echo = true
|
|
|
|
irb.context.echo_on_assignment = true
|
|
|
|
out, err = capture_io do
|
|
|
|
irb.eval_input
|
|
|
|
end
|
|
|
|
assert_empty err
|
|
|
|
assert_equal("=> 1\n=> 1\n=> [2, 3]\n=> 2\n=> 3\n=> 4\n=> 4\n", out)
|
|
|
|
|
|
|
|
# Nothing is output when echo is false
|
|
|
|
input.reset
|
|
|
|
irb.context.echo = false
|
|
|
|
irb.context.echo_on_assignment = false
|
|
|
|
out, err = capture_io do
|
|
|
|
irb.eval_input
|
|
|
|
end
|
|
|
|
assert_empty err
|
|
|
|
assert_equal("", out)
|
|
|
|
|
|
|
|
# Nothing is output when echo is false even if echo_on_assignment is true
|
|
|
|
input.reset
|
|
|
|
irb.context.echo = false
|
|
|
|
irb.context.echo_on_assignment = true
|
|
|
|
out, err = capture_io do
|
|
|
|
irb.eval_input
|
|
|
|
end
|
|
|
|
assert_empty err
|
|
|
|
assert_equal("", out)
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_echo_on_assignment_conf
|
|
|
|
# Default
|
|
|
|
IRB.conf[:ECHO] = nil
|
|
|
|
IRB.conf[:ECHO_ON_ASSIGNMENT] = nil
|
|
|
|
input = TestInputMethod.new()
|
|
|
|
irb = IRB::Irb.new(IRB::WorkSpace.new(Object.new), input)
|
|
|
|
|
|
|
|
assert(irb.context.echo?, "echo? should be true by default")
|
|
|
|
refute(irb.context.echo_on_assignment?, "echo_on_assignment? should be false by default")
|
|
|
|
|
|
|
|
# Explicitly set :ECHO to false
|
|
|
|
IRB.conf[:ECHO] = false
|
|
|
|
irb = IRB::Irb.new(IRB::WorkSpace.new(Object.new), input)
|
|
|
|
|
|
|
|
refute(irb.context.echo?, "echo? should be false when IRB.conf[:ECHO] is set to false")
|
|
|
|
refute(irb.context.echo_on_assignment?, "echo_on_assignment? should be false by default")
|
|
|
|
|
|
|
|
# Explicitly set :ECHO_ON_ASSIGNMENT to true
|
|
|
|
IRB.conf[:ECHO] = nil
|
|
|
|
IRB.conf[:ECHO_ON_ASSIGNMENT] = true
|
|
|
|
irb = IRB::Irb.new(IRB::WorkSpace.new(Object.new), input)
|
|
|
|
|
|
|
|
assert(irb.context.echo?, "echo? should be true by default")
|
|
|
|
assert(irb.context.echo_on_assignment?, "echo_on_assignment? should be true when IRB.conf[:ECHO_ON_ASSIGNMENT] is set to true")
|
|
|
|
end
|
2020-01-16 07:13:35 +03:00
|
|
|
|
|
|
|
def test_multiline_output_on_default_inspector
|
|
|
|
main = Object.new
|
|
|
|
def main.inspect
|
|
|
|
"abc\ndef"
|
|
|
|
end
|
|
|
|
input = TestInputMethod.new([
|
|
|
|
"self"
|
|
|
|
])
|
|
|
|
irb = IRB::Irb.new(IRB::WorkSpace.new(main), input)
|
|
|
|
irb.context.return_format = "=> %s\n"
|
|
|
|
|
|
|
|
# The default
|
|
|
|
irb.context.newline_before_multiline_output = true
|
|
|
|
out, err = capture_io do
|
|
|
|
irb.eval_input
|
|
|
|
end
|
|
|
|
assert_empty err
|
|
|
|
assert_equal("=> \nabc\ndef\n",
|
|
|
|
out)
|
|
|
|
|
|
|
|
# No newline before multiline output
|
|
|
|
input.reset
|
|
|
|
irb.context.newline_before_multiline_output = false
|
|
|
|
out, err = capture_io do
|
|
|
|
irb.eval_input
|
|
|
|
end
|
|
|
|
assert_empty err
|
|
|
|
assert_equal("=> abc\ndef\n",
|
|
|
|
out)
|
|
|
|
end
|
2018-04-14 15:49:30 +03:00
|
|
|
end
|
|
|
|
end
|