web-dev-qa-db-fra.com

Comment résoudre l'erreur "n'est pas dans GROUP BY" dans la requête mysql

J'ai deux modèles: posts et likings qui ont une relation un-à-plusieurs (un poste a donc beaucoup de j'aime). Le modèle Likings possède également un champ isActive qui indique que le fait de aimer est actif ou passif.

Je veux obtenir (trier) les 5 meilleurs messages qui ont reçu le maximum de "j'aime" actifs (seuls les "j'aime" dont le champ isActive est vrai seraient pris en compte).

C'est la requête:

select posts.*, count(likings.id) as likes_count from 'posts'
left join 'likings' on 'likings'.'post_id' = 'posts'.'id' and 'likings'.'isActive' = 1
group by 'posts'.'id'
order by 'likes_count' desc 
limit 5

qui résulte de cette requête laravel:

Post::selectRaw('posts.*, count(likes.id) as likes_count')
    ->leftJoin('likes', function ($join) {
        $join->on('likes.post_id', '=', 'posts.id')
             ->where('likes.is_active', '=', 1);
    })
    ->groupBy('posts.id')
    ->orderBy('likes_count', 'desc')
    ->take(5)
    ->get();

Et c'est l'erreur:

SQLSTATE[42000]: Syntax error or access violation: 1055
'database.posts.user_id' isn't in GROUP BY

Ici, posts.user_id est également un champ de la table posts et indique le propriétaire de la publication.

Comment puis-je résoudre cette erreur? Il semble que ce ne soit pas logique de changer mysql config (probablement en supprimant le mode ONLY_FULL_GROUP_BY), mais un minimum de modification dans la requête.

8
horse

Chaque champ non agrégé doit être groupé. 

C'est un comportement standard, qui dans mysql, dépend du mode ONLY_FULL_GROUP_BY.
Ce mode est activé par défaut dans> = 5.7

select posts.id, post.user_id, count(likings.id) as likes_count
from posts
left join likings on likings.post_id = posts.id and likings.isActive= 1
group by posts.id, posts.user_id
order by likes_count desc 
limit 5

Ou, vous pouvez aggregate il: 

select posts.id, MIN(post.user_id), count(likings.id) as likes_count
from posts
left join likings on likings.post_id = posts.id and likings.isActive= 1
group by posts.id
order by likes_count desc 
limit 5

Autre moyen - changez sql_mode:

SET SESSION sql_mode = '';

En outre, vous pouvez le diviser en 2 requêtes: 

  1. Obtenir des identifiants agrégés et post
  2. Récupère les données de publication avec les identifiants récupérés (avec IN clause)
9
vp_arth

Dans 5.7, sqlmode est défini par défaut sur:

ONLY_FULL_GROUP_BY, NO_AUTO_CREATE_USER, STRICT_TRANS_TABLES, NO_ENGINE_SUBSTITUTION

Pour supprimer la clause ONLY_FULL_GROUP_BY, procédez comme suit:

SET GLOBAL sql_mode=(SELECT REPLACE(@@sql_mode,'ONLY_FULL_GROUP_BY',''));
2
Raugaral