2023-02-17 18:20:17 +00:00
|
|
|
use std::process::ExitCode;
|
2023-01-22 20:23:56 +00:00
|
|
|
|
2023-02-15 12:22:13 +00:00
|
|
|
use lua::task::{TaskScheduler, TaskSchedulerResumeExt, TaskSchedulerScheduleExt};
|
2023-01-23 02:14:13 +00:00
|
|
|
use mlua::prelude::*;
|
2023-02-13 14:28:18 +00:00
|
|
|
use tokio::task::LocalSet;
|
2023-01-21 03:01:02 +00:00
|
|
|
|
2023-02-06 03:32:10 +00:00
|
|
|
pub(crate) mod globals;
|
2023-02-11 21:40:14 +00:00
|
|
|
pub(crate) mod lua;
|
2023-01-21 03:01:02 +00:00
|
|
|
|
2023-03-08 11:56:08 +00:00
|
|
|
mod error;
|
2023-02-10 11:14:28 +00:00
|
|
|
#[cfg(test)]
|
|
|
|
mod tests;
|
2023-01-21 03:01:02 +00:00
|
|
|
|
2023-03-08 11:56:08 +00:00
|
|
|
pub use error::LuneError;
|
2023-02-10 11:14:28 +00:00
|
|
|
pub use globals::LuneGlobal;
|
2023-02-16 15:19:58 +00:00
|
|
|
pub use lua::create_lune_lua;
|
2023-01-22 20:23:56 +00:00
|
|
|
|
|
|
|
#[derive(Clone, Debug, Default)]
|
|
|
|
pub struct Lune {
|
2023-02-17 18:20:17 +00:00
|
|
|
args: Vec<String>,
|
2023-01-22 20:23:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Lune {
|
2023-02-10 11:14:28 +00:00
|
|
|
/**
|
|
|
|
Creates a new Lune script runner.
|
|
|
|
*/
|
2023-01-22 20:23:56 +00:00
|
|
|
pub fn new() -> Self {
|
|
|
|
Self::default()
|
2023-01-21 03:01:02 +00:00
|
|
|
}
|
2023-01-22 20:23:56 +00:00
|
|
|
|
2023-02-10 11:14:28 +00:00
|
|
|
/**
|
2023-02-17 18:20:17 +00:00
|
|
|
Arguments to give in `process.args` for a Lune script.
|
2023-02-10 11:14:28 +00:00
|
|
|
*/
|
2023-02-17 18:20:17 +00:00
|
|
|
pub fn with_args<V>(mut self, args: V) -> Self
|
|
|
|
where
|
|
|
|
V: Into<Vec<String>>,
|
|
|
|
{
|
|
|
|
self.args = args.into();
|
2023-02-10 11:14:28 +00:00
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Runs a Lune script.
|
|
|
|
|
|
|
|
This will create a new sandboxed Luau environment with the configured
|
|
|
|
globals and arguments, running inside of a [`tokio::task::LocalSet`].
|
|
|
|
|
2023-02-17 18:20:17 +00:00
|
|
|
Some Lune globals may spawn separate tokio tasks on other threads, but the Luau
|
|
|
|
environment itself is guaranteed to run on a single thread in the local set.
|
2023-02-11 11:39:39 +00:00
|
|
|
|
2023-02-17 18:20:17 +00:00
|
|
|
Note that this will create a static Lua instance and task scheduler that will
|
|
|
|
both live for the remainer of the program, and that this leaks memory using
|
2023-02-11 11:39:39 +00:00
|
|
|
[`Box::leak`] that will then get deallocated when the program exits.
|
2023-02-10 11:14:28 +00:00
|
|
|
*/
|
2023-03-08 11:56:08 +00:00
|
|
|
pub async fn try_run(
|
|
|
|
&self,
|
|
|
|
script_name: impl AsRef<str>,
|
|
|
|
script_contents: impl AsRef<[u8]>,
|
|
|
|
) -> Result<ExitCode, LuneError> {
|
|
|
|
self.run(script_name, script_contents, true)
|
|
|
|
.await
|
|
|
|
.map_err(LuneError::from_lua_error)
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Tries to run a Lune script directly, returning the underlying
|
|
|
|
result type if loading the script raises a lua error.
|
|
|
|
|
|
|
|
Passing `false` as the third argument `emit_prettified_errors` will
|
|
|
|
bypass any additional error formatting automatically done by Lune
|
|
|
|
for errors that are emitted while the script is running.
|
|
|
|
|
|
|
|
Behavior is otherwise exactly the same as `try_run`
|
|
|
|
and `try_run` should be preferred in all other cases.
|
|
|
|
*/
|
|
|
|
#[doc(hidden)]
|
2023-02-10 11:14:28 +00:00
|
|
|
pub async fn run(
|
|
|
|
&self,
|
2023-03-07 21:07:53 +00:00
|
|
|
script_name: impl AsRef<str>,
|
|
|
|
script_contents: impl AsRef<[u8]>,
|
2023-03-08 11:56:08 +00:00
|
|
|
emit_prettified_errors: bool,
|
2023-02-10 11:14:28 +00:00
|
|
|
) -> Result<ExitCode, LuaError> {
|
2023-02-16 15:19:58 +00:00
|
|
|
// Create our special lune-flavored Lua object with extra registry values
|
2023-03-08 11:56:08 +00:00
|
|
|
let lua = create_lune_lua()?;
|
2023-02-16 23:08:24 +00:00
|
|
|
// Create our task scheduler
|
2023-02-13 20:29:05 +00:00
|
|
|
let sched = TaskScheduler::new(lua)?.into_static();
|
|
|
|
lua.set_app_data(sched);
|
2023-02-16 23:08:24 +00:00
|
|
|
// Create the main thread and schedule it
|
|
|
|
let main_chunk = lua
|
2023-03-07 21:07:53 +00:00
|
|
|
.load(script_contents.as_ref())
|
2023-03-08 11:56:08 +00:00
|
|
|
.set_name(script_name.as_ref())?
|
|
|
|
.into_function()?;
|
|
|
|
let main_thread = lua.create_thread(main_chunk)?;
|
2023-02-16 23:08:24 +00:00
|
|
|
let main_thread_args = LuaValue::Nil.to_lua_multi(lua)?;
|
|
|
|
sched.schedule_blocking(main_thread, main_thread_args)?;
|
2023-02-14 12:17:07 +00:00
|
|
|
// Create our wanted lune globals, some of these need
|
|
|
|
// the task scheduler be available during construction
|
2023-02-17 18:20:17 +00:00
|
|
|
for global in LuneGlobal::all(&self.args) {
|
|
|
|
global.inject(lua)?;
|
2023-02-14 12:17:07 +00:00
|
|
|
}
|
2023-02-13 14:28:18 +00:00
|
|
|
// Keep running the scheduler until there are either no tasks
|
2023-02-13 20:29:05 +00:00
|
|
|
// left to run, or until a task requests to exit the process
|
|
|
|
let exit_code = LocalSet::new()
|
|
|
|
.run_until(async move {
|
2023-02-14 14:27:10 +00:00
|
|
|
let mut got_error = false;
|
2023-02-14 15:39:15 +00:00
|
|
|
loop {
|
|
|
|
let result = sched.resume_queue().await;
|
2023-02-14 14:27:10 +00:00
|
|
|
if let Some(err) = result.get_lua_error() {
|
2023-03-08 11:56:08 +00:00
|
|
|
if emit_prettified_errors {
|
|
|
|
eprintln!("{}", LuneError::from_lua_error(err));
|
|
|
|
} else {
|
|
|
|
eprintln!("{err}");
|
|
|
|
}
|
2023-02-14 14:27:10 +00:00
|
|
|
got_error = true;
|
2023-01-22 21:26:45 +00:00
|
|
|
}
|
2023-02-14 15:39:15 +00:00
|
|
|
if result.is_done() {
|
|
|
|
if let Some(exit_code) = result.get_exit_code() {
|
|
|
|
break exit_code;
|
|
|
|
} else if got_error {
|
|
|
|
break ExitCode::FAILURE;
|
|
|
|
} else {
|
|
|
|
break ExitCode::SUCCESS;
|
|
|
|
}
|
|
|
|
}
|
2023-01-23 18:51:32 +00:00
|
|
|
}
|
2023-01-24 16:31:27 +00:00
|
|
|
})
|
2023-02-13 14:28:18 +00:00
|
|
|
.await;
|
|
|
|
Ok(exit_code)
|
2023-01-21 03:01:02 +00:00
|
|
|
}
|
|
|
|
}
|