如何在Laravel中模拟没有连接数据库的模型

时间:2019-10-27 20:57:37

标签: laravel mocking phpunit laravel-6 phpunit-testing

我尝试测试控制器的方法index()。在这种方法中,有一个模型。

class UserController extends Controller
{        
     public function index()
     {
        return User::all();
     }
}

在测试课中,我有以下内容。

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]连接被拒绝(SQL:从users中选择*   usersdeleted_at为空)

1 个答案:

答案 0 :(得分:1)

您尝试使用模拟对对象实例的函数调用的方法来模拟静态调用。模拟静态函数调用不是直接的,可以使用别名来完成,但是不建议这样做。

一种简单的方法是将逻辑简单地包装到服务中并对其进行模拟。

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或使用{{1} }环境为您提供数据库。比起提供任何有价值的东西,不进行测试更是一个障碍。速度在您的测试方法中至关重要,因为速度会很多,由于时间紧迫,最好先快速然后跳过它,并且模拟所有数据库调用会很困难。