From 2951ed28fcb4941cd4ed635363343e15e74faddb Mon Sep 17 00:00:00 2001 From: Valerio Ageno Date: Thu, 25 Apr 2024 18:56:14 +0200 Subject: [PATCH] feat: handle async functions --- src/ssr.rs | 75 +++++++++++++++++++++++++++++++++++++++++++++++++++++- 1 file changed, 74 insertions(+), 1 deletion(-) diff --git a/src/ssr.rs b/src/ssr.rs index 2aed227..dc4906c 100644 --- a/src/ssr.rs +++ b/src/ssr.rs @@ -147,11 +147,26 @@ where // TODO: transform this into an iterator for key in self.fn_map.keys() { - let result = match self.fn_map[key].call(scope, undef, &[params]) { + let render_fn = self.fn_map[key]; + + let mut result = match render_fn.call(scope, undef, &[params]) { Some(val) => val, None => return Err("Failed to call function"), }; + if result.is_promise() { + let promise = match v8::Local::::try_from(result) { + Ok(val) => val, + Err(_) => return Err("Failed to cast main function to promise"), + }; + + while promise.state() == v8::PromiseState::Pending { + scope.perform_microtask_checkpoint(); + } + + result = promise.result(scope); + } + let result = match result.to_string(scope) { Some(val) => val, None => return Err("Failed to parse the result to string"), @@ -282,4 +297,62 @@ mod tests { assert_eq!(js2.render_to_string(None).unwrap(), "I don't accept params"); } + + #[test] + fn entry_point_is_async() { + init_test(); + + let mut js = Ssr::from( + r##"var SSR = {x: async () => ""};"##.to_string(), + "SSR", + ) + .unwrap(); + + assert_eq!(js.render_to_string(None).unwrap(), ""); + assert_eq!( + js.render_to_string(Some(r#"{"Hello world"}"#)).unwrap(), + "" + ); + } + + #[test] + fn entry_point_is_async_with_params() { + init_test(); + + let mut js = Ssr::from( + r##"var SSR = {x: async (params) => "These are our parameters: " + params};"## + .to_string(), + "SSR", + ) + .unwrap(); + + assert_eq!( + js.render_to_string(Some(r#"{"Hello world"}"#)).unwrap(), + "These are our parameters: {\"Hello world\"}" + ); + } + + #[test] + fn entry_point_is_async_with_nested_async() { + init_test(); + + let mut js = Ssr::from( + r##" + const asyncFn = async () => { + return "Hello world" + } + var SSR = {x: async () => { + return await asyncFn() + }}; + "## + .to_string(), + "SSR", + ) + .unwrap(); + + assert_eq!( + js.render_to_string(Some(r#"{"Hello world"}"#)).unwrap(), + "Hello world" + ); + } }