@@ -3,46 +3,78 @@ use clippy_utils::diagnostics::span_lint_and_sugg;
3
3
use clippy_utils:: source:: { indent_of, reindent_multiline, snippet_opt} ;
4
4
use clippy_utils:: ty:: is_type_diagnostic_item;
5
5
use clippy_utils:: usage:: contains_return_break_continue_macro;
6
- use clippy_utils:: { is_res_lang_ctor, path_to_local_id, sugg} ;
6
+ use clippy_utils:: { is_res_lang_ctor, path_to_local_id, peel_blocks , sugg} ;
7
7
use rustc_errors:: Applicability ;
8
8
use rustc_hir:: def:: { DefKind , Res } ;
9
9
use rustc_hir:: LangItem :: { OptionNone , ResultErr } ;
10
- use rustc_hir:: { Arm , Expr , PatKind } ;
10
+ use rustc_hir:: { Arm , Expr , Pat , PatKind } ;
11
11
use rustc_lint:: LateContext ;
12
+ use rustc_middle:: ty:: Ty ;
12
13
use rustc_span:: sym;
13
14
14
15
use super :: MANUAL_UNWRAP_OR ;
15
16
16
- pub ( super ) fn check < ' tcx > ( cx : & LateContext < ' tcx > , expr : & Expr < ' tcx > , scrutinee : & ' tcx Expr < ' _ > , arms : & ' tcx [ Arm < ' _ > ] ) {
17
+ pub ( super ) fn check_match < ' tcx > (
18
+ cx : & LateContext < ' tcx > ,
19
+ expr : & ' tcx Expr < ' tcx > ,
20
+ scrutinee : & ' tcx Expr < ' _ > ,
21
+ arms : & ' tcx [ Arm < ' _ > ] ,
22
+ ) {
17
23
let ty = cx. typeck_results ( ) . expr_ty ( scrutinee) ;
18
- if let Some ( ty_name) = if is_type_diagnostic_item ( cx, ty, sym:: Option ) {
24
+ if let Some ( ( or_arm, unwrap_arm) ) = applicable_or_arm ( cx, arms) {
25
+ check_and_lint ( cx, expr, unwrap_arm. pat , scrutinee, unwrap_arm. body , or_arm. body , ty) ;
26
+ }
27
+ }
28
+
29
+ pub ( super ) fn check_if_let < ' tcx > (
30
+ cx : & LateContext < ' tcx > ,
31
+ expr : & ' tcx Expr < ' _ > ,
32
+ let_pat : & ' tcx Pat < ' _ > ,
33
+ let_expr : & ' tcx Expr < ' _ > ,
34
+ then_expr : & ' tcx Expr < ' _ > ,
35
+ else_expr : & ' tcx Expr < ' _ > ,
36
+ ) {
37
+ let ty = cx. typeck_results ( ) . expr_ty ( let_expr) ;
38
+ check_and_lint ( cx, expr, let_pat, let_expr, then_expr, peel_blocks ( else_expr) , ty) ;
39
+ }
40
+
41
+ fn check_and_lint < ' tcx > (
42
+ cx : & LateContext < ' tcx > ,
43
+ expr : & ' tcx Expr < ' _ > ,
44
+ let_pat : & ' tcx Pat < ' _ > ,
45
+ let_expr : & ' tcx Expr < ' _ > ,
46
+ then_expr : & ' tcx Expr < ' _ > ,
47
+ else_expr : & ' tcx Expr < ' _ > ,
48
+ ty : Ty < ' tcx > ,
49
+ ) {
50
+ if let PatKind :: TupleStruct ( ref qpath, [ unwrap_pat] , _) = let_pat. kind
51
+ && let Res :: Def ( DefKind :: Ctor ( ..) , ctor_id) = cx. qpath_res ( qpath, let_pat. hir_id )
52
+ && let Some ( variant_id) = cx. tcx . opt_parent ( ctor_id)
53
+ && ( cx. tcx . lang_items ( ) . option_some_variant ( ) == Some ( variant_id)
54
+ || cx. tcx . lang_items ( ) . result_ok_variant ( ) == Some ( variant_id) )
55
+ && let PatKind :: Binding ( _, binding_hir_id, ..) = unwrap_pat. kind
56
+ && path_to_local_id ( peel_blocks ( then_expr) , binding_hir_id)
57
+ && cx. typeck_results ( ) . expr_adjustments ( then_expr) . is_empty ( )
58
+ && let Some ( ty_name) = find_type_name ( cx, ty)
59
+ && let Some ( or_body_snippet) = snippet_opt ( cx, else_expr. span )
60
+ && let Some ( indent) = indent_of ( cx, expr. span )
61
+ && constant_simple ( cx, cx. typeck_results ( ) , else_expr) . is_some ( )
62
+ {
63
+ lint ( cx, expr, let_expr, ty_name, or_body_snippet, indent) ;
64
+ }
65
+ }
66
+
67
+ fn find_type_name < ' tcx > ( cx : & LateContext < ' tcx > , ty : Ty < ' tcx > ) -> Option < & ' static str > {
68
+ if is_type_diagnostic_item ( cx, ty, sym:: Option ) {
19
69
Some ( "Option" )
20
70
} else if is_type_diagnostic_item ( cx, ty, sym:: Result ) {
21
71
Some ( "Result" )
22
72
} else {
23
73
None
24
- } && let Some ( or_arm) = applicable_or_arm ( cx, arms)
25
- && let Some ( or_body_snippet) = snippet_opt ( cx, or_arm. body . span )
26
- && let Some ( indent) = indent_of ( cx, expr. span )
27
- && constant_simple ( cx, cx. typeck_results ( ) , or_arm. body ) . is_some ( )
28
- {
29
- let reindented_or_body = reindent_multiline ( or_body_snippet. into ( ) , true , Some ( indent) ) ;
30
-
31
- let mut app = Applicability :: MachineApplicable ;
32
- let suggestion = sugg:: Sugg :: hir_with_context ( cx, scrutinee, expr. span . ctxt ( ) , ".." , & mut app) . maybe_par ( ) ;
33
- span_lint_and_sugg (
34
- cx,
35
- MANUAL_UNWRAP_OR ,
36
- expr. span ,
37
- format ! ( "this pattern reimplements `{ty_name}::unwrap_or`" ) ,
38
- "replace with" ,
39
- format ! ( "{suggestion}.unwrap_or({reindented_or_body})" , ) ,
40
- app,
41
- ) ;
42
74
}
43
75
}
44
76
45
- fn applicable_or_arm < ' a > ( cx : & LateContext < ' _ > , arms : & ' a [ Arm < ' a > ] ) -> Option < & ' a Arm < ' a > > {
77
+ fn applicable_or_arm < ' a > ( cx : & LateContext < ' _ > , arms : & ' a [ Arm < ' a > ] ) -> Option < ( & ' a Arm < ' a > , & ' a Arm < ' a > ) > {
46
78
if arms. len ( ) == 2
47
79
&& arms. iter ( ) . all ( |arm| arm. guard . is_none ( ) )
48
80
&& let Some ( ( idx, or_arm) ) = arms. iter ( ) . enumerate ( ) . find ( |( _, arm) | match arm. pat . kind {
@@ -54,18 +86,33 @@ fn applicable_or_arm<'a>(cx: &LateContext<'_>, arms: &'a [Arm<'a>]) -> Option<&'
54
86
_ => false ,
55
87
} )
56
88
&& let unwrap_arm = & arms[ 1 - idx]
57
- && let PatKind :: TupleStruct ( ref qpath, [ unwrap_pat] , _) = unwrap_arm. pat . kind
58
- && let Res :: Def ( DefKind :: Ctor ( ..) , ctor_id) = cx. qpath_res ( qpath, unwrap_arm. pat . hir_id )
59
- && let Some ( variant_id) = cx. tcx . opt_parent ( ctor_id)
60
- && ( cx. tcx . lang_items ( ) . option_some_variant ( ) == Some ( variant_id)
61
- || cx. tcx . lang_items ( ) . result_ok_variant ( ) == Some ( variant_id) )
62
- && let PatKind :: Binding ( _, binding_hir_id, ..) = unwrap_pat. kind
63
- && path_to_local_id ( unwrap_arm. body , binding_hir_id)
64
- && cx. typeck_results ( ) . expr_adjustments ( unwrap_arm. body ) . is_empty ( )
65
89
&& !contains_return_break_continue_macro ( or_arm. body )
66
90
{
67
- Some ( or_arm)
91
+ Some ( ( or_arm, unwrap_arm ) )
68
92
} else {
69
93
None
70
94
}
71
95
}
96
+
97
+ fn lint < ' tcx > (
98
+ cx : & LateContext < ' tcx > ,
99
+ expr : & Expr < ' tcx > ,
100
+ scrutinee : & ' tcx Expr < ' _ > ,
101
+ ty_name : & str ,
102
+ or_body_snippet : String ,
103
+ indent : usize ,
104
+ ) {
105
+ let reindented_or_body = reindent_multiline ( or_body_snippet. into ( ) , true , Some ( indent) ) ;
106
+
107
+ let mut app = Applicability :: MachineApplicable ;
108
+ let suggestion = sugg:: Sugg :: hir_with_context ( cx, scrutinee, expr. span . ctxt ( ) , ".." , & mut app) . maybe_par ( ) ;
109
+ span_lint_and_sugg (
110
+ cx,
111
+ MANUAL_UNWRAP_OR ,
112
+ expr. span ,
113
+ format ! ( "this pattern reimplements `{ty_name}::unwrap_or`" ) ,
114
+ "replace with" ,
115
+ format ! ( "{suggestion}.unwrap_or({reindented_or_body})" , ) ,
116
+ app,
117
+ ) ;
118
+ }
0 commit comments