В одной из прошлых статей я рассматривал особенности построения запросов к базе данных на основе Drupal 8 Database API. Безусловно, с помощью данного API можно получать любые данные, содержащиеся в базе, однако, такой подход не всегда оправдан. В случае работы с сущностями правильным решением будет использование Entity API, а если точнее, то сервиса entity.query.

 

Сервис entity.query

Сервис entity.query представляет собой развитие EntityFieldQuery из Drupal 7 и, аналогично своему прародителю, служит для удобного получения сущностей, соответствующих определённому паттерну. Получать таким образом можно любые сущности - и контентные и конфигурационные.

Сервис предоставляет объект запроса (реализует интерфейс \Drupal\Core\Entity\Query\QueryInterface) для заданного типа сущности. Существует два способа получить данный объект: правильный и н̶е̶п̶р̶а̶в̶и̶л̶ь̶н̶ы̶й̶ простой.

Правильный способ заключается в получении объекта с помощью контейнера сервисов. Это наиболее рекомендуемый подход. Он удобен при наличии доступа к контейнеру (например, в методе create() у класса плагина).

/** @var \Symfony\Component\DependencyInjection\ContainerInterface $container */
$service = $container->get('entity.query');
$query = $service->get($entity_type);

Простой способ предполагает использование статического метода entityQuery() класса \Drupal. Данный подход не рекомендуется к применению, поэтому использовать его стоит только в общем случае при отсутствии доступа к контейнеру сервисов.

$query = \Drupal::entityQuery($entity_type);

В обоих случаях в переменную $entity_type нужно передать идентификатор типа сущностей, с которыми требуется работать (например, node, user, comment и т.д.).

 

Загружаем сущности правильно

Перейдём к конкретным примерам. Для простоты будем создавать объект запроса с помощью статического метода класса \Drupal.

Пример: Получить идентификаторы всех опубликованных нод типа article.

$query = \Drupal::entityQuery('node');
$query->condition('type', 'article');
$query->condition('status', 1);
$nids = $query->execute();

Здесь используются методы condition() и execute(). Первый используется для применения условий к запросу, а второй непосредственно для выполнения запроса и получения идентификаторов загружаемых сущностей. В метод condition() можно передавать четыре параметра:

  • field - машинное имя поля сущности, для которого определяется условие (является обязательным параметром);
  • value - значение поля, может быть скалярным значением или массивом. Если параметр опущен, то будет использоваться NULL;
  • operator - оператор условия. Если параметр опущен, то ему будет присвоено значение «=»;
  • langcode - код языка. Если параметр опущен, любой перевод будет удовлетворять условию. В противном случае выборка будет ограничена заданным переводом.

В entity.query не существует различий между свойствами и полями сущности, как это было при использовании EntityFieldQuery в Drupal 7. Теперь определение условий для свойств и полей делается в рамках метода condition().

Параметр field должен содержать имя поля сущности, за которым может следовать имя столбца поля. В свою очередь, столбец может быть ссылочным свойством (в случае ссылочного поля), за которым может следовать имя поля целевой сущности и т. д. Кроме того, тип целевой сущности можно указать, добавив «:target_entity_type_id» после ссылочного свойства.

В качестве оператора допустимы следующие значения:

  • =, <>, >, >=, <, <=, STARTS_WITH, CONTAINS, ENDS_WITH: Эти операторы предполагают, что тип value соответствует типу значений поля;
  • IN, NOT IN: Эти операторы предполагают, что value является массивом и тип каждого элемента соответствует типу значений поля;
  • BETWEEN: Этот оператор предполагает, что value является массивом из двух элементов, тип которых соответствует типу значений поля;

Пример: Получить идентификаторы всех опубликованных нод типа poem или fairy_tale, которые в тизере поля body содержат фразу «У лукоморья дуб зеленый», помечены трёмя и более тегами, один из которых «Сказки Пушкина», и опубликованы пользователем с идентификатором 1799 в течение последней недели.

$query = \Drupal::entityQuery('node');
$query->condition('type', ['poem', 'fairy_tale'], 'IN');
$query->condition('body.summary', 'У лукоморья дуб зеленый', 'CONTAINS');
$query->condition('field_tags.%delta', 2);
$query->condition('field_tags.entity:taxonomy_term.name', 'Сказки Пушкина');
$query->condition('uid', 1799);
$query->condition('created', time() - 604800, '>');
$query->condition('status', 1);
$nids = $query->execute();

Стоит отметить, что такие значения параметра field, как body и body.value будут идентичны, так как value является столбцом по умолчанию. В случае multiple-поля можно определять условие относительно конкретного значения с помощью указания дельты. Например, параметр field_tags.7 будет указывать на восьмое значение поля field_tags. Кроме того, можно задать условие для самой дельты, используя «%delta», как это было реализовано в примере выше.

