crates.io docs.rs ![Build Status] ![Coverage Status]

[ChangeLog] | [FAQ] | [Known issues]

Ergonomic, efficient and Zero-cost rust bindings to Lua5.4

Features

Limits

Examples

See builtin bindings tests

Usage

Feature flags

Basic

First, add ezlua to your dependencies in Cargo.toml toml [dependencies] ezlua = { version = '0.3' }

Then, use ezlua in rust, the code framework like this ```rust use ezlua::prelude::*;

fn main() -> LuaResult<()> { // create a lua VM let lua = Lua::withopenlibs();

// load your lua script and execute it
lua.do_string(r#"function add(a, b) return a + b end"#, None)?;

// get function named add from lua global table
let add = lua.global().get("add")?;

// call add function and get its result
let result = add.pcall::<_, u32>((111, 222))?;
assert_eq!(result, 333);

// ... for the following code

Ok(())

} ```

Bind your function

Of course, you can provide your rust function to lua via ezlua binding, and it's very simple, like this rust lua.global().set("add", lua.new_closure(|a: u32, b: u32| a + b)?)?; lua.do_string("assert(add(111, 222) == 333)", None)?;

And you can bind exists function easily ```rust let string: LuaTable = lua.global().get("string")?.tryinto()?; string.setclosure("trim", str::trim)?; string.setclosure("trimstart", str::trimstart)?; string.setclosure("trimend", str::trimend)?;

let os: LuaTable = lua.global().get("os")?.tryinto()?; os.setclosure("mkdir", std::fs::createdir::<&str>)?; os.setclosure("mkdirs", std::fs::createdirall::<&str>)?; os.setclosure("rmdir", std::fs::removedir::<&str>)?; os.setclosure("chdir", std::env::setcurrentdir::<&str>)?; os.setclosure("getcwd", std::env::currentdir)?; os.setclosure("getexe", std::env::current_exe)?; ```

Bind your type

Implement ToLua trait for your type, and then you can pass it to lua

```rust

[derive(Debug, Default)]

struct Config { name: String, path: String, timeout: u64, // ... }

impl ToLua for Config { fn tolua<'a>(self, lua: &'a LuaState) -> LuaResult> { let conf = lua.newtable()?; conf.set("name", self.name)?; conf.set("path", self.path)?; conf.set("timeout", self.timeout)?; conf.to_lua(lua) } }

lua.global().setclosure("defaultconfig", Config::default)?; ```

Simply bindings via serde

Continuing with the example above, you can simply the binding code via serde

```rust use serde::{Deserialize, Serialize}; use ezlua::serde::SerdeValue;

[derive(Debug, Default, Deserialize, Serialize)]

struct Config { name: String, path: String, timeout: u64, // ... }

// You can use impltoluaasserde macro to simply this after version v0.3.1 // ezlua::impltoluaasserde!(Config); impl ToLua for Config { fn tolua<'a>(self, lua: &'a LuaState) -> LuaResult> { SerdeValue(self).tolua(lua) } }

// You can use implfromluaasserde macro to simply this after version v0.3.1 // ezlua::implfromluaasserde!(Config); impl FromLua<'> for Config { fn fromlua(lua: &LuaState, val: ValRef) -> Option { SerdeValue::::from_lua(lua, val).map(|s| s.0) } }

lua.global().set("DEFAULTCONFIG", SerdeValue(Config::default()))?; lua.global() .setclosure("set_config", |config: Config| { // ... set your config })?; ```

Bind custom object (userdata)

ezlua's userdata binding mechanism is powerful, the following code comes from std bindings

```rust use std::{fs::Metadata, path::*};

impl UserData for Metadata { fn getter(fields: UserdataRegistry) -> Result<()> { fields.setclosure("size", Self::len)?; fields.setclosure("modified", Self::modified)?; fields.setclosure("created", Self::created)?; fields.setclosure("accessed", Self::accessed)?; fields.set_closure("readonly", |this: &Self| this.permissions().readonly())?;

    Ok(())
}

fn methods(mt: UserdataRegistry<Self>) -> Result<()> {
    mt.set_closure("len", Self::len)?;
    mt.set_closure("is_dir", Self::is_dir)?;
    mt.set_closure("is_file", Self::is_file)?;
    mt.set_closure("is_symlink", Self::is_symlink)?;

    Ok(())
}

} ```

Types which impls the UserData trait, ezlua also impls ToLua for it, and impls FromLua for its reference rust lua.global().set("path_metadata", Path::metadata)?;

Defaultly, types binded as userdata is immutable, if you need mutable reference, you can specific a UserData::Trans type, and there is a builtin impl that is RefCell, so the mutable binding impls looks like this ```rust use core::cell::RefCell; use std::process::{Child, Command, ExitStatus, Stdio};

impl UserData for Child { type Trans = RefCell;

fn getter(fields: UserdataRegistry<Self>) -> LuaResult<()> {
    fields.add("id", Self::id)?;

    Ok(())
}

fn methods(mt: UserdataRegistry<Self>) -> Result<()> {
    mt.add_mut("kill", Self::kill)?;
    mt.add_mut("wait", Self::wait)?;

    mt.add_mut("try_wait", |this: &mut Self| {
        this.try_wait().ok().flatten().ok_or(())
    })?;
}

} ```

Under normal circumstances, you need only impl the getter/setter/methods methods when impl the UserData trait, which allows you "read property"/"write property"/"call method" through the userdata value, but also ezlua provides more powerful features for UserData, such as "uservalue access" and "userdata cache".

In order to enable the "uservalue access" feature for an userdata type, just needs to specify const INDEX_USERVALUE: bool = true ```rust struct Test { a: i32, }

impl UserData for Test { type Trans = RefCell;

const INDEX_USERVALUE: bool = true;

fn methods(mt: UserdataRegistry<Self>) -> LuaResult<()> {
    mt.set_closure("inc", |mut this: RefMut<Self>| this.a += 1)?;
    Ok(())
}

}

let uv = lua.newval(Test { a: 0 })?; lua.global().set("uv", uv)?; lua.dostring("uv.abc = 3; assert(uv.abc == 3)", None)?; lua.do_string("assert(debug.getuservalue(uv).abc == 3)", None)?; ```

In order to enable the "userdata cache" feature for an userdata type, you should impl the UserData::key_to_cache method, which returns a pointer, as a lightuserdata key in the cache table in lua. ```rust

[derive(derive_more::Deref, Clone)]

struct RcTest(Rc);

impl UserData for RcTest { fn keytocache(&self) -> *const () { self.as_ref() as *const _ as _ }

fn getter(fields: UserdataRegistry<Self>) -> LuaResult<()> {
    fields.set_closure("a", |this: &Self| this.a)?;
    Ok(())
}

fn methods(_: UserdataRegistry<Self>) -> LuaResult<()> {
    Ok(())
}

}

let test = RcTest(Test { a: 123 }.into()); lua.global().set("uv", test.clone())?; // when converting an UserData type to lua value, ezlua will first use the userdata in the cache table if existing, // otherwise, create a new userdata and insert it to the cache table, so the "uv" and "uv1" will refer to the same userdata object lua.global().set("uv1", test.clone())?; lua.dostring("print(uv, uv1)", None)?; lua.dostring("assert(uv == uv1)", None)?; ```

Register your own module

To register a lua module, you can provide a rust function return a lua table via LuaState::register_module method ```rust lua.registermodule("json", ezlua::binding::json::open, false)?; lua.registermodule("path", |lua| { let t = lua.new_table()?;

t.set_closure("dirname", Path::parent)?;
t.set_closure("exists", Path::exists)?;
t.set_closure("abspath", std::fs::canonicalize::<&str>)?;
t.set_closure("isabs", Path::is_absolute)?;
t.set_closure("isdir", Path::is_dir)?;
t.set_closure("isfile", Path::is_file)?;
t.set_closure("issymlink", Path::is_symlink)?;

return Ok(t);

}, false)?; ```

And then use them in lua ```lua local json = require 'json' local path = require 'path'

local dir = path.abspath('.') assert(json.load(json.dump(dir)) == dir) ```

Multiple thread usage

To use multiple thread feature in lua, you need to specify the thread feature in Cargo.toml, and patch the lua-src crate with ezlua's custom ```toml [dependencies] ezlua = { version = '0.3', features = ['thread'] }

[patch.crates-io] lua-src = { git = "https://github.com/metaworm/lua-src-rs" } ```

And then, register the thread module for lua rust lua.register_module("thread", ezlua::binding::std::thread::init, true)?;

And then, use it in lua ```lua local thread = require 'thread' local threads = {} local tt = { n = 0 } local count = 64 for i = 1, count do threads[i] = thread.spawn(function() tt.n = tt.n + 1 -- print(tt.n) end) end

for i, t in ipairs(threads) do t:join() print('#' .. i .. ' finished') end assert(tt.n == count) ```

In addition, you can also start a new thread with the same lua VM ```rust let co = Coroutine::empty(&lua); std::thread::spawn(move || { let print = co.global().get("print")?; print.pcall_void("running lua in another thread")?;

LuaResult::Ok(())

}) .join() .unwrap(); ```

Module mode

In a module mode ezlua allows to create a compiled Lua module that can be loaded from Lua code using require.

First, disable the default vendored feature, and keep std feature only, and config your crate as a cdylib in Cargo.toml : ``` toml [dependencies] ezlua = {version = '0.3', default-features = false, features = ['std']}

[lib] crate-type = ['cdylib'] ```

Then, export your luaopen_ function by using ezlua::lua_module! macro, where the first argument is luaopen_<Your module name> ``` rust use ezlua::prelude::*;

ezlua::luamodule!(luaopenezluamod, |lua| { let module = lua.new_table()?;

module.set("_VERSION", "0.1.0")?;
// ... else module functions

return Ok(module);

}); ```

Internal design

TODO