1

简短的问题: 有没有办法在 SimpleTest 中重置 Mock 对象,消除所有期望?

更长的解释:

我有一个正在使用SimpleTest测试的类,并且它使用的 Mock 对象存在一些问题。

类是 a Logger,记录器内部是许多Writer对象(FileWriterEmailWriter等)。调用该Logger::log()方法在幕后执行一些逻辑并将消息路由到正确的编写器。写入器缓存在 Logger 类中,以节省每次重新实例化每个写入器。

在我的单元测试中,我设置了一个 Logger,创建并向其中添加了一些 Mock Writer 对象,然后一直使用诸如MockDBWriter->expectOnce()测试 Logger 是否工作的方法。

现在的问题是我想测试 Logger 的另一个功能,但是expectOnce期望仍然有效,导致我的后续测试失败。

function testWritesMessageOK() {
    $log = Logger::getInstance();
    $mock = new MockFileWriter($this);

    $log->addWriter($mock);

    $mock->expectOnce("write", "Message");

    $log->write("Message");    // OK
}

// this is just an example - the actual test is much less inane
function testNumberOfWrites() {
    $log = Logger::getInstance();
    $mock = $log->getWriter();

    $mock->expectCallCount('write', 2);

    $log->write("One");    // fail - it doesn't match "Message"
    $log->write("Two");
}

有没有办法重置 Mock 对象,消除所有期望?

4

1 回答 1

2

使用单独的模拟实例。

任何一个:

$mock = $log->getWriter();
$mock = new $mock;

或者:

$mock = new MockFileWriter($this);
// And then:
$mock = new MockDBWriter($this);
// And then:
$mock = new MockEmailWriter($this);
// etc.

我会质疑缓存作者以节省重新实例化的智慧。如果您使实例化成为一种廉价的操作(即不创建数据库连接或任何东西)并将这种事情推迟到您真正需要连接之前,例如第一个查询,那么您将不需要缓存,整个问题可能离开。

您可以做的另一件事是调用 SimpleMock 构造函数。

$mock = $log->getWriter();
$mock->SimpleMock();

这将完成所有这些:

/**
 *    Creates an empty action list and expectation list.
 *    All call counts are set to zero.
 *    @access public
 */
function SimpleMock() {
    $this->_actions = &new SimpleCallSchedule();
    $this->_expectations = &new SimpleCallSchedule();
    $this->_call_counts = array();
    $this->_expected_counts = array();
    $this->_max_counts = array();
    $this->_expected_args = array();
    $this->_expected_args_at = array();
    $test = &$this->_getCurrentTestCase();
    $test->tell($this);
}

唯一的问题是tell()最后的调用会导致SimpleMock::atTestEnd()在计算期望值时被调用两次。但是,你可以用这个来解决这个问题:

// $this should == the test case in question
array_pop($this->_observers);

此答案基于 SimpleTest 的 1.0.1 版。

于 2009-10-26T10:31:11.040 回答