-
Notifications
You must be signed in to change notification settings - Fork 1.2k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
## Summary Implement [`no-isinstance-type-none`](https://github.com/dosisod/refurb/blob/master/refurb/checks/builtin/no_isinstance_type_none.py) as `isinstance-type-none` (`FURB168`). Auto-fixes calls to `isinstance` to check if an object is `None` to a `None` identity check. For example, ```python isinstance(foo, type(None)) ``` becomes ```python foo is None ``` Related to #1348. ## Test Plan `cargo test` --------- Co-authored-by: Charlie Marsh <[email protected]>
- Loading branch information
1 parent
a151e50
commit 10a50bf
Showing
9 changed files
with
368 additions
and
0 deletions.
There are no files selected for viewing
51 changes: 51 additions & 0 deletions
51
crates/ruff_linter/resources/test/fixtures/refurb/FURB168.py
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,51 @@ | ||
foo: object | ||
|
||
# Errors. | ||
|
||
if isinstance(foo, type(None)): | ||
pass | ||
|
||
if isinstance(foo, (type(None))): | ||
pass | ||
|
||
if isinstance(foo, (type(None), type(None), type(None))): | ||
pass | ||
|
||
if isinstance(foo, None | None): | ||
pass | ||
|
||
if isinstance(foo, (None | None)): | ||
pass | ||
|
||
if isinstance(foo, None | type(None)): | ||
pass | ||
|
||
if isinstance(foo, (None | type(None))): | ||
pass | ||
|
||
# A bit contrived, but is both technically valid and equivalent to the above. | ||
if isinstance(foo, (type(None) | ((((type(None))))) | ((None | type(None))))): | ||
pass | ||
|
||
|
||
# Okay. | ||
|
||
if isinstance(foo, int): | ||
pass | ||
|
||
if isinstance(foo, (int)): | ||
pass | ||
|
||
if isinstance(foo, (int, str)): | ||
pass | ||
|
||
if isinstance(foo, (int, type(None), str)): | ||
pass | ||
|
||
# This is a TypeError, which the rule ignores. | ||
if isinstance(foo, None): | ||
pass | ||
|
||
# This is also a TypeError, which the rule ignores. | ||
if isinstance(foo, (None,)): | ||
pass |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
144 changes: 144 additions & 0 deletions
144
crates/ruff_linter/src/rules/refurb/rules/isinstance_type_none.rs
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,144 @@ | ||
use ruff_diagnostics::{Diagnostic, Edit, Fix, FixAvailability, Violation}; | ||
use ruff_python_ast::{self as ast, Expr, Operator}; | ||
|
||
use ruff_macros::{derive_message_formats, violation}; | ||
use ruff_python_codegen::Generator; | ||
use ruff_text_size::{Ranged, TextRange}; | ||
|
||
use crate::checkers::ast::Checker; | ||
use crate::fix::edits::pad; | ||
|
||
/// ## What it does | ||
/// Checks for uses of `isinstance` that check if an object is of type `None`. | ||
/// | ||
/// ## Why is this bad? | ||
/// There is only ever one instance of `None`, so it is more efficient and | ||
/// readable to use the `is` operator to check if an object is `None`. | ||
/// | ||
/// ## Example | ||
/// ```python | ||
/// isinstance(obj, type(None)) | ||
/// ``` | ||
/// | ||
/// Use instead: | ||
/// ```python | ||
/// obj is None | ||
/// ``` | ||
/// | ||
/// ## References | ||
/// - [Python documentation: `isinstance`](https://docs.python.org/3/library/functions.html#isinstance) | ||
/// - [Python documentation: `None`](https://docs.python.org/3/library/constants.html#None) | ||
/// - [Python documentation: `type`](https://docs.python.org/3/library/functions.html#type) | ||
/// - [Python documentation: Identity comparisons](https://docs.python.org/3/reference/expressions.html#is-not) | ||
#[violation] | ||
pub struct IsinstanceTypeNone; | ||
|
||
impl Violation for IsinstanceTypeNone { | ||
const FIX_AVAILABILITY: FixAvailability = FixAvailability::Sometimes; | ||
|
||
#[derive_message_formats] | ||
fn message(&self) -> String { | ||
format!("Prefer `is` operator over `isinstance` to check if an object is `None`") | ||
} | ||
|
||
fn fix_title(&self) -> Option<String> { | ||
Some("Replace with `is` operator".to_string()) | ||
} | ||
} | ||
|
||
/// FURB168 | ||
pub(crate) fn isinstance_type_none(checker: &mut Checker, call: &ast::ExprCall) { | ||
let Expr::Name(ast::ExprName { id, .. }) = call.func.as_ref() else { | ||
return; | ||
}; | ||
if id.as_str() != "isinstance" { | ||
return; | ||
} | ||
if !checker.semantic().is_builtin(id) { | ||
return; | ||
} | ||
let Some(types) = call.arguments.find_positional(1) else { | ||
return; | ||
}; | ||
|
||
if is_none(types) { | ||
let Some(Expr::Name(ast::ExprName { | ||
id: object_name, .. | ||
})) = call.arguments.find_positional(0) | ||
else { | ||
return; | ||
}; | ||
let mut diagnostic = Diagnostic::new(IsinstanceTypeNone, call.range()); | ||
let replacement = generate_replacement(object_name, checker.generator()); | ||
diagnostic.set_fix(Fix::safe_edit(Edit::range_replacement( | ||
pad(replacement, call.range(), checker.locator()), | ||
call.range(), | ||
))); | ||
checker.diagnostics.push(diagnostic); | ||
} | ||
} | ||
|
||
/// Returns `true` if the given expression is equivalent to checking if the | ||
/// object type is `None` when used with the `isinstance` builtin. | ||
fn is_none(expr: &Expr) -> bool { | ||
fn inner(expr: &Expr, in_union_context: bool) -> bool { | ||
match expr { | ||
// Ex) `None` | ||
// Note: `isinstance` only accepts `None` as a type when used with | ||
// the union operator, so we need to check if we're in a union. | ||
Expr::Constant(ast::ExprConstant { value, .. }) if in_union_context => value.is_none(), | ||
|
||
// Ex) `type(None)` | ||
Expr::Call(ast::ExprCall { | ||
func, arguments, .. | ||
}) if arguments.len() == 1 => { | ||
if let Expr::Name(ast::ExprName { id, .. }) = func.as_ref() { | ||
if id.as_str() == "type" { | ||
if let Some(Expr::Constant(ast::ExprConstant { value, .. })) = | ||
arguments.args.get(0) | ||
{ | ||
return value.is_none(); | ||
} | ||
} | ||
} | ||
false | ||
} | ||
|
||
// Ex) `(type(None),)` | ||
Expr::Tuple(ast::ExprTuple { elts, .. }) => elts.iter().all(|elt| inner(elt, false)), | ||
|
||
// Ex) `type(None) | type(None)` | ||
Expr::BinOp(ast::ExprBinOp { | ||
left, | ||
op: Operator::BitOr, | ||
right, | ||
.. | ||
}) => inner(left, true) && inner(right, true), | ||
|
||
// Otherwise, return false. | ||
_ => false, | ||
} | ||
} | ||
inner(expr, false) | ||
} | ||
|
||
/// Format a code snippet comparing `name` to `None` (e.g., `name is None`). | ||
fn generate_replacement(name: &str, generator: Generator) -> String { | ||
// Construct `name`. | ||
let var = ast::ExprName { | ||
id: name.to_string(), | ||
ctx: ast::ExprContext::Load, | ||
range: TextRange::default(), | ||
}; | ||
// Construct `name is None`. | ||
let compare = ast::ExprCompare { | ||
left: Box::new(var.into()), | ||
ops: vec![ast::CmpOp::Is], | ||
comparators: vec![ast::Expr::Constant(ast::ExprConstant { | ||
value: ast::Constant::None, | ||
range: TextRange::default(), | ||
})], | ||
range: TextRange::default(), | ||
}; | ||
generator.expr(&compare.into()) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.