Skip to content

Commit 56c8235

Browse files
committed
Auto merge of rust-lang#10943 - y21:issue7189, r=llogiq
[`map_identity`]: recognize tuple identity function Fixes rust-lang#7189 This lint now recognizes `.map(|(a, b)| (a, b))` as a useless `map` call. changelog: [`map_identity`]: recognize tuple identity function
2 parents 2b030eb + d6fc606 commit 56c8235

File tree

5 files changed

+96
-10
lines changed

5 files changed

+96
-10
lines changed

clippy_utils/src/lib.rs

+17-8
Original file line numberDiff line numberDiff line change
@@ -2032,17 +2032,26 @@ pub fn is_must_use_func_call(cx: &LateContext<'_>, expr: &Expr<'_>) -> bool {
20322032
/// * `|x| return x`
20332033
/// * `|x| { return x }`
20342034
/// * `|x| { return x; }`
2035+
/// * `|(x, y)| (x, y)`
20352036
///
20362037
/// Consider calling [`is_expr_untyped_identity_function`] or [`is_expr_identity_function`] instead.
20372038
fn is_body_identity_function(cx: &LateContext<'_>, func: &Body<'_>) -> bool {
2038-
let id = if_chain! {
2039-
if let [param] = func.params;
2040-
if let PatKind::Binding(_, id, _, _) = param.pat.kind;
2041-
then {
2042-
id
2043-
} else {
2044-
return false;
2039+
fn check_pat(cx: &LateContext<'_>, pat: &Pat<'_>, expr: &Expr<'_>) -> bool {
2040+
match (pat.kind, expr.kind) {
2041+
(PatKind::Binding(_, id, _, _), _) => {
2042+
path_to_local_id(expr, id) && cx.typeck_results().expr_adjustments(expr).is_empty()
2043+
},
2044+
(PatKind::Tuple(pats, dotdot), ExprKind::Tup(tup))
2045+
if dotdot.as_opt_usize().is_none() && pats.len() == tup.len() =>
2046+
{
2047+
pats.iter().zip(tup).all(|(pat, expr)| check_pat(cx, pat, expr))
2048+
},
2049+
_ => false,
20452050
}
2051+
}
2052+
2053+
let [param] = func.params else {
2054+
return false;
20462055
};
20472056

20482057
let mut expr = func.value;
@@ -2075,7 +2084,7 @@ fn is_body_identity_function(cx: &LateContext<'_>, func: &Body<'_>) -> bool {
20752084
}
20762085
}
20772086
},
2078-
_ => return path_to_local_id(expr, id) && cx.typeck_results().expr_adjustments(expr).is_empty(),
2087+
_ => return check_pat(cx, param.pat, expr),
20792088
}
20802089
}
20812090
}

