Проблема с насмешкой над чертой загрузочной модели Laravel

Я использую загрузочную черту модели для регистрации определенных событий для моделей, использующих мою черту. Однако я столкнулся с проблемой, пытаясь издеваться над моделями, использующими эту черту. В частности, когда создается экземпляр модели Mockery, ее загрузочный код соглашается с тем, что у него должен быть метод bootMyTrait, но не может найти его, когда пытается его вызвать.

Образец репозитория для приведенного ниже, с командами для воспроизведения.

Например, вот черта:

namespace App;
trait MyTrait
{
    public static function bootMyTrait()
    {
        print("Booting MyTrait\n");
    }
}

И модель, использующая его:

namespace App;
use Illuminate\Database\Eloquent\Model;
class MyModel extends Model
{
    use MyTrait;
}

Создание модели регулярно работает нормально. Это показывает желаемый результат:

$model = new MyModel();

Однако попытка издеваться над этой моделью не сотрудничает. Этот:

use Illuminate\Foundation\Testing\WithoutMiddleware;
use Illuminate\Foundation\Testing\DatabaseMigrations;
use Illuminate\Foundation\Testing\DatabaseTransactions;


    class ExampleTest extends TestCase
    {
        /**
         * A basic functional test example.
         *
         * @return void
         */
        public function testTraitBooting()
        {
            $model = $this->getMock('App\MyModel');
        }
    }


Fails. Adding some debugging to Eloquent:


    /**
     * Boot all of the bootable traits on the model.
     *
     * @return void
     */
    protected static function bootTraits()
    {
        $class = static::class;

        foreach (class_uses_recursive($class) as $trait) {
            print("\nTesting that class: $class has method: " . $method = 'boot'.class_basename($trait) . " because of Trait: $trait\n");
            if (method_exists($class, $method = 'boot'.class_basename($trait))) {
                print("Class: $class has method: $method \n");
                try {
                    forward_static_call([$class, $method]);
                } catch (\PHPUnit_Framework_MockObject_BadMethodCallException $e) {
                    print("Class: $class failed calling $method\n");
                    throw $e;
                }
            }
        }
    }

Дает нам этот отказ:

PHPUnit 5.1.0 by Sebastian Bergmann and contributors.

E                                                                   1 / 1 (100%)
Testing that class: Mock_MyModel_9ee820db has method: bootMyTrait because of Trait: App\MyTrait
Class: Mock_MyModel_9ee820db has method: bootMyTrait
Class: Mock_MyModel_9ee820db failed calling bootMyTrait


Time: 129 ms, Memory: 18.00Mb

There was 1 error:

1) ExampleTest::testTraitBooting
PHPUnit_Framework_MockObject_BadMethodCallException:

mock-bootable-laravel-model-trait/vendor/laravel/framework/src/Illuminate/Database/Eloquent/Model.php:326
mock-bootable-laravel-model-trait/vendor/laravel/framework/src/Illuminate/Database/Eloquent/Model.php:309
mock-bootable-laravel-model-trait/vendor/laravel/framework/src/Illuminate/Database/Eloquent/Model.php:296
mock-bootable-laravel-model-trait/vendor/laravel/framework/src/Illuminate/Database/Eloquent/Model.php:277
mock-bootable-laravel-model-trait/tests/ExampleTest.php:16

Я также пытался создать макет несколькими разными способами. На примере DatabaseSoftDeletingTraitTest:

use Illuminate\Foundation\Testing\WithoutMiddleware;
use Illuminate\Foundation\Testing\DatabaseMigrations;
use Illuminate\Foundation\Testing\DatabaseTransactions;
use Mockery as m;

class ExampleTest extends TestCase
{
    /**
     * A basic functional test example.
     *
     * @return void
     */
    public function testTraitBooting()
    {
        $mock = m::mock('App\MyModel');
        $mock->shouldReceive('bootMyTrait')->once();
    }
}

Но здесь bootMyTrait никогда не вызывается:

PHPUnit 5.1.0 by Sebastian Bergmann and contributors.

E                                                                   1 / 1 (100%)

Time: 149 ms, Memory: 19.25Mb

There was 1 error:

1) ExampleTest::testTraitBooting
Mockery\Exception\InvalidCountException: Method bootMyTrait() from Mockery_0_App_MyModel should be called
 exactly 1 times but called 0 times.

mock-bootable-laravel-model-trait/vendor/mockery/mockery/library/Mockery/CountValidator/Exact.php:37
mock-bootable-laravel-model-trait/vendor/mockery/mockery/library/Mockery/Expectation.php:271
mock-bootable-laravel-model-trait/vendor/mockery/mockery/library/Mockery/ExpectationDirector.php:120
mock-bootable-laravel-model-trait/vendor/mockery/mockery/library/Mockery/Container.php:297
mock-bootable-laravel-model-trait/vendor/mockery/mockery/library/Mockery/Container.php:282
mock-bootable-laravel-model-trait/vendor/mockery/mockery/library/Mockery.php:142
mock-bootable-laravel-model-trait/vendor/laravel/framework/src/Illuminate/Foundation/Testing/TestCase.php:122

Итак, я могу переместить код, который я делаю в методе загрузки, в ServiceProvider, но тогда мне нужно будет зарегистрировать каждую модель, которая использует трейт. Это кажется грязным, и использование метода загрузки кажется уместным. Итак, я думаю, что либо наткнулся на ошибку, либо неправильно издеваюсь над моделью использования черт. Я просмотрел getMockForTrait, но мне также нужен издевательский экземпляр для расширения Eloquent (некоторые из методов типажа вызывают красноречивые методы)

Если кто-то увидит что-то, что я пропустил (или если я совершенно неправильно подхожу к этому), очень признателен


person timbroder    schedule 12.04.2016    source источник


Ответы (1)


После некоторых тестов я считаю, что такого метода будет достаточно, чтобы протестировать его:

$mock = m::mock('App\MyModel')->makePartial();
$mock->shouldReceive('bootMyTrait')->once();
$mock->__construct();

Объяснение:

  1. $mock = m::mock('App\MyModel')->makePartial();

    Мы создаем макет, но делаем его частичным, потому что хотим использовать конструктор класса по умолчанию и другие методы. Если сделать его частичным, это означает, что все методы, которые мы не переопределяем, будут использоваться из исходного класса App\MyModel.

  2. $mock->shouldReceive('bootMyTrait')->once();

    Это должно быть очевидно — мы хотим проверить, запускается ли метод bootMyTrait ровно 1 раз.

  3. $mock->__construct();

    Таким образом, мы можем запустить конструктор класса по умолчанию. Создается впечатление, что при создании макета конструктор не используется, поэтому мы не можем протестировать его по-другому. Нам нужно вручную запустить метод конструктора объекта, если мы хотим убедиться, что запускается исходный конструктор класса.

person Marcin Nabiałek    schedule 21.04.2016
comment
Спасибо! Я не уверен, почему это не работает с оболочкой Laravel Mockery, но я соглашусь! - person timbroder; 22.04.2016