mirror of
https://github.com/rwf2/Rocket.git
synced 2025-01-23 18:02:15 +00:00
fd294049c7
This commit completely rewrites Rocket's HTTP serving. In addition to significant internal cleanup, this commit introduces the following major features: * Support for custom, external listeners in the `listener` module. The new `listener` module contains new `Bindable`, `Listener`, and `Connection` traits which enable composable, external implementations of connection listeners. Rocket can launch on any `Listener`, or anything that can be used to create a listener (`Bindable`), via a new `launch_on()` method. * Support for Unix domain socket listeners out of the box. The default listener backwards compatibly supports listening on Unix domain sockets. To do so, configure an `address` of `unix:path/to/socket` and optional set `reuse` to `true` (the default) or `false` which controls whether Rocket will handle creating and deleting the unix domain socket. In addition to these new features, this commit makes the following major improvements: * Rocket now depends on hyper 1. * Rocket no longer depends on hyper to handle connections. This allows us to handle more connection failure conditions which results in an overall more robust server with fewer dependencies. * Logic to work around hyper's inability to reference incoming request data in the response results in a 15% performance improvement. * `Client`s can be marked secure with `Client::{un}tracked_secure()`, allowing Rocket to treat local connections as running under TLS. * The `macros` feature of `tokio` is no longer used by Rocket itself. Dependencies can take advantage of this reduction in compile-time cost by disabling the new default feature `tokio-macros`. * A new `TlsConfig::validate()` method allows checking a TLS config. * New `TlsConfig::{certs,key}_reader()`, `MtlsConfig::ca_certs_reader()` methods return `BufReader`s, which allow reading the configured certs and key directly. * A new `NamedFile::open_with()` constructor allows specifying `OpenOptions`. These improvements resulted in the following breaking changes: * The MSRV is now 1.74. * `hyper` is no longer exported from `rocket::http`. * `IoHandler::io` takes `Box<Self>` instead of `Pin<Box<Self>>`. - Use `Box::into_pin(self)` to recover the previous type. * `Response::upgrade()` now returns an `&mut dyn IoHandler`, not `Pin<& mut _>`. * `Config::{address,port,tls,mtls}` methods have been removed. - Use methods on `Rocket::endpoint()` instead. * `TlsConfig` was moved to `tls::TlsConfig`. * `MutualTls` was renamed and moved to `mtls::MtlsConfig`. * `ErrorKind::TlsBind` was removed. * The second field of `ErrorKind::Shutdown` was removed. * `{Local}Request::{set_}remote()` methods take/return an `Endpoint`. * `Client::new()` was removed; it was previously deprecated. Internally, the following major changes were made: * A new `async_bound` attribute macro was introduced to allow setting bounds on futures returned by `async fn`s in traits while maintaining good docs. * All utility functionality was moved to a new `util` module. Resolves #2671. Resolves #1070.
340 lines
9.6 KiB
Rust
340 lines
9.6 KiB
Rust
use rocket::{*, error::ErrorKind::SentinelAborts};
|
|
|
|
#[get("/two")]
|
|
fn two_states(_one: &State<u32>, _two: &State<String>) {}
|
|
|
|
#[post("/one", data = "<s>")]
|
|
fn one_state<'r>(_three: &'r State<u8>, s: &'r str) -> &'r str { s }
|
|
|
|
#[async_test]
|
|
async fn state_sentinel_works() {
|
|
let err = rocket::build()
|
|
.configure(Config::debug_default())
|
|
.mount("/", routes![two_states])
|
|
.ignite().await
|
|
.unwrap_err();
|
|
|
|
assert!(matches!(err.kind(), SentinelAborts(vec) if vec.len() == 2));
|
|
|
|
let err = rocket::build()
|
|
.configure(Config::debug_default())
|
|
.mount("/", routes![two_states])
|
|
.manage(String::new())
|
|
.ignite().await
|
|
.unwrap_err();
|
|
|
|
assert!(matches!(err.kind(), SentinelAborts(vec) if vec.len() == 1));
|
|
|
|
let err = rocket::build()
|
|
.configure(Config::debug_default())
|
|
.mount("/", routes![two_states])
|
|
.manage(1 as u32)
|
|
.ignite().await
|
|
.unwrap_err();
|
|
|
|
assert!(matches!(err.kind(), SentinelAborts(vec) if vec.len() == 1));
|
|
|
|
let result = rocket::build()
|
|
.configure(Config::debug_default())
|
|
.mount("/", routes![two_states])
|
|
.manage(String::new())
|
|
.manage(1 as u32)
|
|
.ignite().await;
|
|
|
|
assert!(result.is_ok());
|
|
|
|
let err = rocket::build()
|
|
.configure(Config::debug_default())
|
|
.mount("/", routes![one_state])
|
|
.ignite().await
|
|
.unwrap_err();
|
|
|
|
assert!(matches!(err.kind(), SentinelAborts(vec) if vec.len() == 1));
|
|
|
|
let result = rocket::build()
|
|
.configure(Config::debug_default())
|
|
.mount("/", routes![one_state])
|
|
.manage(1 as u8)
|
|
.ignite().await;
|
|
|
|
assert!(result.is_ok());
|
|
|
|
let err = rocket::build()
|
|
.configure(Config::debug_default())
|
|
.mount("/", routes![one_state, two_states])
|
|
.ignite().await
|
|
.unwrap_err();
|
|
|
|
assert!(matches!(err.kind(), SentinelAborts(vec) if vec.len() == 3));
|
|
|
|
let err = rocket::build()
|
|
.configure(Config::debug_default())
|
|
.mount("/", routes![one_state, two_states])
|
|
.manage(1 as u32)
|
|
.ignite().await
|
|
.unwrap_err();
|
|
|
|
assert!(matches!(err.kind(), SentinelAborts(vec) if vec.len() == 2));
|
|
|
|
let err = rocket::build()
|
|
.configure(Config::debug_default())
|
|
.mount("/", routes![one_state, two_states])
|
|
.manage(1 as u8)
|
|
.ignite().await
|
|
.unwrap_err();
|
|
|
|
assert!(matches!(err.kind(), SentinelAborts(vec) if vec.len() == 2));
|
|
|
|
let err = rocket::build()
|
|
.configure(Config::debug_default())
|
|
.mount("/", routes![one_state, two_states])
|
|
.manage(1 as u32)
|
|
.manage(1 as u8)
|
|
.ignite().await
|
|
.unwrap_err();
|
|
|
|
assert!(matches!(err.kind(), SentinelAborts(vec) if vec.len() == 1));
|
|
|
|
let result = rocket::build()
|
|
.configure(Config::debug_default())
|
|
.mount("/", routes![one_state, two_states])
|
|
.manage(1 as u32)
|
|
.manage(1 as u8)
|
|
.manage(String::new())
|
|
.ignite().await;
|
|
|
|
assert!(result.is_ok());
|
|
}
|
|
|
|
struct Data;
|
|
|
|
#[crate::async_trait]
|
|
impl<'r> data::FromData<'r> for Data {
|
|
type Error = Error;
|
|
async fn from_data(_: &'r Request<'_>, _: data::Data<'r>) -> data::Outcome<'r, Self> {
|
|
unimplemented!()
|
|
}
|
|
}
|
|
|
|
impl Sentinel for Data {
|
|
fn abort(rocket: &Rocket<Ignite>) -> bool {
|
|
rocket.state::<Data>().is_none()
|
|
}
|
|
}
|
|
|
|
#[post("/data", data = "<_data>")]
|
|
fn with_data(_data: Data) {}
|
|
|
|
#[async_test]
|
|
async fn data_sentinel_works() {
|
|
let err = rocket::build()
|
|
.configure(Config::debug_default())
|
|
.mount("/", routes![with_data])
|
|
.ignite().await
|
|
.unwrap_err();
|
|
|
|
assert!(matches!(err.kind(), SentinelAborts(vec) if vec.len() == 1));
|
|
|
|
let result = rocket::build()
|
|
.configure(Config::debug_default())
|
|
.mount("/", routes![with_data])
|
|
.manage(Data)
|
|
.ignite().await;
|
|
|
|
assert!(result.is_ok());
|
|
}
|
|
|
|
#[test]
|
|
fn inner_sentinels_detected() {
|
|
use rocket::local::blocking::Client;
|
|
|
|
#[derive(Responder)]
|
|
struct MyThing<T>(T);
|
|
|
|
struct ResponderSentinel;
|
|
|
|
impl<'r, 'o: 'r> response::Responder<'r, 'o> for ResponderSentinel {
|
|
fn respond_to(self, _: &'r Request<'_>) -> response::Result<'o> {
|
|
unimplemented!()
|
|
}
|
|
}
|
|
|
|
impl Sentinel for ResponderSentinel {
|
|
fn abort(_: &Rocket<Ignite>) -> bool {
|
|
true
|
|
}
|
|
}
|
|
|
|
#[get("/")]
|
|
fn route() -> MyThing<ResponderSentinel> { todo!() }
|
|
|
|
let err = Client::debug_with(routes![route]).unwrap_err();
|
|
assert!(matches!(err.kind(), SentinelAborts(vec) if vec.len() == 1));
|
|
|
|
#[derive(Responder)]
|
|
struct Inner<T>(T);
|
|
|
|
#[get("/")]
|
|
fn inner() -> MyThing<Inner<ResponderSentinel>> { todo!() }
|
|
|
|
let err = Client::debug_with(routes![inner]).unwrap_err();
|
|
assert!(matches!(err.kind(), SentinelAborts(vec) if vec.len() == 1));
|
|
|
|
#[get("/")]
|
|
fn inner_either() -> Either<Inner<ResponderSentinel>, ResponderSentinel> { todo!() }
|
|
|
|
let err = Client::debug_with(routes![inner_either]).unwrap_err();
|
|
assert!(matches!(err.kind(), SentinelAborts(vec) if vec.len() == 2));
|
|
|
|
#[derive(Responder)]
|
|
struct Block<T>(T);
|
|
|
|
impl<T> Sentinel for Block<T> {
|
|
fn abort(_: &Rocket<Ignite>) -> bool {
|
|
false
|
|
}
|
|
}
|
|
|
|
#[get("/")]
|
|
fn blocked() -> Block<ResponderSentinel> { todo!() }
|
|
|
|
Client::debug_with(routes![blocked]).expect("no sentinel errors");
|
|
|
|
#[get("/a")]
|
|
fn inner_b() -> Either<Inner<Block<ResponderSentinel>>, Block<ResponderSentinel>> {
|
|
todo!()
|
|
}
|
|
|
|
#[get("/b")]
|
|
fn inner_b2() -> Either<Block<Inner<ResponderSentinel>>, Block<ResponderSentinel>> {
|
|
todo!()
|
|
}
|
|
|
|
Client::debug_with(routes![inner_b, inner_b2]).expect("no sentinel errors");
|
|
|
|
#[get("/")]
|
|
fn half_b() -> Either<Inner<ResponderSentinel>, Block<ResponderSentinel>> {
|
|
todo!()
|
|
}
|
|
|
|
let err = Client::debug_with(routes![half_b]).unwrap_err();
|
|
assert!(matches!(err.kind(), SentinelAborts(vec) if vec.len() == 1));
|
|
|
|
use rocket::response::Responder;
|
|
|
|
#[get("/")]
|
|
fn half_c<'r>() -> Either<
|
|
Inner<impl Responder<'r, 'static>>,
|
|
Result<ResponderSentinel, Inner<ResponderSentinel>>
|
|
> {
|
|
Either::Left(Inner(()))
|
|
}
|
|
|
|
let err = Client::debug_with(routes![half_c]).unwrap_err();
|
|
assert!(matches!(err.kind(), SentinelAborts(vec) if vec.len() == 2));
|
|
|
|
#[get("/")]
|
|
fn half_d<'r>() -> Either<
|
|
Inner<impl Responder<'r, 'static>>,
|
|
Result<Block<ResponderSentinel>, Inner<ResponderSentinel>>
|
|
> {
|
|
Either::Left(Inner(()))
|
|
}
|
|
|
|
let err = Client::debug_with(routes![half_d]).unwrap_err();
|
|
assert!(matches!(err.kind(), SentinelAborts(vec) if vec.len() == 1));
|
|
|
|
// The special `Result` implementation.
|
|
type MyResult = Result<ResponderSentinel, ResponderSentinel>;
|
|
|
|
#[get("/")]
|
|
fn half_e<'r>() -> Either<Inner<impl Responder<'r, 'static>>, MyResult> {
|
|
Either::Left(Inner(()))
|
|
}
|
|
|
|
let err = Client::debug_with(routes![half_e]).unwrap_err();
|
|
assert!(matches!(err.kind(), SentinelAborts(vec) if vec.len() == 1));
|
|
|
|
// Another specialized sentinel.
|
|
|
|
#[get("/")] fn either_route() -> Either<ResponderSentinel, ResponderSentinel> { todo!() }
|
|
let err = Client::debug_with(routes![either_route]).unwrap_err();
|
|
assert!(matches!(err.kind(), SentinelAborts(vec) if vec.len() == 1));
|
|
|
|
#[get("/")] fn either_route2() -> Either<ResponderSentinel, ()> { todo!() }
|
|
let err = Client::debug_with(routes![either_route2]).unwrap_err();
|
|
assert!(matches!(err.kind(), SentinelAborts(vec) if vec.len() == 1));
|
|
|
|
#[get("/")] fn either_route3() -> Either<(), ResponderSentinel> { todo!() }
|
|
let err = Client::debug_with(routes![either_route3]).unwrap_err();
|
|
assert!(matches!(err.kind(), SentinelAborts(vec) if vec.len() == 1));
|
|
|
|
#[get("/")] fn either_route4() -> Either<(), ()> { todo!() }
|
|
Client::debug_with(routes![either_route4]).expect("no sentinel error");
|
|
}
|
|
|
|
#[async_test]
|
|
async fn known_macro_sentinel_works() {
|
|
use rocket::response::stream::{TextStream, ByteStream, ReaderStream};
|
|
use rocket::local::asynchronous::Client;
|
|
use rocket::tokio::io::AsyncRead;
|
|
|
|
#[derive(Responder)]
|
|
struct TextSentinel<'r>(&'r str);
|
|
|
|
impl Sentinel for TextSentinel<'_> {
|
|
fn abort(_: &Rocket<Ignite>) -> bool {
|
|
true
|
|
}
|
|
}
|
|
|
|
impl AsRef<str> for TextSentinel<'_> {
|
|
fn as_ref(&self) -> &str {
|
|
self.0
|
|
}
|
|
}
|
|
|
|
impl AsRef<[u8]> for TextSentinel<'_> {
|
|
fn as_ref(&self) -> &[u8] {
|
|
self.0.as_bytes()
|
|
}
|
|
}
|
|
|
|
impl AsyncRead for TextSentinel<'_> {
|
|
fn poll_read(
|
|
self: std::pin::Pin<&mut Self>,
|
|
_: &mut futures::task::Context<'_>,
|
|
_: &mut tokio::io::ReadBuf<'_>,
|
|
) -> futures::task::Poll<std::io::Result<()>> {
|
|
futures::task::Poll::Ready(Ok(()))
|
|
}
|
|
}
|
|
|
|
#[get("/text")]
|
|
fn text<'r>() -> TextStream![TextSentinel<'r>] {
|
|
TextStream!(yield TextSentinel("hi");)
|
|
}
|
|
|
|
#[get("/<a>")]
|
|
fn byte(a: &str) -> ByteStream![TextSentinel<'_>] {
|
|
ByteStream!(yield TextSentinel(a);)
|
|
}
|
|
|
|
#[get("/<_a>/<b>")]
|
|
fn reader<'a, 'b>(_a: &'a str, b: &'b str) -> ReaderStream![TextSentinel<'b>] {
|
|
ReaderStream!(yield TextSentinel(b);)
|
|
}
|
|
|
|
macro_rules! UnknownStream {
|
|
($t:ty) => (ReaderStream![$t])
|
|
}
|
|
|
|
#[get("/ignore")]
|
|
fn ignore() -> UnknownStream![TextSentinel<'static>] {
|
|
ReaderStream!(yield TextSentinel("hi");)
|
|
}
|
|
|
|
let err = Client::debug_with(routes![text, byte, reader, ignore]).await.unwrap_err();
|
|
assert!(matches!(err.kind(), SentinelAborts(vec) if vec.len() == 3));
|
|
}
|