diff --git a/crates/ruff_linter/resources/test/fixtures/flake8_bugbear/B909.py b/crates/ruff_linter/resources/test/fixtures/flake8_bugbear/B909.py index 4381aa0d5787a..7a1146d84fac0 100644 --- a/crates/ruff_linter/resources/test/fixtures/flake8_bugbear/B909.py +++ b/crates/ruff_linter/resources/test/fixtures/flake8_bugbear/B909.py @@ -134,3 +134,20 @@ def __init__(self, ls): pass else: foo.remove(1) + +# should error +for elem in some_list: + if some_list.pop() == 2: + pass + +# should not error +for elem in some_list: + if some_list.pop() == 2: + break + +# should error +for elem in some_list: + if some_list.pop() == 2: + pass + else: + break diff --git a/crates/ruff_linter/src/rules/flake8_bugbear/rules/loop_iterator_mutation.rs b/crates/ruff_linter/src/rules/flake8_bugbear/rules/loop_iterator_mutation.rs index c2cf1de19bf5c..f8735a9498561 100644 --- a/crates/ruff_linter/src/rules/flake8_bugbear/rules/loop_iterator_mutation.rs +++ b/crates/ruff_linter/src/rules/flake8_bugbear/rules/loop_iterator_mutation.rs @@ -7,8 +7,8 @@ use ruff_python_ast::comparable::ComparableExpr; use ruff_python_ast::name::UnqualifiedName; use ruff_python_ast::{ visitor::{self, Visitor}, - ElifElseClause, Expr, ExprAttribute, ExprCall, ExprSubscript, Operator, Stmt, StmtAssign, - StmtAugAssign, StmtBreak, StmtDelete, StmtFor, StmtIf, + Expr, ExprAttribute, ExprCall, ExprSubscript, Stmt, StmtAssign, StmtAugAssign, StmtBreak, + StmtDelete, StmtFor, StmtIf, }; use ruff_text_size::TextRange; @@ -46,7 +46,7 @@ impl Violation for LoopIteratorMutation { let LoopIteratorMutation { name } = self; if let Some(name) = name.as_ref().and_then(SourceCodeSnippet::full_display) { - format!("Mutation to loop iterable `{}` during iteration", name) + format!("Mutation to loop iterable `{name}` during iteration") } else { format!("Mutation to loop iterable during iteration") } @@ -131,14 +131,11 @@ impl<'a> LoopMutationsVisitor<'a> { /// Register a mutation. fn add_mutation(&mut self, range: TextRange) { - self.mutations - .entry(self.branch) - .or_insert(Vec::new()) - .push(range); + self.mutations.entry(self.branch).or_default().push(range); } /// Handle, e.g., `del items[0]`. - fn handle_delete(&mut self, range: TextRange, targets: &'a Vec) { + fn handle_delete(&mut self, range: TextRange, targets: &'a [Expr]) { for target in targets { if let Expr::Subscript(ExprSubscript { range: _, @@ -155,7 +152,7 @@ impl<'a> LoopMutationsVisitor<'a> { } /// Handle, e.g., `items[0] = 1`. - fn handle_assign(&mut self, range: TextRange, targets: &'a Vec, _value: &Box) { + fn handle_assign(&mut self, range: TextRange, targets: &[Expr]) { for target in targets { if let Expr::Subscript(ExprSubscript { range: _, @@ -172,13 +169,7 @@ impl<'a> LoopMutationsVisitor<'a> { } /// Handle, e.g., `items += [1]`. - fn handle_aug_assign( - &mut self, - range: TextRange, - target: &Box, - _op: &Operator, - _value: &Box, - ) { + fn handle_aug_assign(&mut self, range: TextRange, target: &Expr) { if ComparableExpr::from(self.name) == ComparableExpr::from(target) { self.add_mutation(range); } @@ -200,20 +191,6 @@ impl<'a> LoopMutationsVisitor<'a> { } } } - - fn handle_branches(&mut self, body: &'a [Stmt], elif_else_clauses: &'a [ElifElseClause]) { - self.branch += 1; - self.branches.push(self.branch); - self.visit_body(body); - self.branches.pop(); - - for clause in elif_else_clauses { - self.branch += 1; - self.branches.push(self.branch); - self.visit_body(&clause.body); - self.branches.pop(); - } - } } /// `Visitor` to collect all used identifiers in a statement. @@ -227,38 +204,50 @@ impl<'a> Visitor<'a> for LoopMutationsVisitor<'a> { } // Ex) `items[0] = 1` - Stmt::Assign(StmtAssign { - range, - targets, - value, - }) => { - self.handle_assign(*range, targets, value); + Stmt::Assign(StmtAssign { range, targets, .. }) => { + self.handle_assign(*range, targets); visitor::walk_stmt(self, stmt); } // Ex) `items += [1]` - Stmt::AugAssign(StmtAugAssign { - range, - target, - op, - value, - }) => { - self.handle_aug_assign(*range, target, op, value); + Stmt::AugAssign(StmtAugAssign { range, target, .. }) => { + self.handle_aug_assign(*range, target); visitor::walk_stmt(self, stmt); } // Ex) `if True: items.append(1)` Stmt::If(StmtIf { + test, body, elif_else_clauses, .. - }) => self.handle_branches(body, elif_else_clauses), + }) => { + // Handle the `if` branch. + self.branch += 1; + self.branches.push(self.branch); + self.visit_expr(test); + self.visit_body(body); + self.branches.pop(); + + // Handle the `elif` and `else` branches. + for clause in elif_else_clauses { + self.branch += 1; + self.branches.push(self.branch); + if let Some(test) = &clause.test { + self.visit_expr(test); + } + self.visit_body(&clause.body); + self.branches.pop(); + } + } // On break, clear the mutations for the current branch. - Stmt::Break(StmtBreak { range: _ }) => match self.mutations.get_mut(&self.branch) { - Some(a) => a.clear(), - None => {} - }, + Stmt::Break(StmtBreak { range: _ }) => { + if let Some(mutations) = self.mutations.get_mut(&self.branch) { + mutations.clear(); + } + visitor::walk_stmt(self, stmt); + } // Avoid recursion for class and function definitions. Stmt::ClassDef(_) | Stmt::FunctionDef(_) => {} @@ -273,7 +262,7 @@ impl<'a> Visitor<'a> for LoopMutationsVisitor<'a> { fn visit_expr(&mut self, expr: &'a Expr) { // Ex) `items.append(1)` if let Expr::Call(ExprCall { func, .. }) = expr { - self.handle_call(&*func); + self.handle_call(func); } visitor::walk_expr(self, expr); diff --git a/crates/ruff_linter/src/rules/flake8_bugbear/snapshots/ruff_linter__rules__flake8_bugbear__tests__B909_B909.py.snap b/crates/ruff_linter/src/rules/flake8_bugbear/snapshots/ruff_linter__rules__flake8_bugbear__tests__B909_B909.py.snap index d6cf21e4f18a3..2d03e647fe77e 100644 --- a/crates/ruff_linter/src/rules/flake8_bugbear/snapshots/ruff_linter__rules__flake8_bugbear__tests__B909_B909.py.snap +++ b/crates/ruff_linter/src/rules/flake8_bugbear/snapshots/ruff_linter__rules__flake8_bugbear__tests__B909_B909.py.snap @@ -317,4 +317,25 @@ B909.py:136:9: B909 Mutation to loop iterable `foo` during iteration 135 | else: 136 | foo.remove(1) | ^^^^^^^^^^ B909 +137 | +138 | # should error + | + +B909.py:140:8: B909 Mutation to loop iterable `some_list` during iteration + | +138 | # should error +139 | for elem in some_list: +140 | if some_list.pop() == 2: + | ^^^^^^^^^^^^^ B909 +141 | pass + | + +B909.py:150:8: B909 Mutation to loop iterable `some_list` during iteration + | +148 | # should error +149 | for elem in some_list: +150 | if some_list.pop() == 2: + | ^^^^^^^^^^^^^ B909 +151 | pass +152 | else: |