Skip to content

Commit 8fe840d

Browse files
committed
New lint needless_as_bytes
1 parent 43e3384 commit 8fe840d

8 files changed

+173
-0
lines changed

CHANGELOG.md

+1
Original file line numberDiff line numberDiff line change
@@ -5725,6 +5725,7 @@ Released 2018-09-13
57255725
[`mutex_integer`]: https://rust-lang.github.io/rust-clippy/master/index.html#mutex_integer
57265726
[`naive_bytecount`]: https://rust-lang.github.io/rust-clippy/master/index.html#naive_bytecount
57275727
[`needless_arbitrary_self_type`]: https://rust-lang.github.io/rust-clippy/master/index.html#needless_arbitrary_self_type
5728+
[`needless_as_bytes`]: https://rust-lang.github.io/rust-clippy/master/index.html#needless_as_bytes
57285729
[`needless_bitwise_bool`]: https://rust-lang.github.io/rust-clippy/master/index.html#needless_bitwise_bool
57295730
[`needless_bool`]: https://rust-lang.github.io/rust-clippy/master/index.html#needless_bool
57305731
[`needless_bool_assign`]: https://rust-lang.github.io/rust-clippy/master/index.html#needless_bool_assign
12 KB
Binary file not shown.

clippy_lints/src/declared_lints.rs

+1
Original file line numberDiff line numberDiff line change
@@ -526,6 +526,7 @@ pub static LINTS: &[&crate::LintInfo] = &[
526526
crate::mutex_atomic::MUTEX_ATOMIC_INFO,
527527
crate::mutex_atomic::MUTEX_INTEGER_INFO,
528528
crate::needless_arbitrary_self_type::NEEDLESS_ARBITRARY_SELF_TYPE_INFO,
529+
crate::needless_as_bytes::NEEDLESS_AS_BYTES_INFO,
529530
crate::needless_bool::BOOL_COMPARISON_INFO,
530531
crate::needless_bool::NEEDLESS_BOOL_INFO,
531532
crate::needless_bool::NEEDLESS_BOOL_ASSIGN_INFO,

clippy_lints/src/lib.rs

+2
Original file line numberDiff line numberDiff line change
@@ -249,6 +249,7 @@ mod mut_reference;
249249
mod mutable_debug_assertion;
250250
mod mutex_atomic;
251251
mod needless_arbitrary_self_type;
252+
mod needless_as_bytes;
252253
mod needless_bool;
253254
mod needless_borrowed_ref;
254255
mod needless_borrows_for_generic_args;
@@ -944,5 +945,6 @@ pub fn register_lints(store: &mut rustc_lint::LintStore, conf: &'static Conf) {
944945
store.register_late_pass(|_| Box::new(manual_is_power_of_two::ManualIsPowerOfTwo));
945946
store.register_late_pass(|_| Box::new(non_zero_suggestions::NonZeroSuggestions));
946947
store.register_late_pass(move |_| Box::new(unused_trait_names::UnusedTraitNames::new(conf)));
948+
store.register_late_pass(|_| Box::new(needless_as_bytes::NeedlessAsBytes));
947949
// add lints here, do not remove this comment, it's used in `new_lint`
948950
}

clippy_lints/src/needless_as_bytes.rs

+60
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,60 @@
1+
use clippy_utils::diagnostics::span_lint_and_sugg;
2+
use clippy_utils::sugg::Sugg;
3+
use clippy_utils::ty::is_type_lang_item;
4+
use rustc_errors::Applicability;
5+
use rustc_hir::{Expr, ExprKind, LangItem};
6+
use rustc_lint::{LateContext, LateLintPass};
7+
use rustc_session::declare_lint_pass;
8+
use rustc_span::sym;
9+
10+
declare_clippy_lint! {
11+
/// ### What it does
12+
/// It detects useless calls to `str::as_bytes()` before calling `len()` or `is_empty()`.
13+
///
14+
/// ### Why is this bad?
15+
/// The `len()` and `is_empty()` methods are also directly available on strings, and they
16+
/// return identical results. In particular, `len()` on a string returns the number of
17+
/// bytes.
18+
///
19+
/// ### Example
20+
/// ```
21+
/// let len = "some string".as_bytes().len();
22+
/// let b = "some string".as_bytes().is_empty();
23+
/// ```
24+
/// Use instead:
25+
/// ```
26+
/// let len = "some string".len();
27+
/// let b = "some string".is_empty();
28+
/// ```
29+
#[clippy::version = "1.83.0"]
30+
pub NEEDLESS_AS_BYTES,
31+
complexity,
32+
"detect useless calls to `as_bytes()`"
33+
}
34+
35+
declare_lint_pass!(NeedlessAsBytes => [NEEDLESS_AS_BYTES]);
36+
37+
impl LateLintPass<'_> for NeedlessAsBytes {
38+
fn check_expr(&mut self, cx: &LateContext<'_>, expr: &Expr<'_>) {
39+
if let ExprKind::MethodCall(method2, receiver2, &[], _) = expr.kind
40+
&& cx.typeck_results().expr_ty_adjusted(receiver2).peel_refs().is_slice()
41+
&& (method2.ident.name == sym::len || method2.ident.name.as_str() == "is_empty")
42+
&& let ExprKind::MethodCall(method1, receiver1, &[], _) = receiver2.kind
43+
&& let ty1 = cx.typeck_results().expr_ty_adjusted(receiver1).peel_refs()
44+
&& (is_type_lang_item(cx, ty1, LangItem::String) || ty1.is_str())
45+
&& method1.ident.name.as_str() == "as_bytes"
46+
{
47+
let mut app = Applicability::MachineApplicable;
48+
let sugg = Sugg::hir_with_context(cx, receiver1, expr.span.ctxt(), "..", &mut app);
49+
span_lint_and_sugg(
50+
cx,
51+
NEEDLESS_AS_BYTES,
52+
expr.span,
53+
"needless call to `as_bytes()`",
54+
format!("`{}()` can be called directly on strings", method2.ident.name),
55+
format!("{sugg}.{}()", method2.ident.name),
56+
app,
57+
);
58+
}
59+
}
60+
}

