Skip to content

Commit 5ca42df

Browse files
authored
fix stuck connection when handler doesn't read payload (#2624)
1 parent fc5ecdc commit 5ca42df

File tree

6 files changed

+307
-27
lines changed

6 files changed

+307
-27
lines changed

actix-http/CHANGES.md

+8
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,14 @@
11
# Changes
22

33
## Unreleased - 2021-xx-xx
4+
### Changed
5+
- `error::DispatcherError` enum is now marked `#[non_exhaustive]`. [#2624]
6+
7+
8+
### Fixed
9+
- Issue where handlers that took payload but then dropped without reading it to EOF it would cause keep-alive connections to become stuck. [#2624]
10+
11+
[#2624]: https://github.com/actix/actix-web/pull/2624
412

513

614
## 3.0.0-rc.1 - 2022-01-31

actix-http/src/error.rs

+5
Original file line numberDiff line numberDiff line change
@@ -340,6 +340,7 @@ impl From<PayloadError> for Error {
340340

341341
/// A set of errors that can occur during dispatching HTTP requests.
342342
#[derive(Debug, Display, From)]
343+
#[non_exhaustive]
343344
pub enum DispatchError {
344345
/// Service error.
345346
#[display(fmt = "Service Error")]
@@ -373,6 +374,10 @@ pub enum DispatchError {
373374
#[display(fmt = "Connection shutdown timeout")]
374375
DisconnectTimeout,
375376

377+
/// Handler dropped payload before reading EOF.
378+
#[display(fmt = "Handler dropped payload before reading EOF")]
379+
HandlerDroppedPayload,
380+
376381
/// Internal error.
377382
#[display(fmt = "Internal error")]
378383
InternalError,

actix-http/src/h1/codec.rs

+2
Original file line numberDiff line numberDiff line change
@@ -125,11 +125,13 @@ impl Decoder for Codec {
125125
self.flags.set(Flags::HEAD, head.method == Method::HEAD);
126126
self.version = head.version;
127127
self.conn_type = head.connection_type();
128+
128129
if self.conn_type == ConnectionType::KeepAlive
129130
&& !self.flags.contains(Flags::KEEP_ALIVE_ENABLED)
130131
{
131132
self.conn_type = ConnectionType::Close
132133
}
134+
133135
match payload {
134136
PayloadType::None => self.payload = None,
135137
PayloadType::Payload(pl) => self.payload = Some(pl),

actix-http/src/h1/decoder.rs

+5-3
Original file line numberDiff line numberDiff line change
@@ -209,15 +209,16 @@ impl MessageType for Request {
209209

210210
let (len, method, uri, ver, h_len) = {
211211
// SAFETY:
212-
// Create an uninitialized array of `MaybeUninit`. The `assume_init` is
213-
// safe because the type we are claiming to have initialized here is a
214-
// bunch of `MaybeUninit`s, which do not require initialization.
212+
// Create an uninitialized array of `MaybeUninit`. The `assume_init` is safe because the
213+
// type we are claiming to have initialized here is a bunch of `MaybeUninit`s, which
214+
// do not require initialization.
215215
let mut parsed = unsafe {
216216
MaybeUninit::<[MaybeUninit<httparse::Header<'_>>; MAX_HEADERS]>::uninit()
217217
.assume_init()
218218
};
219219

220220
let mut req = httparse::Request::new(&mut []);
221+
221222
match req.parse_with_uninit_headers(src, &mut parsed)? {
222223
httparse::Status::Complete(len) => {
223224
let method = Method::from_bytes(req.method.unwrap().as_bytes())
@@ -232,6 +233,7 @@ impl MessageType for Request {
232233

233234
(len, method, uri, version, req.headers.len())
234235
}
236+
235237
httparse::Status::Partial => {
236238
return if src.len() >= MAX_BUFFER_SIZE {
237239
trace!("MAX_BUFFER_SIZE unprocessed data reached, closing");

actix-http/src/h1/dispatcher.rs

+68-5
Original file line numberDiff line numberDiff line change
@@ -21,7 +21,7 @@ use crate::{
2121
config::ServiceConfig,
2222
error::{DispatchError, ParseError, PayloadError},
2323
service::HttpFlow,
24-
Error, Extensions, OnConnectData, Request, Response, StatusCode,
24+
ConnectionType, Error, Extensions, OnConnectData, Request, Response, StatusCode,
2525
};
2626

2727
use super::{
@@ -151,7 +151,8 @@ pin_project! {
151151
error: Option<DispatchError>,
152152

153153
#[pin]
154-
state: State<S, B, X>,
154+
pub(super) state: State<S, B, X>,
155+
// when Some(_) dispatcher is in state of receiving request payload
155156
payload: Option<PayloadSender>,
156157
messages: VecDeque<DispatcherMessage>,
157158

@@ -174,7 +175,7 @@ enum DispatcherMessage {
174175

175176
pin_project! {
176177
#[project = StateProj]
177-
enum State<S, B, X>
178+
pub(super) enum State<S, B, X>
178179
where
179180
S: Service<Request>,
180181
X: Service<Request, Response = Request>,
@@ -194,7 +195,7 @@ where
194195
X: Service<Request, Response = Request>,
195196
B: MessageBody,
196197
{
197-
fn is_none(&self) -> bool {
198+
pub(super) fn is_none(&self) -> bool {
198199
matches!(self, State::None)
199200
}
200201
}
@@ -686,12 +687,74 @@ where
686687
let can_not_read = !self.can_read(cx);
687688

688689
// limit amount of non-processed requests
689-
if pipeline_queue_full || can_not_read {
690+
if pipeline_queue_full {
690691
return Ok(false);
691692
}
692693

693694
let mut this = self.as_mut().project();
694695

696+
if can_not_read {
697+
log::debug!("cannot read request payload");
698+
699+
if let Some(sender) = &this.payload {
700+
// ...maybe handler does not want to read any more payload...
701+
if let PayloadStatus::Dropped = sender.need_read(cx) {
702+
log::debug!("handler dropped payload early; attempt to clean connection");
703+
// ...in which case poll request payload a few times
704+
loop {
705+
match this.codec.decode(this.read_buf)? {
706+
Some(msg) => {
707+
match msg {
708+
// payload decoded did not yield EOF yet
709+
Message::Chunk(Some(_)) => {
710+
// if non-clean connection, next loop iter will detect empty
711+
// read buffer and close connection
712+
}
713+
714+
// connection is in clean state for next request
715+
Message::Chunk(None) => {
716+
log::debug!("connection successfully cleaned");
717+
718+
// reset dispatcher state
719+
let _ = this.payload.take();
720+
this.state.set(State::None);
721+
722+
// break out of payload decode loop
723+
break;
724+
}
725+
726+
// Either whole payload is read and loop is broken or more data
727+
// was expected in which case connection is closed. In both
728+
// situations dispatcher cannot get here.
729+
Message::Item(_) => {
730+
unreachable!("dispatcher is in payload receive state")
731+
}
732+
}
733+
}
734+
735+
// not enough info to decide if connection is going to be clean or not
736+
None => {
737+
log::error!(
738+
"handler did not read whole payload and dispatcher could not \
739+
drain read buf; return 500 and close connection"
740+
);
741+
742+
this.flags.insert(Flags::SHUTDOWN);
743+
let mut res = Response::internal_server_error().drop_body();
744+
res.head_mut().set_connection_type(ConnectionType::Close);
745+
this.messages.push_back(DispatcherMessage::Error(res));
746+
*this.error = Some(DispatchError::HandlerDroppedPayload);
747+
return Ok(true);
748+
}
749+
}
750+
}
751+
}
752+
} else {
753+
// can_not_read and no request payload
754+
return Ok(false);
755+
}
756+
}
757+
695758
let mut updated = false;
696759

697760
loop {

0 commit comments

Comments
 (0)