mirror of
https://github.com/ntex-rs/ntex.git
synced 2025-04-04 13:27:39 +03:00
Refactor http1 dispatcher
This commit is contained in:
parent
114a7b6dba
commit
a29c003f55
4 changed files with 156 additions and 140 deletions
|
@ -349,7 +349,7 @@ impl<F> Io<F> {
|
||||||
if self.flags().contains(Flags::KEEPALIVE) {
|
if self.flags().contains(Flags::KEEPALIVE) {
|
||||||
timer::unregister(self.0 .0.keepalive.get(), &self.0);
|
timer::unregister(self.0 .0.keepalive.get(), &self.0);
|
||||||
}
|
}
|
||||||
if timeout != time::Duration::ZERO {
|
if !timeout.is_zero() {
|
||||||
log::debug!("start keep-alive timeout {:?}", timeout);
|
log::debug!("start keep-alive timeout {:?}", timeout);
|
||||||
self.0 .0.insert_flags(Flags::KEEPALIVE);
|
self.0 .0.insert_flags(Flags::KEEPALIVE);
|
||||||
self.0 .0.keepalive.set(timer::register(timeout, &self.0));
|
self.0 .0.keepalive.set(timer::register(timeout, &self.0));
|
||||||
|
|
|
@ -1,5 +1,9 @@
|
||||||
# Changes
|
# Changes
|
||||||
|
|
||||||
|
## [0.5.17] - 2022-05-xx
|
||||||
|
|
||||||
|
* http: Fix handling for zero slow-request timeout
|
||||||
|
|
||||||
## [0.5.16] - 2022-04-05
|
## [0.5.16] - 2022-04-05
|
||||||
|
|
||||||
* ws: Add keep-alive timeout support to websockets client
|
* ws: Add keep-alive timeout support to websockets client
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
[package]
|
[package]
|
||||||
name = "ntex"
|
name = "ntex"
|
||||||
version = "0.5.16"
|
version = "0.5.17"
|
||||||
authors = ["ntex contributors <team@ntex.rs>"]
|
authors = ["ntex contributors <team@ntex.rs>"]
|
||||||
description = "Framework for composable network services"
|
description = "Framework for composable network services"
|
||||||
readme = "README.md"
|
readme = "README.md"
|
||||||
|
|
|
@ -50,6 +50,8 @@ enum State<B> {
|
||||||
Call,
|
Call,
|
||||||
#[error("State::ReadRequest")]
|
#[error("State::ReadRequest")]
|
||||||
ReadRequest,
|
ReadRequest,
|
||||||
|
#[error("State::ReadFirstRequest")]
|
||||||
|
ReadFirstRequest,
|
||||||
#[error("State::ReadPayload")]
|
#[error("State::ReadPayload")]
|
||||||
ReadPayload,
|
ReadPayload,
|
||||||
#[error("State::SendPayload")]
|
#[error("State::SendPayload")]
|
||||||
|
@ -100,16 +102,21 @@ where
|
||||||
io.set_disconnect_timeout(config.client_disconnect.into());
|
io.set_disconnect_timeout(config.client_disconnect.into());
|
||||||
|
|
||||||
// slow-request timer
|
// slow-request timer
|
||||||
io.start_keepalive_timer(config.client_timeout);
|
let flags = if config.client_timeout.is_zero() {
|
||||||
|
Flags::empty()
|
||||||
|
} else {
|
||||||
|
io.start_keepalive_timer(config.client_timeout);
|
||||||
|
Flags::KEEPALIVE_REG
|
||||||
|
};
|
||||||
|
|
||||||
Dispatcher {
|
Dispatcher {
|
||||||
call: CallState::None,
|
call: CallState::None,
|
||||||
st: State::ReadRequest,
|
st: State::ReadFirstRequest,
|
||||||
inner: DispatcherInner {
|
inner: DispatcherInner {
|
||||||
io,
|
io,
|
||||||
|
flags,
|
||||||
codec,
|
codec,
|
||||||
config,
|
config,
|
||||||
flags: Flags::KEEPALIVE_REG,
|
|
||||||
error: None,
|
error: None,
|
||||||
payload: None,
|
payload: None,
|
||||||
_t: marker::PhantomData,
|
_t: marker::PhantomData,
|
||||||
|
@ -281,140 +288,9 @@ where
|
||||||
this.call.set(next);
|
this.call.set(next);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// read request and call service
|
||||||
State::ReadRequest => {
|
State::ReadRequest => {
|
||||||
log::trace!("trying to read http message");
|
*this.st = ready!(this.inner.read_request(cx, &mut this.call));
|
||||||
|
|
||||||
// decode incoming bytes stream
|
|
||||||
match this.inner.io.poll_recv(&this.inner.codec, cx) {
|
|
||||||
Poll::Ready(Ok((mut req, pl))) => {
|
|
||||||
log::trace!(
|
|
||||||
"http message is received: {:?} and payload {:?}",
|
|
||||||
req,
|
|
||||||
pl
|
|
||||||
);
|
|
||||||
|
|
||||||
// configure request payload
|
|
||||||
let upgrade = match pl {
|
|
||||||
PayloadType::None => false,
|
|
||||||
PayloadType::Payload(decoder) => {
|
|
||||||
let (ps, pl) = Payload::create(false);
|
|
||||||
req.replace_payload(http::Payload::H1(pl));
|
|
||||||
this.inner.payload = Some((decoder, ps));
|
|
||||||
false
|
|
||||||
}
|
|
||||||
PayloadType::Stream(decoder) => {
|
|
||||||
if this.inner.config.upgrade.is_none() {
|
|
||||||
let (ps, pl) = Payload::create(false);
|
|
||||||
req.replace_payload(http::Payload::H1(pl));
|
|
||||||
this.inner.payload = Some((decoder, ps));
|
|
||||||
false
|
|
||||||
} else {
|
|
||||||
this.inner.flags.insert(Flags::UPGRADE);
|
|
||||||
true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// slow-request first request
|
|
||||||
this.inner.flags.insert(Flags::STARTED);
|
|
||||||
this.inner.flags.remove(Flags::KEEPALIVE_REG);
|
|
||||||
this.inner.io.remove_keepalive_timer();
|
|
||||||
|
|
||||||
if upgrade {
|
|
||||||
// Handle UPGRADE request
|
|
||||||
log::trace!("prep io for upgrade handler");
|
|
||||||
*this.st = State::Upgrade(Some(req));
|
|
||||||
} else {
|
|
||||||
if req.upgrade() {
|
|
||||||
this.inner.flags.insert(Flags::UPGRADE_HND);
|
|
||||||
let io: IoBoxed = this.inner.io.take().into();
|
|
||||||
req.head_mut().io = CurrentIo::Io(Rc::new((
|
|
||||||
io.get_ref(),
|
|
||||||
RefCell::new(Some(Box::new((
|
|
||||||
io,
|
|
||||||
this.inner.codec.clone(),
|
|
||||||
)))),
|
|
||||||
)));
|
|
||||||
} else {
|
|
||||||
req.head_mut().io =
|
|
||||||
CurrentIo::Ref(this.inner.io.get_ref());
|
|
||||||
}
|
|
||||||
*this.st = State::Call;
|
|
||||||
this.call.set(
|
|
||||||
if let Some(ref f) = this.inner.config.on_request {
|
|
||||||
// Handle filter fut
|
|
||||||
CallState::Filter {
|
|
||||||
fut: f.call((req, this.inner.io.get_ref())),
|
|
||||||
}
|
|
||||||
} else if req.head().expect() {
|
|
||||||
// Handle normal requests with EXPECT: 100-Continue` header
|
|
||||||
CallState::Expect {
|
|
||||||
fut: this.inner.config.expect.call(req),
|
|
||||||
}
|
|
||||||
} else if this.inner.flags.contains(Flags::UPGRADE_HND)
|
|
||||||
{
|
|
||||||
// Handle upgrade requests
|
|
||||||
CallState::ServiceUpgrade {
|
|
||||||
fut: this.inner.config.service.call(req),
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// Handle normal requests
|
|
||||||
CallState::Service {
|
|
||||||
fut: this.inner.config.service.call(req),
|
|
||||||
}
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Poll::Ready(Err(RecvError::WriteBackpressure)) => {
|
|
||||||
if let Err(err) = ready!(this.inner.io.poll_flush(cx, false)) {
|
|
||||||
log::trace!("peer is gone with {:?}", err);
|
|
||||||
*this.st = State::Stop;
|
|
||||||
this.inner.error = Some(DispatchError::PeerGone(Some(err)));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Poll::Ready(Err(RecvError::Decoder(err))) => {
|
|
||||||
// Malformed requests, respond with 400
|
|
||||||
log::trace!("malformed request: {:?}", err);
|
|
||||||
let (res, body) = Response::BadRequest().finish().into_parts();
|
|
||||||
this.inner.error = Some(DispatchError::Parse(err));
|
|
||||||
*this.st = this.inner.send_response(res, body.into_body());
|
|
||||||
}
|
|
||||||
Poll::Ready(Err(RecvError::PeerGone(err))) => {
|
|
||||||
log::trace!("peer is gone with {:?}", err);
|
|
||||||
*this.st = State::Stop;
|
|
||||||
this.inner.error = Some(DispatchError::PeerGone(err));
|
|
||||||
}
|
|
||||||
Poll::Ready(Err(RecvError::Stop)) => {
|
|
||||||
log::trace!("dispatcher is instructed to stop");
|
|
||||||
*this.st = State::Stop;
|
|
||||||
}
|
|
||||||
Poll::Ready(Err(RecvError::KeepAlive)) => {
|
|
||||||
// keep-alive timeout
|
|
||||||
if !this.inner.flags.contains(Flags::STARTED) {
|
|
||||||
log::trace!("slow request timeout");
|
|
||||||
let (req, body) =
|
|
||||||
Response::RequestTimeout().finish().into_parts();
|
|
||||||
let _ = this.inner.send_response(req, body.into_body());
|
|
||||||
this.inner.error = Some(DispatchError::SlowRequestTimeout);
|
|
||||||
} else {
|
|
||||||
log::trace!("keep-alive timeout, close connection");
|
|
||||||
}
|
|
||||||
*this.st = State::Stop;
|
|
||||||
}
|
|
||||||
Poll::Pending => {
|
|
||||||
// register keep-alive timer
|
|
||||||
if this.inner.flags.contains(Flags::KEEPALIVE)
|
|
||||||
&& !this.inner.flags.contains(Flags::KEEPALIVE_REG)
|
|
||||||
{
|
|
||||||
this.inner.flags.insert(Flags::KEEPALIVE_REG);
|
|
||||||
this.inner
|
|
||||||
.io
|
|
||||||
.start_keepalive_timer(this.inner.config.keep_alive);
|
|
||||||
}
|
|
||||||
return Poll::Pending;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
// consume request's payload
|
// consume request's payload
|
||||||
State::ReadPayload => {
|
State::ReadPayload => {
|
||||||
|
@ -444,6 +320,11 @@ where
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// read first request and call service
|
||||||
|
State::ReadFirstRequest => {
|
||||||
|
*this.st = ready!(this.inner.read_request(cx, &mut this.call));
|
||||||
|
this.inner.flags.insert(Flags::STARTED);
|
||||||
|
}
|
||||||
// stop io tasks and call upgrade service
|
// stop io tasks and call upgrade service
|
||||||
State::Upgrade(ref mut req) => {
|
State::Upgrade(ref mut req) => {
|
||||||
let io = this.inner.io.take();
|
let io = this.inner.io.take();
|
||||||
|
@ -485,10 +366,12 @@ where
|
||||||
|
|
||||||
impl<T, S, B, X, U> DispatcherInner<T, S, B, X, U>
|
impl<T, S, B, X, U> DispatcherInner<T, S, B, X, U>
|
||||||
where
|
where
|
||||||
|
T: Filter,
|
||||||
S: Service<Request>,
|
S: Service<Request>,
|
||||||
S::Error: ResponseError + 'static,
|
S::Error: ResponseError + 'static,
|
||||||
S::Response: Into<Response<B>>,
|
S::Response: Into<Response<B>>,
|
||||||
B: MessageBody,
|
B: MessageBody,
|
||||||
|
X: Service<Request>,
|
||||||
{
|
{
|
||||||
fn switch_to_read_request(&mut self) -> State<B> {
|
fn switch_to_read_request(&mut self) -> State<B> {
|
||||||
// connection is not keep-alive, disconnect
|
// connection is not keep-alive, disconnect
|
||||||
|
@ -496,14 +379,20 @@ where
|
||||||
self.io.close();
|
self.io.close();
|
||||||
State::Stop
|
State::Stop
|
||||||
} else {
|
} else {
|
||||||
|
// register keep-alive timer
|
||||||
|
if self.flags.contains(Flags::KEEPALIVE) {
|
||||||
|
self.flags.remove(Flags::KEEPALIVE);
|
||||||
|
self.flags.insert(Flags::KEEPALIVE_REG);
|
||||||
|
self.io.start_keepalive_timer(self.config.keep_alive);
|
||||||
|
}
|
||||||
State::ReadRequest
|
State::ReadRequest
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn unregister_keepalive(&mut self) {
|
fn unregister_keepalive(&mut self) {
|
||||||
if self.flags.contains(Flags::KEEPALIVE) {
|
if self.flags.contains(Flags::KEEPALIVE_REG) {
|
||||||
self.io.remove_keepalive_timer();
|
self.io.remove_keepalive_timer();
|
||||||
self.flags.remove(Flags::KEEPALIVE);
|
self.flags.remove(Flags::KEEPALIVE | Flags::KEEPALIVE_REG);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -529,6 +418,129 @@ where
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn read_request(
|
||||||
|
&mut self,
|
||||||
|
cx: &mut Context<'_>,
|
||||||
|
call_state: &mut std::pin::Pin<&mut CallState<S, X>>,
|
||||||
|
) -> Poll<State<B>> {
|
||||||
|
log::trace!("trying to read http message");
|
||||||
|
|
||||||
|
loop {
|
||||||
|
let result = ready!(self.io.poll_recv(&self.codec, cx));
|
||||||
|
|
||||||
|
// decode incoming bytes stream
|
||||||
|
return match result {
|
||||||
|
Ok((mut req, pl)) => {
|
||||||
|
log::trace!("http message is received: {:?} and payload {:?}", req, pl);
|
||||||
|
|
||||||
|
// keep-alive timer
|
||||||
|
if self.flags.contains(Flags::KEEPALIVE_REG) {
|
||||||
|
self.flags.remove(Flags::KEEPALIVE_REG);
|
||||||
|
self.io.remove_keepalive_timer();
|
||||||
|
}
|
||||||
|
|
||||||
|
// configure request payload
|
||||||
|
let upgrade = match pl {
|
||||||
|
PayloadType::None => false,
|
||||||
|
PayloadType::Payload(decoder) => {
|
||||||
|
let (ps, pl) = Payload::create(false);
|
||||||
|
req.replace_payload(http::Payload::H1(pl));
|
||||||
|
self.payload = Some((decoder, ps));
|
||||||
|
false
|
||||||
|
}
|
||||||
|
PayloadType::Stream(decoder) => {
|
||||||
|
if self.config.upgrade.is_none() {
|
||||||
|
let (ps, pl) = Payload::create(false);
|
||||||
|
req.replace_payload(http::Payload::H1(pl));
|
||||||
|
self.payload = Some((decoder, ps));
|
||||||
|
false
|
||||||
|
} else {
|
||||||
|
self.flags.insert(Flags::UPGRADE);
|
||||||
|
true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
if upgrade {
|
||||||
|
// Handle UPGRADE request
|
||||||
|
log::trace!("prep io for upgrade handler");
|
||||||
|
Poll::Ready(State::Upgrade(Some(req)))
|
||||||
|
} else {
|
||||||
|
if req.upgrade() {
|
||||||
|
self.flags.insert(Flags::UPGRADE_HND);
|
||||||
|
let io: IoBoxed = self.io.take().into();
|
||||||
|
req.head_mut().io = CurrentIo::Io(Rc::new((
|
||||||
|
io.get_ref(),
|
||||||
|
RefCell::new(Some(Box::new((io, self.codec.clone())))),
|
||||||
|
)));
|
||||||
|
} else {
|
||||||
|
req.head_mut().io = CurrentIo::Ref(self.io.get_ref());
|
||||||
|
}
|
||||||
|
call_state.set(if let Some(ref f) = self.config.on_request {
|
||||||
|
// Handle filter fut
|
||||||
|
CallState::Filter {
|
||||||
|
fut: f.call((req, self.io.get_ref())),
|
||||||
|
}
|
||||||
|
} else if req.head().expect() {
|
||||||
|
// Handle normal requests with EXPECT: 100-Continue` header
|
||||||
|
CallState::Expect {
|
||||||
|
fut: self.config.expect.call(req),
|
||||||
|
}
|
||||||
|
} else if self.flags.contains(Flags::UPGRADE_HND) {
|
||||||
|
// Handle upgrade requests
|
||||||
|
CallState::ServiceUpgrade {
|
||||||
|
fut: self.config.service.call(req),
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Handle normal requests
|
||||||
|
CallState::Service {
|
||||||
|
fut: self.config.service.call(req),
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Poll::Ready(State::Call)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Err(RecvError::WriteBackpressure) => {
|
||||||
|
if let Err(err) = ready!(self.io.poll_flush(cx, false)) {
|
||||||
|
log::trace!("peer is gone with {:?}", err);
|
||||||
|
self.error = Some(DispatchError::PeerGone(Some(err)));
|
||||||
|
Poll::Ready(State::Stop)
|
||||||
|
} else {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Err(RecvError::Decoder(err)) => {
|
||||||
|
// Malformed requests, respond with 400
|
||||||
|
log::trace!("malformed request: {:?}", err);
|
||||||
|
let (res, body) = Response::BadRequest().finish().into_parts();
|
||||||
|
self.error = Some(DispatchError::Parse(err));
|
||||||
|
Poll::Ready(self.send_response(res, body.into_body()))
|
||||||
|
}
|
||||||
|
Err(RecvError::PeerGone(err)) => {
|
||||||
|
log::trace!("peer is gone with {:?}", err);
|
||||||
|
self.error = Some(DispatchError::PeerGone(err));
|
||||||
|
Poll::Ready(State::Stop)
|
||||||
|
}
|
||||||
|
Err(RecvError::Stop) => {
|
||||||
|
log::trace!("dispatcher is instructed to stop");
|
||||||
|
Poll::Ready(State::Stop)
|
||||||
|
}
|
||||||
|
Err(RecvError::KeepAlive) => {
|
||||||
|
// keep-alive timeout
|
||||||
|
if !self.flags.contains(Flags::STARTED) {
|
||||||
|
log::trace!("slow request timeout");
|
||||||
|
let (req, body) = Response::RequestTimeout().finish().into_parts();
|
||||||
|
let _ = self.send_response(req, body.into_body());
|
||||||
|
self.error = Some(DispatchError::SlowRequestTimeout);
|
||||||
|
} else {
|
||||||
|
log::trace!("keep-alive timeout, close connection");
|
||||||
|
}
|
||||||
|
Poll::Ready(State::Stop)
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
fn send_response(&mut self, msg: Response<()>, body: ResponseBody<B>) -> State<B> {
|
fn send_response(&mut self, msg: Response<()>, body: ResponseBody<B>) -> State<B> {
|
||||||
trace!("sending response: {:?} body: {:?}", msg, body.size());
|
trace!("sending response: {:?} body: {:?}", msg, body.size());
|
||||||
// we dont need to process responses if socket is disconnected
|
// we dont need to process responses if socket is disconnected
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue