1
Fork 0

Suggest if let/let_else for refutable pat in let

This commit is contained in:
Esteban Kuber 2022-03-08 05:54:38 +00:00
parent b97dc20784
commit 0d92752b8a
15 changed files with 204 additions and 56 deletions

View file

@ -7,7 +7,8 @@ use super::{PatCtxt, PatternError};
use rustc_arena::TypedArena; use rustc_arena::TypedArena;
use rustc_ast::Mutability; use rustc_ast::Mutability;
use rustc_errors::{ use rustc_errors::{
error_code, struct_span_err, Applicability, Diagnostic, DiagnosticBuilder, ErrorGuaranteed, error_code, pluralize, struct_span_err, Applicability, Diagnostic, DiagnosticBuilder,
ErrorGuaranteed,
}; };
use rustc_hir as hir; use rustc_hir as hir;
use rustc_hir::def::*; use rustc_hir::def::*;
@ -20,7 +21,7 @@ use rustc_session::lint::builtin::{
}; };
use rustc_session::Session; use rustc_session::Session;
use rustc_span::source_map::Spanned; use rustc_span::source_map::Spanned;
use rustc_span::{DesugaringKind, ExpnKind, MultiSpan, Span}; use rustc_span::{BytePos, DesugaringKind, ExpnKind, MultiSpan, Span};
crate fn check_match(tcx: TyCtxt<'_>, def_id: DefId) { crate fn check_match(tcx: TyCtxt<'_>, def_id: DefId) {
let body_id = match def_id.as_local() { let body_id = match def_id.as_local() {
@ -241,6 +242,9 @@ impl<'p, 'tcx> MatchVisitor<'_, 'p, 'tcx> {
} }
let joined_patterns = joined_uncovered_patterns(&cx, &witnesses); let joined_patterns = joined_uncovered_patterns(&cx, &witnesses);
let mut bindings = vec![];
let mut err = struct_span_err!( let mut err = struct_span_err!(
self.tcx.sess, self.tcx.sess,
pat.span, pat.span,
@ -257,6 +261,16 @@ impl<'p, 'tcx> MatchVisitor<'_, 'p, 'tcx> {
false false
} }
_ => { _ => {
pat.walk(&mut |pat: &hir::Pat<'_>| {
match pat.kind {
hir::PatKind::Binding(_, _, ident, _) => {
bindings.push(ident);
}
_ => {}
}
true
});
err.span_label(pat.span, pattern_not_covered_label(&witnesses, &joined_patterns)); err.span_label(pat.span, pattern_not_covered_label(&witnesses, &joined_patterns));
true true
} }
@ -267,13 +281,71 @@ impl<'p, 'tcx> MatchVisitor<'_, 'p, 'tcx> {
"`let` bindings require an \"irrefutable pattern\", like a `struct` or \ "`let` bindings require an \"irrefutable pattern\", like a `struct` or \
an `enum` with only one variant", an `enum` with only one variant",
); );
if let Ok(snippet) = self.tcx.sess.source_map().span_to_snippet(span) { if self.tcx.sess.source_map().span_to_snippet(span).is_ok() {
err.span_suggestion( let semi_span = span.shrink_to_hi().with_lo(span.hi() - BytePos(1));
span, let start_span = span.shrink_to_lo();
"you might want to use `if let` to ignore the variant that isn't matched", let end_span = semi_span.shrink_to_lo();
format!("if {} {{ /* */ }}", &snippet[..snippet.len() - 1]), err.multipart_suggestion(
&format!(
"you might want to use `if let` to ignore the variant{} that {} matched",
pluralize!(witnesses.len()),
match witnesses.len() {
1 => "isn't",
_ => "aren't",
},
),
vec![
match &bindings[..] {
[] => (start_span, "if ".to_string()),
[binding] => (start_span, format!("let {} = if ", binding)),
bindings => (
start_span,
format!(
"let ({}) = if ",
bindings
.iter()
.map(|ident| ident.to_string())
.collect::<Vec<_>>()
.join(", ")
),
),
},
match &bindings[..] {
[] => (semi_span, " { todo!() }".to_string()),
[binding] => {
(end_span, format!(" {{ {} }} else {{ todo!() }}", binding))
}
bindings => (
end_span,
format!(
" {{ ({}) }} else {{ todo!() }}",
bindings
.iter()
.map(|ident| ident.to_string())
.collect::<Vec<_>>()
.join(", ")
),
),
},
],
Applicability::HasPlaceholders, Applicability::HasPlaceholders,
); );
if cx.tcx.sess.is_nightly_build() {
err.span_suggestion_verbose(
semi_span.shrink_to_lo(),
&format!(
"alternatively, on nightly, you might want to use \
`#![feature(let_else)]` to handle the variant{} that {} matched",
pluralize!(witnesses.len()),
match witnesses.len() {
1 => "isn't",
_ => "aren't",
},
),
" else { todo!() }".to_string(),
Applicability::HasPlaceholders,
);
}
} }
err.note( err.note(
"for more information, visit \ "for more information, visit \

View file

@ -7,10 +7,14 @@ LL | A = { let 0 = 0; 0 },
= note: `let` bindings require an "irrefutable pattern", like a `struct` or an `enum` with only one variant = note: `let` bindings require an "irrefutable pattern", like a `struct` or an `enum` with only one variant
= note: for more information, visit https://doc.rust-lang.org/book/ch18-02-refutability.html = note: for more information, visit https://doc.rust-lang.org/book/ch18-02-refutability.html
= note: the matched value is of type `i32` = note: the matched value is of type `i32`
help: you might want to use `if let` to ignore the variant that isn't matched help: you might want to use `if let` to ignore the variants that aren't matched
| |
LL | A = { if let 0 = 0 { /* */ } 0 }, LL | A = { if let 0 = 0 { todo!() } 0 },
| ~~~~~~~~~~~~~~~~~~~~~~ | ++ ~~~~~~~~~~~
help: alternatively, on nightly, you might want to use `#![feature(let_else)]` to handle the variants that aren't matched
|
LL | A = { let 0 = 0 else { todo!() }; 0 },
| ++++++++++++++++
error: aborting due to previous error error: aborting due to previous error

View file

@ -7,10 +7,14 @@ LL | let x: [i32; { let 0 = 0; 0 }] = [];
= note: `let` bindings require an "irrefutable pattern", like a `struct` or an `enum` with only one variant = note: `let` bindings require an "irrefutable pattern", like a `struct` or an `enum` with only one variant
= note: for more information, visit https://doc.rust-lang.org/book/ch18-02-refutability.html = note: for more information, visit https://doc.rust-lang.org/book/ch18-02-refutability.html
= note: the matched value is of type `i32` = note: the matched value is of type `i32`
help: you might want to use `if let` to ignore the variant that isn't matched help: you might want to use `if let` to ignore the variants that aren't matched
| |
LL | let x: [i32; { if let 0 = 0 { /* */ } 0 }] = []; LL | let x: [i32; { if let 0 = 0 { todo!() } 0 }] = [];
| ~~~~~~~~~~~~~~~~~~~~~~ | ++ ~~~~~~~~~~~
help: alternatively, on nightly, you might want to use `#![feature(let_else)]` to handle the variants that aren't matched
|
LL | let x: [i32; { let 0 = 0 else { todo!() }; 0 }] = [];
| ++++++++++++++++
error: aborting due to previous error error: aborting due to previous error

View file

@ -7,10 +7,14 @@ LL | const X: i32 = { let 0 = 0; 0 };
= note: `let` bindings require an "irrefutable pattern", like a `struct` or an `enum` with only one variant = note: `let` bindings require an "irrefutable pattern", like a `struct` or an `enum` with only one variant
= note: for more information, visit https://doc.rust-lang.org/book/ch18-02-refutability.html = note: for more information, visit https://doc.rust-lang.org/book/ch18-02-refutability.html
= note: the matched value is of type `i32` = note: the matched value is of type `i32`
help: you might want to use `if let` to ignore the variant that isn't matched help: you might want to use `if let` to ignore the variants that aren't matched
| |
LL | const X: i32 = { if let 0 = 0 { /* */ } 0 }; LL | const X: i32 = { if let 0 = 0 { todo!() } 0 };
| ~~~~~~~~~~~~~~~~~~~~~~ | ++ ~~~~~~~~~~~
help: alternatively, on nightly, you might want to use `#![feature(let_else)]` to handle the variants that aren't matched
|
LL | const X: i32 = { let 0 = 0 else { todo!() }; 0 };
| ++++++++++++++++
error[E0005]: refutable pattern in local binding: `i32::MIN..=-1_i32` and `1_i32..=i32::MAX` not covered error[E0005]: refutable pattern in local binding: `i32::MIN..=-1_i32` and `1_i32..=i32::MAX` not covered
--> $DIR/const-match-check.rs:8:23 --> $DIR/const-match-check.rs:8:23
@ -21,10 +25,14 @@ LL | static Y: i32 = { let 0 = 0; 0 };
= note: `let` bindings require an "irrefutable pattern", like a `struct` or an `enum` with only one variant = note: `let` bindings require an "irrefutable pattern", like a `struct` or an `enum` with only one variant
= note: for more information, visit https://doc.rust-lang.org/book/ch18-02-refutability.html = note: for more information, visit https://doc.rust-lang.org/book/ch18-02-refutability.html
= note: the matched value is of type `i32` = note: the matched value is of type `i32`
help: you might want to use `if let` to ignore the variant that isn't matched help: you might want to use `if let` to ignore the variants that aren't matched
| |
LL | static Y: i32 = { if let 0 = 0 { /* */ } 0 }; LL | static Y: i32 = { if let 0 = 0 { todo!() } 0 };
| ~~~~~~~~~~~~~~~~~~~~~~ | ++ ~~~~~~~~~~~
help: alternatively, on nightly, you might want to use `#![feature(let_else)]` to handle the variants that aren't matched
|
LL | static Y: i32 = { let 0 = 0 else { todo!() }; 0 };
| ++++++++++++++++
error[E0005]: refutable pattern in local binding: `i32::MIN..=-1_i32` and `1_i32..=i32::MAX` not covered error[E0005]: refutable pattern in local binding: `i32::MIN..=-1_i32` and `1_i32..=i32::MAX` not covered
--> $DIR/const-match-check.rs:13:26 --> $DIR/const-match-check.rs:13:26
@ -35,10 +43,14 @@ LL | const X: i32 = { let 0 = 0; 0 };
= note: `let` bindings require an "irrefutable pattern", like a `struct` or an `enum` with only one variant = note: `let` bindings require an "irrefutable pattern", like a `struct` or an `enum` with only one variant
= note: for more information, visit https://doc.rust-lang.org/book/ch18-02-refutability.html = note: for more information, visit https://doc.rust-lang.org/book/ch18-02-refutability.html
= note: the matched value is of type `i32` = note: the matched value is of type `i32`
help: you might want to use `if let` to ignore the variant that isn't matched help: you might want to use `if let` to ignore the variants that aren't matched
| |
LL | const X: i32 = { if let 0 = 0 { /* */ } 0 }; LL | const X: i32 = { if let 0 = 0 { todo!() } 0 };
| ~~~~~~~~~~~~~~~~~~~~~~ | ++ ~~~~~~~~~~~
help: alternatively, on nightly, you might want to use `#![feature(let_else)]` to handle the variants that aren't matched
|
LL | const X: i32 = { let 0 = 0 else { todo!() }; 0 };
| ++++++++++++++++
error[E0005]: refutable pattern in local binding: `i32::MIN..=-1_i32` and `1_i32..=i32::MAX` not covered error[E0005]: refutable pattern in local binding: `i32::MIN..=-1_i32` and `1_i32..=i32::MAX` not covered
--> $DIR/const-match-check.rs:19:26 --> $DIR/const-match-check.rs:19:26
@ -49,10 +61,14 @@ LL | const X: i32 = { let 0 = 0; 0 };
= note: `let` bindings require an "irrefutable pattern", like a `struct` or an `enum` with only one variant = note: `let` bindings require an "irrefutable pattern", like a `struct` or an `enum` with only one variant
= note: for more information, visit https://doc.rust-lang.org/book/ch18-02-refutability.html = note: for more information, visit https://doc.rust-lang.org/book/ch18-02-refutability.html
= note: the matched value is of type `i32` = note: the matched value is of type `i32`
help: you might want to use `if let` to ignore the variant that isn't matched help: you might want to use `if let` to ignore the variants that aren't matched
| |
LL | const X: i32 = { if let 0 = 0 { /* */ } 0 }; LL | const X: i32 = { if let 0 = 0 { todo!() } 0 };
| ~~~~~~~~~~~~~~~~~~~~~~ | ++ ~~~~~~~~~~~
help: alternatively, on nightly, you might want to use `#![feature(let_else)]` to handle the variants that aren't matched
|
LL | const X: i32 = { let 0 = 0 else { todo!() }; 0 };
| ++++++++++++++++
error: aborting due to 4 previous errors error: aborting due to 4 previous errors

View file

@ -16,8 +16,12 @@ LL | T(T, [!; 0]),
= note: the matched value is of type `Helper<T, U>` = note: the matched value is of type `Helper<T, U>`
help: you might want to use `if let` to ignore the variant that isn't matched help: you might want to use `if let` to ignore the variant that isn't matched
| |
LL | if let Helper::U(u) = Helper::T(t, []) { /* */ } LL | let u = if let Helper::U(u) = Helper::T(t, []) { u } else { todo!() };
| ++++++++++ ++++++++++++++++++++++
help: alternatively, on nightly, you might want to use `#![feature(let_else)]` to handle the variant that isn't matched
| |
LL | let Helper::U(u) = Helper::T(t, []) else { todo!() };
| ++++++++++++++++
error: aborting due to previous error error: aborting due to previous error

View file

@ -22,8 +22,12 @@ LL | | }
= note: the matched value is of type `Option<i32>` = note: the matched value is of type `Option<i32>`
help: you might want to use `if let` to ignore the variant that isn't matched help: you might want to use `if let` to ignore the variant that isn't matched
| |
LL | if let Some(y) = x { /* */ } LL | let y = if let Some(y) = x { y } else { todo!() };
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ++++++++++ ++++++++++++++++++++++
help: alternatively, on nightly, you might want to use `#![feature(let_else)]` to handle the variant that isn't matched
|
LL | let Some(y) = x else { todo!() };
| ++++++++++++++++
error: aborting due to previous error error: aborting due to previous error

View file

@ -21,8 +21,12 @@ LL | | }
= note: the matched value is of type `Result<u32, !>` = note: the matched value is of type `Result<u32, !>`
help: you might want to use `if let` to ignore the variant that isn't matched help: you might want to use `if let` to ignore the variant that isn't matched
| |
LL | if let Ok(_x) = foo() { /* */ } LL | let _x = if let Ok(_x) = foo() { _x } else { todo!() };
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | +++++++++++ +++++++++++++++++++++++
help: alternatively, on nightly, you might want to use `#![feature(let_else)]` to handle the variant that isn't matched
|
LL | let Ok(_x) = foo() else { todo!() };
| ++++++++++++++++
error: aborting due to previous error error: aborting due to previous error

