我有一个测试文件来测试服务实例,并且已经用KnpMenuBundle制作了一个自定义菜单.
一切正常,期望PHPunit在测试我的MenuBuilder时返回错误.
class ServiceAvailabilityTest extends KernelTestCase
{
/**
* @dataProvider getServiceIds
*
* @param $serviceId
*/
public function testServiceInstance($serviceId)
{
static::bootKernel();
static::$kernel->getContainer()->get($serviceId);
}
}
在我的MenuBuilder上,我使用authorizationChecker来知道是否授予用户权限,就像这样.
if ($this->authorizationChecker->isGranted('ROLE_ADMIN')) {
$menu->addChild('sidebar.front.administration', ['route' => 'sonata_admin_redirect'])
->setExtra('translation_domain', 'navigation')
->setAttribute('icon', 'fa fa-eye');
}
当我删除所有这些时,测试就可以了
$this->authorizationChecker->isGranted('ROLE_ADMIN')
1) Tests\ServiceAvailabilityTest::testServiceInstance with data set #423 (‘menu.main’)
Symfony\Component\Security\Core\ExceptionAuthenticationCredentialsNotFoundException: The token storage contains no authentication token. One possible reason may be that there is no firewall configured for this URL. /code/vendor/symfony/symfony/src/Symfony/Component/Security/Core/Authorization/AuthorizationChecker.PHP:57
/code/src/AppBundle/Menu/MenuBuilder.PHP:192
/code/src/AppBundle/Menu/MenuBuilder.PHP:101
/code/app/cache/test/appTestDebugProjectContainer.PHP:8311
/code/vendor/symfony/symfony/src/Symfony/Component/DependencyInjection/Container.PHP:312
/code/tests/ServiceAvailabilityTest.PHP:3
如果您需要检查的话,这里有我的菜单服务
menu.builder:
class: AppBundle\Menu\MenuBuilder
arguments: [ '@knp_menu.factory', '@doctrine', '@manager.server','@security.authorization_checker', '@request_stack' ]
menu.main:
class: Knp\Menu\MenuItem
factory: [ '@menu.builder', 'createMainMenu' ]
arguments: [ '@request_stack' ]
tags:
- { name: knp_menu.menu, alias: sidebar }
我已经在互联网上搜索了,但是他们通过在security.yml上添加访问控制来解决此问题
- { path: ^/, role: ROLE_USER }
但是我没有菜单路线.
解决方法:
尝试这个:
/**
* @param string $firewallName
* @param UserInterface $user
* @param array $options
* @param array $server
*/
protected function loginUser($firewallName, UserInterface $user, array $options = array(), array $server = array())
{
$this->client = static::createClient();
$token = new UsernamePasswordToken($user, null, $firewallName, $user->getRoles());
static::$kernel->getContainer()->get('security.token_storage')->setToken($token);
// <2.8 this may be usefull
//$request = new Request();
//$event = new InteractiveLoginEvent($request, $token);
//static::$kernel->getContainer()->get('event_dispatcher')->dispatch('security.interactive_login', $event);
$session = $this->client->getContainer()->get('session');
$session->set('_security_'.$firewallName, serialize($token));
$session->save();
$cookie = new Cookie($session->getName(), $session->getId());
$this->client->getCookieJar()->set($cookie);
}
在您的testCase / setUp中,例如:
static::bootKernel();
$this->loginUser('admin', $testUser);
$this->assertNotFalse(static::$kernel->getContainer()->get('security.authorization_checker')->isGranted('ROLE_ADMIN'));