首页 > 解决方案 > Symfony Doctrine 关系在 PhpUnit 测试中为空

问题描述

我有一个 Symfony 应用程序,User其实体与实体具有多对多关系Cat。我还有一个 PhpUnit 测试,它检查从 1 个用户删除一只猫(属于 2 个用户)实际上并没有删除猫:

    public function testDeletingACatBelongingToTwoUsersOnlyDeletesTheAssociationNotTheCat()
    {
        $cat = $this->createCat();
        // Associate with user 1
        $user1 = new User();
        $user1->setEmail('test@example.com');
        $user1->setPassword('pwdpwd');
        $user1->addCat($cat);
        $this->em->persist($user1);
        // Associate with user 2
        $user2 = new User();
        $user2->setEmail('another@example.com');
        $user2->setPassword('pwdpwd');
        $user2->addCat($cat);
        $this->em->persist($user2);
        $this->em->flush();
        // Sanity check:
        $this->assertCount(1, $user1->getCats()); // PASS
        $this->assertCount(1, $user2->getCats()); // PASS
        $this->assertCount(2, $cat->getUsers()); // FAIL (0)
        // ... perform the test (not shown here)
    }

    private function createCat(): Cat
    {
        $cat = new Cat();
        $cat->setName($this->name);
        $this->em->persist($cat);
        $this->em->flush();

        return $cat;
    }

我的问题是,为什么在我的测试中$cat->getUsers()返回?0在运行时它不会,它返回正确的值。它仅在测试中返回0

以下是我的实体的相关摘录,由 Symfony 自动生成:

/**
 * @ORM\Entity(repositoryClass=UserRepository::class)
 */
class User implements UserInterface
{
    /**
     * @ORM\ManyToMany(targetEntity=Cat::class, inversedBy="users")
     */
    private $cats;

    /**
     * @return Collection|Cat[]
     */
    public function getCats(): Collection
    {
        return $this->cats;
    }

    public function addCat(Cat $cat): self
    {
        if (!$this->cats->contains($cat)) {
            $this->cats[] = $cat;
        }

        return $this;
    }

    public function removeCat(Cat $cat): self
    {
        $this->cats->removeElement($cat);

        return $this;
    }
}

/**
 * @ORM\Entity(repositoryClass=CatRepository::class)
 */
class Cat
{
    /**
     * @ORM\ManyToMany(targetEntity=User::class, mappedBy="cats")
     */
    private $users;

    /**
     * @return Collection|User[]
     */
    public function getUsers(): Collection
    {
        return $this->users;
    }
}

标签: phpsymfonydoctrine-ormphpunit

解决方案


原因是,集合不与数据库同步,并且拥有方和反向方之间的同步也不会自动完成。

您的用户实体的类别条目可能会保存到数据库中(虽然我缺少一些级联语句,但我知道什么)。创建类别时,它的用户集合是空的(显然),然后用户将类别添加到数据库中的多对多关系中。

但是,该系列是一个普通的系列。如果您从数据库加载类别,它将是一个延迟加载的 PersistentCollection(或类似的东西),它只会在访问时从数据库中获取项目(延迟加载的定义)。您的测试代码具有普通集合(因为您自己创建了对象)。

不太确定,如果它会起作用,但你可以尝试刷新 cat ( $em->refresh($cat);) 我不太确定,如果它会替换集合。或者,您可以使您的User::addCatthat 它也调用$cat->addUser($this)(您可能必须添加它,注意无限递归,这已经应该通过“包含”检查来防止。)。


推荐阅读