• Stars
    star
    313
  • Rank 133,714 (Top 3 %)
  • Language
    Elm
  • License
    BSD 3-Clause "New...
  • Created about 8 years ago
  • Updated almost 4 years ago

Reviews

There are no reviews yet. Be the first to send feedback to the community and the maintainers!

Repository Details

A GraphQL library for Elm

elm-graphql logo

jamesmacaulay/elm-graphql

Travis-CI build status

A GraphQL library for Elm, written entirely in Elm.

The goal of this package is to provide a really good interface for working directly with GraphQL queries and schemas in Elm. Right now the main offering of the package is an interface for building up nested queries and mutations in a way that also builds up a decoder capable of decoding successful responses to the request. The package also provides a module for sending these requests to a GraphQL server over HTTP and decoding the responses accordingly.

The docs can be found here.

And here's an end-to-end example that builds a query, sends it to a server, and decodes the response.

Building requests

Building up a GraphQL query with this package feels a lot like building a JSON decoder, especially if you are familiar with the elm-decode-pipeline package. First you define type aliases for each of the nested record types you want to construct out of the response:

import GraphQL.Request.Builder exposing (..)
import GraphQL.Request.Builder.Arg as Arg
import GraphQL.Request.Builder.Variable as Var

type alias Photo =
    { url : String
    , caption : String
    }

type alias User =
    { name : String
    , photos : List Photo }

Then you build a query document:

userQuery : Document Query User { vars | userID : String }
userQuery =
    let
        userIDVar =
            Var.required "userID" .userID Var.id

        photo =
            object Photo
                |> with (field "url" [] string)
                |> with (field "caption" [] string)

        user =
            object User
                |> with (field "name" [] string)
                |> with (field "photos" [] (list photo))
        
        queryRoot =
            extract
                (field "user"
                    [ ( "id", Arg.variable userIDVar ) ]
                    user
                )
    in
        queryDocument queryRoot

The Document type can represent both query and mutation documents. It lets you do two important things:

  • generate GraphQL request documents to send to the server, and
  • decode JSON responses from the server.

Here's what the above Document looks like when you encode it to a string to be sent to the server:

query ($userID: ID!) {
  user(id: $userID) {
    name
    photos {
      url
      caption
    }
  }
}

To supply values for the variables used in the Document, you provide an Elm value that your variables can extract values from according to the getter functions supplied when you define the variables. In this example, the "userID" variable was defined as a string that's extracted from the userID field of some Elm record, so the following code is a valid way to create a Request with the required variable values supplied:

userQueryRequest : Request Query User
userQueryRequest =
    userQuery
        |> request { userID = "123" }

Assuming you've built a query that is valid for the server's schema, sending it to the server will result in a JSON response that can be decoded with a JSON decoder that is built up automatically along with the structure of the query. Here's what a JSON response for userQuery might look like:

{
  "data": {
    "user": {
      "name": "Lola",
      "photos": [
        {
          "url": "http://cdn.catphotos.com/lola.jpg",
          "caption": "Lola curling up on the chair"
        }
      ]
    }
  }
}

When it is decoded with the help of the decoder contained in userQuery, it becomes this:

{ name = "Lola"
, photos =
    [ { url = "http://cdn.catphotos.com/lola.jpg"
      , caption = "Lola curling up on the chair"
      }
    ]
}

Mutations are built just like queries, except that you wrap them up in a call to mutationDocument instead of a call to queryDocument. Here's an example of a mutation that logs in a user and extracts an auth token from the response:

type alias LoginVars =
    { username : String
    , password : String
    }


loginMutation : Document Mutation String LoginVars
loginMutation =
    let
        usernameVar =
            Var.required "username" .username Var.string

        passwordVar =
            Var.required "password" .password Var.string
    in
        mutationDocument <|
            extract
                (field "login"
                    [ ( "username", Arg.variable usernameVar )
                    , ( "password", Arg.variable passwordVar )
                    ]
                    (extract (field "token" [] string))
                )

