1 use crate::utils::{match_qpath, paths, snippet, span_lint_and_then};
3 use rustc::lint::{LateContext, LateLintPass, LintArray, LintPass};
4 use rustc::{declare_lint_pass, declare_tool_lint};
5 use rustc_errors::Applicability;
6 use syntax::ast::LitKind;
8 use syntax::symbol::Symbol;
10 declare_clippy_lint! {
11 /// **What it does:** Lint for redundant pattern matching over `Result` or
14 /// **Why is this bad?** It's more concise and clear to just use the proper
17 /// **Known problems:** None.
22 /// if let Ok(_) = Ok::<i32, i32>(42) {}
23 /// if let Err(_) = Err::<i32, i32>(42) {}
24 /// if let None = None::<()> {}
25 /// if let Some(_) = Some(42) {}
26 /// match Ok::<i32, i32>(42) {
32 /// The more idiomatic use would be:
35 /// if Ok::<i32, i32>(42).is_ok() {}
36 /// if Err::<i32, i32>(42).is_err() {}
37 /// if None::<()>.is_none() {}
38 /// if Some(42).is_some() {}
39 /// Ok::<i32, i32>(42).is_ok();
41 pub REDUNDANT_PATTERN_MATCHING,
43 "use the proper utility function avoiding an `if let`"
46 declare_lint_pass!(RedundantPatternMatching => [REDUNDANT_PATTERN_MATCHING]);
48 impl<'a, 'tcx> LateLintPass<'a, 'tcx> for RedundantPatternMatching {
49 fn check_expr(&mut self, cx: &LateContext<'a, 'tcx>, expr: &'tcx Expr) {
50 if let ExprKind::Match(ref op, ref arms, ref match_source) = expr.node {
52 MatchSource::Normal => find_sugg_for_match(cx, expr, op, arms),
53 MatchSource::IfLetDesugar { .. } => find_sugg_for_if_let(cx, expr, op, arms),
60 fn find_sugg_for_if_let<'a, 'tcx>(cx: &LateContext<'a, 'tcx>, expr: &'tcx Expr, op: &P<Expr>, arms: &HirVec<Arm>) {
61 if arms[0].pats.len() == 1 {
62 let good_method = match arms[0].pats[0].node {
63 PatKind::TupleStruct(ref path, ref patterns, _) if patterns.len() == 1 => {
64 if let PatKind::Wild = patterns[0].node {
65 if match_qpath(path, &*paths::RESULT_OK) {
67 } else if match_qpath(path, &*paths::RESULT_ERR) {
69 } else if match_qpath(path, &*paths::OPTION_SOME) {
79 PatKind::Path(ref path) if match_qpath(path, &*paths::OPTION_NONE) => "is_none()",
86 REDUNDANT_PATTERN_MATCHING,
88 &format!("redundant pattern matching, consider using `{}`", good_method),
90 let span = expr.span.to(op.span);
94 format!("if {}.{}", snippet(cx, op.span, "_"), good_method),
95 Applicability::MachineApplicable, // snippet
104 fn find_sugg_for_match<'a, 'tcx>(cx: &LateContext<'a, 'tcx>, expr: &'tcx Expr, op: &P<Expr>, arms: &HirVec<Arm>) {
106 let node_pair = (&arms[0].pats[0].node, &arms[1].pats[0].node);
108 let found_good_method = match node_pair {
110 PatKind::TupleStruct(ref path_left, ref patterns_left, _),
111 PatKind::TupleStruct(ref path_right, ref patterns_right, _),
112 ) if patterns_left.len() == 1 && patterns_right.len() == 1 => {
113 if let (PatKind::Wild, PatKind::Wild) = (&patterns_left[0].node, &patterns_right[0].node) {
114 find_good_method_for_match(
127 (PatKind::TupleStruct(ref path_left, ref patterns, _), PatKind::Path(ref path_right))
128 | (PatKind::Path(ref path_left), PatKind::TupleStruct(ref path_right, ref patterns, _))
129 if patterns.len() == 1 =>
131 if let PatKind::Wild = patterns[0].node {
132 find_good_method_for_match(
136 &*paths::OPTION_SOME,
137 &*paths::OPTION_NONE,
148 if let Some(good_method) = found_good_method {
151 REDUNDANT_PATTERN_MATCHING,
153 &format!("redundant pattern matching, consider using `{}`", good_method),
155 let span = expr.span.to(op.span);
159 format!("{}.{}", snippet(cx, op.span, "_"), good_method),
160 Applicability::MachineApplicable, // snippet
170 fn find_good_method_for_match<'a>(
174 expected_left: &[Symbol],
175 expected_right: &[Symbol],
176 should_be_left: &'a str,
177 should_be_right: &'a str,
178 ) -> Option<&'a str> {
179 let body_node_pair = if match_qpath(path_left, expected_left) && match_qpath(path_right, expected_right) {
180 (&(*arms[0].body).node, &(*arms[1].body).node)
181 } else if match_qpath(path_right, expected_left) && match_qpath(path_left, expected_right) {
182 (&(*arms[1].body).node, &(*arms[0].body).node)
187 match body_node_pair {
188 (ExprKind::Lit(ref lit_left), ExprKind::Lit(ref lit_right)) => match (&lit_left.node, &lit_right.node) {
189 (LitKind::Bool(true), LitKind::Bool(false)) => Some(should_be_left),
190 (LitKind::Bool(false), LitKind::Bool(true)) => Some(should_be_right),