Scalars
Scalars are the primitive types at the leaves of a GraphQL query: numbers, strings, and booleans. You can create custom scalars to other primitive values, but this often requires coordination with the client library intended to consume the API you're building.
Since any value going over the wire is eventually transformed into JSON, you're also limited in the data types you can use. Typically, you represent your custom scalars as strings.
In Juniper, you use the graphql_scalar!
macro to create a custom scalar. In
this example, we're representing a user ID as a string wrapped in a custom type:
use juniper::Value; struct UserID(String); juniper::graphql_scalar!(UserID where Scalar = <S> { description: "An opaque identifier, represented as a string" resolve(&self) -> Value { Value::scalar(self.0.clone()) } from_input_value(v: &InputValue) -> Option<UserID> { // If there's a parse error here, simply return None. Juniper will // present an error to the client. v.as_scalar_value::<String>().map(|s| UserID(s.to_owned())) } from_str<'a>(value: ScalarToken<'a>) -> juniper::ParseScalarResult<'a, S> { <String as juniper::ParseScalarValue<S>>::from_str(value) } }); #[derive(juniper::GraphQLObject)] struct User { id: UserID, username: String, } # fn main() {}
Built-in scalars
Juniper has built-in support for:
i32
asInt
f64
asFloat
String
and&str
asString
bool
asBoolean
juniper::ID
asID
. This type is defined in the spec as a type that is serialized as a string but can be parsed from both a string and an integer.
Non-standard scalars
Juniper has built-in support for UUIDs from the uuid crate. This support is enabled by default, but can be disabled if you want to reduce the number of dependencies in your application.