如何在 Laravel 中模拟没有连接数据库的模型
How mock the model without connection database in Laravel
我尝试测试控制器的方法index()
。在这个方法中,有一个模型。
class UserController extends Controller
{
public function index()
{
return User::all();
}
}
在测试class中,我有以下内容。
class UserControllerTest extends TestCase
{
public function testIndex():void
{
$user = factory(User::class)->make();
$mock = Mockery::mock(User::class);
$mock->shouldReceive('all')->andReturn($user);
$this->app->instance('User', $mock);
$response = $this->json('GET', 'api/users');
dd($response->getContent()); // error : [2002] Connection refused
}
}
当我 运行 测试时,我的数据库连接出现错误。这很奇怪,因为我模拟了模型,这意味着我不需要建立与数据库的连接。我该如何解决这个错误?
错误
SQLSTATE[HY000] [2002] Connection refused (SQL: select * from users
where users
.deleted_at
is null)
您正在尝试使用模拟对对象实例的函数调用的方法来模拟静态调用。模拟静态函数调用并不直接,可以通过别名来完成,但不推荐。
一种简单的方法是将您的逻辑简单地包装在服务中并对其进行模拟。
class UserService {
public function all(): Collection {
return User::all();
}
}
现在您的模拟代码应该如下所示。
$user = factory(User::class)->make();
$mock = Mockery::mock(UserService::class);
// Teoretically all method will return Eloquent Collection, but should be fine
$mock->shouldReceive('all')->andReturn(collect($user));
$this->app->instance(UserService::class, $mock);
使用 container
和替换实例时,非常依赖于使用 container
而不是 new
关键字来获取这些模拟的 类。因此,控制器应该看起来类似于此。
class UserController extends Controller
{
/** @var UserService **/
private $userService;
public function __construct(UserService $userService) {
// load userService from the container as the mocked instance on tests
$this->userService = $userService;
}
public function index()
{
return $this->userService->all();
}
}
最后一点,我有多个项目是测试、代码覆盖率等的主要驱动力。通过在数据库中创建测试更容易,使用 sqlite
或 docker
环境为你提供数据库。在没有提供任何重要价值的情况下进行测试是一个更大的障碍。速度在您的测试方法中至关重要,因为会有很多测试,最好快速完成,然后由于时间压力而跳过它,并且模拟所有数据库调用会很困难。
我尝试测试控制器的方法index()
。在这个方法中,有一个模型。
class UserController extends Controller
{
public function index()
{
return User::all();
}
}
在测试class中,我有以下内容。
class UserControllerTest extends TestCase
{
public function testIndex():void
{
$user = factory(User::class)->make();
$mock = Mockery::mock(User::class);
$mock->shouldReceive('all')->andReturn($user);
$this->app->instance('User', $mock);
$response = $this->json('GET', 'api/users');
dd($response->getContent()); // error : [2002] Connection refused
}
}
当我 运行 测试时,我的数据库连接出现错误。这很奇怪,因为我模拟了模型,这意味着我不需要建立与数据库的连接。我该如何解决这个错误?
错误
SQLSTATE[HY000] [2002] Connection refused (SQL: select * from
users
whereusers
.deleted_at
is null)
您正在尝试使用模拟对对象实例的函数调用的方法来模拟静态调用。模拟静态函数调用并不直接,可以通过别名来完成,但不推荐。
一种简单的方法是将您的逻辑简单地包装在服务中并对其进行模拟。
class UserService {
public function all(): Collection {
return User::all();
}
}
现在您的模拟代码应该如下所示。
$user = factory(User::class)->make();
$mock = Mockery::mock(UserService::class);
// Teoretically all method will return Eloquent Collection, but should be fine
$mock->shouldReceive('all')->andReturn(collect($user));
$this->app->instance(UserService::class, $mock);
使用 container
和替换实例时,非常依赖于使用 container
而不是 new
关键字来获取这些模拟的 类。因此,控制器应该看起来类似于此。
class UserController extends Controller
{
/** @var UserService **/
private $userService;
public function __construct(UserService $userService) {
// load userService from the container as the mocked instance on tests
$this->userService = $userService;
}
public function index()
{
return $this->userService->all();
}
}
最后一点,我有多个项目是测试、代码覆盖率等的主要驱动力。通过在数据库中创建测试更容易,使用 sqlite
或 docker
环境为你提供数据库。在没有提供任何重要价值的情况下进行测试是一个更大的障碍。速度在您的测试方法中至关重要,因为会有很多测试,最好快速完成,然后由于时间压力而跳过它,并且模拟所有数据库调用会很困难。