Подсчет строк с помощью эскелета
Я пытаюсь сосчитать строки внутреннего соединения с помощью Esqueleto (версия 2.1.2.1). К сожалению, мой код не компилируется, и я не понимаю, почему. Я посмотрел на следующие примеры того, как это сделать, но не смог понять, что я делаю неправильно: пример1, пример2.
Моя схема выглядит так (упрощенно):
User
Game
state
Player
user UserId Maybe
game GameId
Пользователи могут зарегистрироваться на сайте, чтобы играть в игры. Вы также можете играть без регистрации. Следовательно, есть отдельная таблица Player
, В игре есть состояние. Может быть Ongoing
или какая-то форма игры. Теперь я хочу подсчитать все текущие игры, в которые играет пользователь.
Следующий SQL-запрос прекрасно справляется с этой задачей (для фиксированного userId, равного 1):
SELECT COUNT (*)
FROM Player INNER JOIN Game
ON Player.game = Game.id
WHERE Player.user = 1 AND game.state = "Ongoing"
Однако следующий запрос Esqueleto не компилируется:
[count1] <- runDB $ E.select -- Line 25
$ E.from $ \(player `E.InnerJoin` game) -> do
E.on $ player^.PlayerGame E.==. game^.GameId
E.where_ $
player^.PlayerUser E.==. E.just (E.val userId) E.&&.
game^.GameState E.==. E.val MyGame.Ongoing
return (game, player)
return E.countRows -- Line 32
Сообщение об ошибке выглядит так:
Handler/ListUserGames.hs:25:23:
No instance for (E.SqlSelect (expr0 (E.Value a0)) r0)
arising from a use of ‘E.select’
The type variables ‘r0’, ‘expr0’, ‘a0’ are ambiguous
Note: there are several potential instances:
instance (E.SqlSelect a ra, E.SqlSelect b rb) =>
E.SqlSelect (a, b) (ra, rb)
-- Defined in ‘Database.Esqueleto.Internal.Sql’
instance (E.SqlSelect a ra, E.SqlSelect b rb, E.SqlSelect c rc) =>
E.SqlSelect (a, b, c) (ra, rb, rc)
-- Defined in ‘Database.Esqueleto.Internal.Sql’
instance (E.SqlSelect a ra, E.SqlSelect b rb, E.SqlSelect c rc,
E.SqlSelect d rd) =>
E.SqlSelect (a, b, c, d) (ra, rb, rc, rd)
-- Defined in ‘Database.Esqueleto.Internal.Sql’
...plus 13 others
In the expression: E.select
In the second argument of ‘($)’, namely
‘E.select
$ E.from
$ \ (player `E.InnerJoin` game)
-> do { E.on $ player ^. PlayerGame E.==. game ^. GameId;
E.where_
$ player ^. PlayerUser E.==. E.just (E.val userId)
E.&&. game ^. GameState E.==. E.val MyGame.Ongoing;
.... }’
In a stmt of a 'do' block:
[count1] <- runDB
$ E.select
$ E.from
$ \ (player `E.InnerJoin` game)
-> do { E.on $ player ^. PlayerGame E.==. game ^. GameId;
E.where_
$ player ^. PlayerUser E.==. E.just (E.val userId)
E.&&. game ^. GameState E.==. E.val MyGame.Ongoing;
.... }
Handler/ListUserGames.hs:32:32:
No instance for (E.Esqueleto query0 expr0 backend0)
arising from a use of ‘E.countRows’
The type variables ‘query0’, ‘expr0’, ‘backend0’ are ambiguous
Note: there is a potential instance available:
instance E.Esqueleto E.SqlQuery E.SqlExpr SqlBackend
-- Defined in ‘Database.Esqueleto.Internal.Sql’
In the first argument of ‘return’, namely ‘E.countRows’
In a stmt of a 'do' block: return E.countRows
In the expression:
do { E.on $ player ^. PlayerGame E.==. game ^. GameId;
E.where_
$ player ^. PlayerUser E.==. E.just (E.val userId)
E.&&. game ^. GameState E.==. E.val MyGame.Ongoing;
return (game, player);
return E.countRows }
Тем не менее, точно такой же запрос работает, если я удаляю countRows
, Т.е. следующий код компилируется и делает то, что я хочу.
ongoing <- runDB $ E.select
$ E.from $ \(player `E.InnerJoin` game) -> do
E.on $ player^.PlayerGame E.==. game^.GameId
E.where_ $
player^.PlayerUser E.==. E.just (E.val userId) E.&&.
game^.GameState E.==. E.val MyGame.Ongoing
E.orderBy [E.desc $ game^.GameLastActionTime]
E.limit pagelen
E.offset $ max 0 $ (page1 - 1) * pagelen
return (game, player)
Что я делаю неправильно?
1 ответ
Оказывается, что код Esqueleto выше верен. Ошибка была в другой части кода, где отсутствие ограничений вызывало неоднозначность типов.