mod test_utils; use serde_json::json; use smol::{Task, Timer}; use std::time::Duration; use tide::Request; use async_graphql::*; type Result = std::result::Result>; #[test] fn quickstart() -> Result<()> { smol::run(async { let listen_addr = test_utils::find_listen_addr().await; let server = Task::>::spawn(async move { struct QueryRoot; #[Object] impl QueryRoot { #[field(desc = "Returns the sum of a and b")] async fn add(&self, a: i32, b: i32) -> i32 { a + b } } let mut app = tide::new(); app.at("/").post(|req: Request<()>| async move { let schema = Schema::build(QueryRoot, EmptyMutation, EmptySubscription).finish(); async_graphql_tide::graphql(req, schema, |query_builder| query_builder).await }); app.listen(&listen_addr).await?; Ok(()) }); let client = Task::>::spawn(async move { Timer::after(Duration::from_millis(300)).await; let resp = reqwest::Client::new() .post(format!("http://{}", listen_addr).as_str()) .body(r#"{"query":"{ add(a: 10, b: 20) }"}"#) .header(reqwest::header::CONTENT_TYPE, "application/json") .send() .await?; assert_eq!(resp.status(), reqwest::StatusCode::OK); let string = resp.text().await?; println!("{}", string); assert_eq!(string, json!({"data": {"add": 30}}).to_string()); Ok(()) }); client.await?; server.cancel().await; Ok(()) }) } #[test] fn hello() -> Result<()> { smol::run(async { let listen_addr = test_utils::find_listen_addr().await; let server = Task::>::spawn(async move { struct Hello(String); struct QueryRoot; #[Object] impl QueryRoot { #[field(desc = "Returns hello")] async fn hello<'a>(&self, ctx: &'a Context<'_>) -> String { let name = ctx.data_opt::().map(|hello| hello.0.as_str()); format!("Hello, {}!", name.unwrap_or("world")) } } struct AppState { schema: Schema, } let schema = Schema::build(QueryRoot, EmptyMutation, EmptySubscription).finish(); let app_state = AppState { schema }; let mut app = tide::with_state(app_state); app.at("/").post(|req: Request| async move { let schema = req.state().schema.clone(); let name = &req .header(&"name".parse().unwrap()) .and_then(|values| values.first().map(|value| value.to_string())); async_graphql_tide::graphql(req, schema, |mut query_builder| { if let Some(name) = name { query_builder = query_builder.data(Hello(name.to_string())) } query_builder }) .await }); app.listen(&listen_addr).await?; Ok(()) }); let client = Task::>::spawn(async move { Timer::after(Duration::from_millis(300)).await; let resp = reqwest::Client::new() .post(format!("http://{}", listen_addr).as_str()) .body(r#"{"query":"{ hello }"}"#) .header(reqwest::header::CONTENT_TYPE, "application/json") .header("Name", "Foo") .send() .await?; assert_eq!(resp.status(), reqwest::StatusCode::OK); let string = resp.text().await?; println!("{}", string); assert_eq!(string, json!({"data":{"hello":"Hello, Foo!"}}).to_string()); let resp = reqwest::Client::new() .post(format!("http://{}", listen_addr).as_str()) .body(r#"{"query":"{ hello }"}"#) .header(reqwest::header::CONTENT_TYPE, "application/json") .send() .await?; assert_eq!(resp.status(), reqwest::StatusCode::OK); let string = resp.text().await?; println!("{}", string); assert_eq!( string, json!({"data":{"hello":"Hello, world!"}}).to_string() ); Ok(()) }); client.await?; server.cancel().await; Ok(()) }) } #[test] fn upload() -> Result<()> { smol::run(async { let listen_addr = test_utils::find_listen_addr().await; let server = Task::>::spawn(async move { struct QueryRoot; #[Object] impl QueryRoot {} #[async_graphql::SimpleObject] #[derive(Clone)] pub struct FileInfo { filename: String, mime_type: Option, } struct MutationRoot; #[Object] impl MutationRoot { async fn single_upload(&self, file: Upload) -> FileInfo { println!("single_upload: filename={}", file.filename); println!("single_upload: content_type={:?}", file.content_type); println!("single_upload: path={:?}", file.path); let file_path = file.path.clone(); let content = Task::blocking(async move { std::fs::read_to_string(file_path) }) .await .ok(); assert_eq!(content, Some("test\r\n".to_owned())); let file_info = FileInfo { filename: file.filename, mime_type: file.content_type, }; file_info } } let mut app = tide::new(); app.at("/").post(|req: Request<()>| async move { let schema = Schema::build(QueryRoot, MutationRoot, EmptySubscription).finish(); async_graphql_tide::graphql(req, schema, |query_builder| query_builder).await }); app.listen(&listen_addr).await?; Ok(()) }); let client = Task::>::spawn(async move { Timer::after(Duration::from_millis(300)).await; let form = reqwest::multipart::Form::new() .text("operations", r#"{ "query": "mutation ($file: Upload!) { singleUpload(file: $file) { filename, mimeType } }", "variables": { "file": null } }"#) .text("map", r#"{ "0": ["variables.file"] }"#) .part("0", reqwest::multipart::Part::stream("test").file_name("test.txt").mime_str("text/plain")?); let resp = reqwest::Client::new() .post(format!("http://{}", listen_addr).as_str()) .multipart(form) .send() .await?; assert_eq!(resp.status(), reqwest::StatusCode::OK); let string = resp.text().await?; println!("{}", string); assert_eq!( string, json!({"data": {"singleUpload": {"filename": "test.txt", "mimeType": "text/plain"}}}).to_string() ); Ok(()) }); client.await?; server.cancel().await; Ok(()) }) }