编写功能测试时将会话中的用户附加到当前EntityManager

问题描述 投票:0回答:2

我正在为与

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

php symfony testing doctrine-orm functional-testing
2个回答
1
投票

您收到的错误消息表明测试客户端容器中包含的 EntityManager 不了解您的 User 实体。这让我相信您在

createAuthenticatedClient
方法中检索 User 的方式是使用不同的 EntityManager。

我建议您尝试使用测试内核的 EntityManager 来检索 User 实体。例如,您可以从测试客户端的容器中获取它。


1
投票

感谢您的推文,我来完成给定的答案并(尝试)提出解决方案,

问题是您的用户不受 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;
}

这对你的灯具来说是一个遗憾(它们太性感了),但我看不到任何方法可以将你的灯具作为真正的入口附加起来,因为你无法与测试的控制器进行更多交互。

另一种方法是向您的登录端点创建请求,但这会更难看。

© www.soinside.com 2019 - 2024. All rights reserved.