From 36bc725eaa8d0decb4ad880ddb42094a967feb20 Mon Sep 17 00:00:00 2001 From: Philipp Thiel <153559399+Philipp-Thiel@users.noreply.github.com> Date: Wed, 28 Feb 2024 19:19:36 +0100 Subject: [PATCH] [`flake8-bugbear`] Avoid adding default initializers to stubs (`B006`) (#10152) ## Summary Adapts the fix for rule B006 to no longer modify the body of function stubs, while retaining the change in method signature. ## Test Plan The existing tests for B006 were adapted to reflect this change in behavior. ## Relevant issue https://github.com/astral-sh/ruff/issues/10083 --- .../rules/mutable_argument_default.rs | 24 ++- ...flake8_bugbear__tests__B006_B006_1.py.snap | 5 - ...flake8_bugbear__tests__B006_B006_2.py.snap | 7 +- ...flake8_bugbear__tests__B006_B006_3.py.snap | 9 +- ...ke8_bugbear__tests__B006_B006_B008.py.snap | 182 ++++++------------ ...extend_immutable_calls_arg_annotation.snap | 6 +- 6 files changed, 87 insertions(+), 146 deletions(-) diff --git a/crates/ruff_linter/src/rules/flake8_bugbear/rules/mutable_argument_default.rs b/crates/ruff_linter/src/rules/flake8_bugbear/rules/mutable_argument_default.rs index 2f3c8281bae7d1..db495500f4dca3 100644 --- a/crates/ruff_linter/src/rules/flake8_bugbear/rules/mutable_argument_default.rs +++ b/crates/ruff_linter/src/rules/flake8_bugbear/rules/mutable_argument_default.rs @@ -2,7 +2,7 @@ use ast::call_path::{from_qualified_name, CallPath}; use ruff_diagnostics::{Diagnostic, Edit, Fix, FixAvailability, Violation}; use ruff_macros::{derive_message_formats, violation}; use ruff_python_ast::helpers::is_docstring_stmt; -use ruff_python_ast::{self as ast, Expr, Parameter, ParameterWithDefault}; +use ruff_python_ast::{self as ast, Expr, Parameter, ParameterWithDefault, Stmt}; use ruff_python_codegen::{Generator, Stylist}; use ruff_python_index::Indexer; use ruff_python_semantic::analyze::typing::{is_immutable_annotation, is_mutable_expr}; @@ -152,6 +152,11 @@ fn move_initialization( // Set the default argument value to `None`. let default_edit = Edit::range_replacement("None".to_string(), default.range()); + // If the function is a stub, this is the only necessary edit. + if is_stub(function_def) { + return Some(Fix::unsafe_edit(default_edit)); + } + // Add an `if`, to set the argument to its original value if still `None`. let mut content = String::new(); content.push_str(&format!("if {} is None:", parameter.name.as_str())); @@ -204,3 +209,20 @@ fn move_initialization( let initialization_edit = Edit::insertion(content, pos); Some(Fix::unsafe_edits(default_edit, [initialization_edit])) } + +/// Returns `true` if a function has an empty body, and is therefore a stub. +/// +/// A function body is considered to be empty if it contains only `pass` statements, `...` literals, +/// and docstrings. +fn is_stub(function_def: &ast::StmtFunctionDef) -> bool { + function_def.body.iter().all(|stmt| match stmt { + Stmt::Pass(_) => true, + Stmt::Expr(ast::StmtExpr { value, range: _ }) => { + matches!( + value.as_ref(), + Expr::StringLiteral(_) | Expr::EllipsisLiteral(_) + ) + } + _ => false, + }) +} diff --git a/crates/ruff_linter/src/rules/flake8_bugbear/snapshots/ruff_linter__rules__flake8_bugbear__tests__B006_B006_1.py.snap b/crates/ruff_linter/src/rules/flake8_bugbear/snapshots/ruff_linter__rules__flake8_bugbear__tests__B006_B006_1.py.snap index 4720535da33363..1456ef463102a8 100644 --- a/crates/ruff_linter/src/rules/flake8_bugbear/snapshots/ruff_linter__rules__flake8_bugbear__tests__B006_B006_1.py.snap +++ b/crates/ruff_linter/src/rules/flake8_bugbear/snapshots/ruff_linter__rules__flake8_bugbear__tests__B006_B006_1.py.snap @@ -19,8 +19,3 @@ B006_1.py:3:22: B006 [*] Do not use mutable data structures for argument default 3 |+def foobar(foor, bar=None): 4 4 | """ 5 5 | """ - 6 |+ - 7 |+ if bar is None: - 8 |+ bar = {} - - diff --git a/crates/ruff_linter/src/rules/flake8_bugbear/snapshots/ruff_linter__rules__flake8_bugbear__tests__B006_B006_2.py.snap b/crates/ruff_linter/src/rules/flake8_bugbear/snapshots/ruff_linter__rules__flake8_bugbear__tests__B006_B006_2.py.snap index 522733a5cd7318..c050ac73b7212d 100644 --- a/crates/ruff_linter/src/rules/flake8_bugbear/snapshots/ruff_linter__rules__flake8_bugbear__tests__B006_B006_2.py.snap +++ b/crates/ruff_linter/src/rules/flake8_bugbear/snapshots/ruff_linter__rules__flake8_bugbear__tests__B006_B006_2.py.snap @@ -19,9 +19,4 @@ B006_2.py:4:22: B006 [*] Do not use mutable data structures for argument default 4 |-def foobar(foor, bar={}): 4 |+def foobar(foor, bar=None): 5 5 | """ -6 |- """ - 6 |+ """ - 7 |+ if bar is None: - 8 |+ bar = {} - - +6 6 | """ diff --git a/crates/ruff_linter/src/rules/flake8_bugbear/snapshots/ruff_linter__rules__flake8_bugbear__tests__B006_B006_3.py.snap b/crates/ruff_linter/src/rules/flake8_bugbear/snapshots/ruff_linter__rules__flake8_bugbear__tests__B006_B006_3.py.snap index cd7da7f1fac6a5..583e51b5a7cd73 100644 --- a/crates/ruff_linter/src/rules/flake8_bugbear/snapshots/ruff_linter__rules__flake8_bugbear__tests__B006_B006_3.py.snap +++ b/crates/ruff_linter/src/rules/flake8_bugbear/snapshots/ruff_linter__rules__flake8_bugbear__tests__B006_B006_3.py.snap @@ -16,10 +16,5 @@ B006_3.py:4:22: B006 [*] Do not use mutable data structures for argument default 3 3 | 4 |-def foobar(foor, bar={}): 4 |+def foobar(foor, bar=None): - 5 |+ """ -5 6 | """ -6 |- """ - 7 |+ if bar is None: - 8 |+ bar = {} - - +5 5 | """ +6 6 | """ diff --git a/crates/ruff_linter/src/rules/flake8_bugbear/snapshots/ruff_linter__rules__flake8_bugbear__tests__B006_B006_B008.py.snap b/crates/ruff_linter/src/rules/flake8_bugbear/snapshots/ruff_linter__rules__flake8_bugbear__tests__B006_B006_B008.py.snap index eae2c6f6477792..5627954be4cd29 100644 --- a/crates/ruff_linter/src/rules/flake8_bugbear/snapshots/ruff_linter__rules__flake8_bugbear__tests__B006_B006_B008.py.snap +++ b/crates/ruff_linter/src/rules/flake8_bugbear/snapshots/ruff_linter__rules__flake8_bugbear__tests__B006_B006_B008.py.snap @@ -15,11 +15,9 @@ B006_B008.py:63:25: B006 [*] Do not use mutable data structures for argument def 62 62 | 63 |-def this_is_wrong(value=[1, 2, 3]): 63 |+def this_is_wrong(value=None): - 64 |+ if value is None: - 65 |+ value = [1, 2, 3] -64 66 | ... -65 67 | -66 68 | +64 64 | ... +65 65 | +66 66 | B006_B008.py:67:30: B006 [*] Do not use mutable data structures for argument defaults | @@ -35,11 +33,9 @@ B006_B008.py:67:30: B006 [*] Do not use mutable data structures for argument def 66 66 | 67 |-def this_is_also_wrong(value={}): 67 |+def this_is_also_wrong(value=None): - 68 |+ if value is None: - 69 |+ value = {} -68 70 | ... -69 71 | -70 72 | +68 68 | ... +69 69 | +70 70 | B006_B008.py:73:52: B006 [*] Do not use mutable data structures for argument defaults | @@ -57,11 +53,9 @@ B006_B008.py:73:52: B006 [*] Do not use mutable data structures for argument def 72 72 | @staticmethod 73 |- def this_is_also_wrong_and_more_indented(value={}): 73 |+ def this_is_also_wrong_and_more_indented(value=None): - 74 |+ if value is None: - 75 |+ value = {} -74 76 | pass -75 77 | -76 78 | +74 74 | pass +75 75 | +76 76 | B006_B008.py:77:31: B006 [*] Do not use mutable data structures for argument defaults | @@ -82,11 +76,9 @@ B006_B008.py:77:31: B006 [*] Do not use mutable data structures for argument def 78 |- 79 |-}): 77 |+def multiline_arg_wrong(value=None): - 78 |+ if value is None: - 79 |+ value = {} -80 80 | ... -81 81 | -82 82 | def single_line_func_wrong(value = {}): ... +80 78 | ... +81 79 | +82 80 | def single_line_func_wrong(value = {}): ... B006_B008.py:82:36: B006 Do not use mutable data structures for argument defaults | @@ -111,11 +103,9 @@ B006_B008.py:85:20: B006 [*] Do not use mutable data structures for argument def 84 84 | 85 |-def and_this(value=set()): 85 |+def and_this(value=None): - 86 |+ if value is None: - 87 |+ value = set() -86 88 | ... -87 89 | -88 90 | +86 86 | ... +87 87 | +88 88 | B006_B008.py:89:20: B006 [*] Do not use mutable data structures for argument defaults | @@ -131,11 +121,9 @@ B006_B008.py:89:20: B006 [*] Do not use mutable data structures for argument def 88 88 | 89 |-def this_too(value=collections.OrderedDict()): 89 |+def this_too(value=None): - 90 |+ if value is None: - 91 |+ value = collections.OrderedDict() -90 92 | ... -91 93 | -92 94 | +90 90 | ... +91 91 | +92 92 | B006_B008.py:93:32: B006 [*] Do not use mutable data structures for argument defaults | @@ -151,11 +139,9 @@ B006_B008.py:93:32: B006 [*] Do not use mutable data structures for argument def 92 92 | 93 |-async def async_this_too(value=collections.defaultdict()): 93 |+async def async_this_too(value=None): - 94 |+ if value is None: - 95 |+ value = collections.defaultdict() -94 96 | ... -95 97 | -96 98 | +94 94 | ... +95 95 | +96 96 | B006_B008.py:97:26: B006 [*] Do not use mutable data structures for argument defaults | @@ -166,16 +152,14 @@ B006_B008.py:97:26: B006 [*] Do not use mutable data structures for argument def = help: Replace with `None`; initialize within function ℹ Unsafe fix -94 94 | ... -95 95 | -96 96 | -97 |-def dont_forget_me(value=collections.deque()): - 97 |+def dont_forget_me(value=None): - 98 |+ if value is None: - 99 |+ value = collections.deque() -98 100 | ... -99 101 | -100 102 | +94 94 | ... +95 95 | +96 96 | +97 |-def dont_forget_me(value=collections.deque()): + 97 |+def dont_forget_me(value=None): +98 98 | ... +99 99 | +100 100 | B006_B008.py:102:46: B006 [*] Do not use mutable data structures for argument defaults | @@ -192,11 +176,9 @@ B006_B008.py:102:46: B006 [*] Do not use mutable data structures for argument de 101 101 | # N.B. we're also flagging the function call in the comprehension 102 |-def list_comprehension_also_not_okay(default=[i**2 for i in range(3)]): 102 |+def list_comprehension_also_not_okay(default=None): - 103 |+ if default is None: - 104 |+ default = [i ** 2 for i in range(3)] -103 105 | pass -104 106 | -105 107 | +103 103 | pass +104 104 | +105 105 | B006_B008.py:106:46: B006 [*] Do not use mutable data structures for argument defaults | @@ -212,11 +194,9 @@ B006_B008.py:106:46: B006 [*] Do not use mutable data structures for argument de 105 105 | 106 |-def dict_comprehension_also_not_okay(default={i: i**2 for i in range(3)}): 106 |+def dict_comprehension_also_not_okay(default=None): - 107 |+ if default is None: - 108 |+ default = {i: i ** 2 for i in range(3)} -107 109 | pass -108 110 | -109 111 | +107 107 | pass +108 108 | +109 109 | B006_B008.py:110:45: B006 [*] Do not use mutable data structures for argument defaults | @@ -232,11 +212,9 @@ B006_B008.py:110:45: B006 [*] Do not use mutable data structures for argument de 109 109 | 110 |-def set_comprehension_also_not_okay(default={i**2 for i in range(3)}): 110 |+def set_comprehension_also_not_okay(default=None): - 111 |+ if default is None: - 112 |+ default = {i ** 2 for i in range(3)} -111 113 | pass -112 114 | -113 115 | +111 111 | pass +112 112 | +113 113 | B006_B008.py:114:33: B006 [*] Do not use mutable data structures for argument defaults | @@ -252,11 +230,9 @@ B006_B008.py:114:33: B006 [*] Do not use mutable data structures for argument de 113 113 | 114 |-def kwonlyargs_mutable(*, value=[]): 114 |+def kwonlyargs_mutable(*, value=None): - 115 |+ if value is None: - 116 |+ value = [] -115 117 | ... -116 118 | -117 119 | +115 115 | ... +116 116 | +117 117 | B006_B008.py:239:20: B006 [*] Do not use mutable data structures for argument defaults | @@ -274,11 +250,9 @@ B006_B008.py:239:20: B006 [*] Do not use mutable data structures for argument de 238 238 | # We should handle arbitrary nesting of these B008. 239 |-def nested_combo(a=[float(3), dt.datetime.now()]): 239 |+def nested_combo(a=None): - 240 |+ if a is None: - 241 |+ a = [float(3), dt.datetime.now()] -240 242 | pass -241 243 | -242 244 | +240 240 | pass +241 241 | +242 242 | B006_B008.py:276:27: B006 [*] Do not use mutable data structures for argument defaults | @@ -299,12 +273,6 @@ B006_B008.py:276:27: B006 [*] Do not use mutable data structures for argument de 277 277 | b: Optional[Dict[int, int]] = {}, 278 278 | c: Annotated[Union[Set[str], abc.Sized], "annotation"] = set(), 279 279 | d: typing_extensions.Annotated[Union[Set[str], abc.Sized], "annotation"] = set(), -280 280 | ): - 281 |+ if a is None: - 282 |+ a = [] -281 283 | pass -282 284 | -283 285 | B006_B008.py:277:35: B006 [*] Do not use mutable data structures for argument defaults | @@ -326,11 +294,6 @@ B006_B008.py:277:35: B006 [*] Do not use mutable data structures for argument de 278 278 | c: Annotated[Union[Set[str], abc.Sized], "annotation"] = set(), 279 279 | d: typing_extensions.Annotated[Union[Set[str], abc.Sized], "annotation"] = set(), 280 280 | ): - 281 |+ if b is None: - 282 |+ b = {} -281 283 | pass -282 284 | -283 285 | B006_B008.py:278:62: B006 [*] Do not use mutable data structures for argument defaults | @@ -351,11 +314,7 @@ B006_B008.py:278:62: B006 [*] Do not use mutable data structures for argument de 278 |+ c: Annotated[Union[Set[str], abc.Sized], "annotation"] = None, 279 279 | d: typing_extensions.Annotated[Union[Set[str], abc.Sized], "annotation"] = set(), 280 280 | ): - 281 |+ if c is None: - 282 |+ c = set() -281 283 | pass -282 284 | -283 285 | +281 281 | pass B006_B008.py:279:80: B006 [*] Do not use mutable data structures for argument defaults | @@ -375,11 +334,8 @@ B006_B008.py:279:80: B006 [*] Do not use mutable data structures for argument de 279 |- d: typing_extensions.Annotated[Union[Set[str], abc.Sized], "annotation"] = set(), 279 |+ d: typing_extensions.Annotated[Union[Set[str], abc.Sized], "annotation"] = None, 280 280 | ): - 281 |+ if d is None: - 282 |+ d = set() -281 283 | pass -282 284 | -283 285 | +281 281 | pass +282 282 | B006_B008.py:284:52: B006 [*] Do not use mutable data structures for argument defaults | @@ -396,11 +352,8 @@ B006_B008.py:284:52: B006 [*] Do not use mutable data structures for argument de 284 |-def single_line_func_wrong(value: dict[str, str] = {}): 284 |+def single_line_func_wrong(value: dict[str, str] = None): 285 285 | """Docstring""" - 286 |+ if value is None: - 287 |+ value = {} -286 288 | -287 289 | -288 290 | def single_line_func_wrong(value: dict[str, str] = {}): +286 286 | +287 287 | B006_B008.py:288:52: B006 [*] Do not use mutable data structures for argument defaults | @@ -418,11 +371,8 @@ B006_B008.py:288:52: B006 [*] Do not use mutable data structures for argument de 288 |-def single_line_func_wrong(value: dict[str, str] = {}): 288 |+def single_line_func_wrong(value: dict[str, str] = None): 289 289 | """Docstring""" - 290 |+ if value is None: - 291 |+ value = {} -290 292 | ... -291 293 | -292 294 | +290 290 | ... +291 291 | B006_B008.py:293:52: B006 [*] Do not use mutable data structures for argument defaults | @@ -438,11 +388,9 @@ B006_B008.py:293:52: B006 [*] Do not use mutable data structures for argument de 292 292 | 293 |-def single_line_func_wrong(value: dict[str, str] = {}): 293 |+def single_line_func_wrong(value: dict[str, str] = None): - 294 |+ if value is None: - 295 |+ value = {} -294 296 | """Docstring"""; ... -295 297 | -296 298 | +294 294 | """Docstring"""; ... +295 295 | +296 296 | B006_B008.py:297:52: B006 [*] Do not use mutable data structures for argument defaults | @@ -459,11 +407,9 @@ B006_B008.py:297:52: B006 [*] Do not use mutable data structures for argument de 296 296 | 297 |-def single_line_func_wrong(value: dict[str, str] = {}): 297 |+def single_line_func_wrong(value: dict[str, str] = None): - 298 |+ if value is None: - 299 |+ value = {} -298 300 | """Docstring"""; \ -299 301 | ... -300 302 | +298 298 | """Docstring"""; \ +299 299 | ... +300 300 | B006_B008.py:302:52: B006 [*] Do not use mutable data structures for argument defaults | @@ -485,11 +431,8 @@ B006_B008.py:302:52: B006 [*] Do not use mutable data structures for argument de 304 |-}): 302 |+def single_line_func_wrong(value: dict[str, str] = None): 305 303 | """Docstring""" - 304 |+ if value is None: - 305 |+ value = {} -306 306 | -307 307 | -308 308 | def single_line_func_wrong(value: dict[str, str] = {}) \ +306 304 | +307 305 | B006_B008.py:308:52: B006 Do not use mutable data structures for argument defaults | @@ -513,10 +456,5 @@ B006_B008.py:313:52: B006 [*] Do not use mutable data structures for argument de 311 311 | 312 312 | 313 |-def single_line_func_wrong(value: dict[str, str] = {}): -314 |- """Docstring without newline""" 313 |+def single_line_func_wrong(value: dict[str, str] = None): - 314 |+ """Docstring without newline""" - 315 |+ if value is None: - 316 |+ value = {} - - +314 314 | """Docstring without newline""" diff --git a/crates/ruff_linter/src/rules/flake8_bugbear/snapshots/ruff_linter__rules__flake8_bugbear__tests__extend_immutable_calls_arg_annotation.snap b/crates/ruff_linter/src/rules/flake8_bugbear/snapshots/ruff_linter__rules__flake8_bugbear__tests__extend_immutable_calls_arg_annotation.snap index b7b697526c8a78..ff069ea9304453 100644 --- a/crates/ruff_linter/src/rules/flake8_bugbear/snapshots/ruff_linter__rules__flake8_bugbear__tests__extend_immutable_calls_arg_annotation.snap +++ b/crates/ruff_linter/src/rules/flake8_bugbear/snapshots/ruff_linter__rules__flake8_bugbear__tests__extend_immutable_calls_arg_annotation.snap @@ -15,8 +15,4 @@ B006_extended.py:17:55: B006 [*] Do not use mutable data structures for argument 16 16 | 17 |-def error_due_to_missing_import(foo: ImmutableTypeA = []): 17 |+def error_due_to_missing_import(foo: ImmutableTypeA = None): - 18 |+ if foo is None: - 19 |+ foo = [] -18 20 | ... - - +18 18 | ...