【问题标题】:Testing a Doctrine2 Entity with assertEquals results in fatal out-of-memory error使用 assertEquals 测试 Doctrine2 实体会导致致命的内存不足错误
【发布时间】:2013-09-26 19:44:17
【问题描述】:

我有一个使用 Doctrine2 自定义存储库和 Doctrine Fixtures 的 PHPUnit 测试。我想测试一个查询是否从我的夹具中返回了一个预期的实体。

但是当我尝试$this->assertEquals($expectedEntity, $result); 时,我得到了Fatal error: out of memory。我猜它正在递归到所有关系和实体管理器等等。

有没有好方法来测试这种相等性?我应该只对实体的 ID 使用 assertEquals 吗?

编辑: 这是测试代码

<?php
use Liip\FunctionalTestBundle\Test\WebTestCase;

class AbstractRepositoryTestCase extends WebTestCase
{
    /**
     * @var Doctrine\ORM\EntityRepository 
     */
    protected $repo;

    /**
     * @var Doctrine\Common\DataFixtures\Executor\AbstractExecutor
     */
    protected $fixtureExecutor;

    /**
     * @var string Which repository to load, overriden by derived class
     */
    protected $repoName;

    /**
     * @var array Fixture classes to load on setup
     */
    protected $fixtures = array();

    public function setUp()
    {
        $kernel = static::createKernel();
        $this->repo = $kernel->boot();
        $this->repo = $kernel->getContainer()
                             ->get('doctrine.orm.entity_manager')
                             ->getRepository($this->repoName);

        $this->fixtureExecutor = $this->loadFixtures($this->getFixtures());
    }

    public function getFixtures()
    {
        return $this->fixtures;
    }
}

class ArticleRepositoryTest extends AbstractRepositoryTestCase
{
    /**
     * @var string Which repository to load, overriden by derived class
     */
    protected $repoName = 'MyMainBundle:Article';

    /**
     * @var array Fixture classes to load on setup
     */
    protected $fixtures = array(
        'My\MainBundle\DataFixtures\ORM\LoadUserData',
        'My\MainBundle\DataFixtures\ORM\LoadArticleData',
        'My\MainBundle\DataFixtures\ORM\LoadFeedsData',
        'My\MainBundle\DataFixtures\ORM\LoadFeedDataData',
        'My\MainBundle\DataFixtures\ORM\LoadUserReadArticleData',
    );
    public function testGetNextArticle_ExpectCorrect()
    {
        /** @var Doctrine\Common\DataFixtures\ReferenceRepository **/
        $refRepo = $this->fixtureExecutor->getReferenceRepository();

        /** @var FeedStream\MainBundle\Entity\Article **/
        $curr = $refRepo->getReference('feed-1-article-3');
        $expected = $refRepo->getReference('feed-1-article-2');
        $expected2 = $refRepo->getReference('feed-1-article-1');
        $next = $this->repo->getNextArticle($curr->getFeed()->getId(), $curr);

        $this->assertNotNull($next);
        // this is the part that doesn't work
        $this->assertEquals($expected, $next);
        // this is the code I've used instead
        $this->assertEquals($expected->getId(), $next->getId());
    }
}

这里是实体(getter/setter 省略以节省空间)

<?php

namespace My\MainBundle\Entity;

use Doctrine\ORM\Mapping as ORM;

/**
 * My\MainBundle\Entity\Article
 *
 * @ORM\Table(name="articles", uniqueConstraints={
 *   @ORM\UniqueConstraint(name="feed_guid", columns={"feed_id", "guid"}),
 *   @ORM\UniqueConstraint(name="article_slug_unique", columns={"feed_id", "slug"})
 * })
 * @ORM\Entity(repositoryClass="My\MainBundle\Repository\ArticleRepository")
 */
class Article
{
    /**
     * @var integer $id
     *
     * @ORM\Column(name="id", type="integer", nullable=false)
     * @ORM\Id
     * @ORM\GeneratedValue(strategy="IDENTITY")
     */
    private $id;

    /**
     * @var string $guid
     *
     * @ORM\Column(name="guid", type="string", length=255, nullable=false)
     */
    private $guid;

    /**
     * @var string $title
     *
     * @ORM\Column(name="title", type="string", length=255, nullable=false)
     */
    private $title;

    /**
     * @var datetime $pubDate
     *
     * @ORM\Column(name="pub_date", type="datetime", nullable=true)
     */
    private $pubDate;

