2018-12-23 14:41:50 -06:00
|
|
|
# Objects and generics
|
|
|
|
|
|
|
|
Yet another point where GraphQL and Rust differs is in how generics work. In
|
|
|
|
Rust, almost any type could be generic - that is, take type parameters. In
|
|
|
|
GraphQL, there are only two generic types: lists and non-nullables.
|
|
|
|
|
|
|
|
This poses a restriction on what you can expose in GraphQL from Rust: no generic
|
|
|
|
structs can be exposed - all type parameters must be bound. For example, you can
|
|
|
|
not make e.g. `Result<T, E>` into a GraphQL type, but you _can_ make e.g.
|
|
|
|
`Result<User, String>` into a GraphQL type.
|
|
|
|
|
|
|
|
Let's make a slightly more compact but generic implementation of [the last
|
|
|
|
chapter](non_struct_objects.md):
|
|
|
|
|
|
|
|
```rust
|
2019-01-11 10:16:21 -06:00
|
|
|
# #[derive(juniper::GraphQLObject)] struct User { name: String }
|
|
|
|
# #[derive(juniper::GraphQLObject)] struct ForumPost { title: String }
|
2018-12-23 14:41:50 -06:00
|
|
|
|
2019-01-11 10:16:21 -06:00
|
|
|
#[derive(juniper::GraphQLObject)]
|
2018-12-23 14:41:50 -06:00
|
|
|
struct ValidationError {
|
|
|
|
field: String,
|
|
|
|
message: String,
|
|
|
|
}
|
|
|
|
|
|
|
|
# #[allow(dead_code)]
|
|
|
|
struct MutationResult<T>(Result<T, Vec<ValidationError>>);
|
|
|
|
|
2019-11-15 19:43:39 -06:00
|
|
|
#[juniper::graphql_object(
|
2019-05-07 03:56:06 -05:00
|
|
|
name = "UserResult",
|
|
|
|
)]
|
|
|
|
impl MutationResult<User> {
|
|
|
|
fn user(&self) -> Option<&User> {
|
2018-12-23 14:41:50 -06:00
|
|
|
self.0.as_ref().ok()
|
|
|
|
}
|
|
|
|
|
2019-05-07 03:56:06 -05:00
|
|
|
fn error(&self) -> Option<&Vec<ValidationError>> {
|
2018-12-23 14:41:50 -06:00
|
|
|
self.0.as_ref().err()
|
|
|
|
}
|
2019-05-07 03:56:06 -05:00
|
|
|
}
|
2018-12-23 14:41:50 -06:00
|
|
|
|
2019-11-15 19:43:39 -06:00
|
|
|
#[juniper::graphql_object(
|
2019-05-07 03:56:06 -05:00
|
|
|
name = "ForumPostResult",
|
|
|
|
)]
|
|
|
|
impl MutationResult<ForumPost> {
|
|
|
|
fn forum_post(&self) -> Option<&ForumPost> {
|
2018-12-23 14:41:50 -06:00
|
|
|
self.0.as_ref().ok()
|
|
|
|
}
|
|
|
|
|
2019-05-07 03:56:06 -05:00
|
|
|
fn error(&self) -> Option<&Vec<ValidationError>> {
|
2018-12-23 14:41:50 -06:00
|
|
|
self.0.as_ref().err()
|
|
|
|
}
|
2019-05-07 03:56:06 -05:00
|
|
|
}
|
2018-12-23 14:41:50 -06:00
|
|
|
|
|
|
|
# fn main() {}
|
|
|
|
```
|
|
|
|
|
|
|
|
Here, we've made a wrapper around `Result` and exposed some concrete
|
|
|
|
instantiations of `Result<T, E>` as distinct GraphQL objects. The reason we
|
|
|
|
needed the wrapper is of Rust's rules for when you can derive a trait - in this
|
|
|
|
case, both `Result` and Juniper's internal GraphQL trait are from third-party
|
|
|
|
sources.
|
|
|
|
|
|
|
|
Because we're using generics, we also need to specify a name for our
|
|
|
|
instantiated types. Even if Juniper _could_ figure out the name,
|
|
|
|
`MutationResult<User>` wouldn't be a valid GraphQL type name.
|