View file

@ -7,10 +7,14 @@ LL | let (0 | (1 | 2)) = 0;
= note: `let` bindings require an "irrefutable pattern", like a `struct` or an `enum` with only one variant = note: `let` bindings require an "irrefutable pattern", like a `struct` or an `enum` with only one variant
= note: for more information, visit https://doc.rust-lang.org/book/ch18-02-refutability.html = note: for more information, visit https://doc.rust-lang.org/book/ch18-02-refutability.html
= note: the matched value is of type `i32` = note: the matched value is of type `i32`
help: you might want to use `if let` to ignore the variant that isn't matched help: you might want to use `if let` to ignore the variants that aren't matched
| |
LL | if let (0 | (1 | 2)) = 0 { /* */ } LL | if let (0 | (1 | 2)) = 0 { todo!() }
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ++ ~~~~~~~~~~~
help: alternatively, on nightly, you might want to use `#![feature(let_else)]` to handle the variants that aren't matched
|
LL | let (0 | (1 | 2)) = 0 else { todo!() };
| ++++++++++++++++
error[E0004]: non-exhaustive patterns: `i32::MIN..=-1_i32` and `3_i32..=i32::MAX` not covered error[E0004]: non-exhaustive patterns: `i32::MIN..=-1_i32` and `3_i32..=i32::MAX` not covered
--> $DIR/issue-69875-should-have-been-expanded-earlier-non-exhaustive.rs:3:11 --> $DIR/issue-69875-should-have-been-expanded-earlier-non-exhaustive.rs:3:11