    /**
     * @var text $summary
     *
     * @ORM\Column(name="summary", type="text", nullable=true)
     */
    private $summary;

    /**
     * @var text $content
     *
     * @ORM\Column(name="content", type="text", nullable=false)
     */
    private $content;

    /**
     * @var string $sourceUrl
     *
     * @ORM\Column(name="source_url", type="string", length=255, nullable=true)
     */
    private $sourceUrl;

    /**
     * @var string $commentUrl
     *
     * @ORM\Column(name="comment_url", type="string", length=255, nullable=true)
     */
    private $commentUrl;

    /**
     * @var string $slug
     *
     * @ORM\Column(name="slug", type="string", length=64, nullable=false)
     */
    private $slug;

    /**
     * @var string $thumbnailFile
     *
     * @ORM\Column(name="thumbnail_file", type="string", length=64, nullable=true)
     */
    private $thumbnailFile;

    /**
     * @var My\MainBundle\Entity\ArticleEnclosure
     *
     * @ORM\ManyToOne(targetEntity="My\MainBundle\Entity\ArticleEnclosure")
     * @ORM\JoinColumns({
     *   @ORM\JoinColumn(name="thumbnail_enclosure_id", referencedColumnName="id")
     * })
     */
    private $thumbnailEnclosure;

    /**
     * @var My\MainBundle\Entity\ArticleImageScrape
     *
     * @ORM\ManyToOne(targetEntity="My\MainBundle\Entity\ArticleImageScrape")
     * @ORM\JoinColumns({
     *   @ORM\JoinColumn(name="thumbnail_scrape_id", referencedColumnName="id", unique=false)
     * })
     */
    private $thumbnailScrape;

    /**
     * @var My\MainBundle\Entity\ArticleBitly
     *
     * @ORM\OneToOne(targetEntity="My\MainBundle\Entity\ArticleBitly", mappedBy="article", orphanRemoval=true)
     */
    private $bitly;

    /**
     * @var My\MainBundle\Entity\ArticleEnclosure
     *
     * @ORM\OneToMany(targetEntity="My\MainBundle\Entity\ArticleEnclosure", mappedBy="article", orphanRemoval=true)
     */
    private $enclosures;

    /**
     * @var My\MainBundle\Entity\ArticleImageScrape
     *
     * @ORM\OneToMany(targetEntity="My\MainBundle\Entity\ArticleImageScrape", mappedBy="article", orphanRemoval=true)
     */
    private $scrapes;

    /**
     * @var My\MainBundle\Entity\ArticleLink
     *
     * @ORM\OneToMany(targetEntity="My\MainBundle\Entity\ArticleLink", mappedBy="article", orphanRemoval=true)
     */
    private $links;

    /**
     * @var My\MainBundle\Entity\Feed
     *
     * @ORM\ManyToOne(targetEntity="My\MainBundle\Entity\Feed", inversedBy="articles")
     * @ORM\JoinColumns({
     *   @ORM\JoinColumn(name="feed_id", referencedColumnName="id", nullable=false)
     * })
     */
    private $feed;

    /**
     * @var My\MainBundle\Entity\ArticleAuthor
     *
     * @ORM\ManyToOne(targetEntity="My\MainBundle\Entity\ArticleAuthor", inversedBy="articles")
     * @ORM\JoinColumns({
     *   @ORM\JoinColumn(name="author_id", referencedColumnName="id")
     * })
     */
    private $author;

    public function __construct()
    {
        $this->links = new \Doctrine\Common\Collections\ArrayCollection();
    }
}

【问题讨论】:

  • 你能展示一些代码(比如实体)和测试吗?学说 2 中的实体与实体管理器没有任何联系,所以你要么做错了,要么坏了!

标签: php doctrine-orm phpunit


【解决方案1】:

除了 ID 之外,您还应该测试类。

【讨论】:

  • 谢谢,我想这是有道理的,因为任何其他实体类型也会有 getId()
  • 您还可以使用 === 来测试对象引用的相等性,而不是使用 == 来比较对象属性。从理论上讲,它应该可以工作,因为 Doctrine 使用身份映射,但我不确定此映射是否在所有情况下都使用。
  • 有趣。我不会想到从 Fixture 参考和查询结果中的参考是相同的,但正如你所说的那样。
猜你喜欢
  • 2021-02-01
  • 2014-01-25
  • 2014-06-01
  • 2011-10-16
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2011-05-05
相关资源
最近更新 更多