class MiniTest::Mock

def expect(name, retval, args=[])

def expect(name, retval, args=[])
  raise ArgumentError, "args must be an array" unless Array === args
  @expected_calls[name] << { :retval => retval, :args => args }
  self
end

def initialize # :nodoc:

:nodoc:
def initialize # :nodoc:
  @expected_calls = Hash.new { |calls, name| calls[name] = [] }
  @actual_calls   = Hash.new { |calls, name| calls[name] = [] }
end

def method_missing(sym, *args) # :nodoc:

:nodoc:
def method_missing(sym, *args) # :nodoc:
  unless @expected_calls.has_key?(sym) then
    raise NoMethodError, "unmocked method %p, expected one of %p" %
      [sym, @expected_calls.keys.sort_by(&:to_s)]
  end
  expected_calls = @expected_calls[sym].select { |call| call[:args].size == args.size }
  if expected_calls.empty?
    arg_sizes = @expected_calls[sym].map { |call| call[:args].size }.uniq.sort
    raise ArgumentError, "mocked method %p expects %s arguments, got %d" %
      [sym, arg_sizes.join('/'), args.size]
  end
  expected_call = expected_calls.find do |call|
    call[:args].zip(args).all? { |mod, a| mod === a or mod == a }
  end
  unless expected_call
    raise MockExpectationError, "mocked method %p called with unexpected arguments %p" %
      [sym, args]
  end
  expected_args, retval = expected_call[:args], expected_call[:retval]
  
  @actual_calls[sym] << {
    :retval => retval,
    :args => expected_args.zip(args).map { |mod, a| mod === a ? mod : a }
  }
  retval
end

def respond_to?(sym) # :nodoc:

:nodoc:
def respond_to?(sym) # :nodoc:
  return true if @expected_calls.has_key?(sym.to_sym)
  return __respond_to?(sym)
end

def verify

def verify
  @expected_calls.each do |name, calls|
    calls.each do |expected|
      msg1 = "expected #{name}, #{expected.inspect}"
      msg2 = "#{msg1}, got #{@actual_calls[name].inspect}"
      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
  end
  true
end