Per un progetto Symfony2 ho dovuto creare una relazione tra un post di blog e le cosiddette piattaforme. Una piattaforma definisce un filtro specifico basato sul dominio che si usa per visualizzare il sito. Ad esempio: se ti unisci al sito per url first-example.com, il sito fornirà solo post di blog, che sono collegati a questa piattaforma specifica.Utilizzo di EntityRepository :: findBy() con relazioni Many-To-Many porterà a un E_NOTICE in Doctrine
Per fare ciò, ho creato due entità Post e piattaforma. Successivamente li ho mappati insieme con una relazione Many-To-Many. Sto cercando di recuperare i dati tramite questa relazione Molti-A-Molti dalla funzione integrata findBy()
in Doctrines 'EntityRepository
.
// every one of these methods will throw the same error
$posts = $postRepo->findBy(array('platforms' => array($platform)));
$posts = $postRepo->findByPlatforms($platform);
$posts = $postRepo->findByPlatforms(array($platform));
Dove $postRepo
è il repository corretto per il Post
entità e $platform
un Platform
oggetto esistente.
In entrambi i casi: finisco ottenendo il seguente errore:
ErrorException: Notice: Undefined index: joinColumns in [...]/vendor/doctrine/orm/lib/Doctrine/ORM/Persisters/BasicEntityPersister.php line 1495
[...]/vendor/doctrine/orm/lib/Doctrine/ORM/Persisters/BasicEntityPersister.php:1495
[...]/vendor/doctrine/orm/lib/Doctrine/ORM/Persisters/BasicEntityPersister.php:1452
[...]/vendor/doctrine/orm/lib/Doctrine/ORM/Persisters/BasicEntityPersister.php:1525
[...]/vendor/doctrine/orm/lib/Doctrine/ORM/Persisters/BasicEntityPersister.php:1018
[...]/vendor/doctrine/orm/lib/Doctrine/ORM/Persisters/BasicEntityPersister.php:842
[...]/vendor/doctrine/orm/lib/Doctrine/ORM/EntityRepository.php:157
[...]/src/Foobar/BlogBundle/Tests/ORM/PostTest.php:102
E 'anche possibile recuperare entités correlati in un molti-a-molti in questo modo, o sono costretto a scrivere queste funzioni da solo? La cosa strana è: Doctrine non genera alcun errore come: "Non è possibile"., Ma uno interno E_NOTICE
. Ecco perché credo che dovrebbe essere possibile, ma mi mancano alcuni punti qui.
Spogliato verso parti interessanti, le due Entità sono simili a questa.
<?php
namespace Foobar\CommunityBundle\Entity;
use Doctrine\Common\Collections\ArrayCollection;
use Doctrine\ORM\Mapping as ORM;
// [...] other namespace stuff
/**
* @ORM\Entity(repositoryClass="Foobar\CommunityBundle\Entity\Repository\PlatformRepository")
* @ORM\Table(name="platforms")
*/
class Platform
{
/**
* @ORM\Id
* @ORM\Column(type="integer")
* @ORM\GeneratedValue(strategy="AUTO")
*/
protected $id;
// [...] other field stuff
}
<?php
namespace Foobar\BlogBundle\Entity;
use Doctrine\Common\Collections\ArrayCollection;
use Doctrine\ORM\Mapping as ORM;
// [...] other namespace stuff
/**
* @ORM\Entity(repositoryClass="Foobar\BlogBundle\Entity\Repository\PostRepository")
* @ORM\Table(name="posts")
*/
class Post implements Likeable, Commentable, Taggable, PlatformAware
{
/**
* @ORM\Id
* @ORM\Column(type="integer")
* @ORM\GeneratedValue(strategy="AUTO")
*/
protected $id;
/**
* @ORM\ManyToMany(targetEntity="Foobar\CommunityBundle\Entity\Platform", cascade={"persist"})
* @ORM\JoinTable(name="map_post_platform",
* joinColumns={@ORM\JoinColumn(name="post_id", referencedColumnName="id")},
* inverseJoinColumns={@ORM\JoinColumn(name="platform_id", referencedColumnName="id")}
* )
*/
protected $platforms;
// [...] other fields
/**
* Constructor
*/
public function __construct()
{
// [...]
$this->platforms = new ArrayCollection();
}
}
E, naturalmente, il file composer.json (così ridotta alle linee interessate)
{
[...]
"require": {
"php": ">=5.3.3",
"symfony/symfony": "2.1.*",
"doctrine/orm": ">=2.2.3,<2.4-dev",
"doctrine/doctrine-bundle": "1.0.*",
"doctrine/doctrine-fixtures-bundle": "dev-master",
[...]
},
[...]
}
Grazie per la risposta. Solo una piccola aggiunta. Il tuo primo metodo è scrivere una propria funzione in un repository personalizzato. Forse mi sono espresso poco chiaro ma ho provato a recuperare le entità correlate tramite la funzione incorporata 'find *()' di Doctrine. Il secondo metodo non funziona perché ho un'associazione unidirezionale. Pertanto non esiste la proprietà '$ posts' su' Platform' e, a causa di ciò, nessun getter e setter. Tuttavia la tua risposta mi aiuta molto, dato che ora sono più sicuro che non è possibile usare solo i metodi incorporati per ottenere le associazioni molti-a-molti filtrate. – devsheeep