2023-08-11 14:05:13 +02:00
|
|
|
#![allow(unused)]
|
|
|
|
#![warn(clippy::impossible_comparisons)]
|
|
|
|
#![warn(clippy::redundant_comparisons)]
|
|
|
|
#![allow(clippy::no_effect)]
|
|
|
|
#![allow(clippy::short_circuit_statement)]
|
|
|
|
#![allow(clippy::manual_range_contains)]
|
|
|
|
|
|
|
|
const STATUS_BAD_REQUEST: u16 = 400;
|
|
|
|
const STATUS_SERVER_ERROR: u16 = 500;
|
|
|
|
|
|
|
|
struct Status {
|
|
|
|
code: u16,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl PartialEq<u16> for Status {
|
|
|
|
fn eq(&self, other: &u16) -> bool {
|
|
|
|
self.code == *other
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl PartialOrd<u16> for Status {
|
|
|
|
fn partial_cmp(&self, other: &u16) -> Option<std::cmp::Ordering> {
|
|
|
|
self.code.partial_cmp(other)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl PartialEq<Status> for u16 {
|
|
|
|
fn eq(&self, other: &Status) -> bool {
|
|
|
|
*self == other.code
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl PartialOrd<Status> for u16 {
|
|
|
|
fn partial_cmp(&self, other: &Status) -> Option<std::cmp::Ordering> {
|
|
|
|
self.partial_cmp(&other.code)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn main() {
|
|
|
|
let status_code = 500; // Value doesn't matter for the lint
|
|
|
|
let status = Status { code: status_code };
|
|
|
|
|
2023-08-24 21:32:12 +02:00
|
|
|
// Correct
|
|
|
|
status_code >= 400 && status_code < 500;
|
2023-08-11 14:05:13 +02:00
|
|
|
status_code <= 400 && status_code > 500;
|
2023-08-24 21:32:12 +02:00
|
|
|
//~^ ERROR: boolean expression will never evaluate to 'true'
|
|
|
|
//~| NOTE: since `400` < `500`, the expression evaluates to false for any value of `st
|
2023-08-11 14:05:13 +02:00
|
|
|
status_code > 500 && status_code < 400;
|
2023-08-24 21:32:12 +02:00
|
|
|
//~^ ERROR: boolean expression will never evaluate to 'true'
|
|
|
|
//~| NOTE: since `500` > `400`, the expression evaluates to false for any value of `st
|
2023-08-11 14:05:13 +02:00
|
|
|
status_code < 500 && status_code > 500;
|
2023-08-24 21:32:12 +02:00
|
|
|
//~^ ERROR: boolean expression will never evaluate to 'true'
|
|
|
|
//~| NOTE: `status_code` cannot simultaneously be greater than and less than `500`
|
2023-08-11 14:05:13 +02:00
|
|
|
|
|
|
|
// More complex expressions
|
|
|
|
status_code < { 400 } && status_code > { 500 };
|
2023-08-24 21:32:12 +02:00
|
|
|
//~^ ERROR: boolean expression will never evaluate to 'true'
|
|
|
|
//~| NOTE: since `{ 400 }` < `{ 500 }`, the expression evaluates to false for any valu
|
2023-08-11 14:05:13 +02:00
|
|
|
status_code < STATUS_BAD_REQUEST && status_code > STATUS_SERVER_ERROR;
|
2023-08-24 21:32:12 +02:00
|
|
|
//~^ ERROR: boolean expression will never evaluate to 'true'
|
|
|
|
//~| NOTE: since `STATUS_BAD_REQUEST` < `STATUS_SERVER_ERROR`, the expression evaluate
|
2023-08-11 14:05:13 +02:00
|
|
|
status_code <= u16::MIN + 1 && status_code > STATUS_SERVER_ERROR;
|
2023-08-24 21:32:12 +02:00
|
|
|
//~^ ERROR: boolean expression will never evaluate to 'true'
|
|
|
|
//~| NOTE: since `u16::MIN + 1` < `STATUS_SERVER_ERROR`, the expression evaluates to f
|
2023-08-11 14:05:13 +02:00
|
|
|
status_code < STATUS_SERVER_ERROR && status_code > STATUS_SERVER_ERROR;
|
2023-08-24 21:32:12 +02:00
|
|
|
//~^ ERROR: boolean expression will never evaluate to 'true'
|
|
|
|
//~| NOTE: `status_code` cannot simultaneously be greater than and less than `STATUS_S
|
2023-08-11 14:05:13 +02:00
|
|
|
|
|
|
|
// Comparing two different types, via the `impl PartialOrd<u16> for Status`
|
|
|
|
status < { 400 } && status > { 500 };
|
2023-08-24 21:32:12 +02:00
|
|
|
//~^ ERROR: boolean expression will never evaluate to 'true'
|
|
|
|
//~| NOTE: since `{ 400 }` < `{ 500 }`, the expression evaluates to false for any valu
|
2023-08-11 14:05:13 +02:00
|
|
|
status < STATUS_BAD_REQUEST && status > STATUS_SERVER_ERROR;
|
2023-08-24 21:32:12 +02:00
|
|
|
//~^ ERROR: boolean expression will never evaluate to 'true'
|
|
|
|
//~| NOTE: since `STATUS_BAD_REQUEST` < `STATUS_SERVER_ERROR`, the expression evaluate
|
2023-08-11 14:05:13 +02:00
|
|
|
status <= u16::MIN + 1 && status > STATUS_SERVER_ERROR;
|
2023-08-24 21:32:12 +02:00
|
|
|
//~^ ERROR: boolean expression will never evaluate to 'true'
|
|
|
|
//~| NOTE: since `u16::MIN + 1` < `STATUS_SERVER_ERROR`, the expression evaluates to f
|
2023-08-11 14:05:13 +02:00
|
|
|
status < STATUS_SERVER_ERROR && status > STATUS_SERVER_ERROR;
|
2023-08-24 21:32:12 +02:00
|
|
|
//~^ ERROR: boolean expression will never evaluate to 'true'
|
|
|
|
//~| NOTE: `status` cannot simultaneously be greater than and less than `STATUS_SERVER
|
2023-08-11 14:05:13 +02:00
|
|
|
|
|
|
|
// Yoda conditions
|
2023-08-24 21:32:12 +02:00
|
|
|
// Correct
|
|
|
|
500 <= status_code && 600 > status_code;
|
|
|
|
// Correct
|
|
|
|
500 <= status_code && status_code <= 600;
|
|
|
|
// Incorrect
|
|
|
|
500 >= status_code && 600 < status_code;
|
|
|
|
//~^ ERROR: boolean expression will never evaluate to 'true'
|
|
|
|
//~| NOTE: since `500` < `600`, the expression evaluates to false for any value of `st
|
|
|
|
// Incorrect
|
|
|
|
500 >= status_code && status_code > 600;
|
|
|
|
//~^ ERROR: boolean expression will never evaluate to 'true'
|
|
|
|
//~| NOTE: since `500` < `600`, the expression evaluates to false for any value of `st
|
2023-08-11 14:05:13 +02:00
|
|
|
|
|
|
|
// Yoda conditions, comparing two different types
|
2023-08-24 21:32:12 +02:00
|
|
|
// Correct
|
|
|
|
500 <= status && 600 > status;
|
|
|
|
// Correct
|
|
|
|
500 <= status && status <= 600;
|
|
|
|
// Incorrect
|
|
|
|
500 >= status && 600 < status;
|
|
|
|
//~^ ERROR: boolean expression will never evaluate to 'true'
|
|
|
|
//~| NOTE: since `500` < `600`, the expression evaluates to false for any value of `st
|
|
|
|
// Incorrect
|
|
|
|
500 >= status && status > 600;
|
|
|
|
//~^ ERROR: boolean expression will never evaluate to 'true'
|
|
|
|
//~| NOTE: since `500` < `600`, the expression evaluates to false for any value of `st
|
2023-08-11 14:05:13 +02:00
|
|
|
|
|
|
|
// Expressions where one of the sides has no effect
|
|
|
|
status_code < 200 && status_code <= 299;
|
2023-08-24 21:32:12 +02:00
|
|
|
//~^ ERROR: right-hand side of `&&` operator has no effect
|
2023-08-11 14:05:13 +02:00
|
|
|
status_code > 200 && status_code >= 299;
|
2023-08-24 21:32:12 +02:00
|
|
|
//~^ ERROR: left-hand side of `&&` operator has no effect
|
|
|
|
|
|
|
|
// Useless left
|
|
|
|
status_code >= 500 && status_code > 500;
|
|
|
|
//~^ ERROR: left-hand side of `&&` operator has no effect
|
|
|
|
// Useless right
|
|
|
|
status_code > 500 && status_code >= 500;
|
|
|
|
//~^ ERROR: right-hand side of `&&` operator has no effect
|
|
|
|
// Useless left
|
|
|
|
status_code <= 500 && status_code < 500;
|
|
|
|
//~^ ERROR: left-hand side of `&&` operator has no effect
|
|
|
|
// Useless right
|
|
|
|
status_code < 500 && status_code <= 500;
|
|
|
|
//~^ ERROR: right-hand side of `&&` operator has no effect
|
2023-08-11 14:05:13 +02:00
|
|
|
|
|
|
|
// Other types
|
|
|
|
let name = "Steve";
|
|
|
|
name < "Jennifer" && name > "Shannon";
|
2023-08-24 21:32:12 +02:00
|
|
|
//~^ ERROR: boolean expression will never evaluate to 'true'
|
|
|
|
//~| NOTE: since `"Jennifer"` < `"Shannon"`, the expression evaluates to false for any
|
2023-08-11 14:05:13 +02:00
|
|
|
|
|
|
|
let numbers = [1, 2];
|
|
|
|
numbers < [3, 4] && numbers > [5, 6];
|
2023-08-24 21:32:12 +02:00
|
|
|
//~^ ERROR: boolean expression will never evaluate to 'true'
|
|
|
|
//~| NOTE: since `[3, 4]` < `[5, 6]`, the expression evaluates to false for any value
|
2023-08-11 14:05:13 +02:00
|
|
|
|
|
|
|
let letter = 'a';
|
|
|
|
letter < 'b' && letter > 'c';
|
2023-08-24 21:32:12 +02:00
|
|
|
//~^ ERROR: boolean expression will never evaluate to 'true'
|
|
|
|
//~| NOTE: since `'b'` < `'c'`, the expression evaluates to false for any value of `le
|
2023-08-11 14:05:13 +02:00
|
|
|
|
|
|
|
let area = 42.0;
|
|
|
|
area < std::f32::consts::E && area > std::f32::consts::PI;
|
2023-08-24 21:32:12 +02:00
|
|
|
//~^ ERROR: boolean expression will never evaluate to 'true'
|
|
|
|
//~| NOTE: since `std::f32::consts::E` < `std::f32::consts::PI`, the expression evalua
|
2023-08-11 14:05:13 +02:00
|
|
|
}
|