summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authormatz <matz@b2dd03c8-39d4-4d8f-98ff-823fe69b080e>2009-09-24 04:42:28 +0000
committermatz <matz@b2dd03c8-39d4-4d8f-98ff-823fe69b080e>2009-09-24 04:42:28 +0000
commitc4b0b4c91c54b58b02cf94b25d6127b9c3777996 (patch)
treef842b52c2d3f54c2728f0363f178ed86770d1961
parent399dace5edc9380b66d709108195c71fc418f1ab (diff)
* proc.c (mnew): generate method object that wraps method_missing,
when #respond_to_missing? is defined. * test/ruby/test_object.rb (test_respond_to_missing): add test suites for #respond_to_missing? changes. git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@25073 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
-rw-r--r--ChangeLog8
-rw-r--r--proc.c21
-rw-r--r--test/ruby/test_object.rb33
3 files changed, 62 insertions, 0 deletions
diff --git a/ChangeLog b/ChangeLog
index b55133f937..81c01fc96e 100644
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,3 +1,11 @@
+Thu Sep 24 13:32:53 2009 Yukihiro Matsumoto <matz@ruby-lang.org>
+
+ * proc.c (mnew): generate method object that wraps method_missing,
+ when #respond_to_missing? is defined.
+
+ * test/ruby/test_object.rb (test_respond_to_missing): add test
+ suites for #respond_to_missing? changes.
+
Thu Sep 24 09:41:42 2009 Marc-Andre Lafortune <ruby-core@marc-andre.ca>
* lib/mathn.rb (Bignum#**): Fixed bignum**fixnum that was broken when
diff --git a/proc.c b/proc.c
index c5eca31d46..3300ec9aa7 100644
--- a/proc.c
+++ b/proc.c
@@ -884,6 +884,19 @@ rb_obj_is_method(VALUE m)
}
static VALUE
+missing_wrap(VALUE dummy, VALUE args, int argc, VALUE *argv)
+{
+ VALUE new_args = rb_ary_new4(argc, argv);
+ VALUE obj = RARRAY_PTR(args)[0];
+ VALUE sym = RARRAY_PTR(args)[1];
+
+
+ rb_ary_unshift(new_args, sym);
+ return rb_funcall2(obj, rb_intern("method_missing"),
+ check_argc(RARRAY_LEN(new_args)), RARRAY_PTR(new_args));
+}
+
+static VALUE
mnew(VALUE klass, VALUE obj, ID id, VALUE mclass, int scope)
{
VALUE method;
@@ -896,6 +909,14 @@ mnew(VALUE klass, VALUE obj, ID id, VALUE mclass, int scope)
again:
me = rb_method_entry(klass, id);
if (UNDEFINED_METHOD_ENTRY_P(me)) {
+ ID rmiss = rb_intern("respond_to_missing?");
+ VALUE sym = ID2SYM(id);
+
+ if (!rb_method_basic_definition_p(klass, rmiss)) {
+ if (RTEST(rb_funcall(obj, rmiss, 1, sym))) {
+ return rb_proc_new(missing_wrap, rb_assoc_new(obj, sym));
+ }
+ }
rb_print_undef(klass, id, 0);
}
def = me->def;
diff --git a/test/ruby/test_object.rb b/test/ruby/test_object.rb
index eff463f307..47e4a640d9 100644
--- a/test/ruby/test_object.rb
+++ b/test/ruby/test_object.rb
@@ -304,6 +304,39 @@ class TestObject < Test::Unit::TestCase
end
end
+ def test_respond_to_missing
+ c = Class.new
+ c.class_eval do
+ def respond_to_missing?(id)
+ if id == :foobar
+ true
+ else
+ false
+ end
+ end
+ def method_missing(id,*args)
+ if id == :foobar
+ return [:foo, *args]
+ else
+ super
+ end
+ end
+ end
+
+ foo = c.new
+ assert_equal([:foo], foo.foobar);
+ assert_equal([:foo, 1], foo.foobar(1));
+ assert(foo.respond_to?(:foobar))
+ assert_equal(false, foo.respond_to?(:foobarbaz))
+ assert_raise(NoMethodError) do
+ foo.foobarbaz
+ end
+
+ foobar = foo.method(:foobar)
+ assert_equal([:foo], foobar.call);
+ assert_equal([:foo, 1], foobar.call(1));
+ end
+
def test_send_with_no_arguments
assert_raise(ArgumentError) { 1.send }
end