use std::borrow::Cow;
use std::collections::{BTreeMap, BTreeSet, HashMap, HashSet};
use std::ffi::{CStr, CString, OsString};
use std::path::PathBuf;
use bstr::BString;
use maplit::{btreemap, btreeset, hashmap, hashset};
use mlua::{
AnyUserData, BorrowedBytes, BorrowedStr, Either, Error, Function, IntoLua, Lua, RegistryKey, Result,
Table, Thread, UserDataRef, Value,
};
#[test]
fn test_value_into_lua() -> Result<()> {
let lua = Lua::new();
let v = Value::Boolean(true);
let v2 = (&v).into_lua(&lua)?;
assert_eq!(v, v2);
let table = lua.create_table()?;
table.set("v", &v)?;
assert_eq!(v, table.get::<Value>("v")?);
Ok(())
}
#[test]
fn test_string_into_lua() -> Result<()> {
let lua = Lua::new();
let s = lua.create_string("hello, world!")?;
let s2 = (&s).into_lua(&lua)?;
assert_eq!(s, *s2.as_string().unwrap());
let table = lua.create_table()?;
table.set("s", &s)?;
assert_eq!(s, table.get::<String>("s")?);
Ok(())
}
#[test]
fn test_string_from_lua() -> Result<()> {
let lua = Lua::new();
let f = lua.create_function(|_, s: mlua::String| Ok(s))?;
let s = f.call::<String>("hello, world!")?;
assert_eq!(s, "hello, world!");
let s = f.call::<String>(42)?;
assert_eq!(s, "42");
Ok(())
}
#[test]
fn test_borrowedstr_into_lua() -> Result<()> {
let lua = Lua::new();
let s = lua.create_string("hello, world!")?;
let bs = s.to_str()?;
let bs2 = (&bs).into_lua(&lua)?;
assert_eq!(bs2.as_string().unwrap(), "hello, world!");
let table = lua.create_table()?;
table.set("bs", &bs)?;
assert_eq!(bs, table.get::<String>("bs")?);
Ok(())
}
#[test]
fn test_borrowedstr_from_lua() -> Result<()> {
let lua = Lua::new();
let f = lua.create_function(|_, s: BorrowedStr| Ok(s))?;
let s = f.call::<String>("hello, world!")?;
assert_eq!(s, "hello, world!");
Ok(())
}
#[test]
fn test_borrowedbytes_into_lua() -> Result<()> {
let lua = Lua::new();
let s = lua.create_string("hello, world!")?;
let bb = s.as_bytes();
let bb2 = (&bb).into_lua(&lua)?;
assert_eq!(bb2.as_string().unwrap(), "hello, world!");
let table = lua.create_table()?;
table.set("bb", &bb)?;
assert_eq!(bb, table.get::<String>("bb")?.as_bytes());
Ok(())
}
#[test]
fn test_borrowedbytes_from_lua() -> Result<()> {
let lua = Lua::new();
let f = lua.create_function(|_, s: BorrowedBytes| Ok(s))?;
let s = f.call::<String>("hello, world!")?;
assert_eq!(s, "hello, world!");
Ok(())
}
#[test]
fn test_table_into_lua() -> Result<()> {
let lua = Lua::new();
let t = lua.create_table()?;
let t2 = (&t).into_lua(&lua)?;
assert_eq!(&t, t2.as_table().unwrap());
let f = lua.create_function(|_, (t, s): (Table, String)| t.set("s", s))?;
f.call::<()>((&t, "hello"))?;
assert_eq!("hello", t.get::<String>("s")?);
Ok(())
}
#[test]
fn test_function_into_lua() -> Result<()> {
let lua = Lua::new();
let f = lua.create_function(|_, ()| Ok::<_, Error>(()))?;
let f2 = (&f).into_lua(&lua)?;
assert_eq!(&f, f2.as_function().unwrap());
let table = lua.create_table()?;
table.set("f", &f)?;
assert_eq!(f, table.get::<Function>("f")?);
Ok(())
}
#[test]
fn test_function_from_lua() -> Result<()> {
let lua = Lua::new();
assert!(lua.globals().get::<Function>("print").is_ok());
match lua.globals().get::<Function>("math") {
Err(err @ Error::FromLuaConversionError { .. }) => {
assert_eq!(err.to_string(), "error converting Lua table to function");
}
_ => panic!("expected `Error::FromLuaConversionError`"),
}
Ok(())
}
#[test]
fn test_thread_into_lua() -> Result<()> {
let lua = Lua::new();
let f = lua.create_function(|_, ()| Ok::<_, Error>(()))?;
let th = lua.create_thread(f)?;
let th2 = (&th).into_lua(&lua)?;
assert_eq!(&th, th2.as_thread().unwrap());
let table = lua.create_table()?;
table.set("th", &th)?;
assert_eq!(th, table.get::<Thread>("th")?);
Ok(())
}
#[test]
fn test_thread_from_lua() -> Result<()> {
let lua = Lua::new();
match lua.globals().get::<Thread>("print") {
Err(err @ Error::FromLuaConversionError { .. }) => {
assert_eq!(err.to_string(), "error converting Lua function to thread");
}
_ => panic!("expected `Error::FromLuaConversionError`"),
}
Ok(())
}
#[test]
fn test_anyuserdata_into_lua() -> Result<()> {
let lua = Lua::new();
let ud = lua.create_any_userdata(String::from("hello"))?;
let ud2 = (&ud).into_lua(&lua)?;
assert_eq!(&ud, ud2.as_userdata().unwrap());
let table = lua.create_table()?;
table.set("ud", &ud)?;
assert_eq!(ud, table.get::<AnyUserData>("ud")?);
assert_eq!("hello", *table.get::<UserDataRef<String>>("ud")?);
Ok(())
}
#[test]
fn test_anyuserdata_from_lua() -> Result<()> {
let lua = Lua::new();
match lua.globals().get::<AnyUserData>("print") {
Err(err @ Error::FromLuaConversionError { .. }) => {
assert_eq!(err.to_string(), "error converting Lua function to userdata");
}
_ => panic!("expected `Error::FromLuaConversionError`"),
}
Ok(())
}
#[test]
fn test_error_conversion() -> Result<()> {
let lua = Lua::new();
match lua.convert::<Error>(Error::external("external error")) {
Ok(Error::ExternalError(msg)) => assert_eq!(msg.to_string(), "external error"),
res => panic!("expected `Error::ExternalError`, got {res:?}"),
}
match lua.convert::<Error>("abc") {
Ok(Error::RuntimeError(msg)) => assert_eq!(msg, "abc"),
res => panic!("expected `Error::RuntimeError`, got {res:?}"),
}
match lua.convert::<Error>(true) {
Ok(Error::RuntimeError(msg)) => assert_eq!(msg, "true"),
res => panic!("expected `Error::RuntimeError`, got {res:?}"),
}
match lua.convert::<Error>(lua.globals()) {
Ok(Error::RuntimeError(msg)) => assert!(msg.starts_with("table:")),
res => panic!("expected `Error::RuntimeError`, got {res:?}"),
}
Ok(())
}
#[test]
fn test_registry_value_into_lua() -> Result<()> {
let lua = Lua::new();
let s = lua.create_string("hello, world")?;
let r = lua.create_registry_value(&s)?;
let value1 = lua.pack(&r)?;
let value2 = lua.pack(r)?;
assert_eq!(value1.as_str().as_deref(), Some("hello, world"));
assert_eq!(value1.to_pointer(), value2.to_pointer());
let t = lua.create_table()?;
let r = lua.create_registry_value(&t)?;
let f = lua.create_function(|_, (t, k, v): (Table, Value, Value)| t.set(k, v))?;
f.call::<()>((&r, "hello", "world"))?;
f.call::<()>((r, "welcome", "to the jungle"))?;
assert_eq!(t.get::<String>("hello")?, "world");
assert_eq!(t.get::<String>("welcome")?, "to the jungle");
let r_nil = lua.create_registry_value(Value::Nil)?;
t.set("hello", &r_nil)?;
assert_eq!(t.get::<Value>("hello")?, Value::Nil);
let lua2 = Lua::new();
let r2 = lua2.create_registry_value("abc")?;
assert!(matches!(f.call::<()>(&r2), Err(Error::MismatchedRegistryKey)));
Ok(())
}
#[test]
fn test_registry_key_from_lua() -> Result<()> {
let lua = Lua::new();
let fkey = lua.load("function() return 1 end").eval::<RegistryKey>()?;
let f = lua.registry_value::<Function>(&fkey)?;
assert_eq!(f.call::<i32>(())?, 1);
Ok(())
}
#[test]
fn test_bool_into_lua() -> Result<()> {
let lua = Lua::new();
assert!(true.into_lua(&lua)?.is_boolean());
let table = lua.create_table()?;
table.set("b", true)?;
assert_eq!(true, table.get::<bool>("b")?);
Ok(())
}
#[test]
fn test_bool_from_lua() -> Result<()> {
let lua = Lua::new();
assert!(lua.globals().get::<bool>("print")?);
assert!(lua.convert::<bool>(123)?);
assert!(!lua.convert::<bool>(Value::Nil)?);
Ok(())
}
#[test]
fn test_integer_from_lua() -> Result<()> {
let lua = Lua::new();
let f = lua.create_function(|_, i: i32| Ok(i))?;
assert_eq!(f.call::<i32>(42)?, 42);
match f.call::<i32>(i64::MAX).err() {
Some(Error::CallbackError { cause, .. }) => match cause.as_ref() {
Error::BadArgument { cause, .. } => match cause.as_ref() {
Error::FromLuaConversionError { message, .. } => {
assert_eq!(message.as_ref().unwrap(), "out of range");
}
err => panic!("expected Error::FromLuaConversionError, got {err:?}"),
},
err => panic!("expected Error::BadArgument, got {err:?}"),
},
err => panic!("expected Error::CallbackError, got {err:?}"),
}
assert_eq!(f.call::<i32>("42")?, 42);
Ok(())
}
#[test]
fn test_float_from_lua() -> Result<()> {
let lua = Lua::new();
let f = lua.create_function(|_, f: f32| Ok(f))?;
assert_eq!(f.call::<f32>(42.0)?, 42.0);
let val = f.call::<f32>(f64::MAX)?;
assert!(val.is_infinite());
assert_eq!(f.call::<f32>("42.0")?, 42.0);
Ok(())
}
#[test]
fn test_conv_vec() -> Result<()> {
let lua = Lua::new();
let v = vec![1, 2, 3];
lua.globals().set("v", v.clone())?;
let v2: Vec<i32> = lua.globals().get("v")?;
assert_eq!(v, v2);
Ok(())
}
#[test]
fn test_conv_hashmap() -> Result<()> {
let lua = Lua::new();
let map = hashmap! {"hello".to_string() => "world".to_string()};
lua.globals().set("map", map.clone())?;
let map2: HashMap<String, String> = lua.globals().get("map")?;
assert_eq!(map, map2);
Ok(())
}
#[test]
fn test_conv_hashset() -> Result<()> {
let lua = Lua::new();
let set = hashset! {"hello".to_string(), "world".to_string()};
lua.globals().set("set", set.clone())?;
let set2: HashSet<String> = lua.globals().get("set")?;
assert_eq!(set, set2);
let set3 = lua.load(r#"{"a", "b", "c"}"#).eval::<HashSet<String>>()?;
assert_eq!(set3, hashset! { "a".into(), "b".into(), "c".into() });
Ok(())
}
#[test]
fn test_conv_btreemap() -> Result<()> {
let lua = Lua::new();
let map = btreemap! {"hello".to_string() => "world".to_string()};
lua.globals().set("map", map.clone())?;
let map2: BTreeMap<String, String> = lua.globals().get("map")?;
assert_eq!(map, map2);
Ok(())
}
#[test]
fn test_conv_btreeset() -> Result<()> {
let lua = Lua::new();
let set = btreeset! {"hello".to_string(), "world".to_string()};
lua.globals().set("set", set.clone())?;
let set2: BTreeSet<String> = lua.globals().get("set")?;
assert_eq!(set, set2);
let set3 = lua.load(r#"{"a", "b", "c"}"#).eval::<BTreeSet<String>>()?;
assert_eq!(set3, btreeset! { "a".into(), "b".into(), "c".into() });
Ok(())
}
#[test]
fn test_conv_cstring() -> Result<()> {
let lua = Lua::new();
let s = CString::new(b"hello".to_vec()).unwrap();
lua.globals().set("s", s.clone())?;
let s2: CString = lua.globals().get("s")?;
assert_eq!(s, s2);
let cs = CStr::from_bytes_with_nul(b"hello\0").unwrap();
lua.globals().set("cs", cs)?;
let cs2: CString = lua.globals().get("cs")?;
assert_eq!(cs, cs2.as_c_str());
Ok(())
}
#[test]
fn test_conv_cow() -> Result<()> {
let lua = Lua::new();
let s = Cow::from("hello");
lua.globals().set("s", s.clone())?;
let s2: String = lua.globals().get("s")?;
assert_eq!(s, s2);
Ok(())
}
#[test]
fn test_conv_boxed_str() -> Result<()> {
let lua = Lua::new();
let s = String::from("hello").into_boxed_str();
lua.globals().set("s", s.clone())?;
let s2: Box<str> = lua.globals().get("s")?;
assert_eq!(s, s2);
Ok(())
}
#[test]
fn test_conv_boxed_slice() -> Result<()> {
let lua = Lua::new();
let v = vec![1, 2, 3].into_boxed_slice();
lua.globals().set("v", v.clone())?;
let v2: Box<[i32]> = lua.globals().get("v")?;
assert_eq!(v, v2);
Ok(())
}
#[test]
fn test_conv_array() -> Result<()> {
let lua = Lua::new();
let v = [1, 2, 3];
lua.globals().set("v", v)?;
let v2: [i32; 3] = lua.globals().get("v")?;
assert_eq!(v, v2);
let v2 = lua.globals().get::<[i32; 4]>("v");
assert!(matches!(v2, Err(Error::FromLuaConversionError { .. })));
Ok(())
}
#[test]
fn test_bstring_from_lua() -> Result<()> {
let lua = Lua::new();
let s = lua.create_string("hello, world")?;
let bstr = lua.unpack::<BString>(Value::String(s))?;
assert_eq!(bstr, "hello, world");
let bstr = lua.unpack::<BString>(Value::Integer(123))?;
assert_eq!(bstr, "123");
let bstr = lua.unpack::<BString>(Value::Number(-123.55))?;
assert_eq!(bstr, "-123.55");
let f = lua.create_function(|_, bstr: BString| Ok(bstr))?;
let bstr = f.call::<BString>("hello, world")?;
assert_eq!(bstr, "hello, world");
let bstr = f.call::<BString>(-43.22)?;
assert_eq!(bstr, "-43.22");
Ok(())
}
#[cfg(feature = "luau")]
#[test]
fn test_bstring_from_lua_buffer() -> Result<()> {
let lua = Lua::new();
let buf = lua.create_buffer("hello, world")?;
let bstr = lua.convert::<BString>(buf)?;
assert_eq!(bstr, "hello, world");
let f = lua.create_function(|_, bstr: BString| Ok(bstr))?;
let buf = lua.create_buffer("hello, world")?;
let bstr = f.call::<BString>(buf)?;
assert_eq!(bstr, "hello, world");
Ok(())
}
#[test]
fn test_osstring_into_from_lua() -> Result<()> {
let lua = Lua::new();
let s = OsString::from("hello, world");
let v = lua.pack(s.as_os_str())?;
assert!(v.is_string());
assert_eq!(v.as_str().unwrap(), "hello, world");
let v = lua.pack(s)?;
assert!(v.is_string());
assert_eq!(v.as_str().unwrap(), "hello, world");
let s = lua.create_string("hello, world")?;
let bstr = lua.unpack::<OsString>(Value::String(s))?;
assert_eq!(bstr, "hello, world");
let bstr = lua.unpack::<OsString>(Value::Integer(123))?;
assert_eq!(bstr, "123");
let bstr = lua.unpack::<OsString>(Value::Number(-123.55))?;
assert_eq!(bstr, "-123.55");
Ok(())
}
#[test]
fn test_pathbuf_into_from_lua() -> Result<()> {
let lua = Lua::new();
let pb = PathBuf::from(env!("CARGO_TARGET_TMPDIR"));
let pb_str = pb.to_str().unwrap();
let v = lua.pack(pb.as_path())?;
assert!(v.is_string());
assert_eq!(v.as_str().unwrap(), pb_str);
let v = lua.pack(pb.clone())?;
assert!(v.is_string());
assert_eq!(v.as_str().unwrap(), pb_str);
let s = lua.create_string(pb_str)?;
let bstr = lua.unpack::<PathBuf>(Value::String(s))?;
assert_eq!(bstr, pb);
Ok(())
}
#[test]
fn test_option_into_from_lua() -> Result<()> {
let lua = Lua::new();
let v = Some(42);
let v2 = v.into_lua(&lua)?;
assert_eq!(v, v2.as_i32());
let f = lua.create_function(|_, v: Option<i32>| Ok(v))?;
assert_eq!(f.call::<Option<i32>>(Some(42))?, Some(42));
assert_eq!(f.call::<Option<i32>>(Option::<i32>::None)?, None);
assert_eq!(f.call::<Option<i32>>(())?, None);
Ok(())
}
#[test]
fn test_either_enum() -> Result<()> {
let mut either = Either::<_, String>::Left(42);
assert!(either.is_left());
assert_eq!(*either.as_ref().left().unwrap(), 42);
*either.as_mut().left().unwrap() = 44;
assert_eq!(*either.as_ref().left().unwrap(), 44);
assert_eq!(format!("{either}"), "44");
assert_eq!(either.right(), None);
either = Either::Right("hello".to_string());
assert!(either.is_right());
assert_eq!(*either.as_ref().right().unwrap(), "hello");
*either.as_mut().right().unwrap() = "world".to_string();
assert_eq!(*either.as_ref().right().unwrap(), "world");
assert_eq!(format!("{either}"), "world");
assert_eq!(either.left(), None);
Ok(())
}
#[test]
fn test_either_into_lua() -> Result<()> {
let lua = Lua::new();
let mut either = Either::<i32, &Table>::Left(42);
assert_eq!(either.into_lua(&lua)?, Value::Integer(42));
let t = lua.create_table()?;
either = Either::Right(&t);
assert!(matches!(either.into_lua(&lua)?, Value::Table(_)));
let f =
lua.create_function(|_, either: Either<i32, Table>| either.right().unwrap().set("hello", "world"))?;
let t = lua.create_table()?;
either = Either::Right(&t);
f.call::<()>(either)?;
assert_eq!(t.get::<String>("hello")?, "world");
let f = lua.create_function(|_, either: Either<i32, Table>| Ok(either.left().unwrap() + 1))?;
either = Either::Left(42);
assert_eq!(f.call::<i32>(either)?, 43);
Ok(())
}
#[test]
fn test_either_from_lua() -> Result<()> {
let lua = Lua::new();
let mut either = lua.unpack::<Either<i32, Table>>(Value::Integer(42))?;
assert!(either.is_left());
assert_eq!(*either.as_ref().left().unwrap(), 42);
let t = lua.create_table()?;
either = lua.unpack::<Either<i32, Table>>(Value::Table(t.clone()))?;
assert!(either.is_right());
assert_eq!(either.as_ref().right().unwrap(), &t);
match lua.unpack::<Either<i32, Table>>(Value::String(lua.create_string("abc")?)) {
Err(Error::FromLuaConversionError { to, .. }) => assert_eq!(to, "Either<i32, Table>"),
_ => panic!("expected `Error::FromLuaConversionError`"),
}
let f = lua.create_function(|_, either: Either<i32, Table>| Ok(either))?;
let either = f.call::<Either<i32, Table>>(42)?;
assert!(either.is_left());
assert_eq!(*either.as_ref().left().unwrap(), 42);
let either = f.call::<Either<i32, Table>>([5; 5])?;
assert!(either.is_right());
assert_eq!(either.as_ref().right().unwrap(), &[5; 5]);
match f.call::<Value>("hello") {
Ok(_) => panic!("expected error, got Ok"),
Err(ref err @ Error::CallbackError { ref cause, .. }) => {
match cause.as_ref() {
Error::BadArgument { cause, .. } => match cause.as_ref() {
Error::FromLuaConversionError { to, .. } => {
assert_eq!(to, "Either<i32, Table>")
}
err => panic!("expected `Error::FromLuaConversionError`, got {err:?}"),
},
err => panic!("expected `Error::BadArgument`, got {err:?}"),
}
assert!(err
.to_string()
.starts_with("bad argument #1: error converting Lua string to Either<i32, Table>"),);
}
err => panic!("expected `Error::CallbackError`, got {err:?}"),
}
Ok(())
}
#[test]
fn test_char_into_lua() -> Result<()> {
let lua = Lua::new();
let v = '🦀';
let v2 = v.into_lua(&lua)?;
assert_eq!(Some(v.to_string()), v2.as_string_lossy());
Ok(())
}
#[test]
fn test_char_from_lua() -> Result<()> {
let lua = Lua::new();
assert_eq!(lua.convert::<char>("A")?, 'A');
assert_eq!(lua.convert::<char>(65)?, 'A');
assert_eq!(lua.convert::<char>(128175)?, '💯');
assert!(lua
.convert::<char>(5456324)
.is_err_and(|e| e.to_string().contains("integer out of range")));
assert!(lua
.convert::<char>("hello")
.is_err_and(|e| e.to_string().contains("expected string to have exactly one char")));
assert!(lua
.convert::<char>(HashMap::<String, String>::new())
.is_err_and(|e| e.to_string().contains("expected string or integer")));
Ok(())
}