rust/tests/ui/collection_is_never_read.rs

239 lines
5.3 KiB
Rust
Raw Normal View History

2023-06-06 22:56:57 +02:00
#![allow(unused, clippy::useless_vec)]
2023-02-16 18:12:11 +00:00
#![warn(clippy::collection_is_never_read)]
2023-02-28 22:08:15 +00:00
use std::collections::{HashMap, HashSet};
2023-02-16 18:12:11 +00:00
fn main() {}
fn not_a_collection() {
// TODO: Expand `collection_is_never_read` beyond collections?
let mut x = 10; // Ok
x += 1;
}
fn no_access_at_all() {
// Other lints should catch this.
let x = vec![1, 2, 3]; // Ok
}
fn write_without_read() {
// The main use case for `collection_is_never_read`.
let mut x = HashMap::new();
//~^ ERROR: collection is never read
//~| NOTE: `-D clippy::collection-is-never-read` implied by `-D warnings`
2023-02-16 18:12:11 +00:00
x.insert(1, 2);
}
fn read_without_write() {
let mut x = vec![1, 2, 3]; // Ok
let _ = x.len();
}
fn write_and_read() {
let mut x = vec![1, 2, 3]; // Ok
x.push(4);
let _ = x.len();
}
fn write_after_read() {
// TODO: Warn here, but this requires more extensive data flow analysis.
let mut x = vec![1, 2, 3]; // Ok
let _ = x.len();
x.push(4); // Pointless
}
fn write_before_reassign() {
// TODO: Warn here, but this requires more extensive data flow analysis.
let mut x = HashMap::new(); // Ok
x.insert(1, 2); // Pointless
x = HashMap::new();
let _ = x.len();
}
fn read_in_closure() {
let mut x = HashMap::new(); // Ok
x.insert(1, 2);
let _ = || {
let _ = x.len();
};
}
fn write_in_closure() {
let mut x = vec![1, 2, 3];
//~^ ERROR: collection is never read
2023-02-16 18:12:11 +00:00
let _ = || {
x.push(4);
};
}
fn read_in_format() {
let mut x = HashMap::new(); // Ok
x.insert(1, 2);
format!("{x:?}");
}
fn shadowing_1() {
let x = HashMap::<usize, usize>::new(); // Ok
let _ = x.len();
let mut x = HashMap::new();
//~^ ERROR: collection is never read
2023-02-16 18:12:11 +00:00
x.insert(1, 2);
}
fn shadowing_2() {
let mut x = HashMap::new();
//~^ ERROR: collection is never read
2023-02-16 18:12:11 +00:00
x.insert(1, 2);
let x = HashMap::<usize, usize>::new(); // Ok
let _ = x.len();
}
#[allow(clippy::let_unit_value)]
2023-03-12 14:59:10 +00:00
fn fake_read_1() {
let mut x = vec![1, 2, 3];
//~^ ERROR: collection is never read
2023-02-16 18:12:11 +00:00
x.reverse();
let _: () = x.clear();
}
2023-03-12 14:59:10 +00:00
fn fake_read_2() {
let mut x = vec![1, 2, 3];
//~^ ERROR: collection is never read
2023-03-12 14:59:10 +00:00
x.reverse();
println!("{:?}", x.push(5));
}
2023-02-16 18:12:11 +00:00
fn assignment() {
let mut x = vec![1, 2, 3];
//~^ ERROR: collection is never read
2023-02-16 18:12:11 +00:00
let y = vec![4, 5, 6]; // Ok
x = y;
}
#[allow(clippy::self_assignment)]
fn self_assignment() {
let mut x = vec![1, 2, 3];
//~^ ERROR: collection is never read
2023-02-16 18:12:11 +00:00
x = x;
}
fn method_argument_but_not_target() {
struct MyStruct;
impl MyStruct {
fn my_method(&self, _argument: &[usize]) {}
}
let my_struct = MyStruct;
let mut x = vec![1, 2, 3]; // Ok
x.reverse();
my_struct.my_method(&x);
}
fn insert_is_not_a_read() {
let mut x = HashSet::new();
//~^ ERROR: collection is never read
x.insert(5);
}
fn insert_is_a_read() {
let mut x = HashSet::new(); // Ok
if x.insert(5) {
println!("5 was inserted");
}
}
2023-02-28 22:08:15 +00:00
fn not_read_if_return_value_not_used() {
// `is_empty` does not modify the set, so it's a query. But since the return value is not used, the
// lint does not consider it a read here.
let x = vec![1, 2, 3];
//~^ ERROR: collection is never read
2023-02-28 22:08:15 +00:00
x.is_empty();
}
fn extension_traits() {
trait VecExt<T> {
fn method_with_side_effect(&self);
fn method_without_side_effect(&self);
}
impl<T> VecExt<T> for Vec<T> {
fn method_with_side_effect(&self) {
println!("my length: {}", self.len());
}
fn method_without_side_effect(&self) {}
}
let x = vec![1, 2, 3]; // Ok
x.method_with_side_effect();
let y = vec![1, 2, 3]; // Ok (false negative)
y.method_without_side_effect();
}
fn function_argument() {
#[allow(clippy::ptr_arg)]
fn foo<T>(v: &Vec<T>) -> usize {
v.len()
}
let x = vec![1, 2, 3]; // Ok
foo(&x);
}
fn supported_types() {
let mut x = std::collections::BTreeMap::new();
//~^ ERROR: collection is never read
x.insert(true, 1);
let mut x = std::collections::BTreeSet::new();
//~^ ERROR: collection is never read
x.insert(1);
let mut x = std::collections::BinaryHeap::new();
//~^ ERROR: collection is never read
x.push(1);
let mut x = std::collections::HashMap::new();
//~^ ERROR: collection is never read
x.insert(1, 2);
let mut x = std::collections::HashSet::new();
//~^ ERROR: collection is never read
x.insert(1);
let mut x = std::collections::LinkedList::new();
//~^ ERROR: collection is never read
x.push_front(1);
let mut x = Some(true);
//~^ ERROR: collection is never read
x.insert(false);
let mut x = String::from("hello");
//~^ ERROR: collection is never read
x.push('!');
let mut x = Vec::new();
//~^ ERROR: collection is never read
x.clear();
x.push(1);
let mut x = std::collections::VecDeque::new();
//~^ ERROR: collection is never read
x.push_front(1);
}
fn issue11783() {
struct Sender;
impl Sender {
fn send(&self, msg: String) -> Result<(), ()> {
// pretend to send message
println!("{msg}");
Ok(())
}
}
let mut users: Vec<Sender> = vec![];
users.retain(|user| user.send("hello".to_string()).is_ok());
}