View file

@ -17,10 +17,14 @@ LL | Bar,
LL | Baz LL | Baz
| ^^^ not covered | ^^^ not covered
= note: the matched value is of type `Thing` = note: the matched value is of type `Thing`
help: you might want to use `if let` to ignore the variant that isn't matched help: you might want to use `if let` to ignore the variants that aren't matched
| |
LL | if let Thing::Foo(y) = Thing::Foo(1) { /* */ } LL | let y = if let Thing::Foo(y) = Thing::Foo(1) { y } else { todo!() };
| ++++++++++ ++++++++++++++++++++++
help: alternatively, on nightly, you might want to use `#![feature(let_else)]` to handle the variants that aren't matched
| |
LL | let Thing::Foo(y) = Thing::Foo(1) else { todo!() };
| ++++++++++++++++
error: aborting due to previous error error: aborting due to previous error

View file

@ -42,10 +42,14 @@ LL | B,
LL | C LL | C
| ^ not covered | ^ not covered
= note: the matched value is of type `E` = note: the matched value is of type `E`
help: you might want to use `if let` to ignore the variant that isn't matched help: you might want to use `if let` to ignore the variants that aren't matched
| |
LL | if let E::A = e { /* */ } LL | if let E::A = e { todo!() }
| ~~~~~~~~~~~~~~~~~~~~~~~~~ | ++ ~~~~~~~~~~~
help: alternatively, on nightly, you might want to use `#![feature(let_else)]` to handle the variants that aren't matched
|
LL | let E::A = e else { todo!() };
| ++++++++++++++++
error[E0004]: non-exhaustive patterns: `&B` and `&C` not covered error[E0004]: non-exhaustive patterns: `&B` and `&C` not covered
--> $DIR/non-exhaustive-defined-here.rs:52:11 --> $DIR/non-exhaustive-defined-here.rs:52:11
@ -91,10 +95,14 @@ LL | B,
LL | C LL | C
| ^ not covered | ^ not covered
= note: the matched value is of type `&E` = note: the matched value is of type `&E`
help: you might want to use `if let` to ignore the variant that isn't matched help: you might want to use `if let` to ignore the variants that aren't matched
| |
LL | if let E::A = e { /* */ } LL | if let E::A = e { todo!() }
| ~~~~~~~~~~~~~~~~~~~~~~~~~ | ++ ~~~~~~~~~~~
help: alternatively, on nightly, you might want to use `#![feature(let_else)]` to handle the variants that aren't matched
|
LL | let E::A = e else { todo!() };
| ++++++++++++++++
error[E0004]: non-exhaustive patterns: `&&mut &B` and `&&mut &C` not covered error[E0004]: non-exhaustive patterns: `&&mut &B` and `&&mut &C` not covered
--> $DIR/non-exhaustive-defined-here.rs:66:11 --> $DIR/non-exhaustive-defined-here.rs:66:11
@ -140,10 +148,14 @@ LL | B,
LL | C LL | C
| ^ not covered | ^ not covered
= note: the matched value is of type `&&mut &E` = note: the matched value is of type `&&mut &E`
help: you might want to use `if let` to ignore the variant that isn't matched help: you might want to use `if let` to ignore the variants that aren't matched
| |
LL | if let E::A = e { /* */ } LL | if let E::A = e { todo!() }
| ++ ~~~~~~~~~~~
help: alternatively, on nightly, you might want to use `#![feature(let_else)]` to handle the variants that aren't matched
| |
LL | let E::A = e else { todo!() };
| ++++++++++++++++
error[E0004]: non-exhaustive patterns: `None` not covered error[E0004]: non-exhaustive patterns: `None` not covered
--> $DIR/non-exhaustive-defined-here.rs:92:11 --> $DIR/non-exhaustive-defined-here.rs:92:11
@ -185,8 +197,12 @@ LL | None,
= note: the matched value is of type `Opt` = note: the matched value is of type `Opt`
help: you might want to use `if let` to ignore the variant that isn't matched help: you might want to use `if let` to ignore the variant that isn't matched
| |
LL | if let Opt::Some(ref _x) = e { /* */ } LL | let _x = if let Opt::Some(ref _x) = e { _x } else { todo!() };
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | +++++++++++ +++++++++++++++++++++++
help: alternatively, on nightly, you might want to use `#![feature(let_else)]` to handle the variant that isn't matched
|
LL | let Opt::Some(ref _x) = e else { todo!() };
| ++++++++++++++++
error: aborting due to 8 previous errors error: aborting due to 8 previous errors

