从 Rust 中调用 Lua 函数出错:`*mut rlua::ffi::lua_State` 无法在线程之间安全共享
2021-10-11 1:10:24
收藏:0
阅读:225
评论:1
我正在开发一个命令行程序,用于使用 MiniJinja 库(由 mitsuhiko 创建)渲染模板文件。
程序在这里:https://github.com/benwilber/temple。
我希望能够通过允许用户加载自定义 Lua 脚本来扩展程序,以便进行自定义过滤器、函数和测试。然而,我遇到了 Rust 生命周期错误,无法解决。
基本上,我希望能够将 Lua 函数注册为自定义过滤器函数。但是在编译时会出现错误。这是代码:
https://github.com/benwilber/temple/compare/0.3.1..lua
错误信息:
https://gist.github.com/c649a0b240cf299d3dbbe018c24cbcdc
如何从 MiniJinja 的 add_filter
函数中调用 Lua 函数?如果可能的话,我更愿意尝试在常规/安全的方式中解决这个问题。但是如果必须的话,我也可以尝试不安全的替代方案。
谢谢!
编辑:我在 Reddit 和 users.rust-lang.org 上发布了相同的问题。
原文链接 https://stackoverflow.com/questions/69519948
点赞
评论区的留言会收到邮件通知哦~
推荐文章
- 如何在roblox studio中1:1导入真实世界的地形?
- 求解,lua_resume的第二次调用继续执行协程问题。
- 【上海普陀区】内向猫网络招募【Skynet游戏框架Lua后端程序员】
- SF爱好求教:如何用lua实现游戏内调用数据库函数实现账号密码注册?
- Lua实现网站后台开发
- LUA错误显式返回,社区常见的规约是怎么样的
- lua5.3下载库失败
- 请问如何实现文本框内容和某个网页搜索框内容连接,并把网页输出来的结果反馈到另外一个文本框上
- lua lanes多线程使用
- 一个kv数据库
- openresty 有没有比较轻量的 docker 镜像
- 想问一下,有大佬用过luacurl吗
- 在Lua执行过程中使用Load函数出现问题
- 为什么 neovim 里没有显示一些特殊字符?
- Lua比较两个表的值(不考虑键的顺序)
- 有个lua简单的项目,外包,有意者加微信 liuheng600456详谈,最好在成都
- 如何在 Visual Studio 2022 中运行 Lua 代码?
- addEventListener 返回 nil Lua
- Lua中获取用户配置主目录的跨平台方法
- 如何编写 Lua 模式将字符串(嵌套数组)转换为真正的数组?
Lua 使用的状态不安全,不能从多个线程中使用。由此导致
LuaFunction
不是Sync
或Send
。错误信息的一部分正在强制执行:
help: within `LuaFunction<'_>`, the trait `Sync` is not implemented for `*mut rlua::ffi::lua_State`
相反,
minijinja::Filter
必须实现Send + Sync + 'static
。 (请参阅https://docs.rs/minijinja/0.5.0/minijinja/filters/trait.Filter.html)。这意味着不能在调用
Filters
之间共享LuaFunctions
(甚至是LuaContext
)。一种选择是不要将 lua 状态传递给闭包,而是创建一个新的 lua 状态,例如:
env.add_filter( "concat2", |_env: &Environment, s1: String, s2: String| -> anyhow::Result<String, minijinja::Error> { lua.context(|lua_ctx| { lua_ctx.load(include_str!("temple.lua")).exec().unwrap(); let globals = lua_ctx.globals(); let temple: rlua::Table = globals.get("temple").unwrap(); let filters: rlua::Table = temple.get("_filters").unwrap(); let concat2: rlua::Function = filters.get("concat2").unwrap(); let res: String = concat2.call::<_, String>((s1, s2)).unwrap(); Ok(res) } } );
这可能会有相对较高的开销。
另一个选择是在一个线程中创建 rlua 状态,并通过管道进行通信。这将更像这样:
pub fn test() { let mut env = minijinja::Environment::new(); let (to_lua_tx, to_lua_rx) = channel::<(String,String,SyncSender<String>)>(); thread::spawn(move|| { let lua = rlua::Lua::new(); lua.context(move |lua_ctx| { lua_ctx.load("some_code").exec().unwrap(); let globals = lua_ctx.globals(); let temple: rlua::Table = globals.get("temple").unwrap(); let filters: rlua::Table = temple.get("_filters").unwrap(); let concat2: rlua::Function = filters.get("concat2").unwrap(); while let Ok((s1,s2, channel)) = to_lua_rx.recv() { let res: String = concat2.call::<_, String>((s1, s2)).unwrap(); channel.send(res).unwrap() } }) }); let to_lua_tx = Mutex::new(to_lua_tx); env.add_filter( "concat2", move |_env: &minijinja::Environment, s1: String, s2: String| -> anyhow::Result<String, minijinja::Error> { let (tx,rx) = sync_channel::<String>(0); to_lua_tx.lock().unwrap().send((s1,s2,tx)).unwrap(); let res = rx.recv().unwrap(); Ok(res) } ); }
即使以这种方式启动多个 lua 状态也是有可能的,但需要更多的管道。
免责声明:此代码均未经测试 - 但在 playground 中使用了 minijinja 和 rlua 的存根版本。您可能需要更好的错误处理,并可能需要一些额外的代码来处理干净地关闭所有线程。