Fix handling not consumed requests payload

This commit is contained in:
Nikolay Kim 2024-05-29 18:10:38 +05:00
parent 9c29de14cf
commit 5804165d9a
4 changed files with 35 additions and 11 deletions

View file

@ -4,6 +4,8 @@
* http: Fix handling payload timer after payload got consumed
* http: Fix handling not consumed request's payload
## [2.0.0] - 2024-05-28
* Use "async fn" for Service::ready() and Service::shutdown()

View file

@ -208,7 +208,12 @@ where
State::ReadRequest => ready!(inner.poll_read_request(cx)),
// consume request's payload
State::ReadPayload => {
ready!(inner.poll_request_payload(cx)).unwrap_or(State::ReadRequest)
let result = inner.poll_request_payload(cx);
if inner.flags.contains(Flags::SENDPAYLOAD_AND_STOP) {
inner.stop()
} else {
ready!(result).unwrap_or(State::ReadRequest)
}
}
// send response body
State::SendPayload { body } => {
@ -224,6 +229,7 @@ where
let _ = ready!(Pin::new(f).poll(cx));
fut.take();
}
log::debug!("{}: Dispatcher is stopped", inner.io.tag());
return Poll::Ready(
if let Some(io) = io {
@ -654,7 +660,8 @@ where
// wait until future completes and then close
// connection
self.payload = None;
Poll::Ready(Err(Either::Left(ProtocolError::PayloadIsNotConsumed)))
self.flags.insert(Flags::SENDPAYLOAD_AND_STOP);
Poll::Pending
}
}
}

View file

@ -76,10 +76,6 @@ pub enum ProtocolError {
#[error("Payload did not complete within the specified timeout")]
SlowPayloadTimeout,
/// Payload is not consumed
#[error("Task is completed but request's payload is not consumed")]
PayloadIsNotConsumed,
/// Response body processing error
#[error("Response body processing error: {0}")]
ResponsePayload(Box<dyn std::error::Error>),
@ -89,14 +85,10 @@ impl super::ResponseError for ProtocolError {
fn error_response(&self) -> super::Response {
match self {
ProtocolError::Decode(_) => super::Response::BadRequest().into(),
ProtocolError::SlowRequestTimeout | ProtocolError::SlowPayloadTimeout => {
super::Response::RequestTimeout().into()
}
ProtocolError::Encode(_)
| ProtocolError::PayloadIsNotConsumed
| ProtocolError::ResponsePayload(_) => {
ProtocolError::Encode(_) | ProtocolError::ResponsePayload(_) => {
super::Response::InternalServerError().into()
}
}

View file

@ -380,6 +380,29 @@ async fn test_http1_disable_payload_timer_after_whole_pl_has_been_read() {
assert_eq!(&data[..17], b"HTTP/1.1 200 OK\r\n");
}
/// Handle not consumed payload
#[ntex::test]
async fn test_http1_handle_not_consumed_payload() {
let srv = test_server(|| {
HttpService::build()
.h1_control(fn_service(move |msg: Control<_, _>| {
if matches!(msg, Control::ProtocolError(_)) {
panic!()
}
async move { Ok::<_, io::Error>(msg.ack()) }
}))
.h1(|_| async move { Ok::<_, io::Error>(Response::Ok().finish()) })
});
let mut stream = net::TcpStream::connect(srv.addr()).unwrap();
let _ = stream.write_all(b"GET /test/tests/test HTTP/1.1\r\ncontent-length: 4\r\n\r\n");
sleep(Millis(250)).await;
let _ = stream.write_all(b"1234");
let mut data = vec![0; 1024];
let _ = stream.read(&mut data);
assert_eq!(&data[..17], b"HTTP/1.1 200 OK\r\n");
}
#[ntex::test]
async fn test_content_length() {
let srv = test_server(|| {