lintcheck/src/main.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -523,7 +523,7 @@ fn gather_stats(clippy_warnings: &[ClippyWarning]) -> (String, HashMap<&String,
523523
.for_each(|wrn| *counter.entry(&wrn.lint_type).or_insert(0) += 1);
524524

525525
// collect into a tupled list for sorting
526-
let mut stats: Vec<(&&String, &usize)> = counter.iter().map(|(lint, count)| (lint, count)).collect();
526+
let mut stats: Vec<(&&String, &usize)> = counter.iter().collect();
527527
// sort by "000{count} {clippy::lintname}"
528528
// to not have a lint with 200 and 2 warnings take the same spot
529529
stats.sort_by_key(|(lint, count)| format!("{count:0>4}, {lint}"));

tests/ui/map_identity.fixed

+24
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,30 @@ fn main() {
2222
let _ = Ok(1).map_err(std::convert::identity::<u32>);
2323
}
2424

25+
fn issue7189() {
26+
// should lint
27+
let x = [(1, 2), (3, 4)];
28+
let _ = x.iter();
29+
let _ = x.iter();
30+
let _ = x.iter();
31+
32+
let y = [(1, 2, (3, (4,))), (5, 6, (7, (8,)))];
33+
let _ = y.iter();
34+
35+
// should not lint
36+
let _ = x.iter().map(|(x, y)| (x, y, y));
37+
let _ = x.iter().map(|(x, _y)| (x,));
38+
let _ = x.iter().map(|(x, _)| (x,));
39+
let _ = x.iter().map(|(x, ..)| (x,));
40+
let _ = y.iter().map(|(x, y, (z, _))| (x, y, (z, z)));
41+
let _ = y
42+
.iter()
43+
.map(|(x, y, (z, _)): &(i32, i32, (i32, (i32,)))| (x, y, (z, z)));
44+
let _ = y
45+
.iter()
46+
.map(|(x, y, (z, (w,))): &(i32, i32, (i32, (i32,)))| (x, y, (z, (w,))));
47+
}
48+
2549
fn not_identity(x: &u16) -> u16 {
2650
*x
2751
}

tests/ui/map_identity.rs

+26
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,32 @@ fn main() {
2424
let _ = Ok(1).map_err(std::convert::identity::<u32>);
2525
}
2626

27+
fn issue7189() {
28+
// should lint
29+
let x = [(1, 2), (3, 4)];
30+
let _ = x.iter().map(|(x, y)| (x, y));
31+
let _ = x.iter().map(|(x, y)| {
32+
return (x, y);
33+
});
34+
let _ = x.iter().map(|(x, y)| return (x, y));
35+
36+
let y = [(1, 2, (3, (4,))), (5, 6, (7, (8,)))];
37+
let _ = y.iter().map(|(x, y, (z, (w,)))| (x, y, (z, (w,))));
38+
39+
// should not lint
40+
let _ = x.iter().map(|(x, y)| (x, y, y));
41+
let _ = x.iter().map(|(x, _y)| (x,));
42+
let _ = x.iter().map(|(x, _)| (x,));
43+
let _ = x.iter().map(|(x, ..)| (x,));
44+
let _ = y.iter().map(|(x, y, (z, _))| (x, y, (z, z)));
45+
let _ = y
46+
.iter()
47+
.map(|(x, y, (z, _)): &(i32, i32, (i32, (i32,)))| (x, y, (z, z)));
48+
let _ = y
49+
.iter()
50+
.map(|(x, y, (z, (w,))): &(i32, i32, (i32, (i32,)))| (x, y, (z, (w,))));
51+
}
52+
2753
fn not_identity(x: &u16) -> u16 {
2854
*x
2955
}

tests/ui/map_identity.stderr

+28-1
Original file line numberDiff line numberDiff line change
@@ -40,5 +40,32 @@ error: unnecessary map of the identity function
4040
LL | let _: Result<u32, u32> = Ok(1).map_err(|a| a);
4141
| ^^^^^^^^^^^^^^^ help: remove the call to `map_err`
4242

43-
error: aborting due to 6 previous errors
43+
error: unnecessary map of the identity function
44+
--> $DIR/map_identity.rs:30:21
45+
|
46+
LL | let _ = x.iter().map(|(x, y)| (x, y));
47+
| ^^^^^^^^^^^^^^^^^^^^^ help: remove the call to `map`
48+
49+
error: unnecessary map of the identity function
50+
--> $DIR/map_identity.rs:31:21
51+
|
52+
LL | let _ = x.iter().map(|(x, y)| {
53+
| _____________________^
54+
LL | | return (x, y);
55+
LL | | });
56+
| |______^ help: remove the call to `map`
57+
58+
error: unnecessary map of the identity function
59+
--> $DIR/map_identity.rs:34:21
60+
|
61+
LL | let _ = x.iter().map(|(x, y)| return (x, y));
62+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^ help: remove the call to `map`
63+
64+
error: unnecessary map of the identity function
65+
--> $DIR/map_identity.rs:37:21
66+
|
67+
LL | let _ = y.iter().map(|(x, y, (z, (w,)))| (x, y, (z, (w,))));
68+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ help: remove the call to `map`
69+
70+
error: aborting due to 10 previous errors
4471

0 commit comments

Comments
 (0)