# Seeker
This library will help you easily add searching to your Phoenix/Ecto
application, without any additional dependencies.
There are advanced searching solutions around, like ElasticSearch or Algolia.
Seeker will do the job for many Phoenix/Ecto apps, without the need to run
additional infrastructure or work in a different language.
## Installation
Add `seeker` to your list of dependencies in `mix.exs`:
```elixir
def deps do
[
{:seeker, "~> 0.1.0"}
]
end
```
## Usage
Add a `MyApp.Seeker` module as the following example:
```elixir
# lib/my_app/seeker.ex
defmodule MyApp.Seeker do
use Seeker,
otp_app: :my_app,
repo: MyApp.Repo
end
```
Assuming a params map with the following structure:
```elixir
%{
q: %{
first_name_eq: "Foo",
email_end: "gmail.com",
status_in: ["premium", "plus"]
},
s: "last_name+asc"
}
```
Please check [supported predicates](https://github.com/amco/seeker-ex/blob/main/lib/seeker/query.ex#L8).
Then, use `all/2` function in the controller like this:
```elixir
defmodule MyAppWeb.UserController do
use MyAppWeb, :controller
alias MyApp.Seeker
alias MyApp.Accounts.User
def index(conn, params) do
users = User |> Seeker.all(params)
render(conn, :index, users: users)
end
end
```
Use `query/2` function when performing queries after using `Seeker` or
using pagination. Like this:
```elixir
defmodule MyAppWeb.UserController do
use MyAppWeb, :controller
alias MyApp.Accounts.User
alias MyApp.{Seeker, Repo}
def index(conn, params) do
page =
User
|> Seeker.query(params)
|> Repo.paginate(params)
render(conn, :index, page: page)
end
end
```
### Associations
Based on the following schemas:
```elixir
defmodule MyApp.Post do
use Ecto.Schema
schema "posts" do
field(:title, :string)
belongs_to(:author, MyApp.User)
belongs_to(:category, MyApp.Category)
end
end
defmodule MyApp.User do
use Ecto.Schema
schema "users" do
field(:name, :string)
field(:email, :string)
has_many(:posts, MyApp.Post, foreign_key: :author_id)
end
end
defmodule Seeker.Schemas.Category do
use Ecto.Schema
schema "categories" do
field(:name, :string)
has_many(:posts, MyApp.Post)
end
end
```
Posts can be filtered and sorted using double underscore (`__`) in
the params like this:
```elixir
%{
q: %{
category__name_eq: "Foo",
author__email_end: "gmail.com"
},
s: "author__email+desc"
}
```
Documentation can be generated with [ExDoc](https://github.com/elixir-lang/ex_doc)
and published on [HexDocs](https://hexdocs.pm). Once published, the docs can
be found at <https://hexdocs.pm/seeker>.