mirror of
https://github.com/lune-org/mlua-luau-scheduler.git
synced 2025-04-06 19:40:55 +01:00
Thank you clippy
This commit is contained in:
parent
577fa2e272
commit
fa4e6730ea
11 changed files with 86 additions and 36 deletions
17
Cargo.toml
17
Cargo.toml
|
@ -2,9 +2,12 @@
|
||||||
name = "mlua-luau-runtime"
|
name = "mlua-luau-runtime"
|
||||||
version = "0.0.0"
|
version = "0.0.0"
|
||||||
edition = "2021"
|
edition = "2021"
|
||||||
|
license = "MPL-2.0"
|
||||||
[lib]
|
repository = "https://github.com/lune-org/mlua-luau-runtime"
|
||||||
path = "lib/lib.rs"
|
description = "Luau-based async runtime, using mlua and async-executor"
|
||||||
|
readme = "README.md"
|
||||||
|
keywords = ["async", "luau", "runtime"]
|
||||||
|
categories = ["async"]
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
async-executor = "1.8"
|
async-executor = "1.8"
|
||||||
|
@ -23,6 +26,14 @@ mlua = { version = "0.9.5", features = [
|
||||||
async-fs = "2.1"
|
async-fs = "2.1"
|
||||||
async-io = "2.3"
|
async-io = "2.3"
|
||||||
|
|
||||||
|
[lints.clippy]
|
||||||
|
all = { level = "deny", priority = -3 }
|
||||||
|
cargo = { level = "warn", priority = -2 }
|
||||||
|
pedantic = { level = "warn", priority = -1 }
|
||||||
|
|
||||||
|
[lib]
|
||||||
|
path = "lib/lib.rs"
|
||||||
|
|
||||||
[[example]]
|
[[example]]
|
||||||
name = "basic_sleep"
|
name = "basic_sleep"
|
||||||
test = true
|
test = true
|
||||||
|
|
|
@ -16,7 +16,7 @@
|
||||||
|
|
||||||
<br/>
|
<br/>
|
||||||
|
|
||||||
Luau-based async runtime for [`mlua`](https://crates.io/crates/mlua), built on top of [`async-executor`](https://crates.io/crates/async-executor).
|
Luau-based async runtime, using [`mlua`](https://crates.io/crates/mlua) and [`async-executor`](https://crates.io/crates/async-executor).
|
||||||
|
|
||||||
## Example Usage
|
## Example Usage
|
||||||
|
|
||||||
|
|
|
@ -1,9 +1,11 @@
|
||||||
|
#![allow(clippy::missing_errors_doc)]
|
||||||
|
|
||||||
use std::time::{Duration, Instant};
|
use std::time::{Duration, Instant};
|
||||||
|
|
||||||
use async_io::{block_on, Timer};
|
use async_io::{block_on, Timer};
|
||||||
|
|
||||||
use mlua::prelude::*;
|
use mlua::prelude::*;
|
||||||
use mlua_luau_runtime::*;
|
use mlua_luau_runtime::Runtime;
|
||||||
|
|
||||||
const MAIN_SCRIPT: &str = include_str!("./lua/basic_sleep.luau");
|
const MAIN_SCRIPT: &str = include_str!("./lua/basic_sleep.luau");
|
||||||
|
|
||||||
|
@ -20,7 +22,7 @@ pub fn main() -> LuaResult<()> {
|
||||||
)?;
|
)?;
|
||||||
|
|
||||||
// Load the main script into a runtime
|
// Load the main script into a runtime
|
||||||
let rt = Runtime::new(&lua)?;
|
let rt = Runtime::new(&lua);
|
||||||
let main = lua.load(MAIN_SCRIPT);
|
let main = lua.load(MAIN_SCRIPT);
|
||||||
rt.spawn_thread(main, ())?;
|
rt.spawn_thread(main, ())?;
|
||||||
|
|
||||||
|
|
|
@ -1,10 +1,12 @@
|
||||||
|
#![allow(clippy::missing_errors_doc)]
|
||||||
|
|
||||||
use std::io::ErrorKind;
|
use std::io::ErrorKind;
|
||||||
|
|
||||||
use async_fs::read_to_string;
|
use async_fs::read_to_string;
|
||||||
use async_io::block_on;
|
use async_io::block_on;
|
||||||
|
|
||||||
use mlua::prelude::*;
|
use mlua::prelude::*;
|
||||||
use mlua_luau_runtime::*;
|
use mlua_luau_runtime::{LuaSpawnExt, Runtime};
|
||||||
|
|
||||||
const MAIN_SCRIPT: &str = include_str!("./lua/basic_spawn.luau");
|
const MAIN_SCRIPT: &str = include_str!("./lua/basic_spawn.luau");
|
||||||
|
|
||||||
|
@ -27,7 +29,7 @@ pub fn main() -> LuaResult<()> {
|
||||||
)?;
|
)?;
|
||||||
|
|
||||||
// Load the main script into a runtime
|
// Load the main script into a runtime
|
||||||
let rt = Runtime::new(&lua)?;
|
let rt = Runtime::new(&lua);
|
||||||
let main = lua.load(MAIN_SCRIPT);
|
let main = lua.load(MAIN_SCRIPT);
|
||||||
rt.spawn_thread(main, ())?;
|
rt.spawn_thread(main, ())?;
|
||||||
|
|
||||||
|
|
|
@ -1,5 +1,7 @@
|
||||||
|
#![allow(clippy::missing_errors_doc)]
|
||||||
|
|
||||||
use mlua::prelude::*;
|
use mlua::prelude::*;
|
||||||
use mlua_luau_runtime::*;
|
use mlua_luau_runtime::Runtime;
|
||||||
|
|
||||||
use async_io::block_on;
|
use async_io::block_on;
|
||||||
|
|
||||||
|
@ -10,7 +12,7 @@ pub fn main() -> LuaResult<()> {
|
||||||
let lua = Lua::new();
|
let lua = Lua::new();
|
||||||
|
|
||||||
// Create a new runtime with custom callbacks
|
// Create a new runtime with custom callbacks
|
||||||
let rt = Runtime::new(&lua)?;
|
let rt = Runtime::new(&lua);
|
||||||
rt.set_error_callback(|e| {
|
rt.set_error_callback(|e| {
|
||||||
println!(
|
println!(
|
||||||
"Captured error from Lua!\n{}\n{e}\n{}",
|
"Captured error from Lua!\n{}\n{e}\n{}",
|
||||||
|
|
|
@ -1,9 +1,11 @@
|
||||||
|
#![allow(clippy::missing_errors_doc)]
|
||||||
|
|
||||||
use std::time::Duration;
|
use std::time::Duration;
|
||||||
|
|
||||||
use async_io::{block_on, Timer};
|
use async_io::{block_on, Timer};
|
||||||
|
|
||||||
use mlua::prelude::*;
|
use mlua::prelude::*;
|
||||||
use mlua_luau_runtime::*;
|
use mlua_luau_runtime::Runtime;
|
||||||
|
|
||||||
const MAIN_SCRIPT: &str = include_str!("./lua/lots_of_threads.luau");
|
const MAIN_SCRIPT: &str = include_str!("./lua/lots_of_threads.luau");
|
||||||
|
|
||||||
|
@ -12,7 +14,7 @@ const ONE_NANOSECOND: Duration = Duration::from_nanos(1);
|
||||||
pub fn main() -> LuaResult<()> {
|
pub fn main() -> LuaResult<()> {
|
||||||
// Set up persistent Lua environment
|
// Set up persistent Lua environment
|
||||||
let lua = Lua::new();
|
let lua = Lua::new();
|
||||||
let rt = Runtime::new(&lua)?;
|
let rt = Runtime::new(&lua);
|
||||||
|
|
||||||
lua.globals().set("spawn", rt.create_spawn_function()?)?;
|
lua.globals().set("spawn", rt.create_spawn_function()?)?;
|
||||||
lua.globals().set(
|
lua.globals().set(
|
||||||
|
|
|
@ -1,16 +1,18 @@
|
||||||
|
#![allow(clippy::missing_errors_doc)]
|
||||||
|
|
||||||
use std::time::{Duration, Instant};
|
use std::time::{Duration, Instant};
|
||||||
|
|
||||||
use async_io::{block_on, Timer};
|
use async_io::{block_on, Timer};
|
||||||
|
|
||||||
use mlua::prelude::*;
|
use mlua::prelude::*;
|
||||||
use mlua_luau_runtime::*;
|
use mlua_luau_runtime::Runtime;
|
||||||
|
|
||||||
const MAIN_SCRIPT: &str = include_str!("./lua/scheduler_ordering.luau");
|
const MAIN_SCRIPT: &str = include_str!("./lua/scheduler_ordering.luau");
|
||||||
|
|
||||||
pub fn main() -> LuaResult<()> {
|
pub fn main() -> LuaResult<()> {
|
||||||
// Set up persistent Lua environment
|
// Set up persistent Lua environment
|
||||||
let lua = Lua::new();
|
let lua = Lua::new();
|
||||||
let rt = Runtime::new(&lua)?;
|
let rt = Runtime::new(&lua);
|
||||||
|
|
||||||
lua.globals().set("spawn", rt.create_spawn_function()?)?;
|
lua.globals().set("spawn", rt.create_spawn_function()?)?;
|
||||||
lua.globals().set("defer", rt.create_defer_function()?)?;
|
lua.globals().set("defer", rt.create_defer_function()?)?;
|
||||||
|
|
|
@ -31,6 +31,7 @@ impl ThreadErrorCallback {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[allow(clippy::needless_pass_by_value)]
|
||||||
fn default_error_callback(e: LuaError) {
|
fn default_error_callback(e: LuaError) {
|
||||||
eprintln!("{e}");
|
eprintln!("{e}");
|
||||||
}
|
}
|
||||||
|
|
26
lib/queue.rs
26
lib/queue.rs
|
@ -6,8 +6,6 @@ use mlua::prelude::*;
|
||||||
|
|
||||||
use crate::IntoLuaThread;
|
use crate::IntoLuaThread;
|
||||||
|
|
||||||
const ERR_OOM: &str = "out of memory";
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
Queue for storing [`LuaThread`]s with associated arguments.
|
Queue for storing [`LuaThread`]s with associated arguments.
|
||||||
|
|
||||||
|
@ -15,7 +13,7 @@ const ERR_OOM: &str = "out of memory";
|
||||||
well as listening for new items being pushed to the queue.
|
well as listening for new items being pushed to the queue.
|
||||||
*/
|
*/
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug, Clone)]
|
||||||
pub struct ThreadQueue {
|
pub(crate) struct ThreadQueue {
|
||||||
queue: Arc<ConcurrentQueue<ThreadWithArgs>>,
|
queue: Arc<ConcurrentQueue<ThreadWithArgs>>,
|
||||||
event: Arc<Event>,
|
event: Arc<Event>,
|
||||||
}
|
}
|
||||||
|
@ -35,9 +33,9 @@ impl ThreadQueue {
|
||||||
) -> LuaResult<()> {
|
) -> LuaResult<()> {
|
||||||
let thread = thread.into_lua_thread(lua)?;
|
let thread = thread.into_lua_thread(lua)?;
|
||||||
let args = args.into_lua_multi(lua)?;
|
let args = args.into_lua_multi(lua)?;
|
||||||
let stored = ThreadWithArgs::new(lua, thread, args);
|
let stored = ThreadWithArgs::new(lua, thread, args)?;
|
||||||
|
|
||||||
self.queue.push(stored).unwrap();
|
self.queue.push(stored).into_lua_err()?;
|
||||||
self.event.notify(usize::MAX);
|
self.event.notify(usize::MAX);
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
|
@ -61,7 +59,7 @@ impl ThreadQueue {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
Representation of a [`LuaThread`] with associated arguments currently stored in the Lua registry.
|
Representation of a [`LuaThread`] with its associated arguments currently stored in the Lua registry.
|
||||||
*/
|
*/
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
struct ThreadWithArgs {
|
struct ThreadWithArgs {
|
||||||
|
@ -70,19 +68,23 @@ struct ThreadWithArgs {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ThreadWithArgs {
|
impl ThreadWithArgs {
|
||||||
pub fn new<'lua>(lua: &'lua Lua, thread: LuaThread<'lua>, args: LuaMultiValue<'lua>) -> Self {
|
fn new<'lua>(
|
||||||
|
lua: &'lua Lua,
|
||||||
|
thread: LuaThread<'lua>,
|
||||||
|
args: LuaMultiValue<'lua>,
|
||||||
|
) -> LuaResult<Self> {
|
||||||
let argsv = args.into_vec();
|
let argsv = args.into_vec();
|
||||||
|
|
||||||
let key_thread = lua.create_registry_value(thread).expect(ERR_OOM);
|
let key_thread = lua.create_registry_value(thread)?;
|
||||||
let key_args = lua.create_registry_value(argsv).expect(ERR_OOM);
|
let key_args = lua.create_registry_value(argsv)?;
|
||||||
|
|
||||||
Self {
|
Ok(Self {
|
||||||
key_thread,
|
key_thread,
|
||||||
key_args,
|
key_args,
|
||||||
}
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn into_inner(self, lua: &Lua) -> (LuaThread<'_>, LuaMultiValue<'_>) {
|
fn into_inner(self, lua: &Lua) -> (LuaThread<'_>, LuaMultiValue<'_>) {
|
||||||
let thread = lua.registry_value(&self.key_thread).unwrap();
|
let thread = lua.registry_value(&self.key_thread).unwrap();
|
||||||
let argsv = lua.registry_value(&self.key_args).unwrap();
|
let argsv = lua.registry_value(&self.key_args).unwrap();
|
||||||
|
|
||||||
|
|
|
@ -23,17 +23,18 @@ impl<'lua> Runtime<'lua> {
|
||||||
|
|
||||||
This runtime will have a default error callback that prints errors to stderr.
|
This runtime will have a default error callback that prints errors to stderr.
|
||||||
*/
|
*/
|
||||||
pub fn new(lua: &'lua Lua) -> LuaResult<Runtime<'lua>> {
|
#[must_use]
|
||||||
|
pub fn new(lua: &'lua Lua) -> Runtime<'lua> {
|
||||||
let queue_spawn = ThreadQueue::new();
|
let queue_spawn = ThreadQueue::new();
|
||||||
let queue_defer = ThreadQueue::new();
|
let queue_defer = ThreadQueue::new();
|
||||||
let error_callback = ThreadErrorCallback::default();
|
let error_callback = ThreadErrorCallback::default();
|
||||||
|
|
||||||
Ok(Runtime {
|
Runtime {
|
||||||
lua,
|
lua,
|
||||||
queue_spawn,
|
queue_spawn,
|
||||||
queue_defer,
|
queue_defer,
|
||||||
error_callback,
|
error_callback,
|
||||||
})
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -60,6 +61,10 @@ impl<'lua> Runtime<'lua> {
|
||||||
Spawns a chunk / function / thread onto the runtime queue.
|
Spawns a chunk / function / thread onto the runtime queue.
|
||||||
|
|
||||||
Threads are guaranteed to be resumed in the order that they were pushed to the queue.
|
Threads are guaranteed to be resumed in the order that they were pushed to the queue.
|
||||||
|
|
||||||
|
# Errors
|
||||||
|
|
||||||
|
Errors when out of memory.
|
||||||
*/
|
*/
|
||||||
pub fn spawn_thread(
|
pub fn spawn_thread(
|
||||||
&self,
|
&self,
|
||||||
|
@ -75,6 +80,10 @@ impl<'lua> Runtime<'lua> {
|
||||||
Deferred threads are guaranteed to run after all spawned threads either yield or complete.
|
Deferred threads are guaranteed to run after all spawned threads either yield or complete.
|
||||||
|
|
||||||
Threads are guaranteed to be resumed in the order that they were pushed to the queue.
|
Threads are guaranteed to be resumed in the order that they were pushed to the queue.
|
||||||
|
|
||||||
|
# Errors
|
||||||
|
|
||||||
|
Errors when out of memory.
|
||||||
*/
|
*/
|
||||||
pub fn defer_thread(
|
pub fn defer_thread(
|
||||||
&self,
|
&self,
|
||||||
|
@ -88,6 +97,10 @@ impl<'lua> Runtime<'lua> {
|
||||||
Creates a Lua function that can be used to spawn threads / functions onto the runtime queue.
|
Creates a Lua function that can be used to spawn threads / functions onto the runtime queue.
|
||||||
|
|
||||||
The function takes a thread or function as the first argument, and any variadic arguments as the rest.
|
The function takes a thread or function as the first argument, and any variadic arguments as the rest.
|
||||||
|
|
||||||
|
# Errors
|
||||||
|
|
||||||
|
Errors when out of memory.
|
||||||
*/
|
*/
|
||||||
pub fn create_spawn_function(&self) -> LuaResult<LuaFunction<'lua>> {
|
pub fn create_spawn_function(&self) -> LuaResult<LuaFunction<'lua>> {
|
||||||
let error_callback = self.error_callback.clone();
|
let error_callback = self.error_callback.clone();
|
||||||
|
@ -123,6 +136,10 @@ impl<'lua> Runtime<'lua> {
|
||||||
The function takes a thread or function as the first argument, and any variadic arguments as the rest.
|
The function takes a thread or function as the first argument, and any variadic arguments as the rest.
|
||||||
|
|
||||||
Deferred threads are guaranteed to run after all spawned threads either yield or complete.
|
Deferred threads are guaranteed to run after all spawned threads either yield or complete.
|
||||||
|
|
||||||
|
# Errors
|
||||||
|
|
||||||
|
Errors when out of memory.
|
||||||
*/
|
*/
|
||||||
pub fn create_defer_function(&self) -> LuaResult<LuaFunction<'lua>> {
|
pub fn create_defer_function(&self) -> LuaResult<LuaFunction<'lua>> {
|
||||||
let defer_queue = self.queue_defer.clone();
|
let defer_queue = self.queue_defer.clone();
|
||||||
|
@ -142,6 +159,10 @@ impl<'lua> Runtime<'lua> {
|
||||||
|
|
||||||
Note that the given Lua state must be the same one that was
|
Note that the given Lua state must be the same one that was
|
||||||
used to create this runtime, otherwise this method will panic.
|
used to create this runtime, otherwise this method will panic.
|
||||||
|
|
||||||
|
# Panics
|
||||||
|
|
||||||
|
Panics if the given Lua state already has a runtime attached to it.
|
||||||
*/
|
*/
|
||||||
pub async fn run(&self) {
|
pub async fn run(&self) {
|
||||||
/*
|
/*
|
||||||
|
@ -164,13 +185,14 @@ impl<'lua> Runtime<'lua> {
|
||||||
Also ensure we do not already have an executor - this is a definite user error
|
Also ensure we do not already have an executor - this is a definite user error
|
||||||
and may happen if the user tries to run multiple runtimes on the same Lua state.
|
and may happen if the user tries to run multiple runtimes on the same Lua state.
|
||||||
*/
|
*/
|
||||||
if self.lua.app_data_ref::<Weak<Executor>>().is_some() {
|
assert!(
|
||||||
panic!(
|
self.lua.app_data_ref::<Weak<Executor>>().is_none(),
|
||||||
"Lua state already has an executor attached!\
|
"\
|
||||||
\nThis may be caused by running multiple runtimes on the same lua state, or a call to Runtime::run being cancelled.\
|
Lua state already has an executor attached!\
|
||||||
\nOnly one runtime can be used per lua state at once, and runtimes must always run until completion."
|
\nThis may be caused by running multiple runtimes on the same Lua state, or a call to Runtime::run being cancelled.\
|
||||||
|
\nOnly one runtime can be used per Lua state at once, and runtimes must always run until completion.\
|
||||||
|
"
|
||||||
);
|
);
|
||||||
}
|
|
||||||
self.lua.set_app_data(Arc::downgrade(&main_exec));
|
self.lua.set_app_data(Arc::downgrade(&main_exec));
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
|
|
@ -15,6 +15,10 @@ use async_executor::{Executor, Task};
|
||||||
pub trait IntoLuaThread<'lua> {
|
pub trait IntoLuaThread<'lua> {
|
||||||
/**
|
/**
|
||||||
Converts the value into a Lua thread.
|
Converts the value into a Lua thread.
|
||||||
|
|
||||||
|
# Errors
|
||||||
|
|
||||||
|
Errors when out of memory.
|
||||||
*/
|
*/
|
||||||
fn into_lua_thread(self, lua: &'lua Lua) -> LuaResult<LuaThread<'lua>>;
|
fn into_lua_thread(self, lua: &'lua Lua) -> LuaResult<LuaThread<'lua>>;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Reference in a new issue