Получить список объектов, затем установить их во многих отношениях - PullRequest
0 голосов
/ 08 января 2020

Я использую doctrine в проекте.

Я получил класс Card

class Card implements \JsonSerializable
    {

    /**
     * @ORM\Id
     * @ORM\Column(type="string")
     */
    private $id;

    /**
     * @ORM\Column(type="string", length=255)
     */
    private $sign;

    /**
     * @ORM\Column(type="string", length=10)
     */
    private $number;

   /**
     * Card constructor.
     *
     * @param $sign
     * @param $number
     */
    public function __construct($sign, $number)
    {
        $this->sign = $sign;
        $this->number = $number;

        $this->id = \sprintf('%s_%s', $sign, $number);
    }
}

И класс CardPicker, содержащий Card

class CardPicker
{
    /**
     * @ORM\Id
     * @ORM\GeneratedValue
     * @ORM\Column(type="integer")
     */
    private $id;

    /**
     * Card that could be picked.
     *
     * Many CardPicker have Many Card.
     *
     * @ManyToMany(targetEntity="Card", cascade={"persist", "remove"})
     * @JoinTable(name="available_picker_cards",
     *     joinColumns={@JoinColumn(name="card_picker_id", referencedColumnName="id")},
     *     inverseJoinColumns={@JoinColumn(name="card_id", referencedColumnName="id")}
     * )
     */
    private $availableCards;

public function setCards(array $cards): void
    {
        $this->availableCards = new ArrayCollection($cards);
    }
}

Я сохранил несколько карт в своей базе данных , И я хочу, чтобы начать свой CardPicker. Поэтому я пишу этот метод:

 public function initCardPicker(): CardPicker
    {
        // fetch all wanted cards and put them in CardPicker
        $cards = $this->cardRepository->findAll();

        $cardPicker = new CardPicker();

        $cardPicker->setCards($cards);

        return $this->repository->save($cardPicker);
    }

