GraphQL server library for Rust
Find a file
Michael Macias b78aef715d Make rustc-serialize optional
This extracts rustc-serialize JSON serialization into its own module,
decoupling all core structures from its use.

rustc-serialize can now be completely disabled, removing it as a
dependency if serde (or none) is used instead. It is, however, still the
default serializer. In Cargo, the `default-features` field must be set
to `false` to disable it, e.g.,

    [dependencies.juniper]
    version = "0.6"
    default-features = false
    features = ["serde"]

The Iron handlers still require rustc-serialize.

The default values for attributes changed from being formatted as JSON
to how it is defined in [the reference implementation][printer.js].

[printer.js]: https://github.com/graphql/graphql-js/blob/v0.8.2/src/language/printer.js
2017-01-06 21:34:44 -06:00
_build Get travis to upload documentation to gh-pages branch 2016-09-11 20:31:16 +02:00
benches Use bencher crate to run benchmarks on stable rust 2016-12-26 12:16:19 +01:00
examples Move context type parameter to associated type of GraphQLType 2016-12-22 16:25:44 +01:00
src Make rustc-serialize optional 2017-01-06 21:34:44 -06:00
.gitignore Initial import 2016-09-11 18:41:29 +02:00
.travis.yml Remove travis-cargo 2016-10-20 20:44:28 +02:00
Cargo.toml Make rustc-serialize optional 2017-01-06 21:34:44 -06:00
CHANGELOG.md Prepare for 0.6.1 release 2017-01-06 13:15:16 +01:00
LICENSE Initial import 2016-09-11 18:41:29 +02:00
README.md Prepare for 0.6.1 release 2017-01-06 13:15:16 +01:00

Juniper

GraphQL server library for Rust

Build Status Crates.io


GraphQL is a data query language developed by Facebook intended to serve mobile and web application frontends. Juniper makes it possible to write GraphQL servers in Rust that are type-safe and blazingly fast.

Juniper does not include a web server - instead it provides building blocks to make integration with existing servers straightforward. It optionally provides a pre-built integration for the Iron framework.

Installation

Add Juniper to your Cargo.toml:

[dependencies]
juniper = "0.6.1"

If you want the Iron integration enabled, you need to enable the iron-handlers feature flag:

[dependencies]
juniper = { version = "0.6.1", features = ["iron-handlers"] }

Building schemas

GraphQL turns the REST paradigm as it's usually implemented on its head: instead of providing a fixed structure of all types and relations in the system, GraphQL defines a schema which your users can query. The schema defines all types, fields, and relations available, while the query defines which fields and relations a user is interested in.

Juniper expects you to already have the types you want to expose in GraphQL as Rust data types. Other than that, it doesn't make any assumptions whether they are stored in a database or just in memory. Exposing a type is a matter of implementing the GraphQLType for your type. To make things a bit easier, Juniper comes with a set of macros that help you do this, based on what kind of type you want to expose. Let's look at how one could expose parts of the Star Wars Schema:

#[macro_use] extern crate juniper;

use juniper::FieldResult;

enum Episode {
    NewHope,
    Empire,
    Jedi,
}

struct Human {
    id: String,
    name: String,
    appears_in: Vec<Episode>,
    home_planet: String,
}

graphql_enum!(Episode {
    Episode::NewHope => "NEW_HOPE",
    Episode::Empire => "EMPIRE",
    Episode::Jedi => "JEDI",
});

graphql_object!(Human: () |&self| {
    description: "A humanoid creature in the Star Wars universe"

    // Field resolver methods look almost like ordinary methods. The macro picks
    // up arguments and return types for the introspection schema, and verifies
    // it during compilation.
    field id() -> FieldResult<&String> {
        Ok(&self.id)
    }

    field name() -> FieldResult<&String> {
        Ok(&self.name)
    }

    field appears_in() -> FieldResult<&Vec<Episode>> {
        Ok(&self.appears_in)
    }

    field home_planet() -> FieldResult<&String> {
        Ok(&self.home_planet)
    }
});

You can find the full example in src/tests/schema.rs, including polymorphism with traits and interfaces. For an example of the Iron integration, see the examples folder.

Features

Juniper supports the full GraphQL query language according to the specification, including the introspective schema and all validations. It does not, however, support the schema language.

As an exception to other GraphQL libraries for other languages, Juniper builds non-null types by default. A field of type Vec<Episode> will be converted into [Episode!]!. The corresponding Rust type for e.g. [Episode] would be Option<Vec<Option<Episode>>>.

API Stability

Juniper has not reached 1.0 yet, thus some API instability should be expected.

1.0 Roadmap

Version 0.6.1 will probably be re-released as 1.0 to indicate API stability.

The road to 1.0 focuses on two aspects: making sure the API hasn't got any obvious dead-ends with respect to probable future features, and improving test coverage for general execution. There are some chores that need to be completed as well.

  • Extensive execution testing
    • Sending input objects and partial input objects in variables
    • Sending enums in variables
    • General input value type checking and validation
  • Improve helper macros
    • graphql_union! helper completely missing
    • graphql_input_object! helper completely missing
    • Add support for deprecating things
    • Custom enum values and descriptions
    • Improved syntax for fields that can't fail resolution - make FieldResult<T> optional maybe?
  • Investigate asynchronous execution - implementing it is not necessary, but at least look at what API changes will be needed for us to hook into Tokio, for example.
  • Larger examples to illustrate things like database access