Yii « ActiveRecord для SQL-запроса с COUNT и GROUP BY

Всех приветствую!

Есть 2 таблицы: helpdesk_categories(hc_id, hc_name, ...) и helpdesk_subjects(hs_id, hs_name, hs_is_active, hs_category_id, ...). Все модели созданы, все связи установлены. Связь helpdesk_subjects.hs_category_id = helpdesk_categories.hc_id (M:1 соответственно).

Использую СУБД PostgreSQL.

Мне нужно вывести с помощью CGridView список имен всех "helpdesk_category" и количество активных (hs_is_resolved = FALSE) "helpdesk_subjects" у каждой категории, т.е. будет всего 2 столбца в результате.

В PHP-коде представления передаю данные для CGridView в dataProvider:

/** * GeSHi (C) 2004 - 2007 Nigel McNie, 2007 - 2008 Benny Baumann * (http://qbnz.com/highlighter/ and http://geshi.org/) */ .php.geshi_code {font-family:monospace;} .php.geshi_code .imp {font-weight: bold; color: red;} .php.geshi_code .kw1 {color: #b1b100;} .php.geshi_code .kw2 {color: #000000; font-weight: bold;} .php.geshi_code .kw3 {color: #990000;} .php.geshi_code .co1 {color: #666666; font-style: italic;} .php.geshi_code .co2 {color: #666666; font-style: italic;} .php.geshi_code .co3 {color: #0000cc; font-style: italic;} .php.geshi_code .co4 {color: #009933; font-style: italic;} .php.geshi_code .coMULTI {color: #666666; font-style: italic;} .php.geshi_code .es0 {color: #000099; font-weight: bold;} .php.geshi_code .es1 {color: #000099; font-weight: bold;} .php.geshi_code .es2 {color: #660099; font-weight: bold;} .php.geshi_code .es3 {color: #660099; font-weight: bold;} .php.geshi_code .es4 {color: #006699; font-weight: bold;} .php.geshi_code .es5 {color: #006699; font-weight: bold; font-style: italic;} .php.geshi_code .es6 {color: #009933; font-weight: bold;} .php.geshi_code .es_h {color: #000099; font-weight: bold;} .php.geshi_code .br0 {color: #009900;} .php.geshi_code .sy0 {color: #339933;} .php.geshi_code .sy1 {color: #000000; font-weight: bold;} .php.geshi_code .st0 {color: #0000ff;} .php.geshi_code .st_h {color: #0000ff;} .php.geshi_code .nu0 {color: #cc66cc;} .php.geshi_code .nu8 {color: #208080;} .php.geshi_code .nu12 {color: #208080;} .php.geshi_code .nu19 {color:#800080;} .php.geshi_code .me1 {color: #004000;} .php.geshi_code .me2 {color: #004000;} .php.geshi_code .re0 {color: #000088;} .php.geshi_code span.xtra { display:block; }
$this->widget('MyGridView', array(
  'id' => 'helpdesk-category-grid',
  'dataProvider' => $model->with(
        array(
            'subjects' => array(
                'select' => 'COUNT(subjects.hs_id) AS active_subjects_count', // subjects - псевдоним таблицы helpdesk_subjects, автоматически назначаемый ActiveRecord'ом
                'joinType' => 'LEFT JOIN',
                'condition' => 'hs_is_resolved = FALSE OR hs_is_resolved IS NULL',
                'group' => 'hc_id',
            )
        )
    )->findAll(array('select' => 'hc_name')),
  'sortable' => true,
  'columns' => array(
    'name', // птыаюсь вывести в CGridView для начала хотя бы только имя категории
  ),
));


Когда смотрю, что получилось, вылетает CDbException с информацией о том, что поле hs_id присоединяемой таблицы должно быть в выражении GROUP BY или должно быть использовано агрегатной функцией.

А теперь вопрос! Как мне избавиться от того, чтобы он выбирал это несчастное поле hs_id из таблицы helpdesk_subjects. Я даже явно указал список полей, которые мне нужны. Помогите, пожалуйста. Уже всю голову сломал. :(((

SQL-запрос, который я хотел бы получить:
/** * GeSHi (C) 2004 - 2007 Nigel McNie, 2007 - 2008 Benny Baumann * (http://qbnz.com/highlighter/ and http://geshi.org/) */ .sql.geshi_code {font-family:monospace;} .sql.geshi_code .imp {font-weight: bold; color: red;} .sql.geshi_code .kw1 {color: #993333; font-weight: bold;} .sql.geshi_code .co1 {color: #808080; font-style: italic;} .sql.geshi_code .co2 {color: #808080; font-style: italic;} .sql.geshi_code .coMULTI {color: #808080; font-style: italic;} .sql.geshi_code .es0 {color: #000099; font-weight: bold;} .sql.geshi_code .br0 {color: #66cc66;} .sql.geshi_code .sy0 {color: #66cc66;} .sql.geshi_code .st0 {color: #ff0000;} .sql.geshi_code .nu0 {color: #cc66cc;} .sql.geshi_code span.xtra { display:block; }
SELECT
"helpdesk_categories"."hc_name",
COUNT("helpdesk_subjects"."hs_id")
FROM "helpdesk_categories"
LEFT JOIN "helpdesk_subjects" ON "helpdesk_categories"."hc_id" = "helpdesk_subjects"."hs_category_id"
WHERE "hs_is_resolved" = FALSE OR "hs_is_resolved" IS NULL
GROUP BY "hc_id";


SQL-запрос, который выполняет Yii:
/** * GeSHi (C) 2004 - 2007 Nigel McNie, 2007 - 2008 Benny Baumann * (http://qbnz.com/highlighter/ and http://geshi.org/) */ .sql.geshi_code {font-family:monospace;} .sql.geshi_code .imp {font-weight: bold; color: red;} .sql.geshi_code .kw1 {color: #993333; font-weight: bold;} .sql.geshi_code .co1 {color: #808080; font-style: italic;} .sql.geshi_code .co2 {color: #808080; font-style: italic;} .sql.geshi_code .coMULTI {color: #808080; font-style: italic;} .sql.geshi_code .es0 {color: #000099; font-weight: bold;} .sql.geshi_code .br0 {color: #66cc66;} .sql.geshi_code .sy0 {color: #66cc66;} .sql.geshi_code .st0 {color: #ff0000;} .sql.geshi_code .nu0 {color: #cc66cc;} .sql.geshi_code span.xtra { display:block; }
SELECT
"t"."hc_name" AS "t0_c1",
"t"."hc_id" AS "t0_c0",
COUNT(subjects.hs_id) AS active_subjects_count,
"subjects"."hs_id" AS "t1_c0" /* вот из-за того, что он пытается выбрать это поле, и происходит ошибка! */
FROM "helpdesk_categories" "t"
LEFT JOIN "helpdesk_subjects" "subjects" ON ("subjects"."hs_category_id"="t"."hc_id")
WHERE (hs_is_resolved = FALSE OR hs_is_resolved IS NULL)
GROUP BY hc_id;

1 ответов


В общем, так.
Как изначально хотел сделать я - это неправильно.
Свойство 'dataProvider' в CGridView должно быть экземпляром класса CActiveDataProvider, а я пытался запихнуть туда массив, который возвращает метод findAll() класса CActiveRecord.
Кроме того, при использовании в CDbCriteria 'with' => 'something', получается, что он автоматически делает выборку всех полей присоединяемой таблицы. Поэтому здесь нужно быть внимательным при использовании GROUP BY (это по сути ответ на мой вопрос).

UPD1: Для запросов, в которых используется группировка и подсчет строк (GROUP BY + ORDER) в методе relations() модели, стоит указать связь со специальным типом STAT и использовать ее (спасибо за наводку hordesalik).

Запрос, который нужен в моем случае, можно сформировать, используя такой набор свойств для CDbCriteria:

 $criteria = new CDbCriteria(array(
            'alias' => 'hc',
            'select' => '"hc_name", COUNT("hs"."hs_id") AS active_subjects_count', // subjects - псевдоним таблицы helpdesk_subjects, автоматически назначенный ActiveRecord'ом
            'join' => 'LEFT JOIN "helpdesk_subjects" AS "hs" ON "hc"."hc_id" = "hs"."hs_category_id"',
            'condition' => '"hs"."hs_is_resolved" = FALSE OR "hs"."hs_is_resolved" IS NULL',
            'group' => '"hc"."hc_id"',
        ));
Затем передать этот объект $criteria в конструктор CActiveDataProvider внутри метода search() класса модели (метод search() возвращает объект CActiveDataProvider):
public function search() {

/* ... */

        return new CActiveDataProvider($this, array(
            'criteria' => $criteria,
            /* ... */
        ));
}
Ну и уже затем в представлении (главное, не забыть передать $model в представление) в CGridView в качестве dataProvider указать $model->search():

$this->widget('MyGridView', array(
    'id' => 'helpdesk-category-grid',
    'dataProvider' => $model->search(),
    'columns' => array(
        'hc_name',
        array(
            'name'=>'Category name',
            'value'=>'$data->hc_name',
        ),
        array(
            'name'=>'Count of active subjects',
            'value'=>'$data->active_subjects_count', // не забудьте объявить соответствующие свойства в классе модели, если вы обращаетесь к псевдонимам
        ),
       /* ... */
    ),
));


Я совершенно не знаком с Yii, поэтому не представляю как сделать правильно. Но думаю так можно обмануть автогенератор запросов (зачем их только используют, если они создают лишь неудобства). По времени выполнения что с join-ом, что с подзапросом должно быть примерно одинаково.

SELECT
hc."hc_id",
hc."hc_name",
(SELECT COUNT("hs_id")
  FROM "helpdesk_subjects" hs
  WHERE ("hs_is_resolved" = FALSE OR "hs_is_resolved" IS NULL)
    AND hs."hs_category_id" = hc."hc_id"
) cnt
FROM "helpdesk_categories" hc
GROUP BY hc."hc_id";

Я так понимаю отношение "subjects" у Вас HAS_MANY, для подсчета COUNT есть отношение STAT. Допишите отношение activeSubjectsCount с типом STAT, классом HelpdescSubject, ну и поле там объединяющее. + нужно дописать условие выбора по полю hs_is_resolved, это тоже можно сделать в отношении ('condition'=>'hs_is_resolved=0')
http://www.yiiframework.com/doc/guide/1.1/ru/database.arr#sec-9