diff --git a/bonnie.toml b/bonnie.toml index 4c7bf9d6af..21916304ca 100644 --- a/bonnie.toml +++ b/bonnie.toml @@ -197,7 +197,8 @@ test.subcommands.cli.desc = "runs the cli tests (all are long-running, this will test.subcommands.example-all-integrations.cmd = [ "EXAMPLE_INTEGRATION=actix-web bonnie dev example %category %example test", "EXAMPLE_INTEGRATION=warp bonnie dev example %category %example test", - "EXAMPLE_INTEGRATION=axum bonnie dev example %category %example test" + "EXAMPLE_INTEGRATION=axum bonnie dev example %category %example test", + "EXAMPLE_INTEGRATION=rocket bonnie dev example %category %example test" ] test.subcommands.example-all-integrations.args = [ "category", "example" ] test.subcommands.example-all-integrations.desc = "tests a single example with all integrations (assumes geckodriver running in background)" diff --git a/examples/core/custom_server_rocket/.integration_locked b/examples/core/custom_server_rocket/.integration_locked new file mode 100644 index 0000000000..e69de29bb2 diff --git a/examples/core/custom_server_rocket/Cargo.toml b/examples/core/custom_server_rocket/Cargo.toml new file mode 100644 index 0000000000..ee53824274 --- /dev/null +++ b/examples/core/custom_server_rocket/Cargo.toml @@ -0,0 +1,22 @@ +[package] +name = "custom_server_rocket" +version = "0.4.0-beta.22" +edition = "2021" + +# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html + +[dependencies] +perseus = { path = "../../../packages/perseus", features = ["hydrate"] } +sycamore = "^0.8.1" +serde = { version = "1", features = ["derive"] } +serde_json = "1" + +[target.'cfg(engine)'.dev-dependencies] +fantoccini = "0.17" + +[target.'cfg(engine)'.dependencies] +tokio = { version = "1", features = ["macros", "rt", "rt-multi-thread"] } +perseus-rocket = { path = "../../../packages/perseus-rocket", features = [ "dflt-server" ] } +rocket = "0.5.0-rc.2" + +[target.'cfg(client)'.dependencies] diff --git a/examples/core/custom_server_rocket/README.md b/examples/core/custom_server_rocket/README.md new file mode 100644 index 0000000000..22c2f3429e --- /dev/null +++ b/examples/core/custom_server_rocket/README.md @@ -0,0 +1,3 @@ +# Custom Server Rocket Example + +This is an example of setting up a custom server using rocket with Perseus, with one example API route. diff --git a/examples/core/custom_server_rocket/src/main.rs b/examples/core/custom_server_rocket/src/main.rs new file mode 100644 index 0000000000..e9bc4ed662 --- /dev/null +++ b/examples/core/custom_server_rocket/src/main.rs @@ -0,0 +1,50 @@ +mod templates; + +use perseus::prelude::*; + +#[cfg(engine)] +mod api { + use rocket::get; + + #[get("/hello")] + pub fn hello() -> String { + "Hello from an api".to_string() + } +} + +#[cfg(engine)] +pub async fn dflt_server< + M: perseus::stores::MutableStore + 'static, + T: perseus::i18n::TranslationsManager + 'static, +>( + turbine: &'static perseus::turbine::Turbine, + opts: perseus::server::ServerOptions, + (host, port): (String, u16), +) { + use perseus_rocket::perseus_base_app; + use rocket::routes; + + let addr = host.parse().expect("Invalid address provided to bind to."); + + let mut config = rocket::Config::default(); + + let mut app = perseus_base_app(turbine, opts).await; + app = app.mount("/api", routes![api::hello]); + + config.address = addr; + config.port = port; + app = app.configure(config); + + match app.launch().await { + Err(e) => println!("Error lauching rocket app: {}", e), + _ => (), + } +} + +#[perseus::main(dflt_server)] +pub fn main() -> PerseusApp { + PerseusApp::new() + .template(crate::templates::index::get_template()) + .template(crate::templates::about::get_template()) + .error_views(ErrorViews::unlocalized_development_default()) +} diff --git a/examples/core/custom_server_rocket/src/templates/about.rs b/examples/core/custom_server_rocket/src/templates/about.rs new file mode 100644 index 0000000000..12989c60c7 --- /dev/null +++ b/examples/core/custom_server_rocket/src/templates/about.rs @@ -0,0 +1,19 @@ +use perseus::prelude::*; +use sycamore::prelude::*; + +fn about_page(cx: Scope) -> View { + view! { cx, + p { "About." } + } +} + +#[engine_only_fn] +fn head(cx: Scope) -> View { + view! { cx, + title { "About Page | Perseus Example – Basic" } + } +} + +pub fn get_template() -> Template { + Template::build("about").view(about_page).head(head).build() +} diff --git a/examples/core/custom_server_rocket/src/templates/index.rs b/examples/core/custom_server_rocket/src/templates/index.rs new file mode 100644 index 0000000000..09ef5e05f7 --- /dev/null +++ b/examples/core/custom_server_rocket/src/templates/index.rs @@ -0,0 +1,39 @@ +use perseus::prelude::*; +use serde::{Deserialize, Serialize}; +use sycamore::prelude::*; + +#[derive(Serialize, Deserialize, ReactiveState, Clone)] +#[rx(alias = "IndexPageStateRx")] +struct IndexPageState { + greeting: String, +} + +#[auto_scope] +fn index_page(cx: Scope, state: &IndexPageStateRx) -> View { + view! { cx, + p { (state.greeting.get()) } + a(href = "about", id = "about-link") { "About!" } + } +} + +#[engine_only_fn] +fn head(cx: Scope, _props: IndexPageState) -> View { + view! { cx, + title { "Index Page | Perseus Example – Basic" } + } +} + +#[engine_only_fn] +async fn get_build_state(_info: StateGeneratorInfo<()>) -> IndexPageState { + IndexPageState { + greeting: "Hello World!".to_string(), + } +} + +pub fn get_template() -> Template { + Template::build("index") + .build_state_fn(get_build_state) + .view_with_state(index_page) + .head_with_state(head) + .build() +} diff --git a/examples/core/custom_server_rocket/src/templates/mod.rs b/examples/core/custom_server_rocket/src/templates/mod.rs new file mode 100644 index 0000000000..9b9cf18fc5 --- /dev/null +++ b/examples/core/custom_server_rocket/src/templates/mod.rs @@ -0,0 +1,2 @@ +pub mod about; +pub mod index; diff --git a/packages/perseus-integration/Cargo.toml b/packages/perseus-integration/Cargo.toml index 9054b2be8b..dcc78e1074 100644 --- a/packages/perseus-integration/Cargo.toml +++ b/packages/perseus-integration/Cargo.toml @@ -9,6 +9,7 @@ edition = "2021" perseus-actix-web = { path = "../perseus-actix-web", features = [ "dflt-server" ], optional = true } perseus-warp = { path = "../perseus-warp", features = [ "dflt-server" ], optional = true } perseus-axum = { path = "../perseus-axum", features = [ "dflt-server" ], optional = true } +perseus-rocket = { path = "../perseus-rocket", features = [ "dflt-server" ], optional = true } [features] default = [ "warp" ] @@ -16,3 +17,4 @@ default = [ "warp" ] actix-web = [ "perseus-actix-web" ] warp = [ "perseus-warp" ] axum = [ "perseus-axum" ] +rocket = [ "perseus-rocket" ] diff --git a/packages/perseus-integration/src/lib.rs b/packages/perseus-integration/src/lib.rs index 8f67d7ef1d..3a2046cf72 100644 --- a/packages/perseus-integration/src/lib.rs +++ b/packages/perseus-integration/src/lib.rs @@ -4,5 +4,7 @@ pub use perseus_actix_web::dflt_server; #[cfg(feature = "axum")] pub use perseus_axum::dflt_server; +#[cfg(feature = "rocket")] +pub use perseus_rocket::dflt_server; #[cfg(feature = "warp")] pub use perseus_warp::dflt_server; diff --git a/packages/perseus-rocket/Cargo.toml b/packages/perseus-rocket/Cargo.toml new file mode 100644 index 0000000000..cad02d3c77 --- /dev/null +++ b/packages/perseus-rocket/Cargo.toml @@ -0,0 +1,24 @@ +[package] +name = "perseus-rocket" +version = "0.4.0-beta.22" +edition = "2021" +description = "An integration that makes the Perseus framework easy to use with Rocket." +authors = ["Miroito "] +license = "MIT" +repository = "https://github.com/framesurge/perseus" +homepage = "https://framesurge.sh/perseus" +readme = "./README.md" +keywords = ["wasm", "frontend", "webdev", "ssg", "ssr"] +categories = ["wasm", "web-programming::http-server", "development-tools", "asynchronous", "gui"] + +[dependencies] +perseus = { path = "../perseus", version = "0.4.0-beta.21"} +rocket = "0.5.0-rc.2" + +[features] +# Enables the default server configuration, which provides a convenience function if you're not adding any extra routes +dflt-server = [] + +[package.metadata.docs.rs] +rustc-args = ["--cfg=engine"] +rustdoc-args = ["--cfg=engine"] diff --git a/packages/perseus-rocket/README.md b/packages/perseus-rocket/README.md new file mode 100644 index 0000000000..fb1fae06f9 --- /dev/null +++ b/packages/perseus-rocket/README.md @@ -0,0 +1,7 @@ +# Perseus Rocket Integration + +> :warning: This integration uses Rocket 0.5.0-rc.2 which is still in beta + +This is the official [Perseus](https://github.com/framesurge/perseus) integration for making serving your apps on [Rocket](https://rocket.rs/) significantly easier! + +If you're new to Perseus, you should check out [the core package](https://github.com/framesurge/perseus) first. diff --git a/packages/perseus-rocket/README.proj.md b/packages/perseus-rocket/README.proj.md new file mode 120000 index 0000000000..fe84005413 --- /dev/null +++ b/packages/perseus-rocket/README.proj.md @@ -0,0 +1 @@ +../../README.md \ No newline at end of file diff --git a/packages/perseus-rocket/src/lib.rs b/packages/perseus-rocket/src/lib.rs new file mode 100644 index 0000000000..d69165b528 --- /dev/null +++ b/packages/perseus-rocket/src/lib.rs @@ -0,0 +1,323 @@ +#![doc = include_str!("../README.proj.md")] + +/*! +## Packages + +This is the API documentation for the `perseus-rocket` package, which allows Perseus apps to run on Rocket. Note that Perseus mostly uses [the book](https://framesurge.sh/perseus/en-US) for +documentation, and this should mostly be used as a secondary reference source. You can also find full usage examples [here](https://github.com/framesurge/perseus/tree/main/examples). +*/ + +#![cfg(engine)] +#![deny(missing_docs)] +#![deny(missing_debug_implementations)] + +use std::{io::Cursor, path::Path}; + +use perseus::{ + i18n::TranslationsManager, + path::PathMaybeWithLocale, + server::ServerOptions, + stores::MutableStore, + turbine::{ApiResponse as PerseusApiResponse, Turbine}, +}; +use rocket::{ + fs::{FileServer, NamedFile}, + get, + http::{Method, Status}, + response::Responder, + route::{Handler, Outcome}, + routes, + tokio::fs::File, + Build, Data, Request, Response, Rocket, Route, State, +}; + +// ----- Newtype wrapper for response implementation ----- + +#[derive(Debug)] +struct ApiResponse(PerseusApiResponse); +impl From for ApiResponse { + fn from(val: PerseusApiResponse) -> Self { + Self(val) + } +} +impl<'r> Responder<'r, 'static> for ApiResponse { + fn respond_to(self, _request: &'r rocket::Request<'_>) -> rocket::response::Result<'static> { + let mut resp_build = Response::build(); + resp_build + .status(rocket::http::Status { + code: self.0.status.into(), + }) + .sized_body(self.0.body.len(), Cursor::new(self.0.body)); + + for h in self.0.headers.iter() { + // Headers that contain non-visible ascii characters are chopped off here in + // order to make the conversion + if let Ok(value) = h.1.to_str() { + resp_build.raw_header(h.0.to_string(), value.to_string()); + } + } + + resp_build.ok() + } +} + +// ----- Simple routes ----- + +#[get("/bundle.js")] +async fn get_js_bundle(opts: &State) -> std::io::Result { + NamedFile::open(&opts.js_bundle).await +} + +#[get("/bundle.wasm")] +async fn get_wasm_bundle(opts: &State) -> std::io::Result { + NamedFile::open(&opts.wasm_bundle).await +} + +#[get("/bundle.wasm.js")] +async fn get_wasm_js_bundle(opts: &State) -> std::io::Result { + NamedFile::open(&opts.wasm_js_bundle).await +} + +// ----- Turbine dependant route handlers ----- + +async fn perseus_locale<'r, M, T>(req: &'r Request<'_>, turbine: &Turbine) -> Outcome<'r> +where + M: MutableStore + 'static, + T: TranslationsManager + 'static, +{ + match req.routed_segment(1) { + Some(locale) => Outcome::from(req, ApiResponse(turbine.get_translations(locale).await)), + _ => Outcome::Failure(Status::BadRequest), + } +} + +async fn perseus_initial_load_handler<'r, M, T>( + req: &'r Request<'_>, + turbine: &Turbine, +) -> Outcome<'r> +where + M: MutableStore + 'static, + T: TranslationsManager + 'static, +{ + // Since this is a fallback handler, we have to do everything from the request + // itself + let path = req.uri().path().to_string(); + + let mut http_req = rocket::http::hyper::Request::builder(); + http_req = http_req.method("GET"); + for h in req.headers().iter() { + http_req = http_req.header(h.name.to_string(), h.value.to_string()); + } + + match http_req.body(()) { + Ok(r) => Outcome::from( + req, + ApiResponse(turbine.get_initial_load(PathMaybeWithLocale(path), r).await), + ), + _ => Outcome::Failure(Status::BadRequest), + } +} + +async fn perseus_subsequent_load_handler<'r, M, T>( + req: &'r Request<'_>, + turbine: &Turbine, +) -> Outcome<'r> +where + M: MutableStore + 'static, + T: TranslationsManager + 'static, +{ + let locale_opt = req.routed_segment(1); + let entity_name_opt = req + .query_value::<&str>("entity_name") + .and_then(|res| res.ok()); + let was_incremental_match_opt = req + .query_value::("was_incremental_match") + .and_then(|res| res.ok()); + + let (locale, entity_name, was_incremental_match) = + match (locale_opt, entity_name_opt, was_incremental_match_opt) { + (Some(l), Some(e), Some(w)) => (l.to_string(), e.to_string(), w), + _ => return Outcome::Failure(Status::BadRequest), + }; + + let raw_path = req.routed_segments(2..).collect::>().join("/"); + + let mut http_req = rocket::http::hyper::Request::builder(); + http_req = http_req.method("GET"); + for h in req.headers().iter() { + http_req = http_req.header(h.name.to_string(), h.value.to_string()); + } + + match http_req.body(()) { + Ok(r) => Outcome::from( + req, + ApiResponse( + turbine + .get_subsequent_load( + perseus::path::PathWithoutLocale(raw_path), + locale, + entity_name, + was_incremental_match, + r, + ) + .await, + ), + ), + _ => Outcome::Failure(Status::BadRequest), + } +} + +// ----- Rocket handler trait implementation ----- + +#[derive(Clone)] +enum PerseusRouteKind<'a> { + Locale, + StaticAlias(&'a String), + IntialLoadHandler, + SubsequentLoadHandler, +} + +#[derive(Clone)] +struct RocketHandlerWithTurbine<'a, M, T> +where + M: MutableStore + 'static, + T: TranslationsManager + 'static, +{ + turbine: &'a Turbine, + perseus_route: PerseusRouteKind<'a>, +} + +#[rocket::async_trait] +impl Handler for RocketHandlerWithTurbine<'static, M, T> +where + M: MutableStore + 'static, + T: TranslationsManager + 'static, +{ + async fn handle<'r>(&self, req: &'r Request<'_>, _data: Data<'r>) -> Outcome<'r> { + match self.perseus_route { + PerseusRouteKind::Locale => perseus_locale(req, self.turbine).await, + PerseusRouteKind::StaticAlias(static_alias) => { + perseus_static_alias(req, static_alias).await + } + PerseusRouteKind::IntialLoadHandler => { + perseus_initial_load_handler(req, self.turbine).await + } + PerseusRouteKind::SubsequentLoadHandler => { + perseus_subsequent_load_handler(req, self.turbine).await + } + } + } +} + +async fn perseus_static_alias<'r>(req: &'r Request<'_>, static_alias: &String) -> Outcome<'r> { + match File::open(static_alias).await { + Ok(file) => Outcome::from(req, file), + _ => Outcome::Failure(Status::NotFound), + } +} + +// ----- Integration code ----- + +/// Configures an Rocket Web app for Perseus. +/// This returns a rocket app at the build stage that can be built upon further with +/// more routes, fairings etc... +pub async fn perseus_base_app( + turbine: &'static Turbine, + opts: ServerOptions, +) -> Rocket +where + M: MutableStore + 'static, + T: TranslationsManager + 'static, +{ + let get_locale = Route::new( + Method::Get, + "/translations/", + RocketHandlerWithTurbine { + turbine: &turbine, + perseus_route: PerseusRouteKind::Locale, + }, + ); + + // Since this route matches everything, its rank has been set to 100, + // That means that it will be used after routes that have a rank inferior to 100 + // forward, see https://rocket.rs/v0.5-rc/guide/requests/#default-ranking + let get_initial_load_handler = Route::ranked( + 100, + Method::Get, + "/", + RocketHandlerWithTurbine { + turbine: &turbine, + perseus_route: PerseusRouteKind::IntialLoadHandler, + }, + ); + + let get_subsequent_load_handler = Route::new( + Method::Get, + "/page/", + RocketHandlerWithTurbine { + turbine: &turbine, + perseus_route: PerseusRouteKind::SubsequentLoadHandler, + }, + ); + + let mut perseus_routes: Vec = + routes![get_js_bundle, get_wasm_js_bundle, get_wasm_bundle]; + perseus_routes.append(&mut vec![get_locale, get_subsequent_load_handler]); + + let mut app = rocket::build() + .manage(opts.clone()) + .mount("/.perseus/", perseus_routes) + .mount("/", vec![get_initial_load_handler]); + + if Path::new(&opts.snippets).exists() { + app = app.mount("/.perseus/snippets", FileServer::from(opts.snippets)) + } + + if turbine.static_dir.exists() { + app = app.mount("/.perseus/static", FileServer::from(&turbine.static_dir)) + } + + let mut static_aliases: Vec = vec![]; + + for (url, static_path) in turbine.static_aliases.iter() { + let route = Route::new( + Method::Get, + url, + RocketHandlerWithTurbine { + turbine: &turbine, + perseus_route: PerseusRouteKind::StaticAlias(static_path), + }, + ); + static_aliases.push(route) + } + + app = app.mount("/", static_aliases); + + app +} + +// ----- Default server ----- + +/// Creates and starts the default Perseus server with Rocket. This should be +/// run in a `main` function annotated with `#[tokio::main]` (which requires the +/// `macros` and `rt-multi-thread` features on the `tokio` dependency). +#[cfg(feature = "dflt-server")] +pub async fn dflt_server( + turbine: &'static Turbine, + opts: ServerOptions, + (host, port): (String, u16), +) { + let addr = host.parse().expect("Invalid address provided to bind to."); + + let mut app = perseus_base_app(turbine, opts).await; + + let mut config = rocket::Config::default(); + config.port = port; + config.address = addr; + app = app.configure(config); + + match app.launch().await { + Err(e) => println!("Error lauching rocket app: {}", e), + _ => (), + } +}