88 lines
2.2 KiB
Rust
88 lines
2.2 KiB
Rust
// These are simplifications of the tower traits by the same name:
|
|
|
|
pub trait Service<Request> {
|
|
type Response;
|
|
}
|
|
|
|
pub trait Layer<C> {
|
|
type Service;
|
|
}
|
|
|
|
// Any type will do here:
|
|
|
|
pub struct Req;
|
|
pub struct Res;
|
|
|
|
// This is encoding a trait alias.
|
|
|
|
pub trait ParticularService:
|
|
Service<Req, Response = Res> {
|
|
}
|
|
|
|
impl<T> ParticularService for T
|
|
where
|
|
T: Service<Req, Response = Res>,
|
|
{
|
|
}
|
|
|
|
// This is also a trait alias.
|
|
// The weird = <Self as ...> bound is there so that users of the trait do not
|
|
// need to repeat the bounds. See https://github.com/rust-lang/rust/issues/20671
|
|
// for context, and in particular the workaround in:
|
|
// https://github.com/rust-lang/rust/issues/20671#issuecomment-529752828
|
|
|
|
pub trait ParticularServiceLayer<C>:
|
|
Layer<C, Service = <Self as ParticularServiceLayer<C>>::Service>
|
|
{
|
|
type Service: ParticularService;
|
|
}
|
|
|
|
impl<T, C> ParticularServiceLayer<C> for T
|
|
where
|
|
T: Layer<C>,
|
|
T::Service: ParticularService,
|
|
{
|
|
type Service = T::Service;
|
|
}
|
|
|
|
// These are types that implement the traits that the trait aliases refer to.
|
|
// They should also implement the alias traits due to the blanket impls.
|
|
|
|
struct ALayer<C>(C);
|
|
impl<C> Layer<C> for ALayer<C> {
|
|
type Service = AService;
|
|
}
|
|
|
|
struct AService;
|
|
impl Service<Req> for AService {
|
|
// However, AService does _not_ meet the blanket implementation,
|
|
// since its Response type is bool, not Res as it should be.
|
|
type Response = bool;
|
|
}
|
|
|
|
// This is a wrapper type around ALayer that uses the trait alias
|
|
// as a way to communicate the requirements of the provided types.
|
|
struct Client<C>(C);
|
|
|
|
// The method and the free-standing function below both have the same bounds.
|
|
|
|
impl<C> Client<C>
|
|
where
|
|
ALayer<C>: ParticularServiceLayer<C>,
|
|
{
|
|
fn check(&self) {}
|
|
}
|
|
|
|
fn check<C>(_: C) where ALayer<C>: ParticularServiceLayer<C> {}
|
|
|
|
// But, they give very different error messages.
|
|
|
|
fn main() {
|
|
// This gives a very poor error message that does nothing to point the user
|
|
// at the underlying cause of why the types involved do not meet the bounds.
|
|
Client(()).check(); //~ ERROR E0599
|
|
|
|
// This gives a good(ish) error message that points the user at _why_ the
|
|
// bound isn't met, and thus how they might fix it.
|
|
check(()); //~ ERROR E0271
|
|
}
|