use workspaces
|
@ -1,58 +0,0 @@
|
||||||
# doordesk-rs
|
|
||||||
|
|
||||||
## Dev Server
|
|
||||||
|
|
||||||
```bash
|
|
||||||
cargo install cargo-leptos
|
|
||||||
```
|
|
||||||
|
|
||||||
```bash
|
|
||||||
cargo leptos watch
|
|
||||||
```
|
|
||||||
|
|
||||||
By default, `cargo-leptos` uses `nightly` Rust, `cargo-generate`, and `sass`. If you run into any trouble, you may need to install one or more of these tools.
|
|
||||||
|
|
||||||
1. `rustup toolchain install nightly --allow-downgrade` - make sure you have Rust nightly
|
|
||||||
2. `rustup target add wasm32-unknown-unknown` - add the ability to compile Rust to WebAssembly
|
|
||||||
3. `cargo install cargo-generate` - install `cargo-generate` binary (should be installed automatically in future)
|
|
||||||
4. `npm install -g sass` - install `dart-sass` (should be optional in future
|
|
||||||
|
|
||||||
## Compiling for Release
|
|
||||||
```bash
|
|
||||||
cargo leptos build --release
|
|
||||||
```
|
|
||||||
|
|
||||||
Will generate your server binary in target/server/release and your site package in target/site
|
|
||||||
|
|
||||||
## Testing Your Project
|
|
||||||
```bash
|
|
||||||
cargo leptos end-to-end
|
|
||||||
```
|
|
||||||
|
|
||||||
```bash
|
|
||||||
cargo leptos end-to-end --release
|
|
||||||
```
|
|
||||||
|
|
||||||
Cargo-leptos uses Playwright as the end-to-end test tool.
|
|
||||||
Tests are located in end2end/tests directory.
|
|
||||||
|
|
||||||
## Executing a Server on a Remote Machine Without the Toolchain
|
|
||||||
After running a `cargo leptos build --release` the minimum files needed are:
|
|
||||||
|
|
||||||
1. The server binary located in `target/server/release`
|
|
||||||
2. The `site` directory and all files within located in `target/site`
|
|
||||||
|
|
||||||
Copy these files to your remote server. The directory structure should be:
|
|
||||||
```text
|
|
||||||
start-axum
|
|
||||||
site/
|
|
||||||
```
|
|
||||||
Set the following environment variables (updating for your project as needed):
|
|
||||||
```text
|
|
||||||
LEPTOS_OUTPUT_NAME="start-axum"
|
|
||||||
LEPTOS_SITE_ROOT="site"
|
|
||||||
LEPTOS_SITE_PKG_DIR="pkg"
|
|
||||||
LEPTOS_SITE_ADDR="127.0.0.1:3000"
|
|
||||||
LEPTOS_RELOAD_PORT="3001"
|
|
||||||
```
|
|
||||||
Finally, run the server binary.
|
|
|
@ -1,2 +0,0 @@
|
||||||
[toolchain]
|
|
||||||
channel = "nightly"
|
|
|
@ -1,40 +0,0 @@
|
||||||
use cfg_if::cfg_if;
|
|
||||||
|
|
||||||
cfg_if! { if #[cfg(feature = "ssr")] {
|
|
||||||
use axum::{
|
|
||||||
body::{boxed, Body, BoxBody},
|
|
||||||
extract::State,
|
|
||||||
response::IntoResponse,
|
|
||||||
http::{Request, Response, StatusCode, Uri},
|
|
||||||
};
|
|
||||||
use axum::response::Response as AxumResponse;
|
|
||||||
use tower::ServiceExt;
|
|
||||||
use tower_http::services::ServeDir;
|
|
||||||
use leptos::*;
|
|
||||||
use crate::app::App;
|
|
||||||
|
|
||||||
pub async fn file_and_error_handler(uri: Uri, State(options): State<LeptosOptions>, req: Request<Body>) -> AxumResponse {
|
|
||||||
let root = options.site_root.clone();
|
|
||||||
let res = get_static_file(uri.clone(), &root).await.unwrap();
|
|
||||||
|
|
||||||
if res.status() == StatusCode::OK {
|
|
||||||
res.into_response()
|
|
||||||
} else {
|
|
||||||
let handler = leptos_axum::render_app_to_stream(options.to_owned(), move |cx| view!{cx, <App/>});
|
|
||||||
handler(req).await.into_response()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn get_static_file(uri: Uri, root: &str) -> Result<Response<BoxBody>, (StatusCode, String)> {
|
|
||||||
let req = Request::builder().uri(uri.clone()).body(Body::empty()).unwrap();
|
|
||||||
// `ServeDir` implements `tower::Service` so we can call it with `tower::ServiceExt::oneshot`
|
|
||||||
// This path is relative to the cargo root
|
|
||||||
match ServeDir::new(root).oneshot(req).await {
|
|
||||||
Ok(res) => Ok(res.map(boxed)),
|
|
||||||
Err(err) => Err((
|
|
||||||
StatusCode::INTERNAL_SERVER_ERROR,
|
|
||||||
format!("Something went wrong: {err}"),
|
|
||||||
)),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}}
|
|
|
@ -1,24 +0,0 @@
|
||||||
use cfg_if::cfg_if;
|
|
||||||
pub mod app;
|
|
||||||
pub mod error_template;
|
|
||||||
pub mod fileserv;
|
|
||||||
|
|
||||||
pub mod components;
|
|
||||||
pub mod routes;
|
|
||||||
|
|
||||||
cfg_if! { if #[cfg(feature = "hydrate")] {
|
|
||||||
use leptos::*;
|
|
||||||
use wasm_bindgen::prelude::wasm_bindgen;
|
|
||||||
use crate::app::*;
|
|
||||||
|
|
||||||
#[wasm_bindgen]
|
|
||||||
pub fn hydrate() {
|
|
||||||
// initializes logging using the `log` crate
|
|
||||||
_ = console_log::init_with_level(log::Level::Debug);
|
|
||||||
console_error_panic_hook::set_once();
|
|
||||||
|
|
||||||
leptos::mount_to_body(move |cx| {
|
|
||||||
view! { cx, <App/> }
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}}
|
|
|
@ -1,9 +0,0 @@
|
||||||
/** @type {import('tailwindcss').Config} */
|
|
||||||
module.exports = {
|
|
||||||
content: ["*.html", "./src/**/*.rs"],
|
|
||||||
theme: {
|
|
||||||
extend: {},
|
|
||||||
},
|
|
||||||
plugins: [],
|
|
||||||
}
|
|
||||||
|
|
2
doordesk-rs/.gitignore → doordesk/.gitignore
vendored
|
@ -2,7 +2,6 @@
|
||||||
# will have compiled files and executables
|
# will have compiled files and executables
|
||||||
/target/
|
/target/
|
||||||
pkg
|
pkg
|
||||||
Cargo.lock
|
|
||||||
|
|
||||||
# These are backup files generated by rustfmt
|
# These are backup files generated by rustfmt
|
||||||
**/*.rs.bk
|
**/*.rs.bk
|
||||||
|
@ -12,4 +11,3 @@ node_modules/
|
||||||
test-results/
|
test-results/
|
||||||
end2end/playwright-report/
|
end2end/playwright-report/
|
||||||
playwright/.cache/
|
playwright/.cache/
|
||||||
.sass-cache
|
|
2753
doordesk/Cargo.lock
generated
Normal file
|
@ -1,51 +1,45 @@
|
||||||
[package]
|
[workspace]
|
||||||
name = "doordesk"
|
resolver = "2"
|
||||||
version = "0.1.0"
|
members = ["app", "frontend", "server"]
|
||||||
edition = "2021"
|
|
||||||
|
|
||||||
[lib]
|
# need to be applied only to wasm build
|
||||||
crate-type = ["cdylib", "rlib"]
|
[profile.release]
|
||||||
|
codegen-units = 1
|
||||||
|
lto = true
|
||||||
|
opt-level = 'z'
|
||||||
|
|
||||||
[dependencies]
|
[workspace.dependencies]
|
||||||
axum = { version = "0.6.4", optional = true }
|
leptos = { version = "0.5", features = ["nightly"] }
|
||||||
console_error_panic_hook = "0.1"
|
leptos_meta = { version = "0.5", features = ["nightly"] }
|
||||||
console_log = "1"
|
leptos_router = { version = "0.5", features = ["nightly"] }
|
||||||
|
leptos_axum = { version = "0.5" }
|
||||||
|
|
||||||
|
axum = "0.6.20"
|
||||||
cfg-if = "1"
|
cfg-if = "1"
|
||||||
leptos = { version = "0.4", features = ["nightly"] }
|
console_error_panic_hook = "0.1.7"
|
||||||
leptos_axum = { version = "0.4", optional = true }
|
console_log = "1"
|
||||||
leptos_meta = { version = "0.4", features = ["nightly"] }
|
http = "0.2.9"
|
||||||
leptos_router = { version = "0.4", features = ["nightly"] }
|
log = "0.4.20"
|
||||||
log = "0.4"
|
simple_logger = "4.2.0"
|
||||||
simple_logger = "4"
|
thiserror = "1"
|
||||||
tokio = { version = "1.25.0", optional = true }
|
tokio = { version = "1.33.0", features = ["full"] }
|
||||||
tower = { version = "0.4.13", optional = true }
|
tower = { version = "0.4.13", features = ["full"] }
|
||||||
tower-http = { version = "0.4", features = ["fs"], optional = true }
|
tower-http = { version = "0.4", features = ["full"] }
|
||||||
wasm-bindgen = "0.2.89"
|
wasm-bindgen = "=0.2.89"
|
||||||
thiserror = "1.0.38"
|
|
||||||
tracing = { version = "0.1.37", optional = true }
|
|
||||||
http = "0.2.8"
|
|
||||||
|
|
||||||
[features]
|
# See https://github.com/akesson/cargo-leptos for documentation of all the parameters.
|
||||||
hydrate = ["leptos/hydrate", "leptos_meta/hydrate", "leptos_router/hydrate"]
|
|
||||||
ssr = [
|
|
||||||
"dep:axum",
|
|
||||||
"dep:tokio",
|
|
||||||
"dep:tower",
|
|
||||||
"dep:tower-http",
|
|
||||||
"dep:leptos_axum",
|
|
||||||
"leptos/ssr",
|
|
||||||
"leptos_meta/ssr",
|
|
||||||
"leptos_router/ssr",
|
|
||||||
"dep:tracing",
|
|
||||||
]
|
|
||||||
|
|
||||||
[package.metadata.cargo-all-features]
|
# A leptos project defines which workspace members
|
||||||
denylist = ["axum", "tokio", "tower", "tower-http", "leptos_axum"]
|
# that are used together frontend (lib) & server (bin)
|
||||||
skip_feature_sets = [["ssr", "hydrate"]]
|
[[workspace.metadata.leptos]]
|
||||||
|
# this name is used for the wasm, js and css file names
|
||||||
|
name = "doordesk"
|
||||||
|
|
||||||
[package.metadata.leptos]
|
# the package in the workspace that contains the server binary (binary crate)
|
||||||
# The name used by wasm-bindgen/cargo-leptos for the JS/WASM bundle. Defaults to the crate name
|
bin-package = "server"
|
||||||
output-name = "doordesk"
|
|
||||||
|
# the package in the workspace that contains the frontend wasm binary (library crate)
|
||||||
|
lib-package = "frontend"
|
||||||
|
|
||||||
# The site root folder is where cargo-leptos generate all output. WARNING: all content of this folder will be erased on a rebuild. Use it in your server setup.
|
# The site root folder is where cargo-leptos generate all output. WARNING: all content of this folder will be erased on a rebuild. Use it in your server setup.
|
||||||
site-root = "target/site"
|
site-root = "target/site"
|
||||||
|
@ -98,7 +92,7 @@ env = "DEV"
|
||||||
# The features to use when compiling the bin target
|
# The features to use when compiling the bin target
|
||||||
#
|
#
|
||||||
# Optional. Can be over-ridden with the command line parameter --bin-features
|
# Optional. Can be over-ridden with the command line parameter --bin-features
|
||||||
bin-features = ["ssr"]
|
bin-features = []
|
||||||
|
|
||||||
# If the --no-default-features flag should be used when compiling the bin target
|
# If the --no-default-features flag should be used when compiling the bin target
|
||||||
#
|
#
|
||||||
|
@ -108,7 +102,7 @@ bin-default-features = false
|
||||||
# The features to use when compiling the lib target
|
# The features to use when compiling the lib target
|
||||||
#
|
#
|
||||||
# Optional. Can be over-ridden with the command line parameter --lib-features
|
# Optional. Can be over-ridden with the command line parameter --lib-features
|
||||||
lib-features = ["hydrate"]
|
lib-features = []
|
||||||
|
|
||||||
# If the --no-default-features flag should be used when compiling the lib target
|
# If the --no-default-features flag should be used when compiling the lib target
|
||||||
#
|
#
|
87
doordesk/README.md
Normal file
|
@ -0,0 +1,87 @@
|
||||||
|
<picture>
|
||||||
|
<source srcset="https://raw.githubusercontent.com/leptos-rs/leptos/main/docs/logos/Leptos_logo_Solid_White.svg" media="(prefers-color-scheme: dark)">
|
||||||
|
<img src="https://raw.githubusercontent.com/leptos-rs/leptos/main/docs/logos/Leptos_logo_RGB.svg" alt="Leptos Logo">
|
||||||
|
</picture>
|
||||||
|
|
||||||
|
# Leptos Axum Starter Template
|
||||||
|
|
||||||
|
This is a template for use with the [Leptos](https://github.com/leptos-rs/leptos) web framework and the [cargo-leptos](https://github.com/akesson/cargo-leptos) tool using [Axum](https://github.com/tokio-rs/axum).
|
||||||
|
|
||||||
|
## Creating your template repo
|
||||||
|
|
||||||
|
If you don't have `cargo-leptos` installed you can install it with
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cargo install cargo-leptos
|
||||||
|
```
|
||||||
|
|
||||||
|
Then run
|
||||||
|
```bash
|
||||||
|
cargo leptos new --git https://github.com/leptos-rs/start-axum-workspace/
|
||||||
|
```
|
||||||
|
|
||||||
|
to generate a new project template.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cd {projectname}
|
||||||
|
```
|
||||||
|
|
||||||
|
to go to your newly created project.
|
||||||
|
Feel free to explore the project structure, but the best place to start with your application code is in `src/app.rs`.
|
||||||
|
Addtionally, Cargo.toml may need updating as new versions of the dependencies are released, especially if things are not working after a `cargo update`.
|
||||||
|
|
||||||
|
## Running your project
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cargo leptos watch
|
||||||
|
```
|
||||||
|
|
||||||
|
## Installing Additional Tools
|
||||||
|
|
||||||
|
By default, `cargo-leptos` uses `nightly` Rust, `cargo-generate`, and `sass`. If you run into any trouble, you may need to install one or more of these tools.
|
||||||
|
|
||||||
|
1. `rustup toolchain install nightly --allow-downgrade` - make sure you have Rust nightly
|
||||||
|
2. `rustup default nightly` - setup nightly as default, or you can use rust-toolchain file later on
|
||||||
|
3. `rustup target add wasm32-unknown-unknown` - add the ability to compile Rust to WebAssembly
|
||||||
|
4. `cargo install cargo-generate` - install `cargo-generate` binary (should be installed automatically in future)
|
||||||
|
5. `npm install -g sass` - install `dart-sass` (should be optional in future
|
||||||
|
|
||||||
|
## Compiling for Release
|
||||||
|
```bash
|
||||||
|
cargo leptos build --release
|
||||||
|
```
|
||||||
|
|
||||||
|
Will generate your server binary in target/server/release and your site package in target/site
|
||||||
|
|
||||||
|
## Testing Your Project
|
||||||
|
```bash
|
||||||
|
cargo leptos end-to-end
|
||||||
|
```
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cargo leptos end-to-end --release
|
||||||
|
```
|
||||||
|
|
||||||
|
Cargo-leptos uses Playwright as the end-to-end test tool.
|
||||||
|
Tests are located in end2end/tests directory.
|
||||||
|
|
||||||
|
## Executing a Server on a Remote Machine Without the Toolchain
|
||||||
|
After running a `cargo leptos build --release` the minimum files needed are:
|
||||||
|
|
||||||
|
1. The server binary located in `target/server/release`
|
||||||
|
2. The `site` directory and all files within located in `target/site`
|
||||||
|
|
||||||
|
Copy these files to your remote server. The directory structure should be:
|
||||||
|
```text
|
||||||
|
start-axum
|
||||||
|
site/
|
||||||
|
```
|
||||||
|
Set the following enviornment variables (updating for your project as needed):
|
||||||
|
```text
|
||||||
|
LEPTOS_OUTPUT_NAME="start-axum"
|
||||||
|
LEPTOS_SITE_ROOT="site"
|
||||||
|
LEPTOS_SITE_PKG_DIR="pkg"
|
||||||
|
LEPTOS_SITE_ADDR="127.0.0.1:3000"
|
||||||
|
LEPTOS_RELOAD_PORT="3001"
|
||||||
|
```
|
||||||
|
Finally, run the server binary.
|
22
doordesk/app/Cargo.toml
Normal file
|
@ -0,0 +1,22 @@
|
||||||
|
[package]
|
||||||
|
name = "app"
|
||||||
|
version = "0.1.0"
|
||||||
|
edition = "2021"
|
||||||
|
|
||||||
|
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
leptos.workspace = true
|
||||||
|
leptos_meta.workspace = true
|
||||||
|
leptos_router.workspace = true
|
||||||
|
leptos_axum = { workspace = true, optional = true }
|
||||||
|
|
||||||
|
http.workspace = true
|
||||||
|
cfg-if.workspace = true
|
||||||
|
thiserror.workspace = true
|
||||||
|
|
||||||
|
[features]
|
||||||
|
default = []
|
||||||
|
hydrate = ["leptos/hydrate", "leptos_meta/hydrate", "leptos_router/hydrate"]
|
||||||
|
ssr = ["leptos/ssr", "leptos_meta/ssr", "leptos_router/ssr", "dep:leptos_axum"]
|
||||||
|
|
|
@ -1,18 +1,18 @@
|
||||||
use leptos::*;
|
use leptos::*;
|
||||||
|
|
||||||
#[component]
|
#[component]
|
||||||
pub fn Article(cx: Scope) -> impl IntoView {
|
pub fn Article() -> impl IntoView {
|
||||||
let (count, set_count) = create_signal(cx, 0);
|
// Creates a reactive value to update the button
|
||||||
|
let (count, set_count) = create_signal(0);
|
||||||
let on_click = move |_| set_count.update(|count| *count += 1);
|
let on_click = move |_| set_count.update(|count| *count += 1);
|
||||||
|
|
||||||
view! { cx,
|
view! {
|
||||||
<article class="bg-zinc-700 mx-auto p-7 my-5 w-11/12 max-w-screen-xl rounded-md shadow-1g bg-opacity-10">
|
<article class="bg-zinc-700 mx-auto p-7 my-5 w-11/12 max-w-screen-xl rounded-md shadow-1g bg-opacity-10">
|
||||||
<h1 class="max-6-xs text-3xl text-orange-600 font-light capitalize">"ayo"</h1>
|
<h1 class="max-6-xs text-3xl text-orange-600 font-light capitalize">"ayo"</h1>
|
||||||
<hr class="opacity-50" />
|
<hr class="opacity-50" />
|
||||||
<span class="opacity-50 text-xs pt-0 m-t pb-3.5">"today"</span>
|
<span class="opacity-50 text-xs pt-0 m-t pb-3.5">"today"</span>
|
||||||
<div>
|
<div>
|
||||||
<button on:click=on_click>"Click Me: " {count}</button>
|
<button on:click=on_click>"Click Me: " {count}</button>
|
||||||
<p>I DID IT?</p>
|
|
||||||
</div>
|
</div>
|
||||||
</article>
|
</article>
|
||||||
}
|
}
|
|
@ -24,19 +24,18 @@ impl AppError {
|
||||||
// Feel free to do more complicated things here than just displaying the error.
|
// Feel free to do more complicated things here than just displaying the error.
|
||||||
#[component]
|
#[component]
|
||||||
pub fn ErrorTemplate(
|
pub fn ErrorTemplate(
|
||||||
cx: Scope,
|
|
||||||
#[prop(optional)] outside_errors: Option<Errors>,
|
#[prop(optional)] outside_errors: Option<Errors>,
|
||||||
#[prop(optional)] errors: Option<RwSignal<Errors>>,
|
#[prop(optional)] errors: Option<RwSignal<Errors>>,
|
||||||
) -> impl IntoView {
|
) -> impl IntoView {
|
||||||
let errors = match outside_errors {
|
let errors = match outside_errors {
|
||||||
Some(e) => create_rw_signal(cx, e),
|
Some(e) => create_rw_signal(e),
|
||||||
None => match errors {
|
None => match errors {
|
||||||
Some(e) => e,
|
Some(e) => e,
|
||||||
None => panic!("No Errors found and we expected errors!"),
|
None => panic!("No Errors found and we expected errors!"),
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
// Get Errors from Signal
|
// Get Errors from Signal
|
||||||
let errors = errors.get();
|
let errors = errors.get_untracked();
|
||||||
|
|
||||||
// Downcast lets us take a type that implements `std::error::Error`
|
// Downcast lets us take a type that implements `std::error::Error`
|
||||||
let errors: Vec<AppError> = errors
|
let errors: Vec<AppError> = errors
|
||||||
|
@ -48,32 +47,26 @@ pub fn ErrorTemplate(
|
||||||
// Only the response code for the first error is actually sent from the server
|
// Only the response code for the first error is actually sent from the server
|
||||||
// this may be customized by the specific application
|
// this may be customized by the specific application
|
||||||
cfg_if! { if #[cfg(feature="ssr")] {
|
cfg_if! { if #[cfg(feature="ssr")] {
|
||||||
let response = use_context::<ResponseOptions>(cx);
|
let response = use_context::<ResponseOptions>();
|
||||||
if let Some(response) = response {
|
if let Some(response) = response {
|
||||||
response.set_status(errors[0].status_code());
|
response.set_status(errors[0].status_code());
|
||||||
}
|
}
|
||||||
}}
|
}}
|
||||||
|
|
||||||
view! {cx,
|
view! {
|
||||||
<h1>{if errors.len() > 1 {"Errors"} else {"Error"}}</h1>
|
<h1>{if errors.len() > 1 { "Errors" } else { "Error" }}</h1>
|
||||||
<For
|
<For
|
||||||
// a function that returns the items we're iterating over; a signal is fine
|
// a function that returns the items we're iterating over; a signal is fine
|
||||||
each= move || {errors.clone().into_iter().enumerate()}
|
each=move || { errors.clone().into_iter().enumerate() }
|
||||||
// a unique key for each item as a reference
|
// a unique key for each item as a reference
|
||||||
key=|(index, _error)| *index
|
key=|(index, _error)| *index
|
||||||
// renders each item to a view
|
// renders each item to a view
|
||||||
view= move |cx, error| {
|
children=move |error| {
|
||||||
let error_string = error.1.to_string();
|
let error_string = error.1.to_string();
|
||||||
let error_code= error.1.status_code();
|
let error_code = error.1.status_code();
|
||||||
view! {
|
view! {
|
||||||
cx,
|
<h2>{error_code.to_string()}</h2>
|
||||||
<article class="bg-zinc-700 mx-auto p-7 my-5 w-11/12 max-w-screen-xl rounded-md shadow-1g bg-opacity-10">
|
|
||||||
<h1 class="max-6-xs text-3xl text-orange-600 font-light capitalize">
|
|
||||||
{error_code.to_string()}
|
|
||||||
</h1>
|
|
||||||
<hr class="opacity-50" />
|
|
||||||
<p>"Error: " {error_string}</p>
|
<p>"Error: " {error_string}</p>
|
||||||
</article>
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
/>
|
/>
|
|
@ -1,32 +1,32 @@
|
||||||
use crate::error_template::{AppError, ErrorTemplate};
|
use crate::error_template::{AppError, ErrorTemplate};
|
||||||
|
|
||||||
|
//use crate::routes::{blog::*, home::*, projects::*};
|
||||||
use leptos::*;
|
use leptos::*;
|
||||||
use leptos_meta::*;
|
use leptos_meta::*;
|
||||||
use leptos_router::*;
|
use leptos_router::*;
|
||||||
use crate::routes::{blog::*, home::*,projects::*};
|
|
||||||
|
pub mod error_template;
|
||||||
|
pub mod components;
|
||||||
|
pub mod routes;
|
||||||
|
|
||||||
|
use crate::routes::{home::*, blog::*, projects::*};
|
||||||
|
|
||||||
#[component]
|
#[component]
|
||||||
pub fn App(cx: Scope) -> impl IntoView {
|
pub fn App() -> impl IntoView {
|
||||||
// Provides context that manages stylesheets, titles, meta tags, etc.
|
// Provides context that manages stylesheets, titles, meta tags, etc.
|
||||||
provide_meta_context(cx);
|
provide_meta_context();
|
||||||
|
|
||||||
view! {
|
view! {
|
||||||
cx,
|
|
||||||
// injects a stylesheet into the document <head>
|
|
||||||
// id=leptos means cargo-leptos will hot-reload this stylesheet
|
|
||||||
<Stylesheet id="leptos" href="/pkg/doordesk.css"/>
|
<Stylesheet id="leptos" href="/pkg/doordesk.css"/>
|
||||||
|
|
||||||
// sets the document title
|
// sets the document title
|
||||||
<Title text="doordesk"/>
|
<Title text="doordesk"/>
|
||||||
|
|
||||||
// content for this welcome page
|
// content for this welcome page
|
||||||
<Router fallback=|cx| {
|
<Router fallback=|| {
|
||||||
let mut outside_errors = Errors::default();
|
let mut outside_errors = Errors::default();
|
||||||
outside_errors.insert_with_default_key(AppError::NotFound);
|
outside_errors.insert_with_default_key(AppError::NotFound);
|
||||||
view! { cx,
|
view! { <ErrorTemplate outside_errors/> }.into_view()
|
||||||
<ErrorTemplate outside_errors/>
|
|
||||||
}
|
|
||||||
.into_view(cx)
|
|
||||||
}>
|
}>
|
||||||
<nav class="bg-gradient-to-b from-zinc-800 to-zinc-900 shadow-lg sticky top-0">
|
<nav class="bg-gradient-to-b from-zinc-800 to-zinc-900 shadow-lg sticky top-0">
|
||||||
<ul class="container flex items-center p-3">
|
<ul class="container flex items-center p-3">
|
||||||
|
@ -46,9 +46,9 @@ pub fn App(cx: Scope) -> impl IntoView {
|
||||||
</nav>
|
</nav>
|
||||||
<main>
|
<main>
|
||||||
<Routes>
|
<Routes>
|
||||||
<Route path="/" view=|cx| view! { cx, <Home /> }/>
|
<Route path="" view=Home/>
|
||||||
<Route path="/blog" view=|cx| view! { cx, <Blog /> }/>
|
<Route path="blog" view=Blog/>
|
||||||
<Route path="/projects" view=|cx| view! { cx, <Projects /> }/>
|
<Route path="projects" view=Projects/>
|
||||||
</Routes>
|
</Routes>
|
||||||
</main>
|
</main>
|
||||||
<p class="text-center hover:rotate-180 duration-200 w-8 m-auto">
|
<p class="text-center hover:rotate-180 duration-200 w-8 m-auto">
|
|
@ -2,9 +2,9 @@ use crate::components::article::*;
|
||||||
use leptos::*;
|
use leptos::*;
|
||||||
|
|
||||||
#[component]
|
#[component]
|
||||||
pub fn Blog(cx: Scope) -> impl IntoView {
|
pub fn Blog() -> impl IntoView {
|
||||||
view! { cx,
|
view! {
|
||||||
<Article />
|
<Article />
|
||||||
<p>blog</p>
|
<p>Blog</p>
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -1,10 +1,10 @@
|
||||||
use leptos::*;
|
|
||||||
use crate::components::article::*;
|
use crate::components::article::*;
|
||||||
|
use leptos::*;
|
||||||
|
|
||||||
#[component]
|
#[component]
|
||||||
pub fn Home(cx: Scope) -> impl IntoView {
|
pub fn Home() -> impl IntoView {
|
||||||
view! { cx,
|
view! {
|
||||||
<Article />
|
<Article />
|
||||||
<p>home</p>
|
<p>Home</p>
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -1,10 +1,10 @@
|
||||||
use leptos::*;
|
|
||||||
use crate::components::article::*;
|
use crate::components::article::*;
|
||||||
|
use leptos::*;
|
||||||
|
|
||||||
#[component]
|
#[component]
|
||||||
pub fn Projects(cx: Scope) -> impl IntoView {
|
pub fn Projects() -> impl IntoView {
|
||||||
view! { cx,
|
view! {
|
||||||
<Article />
|
<Article />
|
||||||
<p>projects</p>
|
<p>Projects</p>
|
||||||
}
|
}
|
||||||
}
|
}
|
18
doordesk/frontend/Cargo.toml
Normal file
|
@ -0,0 +1,18 @@
|
||||||
|
[package]
|
||||||
|
name = "frontend"
|
||||||
|
version = "0.1.0"
|
||||||
|
edition = "2021"
|
||||||
|
|
||||||
|
[lib]
|
||||||
|
crate-type = ["cdylib", "rlib"]
|
||||||
|
|
||||||
|
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
app = { path = "../app", default-features = false, features = ["hydrate"] }
|
||||||
|
leptos = { workspace = true, features = [ "hydrate" ] }
|
||||||
|
|
||||||
|
console_error_panic_hook.workspace = true
|
||||||
|
console_log.workspace = true
|
||||||
|
log.workspace = true
|
||||||
|
wasm-bindgen.workspace = true
|
12
doordesk/frontend/src/lib.rs
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
use app::*;
|
||||||
|
use leptos::*;
|
||||||
|
use wasm_bindgen::prelude::wasm_bindgen;
|
||||||
|
|
||||||
|
#[wasm_bindgen]
|
||||||
|
pub fn hydrate() {
|
||||||
|
// initializes logging using the `log` crate
|
||||||
|
_ = console_log::init_with_level(log::Level::Debug);
|
||||||
|
console_error_panic_hook::set_once();
|
||||||
|
|
||||||
|
leptos::mount_to_body(App);
|
||||||
|
}
|
Before Width: | Height: | Size: 12 KiB After Width: | Height: | Size: 12 KiB |
Before Width: | Height: | Size: 38 KiB After Width: | Height: | Size: 38 KiB |
Before Width: | Height: | Size: 10 KiB After Width: | Height: | Size: 10 KiB |
Before Width: | Height: | Size: 685 B After Width: | Height: | Size: 685 B |
Before Width: | Height: | Size: 1.5 KiB After Width: | Height: | Size: 1.5 KiB |
Before Width: | Height: | Size: 15 KiB After Width: | Height: | Size: 15 KiB |
2
doordesk/rust-toolchain.toml
Normal file
|
@ -0,0 +1,2 @@
|
||||||
|
[toolchain]
|
||||||
|
channel = "nightly"
|
18
doordesk/server/Cargo.toml
Normal file
|
@ -0,0 +1,18 @@
|
||||||
|
[package]
|
||||||
|
name = "server"
|
||||||
|
version = "0.1.0"
|
||||||
|
edition = "2021"
|
||||||
|
|
||||||
|
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
app = { path = "../app", default-features = false, features = ["ssr"] }
|
||||||
|
leptos = { workspace = true, features = [ "ssr" ]}
|
||||||
|
leptos_axum.workspace = true
|
||||||
|
|
||||||
|
axum.workspace = true
|
||||||
|
simple_logger.workspace = true
|
||||||
|
tokio.workspace = true
|
||||||
|
tower.workspace = true
|
||||||
|
tower-http.workspace = true
|
||||||
|
log.workspace = true
|
45
doordesk/server/src/fileserv.rs
Normal file
|
@ -0,0 +1,45 @@
|
||||||
|
use app::error_template::AppError;
|
||||||
|
use app::error_template::ErrorTemplate;
|
||||||
|
use app::App;
|
||||||
|
use axum::response::Response as AxumResponse;
|
||||||
|
use axum::{
|
||||||
|
body::{boxed, Body, BoxBody},
|
||||||
|
extract::State,
|
||||||
|
http::{Request, Response, StatusCode, Uri},
|
||||||
|
response::IntoResponse,
|
||||||
|
};
|
||||||
|
use leptos::*;
|
||||||
|
use tower::ServiceExt;
|
||||||
|
use tower_http::services::ServeDir;
|
||||||
|
|
||||||
|
pub async fn file_and_error_handler(
|
||||||
|
uri: Uri,
|
||||||
|
State(options): State<LeptosOptions>,
|
||||||
|
req: Request<Body>,
|
||||||
|
) -> AxumResponse {
|
||||||
|
let root = options.site_root.clone();
|
||||||
|
let res = get_static_file(uri.clone(), &root).await.unwrap();
|
||||||
|
|
||||||
|
if res.status() == StatusCode::OK {
|
||||||
|
res.into_response()
|
||||||
|
} else {
|
||||||
|
let handler = leptos_axum::render_app_to_stream(options.to_owned(), move || view! { <App/> });
|
||||||
|
handler(req).await.into_response()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn get_static_file(uri: Uri, root: &str) -> Result<Response<BoxBody>, (StatusCode, String)> {
|
||||||
|
let req = Request::builder()
|
||||||
|
.uri(uri.clone())
|
||||||
|
.body(Body::empty())
|
||||||
|
.unwrap();
|
||||||
|
// `ServeDir` implements `tower::Service` so we can call it with `tower::ServiceExt::oneshot`
|
||||||
|
// This path is relative to the cargo root
|
||||||
|
match ServeDir::new(root).oneshot(req).await {
|
||||||
|
Ok(res) => Ok(res.map(boxed)),
|
||||||
|
Err(err) => Err((
|
||||||
|
StatusCode::INTERNAL_SERVER_ERROR,
|
||||||
|
format!("Something went wrong: {err}"),
|
||||||
|
)),
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,12 +1,13 @@
|
||||||
#[cfg(feature = "ssr")]
|
use app::*;
|
||||||
|
use axum::{routing::post, Router};
|
||||||
|
use fileserv::file_and_error_handler;
|
||||||
|
use leptos::*;
|
||||||
|
use leptos_axum::{generate_route_list, LeptosRoutes};
|
||||||
|
|
||||||
|
pub mod fileserv;
|
||||||
|
|
||||||
#[tokio::main]
|
#[tokio::main]
|
||||||
async fn main() {
|
async fn main() {
|
||||||
use axum::{routing::post, Router};
|
|
||||||
use leptos::*;
|
|
||||||
use leptos_axum::{generate_route_list, LeptosRoutes};
|
|
||||||
use doordesk::app::*;
|
|
||||||
use doordesk::fileserv::file_and_error_handler;
|
|
||||||
|
|
||||||
simple_logger::init_with_level(log::Level::Debug).expect("couldn't initialize logging");
|
simple_logger::init_with_level(log::Level::Debug).expect("couldn't initialize logging");
|
||||||
|
|
||||||
// Setting get_configuration(None) means we'll be using cargo-leptos's env values
|
// Setting get_configuration(None) means we'll be using cargo-leptos's env values
|
||||||
|
@ -17,27 +18,20 @@ async fn main() {
|
||||||
let conf = get_configuration(None).await.unwrap();
|
let conf = get_configuration(None).await.unwrap();
|
||||||
let leptos_options = conf.leptos_options;
|
let leptos_options = conf.leptos_options;
|
||||||
let addr = leptos_options.site_addr;
|
let addr = leptos_options.site_addr;
|
||||||
let routes = generate_route_list(|cx| view! { cx, <App/> }).await;
|
let routes = generate_route_list(App);
|
||||||
|
|
||||||
// build our application with a route
|
// build our application with a route
|
||||||
let app = Router::new()
|
let app = Router::new()
|
||||||
.route("/api/*fn_name", post(leptos_axum::handle_server_fns))
|
.route("/api/*fn_name", post(leptos_axum::handle_server_fns))
|
||||||
.leptos_routes(&leptos_options, routes, |cx| view! { cx, <App/> })
|
.leptos_routes(&leptos_options, routes, App)
|
||||||
.fallback(file_and_error_handler)
|
.fallback(file_and_error_handler)
|
||||||
.with_state(leptos_options);
|
.with_state(leptos_options);
|
||||||
|
|
||||||
// run our app with hyper
|
// run our app with hyper
|
||||||
// `axum::Server` is a re-export of `hyper::Server`
|
// `axum::Server` is a re-export of `hyper::Server`
|
||||||
log!("listening on http://{}", &addr);
|
log::info!("listening on http://{}", &addr);
|
||||||
axum::Server::bind(&addr)
|
axum::Server::bind(&addr)
|
||||||
.serve(app.into_make_service())
|
.serve(app.into_make_service())
|
||||||
.await
|
.await
|
||||||
.unwrap();
|
.unwrap();
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(not(feature = "ssr"))]
|
|
||||||
pub fn main() {
|
|
||||||
// no client-side main function
|
|
||||||
// unless we want this to work with e.g., Trunk for a purely client-side app
|
|
||||||
// see lib.rs for hydration function instead
|
|
||||||
}
|
|
12
doordesk/tailwind.config.js
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
/** @type {import('tailwindcss').Config} */
|
||||||
|
module.exports = {
|
||||||
|
content: {
|
||||||
|
relative: true,
|
||||||
|
files: ["*.html", "./app/**/*.rs"],
|
||||||
|
},
|
||||||
|
theme: {
|
||||||
|
extend: {},
|
||||||
|
},
|
||||||
|
plugins: [],
|
||||||
|
}
|
||||||
|
|