The correct way to do this is to actually use a stubbing framework like Dan says.
For example, in rspec, you'd do:
it "reads the file contents" do
File.should_receive(:open) {|name|
if name.include?("retval")
"0\n"
else
"1\n"
end
}
File.open("foo retval").should == "0\n"
File.open("other file").should == "1\n"
end
But for the curious, here's a semi-safe way to do it without external libs.
The idea is to isolate the stub so as little code is affected by it as possible.
I named this script isolated-patch.rb
:
class File
class << self
alias_method :orig_open, :open
def stubbed_open(name, &block)
if name.include?("retval")
return "0\n"
else
return "1\n"
end
end
end
end
def get_size(path)
# contrived example
File.open(path) {|fh|
# change bytesize to size on ruby 1.8
fh.read.bytesize
}
end
# The stub will apply for the duration of the block.
# That means the block shouldn't be calling any methods where file opening
# needs to work normally.
# Warning: not thread-safe
def stub_file_open
class << File
alias_method :open, :stubbed_open
end
yield
ensure
class << File
alias_method :open, :orig_open
end
end
if __FILE__ == $0
stub_file_open do
val = File.open("what-retval") { puts "this is not run" }
puts "stubbed open() returned: #{val.inspect}"
size = get_size("test.txt")
puts "obviously wrong size of test.txt: #{size.inspect}"
end
# you need to manually create this file before running the script
size = get_size("test.txt")
puts "size of test.txt: #{size.inspect}"
end
Demo:
> echo 'foo bar' > test.txt
> ruby isolated-patch.rb
stubbed open() returned: "0\n"
obviously wrong size of test.txt: "1\n"
size of test.txt: 8