6

我目前正在为 Laravel 4 中的包编写测试。

我在嘲笑Illuminate\Database\Query\Builder它几乎一直有效,除了当where方法使用回调时,我无法检查回调中的方法是否被调用。

我希望你们中的一个人能有所启发。

$query = \Mockery::mock('Illuminate\Database\Query\Builder', function ($mock) {
    /** @var \Mockery\Mock $mock */
    $mock->shouldReceive('where');
    $mock->shouldReceive('orWhere')->twice();
});

where以及应该调用的实际方法orWhere注意:该构建器模拟被传递给下面的类。

$builder = new LaravelBuilder($query);

然后调用$builder->filter()which 包含以下代码。

$this->query->where(function ($query) use ($filterData) {
    /** @var \Illuminate\Database\Query\Builder|\Illuminate\Database\Eloquent\Builder $query */
    foreach($filterData['columns'] as $colData) {
        /** @var \Samvaughton\Ldt\Column $column */
        $column = $colData['column'];

        // See if this column is searchable
        if (!$column->isSearchable() || !$colData['searchable']) continue;

        // If the individual column term is empty, use the main term
        $term = (empty($colData['term'])) ? $filterData['term'] : $colData['term'];

        // Actually apply the filter
        $query->orWhere($column->getSqlColumn(), "LIKE", "%{$term}%");
    }
});

主要部分是底部位$query->orWhere,PHPUnit 测试每次都失败,因为它orWhere甚至一次都没有运行。在您说它可能无法执行之前continue,我传递的数据将允许这样做。

我怀疑这是因为我一开始就嘲笑这个where方法。如果我exit在它之前包含一个foreach它不会执行,表明它甚至没有在回调中运行任何东西。我知道这是默认行为,但如何让 Mockery 运行相同/相似的回调?

我尝试过使用部分模拟,使用shouldExpect但不完全理解它。我也尝试过四处搜索,但这种情况没有运气。

如果我能学习如何在回调中使用嘲弄,那就太好了。

4

1 回答 1

4

好的,所以我认为这可以做得更好,但我采取了部分嘲笑 Laravel 的查询构建器的方法。

见第 324 行Illuminate\Database\Query\Builder

当使用闭包时,Laravel 会产生一个新的查询构建器,它也需要被模拟。这是有效的代码:

$query = \Mockery::mock('Illuminate\Database\Query\Builder', function ($mock) {
    /** @var \Mockery\Mock $mock */
    $mock->makePartial();
    $mock->shouldReceive('where')->once()->passthru();
    $mock->shouldReceive('newQuery')->andReturn(
        \Mockery::mock('Illuminate\Database\Query\Builder', function ($mock) {
            /** @var \Mockery\Mock $mock */
            $mock->makePartial();
            $mock->shouldReceive('orWhere')->twice();
        })
    );
});

我需要使用makePartial(),因此查询构建器保留了其原始闭包功能(这不适用于单元测试),因此这不是最佳解决方案。从这里我模拟了newQuery被调用以生成一个新的查询构建器实例的方法,然后我以类似的格式模拟它。

于 2013-12-21T13:12:48.060 回答