评论中建议的ast
模块最终运行良好。这是我创建的一个类,用于提取在每个函数中使用的包中定义的函数或变量。
import ast
import types
import inspect
class CausalBuilder(ast.NodeVisitor):
def __init__(self, package):
self.forest = []
self.fnames = []
for name, obj in vars(package).items():
if isinstance(obj, types.ModuleType):
with open(obj.__file__) as f:
text = f.read()
tree = ast.parse(text)
self.forest.append(tree)
elif isinstance(obj, types.FunctionType):
mod, *_ = inspect.getmodule(obj).__name__.split('.')
if mod == package.__name__:
self.fnames.append(name)
self.causes = {n: [] for n in self.fnames}
def build(self):
for tree in self.forest:
self.visit(tree)
return self.causes
def visit_FunctionDef(self, node):
self.generic_visit(node)
for b in node.body:
if node.name in self.fnames:
self.causes[node.name] += self.extract_cause(b)
def extract_cause(self, node):
nodes = [node]
cause = []
while nodes:
for i, n in enumerate(nodes):
ntype = type(n)
if ntype == ast.Name:
if n.id in self.fnames:
cause.append(n.id)
elif ntype in (ast.Assign, ast.AugAssign, ast.Attribute,
ast.Subscript, ast.Return):
nodes.append(n.value)
elif ntype in (ast.If, ast.IfExp):
nodes.append(n.test)
nodes.extend(n.body)
nodes.extend(n.orelse)
elif ntype == ast.Compare:
nodes.append(n.left)
nodes.extend(n.comparators)
elif ntype == ast.Call:
nodes.append(n.func)
elif ntype == ast.BinOp:
nodes.append(n.left)
nodes.append(n.right)
elif ntype == ast.UnaryOp:
nodes.append(n.operand)
elif ntype == ast.BoolOp:
nodes.extend(n.values)
elif ntype == ast.Num:
pass
else:
raise TypeError("Node type `{}` not accounted for."
.format(ntype))
nodes.pop(nodes.index(n))
return cause
可以通过首先导入 python 包并传递给构造函数来使用该类,然后build
像这样调用该方法:
import package
cb = CausalBuilder(package)
print(cb.build())
这将打印出一个字典,其中包含一组表示函数名称的键,以及表示函数中使用的函数和/或变量的列表的值。并非每种 ast 类型都被考虑在内,但这对我来说已经足够好了。
该实现递归地将节点分解为更简单的类型,直到它ast.Name
可以提取目标函数中正在使用的变量、函数或方法的名称。