1
0
forked from VimPlug/jedi
Files
jedi-fork/jedi/api/refactoring.py
2020-02-11 01:35:07 +01:00

101 lines
3.1 KiB
Python

from os.path import dirname, basename, join
import re
import difflib
from parso import split_lines
class ChangedFile(object):
def __init__(self, grammar, path, module_node, node_to_str_map):
self._grammar = grammar
self._path = path
self._module_node = module_node
self._node_to_str_map = node_to_str_map
def get_diff(self):
old_lines = split_lines(self._module_node.get_code(), keepends=True)
new_lines = split_lines(self.get_code(), keepends=True)
diff = difflib.unified_diff(
old_lines, new_lines,
fromfile=self._path,
tofile=self._path
)
# Apparently there's a space at the end of the diff - for whatever
# reason.
return ''.join(diff).rstrip(' ')
def get_code(self):
return self._grammar.refactor(self._module_node, self._node_to_str_map)
def apply(self):
with open(self._path, 'w') as f:
f.write(self.get_code())
def __repr__(self):
return '<%s: %s>' % (self.__class__.__name__, self._path)
class Refactoring(object):
def __init__(self, grammar, file_to_node_changes, renames=()):
self._grammar = grammar
self._renames = renames
self._file_to_node_changes = file_to_node_changes
def get_changed_files(self):
return [
ChangedFile(
self._grammar, path,
module_node=next(iter(map_)).get_root_node(),
node_to_str_map=map_
) for path, map_ in self._file_to_node_changes.items()
]
def get_renames(self):
"""
Files can be renamed in a refactoring.
Returns ``Iterable[Tuple[str, str]]``.
"""
return sorted(self._renames)
def get_diff(self):
text = ''
for from_, to in self.get_renames():
text += 'rename from %s\nrename to %s\n' % (from_, to)
return text + ''.join(f.get_diff() for f in self.get_changed_files())
def apply(self):
for old, new in self.get_renames():
rename(old, new)
for f in self.get_changed_files():
f.apply()
def _calculate_rename(path, new_name):
name = basename(path)
dir_ = dirname(path)
if name in ('__init__.py', '__init__.pyi'):
parent_dir = dirname(dir_)
return dir_, join(parent_dir, new_name, name)
ending = re.search(r'\.pyi?$', name).group(0)
return path, join(dir_, new_name + ending)
def rename(grammar, definitions, new_name):
file_renames = set()
file_tree_name_map = {}
for d in definitions:
tree_name = d._name.tree_name
if d.type == 'module' and tree_name is None:
file_renames.add(_calculate_rename(d.module_path, new_name))
else:
# This private access is ok in a way. It's not public to
# protect Jedi users from seeing it.
if tree_name is not None:
fmap = file_tree_name_map.setdefault(d.module_path, {})
fmap[tree_name] = tree_name.prefix + new_name
return Refactoring(grammar, file_tree_name_map, file_renames)