如何使用 ast(抽象语法树)模块获取节点的完整“路径”?

Pie*_*rre 6 python abstract-syntax-tree

我正在修改 Python 的ast(抽象语法树)模块。

使用ast.NodeVisitor,可以遍历源树并显示信息,例如每个节点的名称,如下所示:

class MyAST(ast.NodeVisitor):      

     def visit_ClassDef(self, node):
         print(node.col_offset * " " +node.name)         
         self.generic_visit(node)   

     def visit_FunctionDef(self, node):         
         print(node.col_offset * " " + node.name)     
         self.generic_visit(node)   
Run Code Online (Sandbox Code Playgroud)

然后它可以用于任何代码,如下所示:

tree = ast.parse(open("/path/to/file.py").read())
MyAST().visit(tree)
Run Code Online (Sandbox Code Playgroud)

这将输出类似的内容(例如在requests包中的models.py上使用):

RequestEncodingMixin                                                                                                                                                         
    path_url                                                                                                                                                                 
    _encode_params                                                                                                                                                           
    _encode_files                                                                                                                                                            
RequestHooksMixin                                                                                                                                                            
    register_hook                                                                                                                                                            
    deregister_hook                                                                                                                                                          
Request                                                                                                                                                                      
    __init__
    __repr__
    prepare
(...)
Run Code Online (Sandbox Code Playgroud)

我想知道对于给定节点,它的完整“路径”。在上面的输出示例中,当我在FunctionDef节点 中时deregister_hook,我想知道它的父节点,ClassDef RequestHooksMixin以便我可以输出类似 的内容RequestHooksMixin.deregister_hook

blh*_*ing 0

ast.NodeVisitor设计时并未考虑嵌套级别,因此它不是生成基于级别缩进的输出的最佳工具。

相反,您可以ast.iter_child_nodes递归地访问每个节点的直接子节点,并输出该节点并在它属于类或函数时增加缩进级别:

import ast

def build_tree(node):
    def _visit(node, level=0):
        if isinstance(node, (ast.ClassDef, ast.FunctionDef)):
            yield node.name, level
            level += 1
        for child in ast.iter_child_nodes(node):
            yield from _visit(child, level)
    return '\n'.join(f'{"    " * level}{name}' for name, level in _visit(node))
Run Code Online (Sandbox Code Playgroud)

以便:

source = '''
class Foo:
    def method1(self):
        pass
    def method2(self):
        pass

class Bar:
    def method1(self):
        pass
    def method2(self):
        pass
'''
print(build_tree(ast.parse(source)))
Run Code Online (Sandbox Code Playgroud)

输出:

Foo
    method1
    method2
Bar
    method1
    method2
Run Code Online (Sandbox Code Playgroud)

演示: https: //ideone.com/aQHqbi