Compile-time GraphQL client for Elixir. Parses and validates queries during compilation, generates typed Ecto embedded schemas for responses and variables, and executes queries at runtime via Req.
- Compile-time validation — Syntax errors, schema mismatches, and deprecated usage caught at
mix compile, with errors pointing to file:line:column in your Elixir source - Minimal code generation — Only generates modules for types actually referenced in your queries, not the entire schema
- Zero runtime parsing — All GraphQL parsing happens at compile time
- Typed responses — JSON responses are cast into typed Ecto embedded schemas, accessible via dot notation,
[:field], andget_in - Typed variables — Input validation via Ecto changesets with generated
params()type - Union / Interface support — Automatic type dispatch, no extra fields needed
- Req integration — Full access to Req's middleware/plugin system, test with
Req.Testdirectly - Configurable JSON library — Defaults to Elixir 1.18+ built-in
JSON, falls back toJason, or set your own via:json_library - Auto-generated docs —
defgqlfunctions include@docwith variables, types, and generated module listing
Add grephql to your dependencies in mix.exs:
def deps do
[
{:grephql, "~> 0.9.2"}
]
endmix grephql.download_schema \
--endpoint https://api.example.com/graphql \
--output priv/schemas/schema.json \
--header "Authorization: Bearer token123"defmodule MyApp.GitHub do
use Grephql,
otp_app: :my_app,
source: "priv/schemas/github.json",
endpoint: "https://api.github.com/graphql"
defgql :get_user, ~GQL"""
query GetUser($login: String!) {
user(login: $login) {
name
bio
}
}
"""
defgql :get_viewer, ~GQL"""
query {
viewer {
login
email
}
}
"""
end# With variables — validates input before sending
case MyApp.GitHub.get_user(%{login: "octocat"}) do
{:ok, result} ->
result.data.user.name #=> "The Octocat"
{:error, %Ecto.Changeset{} = changeset} ->
# Variable validation failed
changeset.errors
{:error, %Req.Response{} = response} ->
# HTTP error
response.status
end
# Without variables
{:ok, result} = MyApp.GitHub.get_viewer()
result.data.viewer.loginDefines a public (or private) GraphQL query function. At compile time: parses, validates, generates typed modules, and defines a callable function.
# Public — generates def get_user/2
defgql :get_user, ~GQL"""
query GetUser($id: ID!) {
user(id: $id) { name }
}
"""
# Private — generates defp get_user/2
defgqlp :get_user, ~GQL"""
query GetUser($id: ID!) {
user(id: $id) { name }
}
"""defgql functions automatically include @doc with operation info, variable table, and all generated module names.
Defines a reusable named fragment. The fragment name comes from the GraphQL definition itself, so deffragment only takes the fragment string. Fragments are validated at compile time and automatically appended to queries that reference them via ...FragmentName.
deffragment ~GQL"""
fragment UserFields on User {
name
email
createdAt
}
"""
defgql :get_user, ~GQL"""
query GetUser($id: ID!) {
user(id: $id) {
...UserFields
}
}
"""The fragment generates a typed module at Client.Fragments.UserFields.
defgql only sees fragments defined before it in the module. If the same fragment name is defined multiple times before a query, the latest definition overrides earlier ones for that query.
Configuration is resolved in order (later wins): compile-time defaults -> runtime config -> per-call opts.
use Grephql,
otp_app: :my_app,
source: "priv/schemas/github.json",
endpoint: "https://api.github.com/graphql",
req_options: [receive_timeout: 30_000],
scalars: %{"DateTime" => Grephql.Types.DateTime}# config/runtime.exs
config :my_app, MyApp.GitHub,
endpoint: "https://api.github.com/graphql",
req_options: [auth: {:bearer, System.fetch_env!("GITHUB_TOKEN")}]MyApp.GitHub.get_user(%{login: "octocat"},
endpoint: "https://other.api.com/graphql",
req_options: [receive_timeout: 60_000]
)The ~GQL sigil marks GraphQL strings for automatic formatting by mix format. Plain strings still work with defgql — ~GQL is optional.
Add the formatter plugin to your .formatter.exs:
[
plugins: [Grephql.Formatter],
# ...
]Or via dependency import:
[
import_deps: [:grephql],
# ...
]# Before
defgql :get_user, ~GQL"query GetUser($id: ID!) { user(id: $id) { name email posts { title } } }"
# After mix format
defgql :get_user, ~GQL"""
query GetUser($id: ID!) {
user(id: $id) {
name
email
posts {
title
}
}
}
"""Map GraphQL custom scalars to Ecto types via the :scalars option:
use Grephql,
otp_app: :my_app,
source: "schema.json",
scalars: %{
"DateTime" => Grephql.Types.DateTime,
"JSON" => :map
}Grephql.Types.DateTime is included for ISO 8601 DateTime strings. For other custom scalars, provide any module implementing the Ecto.Type behaviour.
GraphQL enums are automatically converted to snake_cased atoms:
# GraphQL: enum Role { ADMIN, READ_ONLY }
# Elixir: :admin, :read_onlyLoading is case-insensitive — "ADMIN", "admin", and "Admin" all resolve to :admin.
Union and interface types are resolved at decode time using the __typename field:
defgql :search, ~GQL"""
query Search($q: String!) {
search(query: $q) {
... on User {
name
}
... on Repository {
fullName
}
}
}
"""{:ok, result} = MyApp.GitHub.search(%{q: "elixir"})
Enum.each(result.data.search, fn
%MyApp.GitHub.Search.Result.Search.User{} = user -> IO.puts(user.name)
%MyApp.GitHub.Search.Result.Search.Repository{} = repo -> IO.puts(repo.full_name)
end)
# __typename is also available as a snake_cased atom
hd(result.data.search).__typename #=> :userGraphQL aliases affect both the struct field name and the module path:
defgql :get_account, ~GQL"""
query {
account: user(login: "alice") {
displayName: name
}
}
"""
{:ok, result} = MyApp.GitHub.get_account()
result.data.account.display_name # alias becomes the field nameEach defgql generates typed Ecto embedded schema modules at compile time. Given defgql :get_user inside MyApp.GitHub:
| Type | Pattern | Example |
|---|---|---|
| Result | Client.FnName.Result.Field... |
MyApp.GitHub.GetUser.Result.User |
| Nested field | ...Result.Field.NestedField |
MyApp.GitHub.GetUser.Result.User.Posts |
| Variables | Client.FnName.Variables |
MyApp.GitHub.GetUser.Variables |
| Input types | Client.Inputs.TypeName |
MyApp.GitHub.Inputs.CreateUserInput |
| Fragment | Client.Fragments.Name |
MyApp.GitHub.Fragments.UserFields |
| Union variant | ...Result.Field.TypeName |
MyApp.GitHub.Search.Result.Search.User |
- Function name is CamelCased:
:get_user->GetUser - Struct field names are snake_cased:
userName->:user_name - Field aliases override both field name and module path:
author: user { ... }-> field:author, module...Result.Author - Input types are shared across queries under
Client.Inputs.* - Variables are per-query under
Client.FnName.Variables - Fragment modules live under
Client.Fragments.*
defmodule MyApp.GitHub do
use Grephql, otp_app: :my_app, source: "schema.json"
deffragment ~GQL"""
fragment PostFields on Post {
title
body
}
"""
defgql :get_user, ~GQL"""
query GetUser($id: ID!) {
author: user(id: $id) {
name
posts {
...PostFields
}
}
}
"""
end
# Generated modules:
# MyApp.GitHub.Fragments.PostFields — %{title: String.t(), body: String.t()}
# MyApp.GitHub.GetUser.Result — %{author: Author.t()}
# MyApp.GitHub.GetUser.Result.Author — %{name: String.t(), posts: [Posts.t()]}
# MyApp.GitHub.GetUser.Result.Author.Posts — %{title: String.t(), body: String.t()}
# MyApp.GitHub.GetUser.Variables — %{id: String.t()}All generated schemas implement the Access behaviour, so you can use bracket syntax and get_in/2 alongside dot notation:
{:ok, result} = MyApp.GitHub.get_user(%{login: "octocat"})
# Dot notation
result.data.user.name
# Bracket syntax
result.data.user[:name]
# Dynamic / nested access
get_in(result.data, [:user, :name])This is powered by TypedStructor.Plugins.Access, which is registered in every generated typed_embedded_schema block.
Use Req.Test to stub HTTP responses without any network calls:
# config/test.exs
config :my_app, MyApp.GitHub,
req_options: [plug: {Req.Test, MyApp.GitHub}]test "get_user returns user data" do
Req.Test.stub(MyApp.GitHub, fn conn ->
Req.Test.json(conn, %{
"data" => %{"user" => %{"name" => "Alice", "bio" => "Elixirist"}}
})
end)
assert {:ok, result} = MyApp.GitHub.get_user(%{login: "alice"})
assert result.data.user.name == "Alice"
endDownloads a GraphQL schema via introspection and saves it as JSON.
mix grephql.download_schema --endpoint URL --output PATH [--header "Key: Value"]| Option | Required | Description |
|---|---|---|
--endpoint / -e |
yes | GraphQL endpoint URL |
--output / -o |
yes | File path to save the schema JSON |
--header / -h |
no | HTTP header in "Key: Value" format (repeatable) |
| Option | Required | Description |
|---|---|---|
:otp_app |
yes | OTP application for runtime config lookup |
:source |
yes | Path to introspection JSON (relative to caller file) or inline JSON string |
:endpoint |
no | Default GraphQL endpoint URL |
:req_options |
no | Default Req options (keyword list) |
:scalars |
no | Map of GraphQL scalar name to Ecto type (default: %{}) |
Defaults to Elixir 1.18+ built-in JSON, falls back to Jason. To override:
config :grephql, :json_library, JasonAny module implementing encode!/1 and decode/1 works.
- Elixir ~> 1.15
- Erlang/OTP 24+
See LICENSE for details.