mirror of
https://github.com/lune-org/lune.git
synced 2024-12-13 13:30:38 +00:00
More work on task scheduler
This commit is contained in:
parent
744b73db3d
commit
c45c78bdc2
23 changed files with 359 additions and 239 deletions
|
@ -159,7 +159,7 @@ impl Cli {
|
||||||
// Display the file path relative to cwd with no extensions in stack traces
|
// Display the file path relative to cwd with no extensions in stack traces
|
||||||
let file_display_name = file_path.with_extension("").display().to_string();
|
let file_display_name = file_path.with_extension("").display().to_string();
|
||||||
// Create a new lune object with all globals & run the script
|
// Create a new lune object with all globals & run the script
|
||||||
let lune = Lune::new().with_all_globals_and_args(self.script_args);
|
let lune = Lune::new().with_args(self.script_args);
|
||||||
let result = lune.run(&file_display_name, &file_contents).await;
|
let result = lune.run(&file_display_name, &file_contents).await;
|
||||||
Ok(match result {
|
Ok(match result {
|
||||||
Err(e) => {
|
Err(e) => {
|
||||||
|
|
|
@ -1,11 +1,10 @@
|
||||||
use std::{
|
use std::{
|
||||||
env::{self, current_dir},
|
env::{self, current_dir},
|
||||||
io,
|
fs,
|
||||||
path::PathBuf,
|
path::PathBuf,
|
||||||
};
|
};
|
||||||
|
|
||||||
use mlua::prelude::*;
|
use mlua::prelude::*;
|
||||||
use tokio::fs;
|
|
||||||
|
|
||||||
use crate::utils::table::TableBuilder;
|
use crate::utils::table::TableBuilder;
|
||||||
|
|
||||||
|
@ -15,37 +14,63 @@ pub fn create(lua: &'static Lua) -> LuaResult<LuaTable> {
|
||||||
if env::var_os("LUAU_PWD_REQUIRE").is_some() {
|
if env::var_os("LUAU_PWD_REQUIRE").is_some() {
|
||||||
return TableBuilder::new(lua)?.build_readonly();
|
return TableBuilder::new(lua)?.build_readonly();
|
||||||
}
|
}
|
||||||
// Store the current pwd, and make helper functions for path conversions
|
// Store the current pwd, and make the functions for path conversions & loading a file
|
||||||
let require_pwd = current_dir()?.to_string_lossy().to_string();
|
let mut require_pwd = current_dir()?.to_string_lossy().to_string();
|
||||||
|
if !require_pwd.ends_with('/') {
|
||||||
|
require_pwd = format!("{require_pwd}/")
|
||||||
|
}
|
||||||
let require_info: LuaFunction = lua.named_registry_value("dbg.info")?;
|
let require_info: LuaFunction = lua.named_registry_value("dbg.info")?;
|
||||||
let require_error: LuaFunction = lua.named_registry_value("error")?;
|
let require_error: LuaFunction = lua.named_registry_value("error")?;
|
||||||
let require_get_abs_rel_paths = lua
|
let require_get_abs_rel_paths = lua
|
||||||
.create_function(
|
.create_function(
|
||||||
|_, (require_pwd, require_source, require_path): (String, String, String)| {
|
|_, (require_pwd, require_source, require_path): (String, String, String)| {
|
||||||
let mut path_relative_to_pwd = PathBuf::from(
|
let path_relative_to_pwd = PathBuf::from(
|
||||||
&require_source
|
&require_source
|
||||||
.trim_start_matches("[string \"")
|
.trim_start_matches("[string \"")
|
||||||
.trim_end_matches("\"]"),
|
.trim_end_matches("\"]"),
|
||||||
)
|
)
|
||||||
.parent()
|
.parent()
|
||||||
.unwrap()
|
.unwrap()
|
||||||
.join(require_path);
|
.join(&require_path);
|
||||||
// Try to normalize and resolve relative path segments such as './' and '../'
|
// Try to normalize and resolve relative path segments such as './' and '../'
|
||||||
if let Ok(canonicalized) =
|
let file_path = match (
|
||||||
path_relative_to_pwd.with_extension("luau").canonicalize()
|
path_relative_to_pwd.with_extension("luau").canonicalize(),
|
||||||
{
|
path_relative_to_pwd.with_extension("lua").canonicalize(),
|
||||||
path_relative_to_pwd = canonicalized;
|
) {
|
||||||
|
(Ok(luau), _) => luau,
|
||||||
|
(_, Ok(lua)) => lua,
|
||||||
|
_ => {
|
||||||
|
return Err(LuaError::RuntimeError(format!(
|
||||||
|
"File does not exist at path '{require_path}'"
|
||||||
|
)))
|
||||||
}
|
}
|
||||||
if let Ok(canonicalized) = path_relative_to_pwd.with_extension("lua").canonicalize()
|
};
|
||||||
{
|
let absolute = file_path.to_string_lossy().to_string();
|
||||||
path_relative_to_pwd = canonicalized;
|
|
||||||
}
|
|
||||||
let absolute = path_relative_to_pwd.to_string_lossy().to_string();
|
|
||||||
let relative = absolute.trim_start_matches(&require_pwd).to_string();
|
let relative = absolute.trim_start_matches(&require_pwd).to_string();
|
||||||
Ok((absolute, relative))
|
Ok((absolute, relative))
|
||||||
},
|
},
|
||||||
)?
|
)?
|
||||||
.bind(require_pwd)?;
|
.bind(require_pwd)?;
|
||||||
|
// Note that file loading must be blocking to guarantee the require cache works, if it
|
||||||
|
// were async then one lua script may require a module during the file reading process
|
||||||
|
let require_get_loaded_file = lua.create_function(
|
||||||
|
|lua: &Lua, (path_absolute, path_relative): (String, String)| {
|
||||||
|
// Use a name without extensions for loading the chunk, the
|
||||||
|
// above code assumes the require path is without extensions
|
||||||
|
let path_relative_no_extension = path_relative
|
||||||
|
.trim_end_matches(".lua")
|
||||||
|
.trim_end_matches(".luau");
|
||||||
|
// Try to read the wanted file, note that we use bytes instead of reading
|
||||||
|
// to a string since lua scripts are not necessarily valid utf-8 strings
|
||||||
|
match fs::read(path_absolute) {
|
||||||
|
Ok(contents) => lua
|
||||||
|
.load(&contents)
|
||||||
|
.set_name(path_relative_no_extension)?
|
||||||
|
.eval::<LuaValue>(),
|
||||||
|
Err(e) => Err(LuaError::external(e)),
|
||||||
|
}
|
||||||
|
},
|
||||||
|
)?;
|
||||||
/*
|
/*
|
||||||
We need to get the source file where require was
|
We need to get the source file where require was
|
||||||
called to be able to do path-relative requires,
|
called to be able to do path-relative requires,
|
||||||
|
@ -61,12 +86,15 @@ pub fn create(lua: &'static Lua) -> LuaResult<LuaTable> {
|
||||||
.with_value("info", require_info)?
|
.with_value("info", require_info)?
|
||||||
.with_value("error", require_error)?
|
.with_value("error", require_error)?
|
||||||
.with_value("paths", require_get_abs_rel_paths)?
|
.with_value("paths", require_get_abs_rel_paths)?
|
||||||
.with_async_function("load", load_file)?
|
.with_value("load", require_get_loaded_file)?
|
||||||
.build_readonly()?;
|
.build_readonly()?;
|
||||||
let require_fn_lua = lua
|
let require_fn_lua = lua
|
||||||
.load(
|
.load(
|
||||||
r#"
|
r#"
|
||||||
local source = info(2, "s")
|
local source = info(1, "s")
|
||||||
|
if source == '[string "require"]' then
|
||||||
|
source = info(2, "s")
|
||||||
|
end
|
||||||
local absolute, relative = paths(source, ...)
|
local absolute, relative = paths(source, ...)
|
||||||
if loaded[absolute] ~= true then
|
if loaded[absolute] ~= true then
|
||||||
local first, second = load(absolute, relative)
|
local first, second = load(absolute, relative)
|
||||||
|
@ -88,20 +116,3 @@ pub fn create(lua: &'static Lua) -> LuaResult<LuaTable> {
|
||||||
.with_value("require", require_fn_lua)?
|
.with_value("require", require_fn_lua)?
|
||||||
.build_readonly()
|
.build_readonly()
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn load_file(
|
|
||||||
lua: &Lua,
|
|
||||||
(path_absolute, path_relative): (String, String),
|
|
||||||
) -> LuaResult<LuaValue> {
|
|
||||||
// Try to read the wanted file, note that we use bytes instead of reading
|
|
||||||
// to a string since lua scripts are not necessarily valid utf-8 strings
|
|
||||||
match fs::read(&path_absolute).await {
|
|
||||||
Ok(contents) => lua.load(&contents).set_name(path_relative)?.eval(),
|
|
||||||
Err(e) => match e.kind() {
|
|
||||||
io::ErrorKind::NotFound => Err(LuaError::RuntimeError(format!(
|
|
||||||
"No lua module exists at the path '{path_relative}'"
|
|
||||||
))),
|
|
||||||
_ => Err(LuaError::external(e)),
|
|
||||||
},
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
|
@ -1,10 +1,13 @@
|
||||||
use std::time::Duration;
|
|
||||||
|
|
||||||
use mlua::prelude::*;
|
use mlua::prelude::*;
|
||||||
use tokio::time::{sleep, Instant};
|
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
lua::task::{TaskKind, TaskReference, TaskScheduler, TaskSchedulerScheduleExt},
|
lua::{
|
||||||
|
async_ext::LuaAsyncExt,
|
||||||
|
task::{
|
||||||
|
LuaThreadOrFunction, LuaThreadOrTaskReference, TaskKind, TaskReference, TaskScheduler,
|
||||||
|
TaskSchedulerScheduleExt,
|
||||||
|
},
|
||||||
|
},
|
||||||
utils::table::TableBuilder,
|
utils::table::TableBuilder,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -22,7 +25,6 @@ pub fn create(lua: &'static Lua) -> LuaResult<LuaTable<'static>> {
|
||||||
we need to yield right away to allow the
|
we need to yield right away to allow the
|
||||||
spawned task to run until first yield
|
spawned task to run until first yield
|
||||||
*/
|
*/
|
||||||
let task_spawn_env_thread: LuaFunction = lua.named_registry_value("co.thread")?;
|
|
||||||
let task_spawn_env_yield: LuaFunction = lua.named_registry_value("co.yield")?;
|
let task_spawn_env_yield: LuaFunction = lua.named_registry_value("co.yield")?;
|
||||||
let task_spawn = lua
|
let task_spawn = lua
|
||||||
.load(
|
.load(
|
||||||
|
@ -33,10 +35,10 @@ pub fn create(lua: &'static Lua) -> LuaResult<LuaTable<'static>> {
|
||||||
return task
|
return task
|
||||||
",
|
",
|
||||||
)
|
)
|
||||||
.set_name("=task.spawn")?
|
.set_name("task.spawn")?
|
||||||
.set_environment(
|
.set_environment(
|
||||||
TableBuilder::new(lua)?
|
TableBuilder::new(lua)?
|
||||||
.with_value("thread", task_spawn_env_thread)?
|
.with_function("thread", |lua, _: ()| Ok(lua.current_thread()))?
|
||||||
.with_value("yield", task_spawn_env_yield)?
|
.with_value("yield", task_spawn_env_yield)?
|
||||||
.with_function(
|
.with_function(
|
||||||
"scheduleNext",
|
"scheduleNext",
|
||||||
|
@ -63,83 +65,14 @@ pub fn create(lua: &'static Lua) -> LuaResult<LuaTable<'static>> {
|
||||||
coroutine.set("wrap", lua.create_function(coroutine_wrap)?)?;
|
coroutine.set("wrap", lua.create_function(coroutine_wrap)?)?;
|
||||||
// All good, return the task scheduler lib
|
// All good, return the task scheduler lib
|
||||||
TableBuilder::new(lua)?
|
TableBuilder::new(lua)?
|
||||||
|
.with_value("wait", lua.create_waiter_function()?)?
|
||||||
.with_value("spawn", task_spawn)?
|
.with_value("spawn", task_spawn)?
|
||||||
.with_function("cancel", task_cancel)?
|
.with_function("cancel", task_cancel)?
|
||||||
.with_function("defer", task_defer)?
|
.with_function("defer", task_defer)?
|
||||||
.with_function("delay", task_delay)?
|
.with_function("delay", task_delay)?
|
||||||
.with_async_function("wait", task_wait)?
|
|
||||||
.build_readonly()
|
.build_readonly()
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
Proxy enum to deal with both threads & functions
|
|
||||||
*/
|
|
||||||
|
|
||||||
enum LuaThreadOrFunction<'lua> {
|
|
||||||
Thread(LuaThread<'lua>),
|
|
||||||
Function(LuaFunction<'lua>),
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<'lua> LuaThreadOrFunction<'lua> {
|
|
||||||
fn into_thread(self, lua: &'lua Lua) -> LuaResult<LuaThread<'lua>> {
|
|
||||||
match self {
|
|
||||||
Self::Thread(t) => Ok(t),
|
|
||||||
Self::Function(f) => lua.create_thread(f),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<'lua> FromLua<'lua> for LuaThreadOrFunction<'lua> {
|
|
||||||
fn from_lua(value: LuaValue<'lua>, _: &'lua Lua) -> LuaResult<Self> {
|
|
||||||
match value {
|
|
||||||
LuaValue::Thread(t) => Ok(Self::Thread(t)),
|
|
||||||
LuaValue::Function(f) => Ok(Self::Function(f)),
|
|
||||||
value => Err(LuaError::FromLuaConversionError {
|
|
||||||
from: value.type_name(),
|
|
||||||
to: "LuaThreadOrFunction",
|
|
||||||
message: Some(format!(
|
|
||||||
"Expected thread or function, got '{}'",
|
|
||||||
value.type_name()
|
|
||||||
)),
|
|
||||||
}),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
Proxy enum to deal with both threads & task scheduler task references
|
|
||||||
*/
|
|
||||||
|
|
||||||
enum LuaThreadOrTaskReference<'lua> {
|
|
||||||
Thread(LuaThread<'lua>),
|
|
||||||
TaskReference(TaskReference),
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<'lua> FromLua<'lua> for LuaThreadOrTaskReference<'lua> {
|
|
||||||
fn from_lua(value: LuaValue<'lua>, lua: &'lua Lua) -> LuaResult<Self> {
|
|
||||||
let tname = value.type_name();
|
|
||||||
match value {
|
|
||||||
LuaValue::Thread(t) => Ok(Self::Thread(t)),
|
|
||||||
LuaValue::UserData(u) => {
|
|
||||||
if let Ok(task) = TaskReference::from_lua(LuaValue::UserData(u), lua) {
|
|
||||||
Ok(Self::TaskReference(task))
|
|
||||||
} else {
|
|
||||||
Err(LuaError::FromLuaConversionError {
|
|
||||||
from: tname,
|
|
||||||
to: "thread",
|
|
||||||
message: Some(format!("Expected thread, got '{tname}'")),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
_ => Err(LuaError::FromLuaConversionError {
|
|
||||||
from: tname,
|
|
||||||
to: "thread",
|
|
||||||
message: Some(format!("Expected thread, got '{tname}'")),
|
|
||||||
}),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Basic task functions
|
Basic task functions
|
||||||
*/
|
*/
|
||||||
|
@ -166,12 +99,6 @@ fn task_delay(
|
||||||
sched.schedule_blocking_after_seconds(secs, tof.into_thread(lua)?, args)
|
sched.schedule_blocking_after_seconds(secs, tof.into_thread(lua)?, args)
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn task_wait(_: &Lua, secs: Option<f64>) -> LuaResult<f64> {
|
|
||||||
let start = Instant::now();
|
|
||||||
sleep(Duration::from_secs_f64(secs.unwrap_or_default())).await;
|
|
||||||
Ok(start.elapsed().as_secs_f64())
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Type getter overrides for compat with task scheduler
|
Type getter overrides for compat with task scheduler
|
||||||
*/
|
*/
|
||||||
|
@ -207,7 +134,7 @@ fn coroutine_resume<'lua>(
|
||||||
match value {
|
match value {
|
||||||
LuaThreadOrTaskReference::Thread(t) => {
|
LuaThreadOrTaskReference::Thread(t) => {
|
||||||
let sched = lua.app_data_ref::<&TaskScheduler>().unwrap();
|
let sched = lua.app_data_ref::<&TaskScheduler>().unwrap();
|
||||||
let task = sched.create_task(TaskKind::Instant, t, None, None)?;
|
let task = sched.create_task(TaskKind::Instant, t, None, true)?;
|
||||||
sched.resume_task(task, None)
|
sched.resume_task(task, None)
|
||||||
}
|
}
|
||||||
LuaThreadOrTaskReference::TaskReference(t) => lua
|
LuaThreadOrTaskReference::TaskReference(t) => lua
|
||||||
|
@ -222,7 +149,7 @@ fn coroutine_wrap<'lua>(lua: &'lua Lua, func: LuaFunction) -> LuaResult<LuaFunct
|
||||||
TaskKind::Instant,
|
TaskKind::Instant,
|
||||||
lua.create_thread(func)?,
|
lua.create_thread(func)?,
|
||||||
None,
|
None,
|
||||||
None,
|
false,
|
||||||
)?;
|
)?;
|
||||||
lua.create_function(move |lua, args: LuaMultiValue| {
|
lua.create_function(move |lua, args: LuaMultiValue| {
|
||||||
lua.app_data_ref::<&TaskScheduler>()
|
lua.app_data_ref::<&TaskScheduler>()
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
use std::{collections::HashSet, process::ExitCode};
|
use std::process::ExitCode;
|
||||||
|
|
||||||
use lua::task::{TaskScheduler, TaskSchedulerResumeExt, TaskSchedulerScheduleExt};
|
use lua::task::{TaskScheduler, TaskSchedulerResumeExt, TaskSchedulerScheduleExt};
|
||||||
use mlua::prelude::*;
|
use mlua::prelude::*;
|
||||||
|
@ -18,8 +18,7 @@ pub use lua::create_lune_lua;
|
||||||
|
|
||||||
#[derive(Clone, Debug, Default)]
|
#[derive(Clone, Debug, Default)]
|
||||||
pub struct Lune {
|
pub struct Lune {
|
||||||
includes: HashSet<LuneGlobal>,
|
args: Vec<String>,
|
||||||
excludes: HashSet<LuneGlobal>,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Lune {
|
impl Lune {
|
||||||
|
@ -31,42 +30,13 @@ impl Lune {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
Include a global in the lua environment created for running a Lune script.
|
Arguments to give in `process.args` for a Lune script.
|
||||||
*/
|
*/
|
||||||
pub fn with_global(mut self, global: LuneGlobal) -> Self {
|
pub fn with_args<V>(mut self, args: V) -> Self
|
||||||
self.includes.insert(global);
|
where
|
||||||
self
|
V: Into<Vec<String>>,
|
||||||
}
|
{
|
||||||
|
self.args = args.into();
|
||||||
/**
|
|
||||||
Include all globals in the lua environment created for running a Lune script.
|
|
||||||
*/
|
|
||||||
pub fn with_all_globals(mut self) -> Self {
|
|
||||||
for global in LuneGlobal::all::<String>(&[]) {
|
|
||||||
self.includes.insert(global);
|
|
||||||
}
|
|
||||||
self
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
Include all globals in the lua environment created for running a
|
|
||||||
Lune script, as well as supplying args for [`LuneGlobal::Process`].
|
|
||||||
*/
|
|
||||||
pub fn with_all_globals_and_args(mut self, args: Vec<String>) -> Self {
|
|
||||||
for global in LuneGlobal::all(&args) {
|
|
||||||
self.includes.insert(global);
|
|
||||||
}
|
|
||||||
self
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
Exclude a global from the lua environment created for running a Lune script.
|
|
||||||
|
|
||||||
This should be preferred over manually iterating and filtering
|
|
||||||
which Lune globals to add to the global environment.
|
|
||||||
*/
|
|
||||||
pub fn without_global(mut self, global: LuneGlobal) -> Self {
|
|
||||||
self.excludes.insert(global);
|
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -76,12 +46,11 @@ impl Lune {
|
||||||
This will create a new sandboxed Luau environment with the configured
|
This will create a new sandboxed Luau environment with the configured
|
||||||
globals and arguments, running inside of a [`tokio::task::LocalSet`].
|
globals and arguments, running inside of a [`tokio::task::LocalSet`].
|
||||||
|
|
||||||
Some Lune globals such as [`LuneGlobal::Process`] and [`LuneGlobal::Net`]
|
Some Lune globals may spawn separate tokio tasks on other threads, but the Luau
|
||||||
may spawn separate tokio tasks on other threads, but the Luau environment
|
environment itself is guaranteed to run on a single thread in the local set.
|
||||||
itself is guaranteed to run on a single thread in the local set.
|
|
||||||
|
|
||||||
Note that this will create a static Lua instance and task scheduler which both
|
Note that this will create a static Lua instance and task scheduler that will
|
||||||
will live for the remainer of the program, and that this leaks memory using
|
both live for the remainer of the program, and that this leaks memory using
|
||||||
[`Box::leak`] that will then get deallocated when the program exits.
|
[`Box::leak`] that will then get deallocated when the program exits.
|
||||||
*/
|
*/
|
||||||
pub async fn run(
|
pub async fn run(
|
||||||
|
@ -106,11 +75,9 @@ impl Lune {
|
||||||
sched.schedule_blocking(main_thread, main_thread_args)?;
|
sched.schedule_blocking(main_thread, main_thread_args)?;
|
||||||
// Create our wanted lune globals, some of these need
|
// Create our wanted lune globals, some of these need
|
||||||
// the task scheduler be available during construction
|
// the task scheduler be available during construction
|
||||||
for global in self.includes.clone() {
|
for global in LuneGlobal::all(&self.args) {
|
||||||
if !self.excludes.contains(&global) {
|
|
||||||
global.inject(lua)?;
|
global.inject(lua)?;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
// Keep running the scheduler until there are either no tasks
|
// Keep running the scheduler until there are either no tasks
|
||||||
// left to run, or until a task requests to exit the process
|
// left to run, or until a task requests to exit the process
|
||||||
let exit_code = LocalSet::new()
|
let exit_code = LocalSet::new()
|
||||||
|
|
|
@ -4,6 +4,8 @@ use mlua::prelude::*;
|
||||||
|
|
||||||
use crate::{lua::task::TaskScheduler, utils::table::TableBuilder};
|
use crate::{lua::task::TaskScheduler, utils::table::TableBuilder};
|
||||||
|
|
||||||
|
use super::task::TaskSchedulerAsyncExt;
|
||||||
|
|
||||||
#[async_trait(?Send)]
|
#[async_trait(?Send)]
|
||||||
pub trait LuaAsyncExt {
|
pub trait LuaAsyncExt {
|
||||||
fn create_async_function<'lua, A, R, F, FR>(self, func: F) -> LuaResult<LuaFunction<'lua>>
|
fn create_async_function<'lua, A, R, F, FR>(self, func: F) -> LuaResult<LuaFunction<'lua>>
|
||||||
|
@ -12,6 +14,8 @@ pub trait LuaAsyncExt {
|
||||||
R: ToLuaMulti<'static>,
|
R: ToLuaMulti<'static>,
|
||||||
F: 'static + Fn(&'static Lua, A) -> FR,
|
F: 'static + Fn(&'static Lua, A) -> FR,
|
||||||
FR: 'static + Future<Output = LuaResult<R>>;
|
FR: 'static + Future<Output = LuaResult<R>>;
|
||||||
|
|
||||||
|
fn create_waiter_function<'lua>(self) -> LuaResult<LuaFunction<'lua>>;
|
||||||
}
|
}
|
||||||
|
|
||||||
impl LuaAsyncExt for &'static Lua {
|
impl LuaAsyncExt for &'static Lua {
|
||||||
|
@ -31,7 +35,6 @@ impl LuaAsyncExt for &'static Lua {
|
||||||
let async_env_trace: LuaFunction = self.named_registry_value("dbg.trace")?;
|
let async_env_trace: LuaFunction = self.named_registry_value("dbg.trace")?;
|
||||||
let async_env_error: LuaFunction = self.named_registry_value("error")?;
|
let async_env_error: LuaFunction = self.named_registry_value("error")?;
|
||||||
let async_env_unpack: LuaFunction = self.named_registry_value("tab.unpack")?;
|
let async_env_unpack: LuaFunction = self.named_registry_value("tab.unpack")?;
|
||||||
let async_env_thread: LuaFunction = self.named_registry_value("co.thread")?;
|
|
||||||
let async_env_yield: LuaFunction = self.named_registry_value("co.yield")?;
|
let async_env_yield: LuaFunction = self.named_registry_value("co.yield")?;
|
||||||
let async_env = TableBuilder::new(self)?
|
let async_env = TableBuilder::new(self)?
|
||||||
.with_value("makeError", async_env_make_err)?
|
.with_value("makeError", async_env_make_err)?
|
||||||
|
@ -39,8 +42,8 @@ impl LuaAsyncExt for &'static Lua {
|
||||||
.with_value("trace", async_env_trace)?
|
.with_value("trace", async_env_trace)?
|
||||||
.with_value("error", async_env_error)?
|
.with_value("error", async_env_error)?
|
||||||
.with_value("unpack", async_env_unpack)?
|
.with_value("unpack", async_env_unpack)?
|
||||||
.with_value("thread", async_env_thread)?
|
|
||||||
.with_value("yield", async_env_yield)?
|
.with_value("yield", async_env_yield)?
|
||||||
|
.with_function("thread", |lua, _: ()| Ok(lua.current_thread()))?
|
||||||
.with_function(
|
.with_function(
|
||||||
"resumeAsync",
|
"resumeAsync",
|
||||||
move |lua: &Lua, (thread, args): (LuaThread, A)| {
|
move |lua: &Lua, (thread, args): (LuaThread, A)| {
|
||||||
|
@ -48,7 +51,7 @@ impl LuaAsyncExt for &'static Lua {
|
||||||
let sched = lua
|
let sched = lua
|
||||||
.app_data_ref::<&TaskScheduler>()
|
.app_data_ref::<&TaskScheduler>()
|
||||||
.expect("Missing task scheduler as a lua app data");
|
.expect("Missing task scheduler as a lua app data");
|
||||||
sched.queue_async_task(thread, None, None, async {
|
sched.queue_async_task(thread, None, async {
|
||||||
let rets = fut.await?;
|
let rets = fut.await?;
|
||||||
let mult = rets.to_lua_multi(lua)?;
|
let mult = rets.to_lua_multi(lua)?;
|
||||||
Ok(Some(mult))
|
Ok(Some(mult))
|
||||||
|
@ -68,7 +71,36 @@ impl LuaAsyncExt for &'static Lua {
|
||||||
end
|
end
|
||||||
",
|
",
|
||||||
)
|
)
|
||||||
.set_name("asyncWrapper")?
|
.set_name("async")?
|
||||||
|
.set_environment(async_env)?
|
||||||
|
.into_function()?;
|
||||||
|
Ok(async_func)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
Creates a special async function that waits the
|
||||||
|
desired amount of time, inheriting the guid of the
|
||||||
|
current thread / task for proper cancellation.
|
||||||
|
*/
|
||||||
|
fn create_waiter_function<'lua>(self) -> LuaResult<LuaFunction<'lua>> {
|
||||||
|
let async_env_yield: LuaFunction = self.named_registry_value("co.yield")?;
|
||||||
|
let async_env = TableBuilder::new(self)?
|
||||||
|
.with_value("yield", async_env_yield)?
|
||||||
|
.with_function("resumeAfter", move |lua: &Lua, duration: Option<f64>| {
|
||||||
|
let sched = lua
|
||||||
|
.app_data_ref::<&TaskScheduler>()
|
||||||
|
.expect("Missing task scheduler as a lua app data");
|
||||||
|
sched.schedule_wait(lua.current_thread(), duration)
|
||||||
|
})?
|
||||||
|
.build_readonly()?;
|
||||||
|
let async_func = self
|
||||||
|
.load(
|
||||||
|
"
|
||||||
|
resumeAfter(...)
|
||||||
|
return yield()
|
||||||
|
",
|
||||||
|
)
|
||||||
|
.set_name("wait")?
|
||||||
.set_environment(async_env)?
|
.set_environment(async_env)?
|
||||||
.into_function()?;
|
.into_function()?;
|
||||||
Ok(async_func)
|
Ok(async_func)
|
|
@ -64,7 +64,6 @@ end
|
||||||
* `"tostring"` -> `tostring`
|
* `"tostring"` -> `tostring`
|
||||||
* `"tonumber"` -> `tonumber`
|
* `"tonumber"` -> `tonumber`
|
||||||
---
|
---
|
||||||
* `"co.thread"` -> `coroutine.running`
|
|
||||||
* `"co.yield"` -> `coroutine.yield`
|
* `"co.yield"` -> `coroutine.yield`
|
||||||
* `"co.close"` -> `coroutine.close`
|
* `"co.close"` -> `coroutine.close`
|
||||||
---
|
---
|
||||||
|
@ -93,7 +92,6 @@ pub fn create() -> LuaResult<&'static Lua> {
|
||||||
lua.set_named_registry_value("pcall", globals.get::<_, LuaFunction>("pcall")?)?;
|
lua.set_named_registry_value("pcall", globals.get::<_, LuaFunction>("pcall")?)?;
|
||||||
lua.set_named_registry_value("tostring", globals.get::<_, LuaFunction>("tostring")?)?;
|
lua.set_named_registry_value("tostring", globals.get::<_, LuaFunction>("tostring")?)?;
|
||||||
lua.set_named_registry_value("tonumber", globals.get::<_, LuaFunction>("tonumber")?)?;
|
lua.set_named_registry_value("tonumber", globals.get::<_, LuaFunction>("tonumber")?)?;
|
||||||
lua.set_named_registry_value("co.thread", coroutine.get::<_, LuaFunction>("running")?)?;
|
|
||||||
lua.set_named_registry_value("co.yield", coroutine.get::<_, LuaFunction>("yield")?)?;
|
lua.set_named_registry_value("co.yield", coroutine.get::<_, LuaFunction>("yield")?)?;
|
||||||
lua.set_named_registry_value("co.close", coroutine.get::<_, LuaFunction>("close")?)?;
|
lua.set_named_registry_value("co.close", coroutine.get::<_, LuaFunction>("close")?)?;
|
||||||
lua.set_named_registry_value("dbg.info", debug.get::<_, LuaFunction>("info")?)?;
|
lua.set_named_registry_value("dbg.info", debug.get::<_, LuaFunction>("info")?)?;
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
mod create;
|
mod create;
|
||||||
|
|
||||||
pub mod ext;
|
pub mod async_ext;
|
||||||
pub mod net;
|
pub mod net;
|
||||||
pub mod stdio;
|
pub mod stdio;
|
||||||
pub mod task;
|
pub mod task;
|
||||||
|
|
|
@ -1,7 +1,12 @@
|
||||||
|
use std::time::Duration;
|
||||||
|
|
||||||
use async_trait::async_trait;
|
use async_trait::async_trait;
|
||||||
|
|
||||||
use futures_util::Future;
|
use futures_util::Future;
|
||||||
use mlua::prelude::*;
|
use mlua::prelude::*;
|
||||||
|
use tokio::time::{sleep, Instant};
|
||||||
|
|
||||||
|
use crate::lua::task::TaskKind;
|
||||||
|
|
||||||
use super::super::{
|
use super::super::{
|
||||||
async_handle::TaskSchedulerAsyncHandle, message::TaskSchedulerMessage,
|
async_handle::TaskSchedulerAsyncHandle, message::TaskSchedulerMessage,
|
||||||
|
@ -30,6 +35,12 @@ pub trait TaskSchedulerAsyncExt<'fut> {
|
||||||
R: ToLuaMulti<'static>,
|
R: ToLuaMulti<'static>,
|
||||||
F: 'static + Fn(&'static Lua) -> FR,
|
F: 'static + Fn(&'static Lua) -> FR,
|
||||||
FR: 'static + Future<Output = LuaResult<R>>;
|
FR: 'static + Future<Output = LuaResult<R>>;
|
||||||
|
|
||||||
|
fn schedule_wait(
|
||||||
|
&'fut self,
|
||||||
|
reference: LuaThread<'_>,
|
||||||
|
duration: Option<f64>,
|
||||||
|
) -> LuaResult<TaskReference>;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -82,7 +93,7 @@ impl<'fut> TaskSchedulerAsyncExt<'fut> for TaskScheduler<'fut> {
|
||||||
F: 'static + Fn(&'static Lua) -> FR,
|
F: 'static + Fn(&'static Lua) -> FR,
|
||||||
FR: 'static + Future<Output = LuaResult<R>>,
|
FR: 'static + Future<Output = LuaResult<R>>,
|
||||||
{
|
{
|
||||||
self.queue_async_task(thread, None, None, async move {
|
self.queue_async_task(thread, None, async move {
|
||||||
match func(self.lua).await {
|
match func(self.lua).await {
|
||||||
Ok(res) => match res.to_lua_multi(self.lua) {
|
Ok(res) => match res.to_lua_multi(self.lua) {
|
||||||
Ok(multi) => Ok(Some(multi)),
|
Ok(multi) => Ok(Some(multi)),
|
||||||
|
@ -92,4 +103,30 @@ impl<'fut> TaskSchedulerAsyncExt<'fut> for TaskScheduler<'fut> {
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
Schedules a task reference to be resumed after a certain amount of time.
|
||||||
|
|
||||||
|
The given task will be resumed with the elapsed time as its one and only argument.
|
||||||
|
*/
|
||||||
|
fn schedule_wait(
|
||||||
|
&'fut self,
|
||||||
|
thread: LuaThread<'_>,
|
||||||
|
duration: Option<f64>,
|
||||||
|
) -> LuaResult<TaskReference> {
|
||||||
|
let reference = self.create_task(TaskKind::Future, thread, None, true)?;
|
||||||
|
// Insert the future
|
||||||
|
let futs = self
|
||||||
|
.futures
|
||||||
|
.try_lock()
|
||||||
|
.expect("Tried to add future to queue during futures resumption");
|
||||||
|
futs.push(Box::pin(async move {
|
||||||
|
let before = Instant::now();
|
||||||
|
sleep(Duration::from_secs_f64(duration.unwrap_or_default())).await;
|
||||||
|
let elapsed_secs = before.elapsed().as_secs_f64();
|
||||||
|
let args = elapsed_secs.to_lua_multi(self.lua).unwrap();
|
||||||
|
(Some(reference), Ok(Some(args)))
|
||||||
|
}));
|
||||||
|
Ok(reference)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -124,12 +124,15 @@ async fn resume_next_async_task(scheduler: &TaskScheduler<'_>) -> TaskSchedulerS
|
||||||
.await
|
.await
|
||||||
.expect("Tried to resume next queued future but none are queued")
|
.expect("Tried to resume next queued future but none are queued")
|
||||||
};
|
};
|
||||||
|
// The future might not return a reference that it wants to resume
|
||||||
|
if let Some(task) = task {
|
||||||
// Promote this future task to a blocking task and resume it
|
// Promote this future task to a blocking task and resume it
|
||||||
// right away, also taking care to not borrow mutably twice
|
// right away, also taking care to not borrow mutably twice
|
||||||
// by dropping this guard before trying to resume it
|
// by dropping this guard before trying to resume it
|
||||||
let mut queue_guard = scheduler.tasks_queue_blocking.borrow_mut();
|
let mut queue_guard = scheduler.tasks_queue_blocking.borrow_mut();
|
||||||
queue_guard.push_front(task);
|
queue_guard.push_front(task);
|
||||||
drop(queue_guard);
|
drop(queue_guard);
|
||||||
|
}
|
||||||
resume_next_blocking_task(scheduler, result.transpose())
|
resume_next_blocking_task(scheduler, result.transpose())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -52,7 +52,7 @@ impl TaskSchedulerScheduleExt for TaskScheduler<'_> {
|
||||||
thread: LuaThread<'_>,
|
thread: LuaThread<'_>,
|
||||||
thread_args: LuaMultiValue<'_>,
|
thread_args: LuaMultiValue<'_>,
|
||||||
) -> LuaResult<TaskReference> {
|
) -> LuaResult<TaskReference> {
|
||||||
self.queue_blocking_task(TaskKind::Instant, thread, Some(thread_args), None)
|
self.queue_blocking_task(TaskKind::Instant, thread, Some(thread_args))
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -67,7 +67,7 @@ impl TaskSchedulerScheduleExt for TaskScheduler<'_> {
|
||||||
thread: LuaThread<'_>,
|
thread: LuaThread<'_>,
|
||||||
thread_args: LuaMultiValue<'_>,
|
thread_args: LuaMultiValue<'_>,
|
||||||
) -> LuaResult<TaskReference> {
|
) -> LuaResult<TaskReference> {
|
||||||
self.queue_blocking_task(TaskKind::Deferred, thread, Some(thread_args), None)
|
self.queue_blocking_task(TaskKind::Deferred, thread, Some(thread_args))
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -83,7 +83,7 @@ impl TaskSchedulerScheduleExt for TaskScheduler<'_> {
|
||||||
thread: LuaThread<'_>,
|
thread: LuaThread<'_>,
|
||||||
thread_args: LuaMultiValue<'_>,
|
thread_args: LuaMultiValue<'_>,
|
||||||
) -> LuaResult<TaskReference> {
|
) -> LuaResult<TaskReference> {
|
||||||
self.queue_async_task(thread, Some(thread_args), None, async move {
|
self.queue_async_task(thread, Some(thread_args), async move {
|
||||||
sleep(Duration::from_secs_f64(after_secs)).await;
|
sleep(Duration::from_secs_f64(after_secs)).await;
|
||||||
Ok(None)
|
Ok(None)
|
||||||
})
|
})
|
||||||
|
|
|
@ -1,10 +1,12 @@
|
||||||
mod async_handle;
|
mod async_handle;
|
||||||
mod ext;
|
mod ext;
|
||||||
mod message;
|
mod message;
|
||||||
|
mod proxy;
|
||||||
mod result;
|
mod result;
|
||||||
mod scheduler;
|
mod scheduler;
|
||||||
mod task_kind;
|
mod task_kind;
|
||||||
mod task_reference;
|
mod task_reference;
|
||||||
|
|
||||||
pub use ext::*;
|
pub use ext::*;
|
||||||
|
pub use proxy::*;
|
||||||
pub use scheduler::*;
|
pub use scheduler::*;
|
||||||
|
|
116
packages/lib/src/lua/task/proxy.rs
Normal file
116
packages/lib/src/lua/task/proxy.rs
Normal file
|
@ -0,0 +1,116 @@
|
||||||
|
use mlua::prelude::*;
|
||||||
|
|
||||||
|
use super::TaskReference;
|
||||||
|
|
||||||
|
/*
|
||||||
|
Proxy enum to deal with both threads & functions
|
||||||
|
*/
|
||||||
|
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
|
pub enum LuaThreadOrFunction<'lua> {
|
||||||
|
Thread(LuaThread<'lua>),
|
||||||
|
Function(LuaFunction<'lua>),
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'lua> LuaThreadOrFunction<'lua> {
|
||||||
|
pub fn into_thread(self, lua: &'lua Lua) -> LuaResult<LuaThread<'lua>> {
|
||||||
|
match self {
|
||||||
|
Self::Thread(t) => Ok(t),
|
||||||
|
Self::Function(f) => lua.create_thread(f),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'lua> From<LuaThread<'lua>> for LuaThreadOrFunction<'lua> {
|
||||||
|
fn from(value: LuaThread<'lua>) -> Self {
|
||||||
|
Self::Thread(value)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'lua> From<LuaFunction<'lua>> for LuaThreadOrFunction<'lua> {
|
||||||
|
fn from(value: LuaFunction<'lua>) -> Self {
|
||||||
|
Self::Function(value)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'lua> FromLua<'lua> for LuaThreadOrFunction<'lua> {
|
||||||
|
fn from_lua(value: LuaValue<'lua>, _: &'lua Lua) -> LuaResult<Self> {
|
||||||
|
match value {
|
||||||
|
LuaValue::Thread(t) => Ok(Self::Thread(t)),
|
||||||
|
LuaValue::Function(f) => Ok(Self::Function(f)),
|
||||||
|
value => Err(LuaError::FromLuaConversionError {
|
||||||
|
from: value.type_name(),
|
||||||
|
to: "LuaThreadOrFunction",
|
||||||
|
message: Some(format!(
|
||||||
|
"Expected thread or function, got '{}'",
|
||||||
|
value.type_name()
|
||||||
|
)),
|
||||||
|
}),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'lua> ToLua<'lua> for LuaThreadOrFunction<'lua> {
|
||||||
|
fn to_lua(self, _: &'lua Lua) -> LuaResult<LuaValue<'lua>> {
|
||||||
|
match self {
|
||||||
|
Self::Thread(t) => Ok(LuaValue::Thread(t)),
|
||||||
|
Self::Function(f) => Ok(LuaValue::Function(f)),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
Proxy enum to deal with both threads & task scheduler task references
|
||||||
|
*/
|
||||||
|
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
|
pub enum LuaThreadOrTaskReference<'lua> {
|
||||||
|
Thread(LuaThread<'lua>),
|
||||||
|
TaskReference(TaskReference),
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'lua> From<LuaThread<'lua>> for LuaThreadOrTaskReference<'lua> {
|
||||||
|
fn from(value: LuaThread<'lua>) -> Self {
|
||||||
|
Self::Thread(value)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'lua> From<TaskReference> for LuaThreadOrTaskReference<'lua> {
|
||||||
|
fn from(value: TaskReference) -> Self {
|
||||||
|
Self::TaskReference(value)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'lua> FromLua<'lua> for LuaThreadOrTaskReference<'lua> {
|
||||||
|
fn from_lua(value: LuaValue<'lua>, lua: &'lua Lua) -> LuaResult<Self> {
|
||||||
|
let tname = value.type_name();
|
||||||
|
match value {
|
||||||
|
LuaValue::Thread(t) => Ok(Self::Thread(t)),
|
||||||
|
LuaValue::UserData(u) => {
|
||||||
|
if let Ok(task) = TaskReference::from_lua(LuaValue::UserData(u), lua) {
|
||||||
|
Ok(Self::TaskReference(task))
|
||||||
|
} else {
|
||||||
|
Err(LuaError::FromLuaConversionError {
|
||||||
|
from: tname,
|
||||||
|
to: "thread",
|
||||||
|
message: Some(format!("Expected thread, got '{tname}'")),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_ => Err(LuaError::FromLuaConversionError {
|
||||||
|
from: tname,
|
||||||
|
to: "thread",
|
||||||
|
message: Some(format!("Expected thread, got '{tname}'")),
|
||||||
|
}),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'lua> ToLua<'lua> for LuaThreadOrTaskReference<'lua> {
|
||||||
|
fn to_lua(self, lua: &'lua Lua) -> LuaResult<LuaValue<'lua>> {
|
||||||
|
match self {
|
||||||
|
Self::TaskReference(t) => t.to_lua(lua),
|
||||||
|
Self::Thread(t) => Ok(LuaValue::Thread(t)),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -14,7 +14,7 @@ use super::message::TaskSchedulerMessage;
|
||||||
pub use super::{task_kind::TaskKind, task_reference::TaskReference};
|
pub use super::{task_kind::TaskKind, task_reference::TaskReference};
|
||||||
|
|
||||||
type TaskFutureRets<'fut> = LuaResult<Option<LuaMultiValue<'fut>>>;
|
type TaskFutureRets<'fut> = LuaResult<Option<LuaMultiValue<'fut>>>;
|
||||||
type TaskFuture<'fut> = LocalBoxFuture<'fut, (TaskReference, TaskFutureRets<'fut>)>;
|
type TaskFuture<'fut> = LocalBoxFuture<'fut, (Option<TaskReference>, TaskFutureRets<'fut>)>;
|
||||||
|
|
||||||
/// A struct representing a task contained in the task scheduler
|
/// A struct representing a task contained in the task scheduler
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
|
@ -40,10 +40,10 @@ pub struct TaskScheduler<'fut> {
|
||||||
// Internal state & flags
|
// Internal state & flags
|
||||||
pub(super) lua: &'static Lua,
|
pub(super) lua: &'static Lua,
|
||||||
pub(super) guid: Cell<usize>,
|
pub(super) guid: Cell<usize>,
|
||||||
pub(super) guid_running: Cell<Option<usize>>,
|
|
||||||
pub(super) exit_code: Cell<Option<ExitCode>>,
|
pub(super) exit_code: Cell<Option<ExitCode>>,
|
||||||
// Blocking tasks
|
// Blocking tasks
|
||||||
pub(super) tasks: RefCell<HashMap<TaskReference, Task>>,
|
pub(super) tasks: RefCell<HashMap<TaskReference, Task>>,
|
||||||
|
pub(super) tasks_current: Cell<Option<TaskReference>>,
|
||||||
pub(super) tasks_queue_blocking: RefCell<VecDeque<TaskReference>>,
|
pub(super) tasks_queue_blocking: RefCell<VecDeque<TaskReference>>,
|
||||||
// Future tasks & objects for waking
|
// Future tasks & objects for waking
|
||||||
pub(super) futures: AsyncMutex<FuturesUnordered<TaskFuture<'fut>>>,
|
pub(super) futures: AsyncMutex<FuturesUnordered<TaskFuture<'fut>>>,
|
||||||
|
@ -61,9 +61,9 @@ impl<'fut> TaskScheduler<'fut> {
|
||||||
Ok(Self {
|
Ok(Self {
|
||||||
lua,
|
lua,
|
||||||
guid: Cell::new(0),
|
guid: Cell::new(0),
|
||||||
guid_running: Cell::new(None),
|
|
||||||
exit_code: Cell::new(None),
|
exit_code: Cell::new(None),
|
||||||
tasks: RefCell::new(HashMap::new()),
|
tasks: RefCell::new(HashMap::new()),
|
||||||
|
tasks_current: Cell::new(None),
|
||||||
tasks_queue_blocking: RefCell::new(VecDeque::new()),
|
tasks_queue_blocking: RefCell::new(VecDeque::new()),
|
||||||
futures: AsyncMutex::new(FuturesUnordered::new()),
|
futures: AsyncMutex::new(FuturesUnordered::new()),
|
||||||
futures_tx: tx,
|
futures_tx: tx,
|
||||||
|
@ -109,6 +109,14 @@ impl<'fut> TaskScheduler<'fut> {
|
||||||
self.tasks.borrow().contains_key(&reference)
|
self.tasks.borrow().contains_key(&reference)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
Returns the currently running task, if any.
|
||||||
|
*/
|
||||||
|
#[allow(dead_code)]
|
||||||
|
pub fn current_task(&self) -> Option<TaskReference> {
|
||||||
|
self.tasks_current.get()
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
Creates a new task, storing a new Lua thread
|
Creates a new task, storing a new Lua thread
|
||||||
for it, as well as the arguments to give the
|
for it, as well as the arguments to give the
|
||||||
|
@ -123,7 +131,7 @@ impl<'fut> TaskScheduler<'fut> {
|
||||||
kind: TaskKind,
|
kind: TaskKind,
|
||||||
thread: LuaThread<'_>,
|
thread: LuaThread<'_>,
|
||||||
thread_args: Option<LuaMultiValue<'_>>,
|
thread_args: Option<LuaMultiValue<'_>>,
|
||||||
guid_to_reuse: Option<usize>,
|
inherit_current_guid: bool,
|
||||||
) -> LuaResult<TaskReference> {
|
) -> LuaResult<TaskReference> {
|
||||||
// Store the thread and its arguments in the registry
|
// Store the thread and its arguments in the registry
|
||||||
// NOTE: We must convert to a vec since multis
|
// NOTE: We must convert to a vec since multis
|
||||||
|
@ -137,19 +145,22 @@ impl<'fut> TaskScheduler<'fut> {
|
||||||
args: task_args_key,
|
args: task_args_key,
|
||||||
};
|
};
|
||||||
// Create the task ref to use
|
// Create the task ref to use
|
||||||
let task_ref = if let Some(reusable_guid) = guid_to_reuse {
|
let guid = if inherit_current_guid {
|
||||||
TaskReference::new(kind, reusable_guid)
|
self.current_task()
|
||||||
|
.expect("No current guid to inherit")
|
||||||
|
.id()
|
||||||
} else {
|
} else {
|
||||||
let guid = self.guid.get();
|
let guid = self.guid.get();
|
||||||
self.guid.set(guid + 1);
|
self.guid.set(guid + 1);
|
||||||
TaskReference::new(kind, guid)
|
guid
|
||||||
};
|
};
|
||||||
|
let reference = TaskReference::new(kind, guid);
|
||||||
// Add the task to the scheduler
|
// Add the task to the scheduler
|
||||||
{
|
{
|
||||||
let mut tasks = self.tasks.borrow_mut();
|
let mut tasks = self.tasks.borrow_mut();
|
||||||
tasks.insert(task_ref, task);
|
tasks.insert(reference, task);
|
||||||
}
|
}
|
||||||
Ok(task_ref)
|
Ok(reference)
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -181,8 +192,13 @@ impl<'fut> TaskScheduler<'fut> {
|
||||||
.filter(|task_ref| task_ref.id() == reference.id())
|
.filter(|task_ref| task_ref.id() == reference.id())
|
||||||
.copied()
|
.copied()
|
||||||
.collect();
|
.collect();
|
||||||
for task_ref in tasks_to_remove {
|
for task_ref in &tasks_to_remove {
|
||||||
if let Some(task) = tasks.remove(&task_ref) {
|
if let Some(task) = tasks.remove(task_ref) {
|
||||||
|
// NOTE: We need to close the thread here to
|
||||||
|
// make 100% sure that nothing can resume it
|
||||||
|
let close: LuaFunction = self.lua.named_registry_value("co.close")?;
|
||||||
|
let thread: LuaThread = self.lua.registry_value(&task.thread)?;
|
||||||
|
close.call(thread)?;
|
||||||
self.lua.remove_registry_value(task.thread)?;
|
self.lua.remove_registry_value(task.thread)?;
|
||||||
self.lua.remove_registry_value(task.args)?;
|
self.lua.remove_registry_value(task.args)?;
|
||||||
found = true;
|
found = true;
|
||||||
|
@ -204,13 +220,16 @@ impl<'fut> TaskScheduler<'fut> {
|
||||||
reference: TaskReference,
|
reference: TaskReference,
|
||||||
override_args: Option<LuaResult<LuaMultiValue<'a>>>,
|
override_args: Option<LuaResult<LuaMultiValue<'a>>>,
|
||||||
) -> LuaResult<LuaMultiValue<'a>> {
|
) -> LuaResult<LuaMultiValue<'a>> {
|
||||||
|
// Fetch and check if the task was removed, if it got
|
||||||
|
// removed it means it was intentionally cancelled
|
||||||
let task = {
|
let task = {
|
||||||
let mut tasks = self.tasks.borrow_mut();
|
let mut tasks = self.tasks.borrow_mut();
|
||||||
match tasks.remove(&reference) {
|
match tasks.remove(&reference) {
|
||||||
Some(task) => task,
|
Some(task) => task,
|
||||||
None => return Ok(LuaMultiValue::new()), // Task was removed
|
None => return Ok(LuaMultiValue::new()),
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
// Fetch and remove the thread to resume + its arguments
|
||||||
let thread: LuaThread = self.lua.registry_value(&task.thread)?;
|
let thread: LuaThread = self.lua.registry_value(&task.thread)?;
|
||||||
let args_opt_res = override_args.or_else(|| {
|
let args_opt_res = override_args.or_else(|| {
|
||||||
Ok(self
|
Ok(self
|
||||||
|
@ -222,7 +241,9 @@ impl<'fut> TaskScheduler<'fut> {
|
||||||
});
|
});
|
||||||
self.lua.remove_registry_value(task.thread)?;
|
self.lua.remove_registry_value(task.thread)?;
|
||||||
self.lua.remove_registry_value(task.args)?;
|
self.lua.remove_registry_value(task.args)?;
|
||||||
self.guid_running.set(Some(reference.id()));
|
// We got everything we need and our references
|
||||||
|
// were cleaned up properly, resume the thread
|
||||||
|
self.tasks_current.set(Some(reference));
|
||||||
let rets = match args_opt_res {
|
let rets = match args_opt_res {
|
||||||
Some(args_res) => match args_res {
|
Some(args_res) => match args_res {
|
||||||
/*
|
/*
|
||||||
|
@ -235,12 +256,12 @@ impl<'fut> TaskScheduler<'fut> {
|
||||||
that may pass errors as arguments when resuming tasks, other
|
that may pass errors as arguments when resuming tasks, other
|
||||||
native mlua functions will handle this and dont need wrapping
|
native mlua functions will handle this and dont need wrapping
|
||||||
*/
|
*/
|
||||||
Err(err) => thread.resume(err),
|
Err(e) => thread.resume(e),
|
||||||
Ok(args) => thread.resume(args),
|
Ok(args) => thread.resume(args),
|
||||||
},
|
},
|
||||||
None => thread.resume(()),
|
None => thread.resume(()),
|
||||||
};
|
};
|
||||||
self.guid_running.set(None);
|
self.tasks_current.set(None);
|
||||||
rets
|
rets
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -265,12 +286,11 @@ impl<'fut> TaskScheduler<'fut> {
|
||||||
kind: TaskKind,
|
kind: TaskKind,
|
||||||
thread: LuaThread<'_>,
|
thread: LuaThread<'_>,
|
||||||
thread_args: Option<LuaMultiValue<'_>>,
|
thread_args: Option<LuaMultiValue<'_>>,
|
||||||
guid_to_reuse: Option<usize>,
|
|
||||||
) -> LuaResult<TaskReference> {
|
) -> LuaResult<TaskReference> {
|
||||||
if kind == TaskKind::Future {
|
if kind == TaskKind::Future {
|
||||||
panic!("Tried to schedule future using normal task schedule method")
|
panic!("Tried to schedule future using normal task schedule method")
|
||||||
}
|
}
|
||||||
let task_ref = self.create_task(kind, thread, thread_args, guid_to_reuse)?;
|
let task_ref = self.create_task(kind, thread, thread_args, false)?;
|
||||||
// Add the task to the front of the queue, unless it
|
// Add the task to the front of the queue, unless it
|
||||||
// should be deferred, in that case add it to the back
|
// should be deferred, in that case add it to the back
|
||||||
let mut queue = self.tasks_queue_blocking.borrow_mut();
|
let mut queue = self.tasks_queue_blocking.borrow_mut();
|
||||||
|
@ -304,17 +324,16 @@ impl<'fut> TaskScheduler<'fut> {
|
||||||
&self,
|
&self,
|
||||||
thread: LuaThread<'_>,
|
thread: LuaThread<'_>,
|
||||||
thread_args: Option<LuaMultiValue<'_>>,
|
thread_args: Option<LuaMultiValue<'_>>,
|
||||||
guid_to_reuse: Option<usize>,
|
|
||||||
fut: impl Future<Output = TaskFutureRets<'fut>> + 'fut,
|
fut: impl Future<Output = TaskFutureRets<'fut>> + 'fut,
|
||||||
) -> LuaResult<TaskReference> {
|
) -> LuaResult<TaskReference> {
|
||||||
let task_ref = self.create_task(TaskKind::Future, thread, thread_args, guid_to_reuse)?;
|
let task_ref = self.create_task(TaskKind::Future, thread, thread_args, false)?;
|
||||||
let futs = self
|
let futs = self
|
||||||
.futures
|
.futures
|
||||||
.try_lock()
|
.try_lock()
|
||||||
.expect("Failed to get lock on futures");
|
.expect("Tried to add future to queue during futures resumption");
|
||||||
futs.push(Box::pin(async move {
|
futs.push(Box::pin(async move {
|
||||||
let result = fut.await;
|
let result = fut.await;
|
||||||
(task_ref, result)
|
(Some(task_ref), result)
|
||||||
}));
|
}));
|
||||||
Ok(task_ref)
|
Ok(task_ref)
|
||||||
}
|
}
|
||||||
|
|
|
@ -24,8 +24,12 @@ impl TaskReference {
|
||||||
|
|
||||||
impl fmt::Display for TaskReference {
|
impl fmt::Display for TaskReference {
|
||||||
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
||||||
|
if self.guid == 0 {
|
||||||
|
write!(f, "TaskReference(MAIN)")
|
||||||
|
} else {
|
||||||
write!(f, "TaskReference({} - {})", self.kind, self.guid)
|
write!(f, "TaskReference({} - {})", self.kind, self.guid)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl LuaUserData for TaskReference {}
|
impl LuaUserData for TaskReference {}
|
||||||
|
|
|
@ -26,12 +26,12 @@ macro_rules! create_tests {
|
||||||
// The rest of the test logic can continue as normal
|
// The rest of the test logic can continue as normal
|
||||||
let full_name = format!("tests/{}.luau", $value);
|
let full_name = format!("tests/{}.luau", $value);
|
||||||
let script = read_to_string(&full_name).await?;
|
let script = read_to_string(&full_name).await?;
|
||||||
let lune = Lune::new().with_all_globals_and_args(
|
let lune = Lune::new().with_args(
|
||||||
ARGS
|
ARGS
|
||||||
.clone()
|
.clone()
|
||||||
.iter()
|
.iter()
|
||||||
.map(ToString::to_string)
|
.map(ToString::to_string)
|
||||||
.collect()
|
.collect::<Vec<_>>()
|
||||||
);
|
);
|
||||||
let script_name = full_name.strip_suffix(".luau").unwrap();
|
let script_name = full_name.strip_suffix(".luau").unwrap();
|
||||||
let exit_code = lune.run(&script_name, &script).await?;
|
let exit_code = lune.run(&script_name, &script).await?;
|
||||||
|
|
|
@ -240,18 +240,21 @@ pub fn pretty_format_luau_error(e: &LuaError, colorized: bool) -> String {
|
||||||
// The traceback may also start with "override traceback:" which
|
// The traceback may also start with "override traceback:" which
|
||||||
// means it was passed from somewhere that wants a custom trace,
|
// means it was passed from somewhere that wants a custom trace,
|
||||||
// so we should then respect that and get the best override instead
|
// so we should then respect that and get the best override instead
|
||||||
let mut best_trace: &str = traceback;
|
let mut full_trace = traceback.to_string();
|
||||||
let mut root_cause = cause.as_ref();
|
let mut root_cause = cause.as_ref();
|
||||||
let mut trace_override = false;
|
let mut trace_override = false;
|
||||||
while let LuaError::CallbackError { cause, traceback } = root_cause {
|
while let LuaError::CallbackError { cause, traceback } = root_cause {
|
||||||
let is_override = traceback.starts_with("override traceback:");
|
let is_override = traceback.starts_with("override traceback:");
|
||||||
if is_override {
|
if is_override {
|
||||||
if !trace_override || traceback.lines().count() > best_trace.len() {
|
if !trace_override || traceback.lines().count() > full_trace.len() {
|
||||||
best_trace = traceback.strip_prefix("override traceback:").unwrap();
|
full_trace = traceback
|
||||||
|
.strip_prefix("override traceback:")
|
||||||
|
.unwrap()
|
||||||
|
.to_string();
|
||||||
trace_override = true;
|
trace_override = true;
|
||||||
}
|
}
|
||||||
} else if !trace_override && traceback.lines().count() > best_trace.len() {
|
} else if !trace_override {
|
||||||
best_trace = traceback;
|
full_trace = format!("{traceback}\n{full_trace}");
|
||||||
}
|
}
|
||||||
root_cause = cause;
|
root_cause = cause;
|
||||||
}
|
}
|
||||||
|
@ -266,10 +269,10 @@ pub fn pretty_format_luau_error(e: &LuaError, colorized: bool) -> String {
|
||||||
"{}\n{}\n{}\n{}",
|
"{}\n{}\n{}\n{}",
|
||||||
pretty_format_luau_error(root_cause, colorized),
|
pretty_format_luau_error(root_cause, colorized),
|
||||||
stack_begin,
|
stack_begin,
|
||||||
if best_trace.starts_with("stack traceback:") {
|
if full_trace.starts_with("stack traceback:") {
|
||||||
best_trace.strip_prefix("stack traceback:\n").unwrap()
|
full_trace.strip_prefix("stack traceback:\n").unwrap()
|
||||||
} else {
|
} else {
|
||||||
best_trace
|
&full_trace
|
||||||
},
|
},
|
||||||
stack_end
|
stack_end
|
||||||
)
|
)
|
||||||
|
@ -378,7 +381,8 @@ fn transform_stack_line(line: &str) -> String {
|
||||||
let line_num = match after_name.find(':') {
|
let line_num = match after_name.find(':') {
|
||||||
Some(lineno_start) => match after_name[lineno_start + 1..].find(':') {
|
Some(lineno_start) => match after_name[lineno_start + 1..].find(':') {
|
||||||
Some(lineno_end) => &after_name[lineno_start + 1..lineno_end + 1],
|
Some(lineno_end) => &after_name[lineno_start + 1..lineno_end + 1],
|
||||||
None => match after_name.contains("in function") {
|
None => match after_name.contains("in function") || after_name.contains("in ?")
|
||||||
|
{
|
||||||
false => &after_name[lineno_start + 1..],
|
false => &after_name[lineno_start + 1..],
|
||||||
true => "",
|
true => "",
|
||||||
},
|
},
|
||||||
|
@ -418,11 +422,18 @@ fn transform_stack_line(line: &str) -> String {
|
||||||
fn fix_error_nitpicks(full_message: String) -> String {
|
fn fix_error_nitpicks(full_message: String) -> String {
|
||||||
full_message
|
full_message
|
||||||
// Hacky fix for our custom require appearing as a normal script
|
// Hacky fix for our custom require appearing as a normal script
|
||||||
|
// TODO: It's probably better to pull in the regex crate here ..
|
||||||
.replace("'require', Line 5", "'[C]' - function require")
|
.replace("'require', Line 5", "'[C]' - function require")
|
||||||
.replace("'require', Line 7", "'[C]' - function require")
|
.replace("'require', Line 7", "'[C]' - function require")
|
||||||
|
.replace("'require', Line 8", "'[C]' - function require")
|
||||||
// Fix error calls in custom script chunks coming through
|
// Fix error calls in custom script chunks coming through
|
||||||
.replace(
|
.replace(
|
||||||
"'[C]' - function error\n Script '[C]' - function require",
|
"'[C]' - function error\n Script '[C]' - function require",
|
||||||
"'[C]' - function require",
|
"'[C]' - function require",
|
||||||
)
|
)
|
||||||
|
// Fix strange double require
|
||||||
|
.replace(
|
||||||
|
"'[C]' - function require - function require",
|
||||||
|
"'[C]' - function require",
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,7 +2,7 @@ use std::future::Future;
|
||||||
|
|
||||||
use mlua::prelude::*;
|
use mlua::prelude::*;
|
||||||
|
|
||||||
use crate::lua::ext::LuaAsyncExt;
|
use crate::lua::async_ext::LuaAsyncExt;
|
||||||
|
|
||||||
pub struct TableBuilder {
|
pub struct TableBuilder {
|
||||||
lua: &'static Lua,
|
lua: &'static Lua,
|
||||||
|
|
|
@ -1,3 +1,6 @@
|
||||||
|
local PORT = 9090 -- NOTE: This must be different from
|
||||||
|
-- net tests to let them run in parallel with this file
|
||||||
|
|
||||||
local function test(f, ...)
|
local function test(f, ...)
|
||||||
local success, message = pcall(f, ...)
|
local success, message = pcall(f, ...)
|
||||||
assert(not success, "Function did not throw an error")
|
assert(not success, "Function did not throw an error")
|
||||||
|
@ -14,7 +17,7 @@ test(net.request, "https://wxyz.google.com")
|
||||||
|
|
||||||
-- Net serve is async and will throw an OS error when trying to serve twice on the same port
|
-- Net serve is async and will throw an OS error when trying to serve twice on the same port
|
||||||
|
|
||||||
local handle = net.serve(8080, function()
|
local handle = net.serve(PORT, function()
|
||||||
return ""
|
return ""
|
||||||
end)
|
end)
|
||||||
|
|
||||||
|
@ -22,18 +25,4 @@ task.delay(0, function()
|
||||||
handle.stop()
|
handle.stop()
|
||||||
end)
|
end)
|
||||||
|
|
||||||
test(net.serve, 8080, function() end)
|
test(net.serve, PORT, function() end)
|
||||||
|
|
||||||
local function e()
|
|
||||||
task.spawn(function()
|
|
||||||
task.defer(function()
|
|
||||||
task.delay(0, function()
|
|
||||||
error({
|
|
||||||
Hello = "World",
|
|
||||||
})
|
|
||||||
end)
|
|
||||||
end)
|
|
||||||
end)
|
|
||||||
end
|
|
||||||
|
|
||||||
task.defer(e)
|
|
||||||
|
|
|
@ -4,6 +4,8 @@ assert(type(module) == "table", "Required module did not return a table")
|
||||||
assert(module.Foo == "Bar", "Required module did not contain correct values")
|
assert(module.Foo == "Bar", "Required module did not contain correct values")
|
||||||
assert(module.Hello == "World", "Required module did not contain correct values")
|
assert(module.Hello == "World", "Required module did not contain correct values")
|
||||||
|
|
||||||
require("modules/module")
|
module = require("modules/module")
|
||||||
|
assert(module.Foo == "Bar", "Required module did not contain correct values")
|
||||||
|
assert(module.Hello == "World", "Required module did not contain correct values")
|
||||||
|
|
||||||
return true
|
return true
|
||||||
|
|
|
@ -5,7 +5,7 @@ local function test(path: string)
|
||||||
if success then
|
if success then
|
||||||
error(string.format("Invalid require at path '%s' succeeded", path))
|
error(string.format("Invalid require at path '%s' succeeded", path))
|
||||||
else
|
else
|
||||||
print(message)
|
message = tostring(message)
|
||||||
if string.find(message, string.format("%s'", path)) == nil then
|
if string.find(message, string.format("%s'", path)) == nil then
|
||||||
error(
|
error(
|
||||||
string.format(
|
string.format(
|
||||||
|
|
|
@ -4,4 +4,6 @@ assert(type(module) == "table", "Required module did not return a table")
|
||||||
assert(module.Foo == "Bar", "Required module did not contain correct values")
|
assert(module.Foo == "Bar", "Required module did not contain correct values")
|
||||||
assert(module.Hello == "World", "Required module did not contain correct values")
|
assert(module.Hello == "World", "Required module did not contain correct values")
|
||||||
|
|
||||||
require("modules/nested")
|
module = require("modules/nested")
|
||||||
|
assert(module.Foo == "Bar", "Required module did not contain correct values")
|
||||||
|
assert(module.Hello == "World", "Required module did not contain correct values")
|
||||||
|
|
|
@ -15,7 +15,7 @@ function util.fail(method, url, message)
|
||||||
method = method,
|
method = method,
|
||||||
url = url,
|
url = url,
|
||||||
})
|
})
|
||||||
if not response.ok then
|
if response.ok then
|
||||||
error(string.format("%s passed!\nResponse: %s", message, stdio.format(response)))
|
error(string.format("%s passed!\nResponse: %s", message, stdio.format(response)))
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
Loading…
Reference in a new issue