Solve the information leak problem between deprecations. Unfortunately this was caused by a nasty ruby bug.
git-svn-id: https://svn.apache.org/repos/asf/incubator/thrift/trunk@668930 13f79535-47bb-0310-9956-ffa450edef68
diff --git a/lib/rb/lib/thrift/deprecation.rb b/lib/rb/lib/thrift/deprecation.rb
index 68ecb71..094769b 100644
--- a/lib/rb/lib/thrift/deprecation.rb
+++ b/lib/rb/lib/thrift/deprecation.rb
@@ -31,74 +31,91 @@
end
end
-module Thrift::DeprecationProxy
+module Thrift::DeprecationProxy # :nodoc:
+ # there's a really weird bug in Ruby where class variables behave wrong
+ # when used in a Class.new or #class_eval rather than in a class foo block.
+ CLASS_MAPPING = {}
+ MODULE_MAPPING = {}
def self.new_class(obj, name)
- Class.new(obj) do
+ klass = Class.new(obj) do
klass = self
- @@self = klass
- @@obj = obj
instance_methods.sort.reject { |x| [:__id__,:__send__].include? x.to_sym }.each do |sym|
undef_method sym
end
+ define_method :__thrift_deprecation_proxy_klass do
+ klass
+ end
def method_missing(sym, *args, &block)
- @@obj.instance_method(sym).bind(self).call(*args, &block)
+ klass = __thrift_deprecation_proxy_klass
+ obj, name, warned = CLASS_MAPPING[klass.__id__]
+ obj.instance_method(sym).bind(self).call(*args, &block)
end
(class << self;self;end).class_eval do
- @@self = klass
- @@obj = obj
- @@name = name
- @@warned = false
instance_methods.sort.reject { |x| [:__id__,:__send__].include? x.to_sym }.each do |sym|
undef_method sym
end
+ define_method :__thrift_deprecation_proxy_klass do
+ klass
+ end
def method_missing(sym, *args, &block)
- unless @@warned
- STDERR.puts "Warning: class #{@@name} is deprecated"
+ klass = __thrift_deprecation_proxy_klass
+ obj, name, warned = CLASS_MAPPING[klass.__id__]
+ unless warned
+ STDERR.puts "Warning: class #{name} is deprecated"
STDERR.puts " from #{caller.first}"
- @@warned = true
+ CLASS_MAPPING[__thrift_deprecation_proxy_klass.__id__][2] = true
end
- if @@self.__id__ == self.__id__
- @@obj.send sym, *args, &block
+ if klass.__id__ == self.__id__
+ obj.send sym, *args, &block
else
- @@obj.method(sym).unbind.bind(self).call(*args, &block)
+ obj.method(sym).unbind.bind(self).call(*args, &block)
end
end
end
end
+ CLASS_MAPPING[klass.__id__] = [obj, name, false]
+ klass
end
def self.new_module(obj, name)
- Module.new do
- @@obj = obj
- @@warned = false
- @@name = name
+ mod = Module.new do
include obj
instance_methods.sort.reject { |x| [:__id__,:__send__].include? x.to_sym }.each do |sym|
undef_method sym
end
+ define_method :__thrift_deprecation_proxy_module do
+ mod
+ end
def method_missing(sym, *args, &block)
- unless @@warned
- STDERR.puts "Warning: module #{@@name} is deprecated"
+ mod = __thrift_deprecation_proxy_module
+ obj, name, warned = MODULE_MAPPING[mod.__id__]
+ unless warned
+ STDERR.puts "Warning: module #{name} is deprecated"
STDERR.puts " from #{caller.first}"
- @@warned = true
+ MODULE_MAPPING[mod.__id__][2] = true
end
- @@obj.instance_method(sym).bind(self).call(*args, &block)
+ obj.instance_method(sym).bind(self).call(*args, &block)
end
(class << self;self;end).class_eval do
- @@obj = obj
- @@warned = false
instance_methods.sort.reject { |x| [:__id__,:__send__].include? x.to_sym }.each do |sym|
undef_method sym
end
+ define_method :__thrift_deprecation_proxy_module do
+ mod
+ end
def method_missing(sym, *args, &block)
- unless @@warned
- STDERR.puts "Warning: module #{@@name} is deprecated"
+ mod = __thrift_deprecation_proxy_module
+ obj, name, warned = MODULE_MAPPING[mod.__id__]
+ unless warned
+ STDERR.puts "Warning: module #{name} is deprecated"
STDERR.puts " from #{caller.first}"
- @@warned = true
+ MODULE_MAPPING[mod.__id__][2] = true
end
- @@obj.send sym, *args, &block
+ obj.send sym, *args, &block
end
end
end
+ MODULE_MAPPING[mod.__id__] = [obj, name, false]
+ mod
end
end
diff --git a/lib/rb/spec/deprecation_spec.rb b/lib/rb/spec/deprecation_spec.rb
index 6c83ccd..6c3bdc9 100644
--- a/lib/rb/spec/deprecation_spec.rb
+++ b/lib/rb/spec/deprecation_spec.rb
@@ -176,8 +176,8 @@
describe "deprecate_class!" do
it_should_behave_like "deprecation"
- def stub_stderr(callstr, offset=1)
- STDERR.should_receive(:puts).with("Warning: class #{callstr} is deprecated")
+ def stub_stderr(klass, offset=1)
+ STDERR.should_receive(:puts).with("Warning: class #{klass} is deprecated")
line = caller.first[/\d+$/].to_i + offset
STDERR.should_receive(:puts).with(" from #{__FILE__}:#{line}")
end
@@ -259,8 +259,8 @@
describe "deprecate_module!" do
it_should_behave_like "deprecation"
- def stub_stderr(callstr, offset=1)
- STDERR.should_receive(:puts).with("Warning: module #{callstr} is deprecated")
+ def stub_stderr(mod, offset=1)
+ STDERR.should_receive(:puts).with("Warning: module #{mod} is deprecated")
line = caller.first[/\d+$/].to_i + offset
STDERR.should_receive(:puts).with(" from #{__FILE__}:#{line}")
end
@@ -344,4 +344,27 @@
klass.new.foo.should == "foo"
end
end
+
+ it "should not bleed info between deprecations" do
+ ensure_const_removed :DeprecationSpecOldModule do
+ ensure_const_removed :DeprecationSpecOldModule2 do
+ mod = Module.new do
+ def self.foo
+ "foo"
+ end
+ end
+ deprecate_module! :DeprecationSpecOldModule => mod
+ mod2 = Module.new do
+ def self.bar
+ "bar"
+ end
+ end
+ deprecate_module! :DeprecationSpecOldModule2 => mod2
+ stub_stderr(:DeprecationSpecOldModule)
+ ::DeprecationSpecOldModule.foo.should == "foo"
+ stub_stderr(:DeprecationSpecOldModule2)
+ ::DeprecationSpecOldModule2.bar.should == "bar"
+ end
+ end
+ end
end