【问题标题】:How mock the model without connection database in Laravel如何在 Laravel 中模拟没有连接数据库的模型
【发布时间】:2020-02-23 06:29:20
【问题描述】:

我尝试测试控制器的方法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: select * from users 其中users.deleted_at 为空)

【问题讨论】:

    标签: laravel mocking laravel-6 phpunit


    【解决方案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 或使用docker 环境提供数据库为您服务。没有提供任何重要价值的东西相比,测试是一个更大的障碍。速度在您的测试方法中至关重要,因为会有很多,最好快速完成,然后由于时间压力而跳过它,并且模拟所有数据库调用会很困难。

    【讨论】:

    • 谢谢,要使用控制器,我应该进行依赖注入(like here)吗? @mrhn
    • 控制器的使用能详细点吗?控制器将始终通过路由器加载依赖注入:)
    • 好的,谢谢,我在谷歌上寻找如何通过路由器进行依赖注入。
    猜你喜欢
    • 2016-12-17
    • 2015-05-13
    • 2011-03-14
    • 1970-01-01
    • 2016-02-21
    • 2014-11-05
    • 1970-01-01
    • 1970-01-01
    • 2018-06-18
    相关资源
    最近更新 更多