Symfony-更新唯一的OneToMany关系属性

时间:2018-07-09 14:42:10

标签: php symfony doctrine-orm

一个公司可以有多个电子邮件,并且所有电子邮件都必须是唯一的。
这是我的公司和CompanyEmail实体

公司电子邮件实体:

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

    /**
     * @ORM\Column(type="string", length=128, unique=true)
     * @Assert\Email()
     */
    private $name;

    /**
     * @ORM\ManyToOne(targetEntity="App\Entity\Company", inversedBy="emails")
     * @ORM\JoinColumn(nullable=false)
     */
    private $company;

    // ...
}

公司实体:

/**
 * @ORM\Entity(repositoryClass="App\Repository\CompanyRepository")
 */
class Company
{
    // ...

    /**
     * @ORM\OneToMany(targetEntity="App\Entity\CompanyEmail", mappedBy="company", orphanRemoval=true, cascade={"persist"})
     * @Assert\Valid
     */
    private $emails;

    // ...
}

,并且我正在使用自定义的 EmailsInputType ,该自定义 DataTransformer

class EmailArrayToStringTransformer implements DataTransformerInterface
{
    public function transform($emails): string
    {
        return implode(', ', $emails);
    }


    public function reverseTransform($string): array
    {
        if ($string === '' || $string === null) {
            return [];
        }

        $inputEmails = array_filter(array_unique(array_map('trim', explode(',', $string))));

        $cEmails = [];
        foreach($inputEmails as $email){
            $cEmail = new CompanyEmail();
            $cEmail->setName($email);
            $cEmails[] = $cEmail;
        }

        return $cEmails;
    }
}

并在Controller中使用此编辑方法

/**
     * @Route("/edit/{id}", name="admin_company_edit", requirements={"id": "\d+"}, methods={"GET", "POST"})
     */
    public function edit(Request $request, $id): Response
    {
        $entityManager = $this->getDoctrine()->getManager();
        $company = $entityManager->getRepository(Company::class)->find($id);

        $form = $this->createForm(CompanyType::class, $company);

        $form->handleRequest($request);

        if ($form->isSubmitted() && $form->isValid()) {
            $entityManager->flush();
        }
    }

此代码有两个问题

1-在编辑表单中,当我尝试保留已保存的电子邮件时,Symfony会生成验证错误,告诉您该电子邮件已退出。

2-当我从代码中删除验证限制时,Symfony抛出数据库错误“ *完整性约束冲突:1062复制条目... *”

我应该做些什么才能使我的代码按预期工作!

1 个答案:

答案 0 :(得分:1)

问题就在这里

$countDirs = $(Get-ChildItem -path c:\folder\subfolder -Recurse -Directory).count

您需要检索public function reverseTransform($string): array { [...] foreach($inputEmails as $email){ $cEmail = new CompanyEmail(); [...] } [...] } 而不是创建新的。 因此,基本上,注入email,尝试查找是否已存在电子邮件(CompanyEmailRepository),如果不存在,则创建一个新电子邮件,但如果存在,则使用检索到的内容。

只需几个笔记

  • 请注意电子邮件所有者(因此检索应该针对每个用户,因为除非您可以指定一些别名或共享地址,否则没人可以共享同一封邮件)
  • 也许您不需要像findOneBy(['name'])这样的额外实体,因为您可以使用CompanyEmail字段,以逗号分隔的方式存储它们(除非您需要一些额外的参数,除非您需要在电子邮件上执行一些索引/查询操作)