blob: 9e27f7c4e5b3a0d12ddf6a867b14f2711882d227 [file] [log] [blame]
Kevin Clark3eca0782008-06-18 00:54:53 +00001require File.dirname(__FILE__) + '/spec_helper'
2
Kevin Clark96cc5162008-06-18 00:56:08 +00003shared_examples_for "deprecation" do
4 before(:all) do
5 # ensure deprecation is turned on
6 Thrift.send :remove_const, :DEPRECATION
7 Thrift.const_set :DEPRECATION, true
8 end
9
10 after(:all) do
11 # now turn it off again
12 # no other specs should want it
13 Thrift.send :remove_const, :DEPRECATION
14 Thrift.const_set :DEPRECATION, false
15 end
Kevin Clark03a5fb12008-06-18 01:01:25 +000016
17 def ensure_const_removed(name, &block)
18 begin
19 block.call
20 ensure
21 Object.send :remove_const, name if Object.const_defined? name
22 end
23 end
Kevin Clark96cc5162008-06-18 00:56:08 +000024end
25
Kevin Clark3eca0782008-06-18 00:54:53 +000026describe 'deprecate!' do
Kevin Clark96cc5162008-06-18 00:56:08 +000027 it_should_behave_like "deprecation"
28
Kevin Clarkfc964ee2008-06-18 00:57:46 +000029 def stub_stderr(callstr, offset=1)
Kevin Clark3eca0782008-06-18 00:54:53 +000030 STDERR.should_receive(:puts).with("Warning: calling deprecated method #{callstr}")
Kevin Clarkfc964ee2008-06-18 00:57:46 +000031 line = caller.first[/\d+$/].to_i + offset
32 STDERR.should_receive(:puts).with(" from #{__FILE__}:#{line}")
Kevin Clark3eca0782008-06-18 00:54:53 +000033 end
34
35 it "should work for Module methods" do
36 mod = Module.new do
37 class << self
38 def new
39 "new"
40 end
41 deprecate! :old => :new
42 end
43 end
44 stub_stderr "#{mod.inspect}.old"
45 mod.old.should == "new"
46 end
47
48 it "should work with Modules that extend themselves" do
49 mod = Module.new do
50 extend self
51 def new
52 "new"
53 end
54 deprecate! :old => :new
55 end
56 stub_stderr "#{mod.inspect}.old"
57 mod.old.should == "new"
58 end
59
60 it "should work wtih Class methods" do
61 klass = Class.new do
62 class << self
63 def new
64 "new"
65 end
66 deprecate! :old => :new
67 end
68 end
69 stub_stderr "#{klass.inspect}.old"
70 klass.old.should == "new"
71 end
72
73 it "should work with Classes that include Modules" do
74 mod = Module.new do
75 def new
76 "new"
77 end
78 deprecate! :old => :new
79 end
80 klass = Class.new do
81 include mod
82 end
83 stub_stderr "#{klass.inspect}#old"
84 klass.new.old.should == "new"
85 end
86
87 it "should work with instance methods" do
88 klass = Class.new do
89 def new
90 "new"
91 end
92 deprecate! :old => :new
93 end
94 stub_stderr "#{klass.inspect}#old"
95 klass.new.old.should == "new"
96 end
97
98 it "should work with multiple method names" do
99 klass = Class.new do
100 def new1
101 "new 1"
102 end
103 def new2
104 "new 2"
105 end
106 deprecate! :old1 => :new1, :old2 => :new2
107 end
Kevin Clarkfc964ee2008-06-18 00:57:46 +0000108 stub_stderr("#{klass.inspect}#old1", 3).ordered
109 stub_stderr("#{klass.inspect}#old2", 3).ordered
Kevin Clark3eca0782008-06-18 00:54:53 +0000110 inst = klass.new
111 inst.old1.should == "new 1"
112 inst.old2.should == "new 2"
113 end
114
115 it "should only log a message once, even across multiple instances" do
116 klass = Class.new do
117 def new
118 "new"
119 end
120 deprecate! :old => :new
121 end
122 stub_stderr("#{klass.inspect}#old").once
123 klass.new.old.should == "new"
124 klass.new.old.should == "new"
125 end
126
127 it "should pass arguments" do
128 klass = Class.new do
129 def new(a, b)
130 "new: #{a + b}"
131 end
132 deprecate! :old => :new
133 end
134 stub_stderr("#{klass.inspect}#old")
135 klass.new.old(3, 5).should == "new: 8"
136 end
137
138 it "should pass blocks" do
139 klass = Class.new do
140 def new
141 "new #{yield}"
142 end
143 deprecate! :old => :new
144 end
145 stub_stderr("#{klass.inspect}#old")
146 klass.new.old { "block" }.should == "new block"
147 end
148
149 it "should not freeze the definition of the new method" do
150 klass = Class.new do
151 def new
152 "new"
153 end
154 deprecate! :old => :new
155 end
156 klass.send :define_method, :new do
157 "new 2"
158 end
159 stub_stderr("#{klass.inspect}#old")
160 klass.new.old.should == "new 2"
161 end
Kevin Clarkda40e8d2008-06-18 00:58:04 +0000162
163 it "should call the forwarded method in the same context as the original" do
164 klass = Class.new do
165 def myself
166 self
167 end
168 deprecate! :me => :myself
169 end
170 inst = klass.new
171 stub_stderr("#{klass.inspect}#me")
172 inst.me.should eql(inst.myself)
173 end
Kevin Clark3eca0782008-06-18 00:54:53 +0000174end
175
176describe "deprecate_class!" do
Kevin Clark96cc5162008-06-18 00:56:08 +0000177 it_should_behave_like "deprecation"
178
Kevin Clark2a8a7312008-06-18 01:01:07 +0000179 def stub_stderr(callstr, offset=1)
180 STDERR.should_receive(:puts).with("Warning: class #{callstr} is deprecated")
181 line = caller.first[/\d+$/].to_i + offset
182 STDERR.should_receive(:puts).with(" from #{__FILE__}:#{line}")
183 end
184
Kevin Clark3eca0782008-06-18 00:54:53 +0000185 it "should create a new global constant that points to the old one" do
Kevin Clark03a5fb12008-06-18 01:01:25 +0000186 ensure_const_removed :DeprecationSpecOldClass do
Kevin Clark3eca0782008-06-18 00:54:53 +0000187 klass = Class.new do
188 def foo
189 "foo"
190 end
191 end
192 deprecate_class! :DeprecationSpecOldClass => klass
Kevin Clarkeff97fc2008-06-18 01:01:40 +0000193 stub_stderr(:DeprecationSpecOldClass)
Kevin Clarkf5754ac2008-06-18 01:00:12 +0000194 ::DeprecationSpecOldClass.should eql(klass)
195 ::DeprecationSpecOldClass.new.foo.should == "foo"
Kevin Clarkf5754ac2008-06-18 01:00:12 +0000196 end
197 end
198
199 it "should create a global constant even from inside a module" do
Kevin Clark03a5fb12008-06-18 01:01:25 +0000200 ensure_const_removed :DeprecationSpecOldClass do
Kevin Clarkf5754ac2008-06-18 01:00:12 +0000201 klass = nil #define scoping
Kevin Clark03a5fb12008-06-18 01:01:25 +0000202 Module.new do
Kevin Clarkf5754ac2008-06-18 01:00:12 +0000203 klass = Class.new do
204 def foo
205 "foo"
206 end
207 end
208 deprecate_class! :DeprecationSpecOldClass => klass
209 end
Kevin Clarkeff97fc2008-06-18 01:01:40 +0000210 stub_stderr(:DeprecationSpecOldClass)
Kevin Clarkf5754ac2008-06-18 01:00:12 +0000211 ::DeprecationSpecOldClass.should eql(klass)
212 ::DeprecationSpecOldClass.new.foo.should == "foo"
Kevin Clark3eca0782008-06-18 00:54:53 +0000213 end
214 end
Kevin Clark2a8a7312008-06-18 01:01:07 +0000215
216 it "should not prevent the deprecated class from being a superclass" do
Kevin Clark03a5fb12008-06-18 01:01:25 +0000217 ensure_const_removed :DeprecationSpecOldClass do
Kevin Clark2a8a7312008-06-18 01:01:07 +0000218 klass = Class.new do
219 def foo
220 "foo"
221 end
222 end
223 deprecate_class! :DeprecationSpecOldClass => klass
224 subklass = Class.new(::DeprecationSpecOldClass) do
225 def foo
226 "subclass #{super}"
227 end
228 end
Kevin Clarkeff97fc2008-06-18 01:01:40 +0000229 stub_stderr(:DeprecationSpecOldClass)
Kevin Clark2a8a7312008-06-18 01:01:07 +0000230 subklass.superclass.should eql(klass)
231 subklass.new.foo.should == "subclass foo"
Kevin Clark03a5fb12008-06-18 01:01:25 +0000232 end
233 end
234end
235
236describe "deprecate_module!" do
237 it_should_behave_like "deprecation"
238
239 def stub_stderr(callstr, offset=1)
240 STDERR.should_receive(:puts).with("Warning: module #{callstr} is deprecated")
241 line = caller.first[/\d+$/].to_i + offset
242 STDERR.should_receive(:puts).with(" from #{__FILE__}:#{line}")
243 end
244
245 it "should create a new global constant that points to the old one" do
246 ensure_const_removed :DeprecationSpecOldModule do
247 mod = Module.new do
248 def self.foo
249 "foo"
250 end
251 end
252 deprecate_module! :DeprecationSpecOldModule => mod
Kevin Clarkeff97fc2008-06-18 01:01:40 +0000253 stub_stderr(:DeprecationSpecOldModule)
Kevin Clark03a5fb12008-06-18 01:01:25 +0000254 ::DeprecationSpecOldModule.should eql(mod)
255 ::DeprecationSpecOldModule.foo.should == "foo"
256 end
257 end
258
259 it "should create a global constant even from inside a module" do
260 ensure_const_removed :DeprecationSpecOldModule do
261 mod = nil # scoping
262 Module.new do
263 mod = Module.new do
264 def self.foo
265 "foo"
266 end
267 end
268 deprecate_module! :DeprecationSpecOldModule => mod
269 end
Kevin Clarkeff97fc2008-06-18 01:01:40 +0000270 stub_stderr(:DeprecationSpecOldModule)
Kevin Clark03a5fb12008-06-18 01:01:25 +0000271 ::DeprecationSpecOldModule.should eql(mod)
272 ::DeprecationSpecOldModule.foo.should == "foo"
273 end
274 end
275
276 it "should work for modules that extend themselves" do
277 ensure_const_removed :DeprecationSpecOldModule do
278 mod = Module.new do
279 extend self
280 def foo
281 "foo"
282 end
283 end
284 deprecate_module! :DeprecationSpecOldModule => mod
Kevin Clarkeff97fc2008-06-18 01:01:40 +0000285 stub_stderr(:DeprecationSpecOldModule)
Kevin Clark03a5fb12008-06-18 01:01:25 +0000286 ::DeprecationSpecOldModule.should eql(mod)
287 ::DeprecationSpecOldModule.foo.should == "foo"
288 end
289 end
290
291 it "should work for modules included in other modules" do
292 ensure_const_removed :DeprecationSpecOldModule do
293 mod = Module.new do
294 def foo
295 "foo"
296 end
297 end
298 deprecate_module! :DeprecationSpecOldModule => mod
299 mod2 = Module.new do
300 class << self
301 include ::DeprecationSpecOldModule
302 end
303 end
Kevin Clarkeff97fc2008-06-18 01:01:40 +0000304 stub_stderr(:DeprecationSpecOldModule)
Kevin Clark03a5fb12008-06-18 01:01:25 +0000305 mod2.foo.should == "foo"
306 end
307 end
308
309 it "should work for modules included in classes" do
310 ensure_const_removed :DeprecationSpecOldModule do
311 mod = Module.new do
312 def foo
313 "foo"
314 end
315 end
316 deprecate_module! :DeprecationSpecOldModule => mod
317 klass = Class.new do
318 include ::DeprecationSpecOldModule
319 end
Kevin Clarkeff97fc2008-06-18 01:01:40 +0000320 stub_stderr(:DeprecationSpecOldModule)
Kevin Clark03a5fb12008-06-18 01:01:25 +0000321 klass.new.foo.should == "foo"
Kevin Clark2a8a7312008-06-18 01:01:07 +0000322 end
323 end
Kevin Clark3eca0782008-06-18 00:54:53 +0000324end