问:Symfony4 EasyAdmin oneToMany未保存在数据库中

时间:2018-01-17 15:19:15

标签: symfony one-to-many symfony2-easyadmin

我有2个实体,产品(产品)和制造商(Fabricant)。

当我尝试从Easy Admin更新它们时,如果我更改了产品方面的制造商,一切正常,它可以正常工作,制造商管理员确实显示了适量的儿童产品。

但是,如果我这样做 - 从制造商管理员那里选择子产品 - 它不会将其保存在数据库中。它确实保存了名称,但没有保存子列表。

这里和那里的一些主题表明我必须确保在addProduct函数中,我还使用$ product-> setManufacturer($ this)对产品进行操作; ...我做了(见下面的代码)。

其他人提到在admin配置中,我应该将by_reference选项设置为false。我也做了。但没有成功。

另一个建议是确保两个实体之间的级联是正确的,我已将它放到"所有"直到我能弄清楚什么是错的,但它仍然无法奏效。没有错误消息,没有警告,它甚至保存其他字段,但不保存这个。有什么想法吗?

产品:

<?php

namespace App\Entity;

use Doctrine\ORM\Mapping as ORM;

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

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

    /**
     * @ORM\ManyToOne(targetEntity="Fabricant", inversedBy="produits", cascade="all")
     */
    private $fabricant;

    public function __toString()
    {
        return ($this->nom != null) ? $this->nom : '';
    }

    public function getId()
    {
        return $this->id;
    }

    public function setId($id)
    {
        $this->id = $id;

        return $this;
    }

    public function getNom()
    {
        return $this->nom;
    }

    public function setNom($nom)
    {
        $this->nom = $nom;

        return $this;
    }

    public function getFabricant()
    {
        return $this->fabricant;
    }

    public function setFabricant($fabricant)
    {
        $this->fabricant = $fabricant;

        return $this;
    }
}

制造商:

<?php

namespace App\Entity;

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

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

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

    /**
     * @ORM\OneToMany(targetEntity="Produit", mappedBy="fabricant", cascade="all")
     */
    private $produits;

    public function __toString()
    {
        return ($this->nom != null) ? $this->nom : '';
    }

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

    public function getId()
    {
        return $this->id;
    }

    public function setId($id)
    {
        $this->id = $id;

        return $this;
    }

    public function getNom()
    {
        return $this->nom;
    }

    public function setNom($nom)
    {
        $this->nom = $nom;

        return $this;
    }

    public function getProduits()
    {
        return $this->produits;
    }

    public function addProduit(Produit $produit)
    {
        if ($this->produits->contains($produit)) {
            return;
        }
        $this->produits[] = $produit;
        $produit->setFabricant($this);
        return $this;
    }

    public function removeProduit($produit)
    {
        $this->produits->removeElement($produit);
        $produit->setFabricant(null);
    }
}

Easy Admin yaml config:

easy_admin:
    entities:
        Produit:
            class: App\Entity\Produit
            list:
                fields:
                    - id
                    - nom
                    - fabricant
            new:
                fields:
                    - nom
                    - { property: 'fabricant', type_options: { 'by_reference': false } }
        Fabricant:
            class: App\Entity\Fabricant
            list:
                fields:
                    - id
                    - nom
                    - produits
            new:
                fields:
                    - nom
                    - { property: 'produits', type_options: { by_reference: false } }
            edit:
                fields:
                    - nom
                    - { property: 'produits', type_options: { multiple: true, by_reference: false } }

3 个答案:

答案 0 :(得分:2)

原因是bug in EasyAdmin controller使用具体的$ entity调用flush(),因此其他更改不会保留。您可以等待新版本或extend AdminController and override methods

控制类的变化:

easy_admin_bundle:
    resource: 'App\Controller\AdminController'
    prefix: /admin
    type: annotation

扩展管理员控制器:

<?php
namespace App\Controller;

use EasyCorp\Bundle\EasyAdminBundle\Controller\AdminController as BaseAdminController;

class AdminController extends BaseAdminController
{
    protected function persistEntity($entity)
    {
        $this->em->persist($entity);
        $this->em->flush();
    }

    protected function updateEntity($entity)
    {
        $this->em->flush();
    }

    protected function removeEntity($entity)
    {
        $this->em->remove($entity);
        $this->em->flush();
    }
}

答案 1 :(得分:2)

这对我有用

  form:
    fields:
      - { property: 'school' }
      - { property: 'companies', type_options: { by_reference: false } }

答案 2 :(得分:0)

issue appears very often和我自己碰到了它。这是我的解决方案,不覆盖任何控制器操作:

对于使用EasyAdminBundle v1.17.08 - v1.17.22的任何人,您都应该先升级。显然Issue #1679引入了导致此错误的错误。

下一步是确保所有权实体中的cascade选项已设置:

@ORM\OneToMany(targetEntity="PageBlock", mappedBy="page", cascade="all", orphanRemoval=true)

还要在您的表单中设置by_reference选项:

# in your easy_admin config block...
      - property: 'blocks'
        type: 'collection'
        type_options:
          entry_type: ...PageBlockType
          by_reference: false

最后但并非最不重要的一点是,我在父实体中缺少addremove方法。还要确保命名正确。因此,对于名为property的{​​{1}},请使用以下代码:

$block