2003-09-04 12:18:59 -04:00
|
|
|
require 'test/unit'
|
|
|
|
|
|
|
|
class TestProc < Test::Unit::TestCase
|
2008-02-13 10:03:59 -05:00
|
|
|
def setup
|
|
|
|
@verbose = $VERBOSE
|
|
|
|
$VERBOSE = nil
|
|
|
|
end
|
|
|
|
|
|
|
|
def teardown
|
|
|
|
$VERBOSE = @verbose
|
|
|
|
end
|
|
|
|
|
2003-09-04 12:18:59 -04:00
|
|
|
def test_proc
|
2003-11-27 14:45:17 -05:00
|
|
|
p1 = proc{|i| i}
|
|
|
|
assert_equal(2, p1.call(2))
|
|
|
|
assert_equal(3, p1.call(3))
|
2003-09-05 11:15:43 -04:00
|
|
|
|
2003-11-27 14:45:17 -05:00
|
|
|
p1 = proc{|i| i*2}
|
|
|
|
assert_equal(4, p1.call(2))
|
|
|
|
assert_equal(6, p1.call(3))
|
|
|
|
|
|
|
|
p2 = nil
|
|
|
|
x=0
|
2003-09-05 11:15:43 -04:00
|
|
|
|
2003-09-04 12:18:59 -04:00
|
|
|
proc{
|
|
|
|
iii=5 # nested local variable
|
2003-11-27 14:45:17 -05:00
|
|
|
p1 = proc{|i|
|
2003-09-04 12:18:59 -04:00
|
|
|
iii = i
|
|
|
|
}
|
2003-11-27 14:45:17 -05:00
|
|
|
p2 = proc {
|
|
|
|
x = iii # nested variables shared by procs
|
2003-09-04 12:18:59 -04:00
|
|
|
}
|
|
|
|
# scope of nested variables
|
|
|
|
assert(defined?(iii))
|
|
|
|
}.call
|
|
|
|
assert(!defined?(iii)) # out of scope
|
2003-09-05 11:15:43 -04:00
|
|
|
|
2003-09-04 12:18:59 -04:00
|
|
|
loop{iii=5; assert(eval("defined? iii")); break}
|
|
|
|
loop {
|
|
|
|
iii = 10
|
2003-09-05 11:15:43 -04:00
|
|
|
def self.dyna_var_check
|
2003-09-04 12:18:59 -04:00
|
|
|
loop {
|
|
|
|
assert(!defined?(iii))
|
|
|
|
break
|
|
|
|
}
|
|
|
|
end
|
|
|
|
dyna_var_check
|
|
|
|
break
|
|
|
|
}
|
2003-11-27 14:45:17 -05:00
|
|
|
p1.call(5)
|
|
|
|
p2.call
|
|
|
|
assert_equal(5, x)
|
|
|
|
end
|
|
|
|
|
|
|
|
def assert_arity(n)
|
|
|
|
meta = class << self; self; end
|
|
|
|
meta.class_eval {define_method(:foo, Proc.new)}
|
|
|
|
assert_equal(n, method(:foo).arity)
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_arity
|
2004-03-18 05:09:43 -05:00
|
|
|
assert_equal(0, proc{}.arity)
|
2003-11-27 14:45:17 -05:00
|
|
|
assert_equal(0, proc{||}.arity)
|
|
|
|
assert_equal(1, proc{|x|}.arity)
|
|
|
|
assert_equal(2, proc{|x, y|}.arity)
|
|
|
|
assert_equal(-2, proc{|x, *y|}.arity)
|
|
|
|
assert_equal(-1, proc{|*x|}.arity)
|
|
|
|
assert_equal(-1, proc{|*|}.arity)
|
2007-05-01 00:35:58 -04:00
|
|
|
assert_equal(-3, proc{|x, *y, z|}.arity)
|
|
|
|
assert_equal(-4, proc{|x, *y, z, a|}.arity)
|
2003-11-27 14:45:17 -05:00
|
|
|
|
2004-03-18 05:09:43 -05:00
|
|
|
assert_arity(0) {}
|
2003-11-27 14:45:17 -05:00
|
|
|
assert_arity(0) {||}
|
|
|
|
assert_arity(1) {|x|}
|
|
|
|
assert_arity(2) {|x, y|}
|
|
|
|
assert_arity(-2) {|x, *y|}
|
2007-05-01 00:35:58 -04:00
|
|
|
assert_arity(-3) {|x, *y, z|}
|
2003-11-27 14:45:17 -05:00
|
|
|
assert_arity(-1) {|*x|}
|
|
|
|
assert_arity(-1) {|*|}
|
2003-09-04 12:18:59 -04:00
|
|
|
end
|
2004-02-06 04:27:53 -05:00
|
|
|
|
|
|
|
def m(x)
|
|
|
|
lambda { x }
|
|
|
|
end
|
2006-12-31 10:02:22 -05:00
|
|
|
|
2004-02-06 04:27:53 -05:00
|
|
|
def test_eq
|
|
|
|
a = m(1)
|
|
|
|
b = m(2)
|
2007-11-18 02:18:56 -05:00
|
|
|
assert_not_equal(a, b, "[ruby-dev:22592]")
|
|
|
|
assert_not_equal(a.call, b.call, "[ruby-dev:22592]")
|
2004-02-06 04:55:43 -05:00
|
|
|
|
2007-11-18 02:18:56 -05:00
|
|
|
assert_not_equal(proc {||}, proc {|x,y|}, "[ruby-dev:22599]")
|
2004-02-06 04:27:53 -05:00
|
|
|
|
|
|
|
a = lambda {|x| lambda {} }.call(1)
|
|
|
|
b = lambda {}
|
2007-11-18 02:18:56 -05:00
|
|
|
assert_not_equal(a, b, "[ruby-dev:22601]")
|
2004-02-06 04:27:53 -05:00
|
|
|
end
|
2004-05-24 22:54:22 -04:00
|
|
|
|
|
|
|
def test_block_par
|
2007-05-01 00:35:58 -04:00
|
|
|
assert_equal(10, Proc.new{|&b| b.call(10)}.call {|x| x})
|
|
|
|
assert_equal(12, Proc.new{|a,&b| b.call(a)}.call(12) {|x| x})
|
2004-05-24 22:54:22 -04:00
|
|
|
end
|
2004-06-19 00:27:08 -04:00
|
|
|
|
|
|
|
def test_safe
|
|
|
|
safe = $SAFE
|
|
|
|
c = Class.new
|
|
|
|
x = c.new
|
|
|
|
|
|
|
|
p = proc {
|
|
|
|
$SAFE += 1
|
|
|
|
proc {$SAFE}
|
|
|
|
}.call
|
|
|
|
assert_equal(safe, $SAFE)
|
|
|
|
assert_equal(safe + 1, p.call)
|
|
|
|
assert_equal(safe, $SAFE)
|
|
|
|
|
|
|
|
c.class_eval {define_method(:safe, p)}
|
|
|
|
assert_equal(safe, x.safe)
|
|
|
|
assert_equal(safe, x.method(:safe).call)
|
|
|
|
assert_equal(safe, x.method(:safe).to_proc.call)
|
|
|
|
|
|
|
|
p = proc {$SAFE += 1}
|
|
|
|
assert_equal(safe + 1, p.call)
|
|
|
|
assert_equal(safe, $SAFE)
|
|
|
|
|
|
|
|
c.class_eval {define_method(:inc, p)}
|
|
|
|
assert_equal(safe + 1, proc {x.inc; $SAFE}.call)
|
|
|
|
assert_equal(safe, $SAFE)
|
|
|
|
assert_equal(safe + 1, proc {x.method(:inc).call; $SAFE}.call)
|
|
|
|
assert_equal(safe, $SAFE)
|
|
|
|
assert_equal(safe + 1, proc {x.method(:inc).to_proc.call; $SAFE}.call)
|
|
|
|
assert_equal(safe, $SAFE)
|
|
|
|
end
|
2006-12-31 10:02:22 -05:00
|
|
|
|
|
|
|
def m2
|
|
|
|
"OK"
|
|
|
|
end
|
|
|
|
|
|
|
|
def block
|
|
|
|
method(:m2).to_proc
|
|
|
|
end
|
|
|
|
|
|
|
|
# [yarv-dev:777] block made by Method#to_proc
|
|
|
|
def test_method_to_proc
|
|
|
|
b = block()
|
|
|
|
assert_equal "OK", b.call
|
|
|
|
end
|
|
|
|
|
2008-02-13 07:51:31 -05:00
|
|
|
def test_curry
|
|
|
|
b = proc {|x, y, z| (x||0) + (y||0) + (z||0) }
|
|
|
|
assert_equal(6, b.curry[1][2][3])
|
|
|
|
assert_equal(6, b.curry[1, 2][3, 4])
|
|
|
|
assert_equal(6, b.curry(5)[1][2][3][4][5])
|
|
|
|
assert_equal(6, b.curry(5)[1, 2][3, 4][5])
|
|
|
|
assert_equal(1, b.curry(1)[1])
|
|
|
|
|
|
|
|
b = proc {|x, y, z, *w| (x||0) + (y||0) + (z||0) + w.inject(0, &:+) }
|
|
|
|
assert_equal(6, b.curry[1][2][3])
|
|
|
|
assert_equal(10, b.curry[1, 2][3, 4])
|
|
|
|
assert_equal(15, b.curry(5)[1][2][3][4][5])
|
|
|
|
assert_equal(15, b.curry(5)[1, 2][3, 4][5])
|
|
|
|
assert_equal(1, b.curry(1)[1])
|
|
|
|
|
|
|
|
b = lambda {|x, y, z| (x||0) + (y||0) + (z||0) }
|
|
|
|
assert_equal(6, b.curry[1][2][3])
|
|
|
|
assert_raise(ArgumentError) { b.curry[1, 2][3, 4] }
|
|
|
|
assert_raise(ArgumentError) { b.curry(5) }
|
|
|
|
assert_raise(ArgumentError) { b.curry(1) }
|
|
|
|
|
|
|
|
b = lambda {|x, y, z, *w| (x||0) + (y||0) + (z||0) + w.inject(0, &:+) }
|
|
|
|
assert_equal(6, b.curry[1][2][3])
|
|
|
|
assert_equal(10, b.curry[1, 2][3, 4])
|
|
|
|
assert_equal(15, b.curry(5)[1][2][3][4][5])
|
|
|
|
assert_equal(15, b.curry(5)[1, 2][3, 4][5])
|
|
|
|
assert_raise(ArgumentError) { b.curry(1) }
|
|
|
|
|
|
|
|
b = proc { :foo }
|
|
|
|
assert_equal(:foo, b.curry[])
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_curry_ski_fib
|
|
|
|
s = proc {|f, g, x| f[x][g[x]] }.curry
|
|
|
|
k = proc {|x, y| x }.curry
|
|
|
|
i = proc {|x| x }.curry
|
|
|
|
|
|
|
|
fib = []
|
|
|
|
inc = proc {|x| fib[-1] += 1; x }.curry
|
|
|
|
ret = proc {|x| throw :end if fib.size > 10; fib << 0; x }.curry
|
|
|
|
|
|
|
|
catch(:end) do
|
|
|
|
s[
|
|
|
|
s[s[i][i]][k[i]]
|
|
|
|
][
|
|
|
|
k[inc]
|
|
|
|
][
|
|
|
|
s[
|
|
|
|
s[
|
|
|
|
k[s]
|
|
|
|
][
|
|
|
|
s[k[s[k[s]]]
|
|
|
|
][
|
|
|
|
s[s[k[s]][s[k[s[k[ret]]]][s[k[s[i]]][k]]]][k]]
|
|
|
|
]
|
|
|
|
][
|
|
|
|
k[s[k[s]][k]]
|
|
|
|
]
|
|
|
|
]
|
|
|
|
end
|
|
|
|
|
|
|
|
assert_equal(fib, [1, 1, 2, 3, 5, 8, 13, 21, 34, 55, 89])
|
|
|
|
end
|
2008-02-13 10:03:59 -05:00
|
|
|
|
2008-05-19 01:20:21 -04:00
|
|
|
def test_curry_from_knownbug
|
|
|
|
a = lambda {|x, y, &b| b }
|
|
|
|
b = a.curry[1]
|
|
|
|
|
|
|
|
assert_equal(:ok,
|
|
|
|
if b.call(2){} == nil
|
|
|
|
:ng
|
|
|
|
else
|
|
|
|
:ok
|
|
|
|
end, 'moved from btest/knownbug, [ruby-core:15551]')
|
|
|
|
end
|
|
|
|
|
2008-02-13 10:03:59 -05:00
|
|
|
def test_dup_clone
|
|
|
|
b = proc {|x| x + "bar" }
|
|
|
|
class << b; attr_accessor :foo; end
|
|
|
|
|
|
|
|
bd = b.dup
|
|
|
|
assert_equal("foobar", bd.call("foo"))
|
|
|
|
assert_raise(NoMethodError) { bd.foo = :foo }
|
|
|
|
assert_raise(NoMethodError) { bd.foo }
|
|
|
|
|
|
|
|
bc = b.clone
|
|
|
|
assert_equal("foobar", bc.call("foo"))
|
|
|
|
bc.foo = :foo
|
|
|
|
assert_equal(:foo, bc.foo)
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_binding
|
|
|
|
b = proc {|x, y, z| proc {}.binding }.call(1, 2, 3)
|
|
|
|
class << b; attr_accessor :foo; end
|
|
|
|
|
|
|
|
bd = b.dup
|
|
|
|
assert_equal([1, 2, 3], bd.eval("[x, y, z]"))
|
|
|
|
assert_raise(NoMethodError) { bd.foo = :foo }
|
|
|
|
assert_raise(NoMethodError) { bd.foo }
|
|
|
|
|
|
|
|
bc = b.clone
|
|
|
|
assert_equal([1, 2, 3], bc.eval("[x, y, z]"))
|
|
|
|
bc.foo = :foo
|
|
|
|
assert_equal(:foo, bc.foo)
|
|
|
|
|
|
|
|
b = nil
|
|
|
|
1.times { x, y, z = 1, 2, 3; b = binding }
|
|
|
|
assert_equal([1, 2, 3], b.eval("[x, y, z]"))
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_proc_lambda
|
|
|
|
assert_raise(ArgumentError) { proc }
|
|
|
|
assert_raise(ArgumentError) { lambda }
|
|
|
|
|
|
|
|
o = Object.new
|
|
|
|
def o.foo
|
|
|
|
b = nil
|
|
|
|
1.times { b = lambda }
|
|
|
|
b
|
|
|
|
end
|
|
|
|
assert_equal(:foo, o.foo { :foo }.call)
|
|
|
|
|
|
|
|
def o.foo(&b)
|
|
|
|
b = nil
|
|
|
|
1.times { b = lambda }
|
|
|
|
b
|
|
|
|
end
|
|
|
|
assert_equal(:foo, o.foo { :foo }.call)
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_arity2
|
|
|
|
assert_equal(0, method(:proc).to_proc.arity)
|
|
|
|
assert_equal(-1, proc {}.curry.arity)
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_proc_location
|
|
|
|
t = Thread.new { sleep }
|
|
|
|
assert_raise(ThreadError) { t.instance_eval { initialize { } } }
|
|
|
|
t.kill
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_eq2
|
|
|
|
b1 = proc { }
|
|
|
|
b2 = b1.dup
|
|
|
|
assert(b1 == b2)
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_to_proc
|
|
|
|
b = proc { :foo }
|
|
|
|
assert_equal(:foo, b.to_proc.call)
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_localjump_error
|
|
|
|
o = Object.new
|
|
|
|
def foo; yield; end
|
|
|
|
exc = foo rescue $!
|
|
|
|
assert_nil(exc.exit_value)
|
|
|
|
assert_equal(:noreason, exc.reason)
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_binding2
|
|
|
|
assert_raise(ArgumentError) { proc {}.curry.binding }
|
|
|
|
end
|
2003-09-04 12:18:59 -04:00
|
|
|
end
|