//! A shared pool of renderers for efficient server side rendering. use crate::document::ServerDocument; use crate::streaming::{Mount, StreamingRenderer}; use dioxus_interpreter_js::INITIALIZE_STREAMING_JS; use dioxus_isrg::{CachedRender, RenderFreshness}; use dioxus_lib::document::Document; use dioxus_ssr::Renderer; use futures_channel::mpsc::Sender; use futures_util::{Stream, StreamExt}; use std::rc::Rc; use std::sync::Arc; use std::sync::RwLock; use std::{collections::HashMap, future::Future}; use tokio::task::JoinHandle; use crate::prelude::*; use dioxus_lib::prelude::*; /// A suspense boundary that is pending with a placeholder in the client struct PendingSuspenseBoundary { mount: Mount, children: Vec, } /// Spawn a task in the background. If wasm is enabled, this will use the single threaded tokio runtime fn spawn_platform(f: impl FnOnce() -> Fut + Send + 'static) -> JoinHandle where Fut: Future + 'static, Fut::Output: Send + 'static, { #[cfg(not(target_arch = "wasm32"))] { use tokio_util::task::LocalPoolHandle; static TASK_POOL: std::sync::OnceLock = std::sync::OnceLock::new(); let pool = TASK_POOL.get_or_init(|| { let threads = std::thread::available_parallelism() .unwrap_or(std::num::NonZeroUsize::new(1).unwrap()); LocalPoolHandle::new(threads.into()) }); pool.spawn_pinned(f) } #[cfg(target_arch = "wasm32")] { tokio::task::spawn_local(f()) } } struct SsrRendererPool { renderers: RwLock>, incremental_cache: Option>, } impl SsrRendererPool { fn new( initial_size: usize, incremental: Option, ) -> Self { let renderers = RwLock::new((0..initial_size).map(|_| pre_renderer()).collect()); Self { renderers, incremental_cache: incremental.map(|cache| RwLock::new(cache.build())), } } /// Look for a cached route in the incremental cache and send it into the render channel if it exists fn check_cached_route( &self, route: &str, render_into: &mut Sender>, ) -> Option { if let Some(incremental) = &self.incremental_cache { if let Ok(mut incremental) = incremental.write() { match incremental.get(route) { Ok(Some(cached_render)) => { let CachedRender { freshness, response, .. } = cached_render; _ = render_into.start_send(String::from_utf8(response.to_vec()).map_err( |err| dioxus_isrg::IncrementalRendererError::Other(Box::new(err)), )); return Some(freshness); } Err(e) => { tracing::error!( "Failed to get route \"{route}\" from incremental cache: {e}" ); } _ => {} } } } None } /// Render a virtual dom into a stream. This method will return immediately and continue streaming the result in the background /// The streaming is canceled when the stream the function returns is dropped async fn render_to( self: Arc, cfg: &ServeConfig, route: String, virtual_dom_factory: impl FnOnce() -> VirtualDom + Send + Sync + 'static, server_context: &DioxusServerContext, ) -> Result< ( RenderFreshness, impl Stream>, ), dioxus_isrg::IncrementalRendererError, > { struct ReceiverWithDrop { receiver: futures_channel::mpsc::Receiver< Result, >, cancel_task: Option>, } impl Stream for ReceiverWithDrop { type Item = Result; fn poll_next( mut self: std::pin::Pin<&mut Self>, cx: &mut std::task::Context<'_>, ) -> std::task::Poll> { self.receiver.poll_next_unpin(cx) } } // When we drop the stream, we need to cancel the task that is feeding values to the stream impl Drop for ReceiverWithDrop { fn drop(&mut self) { if let Some(cancel_task) = self.cancel_task.take() { cancel_task.abort(); } } } let (mut into, rx) = futures_channel::mpsc::channel::< Result, >(1000); // before we even spawn anything, we can check synchronously if we have the route cached if let Some(freshness) = self.check_cached_route(&route, &mut into) { return Ok(( freshness, ReceiverWithDrop { receiver: rx, cancel_task: None, }, )); } let wrapper = FullstackHTMLTemplate { cfg: cfg.clone() }; let server_context = server_context.clone(); let mut renderer = self .renderers .write() .unwrap() .pop() .unwrap_or_else(pre_renderer); let myself = self.clone(); let join_handle = spawn_platform(move || async move { let mut virtual_dom = virtual_dom_factory(); let document = std::rc::Rc::new(crate::document::server::ServerDocument::default()); virtual_dom.provide_root_context(document.clone()); virtual_dom.provide_root_context(Rc::new( dioxus_history::MemoryHistory::with_initial_path(&route), ) as Rc); virtual_dom.provide_root_context(document.clone() as std::rc::Rc); // poll the future, which may call server_context() tracing::info!("Rebuilding vdom"); with_server_context(server_context.clone(), || virtual_dom.rebuild_in_place()); let mut pre_body = String::new(); if let Err(err) = wrapper.render_head(&mut pre_body, &virtual_dom) { _ = into.start_send(Err(err)); return; } let stream = Arc::new(StreamingRenderer::new(pre_body, into)); let scope_to_mount_mapping = Arc::new(RwLock::new(HashMap::new())); renderer.pre_render = true; { let scope_to_mount_mapping = scope_to_mount_mapping.clone(); let stream = stream.clone(); // We use a stack to keep track of what suspense boundaries we are nested in to add children to the correct boundary // The stack starts with the root scope because the root is a suspense boundary let pending_suspense_boundaries_stack = RwLock::new(vec![]); renderer.set_render_components(move |renderer, to, vdom, scope| { let is_suspense_boundary = SuspenseContext::downcast_suspense_boundary_from_scope( &vdom.runtime(), scope, ) .filter(|s| s.has_suspended_tasks()) .is_some(); if is_suspense_boundary { let mount = stream.render_placeholder( |to| { { pending_suspense_boundaries_stack .write() .unwrap() .push(scope); } let out = renderer.render_scope(to, vdom, scope); { pending_suspense_boundaries_stack.write().unwrap().pop(); } out }, &mut *to, )?; // Add the suspense boundary to the list of pending suspense boundaries // We will replace the mount with the resolved contents later once the suspense boundary is resolved let mut scope_to_mount_mapping_write = scope_to_mount_mapping.write().unwrap(); scope_to_mount_mapping_write.insert( scope, PendingSuspenseBoundary { mount, children: vec![], }, ); // Add the scope to the list of children of the parent suspense boundary let pending_suspense_boundaries_stack = pending_suspense_boundaries_stack.read().unwrap(); // If there is a parent suspense boundary, add the scope to the list of children // This suspense boundary will start capturing errors when the parent is resolved if let Some(parent) = pending_suspense_boundaries_stack.last() { let parent = scope_to_mount_mapping_write.get_mut(parent).unwrap(); parent.children.push(scope); } // Otherwise this is a root suspense boundary, so we need to start capturing errors immediately else { vdom.in_runtime(|| { start_capturing_errors(scope); }); } } else { renderer.render_scope(to, vdom, scope)? } Ok(()) }); } macro_rules! throw_error { ($e:expr) => { stream.close_with_error($e); return; }; } // Render the initial frame with loading placeholders let mut initial_frame = renderer.render(&virtual_dom); // Along with the initial frame, we render the html after the main element, but before the body tag closes. This should include the script that starts loading the wasm bundle. if let Err(err) = wrapper.render_after_main(&mut initial_frame, &virtual_dom) { throw_error!(err); } stream.render(initial_frame); // After the initial render, we need to resolve suspense while virtual_dom.suspended_tasks_remaining() { ProvideServerContext::new( virtual_dom.wait_for_suspense_work(), server_context.clone(), ) .await; let resolved_suspense_nodes = ProvideServerContext::new( virtual_dom.render_suspense_immediate(), server_context.clone(), ) .await; // Just rerender the resolved nodes for scope in resolved_suspense_nodes { let pending_suspense_boundary = { let mut lock = scope_to_mount_mapping.write().unwrap(); lock.remove(&scope) }; // If the suspense boundary was immediately removed, it may not have a mount. We can just skip resolving it if let Some(pending_suspense_boundary) = pending_suspense_boundary { let mut resolved_chunk = String::new(); // After we replace the placeholder in the dom with javascript, we need to send down the resolved data so that the client can hydrate the node let render_suspense = |into: &mut String| { renderer.reset_hydration(); renderer.render_scope(into, &virtual_dom, scope) }; let resolved_data = serialize_server_data(&virtual_dom, scope); if let Err(err) = stream.replace_placeholder( pending_suspense_boundary.mount, render_suspense, resolved_data, &mut resolved_chunk, ) { throw_error!(dioxus_isrg::IncrementalRendererError::RenderError(err)); } stream.render(resolved_chunk); // Freeze the suspense boundary to prevent future reruns of any child nodes of the suspense boundary if let Some(suspense) = SuspenseContext::downcast_suspense_boundary_from_scope( &virtual_dom.runtime(), scope, ) { suspense.freeze(); // Go to every child suspense boundary and add an error boundary. Since we cannot rerun any nodes above the child suspense boundary, // we need to capture the errors and send them to the client as it resolves virtual_dom.in_runtime(|| { for &suspense_scope in pending_suspense_boundary.children.iter() { start_capturing_errors(suspense_scope); } }); } } } } // After suspense is done, we render the html after the body let mut post_streaming = String::new(); if let Err(err) = wrapper.render_after_body(&mut post_streaming) { throw_error!(err); } // If incremental rendering is enabled, add the new render to the cache without the streaming bits if let Some(incremental) = &self.incremental_cache { let mut cached_render = String::new(); if let Err(err) = wrapper.render_head(&mut cached_render, &virtual_dom) { throw_error!(err); } cached_render.push_str(&post_streaming); if let Ok(mut incremental) = incremental.write() { let _ = incremental.cache(route, cached_render); } } stream.render(post_streaming); renderer.reset_render_components(); myself.renderers.write().unwrap().push(renderer); }); Ok(( RenderFreshness::now(None), ReceiverWithDrop { receiver: rx, cancel_task: Some(join_handle), }, )) } } /// Start capturing errors at a suspense boundary. If the parent suspense boundary is frozen, we need to capture the errors in the suspense boundary /// and send them to the client to continue bubbling up fn start_capturing_errors(suspense_scope: ScopeId) { // Add an error boundary to the scope suspense_scope.in_runtime(provide_error_boundary); } fn serialize_server_data(virtual_dom: &VirtualDom, scope: ScopeId) -> String { // After we replace the placeholder in the dom with javascript, we need to send down the resolved data so that the client can hydrate the node // Extract any data we serialized for hydration (from server futures) let html_data = crate::html_storage::HTMLData::extract_from_suspense_boundary(virtual_dom, scope); // serialize the server state into a base64 string html_data.serialized() } /// State used in server side rendering. This utilizes a pool of [`dioxus_ssr::Renderer`]s to cache static templates between renders. #[derive(Clone)] pub struct SSRState { // We keep a pool of renderers to avoid re-creating them on every request. They are boxed to make them very cheap to move renderers: Arc, } impl SSRState { /// Create a new [`SSRState`]. pub fn new(cfg: &ServeConfig) -> Self { Self { renderers: Arc::new(SsrRendererPool::new(4, cfg.incremental.clone())), } } /// Render the application to HTML. pub async fn render<'a>( &'a self, route: String, cfg: &'a ServeConfig, virtual_dom_factory: impl FnOnce() -> VirtualDom + Send + Sync + 'static, server_context: &'a DioxusServerContext, ) -> Result< ( RenderFreshness, impl Stream>, ), dioxus_isrg::IncrementalRendererError, > { self.renderers .clone() .render_to(cfg, route, virtual_dom_factory, server_context) .await } } /// The template that wraps the body of the HTML for a fullstack page. This template contains the data needed to hydrate server functions that were run on the server. pub struct FullstackHTMLTemplate { cfg: ServeConfig, } impl FullstackHTMLTemplate { /// Create a new [`FullstackHTMLTemplate`]. pub fn new(cfg: &ServeConfig) -> Self { Self { cfg: cfg.clone() } } } impl FullstackHTMLTemplate { /// Render any content before the head of the page. pub fn render_head( &self, to: &mut R, virtual_dom: &VirtualDom, ) -> Result<(), dioxus_isrg::IncrementalRendererError> { let ServeConfig { index, .. } = &self.cfg; let title = { let document: Option> = virtual_dom.in_runtime(|| ScopeId::ROOT.consume_context()); // Collect any head content from the document provider and inject that into the head document.and_then(|document| document.title()) }; to.write_str(&index.head_before_title)?; if let Some(title) = title { to.write_str(&title)?; } else { to.write_str(&index.title)?; } to.write_str(&index.head_after_title)?; let document: Option> = virtual_dom.in_runtime(|| ScopeId::ROOT.consume_context()); if let Some(document) = document { // Collect any head content from the document provider and inject that into the head document.render(to)?; // Enable a warning when inserting contents into the head during streaming document.start_streaming(); } self.render_before_body(to)?; Ok(()) } /// Render any content before the body of the page. fn render_before_body( &self, to: &mut R, ) -> Result<(), dioxus_isrg::IncrementalRendererError> { let ServeConfig { index, .. } = &self.cfg; to.write_str(&index.close_head)?; write!(to, "")?; Ok(()) } /// Render all content after the main element of the page. pub fn render_after_main( &self, to: &mut R, virtual_dom: &VirtualDom, ) -> Result<(), dioxus_isrg::IncrementalRendererError> { let ServeConfig { index, .. } = &self.cfg; // Collect the initial server data from the root node. For most apps, no use_server_futures will be resolved initially, so this will be full on `None`s. // Sending down those Nones are still important to tell the client not to run the use_server_futures that are already running on the backend let resolved_data = serialize_server_data(virtual_dom, ScopeId::ROOT); write!( to, r#""#, )?; to.write_str(&index.post_main)?; Ok(()) } /// Render all content after the body of the page. pub fn render_after_body( &self, to: &mut R, ) -> Result<(), dioxus_isrg::IncrementalRendererError> { let ServeConfig { index, .. } = &self.cfg; to.write_str(&index.after_closing_body_tag)?; Ok(()) } /// Wrap a body in the template pub fn wrap_body( &self, to: &mut R, virtual_dom: &VirtualDom, body: impl std::fmt::Display, ) -> Result<(), dioxus_isrg::IncrementalRendererError> { self.render_head(to, virtual_dom)?; write!(to, "{body}")?; self.render_after_main(to, virtual_dom)?; self.render_after_body(to)?; Ok(()) } } fn pre_renderer() -> Renderer { let mut renderer = Renderer::default(); renderer.pre_render = true; renderer }