Skip to content

Commit ea3e228

Browse files
committedSep 15, 2021
fix(server): convert panic to error if Connection::without_shutdown called on HTTP/2 conn
1 parent c88011d commit ea3e228

File tree

2 files changed

+18
-5
lines changed

2 files changed

+18
-5
lines changed
 

‎src/error.rs

+13
Original file line numberDiff line numberDiff line change
@@ -128,6 +128,10 @@ pub(super) enum User {
128128
#[cfg(feature = "http1")]
129129
ManualUpgrade,
130130

131+
/// User called `server::Connection::without_shutdown()` on an HTTP/2 conn.
132+
#[cfg(feature = "server")]
133+
WithoutShutdownNonHttp1,
134+
131135
/// User aborted in an FFI callback.
132136
#[cfg(feature = "ffi")]
133137
AbortedByCallback,
@@ -355,6 +359,11 @@ impl Error {
355359
Error::new_user(User::Body).with(cause)
356360
}
357361

362+
#[cfg(feature = "server")]
363+
pub(super) fn new_without_shutdown_not_h1() -> Error {
364+
Error::new(Kind::User(User::WithoutShutdownNonHttp1))
365+
}
366+
358367
#[cfg(feature = "http1")]
359368
pub(super) fn new_shutdown(cause: std::io::Error) -> Error {
360369
Error::new(Kind::Shutdown).with(cause)
@@ -449,6 +458,10 @@ impl Error {
449458
Kind::User(User::NoUpgrade) => "no upgrade available",
450459
#[cfg(feature = "http1")]
451460
Kind::User(User::ManualUpgrade) => "upgrade expected but low level API in use",
461+
#[cfg(feature = "server")]
462+
Kind::User(User::WithoutShutdownNonHttp1) => {
463+
"without_shutdown() called on a non-HTTP/1 connection"
464+
}
452465
#[cfg(feature = "ffi")]
453466
Kind::User(User::AbortedByCallback) => "operation aborted by an application callback",
454467
}

‎src/server/conn.rs

+5-5
Original file line numberDiff line numberDiff line change
@@ -741,10 +741,6 @@ where
741741
/// upgrade. Once the upgrade is completed, the connection would be "done",
742742
/// but it is not desired to actually shutdown the IO object. Instead you
743743
/// would take it back using `into_parts`.
744-
///
745-
/// Use [`poll_fn`](https://docs.rs/futures/0.1.25/futures/future/fn.poll_fn.html)
746-
/// and [`try_ready!`](https://docs.rs/futures/0.1.25/futures/macro.try_ready.html)
747-
/// to work with this function; or use the `without_shutdown` wrapper.
748744
pub fn poll_without_shutdown(&mut self, cx: &mut task::Context<'_>) -> Poll<crate::Result<()>>
749745
where
750746
S: Unpin,
@@ -782,6 +778,10 @@ where
782778

783779
/// Prevent shutdown of the underlying IO object at the end of service the request,
784780
/// instead run `into_parts`. This is a convenience wrapper over `poll_without_shutdown`.
781+
///
782+
/// # Error
783+
///
784+
/// This errors if the underlying connection protocol is not HTTP/1.
785785
pub fn without_shutdown(self) -> impl Future<Output = crate::Result<Parts<I, S>>>
786786
where
787787
S: Unpin,
@@ -791,7 +791,7 @@ where
791791
let mut conn = Some(self);
792792
futures_util::future::poll_fn(move |cx| {
793793
ready!(conn.as_mut().unwrap().poll_without_shutdown(cx))?;
794-
Poll::Ready(Ok(conn.take().unwrap().into_parts()))
794+
Poll::Ready(conn.take().unwrap().try_into_parts().ok_or_else(crate::Error::new_without_shutdown_not_h1))
795795
})
796796
}
797797

0 commit comments

Comments
 (0)
Please sign in to comment.