如何在 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();
    }
}

最后一点,我有多个项目是测试、代码覆盖率等的主要驱动力。通过在数据库中创建测试更容易,使用 sqlitedocker环境为你提供数据库。在没有提供任何重要价值的情况下进行测试是一个更大的障碍。速度在您的测试方法中至关重要,因为会有很多测试,最好快速完成,然后由于时间压力而跳过它,并且模拟所有数据库调用会很困难。