-
Notifications
You must be signed in to change notification settings - Fork 176
Description
Describe the bug
When MoveModule encounters an import statement inside a function (thus, evaluated at some unknown point later and maybe never), and the import statement concerns the module being moved, the calculated changes include both a modification of the original import statement and an extraneous import statement at the top of the file. This changes the order of imports and, in sufficiently cursed code-bases, may even result in a circular import.
Basically, the issue is that MoveModule appears to consider imports to be side effect-free, which sometimes they aren't.
The extra import is being added here:
Lines 594 to 596 in 25586a6
| if should_import: | |
| pymodule = self.tools.new_pymodule(pymodule, source) | |
| source = self.tools.add_imports(pymodule, [new_import]) |
Lines 769 to 773 in 25586a6
| def _add_imports_to_module(import_tools, pymodule, new_imports): | |
| module_with_imports = import_tools.module_imports(pymodule) | |
| for new_import in new_imports: | |
| module_with_imports.add_import(new_import) | |
| return module_with_imports.get_changed_source() |
To Reproduce
Steps to reproduce the behavior:
- Code before refactoring:
# a.py
def use_b_later():
import b
# b.py
import a
- Describe the refactoring you want to do
Rename (move) b into prefix.b.
- Expected code after refactoring:
# a.py
def use_b_later():
import prefix.b
# b.py
import a
- Describe the error or unexpected result that you are getting
# a.py
import prefix.b
def use_b_later():
import prefix.b
# b.py
import a
Reproducible example
$ python << EOF
import shutil
import tempfile
from contextlib import contextmanager
from os import walk
from pathlib import Path
from rope.base.project import Project
from rope.refactor.move import MoveModule
@contextmanager
def temp_tree():
path = tempfile.mkdtemp()
try:
yield Path(path)
finally:
shutil.rmtree(path, ignore_errors=True)
with temp_tree() as root:
(root / "utils.py").write_text(
"""
def use_lazy():
import lazy"""
)
(root / "lazy.py").write_text(
"""
import utils"""
)
(root / "prefix").mkdir()
(root / "prefix" / "__init__.py").touch()
project = Project(root)
new_package = project.get_folder("prefix")
lazy = project.get_module("lazy").get_resource()
print(MoveModule(project, lazy).get_changes(new_package).get_description())
EOF
Moving module <lazy>:
--- a/lazy.py
+++ b/lazy.py
@@ -1,2 +1 @@
-
-import utils+import utils
--- a/utils.py
+++ b/utils.py
@@ -1,3 +1,3 @@
-
+import prefix.lazy
def use_lazy():
- import lazy+ import prefix.lazy
rename from lazy.py
rename to prefix/lazy.pyEditor information (please complete the following information):
- Project Python version: 3.11
- Rope Python version: 3.11
- Rope version: 1.11.0 (also checked 1.9.0)
- Text editor/IDE and version: ...
Additional context
Context: trying to use Rope for some automation in https://github.com/SomeoneSerge/pkgs/blob/master/python-packages/by-name/pr/prefix-python-modules/prefix_python_modules.py