Skip to content
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

Improved function parser to use ast parser instead of Worder #780

Merged
merged 5 commits into from
Mar 13, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion .github/workflows/main.yml
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@ jobs:

strategy:
matrix:
python-version: ['3.8', '3.9', '3.10', '3.11', '3.12.0-rc.3']
python-version: ['3.8', '3.9', '3.10', '3.11', '3.12']
os: [ubuntu-latest, windows-latest, macos-latest]
fail-fast: false
steps:
Expand Down
3 changes: 2 additions & 1 deletion CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,9 +1,10 @@
# **Upcoming release**

- Check for ast.Attributes when finding occurrences in fstrings (@sandratsy)
- #751 Check for ast.Attributes when finding occurrences in fstrings (@sandratsy)
- #777, #698 add validation to refuse Rename refactoring to a python keyword
- #730 Match on module aliases for autoimport suggestions
- #755 Remove dependency on `build` package being installed while running tests
- #780 Improved function parser to use ast parser instead of Worder

# Release 1.12.0

Expand Down
2 changes: 1 addition & 1 deletion rope/contrib/generate.py
Original file line number Diff line number Diff line change
Expand Up @@ -401,7 +401,7 @@ def get_passed_args(self):
start, end = finder.get_primary_range(self.offset)
parens_start, parens_end = finder.get_word_parens_range(end - 1)
call = source[start:parens_end]
parser = functionutils._FunctionParser(call, False)
parser = functionutils._FunctionCallParser(call, False)
args, keywords = parser.get_parameters()
for arg in args:
if self._is_id(arg):
Expand Down
103 changes: 84 additions & 19 deletions rope/refactor/functionutils.py
Original file line number Diff line number Diff line change
@@ -1,5 +1,9 @@
import ast
from typing import Tuple, List

from rope.base import pyobjects, worder
from rope.base.builtins import Lambda
from rope.base.codeanalyze import SourceLinesAdapter


class DefinitionInfo:
Expand Down Expand Up @@ -33,7 +37,7 @@ def _read(pyfunction, code):
kind = pyfunction.get_kind()
is_method = kind == "method"
is_lambda = kind == "lambda"
info = _FunctionParser(code, is_method, is_lambda)
info = _FunctionDefParser(code, is_method, is_lambda)
args, keywords = info.get_parameters()
args_arg = None
keywords_arg = None
Expand Down Expand Up @@ -108,7 +112,7 @@ def read(primary, pyname, definition_info, code):
is_method_call = CallInfo._is_method_call(primary, pyname)
is_constructor = CallInfo._is_class(pyname)
is_classmethod = CallInfo._is_classmethod(pyname)
info = _FunctionParser(code, is_method_call or is_classmethod)
info = _FunctionCallParser(code, is_method_call or is_classmethod)
args, keywords = info.get_parameters()
args_arg = None
keywords_arg = None
Expand Down Expand Up @@ -202,8 +206,11 @@ def to_call_info(self, definition_info):
)


class _FunctionParser:
def __init__(self, call, implicit_arg, is_lambda=False):
class _BaseFunctionParser:
call: str
implicit_arg: bool

def __init__(self, call: str, implicit_arg: bool, is_lambda: bool = False):
self.call = call
self.implicit_arg = implicit_arg
self.word_finder = worder.Worder(self.call)
Expand All @@ -213,21 +220,6 @@ def __init__(self, call, implicit_arg, is_lambda=False):
self.last_parens = self.call.rindex(")")
self.first_parens = self.word_finder._find_parens_start(self.last_parens)

def get_parameters(self):
args, keywords = self.word_finder.get_parameters(
self.first_parens, self.last_parens
)
if self.is_called_as_a_method():
instance = self.call[: self.call.rindex(".", 0, self.first_parens)]
args.insert(0, instance.strip())
return args, keywords

def get_instance(self):
if self.is_called_as_a_method():
return self.word_finder.get_primary_at(
self.call.rindex(".", 0, self.first_parens) - 1
)

def get_function_name(self):
if self.is_called_as_a_method():
return self.word_finder.get_word_at(self.first_parens - 1)
Expand All @@ -236,3 +228,76 @@ def get_function_name(self):

def is_called_as_a_method(self):
return self.implicit_arg and "." in self.call[: self.first_parens]

def _get_source_range(self, tree):
start = self._lines.get_line_start(tree.lineno) + tree.col_offset
end = self._lines.get_line_start(tree.end_lineno) + tree.end_col_offset
return self._lines.code[start:end]


class _FunctionDefParser(_BaseFunctionParser):
_lines: SourceLinesAdapter

def __init__(self, call, implicit_arg, is_lambda=False):
super().__init__(call, implicit_arg, is_lambda=False)
_modified_call = "def " + call.rstrip(":") + ": pass"
self._lines = SourceLinesAdapter(_modified_call)
self.ast = ast.parse(_modified_call).body[0]
assert isinstance(self.ast, ast.FunctionDef)

