From e8aa9b24f1c95b9fe223370deac769f0865b9867 Mon Sep 17 00:00:00 2001 From: Greg Johnston Date: Mon, 17 Jul 2023 21:59:20 -0400 Subject: [PATCH] fix: memory leak in `leptos_axum` (#1374) --- integrations/axum/src/lib.rs | 30 +++++++----------------------- leptos_reactive/src/runtime.rs | 11 ++++++----- 2 files changed, 13 insertions(+), 28 deletions(-) diff --git a/integrations/axum/src/lib.rs b/integrations/axum/src/lib.rs index da1a6998bc..d569fc45a8 100644 --- a/integrations/axum/src/lib.rs +++ b/integrations/axum/src/lib.rs @@ -605,7 +605,6 @@ where let res_options3 = default_res_options.clone(); let local_pool = get_leptos_pool(); let (tx, rx) = futures::channel::mpsc::channel(8); - let (runtime_tx, runtime_rx) = futures::channel::oneshot::channel(); let current_span = tracing::Span::current(); local_pool.spawn_pinned(move || async move { @@ -630,17 +629,12 @@ where replace_blocks ); - runtime_tx.send(runtime).expect("should be able to send runtime"); - forward_stream(&options, res_options2, bundle, runtime, scope, tx).await; + + runtime.dispose(); }.instrument(current_span)); - async move { - let runtime = runtime_rx - .await - .expect("runtime should be sent by renderer"); - generate_response(res_options3, rx, runtime).await - } + generate_response(res_options3, rx) }) } } @@ -649,7 +643,6 @@ where async fn generate_response( res_options: ResponseOptions, rx: Receiver, - runtime: RuntimeId, ) -> Response> { let mut stream = Box::pin(rx.map(|html| Ok(Bytes::from(html)))); @@ -662,11 +655,7 @@ async fn generate_response( let complete_stream = futures::stream::iter([first_chunk.unwrap(), second_chunk.unwrap()]) - .chain(stream) - .chain(futures::stream::once(async move { - runtime.dispose(); - Ok(Default::default()) - })); + .chain(stream); let mut res = Response::new(StreamBody::new( Box::pin(complete_stream) as PinnedHtmlStream @@ -781,8 +770,6 @@ where let full_path = format!("http://leptos.dev{path}"); let (tx, rx) = futures::channel::mpsc::channel(8); - let (runtime_tx, runtime_rx) = - futures::channel::oneshot::channel(); let local_pool = get_leptos_pool(); let current_span = tracing::Span::current(); local_pool.spawn_pinned(|| async move { @@ -802,15 +789,12 @@ where add_context, ); - runtime_tx.send(runtime).expect("should be able to send runtime"); - forward_stream(&options, res_options2, bundle, runtime, scope, tx).await; + + runtime.dispose(); }.instrument(current_span)); - let runtime = runtime_rx - .await - .expect("runtime should be sent by renderer"); - generate_response(res_options3, rx, runtime).await + generate_response(res_options3, rx).await } }) } diff --git a/leptos_reactive/src/runtime.rs b/leptos_reactive/src/runtime.rs index 9750f304f1..ba839c3a9e 100644 --- a/leptos_reactive/src/runtime.rs +++ b/leptos_reactive/src/runtime.rs @@ -415,11 +415,12 @@ pub struct RuntimeId; impl RuntimeId { /// Removes the runtime, disposing all its child [`Scope`](crate::Scope)s. pub fn dispose(self) { - cfg_if! { - if #[cfg(not(any(feature = "csr", feature = "hydrate")))] { - let runtime = RUNTIMES.with(move |runtimes| runtimes.borrow_mut().remove(self)); - drop(runtime); - } + #[cfg(not(any(feature = "csr", feature = "hydrate")))] + { + let runtime = RUNTIMES.with(move |runtimes| runtimes.borrow_mut().remove(self)) + .expect("Attempted to dispose of a reactive runtime that was not found. This suggests \ + a possible memory leak. Please open an issue with details at https://github.com/leptos-rs/leptos"); + drop(runtime); } }