【问题标题】:Unit test function that return value from database从数据库返回值的单元测试函数
【发布时间】:2021-03-03 16:46:57
【问题描述】:

目前我想创建单元测试来覆盖我开发的功能中的一些功能,但是当我面对这种功能时遇到问题,当功能的目的是从数据库中检索结果时。

public function saveMessage($request)
    {
        $validation = $this->validate($request);

        if($validation['status']) {
            $input = $validation['detail'];

            $data = [
                'id'            => $input['id'],
                'note'          => $input['message'],
                'create_by'     => $input['create_by'],
                'status'        => $input['status'],
                'create_date'   => DATETIME,
            ];

            $result = $this->fooService->insert($data);

            return [
                'success' => true
            ];
        }
        return $validation;
    }

这种功能可以不接触数据库吗?

【问题讨论】:

  • 当然,您可以模拟fooService(可能),但您实际上要测试什么?
  • 这似乎更适合集成测试。单元测试不应处理在基础架构级别执行操作的代码。
  • 就像 El_Vanja 说的。如果您想 tset 是否将值存储在数据库中,最好使用集成测试。如果您真的想要进行单元测试,请像建议的 lagbox 那样模拟数据库
  • 好吧,我明白了。谢谢大家:)

标签: php laravel mocking phpunit


【解决方案1】:

对于单元测试,您应该模拟方法的整个依赖关系,否则它将成为集成测试。

我曾经用 Prophecy 模拟(很快从 phpunit 10 弃用)但你可以使用 MockBuilder 它是相同的原理,如果你想模拟 $this->fooService 一个例子:

protected function setUp(): void
{
    $this->fooService = $this->prophesize(FooService::class);
    $this->classOfMethodToTest = new ClassOfMethodToTest(
        $this->fooService->reveal()
    ); // mock FooService dependency
}

public function testSaveMessage(): void
{
    $this->fooService
        ->insert(Argument::type('array')) // or an real array
        ->shouldBeCalledOnce() // apparently it is called only once
        ->willReturn(true) // for example i don't know real logic
    ;
    
    $this->classOfMethodToTest->saveMessage($fakeData);
}

【讨论】:

  • 很好的答案,但我建议 100% 使用 Mockery 而不是 Prophecy。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2022-10-20
  • 1970-01-01
  • 2016-10-04
  • 2015-05-30
  • 1970-01-01
相关资源
最近更新 更多