use std::time::Duration; use mlua::{Error, Function, Lua, Result, Table, Thread, Value, Variadic}; use tokio::time::{self, Instant}; use crate::utils::table_builder::ReadonlyTableBuilder; type Vararg<'lua> = Variadic>; pub async fn create(lua: &Lua) -> Result<()> { lua.globals().raw_set( "task", 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)? .with_async_function("wait", task_wait)? .build()?, ) } fn get_or_create_thread_from_arg<'a>(lua: &'a Lua, arg: Value<'a>) -> Result> { Ok(match 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 resume_thread(lua: &Lua, thread: Thread<'_>, args: Vararg<'_>) -> Result<()> { let coroutine: Table = lua.globals().raw_get("coroutine")?; let resume: Function = coroutine.raw_get("resume")?; // FIXME: This is blocking, we should spawn a local tokio task, // but doing that moves "thread" and "args", that both have // the lifetime of the outer function, so it doesn't work resume.call_async((thread, args)).await?; Ok(()) } 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: &'a Lua, (tof, args): (Value<'a>, Vararg<'a>)) -> Result> { // TODO: Defer (sleep a minimum amount of time) let thread = get_or_create_thread_from_arg(lua, tof)?; resume_thread(lua, thread.clone(), args).await?; Ok(thread) } async fn task_delay<'a>( lua: &'a Lua, (_delay, tof, args): (Option, Value<'a>, Vararg<'a>), ) -> Result> { // TODO: Delay by the amount of time wanted let thread = get_or_create_thread_from_arg(lua, tof)?; resume_thread(lua, thread.clone(), args).await?; Ok(thread) } async fn task_spawn<'a>(lua: &'a Lua, (tof, args): (Value<'a>, Vararg<'a>)) -> Result> { let thread = get_or_create_thread_from_arg(lua, tof)?; resume_thread(lua, thread.clone(), args).await?; Ok(thread) } // FIXME: It doesn't seem possible to properly make an async wait // function with mlua right now, something breaks when using // the async wait function inside of a coroutine async fn task_wait(_: &Lua, duration: Option) -> Result { 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()) }