-
Notifications
You must be signed in to change notification settings - Fork 1.2k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
F401 - update documentation and deprecate ignore_init_module_imports
#11436
Changes from all commits
e78fee6
c9ac09b
fb1ac97
7706561
36b4331
63a9676
f656ca4
1bfa7b1
4f0b79e
3202de8
97d1526
8d3ffbd
249d848
b6871c9
715da2e
21fe25e
26f80c1
2be489f
943ee25
6028415
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -24,6 +24,7 @@ enum UnusedImportContext { | |
Init { | ||
first_party: bool, | ||
dunder_all_count: usize, | ||
ignore_init_module_imports: bool, | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This field is required to make fix titles follow the "old" behavior in |
||
}, | ||
} | ||
|
||
|
@@ -46,12 +47,29 @@ enum UnusedImportContext { | |
/// from module import member as member | ||
plredmond marked this conversation as resolved.
Show resolved
Hide resolved
|
||
/// ``` | ||
/// | ||
/// Alternatively, you can use `__all__` to declare a symbol as part of the module's | ||
/// interface, as in: | ||
/// | ||
/// ```python | ||
/// # __init__.py | ||
/// import some_module | ||
/// | ||
/// __all__ = [ "some_module"] | ||
/// ``` | ||
/// | ||
/// ## Fix safety | ||
/// | ||
/// When `ignore_init_module_imports` is disabled, fixes can remove for unused imports in `__init__` files. | ||
/// These fixes are considered unsafe because they can change the public interface. | ||
/// Fixes to remove unused imports are safe, except in `__init__.py` files. | ||
/// | ||
/// Applying fixes to `__init__.py` files is currently in preview. The fix offered depends on the | ||
/// type of the unused import. Ruff will suggest a safe fix to export first-party imports with | ||
/// either a redundant alias or, if already present in the file, an `__all__` entry. If multiple | ||
/// `__all__` declarations are present, Ruff will not offer a fix. Ruff will suggest an unsafe fix | ||
/// to remove third-party and standard library imports -- the fix is unsafe because the module's | ||
/// interface changes. | ||
/// | ||
/// ## Example | ||
/// | ||
/// ```python | ||
/// import numpy as np # unused import | ||
/// | ||
|
@@ -61,12 +79,14 @@ enum UnusedImportContext { | |
/// ``` | ||
/// | ||
/// Use instead: | ||
/// | ||
/// ```python | ||
/// def area(radius): | ||
/// return 3.14 * radius**2 | ||
/// ``` | ||
/// | ||
/// To check the availability of a module, use `importlib.util.find_spec`: | ||
/// | ||
/// ```python | ||
/// from importlib.util import find_spec | ||
/// | ||
|
@@ -87,6 +107,8 @@ enum UnusedImportContext { | |
pub struct UnusedImport { | ||
/// Qualified name of the import | ||
name: String, | ||
/// Unqualified name of the import | ||
module: String, | ||
/// Name of the import binding | ||
binding: String, | ||
context: Option<UnusedImportContext>, | ||
|
@@ -117,6 +139,7 @@ impl Violation for UnusedImport { | |
fn fix_title(&self) -> Option<String> { | ||
let UnusedImport { | ||
name, | ||
module, | ||
binding, | ||
multiple, | ||
.. | ||
|
@@ -125,14 +148,14 @@ impl Violation for UnusedImport { | |
Some(UnusedImportContext::Init { | ||
first_party: true, | ||
dunder_all_count: 1, | ||
ignore_init_module_imports: true, | ||
}) => Some(format!("Add unused import `{binding}` to __all__")), | ||
|
||
Some(UnusedImportContext::Init { | ||
first_party: true, | ||
dunder_all_count: 0, | ||
}) => Some(format!( | ||
"Use an explicit re-export: `{binding} as {binding}`" | ||
)), | ||
ignore_init_module_imports: true, | ||
}) => Some(format!("Use an explicit re-export: `{module} as {module}`")), | ||
|
||
_ => Some(if *multiple { | ||
"Remove unused import".to_string() | ||
|
@@ -244,7 +267,8 @@ pub(crate) fn unused_import(checker: &Checker, scope: &Scope, diagnostics: &mut | |
} | ||
|
||
let in_init = checker.path().ends_with("__init__.py"); | ||
let fix_init = checker.settings.preview.is_enabled(); | ||
let fix_init = !checker.settings.ignore_init_module_imports; | ||
let preview_mode = checker.settings.preview.is_enabled(); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. These three lines look fishy because I'd originally replaced this let fix_init = !checker.settings.ignore_init_module_imports; with let fix_init = checker.settings.preview.is_enabled(); which was perhaps lazy. In this PR I'm restoring the old behavior, and so gave |
||
let dunder_all_exprs = find_dunder_all_exprs(checker.semantic()); | ||
|
||
// Generate a diagnostic for every import, but share fixes across all imports within the same | ||
|
@@ -275,6 +299,7 @@ pub(crate) fn unused_import(checker: &Checker, scope: &Scope, diagnostics: &mut | |
checker, | ||
), | ||
dunder_all_count: dunder_all_exprs.len(), | ||
ignore_init_module_imports: !fix_init, | ||
}) | ||
} else { | ||
None | ||
|
@@ -288,30 +313,31 @@ pub(crate) fn unused_import(checker: &Checker, scope: &Scope, diagnostics: &mut | |
first_party: true, | ||
.. | ||
}) | ||
) | ||
) && preview_mode | ||
}); | ||
|
||
// generate fixes that are shared across bindings in the statement | ||
let (fix_remove, fix_reexport) = if (!in_init || fix_init) && !in_except_handler { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This section looks like a large change but isn't. I only added |
||
( | ||
fix_by_removing_imports( | ||
checker, | ||
import_statement, | ||
to_remove.iter().map(|(binding, _)| binding), | ||
in_init, | ||
) | ||
.ok(), | ||
fix_by_reexporting( | ||
checker, | ||
import_statement, | ||
&to_reexport.iter().map(|(b, _)| b).collect::<Vec<_>>(), | ||
&dunder_all_exprs, | ||
let (fix_remove, fix_reexport) = | ||
if (!in_init || fix_init || preview_mode) && !in_except_handler { | ||
( | ||
fix_by_removing_imports( | ||
checker, | ||
import_statement, | ||
to_remove.iter().map(|(binding, _)| binding), | ||
in_init, | ||
) | ||
.ok(), | ||
fix_by_reexporting( | ||
checker, | ||
import_statement, | ||
&to_reexport.iter().map(|(b, _)| b).collect::<Vec<_>>(), | ||
&dunder_all_exprs, | ||
) | ||
.ok(), | ||
) | ||
.ok(), | ||
) | ||
} else { | ||
(None, None) | ||
}; | ||
} else { | ||
(None, None) | ||
}; | ||
|
||
for ((binding, context), fix) in iter::Iterator::chain( | ||
iter::zip(to_remove, iter::repeat(fix_remove)), | ||
|
@@ -320,6 +346,7 @@ pub(crate) fn unused_import(checker: &Checker, scope: &Scope, diagnostics: &mut | |
let mut diagnostic = Diagnostic::new( | ||
UnusedImport { | ||
name: binding.import.qualified_name().to_string(), | ||
module: binding.import.member_name().to_string(), | ||
binding: binding.name.to_string(), | ||
context, | ||
multiple, | ||
|
@@ -344,6 +371,7 @@ pub(crate) fn unused_import(checker: &Checker, scope: &Scope, diagnostics: &mut | |
let mut diagnostic = Diagnostic::new( | ||
UnusedImport { | ||
name: binding.import.qualified_name().to_string(), | ||
module: binding.import.member_name().to_string(), | ||
binding: binding.name.to_string(), | ||
context: None, | ||
multiple: false, | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,47 @@ | ||
--- | ||
source: crates/ruff_linter/src/rules/pyflakes/mod.rs | ||
--- | ||
__init__.py:19:8: F401 [*] `sys` imported but unused; consider removing, adding to `__all__`, or using a redundant alias | ||
| | ||
19 | import sys # F401: remove unused | ||
| ^^^ F401 | ||
| | ||
= help: Remove unused import: `sys` | ||
|
||
ℹ Unsafe fix | ||
16 16 | import argparse as argparse # Ok: is redundant alias | ||
17 17 | | ||
18 18 | | ||
19 |-import sys # F401: remove unused | ||
20 19 | | ||
21 20 | | ||
22 21 | # first-party | ||
|
||
__init__.py:33:15: F401 [*] `.unused` imported but unused; consider removing, adding to `__all__`, or using a redundant alias | ||
| | ||
33 | from . import unused # F401: change to redundant alias | ||
| ^^^^^^ F401 | ||
| | ||
= help: Remove unused import: `.unused` | ||
|
||
ℹ Unsafe fix | ||
30 30 | from . import aliased as aliased # Ok: is redundant alias | ||
31 31 | | ||
32 32 | | ||
33 |-from . import unused # F401: change to redundant alias | ||
34 33 | | ||
35 34 | | ||
36 35 | from . import renamed as bees # F401: no fix | ||
|
||
__init__.py:36:26: F401 [*] `.renamed` imported but unused; consider removing, adding to `__all__`, or using a redundant alias | ||
| | ||
36 | from . import renamed as bees # F401: no fix | ||
| ^^^^ F401 | ||
| | ||
= help: Remove unused import: `.renamed` | ||
|
||
ℹ Unsafe fix | ||
33 33 | from . import unused # F401: change to redundant alias | ||
34 34 | | ||
35 35 | | ||
36 |-from . import renamed as bees # F401: no fix |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,50 @@ | ||
--- | ||
source: crates/ruff_linter/src/rules/pyflakes/mod.rs | ||
--- | ||
__init__.py:19:8: F401 [*] `sys` imported but unused; consider removing, adding to `__all__`, or using a redundant alias | ||
| | ||
19 | import sys # F401: remove unused | ||
| ^^^ F401 | ||
| | ||
= help: Remove unused import: `sys` | ||
|
||
ℹ Unsafe fix | ||
16 16 | import argparse # Ok: is exported in __all__ | ||
17 17 | | ||
18 18 | | ||
19 |-import sys # F401: remove unused | ||
20 19 | | ||
21 20 | | ||
22 21 | # first-party | ||
|
||
__init__.py:36:15: F401 [*] `.unused` imported but unused; consider removing, adding to `__all__`, or using a redundant alias | ||
| | ||
36 | from . import unused # F401: add to __all__ | ||
| ^^^^^^ F401 | ||
| | ||
= help: Remove unused import: `.unused` | ||
|
||
ℹ Unsafe fix | ||
33 33 | from . import exported # Ok: is exported in __all__ | ||
34 34 | | ||
35 35 | | ||
36 |-from . import unused # F401: add to __all__ | ||
37 36 | | ||
38 37 | | ||
39 38 | from . import renamed as bees # F401: add to __all__ | ||
|
||
__init__.py:39:26: F401 [*] `.renamed` imported but unused; consider removing, adding to `__all__`, or using a redundant alias | ||
| | ||
39 | from . import renamed as bees # F401: add to __all__ | ||
| ^^^^ F401 | ||
| | ||
= help: Remove unused import: `.renamed` | ||
|
||
ℹ Unsafe fix | ||
36 36 | from . import unused # F401: add to __all__ | ||
37 37 | | ||
38 38 | | ||
39 |-from . import renamed as bees # F401: add to __all__ | ||
40 39 | | ||
41 40 | | ||
42 41 | __all__ = ["argparse", "exported"] |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think it's good that ruff has different handling of first vs third party imports, but changing behavior based on the name of the module (containing the import) I find unintuitive and I argue is unpythonic.
By that, I mean: if I have
mod/__init__.py
, is the fix any more or less safe than if I instead named itmod.py
? No — the__init__.py
is just an implementation detail and in either case, the module's interface changes with any import changes.It's not a universally-accepted convention that
__init__.py
should store your entire public API.For example, it's hinders import times for users of a large package, if the entire public API is placed in
__init__.py
, because then any import of a submodule will trigger an import__init__.py
and thus an import of every module, if everything is reexported.At most, we have the typing docs (Typing documentation: interface conventions) because there's nothing I can find on python.org that discusses this. But given the typing docs guidance, I would find it slightly easier to teach that:
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Also for mono-repos, there may be dozens or hundreds of first party top-level packages. Therefore this rule should base its decisions on whether an import is intra-package (e.g. a relative import) rather than first vs third party packages.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks for your feedback @Hnasar — would you mind opening a new issue to discuss this preview behavior? I don't think this pull request is a great place for it since it's just updating the docs.