From 4a0c1ea42e9f45a5b45fbfaf812a8bfa34ca9330 Mon Sep 17 00:00:00 2001 From: miffy Date: Sat, 7 Sep 2019 02:32:23 +0200 Subject: [PATCH] Add separate module for pagination of queries --- lib/mobilizon/page.ex | 48 +++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 48 insertions(+) create mode 100644 lib/mobilizon/page.ex diff --git a/lib/mobilizon/page.ex b/lib/mobilizon/page.ex new file mode 100644 index 00000000..ce2d440b --- /dev/null +++ b/lib/mobilizon/page.ex @@ -0,0 +1,48 @@ +defmodule Mobilizon.Page do + @moduledoc """ + Module for pagination of queries. + """ + + import Ecto.Query, warn: false + + alias Mobilizon.Repo + + defstruct [ + :total, + :elements + ] + + @type t :: %__MODULE__{ + total: integer, + elements: struct + } + + @doc """ + Returns a Page struct for a query. + """ + @spec build_page(Ecto.Query.t(), integer | nil, integer | nil) :: t + def build_page(query, page, limit) do + [total, elements] = + [ + fn -> Repo.aggregate(query, :count, :id) end, + fn -> Repo.all(paginate(query, page, limit)) end + ] + |> Enum.map(&Task.async/1) + |> Enum.map(&Task.await/1) + + %__MODULE__{total: total, elements: elements} + end + + @doc """ + Add limit and offset to the query. + """ + @spec paginate(Ecto.Query.t(), integer | nil, integer | nil) :: Ecto.Query.t() + def paginate(query, page \\ 1, size \\ 10) + + def paginate(query, page, _size) when is_nil(page), do: paginate(query) + def paginate(query, page, size) when is_nil(size), do: paginate(query, page) + + def paginate(query, page, size) do + from(query, limit: ^size, offset: ^((page - 1) * size)) + end +end