Auto merge of #68522 - estebank:impl-trait-sugg-2, r=oli-obk
Further improve `impl Trait`/`dyn Trait` suggestions After reading [_Returning Trait Objects_ by Bryce Fisher-Fleig](https://bryce.fisher-fleig.org/blog/returning-trait-objects/), [I noticed that](https://www.reddit.com/r/rust/comments/esueur/returning_trait_objects/ffczl4k/) #68195 had a few bugs due to not ignoring `ty::Error`. - Account for `ty::Error`. - Account for `if`/`else` and `match` blocks when pointing at return types and referencing their types. - Increase the multiline suggestion output from 6 lines to 20.
This commit is contained in:
commit
3d8778d767
5 changed files with 262 additions and 23 deletions
|
|
@ -22,6 +22,39 @@ fn bal() -> dyn Trait { //~ ERROR E0746
|
|||
}
|
||||
42
|
||||
}
|
||||
fn bax() -> dyn Trait { //~ ERROR E0746
|
||||
if true {
|
||||
Struct
|
||||
} else {
|
||||
42
|
||||
}
|
||||
}
|
||||
fn bam() -> Box<dyn Trait> {
|
||||
if true {
|
||||
return Struct; //~ ERROR mismatched types
|
||||
}
|
||||
42 //~ ERROR mismatched types
|
||||
}
|
||||
fn baq() -> Box<dyn Trait> {
|
||||
if true {
|
||||
return 0; //~ ERROR mismatched types
|
||||
}
|
||||
42 //~ ERROR mismatched types
|
||||
}
|
||||
fn baz() -> Box<dyn Trait> {
|
||||
if true {
|
||||
Struct //~ ERROR mismatched types
|
||||
} else {
|
||||
42 //~ ERROR mismatched types
|
||||
}
|
||||
}
|
||||
fn baw() -> Box<dyn Trait> {
|
||||
if true {
|
||||
0 //~ ERROR mismatched types
|
||||
} else {
|
||||
42 //~ ERROR mismatched types
|
||||
}
|
||||
}
|
||||
|
||||
// Suggest using `impl Trait`
|
||||
fn bat() -> dyn Trait { //~ ERROR E0746
|
||||
|
|
@ -30,5 +63,12 @@ fn bat() -> dyn Trait { //~ ERROR E0746
|
|||
}
|
||||
42
|
||||
}
|
||||
fn bay() -> dyn Trait { //~ ERROR E0746
|
||||
if true {
|
||||
0
|
||||
} else {
|
||||
42
|
||||
}
|
||||
}
|
||||
|
||||
fn main() {}
|
||||
|
|
|
|||
|
|
@ -96,7 +96,154 @@ LL | Box::new(42)
|
|||
|
|
||||
|
||||
error[E0746]: return type cannot have an unboxed trait object
|
||||
--> $DIR/dyn-trait-return-should-be-impl-trait.rs:27:13
|
||||
--> $DIR/dyn-trait-return-should-be-impl-trait.rs:25:13
|
||||
|
|
||||
LL | fn bax() -> dyn Trait {
|
||||
| ^^^^^^^^^ doesn't have a size known at compile-time
|
||||
|
|
||||
= note: for information on trait objects, see <https://doc.rust-lang.org/book/ch17-02-trait-objects.html#using-trait-objects-that-allow-for-values-of-different-types>
|
||||
= note: if all the returned values were of the same type you could use `impl Trait` as the return type
|
||||
= note: for information on `impl Trait`, see <https://doc.rust-lang.org/book/ch10-02-traits.html#returning-types-that-implement-traits>
|
||||
= note: you can create a new `enum` with a variant for each returned type
|
||||
help: return a boxed trait object instead
|
||||
|
|
||||
LL | fn bax() -> Box<dyn Trait> {
|
||||
LL | if true {
|
||||
LL | Box::new(Struct)
|
||||
LL | } else {
|
||||
LL | Box::new(42)
|
||||
|
|
||||
|
||||
error[E0308]: mismatched types
|
||||
--> $DIR/dyn-trait-return-should-be-impl-trait.rs:34:16
|
||||
|
|
||||
LL | fn bam() -> Box<dyn Trait> {
|
||||
| -------------- expected `std::boxed::Box<(dyn Trait + 'static)>` because of return type
|
||||
LL | if true {
|
||||
LL | return Struct;
|
||||
| ^^^^^^
|
||||
| |
|
||||
| expected struct `std::boxed::Box`, found struct `Struct`
|
||||
| help: store this in the heap by calling `Box::new`: `Box::new(Struct)`
|
||||
|
|
||||
= note: expected struct `std::boxed::Box<(dyn Trait + 'static)>`
|
||||
found struct `Struct`
|
||||
= note: for more on the distinction between the stack and the heap, read https://doc.rust-lang.org/book/ch15-01-box.html, https://doc.rust-lang.org/rust-by-example/std/box.html, and https://doc.rust-lang.org/std/boxed/index.html
|
||||
|
||||
error[E0308]: mismatched types
|
||||
--> $DIR/dyn-trait-return-should-be-impl-trait.rs:36:5
|
||||
|
|
||||
LL | fn bam() -> Box<dyn Trait> {
|
||||
| -------------- expected `std::boxed::Box<(dyn Trait + 'static)>` because of return type
|
||||
...
|
||||
LL | 42
|
||||
| ^^
|
||||
| |
|
||||
| expected struct `std::boxed::Box`, found integer
|
||||
| help: store this in the heap by calling `Box::new`: `Box::new(42)`
|
||||
|
|
||||
= note: expected struct `std::boxed::Box<(dyn Trait + 'static)>`
|
||||
found type `{integer}`
|
||||
= note: for more on the distinction between the stack and the heap, read https://doc.rust-lang.org/book/ch15-01-box.html, https://doc.rust-lang.org/rust-by-example/std/box.html, and https://doc.rust-lang.org/std/boxed/index.html
|
||||
|
||||
error[E0308]: mismatched types
|
||||
--> $DIR/dyn-trait-return-should-be-impl-trait.rs:40:16
|
||||
|
|
||||
LL | fn baq() -> Box<dyn Trait> {
|
||||
| -------------- expected `std::boxed::Box<(dyn Trait + 'static)>` because of return type
|
||||
LL | if true {
|
||||
LL | return 0;
|
||||
| ^
|
||||
| |
|
||||
| expected struct `std::boxed::Box`, found integer
|
||||
| help: store this in the heap by calling `Box::new`: `Box::new(0)`
|
||||
|
|
||||
= note: expected struct `std::boxed::Box<(dyn Trait + 'static)>`
|
||||
found type `{integer}`
|
||||
= note: for more on the distinction between the stack and the heap, read https://doc.rust-lang.org/book/ch15-01-box.html, https://doc.rust-lang.org/rust-by-example/std/box.html, and https://doc.rust-lang.org/std/boxed/index.html
|
||||
|
||||
error[E0308]: mismatched types
|
||||
--> $DIR/dyn-trait-return-should-be-impl-trait.rs:42:5
|
||||
|
|
||||
LL | fn baq() -> Box<dyn Trait> {
|
||||
| -------------- expected `std::boxed::Box<(dyn Trait + 'static)>` because of return type
|
||||
...
|
||||
LL | 42
|
||||
| ^^
|
||||
| |
|
||||
| expected struct `std::boxed::Box`, found integer
|
||||
| help: store this in the heap by calling `Box::new`: `Box::new(42)`
|
||||
|
|
||||
= note: expected struct `std::boxed::Box<(dyn Trait + 'static)>`
|
||||
found type `{integer}`
|
||||
= note: for more on the distinction between the stack and the heap, read https://doc.rust-lang.org/book/ch15-01-box.html, https://doc.rust-lang.org/rust-by-example/std/box.html, and https://doc.rust-lang.org/std/boxed/index.html
|
||||
|
||||
error[E0308]: mismatched types
|
||||
--> $DIR/dyn-trait-return-should-be-impl-trait.rs:46:9
|
||||
|
|
||||
LL | fn baz() -> Box<dyn Trait> {
|
||||
| -------------- expected `std::boxed::Box<(dyn Trait + 'static)>` because of return type
|
||||
LL | if true {
|
||||
LL | Struct
|
||||
| ^^^^^^
|
||||
| |
|
||||
| expected struct `std::boxed::Box`, found struct `Struct`
|
||||
| help: store this in the heap by calling `Box::new`: `Box::new(Struct)`
|
||||
|
|
||||
= note: expected struct `std::boxed::Box<(dyn Trait + 'static)>`
|
||||
found struct `Struct`
|
||||
= note: for more on the distinction between the stack and the heap, read https://doc.rust-lang.org/book/ch15-01-box.html, https://doc.rust-lang.org/rust-by-example/std/box.html, and https://doc.rust-lang.org/std/boxed/index.html
|
||||
|
||||
error[E0308]: mismatched types
|
||||
--> $DIR/dyn-trait-return-should-be-impl-trait.rs:48:9
|
||||
|
|
||||
LL | fn baz() -> Box<dyn Trait> {
|
||||
| -------------- expected `std::boxed::Box<(dyn Trait + 'static)>` because of return type
|
||||
...
|
||||
LL | 42
|
||||
| ^^
|
||||
| |
|
||||
| expected struct `std::boxed::Box`, found integer
|
||||
| help: store this in the heap by calling `Box::new`: `Box::new(42)`
|
||||
|
|
||||
= note: expected struct `std::boxed::Box<(dyn Trait + 'static)>`
|
||||
found type `{integer}`
|
||||
= note: for more on the distinction between the stack and the heap, read https://doc.rust-lang.org/book/ch15-01-box.html, https://doc.rust-lang.org/rust-by-example/std/box.html, and https://doc.rust-lang.org/std/boxed/index.html
|
||||
|
||||
error[E0308]: mismatched types
|
||||
--> $DIR/dyn-trait-return-should-be-impl-trait.rs:53:9
|
||||
|
|
||||
LL | fn baw() -> Box<dyn Trait> {
|
||||
| -------------- expected `std::boxed::Box<(dyn Trait + 'static)>` because of return type
|
||||
LL | if true {
|
||||
LL | 0
|
||||
| ^
|
||||
| |
|
||||
| expected struct `std::boxed::Box`, found integer
|
||||
| help: store this in the heap by calling `Box::new`: `Box::new(0)`
|
||||
|
|
||||
= note: expected struct `std::boxed::Box<(dyn Trait + 'static)>`
|
||||
found type `{integer}`
|
||||
= note: for more on the distinction between the stack and the heap, read https://doc.rust-lang.org/book/ch15-01-box.html, https://doc.rust-lang.org/rust-by-example/std/box.html, and https://doc.rust-lang.org/std/boxed/index.html
|
||||
|
||||
error[E0308]: mismatched types
|
||||
--> $DIR/dyn-trait-return-should-be-impl-trait.rs:55:9
|
||||
|
|
||||
LL | fn baw() -> Box<dyn Trait> {
|
||||
| -------------- expected `std::boxed::Box<(dyn Trait + 'static)>` because of return type
|
||||
...
|
||||
LL | 42
|
||||
| ^^
|
||||
| |
|
||||
| expected struct `std::boxed::Box`, found integer
|
||||
| help: store this in the heap by calling `Box::new`: `Box::new(42)`
|
||||
|
|
||||
= note: expected struct `std::boxed::Box<(dyn Trait + 'static)>`
|
||||
found type `{integer}`
|
||||
= note: for more on the distinction between the stack and the heap, read https://doc.rust-lang.org/book/ch15-01-box.html, https://doc.rust-lang.org/rust-by-example/std/box.html, and https://doc.rust-lang.org/std/boxed/index.html
|
||||
|
||||
error[E0746]: return type cannot have an unboxed trait object
|
||||
--> $DIR/dyn-trait-return-should-be-impl-trait.rs:60:13
|
||||
|
|
||||
LL | fn bat() -> dyn Trait {
|
||||
| ^^^^^^^^^ doesn't have a size known at compile-time
|
||||
|
|
@ -107,7 +254,19 @@ help: return `impl Trait` instead, as all return paths are of type `{integer}`,
|
|||
LL | fn bat() -> impl Trait {
|
||||
| ^^^^^^^^^^
|
||||
|
||||
error: aborting due to 9 previous errors
|
||||
error[E0746]: return type cannot have an unboxed trait object
|
||||
--> $DIR/dyn-trait-return-should-be-impl-trait.rs:66:13
|
||||
|
|
||||
LL | fn bay() -> dyn Trait {
|
||||
| ^^^^^^^^^ doesn't have a size known at compile-time
|
||||
|
|
||||
= note: for information on `impl Trait`, see <https://doc.rust-lang.org/book/ch10-02-traits.html#returning-types-that-implement-traits>
|
||||
help: return `impl Trait` instead, as all return paths are of type `{integer}`, which implements `Trait`
|
||||
|
|
||||
LL | fn bay() -> impl Trait {
|
||||
| ^^^^^^^^^^
|
||||
|
||||
error: aborting due to 19 previous errors
|
||||
|
||||
Some errors have detailed explanations: E0277, E0308, E0746.
|
||||
For more information about an error, try `rustc --explain E0277`.
|
||||
|
|
|
|||
|
|
@ -5,7 +5,7 @@ LL | fn should_ret_unit() -> impl T {
|
|||
| ^^^^^^ the trait `T` is not implemented for `()`
|
||||
LL |
|
||||
LL | panic!()
|
||||
| -------- this returned value is of type `()`
|
||||
| -------- this returned value is of type `!`
|
||||
|
|
||||
= note: the return type of a function must have a statically known size
|
||||
= note: this error originates in a macro outside of the current crate (in Nightly builds, run with -Z external-macro-backtrace for more info)
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue