rust-clippy/clippy_lints/src/map_clone.rs

138 lines
6 KiB
Rust
Raw Normal View History

2015-10-31 04:58:37 +00:00
use rustc::lint::*;
use rustc::hir::*;
2017-05-10 11:42:19 +00:00
use rustc::ty;
use syntax::ast;
2018-01-14 10:05:01 +00:00
use utils::{get_arg_name, is_adjusted, iter_input_pats, match_qpath, match_trait_method, match_type,
paths, remove_blocks, snippet, span_help_and_lint, walk_ptrs_ty, walk_ptrs_ty_depth};
2015-10-31 04:58:37 +00:00
/// **What it does:** Checks for mapping `clone()` over an iterator.
2015-12-14 12:31:28 +00:00
///
/// **Why is this bad?** It makes the code less readable than using the
/// `.cloned()` adapter.
2015-12-14 12:31:28 +00:00
///
2018-01-17 18:40:47 +00:00
/// **Known problems:** Sometimes `.cloned()` requires stricter trait
/// bound than `.map(|e| e.clone())` (which works because of the coercion).
/// See [#498](https://github.com/rust-lang-nursery/rust-clippy/issues/498).
2015-12-14 12:31:28 +00:00
///
2016-07-15 22:25:44 +00:00
/// **Example:**
/// ```rust
/// x.map(|e| e.clone());
/// ```
declare_lint! {
pub MAP_CLONE,
Warn,
"using `.map(|x| x.clone())` to clone an iterator or option's contents"
}
2015-10-31 04:58:37 +00:00
#[derive(Copy, Clone)]
2016-06-10 14:17:20 +00:00
pub struct Pass;
2015-10-31 04:58:37 +00:00
impl<'a, 'tcx> LateLintPass<'a, 'tcx> for Pass {
fn check_expr(&mut self, cx: &LateContext<'a, 'tcx>, expr: &'tcx Expr) {
2015-11-05 16:11:41 +00:00
// call to .map()
if let ExprMethodCall(ref method, _, ref args) = expr.node {
if method.name == "map" && args.len() == 2 {
2015-11-05 16:11:41 +00:00
match args[1].node {
2017-08-30 08:54:24 +00:00
ExprClosure(_, ref decl, closure_eid, _, _) => {
2017-02-02 16:53:28 +00:00
let body = cx.tcx.hir.body(closure_eid);
let closure_expr = remove_blocks(&body.value);
2017-05-10 11:42:19 +00:00
let ty = cx.tables.pat_ty(&body.arguments[0].pat);
if_chain! {
2015-11-05 16:11:41 +00:00
// nothing special in the argument, besides reference bindings
// (e.g. .map(|&x| x) )
if let Some(first_arg) = iter_input_pats(decl, body).next();
if let Some(arg_ident) = get_arg_name(&first_arg.pat);
2015-11-05 16:11:41 +00:00
// the method is being called on a known type (option or iterator)
if let Some(type_name) = get_type_name(cx, expr, &args[0]);
then {
// look for derefs, for .map(|x| *x)
if only_derefs(cx, &*closure_expr, arg_ident) &&
// .cloned() only removes one level of indirection, don't lint on more
walk_ptrs_ty_depth(cx.tables.pat_ty(&first_arg.pat)).1 == 1
{
// the argument is not an &mut T
if let ty::TyRef(_, tam) = ty.sty {
if tam.mutbl == MutImmutable {
span_help_and_lint(cx, MAP_CLONE, expr.span, &format!(
"you seem to be using .map() to clone the contents of an {}, consider \
using `.cloned()`", type_name),
&format!("try\n{}.cloned()", snippet(cx, args[0].span, "..")));
}
}
}
// explicit clone() calls ( .map(|x| x.clone()) )
else if let ExprMethodCall(ref clone_call, _, ref clone_args) = closure_expr.node {
if clone_call.name == "clone" &&
clone_args.len() == 1 &&
match_trait_method(cx, closure_expr, &paths::CLONE_TRAIT) &&
expr_eq_name(&clone_args[0], arg_ident)
{
2017-05-10 11:42:19 +00:00
span_help_and_lint(cx, MAP_CLONE, expr.span, &format!(
"you seem to be using .map() to clone the contents of an {}, consider \
using `.cloned()`", type_name),
&format!("try\n{}.cloned()", snippet(cx, args[0].span, "..")));
}
}
2016-06-06 00:09:19 +00:00
}
}
2016-12-20 17:21:30 +00:00
},
2017-09-05 09:33:04 +00:00
ExprPath(ref path) => if match_qpath(path, &paths::CLONE) {
let type_name = get_type_name(cx, expr, &args[0]).unwrap_or("_");
span_help_and_lint(
cx,
MAP_CLONE,
expr.span,
&format!(
"you seem to be using .map() to clone the contents of an \
{}, consider using `.cloned()`",
type_name
),
&format!("try\n{}.cloned()", snippet(cx, args[0].span, "..")),
);
2016-12-20 17:21:30 +00:00
},
2015-11-05 16:11:41 +00:00
_ => (),
2015-10-31 04:58:37 +00:00
}
}
}
}
}
fn expr_eq_name(expr: &Expr, id: ast::Name) -> bool {
2015-10-31 04:58:37 +00:00
match expr.node {
ExprPath(QPath::Resolved(None, ref path)) => {
2017-08-09 07:30:56 +00:00
let arg_segment = [
PathSegment {
name: id,
parameters: None,
infer_types: true,
2017-08-09 07:30:56 +00:00
},
];
!path.is_global() && path.segments[..] == arg_segment
2016-12-20 17:21:30 +00:00
},
2015-10-31 04:58:37 +00:00
_ => false,
}
}
fn get_type_name(cx: &LateContext, expr: &Expr, arg: &Expr) -> Option<&'static str> {
if match_trait_method(cx, expr, &paths::ITERATOR) {
2015-10-31 04:58:37 +00:00
Some("iterator")
2017-01-13 16:04:56 +00:00
} else if match_type(cx, walk_ptrs_ty(cx.tables.expr_ty(arg)), &paths::OPTION) {
2015-10-31 04:58:37 +00:00
Some("Option")
} else {
None
}
}
fn only_derefs(cx: &LateContext, expr: &Expr, id: ast::Name) -> bool {
2015-11-04 03:11:40 +00:00
match expr.node {
2016-01-04 04:26:12 +00:00
ExprUnary(UnDeref, ref subexpr) if !is_adjusted(cx, subexpr) => only_derefs(cx, subexpr, id),
_ => expr_eq_name(expr, id),
2015-10-31 04:58:37 +00:00
}
}
2016-06-10 14:17:20 +00:00
impl LintPass for Pass {
2015-10-31 04:58:37 +00:00
fn get_lints(&self) -> LintArray {
lint_array!(MAP_CLONE)
}
}