如何使用 Symfony 在 serviceTest 中模拟 EntityManager?

Ale*_*ant 4 phpunit unit-testing symfony

抱歉我的问题很长。

如何在扩展 KernelTestCase 的服务测试中模拟实体管理器?

现在,解释和我的测试......

我使用的是Symfony3.2。我的应用程序是标准的。我有一些控制器,我使用 WebTestCase 来测试它们。

一般来说,我的控制器验证参数,调用服务/管理器,处理一些变量并将它们推送到视图,我的测试在扩展WebTestCase的测试中非常简单。

/**
 * Test New Game Action
 */
public function testFooAction(){
    //We mock the Service
    $fooService = $this
        ->getMockBuilder(GameService::class)
        ->disableOriginalConstructor()
        ->getMock();
    $fooService->expects(self::once())
        ->method('barMethod')
        ->willReturn($result);

    //We create the client
    $client = static::createClient();
    $container = $client->getContainer();

    //I put my mock here
    $container->set('app.game-service', $fooService);

    //I launch the request
    $client->request('GET', '/foo');
    //I handle the response
    $response = $client->getResponse();

    //I do some tests like this one
    self::assertEquals(200, $response->getStatusCode());

}
Run Code Online (Sandbox Code Playgroud)

如您所见,我没有调用 EntityManger,因为我使用Services和这些行来放置我的 Services 的 Mock

    //We create the client
    $client = static::createClient();
    $container = $client->getContainer();

    //I put my mock here
    $container->set('app.game-service', $fooService);
Run Code Online (Sandbox Code Playgroud)

我在服务中模拟实体管理器时遇到问题。我的控制器经过测试,但我的服务未经过测试。

这里的构造函数初始化了一个简单的受保护属性entityManager。问题是这样受到保护的,你会进一步看到......

   /**
     * FooService constructor.
     * @param EntityManager $entityManager
     */
    public function __construct(EntityManager $entityManager)
    {
           $this->entityManager = $entityManager;
    }
Run Code Online (Sandbox Code Playgroud)

为了测试我的服务,这是我的初始代码:

<?php
class FooServiceTest extends KernelTestCase
{
    /**
     * Foo Service.
     *
     * @var FooService
     */
    private $fooService;

    /**
     * Prepares the environment before running each test.
     */
    protected function setUp()
    {
        parent::setUp();

        self::bootKernel();
        $this->fooService = static::$kernel->getContainer()
            ->get('app.foo-service') //HERE IS HOW I HANDLE MY SERVICE TO TEST IT
        ;

    }
Run Code Online (Sandbox Code Playgroud)

testStart 工作完美,有一些数据库交互。

我知道我可以在测试期间回滚数据。但我想模拟entityManager来验证方法的调用commit

我尝试在我的 setUp 中模拟实体管理器:

    protected function setUp()
    {
        parent::setUp();

        $entityManager = $this
            ->getMockBuilder('Doctrine\ORM\EntityManager')
            ->disableOriginalConstructor()
            ->getMock();
        $entityManager
            ->expects(once())  // I WANT EXACTLY ONE CALL TO COMMIT METHOD
            ->method('commit')  
            ->willReturn(null);

        self::bootKernel();
        $container = static::$kernel->getContainer();
        $container->set('doctrine.orm.entity_manager', $entityManager); // THIS LINE DOES NOTHING <=======
        $this->gameService = static::$kernel->getContainer()
            ->get('app.game-service')
        ;
Run Code Online (Sandbox Code Playgroud)

这些代码不起作用。模拟不到位。我仍然拥有真正的entityManager。我认为这是因为容器已经关闭。设置起来非常有用

像野蛮人一样,我将entityManager属性更改为public

我这样做:

    protected function setUp()
    {
        parent::setUp();

        $entityManagerMock = $this
            ->getMockBuilder('Doctrine\ORM\EntityManager')
            ->disableOriginalConstructor()
            ->getMock();
        $entityManagerMock
            ->expects(once())  // I WANT EXACTLY ONE CALL TO commit METHOD
            ->method('commit')  
            ->willReturn(null);

        self::bootKernel();
        $this->gameService = static::$kernel->getContainer()
            ->get('app.game-service')
        ;
        $this->gameService->entityManager = entityManagerMock;
Run Code Online (Sandbox Code Playgroud)

它工作完美。可以运行测试。但在公共财产中拥有一个entityManager并不是一个好的做法

我的问题是:如何在服务测试中模拟entityManager?

(抱歉,我的英语不太流利)

Nik*_* U. 6

首先在你的代码中

$container->set('doctrine.orm.entity_manager'); // THIS LINE DOES NOTHING <=======
Run Code Online (Sandbox Code Playgroud)

我确信您错过了第二个参数:)

另一个注意事项:不要嘲笑你不拥有的东西。将来您将运行composer updateEntityManager,并且EntityManagercommit将有一些可选isReallyCommit参数。你的代码将会被破坏,但你不会注意到它,因为测试是绿色的。我知道这不太可能,但无论如何这只是一个例子。我认为这里的好习惯是

  • 有一些封装了实体管理器的适配器
  • 在服务单元测试中模拟该适配器
  • 通过针对真实数据库的功能测试来测试您的适配器,而无需模拟任何内容

或者只是不要对您的服务进行单元测试,而是通过真实的数据库交互进行功能测试