Я использую следующий запрос с помощью querybuilder, но как-то многие многие отношения работают не так, как ожидалось.
$shopData = $sm->createQueryBuilder()
->select('v')
->from('AdminBundle:Voucher','v')
->innerJoin('v.shop', 's')
->leftJoin('AdminBundle:VoucherProgram', 'vp', \Doctrine\ORM\Query\Expr\Join::ON, 'vp.id = v.program_id')
->leftJoin('AdminBundle:shopHistory', 'sh', \Doctrine\ORM\Query\Expr\Join::ON, 'sh.shop = s.id')
->where('s.shopStatus = :shopStatus')
->setParameter('shopStatus', Shop::SHOP_ACTIVATED)
->andWhere('s.highlightedHome = :highlightedHome')
->setParameter('highlightedHome', Shop::SHOP_HIGHLIGHTED_HOME)
->andWhere('s.offers = \'voucher\'')
->setFirstResult(0)
->setMaxResults(6)
->addOrderBy('v.discount_amount', 'DESC')
->groupBy('sh.shop')
->getQuery()
->getSql();
Сгенерированный запрос выглядит следующим образом:
SELECT v FROM AdminBundle:Voucher v INNER JOIN v.shop s LEFT JOIN AdminBundle:VoucherPrograms vp ON vp.id = v.program_id LEFT JOIN AdminBundle:shopHistory sh ON sh.shop = s.id WHERE s.shopStatus = :shopStatus AND s.highlightedHome = :highlightedHome AND s.offers = 'voucher' GROUP BY sh.shop ORDER BY v.discount_amount DESC
Если я удаляю все и сохраняю только внутреннее соединение для отношения Many2Many, он работает как ожидалось.
$sm->createQueryBuilder()
->select('v')
->from('AdminBundle:Voucher','v')
->innerJoin('v.shop', 's');
Вот сгенерированный запрос:
SELECT l0.* FROM voucher l0_ INNER JOIN shop_voucher l2_ ON l0_.id = l2_.voucher_id INNER JOIN shop l1_ ON l1_.id = l2_.shop_id;
Так что мне интересно, почему система не поднимает правильное отношение, когда я добавляю больше объединений.
Вот мои основные сущности, имеющие отношение Many2Many:
Shop.php
namespace AdminBundle\Entity;
use Doctrine\ORM\Mapping as ORM;
use Doctrine\Common\Collections\ArrayCollection;
/**
* Shop.
*
* @ORM\Table(name="shop")
* @ORM\Entity(repositoryClass="AdminBundle\Entity\ShopRepository")
*/
class Shop
{
const SHOP_DEACTIVATED = 0;
const SHOP_ACTIVATED = 1;
const SHOP_HIGHLIGHTED_HOME = 1;
................................
/**
* @ORM\ManyToMany(targetEntity="Voucher", inversedBy="shop")
* @ORM\JoinTable(name="shop_voucher")
*/
private $voucher;
................................
Voucher.php
namespace AdminBundle\Entity;
use Doctrine\ORM\Mapping as ORM;
#use Doctrine\Common\Collections\ArrayCollection;
/**
* Voucher.
*
* @ORM\Table(name="voucher")
* @ORM\Entity(repositoryClass="AdminBundle\Entity\VoucherRepository")
*/
class Voucher
{
................................
/**
* @ORM\ManyToMany(targetEntity="Shop", mappedBy="voucher", cascade={"persist"})
*/
private $shop;
................................
Я уже проверял, что в стеке есть те же вопросы, но я хочу знать, как мы можем расширять запрос. У меня есть решение для моего вопроса, как показано ниже, но не совсем то, что происходит в описанном выше сценарии.
$shopDataQuery = $connection->prepare('SELECT v.* FROM voucher AS v INNER JOIN shop_voucher AS sv ON sv.voucher_id = v.id INNER JOIN shop AS s ON s.id = sv.shop_id LEFT JOIN voucher_programs AS vp ON vp.id = v.program_id LEFT JOIN shop_history AS sh ON sh.shop = s.id WHERE s.shopStatus = :shopStatus AND s.highlightedHome = :highlightedHome AND s.offers = 'voucher' GROUP BY sh.shop ORDER BY v.discount_amount DESC LIMIT 6');
Update:
Вот shopHistory.php
namespace AdminBundle\Entity;
use Doctrine\ORM\Mapping as ORM;
use Doctrine\Common\Collections\ArrayCollection;
/**
* shopHistory.
*
* @ORM\Table(name="shop_history")
* @ORM\Entity(repositoryClass="AdminBundle\Entity\shopHistoryRepository")
*/
class shopHistory
{
/**
* @var int
*
* @ORM\Column(name="id", type="integer")
* @ORM\Id
* @ORM\GeneratedValue(strategy="AUTO")
*/
private $id;
/**
* @var int
* @ORM\ManyToOne(targetEntity="Shop", inversedBy="shopHistory")
* @ORM\JoinColumn(name="shop", referencedColumnName="id")
*/
private $shop;
Вот VoucherProgram.php
namespace AdminBundle\Entity;
use Doctrine\ORM\Mapping as ORM;
use Doctrine\Common\Collections\ArrayCollection;
/**
* VoucherProgram.
*
* @ORM\Table(name="voucher_program")
* @ORM\Entity(repositoryClass="AdminBundle\Entity\VoucherProgramRepository")
*/
class VoucherProgram
{
/**
* @var int
*
* @ORM\Column(name="id", type="integer")
* @ORM\Id
* @ORM\GeneratedValue(strategy="AUTO")
*/
private $id;
/**
* @ORM\OneToMany(targetEntity="Voucher", mappedBy="program")
*/
private $voucher;
/**
* @ORM\OneToMany(targetEntity="Shop", mappedBy="vprogram")
*/
private $shop;