def get_parameters(self) -> Tuple[List[str], List[Tuple[str, str]]]:
# FIXME: the weird parsing here is because we're replicating what
# _FunctionParser originally did, which was designed before the
# existence of posonlyargs and kwonlyargs, we'll want to rewrite this
# properly to handle them properly at some point
args = []
kwargs = []
args += [arg.arg for arg in self.ast.args.posonlyargs]
args += [arg.arg for arg in self.ast.args.args]
if self.ast.args.vararg is not None:
args += ["*" + self.ast.args.vararg.arg]
if len(self.ast.args.defaults) > 0:
defaults = self.ast.args.defaults
kwargs += [
(name, self._get_source_range(value))
for name, value in zip(args[-len(defaults) :], defaults)
]
del args[-len(self.ast.args.defaults) :]
if self.ast.args.kwarg is not None:
args += ["**" + self.ast.args.kwarg.arg]
if self.ast.args.kwonlyargs:
args += ["*"] + [arg.arg for arg in self.ast.args.kwonlyargs]
if len(self.ast.args.kw_defaults) > 0:
kw_defaults = self.ast.args.kw_defaults
kwargs += [
(name, self._get_source_range(value))
for name, value in zip(kwargs[-len(kw_defaults) :], kw_defaults)
]
del args[-len(self.ast.args.kw_defaults) :]
return args, kwargs


class _FunctionCallParser(_BaseFunctionParser):
_lines: SourceLinesAdapter
ast: ast.Call

def __init__(self, call, implicit_arg, is_lambda=False):
super().__init__(call, implicit_arg, is_lambda=False)
self._lines = SourceLinesAdapter(call)
self.ast = ast.parse(call).body[0].value
assert isinstance(self.ast, ast.Call)

def get_parameters(self) -> Tuple[List[str], List[Tuple[str, str]]]:
args = []
for arg in self.ast.args:
arg_value = self._get_source_range(arg)
args.append(arg_value)
kwargs = []
for kw in self.ast.keywords:
kw_value = self._get_source_range(kw.value)
assert kw.arg
kwargs.append((kw.arg, kw_value))
if self.is_called_as_a_method():
instance = self.call[: self.call.rindex(".", 0, self.first_parens)]
args.insert(0, instance.strip())
return args, kwargs
96 changes: 96 additions & 0 deletions ropetest/refactor/inlinetest.py
Original file line number Diff line number Diff line change
Expand Up @@ -1425,3 +1425,99 @@ def test_dictionary_with_inline_comment(self):
})
""")
self.assertEqual(expected, refactored)

def test_function_call_with_callsite_inline_comment(self):
code = dedent("""\
def a_func(arg1, arg2):
return arg1 + arg2 + 1
myvar = a_func(
1, # some comment
2, # another comment
)
""")
refactored = self._inline(code, code.index("a_func") + 1)
expected = dedent("""\
myvar = 1 + 2 + 1
""")
self.assertEqual(expected, refactored)

def test_function_call_with_callsite_interline_comment(self):
code = dedent("""\
def a_func(arg1, arg2):
return arg1 + arg2 + 1
myvar = a_func(
# some comment
1,
# another comment
2,
# another comment
)
""")
refactored = self._inline(code, code.index("a_func") + 1)
expected = dedent("""\
myvar = 1 + 2 + 1
""")
self.assertEqual(expected, refactored)

def test_function_call_with_defsite_inline_comment(self):
code = dedent("""\
def a_func(
arg1, # noqa
arg2,
):
return arg1 + arg2 + 1
myvar = a_func(1, 2)
""")
refactored = self._inline(code, code.index("a_func") + 1)
expected = dedent("""\
myvar = 1 + 2 + 1
""")
self.assertEqual(expected, refactored)

def test_function_call_with_defsite_interline_comment(self):
code = dedent("""\
def a_func(
# blah
arg1,
# blah blah
arg2,
# blah blah
):
return arg1 + arg2 + 1
myvar = a_func(1, 2)
""")
refactored = self._inline(code, code.index("a_func") + 1)
expected = dedent("""\
myvar = 1 + 2 + 1
""")
self.assertEqual(expected, refactored)

def test_function_call_with_posonlyargs(self):
code = dedent("""\
def a_func(arg1, /, arg2):
return arg1 + arg2 + 1
myvar = a_func(1, 2)
""")
refactored = self._inline(code, code.index("a_func") + 1)
expected = dedent("""\
myvar = 1 + 2 + 1
""")
self.assertEqual(expected, refactored)

def test_function_call_with_kwonlyargs(self):
code = dedent("""\
def a_func(arg1, *, arg2):
return arg1 + arg2 + 1
myvar = a_func(1, arg2=2)
""")
with self.assertRaises(rope.base.exceptions.RefactoringError):
refactored = self._inline(code, code.index("a_func") + 1)

def test_function_call_with_kwonlyargs2(self):
code = dedent("""\
def a_func(arg1, *, arg2=2):
return arg1 + arg2 + 1
myvar = a_func(1)
""")
with self.assertRaises(rope.base.exceptions.RefactoringError):
refactored = self._inline(code, code.index("a_func") + 1)
Loading