1
0
Fork 0
mirror of https://github.com/actix/actix-web.git synced 2024-10-11 04:32:28 +00:00
actix-web/src/pipeline.rs

751 lines
26 KiB
Rust
Raw Normal View History

2017-12-01 23:45:15 +00:00
use std::{io, mem};
2017-11-25 06:15:52 +00:00
use std::rc::Rc;
2017-12-29 09:01:31 +00:00
use std::cell::RefCell;
2017-12-09 13:54:04 +00:00
use std::marker::PhantomData;
2017-11-25 06:15:52 +00:00
2017-12-01 23:45:15 +00:00
use futures::{Async, Poll, Future, Stream};
2017-12-24 19:58:09 +00:00
use futures::unsync::oneshot;
2017-11-25 06:15:52 +00:00
2017-12-01 23:45:15 +00:00
use body::{Body, BodyStream};
2018-01-01 01:26:32 +00:00
use context::{Frame, ActorHttpContext};
use error::Error;
2017-12-04 22:53:40 +00:00
use handler::{Reply, ReplyItem};
2017-11-25 06:15:52 +00:00
use httprequest::HttpRequest;
use httpresponse::HttpResponse;
2017-12-27 03:59:41 +00:00
use middleware::{Middleware, Finished, Started, Response};
2017-12-29 09:01:31 +00:00
use application::Inner;
2018-01-12 02:35:05 +00:00
use server::{Writer, WriterState, HttpHandlerTask};
2017-11-25 06:15:52 +00:00
2017-12-29 17:16:50 +00:00
pub(crate) trait PipelineHandler<S> {
2017-12-29 09:01:31 +00:00
fn handle(&mut self, req: HttpRequest<S>) -> Reply;
}
2017-11-25 06:15:52 +00:00
2017-12-29 17:16:50 +00:00
pub(crate) struct Pipeline<S, H>(PipelineInfo<S>, PipelineState<S, H>);
2017-11-25 06:15:52 +00:00
2017-12-29 09:01:31 +00:00
enum PipelineState<S, H> {
2017-11-25 06:15:52 +00:00
None,
2017-12-01 23:45:15 +00:00
Error,
2017-12-29 09:01:31 +00:00
Starting(StartMiddlewares<S, H>),
Handler(WaitingResponse<S, H>),
RunMiddlewares(RunMiddlewares<S, H>),
Response(ProcessResponse<S, H>),
Finishing(FinishingMiddlewares<S, H>),
Completed(Completed<S, H>),
2017-11-25 06:15:52 +00:00
}
2018-01-11 00:45:57 +00:00
impl<S: 'static, H: PipelineHandler<S>> PipelineState<S, H> {
fn is_response(&self) -> bool {
match *self {
PipelineState::Response(_) => true,
_ => false,
}
}
fn poll(&mut self, info: &mut PipelineInfo<S>) -> Option<PipelineState<S, H>> {
match *self {
PipelineState::Starting(ref mut state) => state.poll(info),
PipelineState::Handler(ref mut state) => state.poll(info),
PipelineState::RunMiddlewares(ref mut state) => state.poll(info),
PipelineState::Finishing(ref mut state) => state.poll(info),
PipelineState::Completed(ref mut state) => state.poll(info),
PipelineState::Response(_) | PipelineState::None | PipelineState::Error => None,
}
}
}
2017-12-09 12:33:40 +00:00
struct PipelineInfo<S> {
req: HttpRequest<S>,
2017-12-01 23:45:15 +00:00
count: usize,
2017-12-09 12:33:40 +00:00
mws: Rc<Vec<Box<Middleware<S>>>>,
2018-01-01 01:26:32 +00:00
context: Option<Box<ActorHttpContext>>,
2017-12-01 23:45:15 +00:00
error: Option<Error>,
disconnected: Option<bool>,
2017-12-01 23:45:15 +00:00
}
2017-12-09 12:33:40 +00:00
impl<S> PipelineInfo<S> {
fn new(req: HttpRequest<S>) -> PipelineInfo<S> {
2017-12-01 23:45:15 +00:00
PipelineInfo {
req: req,
count: 0,
mws: Rc::new(Vec::new()),
error: None,
context: None,
disconnected: None,
2017-12-01 23:45:15 +00:00
}
}
#[cfg_attr(feature = "cargo-clippy", allow(mut_from_ref))]
2017-12-09 12:33:40 +00:00
fn req_mut(&self) -> &mut HttpRequest<S> {
2017-12-01 23:45:15 +00:00
#[allow(mutable_transmutes)]
unsafe{mem::transmute(&self.req)}
}
fn poll_context(&mut self) -> Poll<(), Error> {
if let Some(ref mut context) = self.context {
match context.poll() {
Err(err) => Err(err),
Ok(Async::NotReady) => Ok(Async::NotReady),
Ok(Async::Ready(_)) => Ok(Async::Ready(())),
2017-11-25 06:15:52 +00:00
}
2017-12-01 23:45:15 +00:00
} else {
Ok(Async::Ready(()))
2017-11-25 06:15:52 +00:00
}
}
2017-12-01 23:45:15 +00:00
}
2017-11-25 06:15:52 +00:00
2018-01-10 04:00:18 +00:00
impl<S: 'static, H: PipelineHandler<S>> Pipeline<S, H> {
2017-12-01 23:45:15 +00:00
2017-12-09 12:33:40 +00:00
pub fn new(req: HttpRequest<S>,
2017-12-09 13:54:04 +00:00
mws: Rc<Vec<Box<Middleware<S>>>>,
2017-12-29 09:01:31 +00:00
handler: Rc<RefCell<H>>) -> Pipeline<S, H>
2017-12-01 23:45:15 +00:00
{
2017-12-09 13:54:04 +00:00
let mut info = PipelineInfo {
req: req,
count: 0,
mws: mws,
error: None,
context: None,
disconnected: None,
2017-12-09 13:54:04 +00:00
};
let state = StartMiddlewares::init(&mut info, handler);
Pipeline(info, state)
2017-12-01 23:45:15 +00:00
}
2017-12-09 12:33:40 +00:00
}
2017-12-01 23:45:15 +00:00
2017-12-29 09:01:31 +00:00
impl Pipeline<(), Inner<()>> {
2017-12-09 12:33:40 +00:00
pub fn error<R: Into<HttpResponse>>(err: R) -> Box<HttpHandlerTask> {
2017-12-29 09:01:31 +00:00
Box::new(Pipeline::<(), Inner<()>>(
PipelineInfo::new(HttpRequest::default()), ProcessResponse::init(err.into())))
2017-12-09 13:54:04 +00:00
}
}
2018-01-10 04:00:18 +00:00
impl<S: 'static, H> Pipeline<S, H> {
2017-12-09 13:54:04 +00:00
fn is_done(&self) -> bool {
match self.1 {
PipelineState::None | PipelineState::Error
| PipelineState::Starting(_) | PipelineState::Handler(_)
| PipelineState::RunMiddlewares(_) | PipelineState::Response(_) => true,
2018-01-11 00:45:57 +00:00
PipelineState::Finishing(_) | PipelineState::Completed(_) => false,
2017-12-09 13:54:04 +00:00
}
2017-12-01 23:45:15 +00:00
}
2017-12-09 12:33:40 +00:00
}
2018-01-10 04:00:18 +00:00
impl<S: 'static, H: PipelineHandler<S>> HttpHandlerTask for Pipeline<S, H> {
2017-12-01 23:45:15 +00:00
2017-12-09 12:33:40 +00:00
fn disconnected(&mut self) {
self.0.disconnected = Some(true);
2017-12-01 23:45:15 +00:00
}
2017-11-25 06:15:52 +00:00
2017-12-09 12:33:40 +00:00
fn poll_io(&mut self, io: &mut Writer) -> Poll<bool, Error> {
2018-01-11 00:45:57 +00:00
let info: &mut PipelineInfo<_> = unsafe{ mem::transmute(&mut self.0) };
2017-11-25 06:15:52 +00:00
loop {
2018-01-11 00:45:57 +00:00
if self.1.is_response() {
let state = mem::replace(&mut self.1, PipelineState::None);
if let PipelineState::Response(st) = state {
match st.poll_io(io, info) {
2017-12-01 23:45:15 +00:00
Ok(state) => {
2017-12-09 13:54:04 +00:00
self.1 = state;
if let Some(error) = self.0.error.take() {
2017-12-01 23:45:15 +00:00
return Err(error)
} else {
2017-12-09 13:54:04 +00:00
return Ok(Async::Ready(self.is_done()))
2017-12-01 23:45:15 +00:00
}
}
Err(state) => {
2017-12-09 13:54:04 +00:00
self.1 = state;
2018-01-11 00:45:57 +00:00
return Ok(Async::NotReady);
2017-12-01 23:45:15 +00:00
}
}
2017-11-25 06:15:52 +00:00
}
}
2018-01-11 00:45:57 +00:00
match self.1 {
PipelineState::None =>
return Ok(Async::Ready(true)),
PipelineState::Error =>
return Err(io::Error::new(io::ErrorKind::Other, "Internal error").into()),
_ => (),
}
match self.1.poll(info) {
Some(state) => self.1 = state,
None => return Ok(Async::NotReady),
}
2017-11-25 06:15:52 +00:00
}
}
2017-12-09 12:33:40 +00:00
fn poll(&mut self) -> Poll<(), Error> {
2018-01-11 00:45:57 +00:00
let info: &mut PipelineInfo<_> = unsafe{ mem::transmute(&mut self.0) };
2017-11-25 06:15:52 +00:00
loop {
2018-01-11 00:45:57 +00:00
match self.1 {
2017-12-01 23:45:15 +00:00
PipelineState::None | PipelineState::Error => {
return Ok(Async::Ready(()))
}
2018-01-11 00:45:57 +00:00
_ => (),
}
if let Some(state) = self.1.poll(info) {
self.1 = state;
} else {
return Ok(Async::NotReady);
2017-11-25 06:15:52 +00:00
}
}
}
}
2017-11-27 05:47:33 +00:00
type Fut = Box<Future<Item=Option<HttpResponse>, Error=Error>>;
2017-11-25 06:15:52 +00:00
/// Middlewares start executor
2017-12-29 09:01:31 +00:00
struct StartMiddlewares<S, H> {
hnd: Rc<RefCell<H>>,
2017-11-25 06:15:52 +00:00
fut: Option<Fut>,
2017-12-29 09:01:31 +00:00
_s: PhantomData<S>,
2017-11-25 06:15:52 +00:00
}
2018-01-10 04:00:18 +00:00
impl<S: 'static, H: PipelineHandler<S>> StartMiddlewares<S, H> {
2017-11-25 06:15:52 +00:00
2018-01-11 04:08:13 +00:00
fn init(info: &mut PipelineInfo<S>, handler: Rc<RefCell<H>>) -> PipelineState<S, H> {
2017-12-01 23:45:15 +00:00
// execute middlewares, we need this stage because middlewares could be non-async
// and we can move to next state immidietly
let len = info.mws.len();
2017-11-25 06:15:52 +00:00
loop {
2017-12-01 23:45:15 +00:00
if info.count == len {
2017-12-29 09:01:31 +00:00
let reply = handler.borrow_mut().handle(info.req.clone());
2017-12-09 12:33:40 +00:00
return WaitingResponse::init(info, reply)
2017-11-25 06:15:52 +00:00
} else {
2017-12-01 23:45:15 +00:00
match info.mws[info.count].start(&mut info.req) {
2018-01-10 06:48:35 +00:00
Ok(Started::Done) =>
2017-12-01 23:45:15 +00:00
info.count += 1,
2018-01-10 06:48:35 +00:00
Ok(Started::Response(resp)) =>
2017-12-09 12:33:40 +00:00
return RunMiddlewares::init(info, resp),
2018-01-10 06:48:35 +00:00
Ok(Started::Future(mut fut)) =>
2017-11-25 06:15:52 +00:00
match fut.poll() {
2017-12-01 23:45:15 +00:00
Ok(Async::NotReady) =>
return PipelineState::Starting(StartMiddlewares {
2017-12-29 09:01:31 +00:00
hnd: handler,
fut: Some(fut),
_s: PhantomData}),
2017-11-27 05:47:33 +00:00
Ok(Async::Ready(resp)) => {
2017-11-25 06:15:52 +00:00
if let Some(resp) = resp {
2017-12-16 00:24:15 +00:00
return RunMiddlewares::init(info, resp);
2017-11-25 06:15:52 +00:00
}
2017-12-01 23:45:15 +00:00
info.count += 1;
2017-11-25 06:15:52 +00:00
}
2017-12-01 23:45:15 +00:00
Err(err) =>
2017-12-16 00:24:15 +00:00
return ProcessResponse::init(err.into()),
2017-11-27 05:47:33 +00:00
},
2018-01-10 06:48:35 +00:00
Err(err) =>
2017-12-16 00:24:15 +00:00
return ProcessResponse::init(err.into()),
2017-11-25 06:15:52 +00:00
}
}
}
}
2018-01-11 04:08:13 +00:00
fn poll(&mut self, info: &mut PipelineInfo<S>) -> Option<PipelineState<S, H>> {
2017-12-09 13:54:04 +00:00
let len = info.mws.len();
2017-11-25 06:15:52 +00:00
'outer: loop {
match self.fut.as_mut().unwrap().poll() {
2018-01-11 00:45:57 +00:00
Ok(Async::NotReady) => return None,
2017-11-27 05:47:33 +00:00
Ok(Async::Ready(resp)) => {
2017-12-09 13:54:04 +00:00
info.count += 1;
2017-11-25 06:15:52 +00:00
if let Some(resp) = resp {
2018-01-11 00:45:57 +00:00
return Some(RunMiddlewares::init(info, resp));
2017-11-25 06:15:52 +00:00
}
2017-12-09 13:54:04 +00:00
if info.count == len {
2017-12-29 09:01:31 +00:00
let reply = (*self.hnd.borrow_mut()).handle(info.req.clone());
2018-01-11 00:45:57 +00:00
return Some(WaitingResponse::init(info, reply));
2017-11-25 06:15:52 +00:00
} else {
loop {
2017-12-09 13:54:04 +00:00
match info.mws[info.count].start(info.req_mut()) {
2018-01-10 06:48:35 +00:00
Ok(Started::Done) =>
2017-12-09 13:54:04 +00:00
info.count += 1,
2018-01-10 06:48:35 +00:00
Ok(Started::Response(resp)) => {
2018-01-11 00:45:57 +00:00
return Some(RunMiddlewares::init(info, resp));
2017-11-25 06:15:52 +00:00
},
2018-01-10 06:48:35 +00:00
Ok(Started::Future(fut)) => {
2017-11-25 06:15:52 +00:00
self.fut = Some(fut);
continue 'outer
},
2018-01-10 06:48:35 +00:00
Err(err) =>
2018-01-11 00:45:57 +00:00
return Some(ProcessResponse::init(err.into()))
2017-11-25 06:15:52 +00:00
}
}
}
}
2017-12-01 23:45:15 +00:00
Err(err) =>
2018-01-11 00:45:57 +00:00
return Some(ProcessResponse::init(err.into()))
2017-11-25 06:15:52 +00:00
}
}
}
}
2017-12-01 23:45:15 +00:00
// waiting for response
2017-12-29 09:01:31 +00:00
struct WaitingResponse<S, H> {
2018-01-01 01:26:32 +00:00
fut: Box<Future<Item=HttpResponse, Error=Error>>,
2017-12-09 13:54:04 +00:00
_s: PhantomData<S>,
2017-12-29 09:01:31 +00:00
_h: PhantomData<H>,
2017-11-25 20:05:27 +00:00
}
2018-01-10 04:00:18 +00:00
impl<S: 'static, H> WaitingResponse<S, H> {
2017-11-25 20:05:27 +00:00
2017-12-16 04:00:12 +00:00
#[inline]
2018-01-11 04:08:13 +00:00
fn init(info: &mut PipelineInfo<S>, reply: Reply) -> PipelineState<S, H> {
2017-12-16 04:00:12 +00:00
match reply.into() {
2017-12-01 23:45:15 +00:00
ReplyItem::Message(resp) =>
2017-12-16 04:00:12 +00:00
RunMiddlewares::init(info, resp),
2017-12-01 23:45:15 +00:00
ReplyItem::Future(fut) =>
2017-12-16 04:00:12 +00:00
PipelineState::Handler(
2018-01-01 01:26:32 +00:00
WaitingResponse { fut: fut, _s: PhantomData, _h: PhantomData }),
2017-12-16 04:00:12 +00:00
}
2017-11-25 20:05:27 +00:00
}
2018-01-11 04:08:13 +00:00
fn poll(&mut self, info: &mut PipelineInfo<S>) -> Option<PipelineState<S, H>> {
2018-01-01 01:26:32 +00:00
match self.fut.poll() {
2018-01-11 00:45:57 +00:00
Ok(Async::NotReady) => None,
2018-01-01 01:26:32 +00:00
Ok(Async::Ready(response)) =>
2018-01-11 00:45:57 +00:00
Some(RunMiddlewares::init(info, response)),
2018-01-01 01:26:32 +00:00
Err(err) =>
2018-01-11 00:45:57 +00:00
Some(ProcessResponse::init(err.into())),
2017-11-25 20:05:27 +00:00
}
}
}
2017-11-25 06:15:52 +00:00
/// Middlewares response executor
2017-12-29 09:01:31 +00:00
struct RunMiddlewares<S, H> {
2017-12-01 23:45:15 +00:00
curr: usize,
2017-11-25 17:28:25 +00:00
fut: Option<Box<Future<Item=HttpResponse, Error=Error>>>,
2017-12-09 13:54:04 +00:00
_s: PhantomData<S>,
2017-12-29 09:01:31 +00:00
_h: PhantomData<H>,
2017-11-25 06:15:52 +00:00
}
2018-01-10 04:00:18 +00:00
impl<S: 'static, H> RunMiddlewares<S, H> {
2017-11-25 06:15:52 +00:00
2018-01-11 04:08:13 +00:00
fn init(info: &mut PipelineInfo<S>, mut resp: HttpResponse) -> PipelineState<S, H> {
2017-12-01 23:45:15 +00:00
if info.count == 0 {
2017-12-09 13:54:04 +00:00
return ProcessResponse::init(resp);
2017-12-01 23:45:15 +00:00
}
let mut curr = 0;
let len = info.mws.len();
2017-11-25 06:15:52 +00:00
loop {
2017-12-01 23:45:15 +00:00
resp = match info.mws[curr].response(info.req_mut(), resp) {
2018-01-10 06:48:35 +00:00
Err(err) => {
2017-12-01 23:45:15 +00:00
info.count = curr + 1;
2017-12-16 00:24:15 +00:00
return ProcessResponse::init(err.into())
2017-12-01 23:45:15 +00:00
}
2018-01-10 06:48:35 +00:00
Ok(Response::Done(r)) => {
2017-12-01 23:45:15 +00:00
curr += 1;
if curr == len {
2017-12-09 13:54:04 +00:00
return ProcessResponse::init(r)
2017-11-25 06:15:52 +00:00
} else {
r
}
},
2018-01-10 06:48:35 +00:00
Ok(Response::Future(fut)) => {
2017-12-01 23:45:15 +00:00
return PipelineState::RunMiddlewares(
2017-12-29 09:01:31 +00:00
RunMiddlewares { curr: curr, fut: Some(fut),
_s: PhantomData, _h: PhantomData })
2017-11-25 06:15:52 +00:00
},
};
}
}
2018-01-11 00:45:57 +00:00
fn poll(&mut self, info: &mut PipelineInfo<S>) -> Option<PipelineState<S, H>> {
2017-12-09 13:54:04 +00:00
let len = info.mws.len();
2017-12-01 23:45:15 +00:00
2017-11-25 06:15:52 +00:00
loop {
// poll latest fut
let mut resp = match self.fut.as_mut().unwrap().poll() {
2017-12-29 09:01:31 +00:00
Ok(Async::NotReady) => {
2018-01-11 00:45:57 +00:00
return None
2017-12-29 09:01:31 +00:00
}
2017-11-25 17:28:25 +00:00
Ok(Async::Ready(resp)) => {
2017-12-01 23:45:15 +00:00
self.curr += 1;
2017-12-16 00:24:15 +00:00
resp
2017-11-25 06:15:52 +00:00
}
2017-12-01 23:45:15 +00:00
Err(err) =>
2018-01-11 00:45:57 +00:00
return Some(ProcessResponse::init(err.into())),
2017-11-25 06:15:52 +00:00
};
loop {
2017-12-01 23:45:15 +00:00
if self.curr == len {
2018-01-11 00:45:57 +00:00
return Some(ProcessResponse::init(resp));
2017-11-25 06:15:52 +00:00
} else {
2017-12-09 13:54:04 +00:00
match info.mws[self.curr].response(info.req_mut(), resp) {
2018-01-10 06:48:35 +00:00
Err(err) =>
2018-01-11 00:45:57 +00:00
return Some(ProcessResponse::init(err.into())),
2018-01-10 06:48:35 +00:00
Ok(Response::Done(r)) => {
2017-12-01 23:45:15 +00:00
self.curr += 1;
2017-11-25 06:15:52 +00:00
resp = r
},
2018-01-10 06:48:35 +00:00
Ok(Response::Future(fut)) => {
2017-11-25 06:15:52 +00:00
self.fut = Some(fut);
break
},
}
}
}
}
}
}
2017-12-01 23:45:15 +00:00
2017-12-29 09:01:31 +00:00
struct ProcessResponse<S, H> {
2017-12-16 00:24:15 +00:00
resp: HttpResponse,
2017-12-01 23:45:15 +00:00
iostate: IOState,
running: RunningState,
2017-12-24 19:58:09 +00:00
drain: Option<oneshot::Sender<()>>,
2017-12-09 13:54:04 +00:00
_s: PhantomData<S>,
2017-12-29 09:01:31 +00:00
_h: PhantomData<H>,
2017-12-01 23:45:15 +00:00
}
#[derive(PartialEq)]
enum RunningState {
Running,
Paused,
Done,
}
impl RunningState {
#[inline]
fn pause(&mut self) {
if *self != RunningState::Done {
*self = RunningState::Paused
}
}
#[inline]
fn resume(&mut self) {
if *self != RunningState::Done {
*self = RunningState::Running
}
}
}
enum IOState {
Response,
Payload(BodyStream),
2018-01-01 01:26:32 +00:00
Actor(Box<ActorHttpContext>),
2017-12-01 23:45:15 +00:00
Done,
}
2018-01-10 04:00:18 +00:00
impl<S: 'static, H> ProcessResponse<S, H> {
2017-12-01 23:45:15 +00:00
2017-12-16 04:00:12 +00:00
#[inline]
2018-01-11 04:08:13 +00:00
fn init(resp: HttpResponse) -> PipelineState<S, H> {
2017-12-01 23:45:15 +00:00
PipelineState::Response(
ProcessResponse{ resp: resp,
iostate: IOState::Response,
running: RunningState::Running,
drain: None, _s: PhantomData, _h: PhantomData})
2017-12-01 23:45:15 +00:00
}
2017-12-09 13:54:04 +00:00
fn poll_io(mut self, io: &mut Writer, info: &mut PipelineInfo<S>)
2017-12-29 09:01:31 +00:00
-> Result<PipelineState<S, H>, PipelineState<S, H>>
2017-12-09 13:54:04 +00:00
{
2017-12-24 19:58:09 +00:00
if self.drain.is_none() && self.running != RunningState::Paused {
2017-12-01 23:45:15 +00:00
// if task is paused, write buffer is probably full
'outter: loop {
2017-12-01 23:45:15 +00:00
let result = match mem::replace(&mut self.iostate, IOState::Done) {
IOState::Response => {
2017-12-31 21:22:11 +00:00
let result = match io.start(info.req_mut().get_inner(), &mut self.resp) {
2017-12-01 23:45:15 +00:00
Ok(res) => res,
Err(err) => {
2017-12-09 13:54:04 +00:00
info.error = Some(err.into());
return Ok(FinishingMiddlewares::init(info, self.resp))
2017-12-01 23:45:15 +00:00
}
};
match self.resp.replace_body(Body::Empty) {
2018-01-01 01:26:32 +00:00
Body::Streaming(stream) =>
2017-12-01 23:45:15 +00:00
self.iostate = IOState::Payload(stream),
2018-01-01 01:26:32 +00:00
Body::Actor(ctx) =>
self.iostate = IOState::Actor(ctx),
2017-12-01 23:45:15 +00:00
_ => (),
}
result
},
IOState::Payload(mut body) => {
match body.poll() {
Ok(Async::Ready(None)) => {
self.iostate = IOState::Done;
if let Err(err) = io.write_eof() {
2017-12-09 13:54:04 +00:00
info.error = Some(err.into());
return Ok(FinishingMiddlewares::init(info, self.resp))
2017-12-01 23:45:15 +00:00
}
break
},
Ok(Async::Ready(Some(chunk))) => {
self.iostate = IOState::Payload(body);
2018-01-14 21:50:38 +00:00
match io.write(chunk.into()) {
2017-12-01 23:45:15 +00:00
Err(err) => {
2017-12-09 13:54:04 +00:00
info.error = Some(err.into());
return Ok(FinishingMiddlewares::init(info, self.resp))
2017-12-01 23:45:15 +00:00
},
Ok(result) => result
}
}
Ok(Async::NotReady) => {
self.iostate = IOState::Payload(body);
break
},
Err(err) => {
2017-12-09 13:54:04 +00:00
info.error = Some(err);
return Ok(FinishingMiddlewares::init(info, self.resp))
2017-12-01 23:45:15 +00:00
}
}
},
2018-01-01 01:26:32 +00:00
IOState::Actor(mut ctx) => {
if info.disconnected.take().is_some() {
ctx.disconnected();
}
2018-01-01 01:26:32 +00:00
match ctx.poll() {
Ok(Async::Ready(Some(vec))) => {
if vec.is_empty() {
self.iostate = IOState::Actor(ctx);
break
}
let mut res = None;
for frame in vec {
match frame {
Frame::Chunk(None) => {
info.context = Some(ctx);
self.iostate = IOState::Done;
if let Err(err) = io.write_eof() {
2017-12-09 13:54:04 +00:00
info.error = Some(err.into());
2018-01-11 00:45:57 +00:00
return Ok(
FinishingMiddlewares::init(info, self.resp))
}
break 'outter
},
Frame::Chunk(Some(chunk)) => {
2018-01-14 21:50:38 +00:00
match io.write(chunk) {
Err(err) => {
info.error = Some(err.into());
return Ok(
FinishingMiddlewares::init(info, self.resp))
},
Ok(result) => res = Some(result),
}
},
Frame::Drain(fut) =>
self.drain = Some(fut),
2017-12-01 23:45:15 +00:00
}
}
self.iostate = IOState::Actor(ctx);
if self.drain.is_some() {
self.running.resume();
break 'outter
}
res.unwrap()
2017-12-01 23:45:15 +00:00
},
Ok(Async::Ready(None)) => {
self.iostate = IOState::Done;
break
}
Ok(Async::NotReady) => {
2018-01-01 01:26:32 +00:00
self.iostate = IOState::Actor(ctx);
2017-12-01 23:45:15 +00:00
break
}
Err(err) => {
2017-12-09 13:54:04 +00:00
info.error = Some(err);
return Ok(FinishingMiddlewares::init(info, self.resp))
2017-12-01 23:45:15 +00:00
}
}
}
IOState::Done => break,
};
match result {
WriterState::Pause => {
self.running.pause();
break
}
WriterState::Done => {
self.running.resume()
},
}
}
}
2017-12-15 03:34:31 +00:00
// flush io but only if we need to
2017-12-24 19:58:09 +00:00
if self.running == RunningState::Paused || self.drain.is_some() {
match io.poll_completed(false) {
Ok(Async::Ready(_)) => {
self.running.resume();
// resolve drain futures
if let Some(tx) = self.drain.take() {
let _ = tx.send(());
}
// restart io processing
return self.poll_io(io, info);
},
2018-01-11 00:45:57 +00:00
Ok(Async::NotReady) => return Err(PipelineState::Response(self)),
2017-12-15 03:34:31 +00:00
Err(err) => {
debug!("Error sending data: {}", err);
info.error = Some(err.into());
return Ok(FinishingMiddlewares::init(info, self.resp))
}
2017-12-01 23:45:15 +00:00
}
}
// response is completed
2017-12-16 04:00:12 +00:00
match self.iostate {
IOState::Done => {
match io.write_eof() {
Ok(_) => (),
Err(err) => {
debug!("Error sending data: {}", err);
info.error = Some(err.into());
return Ok(FinishingMiddlewares::init(info, self.resp))
}
}
2017-12-16 04:00:12 +00:00
self.resp.set_response_size(io.written());
Ok(FinishingMiddlewares::init(info, self.resp))
}
2018-01-11 00:45:57 +00:00
_ => Err(PipelineState::Response(self)),
2017-12-01 23:45:15 +00:00
}
}
}
/// Middlewares start executor
2017-12-29 09:01:31 +00:00
struct FinishingMiddlewares<S, H> {
2017-12-16 00:24:15 +00:00
resp: HttpResponse,
2017-12-01 23:45:15 +00:00
fut: Option<Box<Future<Item=(), Error=Error>>>,
2017-12-09 13:54:04 +00:00
_s: PhantomData<S>,
2017-12-29 09:01:31 +00:00
_h: PhantomData<H>,
2017-12-01 23:45:15 +00:00
}
2018-01-10 04:00:18 +00:00
impl<S: 'static, H> FinishingMiddlewares<S, H> {
2017-12-01 23:45:15 +00:00
2017-12-29 09:01:31 +00:00
fn init(info: &mut PipelineInfo<S>, resp: HttpResponse) -> PipelineState<S, H> {
2017-12-01 23:45:15 +00:00
if info.count == 0 {
Completed::init(info)
} else {
2018-01-11 00:45:57 +00:00
let mut state = FinishingMiddlewares{resp: resp, fut: None,
_s: PhantomData, _h: PhantomData};
if let Some(st) = state.poll(info) {
st
} else {
PipelineState::Finishing(state)
2017-12-01 23:45:15 +00:00
}
}
}
2018-01-11 00:45:57 +00:00
fn poll(&mut self, info: &mut PipelineInfo<S>) -> Option<PipelineState<S, H>> {
2017-12-01 23:45:15 +00:00
loop {
// poll latest fut
let not_ready = if let Some(ref mut fut) = self.fut {
match fut.poll() {
Ok(Async::NotReady) => {
true
},
Ok(Async::Ready(())) => {
false
},
Err(err) => {
error!("Middleware finish error: {}", err);
false
}
}
} else {
false
};
if not_ready {
2018-01-11 00:45:57 +00:00
return None;
2017-12-01 23:45:15 +00:00
}
self.fut = None;
2017-12-09 13:54:04 +00:00
info.count -= 1;
2017-12-01 23:45:15 +00:00
2017-12-09 13:54:04 +00:00
match info.mws[info.count].finish(info.req_mut(), &self.resp) {
2017-12-01 23:45:15 +00:00
Finished::Done => {
2017-12-09 13:54:04 +00:00
if info.count == 0 {
2018-01-11 00:45:57 +00:00
return Some(Completed::init(info))
2017-12-01 23:45:15 +00:00
}
}
Finished::Future(fut) => {
self.fut = Some(fut);
},
}
}
}
}
#[derive(Debug)]
2017-12-29 09:01:31 +00:00
struct Completed<S, H>(PhantomData<S>, PhantomData<H>);
2017-12-01 23:45:15 +00:00
2017-12-29 09:01:31 +00:00
impl<S, H> Completed<S, H> {
2017-12-01 23:45:15 +00:00
2017-12-16 04:00:12 +00:00
#[inline]
2017-12-29 09:01:31 +00:00
fn init(info: &mut PipelineInfo<S>) -> PipelineState<S, H> {
2017-12-01 23:45:15 +00:00
if info.context.is_none() {
PipelineState::None
} else {
2017-12-29 09:01:31 +00:00
PipelineState::Completed(Completed(PhantomData, PhantomData))
2017-12-01 23:45:15 +00:00
}
}
2017-12-16 04:00:12 +00:00
#[inline]
2018-01-11 00:45:57 +00:00
fn poll(&mut self, info: &mut PipelineInfo<S>) -> Option<PipelineState<S, H>> {
2017-12-09 13:54:04 +00:00
match info.poll_context() {
2018-01-11 00:45:57 +00:00
Ok(Async::NotReady) => None,
Ok(Async::Ready(())) => Some(PipelineState::None),
Err(_) => Some(PipelineState::Error),
2017-12-01 23:45:15 +00:00
}
}
}
2017-12-02 00:10:01 +00:00
#[cfg(test)]
mod tests {
use super::*;
use actix::*;
use context::HttpContext;
2017-12-02 03:57:34 +00:00
use tokio_core::reactor::Core;
use futures::future::{lazy, result};
2017-12-02 00:10:01 +00:00
2017-12-29 09:01:31 +00:00
impl<S, H> PipelineState<S, H> {
2017-12-02 00:10:01 +00:00
fn is_none(&self) -> Option<bool> {
if let PipelineState::None = *self { Some(true) } else { None }
}
2017-12-29 09:01:31 +00:00
fn completed(self) -> Option<Completed<S, H>> {
2017-12-02 03:57:34 +00:00
if let PipelineState::Completed(c) = self { Some(c) } else { None }
2017-12-02 00:10:01 +00:00
}
}
struct MyActor;
impl Actor for MyActor {
type Context = HttpContext<MyActor>;
}
#[test]
fn test_completed() {
2017-12-02 03:57:34 +00:00
Core::new().unwrap().run(lazy(|| {
2017-12-09 13:54:04 +00:00
let mut info = PipelineInfo::new(HttpRequest::default());
2017-12-29 09:01:31 +00:00
Completed::<(), Inner<()>>::init(&mut info).is_none().unwrap();
2017-12-02 03:57:34 +00:00
let req = HttpRequest::default();
let mut ctx = HttpContext::new(req.clone(), MyActor);
let addr: Address<_> = ctx.address();
2017-12-09 13:54:04 +00:00
let mut info = PipelineInfo::new(req);
2017-12-16 02:49:11 +00:00
info.context = Some(Box::new(ctx));
2017-12-29 09:01:31 +00:00
let mut state = Completed::<(), Inner<()>>::init(&mut info).completed().unwrap();
2017-12-02 03:57:34 +00:00
2018-01-11 00:45:57 +00:00
assert!(state.poll(&mut info).is_none());
let pp = Pipeline(info, PipelineState::Completed(state));
2017-12-09 13:54:04 +00:00
assert!(!pp.is_done());
2017-12-02 03:57:34 +00:00
2017-12-09 13:54:04 +00:00
let Pipeline(mut info, st) = pp;
2018-01-11 00:45:57 +00:00
let mut st = st.completed().unwrap();
2017-12-02 03:57:34 +00:00
drop(addr);
2018-01-11 00:45:57 +00:00
assert!(st.poll(&mut info).unwrap().is_none().unwrap());
2017-12-02 03:57:34 +00:00
result(Ok::<_, ()>(()))
2018-01-11 00:45:57 +00:00
})).unwrap();
2017-12-02 00:10:01 +00:00
}
}