Future plans for this package

There are a lot of things that this package can't do right now, but might do in the future. What gets done depends on how the package ends up being used, and how much demand there is for each feature. Here are some likely possibilities:

  • support for subscriptions
  • generating code from a GraphQL schemas and queries
  • providing functions to validate a query against a target schema
  • leveraging Relay-compliant schemas to cache response data and transform queries so that the client only asks the server for what it doesn't have already
  • providing an interface to implement a GraphQL schema that you can run queries against

Getting and giving help

If you're having trouble figuring out how to do something with this package, check out the #graphql channel on the Elm Slack — there are usually people there who can help. And if you can be one of those people who help other people, then thank you!

Running the tests

Install the elm-test npm package globally:

npm install -g elm-test

Make sure you have the latest version of elm-test, otherwise it may not be able to find the the test files in the right way. You can update an older version with npm update -g elm-test.

Now run elm-test in the root folder of this project:

elm-test

More Repositories

1

zelkova

Elm-style FRP for Clojure and ClojureScript
Clojure
403
star
2

react-bacon

A little module for using React with Bacon.js
JavaScript
118
star
3

cljs-promises

A ClojureScript library for using JS promises with core.async
Clojure
90
star
4

elm-composition-trees

A story about composing trees of computation
Elm
41
star
5

http_spy

Watch HTTP requests as they happen
Elixir
18
star
6

shopify-clj

A Clojure library for interacting with the Shopify API
Clojure
14
star
7

poker-hands

A little library to score poker hands in Clojure
Clojure
12
star
8

elm-json-bidirectional

Helps you construct two-way JSON encoder-decoders in Elm.
Elm
10
star
9

graphqlbin

Mock GraphQL schema server
TypeScript
9
star
10

ruler

Elixir
8
star
11

kozu

Promise-agnostic functional composition in JavaScript.
JavaScript
7
star
12

sprockets_rails3_backport

Backport of Rails 3.1.x Sprockets integration to Rails 3.0.x
Ruby
6
star
13

backfire

backpack => campfire bridge
Ruby
6
star
14

redoc-autogen

Automatically build docs for ReasonML packages
Dockerfile
5
star
15

zelkova-todomvc

TodoMVC with Zelkova and Reagent, translated from evancz/elm-todomvc
Clojure
4
star
16

elm-gif-lab

A little sandbox for making animated GIFs with Elm
Elm
4
star
17

glitch-elm-example

The glitch.com example project ported to Elm!
Elm
3
star
18

zelkova-quil-flyer

A little demo using Zelkova and Quil
Clojure
3
star
19

zelkova-om-searcher

A little demo using Zelkova and Om
Clojure
3
star
20

rule-of-thirds

evaluates objects from peekaboom data for their adherence to the rule of thirds
Python
3
star
21

elm-protocol-records

A library of Elm type protocols, implemented as extensible records
Elm
3
star
22

office-party

iTunes web UI built with Rails & Shoes
Ruby
2
star
23

old-office-party

Awesome jukeboxing for shared spaces.
Ruby
2
star
24

elm-secd

An implementation of Peter Landin's SECD machine in Elm
Elm
2
star
25

batman_workshop

Ruby
1
star
26

mephisto_export

Plugin which adds a couple of rake tasks for managing exports of Mephisto data.
Ruby
1
star
27

bin

~/bin
Ruby
1
star
28

async-tools

Tools for core.async
Clojure
1
star
29

quantified

Small library for working with units of measurement
Ruby
1
star
30

rust-secd

An implementation of Peter Landin's SECD machine in Rust
Rust
1
star
31

gh-pull-notifier

CoffeeScript
1
star
32

comp4106

AI programming
Ruby
1
star
33

tagflow

Instagram hashtag slideshows
JavaScript
1
star
34

channel

An incomplete implementation of CSP channels in Elixir
Elixir
1
star