tests/ui/needless_as_bytes.fixed

+40
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,40 @@
1+
#![warn(clippy::needless_as_bytes)]
2+
#![allow(clippy::const_is_empty)]
3+
4+
struct S;
5+
6+
impl S {
7+
fn as_bytes(&self) -> &[u8] {
8+
&[]
9+
}
10+
}
11+
12+
fn main() {
13+
if "some string".is_empty() {
14+
//~^ needless_as_bytes
15+
println!("len = {}", "some string".len());
16+
//~^ needless_as_bytes
17+
}
18+
19+
let s = String::from("yet another string");
20+
if s.is_empty() {
21+
//~^ needless_as_bytes
22+
println!("len = {}", s.len());
23+
//~^ needless_as_bytes
24+
}
25+
26+
// Do not lint
27+
let _ = S.as_bytes().is_empty();
28+
let _ = S.as_bytes().len();
29+
let _ = (&String::new() as &dyn AsBytes).as_bytes().len();
30+
}
31+
32+
pub trait AsBytes {
33+
fn as_bytes(&self) -> &[u8];
34+
}
35+
36+
impl AsBytes for String {
37+
fn as_bytes(&self) -> &[u8] {
38+
&[]
39+
}
40+
}

tests/ui/needless_as_bytes.rs

+40
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,40 @@
1+
#![warn(clippy::needless_as_bytes)]
2+
#![allow(clippy::const_is_empty)]
3+
4+
struct S;
5+
6+
impl S {
7+
fn as_bytes(&self) -> &[u8] {
8+
&[]
9+
}
10+
}
11+
12+
fn main() {
13+
if "some string".as_bytes().is_empty() {
14+
//~^ needless_as_bytes
15+
println!("len = {}", "some string".as_bytes().len());
16+
//~^ needless_as_bytes
17+
}
18+
19+
let s = String::from("yet another string");
20+
if s.as_bytes().is_empty() {
21+
//~^ needless_as_bytes
22+
println!("len = {}", s.as_bytes().len());
23+
//~^ needless_as_bytes
24+
}
25+
26+
// Do not lint
27+
let _ = S.as_bytes().is_empty();
28+
let _ = S.as_bytes().len();
29+
let _ = (&String::new() as &dyn AsBytes).as_bytes().len();
30+
}
31+
32+
pub trait AsBytes {
33+
fn as_bytes(&self) -> &[u8];
34+
}
35+
36+
impl AsBytes for String {
37+
fn as_bytes(&self) -> &[u8] {
38+
&[]
39+
}
40+
}

tests/ui/needless_as_bytes.stderr

+29
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
error: needless call to `as_bytes()`
2+
--> tests/ui/needless_as_bytes.rs:13:8
3+
|
4+
LL | if "some string".as_bytes().is_empty() {
5+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ help: `is_empty()` can be called directly on strings: `"some string".is_empty()`
6+
|
7+
= note: `-D clippy::needless-as-bytes` implied by `-D warnings`
8+
= help: to override `-D warnings` add `#[allow(clippy::needless_as_bytes)]`
9+
10+
error: needless call to `as_bytes()`
11+
--> tests/ui/needless_as_bytes.rs:15:30
12+
|
13+
LL | println!("len = {}", "some string".as_bytes().len());
14+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ help: `len()` can be called directly on strings: `"some string".len()`
15+
16+
error: needless call to `as_bytes()`
17+
--> tests/ui/needless_as_bytes.rs:20:8
18+
|
19+
LL | if s.as_bytes().is_empty() {
20+
| ^^^^^^^^^^^^^^^^^^^^^^^ help: `is_empty()` can be called directly on strings: `s.is_empty()`
21+
22+
error: needless call to `as_bytes()`
23+
--> tests/ui/needless_as_bytes.rs:22:30
24+
|
25+
LL | println!("len = {}", s.as_bytes().len());
26+
| ^^^^^^^^^^^^^^^^^^ help: `len()` can be called directly on strings: `s.len()`
27+
28+
error: aborting due to 4 previous errors
29+

0 commit comments

Comments
 (0)