summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authornobu <nobu@b2dd03c8-39d4-4d8f-98ff-823fe69b080e>2018-11-22 05:51:42 +0000
committernobu <nobu@b2dd03c8-39d4-4d8f-98ff-823fe69b080e>2018-11-22 05:51:42 +0000
commit3b7b70650c744f8d045328f782fcad360bdd9f46 (patch)
tree82626e1d7fbe5a1dd3515f612b307024e2bb53f6
parent4eaf22cc3f6639003ed3b64c3fee82c5867d27ea (diff)
downloadruby-3b7b70650c744f8d045328f782fcad360bdd9f46.tar.gz
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
-rw-r--r--proc.c16
-rw-r--r--test/ruby/test_method.rb20
-rw-r--r--test/ruby/test_proc.rb17
3 files changed, 34 insertions, 19 deletions
diff --git a/proc.c b/proc.c
index 3c3544ecf6..3c75f1f0eb 100644
--- a/proc.c
+++ b/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.
*
diff --git a/test/ruby/test_method.rb b/test/ruby/test_method.rb
index 80a2669633..5193ac6889 100644
--- a/test/ruby/test_method.rb
+++ b/test/ruby/test_method.rb
@@ -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
diff --git a/test/ruby/test_proc.rb b/test/ruby/test_proc.rb
index 7ea4556e8a..f70345ed74 100644
--- a/test/ruby/test_proc.rb
+++ b/test/ruby/test_proc.rb
@@ -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_equal(6, g.call(2))
+ end
+
+ def test_compose_with_noncallable
+ f = proc {|x| x * 2}
+ g = f * 5
- assert_raise(TypeError) {
- f * 5
+ assert_raise(NoMethodError) {
+ g.call(2)
}
end
end