Есть некоторый legacy код на С++, использующий boost::python и импортирующий самописные пиноновские модули спользуя модуль imp
.
Делалось это примерно так:
test_module.py
:
def test_out(expr):
print(expr)
test_imp.cpp
:
#include <boost/python.hpp>
#include <stdio.h>
int main()
{
Py_InitializeEx(0);
try {
boost::python::object modImp = boost::python::import("imp");
PyImport_AddModule("test_module");
modImp.attr("load_source")("test_module", "test_module.py");
boost::python::exec("test_out(\"Hello, world!\")", boost::python::import("test_module").attr("__dict__"));
}
catch (const boost::python::error_already_set&)
{
PyErr_Print();
}
Py_Finalize();
}
Но начиная с 3.12’го питона имодуль imp
изъяли из обращения. Я попытался переехать на importlib.util
как рекомендовано тут, в меру своего понмания переложив это на С++ный код. Получилось что-то вроде этого:
test_import.cpp
:
#include <boost/python.hpp>
#include <stdio.h>
int main()
{
Py_InitializeEx(0);
try {
boost::python::object modImpUtil = boost::python::import("importlib.util");
PyImport_AddModule("test_module");
boost::python::object spec = modImpUtil.attr("spec_from_file_location")("test_module", "test_module.py");
boost::python::object mod = modImpUtil.attr("module_from_spec")(spec);
spec.attr("loader").attr("exec_module")(mod);
boost::python::exec("test_out(\"Hello, World!\")", boost::python::import("test_module").attr("__dict__"));
}
catch (const boost::python::error_already_set&)
{
PyErr_Print();
}
Py_Finalize();
}
Получилось, но не работает. Пишет NameError: name 'test_out' is not defined
Команды для сборки обоих сишников:
g++ test_imp.cpp -I /usr/include/x86_64-linux-gnu/python3.11 -I /usr/include/python3.11 -lpython3.11 -lboost_python311 -lboost_system -o test_imp
g++ test_import.cpp -I /usr/include/x86_64-linux-gnu/python3.11 -I /usr/include/python3.11 -lpython3.11 -lboost_python311 -lboost_system -o test_import
Я далек от программирования на питоне, поэтому могу упустить что-нибудь очевидное. Но этот код кто-то должен поддерживать. Поэтому с благодарностью приму помощь…
Update: Правильный ответ
#include <boost/python.hpp>
#include <stdio.h>
int main()
{
Py_InitializeEx(0);
try {
boost::python::object modImpUtil = boost::python::import("importlib.util");
boost::python::object modSys = boost::python::import("sys");
PyImport_AddModule("test_module");
boost::python::object spec = modImpUtil.attr("spec_from_file_location")("test_module", "test_module.py");
boost::python::object mod = modImpUtil.attr("module_from_spec")(spec);
spec.attr("loader").attr("exec_module")(mod);
modSys.attr("modules")["test_module"] = mod;
boost::python::exec("test_out(\"Hello, World!\")", boost::python::import("test_module").attr("__dict__"));
}
catch (const boost::python::error_already_set&)
{
PyErr_Print();
}
Py_Finalize();
}