use async_graphql::*; use futures::{SinkExt, Stream, StreamExt}; use std::sync::Arc; #[async_std::test] pub async fn test_subscription() { struct QueryRoot; #[SimpleObject] struct Event { a: i32, b: i32, } #[Object] impl QueryRoot {} struct SubscriptionRoot; #[Subscription] impl SubscriptionRoot { async fn values(&self, start: i32, end: i32) -> impl Stream { futures::stream::iter(start..end) } async fn events(&self, start: i32, end: i32) -> impl Stream { futures::stream::iter((start..end).map(|n| Event { a: n, b: n * 10 })) } } let schema = Schema::new(QueryRoot, EmptyMutation, SubscriptionRoot); { let mut stream = schema .create_subscription_stream( "subscription { values(start: 10, end: 20) }", None, Default::default(), None, ) .await .unwrap(); for i in 10..20 { assert_eq!( Some(serde_json::json!({ "values": i })), stream.next().await ); } assert!(stream.next().await.is_none()); } { let mut stream = schema .create_subscription_stream( "subscription { events(start: 10, end: 20) { a b } }", None, Default::default(), None, ) .await .unwrap(); for i in 10..20 { assert_eq!( Some(serde_json::json!({ "events": {"a": i, "b": i * 10} })), stream.next().await ); } assert!(stream.next().await.is_none()); } } #[async_std::test] pub async fn test_simple_broker() { struct QueryRoot; #[SimpleObject] #[derive(Clone)] struct Event1 { value: i32, } #[SimpleObject] #[derive(Clone)] struct Event2 { value: i32, } #[Object] impl QueryRoot {} struct SubscriptionRoot; #[Subscription] impl SubscriptionRoot { async fn events1(&self) -> impl Stream { SimpleBroker::::subscribe() } async fn events2(&self) -> impl Stream { SimpleBroker::::subscribe() } } let schema = Schema::new(QueryRoot, EmptyMutation, SubscriptionRoot); let mut stream1 = schema .create_subscription_stream( "subscription { events1 { value } }", None, Default::default(), None, ) .await .unwrap(); let mut stream2 = schema .create_subscription_stream( "subscription { events2 { value } }", None, Default::default(), None, ) .await .unwrap(); SimpleBroker::publish(Event1 { value: 10 }); SimpleBroker::publish(Event2 { value: 88 }); SimpleBroker::publish(Event1 { value: 15 }); SimpleBroker::publish(Event2 { value: 99 }); assert_eq!( stream1.next().await, Some(serde_json::json!({ "events1": {"value": 10} })) ); assert_eq!( stream1.next().await, Some(serde_json::json!({ "events1": {"value": 15} })) ); assert_eq!( stream2.next().await, Some(serde_json::json!({ "events2": {"value": 88} })) ); assert_eq!( stream2.next().await, Some(serde_json::json!({ "events2": {"value": 99} })) ); } #[async_std::test] pub async fn test_subscription_with_ctx_data() { struct QueryRoot; #[Object] impl QueryRoot {} struct MyObject; #[Object] impl MyObject { async fn value(&self, ctx: &Context<'_>) -> i32 { *ctx.data::() } } struct SubscriptionRoot; #[Subscription] impl SubscriptionRoot { async fn values(&self, ctx: &Context<'_>) -> impl Stream { let value = *ctx.data::(); futures::stream::once(async move { value }) } async fn objects(&self) -> impl Stream { futures::stream::once(async move { MyObject }) } } let schema = Schema::new(QueryRoot, EmptyMutation, SubscriptionRoot); { let mut stream = schema .create_subscription_stream( "subscription { values objects { value } }", None, Default::default(), Some(Arc::new({ let mut data = Data::default(); data.insert(100i32); data })), ) .await .unwrap(); assert_eq!( Some(serde_json::json!({ "values": 100 })), stream.next().await ); assert_eq!( Some(serde_json::json!({ "objects": { "value": 100 } })), stream.next().await ); assert!(stream.next().await.is_none()); } } #[async_std::test] pub async fn test_subscription_with_token() { struct QueryRoot; #[Object] impl QueryRoot {} struct SubscriptionRoot; struct Token(String); #[Subscription] impl SubscriptionRoot { async fn values(&self, ctx: &Context<'_>) -> FieldResult> { if ctx.data::().0 != "123456" { return Err("forbidden".into()); } Ok(futures::stream::once(async move { 100 })) } } let schema = Schema::new(QueryRoot, EmptyMutation, SubscriptionRoot); { let mut stream = schema .create_subscription_stream( "subscription { values }", None, Default::default(), Some(Arc::new({ let mut data = Data::default(); data.insert(Token("123456".to_string())); data })), ) .await .unwrap(); assert_eq!( Some(serde_json::json!({ "values": 100 })), stream.next().await ); assert!(stream.next().await.is_none()); } { assert!(schema .create_subscription_stream( "subscription { values }", None, Default::default(), Some(Arc::new({ let mut data = Data::default(); data.insert(Token("654321".to_string())); data })), ) .await .is_err()); } } #[async_std::test] pub async fn test_subscription_ws_transport() { struct QueryRoot; #[Object] impl QueryRoot {} struct SubscriptionRoot; #[Subscription] impl SubscriptionRoot { async fn values(&self) -> impl Stream { futures::stream::iter(0..10) } } let schema = Schema::new(QueryRoot, EmptyMutation, SubscriptionRoot); let (mut sink, mut stream) = schema.subscription_connection(WebSocketTransport::default()); sink.send( serde_json::to_vec(&serde_json::json!({ "type": "connection_init", "payload": { "token": "123456" } })) .unwrap() .into(), ) .await .unwrap(); assert_eq!( Some(serde_json::json!({ "type": "connection_ack", })), serde_json::from_slice(&stream.next().await.unwrap()).unwrap() ); sink.send( serde_json::to_vec(&serde_json::json!({ "type": "start", "id": "1", "payload": { "query": "subscription { values }" }, })) .unwrap() .into(), ) .await .unwrap(); for i in 0..10 { assert_eq!( Some(serde_json::json!({ "type": "data", "id": "1", "payload": { "data": { "values": i } }, })), serde_json::from_slice(&stream.next().await.unwrap()).unwrap() ); } } #[async_std::test] pub async fn test_subscription_ws_transport_with_token() { struct Token(String); struct QueryRoot; #[Object] impl QueryRoot {} struct SubscriptionRoot; #[Subscription] impl SubscriptionRoot { async fn values(&self, ctx: &Context<'_>) -> FieldResult> { if ctx.data::().0 != "123456" { return Err("forbidden".into()); } Ok(futures::stream::iter(0..10)) } } let schema = Schema::new(QueryRoot, EmptyMutation, SubscriptionRoot); let (mut sink, mut stream) = schema.subscription_connection(WebSocketTransport::new(|value| { #[derive(serde_derive::Deserialize)] struct Payload { token: String, } let payload: Payload = serde_json::from_value(value).unwrap(); let mut data = Data::default(); data.insert(Token(payload.token)); Ok(data) })); sink.send( serde_json::to_vec(&serde_json::json!({ "type": "connection_init", "payload": { "token": "123456" } })) .unwrap() .into(), ) .await .unwrap(); assert_eq!( Some(serde_json::json!({ "type": "connection_ack", })), serde_json::from_slice(&stream.next().await.unwrap()).unwrap() ); sink.send( serde_json::to_vec(&serde_json::json!({ "type": "start", "id": "1", "payload": { "query": "subscription { values }" }, })) .unwrap() .into(), ) .await .unwrap(); for i in 0..10 { assert_eq!( Some(serde_json::json!({ "type": "data", "id": "1", "payload": { "data": { "values": i } }, })), serde_json::from_slice(&stream.next().await.unwrap()).unwrap() ); } }