mirror of
https://github.com/ruby/ruby.git
synced 2022-11-09 12:17:21 -05:00
* compile.c (NODE_CLASS, NODE_MODULE), insns.def (defineclass): raise
an exception when "class Foo::Bar" is evaluated and Foo::Bar is private. To implement this, define_type of "defineclass" is added so that the instruction can distinguish whether the class definition is scoped (class Foo::Bar) or not (class Bar). * test/ruby/test_class.rb (test_redefine_private_class), test/ruby/test_module.rb (test_define_module_under_private_constant): add tests for above. git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@30714 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
This commit is contained in:
parent
f0483c494f
commit
a2ec8666cf
5 changed files with 58 additions and 8 deletions
12
ChangeLog
12
ChangeLog
|
@ -1,3 +1,15 @@
|
||||||
|
Sat Jan 29 01:24:57 2011 Yusuke Endoh <mame@tsg.ne.jp>
|
||||||
|
|
||||||
|
* compile.c (NODE_CLASS, NODE_MODULE), insns.def (defineclass): raise
|
||||||
|
an exception when "class Foo::Bar" is evaluated and Foo::Bar is
|
||||||
|
private. To implement this, define_type of "defineclass" is added
|
||||||
|
so that the instruction can distinguish whether the class definition
|
||||||
|
is scoped (class Foo::Bar) or not (class Bar).
|
||||||
|
|
||||||
|
* test/ruby/test_class.rb (test_redefine_private_class),
|
||||||
|
test/ruby/test_module.rb
|
||||||
|
(test_define_module_under_private_constant): add tests for above.
|
||||||
|
|
||||||
Sat Jan 29 01:19:17 2011 Yusuke Endoh <mame@tsg.ne.jp>
|
Sat Jan 29 01:19:17 2011 Yusuke Endoh <mame@tsg.ne.jp>
|
||||||
|
|
||||||
* constant.h, variable.c: to ensure compatibility, rb_const_get_* must
|
* constant.h, variable.c: to ensure compatibility, rb_const_get_* must
|
||||||
|
|
|
@ -4680,10 +4680,10 @@ iseq_compile_each(rb_iseq_t *iseq, LINK_ANCHOR *ret, NODE * node, int poped)
|
||||||
node->nd_body,
|
node->nd_body,
|
||||||
rb_sprintf("<class:%s>", rb_id2name(node->nd_cpath->nd_mid)),
|
rb_sprintf("<class:%s>", rb_id2name(node->nd_cpath->nd_mid)),
|
||||||
ISEQ_TYPE_CLASS, nd_line(node));
|
ISEQ_TYPE_CLASS, nd_line(node));
|
||||||
compile_cpath(ret, iseq, node->nd_cpath);
|
VALUE noscope = compile_cpath(ret, iseq, node->nd_cpath);
|
||||||
COMPILE(ret, "super", node->nd_super);
|
COMPILE(ret, "super", node->nd_super);
|
||||||
ADD_INSN3(ret, nd_line(node), defineclass,
|
ADD_INSN3(ret, nd_line(node), defineclass,
|
||||||
ID2SYM(node->nd_cpath->nd_mid), iseqval, INT2FIX(0));
|
ID2SYM(node->nd_cpath->nd_mid), iseqval, INT2FIX(noscope ? 3 : 0));
|
||||||
|
|
||||||
if (poped) {
|
if (poped) {
|
||||||
ADD_INSN(ret, nd_line(node), pop);
|
ADD_INSN(ret, nd_line(node), pop);
|
||||||
|
@ -4696,10 +4696,10 @@ iseq_compile_each(rb_iseq_t *iseq, LINK_ANCHOR *ret, NODE * node, int poped)
|
||||||
rb_sprintf("<module:%s>", rb_id2name(node->nd_cpath->nd_mid)),
|
rb_sprintf("<module:%s>", rb_id2name(node->nd_cpath->nd_mid)),
|
||||||
ISEQ_TYPE_CLASS, nd_line(node));
|
ISEQ_TYPE_CLASS, nd_line(node));
|
||||||
|
|
||||||
compile_cpath(ret, iseq, node->nd_cpath);
|
VALUE noscope = compile_cpath(ret, iseq, node->nd_cpath);
|
||||||
ADD_INSN (ret, nd_line(node), putnil); /* dummy */
|
ADD_INSN (ret, nd_line(node), putnil); /* dummy */
|
||||||
ADD_INSN3(ret, nd_line(node), defineclass,
|
ADD_INSN3(ret, nd_line(node), defineclass,
|
||||||
ID2SYM(node->nd_cpath->nd_mid), iseqval, INT2FIX(2));
|
ID2SYM(node->nd_cpath->nd_mid), iseqval, INT2FIX(noscope ? 5 : 2));
|
||||||
if (poped) {
|
if (poped) {
|
||||||
ADD_INSN(ret, nd_line(node), pop);
|
ADD_INSN(ret, nd_line(node), pop);
|
||||||
}
|
}
|
||||||
|
|
10
insns.def
10
insns.def
|
@ -894,7 +894,8 @@ defineclass
|
||||||
VALUE klass;
|
VALUE klass;
|
||||||
|
|
||||||
switch ((int)define_type) {
|
switch ((int)define_type) {
|
||||||
case 0:
|
case 0: /* scoped: class Foo::Bar */
|
||||||
|
case 3: /* no scope: class Bar */
|
||||||
/* val is dummy. classdef returns class scope value */
|
/* val is dummy. classdef returns class scope value */
|
||||||
|
|
||||||
if (super == Qnil) {
|
if (super == Qnil) {
|
||||||
|
@ -907,7 +908,7 @@ defineclass
|
||||||
rb_autoload_load(cbase, id);
|
rb_autoload_load(cbase, id);
|
||||||
if (rb_const_defined_at(cbase, id)) {
|
if (rb_const_defined_at(cbase, id)) {
|
||||||
/* already exist */
|
/* already exist */
|
||||||
klass = rb_const_get_at(cbase, id);
|
klass = define_type == 0 ? rb_public_const_get(cbase, id) : rb_const_get_at(cbase, id);
|
||||||
if (TYPE(klass) != T_CLASS) {
|
if (TYPE(klass) != T_CLASS) {
|
||||||
rb_raise(rb_eTypeError, "%s is not a class", rb_id2name(id));
|
rb_raise(rb_eTypeError, "%s is not a class", rb_id2name(id));
|
||||||
}
|
}
|
||||||
|
@ -935,7 +936,8 @@ defineclass
|
||||||
/* super is dummy */
|
/* super is dummy */
|
||||||
klass = rb_singleton_class(cbase);
|
klass = rb_singleton_class(cbase);
|
||||||
break;
|
break;
|
||||||
case 2:
|
case 2: /* scoped: module Foo::Bar or module ::Bar */
|
||||||
|
case 5: /* no scope: module Bar */
|
||||||
/* val is dummy. classdef returns class scope value */
|
/* val is dummy. classdef returns class scope value */
|
||||||
/* super is dummy */
|
/* super is dummy */
|
||||||
|
|
||||||
|
@ -943,7 +945,7 @@ defineclass
|
||||||
|
|
||||||
/* find klass */
|
/* find klass */
|
||||||
if (rb_const_defined_at(cbase, id)) {
|
if (rb_const_defined_at(cbase, id)) {
|
||||||
klass = rb_const_get_at(cbase, id);
|
klass = define_type == 2 ? rb_public_const_get(cbase, id) : rb_const_get_at(cbase, id);
|
||||||
/* already exist */
|
/* already exist */
|
||||||
if (TYPE(klass) != T_MODULE) {
|
if (TYPE(klass) != T_MODULE) {
|
||||||
rb_raise(rb_eTypeError, "%s is not a module", rb_id2name(id));
|
rb_raise(rb_eTypeError, "%s is not a module", rb_id2name(id));
|
||||||
|
|
|
@ -240,4 +240,22 @@ class TestClass < Test::Unit::TestCase
|
||||||
def test_nested_class_removal
|
def test_nested_class_removal
|
||||||
assert_normal_exit('File.__send__(:remove_const, :Stat); at_exit{File.stat(".")}; GC.start')
|
assert_normal_exit('File.__send__(:remove_const, :Stat); at_exit{File.stat(".")}; GC.start')
|
||||||
end
|
end
|
||||||
|
|
||||||
|
class PrivateClass
|
||||||
|
end
|
||||||
|
private_constant :PrivateClass
|
||||||
|
|
||||||
|
def test_redefine_private_class
|
||||||
|
assert_raise(NameError) do
|
||||||
|
eval("class ::TestClass::PrivateClass; end")
|
||||||
|
end
|
||||||
|
eval <<-END
|
||||||
|
class ::TestClass
|
||||||
|
class PrivateClass
|
||||||
|
def foo; 42; end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
END
|
||||||
|
assert_equal(42, PrivateClass.new.foo)
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -947,6 +947,24 @@ class TestModule < Test::Unit::TestCase
|
||||||
c.private_constant(:FOO)
|
c.private_constant(:FOO)
|
||||||
assert_raise(NameError) { c::FOO }
|
assert_raise(NameError) { c::FOO }
|
||||||
assert_equal("foo", c.class_eval("FOO"))
|
assert_equal("foo", c.class_eval("FOO"))
|
||||||
|
assert_equal("foo", c.const_get("FOO"))
|
||||||
|
end
|
||||||
|
|
||||||
|
class PrivateClass
|
||||||
|
end
|
||||||
|
private_constant :PrivateClass
|
||||||
|
|
||||||
|
def test_define_module_under_private_constant
|
||||||
|
assert_raise(NameError) do
|
||||||
|
eval %q{class TestModule::PrivateClass; end}
|
||||||
|
end
|
||||||
|
assert_raise(NameError) do
|
||||||
|
eval %q{module TestModule::PrivateClass::TestModule; end}
|
||||||
|
end
|
||||||
|
eval %q{class PrivateClass; end}
|
||||||
|
eval %q{module PrivateClass::TestModule; end}
|
||||||
|
assert_instance_of(Module, PrivateClass::TestModule)
|
||||||
|
PrivateClass.class_eval { remove_const(:TestModule) }
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_public_constant
|
def test_public_constant
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue