在不重启服务器的情况下重新加载模块

reload module without restart server

美好的一天! 我有一个关于在 tarantool 中重新加载 c 模块的小问题 例如:我有一个公开方法的 c 模块:

int calculate(lua_State* L);

另外我声明了入口点:

extern "C"
{    
    LUA_API int luaopen_cuendemodule(lua_State *L);
}

现在,我在 tarantool 中加载此模块 ("testmodule.so"):

require('testmodule')
box.schema.func.create('testmodule.calculate')
box.schema.user.grant('user', 'execute', 'function', 'testmodule.calculate')

现在我从我的 C# 客户端调用这个方法:

await tarantoolClient.Call<TarantoolTuple<CalculateParameters>, CalculationResults>("testmodule.calculate", TarantoolTuple.Create(....));

它按预期工作 - 执行了方法计算并返回了结果

但如果我想更新我的模块,那么问题就开始了:在我替换 so 文件并调用计算方法后,我的 tarantool 重新启动,我可以在 dmesg

中看到类似 "tarntool invalid opcode in testmodule.so" 的内容

阅读文档后,我在函数定义中看到附加参数,如下所示:

box.schema.func.create('testmodule.calculate', {language = 'C'})

但在此之后,如果我从 C# 调用它,我会收到异常消息 "failed to dynamically load function undefined symbol calculate"

我在 ubuntu 上使用 tarantool 1.7 我是用 gcc 8.1.0

编译的

没有很好的方法来做到这一点(意味着 - 一种可移植的方式)。我认为,最好的方法是使用类似 dlopen()[1] 的方法,该函数允许打开(一般管理共享对象)共享对象,但您必须对此非常谨慎,它也可能会使您的代码失败甚至你可以有 sigfault。

一个很好的例子是:https://github.com/tarantool/mqtt,模块不使用这些功能(func.create等等),但它也可以扩展。

所以重点是:如果你开发一个C-module,你必须考虑重新加载策略。 例如,*-unix 类系统有很多允许重新加载一些共享对象的功能,而且 tarantool 也有一些功能。

另外,我建议你开始考虑模块,就像 Lua 的 C 模块一样,实际上是一样的。

PS

一些重载模块也可用:https://github.com/Mons/tnt-package-reload (I didn't test the module), https://github.com/tarantool/reload(我没有测试模块)

[1] http://man7.org/linux/man-pages/man3/dlopen.3.html

我想,您可以试试这个模块来重新加载您的应用程序 -https://github.com/moonlibs/package-reload.