2012-05-04 17:46:01 -04:00
|
|
|
# encoding: utf-8
|
2011-02-19 15:44:08 -05:00
|
|
|
######################################################################
|
|
|
|
# This file is imported from the minitest project.
|
|
|
|
# DO NOT make modifications in this repo. They _will_ be reverted!
|
|
|
|
# File a patch instead and assign it to Ryan Davis.
|
|
|
|
######################################################################
|
2008-10-09 21:18:03 -04:00
|
|
|
|
2012-05-04 17:46:01 -04:00
|
|
|
class MockExpectationError < StandardError # :nodoc:
|
|
|
|
end # omg... worst bug ever. rdoc doesn't allow 1-liners
|
2008-10-09 21:18:03 -04:00
|
|
|
|
2010-12-24 23:55:15 -05:00
|
|
|
##
|
|
|
|
# A simple and clean mock object framework.
|
|
|
|
|
2008-10-09 21:18:03 -04:00
|
|
|
module MiniTest
|
2010-12-24 23:55:15 -05:00
|
|
|
|
|
|
|
##
|
|
|
|
# All mock objects are an instance of Mock
|
|
|
|
|
2008-10-09 21:18:03 -04:00
|
|
|
class Mock
|
2011-08-23 17:47:25 -04:00
|
|
|
alias :__respond_to? :respond_to?
|
|
|
|
|
|
|
|
skip_methods = %w(object_id respond_to_missing? inspect === to_s)
|
|
|
|
|
|
|
|
instance_methods.each do |m|
|
|
|
|
undef_method m unless skip_methods.include?(m.to_s) || m =~ /^__/
|
|
|
|
end
|
|
|
|
|
2010-12-24 23:55:15 -05:00
|
|
|
def initialize # :nodoc:
|
2012-05-04 17:46:01 -04:00
|
|
|
@expected_calls = Hash.new { |calls, name| calls[name] = [] }
|
|
|
|
@actual_calls = Hash.new { |calls, name| calls[name] = [] }
|
2008-10-09 21:18:03 -04:00
|
|
|
end
|
|
|
|
|
2010-12-24 23:55:15 -05:00
|
|
|
##
|
2011-08-23 17:47:25 -04:00
|
|
|
# Expect that method +name+ is called, optionally with +args+, and returns
|
|
|
|
# +retval+.
|
2010-12-24 23:55:15 -05:00
|
|
|
#
|
|
|
|
# @mock.expect(:meaning_of_life, 42)
|
|
|
|
# @mock.meaning_of_life # => 42
|
|
|
|
#
|
|
|
|
# @mock.expect(:do_something_with, true, [some_obj, true])
|
|
|
|
# @mock.do_something_with(some_obj, true) # => true
|
2011-08-23 17:47:25 -04:00
|
|
|
#
|
|
|
|
# +args+ is compared to the expected args using case equality (ie, the
|
|
|
|
# '===' operator), allowing for less specific expectations.
|
|
|
|
#
|
|
|
|
# @mock.expect(:uses_any_string, true, [String])
|
|
|
|
# @mock.uses_any_string("foo") # => true
|
|
|
|
# @mock.verify # => true
|
|
|
|
#
|
|
|
|
# @mock.expect(:uses_one_string, true, ["foo"]
|
|
|
|
# @mock.uses_one_string("bar") # => true
|
|
|
|
# @mock.verify # => raises MockExpectationError
|
2010-12-24 23:55:15 -05:00
|
|
|
|
2008-10-09 21:18:03 -04:00
|
|
|
def expect(name, retval, args=[])
|
2012-05-04 17:46:01 -04:00
|
|
|
raise ArgumentError, "args must be an array" unless Array === args
|
|
|
|
@expected_calls[name] << { :retval => retval, :args => args }
|
2008-10-09 21:18:03 -04:00
|
|
|
self
|
|
|
|
end
|
|
|
|
|
2012-05-04 17:46:01 -04:00
|
|
|
def call name, data
|
|
|
|
case data
|
|
|
|
when Hash then
|
|
|
|
"#{name}(#{data[:args].inspect[1..-2]}) => #{data[:retval].inspect}"
|
|
|
|
else
|
|
|
|
data.map { |d| call name, d }.join ", "
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2010-12-24 23:55:15 -05:00
|
|
|
##
|
|
|
|
# Verify that all methods were called as expected. Raises
|
|
|
|
# +MockExpectationError+ if the mock object was not called as
|
|
|
|
# expected.
|
|
|
|
|
2008-10-09 21:18:03 -04:00
|
|
|
def verify
|
2012-05-04 17:46:01 -04:00
|
|
|
@expected_calls.each do |name, calls|
|
|
|
|
calls.each do |expected|
|
|
|
|
msg1 = "expected #{call name, expected}"
|
|
|
|
msg2 = "#{msg1}, got [#{call name, @actual_calls[name]}]"
|
|
|
|
|
|
|
|
raise MockExpectationError, msg2 if
|
|
|
|
@actual_calls.has_key? name and
|
|
|
|
not @actual_calls[name].include?(expected)
|
|
|
|
|
|
|
|
raise MockExpectationError, msg1 unless
|
|
|
|
@actual_calls.has_key? name and @actual_calls[name].include?(expected)
|
|
|
|
end
|
2008-10-09 21:18:03 -04:00
|
|
|
end
|
|
|
|
true
|
|
|
|
end
|
2010-12-01 00:33:32 -05:00
|
|
|
|
2010-12-24 23:55:15 -05:00
|
|
|
def method_missing(sym, *args) # :nodoc:
|
2012-05-04 17:46:01 -04:00
|
|
|
unless @expected_calls.has_key?(sym) then
|
2011-08-23 17:47:25 -04:00
|
|
|
raise NoMethodError, "unmocked method %p, expected one of %p" %
|
|
|
|
[sym, @expected_calls.keys.sort_by(&:to_s)]
|
|
|
|
end
|
|
|
|
|
2012-05-04 17:46:01 -04:00
|
|
|
index = @actual_calls[sym].length
|
|
|
|
expected_call = @expected_calls[sym][index]
|
2011-08-23 17:47:25 -04:00
|
|
|
|
2012-05-04 17:46:01 -04:00
|
|
|
unless expected_call then
|
|
|
|
raise MockExpectationError, "No more expects available for %p: %p" %
|
|
|
|
[sym, args]
|
|
|
|
end
|
|
|
|
|
|
|
|
expected_args, retval = expected_call[:args], expected_call[:retval]
|
|
|
|
|
|
|
|
if expected_args.size != args.size then
|
2011-08-23 17:47:25 -04:00
|
|
|
raise ArgumentError, "mocked method %p expects %d arguments, got %d" %
|
2012-05-04 17:46:01 -04:00
|
|
|
[sym, expected_args.size, args.size]
|
|
|
|
end
|
|
|
|
|
|
|
|
fully_matched = expected_args.zip(args).all? { |mod, a|
|
|
|
|
mod === a or mod == a
|
|
|
|
}
|
|
|
|
|
|
|
|
unless fully_matched then
|
|
|
|
raise MockExpectationError, "mocked method %p called with unexpected arguments %p" %
|
|
|
|
[sym, args]
|
2011-08-23 17:47:25 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
@actual_calls[sym] << {
|
|
|
|
:retval => retval,
|
2011-11-17 18:02:16 -05:00
|
|
|
:args => expected_args.zip(args).map { |mod, a| mod === a ? mod : a }
|
2011-08-23 17:47:25 -04:00
|
|
|
}
|
|
|
|
|
2010-12-01 00:33:32 -05:00
|
|
|
retval
|
|
|
|
end
|
|
|
|
|
2010-12-24 23:55:15 -05:00
|
|
|
def respond_to?(sym) # :nodoc:
|
2011-10-19 16:34:21 -04:00
|
|
|
return true if @expected_calls.has_key?(sym.to_sym)
|
2011-08-23 17:47:25 -04:00
|
|
|
return __respond_to?(sym)
|
2010-12-01 00:33:32 -05:00
|
|
|
end
|
2008-10-09 21:18:03 -04:00
|
|
|
end
|
|
|
|
end
|