-
Notifications
You must be signed in to change notification settings - Fork 13.2k
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
Better error for missing tuple pattern in args #45069
Conversation
Thanks for the pull request, and welcome! The Rust team is excited to review your changes, and you should hear from @eddyb (or someone else) soon. If any changes to this PR are deemed necessary, please add them as extra commits. This ensures that the reviewer can see what has changed since they last reviewed the code. Due to the way GitHub handles out-of-date commits, this should also make it reasonably obvious what issues have or haven't been addressed. Large or tricky changes may require several passes of review and changes. Please see the contribution instructions for more information. |
None | ||
}; | ||
|
||
// FIXME(#44150): Expand this to "N args expected but a N-tuple found." |
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.
Why does this say "FIXME"?
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.
This PR improves message for "expected |(x, y)|, found |x, y|" case but not "expected |x, y|, found |(x, y)|" case. In the latter case, found_trait_ref
should be FnMut<((_, _),)>
, but it becomes FnMut<(_,)>
and I don't know how to fix.
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.
@nikomatsakis, I believe @sinkuu is referring to the following case:
18 | f(|(x, y)| x + y);
| ^ -------- takes a single tuple as an argument
| | |
| | help: consider changing to: `|x, y|`
| |
| expected closure that takes 2 arguments
We should create an issue after this PR lands to track that case.
This seems like a great thing to fix. I am wondering though if we should tweak the output. I personally found the new message a bit hard to parse, and I think it could be confusing. Instead of changing the actual text of the error message, I wonder if it'd be better to offer a suggestion. I am imagining something like this:
I imagine it looking something like this in the end:
Thoughts @estebank? To do this, you would basically invoke We could also keep things just as they are, but also include the suggestion. I'm not sure which is better, but I definitely the suggestion is a good idea, since I think it will have a more obvious meaning to people than talking about 2-tuples. |
Another option is to do something more dramatic: error[E0593]: closure takes multiple arguments, but a tuple argument is required
--> test.rs:5:40
|
5 | let it = v.into_iter().enumerate().map(|i, x| i);
| ^^^ ------ consider changing to `|(i, x)|`
| |
| expected closure that takes 1 argument, a 2-tuple |
Added suggestions. |
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.
This is awesome! What do you think about my suggest changes to the error message wording?
cc @estebank -- this is definitely the kind of thing we should have a style guide for!
Cow::from(args_str(found)) | ||
}, | ||
if expected_tuple.is_some() { | ||
Cow::from("a tuple argument") |
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.
I think I figured out what is bothering me about this wording. The construction is not quite parallel. It says "multiple arguments" are expected, but then says "but a tuple argument", which is not obviously the opposite. Also, our usual construction is "expected X, but found Y", whereas the message here says "found X, but Y is required" (pre-existing, I know, but I still find it a bit surprising).
Perhaps this?
expected a closure that takes a single tuple as argument but found a closure that takes N distinct arguments
Maybe that's too wordy. We could tweak it, though it goes a bit away from the "expected X but found Y" form:
expected closure to take a single tuple as argument but it takes N distinct arguments
or, maybe better:
closure is expected to take a single tuple as argument, but it takes N distinct arguments
(I would probably change the existing message that doesn't mention tuples to match, as well.)
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.
It looks better, thanks! Reworded the message to the last one.
--> $DIR/closure-arg-count.rs:21:53 | ||
| | ||
21 | let _it = vec![1, 2, 3].into_iter().enumerate().map(|i: usize, x| i); | ||
| ^^^ ------------- help: consider changing to: `|(i, x): (usize, _)|` |
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.
this looks great
20 | let _it = vec![1, 2, 3].into_iter().enumerate().map(|i, x| i); | ||
| ^^^ ------ help: consider changing to: `|(i, x)|` | ||
| | | ||
| expected closure that takes 1 argument, a 2-tuple |
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.
This is looking really good! I hope you don't mind me nitpicking some more though. =)
Reading this "expected closure that takes 1 argument, a 2-tuple" -- which I think I may even have initially suggested -- I see now that it is actually ambiguous. In particular, the intention was something like "that takes 1 argument, which is a 2-tuple", but it could also be parsed as a list, like "that takes 1 argument and a 2-tuple". In fact, this second interpretation is what I got this time, and I had to re-read it a few times to understand what was meant.
Therefore, I suggest:
"expected closure that takes a single tuple as argument"
which echoes the wording from above
None | ||
}; | ||
|
||
// FIXME(#44150): Expand this to "N args expected but a N-tuple found." |
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.
@nikomatsakis, I believe @sinkuu is referring to the following case:
18 | f(|(x, y)| x + y);
| ^ -------- takes a single tuple as an argument
| | |
| | help: consider changing to: `|x, y|`
| |
| expected closure that takes 2 arguments
We should create an issue after this PR lands to track that case.
--> $DIR/closure-arg-count.rs:20:53 | ||
| | ||
20 | let _it = vec![1, 2, 3].into_iter().enumerate().map(|i, x| i); | ||
| ^^^ ------ help: consider changing to: `|(i, x)|` |
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.
Could you modify the suggestion text to be freestanding without the snippet? Something along the lines of consider changing the closure to accept a tuple
. rustc
is not the only way these messages can be presented, so they should make sense even when the suggested code is not presented next to it.
@bors r+ |
📌 Commit f577847 has been approved by |
Better error for missing tuple pattern in args #44150 Before: ``` error[E0593]: closure takes 2 arguments but 1 argument is required --> test.rs:5:40 | 5 | let it = v.into_iter().enumerate().map(|i, x| i); | ^^^ -------- takes 2 arguments | | | expected closure that takes 1 argument ``` After: ``` error[E0593]: closure takes 2 arguments but a 2-tuple is required --> test.rs:5:40 | 5 | let it = v.into_iter().enumerate().map(|i, x| i); | ^^^ ------ takes 2 arguments | | | expected closure that takes a 2-tuple ```
☀️ Test successful - status-appveyor, status-travis |
Seeing this error message actually brightened up my day. So many times have I made this mistake... |
#44150
Before:
After: