use std::convert::TryInto; use napi::{ CallContext, Env, JsBuffer, JsBufferValue, JsNumber, JsObject, Module, Ref, Result, Task, }; struct ComputeFib { n: u32, } impl ComputeFib { pub fn new(n: u32) -> ComputeFib { ComputeFib { n } } } impl Task for ComputeFib { type Output = u32; type JsValue = JsNumber; fn compute(&mut self) -> Result { Ok(fibonacci_native(self.n)) } fn resolve(self, env: Env, output: Self::Output) -> Result { env.create_uint32(output) } } #[inline] fn fibonacci_native(n: u32) -> u32 { match n { 1 | 2 => 1, _ => fibonacci_native(n - 1) + fibonacci_native(n - 2), } } #[js_function(1)] fn test_spawn_thread(ctx: CallContext) -> Result { let n = ctx.get::(0)?; let task = ComputeFib::new(n.try_into()?); let async_promise = ctx.env.spawn(task)?; Ok(async_promise.promise_object()) } struct CountBufferLength { data: Ref, } impl CountBufferLength { pub fn new(data: Ref) -> Self { Self { data } } } impl Task for CountBufferLength { type Output = usize; type JsValue = JsNumber; fn compute(&mut self) -> Result { Ok((&self.data).len()) } fn resolve(self, env: Env, output: Self::Output) -> Result { self.data.unref(env)?; env.create_uint32(output as _) } } #[js_function(1)] fn test_spawn_thread_with_ref(ctx: CallContext) -> Result { let n = ctx.get::(0)?.into_ref()?; let task = CountBufferLength::new(n); let async_work_promise = ctx.env.spawn(task)?; Ok(async_work_promise.promise_object()) } pub fn register_js(module: &mut Module) -> Result<()> { module.create_named_method("testSpawnThread", test_spawn_thread)?; module.create_named_method("testSpawnThreadWithRef", test_spawn_thread_with_ref)?; Ok(()) }