An easy way to transparently create serializable types that are fully legible from lua as plain lua types.
The to_lua and from_lua functions are the easiest way to serialize any
datatype that implements serde's serialize and deserialize traits,
respectively.
extern crate hlua;
extern crate serde;
#[macro_use] extern crate serde_derive;
extern crate serde_hlua;
#[derive(Debug, Deserialize, Serialize, PartialEq)]
struct Point {
    x: f32,
    y: f32
}
fn main() {
    let mut lua = hlua::Lua::new();
    let my_point = Point { x: 3.0, y: 4.0 };
    lua.set("my_value", serde_hlua::to_lua(&my_point).unwrap());
    let retrieved_point: Point = serde_hlua::from_lua(
        lua.get::<hlua::AnyLuaValue, _>("my_value").unwrap()
    ).unwrap();
    assert_eq!(my_point, retrieved_point);
}The serialization implemented by this crate doesn't use lua's userdata system; all types are serialized to plain lua types.
let lua_point: Point = serde_hlua::from_lua(lua.execute::<hlua::AnyLuaValue>("
    return { x = 3.5, y = 7 }
").unwrap()).unwrap();
assert_eq!(lua_point, Point { x: 3.5, y: 7.0 });Typechecking is performed automatically, so an invalid lua value can't deserialize to a type that doesn't support it.
assert!(serde_hlua::from_lua::<Point>(lua.execute::<hlua::AnyLuaValue>("
    return { x = 3.5, y = 'wrong type!' }
").unwrap()).is_err());- 
Mappings and sequences to unit structs are not injective. Lua will convert all of [],[()], and[(), ()]to the empty array{}. This is because lua considers setting a value to nil to be equivalent to erasing it from a table, and so the table{a = nil, b = nil}is the same as the table{}. Because of this, round-tripping a structure that uses unit structs as values will be lossy.However, this is not always an issue; Marking a unit-struct field with #[serde(default)]will allow serde to fill it in even when it's not present, so deserializingstruct {a: (), b: ()}will succeed when given the lua table{}.Unit enum variants are also encoded losslessly, as they are encoded as the name of the variant as a string. 
- 
Integer values are only serialized and deserialized if they can do so losslessly. std::i64::MINcan be losslessly encoded, butstd::i64::MIN + 1cannot, as it is rounded to a different value.f32values are always encoded intof64, as otherwisef64values with too many significant digits (such as1/3) would not encode. They are cast using rust'sasoperator.