Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Streamed SSR Response #2697

Merged
merged 59 commits into from
Jul 1, 2022
Merged
Show file tree
Hide file tree
Changes from 6 commits
Commits
Show all changes
59 commits
Select commit Hold shift + click to select a range
199ca5e
yew::platform?
futursolo May 11, 2022
e6fd176
Stream Response.
futursolo May 22, 2022
a652578
Migrate example
futursolo May 22, 2022
bb41baa
Remove old implementation.
futursolo May 22, 2022
5c5f9e0
Remove extra implementation.
futursolo May 22, 2022
fca682d
Prefer String instead of Cow.
futursolo May 22, 2022
9f82906
Fix MSRV.
futursolo May 22, 2022
3eb9f3b
Fix trybuild.
futursolo May 22, 2022
89cce2f
Merge branch 'master' into platform
futursolo May 22, 2022
bcf4182
Optimise Memory Allocation.
futursolo May 22, 2022
ff0ad94
More optimisation.
futursolo May 22, 2022
e5b76a7
BufWriter.
futursolo May 22, 2022
d4b15f6
Fix tests.
futursolo May 22, 2022
39f1878
Optimise BufWriter.
futursolo May 24, 2022
511b1c8
Remove more allocations.
futursolo May 24, 2022
7f2f46a
Merge branch 'master' into platform
futursolo May 24, 2022
13827e9
Allow setting of buffer capacity.
futursolo May 24, 2022
8b159fc
Fix capacity size.
futursolo May 24, 2022
65a4c20
Fix capacity size.
futursolo May 24, 2022
03e10a6
Remove unneeded const notation.
futursolo May 24, 2022
f8f198b
Fix macro tests.
futursolo May 24, 2022
7c5c7bd
Merge branch 'master' into platform
futursolo May 27, 2022
45d218c
Slightly optimises BufWriter committing logic.
futursolo May 27, 2022
12c0e38
Optimise Implementation.
futursolo Jun 12, 2022
49bbd55
Move BufWriter to a separate file.
futursolo Jun 12, 2022
c01fced
Merge branch 'master' into platform
futursolo Jun 12, 2022
c878adb
Additional Implementation Note.
futursolo Jun 12, 2022
6811275
merge branch 'master' into platform
futursolo Jun 20, 2022
52ee6cc
Merge branch 'master' into platform
futursolo Jun 23, 2022
8694cc9
Merge branch 'master' into platform
futursolo Jun 24, 2022
b784c60
Merge branch 'master' into platform
futursolo Jun 24, 2022
88085ac
Adjust API so it matches `std::channel::mpsc::channel`.
futursolo Jun 24, 2022
08e3bad
Fix feature soundness.
futursolo Jun 24, 2022
05d97da
Make a compatibility layer on channels.
futursolo Jun 25, 2022
6904da4
Fix clippy.
futursolo Jun 25, 2022
848f8f7
Fix feature soundness.
futursolo Jun 25, 2022
64d2ee4
Fix CI.
futursolo Jun 25, 2022
47caa52
Inlining.
futursolo Jun 25, 2022
f277d41
Add documentation.
futursolo Jun 25, 2022
704f2a8
Punctuation.
futursolo Jun 25, 2022
dacdce8
Switch to tokio channel.
futursolo Jun 25, 2022
901cffd
Remvoe pin-project.
futursolo Jun 25, 2022
4511ffd
Fix feature soundness.
futursolo Jun 25, 2022
318f1ef
Typo.
futursolo Jun 25, 2022
d656fc7
Move io to platform.
futursolo Jun 25, 2022
6b4f31f
Tokio does not compile.
futursolo Jun 25, 2022
8cc2a57
Fix workflow.
futursolo Jun 25, 2022
4e03969
Restore wrongly removed docs.
futursolo Jun 25, 2022
5f90233
Does tokio work?
futursolo Jun 25, 2022
a6a3423
Switch back to tokio.
futursolo Jun 25, 2022
ecbcaca
Remove pin-project.
futursolo Jun 25, 2022
01494ce
Use cargo resolver 2.
futursolo Jun 25, 2022
466bed4
Add panic notice.
futursolo Jun 26, 2022
4855da4
Update documentation.
futursolo Jun 26, 2022
e5cd82c
Merge branch 'master' into platform
futursolo Jun 28, 2022
0d477f4
Properties does not have to be send.
futursolo Jun 28, 2022
9c36639
Fix capacity checking as pointed in the review.
futursolo Jun 29, 2022
ba23671
Implementation order.
futursolo Jun 29, 2022
8462ec4
Update note.
futursolo Jun 29, 2022
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 1 addition & 2 deletions examples/function_router/src/app.rs
Original file line number Diff line number Diff line change
@@ -1,7 +1,6 @@
use std::collections::HashMap;

