use mlua::prelude::*; type ValueCallback = Box Fn(&'lua Lua, LuaThread<'lua>, LuaValue<'lua>) + 'static>; type ErrorCallback = Box Fn(&'lua Lua, LuaThread<'lua>, LuaError) + 'static>; const FORWARD_VALUE_KEY: &str = "__runtime__forwardValue"; const FORWARD_ERROR_KEY: &str = "__runtime__forwardError"; /** A set of callbacks for thread values and errors. These callbacks are used to forward values and errors from Lua threads back to Rust. By default, the runtime will print any errors to stderr and not do any operations with values. You can set your own callbacks using the `on_value` and `on_error` builder methods. */ pub struct Callbacks { on_value: Option, on_error: Option, } impl Callbacks { /** Creates a new set of callbacks with no callbacks set. */ pub fn new() -> Self { Self { on_value: None, on_error: None, } } /** Sets the callback for thread values being yielded / returned. */ pub fn on_value(mut self, f: F) -> Self where F: Fn(&Lua, LuaThread, LuaValue) + 'static, { self.on_value.replace(Box::new(f)); self } /** Sets the callback for thread errors. */ pub fn on_error(mut self, f: F) -> Self where F: Fn(&Lua, LuaThread, LuaError) + 'static, { self.on_error.replace(Box::new(f)); self } /** Removes any current thread value callback. */ pub fn without_value_callback(mut self) -> Self { self.on_value.take(); self } /** Removes any current thread error callback. */ pub fn without_error_callback(mut self) -> Self { self.on_error.take(); self } pub(crate) fn inject(self, lua: &Lua) { // Remove any previously injected callbacks lua.unset_named_registry_value(FORWARD_VALUE_KEY).ok(); lua.unset_named_registry_value(FORWARD_ERROR_KEY).ok(); // Create functions to forward values & errors if let Some(f) = self.on_value { lua.set_named_registry_value( FORWARD_VALUE_KEY, lua.create_function(move |lua, (thread, val): (LuaThread, LuaValue)| { f(lua, thread, val); Ok(()) }) .expect("failed to create value callback function"), ) .expect("failed to store value callback function"); } if let Some(f) = self.on_error { lua.set_named_registry_value( FORWARD_ERROR_KEY, lua.create_function(move |lua, (thread, err): (LuaThread, LuaError)| { f(lua, thread, err); Ok(()) }) .expect("failed to create error callback function"), ) .expect("failed to store error callback function"); } } pub(crate) fn forward_value(lua: &Lua, thread: LuaThread, value: LuaValue) { if let Ok(f) = lua.named_registry_value::(FORWARD_VALUE_KEY) { f.call::<_, ()>((thread, value)).unwrap(); } } pub(crate) fn forward_error(lua: &Lua, thread: LuaThread, error: LuaError) { if let Ok(f) = lua.named_registry_value::(FORWARD_ERROR_KEY) { f.call::<_, ()>((thread, error)).unwrap(); } } } impl Default for Callbacks { fn default() -> Self { Callbacks { on_value: Some(Box::new(default_value_callback)), on_error: Some(Box::new(default_error_callback)), } } } fn default_value_callback(_: &Lua, _: LuaThread, _: LuaValue) {} fn default_error_callback(_: &Lua, _: LuaThread, e: LuaError) { eprintln!("{e}"); }