mirror of
https://github.com/actix/actix-web.git
synced 2025-01-04 06:18:51 +00:00
refactor keep-alive; fixed write to socket for upgraded connection
This commit is contained in:
parent
8169149554
commit
05f5ba0084
6 changed files with 187 additions and 181 deletions
244
src/server/h1.rs
244
src/server/h1.rs
|
@ -192,134 +192,112 @@ impl<T, H> Http1<T, H>
|
||||||
|
|
||||||
let retry = self.reader.need_read() == PayloadStatus::Read;
|
let retry = self.reader.need_read() == PayloadStatus::Read;
|
||||||
|
|
||||||
loop {
|
// check in-flight messages
|
||||||
// check in-flight messages
|
let mut io = false;
|
||||||
let mut io = false;
|
let mut idx = 0;
|
||||||
let mut idx = 0;
|
while idx < self.tasks.len() {
|
||||||
while idx < self.tasks.len() {
|
let item = &mut self.tasks[idx];
|
||||||
let item = &mut self.tasks[idx];
|
|
||||||
|
|
||||||
if !io && !item.flags.contains(EntryFlags::EOF) {
|
if !io && !item.flags.contains(EntryFlags::EOF) {
|
||||||
// io is corrupted, send buffer
|
// io is corrupted, send buffer
|
||||||
if item.flags.contains(EntryFlags::ERROR) {
|
if item.flags.contains(EntryFlags::ERROR) {
|
||||||
|
if let Ok(Async::NotReady) = self.stream.poll_completed(true) {
|
||||||
|
return Ok(Async::NotReady)
|
||||||
|
}
|
||||||
|
return Err(())
|
||||||
|
}
|
||||||
|
|
||||||
|
match item.pipe.poll_io(&mut self.stream) {
|
||||||
|
Ok(Async::Ready(ready)) => {
|
||||||
|
// override keep-alive state
|
||||||
|
if self.stream.keepalive() {
|
||||||
|
self.flags.insert(Flags::KEEPALIVE);
|
||||||
|
} else {
|
||||||
|
self.flags.remove(Flags::KEEPALIVE);
|
||||||
|
}
|
||||||
|
// prepare stream for next response
|
||||||
|
self.stream.reset();
|
||||||
|
|
||||||
|
if ready {
|
||||||
|
item.flags.insert(EntryFlags::EOF | EntryFlags::FINISHED);
|
||||||
|
} else {
|
||||||
|
item.flags.insert(EntryFlags::FINISHED);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
// no more IO for this iteration
|
||||||
|
Ok(Async::NotReady) => {
|
||||||
|
if self.reader.need_read() == PayloadStatus::Read && !retry {
|
||||||
|
return Ok(Async::Ready(true));
|
||||||
|
}
|
||||||
|
io = true;
|
||||||
|
}
|
||||||
|
Err(err) => {
|
||||||
|
// it is not possible to recover from error
|
||||||
|
// during pipe handling, so just drop connection
|
||||||
|
error!("Unhandled error: {}", err);
|
||||||
|
item.flags.insert(EntryFlags::ERROR);
|
||||||
|
|
||||||
|
// check stream state, we still can have valid data in buffer
|
||||||
if let Ok(Async::NotReady) = self.stream.poll_completed(true) {
|
if let Ok(Async::NotReady) = self.stream.poll_completed(true) {
|
||||||
return Ok(Async::NotReady)
|
return Ok(Async::NotReady)
|
||||||
}
|
}
|
||||||
return Err(())
|
return Err(())
|
||||||
}
|
}
|
||||||
|
}
|
||||||
match item.pipe.poll_io(&mut self.stream) {
|
} else if !item.flags.contains(EntryFlags::FINISHED) {
|
||||||
Ok(Async::Ready(ready)) => {
|
match item.pipe.poll() {
|
||||||
// override keep-alive state
|
Ok(Async::NotReady) => (),
|
||||||
if self.stream.keepalive() {
|
Ok(Async::Ready(_)) => item.flags.insert(EntryFlags::FINISHED),
|
||||||
self.flags.insert(Flags::KEEPALIVE);
|
Err(err) => {
|
||||||
} else {
|
item.flags.insert(EntryFlags::ERROR);
|
||||||
self.flags.remove(Flags::KEEPALIVE);
|
error!("Unhandled error: {}", err);
|
||||||
}
|
|
||||||
// prepare stream for next response
|
|
||||||
self.stream.reset();
|
|
||||||
|
|
||||||
if ready {
|
|
||||||
item.flags.insert(EntryFlags::EOF | EntryFlags::FINISHED);
|
|
||||||
} else {
|
|
||||||
item.flags.insert(EntryFlags::FINISHED);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
// no more IO for this iteration
|
|
||||||
Ok(Async::NotReady) => {
|
|
||||||
if self.reader.need_read() == PayloadStatus::Read && !retry {
|
|
||||||
return Ok(Async::Ready(true));
|
|
||||||
}
|
|
||||||
io = true;
|
|
||||||
}
|
|
||||||
Err(err) => {
|
|
||||||
// it is not possible to recover from error
|
|
||||||
// during pipe handling, so just drop connection
|
|
||||||
error!("Unhandled error: {}", err);
|
|
||||||
item.flags.insert(EntryFlags::ERROR);
|
|
||||||
|
|
||||||
// check stream state, we still can have valid data in buffer
|
|
||||||
if let Ok(Async::NotReady) = self.stream.poll_completed(true) {
|
|
||||||
return Ok(Async::NotReady)
|
|
||||||
}
|
|
||||||
return Err(())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else if !item.flags.contains(EntryFlags::FINISHED) {
|
|
||||||
match item.pipe.poll() {
|
|
||||||
Ok(Async::NotReady) => (),
|
|
||||||
Ok(Async::Ready(_)) => item.flags.insert(EntryFlags::FINISHED),
|
|
||||||
Err(err) => {
|
|
||||||
item.flags.insert(EntryFlags::ERROR);
|
|
||||||
error!("Unhandled error: {}", err);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
idx += 1;
|
|
||||||
}
|
}
|
||||||
|
idx += 1;
|
||||||
|
}
|
||||||
|
|
||||||
// cleanup finished tasks
|
// cleanup finished tasks
|
||||||
let mut popped = false;
|
let mut popped = false;
|
||||||
while !self.tasks.is_empty() {
|
while !self.tasks.is_empty() {
|
||||||
if self.tasks[0].flags.contains(EntryFlags::EOF | EntryFlags::FINISHED) {
|
if self.tasks[0].flags.contains(EntryFlags::EOF | EntryFlags::FINISHED) {
|
||||||
popped = true;
|
popped = true;
|
||||||
self.tasks.pop_front();
|
self.tasks.pop_front();
|
||||||
} else {
|
} else {
|
||||||
break
|
break
|
||||||
}
|
|
||||||
}
|
|
||||||
if need_read && popped {
|
|
||||||
return self.poll_io()
|
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
if need_read && popped {
|
||||||
|
return self.poll_io()
|
||||||
|
}
|
||||||
|
|
||||||
// no keep-alive
|
// check stream state
|
||||||
if !self.flags.contains(Flags::KEEPALIVE) && self.tasks.is_empty() {
|
if !self.poll_completed(true)? {
|
||||||
// check stream state
|
return Ok(Async::NotReady)
|
||||||
if !self.poll_completed(true)? {
|
}
|
||||||
return Ok(Async::NotReady)
|
|
||||||
}
|
// deal with keep-alive
|
||||||
|
if self.tasks.is_empty() {
|
||||||
|
// no keep-alive situations
|
||||||
|
if self.flags.contains(Flags::ERROR)
|
||||||
|
|| !self.flags.contains(Flags::KEEPALIVE)
|
||||||
|
|| !self.settings.keep_alive_enabled()
|
||||||
|
{
|
||||||
return Ok(Async::Ready(false))
|
return Ok(Async::Ready(false))
|
||||||
}
|
}
|
||||||
|
|
||||||
// start keep-alive timer, this also is slow request timeout
|
// start keep-alive timer
|
||||||
if self.tasks.is_empty() {
|
let keep_alive = self.settings.keep_alive();
|
||||||
// check stream state
|
if self.keepalive_timer.is_none() && keep_alive > 0 {
|
||||||
if self.flags.contains(Flags::ERROR) {
|
trace!("Start keep-alive timer");
|
||||||
return Ok(Async::Ready(false))
|
let mut timer = Timeout::new(
|
||||||
}
|
Duration::new(keep_alive, 0), Arbiter::handle()).unwrap();
|
||||||
|
// register timer
|
||||||
if self.settings.keep_alive_enabled() {
|
let _ = timer.poll();
|
||||||
let keep_alive = self.settings.keep_alive();
|
self.keepalive_timer = Some(timer);
|
||||||
if keep_alive > 0 && self.flags.contains(Flags::KEEPALIVE) {
|
|
||||||
if self.keepalive_timer.is_none() {
|
|
||||||
trace!("Start keep-alive timer");
|
|
||||||
let mut to = Timeout::new(
|
|
||||||
Duration::new(keep_alive, 0), Arbiter::handle()).unwrap();
|
|
||||||
// register timeout
|
|
||||||
let _ = to.poll();
|
|
||||||
self.keepalive_timer = Some(to);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// check stream state
|
|
||||||
if !self.poll_completed(true)? {
|
|
||||||
return Ok(Async::NotReady)
|
|
||||||
}
|
|
||||||
// keep-alive is disabled, drop connection
|
|
||||||
return Ok(Async::Ready(false))
|
|
||||||
}
|
|
||||||
} else if !self.poll_completed(false)? ||
|
|
||||||
self.flags.contains(Flags::KEEPALIVE) {
|
|
||||||
// check stream state or
|
|
||||||
// if keep-alive unset, rely on operating system
|
|
||||||
return Ok(Async::NotReady)
|
|
||||||
} else {
|
|
||||||
return Ok(Async::Ready(false))
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
self.poll_completed(false)?;
|
|
||||||
return Ok(Async::NotReady)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Ok(Async::NotReady)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -868,7 +846,7 @@ mod tests {
|
||||||
use httpmessage::HttpMessage;
|
use httpmessage::HttpMessage;
|
||||||
use application::HttpApplication;
|
use application::HttpApplication;
|
||||||
use server::settings::WorkerSettings;
|
use server::settings::WorkerSettings;
|
||||||
use server::IoStream;
|
use server::{IoStream, KeepAlive};
|
||||||
|
|
||||||
struct Buffer {
|
struct Buffer {
|
||||||
buf: Bytes,
|
buf: Bytes,
|
||||||
|
@ -939,7 +917,8 @@ mod tests {
|
||||||
|
|
||||||
macro_rules! parse_ready {
|
macro_rules! parse_ready {
|
||||||
($e:expr) => ({
|
($e:expr) => ({
|
||||||
let settings = WorkerSettings::<HttpApplication>::new(Vec::new(), None);
|
let settings = WorkerSettings::<HttpApplication>::new(
|
||||||
|
Vec::new(), KeepAlive::Os);
|
||||||
match Reader::new().parse($e, &mut BytesMut::new(), &settings) {
|
match Reader::new().parse($e, &mut BytesMut::new(), &settings) {
|
||||||
Ok(Async::Ready(req)) => req,
|
Ok(Async::Ready(req)) => req,
|
||||||
Ok(_) => panic!("Eof during parsing http request"),
|
Ok(_) => panic!("Eof during parsing http request"),
|
||||||
|
@ -961,7 +940,8 @@ mod tests {
|
||||||
macro_rules! expect_parse_err {
|
macro_rules! expect_parse_err {
|
||||||
($e:expr) => ({
|
($e:expr) => ({
|
||||||
let mut buf = BytesMut::new();
|
let mut buf = BytesMut::new();
|
||||||
let settings = WorkerSettings::<HttpApplication>::new(Vec::new(), None);
|
let settings = WorkerSettings::<HttpApplication>::new(
|
||||||
|
Vec::new(), KeepAlive::Os);
|
||||||
|
|
||||||
match Reader::new().parse($e, &mut buf, &settings) {
|
match Reader::new().parse($e, &mut buf, &settings) {
|
||||||
Err(err) => match err {
|
Err(err) => match err {
|
||||||
|
@ -979,7 +959,8 @@ mod tests {
|
||||||
fn test_parse() {
|
fn test_parse() {
|
||||||
let mut buf = Buffer::new("GET /test HTTP/1.1\r\n\r\n");
|
let mut buf = Buffer::new("GET /test HTTP/1.1\r\n\r\n");
|
||||||
let mut readbuf = BytesMut::new();
|
let mut readbuf = BytesMut::new();
|
||||||
let settings = WorkerSettings::<HttpApplication>::new(Vec::new(), None);
|
let settings = WorkerSettings::<HttpApplication>::new(
|
||||||
|
Vec::new(), KeepAlive::Os);
|
||||||
|
|
||||||
let mut reader = Reader::new();
|
let mut reader = Reader::new();
|
||||||
match reader.parse(&mut buf, &mut readbuf, &settings) {
|
match reader.parse(&mut buf, &mut readbuf, &settings) {
|
||||||
|
@ -996,7 +977,8 @@ mod tests {
|
||||||
fn test_parse_partial() {
|
fn test_parse_partial() {
|
||||||
let mut buf = Buffer::new("PUT /test HTTP/1");
|
let mut buf = Buffer::new("PUT /test HTTP/1");
|
||||||
let mut readbuf = BytesMut::new();
|
let mut readbuf = BytesMut::new();
|
||||||
let settings = WorkerSettings::<HttpApplication>::new(Vec::new(), None);
|
let settings = WorkerSettings::<HttpApplication>::new(
|
||||||
|
Vec::new(), KeepAlive::Os);
|
||||||
|
|
||||||
let mut reader = Reader::new();
|
let mut reader = Reader::new();
|
||||||
match reader.parse(&mut buf, &mut readbuf, &settings) {
|
match reader.parse(&mut buf, &mut readbuf, &settings) {
|
||||||
|
@ -1019,7 +1001,8 @@ mod tests {
|
||||||
fn test_parse_post() {
|
fn test_parse_post() {
|
||||||
let mut buf = Buffer::new("POST /test2 HTTP/1.0\r\n\r\n");
|
let mut buf = Buffer::new("POST /test2 HTTP/1.0\r\n\r\n");
|
||||||
let mut readbuf = BytesMut::new();
|
let mut readbuf = BytesMut::new();
|
||||||
let settings = WorkerSettings::<HttpApplication>::new(Vec::new(), None);
|
let settings = WorkerSettings::<HttpApplication>::new(
|
||||||
|
Vec::new(), KeepAlive::Os);
|
||||||
|
|
||||||
let mut reader = Reader::new();
|
let mut reader = Reader::new();
|
||||||
match reader.parse(&mut buf, &mut readbuf, &settings) {
|
match reader.parse(&mut buf, &mut readbuf, &settings) {
|
||||||
|
@ -1036,7 +1019,8 @@ mod tests {
|
||||||
fn test_parse_body() {
|
fn test_parse_body() {
|
||||||
let mut buf = Buffer::new("GET /test HTTP/1.1\r\nContent-Length: 4\r\n\r\nbody");
|
let mut buf = Buffer::new("GET /test HTTP/1.1\r\nContent-Length: 4\r\n\r\nbody");
|
||||||
let mut readbuf = BytesMut::new();
|
let mut readbuf = BytesMut::new();
|
||||||
let settings = WorkerSettings::<HttpApplication>::new(Vec::new(), None);
|
let settings = WorkerSettings::<HttpApplication>::new(
|
||||||
|
Vec::new(), KeepAlive::Os);
|
||||||
|
|
||||||
let mut reader = Reader::new();
|
let mut reader = Reader::new();
|
||||||
match reader.parse(&mut buf, &mut readbuf, &settings) {
|
match reader.parse(&mut buf, &mut readbuf, &settings) {
|
||||||
|
@ -1055,7 +1039,8 @@ mod tests {
|
||||||
let mut buf = Buffer::new(
|
let mut buf = Buffer::new(
|
||||||
"\r\nGET /test HTTP/1.1\r\nContent-Length: 4\r\n\r\nbody");
|
"\r\nGET /test HTTP/1.1\r\nContent-Length: 4\r\n\r\nbody");
|
||||||
let mut readbuf = BytesMut::new();
|
let mut readbuf = BytesMut::new();
|
||||||
let settings = WorkerSettings::<HttpApplication>::new(Vec::new(), None);
|
let settings = WorkerSettings::<HttpApplication>::new(
|
||||||
|
Vec::new(), KeepAlive::Os);
|
||||||
|
|
||||||
let mut reader = Reader::new();
|
let mut reader = Reader::new();
|
||||||
match reader.parse(&mut buf, &mut readbuf, &settings) {
|
match reader.parse(&mut buf, &mut readbuf, &settings) {
|
||||||
|
@ -1073,7 +1058,8 @@ mod tests {
|
||||||
fn test_parse_partial_eof() {
|
fn test_parse_partial_eof() {
|
||||||
let mut buf = Buffer::new("GET /test HTTP/1.1\r\n");
|
let mut buf = Buffer::new("GET /test HTTP/1.1\r\n");
|
||||||
let mut readbuf = BytesMut::new();
|
let mut readbuf = BytesMut::new();
|
||||||
let settings = WorkerSettings::<HttpApplication>::new(Vec::new(), None);
|
let settings = WorkerSettings::<HttpApplication>::new(
|
||||||
|
Vec::new(), KeepAlive::Os);
|
||||||
|
|
||||||
let mut reader = Reader::new();
|
let mut reader = Reader::new();
|
||||||
not_ready!{ reader.parse(&mut buf, &mut readbuf, &settings) }
|
not_ready!{ reader.parse(&mut buf, &mut readbuf, &settings) }
|
||||||
|
@ -1093,7 +1079,8 @@ mod tests {
|
||||||
fn test_headers_split_field() {
|
fn test_headers_split_field() {
|
||||||
let mut buf = Buffer::new("GET /test HTTP/1.1\r\n");
|
let mut buf = Buffer::new("GET /test HTTP/1.1\r\n");
|
||||||
let mut readbuf = BytesMut::new();
|
let mut readbuf = BytesMut::new();
|
||||||
let settings = WorkerSettings::<HttpApplication>::new(Vec::new(), None);
|
let settings = WorkerSettings::<HttpApplication>::new(
|
||||||
|
Vec::new(), KeepAlive::Os);
|
||||||
|
|
||||||
let mut reader = Reader::new();
|
let mut reader = Reader::new();
|
||||||
not_ready!{ reader.parse(&mut buf, &mut readbuf, &settings) }
|
not_ready!{ reader.parse(&mut buf, &mut readbuf, &settings) }
|
||||||
|
@ -1123,7 +1110,8 @@ mod tests {
|
||||||
Set-Cookie: c1=cookie1\r\n\
|
Set-Cookie: c1=cookie1\r\n\
|
||||||
Set-Cookie: c2=cookie2\r\n\r\n");
|
Set-Cookie: c2=cookie2\r\n\r\n");
|
||||||
let mut readbuf = BytesMut::new();
|
let mut readbuf = BytesMut::new();
|
||||||
let settings = WorkerSettings::<HttpApplication>::new(Vec::new(), None);
|
let settings = WorkerSettings::<HttpApplication>::new(
|
||||||
|
Vec::new(), KeepAlive::Os);
|
||||||
|
|
||||||
let mut reader = Reader::new();
|
let mut reader = Reader::new();
|
||||||
match reader.parse(&mut buf, &mut readbuf, &settings) {
|
match reader.parse(&mut buf, &mut readbuf, &settings) {
|
||||||
|
@ -1358,7 +1346,8 @@ mod tests {
|
||||||
"GET /test HTTP/1.1\r\n\
|
"GET /test HTTP/1.1\r\n\
|
||||||
transfer-encoding: chunked\r\n\r\n");
|
transfer-encoding: chunked\r\n\r\n");
|
||||||
let mut readbuf = BytesMut::new();
|
let mut readbuf = BytesMut::new();
|
||||||
let settings = WorkerSettings::<HttpApplication>::new(Vec::new(), None);
|
let settings = WorkerSettings::<HttpApplication>::new(
|
||||||
|
Vec::new(), KeepAlive::Os);
|
||||||
|
|
||||||
let mut reader = Reader::new();
|
let mut reader = Reader::new();
|
||||||
let mut req = reader_parse_ready!(reader.parse(&mut buf, &mut readbuf, &settings));
|
let mut req = reader_parse_ready!(reader.parse(&mut buf, &mut readbuf, &settings));
|
||||||
|
@ -1379,7 +1368,8 @@ mod tests {
|
||||||
"GET /test HTTP/1.1\r\n\
|
"GET /test HTTP/1.1\r\n\
|
||||||
transfer-encoding: chunked\r\n\r\n");
|
transfer-encoding: chunked\r\n\r\n");
|
||||||
let mut readbuf = BytesMut::new();
|
let mut readbuf = BytesMut::new();
|
||||||
let settings = WorkerSettings::<HttpApplication>::new(Vec::new(), None);
|
let settings = WorkerSettings::<HttpApplication>::new(
|
||||||
|
Vec::new(), KeepAlive::Os);
|
||||||
|
|
||||||
let mut reader = Reader::new();
|
let mut reader = Reader::new();
|
||||||
|
|
||||||
|
@ -1408,7 +1398,8 @@ mod tests {
|
||||||
"GET /test HTTP/1.1\r\n\
|
"GET /test HTTP/1.1\r\n\
|
||||||
transfer-encoding: chunked\r\n\r\n");
|
transfer-encoding: chunked\r\n\r\n");
|
||||||
let mut readbuf = BytesMut::new();
|
let mut readbuf = BytesMut::new();
|
||||||
let settings = WorkerSettings::<HttpApplication>::new(Vec::new(), None);
|
let settings = WorkerSettings::<HttpApplication>::new(
|
||||||
|
Vec::new(), KeepAlive::Os);
|
||||||
|
|
||||||
let mut reader = Reader::new();
|
let mut reader = Reader::new();
|
||||||
let mut req = reader_parse_ready!(reader.parse(&mut buf, &mut readbuf, &settings));
|
let mut req = reader_parse_ready!(reader.parse(&mut buf, &mut readbuf, &settings));
|
||||||
|
@ -1458,7 +1449,8 @@ mod tests {
|
||||||
"GET /test HTTP/1.1\r\n\
|
"GET /test HTTP/1.1\r\n\
|
||||||
transfer-encoding: chunked\r\n\r\n");
|
transfer-encoding: chunked\r\n\r\n");
|
||||||
let mut readbuf = BytesMut::new();
|
let mut readbuf = BytesMut::new();
|
||||||
let settings = WorkerSettings::<HttpApplication>::new(Vec::new(), None);
|
let settings = WorkerSettings::<HttpApplication>::new(
|
||||||
|
Vec::new(), KeepAlive::Os);
|
||||||
|
|
||||||
let mut reader = Reader::new();
|
let mut reader = Reader::new();
|
||||||
let mut req = reader_parse_ready!(reader.parse(&mut buf, &mut readbuf, &settings));
|
let mut req = reader_parse_ready!(reader.parse(&mut buf, &mut readbuf, &settings));
|
||||||
|
|
|
@ -68,27 +68,22 @@ impl<T: AsyncWrite> H1Writer<T> {
|
||||||
self.flags.contains(Flags::KEEPALIVE) && !self.flags.contains(Flags::UPGRADE)
|
self.flags.contains(Flags::KEEPALIVE) && !self.flags.contains(Flags::UPGRADE)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn write_to_stream(&mut self) -> io::Result<WriterState> {
|
fn write_data(&mut self, data: &[u8]) -> io::Result<(usize, bool)> {
|
||||||
while !self.buffer.is_empty() {
|
let mut written = 0;
|
||||||
match self.stream.write(self.buffer.as_ref()) {
|
while written < data.len() {
|
||||||
|
match self.stream.write(&data[written..]) {
|
||||||
Ok(0) => {
|
Ok(0) => {
|
||||||
self.disconnected();
|
self.disconnected();
|
||||||
return Ok(WriterState::Done);
|
return Ok((0, true));
|
||||||
},
|
|
||||||
Ok(n) => {
|
|
||||||
let _ = self.buffer.split_to(n);
|
|
||||||
},
|
},
|
||||||
|
Ok(n) => written += n,
|
||||||
Err(ref e) if e.kind() == io::ErrorKind::WouldBlock => {
|
Err(ref e) if e.kind() == io::ErrorKind::WouldBlock => {
|
||||||
if self.buffer.len() > self.buffer_capacity {
|
return Ok((written, false))
|
||||||
return Ok(WriterState::Pause)
|
|
||||||
} else {
|
|
||||||
return Ok(WriterState::Done)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
Err(err) => return Err(err),
|
Err(err) => return Err(err),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Ok(WriterState::Done)
|
Ok((written, false))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -216,18 +211,12 @@ impl<T: AsyncWrite> Writer for H1Writer<T> {
|
||||||
// shortcut for upgraded connection
|
// shortcut for upgraded connection
|
||||||
if self.flags.contains(Flags::UPGRADE) {
|
if self.flags.contains(Flags::UPGRADE) {
|
||||||
if self.buffer.is_empty() {
|
if self.buffer.is_empty() {
|
||||||
match self.stream.write(payload.as_ref()) {
|
match self.write_data(payload.as_ref())? {
|
||||||
Ok(0) => {
|
(_, true) => return Ok(WriterState::Done),
|
||||||
self.disconnected();
|
(n, false) => if payload.len() < n {
|
||||||
|
self.buffer.extend_from_slice(&payload.as_ref()[n..]);
|
||||||
return Ok(WriterState::Done);
|
return Ok(WriterState::Done);
|
||||||
},
|
|
||||||
Ok(n) => if payload.len() < n {
|
|
||||||
self.buffer.extend_from_slice(&payload.as_ref()[n..])
|
|
||||||
},
|
|
||||||
Err(ref e) if e.kind() == io::ErrorKind::WouldBlock => {
|
|
||||||
return Ok(WriterState::Done)
|
|
||||||
}
|
}
|
||||||
Err(err) => return Err(err),
|
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
self.buffer.extend(payload);
|
self.buffer.extend(payload);
|
||||||
|
@ -264,16 +253,22 @@ impl<T: AsyncWrite> Writer for H1Writer<T> {
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn poll_completed(&mut self, shutdown: bool) -> Poll<(), io::Error> {
|
fn poll_completed(&mut self, shutdown: bool) -> Poll<(), io::Error> {
|
||||||
match self.write_to_stream() {
|
if !self.buffer.is_empty() {
|
||||||
Ok(WriterState::Done) => {
|
let buf: &[u8] = unsafe{mem::transmute(self.buffer.as_ref())};
|
||||||
if shutdown {
|
match self.write_data(buf)? {
|
||||||
self.stream.shutdown()
|
(_, true) => (),
|
||||||
} else {
|
(n, false) => {
|
||||||
Ok(Async::Ready(()))
|
let _ = self.buffer.split_to(n);
|
||||||
|
if self.buffer.len() > self.buffer_capacity {
|
||||||
|
return Ok(Async::NotReady)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
Ok(WriterState::Pause) => Ok(Async::NotReady),
|
}
|
||||||
Err(err) => Err(err)
|
if shutdown {
|
||||||
|
self.stream.shutdown()
|
||||||
|
} else {
|
||||||
|
Ok(Async::Ready(()))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -31,6 +31,19 @@ use httpresponse::HttpResponse;
|
||||||
/// max buffer size 64k
|
/// max buffer size 64k
|
||||||
pub(crate) const MAX_WRITE_BUFFER_SIZE: usize = 65_536;
|
pub(crate) const MAX_WRITE_BUFFER_SIZE: usize = 65_536;
|
||||||
|
|
||||||
|
#[derive(Debug, PartialEq, Clone, Copy)]
|
||||||
|
/// Server keep-alive setting
|
||||||
|
pub enum KeepAlive {
|
||||||
|
/// Keep alive in seconds
|
||||||
|
Timeout(usize),
|
||||||
|
/// Use `SO_KEEPALIVE` socket option, value in seconds
|
||||||
|
Tcp(usize),
|
||||||
|
/// Relay on OS to shutdown tcp connection
|
||||||
|
Os,
|
||||||
|
/// Disabled
|
||||||
|
Disabled,
|
||||||
|
}
|
||||||
|
|
||||||
/// Pause accepting incoming connections
|
/// Pause accepting incoming connections
|
||||||
///
|
///
|
||||||
/// If socket contains some pending connection, they might be dropped.
|
/// If socket contains some pending connection, they might be dropped.
|
||||||
|
|
|
@ -5,6 +5,7 @@ use std::cell::{Cell, RefCell, RefMut, UnsafeCell};
|
||||||
use futures_cpupool::{Builder, CpuPool};
|
use futures_cpupool::{Builder, CpuPool};
|
||||||
|
|
||||||
use helpers;
|
use helpers;
|
||||||
|
use super::KeepAlive;
|
||||||
use super::channel::Node;
|
use super::channel::Node;
|
||||||
use super::shared::{SharedBytes, SharedBytesPool};
|
use super::shared::{SharedBytes, SharedBytesPool};
|
||||||
|
|
||||||
|
@ -97,8 +98,8 @@ impl ServerSettings {
|
||||||
|
|
||||||
pub(crate) struct WorkerSettings<H> {
|
pub(crate) struct WorkerSettings<H> {
|
||||||
h: RefCell<Vec<H>>,
|
h: RefCell<Vec<H>>,
|
||||||
enabled: bool,
|
|
||||||
keep_alive: u64,
|
keep_alive: u64,
|
||||||
|
ka_enabled: bool,
|
||||||
bytes: Rc<SharedBytesPool>,
|
bytes: Rc<SharedBytesPool>,
|
||||||
messages: Rc<helpers::SharedMessagePool>,
|
messages: Rc<helpers::SharedMessagePool>,
|
||||||
channels: Cell<usize>,
|
channels: Cell<usize>,
|
||||||
|
@ -106,11 +107,16 @@ pub(crate) struct WorkerSettings<H> {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<H> WorkerSettings<H> {
|
impl<H> WorkerSettings<H> {
|
||||||
pub(crate) fn new(h: Vec<H>, keep_alive: Option<u64>) -> WorkerSettings<H> {
|
pub(crate) fn new(h: Vec<H>, keep_alive: KeepAlive) -> WorkerSettings<H> {
|
||||||
|
let (keep_alive, ka_enabled) = match keep_alive {
|
||||||
|
KeepAlive::Timeout(val) => (val as u64, true),
|
||||||
|
KeepAlive::Os | KeepAlive::Tcp(_) => (0, true),
|
||||||
|
KeepAlive::Disabled => (0, false),
|
||||||
|
};
|
||||||
|
|
||||||
WorkerSettings {
|
WorkerSettings {
|
||||||
|
keep_alive, ka_enabled,
|
||||||
h: RefCell::new(h),
|
h: RefCell::new(h),
|
||||||
enabled: if let Some(ka) = keep_alive { ka > 0 } else { false },
|
|
||||||
keep_alive: keep_alive.unwrap_or(0),
|
|
||||||
bytes: Rc::new(SharedBytesPool::new()),
|
bytes: Rc::new(SharedBytesPool::new()),
|
||||||
messages: Rc::new(helpers::SharedMessagePool::new()),
|
messages: Rc::new(helpers::SharedMessagePool::new()),
|
||||||
channels: Cell::new(0),
|
channels: Cell::new(0),
|
||||||
|
@ -135,7 +141,7 @@ impl<H> WorkerSettings<H> {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn keep_alive_enabled(&self) -> bool {
|
pub fn keep_alive_enabled(&self) -> bool {
|
||||||
self.enabled
|
self.ka_enabled
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn get_shared_bytes(&self) -> SharedBytes {
|
pub fn get_shared_bytes(&self) -> SharedBytes {
|
||||||
|
|
|
@ -20,13 +20,12 @@ use native_tls::TlsAcceptor;
|
||||||
use openssl::ssl::{AlpnError, SslAcceptorBuilder};
|
use openssl::ssl::{AlpnError, SslAcceptorBuilder};
|
||||||
|
|
||||||
use helpers;
|
use helpers;
|
||||||
use super::{IntoHttpHandler, IoStream};
|
use super::{IntoHttpHandler, IoStream, KeepAlive};
|
||||||
use super::{PauseServer, ResumeServer, StopServer};
|
use super::{PauseServer, ResumeServer, StopServer};
|
||||||
use super::channel::{HttpChannel, WrapperStream};
|
use super::channel::{HttpChannel, WrapperStream};
|
||||||
use super::worker::{Conn, Worker, StreamHandlerType, StopWorker};
|
use super::worker::{Conn, Worker, StreamHandlerType, StopWorker};
|
||||||
use super::settings::{ServerSettings, WorkerSettings};
|
use super::settings::{ServerSettings, WorkerSettings};
|
||||||
|
|
||||||
|
|
||||||
/// An HTTP Server
|
/// An HTTP Server
|
||||||
pub struct HttpServer<H> where H: IntoHttpHandler + 'static
|
pub struct HttpServer<H> where H: IntoHttpHandler + 'static
|
||||||
{
|
{
|
||||||
|
@ -34,7 +33,7 @@ pub struct HttpServer<H> where H: IntoHttpHandler + 'static
|
||||||
threads: usize,
|
threads: usize,
|
||||||
backlog: i32,
|
backlog: i32,
|
||||||
host: Option<String>,
|
host: Option<String>,
|
||||||
keep_alive: Option<u64>,
|
keep_alive: KeepAlive,
|
||||||
factory: Arc<Fn() -> Vec<H> + Send + Sync>,
|
factory: Arc<Fn() -> Vec<H> + Send + Sync>,
|
||||||
#[cfg_attr(feature="cargo-clippy", allow(type_complexity))]
|
#[cfg_attr(feature="cargo-clippy", allow(type_complexity))]
|
||||||
workers: Vec<(usize, Addr<Syn, Worker<H::Handler>>)>,
|
workers: Vec<(usize, Addr<Syn, Worker<H::Handler>>)>,
|
||||||
|
@ -83,7 +82,7 @@ impl<H> HttpServer<H> where H: IntoHttpHandler + 'static
|
||||||
threads: num_cpus::get(),
|
threads: num_cpus::get(),
|
||||||
backlog: 2048,
|
backlog: 2048,
|
||||||
host: None,
|
host: None,
|
||||||
keep_alive: None,
|
keep_alive: KeepAlive::Os,
|
||||||
factory: Arc::new(f),
|
factory: Arc::new(f),
|
||||||
workers: Vec::new(),
|
workers: Vec::new(),
|
||||||
sockets: HashMap::new(),
|
sockets: HashMap::new(),
|
||||||
|
@ -124,14 +123,8 @@ impl<H> HttpServer<H> where H: IntoHttpHandler + 'static
|
||||||
|
|
||||||
/// Set server keep-alive setting.
|
/// Set server keep-alive setting.
|
||||||
///
|
///
|
||||||
/// By default keep alive is enabled.
|
/// By default keep alive is set to a `Os`.
|
||||||
///
|
pub fn keep_alive(mut self, val: KeepAlive) -> Self {
|
||||||
/// - `Some(75)` - enable
|
|
||||||
///
|
|
||||||
/// - `Some(0)` - disable
|
|
||||||
///
|
|
||||||
/// - `None` - use `SO_KEEPALIVE` socket option
|
|
||||||
pub fn keep_alive(mut self, val: Option<u64>) -> Self {
|
|
||||||
self.keep_alive = val;
|
self.keep_alive = val;
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
|
@ -23,7 +23,7 @@ use actix::*;
|
||||||
use actix::msgs::StopArbiter;
|
use actix::msgs::StopArbiter;
|
||||||
|
|
||||||
use helpers;
|
use helpers;
|
||||||
use server::HttpHandler;
|
use server::{HttpHandler, KeepAlive};
|
||||||
use server::channel::HttpChannel;
|
use server::channel::HttpChannel;
|
||||||
use server::settings::WorkerSettings;
|
use server::settings::WorkerSettings;
|
||||||
|
|
||||||
|
@ -48,21 +48,30 @@ impl Message for StopWorker {
|
||||||
/// Http worker
|
/// Http worker
|
||||||
///
|
///
|
||||||
/// Worker accepts Socket objects via unbounded channel and start requests processing.
|
/// Worker accepts Socket objects via unbounded channel and start requests processing.
|
||||||
pub(crate) struct Worker<H> where H: HttpHandler + 'static {
|
pub(crate)
|
||||||
|
struct Worker<H> where H: HttpHandler + 'static {
|
||||||
settings: Rc<WorkerSettings<H>>,
|
settings: Rc<WorkerSettings<H>>,
|
||||||
hnd: Handle,
|
hnd: Handle,
|
||||||
handler: StreamHandlerType,
|
handler: StreamHandlerType,
|
||||||
|
tcp_ka: Option<time::Duration>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<H: HttpHandler + 'static> Worker<H> {
|
impl<H: HttpHandler + 'static> Worker<H> {
|
||||||
|
|
||||||
pub(crate) fn new(h: Vec<H>, handler: StreamHandlerType, keep_alive: Option<u64>)
|
pub(crate) fn new(h: Vec<H>, handler: StreamHandlerType, keep_alive: KeepAlive)
|
||||||
-> Worker<H>
|
-> Worker<H>
|
||||||
{
|
{
|
||||||
|
let tcp_ka = if let KeepAlive::Tcp(val) = keep_alive {
|
||||||
|
Some(time::Duration::new(val as u64, 0))
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
};
|
||||||
|
|
||||||
Worker {
|
Worker {
|
||||||
settings: Rc::new(WorkerSettings::new(h, keep_alive)),
|
settings: Rc::new(WorkerSettings::new(h, keep_alive)),
|
||||||
hnd: Arbiter::handle().clone(),
|
hnd: Arbiter::handle().clone(),
|
||||||
handler,
|
handler,
|
||||||
|
tcp_ka,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -106,9 +115,7 @@ impl<H> Handler<Conn<net::TcpStream>> for Worker<H>
|
||||||
|
|
||||||
fn handle(&mut self, msg: Conn<net::TcpStream>, _: &mut Context<Self>)
|
fn handle(&mut self, msg: Conn<net::TcpStream>, _: &mut Context<Self>)
|
||||||
{
|
{
|
||||||
if !self.settings.keep_alive_enabled() &&
|
if self.tcp_ka.is_some() && msg.io.set_keepalive(self.tcp_ka).is_err() {
|
||||||
msg.io.set_keepalive(Some(time::Duration::new(75, 0))).is_err()
|
|
||||||
{
|
|
||||||
error!("Can not set socket keep-alive option");
|
error!("Can not set socket keep-alive option");
|
||||||
}
|
}
|
||||||
self.handler.handle(Rc::clone(&self.settings), &self.hnd, msg);
|
self.handler.handle(Rc::clone(&self.settings), &self.hnd, msg);
|
||||||
|
|
Loading…
Reference in a new issue