mirror of
https://github.com/rust-lang/rust.git
synced 2024-11-22 14:55:26 +00:00
Rollup merge of #91530 - bobrippling:suggest-1-tuple-parens, r=camelid
Suggest 1-tuple parentheses on exprs without existing parens A follow-on from #86116, split out from #90677. This alters the suggestion to add a trailing comma to create a 1-tuple - previously we would only apply this if the relevant expression was parenthesised. We now make the suggestion regardless of parentheses, which reduces the fragility of the check (w.r.t formatting). e.g. ```rust let a: Option<(i32,)> = Some(3); ``` gets the below suggestion: ```rust let a: Option<(i32,)> = Some((3,)); // ^ ^^ ``` This change also improves the suggestion in other ways, such as by only making the suggestion if the types would match after the suggestion is applied and making the suggestion a multipart suggestion.
This commit is contained in:
commit
557d300e1b
@ -2044,19 +2044,8 @@ impl<'a, 'tcx> InferCtxt<'a, 'tcx> {
|
|||||||
// If a tuple of length one was expected and the found expression has
|
// If a tuple of length one was expected and the found expression has
|
||||||
// parentheses around it, perhaps the user meant to write `(expr,)` to
|
// parentheses around it, perhaps the user meant to write `(expr,)` to
|
||||||
// build a tuple (issue #86100)
|
// build a tuple (issue #86100)
|
||||||
(ty::Tuple(_), _) if expected.tuple_fields().count() == 1 => {
|
(ty::Tuple(_), _) => {
|
||||||
if let Ok(code) = self.tcx.sess().source_map().span_to_snippet(span) {
|
self.emit_tuple_wrap_err(&mut err, span, found, expected)
|
||||||
if let Some(code) =
|
|
||||||
code.strip_prefix('(').and_then(|s| s.strip_suffix(')'))
|
|
||||||
{
|
|
||||||
err.span_suggestion(
|
|
||||||
span,
|
|
||||||
"use a trailing comma to create a tuple with one element",
|
|
||||||
format!("({},)", code),
|
|
||||||
Applicability::MaybeIncorrect,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
// If a character was expected and the found expression is a string literal
|
// If a character was expected and the found expression is a string literal
|
||||||
// containing a single character, perhaps the user meant to write `'c'` to
|
// containing a single character, perhaps the user meant to write `'c'` to
|
||||||
@ -2119,6 +2108,41 @@ impl<'a, 'tcx> InferCtxt<'a, 'tcx> {
|
|||||||
diag
|
diag
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn emit_tuple_wrap_err(
|
||||||
|
&self,
|
||||||
|
err: &mut DiagnosticBuilder<'tcx>,
|
||||||
|
span: Span,
|
||||||
|
found: Ty<'tcx>,
|
||||||
|
expected: Ty<'tcx>,
|
||||||
|
) {
|
||||||
|
let [expected_tup_elem] = &expected.tuple_fields().collect::<Vec<_>>()[..]
|
||||||
|
else { return };
|
||||||
|
|
||||||
|
if !same_type_modulo_infer(expected_tup_elem, found) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let Ok(code) = self.tcx.sess().source_map().span_to_snippet(span)
|
||||||
|
else { return };
|
||||||
|
|
||||||
|
let msg = "use a trailing comma to create a tuple with one element";
|
||||||
|
if code.starts_with('(') && code.ends_with(')') {
|
||||||
|
let before_close = span.hi() - BytePos::from_u32(1);
|
||||||
|
err.span_suggestion(
|
||||||
|
span.with_hi(before_close).shrink_to_hi(),
|
||||||
|
msg,
|
||||||
|
",".into(),
|
||||||
|
Applicability::MachineApplicable,
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
err.multipart_suggestion(
|
||||||
|
msg,
|
||||||
|
vec![(span.shrink_to_lo(), "(".into()), (span.shrink_to_hi(), ",)".into())],
|
||||||
|
Applicability::MachineApplicable,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
fn values_str(
|
fn values_str(
|
||||||
&self,
|
&self,
|
||||||
values: ValuePairs<'tcx>,
|
values: ValuePairs<'tcx>,
|
||||||
|
@ -6,6 +6,10 @@ LL | const TUP: (usize,) = 5usize << 64;
|
|||||||
|
|
|
|
||||||
= note: expected tuple `(usize,)`
|
= note: expected tuple `(usize,)`
|
||||||
found type `usize`
|
found type `usize`
|
||||||
|
help: use a trailing comma to create a tuple with one element
|
||||||
|
|
|
||||||
|
LL | const TUP: (usize,) = (5usize << 64,);
|
||||||
|
| + ++
|
||||||
|
|
||||||
error: aborting due to previous error
|
error: aborting due to previous error
|
||||||
|
|
||||||
|
@ -10,6 +10,12 @@ fn main() {
|
|||||||
|
|
||||||
let _: Option<(i8,)> = Some();
|
let _: Option<(i8,)> = Some();
|
||||||
//~^ ERROR this enum variant takes 1 argument but 0 arguments were supplied
|
//~^ ERROR this enum variant takes 1 argument but 0 arguments were supplied
|
||||||
|
|
||||||
|
let _: Option<(i32,)> = Some(5_usize);
|
||||||
|
//~^ ERROR mismatched types
|
||||||
|
|
||||||
|
let _: Option<(i32,)> = Some((5_usize));
|
||||||
|
//~^ ERROR mismatched types
|
||||||
}
|
}
|
||||||
|
|
||||||
fn int_bool(_: (i32, bool)) {
|
fn int_bool(_: (i32, bool)) {
|
||||||
|
@ -15,7 +15,7 @@ LL | int_bool(1, 2);
|
|||||||
| expected 1 argument
|
| expected 1 argument
|
||||||
|
|
|
|
||||||
note: function defined here
|
note: function defined here
|
||||||
--> $DIR/args-instead-of-tuple-errors.rs:15:4
|
--> $DIR/args-instead-of-tuple-errors.rs:21:4
|
||||||
|
|
|
|
||||||
LL | fn int_bool(_: (i32, bool)) {
|
LL | fn int_bool(_: (i32, bool)) {
|
||||||
| ^^^^^^^^ --------------
|
| ^^^^^^^^ --------------
|
||||||
@ -28,6 +28,25 @@ LL | let _: Option<(i8,)> = Some();
|
|||||||
| |
|
| |
|
||||||
| expected 1 argument
|
| expected 1 argument
|
||||||
|
|
||||||
error: aborting due to 3 previous errors
|
error[E0308]: mismatched types
|
||||||
|
--> $DIR/args-instead-of-tuple-errors.rs:14:34
|
||||||
|
|
|
||||||
|
LL | let _: Option<(i32,)> = Some(5_usize);
|
||||||
|
| ^^^^^^^ expected tuple, found `usize`
|
||||||
|
|
|
||||||
|
= note: expected tuple `(i32,)`
|
||||||
|
found type `usize`
|
||||||
|
|
||||||
For more information about this error, try `rustc --explain E0061`.
|
error[E0308]: mismatched types
|
||||||
|
--> $DIR/args-instead-of-tuple-errors.rs:17:34
|
||||||
|
|
|
||||||
|
LL | let _: Option<(i32,)> = Some((5_usize));
|
||||||
|
| ^^^^^^^^^ expected tuple, found `usize`
|
||||||
|
|
|
||||||
|
= note: expected tuple `(i32,)`
|
||||||
|
found type `usize`
|
||||||
|
|
||||||
|
error: aborting due to 5 previous errors
|
||||||
|
|
||||||
|
Some errors have detailed explanations: E0061, E0308.
|
||||||
|
For more information about an error, try `rustc --explain E0061`.
|
||||||
|
@ -11,6 +11,12 @@ fn main() {
|
|||||||
let _: Option<()> = Some(());
|
let _: Option<()> = Some(());
|
||||||
//~^ ERROR this enum variant takes 1 argument but 0 arguments were supplied
|
//~^ ERROR this enum variant takes 1 argument but 0 arguments were supplied
|
||||||
|
|
||||||
|
let _: Option<(i32,)> = Some((3,));
|
||||||
|
//~^ ERROR mismatched types
|
||||||
|
|
||||||
|
let _: Option<(i32,)> = Some((3,));
|
||||||
|
//~^ ERROR mismatched types
|
||||||
|
|
||||||
two_ints((1, 2)); //~ ERROR this function takes 1 argument
|
two_ints((1, 2)); //~ ERROR this function takes 1 argument
|
||||||
|
|
||||||
with_generic((3, 4)); //~ ERROR this function takes 1 argument
|
with_generic((3, 4)); //~ ERROR this function takes 1 argument
|
||||||
|
@ -11,6 +11,12 @@ fn main() {
|
|||||||
let _: Option<()> = Some();
|
let _: Option<()> = Some();
|
||||||
//~^ ERROR this enum variant takes 1 argument but 0 arguments were supplied
|
//~^ ERROR this enum variant takes 1 argument but 0 arguments were supplied
|
||||||
|
|
||||||
|
let _: Option<(i32,)> = Some(3);
|
||||||
|
//~^ ERROR mismatched types
|
||||||
|
|
||||||
|
let _: Option<(i32,)> = Some((3));
|
||||||
|
//~^ ERROR mismatched types
|
||||||
|
|
||||||
two_ints(1, 2); //~ ERROR this function takes 1 argument
|
two_ints(1, 2); //~ ERROR this function takes 1 argument
|
||||||
|
|
||||||
with_generic(3, 4); //~ ERROR this function takes 1 argument
|
with_generic(3, 4); //~ ERROR this function takes 1 argument
|
||||||
|
@ -31,14 +31,40 @@ help: expected the unit value `()`; create it with empty parentheses
|
|||||||
LL | let _: Option<()> = Some(());
|
LL | let _: Option<()> = Some(());
|
||||||
| ++
|
| ++
|
||||||
|
|
||||||
|
error[E0308]: mismatched types
|
||||||
|
--> $DIR/args-instead-of-tuple.rs:14:34
|
||||||
|
|
|
||||||
|
LL | let _: Option<(i32,)> = Some(3);
|
||||||
|
| ^ expected tuple, found integer
|
||||||
|
|
|
||||||
|
= note: expected tuple `(i32,)`
|
||||||
|
found type `{integer}`
|
||||||
|
help: use a trailing comma to create a tuple with one element
|
||||||
|
|
|
||||||
|
LL | let _: Option<(i32,)> = Some((3,));
|
||||||
|
| + ++
|
||||||
|
|
||||||
|
error[E0308]: mismatched types
|
||||||
|
--> $DIR/args-instead-of-tuple.rs:17:34
|
||||||
|
|
|
||||||
|
LL | let _: Option<(i32,)> = Some((3));
|
||||||
|
| ^^^ expected tuple, found integer
|
||||||
|
|
|
||||||
|
= note: expected tuple `(i32,)`
|
||||||
|
found type `{integer}`
|
||||||
|
help: use a trailing comma to create a tuple with one element
|
||||||
|
|
|
||||||
|
LL | let _: Option<(i32,)> = Some((3,));
|
||||||
|
| +
|
||||||
|
|
||||||
error[E0061]: this function takes 1 argument but 2 arguments were supplied
|
error[E0061]: this function takes 1 argument but 2 arguments were supplied
|
||||||
--> $DIR/args-instead-of-tuple.rs:14:5
|
--> $DIR/args-instead-of-tuple.rs:20:5
|
||||||
|
|
|
|
||||||
LL | two_ints(1, 2);
|
LL | two_ints(1, 2);
|
||||||
| ^^^^^^^^ - - supplied 2 arguments
|
| ^^^^^^^^ - - supplied 2 arguments
|
||||||
|
|
|
|
||||||
note: function defined here
|
note: function defined here
|
||||||
--> $DIR/args-instead-of-tuple.rs:19:4
|
--> $DIR/args-instead-of-tuple.rs:25:4
|
||||||
|
|
|
|
||||||
LL | fn two_ints(_: (i32, i32)) {
|
LL | fn two_ints(_: (i32, i32)) {
|
||||||
| ^^^^^^^^ -------------
|
| ^^^^^^^^ -------------
|
||||||
@ -48,13 +74,13 @@ LL | two_ints((1, 2));
|
|||||||
| + +
|
| + +
|
||||||
|
|
||||||
error[E0061]: this function takes 1 argument but 2 arguments were supplied
|
error[E0061]: this function takes 1 argument but 2 arguments were supplied
|
||||||
--> $DIR/args-instead-of-tuple.rs:16:5
|
--> $DIR/args-instead-of-tuple.rs:22:5
|
||||||
|
|
|
|
||||||
LL | with_generic(3, 4);
|
LL | with_generic(3, 4);
|
||||||
| ^^^^^^^^^^^^ - - supplied 2 arguments
|
| ^^^^^^^^^^^^ - - supplied 2 arguments
|
||||||
|
|
|
|
||||||
note: function defined here
|
note: function defined here
|
||||||
--> $DIR/args-instead-of-tuple.rs:22:4
|
--> $DIR/args-instead-of-tuple.rs:28:4
|
||||||
|
|
|
|
||||||
LL | fn with_generic<T: Copy + Send>((a, b): (i32, T)) {
|
LL | fn with_generic<T: Copy + Send>((a, b): (i32, T)) {
|
||||||
| ^^^^^^^^^^^^ ----------------
|
| ^^^^^^^^^^^^ ----------------
|
||||||
@ -64,13 +90,13 @@ LL | with_generic((3, 4));
|
|||||||
| + +
|
| + +
|
||||||
|
|
||||||
error[E0061]: this function takes 1 argument but 2 arguments were supplied
|
error[E0061]: this function takes 1 argument but 2 arguments were supplied
|
||||||
--> $DIR/args-instead-of-tuple.rs:25:9
|
--> $DIR/args-instead-of-tuple.rs:31:9
|
||||||
|
|
|
|
||||||
LL | with_generic(a, b);
|
LL | with_generic(a, b);
|
||||||
| ^^^^^^^^^^^^ - - supplied 2 arguments
|
| ^^^^^^^^^^^^ - - supplied 2 arguments
|
||||||
|
|
|
|
||||||
note: function defined here
|
note: function defined here
|
||||||
--> $DIR/args-instead-of-tuple.rs:22:4
|
--> $DIR/args-instead-of-tuple.rs:28:4
|
||||||
|
|
|
|
||||||
LL | fn with_generic<T: Copy + Send>((a, b): (i32, T)) {
|
LL | fn with_generic<T: Copy + Send>((a, b): (i32, T)) {
|
||||||
| ^^^^^^^^^^^^ ----------------
|
| ^^^^^^^^^^^^ ----------------
|
||||||
@ -79,6 +105,7 @@ help: use parentheses to construct a tuple
|
|||||||
LL | with_generic((a, b));
|
LL | with_generic((a, b));
|
||||||
| + +
|
| + +
|
||||||
|
|
||||||
error: aborting due to 6 previous errors
|
error: aborting due to 8 previous errors
|
||||||
|
|
||||||
For more information about this error, try `rustc --explain E0061`.
|
Some errors have detailed explanations: E0061, E0308.
|
||||||
|
For more information about an error, try `rustc --explain E0061`.
|
||||||
|
@ -11,7 +11,7 @@ LL | let _x: (i32,) = (5);
|
|||||||
help: use a trailing comma to create a tuple with one element
|
help: use a trailing comma to create a tuple with one element
|
||||||
|
|
|
|
||||||
LL | let _x: (i32,) = (5,);
|
LL | let _x: (i32,) = (5,);
|
||||||
| ~~~~
|
| +
|
||||||
|
|
||||||
error[E0308]: mismatched types
|
error[E0308]: mismatched types
|
||||||
--> $DIR/issue-86100-tuple-paren-comma.rs:13:9
|
--> $DIR/issue-86100-tuple-paren-comma.rs:13:9
|
||||||
@ -24,7 +24,7 @@ LL | foo((Some(3)));
|
|||||||
help: use a trailing comma to create a tuple with one element
|
help: use a trailing comma to create a tuple with one element
|
||||||
|
|
|
|
||||||
LL | foo((Some(3),));
|
LL | foo((Some(3),));
|
||||||
| ~~~~~~~~~~
|
| +
|
||||||
|
|
||||||
error[E0308]: mismatched types
|
error[E0308]: mismatched types
|
||||||
--> $DIR/issue-86100-tuple-paren-comma.rs:17:22
|
--> $DIR/issue-86100-tuple-paren-comma.rs:17:22
|
||||||
@ -37,7 +37,7 @@ LL | let _s = S { _s: ("abc".to_string()) };
|
|||||||
help: use a trailing comma to create a tuple with one element
|
help: use a trailing comma to create a tuple with one element
|
||||||
|
|
|
|
||||||
LL | let _s = S { _s: ("abc".to_string(),) };
|
LL | let _s = S { _s: ("abc".to_string(),) };
|
||||||
| ~~~~~~~~~~~~~~~~~~~~
|
| +
|
||||||
|
|
||||||
error[E0308]: mismatched types
|
error[E0308]: mismatched types
|
||||||
--> $DIR/issue-86100-tuple-paren-comma.rs:23:22
|
--> $DIR/issue-86100-tuple-paren-comma.rs:23:22
|
||||||
|
Loading…
Reference in New Issue
Block a user