# `Flop.Meta`
[🔗](https://github.com/woylie/flop/blob/0.26.4/lib/flop/meta.ex#L1)

Defines a struct for holding meta information of a query result.

# `t`

```elixir
@type t() :: %Flop.Meta{
  backend: module() | nil,
  current_offset: non_neg_integer() | nil,
  current_page: pos_integer() | nil,
  end_cursor: String.t() | nil,
  errors: [{atom(), term()}],
  flop: Flop.t(),
  has_next_page?: boolean(),
  has_previous_page?: boolean(),
  next_offset: non_neg_integer() | nil,
  next_page: pos_integer() | nil,
  opts: keyword(),
  page_size: pos_integer() | nil,
  params: %{optional(String.t()) =&gt; term()},
  previous_offset: non_neg_integer() | nil,
  previous_page: pos_integer() | nil,
  schema: module() | nil,
  start_cursor: String.t() | nil,
  total_count: non_neg_integer() | nil,
  total_pages: non_neg_integer() | nil
}
```

Meta information for a query result.

- `:flop` - The `Flop` struct used in the query.
- `:schema` - The schema module passed as `for` option.
- `:backend` - The backend module if the query was made using a module with
  `use Flop`.
- `:current_offset` - The `:offset` value used in the query when using
  offset-based pagination or a derived value when using page-based pagination.
  Always `nil` when using cursor-based pagination.
- `:current_page` - The `:page` value used in the query when using page-based
  pagination or a derived value when using offset-based pagination. Note that
  the value will be rounded if the offset lies between pages. Always `nil`
  when using cursor-based pagination.
- `:errors` - Any validation errors that occurred. The format is the same as
  the result of `Ecto.Changeset.traverse_errors(changeset, & &1)`.
- `:previous_offset`, `:next_offset`, `:previous_page`, `:next_page` - Values
  based on `:current_page` and `:current_offset`/`page_size`. Always `nil`
  when using cursor-based pagination.
- `:start_cursor`, `:end_cursor` - The cursors of the first and last record
  in the result set. Only set when using cursor-based pagination with
  `:first`/`:after` or `:last`/`:before`.
- `:has_previous_page?`, `:has_next_page?` - Set in all pagination types.
  Note that `:has_previous_page?` is always `true` when using cursor-based
  pagination with `:first` and `:after` is set; likewise, `:has_next_page?` is
  always `true` when using cursor-based pagination with `:before` and `:last`
  is set.
- `:page_size` - The page size or limit of the query. Set to the `:first`
  or `:last` parameter when using cursor-based pagination.
- `:params` - The original, unvalidated params that were passed. Only set
  if validation errors occurred.
- `:total_count` - The total count of records for the given query. Always
  `nil` when using cursor-based pagination.
- `:total_pages` - The total page count based on the total record count and
  the page size. Always `nil` when using cursor-based pagination.

# `with_errors`
*since 0.19.0* 

```elixir
@spec with_errors(map(), keyword(), keyword()) :: t()
```

Returns a `Flop.Meta` struct with the given params, errors, and opts.

This function is used internally to build error responses in case of
validation errors. You can use it to add additional parameter validation.

The given parameters parameters are normalized before being added to the
struct. The errors have to be passed as a keyword list (same format as the
result of `Ecto.Changeset.traverse_errors(changeset, & &1)`).

## Example

In this list function, the given parameters are first validated with
`Flop.validate/2`, which returns a `Flop` struct on success. You can then pass
that struct to a custom validation function, along with the original
parameters and the opts, which both are needed to call this function.

    def list_pets(%{} = params) do
      opts = [for: Pet]

      with {:ok, %Flop{} = flop} <- Flop.validate(params, opts),
           {:ok, %Flop{} = flop} <- custom_validation(flop, params, opts) do
        Flop.run(Pet, flop, for: Pet)
      end
    end

In your custom validation function, you can retrieve and manipulate the filter
values in the `Flop` struct with the functions defined in the `Flop.Filter`
module.

    defp custom_validation(%Flop{} = flop, %{} = params, opts) do
      %{value: date} = Flop.Filter.get(flop.filters, :date)

      if date && Date.compare(date, Date.utc_today()) != :lt do
        errors = [filters: [{"date must be in the past", []}]]
        {:error, Flop.Meta.with_errors(params, errors, opts)}
      else
        {:ok, flop}
      end
    end

Note that in this example, `Flop.Filter.get/2` is used, which only returns the
first filter in the given filter list. Depending on how you use Flop, the
filter list may have multiple entries for the same field. In that case, you
may need to either use `Flop.Filter.get_all/2` and apply the validation on all
returned filters, or reduce over the whole filter list. The latter has the
advantage that you can attach the error to the actual list entry.

    def custom_validation(%Flop{} = flop, %{} = params, opts) do
      filter_errors =
        flop.filters
        |> Enum.reduce([], &validate_filter/2)
        |> Enum.reverse()

      if Enum.any?(filter_errors, &(&1 != [])) do
        errors = [filters: filter_errors]
        {:error, Flop.Meta.with_errors(params, errors, opts)}
      else
        {:ok, flop}
      end
    end

    defp validate_filter(%Flop.Filter{field: :date, value: date}, acc)
         when is_binary(date) do
      date = Date.from_iso8601!(date)

      if Date.compare(date, Date.utc_today()) != :lt,
        do: [[value: [{"date must be in the past", []}]] | acc],
        else: [[] | acc]
    end

    defp validate_filter(%Flop.Filter{}, acc), do: [[] | acc]

---

*Consult [api-reference.md](api-reference.md) for complete listing*
