Doctrine2 - проблема с запросом двойного левого соединения

Я использую Symfony v3.4 branch с Doctrine.

У меня проблемы с переводом SQL-запроса в запрос Doctrine ORM.

У меня 3 стола.

  • Магазин
  • Фирма
  • Пользователь

Пользователь -> 1: 1 -> Фирма -> 1: 1 -> Магазин

(Панель инструментов разработчика Symfony сообщает, что ассоциации в таблицах сделаны правильно).

Я хочу получить данные Shop, соответствующие созданию User, в одном запросе.

Мой SQL, который дает результат:

SELECT *
FROM mp2_fos_user as u
LEFT JOIN mp2_firm AS f ON u.id = f.firmUserId
LEFT JOIN mp2_shop AS s ON f.id = s.shopFirmId
WHERE u.id = 1

Мой запрос ORM Doctrine

$query = $em->createQueryBuilder()
->select('u, f, s')
->from('App:User', 'u')
->leftJoin('u.userFirm WITH u.id = f.firmUserId', 'f')
->leftJoin('f.firmShop WITH f.id = s.shopFirmId', 's')
->where('u.id = :user_id')
->setParameter('user_id', $user_id)
->getQuery();

в настоящий момент запуск кода приводит к ошибке

[Syntax Error] line 0, col 57: Error: Expected end of string, got 'u'

Что было бы лучше всего для моей проблемы?

Помощь будет очень признательна, Спасибо!

Обновлено

пытался:

$query = $em->createQueryBuilder()
->select('s.id')
->from('App:User', 'u')
->leftJoin('u.userFirm WITH f.firmUser = u', 'f')
->leftJoin('f.firmShop WITH s.shopFirm = f', 's')
->where('u.id = :user_id')
->setParameter('user_id', $user_id)
->getQuery();

получил [Syntax Error] line 0, col 54: Error: Expected end of string, got 'f'

Если объект mp2_firm связан с объектом mp2_fos_user через свойство firmUser, попробуйте оператор WITH как WITH f.firmUser = u. То же самое с идеей с firmShop

geoB 09.06.2018 20:19

@geoB спасибо за предложение. К сожалению, проблема не устранена. Я обновил вопрос, чтобы отразить это. Хотя погрешность немного изменилась ...

Rikijs 09.06.2018 21:39
Стоит ли изучать PHP в 2026-2027 годах?
Стоит ли изучать PHP в 2026-2027 годах?
Привет всем, сегодня я хочу высказать свои соображения по поводу вопроса, который я уже много раз получал в своем сообществе: "Стоит ли изучать PHP в...
Symfony Station Communiqué - 7 июля 2023 г
Symfony Station Communiqué - 7 июля 2023 г
Это коммюнике первоначально появилось на Symfony Station .
Оживление вашего приложения Laravel: Понимание режима обслуживания
Оживление вашего приложения Laravel: Понимание режима обслуживания
Здравствуйте, разработчики! В сегодняшней статье мы рассмотрим важный аспект управления приложениями, который часто упускается из виду в суете...
Установка и настройка Nginx и PHP на Ubuntu-сервере
Установка и настройка Nginx и PHP на Ubuntu-сервере
В этот раз я сделаю руководство по установке и настройке nginx и php на Ubuntu OS.
Коллекции в Laravel более простым способом
Коллекции в Laravel более простым способом
Привет, читатели, сегодня мы узнаем о коллекциях. В Laravel коллекции - это способ манипулировать массивами и играть с массивами данных. Благодаря...
Как установить PHP на Mac
Как установить PHP на Mac
PHP - это популярный язык программирования, который используется для разработки веб-приложений. Если вы используете Mac и хотите разрабатывать...
1
2
45
2
Перейти к ответу Данный вопрос помечен как решенный

Ответы 2

Я думал о чем-то более похожем на это (при условии, что имена сущностей верны):

$query = $em->createQueryBuilder()
    ->select('u, f, s')
    ->from('App:User', 'u')
    ->leftJoin('App:UserFirm f WITH f.firmUser = u')
    ->leftJoin('App:FirmShop s WITH s.shopFirm = f')
    ->where('u.id = :user_id')
    ->setParameter('user_id', $user_id)
    ->getQuery();

Спасибо за идеи, но ваш пример приводит к ошибке.

Rikijs 09.06.2018 22:33
Ответ принят как подходящий

Нет необходимости использовать предложение WITH, если вы определили сопоставление в своих объектах, предложение WITH используется, когда вы хотите объединить свои объекты с дополнительными критериями соответствия.

class User
{
    /**
     * @ORM\YourRelationShipNature(targetEntity = "App\Entity\Firm", mappedBy = "user")
     */
    private $userFirm;
}

class Firm
{
    /**
     * @ORM\YourRelationShipNature(targetEntity = "App\Entity\Shop", mappedBy = "firm")
     */
    private $firmShop;
}


class Shop
{
    //.....
}

И тогда вы можете просто использовать свойства, чтобы присоединиться к своим объектам

$query = $em->createQueryBuilder()
    ->select('u, f, s')
    ->from('App:User', 'u')
    ->leftJoin('u.userFirm', 'f')
    ->leftJoin('f.firmShop', 's')
    ->where('u.id = :user_id')
    ->setParameter('user_id', $user_id)
    ->getQuery();

Другие вопросы по теме