271 lines
8.7 KiB
Rust
271 lines
8.7 KiB
Rust
mod test_utils;
|
|
use serde_json::json;
|
|
use smol::{Task, Timer};
|
|
use std::io::Read;
|
|
use std::time::Duration;
|
|
|
|
use async_graphql::*;
|
|
|
|
type Result<T> = std::result::Result<T, Box<dyn std::error::Error + Send + Sync>>;
|
|
|
|
#[test]
|
|
fn quickstart() -> Result<()> {
|
|
smol::run(async {
|
|
let listen_addr = test_utils::find_listen_addr().await;
|
|
|
|
let server = Task::<Result<()>>::spawn(async move {
|
|
use async_graphql_tide::{RequestExt, ResponseExt};
|
|
use tide::{http::StatusCode, Request, Response};
|
|
|
|
struct QueryRoot;
|
|
#[GQLObject]
|
|
impl QueryRoot {
|
|
#[field(desc = "Returns the sum of a and b")]
|
|
async fn add(&self, a: i32, b: i32) -> i32 {
|
|
a + b
|
|
}
|
|
}
|
|
|
|
async fn graphql(req: Request<()>) -> tide::Result<Response> {
|
|
let schema = Schema::build(QueryRoot, EmptyMutation, EmptySubscription).finish();
|
|
let request = req.body_graphql().await?;
|
|
Ok(Response::new(StatusCode::Ok).body_graphql(schema.execute(request).await)?)
|
|
}
|
|
|
|
let mut app = tide::new();
|
|
app.at("/").post(graphql).get(graphql);
|
|
app.listen(listen_addr).await?;
|
|
|
|
Ok(())
|
|
});
|
|
|
|
let client = Task::<Result<()>>::spawn(async move {
|
|
Timer::after(Duration::from_millis(300)).await;
|
|
|
|
let resp = reqwest::Client::builder()
|
|
.no_proxy()
|
|
.build()
|
|
.unwrap()
|
|
.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!("via post {}", string);
|
|
|
|
assert_eq!(string, json!({"data": {"add": 30}}).to_string());
|
|
|
|
//
|
|
let resp = reqwest::Client::builder()
|
|
.no_proxy()
|
|
.build()
|
|
.unwrap()
|
|
.get(
|
|
format!(
|
|
"http://{}?query=%7B%20add%28a%3A%2010%2C%20b%3A%2020%29%20%7D",
|
|
listen_addr
|
|
)
|
|
.as_str(),
|
|
)
|
|
.send()
|
|
.await?;
|
|
|
|
assert_eq!(resp.status(), reqwest::StatusCode::OK);
|
|
let string = resp.text().await?;
|
|
println!("via get {}", 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::<Result<()>>::spawn(async move {
|
|
use tide::Request;
|
|
|
|
struct Hello(String);
|
|
struct QueryRoot;
|
|
#[GQLObject]
|
|
impl QueryRoot {
|
|
#[field(desc = "Returns hello")]
|
|
async fn hello<'a>(&self, ctx: &'a Context<'_>) -> String {
|
|
let name = ctx.data_opt::<Hello>().map(|hello| hello.0.as_str());
|
|
format!("Hello, {}!", name.unwrap_or("world"))
|
|
}
|
|
}
|
|
|
|
#[derive(Clone)]
|
|
struct AppState {
|
|
schema: Schema<QueryRoot, EmptyMutation, EmptySubscription>,
|
|
}
|
|
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<AppState>| async move {
|
|
let schema = req.state().schema.clone();
|
|
let name = &req
|
|
.header("name")
|
|
.and_then(|values| values.get(0).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::<Result<()>>::spawn(async move {
|
|
Timer::after(Duration::from_millis(300)).await;
|
|
|
|
let resp = reqwest::Client::builder()
|
|
.no_proxy()
|
|
.build()
|
|
.unwrap()
|
|
.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::builder()
|
|
.no_proxy()
|
|
.build()
|
|
.unwrap()
|
|
.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::<Result<()>>::spawn(async move {
|
|
use tide::Request;
|
|
|
|
struct QueryRoot;
|
|
#[GQLObject]
|
|
impl QueryRoot {}
|
|
|
|
#[derive(Clone, GQLSimpleObject)]
|
|
pub struct FileInfo {
|
|
filename: String,
|
|
mime_type: Option<String>,
|
|
}
|
|
|
|
struct MutationRoot;
|
|
#[GQLObject]
|
|
impl MutationRoot {
|
|
async fn single_upload(&self, file: Upload) -> FileInfo {
|
|
println!("single_upload: filename={}", file.filename());
|
|
println!("single_upload: content_type={:?}", file.content_type());
|
|
|
|
let file_info = FileInfo {
|
|
filename: file.filename().into(),
|
|
mime_type: file.content_type().map(ToString::to_string),
|
|
};
|
|
|
|
let mut content = String::new();
|
|
file.into_read().read_to_string(&mut content).unwrap();
|
|
assert_eq!(content, "test".to_owned());
|
|
|
|
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::<Result<()>>::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::builder()
|
|
.no_proxy()
|
|
.build()
|
|
.unwrap()
|
|
.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(())
|
|
})
|
|
}
|