2020-08-09 17:19:34 -05:00
|
|
|
use std::{env, pin::Pin, time::Duration};
|
|
|
|
|
|
|
|
use actix_cors::Cors;
|
2021-06-29 01:22:45 -05:00
|
|
|
use actix_web::{http::header, middleware, web, App, Error, HttpRequest, HttpResponse, HttpServer};
|
2020-08-09 17:19:34 -05:00
|
|
|
|
|
|
|
use juniper::{
|
2020-11-06 20:15:18 -06:00
|
|
|
graphql_object, graphql_subscription,
|
Make interfaces great again! (#682)
* Bootstrap
* Upd
* Bootstrap macro
* Revert stuff
* Correct PoC to compile
* Bootstrap #[graphql_interface] expansion
* Bootstrap #[graphql_interface] meta parsing
* Bootstrap #[graphql_interface] very basic code generation [skip ci]
* Upd trait code generation and fix keywords usage [skip ci]
* Expand trait impls [skip ci]
* Tune up objects [skip ci]
* Finally! Complies at least... [skip ci]
* Parse meta for fields and its arguments [skip ci]
- also, refactor and bikeshed new macros code
* Impl filling fields meta and bootstrap field resolution [skip ci]
* Poking with fields resolution [skip ci]
* Solve Rust's teen async HRTB problems [skip ci]
* Start parsing trait methods [skip ci]
* Finish parsing fields from trait methods [skip ci]
* Autodetect trait asyncness and allow to specify it [skip ci]
* Allow to autogenerate trait object alias via attribute
* Support generics in trait definition and asyncify them correctly
* Temporary disable explicit async
* Cover arguments and custom names/descriptions in tests
* Re-enable tests with explicit async and fix the codegen to satisfy it
* Check implementers are registered in schema and vice versa
* Check argument camelCases
* Test argument defaults, and allow Into coercions for them
* Re-enable markers
* Re-enable markers and relax Sized requirement on IsInputType/IsOutputType marker traits
* Revert 'juniper_actix' fmt
* Fix missing marks for object
* Fix subscriptions marks
* Deduce result type correctly via traits
* Final fixes
* Fmt
* Restore marks checking
* Support custom ScalarValue
* Cover deprecations with tests
* Impl dowcasting via methods
* Impl dowcasting via external functions
* Support custom context, vol. 1
* Support custom context, vol. 2
* Cover fallible field with test
* Impl explicit generic ScalarValue, vol.1
* Impl explicit generic ScalarValue, vol.2
* Allow passing executor into methods
* Generating enum, vol.1
* Generating enum, vol.2
* Generating enum, vol.3
* Generating enum, vol.3
* Generating enum, vol.4
* Generating enum, vol.5
* Generating enum, vol.6
* Generating enum, vol.7
* Generating enum, vol.8
* Refactor juniper stuff
* Fix juniper tests, vol.1
* Fix juniper tests, vol.2
* Polish 'juniper' crate changes, vol.1
* Polish 'juniper' crate changes, vol.2
* Remove redundant stuf
* Polishing 'juniper_codegen', vol.1
* Polishing 'juniper_codegen', vol.2
* Polishing 'juniper_codegen', vol.3
* Polishing 'juniper_codegen', vol.4
* Polishing 'juniper_codegen', vol.5
* Polishing 'juniper_codegen', vol.6
* Polishing 'juniper_codegen', vol.7
* Polishing 'juniper_codegen', vol.8
* Polishing 'juniper_codegen', vol.9
* Fix other crates tests and make Clippy happier
* Fix examples
* Add codegen failure tests, vol. 1
* Add codegen failure tests, vol. 2
* Add codegen failure tests, vol.3
* Fix codegen failure tests accordingly to latest nightly Rust
* Fix codegen when interface has no implementers
* Fix warnings in book tests
* Describing new interfaces in Book, vol.1
Co-authored-by: Christian Legnitto <LegNeato@users.noreply.github.com>
2020-10-06 02:21:01 -05:00
|
|
|
tests::fixtures::starwars::schema::{Character as _, Database, Query},
|
2020-08-09 17:19:34 -05:00
|
|
|
DefaultScalarValue, EmptyMutation, FieldError, RootNode,
|
|
|
|
};
|
|
|
|
use juniper_actix::{graphql_handler, playground_handler, subscriptions::subscriptions_handler};
|
|
|
|
use juniper_graphql_ws::ConnectionConfig;
|
|
|
|
|
|
|
|
type Schema = RootNode<'static, Query, EmptyMutation<Database>, Subscription>;
|
|
|
|
|
|
|
|
fn schema() -> Schema {
|
|
|
|
Schema::new(Query, EmptyMutation::<Database>::new(), Subscription)
|
|
|
|
}
|
|
|
|
|
|
|
|
async fn playground() -> Result<HttpResponse, Error> {
|
|
|
|
playground_handler("/graphql", Some("/subscriptions")).await
|
|
|
|
}
|
|
|
|
|
|
|
|
async fn graphql(
|
|
|
|
req: actix_web::HttpRequest,
|
|
|
|
payload: actix_web::web::Payload,
|
|
|
|
schema: web::Data<Schema>,
|
|
|
|
) -> Result<HttpResponse, Error> {
|
|
|
|
let context = Database::new();
|
|
|
|
graphql_handler(&schema, &context, req, payload).await
|
|
|
|
}
|
|
|
|
|
|
|
|
struct Subscription;
|
|
|
|
|
|
|
|
struct RandomHuman {
|
|
|
|
id: String,
|
|
|
|
name: String,
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: remove this when async interfaces are merged
|
2020-11-06 20:15:18 -06:00
|
|
|
#[graphql_object(context = Database)]
|
2020-08-09 17:19:34 -05:00
|
|
|
impl RandomHuman {
|
|
|
|
fn id(&self) -> &str {
|
|
|
|
&self.id
|
|
|
|
}
|
|
|
|
|
|
|
|
fn name(&self) -> &str {
|
|
|
|
&self.name
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-19 02:08:53 -05:00
|
|
|
type RandomHumanStream =
|
|
|
|
Pin<Box<dyn futures::Stream<Item = Result<RandomHuman, FieldError>> + Send>>;
|
|
|
|
|
2020-11-06 20:15:18 -06:00
|
|
|
#[graphql_subscription(context = Database)]
|
2020-08-09 17:19:34 -05:00
|
|
|
impl Subscription {
|
|
|
|
#[graphql(
|
|
|
|
description = "A random humanoid creature in the Star Wars universe every 3 seconds. Second result will be an error."
|
|
|
|
)]
|
2020-08-19 02:08:53 -05:00
|
|
|
async fn random_human(context: &Database) -> RandomHumanStream {
|
2020-08-09 17:19:34 -05:00
|
|
|
let mut counter = 0;
|
|
|
|
|
|
|
|
let context = (*context).clone();
|
|
|
|
|
|
|
|
use rand::{rngs::StdRng, Rng, SeedableRng};
|
|
|
|
let mut rng = StdRng::from_entropy();
|
2021-06-29 01:22:45 -05:00
|
|
|
let mut interval = tokio::time::interval(Duration::from_secs(5));
|
|
|
|
let stream = async_stream::stream! {
|
2020-08-09 17:19:34 -05:00
|
|
|
counter += 1;
|
2021-06-29 01:22:45 -05:00
|
|
|
loop {
|
|
|
|
interval.tick().await;
|
|
|
|
if counter == 2 {
|
|
|
|
yield Err(FieldError::new(
|
|
|
|
"some field error from handler",
|
|
|
|
Value::Scalar(DefaultScalarValue::String(
|
|
|
|
"some additional string".to_string(),
|
|
|
|
)),
|
|
|
|
))
|
|
|
|
} else {
|
|
|
|
let random_id = rng.gen_range(1000..1005).to_string();
|
|
|
|
let human = context.get_human(&random_id).unwrap().clone();
|
|
|
|
|
|
|
|
yield Ok(RandomHuman {
|
|
|
|
id: human.id().to_owned(),
|
|
|
|
name: human.name().unwrap().to_owned(),
|
|
|
|
})
|
|
|
|
}
|
2020-08-09 17:19:34 -05:00
|
|
|
}
|
2021-06-29 01:22:45 -05:00
|
|
|
};
|
2020-08-09 17:19:34 -05:00
|
|
|
|
|
|
|
Box::pin(stream)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async fn subscriptions(
|
|
|
|
req: HttpRequest,
|
|
|
|
stream: web::Payload,
|
|
|
|
schema: web::Data<Schema>,
|
|
|
|
) -> Result<HttpResponse, Error> {
|
|
|
|
let context = Database::new();
|
|
|
|
let schema = schema.into_inner();
|
|
|
|
let config = ConnectionConfig::new(context);
|
|
|
|
// set the keep alive interval to 15 secs so that it doesn't timeout in playground
|
|
|
|
// playground has a hard-coded timeout set to 20 secs
|
|
|
|
let config = config.with_keep_alive_interval(Duration::from_secs(15));
|
|
|
|
|
|
|
|
subscriptions_handler(req, stream, schema, config).await
|
|
|
|
}
|
|
|
|
|
2020-09-12 11:32:15 -05:00
|
|
|
#[actix_web::main]
|
2020-08-09 17:19:34 -05:00
|
|
|
async fn main() -> std::io::Result<()> {
|
|
|
|
env::set_var("RUST_LOG", "info");
|
|
|
|
env_logger::init();
|
|
|
|
|
|
|
|
HttpServer::new(move || {
|
|
|
|
App::new()
|
|
|
|
.data(schema())
|
|
|
|
.wrap(middleware::Compress::default())
|
|
|
|
.wrap(middleware::Logger::default())
|
|
|
|
.wrap(
|
2020-10-20 04:57:14 -05:00
|
|
|
Cors::default()
|
2021-06-29 01:22:45 -05:00
|
|
|
.allowed_origin("http://127.0.0.1:8080")
|
2020-08-09 17:19:34 -05:00
|
|
|
.allowed_methods(vec!["POST", "GET"])
|
2021-06-29 01:22:45 -05:00
|
|
|
.allowed_headers(vec![header::AUTHORIZATION, header::ACCEPT])
|
|
|
|
.allowed_header(header::CONTENT_TYPE)
|
2020-08-09 17:19:34 -05:00
|
|
|
.supports_credentials()
|
2020-10-20 04:57:14 -05:00
|
|
|
.max_age(3600),
|
2020-08-09 17:19:34 -05:00
|
|
|
)
|
|
|
|
.service(web::resource("/subscriptions").route(web::get().to(subscriptions)))
|
|
|
|
.service(
|
|
|
|
web::resource("/graphql")
|
|
|
|
.route(web::post().to(graphql))
|
|
|
|
.route(web::get().to(graphql)),
|
|
|
|
)
|
|
|
|
.service(web::resource("/playground").route(web::get().to(playground)))
|
|
|
|
.default_service(web::route().to(|| {
|
|
|
|
HttpResponse::Found()
|
2021-06-29 01:22:45 -05:00
|
|
|
.append_header((header::LOCATION, "/playground"))
|
2020-08-09 17:19:34 -05:00
|
|
|
.finish()
|
|
|
|
}))
|
|
|
|
})
|
|
|
|
.bind(format!("{}:{}", "127.0.0.1", 8080))?
|
|
|
|
.run()
|
|
|
|
.await
|
|
|
|
}
|