我正在为与
Action
实体有关系的 User
实体编写功能测试:
<?php
namespace Acme\AppBundle\Entity;
/**
* Class Action
*
* @ORM\Table()
* @ORM\Entity(repositoryClass="Acme\AppBundle\Repository\ActionRepository")
*/
class Action
{
/**
* @var int
*
* @ORM\Column(type="integer")
* @ORM\Id
* @ORM\GeneratedValue(strategy="AUTO")
*/
private $id;
/**
* @var \Acme\AppBundle\Entity\User
*
* @ORM\ManyToOne(targetEntity="\Acme\AppBundle\Entity\User", inversedBy="actions")
* @ORM\JoinColumn(name="user_id", referencedColumnName="id")
*/
private $createdBy;
}
用户:
namespace Acme\AppBundle\Entity;
/**
* @ORM\Entity
* @ORM\Table(name="`user`")
*/
class User extends BaseUser
{
/**
* @ORM\Id
* @ORM\Column(type="integer")
* @ORM\GeneratedValue(strategy="AUTO")
*/
protected $id;
/**
* @var ArrayCollection
*
* @ORM\OneToMany(targetEntity="Action", mappedBy="createdBy")
*/
private $actions;
}
用户使用以下代码片段在控制器中设置:
<?php
namespace Acme\ApiBundle\Controller;
/**
*
* @Route("/actions")
*/
class ActionController extends FOSRestController
{
public function postAction(Request $request)
{
$action = new Action();
$action->setCreatedBy($this->getUser());
return $this->processForm($action, $request->request->all(), Request::METHOD_POST);
}
}
例如,当使用 REST 客户端调用操作时,一切正常,
Action
和 User
之间的关系正确保留。
现在,当使用功能测试来测试操作时,由于以下错误,该关系不起作用:
通过关系“Acme\AppBundle\Entity\Action#createdBy”找到了一个新实体,该实体未配置为实体的级联持久操作:测试。要解决此问题:对此未知实体显式调用 EntityManager#persist() 或配置级联在映射中保留此关联,例如 @ManyToOne(..,cascade={"persist"})。
对于我的功能测试,我需要注入 JWT 和会话令牌,因为我的路由由 JWT 保护,并且我需要在会话中拥有用户。
这是我注入的方法:
<?php
namespace Acme\ApiBundle\Tests;
class ApiWebTestCase extends WebTestCase
{
/**
* @var ReferenceRepository
*/
protected $fixturesRepo;
/**
* @var Client
*/
protected $authClient;
/**
* @var array
*/
private $fixtures = [];
protected function setUp()
{
$fixtures = array_merge([
'Acme\AppBundle\DataFixtures\ORM\LoadUserData'
], $this->fixtures);
$this->fixturesRepo = $this->loadFixtures($fixtures)->getReferenceRepository();
$this->authClient = $this->createAuthenticatedClient();
}
/**
* Create a client with a default Authorization header.
*
* @return \Symfony\Bundle\FrameworkBundle\Client
*/
protected function createAuthenticatedClient()
{
/** @var User $user */
$user = $this->fixturesRepo->getReference('user-1');
$jwtManager = $this->getContainer()->get('lexik_jwt_authentication.jwt_manager');
$token = $jwtManager->create($user);
$this->loginAs($user, 'api');
$client = static::makeClient([], [
'AUTHENTICATION' => 'Bearer ' . $token,
'CONTENT_TYPE' => 'application/json'
]);
$client->disableReboot();
return $client;
}
}
现在,问题是注入的
UsernamePasswordToken
包含一个与当前 User
分离的 EntityManager
实例,从而导致上面的 Doctrine 错误。
我可以将
$user
方法中的 postAction
对象合并到 EntityManager
中,但我不想这样做,因为这意味着我修改我的工作代码以使测试通过。
我还尝试将测试中的 $user
对象直接合并到 EntityManager
中,如下所示:
$em = $client->getContainer()->get('doctrine')->getManager();
$em->merge($user);
但它也不起作用。
所以现在,我陷入了困境,我真的不知道该怎么办,除了我需要将会话中的用户附加回当前
EntityManager
。
您收到的错误消息表明测试客户端容器中包含的 EntityManager 不了解您的 User 实体。这让我相信您在
createAuthenticatedClient
方法中检索 User 的方式是使用不同的 EntityManager。
我建议您尝试使用测试内核的 EntityManager 来检索 User 实体。例如,您可以从测试客户端的容器中获取它。
感谢您的推文,我来完成给定的答案并(尝试)提出解决方案,
问题是您的用户不受 EntityManager 管理,更简单地说,因为它不是在数据库中注册的真实现有用户。
要解决此问题,您需要有一个真正的(托管)用户,该规则可以用于您的操作试图创建的关联。
因此,您可以在每次执行功能测试用例时创建此用户(并在完成后将其删除),或者仅在新环境中首次执行测试用例时创建一次。
这样的事情应该可以解决问题:
/** @var EntityManager */
private $em;
/**
*/
public function setUp()
{
$client = static::createClient();
$this->em = $client->getKernel()
->getContainer()
->get('doctrine');
$this->authClient = $this->createAuthenticatedClient();
}
/**
*/
protected function createAuthenticatedClient()
{
/** @var User $user */
$user = $this->em
->getRepository('Acme\AppBundle\Entity\User')
->findOneBy([], ['id' => DESC]; // Fetch the last created
// ...
return $client;
}
这对你的灯具来说是一个遗憾(它们太性感了),但我看不到任何方法可以将你的灯具作为真正的入口附加起来,因为你无法与测试的控制器进行更多交互。
另一种方法是向您的登录端点创建请求,但这会更难看。