-
Notifications
You must be signed in to change notification settings - Fork 1.6k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
New lint: unit_as_impl_trait
#13925
base: master
Are you sure you want to change the base?
New lint: unit_as_impl_trait
#13925
Conversation
6b2e881
to
6c3ca5b
Compare
I feel that having On the other hand, the suggestion is not too helpful in my opinion, given my reasoning stated above. |
Yes, and as seen in the issue, it exists in production, for example in Axum. |
"consider being explicit, and terminate the body with `()`", | ||
); | ||
} else if let Some(last) = block.stmts.last() { | ||
diag.span_note(last.span, "this statement evaluates to `()` because it ends with `;`") |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Shouldn't we in this case at least check that the expression the semicolon consumed also implements the required trait? Otherwise I could create an example that won't compile when removing the semicolon, and reducing false positives is always a good thing.
/// if returning `()` is intentional. | ||
#[clippy::version = "1.85.0"] | ||
pub UNIT_AS_IMPL_TRAIT, | ||
suspicious, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Given that this is warn by default, I'd like to make sure we remove all possible false positives:
- Whenever the swallowed expression would not impl the returned trait
- Perhaps there are some
#[cfg(..)]
shenanigans going on that change the final call's return to()
because there's nothing meaningful to return on some systems
diag.span_note(expr.span, "this expression evaluates to `()`") | ||
.span_help( | ||
expr.span.shrink_to_hi(), | ||
"consider being explicit, and terminate the body with `()`", |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
also the messaging might be shorter:
"consider being explicit, and terminate the body with `()`", | |
"add `; ()` for explicitness", |
"if this is intentional, consider being explicit, and terminate the body with `()`", | ||
); | ||
} else { | ||
diag.span_note(block.span, "the empty body evaluates to `()`") |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Especially in this case, having some #[cfg(..)]
in the plaintext source would suggest this is intentional.
"this function returns `()` which implements the required trait", | ||
|diag| { | ||
if let Some(expr) = block.expr { | ||
diag.span_note(expr.span, "this expression evaluates to `()`") |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Here it gets tricky, because the expression might return something else on a different target.
() implements traits from And I think it's safe to assume that people in general won't make MyHash and MyEq traits just to avoid this. |
@rustbot author |
6c3ca5b
to
caf5930
Compare
unit_as_impl_trait
caf5930
to
328069a
Compare
328069a
to
b2e952c
Compare
@@ -0,0 +1,45 @@ | |||
#![warn(clippy::unit_as_impl_trait)] | |||
#![allow(clippy::unused_unit)] | |||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
How about a test against the type not implementing the trait, e.g.
struct NoEq;
fn false_positive_no_trait_impl() -> impl Eq { NoEq; }
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Sure, will do. I'll have to investigate how to properly check that the type of the expression implements the bounds of the RPIT, probably similar to what is done in needless_borrows_for_generic_arg
which also has to do a similar check.
This lint implements the suggestion from #13909, and requires that an explicit
()
is used when a function returns()
while declaring an impl trait return type.I have used specialized messages for three different situations:
()
One should note that adding the explicit
()
as suggested will cause theunused_unit
lint (default: warn) to trigger. This should be possible to silence it inside functions returning impl traits if needed, to avoid having contradictory suggestions. Thoughts?Close #13909
changelog: [
unit_as_impl_trait
]: new lint