mirror of
https://github.com/fafhrd91/actix-web
synced 2024-11-28 01:52:57 +01:00
simplify data factory future polling (#1473)
Co-authored-by: Yuki Okushi <huyuumi.dev@gmail.com>
This commit is contained in:
parent
b047413b39
commit
bb17280f51
15
src/app.rs
15
src/app.rs
@ -476,13 +476,13 @@ where
|
|||||||
mod tests {
|
mod tests {
|
||||||
use actix_service::Service;
|
use actix_service::Service;
|
||||||
use bytes::Bytes;
|
use bytes::Bytes;
|
||||||
use futures::future::ok;
|
use futures::future::{ok, err};
|
||||||
|
|
||||||
use super::*;
|
use super::*;
|
||||||
use crate::http::{header, HeaderValue, Method, StatusCode};
|
use crate::http::{header, HeaderValue, Method, StatusCode};
|
||||||
use crate::middleware::DefaultHeaders;
|
use crate::middleware::DefaultHeaders;
|
||||||
use crate::service::ServiceRequest;
|
use crate::service::ServiceRequest;
|
||||||
use crate::test::{call_service, init_service, read_body, TestRequest};
|
use crate::test::{call_service, init_service, try_init_service, read_body, TestRequest};
|
||||||
use crate::{web, HttpRequest, HttpResponse};
|
use crate::{web, HttpRequest, HttpResponse};
|
||||||
|
|
||||||
#[actix_rt::test]
|
#[actix_rt::test]
|
||||||
@ -551,6 +551,17 @@ mod tests {
|
|||||||
assert_eq!(resp.status(), StatusCode::INTERNAL_SERVER_ERROR);
|
assert_eq!(resp.status(), StatusCode::INTERNAL_SERVER_ERROR);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[actix_rt::test]
|
||||||
|
async fn test_data_factory_errors() {
|
||||||
|
let srv =
|
||||||
|
try_init_service(App::new().data_factory(|| err::<u32, _>(())).service(
|
||||||
|
web::resource("/").to(|_: web::Data<usize>| HttpResponse::Ok()),
|
||||||
|
))
|
||||||
|
.await;
|
||||||
|
|
||||||
|
assert!(srv.is_err());
|
||||||
|
}
|
||||||
|
|
||||||
#[actix_rt::test]
|
#[actix_rt::test]
|
||||||
async fn test_extension() {
|
async fn test_extension() {
|
||||||
let mut srv = init_service(App::new().app_data(10usize).service(
|
let mut srv = init_service(App::new().app_data(10usize).service(
|
||||||
|
@ -9,7 +9,7 @@ use actix_http::{Extensions, Request, Response};
|
|||||||
use actix_router::{Path, ResourceDef, ResourceInfo, Router, Url};
|
use actix_router::{Path, ResourceDef, ResourceInfo, Router, Url};
|
||||||
use actix_service::boxed::{self, BoxService, BoxServiceFactory};
|
use actix_service::boxed::{self, BoxService, BoxServiceFactory};
|
||||||
use actix_service::{fn_service, Service, ServiceFactory};
|
use actix_service::{fn_service, Service, ServiceFactory};
|
||||||
use futures::future::{ok, FutureExt, LocalBoxFuture};
|
use futures::future::{join_all, ok, FutureExt, LocalBoxFuture};
|
||||||
|
|
||||||
use crate::config::{AppConfig, AppService};
|
use crate::config::{AppConfig, AppService};
|
||||||
use crate::data::DataFactory;
|
use crate::data::DataFactory;
|
||||||
@ -109,12 +109,15 @@ where
|
|||||||
let rmap = Rc::new(rmap);
|
let rmap = Rc::new(rmap);
|
||||||
rmap.finish(rmap.clone());
|
rmap.finish(rmap.clone());
|
||||||
|
|
||||||
|
// start all data factory futures
|
||||||
|
let factory_futs = join_all(self.data_factories.iter().map(|f| f()));
|
||||||
|
|
||||||
AppInitResult {
|
AppInitResult {
|
||||||
endpoint: None,
|
endpoint: None,
|
||||||
endpoint_fut: self.endpoint.new_service(()),
|
endpoint_fut: self.endpoint.new_service(()),
|
||||||
data: self.data.clone(),
|
data: self.data.clone(),
|
||||||
data_factories: Vec::new(),
|
data_factories: None,
|
||||||
data_factories_fut: self.data_factories.iter().map(|f| f()).collect(),
|
data_factories_fut: factory_futs.boxed_local(),
|
||||||
extensions: Some(
|
extensions: Some(
|
||||||
self.extensions
|
self.extensions
|
||||||
.borrow_mut()
|
.borrow_mut()
|
||||||
@ -133,15 +136,21 @@ pub struct AppInitResult<T, B>
|
|||||||
where
|
where
|
||||||
T: ServiceFactory,
|
T: ServiceFactory,
|
||||||
{
|
{
|
||||||
endpoint: Option<T::Service>,
|
|
||||||
#[pin]
|
#[pin]
|
||||||
endpoint_fut: T::Future,
|
endpoint_fut: T::Future,
|
||||||
|
// a Some signals completion of endpoint creation
|
||||||
|
endpoint: Option<T::Service>,
|
||||||
|
|
||||||
|
#[pin]
|
||||||
|
data_factories_fut: LocalBoxFuture<'static, Vec<Result<Box<dyn DataFactory>, ()>>>,
|
||||||
|
// a Some signals completion of factory futures
|
||||||
|
data_factories: Option<Vec<Box<dyn DataFactory>>>,
|
||||||
|
|
||||||
rmap: Rc<ResourceMap>,
|
rmap: Rc<ResourceMap>,
|
||||||
config: AppConfig,
|
config: AppConfig,
|
||||||
data: Rc<Vec<Box<dyn DataFactory>>>,
|
data: Rc<Vec<Box<dyn DataFactory>>>,
|
||||||
data_factories: Vec<Box<dyn DataFactory>>,
|
|
||||||
data_factories_fut: Vec<LocalBoxFuture<'static, Result<Box<dyn DataFactory>, ()>>>,
|
|
||||||
extensions: Option<Extensions>,
|
extensions: Option<Extensions>,
|
||||||
|
|
||||||
_t: PhantomData<B>,
|
_t: PhantomData<B>,
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -161,44 +170,46 @@ where
|
|||||||
let this = self.project();
|
let this = self.project();
|
||||||
|
|
||||||
// async data factories
|
// async data factories
|
||||||
let mut idx = 0;
|
if let Poll::Ready(factories) = this.data_factories_fut.poll(cx) {
|
||||||
while idx < this.data_factories_fut.len() {
|
let factories: Result<Vec<_>, ()> = factories.into_iter().collect();
|
||||||
match Pin::new(&mut this.data_factories_fut[idx]).poll(cx)? {
|
|
||||||
Poll::Ready(f) => {
|
if let Ok(factories) = factories {
|
||||||
this.data_factories.push(f);
|
this.data_factories.replace(factories);
|
||||||
let _ = this.data_factories_fut.remove(idx);
|
} else {
|
||||||
}
|
return Poll::Ready(Err(()));
|
||||||
Poll::Pending => idx += 1,
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// app service and middleware
|
||||||
if this.endpoint.is_none() {
|
if this.endpoint.is_none() {
|
||||||
if let Poll::Ready(srv) = this.endpoint_fut.poll(cx)? {
|
if let Poll::Ready(srv) = this.endpoint_fut.poll(cx)? {
|
||||||
*this.endpoint = Some(srv);
|
*this.endpoint = Some(srv);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if this.endpoint.is_some() && this.data_factories_fut.is_empty() {
|
// not using if let so condition only needs shared ref
|
||||||
|
if this.endpoint.is_some() && this.data_factories.is_some() {
|
||||||
// create app data container
|
// create app data container
|
||||||
let mut data = this.extensions.take().unwrap();
|
let mut data = this.extensions.take().unwrap();
|
||||||
|
|
||||||
for f in this.data.iter() {
|
for f in this.data.iter() {
|
||||||
f.create(&mut data);
|
f.create(&mut data);
|
||||||
}
|
}
|
||||||
|
|
||||||
for f in this.data_factories.iter() {
|
for f in this.data_factories.take().unwrap().iter() {
|
||||||
f.create(&mut data);
|
f.create(&mut data);
|
||||||
}
|
}
|
||||||
|
|
||||||
Poll::Ready(Ok(AppInitService {
|
return Poll::Ready(Ok(AppInitService {
|
||||||
service: this.endpoint.take().unwrap(),
|
service: this.endpoint.take().unwrap(),
|
||||||
rmap: this.rmap.clone(),
|
rmap: this.rmap.clone(),
|
||||||
config: this.config.clone(),
|
config: this.config.clone(),
|
||||||
data: Rc::new(data),
|
data: Rc::new(data),
|
||||||
pool: HttpRequestPool::create(),
|
pool: HttpRequestPool::create(),
|
||||||
}))
|
}));
|
||||||
} else {
|
|
||||||
Poll::Pending
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Poll::Pending
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
118
src/test.rs
118
src/test.rs
@ -78,6 +78,26 @@ pub fn default_service(
|
|||||||
pub async fn init_service<R, S, B, E>(
|
pub async fn init_service<R, S, B, E>(
|
||||||
app: R,
|
app: R,
|
||||||
) -> impl Service<Request = Request, Response = ServiceResponse<B>, Error = E>
|
) -> impl Service<Request = Request, Response = ServiceResponse<B>, Error = E>
|
||||||
|
where
|
||||||
|
R: IntoServiceFactory<S>,
|
||||||
|
S: ServiceFactory<
|
||||||
|
Config = AppConfig,
|
||||||
|
Request = Request,
|
||||||
|
Response = ServiceResponse<B>,
|
||||||
|
Error = E,
|
||||||
|
>,
|
||||||
|
S::InitError: std::fmt::Debug,
|
||||||
|
{
|
||||||
|
try_init_service(app).await.expect("service initilization failed")
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Fallible version of init_service that allows testing data factory errors.
|
||||||
|
pub(crate) async fn try_init_service<R, S, B, E>(
|
||||||
|
app: R,
|
||||||
|
) -> Result<
|
||||||
|
impl Service<Request = Request, Response = ServiceResponse<B>, Error = E>,
|
||||||
|
S::InitError,
|
||||||
|
>
|
||||||
where
|
where
|
||||||
R: IntoServiceFactory<S>,
|
R: IntoServiceFactory<S>,
|
||||||
S: ServiceFactory<
|
S: ServiceFactory<
|
||||||
@ -89,7 +109,7 @@ where
|
|||||||
S::InitError: std::fmt::Debug,
|
S::InitError: std::fmt::Debug,
|
||||||
{
|
{
|
||||||
let srv = app.into_factory();
|
let srv = app.into_factory();
|
||||||
srv.new_service(AppConfig::default()).await.unwrap()
|
srv.new_service(AppConfig::default()).await
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Calls service and waits for response future completion.
|
/// Calls service and waits for response future completion.
|
||||||
@ -580,7 +600,7 @@ impl TestRequest {
|
|||||||
pub async fn send_request<S, B, E>(self, app: &mut S) -> S::Response
|
pub async fn send_request<S, B, E>(self, app: &mut S) -> S::Response
|
||||||
where
|
where
|
||||||
S: Service<Request = Request, Response = ServiceResponse<B>, Error = E>,
|
S: Service<Request = Request, Response = ServiceResponse<B>, Error = E>,
|
||||||
E: std::fmt::Debug
|
E: std::fmt::Debug,
|
||||||
{
|
{
|
||||||
let req = self.to_request();
|
let req = self.to_request();
|
||||||
call_service(app, req).await
|
call_service(app, req).await
|
||||||
@ -1125,8 +1145,8 @@ mod tests {
|
|||||||
#[actix_rt::test]
|
#[actix_rt::test]
|
||||||
async fn test_response_json() {
|
async fn test_response_json() {
|
||||||
let mut app = init_service(App::new().service(web::resource("/people").route(
|
let mut app = init_service(App::new().service(web::resource("/people").route(
|
||||||
web::post().to(|person: web::Json<Person>| {
|
web::post().to(|person: web::Json<Person>| async {
|
||||||
async { HttpResponse::Ok().json(person.into_inner()) }
|
HttpResponse::Ok().json(person.into_inner())
|
||||||
}),
|
}),
|
||||||
)))
|
)))
|
||||||
.await;
|
.await;
|
||||||
@ -1146,8 +1166,8 @@ mod tests {
|
|||||||
#[actix_rt::test]
|
#[actix_rt::test]
|
||||||
async fn test_body_json() {
|
async fn test_body_json() {
|
||||||
let mut app = init_service(App::new().service(web::resource("/people").route(
|
let mut app = init_service(App::new().service(web::resource("/people").route(
|
||||||
web::post().to(|person: web::Json<Person>| {
|
web::post().to(|person: web::Json<Person>| async {
|
||||||
async { HttpResponse::Ok().json(person.into_inner()) }
|
HttpResponse::Ok().json(person.into_inner())
|
||||||
}),
|
}),
|
||||||
)))
|
)))
|
||||||
.await;
|
.await;
|
||||||
@ -1168,8 +1188,8 @@ mod tests {
|
|||||||
#[actix_rt::test]
|
#[actix_rt::test]
|
||||||
async fn test_request_response_form() {
|
async fn test_request_response_form() {
|
||||||
let mut app = init_service(App::new().service(web::resource("/people").route(
|
let mut app = init_service(App::new().service(web::resource("/people").route(
|
||||||
web::post().to(|person: web::Form<Person>| {
|
web::post().to(|person: web::Form<Person>| async {
|
||||||
async { HttpResponse::Ok().json(person.into_inner()) }
|
HttpResponse::Ok().json(person.into_inner())
|
||||||
}),
|
}),
|
||||||
)))
|
)))
|
||||||
.await;
|
.await;
|
||||||
@ -1194,8 +1214,8 @@ mod tests {
|
|||||||
#[actix_rt::test]
|
#[actix_rt::test]
|
||||||
async fn test_request_response_json() {
|
async fn test_request_response_json() {
|
||||||
let mut app = init_service(App::new().service(web::resource("/people").route(
|
let mut app = init_service(App::new().service(web::resource("/people").route(
|
||||||
web::post().to(|person: web::Json<Person>| {
|
web::post().to(|person: web::Json<Person>| async {
|
||||||
async { HttpResponse::Ok().json(person.into_inner()) }
|
HttpResponse::Ok().json(person.into_inner())
|
||||||
}),
|
}),
|
||||||
)))
|
)))
|
||||||
.await;
|
.await;
|
||||||
@ -1259,53 +1279,53 @@ mod tests {
|
|||||||
assert!(res.status().is_success());
|
assert!(res.status().is_success());
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
|
||||||
Comment out until actix decoupled of actix-http:
|
Comment out until actix decoupled of actix-http:
|
||||||
https://github.com/actix/actix/issues/321
|
https://github.com/actix/actix/issues/321
|
||||||
|
|
||||||
use futures::FutureExt;
|
use futures::FutureExt;
|
||||||
|
|
||||||
#[actix_rt::test]
|
#[actix_rt::test]
|
||||||
async fn test_actor() {
|
async fn test_actor() {
|
||||||
use actix::Actor;
|
use actix::Actor;
|
||||||
|
|
||||||
struct MyActor;
|
struct MyActor;
|
||||||
|
|
||||||
struct Num(usize);
|
struct Num(usize);
|
||||||
impl actix::Message for Num {
|
impl actix::Message for Num {
|
||||||
type Result = usize;
|
type Result = usize;
|
||||||
}
|
}
|
||||||
impl actix::Actor for MyActor {
|
impl actix::Actor for MyActor {
|
||||||
type Context = actix::Context<Self>;
|
type Context = actix::Context<Self>;
|
||||||
}
|
}
|
||||||
impl actix::Handler<Num> for MyActor {
|
impl actix::Handler<Num> for MyActor {
|
||||||
type Result = usize;
|
type Result = usize;
|
||||||
fn handle(&mut self, msg: Num, _: &mut Self::Context) -> Self::Result {
|
fn handle(&mut self, msg: Num, _: &mut Self::Context) -> Self::Result {
|
||||||
msg.0
|
msg.0
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
let mut app = init_service(App::new().service(web::resource("/index.html").to(
|
let mut app = init_service(App::new().service(web::resource("/index.html").to(
|
||||||
move || {
|
move || {
|
||||||
addr.send(Num(1)).map(|res| match res {
|
addr.send(Num(1)).map(|res| match res {
|
||||||
Ok(res) => {
|
Ok(res) => {
|
||||||
if res == 1 {
|
if res == 1 {
|
||||||
Ok(HttpResponse::Ok())
|
Ok(HttpResponse::Ok())
|
||||||
} else {
|
} else {
|
||||||
Ok(HttpResponse::BadRequest())
|
Ok(HttpResponse::BadRequest())
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
Err(err) => Err(err),
|
||||||
Err(err) => Err(err),
|
})
|
||||||
})
|
},
|
||||||
},
|
)))
|
||||||
)))
|
.await;
|
||||||
.await;
|
|
||||||
|
|
||||||
let req = TestRequest::post().uri("/index.html").to_request();
|
let req = TestRequest::post().uri("/index.html").to_request();
|
||||||
let res = app.call(req).await.unwrap();
|
let res = app.call(req).await.unwrap();
|
||||||
assert!(res.status().is_success());
|
assert!(res.status().is_success());
|
||||||
}
|
}
|
||||||
*/
|
*/
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user