73 lines
2.5 KiB
Rust
73 lines
2.5 KiB
Rust
|
use clippy_utils::diagnostics::span_lint_and_sugg;
|
||
|
use clippy_utils::msrvs::{self, Msrv};
|
||
|
use clippy_utils::{is_trait_method, match_def_path, paths, peel_hir_expr_refs};
|
||
|
use rustc_errors::Applicability;
|
||
|
use rustc_hir::def::{DefKind, Res};
|
||
|
use rustc_hir::*;
|
||
|
use rustc_lint::{LateContext, LateLintPass};
|
||
|
use rustc_middle::ty;
|
||
|
use rustc_session::{declare_tool_lint, impl_lint_pass};
|
||
|
use rustc_span::sym;
|
||
|
|
||
|
declare_clippy_lint! {
|
||
|
/// ### What it does
|
||
|
/// Checks for references on `std::path::MAIN_SEPARATOR.to_string()` used
|
||
|
/// to build a `&str`.
|
||
|
///
|
||
|
/// ### Why is this bad?
|
||
|
/// There exists a `std::path::MAIN_SEPARATOR_STR` which does not require
|
||
|
/// an extra memory allocation.
|
||
|
///
|
||
|
/// ### Example
|
||
|
/// ```rust
|
||
|
/// let s: &str = &std::path::MAIN_SEPARATOR.to_string();
|
||
|
/// ```
|
||
|
/// Use instead:
|
||
|
/// ```rust
|
||
|
/// let s: &str = std::path::MAIN_SEPARATOR_STR;
|
||
|
/// ```
|
||
|
#[clippy::version = "1.70.0"]
|
||
|
pub MANUAL_MAIN_SEPARATOR_STR,
|
||
|
complexity,
|
||
|
"`&std::path::MAIN_SEPARATOR.to_string()` can be replaced by `std::path::MAIN_SEPARATOR_STR`"
|
||
|
}
|
||
|
|
||
|
pub struct ManualMainSeparatorStr {
|
||
|
msrv: Msrv,
|
||
|
}
|
||
|
|
||
|
impl ManualMainSeparatorStr {
|
||
|
#[must_use]
|
||
|
pub fn new(msrv: Msrv) -> Self {
|
||
|
Self { msrv }
|
||
|
}
|
||
|
}
|
||
|
|
||
|
impl_lint_pass!(ManualMainSeparatorStr => [MANUAL_MAIN_SEPARATOR_STR]);
|
||
|
|
||
|
impl LateLintPass<'_> for ManualMainSeparatorStr {
|
||
|
fn check_expr(&mut self, cx: &LateContext<'_>, expr: &Expr<'_>) {
|
||
|
if self.msrv.meets(msrvs::PATH_MAIN_SEPARATOR_STR) &&
|
||
|
let (target, _) = peel_hir_expr_refs(expr) &&
|
||
|
is_trait_method(cx, target, sym::ToString) &&
|
||
|
let ExprKind::MethodCall(path, receiver, &[], _) = target.kind &&
|
||
|
path.ident.name == sym::to_string &&
|
||
|
let ExprKind::Path(QPath::Resolved(None, path)) = receiver.kind &&
|
||
|
let Res::Def(DefKind::Const, receiver_def_id) = path.res &&
|
||
|
match_def_path(cx, receiver_def_id, &paths::PATH_MAIN_SEPARATOR) &&
|
||
|
let ty::Ref(_, ty, Mutability::Not) = cx.typeck_results().expr_ty_adjusted(expr).kind() &&
|
||
|
ty.is_str()
|
||
|
{
|
||
|
span_lint_and_sugg(
|
||
|
cx,
|
||
|
MANUAL_MAIN_SEPARATOR_STR,
|
||
|
expr.span,
|
||
|
"taking a reference on `std::path::MAIN_SEPARATOR` conversion to `String`",
|
||
|
"replace with",
|
||
|
"std::path::MAIN_SEPARATOR_STR".to_owned(),
|
||
|
Applicability::MachineApplicable,
|
||
|
);
|
||
|
}
|
||
|
}
|
||
|
}
|