2022-03-10 01:44:25 +00:00
|
|
|
use super::NEEDLESS_MATCH;
|
2022-02-25 10:08:52 +00:00
|
|
|
use clippy_utils::diagnostics::span_lint_and_sugg;
|
2022-03-07 10:12:35 +00:00
|
|
|
use clippy_utils::source::snippet_with_applicability;
|
2022-03-08 09:37:53 +00:00
|
|
|
use clippy_utils::ty::is_type_diagnostic_item;
|
|
|
|
use clippy_utils::{eq_expr_value, get_parent_expr, higher, is_else_clause, is_lang_ctor, peel_blocks_with_stmt};
|
2022-02-25 10:08:52 +00:00
|
|
|
use rustc_errors::Applicability;
|
2022-03-08 09:37:53 +00:00
|
|
|
use rustc_hir::LangItem::OptionNone;
|
2022-03-16 08:26:56 +00:00
|
|
|
use rustc_hir::{Arm, BindingAnnotation, Expr, ExprKind, Pat, PatKind, Path, PathSegment, QPath};
|
2022-03-07 10:12:35 +00:00
|
|
|
use rustc_lint::LateContext;
|
2022-03-08 09:37:53 +00:00
|
|
|
use rustc_span::sym;
|
2022-02-25 10:08:52 +00:00
|
|
|
|
|
|
|
pub(crate) fn check_match(cx: &LateContext<'_>, ex: &Expr<'_>, arms: &[Arm<'_>]) {
|
2022-03-07 10:12:35 +00:00
|
|
|
// This is for avoiding collision with `match_single_binding`.
|
|
|
|
if arms.len() < 2 {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for arm in arms {
|
|
|
|
if let PatKind::Wild = arm.pat.kind {
|
|
|
|
let ret_expr = strip_return(arm.body);
|
|
|
|
if !eq_expr_value(cx, ex, ret_expr) {
|
|
|
|
return;
|
|
|
|
}
|
2022-03-16 08:26:56 +00:00
|
|
|
} else if !pat_same_as_expr(arm.pat, peel_blocks_with_stmt(arm.body)) {
|
2022-03-07 10:12:35 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if let Some(match_expr) = get_parent_expr(cx, ex) {
|
|
|
|
let mut applicability = Applicability::MachineApplicable;
|
2022-02-25 10:08:52 +00:00
|
|
|
span_lint_and_sugg(
|
|
|
|
cx,
|
2022-03-10 01:44:25 +00:00
|
|
|
NEEDLESS_MATCH,
|
2022-03-07 10:12:35 +00:00
|
|
|
match_expr.span,
|
2022-02-25 10:08:52 +00:00
|
|
|
"this match expression is unnecessary",
|
|
|
|
"replace it with",
|
2022-03-07 10:12:35 +00:00
|
|
|
snippet_with_applicability(cx, ex.span, "..", &mut applicability).to_string(),
|
|
|
|
applicability,
|
2022-02-25 10:08:52 +00:00
|
|
|
);
|
|
|
|
}
|
2022-03-07 10:12:35 +00:00
|
|
|
}
|
|
|
|
|
2022-03-08 09:37:53 +00:00
|
|
|
/// Check for nop `if let` expression that assembled as unnecessary match
|
|
|
|
///
|
|
|
|
/// ```rust,ignore
|
|
|
|
/// if let Some(a) = option {
|
|
|
|
/// Some(a)
|
|
|
|
/// } else {
|
|
|
|
/// None
|
|
|
|
/// }
|
|
|
|
/// ```
|
|
|
|
/// OR
|
|
|
|
/// ```rust,ignore
|
|
|
|
/// if let SomeEnum::A = some_enum {
|
|
|
|
/// SomeEnum::A
|
|
|
|
/// } else if let SomeEnum::B = some_enum {
|
|
|
|
/// SomeEnum::B
|
|
|
|
/// } else {
|
|
|
|
/// some_enum
|
|
|
|
/// }
|
|
|
|
/// ```
|
|
|
|
pub(crate) fn check(cx: &LateContext<'_>, ex: &Expr<'_>) {
|
|
|
|
if_chain! {
|
|
|
|
if let Some(ref if_let) = higher::IfLet::hir(cx, ex);
|
|
|
|
if !is_else_clause(cx.tcx, ex);
|
|
|
|
if check_if_let(cx, if_let);
|
|
|
|
then {
|
|
|
|
let mut applicability = Applicability::MachineApplicable;
|
|
|
|
span_lint_and_sugg(
|
|
|
|
cx,
|
2022-03-10 01:44:25 +00:00
|
|
|
NEEDLESS_MATCH,
|
2022-03-08 09:37:53 +00:00
|
|
|
ex.span,
|
|
|
|
"this if-let expression is unnecessary",
|
|
|
|
"replace it with",
|
|
|
|
snippet_with_applicability(cx, if_let.let_expr.span, "..", &mut applicability).to_string(),
|
|
|
|
applicability,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn check_if_let(cx: &LateContext<'_>, if_let: &higher::IfLet<'_>) -> bool {
|
2022-03-08 10:15:11 +00:00
|
|
|
if let Some(if_else) = if_let.if_else {
|
2022-03-08 09:37:53 +00:00
|
|
|
if !pat_same_as_expr(if_let.let_pat, peel_blocks_with_stmt(if_let.if_then)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Recurrsively check for each `else if let` phrase,
|
2022-03-08 10:15:11 +00:00
|
|
|
if let Some(ref nested_if_let) = higher::IfLet::hir(cx, if_else) {
|
2022-03-08 09:37:53 +00:00
|
|
|
return check_if_let(cx, nested_if_let);
|
|
|
|
}
|
2022-03-08 10:15:11 +00:00
|
|
|
|
|
|
|
if matches!(if_else.kind, ExprKind::Block(..)) {
|
|
|
|
let else_expr = peel_blocks_with_stmt(if_else);
|
2022-03-16 08:26:56 +00:00
|
|
|
if matches!(else_expr.kind, ExprKind::Block(..)) {
|
|
|
|
return false;
|
|
|
|
}
|
2022-03-08 10:15:11 +00:00
|
|
|
let ret = strip_return(else_expr);
|
|
|
|
let let_expr_ty = cx.typeck_results().expr_ty(if_let.let_expr);
|
|
|
|
if is_type_diagnostic_item(cx, let_expr_ty, sym::Option) {
|
|
|
|
if let ExprKind::Path(ref qpath) = ret.kind {
|
|
|
|
return is_lang_ctor(cx, qpath, OptionNone) || eq_expr_value(cx, if_let.let_expr, ret);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return eq_expr_value(cx, if_let.let_expr, ret);
|
2022-03-08 09:37:53 +00:00
|
|
|
}
|
2022-03-08 10:15:11 +00:00
|
|
|
return true;
|
2022-03-08 09:37:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
false
|
|
|
|
}
|
|
|
|
|
2022-03-10 06:46:58 +00:00
|
|
|
/// Strip `return` keyword if the expression type is `ExprKind::Ret`.
|
2022-03-07 10:12:35 +00:00
|
|
|
fn strip_return<'hir>(expr: &'hir Expr<'hir>) -> &'hir Expr<'hir> {
|
|
|
|
if let ExprKind::Ret(Some(ret)) = expr.kind {
|
|
|
|
ret
|
|
|
|
} else {
|
|
|
|
expr
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn pat_same_as_expr(pat: &Pat<'_>, expr: &Expr<'_>) -> bool {
|
|
|
|
let expr = strip_return(expr);
|
|
|
|
match (&pat.kind, &expr.kind) {
|
2022-03-10 06:46:58 +00:00
|
|
|
// Example: `Some(val) => Some(val)`
|
2022-03-16 08:26:56 +00:00
|
|
|
(PatKind::TupleStruct(QPath::Resolved(_, path), tuple_params, _), ExprKind::Call(call_expr, call_params)) => {
|
2022-03-07 10:12:35 +00:00
|
|
|
if let ExprKind::Path(QPath::Resolved(_, call_path)) = call_expr.kind {
|
2022-03-16 08:26:56 +00:00
|
|
|
return has_identical_segments(path.segments, call_path.segments)
|
|
|
|
&& has_same_non_ref_symbols(tuple_params, call_params);
|
2022-03-07 10:12:35 +00:00
|
|
|
}
|
|
|
|
},
|
2022-03-16 08:26:56 +00:00
|
|
|
// Example: `val => val`
|
|
|
|
(
|
|
|
|
PatKind::Binding(annot, _, pat_ident, _),
|
|
|
|
ExprKind::Path(QPath::Resolved(
|
2022-03-10 06:46:58 +00:00
|
|
|
_,
|
|
|
|
Path {
|
|
|
|
segments: [first_seg, ..],
|
|
|
|
..
|
|
|
|
},
|
2022-03-16 08:26:56 +00:00
|
|
|
)),
|
|
|
|
) => {
|
|
|
|
return !matches!(annot, BindingAnnotation::Ref | BindingAnnotation::RefMut)
|
|
|
|
&& pat_ident.name == first_seg.ident.name;
|
2022-03-10 06:46:58 +00:00
|
|
|
},
|
|
|
|
// Example: `Custom::TypeA => Custom::TypeB`, or `None => None`
|
2022-03-07 10:12:35 +00:00
|
|
|
(PatKind::Path(QPath::Resolved(_, p_path)), ExprKind::Path(QPath::Resolved(_, e_path))) => {
|
2022-03-08 09:37:53 +00:00
|
|
|
return has_identical_segments(p_path.segments, e_path.segments);
|
2022-03-07 10:12:35 +00:00
|
|
|
},
|
2022-03-10 06:46:58 +00:00
|
|
|
// Example: `5 => 5`
|
2022-03-07 10:12:35 +00:00
|
|
|
(PatKind::Lit(pat_lit_expr), ExprKind::Lit(expr_spanned)) => {
|
|
|
|
if let ExprKind::Lit(pat_spanned) = &pat_lit_expr.kind {
|
|
|
|
return pat_spanned.node == expr_spanned.node;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
_ => {},
|
|
|
|
}
|
|
|
|
|
|
|
|
false
|
|
|
|
}
|
|
|
|
|
2022-03-08 09:37:53 +00:00
|
|
|
fn has_identical_segments(left_segs: &[PathSegment<'_>], right_segs: &[PathSegment<'_>]) -> bool {
|
2022-03-07 10:12:35 +00:00
|
|
|
if left_segs.len() != right_segs.len() {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
for i in 0..left_segs.len() {
|
|
|
|
if left_segs[i].ident.name != right_segs[i].ident.name {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
true
|
|
|
|
}
|
|
|
|
|
2022-03-16 08:26:56 +00:00
|
|
|
fn has_same_non_ref_symbols(pats: &[Pat<'_>], exprs: &[Expr<'_>]) -> bool {
|
|
|
|
if pats.len() != exprs.len() {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
for i in 0..pats.len() {
|
|
|
|
if !pat_same_as_expr(&pats[i], &exprs[i]) {
|
|
|
|
return false;
|
2022-03-07 10:12:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-16 08:26:56 +00:00
|
|
|
true
|
2022-03-07 10:12:35 +00:00
|
|
|
}
|