2019-11-15 15:54:11 +06:00
|
|
|
use std::future::Future;
|
|
|
|
use std::pin::Pin;
|
|
|
|
use std::task::{Context, Poll};
|
|
|
|
|
2019-04-10 12:24:17 -07:00
|
|
|
use actix_codec::{AsyncRead, AsyncWrite, Framed};
|
|
|
|
|
|
|
|
use crate::body::{BodySize, MessageBody, ResponseBody};
|
|
|
|
use crate::error::Error;
|
|
|
|
use crate::h1::{Codec, Message};
|
|
|
|
use crate::response::Response;
|
|
|
|
|
2020-08-24 10:13:35 +01:00
|
|
|
/// Send HTTP/1 response
|
2019-11-19 18:54:19 +06:00
|
|
|
#[pin_project::pin_project]
|
2019-04-10 12:24:17 -07:00
|
|
|
pub struct SendResponse<T, B> {
|
|
|
|
res: Option<Message<(Response<()>, BodySize)>>,
|
2020-01-29 11:15:13 +03:00
|
|
|
#[pin]
|
2019-04-10 12:24:17 -07:00
|
|
|
body: Option<ResponseBody<B>>,
|
2020-08-24 10:13:35 +01:00
|
|
|
#[pin]
|
2019-04-10 12:24:17 -07:00
|
|
|
framed: Option<Framed<T, Codec>>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<T, B> SendResponse<T, B>
|
|
|
|
where
|
|
|
|
B: MessageBody,
|
|
|
|
{
|
|
|
|
pub fn new(framed: Framed<T, Codec>, response: Response<B>) -> Self {
|
|
|
|
let (res, body) = response.into_parts();
|
|
|
|
|
|
|
|
SendResponse {
|
|
|
|
res: Some((res, body.size()).into()),
|
|
|
|
body: Some(body),
|
|
|
|
framed: Some(framed),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<T, B> Future for SendResponse<T, B>
|
|
|
|
where
|
2020-08-24 10:13:35 +01:00
|
|
|
T: AsyncRead + AsyncWrite + Unpin,
|
2020-02-03 22:55:49 +02:00
|
|
|
B: MessageBody + Unpin,
|
2019-04-10 12:24:17 -07:00
|
|
|
{
|
2019-11-15 15:54:11 +06:00
|
|
|
type Output = Result<Framed<T, Codec>, Error>;
|
|
|
|
|
2020-01-29 11:15:13 +03:00
|
|
|
// TODO: rethink if we need loops in polls
|
2020-08-24 10:13:35 +01:00
|
|
|
fn poll(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
|
|
|
|
let mut this = self.as_mut().project();
|
2019-04-10 12:24:17 -07:00
|
|
|
|
2020-01-29 11:15:13 +03:00
|
|
|
let mut body_done = this.body.is_none();
|
2019-04-10 12:24:17 -07:00
|
|
|
loop {
|
2020-01-29 11:15:13 +03:00
|
|
|
let mut body_ready = !body_done;
|
2019-04-10 12:24:17 -07:00
|
|
|
|
|
|
|
// send body
|
2020-01-29 11:15:13 +03:00
|
|
|
if this.res.is_none() && body_ready {
|
2020-08-24 10:13:35 +01:00
|
|
|
while body_ready
|
|
|
|
&& !body_done
|
|
|
|
&& !this
|
|
|
|
.framed
|
|
|
|
.as_ref()
|
|
|
|
.as_pin_ref()
|
|
|
|
.unwrap()
|
|
|
|
.is_write_buf_full()
|
|
|
|
{
|
2020-01-29 11:15:13 +03:00
|
|
|
match this.body.as_mut().as_pin_mut().unwrap().poll_next(cx)? {
|
2019-11-15 15:54:11 +06:00
|
|
|
Poll::Ready(item) => {
|
2020-01-29 11:15:13 +03:00
|
|
|
// body is done when item is None
|
|
|
|
body_done = item.is_none();
|
|
|
|
if body_done {
|
2019-11-15 15:54:11 +06:00
|
|
|
let _ = this.body.take();
|
2019-04-10 12:24:17 -07:00
|
|
|
}
|
2020-08-24 10:13:35 +01:00
|
|
|
let framed = this.framed.as_mut().as_pin_mut().unwrap();
|
2019-11-18 18:42:27 +06:00
|
|
|
framed.write(Message::Chunk(item))?;
|
2019-04-10 12:24:17 -07:00
|
|
|
}
|
2019-11-15 15:54:11 +06:00
|
|
|
Poll::Pending => body_ready = false,
|
2019-04-10 12:24:17 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-24 10:13:35 +01:00
|
|
|
let framed = this.framed.as_mut().as_pin_mut().unwrap();
|
|
|
|
|
2019-04-10 12:24:17 -07:00
|
|
|
// flush write buffer
|
2019-04-11 14:18:58 -07:00
|
|
|
if !framed.is_write_buf_empty() {
|
2019-11-15 15:54:11 +06:00
|
|
|
match framed.flush(cx)? {
|
|
|
|
Poll::Ready(_) => {
|
2019-04-10 12:24:17 -07:00
|
|
|
if body_ready {
|
|
|
|
continue;
|
|
|
|
} else {
|
2019-11-15 15:54:11 +06:00
|
|
|
return Poll::Pending;
|
2019-04-10 12:24:17 -07:00
|
|
|
}
|
|
|
|
}
|
2019-11-15 15:54:11 +06:00
|
|
|
Poll::Pending => return Poll::Pending,
|
2019-04-10 12:24:17 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// send response
|
2019-11-15 15:54:11 +06:00
|
|
|
if let Some(res) = this.res.take() {
|
2019-11-18 18:42:27 +06:00
|
|
|
framed.write(res)?;
|
2019-04-10 12:24:17 -07:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2020-02-10 16:19:48 +02:00
|
|
|
if !body_done {
|
2019-04-10 12:24:17 -07:00
|
|
|
if body_ready {
|
|
|
|
continue;
|
|
|
|
} else {
|
2019-11-15 15:54:11 +06:00
|
|
|
return Poll::Pending;
|
2019-04-10 12:24:17 -07:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2020-08-24 10:13:35 +01:00
|
|
|
|
|
|
|
let framed = this.framed.take().unwrap();
|
|
|
|
|
|
|
|
Poll::Ready(Ok(framed))
|
2019-04-10 12:24:17 -07:00
|
|
|
}
|
|
|
|
}
|