Приглашение Symfony2 FOSUserBundle: работает только с сторонними ассоциациями

Entity/ Пользователь

namespace My\SampleBundle\Entity;

use Doctrine\ORM\Mapping as ORM;
use Symfony\Component\Validator\Constraints as Assert;

/**
 * @ORM\Entity
 * @ORM\Table(name="fos_user")
 */
class User extends \FOS\UserBundle\Entity\User
{
    /** @ORM\Id @ORM\Column(type="integer") @ORM\GeneratedValue(strategy="AUTO") */
    protected $id;

    /**
     * @ORM\OneToOne(targetEntity="Invitation", inversedBy="user")
     * @ORM\JoinColumn(referencedColumnName="code")
     * @Assert\NotNull(message="Your invitation is wrong")
     */
    protected $invitation;

    public function setInvitation(Invitation $invitation)
    {
        $this->invitation = $invitation;
    }

    public function getInvitation()
    {
        return $this->invitation;
    }
}

Entity/ Приглашение

namespace My\SampleBundle\Entity;

use Doctrine\ORM\Mapping as ORM;

/** @ORM\Entity */
class Invitation
{
    /** @ORM\OneToOne(targetEntity="User", mappedBy="invitation", cascade={"persist", "merge"}) */
    protected $user;

    /** @ORM\Id @ORM\Column(type="string", length=6) */
    protected $code;

    /** @ORM\Column(type="string", length=256) */
    protected $email;

    /**
     * When sending invitation be sure to set this value to `true`
     *
     * It can prevent invitations from being sent twice
     *
     * @ORM\Column(type="boolean")
     */
    protected $sent = false;

    public function __construct()
    {
        // generate identifier only once, here a 6 characters length code
        $this->code = substr(md5(uniqid(rand(), true)), 0, 6);
    }

    public function getCode()
    {
        return $this->code;
    }

    public function getEmail()
    {
        return $this->email;
    }

    public function setEmail($email)
    {
        $this->email = $email;
    }

    public function isSent()
    {
        return $this->sent;
    }

    public function send()
    {
        $this->sent = true;
    }

    public function getUser()
    {
        return $this->user;
    }

    public function setUser(User $user)
    {
        $this->user = $user;
    }

    /**
     * Set code
     *
     * @param string $code
     * @return Invitation
     */
    public function setCode($code)
    {
        $this->code = $code;

        return $this;
    }

    /**
     * Set sent
     *
     * @param boolean $sent
     * @return Invitation
     */
    public function setSent($sent)
    {
        $this->sent = $sent;

        return $this;
    }

    /**
     * Get sent
     *
     * @return boolean 
     */
    public function getSent()
    {
        return $this->sent;
    }
}

ошибка

Вы не можете искать поле ассоциации 'My\SampleBundle\Entity\Invitation#user', потому что это обратная сторона ассоциации. Методы поиска работают только на сторонних ассоциациях. 500 Внутренняя ошибка сервера - ORMException

Я стадия, которая только что выполнила документацию.

https://github.com/FriendsOfSymfony/FOSUserBundle/blob/master/Resources/doc/adding_invitation_registration.md

Дисплей в комплекте нормальный. Тем не менее, это стало ошибкой, если я отправляю регистрационную форму.

Любая помощь?

РЕДАКТИРОВАТЬ:

На самом деле, я сначала настроил "inversedBy".
Предварительный вопрос.
Приглашение Symfony2 FOSUserBundle: ошибки отображения "inversedBy"
На первый взгляд, это работает. Однако ошибки отображения отображаются профилировщиком.

Мой \ SampleBundle \ Entity \ Invitation # не содержит обязательного атрибута "inversedBy".

Итак, я изменил его в ответ на совет. Я не знаю, что об этом думать.

3 ответа

Решение

Было решение.
https://github.com/FriendsOfSymfony/FOSUserBundle/issues/800

public function reverseTransform($value)
{
    // ...

    return $this->entityManager
        ->getRepository('My\SampleBundle\Entity\Invitation')
        ->findOneBy(array(
            'code' => $value,
            'user' => null,        <= Removing 'user' solves the issue
        ));
}

Это как говорится в ошибке. Вместо mappedBy вы должны использовать inversedBy для объекта Invitation и использовать mappedBy для объекта User для $ приглашения.

/** @ORM\OneToOne(targetEntity="User", inversedBy="invitation", cascade={"persist", "merge"}) */
    protected $user;

Вы также можете преодолеть эту проблему, создав собственный метод хранилища для поиска пользователей по приглашению.

В итоге я изменил свой трансформатор следующим образом:

public function reverseTransform($value)
{
    if (null === $value || '' === $value) {
        return null;
    }

    if (!is_string($value)) {
        throw new UnexpectedTypeException($value, 'string');
    }

    $invitation = $this->entityManager
        ->getRepository('SixString\PearBundle\Entity\Invitation')
        ->findOneBy(array(
            'code' => $value,
        ));
    if($this->entityManager->getRepository('SixString\PearBundle\Entity\User')->findOneBy(array("invitation" => $invitation))){
        return null;
    }

    return $invitation;
}

Я раздели 'user' => null но добавил проверку, чтобы увидеть, было ли уже использовано приглашение

Другие вопросы по тегам