我需要通过重载来模拟CurrencyEnum,但这并不是最终目的,因为我需要向该模拟中添加接口。 这不起作用:
Mockery::mock('overload:'.CurrencyEnum::class);
错误:(..) must be an instance of \BaseCurrency, instance of \CurrencyEnum given
。
我看着Mockery\Container::mock
,但不知道该怎么做。
在示例中,我想测试TestingClass::first()
方法
class CurrencyEnum implements BaseCurrency
{
/* methods */
}
class TestingClass
{
public function first(string $currencySymbol)
{
$abc = 'some_string';
return $this->second($abc, new CurrencyEnum($currencySymbol));
}
private function second(string $abc, BaseCurrency $currency)
{
/* code */
}
}
答案 0 :(得分:4)
重载方法通过拦截autoload
机制来工作:它为overloaded
类注册一个自动加载器,加载该类的模拟版本,而不是原始版本。
默认情况下,它不会向模拟类添加很多东西。但是,您可以配置几乎任何您需要的东西。
通常,implementing one or more interfaces可以通过用逗号分隔的全限定名称列表来完成,第一个是类:
$mock = Mockery::mock('MyClass, MyInterface, OtherInterface');
由于Mockery::mock
方法的设置方式,该方法无效。 (The author apologises in the source code)
但是,我们可以将接口作为第二个参数传递给模拟方法:
Mockery::mock('overload:'.CurrencyEnum::class, BaseCurrency::class);
这将导致MockConfigurationBuilder
将BaseCurrency
添加为目标;因为它是一个接口,所以它将使模拟实现该接口。
上述的另一种表示法是直接使用构建器:
Mockery::mock(
(new MockConfigurationBuilder())
->setInstanceMock(true)
->setName(CurrencyEnum::class)
->addTarget('stdClass')
->addTarget(BaseCurrency::class)
)
话虽如此,嘲笑诸如枚举和值对象之类的东西却是一个臭名昭著的坏习惯。为什么不只使用 actual CurrencyEnum
?诸如货币代码这样的简单操作根本不能保证完全模拟。可能需要进行结构上的改进,这将同时为您的测试增加大量价值,并使它们更易于阅读。