mirror of
https://github.com/lune-org/mlua-luau-scheduler.git
synced 2025-04-05 19:10:59 +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"
|
||||
version = "0.0.0"
|
||||
edition = "2021"
|
||||
|
||||
[lib]
|
||||
path = "lib/lib.rs"
|
||||
license = "MPL-2.0"
|
||||
repository = "https://github.com/lune-org/mlua-luau-runtime"
|
||||
description = "Luau-based async runtime, using mlua and async-executor"
|
||||
readme = "README.md"
|
||||
keywords = ["async", "luau", "runtime"]
|
||||
categories = ["async"]
|
||||
|
||||
[dependencies]
|
||||
async-executor = "1.8"
|
||||
|
@ -23,6 +26,14 @@ mlua = { version = "0.9.5", features = [
|
|||
async-fs = "2.1"
|
||||
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]]
|
||||
name = "basic_sleep"
|
||||
test = true
|
||||
|
|
|
@ -16,7 +16,7 @@
|
|||
|
||||
<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
|
||||
|
||||
|
|
|
@ -1,9 +1,11 @@
|
|||
#![allow(clippy::missing_errors_doc)]
|
||||
|
||||
use std::time::{Duration, Instant};
|
||||
|
||||
use async_io::{block_on, Timer};
|
||||
|
||||
use mlua::prelude::*;
|
||||
use mlua_luau_runtime::*;
|
||||
use mlua_luau_runtime::Runtime;
|
||||
|
||||
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
|
||||
let rt = Runtime::new(&lua)?;
|
||||
let rt = Runtime::new(&lua);
|
||||
let main = lua.load(MAIN_SCRIPT);
|
||||
rt.spawn_thread(main, ())?;
|
||||
|
||||
|
|
|
@ -1,10 +1,12 @@
|
|||
#![allow(clippy::missing_errors_doc)]
|
||||
|
||||
use std::io::ErrorKind;
|
||||
|
||||
use async_fs::read_to_string;
|
||||
use async_io::block_on;
|
||||
|
||||
use mlua::prelude::*;
|
||||
use mlua_luau_runtime::*;
|
||||
use mlua_luau_runtime::{LuaSpawnExt, Runtime};
|
||||
|
||||
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
|
||||
let rt = Runtime::new(&lua)?;
|
||||
let rt = Runtime::new(&lua);
|
||||
let main = lua.load(MAIN_SCRIPT);
|
||||
rt.spawn_thread(main, ())?;
|
||||
|
||||
|
|
|
@ -1,5 +1,7 @@
|
|||
#![allow(clippy::missing_errors_doc)]
|
||||
|
||||
use mlua::prelude::*;
|
||||
use mlua_luau_runtime::*;
|
||||
use mlua_luau_runtime::Runtime;
|
||||
|
||||
use async_io::block_on;
|
||||
|
||||
|
@ -10,7 +12,7 @@ pub fn main() -> LuaResult<()> {
|
|||
let lua = Lua::new();
|
||||
|
||||
// Create a new runtime with custom callbacks
|
||||
let rt = Runtime::new(&lua)?;
|
||||
let rt = Runtime::new(&lua);
|
||||
rt.set_error_callback(|e| {
|
||||
println!(
|
||||
"Captured error from Lua!\n{}\n{e}\n{}",
|
||||
|
|
|
@ -1,9 +1,11 @@
|
|||
#![allow(clippy::missing_errors_doc)]
|
||||
|
||||
use std::time::Duration;
|
||||
|
||||
use async_io::{block_on, Timer};
|
||||
|
||||
use mlua::prelude::*;
|
||||
use mlua_luau_runtime::*;
|
||||
use mlua_luau_runtime::Runtime;
|
||||
|
||||
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<()> {
|
||||
// Set up persistent Lua environment
|
||||
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(
|
||||
|
|
|
@ -1,16 +1,18 @@
|
|||
#![allow(clippy::missing_errors_doc)]
|
||||
|
||||
use std::time::{Duration, Instant};
|
||||
|
||||
use async_io::{block_on, Timer};
|
||||
|
||||
use mlua::prelude::*;
|
||||
use mlua_luau_runtime::*;
|
||||
use mlua_luau_runtime::Runtime;
|
||||
|
||||
const MAIN_SCRIPT: &str = include_str!("./lua/scheduler_ordering.luau");
|
||||
|
||||
pub fn main() -> LuaResult<()> {
|
||||
// Set up persistent Lua environment
|
||||
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("defer", rt.create_defer_function()?)?;
|
||||
|
|
|
@ -31,6 +31,7 @@ impl ThreadErrorCallback {
|
|||
}
|
||||
}
|
||||
|
||||
#[allow(clippy::needless_pass_by_value)]
|
||||
fn default_error_callback(e: LuaError) {
|
||||
eprintln!("{e}");
|
||||
}
|
||||
|
|
26
lib/queue.rs
26
lib/queue.rs
|
@ -6,8 +6,6 @@ use mlua::prelude::*;
|
|||
|
||||
use crate::IntoLuaThread;
|
||||
|
||||
const ERR_OOM: &str = "out of memory";
|
||||
|
||||
/**
|
||||
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.
|
||||
*/
|
||||
#[derive(Debug, Clone)]
|
||||
pub struct ThreadQueue {
|
||||
pub(crate) struct ThreadQueue {
|
||||
queue: Arc<ConcurrentQueue<ThreadWithArgs>>,
|
||||
event: Arc<Event>,
|
||||
}
|
||||
|
@ -35,9 +33,9 @@ impl ThreadQueue {
|
|||
) -> LuaResult<()> {
|
||||
let thread = thread.into_lua_thread(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);
|
||||
|
||||
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)]
|
||||
struct ThreadWithArgs {
|
||||
|
@ -70,19 +68,23 @@ struct 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 key_thread = lua.create_registry_value(thread).expect(ERR_OOM);
|
||||
let key_args = lua.create_registry_value(argsv).expect(ERR_OOM);
|
||||
let key_thread = lua.create_registry_value(thread)?;
|
||||
let key_args = lua.create_registry_value(argsv)?;
|
||||
|
||||
Self {
|
||||
Ok(Self {
|
||||
key_thread,
|
||||
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 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.
|
||||
*/
|
||||
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_defer = ThreadQueue::new();
|
||||
let error_callback = ThreadErrorCallback::default();
|
||||
|
||||
Ok(Runtime {
|
||||
Runtime {
|
||||
lua,
|
||||
queue_spawn,
|
||||
queue_defer,
|
||||
error_callback,
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -60,6 +61,10 @@ impl<'lua> Runtime<'lua> {
|
|||
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.
|
||||
|
||||
# Errors
|
||||
|
||||
Errors when out of memory.
|
||||
*/
|
||||
pub fn spawn_thread(
|
||||
&self,
|
||||
|
@ -75,6 +80,10 @@ impl<'lua> Runtime<'lua> {
|
|||
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.
|
||||
|
||||
# Errors
|
||||
|
||||
Errors when out of memory.
|
||||
*/
|
||||
pub fn defer_thread(
|
||||
&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.
|
||||
|
||||
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>> {
|
||||
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.
|
||||
|
||||
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>> {
|
||||
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
|
||||
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) {
|
||||
/*
|
||||
|
@ -164,13 +185,14 @@ impl<'lua> Runtime<'lua> {
|
|||
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.
|
||||
*/
|
||||
if self.lua.app_data_ref::<Weak<Executor>>().is_some() {
|
||||
panic!(
|
||||
"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.\
|
||||
\nOnly one runtime can be used per lua state at once, and runtimes must always run until completion."
|
||||
assert!(
|
||||
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.\
|
||||
\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));
|
||||
|
||||
/*
|
||||
|
|
|
@ -15,6 +15,10 @@ use async_executor::{Executor, Task};
|
|||
pub trait IntoLuaThread<'lua> {
|
||||
/**
|
||||
Converts the value into a Lua thread.
|
||||
|
||||
# Errors
|
||||
|
||||
Errors when out of memory.
|
||||
*/
|
||||
fn into_lua_thread(self, lua: &'lua Lua) -> LuaResult<LuaThread<'lua>>;
|
||||
}
|
||||
|
|
Loading…
Add table
Reference in a new issue