2017-01-06 05:05:35 +03:00
|
|
|
# frozen_string_literal: true
|
2008-01-16 11:36:56 +03:00
|
|
|
require 'test/unit'
|
|
|
|
require 'delegate'
|
|
|
|
|
|
|
|
class TestDelegateClass < Test::Unit::TestCase
|
2017-06-24 06:35:29 +03:00
|
|
|
module PP
|
|
|
|
def mu_pp(obj)
|
|
|
|
str = super
|
|
|
|
str = "#<#{obj.class}: #{str}>" if Delegator === obj
|
|
|
|
str
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2008-01-16 11:36:56 +03:00
|
|
|
module M
|
|
|
|
attr_reader :m
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_extend
|
|
|
|
obj = DelegateClass(Array).new([])
|
|
|
|
obj.instance_eval { @m = :m }
|
|
|
|
obj.extend M
|
|
|
|
assert_equal(:m, obj.m, "[ruby-dev:33116]")
|
|
|
|
end
|
2008-05-24 05:34:37 +04:00
|
|
|
|
2019-05-12 02:32:00 +03:00
|
|
|
def test_delegate_class_block
|
|
|
|
klass = DelegateClass(Array) do
|
|
|
|
alias foo first
|
|
|
|
end
|
|
|
|
assert_equal(1, klass.new([1]).foo)
|
|
|
|
end
|
|
|
|
|
2008-05-24 05:34:37 +04:00
|
|
|
def test_systemcallerror_eq
|
|
|
|
e = SystemCallError.new(0)
|
|
|
|
assert((SimpleDelegator.new(e) == e) == (e == SimpleDelegator.new(e)), "[ruby-dev:34808]")
|
|
|
|
end
|
2008-10-18 19:18:49 +04:00
|
|
|
|
|
|
|
class Myclass < DelegateClass(Array);end
|
|
|
|
|
|
|
|
def test_delegateclass_class
|
|
|
|
myclass=Myclass.new([])
|
|
|
|
assert_equal(Myclass,myclass.class)
|
2010-02-05 10:38:10 +03:00
|
|
|
assert_equal(Myclass,myclass.dup.class,'[ruby-dev:40313]')
|
|
|
|
assert_equal(Myclass,myclass.clone.class,'[ruby-dev:40313]')
|
2008-10-18 19:18:49 +04:00
|
|
|
end
|
|
|
|
|
|
|
|
def test_simpledelegator_class
|
|
|
|
simple=SimpleDelegator.new([])
|
|
|
|
assert_equal(SimpleDelegator,simple.class)
|
2010-02-05 10:38:10 +03:00
|
|
|
assert_equal(SimpleDelegator,simple.dup.class)
|
|
|
|
assert_equal(SimpleDelegator,simple.clone.class)
|
2008-10-18 19:18:49 +04:00
|
|
|
end
|
2009-10-06 17:07:12 +04:00
|
|
|
|
2019-08-26 07:11:46 +03:00
|
|
|
def test_simpledelegator_clone
|
|
|
|
simple=SimpleDelegator.new([])
|
|
|
|
simple.freeze
|
|
|
|
|
|
|
|
clone = simple.clone
|
|
|
|
assert_predicate clone, :frozen?
|
|
|
|
assert_predicate clone.__getobj__, :frozen?
|
|
|
|
assert_equal true, Kernel.instance_method(:frozen?).bind(clone).call
|
|
|
|
|
|
|
|
clone = simple.clone(freeze: false)
|
|
|
|
assert_not_predicate clone, :frozen?
|
|
|
|
assert_not_predicate clone.__getobj__, :frozen?
|
|
|
|
assert_equal false, Kernel.instance_method(:frozen?).bind(clone).call
|
|
|
|
end
|
|
|
|
|
2009-10-06 17:07:12 +04:00
|
|
|
class Object
|
|
|
|
def m
|
|
|
|
:o
|
|
|
|
end
|
2010-02-04 02:15:55 +03:00
|
|
|
private
|
|
|
|
def delegate_test_m
|
|
|
|
:o
|
|
|
|
end
|
2009-10-06 17:07:12 +04:00
|
|
|
end
|
|
|
|
|
|
|
|
class Foo
|
|
|
|
def m
|
|
|
|
:m
|
|
|
|
end
|
2010-02-04 02:15:55 +03:00
|
|
|
def delegate_test_m
|
|
|
|
:m
|
|
|
|
end
|
2009-10-06 17:07:12 +04:00
|
|
|
end
|
|
|
|
|
|
|
|
class Bar < DelegateClass(Foo)
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_override
|
2010-02-05 10:38:10 +03:00
|
|
|
foo = Foo.new
|
|
|
|
foo2 = SimpleDelegator.new(foo)
|
|
|
|
bar = Bar.new(foo)
|
2009-10-06 17:07:12 +04:00
|
|
|
assert_equal(:o, Object.new.m)
|
2010-02-05 10:38:10 +03:00
|
|
|
assert_equal(:m, foo.m)
|
|
|
|
assert_equal(:m, foo2.m)
|
|
|
|
assert_equal(:m, bar.m)
|
2010-02-04 02:15:55 +03:00
|
|
|
bug = '[ruby-dev:39154]'
|
2010-02-05 10:38:10 +03:00
|
|
|
assert_equal(:m, foo2.send(:delegate_test_m), bug)
|
|
|
|
assert_equal(:m, bar.send(:delegate_test_m), bug)
|
2009-10-06 17:07:12 +04:00
|
|
|
end
|
2009-12-07 08:11:10 +03:00
|
|
|
|
2019-05-24 07:10:40 +03:00
|
|
|
class Parent
|
|
|
|
def parent_public; end
|
|
|
|
|
|
|
|
protected
|
|
|
|
|
|
|
|
def parent_protected; end
|
2020-07-10 01:01:10 +03:00
|
|
|
|
|
|
|
private
|
|
|
|
|
|
|
|
def parent_private; end
|
2019-05-24 07:10:40 +03:00
|
|
|
end
|
|
|
|
|
|
|
|
class Child < DelegateClass(Parent)
|
|
|
|
end
|
|
|
|
|
|
|
|
class Parent
|
|
|
|
def parent_public_added; end
|
|
|
|
|
|
|
|
protected
|
|
|
|
|
|
|
|
def parent_protected_added; end
|
2020-07-10 01:01:10 +03:00
|
|
|
|
|
|
|
private
|
|
|
|
|
|
|
|
def parent_private_added; end
|
2019-05-24 07:10:40 +03:00
|
|
|
end
|
|
|
|
|
|
|
|
def test_public_instance_methods
|
|
|
|
ignores = Object.public_instance_methods | Delegator.public_instance_methods
|
|
|
|
assert_equal([:parent_public, :parent_public_added], (Child.public_instance_methods - ignores).sort)
|
|
|
|
assert_equal([:parent_public, :parent_public_added], (Child.new(Parent.new).public_methods - ignores).sort)
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_protected_instance_methods
|
|
|
|
ignores = Object.protected_instance_methods | Delegator.protected_instance_methods
|
|
|
|
assert_equal([:parent_protected, :parent_protected_added], (Child.protected_instance_methods - ignores).sort)
|
|
|
|
assert_equal([:parent_protected, :parent_protected_added], (Child.new(Parent.new).protected_methods - ignores).sort)
|
|
|
|
end
|
|
|
|
|
2020-07-10 01:01:10 +03:00
|
|
|
def test_instance_methods
|
|
|
|
ignores = Object.instance_methods | Delegator.instance_methods
|
|
|
|
assert_equal([:parent_protected, :parent_protected_added, :parent_public, :parent_public_added], (Child.instance_methods - ignores).sort)
|
|
|
|
assert_equal([:parent_protected, :parent_protected_added, :parent_public, :parent_public_added], (Child.new(Parent.new).methods - ignores).sort)
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_DelegateClass_instance_method
|
|
|
|
assert_instance_of UnboundMethod, Child.instance_method(:parent_public)
|
|
|
|
assert_instance_of UnboundMethod, Child.instance_method(:parent_public_added)
|
|
|
|
assert_instance_of UnboundMethod, Child.instance_method(:parent_protected)
|
|
|
|
assert_instance_of UnboundMethod, Child.instance_method(:parent_protected_added)
|
|
|
|
assert_raise(NameError) { Child.instance_method(:parent_private) }
|
|
|
|
assert_raise(NameError) { Child.instance_method(:parent_private_added) }
|
|
|
|
assert_instance_of UnboundMethod, Child.instance_method(:to_s)
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_DelegateClass_public_instance_method
|
|
|
|
assert_instance_of UnboundMethod, Child.public_instance_method(:parent_public)
|
|
|
|
assert_instance_of UnboundMethod, Child.public_instance_method(:parent_public_added)
|
|
|
|
assert_raise(NameError) { Child.public_instance_method(:parent_protected) }
|
|
|
|
assert_raise(NameError) { Child.public_instance_method(:parent_protected_added) }
|
|
|
|
assert_raise(NameError) { Child.instance_method(:parent_private) }
|
|
|
|
assert_raise(NameError) { Child.instance_method(:parent_private_added) }
|
|
|
|
assert_instance_of UnboundMethod, Child.public_instance_method(:to_s)
|
|
|
|
end
|
|
|
|
|
2009-12-07 08:11:10 +03:00
|
|
|
class IV < DelegateClass(Integer)
|
|
|
|
attr_accessor :var
|
|
|
|
|
|
|
|
def initialize
|
|
|
|
@var = 1
|
2010-02-06 05:41:04 +03:00
|
|
|
super(0)
|
2009-12-07 08:11:10 +03:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_marshal
|
|
|
|
bug1744 = '[ruby-core:24211]'
|
|
|
|
c = IV.new
|
|
|
|
assert_equal(1, c.var)
|
|
|
|
d = Marshal.load(Marshal.dump(c))
|
|
|
|
assert_equal(1, d.var, bug1744)
|
|
|
|
end
|
2010-02-04 02:15:55 +03:00
|
|
|
|
|
|
|
def test_copy_frozen
|
|
|
|
bug2679 = '[ruby-dev:40242]'
|
|
|
|
a = [42, :hello].freeze
|
|
|
|
d = SimpleDelegator.new(a)
|
|
|
|
assert_nothing_raised(bug2679) {d.dup[0] += 1}
|
2017-12-12 03:46:34 +03:00
|
|
|
assert_raise(FrozenError) {d.clone[0] += 1}
|
2010-02-08 10:43:54 +03:00
|
|
|
d.freeze
|
|
|
|
assert(d.clone.frozen?)
|
|
|
|
assert(!d.dup.frozen?)
|
2010-02-04 02:15:55 +03:00
|
|
|
end
|
2010-02-05 10:38:10 +03:00
|
|
|
|
|
|
|
def test_frozen
|
|
|
|
d = SimpleDelegator.new([1, :foo])
|
|
|
|
d.freeze
|
2017-12-12 03:46:34 +03:00
|
|
|
assert_raise(FrozenError, '[ruby-dev:40314]#1') {d.__setobj__("foo")}
|
2010-02-05 10:38:10 +03:00
|
|
|
assert_equal([1, :foo], d)
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_instance_method
|
|
|
|
s = SimpleDelegator.new("foo")
|
|
|
|
m = s.method("upcase")
|
|
|
|
s.__setobj__([1,2,3])
|
|
|
|
assert_raise(NoMethodError, '[ruby-dev:40314]#3') {m.call}
|
|
|
|
end
|
|
|
|
|
2011-12-04 07:05:03 +04:00
|
|
|
def test_methods
|
|
|
|
s = SimpleDelegator.new("foo")
|
|
|
|
assert_equal([], s.methods(false))
|
|
|
|
def s.bar; end
|
|
|
|
assert_equal([:bar], s.methods(false))
|
|
|
|
end
|
|
|
|
|
2017-06-24 06:35:29 +03:00
|
|
|
def test_eql?
|
|
|
|
extend PP
|
|
|
|
s0 = SimpleDelegator.new("foo")
|
|
|
|
s1 = SimpleDelegator.new("bar")
|
|
|
|
s2 = SimpleDelegator.new("foo")
|
|
|
|
assert_operator(s0, :eql?, s0)
|
|
|
|
assert_operator(s0, :eql?, "foo")
|
|
|
|
assert_operator(s0, :eql?, s2)
|
|
|
|
assert_not_operator(s0, :eql?, s1)
|
|
|
|
assert_not_operator(s0, :eql?, "bar")
|
|
|
|
end
|
|
|
|
|
Add Module#ruby2_keywords for passing keywords through regular argument splats
This approach uses a flag bit on the final hash object in the regular splat,
as opposed to a previous approach that used a VM frame flag. The hash flag
approach is less invasive, and handles some cases that the VM frame flag
approach does not, such as saving the argument splat array and splatting it
later:
ruby2_keywords def foo(*args)
@args = args
bar
end
def bar
baz(*@args)
end
def baz(*args, **kw)
[args, kw]
end
foo(a:1) #=> [[], {a: 1}]
foo({a: 1}, **{}) #=> [[{a: 1}], {}]
foo({a: 1}) #=> 2.7: [[], {a: 1}] # and warning
foo({a: 1}) #=> 3.0: [[{a: 1}], {}]
It doesn't handle some cases that the VM frame flag handles, such as when
the final hash object is replaced using Hash#merge, but those cases are
probably less common and are unlikely to properly support keyword
argument separation.
Use ruby2_keywords to handle argument delegation in the delegate library.
2019-09-21 19:03:36 +03:00
|
|
|
def test_keyword_and_hash
|
|
|
|
foo = Object.new
|
|
|
|
def foo.bar(*args)
|
|
|
|
args
|
|
|
|
end
|
|
|
|
def foo.foo(*args, **kw)
|
|
|
|
[args, kw]
|
|
|
|
end
|
|
|
|
d = SimpleDelegator.new(foo)
|
|
|
|
assert_equal([[], {}], d.foo)
|
|
|
|
assert_equal([], d.bar)
|
|
|
|
assert_equal([[], {:a=>1}], d.foo(:a=>1))
|
|
|
|
assert_equal([{:a=>1}], d.bar(:a=>1))
|
2019-10-05 01:41:13 +03:00
|
|
|
assert_equal([[{:a=>1}], {}], d.foo({:a=>1}))
|
Add Module#ruby2_keywords for passing keywords through regular argument splats
This approach uses a flag bit on the final hash object in the regular splat,
as opposed to a previous approach that used a VM frame flag. The hash flag
approach is less invasive, and handles some cases that the VM frame flag
approach does not, such as saving the argument splat array and splatting it
later:
ruby2_keywords def foo(*args)
@args = args
bar
end
def bar
baz(*@args)
end
def baz(*args, **kw)
[args, kw]
end
foo(a:1) #=> [[], {a: 1}]
foo({a: 1}, **{}) #=> [[{a: 1}], {}]
foo({a: 1}) #=> 2.7: [[], {a: 1}] # and warning
foo({a: 1}) #=> 3.0: [[{a: 1}], {}]
It doesn't handle some cases that the VM frame flag handles, such as when
the final hash object is replaced using Hash#merge, but those cases are
probably less common and are unlikely to properly support keyword
argument separation.
Use ruby2_keywords to handle argument delegation in the delegate library.
2019-09-21 19:03:36 +03:00
|
|
|
assert_equal([{:a=>1}], d.bar({:a=>1}))
|
|
|
|
end
|
|
|
|
|
2010-02-05 10:38:10 +03:00
|
|
|
class Foo
|
|
|
|
private
|
|
|
|
def delegate_test_private
|
|
|
|
:m
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_private_method
|
|
|
|
foo = Foo.new
|
|
|
|
d = SimpleDelegator.new(foo)
|
|
|
|
assert_raise(NoMethodError) {foo.delegate_test_private}
|
|
|
|
assert_equal(:m, foo.send(:delegate_test_private))
|
|
|
|
assert_raise(NoMethodError, '[ruby-dev:40314]#4') {d.delegate_test_private}
|
|
|
|
assert_raise(NoMethodError, '[ruby-dev:40314]#5') {d.send(:delegate_test_private)}
|
|
|
|
end
|
2013-11-15 12:07:39 +04:00
|
|
|
|
|
|
|
def test_global_function
|
|
|
|
klass = Class.new do
|
|
|
|
def open
|
|
|
|
end
|
|
|
|
end
|
|
|
|
obj = klass.new
|
|
|
|
d = SimpleDelegator.new(obj)
|
|
|
|
assert_nothing_raised(ArgumentError) {obj.open}
|
|
|
|
assert_nothing_raised(ArgumentError) {d.open}
|
|
|
|
assert_nothing_raised(ArgumentError) {d.send(:open)}
|
|
|
|
end
|
2013-11-19 20:27:40 +04:00
|
|
|
|
|
|
|
def test_send_method_in_delegator
|
|
|
|
d = Class.new(SimpleDelegator) do
|
|
|
|
def foo
|
|
|
|
"foo"
|
|
|
|
end
|
|
|
|
end.new(Object.new)
|
|
|
|
assert_equal("foo", d.send(:foo))
|
|
|
|
end
|
2013-11-21 13:47:31 +04:00
|
|
|
|
|
|
|
def test_unset_simple_delegator
|
|
|
|
d = SimpleDelegator.allocate
|
|
|
|
assert_raise_with_message(ArgumentError, /not delegated/) {
|
|
|
|
d.__getobj__
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_unset_delegate_class
|
|
|
|
d = IV.allocate
|
|
|
|
assert_raise_with_message(ArgumentError, /not delegated/) {
|
|
|
|
d.__getobj__
|
|
|
|
}
|
|
|
|
end
|
2013-12-04 07:47:57 +04:00
|
|
|
|
|
|
|
class Bug9155 < DelegateClass(Integer)
|
|
|
|
def initialize(value)
|
|
|
|
super(Integer(value))
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_global_method_if_no_target
|
|
|
|
bug9155 = '[ruby-core:58572] [Bug #9155]'
|
|
|
|
x = assert_nothing_raised(ArgumentError, bug9155) {break Bug9155.new(1)}
|
|
|
|
assert_equal(1, x.to_i, bug9155)
|
|
|
|
end
|
2014-01-17 15:05:03 +04:00
|
|
|
|
|
|
|
class Bug9403
|
|
|
|
Name = '[ruby-core:59718] [Bug #9403]'
|
|
|
|
SD = SimpleDelegator.new(new)
|
|
|
|
class << SD
|
|
|
|
def method_name
|
|
|
|
__method__
|
|
|
|
end
|
|
|
|
def callee_name
|
|
|
|
__callee__
|
|
|
|
end
|
|
|
|
alias aliased_name callee_name
|
|
|
|
def dir_name
|
|
|
|
__dir__
|
|
|
|
end
|
|
|
|
end
|
|
|
|
dc = DelegateClass(self)
|
|
|
|
dc.class_eval do
|
|
|
|
def method_name
|
|
|
|
__method__
|
|
|
|
end
|
|
|
|
def callee_name
|
|
|
|
__callee__
|
|
|
|
end
|
|
|
|
alias aliased_name callee_name
|
|
|
|
def dir_name
|
|
|
|
__dir__
|
|
|
|
end
|
|
|
|
end
|
|
|
|
DC = dc.new(new)
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_method_in_simple_delegator
|
|
|
|
assert_equal(:method_name, Bug9403::SD.method_name, Bug9403::Name)
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_callee_in_simple_delegator
|
|
|
|
assert_equal(:callee_name, Bug9403::SD.callee_name, Bug9403::Name)
|
|
|
|
assert_equal(:aliased_name, Bug9403::SD.aliased_name, Bug9403::Name)
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_dir_in_simple_delegator
|
|
|
|
assert_equal(__dir__, Bug9403::SD.dir_name, Bug9403::Name)
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_method_in_delegator_class
|
|
|
|
assert_equal(:method_name, Bug9403::DC.method_name, Bug9403::Name)
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_callee_in_delegator_class
|
|
|
|
assert_equal(:callee_name, Bug9403::DC.callee_name, Bug9403::Name)
|
|
|
|
assert_equal(:aliased_name, Bug9403::DC.aliased_name, Bug9403::Name)
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_dir_in_delegator_class
|
|
|
|
assert_equal(__dir__, Bug9403::DC.dir_name, Bug9403::Name)
|
|
|
|
end
|
2016-08-30 20:29:07 +03:00
|
|
|
|
|
|
|
def test_module_methods_vs_kernel_methods
|
|
|
|
delegate = SimpleDelegator.new(Object.new)
|
|
|
|
assert_raise(NoMethodError) do
|
|
|
|
delegate.constants
|
|
|
|
end
|
|
|
|
end
|
2019-08-25 10:04:14 +03:00
|
|
|
|
|
|
|
def test_basicobject
|
|
|
|
o = BasicObject.new
|
|
|
|
def o.bar; 1; end
|
|
|
|
delegate = SimpleDelegator.new(o)
|
|
|
|
assert_equal(1, delegate.bar)
|
|
|
|
assert_raise(NoMethodError, /undefined method `foo' for/) { delegate.foo }
|
|
|
|
end
|
2020-01-21 05:47:04 +03:00
|
|
|
|
2020-02-03 14:29:37 +03:00
|
|
|
def test_basicobject_respond_to
|
|
|
|
o = BasicObject.new
|
|
|
|
def o.bar
|
|
|
|
nil
|
|
|
|
end
|
|
|
|
|
|
|
|
def o.respond_to?(method, include_private=false)
|
|
|
|
return false if method == :bar
|
|
|
|
::Kernel.instance_method(:respond_to?).bind_call(self, method, include_private)
|
|
|
|
end
|
|
|
|
delegate = SimpleDelegator.new(o)
|
|
|
|
refute delegate.respond_to?(:bar)
|
|
|
|
end
|
|
|
|
|
2020-01-21 05:47:04 +03:00
|
|
|
def test_keyword_argument
|
|
|
|
k = EnvUtil.labeled_class("Target") do
|
|
|
|
def test(a, k:) [a, k]; end
|
|
|
|
end
|
|
|
|
a = DelegateClass(k).new(k.new)
|
|
|
|
assert_equal([1, 0], a.test(1, k: 0))
|
|
|
|
end
|
2008-01-16 11:36:56 +03:00
|
|
|
end
|