Skip to content

Commit 842c655

Browse files
authoredNov 18, 2021
feat(server): add HTTP/1 header read timeout option (#2675)
Adds `Server::http1_header_read_timeout(Duration)`. Setting a duration will determine how long a client has to finish sending all the request headers before trigger a timeout test. This can help reduce resource usage when bad actors open connections without sending full requests. Closes #2457
1 parent d0b1d9e commit 842c655

File tree

9 files changed

+308
-3
lines changed

9 files changed

+308
-3
lines changed
 

‎Cargo.toml

+1
Original file line numberDiff line numberDiff line change
@@ -100,6 +100,7 @@ stream = []
100100
runtime = [
101101
"tcp",
102102
"tokio/rt",
103+
"tokio/time",
103104
]
104105
tcp = [
105106
"socket2",

‎src/error.rs

+10
Original file line numberDiff line numberDiff line change
@@ -44,6 +44,9 @@ pub(super) enum Kind {
4444
#[cfg(any(feature = "http1", feature = "http2"))]
4545
#[cfg(feature = "server")]
4646
Accept,
47+
/// User took too long to send headers
48+
#[cfg(all(feature = "http1", feature = "server", feature = "runtime"))]
49+
HeaderTimeout,
4750
/// Error while reading a body from connection.
4851
#[cfg(any(feature = "http1", feature = "http2", feature = "stream"))]
4952
Body,
@@ -310,6 +313,11 @@ impl Error {
310313
Error::new_user(User::UnexpectedHeader)
311314
}
312315

316+
#[cfg(all(feature = "http1", feature = "server", feature = "runtime"))]
317+
pub(super) fn new_header_timeout() -> Error {
318+
Error::new(Kind::HeaderTimeout)
319+
}
320+
313321
#[cfg(any(feature = "http1", feature = "http2"))]
314322
#[cfg(feature = "client")]
315323
pub(super) fn new_user_unsupported_version() -> Error {
@@ -419,6 +427,8 @@ impl Error {
419427
#[cfg(any(feature = "http1", feature = "http2"))]
420428
#[cfg(feature = "server")]
421429
Kind::Accept => "error accepting connection",
430+
#[cfg(all(feature = "http1", feature = "server", feature = "runtime"))]
431+
Kind::HeaderTimeout => "read header from client timeout",
422432
#[cfg(any(feature = "http1", feature = "http2", feature = "stream"))]
423433
Kind::Body => "error reading a body from connection",
424434
#[cfg(any(feature = "http1", feature = "http2"))]

‎src/proto/h1/conn.rs

+26
Original file line numberDiff line numberDiff line change
@@ -1,12 +1,15 @@
11
use std::fmt;
22
use std::io;
33
use std::marker::PhantomData;
4+
use std::time::Duration;
45

56
use bytes::{Buf, Bytes};
67
use http::header::{HeaderValue, CONNECTION};
78
use http::{HeaderMap, Method, Version};
89
use httparse::ParserConfig;
910
use tokio::io::{AsyncRead, AsyncWrite};
11+
#[cfg(all(feature = "server", feature = "runtime"))]
12+
use tokio::time::Sleep;
1013
use tracing::{debug, error, trace};
1114

1215
use super::io::Buffered;
@@ -47,6 +50,12 @@ where
4750
keep_alive: KA::Busy,
4851
method: None,
4952
h1_parser_config: ParserConfig::default(),
53+
#[cfg(all(feature = "server", feature = "runtime"))]
54+
h1_header_read_timeout: None,
55+
#[cfg(all(feature = "server", feature = "runtime"))]
56+
h1_header_read_timeout_fut: None,
57+
#[cfg(all(feature = "server", feature = "runtime"))]
58+
h1_header_read_timeout_running: false,
5059
preserve_header_case: false,
5160
title_case_headers: false,
5261
h09_responses: false,
@@ -106,6 +115,11 @@ where
106115
self.state.h09_responses = true;
107116
}
108117

118+
#[cfg(all(feature = "server", feature = "runtime"))]
119+
pub(crate) fn set_http1_header_read_timeout(&mut self, val: Duration) {
120+
self.state.h1_header_read_timeout = Some(val);
121+
}
122+
109123
#[cfg(feature = "server")]
110124
pub(crate) fn set_allow_half_close(&mut self) {
111125
self.state.allow_half_close = true;
@@ -178,6 +192,12 @@ where
178192
cached_headers: &mut self.state.cached_headers,
179193
req_method: &mut self.state.method,
180194
h1_parser_config: self.state.h1_parser_config.clone(),
195+
#[cfg(all(feature = "server", feature = "runtime"))]
196+
h1_header_read_timeout: self.state.h1_header_read_timeout,
197+
#[cfg(all(feature = "server", feature = "runtime"))]
198+
h1_header_read_timeout_fut: &mut self.state.h1_header_read_timeout_fut,
199+
#[cfg(all(feature = "server", feature = "runtime"))]
200+
h1_header_read_timeout_running: &mut self.state.h1_header_read_timeout_running,
181201
preserve_header_case: self.state.preserve_header_case,
182202
h09_responses: self.state.h09_responses,
183203
#[cfg(feature = "ffi")]
@@ -798,6 +818,12 @@ struct State {
798818
/// a body or not.
799819
method: Option<Method>,
800820
h1_parser_config: ParserConfig,
821+
#[cfg(all(feature = "server", feature = "runtime"))]
822+
h1_header_read_timeout: Option<Duration>,
823+
#[cfg(all(feature = "server", feature = "runtime"))]
824+
h1_header_read_timeout_fut: Option<Pin<Box<Sleep>>>,
825+
#[cfg(all(feature = "server", feature = "runtime"))]
826+
h1_header_read_timeout_running: bool,
801827
preserve_header_case: bool,
802828
title_case_headers: bool,
803829
h09_responses: bool,

‎src/proto/h1/io.rs

+37-1
Original file line numberDiff line numberDiff line change
@@ -3,10 +3,15 @@ use std::fmt;
33
use std::io::{self, IoSlice};
44
use std::marker::Unpin;
55
use std::mem::MaybeUninit;
6+
use std::future::Future;
7+
#[cfg(all(feature = "server", feature = "runtime"))]
8+
use std::time::Duration;
69

10+
#[cfg(all(feature = "server", feature = "runtime"))]
11+
use tokio::time::Instant;
712
use bytes::{Buf, BufMut, Bytes, BytesMut};
813
use tokio::io::{AsyncRead, AsyncWrite, ReadBuf};
9-
use tracing::{debug, trace};
14+
use tracing::{debug, warn, trace};
1015

1116
use super::{Http1Transaction, ParseContext, ParsedMessage};
1217
use crate::common::buf::BufList;
@@ -181,6 +186,12 @@ where
181186
cached_headers: parse_ctx.cached_headers,
182187
req_method: parse_ctx.req_method,
183188
h1_parser_config: parse_ctx.h1_parser_config.clone(),
189+
#[cfg(all(feature = "server", feature = "runtime"))]
190+
h1_header_read_timeout: parse_ctx.h1_header_read_timeout,
191+
#[cfg(all(feature = "server", feature = "runtime"))]
192+
h1_header_read_timeout_fut: parse_ctx.h1_header_read_timeout_fut,
193+
#[cfg(all(feature = "server", feature = "runtime"))]
194+
h1_header_read_timeout_running: parse_ctx.h1_header_read_timeout_running,
184195
preserve_header_case: parse_ctx.preserve_header_case,
185196
h09_responses: parse_ctx.h09_responses,
186197
#[cfg(feature = "ffi")]
@@ -191,6 +202,16 @@ where
191202
)? {
192203
Some(msg) => {
193204
debug!("parsed {} headers", msg.head.headers.len());
205+
206+
#[cfg(all(feature = "server", feature = "runtime"))]
207+
{
208+
*parse_ctx.h1_header_read_timeout_running = false;
209+
210+
if let Some(h1_header_read_timeout_fut) = parse_ctx.h1_header_read_timeout_fut {
211+
// Reset the timer in order to avoid woken up when the timeout finishes
212+
h1_header_read_timeout_fut.as_mut().reset(Instant::now() + Duration::from_secs(30 * 24 * 60 * 60));
213+
}
214+
}
194215
return Poll::Ready(Ok(msg));
195216
}
196217
None => {
@@ -199,6 +220,18 @@ where
199220
debug!("max_buf_size ({}) reached, closing", max);
200221
return Poll::Ready(Err(crate::Error::new_too_large()));
201222
}
223+
224+
#[cfg(all(feature = "server", feature = "runtime"))]
225+
if *parse_ctx.h1_header_read_timeout_running {
226+
if let Some(h1_header_read_timeout_fut) = parse_ctx.h1_header_read_timeout_fut {
227+
if Pin::new( h1_header_read_timeout_fut).poll(cx).is_ready() {
228+
*parse_ctx.h1_header_read_timeout_running = false;
229+
230+
warn!("read header from client timeout");
231+
return Poll::Ready(Err(crate::Error::new_header_timeout()))
232+
}
233+
}
234+
}
202235
}
203236
}
204237
if ready!(self.poll_read_from_io(cx)).map_err(crate::Error::new_io)? == 0 {
@@ -693,6 +726,9 @@ mod tests {
693726
cached_headers: &mut None,
694727
req_method: &mut None,
695728
h1_parser_config: Default::default(),
729+
h1_header_read_timeout: None,
730+
h1_header_read_timeout_fut: &mut None,
731+
h1_header_read_timeout_running: &mut false,
696732
preserve_header_case: false,
697733
h09_responses: false,
698734
#[cfg(feature = "ffi")]

‎src/proto/h1/mod.rs

+11
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,11 @@
1+
use std::pin::Pin;
2+
use std::time::Duration;
3+
14
use bytes::BytesMut;
25
use http::{HeaderMap, Method};
36
use httparse::ParserConfig;
7+
#[cfg(all(feature = "server", feature = "runtime"))]
8+
use tokio::time::Sleep;
49

510
use crate::body::DecodedLength;
611
use crate::proto::{BodyLength, MessageHead};
@@ -72,6 +77,12 @@ pub(crate) struct ParseContext<'a> {
7277
cached_headers: &'a mut Option<HeaderMap>,
7378
req_method: &'a mut Option<Method>,
7479
h1_parser_config: ParserConfig,
80+
#[cfg(all(feature = "server", feature = "runtime"))]
81+
h1_header_read_timeout: Option<Duration>,
82+
#[cfg(all(feature = "server", feature = "runtime"))]
83+
h1_header_read_timeout_fut: &'a mut Option<Pin<Box<Sleep>>>,
84+
#[cfg(all(feature = "server", feature = "runtime"))]
85+
h1_header_read_timeout_running: &'a mut bool,
7586
preserve_header_case: bool,
7687
h09_responses: bool,
7788
#[cfg(feature = "ffi")]

0 commit comments

Comments
 (0)
Please sign in to comment.