Тип целевой сущности указывать необязательно, если он точно известен. Например, если известно, что поле field_tags хранит ссылки на термины таксономии, то параметры field_tags.entity.name и field_tags.entity:taxonomy_term.name будут тождественны.

Пример: Получить идентификаторы всех нод типа article, которые имеют тизер поля body и не отмечены тегами.

$query = \Drupal::entityQuery('node');
$query->condition('type', 'article');
$query->exists('body.summary');
$query->notExists('field_tags');
$nids = $query->execute();

За проверку поля на пустоту отвечают методы exists() и notExists(). Важно обратить внимание, что эти методы не выполняют проверку на наличие поля у сущности. Это значит, что при проверке несуществующего поля будет получена ошибка. При этом вызов exists($field) идентичен вызову condition($field, NULL, 'IS NOT NULL'), а вызов notExists($field) идентичен condition($field, NULL, 'IS NULL').

Пример: Получить идентификаторы всех нод, упорядоченные по дате создания.

$query = \Drupal::entityQuery('node')->sort('created', 'DESC');
$nids = $query->execute();

Пример: Получить количество всех нод.

$query = \Drupal::entityQuery('node')->count();
$count = $query->execute();

Пример: Получить идентификаторы последних 10 нод.

$query = \Drupal::entityQuery('node');
$query->sort('created', 'DESC');
$query->range(0, 10);
$nids = $query->execute();

С помощью метода pager() также есть возможность управления результатами выборки на основе пагинации.

Пример: Получить идентификаторы всех нод, опубликованных пользователем с идентификатором 1 или с именем «superduperuser».

$query = \Drupal::entityQuery('node', 'OR');
$query->condition('uid', 1);
$query->condition('uid.entity.name', 'superduperuser');
$nids = $query->execute();

Посредством значения параметра conjunction в методе создания объекта запроса можно управлять объединением условий. Допустимы значения AND и OR.  В первом случае условия будут соединены конъюнкцией, а во втором - дизъюнкцией. По умолчанию условия объединяются с помощью конъюнкции.

Пример: Получить идентификаторы всех нод типа article, опубликованных пользователем с идентификатором 1 или с именем «superduperuser».

$query = \Drupal::entityQuery('node');
$query->condition('type', 'article');
 
$group = $query->orConditionGroup();
$group->condition('uid', 1);
$group->condition('uid.entity.name', 'superduperuser');
 
$query->condition($group);
$nids = $query->execute();

Методы andConditionGroup() и orConditionGroup() позволяют добавлять условия, связанные конъюнкцией и дизъюнкцией, независимо от параметра conjunction. Эти методы позволяют реализовывать действительно гибкие условия.

Пример: Получить объекты нод, идентификаторы которых были получены с помощью entity.query.

$nids = \Drupal::entityQuery('node')->execute();
$nodes = \Drupal\node\Entity\Node::loadMultiple($nids);

Метод execute() возвращает массив (при использовании метода count() будет возвращено число), значения которого всегда будут содержать идентификаторы сущностей. Ключи массива могут быть как идентификаторами ревизий сущностей, так и идентификаторами самих сущностей в зависимости от наличия поддержки ревизий у заданного типа сущностей.

Методы currentRevision()latestRevision() и allRevisions() позволяют управлять идентификаторами ревизий в результирующем массиве. При использовании первого метода ключами массива будут идентификаторы текущих ревизий, а при использовании второго метода - идентификаторы самых поздних. Наконец, при использовании третьего метода ключами массива будут идентификаторы всех ревизий.

 

Полезные материалы

Ещё по теме

Комментарии

Ключи, указываемые в "condition", у разных типов сущностей отличаются. Но можно унифицировать свой запрос, предварительно получая список ключей, примерно так:
 

$keys = \Drupal::entityTypeManager()->getDefinition('node')->getKeys();
Alexey Murz Korepov

А можно каким-то образом в результате entityQuery сразу получить готовую загруженную сущность без промежуточного массива?
Т.е. вместо

$query = \Drupal::entityQuery('node', 'OR');
$query->condition('uid', 1);
$query->condition('uid.entity.name', 'superduperuser');
$query->range(0, 1);
$nids = $query->execute();
$node = Node::load(current($nids));

иметь что-то вида:

$node = $query->getFirstAndLoad();

Такой возможности нет.
Антон Банников

Я долго изучал, как работает правильный метод вызова запросов для сущностей. В итоге нашел статью, что он уже устарел. https://www.drupal.org/node/2849874

Да, сервис entity.query объявлен устаревшим и в Drupal 9 его уже нет. Теперь объект запроса нужно получать через хранилище конкретного типа сущностей. При этом метод \Drupal::entityQuery() всё ещё доступен для не объектно-ориентированного кода.

как подписаться на RSS?

https://drupal-coder.ru/rss.xml

Добавить комментарий