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 而不是原始服务。
所以测试通过了,我(相对)很高兴。至少目前是这样。
我正在为控制器编写功能测试。 它使用 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 而不是原始服务。 所以测试通过了,我(相对)很高兴。至少目前是这样。