//! This example demonstrates asynchronous subscriptions with warp and tokio 0.2 use std::{pin::Pin, sync::Arc, time::Duration}; use futures::{Future, FutureExt as _, Stream}; use juniper::{DefaultScalarValue, EmptyMutation, FieldError, RootNode}; use juniper_subscriptions::Coordinator; use juniper_warp::{playground_filter, subscriptions::graphql_subscriptions}; use warp::{http::Response, Filter}; #[derive(Clone)] struct Context {} impl juniper::Context for Context {} #[derive(Clone, Copy, juniper::GraphQLEnum)] enum UserKind { Admin, User, Guest, } struct User { id: i32, kind: UserKind, name: String, } // Field resolvers implementation #[juniper::graphql_object(Context = Context)] impl User { fn id(&self) -> i32 { self.id } fn kind(&self) -> UserKind { self.kind } fn name(&self) -> &str { &self.name } async fn friends(&self) -> Vec { if self.id == 1 { return vec![ User { id: 11, kind: UserKind::User, name: "user11".into(), }, User { id: 12, kind: UserKind::Admin, name: "user12".into(), }, User { id: 13, kind: UserKind::Guest, name: "user13".into(), }, ]; } else if self.id == 2 { return vec![User { id: 21, kind: UserKind::User, name: "user21".into(), }]; } else if self.id == 3 { return vec![ User { id: 31, kind: UserKind::User, name: "user31".into(), }, User { id: 32, kind: UserKind::Guest, name: "user32".into(), }, ]; } else { return vec![]; } } } struct Query; #[juniper::graphql_object(Context = Context)] impl Query { async fn users(id: i32) -> Vec { vec![User { id, kind: UserKind::Admin, name: "User Name".into(), }] } } type UsersStream = Pin> + Send>>; struct Subscription; #[juniper::graphql_subscription(Context = Context)] impl Subscription { async fn users() -> UsersStream { let mut counter = 0; let stream = tokio::time::interval(Duration::from_secs(5)).map(move |_| { counter += 1; if counter == 2 { Err(FieldError::new( "some field error from handler", Value::Scalar(DefaultScalarValue::String( "some additional string".to_string(), )), )) } else { Ok(User { id: counter, kind: UserKind::Admin, name: "stream user".to_string(), }) } }); Box::pin(stream) } } type Schema = RootNode<'static, Query, EmptyMutation, Subscription>; fn schema() -> Schema { Schema::new(Query, EmptyMutation::new(), Subscription) } #[tokio::main] async fn main() { ::std::env::set_var("RUST_LOG", "warp_subscriptions"); env_logger::init(); let log = warp::log("warp_server"); let homepage = warp::path::end().map(|| { Response::builder() .header("content-type", "text/html") .body("

juniper_subscriptions demo

visit graphql playground".to_string()) }); let qm_schema = schema(); let qm_state = warp::any().map(move || Context {}); let qm_graphql_filter = juniper_warp::make_graphql_filter(qm_schema, qm_state.boxed()); let sub_state = warp::any().map(move || Context {}); let coordinator = Arc::new(juniper_subscriptions::Coordinator::new(schema())); log::info!("Listening on 127.0.0.1:8080"); let routes = (warp::path("subscriptions") .and(warp::ws()) .and(sub_state.clone()) .and(warp::any().map(move || Arc::clone(&coordinator))) .map( |ws: warp::ws::Ws, ctx: Context, coordinator: Arc>| { ws.on_upgrade(|websocket| -> Pin + Send>> { graphql_subscriptions(websocket, coordinator, ctx).boxed() }) }, )) .map(|reply| { // TODO#584: remove this workaround warp::reply::with_header(reply, "Sec-WebSocket-Protocol", "graphql-ws") }) .or(warp::post() .and(warp::path("graphql")) .and(qm_graphql_filter)) .or(warp::get() .and(warp::path("playground")) .and(playground_filter("/graphql", Some("/subscriptions")))) .or(homepage) .with(log); warp::serve(routes).run(([127, 0, 0, 1], 8080)).await; }