View file

@ -15,10 +15,14 @@ LL | let (1, (Some(1), 2..=3)) = (1, (None, 2));
= note: `let` bindings require an "irrefutable pattern", like a `struct` or an `enum` with only one variant = note: `let` bindings require an "irrefutable pattern", like a `struct` or an `enum` with only one variant
= note: for more information, visit https://doc.rust-lang.org/book/ch18-02-refutability.html = note: for more information, visit https://doc.rust-lang.org/book/ch18-02-refutability.html
= note: the matched value is of type `(i32, (Option<i32>, i32))` = note: the matched value is of type `(i32, (Option<i32>, i32))`
help: you might want to use `if let` to ignore the variant that isn't matched help: you might want to use `if let` to ignore the variants that aren't matched
| |
LL | if let (1, (Some(1), 2..=3)) = (1, (None, 2)) { /* */ } LL | if let (1, (Some(1), 2..=3)) = (1, (None, 2)) { todo!() }
| ++ ~~~~~~~~~~~
help: alternatively, on nightly, you might want to use `#![feature(let_else)]` to handle the variants that aren't matched
| |
LL | let (1, (Some(1), 2..=3)) = (1, (None, 2)) else { todo!() };
| ++++++++++++++++
error: aborting due to 2 previous errors error: aborting due to 2 previous errors

