# frozen_string_literal: false require 'test/unit' class TestCall < Test::Unit::TestCase def aaa(a, b=100, *rest) res = [a, b] res += rest if rest return res end def test_call assert_raise(ArgumentError) {aaa()} assert_raise(ArgumentError) {aaa} assert_equal([1, 100], aaa(1)) assert_equal([1, 2], aaa(1, 2)) assert_equal([1, 2, 3, 4], aaa(1, 2, 3, 4)) assert_equal([1, 2, 3, 4], aaa(1, *[2, 3, 4])) end def test_callinfo bug9622 = '[ruby-core:61422] [Bug #9622]' o = Class.new do def foo(*args) bar(:foo, *args) end def bar(name) name end end.new e = assert_raise(ArgumentError) {o.foo(100)} assert_nothing_raised(ArgumentError) {o.foo} assert_raise_with_message(ArgumentError, e.message, bug9622) {o.foo(100)} end def test_safe_call s = Struct.new(:x, :y, :z) o = s.new("x") assert_equal("X", o.x&.upcase) assert_nil(o.y&.upcase) assert_equal("x", o.x) o&.x = 6 assert_equal(6, o.x) o&.x *= 7 assert_equal(42, o.x) o&.y = 5 assert_equal(5, o.y) o&.z ||= 6 assert_equal(6, o.z) o&.z &&= 7 assert_equal(7, o.z) o = nil assert_nil(o&.x) assert_nothing_raised(NoMethodError) {o&.x = raise} assert_nothing_raised(NoMethodError) {o&.x = raise; nil} assert_nothing_raised(NoMethodError) {o&.x *= raise} assert_nothing_raised(NoMethodError) {o&.x *= raise; nil} assert_nothing_raised(NoMethodError) {o&.x ||= raise} assert_nothing_raised(NoMethodError) {o&.x ||= raise; nil} assert_nothing_raised(NoMethodError) {o&.x &&= raise} assert_nothing_raised(NoMethodError) {o&.x &&= raise; nil} end def test_safe_call_evaluate_arguments_only_method_call_is_made count = 0 proc = proc { count += 1; 1 } s = Struct.new(:x, :y) o = s.new(["a", "b", "c"]) o.y&.at(proc.call) assert_equal(0, count) o.x&.at(proc.call) assert_equal(1, count) end def test_safe_call_block_command assert_nil(("a".sub! "b" do end&.foo 1)) end def test_safe_call_block_call assert_nil(("a".sub! "b" do end&.foo)) end def test_safe_call_block_call_brace assert_nil(("a".sub! "b" do end&.foo {})) assert_nil(("a".sub! "b" do end&.foo do end)) end def test_safe_call_block_call_command assert_nil(("a".sub! "b" do end&.foo 1 do end)) end def test_invalid_safe_call h = nil assert_raise(NoMethodError) { h[:foo] = nil } end def test_call_splat_order bug12860 = '[ruby-core:77701] [Bug# 12860]' ary = [1, 2] assert_equal([1, 2, 1], aaa(*ary, ary.shift), bug12860) ary = [1, 2] assert_equal([0, 1, 2, 1], aaa(0, *ary, ary.shift), bug12860) end def test_call_block_order bug16504 = '[ruby-core:96769] [Bug# 16504]' b = proc{} ary = [1, 2, b] assert_equal([1, 2, b], aaa(*ary, &ary.pop), bug16504) ary = [1, 2, b] assert_equal([0, 1, 2, b], aaa(0, *ary, &ary.pop), bug16504) end def test_call_cfunc_splat_large_array_bug_4040 a = 1380.times.to_a # Greater than VM_ARGC_STACK_MAX assert_equal(a, [].push(*a)) assert_equal(a, [].push(a[0], *a[1..])) assert_equal(a, [].push(a[0], a[1], *a[2..])) assert_equal(a, [].push(*a[0..1], *a[2..])) assert_equal(a, [].push(*a[...-1], a[-1])) assert_equal(a, [].push(a[0], *a[1...-1], a[-1])) assert_equal(a, [].push(a[0], a[1], *a[2...-1], a[-1])) assert_equal(a, [].push(*a[0..1], *a[2...-1], a[-1])) assert_equal(a, [].push(*a[...-2], a[-2], a[-1])) assert_equal(a, [].push(a[0], *a[1...-2], a[-2], a[-1])) assert_equal(a, [].push(a[0], a[1], *a[2...-2], a[-2], a[-1])) assert_equal(a, [].push(*a[0..1], *a[2...-2], a[-2], a[-1])) kw = {x: 1} a_kw = a + [kw] assert_equal(a_kw, [].push(*a, **kw)) assert_equal(a_kw, [].push(a[0], *a[1..], **kw)) assert_equal(a_kw, [].push(a[0], a[1], *a[2..], **kw)) assert_equal(a_kw, [].push(*a[0..1], *a[2..], **kw)) assert_equal(a_kw, [].push(*a[...-1], a[-1], **kw)) assert_equal(a_kw, [].push(a[0], *a[1...-1], a[-1], **kw)) assert_equal(a_kw, [].push(a[0], a[1], *a[2...-1], a[-1], **kw)) assert_equal(a_kw, [].push(*a[0..1], *a[2...-1], a[-1], **kw)) assert_equal(a_kw, [].push(*a[...-2], a[-2], a[-1], **kw)) assert_equal(a_kw, [].push(a[0], *a[1...-2], a[-2], a[-1], **kw)) assert_equal(a_kw, [].push(a[0], a[1], *a[2...-2], a[-2], a[-1], **kw)) assert_equal(a_kw, [].push(*a[0..1], *a[2...-2], a[-2], a[-1], **kw)) assert_equal(a_kw, [].push(*a, x: 1)) assert_equal(a_kw, [].push(a[0], *a[1..], x: 1)) assert_equal(a_kw, [].push(a[0], a[1], *a[2..], x: 1)) assert_equal(a_kw, [].push(*a[0..1], *a[2..], x: 1)) assert_equal(a_kw, [].push(*a[...-1], a[-1], x: 1)) assert_equal(a_kw, [].push(a[0], *a[1...-1], a[-1], x: 1)) assert_equal(a_kw, [].push(a[0], a[1], *a[2...-1], a[-1], x: 1)) assert_equal(a_kw, [].push(*a[0..1], *a[2...-1], a[-1], x: 1)) assert_equal(a_kw, [].push(*a[...-2], a[-2], a[-1], x: 1)) assert_equal(a_kw, [].push(a[0], *a[1...-2], a[-2], a[-1], x: 1)) assert_equal(a_kw, [].push(a[0], a[1], *a[2...-2], a[-2], a[-1], x: 1)) assert_equal(a_kw, [].push(*a[0..1], *a[2...-2], a[-2], a[-1], x: 1)) a_kw[-1][:y] = 2 kw = {y: 2} assert_equal(a_kw, [].push(*a, x: 1, **kw)) assert_equal(a_kw, [].push(a[0], *a[1..], x: 1, **kw)) assert_equal(a_kw, [].push(a[0], a[1], *a[2..], x: 1, **kw)) assert_equal(a_kw, [].push(*a[0..1], *a[2..], x: 1, **kw)) assert_equal(a_kw, [].push(*a[...-1], a[-1], x: 1, **kw)) assert_equal(a_kw, [].push(a[0], *a[1...-1], a[-1], x: 1, **kw)) assert_equal(a_kw, [].push(a[0], a[1], *a[2...-1], a[-1], x: 1, **kw)) assert_equal(a_kw, [].push(*a[0..1], *a[2...-1], a[-1], x: 1, **kw)) assert_equal(a_kw, [].push(*a[...-2], a[-2], a[-1], x: 1, **kw)) assert_equal(a_kw, [].push(a[0], *a[1...-2], a[-2], a[-1], x: 1, **kw)) assert_equal(a_kw, [].push(a[0], a[1], *a[2...-2], a[-2], a[-1], x: 1, **kw)) assert_equal(a_kw, [].push(*a[0..1], *a[2...-2], a[-2], a[-1], x: 1, **kw)) kw = {} assert_equal(a, [].push(*a, **kw)) assert_equal(a, [].push(a[0], *a[1..], **kw)) assert_equal(a, [].push(a[0], a[1], *a[2..], **kw)) assert_equal(a, [].push(*a[0..1], *a[2..], **kw)) assert_equal(a, [].push(*a[...-1], a[-1], **kw)) assert_equal(a, [].push(a[0], *a[1...-1], a[-1], **kw)) assert_equal(a, [].push(a[0], a[1], *a[2...-1], a[-1], **kw)) assert_equal(a, [].push(*a[0..1], *a[2...-1], a[-1], **kw)) assert_equal(a, [].push(*a[...-2], a[-2], a[-1], **kw)) assert_equal(a, [].push(a[0], *a[1...-2], a[-2], a[-1], **kw)) assert_equal(a, [].push(a[0], a[1], *a[2...-2], a[-2], a[-1], **kw)) assert_equal(a, [].push(*a[0..1], *a[2...-2], a[-2], a[-1], **kw)) a_kw = a + [Hash.ruby2_keywords_hash({})] assert_equal(a, [].push(*a_kw)) # Single test with value that would cause SystemStackError. # Not all tests use such a large array to reduce testing time. assert_equal(1380888, [].push(*1380888.times.to_a).size) end end