Но я получил это сообщение об ошибке: SQLSTATE[23000]: Integrity constraint violation: 19 UNIQUE constraint failed: card.id (500 Internal Server

Кажется, что когда я сохраняю CardPicker, он пытается re-persist мои карты, которые должен быть уникальным в моей системе.

Что-то не так с отображениями?

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

Репозиторий:

/**
 * @method CardPicker|null find($id, $lockMode = null, $lockVersion = null)
 * @method CardPicker|null findOneBy(array $criteria, array $orderBy = null)
 * @method CardPicker[]    findAll()
 * @method CardPicker[]    findBy(array $criteria, array $orderBy = null, $limit = null, $offset = null)
 */
class CardPickerRepository extends ServiceEntityRepository
{
    public function __construct(ManagerRegistry $registry)
    {
        parent::__construct($registry, CardPicker::class);
    }

    /**
     * Save cardPicker and return it.
     *
     * @throws \Doctrine\Common\Persistence\Mapping\MappingException
     * @throws \Doctrine\ORM\ORMException
     * @throws \Doctrine\ORM\OptimisticLockException
     */
    public function save(CardPicker $picker): CardPicker
    {
        $this->_em->persist($picker);
        $this->_em->flush();
        $this->_em->clear();

        return $picker;
    }
}

1 Ответ

1 голос
/ 08 января 2020

@ grenierj Я создал классы, и они работали для меня.

выполнить после создания объекта

php bin/console doctrine:migration:diff
php bin/console doctrine:migration:migrate

config / services.yaml

    App\Repository\CardRepository:
        arguments:
            - '@doctrine'

    App\Entity\CardPicker:
        arguments:
            - '@doctrine'
            - 'App\Repository\CardRepository'

App / Entity / Card. php

<?php

namespace App\Entity;

use Doctrine\ORM\Mapping as ORM;

/**
 * @ORM\Entity(repositoryClass="App\Repository\CardRepository")
 */
class Card
{
    /**
     * @ORM\Id()
     * @ORM\GeneratedValue()
     * @ORM\Column(type="string")
     */
    private $id;

    /**
     * @ORM\Column(type="string", length=255)
     */
    private $sign;

    /**
     * @ORM\Column(type="string", length=10)
     */
    private $number;

    /**
     * Card constructor.
     *
     * @param $sign
     * @param $number
     */
    public function __construct($sign, $number)
    {
        $this->sign = $sign;
        $this->number = $number;

        $this->id = \sprintf('%s_%s', $sign, $number);
    }

    /**
     * @return string
     */
    public function getId(): string
    {
        return $this->id;
    }

    /**
     * @param string $id
     * @return Card
     */
    public function setId(string $id): Card
    {
        $this->id = $id;
        return $this;
    }
}

Приложение / Entity / CardPicker. php

<?php

namespace App\Entity;

use Doctrine\Common\Collections\ArrayCollection;
use Doctrine\ORM\Mapping as ORM;

/**
 * @ORM\Entity(repositoryClass="App\Repository\CardPickerRepository")
 */
class CardPicker
{
    /**
     * @ORM\Id()
     * @ORM\GeneratedValue()
     * @ORM\Column(type="integer")
     */
    private $id;

    /**
     * Card that could be picked.
     *
     * Many CardPicker have Many Card.
     *
     * @ORM\ManyToMany(targetEntity="Card", cascade={"persist", "remove"})
     * @ORM\JoinTable(name="available_picker_cards",
     *     joinColumns={@ORM\JoinColumn(name="card_picker_id", referencedColumnName="id")},
     *     inverseJoinColumns={@ORM\JoinColumn(name="card_id", referencedColumnName="id")}
     * )
     */
    private $availableCards;

    public function __construct()
    {
        $this->availableCards = new ArrayCollection();
    }

    /**
     * @return mixed
     */
    public function getId()
    {
        return $this->id;
    }

    /**
     * @param mixed $id
     * @return CardPicker
     */
    public function setId($id)
    {
        $this->id = $id;
        return $this;
    }

    /**
     * @return mixed
     */
    public function getAvailableCards()
    {
        return $this->availableCards;
    }

    /**
     * @param mixed $availableCards
     * @return CardPicker
     */
    public function setAvailableCards(array $availableCards)
    {
        $this->availableCards = $availableCards;
        return $this;
    }

    public function addAvailableCard(Card $card)
    {
        if (!$this->availableCards->contains($card)) {
            $this->availableCards->add($card);
        }
    }

    public function removeAvailableCard(Card $card)
    {
        $this->availableCards->removeElement($card);
    }
}

App / Repository / CardPickerRepository. php

<?php

namespace App\Repository;

use App\Entity\CardPicker;
use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository;
use Doctrine\Common\Persistence\ManagerRegistry;

/**
 * @method CardPicker|null find($id, $lockMode = null, $lockVersion = null)
 * @method CardPicker|null findOneBy(array $criteria, array $orderBy = null)
 * @method CardPicker[]    findAll()
 * @method CardPicker[]    findBy(array $criteria, array $orderBy = null, $limit = null, $offset = null)
 */
class CardPickerRepository extends ServiceEntityRepository
{
    protected $cardRepository;

    public function __construct(ManagerRegistry $registry, CardRepository $cardRepository)
    {
        $this->cardRepository = $cardRepository;
        parent::__construct($registry, CardPicker::class);
    }

    public function initCardPicker(): CardPicker
    {
        $em = $this->getEntityManager();
        $cards = $this->cardRepository->findAll();

        $picker = new CardPicker();
        $picker->setAvailableCards($cards);

        $em->persist($picker);
        $em->flush();

        return $picker;
    }
}

Приложение / Хранилище / CardRepository. php

<?php

namespace App\Repository;

use App\Entity\Card;
use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository;
use Doctrine\Common\Persistence\ManagerRegistry;

/**
 * @method Card|null find($id, $lockMode = null, $lockVersion = null)
 * @method Card|null findOneBy(array $criteria, array $orderBy = null)
 * @method Card[]    findAll()
 * @method Card[]    findBy(array $criteria, array $orderBy = null, $limit = null, $offset = null)
 */
class CardRepository extends ServiceEntityRepository
{
    public function __construct(ManagerRegistry $registry)
    {
        parent::__construct($registry, Card::class);
    }
}

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...