use yew::prelude::*;
use yew::virtual_dom::AttrValue;
use yew_router::history::{AnyHistory, History, MemoryHistory};
use yew_router::prelude::*;

Expand Down Expand Up @@ -55,7 +54,7 @@ pub fn App() -> Html {

#[derive(Properties, PartialEq, Debug)]
pub struct ServerAppProps {
pub url: AttrValue,
pub url: String,
pub queries: HashMap<String, String>,
}

Expand Down
5 changes: 2 additions & 3 deletions examples/simple_ssr/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,8 @@ yew = { path = "../../packages/yew", features = ["ssr", "hydration"] }
reqwest = { version = "0.11.8", features = ["json"] }
serde = { version = "1.0.132", features = ["derive"] }
uuid = { version = "1.0.0", features = ["serde"] }
futures = "0.3"
bytes = "1.0"

[target.'cfg(target_arch = "wasm32")'.dependencies]
wasm-bindgen-futures = "0.4"
Expand All @@ -19,7 +21,4 @@ log = "0.4"
[target.'cfg(not(target_arch = "wasm32"))'.dependencies]
tokio = { version = "1.15.0", features = ["full"] }
warp = "0.3"
num_cpus = "1.13"
tokio-util = { version = "0.7", features = ["rt"] }
once_cell = "1.5"
clap = { version = "3.1.7", features = ["derive"] }
43 changes: 24 additions & 19 deletions examples/simple_ssr/src/bin/simple_ssr_server.rs
Original file line number Diff line number Diff line change
@@ -1,13 +1,13 @@
use std::error::Error;
use std::path::PathBuf;

use bytes::Bytes;
use clap::Parser;
use once_cell::sync::Lazy;
use futures::stream::{self, Stream, StreamExt};
use simple_ssr::App;
use tokio_util::task::LocalPoolHandle;
use warp::Filter;

// We spawn a local pool that is as big as the number of cpu threads.
static LOCAL_POOL: Lazy<LocalPoolHandle> = Lazy::new(|| LocalPoolHandle::new(num_cpus::get()));
type BoxedError = Box<dyn Error + Send + Sync + 'static>;

/// A basic example
#[derive(Parser, Debug)]
Expand All @@ -17,19 +17,18 @@ struct Opt {
dir: PathBuf,
}

async fn render(index_html_s: &str) -> String {
let content = LOCAL_POOL
.spawn_pinned(move || async move {
let renderer = yew::ServerRenderer::<App>::new();

renderer.render().await
})
.await
.expect("the task has failed.");

// Good enough for an example, but developers should avoid the replace and extra allocation
// here in an actual app.
index_html_s.replace("<body>", &format!("<body>{}", content))
async fn render(
index_html_before: String,
index_html_after: String,
) -> Box<dyn Stream<Item = Result<Bytes, BoxedError>> + Send> {
let renderer = yew::ServerRenderer::<App>::new();

Box::new(
stream::once(async move { index_html_before })
.chain(renderer.render_streamed().await)
.chain(stream::once(async move { index_html_after }))
.map(|m| Result::<_, BoxedError>::Ok(m.into())),
)
}

#[tokio::main]
Expand All @@ -40,10 +39,16 @@ async fn main() {
.await
.expect("failed to read index.html");

let (index_html_before, index_html_after) = index_html_s.split_once("<body>").unwrap();
let mut index_html_before = index_html_before.to_owned();
index_html_before.push_str("<body>");
let index_html_after = index_html_after.to_owned();

let html = warp::path::end().then(move || {
let index_html_s = index_html_s.clone();
let index_html_before = index_html_before.clone();
let index_html_after = index_html_after.clone();

async move { warp::reply::html(render(&index_html_s).await) }
async move { warp::reply::html(render(index_html_before, index_html_after).await) }
});

let routes = html.or(warp::fs::dir(opts.dir));
Expand Down
4 changes: 1 addition & 3 deletions examples/ssr_router/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ edition = "2021"
yew = { path = "../../packages/yew", features = ["ssr", "hydration"] }
function_router = { path = "../function_router" }
log = "0.4"
futures = "0.3"

[target.'cfg(target_arch = "wasm32")'.dependencies]
wasm-bindgen-futures = "0.4"
Expand All @@ -20,7 +21,4 @@ axum = "0.5"
tower = { version = "0.4", features = ["make"] }
tower-http = { version = "0.3", features = ["fs"] }
env_logger = "0.9"
num_cpus = "1.13"
tokio-util = { version = "0.7", features = ["rt"] }
once_cell = "1.5"
clap = { version = "3.1.7", features = ["derive"] }
58 changes: 31 additions & 27 deletions examples/ssr_router/src/bin/ssr_router_server.rs
Original file line number Diff line number Diff line change
@@ -1,24 +1,21 @@
use std::collections::HashMap;
use std::convert::Infallible;
use std::path::PathBuf;

use axum::body::Body;
use axum::body::{Body, StreamBody};
use axum::error_handling::HandleError;
use axum::extract::Query;
use axum::handler::Handler;
use axum::http::{Request, StatusCode};
use axum::response::Html;
use axum::response::IntoResponse;
use axum::routing::get;
use axum::{Extension, Router};
use clap::Parser;
use function_router::{ServerApp, ServerAppProps};
use once_cell::sync::Lazy;
use tokio_util::task::LocalPoolHandle;
use futures::stream::{self, StreamExt};
use tower::ServiceExt;
use tower_http::services::ServeDir;

// We spawn a local pool that is as big as the number of cpu threads.
static LOCAL_POOL: Lazy<LocalPoolHandle> = Lazy::new(|| LocalPoolHandle::new(num_cpus::get()));

/// A basic example
#[derive(Parser, Debug)]
struct Opt {
Expand All @@ -28,29 +25,21 @@ struct Opt {
}

async fn render(
Extension(index_html_s): Extension<String>,
Extension((index_html_before, index_html_after)): Extension<(String, String)>,
url: Request<Body>,
Query(queries): Query<HashMap<String, String>>,
) -> Html<String> {
) -> impl IntoResponse {
let url = url.uri().to_string();

let content = LOCAL_POOL
.spawn_pinned(move || async move {
let server_app_props = ServerAppProps {
url: url.into(),
queries,
};

let renderer = yew::ServerRenderer::<ServerApp>::with_props(server_app_props);

renderer.render().await
})
.await
.expect("the task has failed.");
let server_app_props = ServerAppProps { url, queries };
let renderer = yew::ServerRenderer::<ServerApp>::with_props(server_app_props);

// Good enough for an example, but developers should avoid the replace and extra allocation
// here in an actual app.
Html(index_html_s.replace("<body>", &format!("<body>{}", content)))
StreamBody::new(
stream::once(async move { index_html_before })
.chain(renderer.render_streamed().await)
.chain(stream::once(async move { index_html_after }))
.map(Result::<_, Infallible>::Ok),
)
}

#[tokio::main]
Expand All @@ -63,6 +52,12 @@ async fn main() {
.await
.expect("failed to read index.html");

let (index_html_before, index_html_after) = index_html_s.split_once("<body>").unwrap();
let mut index_html_before = index_html_before.to_owned();
index_html_before.push_str("<body>");

let index_html_after = index_html_after.to_owned();

let handle_error = |e| async move {
(
StatusCode::INTERNAL_SERVER_ERROR,
Expand All @@ -73,13 +68,22 @@ async fn main() {
let app = Router::new()
.route("/api/test", get(|| async move { "Hello World" }))
// needed because /~https://github.com/tower-rs/tower-http/issues/262
.route("/", get(render))
.route(
"/",
get(render.layer(Extension((
index_html_before.clone(),
index_html_after.clone(),
)))),
)
.fallback(HandleError::new(
ServeDir::new(opts.dir)
.append_index_html_on_directories(false)
.fallback(
render
.layer(Extension(index_html_s))
.layer(Extension((
index_html_before.clone(),
index_html_after.clone(),
)))
.into_service()
.map_err(|err| -> std::io::Error { match err {} }),
),
Expand Down
10 changes: 6 additions & 4 deletions packages/yew/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,7 @@ keywords = ["web", "webasm", "javascript"]
categories = ["gui", "wasm", "web-programming"]
description = "A framework for making client-side single-page apps"
readme = "../../README.md"
rust-version = "1.56.0"
rust-version = "1.60.0"
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The MSRV has been updated to 1.60 due to the requirement of namespaced dependency.


[dependencies]
console_error_panic_hook = "0.1"
Expand Down Expand Up @@ -70,6 +70,9 @@ wasm-bindgen-futures = "0.4"

[target.'cfg(not(target_arch = "wasm32"))'.dependencies]
tokio = { version = "1.15.0", features = ["rt"], optional = true }
num_cpus = { version = "1.13", optional = true }
tokio-util = { version = "0.7", features = ["rt"], optional = true }
once_cell = "1"

[dev-dependencies]
wasm-bindgen-test = "0.3"
Expand All @@ -87,9 +90,8 @@ features = [
]

[features]
# TODO: `dep:` syntax only supported with MSRV 1.60, would be more precise
# tokio = ["dep:tokio"]
ssr = ["futures", "html-escape"] # dep:html-escape
tokio = ["dep:tokio", "dep:num_cpus", "dep:tokio-util"]
ssr = ["dep:futures", "dep:html-escape"]
csr = []
hydration = ["csr"]
default = []
Expand Down
2 changes: 1 addition & 1 deletion packages/yew/Makefile.toml
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
[tasks.native-test]
command = "cargo"
args = ["test", "--features", "csr,ssr,hydration"]
args = ["test", "--features", "csr,ssr,hydration,tokio"]

[tasks.wasm-test]
command = "wasm-pack"
Expand Down
24 changes: 14 additions & 10 deletions packages/yew/src/html/component/scope.rs
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,7 @@ use super::lifecycle::ComponentState;
use super::BaseComponent;
use crate::callback::Callback;
use crate::context::{ContextHandle, ContextProvider};
use crate::io_coop::spawn_local;
use crate::platform::spawn_local;
#[cfg(any(feature = "csr", feature = "ssr"))]
use crate::scheduler::Shared;

Expand Down Expand Up @@ -260,6 +260,7 @@ impl<COMP: BaseComponent> Scope<COMP> {

#[cfg(feature = "ssr")]
mod feat_ssr {
use futures::channel::mpsc::UnboundedSender;
use futures::channel::oneshot;

use super::*;
Expand All @@ -270,14 +271,16 @@ mod feat_ssr {
use crate::virtual_dom::Collectable;

impl<COMP: BaseComponent> Scope<COMP> {
pub(crate) async fn render_to_string(
self,
w: &mut String,
pub(crate) async fn render_into_stream(
&self,
tx: &mut UnboundedSender<String>,
props: Rc<COMP::Properties>,
hydratable: bool,
) {
let (tx, rx) = oneshot::channel();
let state = ComponentRenderState::Ssr { sender: Some(tx) };
let (html_tx, html_rx) = oneshot::channel();
let state = ComponentRenderState::Ssr {
sender: Some(html_tx),
};

scheduler::push_component_create(
self.id,
Expand All @@ -295,16 +298,17 @@ mod feat_ssr {
let collectable = Collectable::for_component::<COMP>();

if hydratable {
collectable.write_open_tag(w);
collectable.write_open_tag(tx);
}

let html = rx.await.unwrap();
let html = html_rx.await.unwrap();

let self_any_scope = AnyScope::from(self.clone());
html.render_to_string(w, &self_any_scope, hydratable).await;
html.render_into_stream(tx, &self_any_scope, hydratable)
.await;

if hydratable {
collectable.write_close_tag(w);
collectable.write_close_tag(tx);
}

scheduler::push_component_destroy(Box::new(DestroyRunner {
Expand Down
32 changes: 0 additions & 32 deletions packages/yew/src/io_coop.rs

This file was deleted.

6 changes: 2 additions & 4 deletions packages/yew/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -24,9 +24,7 @@
//! - `csr`: Enables Client-side Rendering support and [`Renderer`]. Only enable this feature if you
//! are making a Yew application (not a library).
//! - `ssr`: Enables Server-side Rendering support and [`ServerRenderer`].
//! - `tokio`: Enables future-based APIs on non-wasm32 targets with tokio runtime. (You may want to
//! enable this if your application uses future-based APIs and it does not compile / lint on
//! non-wasm32 targets.)
//! - `tokio`: Enables future-based APIs on non-wasm32 targets with tokio runtime.
//! - `hydration`: Enables Hydration support.
//!
//! ## Example
Expand Down Expand Up @@ -280,7 +278,7 @@ pub mod context;
mod dom_bundle;
pub mod functional;
pub mod html;
mod io_coop;
pub mod platform;
pub mod scheduler;
mod sealed;
#[cfg(feature = "ssr")]
Expand Down
Loading