1
0
mirror of https://github.com/actix/actix-extras.git synced 2024-11-25 00:12:59 +01:00
actix-extras/src/client/pipeline.rs

389 lines
13 KiB
Rust
Raw Normal View History

2018-02-19 12:11:11 +01:00
use std::{io, mem};
use bytes::{Bytes, BytesMut};
2018-02-24 05:29:35 +01:00
use http::header::CONTENT_ENCODING;
2018-02-19 12:11:11 +01:00
use futures::{Async, Future, Poll};
2018-02-20 07:48:27 +01:00
use futures::unsync::oneshot;
2018-02-19 12:11:11 +01:00
use actix::prelude::*;
2018-02-20 07:48:27 +01:00
use error::Error;
use body::{Body, BodyStream};
use context::{Frame, ActorHttpContext};
2018-02-24 05:29:35 +01:00
use headers::ContentEncoding;
2018-02-19 12:11:11 +01:00
use error::PayloadError;
2018-02-20 07:48:27 +01:00
use server::WriterState;
2018-02-19 12:11:11 +01:00
use server::shared::SharedBytes;
2018-02-25 09:43:00 +01:00
use server::encoding::PayloadStream;
2018-02-19 12:11:11 +01:00
use super::{ClientRequest, ClientResponse};
use super::{Connect, Connection, ClientConnector, ClientConnectorError};
use super::HttpClientWriter;
use super::{HttpResponseParser, HttpResponseParserError};
2018-02-19 22:18:18 +01:00
/// A set of errors that can occur during sending request and reading response
#[derive(Fail, Debug)]
2018-02-19 12:11:11 +01:00
pub enum SendRequestError {
2018-02-19 22:18:18 +01:00
/// Failed to connect to host
#[fail(display="Failed to connect to host: {}", _0)]
Connector(#[cause] ClientConnectorError),
/// Error parsing response
#[fail(display="{}", _0)]
ParseError(#[cause] HttpResponseParserError),
/// Error reading response payload
#[fail(display="Error reading response payload: {}", _0)]
Io(#[cause] io::Error),
2018-02-19 12:11:11 +01:00
}
impl From<io::Error> for SendRequestError {
fn from(err: io::Error) -> SendRequestError {
SendRequestError::Io(err)
}
}
enum State {
New,
Connect(actix::dev::Request<Unsync, ClientConnector, Connect>),
Connection(Connection),
2018-02-19 12:11:11 +01:00
Send(Box<Pipeline>),
None,
}
/// `SendRequest` is a `Future` which represents asynchronous request sending process.
2018-02-20 07:48:27 +01:00
#[must_use = "SendRequest does nothing unless polled"]
2018-02-19 12:11:11 +01:00
pub struct SendRequest {
req: ClientRequest,
state: State,
conn: Addr<Unsync, ClientConnector>,
2018-02-19 12:11:11 +01:00
}
impl SendRequest {
pub(crate) fn new(req: ClientRequest) -> SendRequest {
SendRequest::with_connector(req, ClientConnector::from_registry())
}
pub(crate) fn with_connector(req: ClientRequest, conn: Addr<Unsync, ClientConnector>)
-> SendRequest
{
2018-02-19 12:11:11 +01:00
SendRequest{
req: req,
state: State::New,
conn: conn}
2018-02-19 12:11:11 +01:00
}
pub(crate) fn with_connection(req: ClientRequest, conn: Connection) -> SendRequest
{
SendRequest{
req: req,
state: State::Connection(conn),
conn: ClientConnector::from_registry()}
}
2018-02-19 12:11:11 +01:00
}
impl Future for SendRequest {
type Item = ClientResponse;
type Error = SendRequestError;
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
loop {
let state = mem::replace(&mut self.state, State::None);
match state {
State::New =>
self.state = State::Connect(self.conn.send(Connect(self.req.uri().clone()))),
2018-02-19 12:11:11 +01:00
State::Connect(mut conn) => match conn.poll() {
Ok(Async::NotReady) => {
self.state = State::Connect(conn);
return Ok(Async::NotReady);
},
Ok(Async::Ready(result)) => match result {
Ok(stream) => {
self.state = State::Connection(stream)
2018-02-19 12:11:11 +01:00
},
Err(err) => return Err(SendRequestError::Connector(err)),
},
Err(_) => return Err(SendRequestError::Connector(
ClientConnectorError::Disconnected))
},
State::Connection(stream) => {
let mut writer = HttpClientWriter::new(SharedBytes::default());
writer.start(&mut self.req)?;
let body = match self.req.replace_body(Body::Empty) {
Body::Streaming(stream) => IoBody::Payload(stream),
Body::Actor(ctx) => IoBody::Actor(ctx),
_ => IoBody::Done,
};
let mut pl = Box::new(Pipeline {
body: body,
conn: stream,
writer: writer,
2018-02-24 05:29:35 +01:00
parser: Some(HttpResponseParser::default()),
parser_buf: BytesMut::new(),
disconnected: false,
drain: None,
2018-02-24 05:29:35 +01:00
decompress: None,
should_decompress: self.req.response_decompress(),
write_state: RunningState::Running,
});
self.state = State::Send(pl);
2018-02-19 12:11:11 +01:00
},
State::Send(mut pl) => {
2018-02-20 07:48:27 +01:00
pl.poll_write()
.map_err(|e| io::Error::new(
io::ErrorKind::Other, format!("{}", e).as_str()))?;
2018-02-19 12:11:11 +01:00
match pl.parse() {
Ok(Async::Ready(mut resp)) => {
resp.set_pipeline(pl);
return Ok(Async::Ready(resp))
},
Ok(Async::NotReady) => {
self.state = State::Send(pl);
return Ok(Async::NotReady)
},
Err(err) => return Err(SendRequestError::ParseError(err))
}
}
State::None => unreachable!(),
}
}
}
}
pub(crate) struct Pipeline {
2018-02-20 07:48:27 +01:00
body: IoBody,
2018-02-19 12:11:11 +01:00
conn: Connection,
writer: HttpClientWriter,
2018-02-24 05:29:35 +01:00
parser: Option<HttpResponseParser>,
2018-02-19 12:11:11 +01:00
parser_buf: BytesMut,
2018-02-20 07:48:27 +01:00
disconnected: bool,
drain: Option<oneshot::Sender<()>>,
2018-02-24 05:29:35 +01:00
decompress: Option<PayloadStream>,
should_decompress: bool,
write_state: RunningState,
2018-02-20 07:48:27 +01:00
}
enum IoBody {
Payload(BodyStream),
Actor(Box<ActorHttpContext>),
Done,
}
2018-02-24 05:29:35 +01:00
#[derive(Debug, PartialEq)]
2018-02-20 07:48:27 +01:00
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
}
}
2018-02-19 12:11:11 +01:00
}
impl Pipeline {
#[inline]
pub fn parse(&mut self) -> Poll<ClientResponse, HttpResponseParserError> {
2018-02-24 05:29:35 +01:00
match self.parser.as_mut().unwrap().parse(&mut self.conn, &mut self.parser_buf) {
Ok(Async::Ready(resp)) => {
// check content-encoding
if self.should_decompress {
if let Some(enc) = resp.headers().get(CONTENT_ENCODING) {
if let Ok(enc) = enc.to_str() {
match ContentEncoding::from(enc) {
ContentEncoding::Auto | ContentEncoding::Identity => (),
enc => self.decompress = Some(PayloadStream::new(enc)),
}
}
}
}
Ok(Async::Ready(resp))
}
val => val,
}
2018-02-19 12:11:11 +01:00
}
#[inline]
pub fn poll(&mut self) -> Poll<Option<Bytes>, PayloadError> {
2018-02-24 05:29:35 +01:00
let mut need_run = false;
// need write?
match self.poll_write()
.map_err(|e| io::Error::new(io::ErrorKind::Other, format!("{}", e)))?
{
Async::NotReady => need_run = true,
_ => (),
}
// need read?
if self.parser.is_some() {
loop {
match self.parser.as_mut().unwrap()
.parse_payload(&mut self.conn, &mut self.parser_buf)?
{
Async::Ready(Some(b)) => {
if let Some(ref mut decompress) = self.decompress {
match decompress.feed_data(b) {
Ok(Some(b)) => return Ok(Async::Ready(Some(b))),
Ok(None) => return Ok(Async::NotReady),
Err(ref err) if err.kind() == io::ErrorKind::WouldBlock =>
continue,
Err(err) => return Err(err.into()),
}
} else {
return Ok(Async::Ready(Some(b)))
}
},
Async::Ready(None) => {
let _ = self.parser.take();
break
}
Async::NotReady => return Ok(Async::NotReady),
}
}
}
// eof
if let Some(mut decompress) = self.decompress.take() {
let res = decompress.feed_eof();
if let Some(b) = res? {
return Ok(Async::Ready(Some(b)))
}
}
if need_run {
Ok(Async::NotReady)
} else {
Ok(Async::Ready(None))
}
2018-02-19 12:11:11 +01:00
}
#[inline]
2018-02-20 07:48:27 +01:00
pub fn poll_write(&mut self) -> Poll<(), Error> {
2018-02-24 05:29:35 +01:00
if self.write_state == RunningState::Done {
2018-02-20 07:48:27 +01:00
return Ok(Async::Ready(()))
}
let mut done = false;
2018-02-24 05:29:35 +01:00
if self.drain.is_none() && self.write_state != RunningState::Paused {
2018-02-20 07:48:27 +01:00
'outter: loop {
let result = match mem::replace(&mut self.body, IoBody::Done) {
IoBody::Payload(mut body) => {
match body.poll()? {
Async::Ready(None) => {
self.writer.write_eof()?;
self.disconnected = true;
break
},
Async::Ready(Some(chunk)) => {
self.body = IoBody::Payload(body);
self.writer.write(chunk.into())?
}
Async::NotReady => {
done = true;
self.body = IoBody::Payload(body);
break
},
}
},
IoBody::Actor(mut ctx) => {
if self.disconnected {
ctx.disconnected();
}
match ctx.poll()? {
Async::Ready(Some(vec)) => {
if vec.is_empty() {
self.body = IoBody::Actor(ctx);
break
}
let mut res = None;
for frame in vec {
match frame {
Frame::Chunk(None) => {
// info.context = Some(ctx);
2018-02-24 05:29:35 +01:00
self.disconnected = true;
2018-02-20 07:48:27 +01:00
self.writer.write_eof()?;
break 'outter
},
Frame::Chunk(Some(chunk)) =>
res = Some(self.writer.write(chunk)?),
Frame::Drain(fut) => self.drain = Some(fut),
}
}
self.body = IoBody::Actor(ctx);
if self.drain.is_some() {
2018-02-24 05:29:35 +01:00
self.write_state.resume();
2018-02-20 07:48:27 +01:00
break
}
res.unwrap()
},
Async::Ready(None) => {
done = true;
break
}
Async::NotReady => {
done = true;
self.body = IoBody::Actor(ctx);
break
}
}
},
IoBody::Done => {
2018-02-24 05:29:35 +01:00
self.disconnected = true;
2018-02-20 07:48:27 +01:00
done = true;
break
}
};
match result {
WriterState::Pause => {
2018-02-24 05:29:35 +01:00
self.write_state.pause();
2018-02-20 07:48:27 +01:00
break
}
WriterState::Done => {
2018-02-24 05:29:35 +01:00
self.write_state.resume()
2018-02-20 07:48:27 +01:00
},
}
}
}
// flush io but only if we need to
match self.writer.poll_completed(&mut self.conn, false) {
Ok(Async::Ready(_)) => {
2018-02-24 05:29:35 +01:00
if self.disconnected {
self.write_state = RunningState::Done;
} else {
self.write_state.resume();
}
2018-02-20 07:48:27 +01:00
// resolve drain futures
if let Some(tx) = self.drain.take() {
let _ = tx.send(());
}
// restart io processing
2018-02-24 05:29:35 +01:00
if !done || self.write_state == RunningState::Done {
2018-02-20 07:48:27 +01:00
self.poll_write()
} else {
Ok(Async::NotReady)
}
},
Ok(Async::NotReady) => Ok(Async::NotReady),
Err(err) => Err(err.into()),
}
2018-02-19 12:11:11 +01:00
}
}