From de346adfa3bf1d65e4a41ad627dcf293dd1d21d0 Mon Sep 17 00:00:00 2001 From: Eric Huss Date: Sun, 15 Dec 2024 15:11:35 -0800 Subject: [PATCH] Rewrite do_not_recommend example This rewrites the example to use a more realistic example that actually demonstrates an improvement in the error message. --- src/attributes/diagnostics.md | 112 +++++++++++++++++++++++----------- 1 file changed, 78 insertions(+), 34 deletions(-) diff --git a/src/attributes/diagnostics.md b/src/attributes/diagnostics.md index 5e0c95b6e..8210b9d48 100644 --- a/src/attributes/diagnostics.md +++ b/src/attributes/diagnostics.md @@ -576,58 +576,102 @@ The attribute should be placed on a [trait implementation item][trait-impl], tho r[attributes.diagnostic.do_not_recommend.syntax] The attribute does not accept any arguments, though unexpected arguments are not considered as an error. -In this example, +In the following example, there is a trait called `AsExpression` which is used for casting arbitrary types to the `Expression` type used in an SQL library. There is a method called `check` which takes an `AsExpression`. ```rust,compile_fail,E0277 -trait Foo {} - -impl Foo for T where T: Send {} +# pub trait Expression { +# type SqlType; +# } +# +# pub trait AsExpression { +# type Expression: Expression; +# } +# +# pub struct Text; +# pub struct Integer; +# +# pub struct Bound(T); +# pub struct SelectInt; +# +# impl Expression for SelectInt { +# type SqlType = Integer; +# } +# +# impl Expression for Bound { +# type SqlType = T; +# } +# +# impl AsExpression for i32 { +# type Expression = Bound; +# } +# +# impl AsExpression for &'_ str { +# type Expression = Bound; +# } +# +# impl Foo for T where T: Expression {} + +// Uncomment this line to change the recommendation. +// #[diagnostic::do_not_recommend] +impl AsExpression for T +where + T: Expression, +{ + type Expression = T; +} -fn needs_foo() {} +trait Foo: Expression + Sized { + fn check(&self, _: T) -> ::SqlType>>::Expression + where + T: AsExpression, + { + todo!() + } +} fn main() { - // Mutable pointers do not implement `Send`, so this will not work. - needs_foo::<*mut ()>(); + SelectInt.check("bar"); } ``` -the compiler may generate an error message about the `Send` bound in the impl which looks like this: +The `SelectInt` type's `check` method is expecting an `Integer` type. Calling it with an i32 type works, as it gets converted to an `Integer` by the `AsExpression` trait. However, calling it with a string does not, and generates a an error that may look like this: ```text -error[E0277]: the trait bound `*mut (): Foo` is not satisfied - --> src/main.rs:9:17 - | -9 | needs_foo::<*mut ()>(); - | ^^^^^^^ the trait `Send` is not implemented for `*mut ()` - | -note: required for `*mut ()` to implement `Foo` - --> src/main.rs:3:9 - | -3 | impl Foo for T where T: Send {} - | ^^^ ^ ---- unsatisfied trait bound introduced here -note: required by a bound in `needs_foo` - --> src/main.rs:5:17 - | -5 | fn needs_foo() {} - | ^^^ required by this bound in `needs_foo` +error[E0277]: the trait bound `&str: Expression` is not satisfied + --> src/main.rs:53:15 + | +53 | SelectInt.check("bar"); + | ^^^^^ the trait `Expression` is not implemented for `&str` + | + = help: the following other types implement trait `Expression`: + Bound + SelectInt +note: required for `&str` to implement `AsExpression` + --> src/main.rs:45:13 + | +45 | impl AsExpression for T + | ^^^^^^^^^^^^^^^^ ^ +46 | where +47 | T: Expression, + | ------------------------ unsatisfied trait bound introduced here ``` -By adding the `#[diagnostic::do_no_recommend]` attribute to the `impl`, the message would no longer suggest it: +By adding the `#[diagnostic::do_no_recommend]` attribute to the blanket `impl` for `AsExpression`, the message changes to: ```text -error[E0277]: the trait bound `*mut (): Foo` is not satisfied - --> src/main.rs:11:17 +error[E0277]: the trait bound `&str: AsExpression` is not satisfied + --> src/main.rs:53:15 | -11 | needs_foo::<*mut ()>(); - | ^^^^^^^ the trait `Foo` is not implemented for `*mut ()` +53 | SelectInt.check("bar"); + | ^^^^^ the trait `AsExpression` is not implemented for `&str` | -note: required by a bound in `needs_foo` - --> src/main.rs:7:17 - | -7 | fn needs_foo() {} - | ^^^ required by this bound in `needs_foo` + = help: the trait `AsExpression` is not implemented for `&str` + but trait `AsExpression` is implemented for it + = help: for that trait implementation, expected `Text`, found `Integer` ``` +The first error message includes a somewhat confusing error message about the relationship of `&str` and `Expression`, as well as the unsatisfied trait bound in the blanket impl. After adding `#[diagnostic::do_no_recommend]`, it no longer considers the blanket impl for the recommendation. The message should be a little clearer, with an indication that a string cannot be converted to an `Integer`. + [Clippy]: https://github.com/rust-lang/rust-clippy [_MetaListNameValueStr_]: ../attributes.md#meta-item-attribute-syntax [_MetaListPaths_]: ../attributes.md#meta-item-attribute-syntax