Symfony2: 容器中设置了模拟服务但控制器没有使用它(它仍然使用原始服务)

Symfony2: Mocked service is set in the container but not used by the controller (it still uses the original service)

我正在为控制器编写功能测试。 它使用 class 从第三方网站导入一些数据,为此我写了一个 class 用于 Symfony 并将其设置为服务。

现在,在我的功能测试中,我想用模拟服务替换此服务,将其设置在容器中并在我的功能测试中使用它。

所以我的代码如下:

    // Mock the ImportDataManager and substitute it in the services container
    $mockDataImportManager = $this->getMockBuilder('\AppBundle\Manager\DataImportManager')->disableOriginalConstructor()->getMock();
    $client->getContainer()->set('shq.manager.DataImport', $mockDataImportManager);

    $client->submit($form);
    $crawler = $client->followRedirect();

据我所知,在每次请求之间客户端都会重新启动内核,我必须再次设置模拟 class,我在调用 $client->submit 之前立即设置了模拟。

但这种方法似乎对我不起作用,控制器仍然继续使用服务的真实版本而不是模拟版本。

如何使用模拟的 class 来避免在我的功能测试期间调用远程网站?

如果我转储设置的模拟服务,我可以看到它设置正确:

dump($client->getContainer()->get('shq.manager.DataImport'));die;

returns

.SetUpControllerTest.php on line 145:
Mock_DataImportManager_d2bab1e7 {#4807
  -__phpunit_invocationMocker: null
  -__phpunit_originalObject: null
  -em: null
  -remotes: null
  -tokenGenerator: null
  -passwordEncoder: null
  -userManager: null
}

但在 $form->submit($form) 调用期间未使用它,而是使用原始服务。

更新 继续寻找解决方案,我找到了 Symfony 项目的 this GitHub page,用户在该项目中寻求解决我同样问题的方法。 第二次调用没有使用他 class 的 mocked/substituted 版本,而是使用原始版本。

这是正确的行为吗?那么,我是否真的不能在第二次调用客户端时修改服务容器?

但是,我不明白为什么服务没有在容器中被替换,我也没有真正解决这个问题的方法。

无论如何我找到了某种解决方法,实际上更正确的解决方案(如果仍然不清楚为什么服务没有被替换,这是我想解决的好奇心 - 可能是因为 $client-> submit() 方法使用 POST 方法?)。

我的解决方法是一个简单的测试替身。

我在 AppBundle/Tests/TestDouble 中创建了一个新的 class 并将其命名为 DataImportManagerTestDouble.php

它包含控制器使用的独特方法:

namespace AppBundle\Tests\TestDouble;

use AppBundle\Entity\User;

class DataImportManagerTestDouble
{
    public function importData(User $user)
    {
        return true;
    }
}

然后,我在 config_test.yml (app/config/config_test.yml) 文件中按以下方式实例化它:

services:
    shq.manager.DataImport:
        class: AppBundle\Tests\TestDouble\DataImportManagerTestDouble

这样,在测试期间,并且仅在测试期间,作为服务加载的 class 是 TestDouble 而不是原始服务。 所以测试通过了,我(相对)很高兴。至少目前是这样。