匹配 lib2to3 的 1-or-2-arg 函数调用的模式
pattern to match 1-or-2-arg function call for lib2to3
我有一个脚本可以重写 Python 模块,以便将所有出现的 func(a)
转换为 func2(a is None)
。我现在也想支持 func(a, msg)
变成 func2(a is None, msg)
,但我找不到能做到这一点的模式。以下显示了我的尝试:
from lib2to3 import refactor, fixer_base
from textwrap import dedent
PATTERN_ONE_ARG = """power< 'func' trailer< '(' arglist< obj1=any > ')' > >"""
PATTERN_ONE_OR_TWO_ARGS = """power< 'func' trailer< '(' arglist< obj1=any [',' obj2=any] > ')' > >"""
class TestFixer(fixer_base.BaseFix):
def __init__(self, options, fixer_log):
# self.PATTERN = PATTERN_ONE_ARG
self.PATTERN = PATTERN_ONE_OR_TWO_ARGS
super().__init__(options, fixer_log)
def transform(self, node, results):
print("found it")
return node
class TestRefactoringTool(refactor.MultiprocessRefactoringTool):
def get_fixers(self):
fixer = TestFixer(self.options, self.fixer_log)
return [fixer], []
def test():
test_script = """
log.print("hi")
func(a, "12345")
func(a, msg="12345")
func(a)
"""
refac.refactor_string(dedent(test_script), 'script')
flags = dict(print_function=True)
refac = TestRefactoringTool([], flags)
test()
对于在 test_script
字符串中找到的每个 func
,我应该看到一个 "found it",所以总共有 3 个,但我只看到打印了 2 个,这意味着 func(a)
没有被模式匹配器找到。我的模式基于 lib2to3.fixes
中可用的修复程序,但我一定遗漏了一个微妙之处。任何人都知道谁可以修复 PATTERN_ONE_OR_TWO_ARGS 以便找到所有 3 个函数?
我知道我可以创建一个单独的修复器实例,但是使用该模式可以使我免于编写大量代码(我有几十个这样的修复器,那么总数将是 24 个!)。
找到了:
PATTERN_ONE_OR_TWO_ARGS = """
power< 'func' trailer< '('
( not(arglist | argument<any '=' any>) obj1=any
| arglist< obj1=any ',' obj2=any > )
')' > >
"""
如果 transform() 是:
def transform(self, node, results):
if 'obj2' in results:
print("found 2", results['obj1'], results['obj2'])
else:
print("found 1", results['obj1'])
return node
然后
test_script = """
log.print("hi")
func(a, "12345")
func(a, msg="12345")
func(a)
func(k=a)
"""
输出是
found 2 a "12345"
found 2 a msg="12345"
found 1 a
我还发现 http://python3porting.com/fixers.html#fixers-chapter 这表明我可以覆盖 match() 方法而不是使用模式。感兴趣的四种模式是:
PATTERN_ONE_ARG_OR_KWARG = """power< 'func' trailer< '(' not(arglist) obj1=any ')' > >"""
PATTERN_ONE_ARG = """power< 'func' trailer< '(' not(arglist | argument<any '=' any>) obj1=any ')' > >"""
PATTERN_ONE_KWARG = """power< 'func' trailer< '(' obj1=argument< any '=' any > ')' > >"""
PATTERN_TWO_ARGS_OR_KWARGS = """power< 'func' trailer< '(' arglist< obj1=any ',' obj2=any > ')' > >"""
我有一个脚本可以重写 Python 模块,以便将所有出现的 func(a)
转换为 func2(a is None)
。我现在也想支持 func(a, msg)
变成 func2(a is None, msg)
,但我找不到能做到这一点的模式。以下显示了我的尝试:
from lib2to3 import refactor, fixer_base
from textwrap import dedent
PATTERN_ONE_ARG = """power< 'func' trailer< '(' arglist< obj1=any > ')' > >"""
PATTERN_ONE_OR_TWO_ARGS = """power< 'func' trailer< '(' arglist< obj1=any [',' obj2=any] > ')' > >"""
class TestFixer(fixer_base.BaseFix):
def __init__(self, options, fixer_log):
# self.PATTERN = PATTERN_ONE_ARG
self.PATTERN = PATTERN_ONE_OR_TWO_ARGS
super().__init__(options, fixer_log)
def transform(self, node, results):
print("found it")
return node
class TestRefactoringTool(refactor.MultiprocessRefactoringTool):
def get_fixers(self):
fixer = TestFixer(self.options, self.fixer_log)
return [fixer], []
def test():
test_script = """
log.print("hi")
func(a, "12345")
func(a, msg="12345")
func(a)
"""
refac.refactor_string(dedent(test_script), 'script')
flags = dict(print_function=True)
refac = TestRefactoringTool([], flags)
test()
对于在 test_script
字符串中找到的每个 func
,我应该看到一个 "found it",所以总共有 3 个,但我只看到打印了 2 个,这意味着 func(a)
没有被模式匹配器找到。我的模式基于 lib2to3.fixes
中可用的修复程序,但我一定遗漏了一个微妙之处。任何人都知道谁可以修复 PATTERN_ONE_OR_TWO_ARGS 以便找到所有 3 个函数?
我知道我可以创建一个单独的修复器实例,但是使用该模式可以使我免于编写大量代码(我有几十个这样的修复器,那么总数将是 24 个!)。
找到了:
PATTERN_ONE_OR_TWO_ARGS = """
power< 'func' trailer< '('
( not(arglist | argument<any '=' any>) obj1=any
| arglist< obj1=any ',' obj2=any > )
')' > >
"""
如果 transform() 是:
def transform(self, node, results):
if 'obj2' in results:
print("found 2", results['obj1'], results['obj2'])
else:
print("found 1", results['obj1'])
return node
然后
test_script = """
log.print("hi")
func(a, "12345")
func(a, msg="12345")
func(a)
func(k=a)
"""
输出是
found 2 a "12345"
found 2 a msg="12345"
found 1 a
我还发现 http://python3porting.com/fixers.html#fixers-chapter 这表明我可以覆盖 match() 方法而不是使用模式。感兴趣的四种模式是:
PATTERN_ONE_ARG_OR_KWARG = """power< 'func' trailer< '(' not(arglist) obj1=any ')' > >"""
PATTERN_ONE_ARG = """power< 'func' trailer< '(' not(arglist | argument<any '=' any>) obj1=any ')' > >"""
PATTERN_ONE_KWARG = """power< 'func' trailer< '(' obj1=argument< any '=' any > ')' > >"""
PATTERN_TWO_ARGS_OR_KWARGS = """power< 'func' trailer< '(' arglist< obj1=any ',' obj2=any > ')' > >"""