2023-01-22 19:39:57 +00:00
|
|
|
use std::time::Duration;
|
2023-01-21 18:33:33 +00:00
|
|
|
|
2023-01-22 21:26:45 +00:00
|
|
|
use mlua::{Error, Function, Lua, Result, Table, Thread, Value, Variadic};
|
|
|
|
use tokio::time::{self, Instant};
|
2023-01-21 18:33:33 +00:00
|
|
|
|
2023-01-21 20:48:56 +00:00
|
|
|
use crate::utils::table_builder::ReadonlyTableBuilder;
|
|
|
|
|
2023-01-22 21:26:45 +00:00
|
|
|
pub async fn create(lua: &Lua) -> Result<()> {
|
2023-01-22 20:23:56 +00:00
|
|
|
lua.globals().raw_set(
|
|
|
|
"task",
|
2023-01-22 21:26:45 +00:00
|
|
|
ReadonlyTableBuilder::new(lua)?
|
|
|
|
.with_async_function("cancel", task_cancel)?
|
|
|
|
.with_async_function("defer", task_defer)?
|
|
|
|
.with_async_function("delay", task_delay)?
|
|
|
|
.with_async_function("spawn", task_spawn)?
|
2023-01-22 20:23:56 +00:00
|
|
|
.with_async_function("wait", task_wait)?
|
|
|
|
.build()?,
|
2023-01-22 21:26:45 +00:00
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn get_thread_from_arg<'a>(lua: &'a Lua, thread_or_function_arg: Value<'a>) -> Result<Thread<'a>> {
|
|
|
|
Ok(match thread_or_function_arg {
|
|
|
|
Value::Thread(thread) => thread,
|
|
|
|
Value::Function(func) => lua.create_thread(func)?,
|
|
|
|
val => {
|
|
|
|
return Err(Error::RuntimeError(format!(
|
|
|
|
"Expected type thread or function, got {}",
|
|
|
|
val.type_name()
|
|
|
|
)))
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
async fn task_cancel(lua: &Lua, thread: Thread<'_>) -> Result<()> {
|
|
|
|
let coroutine: Table = lua.globals().raw_get("coroutine")?;
|
|
|
|
let close: Function = coroutine.raw_get("close")?;
|
|
|
|
close.call_async(thread).await?;
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
async fn task_defer<'a>(lua: &Lua, (tof, args): (Value<'a>, Variadic<Value<'a>>)) -> Result<()> {
|
|
|
|
task_wait(lua, None).await?;
|
|
|
|
get_thread_from_arg(lua, tof)?
|
|
|
|
.into_async::<_, Variadic<Value<'_>>>(args)
|
|
|
|
.await?;
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
async fn task_delay<'a>(
|
|
|
|
lua: &Lua,
|
|
|
|
(delay, tof, args): (Option<f32>, Value<'a>, Variadic<Value<'a>>),
|
|
|
|
) -> Result<()> {
|
|
|
|
task_wait(lua, delay).await?;
|
|
|
|
get_thread_from_arg(lua, tof)?
|
|
|
|
.into_async::<_, Variadic<Value<'_>>>(args)
|
|
|
|
.await?;
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
async fn task_spawn<'a>(lua: &Lua, (tof, args): (Value<'a>, Variadic<Value<'a>>)) -> Result<()> {
|
|
|
|
get_thread_from_arg(lua, tof)?
|
|
|
|
.into_async::<_, Variadic<Value<'_>>>(args)
|
|
|
|
.await?;
|
|
|
|
Ok(())
|
2023-01-21 18:33:33 +00:00
|
|
|
}
|
2023-01-22 01:11:17 +00:00
|
|
|
|
2023-01-22 21:26:45 +00:00
|
|
|
// FIXME: It doesn't seem possible to properly make an async wait
|
2023-01-22 01:11:17 +00:00
|
|
|
// function with mlua right now, something breaks when using
|
2023-01-22 21:26:45 +00:00
|
|
|
// the async wait function inside of a coroutine
|
2023-01-22 20:23:56 +00:00
|
|
|
async fn task_wait(_: &Lua, duration: Option<f32>) -> Result<f32> {
|
2023-01-22 21:26:45 +00:00
|
|
|
let start = Instant::now();
|
|
|
|
time::sleep(
|
|
|
|
duration
|
|
|
|
.map(Duration::from_secs_f32)
|
|
|
|
.unwrap_or(Duration::ZERO),
|
|
|
|
)
|
|
|
|
.await;
|
|
|
|
let end = Instant::now();
|
|
|
|
Ok((end - start).as_secs_f32())
|
2023-01-22 01:11:17 +00:00
|
|
|
}
|