Doctrine Proxy没有初始化

时间:2013-12-18 12:08:51

标签: php mysql symfony doctrine-orm

当我将文件上传到Symfony时,它就像上传一样。我在文件上传时使用了Symfony教程并对其进行了修改以满足我的需求。

if($form->isValid())
{
    $em = $this->oStarter->getEntityManager();

    // Save file to database
    $uploadedFile = new ProfilePicture();
    $uploadedFile->setFile($formData["profile_picture"]);
    $user->setProfilePicture($uploadedFile);
    $uploadedFile->setUser($user);

    $em->persist($uploadedFile);
    $em->persist($user);

    $em->flush();

    // Other things like Twig templates etc..

此代码用于上传图像并将其设置为用户的个人资料图片。通过Controller中的$this->getUser()找到用户。当我在刷新后输出实体时,它会向我显示有效实体的转储,就像我期望的那样。

当我访问此用户的个人资料页面时,找不到图像。当我检查MySQL表时,我找到了具有正确ID和路径的ProfilePicture的有效条目。用户还可以像您期望的那样引用ProfilePicture的ID。相反,该页面显示以下转储:

$avatar = $user->getProfilePicture();
$path = $avatar->getWebPath();
Debug::dump($avatar);


object(stdClass)#938 (8) 
{
  ["__CLASS__"]=>
  string(42) "Takeabyte\CoreBundle\Entity\ProfilePicture"
  ["__IS_PROXY__"]=>
  bool(true)
  ["__PROXY_INITIALIZED__"]=>
  bool(false)
  ["id"]=>
  NULL
  ["user"]=>
  object(stdClass)#1011 (52) {
    ["__CLASS__"]=>
    string(32) "Takeabyte\CoreBundle\Entity\User"
    ["id"]=>
    int(11)
    // lots of user info
    }
  ["file"]=>
  NULL
  ["path"]=>
  NULL
  ["temp"]=>
  NULL
}

转储显示没有设置路径。即使在调用代理的功能之后,似乎也没有加载真实数据。我做错了什么?

修改 实体如下:

/**
 * @author Tim Cocu
 * @author Rick Slinkman
 *
 * @ORM\Entity
 * @ORM\Table(name="profilepictures")
 * @Database(target="client")
 */
class ProfilePicture extends Image
{
    /**
     * @ORM\Id
     * @ORM\Column(type="integer")
     * @ORM\GeneratedValue(strategy="AUTO")
     */
    private $id;

    /**
     * @ORM\OneToOne(targetEntity="\Takeabyte\CoreBundle\Entity\User", mappedBy="profilePicture")
     */
    private $user;

    // accessors and mutators
}

/**
 * Description of Image
 *
 * @ORM\MappedSuperclass
 * @Database(target="client")
 * @author Rick Slinkman (r.slinkman@take-a-byte.eu)
 */
class Image extends MediaFile
{
    /**
     * @param ClassMetadata $metadata
     */
    public static function loadValidatorMetadata(ClassMetadata $metadata)
    {
        $metadata->addPropertyConstraint('file', new Assert\File(array(
            'maxSize' => 6000000,
            'mimeTypes' => array(
                "image/jpeg",
                "image/png",
                "image/gif"
            ),
        )));
    }

    // other functions
}

/**
 * Standard container of an uploaded media file
 * @author Rick Slinkman
 * @author Tim Cocu
 * 
 * @ORM\HasLifecycleCallbacks
 * @ORM\MappedSuperclass
 * @Database(target="client")
 * 
 * Based on:
 * http://symfony.com/doc/current/cookbook/doctrine/file_uploads.html
 */
class MediaFile 
{
    /**
     * @Assert\File(maxSize="6000000")
     */
    protected $file;

    /**
     * @ORM\Column(type="string", length=255, nullable=true)
     */
    protected $path;

    /**
     * Temporary storage on file moving.
     */
    protected $temp;

    /**
     * @ORM\PrePersist()
     * @ORM\PreUpdate()
     */
    public function preUpload()
    {
        if (null !== $this->getFile()) 
        {
            // do whatever you want to generate a unique name
            $filename = sha1(uniqid(mt_rand(), true));
            $this->path = $filename.'.'.$this->getFile()->guessExtension();
        }
    }

    /**
     * @ORM\PostPersist()
     * @ORM\PostUpdate()
     */
    public function upload()
    {
        if (null === $this->getFile()) 
        {
            return;
        }

        // if there is an error when moving the file, an exception will
        // be automatically thrown by move(). This will properly prevent
        // the entity from being persisted to the database on error
        $this->getFile()->move($this->getUploadRootDir(), $this->path);

        // check if we have an old image
        if (isset($this->temp)) 
        {
            // delete the old image
            unlink($this->getUploadRootDir().'/'.$this->temp);
            // clear the temp image path
            $this->temp = null;
        }
        $this->file = null;
    }

    /**
     * @ORM\PostRemove()
     */
    public function removeUpload()
    {
        if ($file = $this->getAbsolutePath()) 
        {
            unlink($file);
        }
    }

    // other functions 
}

/**
 * @author: Jordy - j.deruijter@take-a-byte.eu
 * @author: Rick - r.slinkman@take-a-byte.eu
 * @author: Tim - t.cocu@take-a-byte.eu
 * @since: 25-10-13
 *
 * @ORM\Entity
 * @ORM\Table(name="fos_user_user")
 * @Database(target="client")
 */
class User extends BaseUser
{
    /**
     * @ORM\Id
     * @ORM\Column(type="integer")
     * @ORM\GeneratedValue(strategy="AUTO")
     */
    protected $id;

    // Lots of data

    /**
     * @var ProfilePicture
     * @ORM\OneToOne(targetEntity="\Takeabyte\CoreBundle\Entity\ProfilePicture", inversedBy="user")
     */
    protected $profilePicture;

    // Even more data
}

1 个答案:

答案 0 :(得分:4)

我想我在一年前遇到过类似的错误。

您会看到,在进行身份验证时,User实体的序列化(文本)版本会存储在您的会话中。当您访问防火墙后面的页面时,它将被反序列化并再次转换为User。但是,由于您与ProfilePicture 的关系并不急切,并且在序列化期间,此属性未序列化。代理对象不可序列化......

因此,当它尝试从会话中检索经过身份验证的用户时,其$profilePicture属性设置为NULL

这可能是你的情况吗?

IDEA#1:

  1. EAGER实体中将关系设置为User
  2. 尝试设置 always_authenticate_before_granting: true中的confir.ymlsecurity阻止)
  3. 我相信这会导致安全性进入数据库并在每次访问页面时重新获取User实体...

    IDEA#2:

    刷新您的用户实体并手动提取profilePicture 。也许您可以单独在会话中存储用户个人资料图片?