中文
Mimic Generic Specialization in Rust
Last weekend I tried to write the sentinel middleware for tonic and volo, and struggled fighting against rustc again.
Background
The tonic supports two ways to implement middleware.
Different from tonic, volo utilizes motore as the service abstraction.
How the problem arose?
I first implement the tonic::Servie
. Intuitively, I wrote
rust
impl<S, B> Service<http::Request<B>> for SentinelService<S, http::Request<B>, B>
where
S: Service<http::Request<B>>,
{}
impl<S, R> Service<R> for SentinelService<S, R>
where
S: Service<R>,
{}
So what I expected is that: For tonic, whose request is in fact http::Request<http_body::combinators::UnsyncBoxBody<bytes::Bytes, tonic::Status>>
, will be substituted in the first generic, instead of the second. In the first case, we can call methods on http::Request
to provide a default resource extractor for sentinel, while in the second one, the custom resource extractor is necessary.
I thought that there is a similar feature in rust as SFINAE in C++. The best specialization of the generic would be chosen.
However, rustc reminded me that two implementation of trait Service
was contradicted with each other. I realized that there was no SFINAE
in rust!
Mimic the SFINAE!
From Pick preferred implementation on conflicting trait implementation (using negative bounds) - Stack Overflow, I learnt that we could use the unstable negative_impls and auto_traits feature, to do this magic
rust
#![cfg_attr(feature = "nightly", feature(auto_traits, negative_impls))]
trait WithoutDefaultExtractor {}
impl<B> !WithoutDefaultExtractor for http::Request<B> {}
impl<S, B> Service<http::Request<B>> for SentinelService<S, http::Request<B>, B>
where
S: Service<R>,
{}
impl<S, R> Service<R> for SentinelService<S, R>
where
S: Service<R>,
R: WithoutDefaultExtractor,
{}
Is it perfect?
If the generic is the same, say, if we have the following code,
rust
impl<S, R> Service<R> for SentinelService<S, R>
where
S: Service<R>,
{}
impl<S, R> Service<R> for SentinelService<S, R>
where
S: Service<R>,
R: AnotherTrait,
{}
the above method does not work anymore. Or we can use a Higher-Rank Trait Bounds (HRTBs) and apply the similar method? I didn't try.
Fine, I give up ...
Finally, I choose to add some feature items in my Cargo.toml
, so that I can use the #[cfg(feature="http")]
attribute to control the specialization by hand 😦
Then the code becomes
rust
#[cfg(feature = "http")]
impl<S, B> Service<http::Request<B>> for SentinelService<S, http::Request<B>, B>
where
S: Service<http::Request<B>>,
{}
#[cfg(not(feature = "http"))]
impl<S, R> Service<R> for SentinelService<S, R>
where
S: Service<R>,
{}
Related questions
Generics partial specialization in Rust - Stack Overflow
Equivalent of specific template usage in C++ for Rust - Stack Overflow
Appendix
Constraints on GAT
During implementing the middleware, I work a lot around the GAT in tower
and motore
. Generally I found there are two scenes where we may impose constraints on GAT.
- impose a trait constraint on GAT
- instantiate the trait with specific GAT, i.e., the equality constraint.
The first one is widely used in the source code of tower
and motore
, and the second one is related to a question on StackOverflow answered by me.
Here I made a complied example to illustrate them.
See how we impose constraints on GAT Builder::InstanceForBuilder
and Builder::Useless
and the differences between them.
rust
// Trait definitions.
trait Builder {
type InstanceForBuilder<'a>: Instance<'a>;
type Useless<'a>;
fn build<'a>(&self, val: &'a usize) -> Self::InstanceForBuilder<'a>;
}
trait Instance<'a> {
// Some functions will only work when the instance has some concrete associated type.
type InstanceProperty;
}
fn build_with_42_for_bool_instance<'a, B, I>(builder: B)
where
B : Builder<InstanceForBuilder<'a>=I>,
<B as Builder>::Useless<'a> : std::fmt::Debug,
I : Instance<'a, InstanceProperty=bool>+std::fmt::Debug,
{
builder.build(&42);
}
// Now try it out.
struct MyBuilder;
#[derive(Debug)]
struct MyInstance<'a> {
val: &'a usize,
}
impl Builder for MyBuilder {
type InstanceForBuilder<'a> = MyInstance<'a>;
type Useless<'a> = &'a str;
fn build<'a>(&self, val: &'a usize) -> Self::InstanceForBuilder<'a> {
MyInstance { val }
}
}
impl<'a> Instance<'a> for MyInstance<'a> {
type InstanceProperty = bool;
}
fn main() {
let builder = MyBuilder;
build_with_42_for_bool_instance(builder); // TODO: Doesn't work
}
Differences between Tower and Motore
The Service
trait in motore
is different from that in tower. In the motore, the metadata and extension is moved to the context argument passed along the call chain, and the request is kept by another argument.
The poll_ready
is hided. In fact, actix-web
shares similar opinions with motore
, it provides actix_web::dev::forward_ready and actix_web::dev::always_ready to help developers reduce boilerplate codes.
Erase message from the Request in Tonic
In tower, the type of the message is erased in tonic::service::interceptor::Interceptor
by the following magical code. The interceptor cannot modify the message of requests.
rust
// tonic/tonic/src/service/interceptor.rs
impl<S, F, ReqBody, ResBody> Service<http::Request<ReqBody>> for InterceptedService<S, F>
{
fn call(&mut self, req: http::Request<ReqBody>) -> Self::Future {
// It is bad practice to modify the body (i.e. Message) of the request via an interceptor.
// To avoid exposing the body of the request to the interceptor function, we first remove it
// here, allow the interceptor to modify the metadata and extensions, and then recreate the
// HTTP request with the body. Tonic requests do not preserve the URI, HTTP version, and
// HTTP method of the HTTP request, so we extract them here and then add them back in below.
let uri = req.uri().clone();
let method = req.method().clone();
let version = req.version();
let req = crate::Request::from_http(req);
let (metadata, extensions, msg) = req.into_parts();
// Here the `msg` is erased from the `Request`:)
match self
.f
.call(crate::Request::from_parts(metadata, extensions, ()))
{
Ok(req) => {
let (metadata, extensions, _) = req.into_parts();
let req = crate::Request::from_parts(metadata, extensions, msg);
let req = req.into_http(uri, method, version, SanitizeHeaders::No);
ResponseFuture::future(self.inner.call(req))
}
Err(status) => ResponseFuture::status(status),
}
}
}