首页 > 解决方案 > 使用 pybind11 或 Python C API 编译和执行 AST

问题描述

我想使用 Pybind11 或直接使用 Python C API 将此 Python 代码转换为 C++ 代码:

import ast

code = "print('Hello World!')"

code_ast = ast.parse(code, mode="exec") # "code" being a string containing code
# ... perform some modifications on "code_ast"
exec(compile(code_ast, filename="<ast>", mode="exec"))

这是我目前使用的 Pybind11:

#include <iostream>
#include "pybind11/embed.h"

namespace py = pybind11;

std::string code = "print('Hello World!')";

py::module ast = py::module::import("ast");
py::module builtins = py::module::import("builtins");

py::object code_ast = ast.attr("parse")(code, "<unknown>", "exec");
// ... perform some modifications on "code_ast"
py::object compiled_code = builtins.attr("compile")(code_ast, "<ast>", "exec");
builtins.attr("exec")(compiled_code);

不幸的是,最后一行 C++ 代码引发了运行时错误:SystemError: frame does not exist.

我不确定我是否理解此错误,我尝试将globalsand传递localsexec,但没有解决问题。

编辑:通过时globals,它说:SystemError: <built-in function globals> returned NULL without setting an error

关于如何正确实现这一目标的任何想法?

标签: pythonc++abstract-syntax-treepython-c-apipybind11

解决方案


我找到了答案,我没有使用 Pybind11 中的函数,而是 Python ( )globals中的内置函数。builtins.attr("globals")()

这是工作版本:

py::module ast = py::module::import("ast");
py::module builtins = py::module::import("builtins");

py::object code_ast = ast.attr("parse")(code, "<unknown>", "exec");
// ... perform some modifications on "code_ast"
py::object compiled_code = builtins.attr("compile")(code_ast, "<ast>", "exec");
builtins.attr("exec")(compiled_code, py::globals());

推荐阅读