如何编写在日期时间字段上执行 group_by MONTH 的 Ecto 查询

How to write a Ecto query that does a group_by MONTH on a datetime field

我正在执行外查询并尝试按 q.created_date 进行分组。此查询成功执行了 GROUP BY,但它是按秒执行的。我正在尝试按月分组。

MYQUERY |> group_by([q], [q.created_date, q.id])

是否有类似的东西:

MYQUERY |> group_by([q], [month(q.created_date), q.id])

您可以使用 fragmentdate_part('month', field) 在 PostgreSQL 中提取月份:fragment("date_part('month', ?)", p.inserted_at))

iex(1)> from(Post) |> select([p], p.inserted_at) |> Repo.all
[debug] QUERY OK db=1.1ms queue=0.1ms
SELECT p0."inserted_at" FROM "posts" AS p0 []
[#Ecto.DateTime<2000-01-01 00:00:00>, #Ecto.DateTime<2000-02-02 00:00:00>,
 #Ecto.DateTime<2000-03-03 00:00:00>, #Ecto.DateTime<2000-04-04 00:00:00>,
 #Ecto.DateTime<2000-04-02 00:00:00>, #Ecto.DateTime<2000-03-02 00:00:00>,
 #Ecto.DateTime<2000-02-02 00:00:00>, #Ecto.DateTime<2000-01-02 00:00:00>]
iex(2)> from(Post) |> group_by([p], fragment("date_part('month', ?)", p.inserted_at)) |> select([p], count("*")) |> Repo.all
[debug] QUERY OK db=1.7ms
SELECT count('*') FROM "posts" AS p0 GROUP BY date_part('month', p0."inserted_at") []
[2, 2, 2, 2]

使用 Dogbert 的回答和 arvidkahl 的出色提示,可以使用以下方法在片段维度上按月分组而不会产生歧义:

from q in query,
  join: i in Item,
  on: i.id == q.item_id,
  group_by: [i.category_id, fragment("date_trunc('month', ?)", q.inserted_at)],
  select: %{category_id: i.category_id, date: fragment("date_trunc('day', ?)", q.inserted_at), quantity: sum(q.quantity)}