我正在使用带有php单元的laravel 5来创建一个laravel包。我有Repository
..
namespace Myname\Myapp\Repositories;
use Myname\Myapp\Models\PersonModel;
class PersonRepository
{
protected $personModel;
public function __construct(PersonModel $personModel)
{
$this->personModel = $personModel;
}
public function testFunction($var)
{
return $this->personModel->find($var);
}
}
..实现Model
。
namespace Myname\Myapp\Models;
use Illuminate\Database\Eloquent\Model;
class PersonModel extends Model
{
protected $table = 'person';
}
Laravels IoC自动将PersonModel
注入PersonRepository
的构造函数。
我正在编写单元测试,我想使用mock来模拟PersonModel
模型,所以我在测试期间没有访问数据库。
namespace Myname\Myapptests\unit;
use Mockery;
class PersonRepositoryTest extends \Myname\Myapptests\TestCase
{
/**
* @test
*/
public function it_returns_the_test_find()
{
$mock = Mockery::mock('Myname\Myapp\Models\PersonModel')
->shouldReceive('find')
->with('var');
$this->app->instance('Myname\Myapp\Models\PersonModel', $mock);
$repo = $this->app->make('Myname\Myapp\Repositories\PersonRepository');
$result = $repo->testFunction('var');
$this->assert...
}
}
当我运行测试时出现错误
1)Myname \ Myapptests \ unit \ PersonRepositoryTest :: it_returns_the_test_find ErrorException:传递给Myname \ Myapp \ Repositories \ PersonRepository :: __ construct()的参数1必须是Myname \ Myapp \ Models \ PersonModel的实例,Mockery \ CompositeExpectation的实例给出
从我所看到的,mockery扩展了它正在模拟的类,因此应该没有问题注入扩展类来代替类型提示父类(PersonModel)
显然我错过了一些东西。其他示例明确地将模拟对象注入到他们正在测试的类中。 Laravels IoC(应该)为我做这件事。我必须绑定任何东西吗?
我有一种感觉,虽然嘲弄对象没有按我想象的方式创建(扩展PersonModel),否则我认为我不会看到这个错误。
答案 0 :(得分:7)
问题在于你创建模拟时:
$mock = Mockery::mock('Myname\Myapp\Models\PersonModel')
->shouldReceive('find')
->with('var');
所以这个:
$mock = Mockery::mock('Myname\Myapp\Models\PersonModel')
var_dump($mock);
die();
将输出如下内容:Mockery_0_Myname_Myapp_Models_PersonModel
但是这个:
$mock = Mockery::mock('Myname\Myapp\Models\PersonModel')
->shouldReceive('find')
->with('var');
var_dump($mock);
die();
将输出:Mockery\CompositeExpectation
所以尝试做这样的事情:
$mock = Mockery::mock('Myname\Myapp\Models\PersonModel');
$mock->shouldReceive('find')->with('var');
$this->app->instance('Myname\Myapp\Models\PersonModel', $mock);
$repo = $this->app>make('Myname\Myapp\Repositories\PersonRepository');
$result = $repo->testFunction('var');
答案 1 :(得分:7)
虽然Fabio提供great answer,但这里的问题实际上是测试设置。 Mockery's mock objects do comply to contracts并将传递instanceof
个测试并在方法参数中输入提示。
原始代码的问题在于被调用的方法链最终返回期望而不是模拟。我们应该首先创建一个模拟,然后为模拟添加期望。
要解决此问题,请更改此信息:
$mock = Mockery::mock('Myname\Myapp\Models\PersonModel')
->shouldReceive('find')
->with('var');
进入这个:
$mock = Mockery::mock('Myname\Myapp\Models\PersonModel');
$mock->shouldReceive('find')->with('var');
变量$mock
现在将实现PersonModel
。
奖金:
而不是'Myname\Myapp\Models\PersonModel'
,请使用PersonModel::class
。这对IDE更加友好,可以在以后重构代码时为您提供帮助。