View file

@ -21,8 +21,12 @@ LL | | }
= note: the matched value is of type `Result<u32, &R>` = note: the matched value is of type `Result<u32, &R>`
help: you might want to use `if let` to ignore the variant that isn't matched help: you might want to use `if let` to ignore the variant that isn't matched
| |
LL | if let Ok(x) = res { /* */ } LL | let x = if let Ok(x) = res { x } else { todo!() };
| ++++++++++ ++++++++++++++++++++++
help: alternatively, on nightly, you might want to use `#![feature(let_else)]` to handle the variant that isn't matched
| |
LL | let Ok(x) = res else { todo!() };
| ++++++++++++++++
error: aborting due to previous error error: aborting due to previous error

View file

@ -19,10 +19,10 @@ enum Foo {
A(foo::SecretlyEmpty), A(foo::SecretlyEmpty),
B(foo::NotSoSecretlyEmpty), B(foo::NotSoSecretlyEmpty),
C(NotSoSecretlyEmpty), C(NotSoSecretlyEmpty),
D(u32), D(u32, u32),
} }
fn main() { fn main() {
let x: Foo = Foo::D(123); let x: Foo = Foo::D(123, 456);
let Foo::D(_y) = x; //~ ERROR refutable pattern in local binding: `A(_)` not covered let Foo::D(_y, _z) = x; //~ ERROR refutable pattern in local binding: `A(_)` not covered
} }

