mirror of
https://github.com/ruby/ruby.git
synced 2022-11-09 12:17:21 -05:00
proc.c: Support any callable when composing Procs
* proc.c (proc_compose): support any object with a call method rather than supporting only procs. [Feature #6284] * proc.c (compose): use the function call on the given object rather than rb_proc_call_with_block in order to support any object. * test/ruby/test_proc.rb: Add test cases for composing Procs with callable objects. * test/ruby/test_method.rb: Add test cases for composing Methods with callable objects. From: Paul Mucur <paul@altmetric.com> git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@65913 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
This commit is contained in:
parent
4eaf22cc3f
commit
3b7b70650c
3 changed files with 34 additions and 19 deletions
16
proc.c
16
proc.c
|
@ -3052,7 +3052,7 @@ compose(VALUE dummy, VALUE args, int argc, VALUE *argv, VALUE passed_proc)
|
|||
VALUE f, g, fargs;
|
||||
f = RARRAY_AREF(args, 0);
|
||||
g = RARRAY_AREF(args, 1);
|
||||
fargs = rb_ary_new3(1, rb_proc_call_with_block(g, argc, argv, passed_proc));
|
||||
fargs = rb_ary_new3(1, rb_funcall_with_block(g, idCall, argc, argv, passed_proc));
|
||||
|
||||
return rb_proc_call(f, fargs);
|
||||
}
|
||||
|
@ -3061,7 +3061,7 @@ compose(VALUE dummy, VALUE args, int argc, VALUE *argv, VALUE passed_proc)
|
|||
* call-seq:
|
||||
* prc * g -> a_proc
|
||||
*
|
||||
* Returns a proc that is the composition of this proc and the given proc <i>g</i>.
|
||||
* Returns a proc that is the composition of this proc and the given <i>g</i>.
|
||||
* The returned proc takes a variable number of arguments, calls <i>g</i> with them
|
||||
* then calls this proc with the result.
|
||||
*
|
||||
|
@ -3077,16 +3077,6 @@ proc_compose(VALUE self, VALUE g)
|
|||
rb_proc_t *procp;
|
||||
int is_lambda;
|
||||
|
||||
if (!rb_obj_is_method(g) && !rb_obj_is_proc(g)) {
|
||||
rb_raise(rb_eTypeError,
|
||||
"wrong argument type %s (expected Proc/Method)",
|
||||
rb_obj_classname(g));
|
||||
}
|
||||
|
||||
if (rb_obj_is_method(g)) {
|
||||
g = method_to_proc(g);
|
||||
}
|
||||
|
||||
args = rb_ary_new3(2, self, g);
|
||||
|
||||
GetProcPtr(self, procp);
|
||||
|
@ -3103,7 +3093,7 @@ proc_compose(VALUE self, VALUE g)
|
|||
* call-seq:
|
||||
* meth * g -> a_proc
|
||||
*
|
||||
* Returns a proc that is the composition of this method and the given proc <i>g</i>.
|
||||
* Returns a proc that is the composition of this method and the given <i>g</i>.
|
||||
* The returned proc takes a variable number of arguments, calls <i>g</i> with them
|
||||
* then calls this method with the result.
|
||||
*
|
||||
|
|
|
@ -1064,14 +1064,28 @@ class TestMethod < Test::Unit::TestCase
|
|||
assert_equal(6, h.call(2))
|
||||
end
|
||||
|
||||
def test_compose_with_nonproc_or_method
|
||||
def test_compose_with_callable
|
||||
c = Class.new {
|
||||
def f(x) x * 2 end
|
||||
}
|
||||
c2 = Class.new {
|
||||
def call(x) x + 1 end
|
||||
}
|
||||
f = c.new.method(:f)
|
||||
g = f * c2.new
|
||||
|
||||
assert_equal(6, g.call(2))
|
||||
end
|
||||
|
||||
def test_compose_with_noncallable
|
||||
c = Class.new {
|
||||
def f(x) x * 2 end
|
||||
}
|
||||
f = c.new.method(:f)
|
||||
g = f * 5
|
||||
|
||||
assert_raise(TypeError) {
|
||||
f * 5
|
||||
assert_raise(NoMethodError) {
|
||||
g.call(2)
|
||||
}
|
||||
end
|
||||
end
|
||||
|
|
|
@ -1460,11 +1460,22 @@ class TestProc < Test::Unit::TestCase
|
|||
assert_equal(6, h.call(2))
|
||||
end
|
||||
|
||||
def test_compose_with_nonproc_or_method
|
||||
def test_compose_with_callable
|
||||
f = proc {|x| x * 2}
|
||||
c = Class.new {
|
||||
def call(x) x + 1 end
|
||||
}
|
||||
g = f * c.new
|
||||
|
||||
assert_raise(TypeError) {
|
||||
f * 5
|
||||
assert_equal(6, g.call(2))
|
||||
end
|
||||
|
||||
def test_compose_with_noncallable
|
||||
f = proc {|x| x * 2}
|
||||
g = f * 5
|
||||
|
||||
assert_raise(NoMethodError) {
|
||||
g.call(2)
|
||||
}
|
||||
end
|
||||
end
|
||||
|
|
Loading…
Add table
Reference in a new issue