Query Arguments

Our GraphQL API would be pretty boring (and useless) if clients couldn't retrieve filtered data.

Let's assume that our API needs to add the ability to look-up users by their ID and get the posts that they've authored. Here's what a basic query to do that might look like:

{
  user(id: "1") {
    name
    posts {
      id
      title
    }
  }
}

The query includes a field argument, id, contained within the parentheses after the user field name. To make this all work, we need to modify our schema a bit.

Defining Arguments

First, let's create a :user type and define its relationship to :post while we're at it. We'll create a new module for the account-related types and put it there; in blog_web/schema/account_types.ex:

defmodule BlogWeb.Schema.AccountTypes do
  use Absinthe.Schema.Notation

  @desc "A user of the blog"
  object :user do
    field :id, :id
    field :name, :string
    field :email, :string
    field :posts, list_of(:post)
  end

end

The :posts field points to a list of :post results. (This matches up with what we have on the Ecto side, where Blog.Accounts.User defines a has_many association with Blog.Content.Post.)

We've already defined the :post type, but let's go ahead and add an :author field that points back to our :user type. In blog_web/schema/content_types.ex:

object :post do

  # post fields we defined earlier...

  field :author, :user

end

Now let's add the :user field to our query root object in our schema, defining a mandatory :id argument and using the Resolvers.Accounts.find_user/3 resolver function. We also need to make sure we import the types from BlogWeb.Schema.AccountTypes so that :user is available.

In blog_web/schema.ex:

defmodule BlogWeb.Schema do
  use Absinthe.Schema

  import_types Absinthe.Type.Custom

  # Add this `import_types`:
  import_types BlogWeb.Schema.AccountTypes

  import_types BlogWeb.Schema.ContentTypes

  alias BlogWeb.Resolvers

  query do

    @desc "Get all posts"
    field :posts, list_of(:post) do
      resolve &Resolvers.Content.list_posts/3
    end

    # Add this field:
    @desc "Get a user of the blog"
    field :user, :user do
      arg :id, non_null(:id)
      resolve &Resolvers.Accounts.find_user/3
    end

  end

end

Now lets use the argument in our resolver. In blog_web/resolvers/accounts.ex:

defmodule BlogWeb.Resolvers.Accounts do

  def find_user(_parent, %{id: id}, _resolution) do
    case Blog.Accounts.find_user(id) do
      nil ->
        {:error, "User ID #{id} not found"}
      user ->
        {:ok, user}
    end
  end

end

Our schema marks the :id argument as non_null, so we can be certain we will receive it. If :id is left out of the query, Absinthe will return an informative error to the user, and the resolve function will not be called.

If you have experience writing Phoenix controller actions, you might wonder why we can match incoming arguments with atoms instead of having to use strings.

The answer is simple: you've defined the arguments in the schema using atom identifiers, so Absinthe knows what arguments will be used ahead of time, and will coerce as appropriate---culling any extraneous arguments given to a query. This means that all arguments can be supplied to the resolve functions with atom keys.

Finally you'll see that we can handle the possibility that the query, while valid from GraphQL's perspective, may still ask for a user that does not exist. We've decided to return an error in that case.

There's a valid argument for just returning {:ok, nil} when a record can't be found. Whether the absence of data constitutes an error is a decision you get to make.

Arguments and Non-Root Fields

Let's assume we want to query all posts from a user published within a given time range. First, let's add a new field to our :post object type, :published_at.

The GraphQL specification doesn't define any official date or time types, but it does support custom scalar types (you can read more about them in the related guide, and Absinthe ships with several built-in scalar types. We'll use :naive_datetime (which doesn't include timezone information) here.

Edit blog_web/schema/content_types.ex:

defmodule BlogWeb.Schema.ContentTypes do
  use Absinthe.Schema.Notation

  @desc "A blog post"
  object :post do
    field :id, :id
    field :title, :string
    field :body, :string
    field :author, :user
    # Add this:
    field :published_at, :naive_datetime
  end
end

To make the :naive_datetime type available, add an import_types line to your blog_web/schema.ex:

import_types Absinthe.Type.Custom

For more information about how types are imported, read the guide on the topic.

For now, just remember that import_types should only be used in top-level schema module. (Think of it like a manifest.)

Here's the query we'd like to be able to use, getting the posts for a user on a given date:

{
  user(id: "1") {
    name
    posts(date: "2017-01-01") {
      title
      body
      publishedAt
    }
  }
}

To use the passed date, we need to update our :user object type and make some changes to its :posts field; it needs to support a :date argument and use a custom resolver. In blog_web/schema/account_types.ex:

defmodule BlogWeb.Schema.AccountTypes do
  use Absinthe.Schema.Notation

  alias BlogWeb.Resolvers

  object :user do
    field :id, :id
    field :name, :string
    field :email, :string
    # Add the block here:
    field :posts, list_of(:post) do
      arg :date, :date
      resolve &Resolvers.Content.list_posts/3
    end
  end

end

For the resolver, we've added another function head to Resolvers.Content.list_posts/3. This illustrates how you can use the first argument to a resolver to match the parent object of a field. In this case, that parent object would be a Blog.Accounts.User Ecto schema:

# Add this:
def list_posts(%Blog.Accounts.User{} = author, args, _resolution) do
  {:ok, Blog.Content.list_posts(author, args)}
end
# Before this:
def list_posts(_parent, _args, _resolution) do
  {:ok, Blog.Content.list_posts()}
end

Here we pass on the user and arguments to the domain logic function, Blog.Content.list_posts/3, which will find the posts for the user and date (if it's provided; the :date argument is optional). The resolver, just as when it's used for the top level query :posts, returns the posts in an :ok tuple.

Check out the full implementation of logic for Blog.Content.list_posts/3--and some simple seed data--in the absinthe_tutorial repository.

If you've done everything correctly (and have some data handy), if you start up your server with mix phx.server and head over to http://localhost:4000/api/graphiql, you should be able to play with the query.

It should look something like this:

Next Step

Next up, we look at how to modify our data using mutations.