forked from VimPlug/jedi
111 lines
3.5 KiB
Python
111 lines
3.5 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, from_path, to_path, module_node, node_to_str_map):
|
|
self._grammar = grammar
|
|
self._from_path = from_path
|
|
self._to_path = to_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._from_path,
|
|
tofile=self._to_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._from_path, 'w') as f:
|
|
f.write(self.get_code())
|
|
|
|
def __repr__(self):
|
|
return '<%s: %s>' % (self.__class__.__name__, self._from_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):
|
|
def calculate_to_path(p):
|
|
for from_, to in renames:
|
|
if p.startswith(from_):
|
|
p = to + p[len(from_):]
|
|
return p
|
|
|
|
renames = self.get_renames()
|
|
return [
|
|
ChangedFile(
|
|
self._grammar,
|
|
from_path=path,
|
|
to_path=calculate_to_path(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, key=lambda x: (-len(x), x))
|
|
|
|
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 f in self.get_changed_files():
|
|
f.apply()
|
|
|
|
for old, new in self.get_renames():
|
|
rename(old, new)
|
|
|
|
|
|
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)
|
|
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)
|