Skip to content

Commit

Permalink
feat(server): add http1_half_close(bool) option
Browse files Browse the repository at this point in the history
This option determines whether a read EOF should close the connection
automatically. The behavior was to always allow read EOF while waiting
to respond, so this option has a default of `true`.

Setting this option to `false` will allow Service futures to be canceled
as soon as disconnect is noticed.

Closes #1716
  • Loading branch information
seanmonstar committed Nov 27, 2018
1 parent 69368f4 commit 8c8c1f4
Show file tree
Hide file tree
Showing 6 changed files with 121 additions and 14 deletions.
27 changes: 22 additions & 5 deletions src/proto/h1/conn.rs
Original file line number Diff line number Diff line change
Expand Up @@ -38,6 +38,7 @@ where I: AsyncRead + AsyncWrite,
Conn {
io: Buffered::new(io),
state: State {
allow_half_close: true,
cached_headers: None,
error: None,
keep_alive: KA::Busy,
Expand Down Expand Up @@ -75,6 +76,10 @@ where I: AsyncRead + AsyncWrite,
self.state.title_case_headers = true;
}

pub(crate) fn set_disable_half_close(&mut self) {
self.state.allow_half_close = false;
}

pub fn into_inner(self) -> (I, Bytes) {
self.io.into_inner()
}
Expand Down Expand Up @@ -117,7 +122,7 @@ where I: AsyncRead + AsyncWrite,

fn should_error_on_eof(&self) -> bool {
// If we're idle, it's probably just the connection closing gracefully.
T::should_error_on_parse_eof() && !self.state.is_idle()
!self.state.is_idle()
}

fn has_h2_prefix(&self) -> bool {
Expand Down Expand Up @@ -228,10 +233,11 @@ where I: AsyncRead + AsyncWrite,

trace!("read_keep_alive; is_mid_message={}", self.is_mid_message());

if !self.is_mid_message() {
self.require_empty_read().map_err(::Error::new_io)?;
if self.is_mid_message() {
self.mid_message_keep_alive().map_err(::Error::new_io)
} else {
self.require_empty_read().map_err(::Error::new_io)
}
Ok(())
}

fn is_mid_message(&self) -> bool {
Expand All @@ -252,7 +258,7 @@ where I: AsyncRead + AsyncWrite,
// This should only be called for Clients wanting to enter the idle
// state.
fn require_empty_read(&mut self) -> io::Result<()> {
assert!(!self.can_read_head() && !self.can_read_body());
debug_assert!(!self.can_read_head() && !self.can_read_body());

if !self.io.read_buf().is_empty() {
debug!("received an unexpected {} bytes", self.io.read_buf().len());
Expand All @@ -279,6 +285,16 @@ where I: AsyncRead + AsyncWrite,
}
}

fn mid_message_keep_alive(&mut self) -> io::Result<()> {
debug_assert!(!self.can_read_head() && !self.can_read_body());

if self.state.allow_half_close || !self.io.read_buf().is_empty() {
Ok(())
} else {
self.try_io_read().map(|_| ())
}
}

fn try_io_read(&mut self) -> Poll<usize, io::Error> {
match self.io.read_from_io() {
Ok(Async::Ready(0)) => {
Expand Down Expand Up @@ -655,6 +671,7 @@ impl<I, B: Buf, T> fmt::Debug for Conn<I, B, T> {
}

struct State {
allow_half_close: bool,
/// Re-usable HeaderMap to reduce allocating new ones.
cached_headers: Option<HeaderMap>,
/// If an error occurs when there wasn't a direct way to return it
Expand Down
1 change: 0 additions & 1 deletion src/proto/h1/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -31,7 +31,6 @@ pub(crate) trait Http1Transaction {

fn on_error(err: &::Error) -> Option<MessageHead<Self::Outgoing>>;

fn should_error_on_parse_eof() -> bool;
fn should_read_first() -> bool;

fn update_date() {}
Expand Down
8 changes: 0 additions & 8 deletions src/proto/h1/role.rs
Original file line number Diff line number Diff line change
Expand Up @@ -527,10 +527,6 @@ impl Http1Transaction for Server {
Some(msg)
}

fn should_error_on_parse_eof() -> bool {
false
}

fn should_read_first() -> bool {
true
}
Expand Down Expand Up @@ -687,10 +683,6 @@ impl Http1Transaction for Client {
None
}

fn should_error_on_parse_eof() -> bool {
true
}

fn should_read_first() -> bool {
false
}
Expand Down
20 changes: 20 additions & 0 deletions src/server/conn.rs
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,7 @@ pub(super) use self::upgrades::UpgradeableConnection;
#[derive(Clone, Debug)]
pub struct Http<E = Exec> {
exec: E,
h1_half_close: bool,
h1_writev: bool,
mode: ConnectionMode,
keep_alive: bool,
Expand Down Expand Up @@ -163,6 +164,7 @@ impl Http {
pub fn new() -> Http {
Http {
exec: Exec::Default,
h1_half_close: true,
h1_writev: true,
mode: ConnectionMode::Fallback,
keep_alive: true,
Expand Down Expand Up @@ -195,6 +197,20 @@ impl<E> Http<E> {
self
}

/// Set whether HTTP/1 connections should support half-closures.
///
/// Clients can chose to shutdown their write-side while waiting
/// for the server to respond. Setting this to `false` will
/// automatically close any connection immediately if `read`
/// detects an EOF.
///
/// Default is `true`.
#[inline]
pub fn http1_half_close(&mut self, val: bool) -> &mut Self {
self.h1_half_close = val;
self
}

/// Set whether HTTP/1 connections should try to use vectored writes,
/// or always flatten into a single buffer.
///
Expand Down Expand Up @@ -261,6 +277,7 @@ impl<E> Http<E> {
pub fn with_executor<E2>(self, exec: E2) -> Http<E2> {
Http {
exec,
h1_half_close: self.h1_half_close,
h1_writev: self.h1_writev,
mode: self.mode,
keep_alive: self.keep_alive,
Expand Down Expand Up @@ -319,6 +336,9 @@ impl<E> Http<E> {
if !self.keep_alive {
conn.disable_keep_alive();
}
if !self.h1_half_close {
conn.set_disable_half_close();
}
if !self.h1_writev {
conn.set_write_strategy_flatten();
}
Expand Down
14 changes: 14 additions & 0 deletions src/server/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -247,6 +247,20 @@ impl<I, E> Builder<I, E> {
self
}


/// Set whether HTTP/1 connections should support half-closures.
///
/// Clients can chose to shutdown their write-side while waiting
/// for the server to respond. Setting this to `false` will
/// automatically close any connection immediately if `read`
/// detects an EOF.
///
/// Default is `true`.
pub fn http1_half_close(mut self, val: bool) -> Self {
self.protocol.http1_half_close(val);
self
}

/// Sets whether HTTP/1 is required.
///
/// Default is `false`.
Expand Down
65 changes: 65 additions & 0 deletions tests/server.rs
Original file line number Diff line number Diff line change
Expand Up @@ -991,6 +991,71 @@ fn nonempty_parse_eof_returns_error() {
rt.block_on(fut).expect_err("partial parse eof is error");
}

#[test]
fn socket_half_closed() {
let _ = pretty_env_logger::try_init();
let mut rt = Runtime::new().unwrap();
let listener = tcp_bind(&"127.0.0.1:0".parse().unwrap()).unwrap();
let addr = listener.local_addr().unwrap();

thread::spawn(move || {
let mut tcp = connect(&addr);
tcp.write_all(b"GET / HTTP/1.1\r\n\r\n").unwrap();
tcp.shutdown(::std::net::Shutdown::Write).expect("SHDN_WR");

let mut buf = [0; 256];
tcp.read(&mut buf).unwrap();
let expected = "HTTP/1.1 200 OK\r\n";
assert_eq!(s(&buf[..expected.len()]), expected);
});

let fut = listener.incoming()
.into_future()
.map_err(|_| unreachable!())
.and_then(|(item, _incoming)| {
let socket = item.unwrap();
Http::new()
.serve_connection(socket, service_fn(|_| {
Delay::new(Duration::from_millis(500))
.map(|_| {
Response::new(Body::empty())
})
}))
});

rt.block_on(fut).unwrap();
}

#[test]
fn disconnect_after_reading_request_before_responding() {
let _ = pretty_env_logger::try_init();
let mut rt = Runtime::new().unwrap();
let listener = tcp_bind(&"127.0.0.1:0".parse().unwrap()).unwrap();
let addr = listener.local_addr().unwrap();

thread::spawn(move || {
let mut tcp = connect(&addr);
tcp.write_all(b"GET / HTTP/1.1\r\n\r\n").unwrap();
});

let fut = listener.incoming()
.into_future()
.map_err(|_| unreachable!())
.and_then(|(item, _incoming)| {
let socket = item.unwrap();
Http::new()
.http1_half_close(false)
.serve_connection(socket, service_fn(|_| {
Delay::new(Duration::from_secs(2))
.map(|_| -> Response<Body> {
panic!("response future should have been dropped");
})
}))
});

rt.block_on(fut).expect_err("socket disconnected");
}

#[test]
fn returning_1xx_response_is_error() {
let mut rt = Runtime::new().unwrap();
Expand Down

0 comments on commit 8c8c1f4

Please sign in to comment.