View file

@ -1,8 +1,8 @@
error[E0005]: refutable pattern in local binding: `A(_)` not covered error[E0005]: refutable pattern in local binding: `A(_)` not covered
--> $DIR/uninhabited-irrefutable.rs:27:9 --> $DIR/uninhabited-irrefutable.rs:27:9
| |
LL | let Foo::D(_y) = x; LL | let Foo::D(_y, _z) = x;
| ^^^^^^^^^^ pattern `A(_)` not covered | ^^^^^^^^^^^^^^ pattern `A(_)` not covered
| |
= note: `let` bindings require an "irrefutable pattern", like a `struct` or an `enum` with only one variant = note: `let` bindings require an "irrefutable pattern", like a `struct` or an `enum` with only one variant
= note: for more information, visit https://doc.rust-lang.org/book/ch18-02-refutability.html = note: for more information, visit https://doc.rust-lang.org/book/ch18-02-refutability.html
@ -16,8 +16,12 @@ LL | A(foo::SecretlyEmpty),
= note: the matched value is of type `Foo` = note: the matched value is of type `Foo`
help: you might want to use `if let` to ignore the variant that isn't matched help: you might want to use `if let` to ignore the variant that isn't matched
| |
LL | if let Foo::D(_y) = x { /* */ } LL | let (_y, _z) = if let Foo::D(_y, _z) = x { (_y, _z) } else { todo!() };
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | +++++++++++++++++ +++++++++++++++++++++++++++++
help: alternatively, on nightly, you might want to use `#![feature(let_else)]` to handle the variant that isn't matched
|
LL | let Foo::D(_y, _z) = x else { todo!() };
| ++++++++++++++++
error: aborting due to previous error error: aborting due to previous error

View file

@ -132,8 +132,12 @@ LL | | }
= note: the matched value is of type `Result<u32, Void>` = note: the matched value is of type `Result<u32, Void>`
help: you might want to use `if let` to ignore the variant that isn't matched help: you might want to use `if let` to ignore the variant that isn't matched
| |
LL | if let Ok(x) = x { /* */ } LL | let x = if let Ok(x) = x { x } else { todo!() };
| ++++++++++ ++++++++++++++++++++++
help: alternatively, on nightly, you might want to use `#![feature(let_else)]` to handle the variant that isn't matched
| |
LL | let Ok(x) = x else { todo!() };
| ++++++++++++++++
error: aborting due to 7 previous errors error: aborting due to 7 previous errors