forked from VimPlug/jedi
Get the first rename test passing
This commit is contained in:
@@ -523,6 +523,37 @@ class Script(object):
|
|||||||
]
|
]
|
||||||
return sorted(filter(def_ref_filter, defs), key=lambda x: (x.line, x.column))
|
return sorted(filter(def_ref_filter, defs), key=lambda x: (x.line, x.column))
|
||||||
|
|
||||||
|
def rename(self, line, column, new_name):
|
||||||
|
"""
|
||||||
|
Returns an object that you can use to rename the variable under the
|
||||||
|
cursor and its references to a different name.
|
||||||
|
"""
|
||||||
|
definitions = self.get_references(line, column, include_builtins=False)
|
||||||
|
file_renames = []
|
||||||
|
file_tree_name_map = {}
|
||||||
|
for d in definitions:
|
||||||
|
if d.type == 'module':
|
||||||
|
file_renames.append((d.module_path, new))
|
||||||
|
else:
|
||||||
|
# This private access is ok in a way. It's not public to
|
||||||
|
# protect Jedi users from seeing it.
|
||||||
|
tree_name = d._name.tree_name
|
||||||
|
if tree_name is not None:
|
||||||
|
fmap = file_tree_name_map.setdefault(d.module_path, {})
|
||||||
|
fmap[tree_name] = tree_name.prefix + new_name
|
||||||
|
from jedi.api.refactoring import Refactoring
|
||||||
|
return Refactoring(self._grammar, file_tree_name_map, file_renames)
|
||||||
|
|
||||||
|
def extract_variable(self):
|
||||||
|
"""
|
||||||
|
"""
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
def extract_method(self):
|
||||||
|
"""
|
||||||
|
"""
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
|
||||||
class Interpreter(Script):
|
class Interpreter(Script):
|
||||||
"""
|
"""
|
||||||
|
|||||||
@@ -8,3 +8,7 @@ class InternalError(_JediError):
|
|||||||
|
|
||||||
class WrongVersion(_JediError):
|
class WrongVersion(_JediError):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class RefactoringError(_JediError):
|
||||||
|
pass
|
||||||
|
|||||||
64
jedi/api/refactoring.py
Normal file
64
jedi/api/refactoring.py
Normal file
@@ -0,0 +1,64 @@
|
|||||||
|
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, next(iter(map_)).get_root_node(), 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 self._renames
|
||||||
|
|
||||||
|
def get_diff(self):
|
||||||
|
return ''.join(f.get_diff() for f in self.get_changed_files())
|
||||||
|
|
||||||
|
def apply(self):
|
||||||
|
for old, new in self._renames:
|
||||||
|
rename(old, new)
|
||||||
|
|
||||||
|
for f in self.get_changed_files():
|
||||||
|
f.apply()
|
||||||
@@ -16,14 +16,13 @@ import jedi
|
|||||||
class RefactoringCase(object):
|
class RefactoringCase(object):
|
||||||
|
|
||||||
def __init__(self, name, code, line_nr, index, path,
|
def __init__(self, name, code, line_nr, index, path,
|
||||||
new_name, start_line_test, desired):
|
args, desired):
|
||||||
self.name = name
|
self.name = name
|
||||||
self.code = code
|
self.code = code
|
||||||
self.line_nr = line_nr
|
self.line_nr = line_nr
|
||||||
self.index = index
|
self.index = index
|
||||||
self.path = path
|
self.path = path
|
||||||
self.new_name = new_name
|
self._args = args
|
||||||
self.start_line_test = start_line_test
|
|
||||||
self.desired = desired
|
self.desired = desired
|
||||||
|
|
||||||
@property
|
@property
|
||||||
@@ -32,29 +31,13 @@ class RefactoringCase(object):
|
|||||||
return f_name.replace('.py', '')
|
return f_name.replace('.py', '')
|
||||||
|
|
||||||
def refactor(self):
|
def refactor(self):
|
||||||
script = jedi.Script(self.code, self.line_nr, self.index, self.path)
|
script = jedi.Script(self.code, path=self.path)
|
||||||
refactor_func = getattr(script, self.refactor_type)
|
refactor_func = getattr(script, self.refactor_type)
|
||||||
args = (self.new_name,) if self.new_name else ()
|
return refactor_func(self.line_nr, self.index, *self._args)
|
||||||
return refactor_func(script, *args)
|
|
||||||
|
|
||||||
def run(self):
|
def run(self):
|
||||||
refactor_object = self.refactor()
|
refactor_object = self.refactor()
|
||||||
|
return refactor_object.get_diff()
|
||||||
# try to get the right excerpt of the newfile
|
|
||||||
f = refactor_object.new_files()[self.path]
|
|
||||||
lines = f.splitlines()[self.start_line_test:]
|
|
||||||
|
|
||||||
end = self.start_line_test + len(lines)
|
|
||||||
pop_start = None
|
|
||||||
for i, l in enumerate(lines):
|
|
||||||
if l.startswith('# +++'):
|
|
||||||
end = i
|
|
||||||
break
|
|
||||||
elif '#? ' in l:
|
|
||||||
pop_start = i
|
|
||||||
lines.pop(pop_start)
|
|
||||||
self.result = '\n'.join(lines[:end - 1]).strip()
|
|
||||||
return self.result
|
|
||||||
|
|
||||||
def check(self):
|
def check(self):
|
||||||
return self.run() == self.desired
|
return self.run() == self.desired
|
||||||
@@ -65,13 +48,12 @@ class RefactoringCase(object):
|
|||||||
|
|
||||||
|
|
||||||
def collect_file_tests(code, path, lines_to_execute):
|
def collect_file_tests(code, path, lines_to_execute):
|
||||||
r = r'^# --- ?([^\n]*)\n((?:(?!\n# \+\+\+).)*)' \
|
r = r'^# -{5} ?([^\n]*)\n((?:(?!\n# \+{5}).)*\n)' \
|
||||||
r'\n# \+\+\+((?:(?!\n# ---).)*)'
|
r'# \+{5}\n((?:(?!\n# -{5}).)*\n)'
|
||||||
for match in re.finditer(r, code, re.DOTALL | re.MULTILINE):
|
for match in re.finditer(r, code, re.DOTALL | re.MULTILINE):
|
||||||
name = match.group(1).strip()
|
name = match.group(1).strip()
|
||||||
first = match.group(2).strip()
|
first = match.group(2)
|
||||||
second = match.group(3).strip()
|
second = match.group(3)
|
||||||
start_line_test = code[:match.start()].count('\n') + 1
|
|
||||||
|
|
||||||
# get the line with the position of the operation
|
# get the line with the position of the operation
|
||||||
p = re.match(r'((?:(?!#\?).)*)#\? (\d*) ?([^\n]*)', first, re.DOTALL)
|
p = re.match(r'((?:(?!#\?).)*)#\? (\d*) ?([^\n]*)', first, re.DOTALL)
|
||||||
@@ -81,13 +63,13 @@ def collect_file_tests(code, path, lines_to_execute):
|
|||||||
until = p.group(1)
|
until = p.group(1)
|
||||||
index = int(p.group(2))
|
index = int(p.group(2))
|
||||||
new_name = p.group(3)
|
new_name = p.group(3)
|
||||||
|
args = (new_name,) if new_name else ()
|
||||||
|
|
||||||
line_nr = start_line_test + until.count('\n') + 2
|
line_nr = until.count('\n') + 2
|
||||||
if lines_to_execute and line_nr - 1 not in lines_to_execute:
|
if lines_to_execute and line_nr - 1 not in lines_to_execute:
|
||||||
continue
|
continue
|
||||||
|
|
||||||
yield RefactoringCase(name, code, line_nr, index, path,
|
yield RefactoringCase(name, first, line_nr, index, path, args, second)
|
||||||
new_name, start_line_test, second)
|
|
||||||
|
|
||||||
|
|
||||||
def collect_dir_tests(base_dir, test_files):
|
def collect_dir_tests(base_dir, test_files):
|
||||||
|
|||||||
@@ -3,15 +3,21 @@ Test coverage for renaming is mostly being done by testing
|
|||||||
`Script.get_references`.
|
`Script.get_references`.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
# --- simple
|
# ----- simple
|
||||||
def test1():
|
def test1():
|
||||||
#? 7 blabla
|
#? 7 blabla
|
||||||
test1()
|
test1()
|
||||||
AssertionError
|
AssertionError
|
||||||
return test1, test1.not_existing
|
return test1, test1.not_existing
|
||||||
# +++
|
# +++++
|
||||||
def blabla():
|
--- /home/dave/source/jedi/test/refactor/rename.py
|
||||||
blabla()
|
+++ /home/dave/source/jedi/test/refactor/rename.py
|
||||||
AssertionError
|
@@ -1,6 +1,6 @@
|
||||||
return blabla, blabla.not_existing
|
-def test1():
|
||||||
|
+def blabla():
|
||||||
|
#? 7 blabla
|
||||||
|
- test1()
|
||||||
|
+ blabla()
|
||||||
|
AssertionError
|
||||||
|
- return test1, test1.not_existing
|
||||||
|
+ return blabla, blabla.not_existing
|
||||||
|
|||||||
@@ -58,5 +58,5 @@ def test_refactor(refactor_case):
|
|||||||
|
|
||||||
:type refactor_case: :class:`.refactor.RefactoringCase`
|
:type refactor_case: :class:`.refactor.RefactoringCase`
|
||||||
"""
|
"""
|
||||||
refactor_case.run()
|
diff = refactor_case.run()
|
||||||
assert_case_equal(refactor_case, refactor_case.result, refactor_case.desired)
|
assert_case_equal(refactor_case, diff, refactor_case.desired)
|
||||||
|
|||||||
Reference in New Issue
Block a user