use std::error::Error as _;
use std::{fmt, io};
use mlua::{Error, ErrorContext, Lua, Result};
#[test]
fn test_error_context() -> Result<()> {
let lua = Lua::new();
let func =
lua.create_function(|_, ()| Err::<(), _>(Error::runtime("runtime error")).context("some context"))?;
lua.globals().set("func", func)?;
let msg = lua
.load("local _, err = pcall(func); return tostring(err)")
.eval::<String>()?;
assert!(msg.contains("some context"));
assert!(msg.contains("runtime error"));
let func2 = lua.create_function(|lua, ()| {
lua.globals()
.get::<String>("nonextant")
.with_context(|_| "failed to find global")
})?;
lua.globals().set("func2", func2)?;
let msg2 = lua
.load("local _, err = pcall(func2); return tostring(err)")
.eval::<String>()?;
assert!(msg2.contains("failed to find global"));
assert!(msg2.contains("error converting Lua nil to String"));
let func3 = lua.create_function(|_, ()| {
Err::<(), _>(Error::external(io::Error::new(io::ErrorKind::Other, "other")))
.context("some context")
.context("some new context")
})?;
let err = func3.call::<()>(()).unwrap_err();
let err = err.parent().unwrap();
assert!(!err.to_string().contains("some context"));
assert!(err.to_string().contains("some new context"));
assert!(err.downcast_ref::<io::Error>().is_some());
assert!(err.downcast_ref::<fmt::Error>().is_none());
Ok(())
}
#[test]
fn test_error_chain() -> Result<()> {
let lua = Lua::new();
let io_err = io::Error::new(io::ErrorKind::Other, "other");
assert_eq!(Error::external(io_err).chain().count(), 1);
let func = lua.create_function(|_, ()| {
let err = Error::external(io::Error::new(io::ErrorKind::Other, "other")).context("io error");
Err::<(), _>(err)
})?;
let err = func.call::<()>(()).unwrap_err();
assert_eq!(err.chain().count(), 3);
for (i, err) in err.chain().enumerate() {
match i {
0 => assert!(matches!(err.downcast_ref(), Some(Error::CallbackError { .. }))),
1 => assert!(matches!(err.downcast_ref(), Some(Error::WithContext { .. }))),
2 => assert!(matches!(err.downcast_ref(), Some(io::Error { .. }))),
_ => unreachable!(),
}
}
let err = err.parent().unwrap();
assert!(err.source().is_none()); assert!(err.to_string().contains("io error"));
assert!(err.to_string().contains("other"));
Ok(())
}
#[cfg(feature = "anyhow")]
#[test]
fn test_error_anyhow() -> Result<()> {
use mlua::IntoLua;
let lua = Lua::new();
let err = anyhow::Error::msg("anyhow error");
let val = err.into_lua(&lua)?;
assert!(val.is_error());
assert_eq!(val.as_error().unwrap().to_string(), "anyhow error");
let err = Error::runtime("runtime error");
let err = anyhow::Error::new(err);
let err = err.into_lua(&lua)?;
assert!(err.is_error());
let err = err.as_error().unwrap();
assert!(matches!(err, Error::RuntimeError(msg) if msg == "runtime error"));
Ok(())
}