2018-03-04 10:09:32 -05:00
|
|
|
require_relative 'spec_helper'
|
|
|
|
require_relative 'fixtures/module'
|
2017-05-07 08:04:49 -04:00
|
|
|
|
|
|
|
load_extension('module')
|
2017-10-28 13:45:46 -04:00
|
|
|
compile_extension("module_under_autoload")
|
2017-05-07 08:04:49 -04:00
|
|
|
|
|
|
|
describe "CApiModule" do
|
|
|
|
|
|
|
|
before :each do
|
|
|
|
@m = CApiModuleSpecs.new
|
|
|
|
end
|
|
|
|
|
|
|
|
describe "rb_define_global_const" do
|
|
|
|
it "defines a constant on Object" do
|
|
|
|
@m.rb_define_global_const("CApiModuleSpecsGlobalConst", 7)
|
|
|
|
::CApiModuleSpecsGlobalConst.should == 7
|
|
|
|
Object.send :remove_const, :CApiModuleSpecsGlobalConst
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe "rb_const_set given a symbol name and a value" do
|
|
|
|
it "sets a new constant on a module" do
|
|
|
|
@m.rb_const_set(CApiModuleSpecs::C, :W, 7)
|
|
|
|
CApiModuleSpecs::C::W.should == 7
|
|
|
|
end
|
|
|
|
|
|
|
|
it "sets an existing constant's value" do
|
|
|
|
-> {
|
|
|
|
@m.rb_const_set(CApiModuleSpecs::C, :Z, 8)
|
|
|
|
}.should complain(/already initialized constant/)
|
|
|
|
CApiModuleSpecs::C::Z.should == 8
|
|
|
|
end
|
2018-08-03 12:19:40 -04:00
|
|
|
|
|
|
|
it "allows arbitrary names, including constant names not valid in Ruby" do
|
|
|
|
-> {
|
|
|
|
CApiModuleSpecs::C.const_set(:_INVALID, 1)
|
|
|
|
}.should raise_error(NameError, /wrong constant name/)
|
|
|
|
|
|
|
|
@m.rb_const_set(CApiModuleSpecs::C, :_INVALID, 2)
|
|
|
|
@m.rb_const_get(CApiModuleSpecs::C, :_INVALID).should == 2
|
|
|
|
|
|
|
|
# Ruby-level should still not allow access
|
|
|
|
-> {
|
|
|
|
CApiModuleSpecs::C.const_get(:_INVALID)
|
|
|
|
}.should raise_error(NameError, /wrong constant name/)
|
|
|
|
end
|
2017-05-07 08:04:49 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
describe "rb_define_module" do
|
|
|
|
it "returns the module if it is already defined" do
|
|
|
|
mod = @m.rb_define_module("CApiModuleSpecsModuleA")
|
|
|
|
mod.const_get(:X).should == 1
|
|
|
|
end
|
|
|
|
|
|
|
|
it "raises a TypeError if the constant is not a module" do
|
|
|
|
::CApiModuleSpecsGlobalConst = 7
|
2019-07-27 06:40:09 -04:00
|
|
|
-> { @m.rb_define_module("CApiModuleSpecsGlobalConst") }.should raise_error(TypeError)
|
2017-05-07 08:04:49 -04:00
|
|
|
Object.send :remove_const, :CApiModuleSpecsGlobalConst
|
|
|
|
end
|
|
|
|
|
|
|
|
it "defines a new module at toplevel" do
|
|
|
|
mod = @m.rb_define_module("CApiModuleSpecsModuleB")
|
|
|
|
mod.should be_kind_of(Module)
|
|
|
|
mod.name.should == "CApiModuleSpecsModuleB"
|
|
|
|
::CApiModuleSpecsModuleB.should be_kind_of(Module)
|
|
|
|
Object.send :remove_const, :CApiModuleSpecsModuleB
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe "rb_define_module_under" do
|
|
|
|
it "creates a new module inside the inner class" do
|
|
|
|
mod = @m.rb_define_module_under(CApiModuleSpecs, "ModuleSpecsModuleUnder1")
|
|
|
|
mod.should be_kind_of(Module)
|
|
|
|
end
|
|
|
|
|
|
|
|
it "sets the module name" do
|
|
|
|
mod = @m.rb_define_module_under(CApiModuleSpecs, "ModuleSpecsModuleUnder2")
|
|
|
|
mod.name.should == "CApiModuleSpecs::ModuleSpecsModuleUnder2"
|
|
|
|
end
|
2017-06-15 08:48:52 -04:00
|
|
|
end
|
2017-05-07 08:04:49 -04:00
|
|
|
|
2017-06-15 08:48:52 -04:00
|
|
|
describe "rb_define_module_under" do
|
|
|
|
it "defines a module for an existing Autoload with an extension" do
|
2017-05-07 08:04:49 -04:00
|
|
|
CApiModuleSpecs::ModuleUnderAutoload.name.should == "CApiModuleSpecs::ModuleUnderAutoload"
|
|
|
|
end
|
|
|
|
|
|
|
|
it "defines a module for an existing Autoload with a ruby object" do
|
|
|
|
CApiModuleSpecs::RubyUnderAutoload.name.should == "CApiModuleSpecs::RubyUnderAutoload"
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe "rb_define_const given a String name and a value" do
|
|
|
|
it "defines a new constant on a module" do
|
|
|
|
@m.rb_define_const(CApiModuleSpecs::C, "V", 7)
|
|
|
|
CApiModuleSpecs::C::V.should == 7
|
|
|
|
end
|
|
|
|
|
|
|
|
it "sets an existing constant's value" do
|
|
|
|
-> {
|
|
|
|
@m.rb_define_const(CApiModuleSpecs::C, "Z", 9)
|
|
|
|
}.should complain(/already initialized constant/)
|
|
|
|
CApiModuleSpecs::C::Z.should == 9
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe "rb_const_defined" do
|
|
|
|
# The fixture converts C boolean test to Ruby 'true' / 'false'
|
|
|
|
it "returns C non-zero if a constant is defined" do
|
|
|
|
@m.rb_const_defined(CApiModuleSpecs::A, :X).should be_true
|
|
|
|
end
|
|
|
|
|
|
|
|
it "returns C non-zero if a constant is defined in Object" do
|
|
|
|
@m.rb_const_defined(CApiModuleSpecs::A, :Module).should be_true
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe "rb_const_defined_at" do
|
|
|
|
# The fixture converts C boolean test to Ruby 'true' / 'false'
|
|
|
|
it "returns C non-zero if a constant is defined" do
|
|
|
|
@m.rb_const_defined_at(CApiModuleSpecs::A, :X).should be_true
|
|
|
|
end
|
|
|
|
|
|
|
|
it "does not search in ancestors for the constant" do
|
|
|
|
@m.rb_const_defined_at(CApiModuleSpecs::B, :X).should be_false
|
|
|
|
end
|
|
|
|
|
|
|
|
it "does not search in Object" do
|
|
|
|
@m.rb_const_defined_at(CApiModuleSpecs::A, :Module).should be_false
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe "rb_const_get" do
|
|
|
|
it "returns a constant defined in the module" do
|
|
|
|
@m.rb_const_get(CApiModuleSpecs::A, :X).should == 1
|
|
|
|
end
|
|
|
|
|
2021-01-28 11:08:57 -05:00
|
|
|
it "returns a constant defined in the module for multiple constants" do
|
|
|
|
[:Q, :R, :S, :T].each { |x| @m.rb_const_get(CApiModuleSpecs::A, x).should == CApiModuleSpecs::A.const_get(x) }
|
|
|
|
end
|
|
|
|
|
2017-05-07 08:04:49 -04:00
|
|
|
it "returns a constant defined at toplevel" do
|
2020-11-13 07:17:24 -05:00
|
|
|
@m.rb_const_get(CApiModuleSpecs::A, :Integer).should == Integer
|
2017-05-07 08:04:49 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
it "returns a constant defined in a superclass" do
|
|
|
|
@m.rb_const_get(CApiModuleSpecs::B, :X).should == 1
|
|
|
|
end
|
|
|
|
|
|
|
|
it "calls #const_missing if the constant is not defined in the class or ancestors" do
|
|
|
|
CApiModuleSpecs::A.should_receive(:const_missing).with(:CApiModuleSpecsUndefined)
|
|
|
|
@m.rb_const_get(CApiModuleSpecs::A, :CApiModuleSpecsUndefined)
|
|
|
|
end
|
|
|
|
|
|
|
|
it "resolves autoload constants in classes" do
|
|
|
|
@m.rb_const_get(CApiModuleSpecs::A, :D).should == 123
|
|
|
|
end
|
|
|
|
|
|
|
|
it "resolves autoload constants in Object" do
|
|
|
|
@m.rb_const_get(Object, :CApiModuleSpecsAutoload).should == 123
|
|
|
|
end
|
2018-08-03 12:19:40 -04:00
|
|
|
|
|
|
|
it "allows arbitrary names, including constant names not valid in Ruby" do
|
|
|
|
-> {
|
|
|
|
CApiModuleSpecs::A.const_get(:_INVALID)
|
|
|
|
}.should raise_error(NameError, /wrong constant name/)
|
|
|
|
|
|
|
|
-> {
|
|
|
|
@m.rb_const_get(CApiModuleSpecs::A, :_INVALID)
|
|
|
|
}.should raise_error(NameError, /uninitialized constant/)
|
|
|
|
end
|
2017-05-07 08:04:49 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
describe "rb_const_get_from" do
|
|
|
|
it "returns a constant defined in the module" do
|
|
|
|
@m.rb_const_get_from(CApiModuleSpecs::B, :Y).should == 2
|
|
|
|
end
|
|
|
|
|
|
|
|
it "returns a constant defined in a superclass" do
|
|
|
|
@m.rb_const_get_from(CApiModuleSpecs::B, :X).should == 1
|
|
|
|
end
|
|
|
|
|
|
|
|
it "calls #const_missing if the constant is not defined in the class or ancestors" do
|
2020-11-13 07:17:24 -05:00
|
|
|
CApiModuleSpecs::M.should_receive(:const_missing).with(:Integer)
|
|
|
|
@m.rb_const_get_from(CApiModuleSpecs::M, :Integer)
|
2017-05-07 08:04:49 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
it "resolves autoload constants" do
|
|
|
|
@m.rb_const_get_from(CApiModuleSpecs::A, :C).should == 123
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe "rb_const_get_at" do
|
|
|
|
it "returns a constant defined in the module" do
|
|
|
|
@m.rb_const_get_at(CApiModuleSpecs::B, :Y).should == 2
|
|
|
|
end
|
|
|
|
|
|
|
|
it "resolves autoload constants" do
|
|
|
|
@m.rb_const_get_at(CApiModuleSpecs::A, :B).should == 123
|
|
|
|
end
|
|
|
|
|
|
|
|
it "calls #const_missing if the constant is not defined in the module" do
|
|
|
|
CApiModuleSpecs::B.should_receive(:const_missing).with(:X)
|
|
|
|
@m.rb_const_get_at(CApiModuleSpecs::B, :X)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe "rb_define_alias" do
|
|
|
|
it "defines an alias for an existing method" do
|
|
|
|
cls = Class.new do
|
|
|
|
def method_to_be_aliased
|
|
|
|
:method_to_be_aliased
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
@m.rb_define_alias cls, "method_alias", "method_to_be_aliased"
|
|
|
|
cls.new.method_alias.should == :method_to_be_aliased
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe "rb_alias" do
|
|
|
|
it "defines an alias for an existing method" do
|
|
|
|
cls = Class.new do
|
|
|
|
def method_to_be_aliased
|
|
|
|
:method_to_be_aliased
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
@m.rb_alias cls, :method_alias, :method_to_be_aliased
|
|
|
|
cls.new.method_alias.should == :method_to_be_aliased
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe "rb_define_global_function" do
|
|
|
|
it "defines a method on Kernel" do
|
|
|
|
@m.rb_define_global_function("module_specs_global_function")
|
|
|
|
Kernel.should have_method(:module_specs_global_function)
|
|
|
|
module_specs_global_function.should == :test_method
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe "rb_define_method" do
|
|
|
|
it "defines a method on a class" do
|
|
|
|
cls = Class.new
|
|
|
|
@m.rb_define_method(cls, "test_method")
|
|
|
|
cls.should have_instance_method(:test_method)
|
|
|
|
cls.new.test_method.should == :test_method
|
|
|
|
end
|
|
|
|
|
2021-03-27 08:02:41 -04:00
|
|
|
it "returns the correct arity when argc of the method in class is 0" do
|
2021-02-27 07:00:26 -05:00
|
|
|
cls = Class.new
|
|
|
|
@m.rb_define_method(cls, "test_method")
|
|
|
|
cls.new.method(:test_method).arity.should == 0
|
|
|
|
end
|
|
|
|
|
2021-03-27 08:02:41 -04:00
|
|
|
it "returns the correct arity when argc of the method in class is -1" do
|
|
|
|
cls = Class.new
|
|
|
|
@m.rb_define_method_c_array(cls, "test_method_c_array")
|
|
|
|
cls.new.method(:test_method_c_array).arity.should == -1
|
|
|
|
end
|
|
|
|
|
|
|
|
it "returns the correct arity when argc of the method in class is -2" do
|
|
|
|
cls = Class.new
|
|
|
|
@m.rb_define_method_ruby_array(cls, "test_method_ruby_array")
|
|
|
|
cls.new.method(:test_method_ruby_array).arity.should == -1
|
|
|
|
end
|
|
|
|
|
|
|
|
it "returns the correct arity when argc of the method in class is 2" do
|
|
|
|
cls = Class.new
|
|
|
|
@m.rb_define_method_2required(cls, "test_method_2required")
|
|
|
|
cls.new.method(:test_method_2required).arity.should == 2
|
|
|
|
end
|
|
|
|
|
2017-05-07 08:04:49 -04:00
|
|
|
it "defines a method on a module" do
|
|
|
|
mod = Module.new
|
|
|
|
@m.rb_define_method(mod, "test_method")
|
|
|
|
mod.should have_instance_method(:test_method)
|
|
|
|
end
|
2021-02-27 07:00:26 -05:00
|
|
|
|
|
|
|
it "returns the correct arity of the method in module" do
|
|
|
|
mod = Module.new
|
|
|
|
@m.rb_define_method(mod, "test_method")
|
|
|
|
mod.instance_method(:test_method).arity.should == 0
|
|
|
|
end
|
2017-05-07 08:04:49 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
describe "rb_define_module_function" do
|
|
|
|
before :each do
|
|
|
|
@mod = Module.new
|
|
|
|
@m.rb_define_module_function @mod, "test_module_function"
|
|
|
|
end
|
|
|
|
|
|
|
|
it "defines a module function" do
|
|
|
|
@mod.test_module_function.should == :test_method
|
|
|
|
end
|
|
|
|
|
2021-02-27 07:00:26 -05:00
|
|
|
it "returns the correct arity of the module function" do
|
|
|
|
@mod.method(:test_module_function).arity.should == 0
|
|
|
|
end
|
|
|
|
|
2017-05-07 08:04:49 -04:00
|
|
|
it "defines a private instance method" do
|
|
|
|
cls = Class.new
|
|
|
|
cls.include(@mod)
|
|
|
|
|
|
|
|
cls.should have_private_instance_method(:test_module_function)
|
|
|
|
end
|
2021-02-27 07:00:26 -05:00
|
|
|
|
|
|
|
it "returns the correct arity for private instance method" do
|
|
|
|
cls = Class.new
|
|
|
|
cls.include(@mod)
|
|
|
|
|
|
|
|
@mod.instance_method(:test_module_function).arity.should == 0
|
|
|
|
end
|
2017-05-07 08:04:49 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
describe "rb_define_private_method" do
|
|
|
|
it "defines a private method on a class" do
|
|
|
|
cls = Class.new
|
|
|
|
@m.rb_define_private_method(cls, "test_method")
|
|
|
|
cls.should have_private_instance_method(:test_method)
|
|
|
|
cls.new.send(:test_method).should == :test_method
|
|
|
|
end
|
|
|
|
|
|
|
|
it "defines a private method on a module" do
|
|
|
|
mod = Module.new
|
|
|
|
@m.rb_define_private_method(mod, "test_method")
|
|
|
|
mod.should have_private_instance_method(:test_method)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe "rb_define_protected_method" do
|
|
|
|
it "defines a protected method on a class" do
|
|
|
|
cls = Class.new
|
|
|
|
@m.rb_define_protected_method(cls, "test_method")
|
|
|
|
cls.should have_protected_instance_method(:test_method)
|
|
|
|
cls.new.send(:test_method).should == :test_method
|
|
|
|
end
|
|
|
|
|
|
|
|
it "defines a protected method on a module" do
|
|
|
|
mod = Module.new
|
|
|
|
@m.rb_define_protected_method(mod, "test_method")
|
|
|
|
mod.should have_protected_instance_method(:test_method)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe "rb_define_singleton_method" do
|
|
|
|
it "defines a method on the singleton class" do
|
|
|
|
cls = Class.new
|
|
|
|
a = cls.new
|
|
|
|
@m.rb_define_singleton_method a, "module_specs_singleton_method"
|
|
|
|
a.module_specs_singleton_method.should == :test_method
|
2019-07-27 06:40:09 -04:00
|
|
|
-> { cls.new.module_specs_singleton_method }.should raise_error(NoMethodError)
|
2017-05-07 08:04:49 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe "rb_undef_method" do
|
|
|
|
before :each do
|
|
|
|
@class = Class.new do
|
|
|
|
def ruby_test_method
|
|
|
|
:ruby_test_method
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
it "undef'ines a method on a class" do
|
|
|
|
@class.new.ruby_test_method.should == :ruby_test_method
|
|
|
|
@m.rb_undef_method @class, "ruby_test_method"
|
|
|
|
@class.should_not have_instance_method(:ruby_test_method)
|
|
|
|
end
|
|
|
|
|
2019-09-29 12:01:32 -04:00
|
|
|
it "undefines private methods also" do
|
|
|
|
@m.rb_undef_method @class, "initialize_copy"
|
|
|
|
-> { @class.new.dup }.should raise_error(NoMethodError)
|
|
|
|
end
|
|
|
|
|
2017-05-07 08:04:49 -04:00
|
|
|
it "does not raise exceptions when passed a missing name" do
|
2019-07-27 06:40:09 -04:00
|
|
|
-> { @m.rb_undef_method @class, "not_exist" }.should_not raise_error
|
2017-05-07 08:04:49 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
describe "when given a frozen Class" do
|
|
|
|
before :each do
|
|
|
|
@frozen = @class.dup.freeze
|
|
|
|
end
|
|
|
|
|
2020-02-08 21:07:01 -05:00
|
|
|
it "raises a FrozenError when passed a name" do
|
|
|
|
-> { @m.rb_undef_method @frozen, "ruby_test_method" }.should raise_error(FrozenError)
|
2017-05-07 08:04:49 -04:00
|
|
|
end
|
|
|
|
|
2020-02-08 21:07:01 -05:00
|
|
|
it "raises a FrozenError when passed a missing name" do
|
|
|
|
-> { @m.rb_undef_method @frozen, "not_exist" }.should raise_error(FrozenError)
|
2017-05-07 08:04:49 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe "rb_undef" do
|
|
|
|
it "undef'ines a method on a class" do
|
|
|
|
cls = Class.new do
|
|
|
|
def ruby_test_method
|
|
|
|
:ruby_test_method
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
cls.new.ruby_test_method.should == :ruby_test_method
|
|
|
|
@m.rb_undef cls, :ruby_test_method
|
|
|
|
cls.should_not have_instance_method(:ruby_test_method)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe "rb_class2name" do
|
|
|
|
it "returns the module name" do
|
|
|
|
@m.rb_class2name(CApiModuleSpecs::M).should == "CApiModuleSpecs::M"
|
|
|
|
end
|
|
|
|
end
|
2017-09-28 05:19:59 -04:00
|
|
|
|
|
|
|
describe "rb_mod_ancestors" do
|
|
|
|
it "returns an array of ancestors" do
|
|
|
|
one = Module.new
|
|
|
|
two = Module.new do
|
|
|
|
include one
|
|
|
|
end
|
|
|
|
@m.rb_mod_ancestors(two).should == [two, one]
|
|
|
|
end
|
|
|
|
end
|
2017-05-07 08:04:49 -04:00
|
|
|
end
|