From 24b04c0f5f1b31580b453659de1160389bfcbbba Mon Sep 17 00:00:00 2001 From: Elad Kaplan Date: Fri, 3 May 2024 10:29:50 +0300 Subject: [PATCH 01/23] cache mechanism to app context --- Cargo.toml | 8 +- examples/demo/Cargo.lock | 382 +++++++++++++++++- examples/demo/src/app.rs | 1 + examples/demo/src/controllers/cache.rs | 24 ++ examples/demo/src/controllers/mod.rs | 1 + examples/demo/tests/cmd/cli.trycmd | 2 + examples/demo/tests/requests/cache.rs | 29 ++ examples/demo/tests/requests/mod.rs | 1 + .../requests/snapshots/insert@cache.snap | 8 + .../snapshots/key_not_exists@cache.snap | 8 + .../snapshots/read_cache_key@cache.snap | 8 + src/app.rs | 13 +- src/boot.rs | 2 + src/cache/drivers/inmem.rs | 151 +++++++ src/cache/drivers/mod.rs | 53 +++ src/cache/mod.rs | 121 ++++++ src/lib.rs | 2 + 17 files changed, 801 insertions(+), 13 deletions(-) create mode 100644 examples/demo/src/controllers/cache.rs create mode 100644 examples/demo/tests/requests/cache.rs create mode 100644 examples/demo/tests/requests/snapshots/insert@cache.snap create mode 100644 examples/demo/tests/requests/snapshots/key_not_exists@cache.snap create mode 100644 examples/demo/tests/requests/snapshots/read_cache_key@cache.snap create mode 100644 src/cache/drivers/inmem.rs create mode 100644 src/cache/drivers/mod.rs create mode 100644 src/cache/mod.rs diff --git a/Cargo.toml b/Cargo.toml index 00746ed0c..a48ae2aa5 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -22,7 +22,7 @@ rust-version.workspace = true # See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html [features] -default = ["auth_jwt", "cli", "with-db"] +default = ["auth_jwt", "cli", "with-db", "cache_inmem"] auth_jwt = ["dep:jsonwebtoken"] cli = ["dep:clap"] testing = ["dep:axum-test"] @@ -33,6 +33,9 @@ all_storage = ["storage_aws_s3", "storage_azure", "storage_gcp"] storage_aws_s3 = ["object_store/aws"] storage_azure = ["object_store/azure"] storage_gcp = ["object_store/gcp"] +# Cache feature +cache = [] +cache_inmem = ["cache", "dep:moka"] [dependencies] @@ -119,6 +122,9 @@ socketioxide = { version = "0.10.0", features = ["state"], optional = true } # File Upload object_store = { version = "0.9.0", default-features = false } +# cache +moka = { version = "0.12.7", features = ["sync"], optional = true } + [workspace.dependencies] async-trait = { version = "0.1.74" } axum = { version = "0.7.1", features = ["macros"] } diff --git a/examples/demo/Cargo.lock b/examples/demo/Cargo.lock index fce7cded5..175a55f16 100644 --- a/examples/demo/Cargo.lock +++ b/examples/demo/Cargo.lock @@ -458,6 +458,34 @@ version = "1.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "d468802bab17cbc0cc575e9b053f41e72aa36bfa6b7f55e3529ffa43161b97fa" +[[package]] +name = "axum" +version = "0.6.20" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3b829e4e32b91e643de6eafe82b1d90675f5874230191a4ffbc1b336dec4d6bf" +dependencies = [ + "async-trait", + "axum-core 0.3.4", + "bitflags 1.3.2", + "bytes", + "futures-util", + "http 0.2.11", + "http-body 0.4.6", + "hyper 0.14.28", + "itoa", + "matchit", + "memchr", + "mime", + "percent-encoding", + "pin-project-lite", + "rustversion", + "serde", + "sync_wrapper", + "tower", + "tower-layer", + "tower-service", +] + [[package]] name = "axum" version = "0.7.4" @@ -465,7 +493,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "1236b4b292f6c4d6dc34604bb5120d85c3fe1d1aa596bd5cc52ca054d13e7b9e" dependencies = [ "async-trait", - "axum-core", + "axum-core 0.4.3", "axum-macros", "bytes", "futures-util", @@ -494,6 +522,23 @@ dependencies = [ "tracing", ] +[[package]] +name = "axum-core" +version = "0.3.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "759fa577a247914fd3f7f76d62972792636412fbfd634cd452f6a385a74d2d2c" +dependencies = [ + "async-trait", + "bytes", + "futures-util", + "http 0.2.11", + "http-body 0.4.6", + "mime", + "rustversion", + "tower-layer", + "tower-service", +] + [[package]] name = "axum-core" version = "0.4.3" @@ -521,8 +566,8 @@ version = "0.9.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "895ff42f72016617773af68fb90da2a9677d89c62338ec09162d4909d86fdd8f" dependencies = [ - "axum", - "axum-core", + "axum 0.7.4", + "axum-core 0.4.3", "bytes", "cookie", "futures-util", @@ -555,7 +600,7 @@ version = "0.6.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "b683cbc43010e9a3d72c2f31ca464155ff4f95819e88a32924b0f47a43898978" dependencies = [ - "axum", + "axum 0.7.4", "bytes", "futures", "futures-core", @@ -580,7 +625,7 @@ dependencies = [ "anyhow", "async-trait", "auto-future", - "axum", + "axum 0.7.4", "bytes", "cookie", "http 1.0.0", @@ -600,6 +645,24 @@ dependencies = [ "url", ] +[[package]] +name = "axum-tracing-opentelemetry" +version = "0.18.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f26965dca35be1ca4a128177b9302c4c04f0661443621171adb09d002dcdf1d5" +dependencies = [ + "axum 0.7.4", + "futures-core", + "futures-util", + "http 1.0.0", + "opentelemetry", + "pin-project-lite", + "tower", + "tracing", + "tracing-opentelemetry", + "tracing-opentelemetry-instrumentation-sdk", +] + [[package]] name = "axum_session" version = "0.10.1" @@ -608,7 +671,7 @@ checksum = "d98669d6e652c4688e12f5d9f9a6a0077f5be101a1bfaa169062d3df70ad4e38" dependencies = [ "aes-gcm", "async-trait", - "axum-core", + "axum-core 0.4.3", "base64", "bytes", "chrono", @@ -732,7 +795,7 @@ name = "blo" version = "0.1.0" dependencies = [ "async-trait", - "axum", + "axum 0.7.4", "axum-extra", "axum-test", "axum_session", @@ -1178,6 +1241,15 @@ dependencies = [ "once_cell", ] +[[package]] +name = "crossbeam-channel" +version = "0.5.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ab3db02a9c5b5121e1e42fbdb1aeb65f5e02624cc58c43f2884c6ccac0b82f95" +dependencies = [ + "crossbeam-utils", +] + [[package]] name = "crossbeam-deque" version = "0.8.5" @@ -1925,6 +1997,25 @@ dependencies = [ "wasm-bindgen", ] +[[package]] +name = "h2" +version = "0.3.26" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "81fe527a889e1532da5c525686d96d4c2e74cdd345badf8dfef9f6b39dd5f5e8" +dependencies = [ + "bytes", + "fnv", + "futures-core", + "futures-sink", + "futures-util", + "http 0.2.11", + "indexmap 2.2.2", + "slab", + "tokio", + "tokio-util", + "tracing", +] + [[package]] name = "h2" version = "0.4.2" @@ -2158,6 +2249,7 @@ dependencies = [ "futures-channel", "futures-core", "futures-util", + "h2 0.3.26", "http 0.2.11", "http-body 0.4.6", "httparse", @@ -2180,7 +2272,7 @@ dependencies = [ "bytes", "futures-channel", "futures-util", - "h2", + "h2 0.4.2", "http 1.0.0", "http-body 1.0.0", "httparse", @@ -2191,6 +2283,18 @@ dependencies = [ "want", ] +[[package]] +name = "hyper-timeout" +version = "0.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bbb958482e8c7be4bc3cf272a766a2b0bf1a6755e7a6ae777f017a31d11b13b1" +dependencies = [ + "hyper 0.14.28", + "pin-project-lite", + "tokio", + "tokio-io-timeout", +] + [[package]] name = "hyper-util" version = "0.1.3" @@ -2332,6 +2436,22 @@ dependencies = [ "syn 2.0.48", ] +[[package]] +name = "init-tracing-opentelemetry" +version = "0.18.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "754367a7e7f9314afe3ee9780ba8b8ea2cab487cdc8017c9794793051242f878" +dependencies = [ + "opentelemetry", + "opentelemetry-otlp", + "opentelemetry-semantic-conventions", + "opentelemetry_sdk", + "thiserror", + "tracing", + "tracing-opentelemetry", + "tracing-subscriber", +] + [[package]] name = "inout" version = "0.1.3" @@ -2566,12 +2686,19 @@ name = "loco-extras" version = "0.4.0" dependencies = [ "async-trait", - "axum", + "axum 0.7.4", "axum-prometheus", + "axum-tracing-opentelemetry", + "init-tracing-opentelemetry", "loco-rs", + "opentelemetry", + "opentelemetry-otlp", "serde_json", "tower", "tower-http", + "tracing", + "tracing-opentelemetry-instrumentation-sdk", + "tracing-subscriber", ] [[package]] @@ -2588,7 +2715,7 @@ version = "0.4.0" dependencies = [ "argon2", "async-trait", - "axum", + "axum 0.7.4", "axum-extra", "axum-test", "backtrace_printer", @@ -2610,6 +2737,7 @@ dependencies = [ "lazy_static", "lettre", "mime", + "moka", "object_store", "rand", "regex", @@ -2773,6 +2901,26 @@ dependencies = [ "windows-sys 0.48.0", ] +[[package]] +name = "moka" +version = "0.12.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9e0d88686dc561d743b40de8269b26eaf0dc58781bde087b0984646602021d08" +dependencies = [ + "crossbeam-channel", + "crossbeam-epoch", + "crossbeam-utils", + "once_cell", + "parking_lot", + "quanta", + "rustc_version", + "smallvec", + "tagptr", + "thiserror", + "triomphe", + "uuid", +] + [[package]] name = "multer" version = "3.0.0" @@ -2953,6 +3101,93 @@ version = "0.3.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "624a8340c38c1b80fd549087862da4ba43e08858af025b236e509b6649fc13d5" +[[package]] +name = "opentelemetry" +version = "0.22.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "900d57987be3f2aeb70d385fff9b27fb74c5723cc9a52d904d4f9c807a0667bf" +dependencies = [ + "futures-core", + "futures-sink", + "js-sys", + "once_cell", + "pin-project-lite", + "thiserror", + "urlencoding", +] + +[[package]] +name = "opentelemetry-http" +version = "0.11.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7690dc77bf776713848c4faa6501157469017eaf332baccd4eb1cea928743d94" +dependencies = [ + "async-trait", + "bytes", + "http 0.2.11", + "opentelemetry", +] + +[[package]] +name = "opentelemetry-otlp" +version = "0.15.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1a016b8d9495c639af2145ac22387dcb88e44118e45320d9238fbf4e7889abcb" +dependencies = [ + "async-trait", + "futures-core", + "http 0.2.11", + "opentelemetry", + "opentelemetry-http", + "opentelemetry-proto", + "opentelemetry-semantic-conventions", + "opentelemetry_sdk", + "prost", + "thiserror", + "tokio", + "tonic", +] + +[[package]] +name = "opentelemetry-proto" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3a8fddc9b68f5b80dae9d6f510b88e02396f006ad48cac349411fbecc80caae4" +dependencies = [ + "opentelemetry", + "opentelemetry_sdk", + "prost", + "tonic", +] + +[[package]] +name = "opentelemetry-semantic-conventions" +version = "0.14.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f9ab5bd6c42fb9349dcf28af2ba9a0667f697f9bdcca045d39f2cec5543e2910" + +[[package]] +name = "opentelemetry_sdk" +version = "0.22.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9e90c7113be649e31e9a0f8b5ee24ed7a16923b322c3c5ab6367469c049d6b7e" +dependencies = [ + "async-trait", + "crossbeam-channel", + "futures-channel", + "futures-executor", + "futures-util", + "glob", + "once_cell", + "opentelemetry", + "ordered-float 4.2.0", + "percent-encoding", + "rand", + "thiserror", + "tokio", + "tokio-stream", +] + [[package]] name = "ordered-float" version = "3.9.2" @@ -2962,6 +3197,15 @@ dependencies = [ "num-traits", ] +[[package]] +name = "ordered-float" +version = "4.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a76df7075c7d4d01fdcb46c912dd17fba5b60c78ea480b475f2b6ab6f666584e" +dependencies = [ + "num-traits", +] + [[package]] name = "os_pipe" version = "1.1.5" @@ -3353,6 +3597,29 @@ dependencies = [ "unicode-ident", ] +[[package]] +name = "prost" +version = "0.12.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d0f5d036824e4761737860779c906171497f6d55681139d8312388f8fe398922" +dependencies = [ + "bytes", + "prost-derive", +] + +[[package]] +name = "prost-derive" +version = "0.12.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "19de2de2a00075bf566bee3bd4db014b11587e84184d3f7a791bc17f1a8e9e48" +dependencies = [ + "anyhow", + "itertools", + "proc-macro2", + "quote", + "syn 2.0.48", +] + [[package]] name = "psm" version = "0.1.21" @@ -4036,7 +4303,7 @@ dependencies = [ "chrono", "derivative", "inherent", - "ordered-float", + "ordered-float 3.9.2", "rust_decimal", "sea-query-derive", "serde_json", @@ -4846,6 +5113,12 @@ version = "0.1.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "2047c6ded9c721764247e62cd3b03c09ffc529b2ba5b10ec482ae507a4a70160" +[[package]] +name = "tagptr" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7b2093cf4c8eb1e67749a6762251bc9cd836b6fc171623bd0a9d324d37af2417" + [[package]] name = "tap" version = "1.0.1" @@ -5014,6 +5287,16 @@ dependencies = [ "windows-sys 0.48.0", ] +[[package]] +name = "tokio-io-timeout" +version = "1.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "30b74022ada614a1b4834de765f9bb43877f910cc8ce4be40e89042c9223a8bf" +dependencies = [ + "pin-project-lite", + "tokio", +] + [[package]] name = "tokio-macros" version = "2.2.0" @@ -5094,6 +5377,33 @@ dependencies = [ "winnow", ] +[[package]] +name = "tonic" +version = "0.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "76c4eb7a4e9ef9d4763600161f12f5070b92a578e1b634db88a6887844c91a13" +dependencies = [ + "async-stream", + "async-trait", + "axum 0.6.20", + "base64", + "bytes", + "h2 0.3.26", + "http 0.2.11", + "http-body 0.4.6", + "hyper 0.14.28", + "hyper-timeout", + "percent-encoding", + "pin-project", + "prost", + "tokio", + "tokio-stream", + "tower", + "tower-layer", + "tower-service", + "tracing", +] + [[package]] name = "tower" version = "0.4.13" @@ -5102,9 +5412,13 @@ checksum = "b8fa9be0de6cf49e536ce1851f987bd21a43b771b09473c3549a6c853db37c1c" dependencies = [ "futures-core", "futures-util", + "indexmap 1.9.3", "pin-project", "pin-project-lite", + "rand", + "slab", "tokio", + "tokio-util", "tower-layer", "tower-service", "tracing", @@ -5193,6 +5507,36 @@ dependencies = [ "tracing-core", ] +[[package]] +name = "tracing-opentelemetry" +version = "0.23.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a9be14ba1bbe4ab79e9229f7f89fab8d120b865859f10527f31c033e599d2284" +dependencies = [ + "js-sys", + "once_cell", + "opentelemetry", + "opentelemetry_sdk", + "smallvec", + "tracing", + "tracing-core", + "tracing-log", + "tracing-subscriber", + "web-time", +] + +[[package]] +name = "tracing-opentelemetry-instrumentation-sdk" +version = "0.18.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9a276058193f1b03d8279356215ec4c8c1bb21e40e5554bb239aa94fb2d8e189" +dependencies = [ + "http 1.0.0", + "opentelemetry", + "tracing", + "tracing-opentelemetry", +] + [[package]] name = "tracing-serde" version = "0.1.3" @@ -5224,6 +5568,12 @@ dependencies = [ "tracing-serde", ] +[[package]] +name = "triomphe" +version = "0.1.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "859eb650cfee7434994602c3a68b25d77ad9e68c8a6cd491616ef86661382eb3" + [[package]] name = "try-lock" version = "0.2.5" @@ -5657,6 +6007,16 @@ dependencies = [ "wasm-bindgen", ] +[[package]] +name = "web-time" +version = "1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5a6580f308b1fad9207618087a65c04e7a10bc77e02c8e84e9b00dd4b12fa0bb" +dependencies = [ + "js-sys", + "wasm-bindgen", +] + [[package]] name = "webpki-roots" version = "0.25.4" diff --git a/examples/demo/src/app.rs b/examples/demo/src/app.rs index 1edeaebb1..ba34d4d81 100644 --- a/examples/demo/src/app.rs +++ b/examples/demo/src/app.rs @@ -67,6 +67,7 @@ impl Hooks for App { .add_route(controllers::user::routes()) .add_route(controllers::upload::routes()) .add_route(controllers::responses::routes()) + .add_route(controllers::cache::routes()) } async fn boot(mode: StartMode, environment: &Environment) -> Result { diff --git a/examples/demo/src/controllers/cache.rs b/examples/demo/src/controllers/cache.rs new file mode 100644 index 000000000..976032a5f --- /dev/null +++ b/examples/demo/src/controllers/cache.rs @@ -0,0 +1,24 @@ +use loco_rs::prelude::*; +use serde::{Deserialize, Serialize}; + +#[derive(Serialize, Deserialize)] +pub struct CacheResponse { + value: Option, +} + +async fn get_cache(State(ctx): State) -> Result { + format::json(CacheResponse { + value: ctx.cache.get("value").await.unwrap(), + }) +} +async fn insert(State(ctx): State) -> Result { + ctx.cache.insert("value", "loco cache value").await.unwrap(); + format::empty() +} + +pub fn routes() -> Routes { + Routes::new() + .prefix("cache") + .add("/", get(get_cache)) + .add("/insert", post(insert)) +} diff --git a/examples/demo/src/controllers/mod.rs b/examples/demo/src/controllers/mod.rs index a63021c74..4645b9c6f 100644 --- a/examples/demo/src/controllers/mod.rs +++ b/examples/demo/src/controllers/mod.rs @@ -1,4 +1,5 @@ pub mod auth; +pub mod cache; pub mod dashboard; pub mod mysession; pub mod notes; diff --git a/examples/demo/tests/cmd/cli.trycmd b/examples/demo/tests/cmd/cli.trycmd index fe98eba25..42494d60e 100644 --- a/examples/demo/tests/cmd/cli.trycmd +++ b/examples/demo/tests/cmd/cli.trycmd @@ -87,6 +87,8 @@ $ blo-cli routes --environment test [POST] /auth/register [POST] /auth/reset [POST] /auth/verify +[GET] /cache +[POST] /cache/insert [GET] /dashboard/hello [GET] /dashboard/home [GET] /mysession diff --git a/examples/demo/tests/requests/cache.rs b/examples/demo/tests/requests/cache.rs new file mode 100644 index 000000000..028422aa2 --- /dev/null +++ b/examples/demo/tests/requests/cache.rs @@ -0,0 +1,29 @@ +use blo::app::App; +use insta::assert_debug_snapshot; +use loco_rs::testing; + +// TODO: see how to dedup / extract this to app-local test utils +// not to framework, because that would require a runtime dep on insta +macro_rules! configure_insta { + ($($expr:expr),*) => { + let mut settings = insta::Settings::clone_current(); + settings.set_prepend_module_to_snapshot(false); + settings.set_snapshot_suffix("cache"); + let _guard = settings.bind_to_scope(); + }; +} + +#[tokio::test] +async fn ping() { + configure_insta!(); + + testing::request::(|request, _ctx| async move { + let response = request.get("cache").await; + assert_debug_snapshot!("key_not_exists", (response.text(), response.status_code())); + let response = request.post("cache/insert").await; + assert_debug_snapshot!("insert", (response.text(), response.status_code())); + let response = request.get("cache").await; + assert_debug_snapshot!("read_cache_key", (response.text(), response.status_code())); + }) + .await; +} diff --git a/examples/demo/tests/requests/mod.rs b/examples/demo/tests/requests/mod.rs index 5eca40a02..1dadf7044 100644 --- a/examples/demo/tests/requests/mod.rs +++ b/examples/demo/tests/requests/mod.rs @@ -1,4 +1,5 @@ mod auth; +mod cache; mod notes; mod ping; mod prepare_data; diff --git a/examples/demo/tests/requests/snapshots/insert@cache.snap b/examples/demo/tests/requests/snapshots/insert@cache.snap new file mode 100644 index 000000000..e4108c4da --- /dev/null +++ b/examples/demo/tests/requests/snapshots/insert@cache.snap @@ -0,0 +1,8 @@ +--- +source: tests/requests/cache.rs +expression: "(response.text(), response.status_code())" +--- +( + "", + 200, +) diff --git a/examples/demo/tests/requests/snapshots/key_not_exists@cache.snap b/examples/demo/tests/requests/snapshots/key_not_exists@cache.snap new file mode 100644 index 000000000..8fd34bc61 --- /dev/null +++ b/examples/demo/tests/requests/snapshots/key_not_exists@cache.snap @@ -0,0 +1,8 @@ +--- +source: tests/requests/cache.rs +expression: "(response.text(), response.status_code())" +--- +( + "{\"value\":null}", + 200, +) diff --git a/examples/demo/tests/requests/snapshots/read_cache_key@cache.snap b/examples/demo/tests/requests/snapshots/read_cache_key@cache.snap new file mode 100644 index 000000000..a06ae0dc3 --- /dev/null +++ b/examples/demo/tests/requests/snapshots/read_cache_key@cache.snap @@ -0,0 +1,8 @@ +--- +source: tests/requests/cache.rs +expression: "(response.text(), response.status_code())" +--- +( + "{\"value\":\"loco cache value\"}", + 200, +) diff --git a/src/app.rs b/src/app.rs index 878aa6dc2..ddfde6d2a 100644 --- a/src/app.rs +++ b/src/app.rs @@ -12,6 +12,8 @@ use std::sync::Arc; use async_trait::async_trait; use axum::Router as AxumRouter; +#[cfg(feature = "cache")] +use crate::cache; #[cfg(feature = "channels")] use crate::controller::channels::AppChannels; use crate::{ @@ -46,8 +48,11 @@ pub struct AppContext { pub config: Config, /// An optional email sender component that can be used to send email. pub mailer: Option, - // Ab optional storage instance for the application + // An optional storage instance for the application pub storage: Option>, + #[cfg(feature = "cache")] + // Cache instance for the application + pub cache: Arc, } /// A trait that defines hooks for customizing and extending the behavior of a @@ -163,6 +168,12 @@ pub trait Hooks { Ok(None) } + #[cfg(feature = "cache")] + /// Defines the cache configuration for the application. + async fn cache(_config: &config::Config, _environment: &Environment) -> Result { + Ok(cache::Cache::new(cache::drivers::inmem::new())) + } + #[cfg(feature = "channels")] /// Register channels endpoints to the application routers fn register_channels(_ctx: &AppContext) -> AppChannels; diff --git a/src/boot.rs b/src/boot.rs index e97e72e12..e2d48d792 100644 --- a/src/boot.rs +++ b/src/boot.rs @@ -205,6 +205,8 @@ pub async fn create_context(environment: &Environment) -> Result Box { + let cache = Cache::builder().max_capacity(32 * 1024 * 1024).build(); + Inmem::from(cache) +} + +/// Represents the in-memory cache driver. +pub struct Inmem { + cache: Cache, +} + +impl Inmem { + /// Constructs a new [`Inmem`] instance from a given cache. + /// + /// # Returns + /// + /// A boxed [`CacheDriver`] instance. + #[must_use] + pub fn from(cache: Cache) -> Box { + Box::new(Self { cache }) + } +} + +#[async_trait] +impl CacheDriver for Inmem { + /// Checks if a key exists in the cache. + /// + /// # Errors + /// + /// Returns a `CacheError` if there is an error during the operation. + async fn contains_key(&self, key: &str) -> CacheResult { + Ok(self.cache.contains_key(key)) + } + + /// Retrieves a value from the cache based on the provided key. + /// + /// # Errors + /// + /// Returns a `CacheError` if there is an error during the operation. + async fn get(&self, key: &str) -> CacheResult> { + Ok(self.cache.get(key)) + } + + /// Inserts a key-value pair into the cache. + /// + /// # Errors + /// + /// Returns a `CacheError` if there is an error during the operation. + async fn insert(&self, key: &str, value: &str) -> CacheResult<()> { + self.cache + .insert(key.to_string(), Arc::new(value).to_string()); + Ok(()) + } + + /// Removes a key-value pair from the cache. + /// + /// # Errors + /// + /// Returns a `CacheError` if there is an error during the operation. + async fn remove(&self, key: &str) -> CacheResult<()> { + self.cache.remove(key); + Ok(()) + } + + /// Clears all key-value pairs from the cache. + /// + /// # Errors + /// + /// Returns a `CacheError` if there is an error during the operation. + async fn clear(&self) -> CacheResult<()> { + self.cache.invalidate_all(); + Ok(()) + } +} + +#[cfg(test)] +mod tests { + + use super::*; + + #[tokio::test] + async fn is_contains_key() { + let mem = new(); + assert!(!mem.contains_key("key").await.unwrap()); + assert!(mem.insert("key", "loco").await.is_ok()); + assert!(mem.contains_key("key").await.unwrap()); + } + + #[tokio::test] + async fn can_get_key_value() { + let mem = new(); + assert!(mem.insert("key", "loco").await.is_ok()); + assert_eq!(mem.get("key").await.unwrap(), Some("loco".to_string())); + + //try getting key that not exists + assert_eq!(mem.get("not-found").await.unwrap(), None); + } + + #[tokio::test] + async fn can_remove_key() { + let mem = new(); + assert!(mem.insert("key", "loco").await.is_ok()); + assert!(mem.contains_key("key").await.unwrap()); + mem.remove("key").await.unwrap(); + assert!(!mem.contains_key("key").await.unwrap()); + } + + #[tokio::test] + async fn can_clear() { + let mem = new(); + + let keys = vec!["key", "key2", "key3"]; + for key in &keys { + assert!(mem.insert(key, "loco").await.is_ok()); + } + for key in &keys { + assert!(mem.contains_key(key).await.is_ok()); + } + assert!(mem.clear().await.is_ok()); + for key in &keys { + assert!(!mem.contains_key(key).await.unwrap()); + } + } +} diff --git a/src/cache/drivers/mod.rs b/src/cache/drivers/mod.rs new file mode 100644 index 000000000..db6e63e44 --- /dev/null +++ b/src/cache/drivers/mod.rs @@ -0,0 +1,53 @@ +//! # Cache Drivers Module +//! +//! This module defines traits and implementations for cache drivers. +use async_trait::async_trait; + +use super::CacheResult; + +#[cfg(feature = "cache_inmem")] +pub mod inmem; + +/// Trait representing a cache driver. +#[async_trait] +pub trait CacheDriver: Sync + Send { + /// Checks if a key exists in the cache. + /// + /// # Errors + /// + /// Returns a [`super::CacheError`] if there is an error during the + /// operation. + async fn contains_key(&self, key: &str) -> CacheResult; + + /// Retrieves a value from the cache based on the provided key. + /// + /// # Errors + /// + /// Returns a [`super::CacheError`] if there is an error during the + /// operation. + async fn get(&self, key: &str) -> CacheResult>; + + /// Inserts a key-value pair into the cache. + /// + /// # Errors + /// + /// Returns a [`super::CacheError`] if there is an error during the + /// operation. + async fn insert(&self, key: &str, value: &str) -> CacheResult<()>; + + /// Removes a key-value pair from the cache. + /// + /// # Errors + /// + /// Returns a [`super::CacheError`] if there is an error during the + /// operation. + async fn remove(&self, key: &str) -> CacheResult<()>; + + /// Clears all key-value pairs from the cache. + /// + /// # Errors + /// + /// Returns a [`super::CacheError`] if there is an error during the + /// operation. + async fn clear(&self) -> CacheResult<()>; +} diff --git a/src/cache/mod.rs b/src/cache/mod.rs new file mode 100644 index 000000000..ff4190071 --- /dev/null +++ b/src/cache/mod.rs @@ -0,0 +1,121 @@ +//! # Cache Module +//! +//! This module provides a generic cache interface for various cache drivers. +pub mod drivers; + +use self::drivers::CacheDriver; + +/// Errors related to cache operations +#[derive(thiserror::Error, Debug)] +#[allow(clippy::module_name_repetitions)] +pub enum CacheError {} + +pub type CacheResult = std::result::Result; + +/// Represents a cache instance +pub struct Cache { + /// The cache driver used for underlying operations + pub driver: Box, +} + +impl Cache { + /// Creates a new cache instance with the specified cache driver. + #[must_use] + pub fn new(driver: Box) -> Self { + Self { driver } + } + + /// Checks if a key exists in the cache. + /// + /// # Example + /// ``` + /// use loco_rs::cache::{self, CacheResult}; + /// + /// pub async fn contains_key() -> CacheResult { + /// let cache = cache::Cache::new(cache::drivers::mem::new()); + /// cache.contains_key("key").await + /// } + /// ``` + /// + /// # Errors + /// A [`CacheResult`] indicating whether the key exists in the cache. + pub async fn contains_key(&self, key: &str) -> CacheResult { + self.driver.contains_key(key).await + } + + /// Retrieves a value from the cache based on the provided key. + /// + /// # Example + /// ``` + /// use loco_rs::cache::{self, CacheResult}; + /// + /// pub async fn get_key() -> CacheResult> { + /// let cache = cache::Cache::new(cache::drivers::mem::new()); + /// cache.get("key").await + /// } + /// ``` + /// + /// # Errors + /// A [`CacheResult`] containing an `Option` representing the retrieved + /// value. + pub async fn get(&self, key: &str) -> CacheResult> { + self.driver.get(key).await + } + + /// Inserts a key-value pair into the cache. + /// + /// # Example + /// ``` + /// use loco_rs::cache::{self, CacheResult}; + /// + /// pub async fn insert() -> CacheResult<()> { + /// let cache = cache::Cache::new(cache::drivers::mem::new()); + /// cache.insert("key", "value").await + /// } + /// ``` + /// + /// # Errors + /// + /// A [`CacheResult`] indicating the success of the operation. + pub async fn insert(&self, key: &str, value: &str) -> CacheResult<()> { + self.driver.insert(key, value).await + } + + /// Removes a key-value pair from the cache. + /// + /// # Example + /// ``` + /// use loco_rs::cache::{self, CacheResult}; + /// + /// pub async fn remove() -> CacheResult<()> { + /// let cache = cache::Cache::new(cache::drivers::mem::new()); + /// cache.remove("key").await + /// } + /// ``` + /// + /// # Errors + /// + /// A [`CacheResult`] indicating the success of the operation. + pub async fn remove(&self, key: &str) -> CacheResult<()> { + self.driver.remove(key).await + } + + /// Clears all key-value pairs from the cache. + /// + /// # Example + /// ``` + /// use loco_rs::cache::{self, CacheResult}; + /// + /// pub async fn clear() -> CacheResult<()> { + /// let cache = cache::Cache::new(cache::drivers::mem::new()); + /// cache.clear().await + /// } + /// ``` + /// + /// # Errors + /// + /// A [`CacheResult`] indicating the success of the operation. + pub async fn clear(&self) -> CacheResult<()> { + self.driver.clear().await + } +} diff --git a/src/lib.rs b/src/lib.rs index 97b5c1d5d..62855a0ab 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -47,6 +47,8 @@ pub mod cli; pub mod auth; pub mod boot; +#[cfg(feature = "cache")] +pub mod cache; pub mod config; pub mod controller; pub mod environment; From 24cf1e73b1e76189736d33c27d0f37a7dff3f705 Mon Sep 17 00:00:00 2001 From: Elad Kaplan Date: Fri, 3 May 2024 10:30:53 +0300 Subject: [PATCH 02/23] cache mechanism to app context --- src/cache/drivers/inmem.rs | 10 ---------- 1 file changed, 10 deletions(-) diff --git a/src/cache/drivers/inmem.rs b/src/cache/drivers/inmem.rs index 0185ede2a..741b9644a 100644 --- a/src/cache/drivers/inmem.rs +++ b/src/cache/drivers/inmem.rs @@ -1,16 +1,6 @@ //! # In-Memory Cache Driver //! //! This module implements a cache driver using an in-memory cache. -//! -//! ## Usage TODO.... -//! -//! ```rust -//! use super::CacheDriver; -//! use crate::cache::CacheResult; -//! -//! // Create a new instance of the in-memory cache driver -//! let mem = new(); -//! ``` use std::sync::Arc; use async_trait::async_trait; From d97c30a38acdbfd67efc3e03c53a752c2849f962 Mon Sep 17 00:00:00 2001 From: Elad Kaplan Date: Fri, 3 May 2024 10:46:04 +0300 Subject: [PATCH 03/23] fix docs example --- examples/demo/Cargo.lock | 2 ++ src/cache/mod.rs | 10 +++++----- 2 files changed, 7 insertions(+), 5 deletions(-) diff --git a/examples/demo/Cargo.lock b/examples/demo/Cargo.lock index 175a55f16..1b92f6caf 100644 --- a/examples/demo/Cargo.lock +++ b/examples/demo/Cargo.lock @@ -802,6 +802,7 @@ dependencies = [ "chrono", "eyre", "fluent-templates", + "futures-util", "include_dir", "insta", "loco-extras", @@ -815,6 +816,7 @@ dependencies = [ "serial_test", "tera", "tokio", + "tower", "tracing", "tracing-subscriber", "trycmd", diff --git a/src/cache/mod.rs b/src/cache/mod.rs index ff4190071..cb6161a4f 100644 --- a/src/cache/mod.rs +++ b/src/cache/mod.rs @@ -32,7 +32,7 @@ impl Cache { /// use loco_rs::cache::{self, CacheResult}; /// /// pub async fn contains_key() -> CacheResult { - /// let cache = cache::Cache::new(cache::drivers::mem::new()); + /// let cache = cache::Cache::new(cache::drivers::inmem::new()); /// cache.contains_key("key").await /// } /// ``` @@ -50,7 +50,7 @@ impl Cache { /// use loco_rs::cache::{self, CacheResult}; /// /// pub async fn get_key() -> CacheResult> { - /// let cache = cache::Cache::new(cache::drivers::mem::new()); + /// let cache = cache::Cache::new(cache::drivers::inmem::new()); /// cache.get("key").await /// } /// ``` @@ -69,7 +69,7 @@ impl Cache { /// use loco_rs::cache::{self, CacheResult}; /// /// pub async fn insert() -> CacheResult<()> { - /// let cache = cache::Cache::new(cache::drivers::mem::new()); + /// let cache = cache::Cache::new(cache::drivers::inmem::new()); /// cache.insert("key", "value").await /// } /// ``` @@ -88,7 +88,7 @@ impl Cache { /// use loco_rs::cache::{self, CacheResult}; /// /// pub async fn remove() -> CacheResult<()> { - /// let cache = cache::Cache::new(cache::drivers::mem::new()); + /// let cache = cache::Cache::new(cache::drivers::inmem::new()); /// cache.remove("key").await /// } /// ``` @@ -107,7 +107,7 @@ impl Cache { /// use loco_rs::cache::{self, CacheResult}; /// /// pub async fn clear() -> CacheResult<()> { - /// let cache = cache::Cache::new(cache::drivers::mem::new()); + /// let cache = cache::Cache::new(cache::drivers::inmem::new()); /// cache.clear().await /// } /// ``` From ea5b65cd5b2f6b2a85b530815f8ea104ca10fb5f Mon Sep 17 00:00:00 2001 From: Elad Kaplan Date: Tue, 21 May 2024 21:59:01 +0300 Subject: [PATCH 04/23] create a default null providers for storage and cache --- Cargo.toml | 3 +- examples/demo/src/app.rs | 12 ++- examples/demo/src/controllers/upload.rs | 1 - examples/demo/tests/requests/upload.rs | 2 +- .../src/initializers/opentelemetry/mod.rs | 2 +- src/app.rs | 34 ++++--- src/boot.rs | 5 +- src/cache/drivers/mod.rs | 1 + src/cache/drivers/null.rs | 79 +++++++++++++++ src/cache/mod.rs | 5 +- src/lib.rs | 1 - src/storage/drivers/mod.rs | 1 + src/storage/drivers/null.rs | 95 +++++++++++++++++++ src/storage/mod.rs | 3 + 14 files changed, 214 insertions(+), 30 deletions(-) create mode 100644 src/cache/drivers/null.rs create mode 100644 src/storage/drivers/null.rs diff --git a/Cargo.toml b/Cargo.toml index a48ae2aa5..905d4ba28 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -34,8 +34,7 @@ storage_aws_s3 = ["object_store/aws"] storage_azure = ["object_store/azure"] storage_gcp = ["object_store/gcp"] # Cache feature -cache = [] -cache_inmem = ["cache", "dep:moka"] +cache_inmem = ["dep:moka"] [dependencies] diff --git a/examples/demo/src/app.rs b/examples/demo/src/app.rs index b6599bb86..1c4d03064 100644 --- a/examples/demo/src/app.rs +++ b/examples/demo/src/app.rs @@ -5,6 +5,7 @@ use loco_extras; use loco_rs::{ app::{AppContext, Hooks, Initializer}, boot::{create_app, BootResult, StartMode}, + cache, config::Config, controller::AppRoutes, db::{self, truncate_table}, @@ -80,10 +81,7 @@ impl Hooks for App { create_app::(mode, environment).await } - async fn storage( - _config: &Config, - environment: &Environment, - ) -> Result> { + async fn storage(_config: &Config, environment: &Environment) -> Result { let store = if environment == &Environment::Test { storage::drivers::mem::new() } else { @@ -91,7 +89,11 @@ impl Hooks for App { }; let storage = Storage::single(store); - return Ok(Some(storage)); + return Ok(storage); + } + + async fn cache(_config: &Config, _environment: &Environment) -> Result { + Ok(cache::Cache::new(cache::drivers::inmem::new())) } fn connect_workers<'a>(p: &'a mut Processor, ctx: &'a AppContext) { diff --git a/examples/demo/src/controllers/upload.rs b/examples/demo/src/controllers/upload.rs index dee168553..103cd7566 100644 --- a/examples/demo/src/controllers/upload.rs +++ b/examples/demo/src/controllers/upload.rs @@ -30,7 +30,6 @@ async fn upload_file(State(ctx): State, mut multipart: Multipart) -> let path = PathBuf::from("folder").join(file_name); ctx.storage .as_ref() - .unwrap() .upload(path.as_path(), &content) .await?; diff --git a/examples/demo/tests/requests/upload.rs b/examples/demo/tests/requests/upload.rs index d7c34865c..bc98eee5b 100644 --- a/examples/demo/tests/requests/upload.rs +++ b/examples/demo/tests/requests/upload.rs @@ -18,7 +18,7 @@ async fn can_upload_file() { let res: views::upload::Response = serde_json::from_str(&response.text()).unwrap(); - let stored_file: String = ctx.storage.unwrap().download(&res.path).await.unwrap(); + let stored_file: String = ctx.storage.download(&res.path).await.unwrap(); assert_eq!(stored_file, file_content); }) diff --git a/loco-extras/src/initializers/opentelemetry/mod.rs b/loco-extras/src/initializers/opentelemetry/mod.rs index b044cec8c..dc7330fa7 100644 --- a/loco-extras/src/initializers/opentelemetry/mod.rs +++ b/loco-extras/src/initializers/opentelemetry/mod.rs @@ -1,4 +1,4 @@ -use axum::{async_trait, Extension, Router as AxumRouter}; +use axum::{async_trait, Router as AxumRouter}; use axum_tracing_opentelemetry::middleware::{OtelAxumLayer, OtelInResponseLayer}; use loco_rs::{ app::{AppContext, Initializer}, diff --git a/src/app.rs b/src/app.rs index ddfde6d2a..9810ef3d3 100644 --- a/src/app.rs +++ b/src/app.rs @@ -12,17 +12,16 @@ use std::sync::Arc; use async_trait::async_trait; use axum::Router as AxumRouter; -#[cfg(feature = "cache")] -use crate::cache; #[cfg(feature = "channels")] use crate::controller::channels::AppChannels; use crate::{ boot::{BootResult, ServeParams, StartMode}, + cache::{self, Cache}, config::{self, Config}, controller::AppRoutes, environment::Environment, mailer::EmailSender, - storage::Storage, + storage::{self, Storage}, task::Tasks, worker::{Pool, Processor, RedisConnectionManager}, Result, @@ -49,8 +48,7 @@ pub struct AppContext { /// An optional email sender component that can be used to send email. pub mailer: Option, // An optional storage instance for the application - pub storage: Option>, - #[cfg(feature = "cache")] + pub storage: Arc, // Cache instance for the application pub cache: Arc, } @@ -160,18 +158,24 @@ pub trait Hooks { /// Defines the application's routing configuration. fn routes(_ctx: &AppContext) -> AppRoutes; - /// Defines the storage configuration for the application - async fn storage( - _config: &config::Config, - _environment: &Environment, - ) -> Result> { - Ok(None) + // Provides a default null storage implementation for the application. + /// + /// This function returns a [`Storage`] instance that always returns errors. + /// It simplifies user workflow by avoiding the need for feature flags or + /// configuring a specific storage driver during development or testing. + async fn storage(_config: &config::Config, _environment: &Environment) -> Result { + let storage = Storage::single(storage::drivers::null::new()); + Ok(storage) } - #[cfg(feature = "cache")] - /// Defines the cache configuration for the application. - async fn cache(_config: &config::Config, _environment: &Environment) -> Result { - Ok(cache::Cache::new(cache::drivers::inmem::new())) + /// Provides a default null cache implementation for the application. + /// + /// This function returns a [`Cache`] instance that always returns + /// errors. It simplifies user workflow by avoiding the need for feature + /// flags or optional cache driver configurations during development or + /// testing. + async fn cache(_config: &config::Config, _environment: &Environment) -> Result { + Ok(cache::Cache::new(cache::drivers::null::new())) } #[cfg(feature = "channels")] diff --git a/src/boot.rs b/src/boot.rs index e2d48d792..533e43ed9 100644 --- a/src/boot.rs +++ b/src/boot.rs @@ -1,7 +1,7 @@ //! # Application Bootstrapping and Logic //! This module contains functions and structures for bootstrapping and running //! your application. -use std::{collections::BTreeMap, sync::Arc}; +use std::collections::BTreeMap; use axum::Router; #[cfg(feature = "with-db")] @@ -204,8 +204,7 @@ pub async fn create_context(environment: &Environment) -> Result Box { + Box::new(Null {}) +} + +#[async_trait] +impl CacheDriver for Null { + /// Checks if a key exists in the cache. + /// + /// # Errors + /// + /// Returns always error + async fn contains_key(&self, _key: &str) -> CacheResult { + Err(CacheError::Any( + "Operation not supported by null cache".into(), + )) + } + + /// Retrieves a value from the cache based on the provided key. + /// + /// # Errors + /// + /// Returns always error + async fn get(&self, _key: &str) -> CacheResult> { + Ok(None) + } + + /// Inserts a key-value pair into the cache. + /// + /// # Errors + /// + /// Returns always error + async fn insert(&self, _key: &str, _value: &str) -> CacheResult<()> { + Err(CacheError::Any( + "Operation not supported by null cache".into(), + )) + } + + /// Removes a key-value pair from the cache. + /// + /// # Errors + /// + /// Returns always error + async fn remove(&self, _key: &str) -> CacheResult<()> { + Err(CacheError::Any( + "Operation not supported by null cache".into(), + )) + } + + /// Clears all key-value pairs from the cache. + /// + /// # Errors + /// + /// Returns always error + async fn clear(&self) -> CacheResult<()> { + Err(CacheError::Any( + "Operation not supported by null cache".into(), + )) + } +} diff --git a/src/cache/mod.rs b/src/cache/mod.rs index cb6161a4f..bf0e90316 100644 --- a/src/cache/mod.rs +++ b/src/cache/mod.rs @@ -8,7 +8,10 @@ use self::drivers::CacheDriver; /// Errors related to cache operations #[derive(thiserror::Error, Debug)] #[allow(clippy::module_name_repetitions)] -pub enum CacheError {} +pub enum CacheError { + #[error(transparent)] + Any(#[from] Box), +} pub type CacheResult = std::result::Result; diff --git a/src/lib.rs b/src/lib.rs index 62855a0ab..e127af7d1 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -47,7 +47,6 @@ pub mod cli; pub mod auth; pub mod boot; -#[cfg(feature = "cache")] pub mod cache; pub mod config; pub mod controller; diff --git a/src/storage/drivers/mod.rs b/src/storage/drivers/mod.rs index 7ea0a024f..3cdb6b367 100644 --- a/src/storage/drivers/mod.rs +++ b/src/storage/drivers/mod.rs @@ -10,6 +10,7 @@ pub mod azure; pub mod gcp; pub mod local; pub mod mem; +pub mod null; pub mod object_store_adapter; use super::StorageResult; diff --git a/src/storage/drivers/null.rs b/src/storage/drivers/null.rs new file mode 100644 index 000000000..73a762ed1 --- /dev/null +++ b/src/storage/drivers/null.rs @@ -0,0 +1,95 @@ +//! # Null Storage Driver +//! +//! The Null storage Driver is the default storage driver implemented when the +//! Loco framework is initialized. The primary purpose of this driver is to +//! simplify the user workflow by avoiding the need for feature flags or +//! optional storage driver configurations. +use std::path::Path; + +use async_trait::async_trait; +use bytes::Bytes; + +use super::{GetResponse, StorageResult, StoreDriver, UploadResponse}; +use crate::storage::StorageError; + +pub struct NullStorage {} + +/// Constructor for creating a new `Store` instance. +#[must_use] +pub fn new() -> Box { + Box::new(NullStorage {}) +} + +#[async_trait] +impl StoreDriver for NullStorage { + /// Uploads the content represented by `Bytes` to the specified path in the + /// object store. + /// + /// # Errors + /// + /// Returns a `StorageResult` with the result of the upload operation. + async fn upload(&self, _path: &Path, _content: &Bytes) -> StorageResult { + Err(StorageError::Any( + "Operation not supported by null storage".into(), + )) + } + + /// Retrieves the content from the specified path in the object store. + /// + /// # Errors + /// + /// Returns a `StorageResult` with the result of the retrieval operation. + async fn get(&self, _path: &Path) -> StorageResult { + Err(StorageError::Any( + "Operation not supported by null storage".into(), + )) + } + + /// Deletes the content at the specified path in the object store. + /// + /// # Errors + /// + /// Returns a `StorageResult` indicating the success of the deletion + /// operation. + async fn delete(&self, _path: &Path) -> StorageResult<()> { + Err(StorageError::Any( + "Operation not supported by null storage".into(), + )) + } + + /// Renames or moves the content from one path to another in the object + /// store. + /// + /// # Errors + /// + /// Returns a `StorageResult` indicating the success of the rename/move + /// operation. + async fn rename(&self, _from: &Path, _to: &Path) -> StorageResult<()> { + Err(StorageError::Any( + "Operation not supported by null storage".into(), + )) + } + + /// Copies the content from one path to another in the object store. + /// + /// # Errors + /// + /// Returns a `StorageResult` indicating the success of the copy operation. + async fn copy(&self, _from: &Path, _to: &Path) -> StorageResult<()> { + Err(StorageError::Any( + "Operation not supported by null storage".into(), + )) + } + + /// Checks if the content exists at the specified path in the object store. + /// + /// # Errors + /// + /// Returns a `StorageResult` with a boolean indicating the existence of the + /// content. + async fn exists(&self, _path: &Path) -> StorageResult { + Err(StorageError::Any( + "Operation not supported by null storage".into(), + )) + } +} diff --git a/src/storage/mod.rs b/src/storage/mod.rs index 8fcec1741..dc8db80d8 100644 --- a/src/storage/mod.rs +++ b/src/storage/mod.rs @@ -36,6 +36,9 @@ pub enum StorageError { #[error("secondaries errors")] Multi(BTreeMap), + + #[error(transparent)] + Any(#[from] Box), } pub type StorageResult = std::result::Result; From 4ae6c9bdc490af24140eb05354582dbcd23e314f Mon Sep 17 00:00:00 2001 From: Elad Kaplan Date: Tue, 21 May 2024 22:12:07 +0300 Subject: [PATCH 05/23] short docs --- docs-site/content/docs/the-app/cache.md | 38 +++++++++++++++++++++++ docs-site/content/docs/the-app/storage.md | 2 ++ 2 files changed, 40 insertions(+) create mode 100644 docs-site/content/docs/the-app/cache.md diff --git a/docs-site/content/docs/the-app/cache.md b/docs-site/content/docs/the-app/cache.md new file mode 100644 index 000000000..e3384294b --- /dev/null +++ b/docs-site/content/docs/the-app/cache.md @@ -0,0 +1,38 @@ ++++ +title = "Cache" +description = "" +date = 2024-02-07T08:00:00+00:00 +updated = 2024-02-07T08:00:00+00:00 +draft = false +weight = 20 +sort_by = "weight" +template = "docs/page.html" + +[extra] +lead = "" +toc = true +top = false +flair =[] ++++ + +`Loco` provides an cache layer to improve application performance by storing frequently accessed data. + +## Default Behavior + +By default, `Loco` initializes a `Null` cache driver. This means any interaction with the cache will return an error, effectively bypassing the cache functionality. + +## Enabling Caching + +To enable caching and configure a specific cache driver, you can replace the default `Null` driver with your preferred implementation. + +In your `app.rs` file, define a function named `cache` function as a Hook in the `app.rs` file and import the `cache` module from `loco_rs`. + +Here's an example using an in-memory cache driver: + +```rust +use loco_rs::cache; + +async fn cache(_config: &Config, _environment: &Environment) -> Result { + Ok(cache::Cache::new(cache::drivers::inmem::new())) +} +``` \ No newline at end of file diff --git a/docs-site/content/docs/the-app/storage.md b/docs-site/content/docs/the-app/storage.md index 52284c81c..dba441855 100644 --- a/docs-site/content/docs/the-app/storage.md +++ b/docs-site/content/docs/the-app/storage.md @@ -25,6 +25,8 @@ By default, in-memory and disk storage come out of the box. To work with cloud p - `storage_gcp` - `all_storage` +By default loco initialize a `Null` provider, meaning any work with the storage will return an error. + ## Setup Add the `storage` function as a Hook in the `app.rs` file and import the `storage` module from `loco_rs`. From 10242fde658366efa435ef55747884b703dcce22 Mon Sep 17 00:00:00 2001 From: Elad Kaplan Date: Tue, 21 May 2024 22:14:04 +0300 Subject: [PATCH 06/23] short docs --- examples/demo/Cargo.lock | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/examples/demo/Cargo.lock b/examples/demo/Cargo.lock index 1b92f6caf..62ac76fc6 100644 --- a/examples/demo/Cargo.lock +++ b/examples/demo/Cargo.lock @@ -1245,9 +1245,9 @@ dependencies = [ [[package]] name = "crossbeam-channel" -version = "0.5.12" +version = "0.5.13" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ab3db02a9c5b5121e1e42fbdb1aeb65f5e02624cc58c43f2884c6ccac0b82f95" +checksum = "33480d6946193aa8033910124896ca395333cae7e2d1113d1fef6c3272217df2" dependencies = [ "crossbeam-utils", ] @@ -3601,9 +3601,9 @@ dependencies = [ [[package]] name = "prost" -version = "0.12.4" +version = "0.12.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d0f5d036824e4761737860779c906171497f6d55681139d8312388f8fe398922" +checksum = "deb1435c188b76130da55f17a466d252ff7b1418b2ad3e037d127b94e3411f29" dependencies = [ "bytes", "prost-derive", @@ -3611,9 +3611,9 @@ dependencies = [ [[package]] name = "prost-derive" -version = "0.12.4" +version = "0.12.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "19de2de2a00075bf566bee3bd4db014b11587e84184d3f7a791bc17f1a8e9e48" +checksum = "81bddcdb20abf9501610992b6759a4c888aef7d1a7247ef75e2404275ac24af1" dependencies = [ "anyhow", "itertools", From f8fcadfd5078a448e62114c710a7e5f8c592e479 Mon Sep 17 00:00:00 2001 From: Dotan Nahum Date: Fri, 24 May 2024 14:24:00 +0300 Subject: [PATCH 07/23] docs: fix big_int --- docs-site/content/docs/the-app/models.md | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/docs-site/content/docs/the-app/models.md b/docs-site/content/docs/the-app/models.md index ecdf07416..70635dc55 100644 --- a/docs-site/content/docs/the-app/models.md +++ b/docs-site/content/docs/the-app/models.md @@ -103,9 +103,9 @@ For schema data types, you can use the following mapping to understand the schem ("int", "integer_null"), ("int!", "integer"), ("int^", "integer_uniq"), -("big_integer", "big_integer_null"), -("big_integer!", "big_integer"), -("big_integer^", "big_integer_uniq"), +("big_int", "big_integer_null"), +("big_int!", "big_integer"), +("big_int^", "big_integer_uniq"), ("float", "float_null"), ("float!", "float"), ("double", "double_null"), From acf138cbaacfbec056ec05afa2a23ffe3b7dcda3 Mon Sep 17 00:00:00 2001 From: Cam Date: Sat, 25 May 2024 16:47:23 +1200 Subject: [PATCH 08/23] Update tour and guide docs to match current behaviour. Co-Authored-By: Tomas Chmelevskij --- .../content/docs/getting-started/guide.md | 22 +++++++++---------- .../docs/getting-started/tour/index.md | 4 ++-- 2 files changed, 13 insertions(+), 13 deletions(-) diff --git a/docs-site/content/docs/getting-started/guide.md b/docs-site/content/docs/getting-started/guide.md index c43841c48..cba50f50f 100644 --- a/docs-site/content/docs/getting-started/guide.md +++ b/docs-site/content/docs/getting-started/guide.md @@ -190,7 +190,7 @@ pub async fn echo(req_body: String) -> String { req_body } -pub async fn hello(State(_ctx): State) -> Result { +pub async fn hello(State(_ctx): State) -> Result { // do something with context (database, etc) format::text("hello") } @@ -212,7 +212,7 @@ $ cargo loco start Now, let's test it out: ```sh -$ curl localhost:3000/api/guide +$ curl localhost:3000/guide hello ``` @@ -238,12 +238,12 @@ Next, set up a _hello_ route, this is the contents of `home.rs`: use loco_rs::prelude::*; // _ctx contains your database connection, as well as other app resource that you'll need -async fn hello(State(_ctx): State) -> Result { +async fn hello(State(_ctx): State) -> Result { format::text("ola, mundo") } pub fn routes() -> Routes { - Routes::new().prefix("api/home").add("/hello", get(hello)) + Routes::new().prefix("home").add("/hello", get(hello)) } ``` @@ -283,7 +283,7 @@ $ cargo loco start And hit `/home/hello`: ```sh -$ curl localhost:3000/api/home/hello +$ curl localhost:3000/home/hello ola, mundo ``` @@ -297,7 +297,7 @@ $ cargo loco routes [POST] /api/auth/register [POST] /api/auth/reset [POST] /api/auth/verify -[GET] /api/home/hello <---- this is our new route! +[GET] /home/hello <---- this is our new route! [GET] /api/notes [POST] /api/notes .. @@ -530,7 +530,7 @@ $ cargo loco start And make a request: ```sh -$ curl localhost:3000/api/articles +$ curl localhost:3000/articles [{"created_at":"...","updated_at":"...","id":1,"title":"how to build apps in 3 steps","content":"use Loco: https://loco.rs"}] ``` @@ -638,14 +638,14 @@ Add a new article: $ curl -X POST -H "Content-Type: application/json" -d '{ "title": "Your Title", "content": "Your Content xxx" -}' localhost:3000/api/articles +}' localhost:3000/articles {"created_at":"...","updated_at":"...","id":2,"title":"Your Title","content":"Your Content xxx"} ``` Get a list: ```sh -$ curl localhost:3000/api/articles +$ curl localhost:3000/articles [{"created_at":"...","updated_at":"...","id":1,"title":"how to build apps in 3 steps","content":"use Loco: https://loco.rs"},{"created_at":"...","updated_at":"...","id":2,"title":"Your Title","content":"Your Content xxx"} ``` @@ -752,14 +752,14 @@ Now let's add a comment to Article `1`: $ curl -X POST -H "Content-Type: application/json" -d '{ "content": "this rocks", "article_id": 1 -}' localhost:3000/api/comments +}' localhost:3000/comments {"created_at":"...","updated_at":"...","id":4,"content":"this rocks","article_id":1} ``` And, fetch the relation: ```sh -$ curl localhost:3000/api/articles/1/comments +$ curl localhost:3000/articles/1/comments [{"created_at":"...","updated_at":"...","id":4,"content":"this rocks","article_id":1}] ``` diff --git a/docs-site/content/docs/getting-started/tour/index.md b/docs-site/content/docs/getting-started/tour/index.md index 990351953..30de93b77 100644 --- a/docs-site/content/docs/getting-started/tour/index.md +++ b/docs-site/content/docs/getting-started/tour/index.md @@ -121,13 +121,13 @@ Next, try adding a `post` with `curl`: $ curl -X POST -H "Content-Type: application/json" -d '{ "title": "Your Title", "content": "Your Content xxx" -}' localhost:3000/api/posts +}' localhost:3000/posts ``` You can list your posts: ```sh -$ curl localhost:3000/api/posts +$ curl localhost:3000/posts ``` For those counting -- the commands for creating a blog backend were: From 883dab12df216184184e4cc21485e46fb94945ac Mon Sep 17 00:00:00 2001 From: Elad Kaplan Date: Sun, 26 May 2024 09:32:06 +0300 Subject: [PATCH 09/23] create override_context hook --- docs-site/content/docs/the-app/cache.md | 7 +++--- docs-site/content/docs/the-app/storage.md | 28 ++++++++--------------- examples/demo/src/app.rs | 13 ++++------- src/app.rs | 6 +++++ src/boot.rs | 13 +++++++---- 5 files changed, 34 insertions(+), 33 deletions(-) diff --git a/docs-site/content/docs/the-app/cache.md b/docs-site/content/docs/the-app/cache.md index e3384294b..bc53548ec 100644 --- a/docs-site/content/docs/the-app/cache.md +++ b/docs-site/content/docs/the-app/cache.md @@ -25,14 +25,15 @@ By default, `Loco` initializes a `Null` cache driver. This means any interaction To enable caching and configure a specific cache driver, you can replace the default `Null` driver with your preferred implementation. -In your `app.rs` file, define a function named `cache` function as a Hook in the `app.rs` file and import the `cache` module from `loco_rs`. +In your `app.rs` file, define a function named `override_context` function as a Hook in the `app.rs` file and import the `cache` module from `loco_rs`. Here's an example using an in-memory cache driver: ```rust use loco_rs::cache; -async fn cache(_config: &Config, _environment: &Environment) -> Result { - Ok(cache::Cache::new(cache::drivers::inmem::new())) +async fn override_context(mut ctx: AppContext) -> Result { + ctx.cache = cache::Cache::new(cache::drivers::inmem::new()).into(); + Ok(ctx) } ``` \ No newline at end of file diff --git a/docs-site/content/docs/the-app/storage.md b/docs-site/content/docs/the-app/storage.md index dba441855..0e3a7a825 100644 --- a/docs-site/content/docs/the-app/storage.md +++ b/docs-site/content/docs/the-app/storage.md @@ -29,15 +29,13 @@ By default loco initialize a `Null` provider, meaning any work with the storage ## Setup -Add the `storage` function as a Hook in the `app.rs` file and import the `storage` module from `loco_rs`. +Add the `override_context` function as a Hook in the `app.rs` file and import the `storage` module from `loco_rs`. ```rust use loco_rs::storage; -impl Hooks for App { - async fn storage(_config: &Config, environment: &Environment) -> Result> { - return Ok(None); - } +async fn override_context(mut ctx: AppContext) -> Result { + Ok(ctx) } ``` @@ -61,13 +59,11 @@ In this example, we initialize the in-memory driver and create a new storage wit ```rust use loco_rs::storage; -async fn storage( - _config: &Config, - environment: &Environment, - ) -> Result> { - let storage = Storage::single(storage::drivers::mem::new()); - return Ok(Some(storage)); - } + +async fn override_context(mut ctx: AppContext) -> Result { + ctx.storage = Storage::single(storage::drivers::mem::new()).into(); + Ok(ctx) +} ``` ### Multiple Drivers @@ -177,11 +173,7 @@ async fn upload_file( })?; let path = PathBuf::from("folder").join(file_name); - ctx.storage - .as_ref() - .unwrap() - .upload(path.as_path(), &content) - .await?; + ctx.storage.as_ref().upload(path.as_path(), &content).await?; file = Some(path); } @@ -211,7 +203,7 @@ async fn can_register() { let res: views::upload::Response = serde_json::from_str(&response.text()).unwrap(); - let stored_file: String = ctx.storage.unwrap().download(&res.path).await.unwrap(); + let stored_file: String = ctx.storage.as_ref().download(&res.path).await.unwrap(); assert_eq!(stored_file, file_content); }) diff --git a/examples/demo/src/app.rs b/examples/demo/src/app.rs index 1c4d03064..8e1dff22e 100644 --- a/examples/demo/src/app.rs +++ b/examples/demo/src/app.rs @@ -81,19 +81,16 @@ impl Hooks for App { create_app::(mode, environment).await } - async fn storage(_config: &Config, environment: &Environment) -> Result { - let store = if environment == &Environment::Test { + async fn override_context(mut ctx: AppContext) -> Result { + let store = if ctx.environment == Environment::Test { storage::drivers::mem::new() } else { storage::drivers::local::new_with_prefix("storage-uploads").map_err(Box::from)? }; + ctx.storage = Storage::single(store).into(); + ctx.cache = cache::Cache::new(cache::drivers::inmem::new()).into(); - let storage = Storage::single(store); - return Ok(storage); - } - - async fn cache(_config: &Config, _environment: &Environment) -> Result { - Ok(cache::Cache::new(cache::drivers::inmem::new())) + Ok(ctx) } fn connect_workers<'a>(p: &'a mut Processor, ctx: &'a AppContext) { diff --git a/src/app.rs b/src/app.rs index 9810ef3d3..12500068f 100644 --- a/src/app.rs +++ b/src/app.rs @@ -178,6 +178,12 @@ pub trait Hooks { Ok(cache::Cache::new(cache::drivers::null::new())) } + // Provides the options to override app context. + #[warn(unused_mut)] + async fn override_context(mut ctx: AppContext) -> Result { + Ok(ctx) + } + #[cfg(feature = "channels")] /// Register channels endpoints to the application routers fn register_channels(_ctx: &AppContext) -> AppChannels; diff --git a/src/boot.rs b/src/boot.rs index 533e43ed9..64f4474a6 100644 --- a/src/boot.rs +++ b/src/boot.rs @@ -13,12 +13,14 @@ use crate::db; use crate::{ app::{AppContext, Hooks}, banner::print_banner, + cache, config::{self, Config}, controller::ListRoutes, environment::Environment, errors::Error, mailer::{EmailSender, MailerWorker}, redis, + storage::{self, Storage}, task::Tasks, worker::{self, AppWorker, Pool, Processor, RedisConnectionManager, DEFAULT_QUEUES}, Result, @@ -199,16 +201,19 @@ pub async fn create_context(environment: &Environment) -> Result Date: Sun, 26 May 2024 09:37:13 +0300 Subject: [PATCH 10/23] fix clippy --- src/app.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/app.rs b/src/app.rs index 12500068f..1fff865ae 100644 --- a/src/app.rs +++ b/src/app.rs @@ -179,7 +179,7 @@ pub trait Hooks { } // Provides the options to override app context. - #[warn(unused_mut)] + #[allow(unused_mut)] async fn override_context(mut ctx: AppContext) -> Result { Ok(ctx) } From c1fb4480ac54b06b480693078d4cac53b793254c Mon Sep 17 00:00:00 2001 From: Elad Kaplan Date: Sun, 26 May 2024 09:56:16 +0300 Subject: [PATCH 11/23] change to after_context --- docs-site/content/docs/the-app/cache.md | 5 +++-- docs-site/content/docs/the-app/storage.md | 7 ++++--- examples/demo/src/app.rs | 4 ++-- src/app.rs | 5 ++--- src/boot.rs | 2 +- 5 files changed, 12 insertions(+), 11 deletions(-) diff --git a/docs-site/content/docs/the-app/cache.md b/docs-site/content/docs/the-app/cache.md index bc53548ec..204d0a9e1 100644 --- a/docs-site/content/docs/the-app/cache.md +++ b/docs-site/content/docs/the-app/cache.md @@ -25,14 +25,15 @@ By default, `Loco` initializes a `Null` cache driver. This means any interaction To enable caching and configure a specific cache driver, you can replace the default `Null` driver with your preferred implementation. -In your `app.rs` file, define a function named `override_context` function as a Hook in the `app.rs` file and import the `cache` module from `loco_rs`. +In your `app.rs` file, define a function named `after_context` function as a Hook in the `app.rs` file and import the `cache` module from `loco_rs`. Here's an example using an in-memory cache driver: ```rust use loco_rs::cache; -async fn override_context(mut ctx: AppContext) -> Result { +async fn after_context(ctx: AppContext) -> Result { + let mut ctx = ctx.clone() ctx.cache = cache::Cache::new(cache::drivers::inmem::new()).into(); Ok(ctx) } diff --git a/docs-site/content/docs/the-app/storage.md b/docs-site/content/docs/the-app/storage.md index 0e3a7a825..38b4dc90c 100644 --- a/docs-site/content/docs/the-app/storage.md +++ b/docs-site/content/docs/the-app/storage.md @@ -29,12 +29,12 @@ By default loco initialize a `Null` provider, meaning any work with the storage ## Setup -Add the `override_context` function as a Hook in the `app.rs` file and import the `storage` module from `loco_rs`. +Add the `after_context` function as a Hook in the `app.rs` file and import the `storage` module from `loco_rs`. ```rust use loco_rs::storage; -async fn override_context(mut ctx: AppContext) -> Result { +async fn after_context(ctx: AppContext) -> Result { Ok(ctx) } ``` @@ -60,7 +60,8 @@ In this example, we initialize the in-memory driver and create a new storage wit ```rust use loco_rs::storage; -async fn override_context(mut ctx: AppContext) -> Result { +async fn after_context(ctx: AppContext) -> Result { + let mut ctx = ctx.clone() ctx.storage = Storage::single(storage::drivers::mem::new()).into(); Ok(ctx) } diff --git a/examples/demo/src/app.rs b/examples/demo/src/app.rs index 8e1dff22e..297916ef0 100644 --- a/examples/demo/src/app.rs +++ b/examples/demo/src/app.rs @@ -6,7 +6,6 @@ use loco_rs::{ app::{AppContext, Hooks, Initializer}, boot::{create_app, BootResult, StartMode}, cache, - config::Config, controller::AppRoutes, db::{self, truncate_table}, environment::Environment, @@ -81,7 +80,8 @@ impl Hooks for App { create_app::(mode, environment).await } - async fn override_context(mut ctx: AppContext) -> Result { + async fn after_context(ctx: AppContext) -> Result { + let mut ctx = ctx.clone(); let store = if ctx.environment == Environment::Test { storage::drivers::mem::new() } else { diff --git a/src/app.rs b/src/app.rs index 1fff865ae..a3d468d3e 100644 --- a/src/app.rs +++ b/src/app.rs @@ -178,9 +178,8 @@ pub trait Hooks { Ok(cache::Cache::new(cache::drivers::null::new())) } - // Provides the options to override app context. - #[allow(unused_mut)] - async fn override_context(mut ctx: AppContext) -> Result { + // Provides the options to change Loco [`AppContext`] after initialization. + async fn after_context(ctx: AppContext) -> Result { Ok(ctx) } diff --git a/src/boot.rs b/src/boot.rs index 64f4474a6..c0d4f54a5 100644 --- a/src/boot.rs +++ b/src/boot.rs @@ -213,7 +213,7 @@ pub async fn create_context(environment: &Environment) -> Result Date: Sun, 26 May 2024 10:13:31 +0300 Subject: [PATCH 12/23] changes --- docs-site/content/docs/the-app/cache.md | 7 ++++--- docs-site/content/docs/the-app/storage.md | 7 ++++--- examples/demo/src/app.rs | 11 +++++++---- src/app.rs | 20 -------------------- 4 files changed, 15 insertions(+), 30 deletions(-) diff --git a/docs-site/content/docs/the-app/cache.md b/docs-site/content/docs/the-app/cache.md index 204d0a9e1..ec590f161 100644 --- a/docs-site/content/docs/the-app/cache.md +++ b/docs-site/content/docs/the-app/cache.md @@ -33,8 +33,9 @@ Here's an example using an in-memory cache driver: use loco_rs::cache; async fn after_context(ctx: AppContext) -> Result { - let mut ctx = ctx.clone() - ctx.cache = cache::Cache::new(cache::drivers::inmem::new()).into(); - Ok(ctx) + Ok(AppContext { + cache: cache::Cache::new(cache::drivers::inmem::new()).into(), + ..ctx + }) } ``` \ No newline at end of file diff --git a/docs-site/content/docs/the-app/storage.md b/docs-site/content/docs/the-app/storage.md index 38b4dc90c..665fcb247 100644 --- a/docs-site/content/docs/the-app/storage.md +++ b/docs-site/content/docs/the-app/storage.md @@ -61,9 +61,10 @@ In this example, we initialize the in-memory driver and create a new storage wit use loco_rs::storage; async fn after_context(ctx: AppContext) -> Result { - let mut ctx = ctx.clone() - ctx.storage = Storage::single(storage::drivers::mem::new()).into(); - Ok(ctx) + Ok(AppContext { + storage: Storage::single(storage::drivers::mem::new()).into(), + ..ctx + }) } ``` diff --git a/examples/demo/src/app.rs b/examples/demo/src/app.rs index 297916ef0..97eae6292 100644 --- a/examples/demo/src/app.rs +++ b/examples/demo/src/app.rs @@ -81,16 +81,19 @@ impl Hooks for App { } async fn after_context(ctx: AppContext) -> Result { - let mut ctx = ctx.clone(); let store = if ctx.environment == Environment::Test { storage::drivers::mem::new() } else { storage::drivers::local::new_with_prefix("storage-uploads").map_err(Box::from)? }; - ctx.storage = Storage::single(store).into(); - ctx.cache = cache::Cache::new(cache::drivers::inmem::new()).into(); - Ok(ctx) + Ok(AppContext { + storage: Storage::single(store).into(), + cache: cache::Cache::new(cache::drivers::inmem::new()).into(), + ..ctx + }) + + // Ok(ctx) } fn connect_workers<'a>(p: &'a mut Processor, ctx: &'a AppContext) { diff --git a/src/app.rs b/src/app.rs index a3d468d3e..88c73b6e0 100644 --- a/src/app.rs +++ b/src/app.rs @@ -158,26 +158,6 @@ pub trait Hooks { /// Defines the application's routing configuration. fn routes(_ctx: &AppContext) -> AppRoutes; - // Provides a default null storage implementation for the application. - /// - /// This function returns a [`Storage`] instance that always returns errors. - /// It simplifies user workflow by avoiding the need for feature flags or - /// configuring a specific storage driver during development or testing. - async fn storage(_config: &config::Config, _environment: &Environment) -> Result { - let storage = Storage::single(storage::drivers::null::new()); - Ok(storage) - } - - /// Provides a default null cache implementation for the application. - /// - /// This function returns a [`Cache`] instance that always returns - /// errors. It simplifies user workflow by avoiding the need for feature - /// flags or optional cache driver configurations during development or - /// testing. - async fn cache(_config: &config::Config, _environment: &Environment) -> Result { - Ok(cache::Cache::new(cache::drivers::null::new())) - } - // Provides the options to change Loco [`AppContext`] after initialization. async fn after_context(ctx: AppContext) -> Result { Ok(ctx) From dc4936ec6715ed5c2c25ab91e4a7f50ccf9e67f0 Mon Sep 17 00:00:00 2001 From: Elad Kaplan Date: Sun, 26 May 2024 10:16:38 +0300 Subject: [PATCH 13/23] clippy --- src/app.rs | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/app.rs b/src/app.rs index 88c73b6e0..acb0d35c2 100644 --- a/src/app.rs +++ b/src/app.rs @@ -16,12 +16,12 @@ use axum::Router as AxumRouter; use crate::controller::channels::AppChannels; use crate::{ boot::{BootResult, ServeParams, StartMode}, - cache::{self, Cache}, + cache::{self}, config::{self, Config}, controller::AppRoutes, environment::Environment, mailer::EmailSender, - storage::{self, Storage}, + storage::Storage, task::Tasks, worker::{Pool, Processor, RedisConnectionManager}, Result, From 2cbd9c8c6d8680ff80bdfbadb5e4c086f4b70e67 Mon Sep 17 00:00:00 2001 From: Elad Kaplan Date: Sun, 26 May 2024 12:35:33 +0300 Subject: [PATCH 14/23] improve docs with snipdoc --- .github/workflows/docs.yml | 33 ++++ README.md | 152 +++++++++--------- .../docs/getting-started/tour/index.md | 139 ++++++++-------- src/lib.rs | 119 ++++++++++++-- 4 files changed, 289 insertions(+), 154 deletions(-) create mode 100644 .github/workflows/docs.yml diff --git a/.github/workflows/docs.yml b/.github/workflows/docs.yml new file mode 100644 index 000000000..b2eac0eb5 --- /dev/null +++ b/.github/workflows/docs.yml @@ -0,0 +1,33 @@ +name: docs + +on: + push: + branches: + - master + pull_request: + +env: + RUST_TOOLCHAIN: stable + TOOLCHAIN_PROFILE: minimal + +jobs: + check: + name: Check + runs-on: ubuntu-latest + + permissions: + contents: read + + steps: + - name: Checkout the code + uses: actions/checkout@v4 + - uses: actions-rs/toolchain@v1 + with: + profile: ${{ env.TOOLCHAIN_PROFILE }} + toolchain: ${{ env.RUST_TOOLCHAIN }} + override: true + components: rustfmt + - run: cargo install snipdoc + - run: snipdoc check + + \ No newline at end of file diff --git a/README.md b/README.md index ae98e8ee3..fb0abe151 100644 --- a/README.md +++ b/README.md @@ -1,103 +1,111 @@ + +
-![Loco.rs](https://github.com/loco-rs/loco/assets/83390/992d215a-3cd3-42ee-a1c7-de9fd25a5bac) -[![Current Crates.io Version](https://img.shields.io/crates/v/loco-rs.svg)](https://crates.io/crates/loco-rs) -[![Discord channel](https://img.shields.io/badge/discord-Join-us)](https://discord.gg/fTvyBzwKS8) + -# Welcome to Loco! +

Loco

-https://loco.rs +

๐Ÿš‚ Loco is "Rust on Rails".

+ [![crate](https://img.shields.io/crates/v/loco-rs.svg)](https://crates.io/crates/loco-rs) + [![docs](https://docs.rs/loco-rs/badge.svg)](https://docs.rs/loco-rs) + [![Discord channel](https://img.shields.io/badge/discord-Join-us)](https://discord.gg/fTvyBzwKS8) -Loco is "Rust on Rails". +
-Loco is strongly inspired by Rails. If you know Rails and Rust, you'll feel at home. If you only know Rails and new to Rust, you'll find Loco refreshing. We do not assume you know Rails. + # Loco + #### Loco is strongly inspired by Rails. If you know Rails and Rust, you'll feel at home. If you only know Rails and new to Rust, you'll find Loco refreshing. We do not assume you know Rails. -## Quick Start + ## Quick Start + ```sh + $ cargo install loco-cli + $ cargo install sea-orm-cli # Only when DB is needed + ``` -```sh -$ cargo install loco-cli -``` + Now you can create your new app (choose "SaaS app"). -Now you can create your new app (choose "SaaS app"). + ```sh + $ loco new + โœ” โฏ App name? ยท myapp + ? โฏ What would you like to build? โ€บ + lightweight-service (minimal, only controllers and views) + Rest API (with DB and user auth) + โฏ SaaS app (with DB and user auth) + ๐Ÿš‚ Loco app generated successfully in: + myapp + ``` -```sh -$ loco new -โฏ App name? [myapp]: -โฏ SaaS app (with DB and user auth) - Stateless service (minimal, no db) -๐Ÿš‚ Loco app generated successfully in: -myapp -``` +
+ To configure a database , please run a local postgres database with + loco:loco and a db named is the [insert app]_development. +
-To configure a database , please run a local postgres database with loco:loco and a db named [insert app]_development. + You can use Docker to run a Postgres instance: -``` -$ docker run -d -p 5432:5432 -e POSTGRES_USER=loco -e POSTGRES_DB=myapp_development -e POSTGRES_PASSWORD="loco" postgres:15.3-alpine -``` + When generating a starter, the database name incorporates your application + name and the environment. For instance, if you include `myapp`, the database + name in the `test.yaml`configuration will be `myapp_test`, and in the + `development.yaml` configuration, it will be `myapp_development`. -Now `cd` into your `myapp` and start your app: -``` -$ cd myapp -$ cargo loco start -Finished dev [unoptimized + debuginfo] target(s) in 21.63s - Running `target/debug/myapp start` + A more advanced set of `docker-compose.yml` and `Dockerfiles` that include Redis and the `mailtutan` mailer are available for [each starter on GitHub](https://github.com/loco-rs/loco/blob/master/starters/saas/.devcontainer/docker-compose.yml). - : - : - : + Now `cd` into your `myapp` and start your app: -controller/app_routes.rs:203: [Middleware] Adding log trace id + ``` + $ cd myapp + $ cargo loco start + Finished dev [unoptimized + debuginfo] target(s) in 21.63s + Running `target/debug/myapp start` - โ–„ โ–€ - โ–€ โ–„ - โ–„ โ–€ โ–„ โ–„ โ–„โ–€ - โ–„ โ–€โ–„โ–„ - โ–„ โ–€ โ–€ โ–€โ–„โ–€โ–ˆโ–„ - โ–€โ–ˆโ–„ -โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–€โ–€โ–ˆ - โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–€โ–ˆ - โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–€โ–€โ–€ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–„โ–ˆโ–„ - โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–„ - โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–„โ–„โ–„ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ - โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–€ - โ–€โ–€โ–€โ–ˆโ–ˆโ–„ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–ˆโ–ˆโ–€ - โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ + : + : + : -started on port 3000 -``` + controller/app_routes.rs:203: [Middleware] Adding log trace id -## Project Status + โ–„ โ–€ + โ–€ โ–„ + โ–„ โ–€ โ–„ โ–„ โ–„โ–€ + โ–„ โ–€โ–„โ–„ + โ–„ โ–€ โ–€ โ–€โ–„โ–€โ–ˆโ–„ + โ–€โ–ˆโ–„ + โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–€โ–€โ–ˆ + โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–€โ–ˆ + โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–€โ–€โ–€ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–„โ–ˆโ–„ + โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–„ + โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–„โ–„โ–„ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ + โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–€ + โ–€โ–€โ–€โ–ˆโ–ˆโ–„ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–ˆโ–ˆโ–€ + โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ -Loco is feature complete, but features are still being added rapidly. + started on port 3000 + ``` -### What can you build? +
+ You don't have to run things through `cargo` but in development it's highly + recommended. If you build `--release`, your binary contains everything + including your code and `cargo` or Rust is not needed.
-- Stateless APIs -- Complete SaaS products with user authentication -- Purpose-built services such as ML inference endpoints -- Full stack projects with separate frontend project integrated with Loco -- Hobby projects full-stack with backend and HTML frontend + ## Project Status + + Stateless APIs + + Complete SaaS products with user authentication + + Purpose-built services such as ML inference endpoints + + Full stack projects with separate frontend project integrated with Loco + + Hobby projects full-stack with backend and HTML frontend -### What's being done now? + ## Powered by Loco + + [SpectralOps](https://spectralops.io) - various services powered by Loco + framework + + [Nativish](https://nativi.sh) - app backend powered by Loco framework -- View [issues](https://github.com/loco-rs/loco/issues) for what we plan next and what we work on (you're welcome to submit PRs!) -- View [CHANGELOG](https://github.com/loco-rs/loco/blob/master/CHANGELOG.md) for what we already added + ## Contributors โœจ -## Powered by Loco - -* [SpectralOps](https://spectralops.io) - various services powered by Loco framework -* [Nativish](https://nativi.sh) - app backend powered by Loco framework - -[open an issue to add yourself here](https://github.com/loco-rs/loco/issues) - - -## Contributors โœจ - -Thanks goes to these wonderful people: + Thanks goes to these wonderful people: + \ No newline at end of file diff --git a/docs-site/content/docs/getting-started/tour/index.md b/docs-site/content/docs/getting-started/tour/index.md index 30de93b77..af9c7a2ba 100644 --- a/docs-site/content/docs/getting-started/tour/index.md +++ b/docs-site/content/docs/getting-started/tour/index.md @@ -19,73 +19,78 @@ flair =[]
Let's create a blog backend on `loco` in 4 commands. First install `loco-cli` and `sea-orm-cli`: -```sh -$ cargo install loco-cli -$ cargo install sea-orm-cli -``` - -Now you can create your new app (choose "SaaS app"). - -```sh -$ loco new -โœ” โฏ App name? ยท myapp -? โฏ What would you like to build? โ€บ - lightweight-service (minimal, only controllers and views) - Rest API (with DB and user auth) -โฏ SaaS app (with DB and user auth) -๐Ÿš‚ Loco app generated successfully in: -myapp -``` - -
-To configure a database , please run a local postgres database with loco:loco and a db named is the [insert app]_development. -
- -You can use Docker to run a Postgres instance: - -When generating a starter, the database name incorporates your application name and the environment. For instance, if you include `myapp`, the database name in the `test.yaml`configuration will be `myapp_test`, and in the `development.yaml` configuration, it will be `myapp_development`. - -``` -$ docker run -d -p 5432:5432 -e POSTGRES_USER=loco -e POSTGRES_DB=myapp_development -e POSTGRES_PASSWORD="loco" postgres:15.3-alpine -``` - -A more advanced set of `docker-compose.yml` and `Dockerfiles` that include Redis and the `mailtutan` mailer are available for [each starter on GitHub](https://github.com/loco-rs/loco/blob/master/starters/saas/.devcontainer/docker-compose.yml). - -Now `cd` into your `myapp` and start your app: - -``` -$ cd myapp -$ cargo loco start -Finished dev [unoptimized + debuginfo] target(s) in 21.63s - Running `target/debug/myapp start` - - : - : - : - -controller/app_routes.rs:203: [Middleware] Adding log trace id - - โ–„ โ–€ - โ–€ โ–„ - โ–„ โ–€ โ–„ โ–„ โ–„โ–€ - โ–„ โ–€โ–„โ–„ - โ–„ โ–€ โ–€ โ–€โ–„โ–€โ–ˆโ–„ - โ–€โ–ˆโ–„ -โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–€โ–€โ–ˆ - โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–€โ–ˆ - โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–€โ–€โ–€ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–„โ–ˆโ–„ - โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–„ - โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–„โ–„โ–„ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ - โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–€ - โ–€โ–€โ–€โ–ˆโ–ˆโ–„ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–ˆโ–ˆโ–€ - โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ - -started on port 3000 -``` - -
-You don't have to run things through `cargo` but in development it's highly recommended. If you build `--release`, your binary contains everything including your code and `cargo` or Rust is not needed. -
+ + ```sh + $ cargo install loco-cli + $ cargo install sea-orm-cli # Only when DB is needed + ``` + + Now you can create your new app (choose "SaaS app"). + + ```sh + $ loco new + โœ” โฏ App name? ยท myapp + ? โฏ What would you like to build? โ€บ + lightweight-service (minimal, only controllers and views) + Rest API (with DB and user auth) + โฏ SaaS app (with DB and user auth) + ๐Ÿš‚ Loco app generated successfully in: + myapp + ``` + +
+ To configure a database , please run a local postgres database with + loco:loco and a db named is the [insert app]_development. +
+ + You can use Docker to run a Postgres instance: + + When generating a starter, the database name incorporates your application + name and the environment. For instance, if you include `myapp`, the database + name in the `test.yaml`configuration will be `myapp_test`, and in the + `development.yaml` configuration, it will be `myapp_development`. + + + + A more advanced set of `docker-compose.yml` and `Dockerfiles` that include Redis and the `mailtutan` mailer are available for [each starter on GitHub](https://github.com/loco-rs/loco/blob/master/starters/saas/.devcontainer/docker-compose.yml). + + Now `cd` into your `myapp` and start your app: + + ``` + $ cd myapp + $ cargo loco start + Finished dev [unoptimized + debuginfo] target(s) in 21.63s + Running `target/debug/myapp start` + + : + : + : + + controller/app_routes.rs:203: [Middleware] Adding log trace id + + โ–„ โ–€ + โ–€ โ–„ + โ–„ โ–€ โ–„ โ–„ โ–„โ–€ + โ–„ โ–€โ–„โ–„ + โ–„ โ–€ โ–€ โ–€โ–„โ–€โ–ˆโ–„ + โ–€โ–ˆโ–„ + โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–€โ–€โ–ˆ + โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–€โ–ˆ + โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–€โ–€โ–€ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–„โ–ˆโ–„ + โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–„ + โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–„โ–„โ–„ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ + โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–€ + โ–€โ–€โ–€โ–ˆโ–ˆโ–„ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–ˆโ–ˆโ–€ + โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ + + started on port 3000 + ``` + +
+ You don't have to run things through `cargo` but in development it's highly + recommended. If you build `--release`, your binary contains everything + including your code and `cargo` or Rust is not needed.
+ ## Adding a CRUD API diff --git a/src/lib.rs b/src/lib.rs index e127af7d1..0ccdd815e 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -1,30 +1,119 @@ #![allow(clippy::missing_const_for_fn)] #![allow(clippy::module_name_repetitions)] -//! ## Starting A New Project +// +//!
//! -//! To start a new project, you can use cargo-generate: +//! +//! +//!

Loco

+//! +//!

๐Ÿš‚ Loco is "Rust on Rails".

+//! +//! [![crate](https://img.shields.io/crates/v/loco-rs.svg)](https://crates.io/crates/loco-rs) +//! [![docs](https://docs.rs/loco-rs/badge.svg)](https://docs.rs/loco-rs) +//! [![Discord channel](https://img.shields.io/badge/discord-Join-us)](https://discord.gg/fTvyBzwKS8) +//! +//!
+//! +//! # Loco +//! +//! #### Loco is strongly inspired by Rails. If you know Rails and Rust, you'll feel at home. If you only know Rails and new to Rust, you'll find Loco refreshing. We do not assume you know Rails. +//! +//! ## Quick Start +// +//! ```sh +//! $ cargo install loco-cli +//! $ cargo install sea-orm-cli # Only when DB is needed +//! ``` +//! +//! Now you can create your new app (choose "SaaS app"). //! //! ```sh -//! cargo install loco-cli -//! โฏ loco new +//! $ loco new //! โœ” โฏ App name? ยท myapp //! ? โฏ What would you like to build? โ€บ -//! โฏ lightweight-service (minimal, only controllers and views) +//! lightweight-service (minimal, only controllers and views) //! Rest API (with DB and user auth) -//! SaaS app (with DB and user auth) +//! โฏ SaaS app (with DB and user auth) +//! ๐Ÿš‚ Loco app generated successfully in: +//! myapp +//! ``` +//! +//!
+//! To configure a database , please run a local postgres database with +//! loco:loco and a db named is the [insert app]_development. +//!
+//! +//! You can use Docker to run a Postgres instance: +//! +//! When generating a starter, the database name incorporates your application +//! name and the environment. For instance, if you include `myapp`, the database +//! name in the `test.yaml`configuration will be `myapp_test`, and in the +//! `development.yaml` configuration, it will be `myapp_development`. +//! +//! +//! +//! A more advanced set of `docker-compose.yml` and `Dockerfiles` that include Redis and the `mailtutan` mailer are available for [each starter on GitHub](https://github.com/loco-rs/loco/blob/master/starters/saas/.devcontainer/docker-compose.yml). +//! +//! Now `cd` into your `myapp` and start your app: +//! +//! ``` +//! $ cd myapp +//! $ cargo loco start +//! Finished dev [unoptimized + debuginfo] target(s) in 21.63s +//! Running `target/debug/myapp start` +//! +//! : +//! : +//! : +//! +//! controller/app_routes.rs:203: [Middleware] Adding log trace id +//! +//! โ–„ โ–€ +//! โ–€ โ–„ +//! โ–„ โ–€ โ–„ โ–„ โ–„โ–€ +//! โ–„ โ–€โ–„โ–„ +//! โ–„ โ–€ โ–€ โ–€โ–„โ–€โ–ˆโ–„ +//! โ–€โ–ˆโ–„ +//! โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–€โ–€โ–ˆ +//! โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–€โ–ˆ +//! โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–€โ–€โ–€ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–„โ–ˆโ–„ +//! โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–„ +//! โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–„โ–„โ–„ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ +//! โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–€ +//! โ–€โ–€โ–€โ–ˆโ–ˆโ–„ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–ˆโ–ˆโ–€ +//! โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ +//! +//! started on port 3000 //! ``` //! -//! ## Available Features +//!
+//! You don't have to run things through `cargo` but in development it's highly +//! recommended. If you build `--release`, your binary contains everything +//! including your code and `cargo` or Rust is not needed.
+//
+//! +//! ## Project Status +//! + Stateless APIs +//! + Complete SaaS products with user authentication +//! + Purpose-built services such as ML inference endpoints +//! + Full stack projects with separate frontend project integrated with Loco +//! + Hobby projects full-stack with backend and HTML frontend //! -//! To avoid compiling unused dependencies, loco gates certain features. +//! ## Powered by Loco +//! + [SpectralOps](https://spectralops.io) - various services powered by Loco +//! framework +//! + [Nativish](https://nativi.sh) - app backend powered by Loco framework //! -//! | Feature | Default | Description | -//! |------------|---------|-----------------------------| -//! | `auth_jwt` | true | Enable user authentication. | -//! | `cli` | true | Expose Cli commands. | -//! | `testing | false | Test Utilities Module. | -//! | `with-db` | true | with-db. | -//! | `channels` | false | Enable socket channels. | +//! ## Contributors โœจ +//! +//! Thanks goes to these wonderful people: +//! +//! +//! +//! +//
+ pub use self::errors::Error; mod banner; From 3cf52a8a8ff90f64bd596c96cb6ef732092aa778 Mon Sep 17 00:00:00 2001 From: Elad Kaplan Date: Sun, 26 May 2024 12:42:00 +0300 Subject: [PATCH 15/23] validate ci --- src/lib.rs | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/lib.rs b/src/lib.rs index 0ccdd815e..a62787694 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -26,7 +26,7 @@ //! $ cargo install sea-orm-cli # Only when DB is needed //! ``` //! -//! Now you can create your new app (choose "SaaS app"). +//! Now you can create your new app (choose "`SaaS` app"). //! //! ```sh //! $ loco new @@ -95,7 +95,7 @@ //! //! ## Project Status //! + Stateless APIs -//! + Complete SaaS products with user authentication +//! + Complete `SaaS` products with user authentication //! + Purpose-built services such as ML inference endpoints //! + Full stack projects with separate frontend project integrated with Loco //! + Hobby projects full-stack with backend and HTML frontend From a1140ebaa1bc58fbeac5f130f1a667762019767d Mon Sep 17 00:00:00 2001 From: Elad Kaplan Date: Sun, 26 May 2024 12:43:59 +0300 Subject: [PATCH 16/23] alignment --- README.md | 4 ++-- docs-site/content/docs/getting-started/tour/index.md | 2 +- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/README.md b/README.md index fb0abe151..17d1d5e08 100644 --- a/README.md +++ b/README.md @@ -23,7 +23,7 @@ $ cargo install sea-orm-cli # Only when DB is needed ``` - Now you can create your new app (choose "SaaS app"). + Now you can create your new app (choose "`SaaS` app"). ```sh $ loco new @@ -91,7 +91,7 @@ ## Project Status + Stateless APIs - + Complete SaaS products with user authentication + + Complete `SaaS` products with user authentication + Purpose-built services such as ML inference endpoints + Full stack projects with separate frontend project integrated with Loco + Hobby projects full-stack with backend and HTML frontend diff --git a/docs-site/content/docs/getting-started/tour/index.md b/docs-site/content/docs/getting-started/tour/index.md index af9c7a2ba..2330ec501 100644 --- a/docs-site/content/docs/getting-started/tour/index.md +++ b/docs-site/content/docs/getting-started/tour/index.md @@ -25,7 +25,7 @@ Let's create a blog backend on `loco` in 4 commands. First install `loco-cli` an $ cargo install sea-orm-cli # Only when DB is needed ``` - Now you can create your new app (choose "SaaS app"). + Now you can create your new app (choose "`SaaS` app"). ```sh $ loco new From 0bf15914ea0b5f96d97fabcc02f3d51499b71628 Mon Sep 17 00:00:00 2001 From: Elad Kaplan Date: Mon, 27 May 2024 09:14:23 +0300 Subject: [PATCH 17/23] snipdoc --- README.md | 147 ++++++++---------- docs-site/content/blog/frontend-website.md | 1 - docs-site/content/docs/getting-started/cli.md | 26 ++-- .../content/docs/getting-started/config.md | 26 ++-- .../docs/getting-started/deployment.md | 113 ++++++++++---- .../content/docs/getting-started/guide.md | 71 +++++---- .../content/docs/getting-started/starters.md | 15 +- .../docs/getting-started/tour/index.md | 106 ++++++++----- docs-site/templates/shortcodes/get_env.html | 0 examples/demo/config/development.yaml | 15 ++ snipdoc.yml | 124 +++++++++++++++ src/lib.rs | 114 +------------- 12 files changed, 436 insertions(+), 322 deletions(-) create mode 100644 docs-site/templates/shortcodes/get_env.html create mode 100644 snipdoc.yml diff --git a/README.md b/README.md index 17d1d5e08..e2fb037ea 100644 --- a/README.md +++ b/README.md @@ -1,11 +1,14 @@ -

Loco

-

๐Ÿš‚ Loco is "Rust on Rails".

+

+ +๐Ÿš‚ Loco is Rust on Rails. + +

[![crate](https://img.shields.io/crates/v/loco-rs.svg)](https://crates.io/crates/loco-rs) [![docs](https://docs.rs/loco-rs/badge.svg)](https://docs.rs/loco-rs) @@ -18,94 +21,82 @@ #### Loco is strongly inspired by Rails. If you know Rails and Rust, you'll feel at home. If you only know Rails and new to Rust, you'll find Loco refreshing. We do not assume you know Rails. ## Quick Start - ```sh - $ cargo install loco-cli - $ cargo install sea-orm-cli # Only when DB is needed - ``` + +```sh +cargo install loco-cli +cargo install sea-orm-cli # Only when DB is needed +``` + Now you can create your new app (choose "`SaaS` app"). - ```sh - $ loco new - โœ” โฏ App name? ยท myapp - ? โฏ What would you like to build? โ€บ - lightweight-service (minimal, only controllers and views) - Rest API (with DB and user auth) - โฏ SaaS app (with DB and user auth) - ๐Ÿš‚ Loco app generated successfully in: - myapp - ``` - -
- To configure a database , please run a local postgres database with - loco:loco and a db named is the [insert app]_development. -
+ +```sh +โฏ loco new +โœ” โฏ App name? ยท myapp +โœ” โฏ What would you like to build? ยท SaaS app (with DB and user auth) - You can use Docker to run a Postgres instance: +๐Ÿš‚ Loco app generated successfully in: +myapp +``` + - When generating a starter, the database name incorporates your application - name and the environment. For instance, if you include `myapp`, the database - name in the `test.yaml`configuration will be `myapp_test`, and in the - `development.yaml` configuration, it will be `myapp_development`. +To configure a database , please run a local postgres database with loco:loco and a db named [insert app]_development. + +```sh +docker run -d -p 5432:5432 \ + -e POSTGRES_USER=loco \ + -e POSTGRES_DB=myapp_development \ + -e POSTGRES_PASSWORD="loco" \ + postgres:15.3-alpine +``` + A more advanced set of `docker-compose.yml` and `Dockerfiles` that include Redis and the `mailtutan` mailer are available for [each starter on GitHub](https://github.com/loco-rs/loco/blob/master/starters/saas/.devcontainer/docker-compose.yml). Now `cd` into your `myapp` and start your app: - ``` - $ cd myapp - $ cargo loco start - Finished dev [unoptimized + debuginfo] target(s) in 21.63s - Running `target/debug/myapp start` - - : - : - : - - controller/app_routes.rs:203: [Middleware] Adding log trace id - - โ–„ โ–€ - โ–€ โ–„ - โ–„ โ–€ โ–„ โ–„ โ–„โ–€ - โ–„ โ–€โ–„โ–„ - โ–„ โ–€ โ–€ โ–€โ–„โ–€โ–ˆโ–„ - โ–€โ–ˆโ–„ - โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–€โ–€โ–ˆ - โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–€โ–ˆ - โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–€โ–€โ–€ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–„โ–ˆโ–„ - โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–„ - โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–„โ–„โ–„ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ - โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–€ - โ–€โ–€โ–€โ–ˆโ–ˆโ–„ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–ˆโ–ˆโ–€ - โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ - - started on port 3000 - ``` - -
- You don't have to run things through `cargo` but in development it's highly - recommended. If you build `--release`, your binary contains everything - including your code and `cargo` or Rust is not needed.
- - ## Project Status - + Stateless APIs - + Complete `SaaS` products with user authentication - + Purpose-built services such as ML inference endpoints - + Full stack projects with separate frontend project integrated with Loco - + Hobby projects full-stack with backend and HTML frontend - - ## Powered by Loco - + [SpectralOps](https://spectralops.io) - various services powered by Loco - framework - + [Nativish](https://nativi.sh) - app backend powered by Loco framework - - ## Contributors โœจ - - Thanks goes to these wonderful people: + +```sh +$ cargo loco start + + โ–„ โ–€ + โ–€ โ–„ + โ–„ โ–€ โ–„ โ–„ โ–„โ–€ + โ–„ โ–€โ–„โ–„ + โ–„ โ–€ โ–€ โ–€โ–„โ–€โ–ˆโ–„ + โ–€โ–ˆโ–„ +โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–€โ–€โ–ˆ +โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–€โ–ˆ +โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–€โ–€โ–€ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–„โ–ˆโ–„ +โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–„ +โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–„โ–„โ–„ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ +โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–€ + โ–€โ–€โ–€โ–ˆโ–ˆโ–„ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–ˆโ–ˆโ–€ + โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ + https://loco.rs + +listening on port 3000 +``` + + +## Project Status ++ Stateless APIs ++ Complete `SaaS` products with user authentication ++ Purpose-built services such as ML inference endpoints ++ Full stack projects with separate frontend project integrated with Loco ++ Hobby projects full-stack with backend and HTML frontend + +## Powered by Loco ++ [SpectralOps](https://spectralops.io) - various services powered by Loco + framework ++ [Nativish](https://nativi.sh) - app backend powered by Loco framework + +## Contributors โœจ +Thanks goes to these wonderful people: - - \ No newline at end of file + \ No newline at end of file diff --git a/docs-site/content/blog/frontend-website.md b/docs-site/content/blog/frontend-website.md index bba39181e..b3b70fff5 100644 --- a/docs-site/content/blog/frontend-website.md +++ b/docs-site/content/blog/frontend-website.md @@ -147,7 +147,6 @@ server: ``` Now, run the Loco server again and you should see frontend app serving via Loco - ```sh $ cargo loco start ``` diff --git a/docs-site/content/docs/getting-started/cli.md b/docs-site/content/docs/getting-started/cli.md index 480edb1f0..4d7c4a7bf 100644 --- a/docs-site/content/docs/getting-started/cli.md +++ b/docs-site/content/docs/getting-started/cli.md @@ -16,18 +16,24 @@ flair =[] Create your starter app: -```rust -$ cargo install loco-cli -$ loco new -< follow the guide > + +```sh +โฏ loco new +โœ” โฏ App name? ยท myapp +โœ” โฏ What would you like to build? ยท SaaS app (with DB and user auth) + +๐Ÿš‚ Loco app generated successfully in: +myapp ``` + Now `cd` into your app, set up a convenience `rr` alias and try out the various commands: + +```sh +cargo loco --help ``` -$ cd myapp -$ cargo loco --help -``` + You can now drive your development through the CLI: @@ -49,9 +55,11 @@ $ cargo test To run you app, run: + +```sh +cargo loco start ``` -$ cargo loco start -``` + ## Background workers diff --git a/docs-site/content/docs/getting-started/config.md b/docs-site/content/docs/getting-started/config.md index 80d6a381d..4c460cb44 100644 --- a/docs-site/content/docs/getting-started/config.md +++ b/docs-site/content/docs/getting-started/config.md @@ -61,27 +61,23 @@ config/ ``` To run the application using the 'qa' environment, execute the following command: - -``` -$ LOCO_ENV=qa cargo loco start + +```sh +LOCO_ENV=qa cargo loco start ``` + ## Settings -The configuration files contain knobs to set up your Loco app. You can also have your custom settings, with the `settings:` section. - - -```yaml -# in config/development.yaml -# add the `settings:` section -settings: +The configuration files contain knobs to set up your Loco app. You can also have your custom settings, with the `settings:` section. in `config/development.yaml` add the `settings:` section + +```yaml + settings: allow_list: - google.com - - apple.com - -logger: - # ... -``` + - apple.com + ``` + These setting will appear in `ctx.config.settings` as `serde_json::Value`. You can create your strongly typed settings by adding a struct: diff --git a/docs-site/content/docs/getting-started/deployment.md b/docs-site/content/docs/getting-started/deployment.md index 9b107c936..77e6aa959 100644 --- a/docs-site/content/docs/getting-started/deployment.md +++ b/docs-site/content/docs/getting-started/deployment.md @@ -17,9 +17,11 @@ Deployment is super simple in Loco, and this is why this guide is super short. A To deploy, build your production binary for your relevant server architecture: + +```sh +cargo build --release ``` -$ cargo build --release -``` + And copy your binary along with your `config/` folder to the server. You can then run `myapp start` on your server. @@ -33,63 +35,114 @@ There are a few configuration sections that are important to review and set acco - Logger: -```yaml + +```yaml + # Application logging configuration logger: - level: -``` + # Enable or disable logging. + enable: true + # Enable pretty backtrace (sets RUST_BACKTRACE=1) + pretty_backtrace: true + # Log level, options: trace, debug, info, warn or error. + level: debug + # Define the logging format. options: compact, pretty or Json + format: compact + # By default the logger has filtering only logs that came from your code or logs that came from `loco` framework. to see all third party libraries + # Uncomment the line below to override to see all third party libraries you can enable this config and override the logger filters. + # override_filter: trace + ``` + + - Server: - -```yaml -server: + +```yaml + server: # Port on which the server will listen. the server binding is 0.0.0.0:{PORT} - port: 3000 + port: {{get_env(name="NODE_PORT", default=3000)}} # The UI hostname or IP address that mailers will point to. host: http://localhost -``` + # Out of the box middleware configuration. to disable middleware you can changed the `enable` field to `false` of comment the middleware block + ``` + -- Database: -```yaml -database: +- Database: + +```yaml + database: # Database connection URI - uri: postgres://loco:loco@localhost:5432/loco_app -``` + uri: {{get_env(name="DATABASE_URL", default="postgres://loco:loco@localhost:5432/loco_app")}} + # When enabled, the sql query will be logged. + enable_logging: false + # Set the timeout duration when acquiring a connection. + connect_timeout: 500 + # Set the idle duration before closing a connection. + idle_timeout: 500 + # Minimum number of connections for a pool. + min_connections: 1 + # Maximum number of connections for a pool. + max_connections: 1 + # Run migration up when application loaded + auto_migrate: true + # Truncate database when application loaded. This is a dangerous operation, make sure that you using this flag only on dev environments or test mode + dangerously_truncate: false + # Recreating schema when application loaded. This is a dangerous operation, make sure that you using this flag only on dev environments or test mode + dangerously_recreate: false + ``` + -- Mailer: -```yaml -mailer: +- Mailer: + +```yaml + mailer: # SMTP mailer configuration. smtp: # Enable/Disable smtp mailer. enable: true # SMTP server host. e.x localhost, smtp.gmail.com host: localhost -``` + # SMTP server port + port: 1025 + # Use secure connection (SSL/TLS). + secure: false + # auth: + # user: + # password: + ``` + - Redis: - -``` -redis: + +```yaml + redis: # Redis connection URI - uri: redis://127.0.0.1/ -``` + uri: {{get_env(name="REDIS_URL", default="redis://127.0.0.1")}} + # Dangerously flush all data in Redis on startup. dangerous operation, make sure that you using this flag only on dev environments or test mode + dangerously_flush: false + ``` + - JWT secret: - -```yaml -auth: + +```yaml + auth: # JWT authentication jwt: # Secret key for token generation and verification - secret: ... -``` + secret: PqRwLF2rhHe8J22oBeHy + # Token expiration time in seconds + expiration: 604800 # 7 days + ``` + + ## Generate Loco offers a deployment template enabling the creation of a deployment infrastructure. + ```sh cargo loco generate deployment ? โฏ Choose your deployment โ€บ @@ -102,6 +155,8 @@ cargo loco generate deployment skipped (exists): "dockerfile" added: ".dockerignore" ``` + + Deployment Options: diff --git a/docs-site/content/docs/getting-started/guide.md b/docs-site/content/docs/getting-started/guide.md index cba50f50f..5201e8d46 100644 --- a/docs-site/content/docs/getting-started/guide.md +++ b/docs-site/content/docs/getting-started/guide.md @@ -43,24 +43,29 @@ You can follow this guide for a step-by-step "bottom up" learning, or you can ju ### Installing + ```sh -$ cargo install loco-cli +cargo install loco-cli +cargo install sea-orm-cli # Only when DB is needed ``` + + ### Creating a new Loco app Now you can create your new app (choose "SaaS app" for built-in authentication). + ```sh -$ loco new +โฏ loco new โœ” โฏ App name? ยท myapp -? โฏ What would you like to build? โ€บ - lightweight-service (minimal, only controllers and views) - Rest API (with DB and user auth) -โฏ SaaS app (with DB and user auth) +โœ” โฏ What would you like to build? ยท SaaS app (with DB and user auth) + ๐Ÿš‚ Loco app generated successfully in: myapp ``` + + You can now switch to to `myapp`: @@ -79,9 +84,15 @@ To configure a database, please run a local postgres database with loco:lo This docker command start up postgresql database server. + ```sh -docker run -d -p 5432:5432 -e POSTGRES_USER=loco -e POSTGRES_DB=myapp_development -e POSTGRES_PASSWORD="loco" postgres:15.3-alpine +docker run -d -p 5432:5432 \ + -e POSTGRES_USER=loco \ + -e POSTGRES_DB=myapp_development \ + -e POSTGRES_PASSWORD="loco" \ + postgres:15.3-alpine ``` + This docker command start up redis server: @@ -91,14 +102,16 @@ docker run -p 6379:6379 -d redis redis-server Use doctor command to check the needed resources: -``` + +```sh $ cargo loco doctor Finished dev [unoptimized + debuginfo] target(s) in 0.32s - Running `target/debug/myapp-cli doctor` + Running `target/debug/myapp-cli doctor` โœ… SeaORM CLI is installed โœ… DB connection: success โœ… Redis connection: success ``` + Here's a rundown of what Loco creates for you by default: @@ -125,27 +138,11 @@ Let's get some responses quickly. For this, we need to start up the server. ### Starting the server + ```sh -$ cargo loco start - - โ–„ โ–€ - โ–€ โ–„ - โ–„ โ–€ โ–„ โ–„ โ–„โ–€ - โ–„ โ–€โ–„โ–„ - โ–„ โ–€ โ–€ โ–€โ–„โ–€โ–ˆโ–„ - โ–€โ–ˆโ–„ -โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–€โ–€โ–ˆ - โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–€โ–ˆ - โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–€โ–€โ–€ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–„โ–ˆโ–„ - โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–„ - โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–„โ–„โ–„ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ - โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–€ - โ–€โ–€โ–€โ–ˆโ–ˆโ–„ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–ˆโ–ˆโ–€ - โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ - https://loco.rs - -listening on port 3000 +cargo loco start ``` + And now, let's see that it's alive: @@ -205,9 +202,11 @@ pub fn routes() -> Routes { Start the server: + ```sh -$ cargo loco start +cargo loco start ``` + Now, let's test it out: @@ -276,9 +275,11 @@ impl Hooks for App { That's it. Kill the server and bring it up again: + +```sh +cargo loco start ``` -$ cargo loco start -``` + And hit `/home/hello`: @@ -523,9 +524,11 @@ pub fn routes() -> Routes { Now, start the app: + ```sh -$ cargo loco start +cargo loco start ``` + And make a request: @@ -628,9 +631,11 @@ The order of the extractors is important, as changing the order of them can lead You can now test that it works, start the app: + ```sh -$ cargo loco start +cargo loco start ``` + Add a new article: diff --git a/docs-site/content/docs/getting-started/starters.md b/docs-site/content/docs/getting-started/starters.md index f9b9e17c6..4f6bb519a 100644 --- a/docs-site/content/docs/getting-started/starters.md +++ b/docs-site/content/docs/getting-started/starters.md @@ -15,20 +15,25 @@ flair =[] Simplify your project setup with Loco's predefined boilerplates, designed to make your development journey smoother. To get started, install our CLI and choose the template that suits your needs. + ```sh cargo install loco-cli +cargo install sea-orm-cli # Only when DB is needed ``` + Create a starter: + ```sh -loco new +โฏ loco new โœ” โฏ App name? ยท myapp -? โฏ What would you like to build? โ€บ -โฏ lightweight-service (minimal, only controllers and views) - Rest API (with DB and user auth) - SaaS app (with DB and user auth) +โœ” โฏ What would you like to build? ยท SaaS app (with DB and user auth) + +๐Ÿš‚ Loco app generated successfully in: +myapp ``` + ## Available Starters diff --git a/docs-site/content/docs/getting-started/tour/index.md b/docs-site/content/docs/getting-started/tour/index.md index 2330ec501..074a3e8db 100644 --- a/docs-site/content/docs/getting-started/tour/index.md +++ b/docs-site/content/docs/getting-started/tour/index.md @@ -19,11 +19,13 @@ flair =[]
Let's create a blog backend on `loco` in 4 commands. First install `loco-cli` and `sea-orm-cli`: - - ```sh - $ cargo install loco-cli - $ cargo install sea-orm-cli # Only when DB is needed - ``` + +```sh +cargo install loco-cli +cargo install sea-orm-cli # Only when DB is needed +``` + + Now you can create your new app (choose "`SaaS` app"). @@ -50,47 +52,50 @@ Let's create a blog backend on `loco` in 4 commands. First install `loco-cli` an name in the `test.yaml`configuration will be `myapp_test`, and in the `development.yaml` configuration, it will be `myapp_development`. + +```sh +docker run -d -p 5432:5432 \ + -e POSTGRES_USER=loco \ + -e POSTGRES_DB=myapp_development \ + -e POSTGRES_PASSWORD="loco" \ + postgres:15.3-alpine +``` + A more advanced set of `docker-compose.yml` and `Dockerfiles` that include Redis and the `mailtutan` mailer are available for [each starter on GitHub](https://github.com/loco-rs/loco/blob/master/starters/saas/.devcontainer/docker-compose.yml). Now `cd` into your `myapp` and start your app: - ``` - $ cd myapp - $ cargo loco start - Finished dev [unoptimized + debuginfo] target(s) in 21.63s - Running `target/debug/myapp start` - - : - : - : - - controller/app_routes.rs:203: [Middleware] Adding log trace id - - โ–„ โ–€ - โ–€ โ–„ - โ–„ โ–€ โ–„ โ–„ โ–„โ–€ - โ–„ โ–€โ–„โ–„ - โ–„ โ–€ โ–€ โ–€โ–„โ–€โ–ˆโ–„ - โ–€โ–ˆโ–„ - โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–€โ–€โ–ˆ - โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–€โ–ˆ - โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–€โ–€โ–€ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–„โ–ˆโ–„ - โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–„ - โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–„โ–„โ–„ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ - โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–€ - โ–€โ–€โ–€โ–ˆโ–ˆโ–„ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–ˆโ–ˆโ–€ - โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ - - started on port 3000 - ``` + +```sh +$ cargo loco start + + โ–„ โ–€ + โ–€ โ–„ + โ–„ โ–€ โ–„ โ–„ โ–„โ–€ + โ–„ โ–€โ–„โ–„ + โ–„ โ–€ โ–€ โ–€โ–„โ–€โ–ˆโ–„ + โ–€โ–ˆโ–„ +โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–€โ–€โ–ˆ +โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–€โ–ˆ +โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–€โ–€โ–€ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–„โ–ˆโ–„ +โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–„ +โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–„โ–„โ–„ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ +โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–€ + โ–€โ–€โ–€โ–ˆโ–ˆโ–„ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–ˆโ–ˆโ–€ + โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ + https://loco.rs + +listening on port 3000 +``` + +
You don't have to run things through `cargo` but in development it's highly recommended. If you build `--release`, your binary contains everything including your code and `cargo` or Rust is not needed.
- ## Adding a CRUD API @@ -115,10 +120,29 @@ injected: "tests/requests/mod.rs" Your database have been migrated and model, entities, and a full CRUD controller have been generated automatically. Start your app: - + ```sh $ cargo loco start + + โ–„ โ–€ + โ–€ โ–„ + โ–„ โ–€ โ–„ โ–„ โ–„โ–€ + โ–„ โ–€โ–„โ–„ + โ–„ โ–€ โ–€ โ–€โ–„โ–€โ–ˆโ–„ + โ–€โ–ˆโ–„ +โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–€โ–€โ–ˆ +โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–€โ–ˆ +โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–€โ–€โ–€ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–„โ–ˆโ–„ +โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–„ +โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–„โ–„โ–„ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ +โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–€ + โ–€โ–€โ–€โ–ˆโ–ˆโ–„ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–ˆโ–ˆโ–€ + โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ + https://loco.rs + +listening on port 3000 ``` + Next, try adding a `post` with `curl`: @@ -152,20 +176,24 @@ To authenticate, you will need a running redis server. This docker command starts up a redis server: -``` + +```sh docker run -p 6379:6379 -d redis redis-server ``` + Use doctor command to check the needed resources: -``` + +```sh $ cargo loco doctor Finished dev [unoptimized + debuginfo] target(s) in 0.32s - Running `target/debug/myapp-cli doctor` + Running `target/debug/myapp-cli doctor` โœ… SeaORM CLI is installed โœ… DB connection: success โœ… Redis connection: success ``` + ### Registering a New User diff --git a/docs-site/templates/shortcodes/get_env.html b/docs-site/templates/shortcodes/get_env.html new file mode 100644 index 000000000..e69de29bb diff --git a/examples/demo/config/development.yaml b/examples/demo/config/development.yaml index f932a6338..9e3949834 100644 --- a/examples/demo/config/development.yaml +++ b/examples/demo/config/development.yaml @@ -1,10 +1,13 @@ # Loco configuration file documentation +# settings: allow_list: - google.com - apple.com +# +# # Application logging configuration logger: # Enable or disable logging. @@ -18,14 +21,17 @@ logger: # By default the logger has filtering only logs that came from your code or logs that came from `loco` framework. to see all third party libraries # Uncomment the line below to override to see all third party libraries you can enable this config and override the logger filters. # override_filter: trace +# # Web server configuration +# server: # Port on which the server will listen. the server binding is 0.0.0.0:{PORT} port: {{get_env(name="NODE_PORT", default=3000)}} # The UI hostname or IP address that mailers will point to. host: http://localhost # Out of the box middleware configuration. to disable middleware you can changed the `enable` field to `false` of comment the middleware block +# middlewares: # Allows to limit the payload size request. payload that bigger than this file will blocked the request. limit_payload: @@ -80,6 +86,7 @@ workers: mode: BackgroundQueue # Mailer Configuration. +# mailer: # SMTP mailer configuration. smtp: @@ -94,6 +101,7 @@ mailer: # auth: # user: # password: +# # Initializers Configuration # initializers: @@ -104,6 +112,7 @@ mailer: # Database Configuration +# database: # Database connection URI uri: {{get_env(name="DATABASE_URL", default="postgres://loco:loco@localhost:5432/loco_app")}} @@ -123,15 +132,19 @@ database: dangerously_truncate: false # Recreating schema when application loaded. This is a dangerous operation, make sure that you using this flag only on dev environments or test mode dangerously_recreate: false +# # Redis Configuration +# redis: # Redis connection URI uri: {{get_env(name="REDIS_URL", default="redis://127.0.0.1")}} # Dangerously flush all data in Redis on startup. dangerous operation, make sure that you using this flag only on dev environments or test mode dangerously_flush: false +# # Authentication Configuration +# auth: # JWT authentication jwt: @@ -139,3 +152,5 @@ auth: secret: PqRwLF2rhHe8J22oBeHy # Token expiration time in seconds expiration: 604800 # 7 days +# + diff --git a/snipdoc.yml b/snipdoc.yml new file mode 100644 index 000000000..1a7f61b14 --- /dev/null +++ b/snipdoc.yml @@ -0,0 +1,124 @@ +snippets: + description: + content: ๐Ÿš‚ Loco is Rust on Rails. + kind: Yaml + path: ./snipdoc.yml + loco-help-command: + content: |- + ```sh + cargo loco --help + ``` + kind: Yaml + path: ./snipdoc.yml + loco-build-command: + content: |- + ```sh + cargo build --release + ``` + kind: Yaml + path: ./snipdoc.yml + quick-installation-command: + content: |- + ```sh + cargo install loco-cli + cargo install sea-orm-cli # Only when DB is needed + ``` + kind: Yaml + path: ./snipdoc.yml + loco-cli-new-from-template: + content: |- + ```sh + โฏ loco new + โœ” โฏ App name? ยท myapp + โœ” โฏ What would you like to build? ยท SaaS app (with DB and user auth) + + ๐Ÿš‚ Loco app generated successfully in: + myapp + ``` + kind: Yaml + path: ./snipdoc.yml + postgres-run-docker-command: + content: |- + ```sh + docker run -d -p 5432:5432 \ + -e POSTGRES_USER=loco \ + -e POSTGRES_DB=myapp_development \ + -e POSTGRES_PASSWORD="loco" \ + postgres:15.3-alpine + ``` + kind: Yaml + path: ./snipdoc.yml + redis-run-docker-command: + content: |- + ```sh + docker run -p 6379:6379 -d redis redis-server + ``` + kind: Yaml + path: ./snipdoc.yml + starting-the-server-command: + content: |- + ```sh + cargo loco start + ``` + kind: Yaml + path: ./snipdoc.yml + starting-the-server-command-with-environment-env-var: + content: |- + ```sh + LOCO_ENV=qa cargo loco start + ``` + kind: Yaml + path: ./snipdoc.yml + starting-the-server-command-with-output: + content: |- + ```sh + $ cargo loco start + + โ–„ โ–€ + โ–€ โ–„ + โ–„ โ–€ โ–„ โ–„ โ–„โ–€ + โ–„ โ–€โ–„โ–„ + โ–„ โ–€ โ–€ โ–€โ–„โ–€โ–ˆโ–„ + โ–€โ–ˆโ–„ + โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–€โ–€โ–ˆ + โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–€โ–ˆ + โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–€โ–€โ–€ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–„โ–ˆโ–„ + โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–„ + โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–„โ–„โ–„ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ + โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–€ + โ–€โ–€โ–€โ–ˆโ–ˆโ–„ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–ˆโ–ˆโ–€ + โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ + https://loco.rs + + listening on port 3000 + ``` + kind: Yaml + path: ./snipdoc.yml + doctor-command: + content: |- + ```sh + $ cargo loco doctor + Finished dev [unoptimized + debuginfo] target(s) in 0.32s + Running `target/debug/myapp-cli doctor` + โœ… SeaORM CLI is installed + โœ… DB connection: success + โœ… Redis connection: success + ``` + kind: Yaml + path: ./snipdoc.yml + generate-deployment-command: + content: |- + ```sh + cargo loco generate deployment + ? โฏ Choose your deployment โ€บ + โฏ Docker + โฏ Shuttle + โฏ Nginx + + .. + โœ” โฏ Choose your deployment ยท Docker + skipped (exists): "dockerfile" + added: ".dockerignore" + ``` + kind: Yaml + path: ./snipdoc.yml diff --git a/src/lib.rs b/src/lib.rs index a62787694..29f47c3b8 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -1,118 +1,6 @@ #![allow(clippy::missing_const_for_fn)] #![allow(clippy::module_name_repetitions)] -// -//!
-//! -//! -//! -//!

Loco

-//! -//!

๐Ÿš‚ Loco is "Rust on Rails".

-//! -//! [![crate](https://img.shields.io/crates/v/loco-rs.svg)](https://crates.io/crates/loco-rs) -//! [![docs](https://docs.rs/loco-rs/badge.svg)](https://docs.rs/loco-rs) -//! [![Discord channel](https://img.shields.io/badge/discord-Join-us)](https://discord.gg/fTvyBzwKS8) -//! -//!
-//! -//! # Loco -//! -//! #### Loco is strongly inspired by Rails. If you know Rails and Rust, you'll feel at home. If you only know Rails and new to Rust, you'll find Loco refreshing. We do not assume you know Rails. -//! -//! ## Quick Start -// -//! ```sh -//! $ cargo install loco-cli -//! $ cargo install sea-orm-cli # Only when DB is needed -//! ``` -//! -//! Now you can create your new app (choose "`SaaS` app"). -//! -//! ```sh -//! $ loco new -//! โœ” โฏ App name? ยท myapp -//! ? โฏ What would you like to build? โ€บ -//! lightweight-service (minimal, only controllers and views) -//! Rest API (with DB and user auth) -//! โฏ SaaS app (with DB and user auth) -//! ๐Ÿš‚ Loco app generated successfully in: -//! myapp -//! ``` -//! -//!
-//! To configure a database , please run a local postgres database with -//! loco:loco and a db named is the [insert app]_development. -//!
-//! -//! You can use Docker to run a Postgres instance: -//! -//! When generating a starter, the database name incorporates your application -//! name and the environment. For instance, if you include `myapp`, the database -//! name in the `test.yaml`configuration will be `myapp_test`, and in the -//! `development.yaml` configuration, it will be `myapp_development`. -//! -//! -//! -//! A more advanced set of `docker-compose.yml` and `Dockerfiles` that include Redis and the `mailtutan` mailer are available for [each starter on GitHub](https://github.com/loco-rs/loco/blob/master/starters/saas/.devcontainer/docker-compose.yml). -//! -//! Now `cd` into your `myapp` and start your app: -//! -//! ``` -//! $ cd myapp -//! $ cargo loco start -//! Finished dev [unoptimized + debuginfo] target(s) in 21.63s -//! Running `target/debug/myapp start` -//! -//! : -//! : -//! : -//! -//! controller/app_routes.rs:203: [Middleware] Adding log trace id -//! -//! โ–„ โ–€ -//! โ–€ โ–„ -//! โ–„ โ–€ โ–„ โ–„ โ–„โ–€ -//! โ–„ โ–€โ–„โ–„ -//! โ–„ โ–€ โ–€ โ–€โ–„โ–€โ–ˆโ–„ -//! โ–€โ–ˆโ–„ -//! โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„โ–„ โ–€โ–€โ–ˆ -//! โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–€โ–ˆ -//! โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–€โ–€โ–€ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–„โ–ˆโ–„ -//! โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–„ -//! โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–„โ–„โ–„ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ -//! โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆ โ–ˆโ–ˆโ–ˆโ–ˆโ–€ -//! โ–€โ–€โ–€โ–ˆโ–ˆโ–„ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ โ–ˆโ–ˆโ–€ -//! โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€โ–€ -//! -//! started on port 3000 -//! ``` -//! -//!
-//! You don't have to run things through `cargo` but in development it's highly -//! recommended. If you build `--release`, your binary contains everything -//! including your code and `cargo` or Rust is not needed.
-//
-//! -//! ## Project Status -//! + Stateless APIs -//! + Complete `SaaS` products with user authentication -//! + Purpose-built services such as ML inference endpoints -//! + Full stack projects with separate frontend project integrated with Loco -//! + Hobby projects full-stack with backend and HTML frontend -//! -//! ## Powered by Loco -//! + [SpectralOps](https://spectralops.io) - various services powered by Loco -//! framework -//! + [Nativish](https://nativi.sh) - app backend powered by Loco framework -//! -//! ## Contributors โœจ -//! -//! Thanks goes to these wonderful people: -//! -//! -//! -//! -//
+#![doc = include_str!("../README.md")] pub use self::errors::Error; From 02189c2f97e35841cc6ac404f3b23baea6d8c05c Mon Sep 17 00:00:00 2001 From: Elad Kaplan Date: Mon, 27 May 2024 09:37:18 +0300 Subject: [PATCH 18/23] rename redis in app context and config to queue --- examples/demo/config/development.yaml | 4 ++-- examples/demo/config/test.yaml | 4 ++-- examples/demo/config/teste2e.yaml | 4 ++-- src/app.rs | 4 ++-- src/boot.rs | 16 +++++++--------- src/config.rs | 2 +- src/controller/health.rs | 2 +- src/worker.rs | 6 +++--- 8 files changed, 20 insertions(+), 22 deletions(-) diff --git a/examples/demo/config/development.yaml b/examples/demo/config/development.yaml index f932a6338..f49d49bdc 100644 --- a/examples/demo/config/development.yaml +++ b/examples/demo/config/development.yaml @@ -124,8 +124,8 @@ database: # Recreating schema when application loaded. This is a dangerous operation, make sure that you using this flag only on dev environments or test mode dangerously_recreate: false -# Redis Configuration -redis: +# Queue Configuration +queue: # Redis connection URI uri: {{get_env(name="REDIS_URL", default="redis://127.0.0.1")}} # Dangerously flush all data in Redis on startup. dangerous operation, make sure that you using this flag only on dev environments or test mode diff --git a/examples/demo/config/test.yaml b/examples/demo/config/test.yaml index da7dc833f..3734579b6 100644 --- a/examples/demo/config/test.yaml +++ b/examples/demo/config/test.yaml @@ -116,8 +116,8 @@ database: # Recreating schema when application loaded. This is a dangerous operation, make sure that you using this flag only on dev environments or test mode dangerously_recreate: true -# Redis Configuration -redis: +# Queue Configuration +queue: # Redis connection URI uri: {{get_env(name="REDIS_URL", default="redis://127.0.0.1")}} # Dangerously flush all data in Redis on startup. dangerous operation, make sure that you using this flag only on dev environments or test mode diff --git a/examples/demo/config/teste2e.yaml b/examples/demo/config/teste2e.yaml index 3679cb8fb..e0e9cae38 100644 --- a/examples/demo/config/teste2e.yaml +++ b/examples/demo/config/teste2e.yaml @@ -95,8 +95,8 @@ database: # Recreating schema when application loaded. This is a dangerous operation, make sure that you using this flag only on dev environments or test mode dangerously_recreate: true -# Redis Configuration -redis: +# Queue Configuration +queue: # Redis connection URI uri: {{get_env(name="APP_REDIS_URI", default="redis://127.0.0.1")}} # Dangerously flush all data in Redis on startup. dangerous operation, make sure that you using this flag only on dev environments or test mode diff --git a/src/app.rs b/src/app.rs index acb0d35c2..0dab04631 100644 --- a/src/app.rs +++ b/src/app.rs @@ -41,8 +41,8 @@ pub struct AppContext { #[cfg(feature = "with-db")] /// A database connection used by the application. pub db: DatabaseConnection, - /// An optional connection pool for Redis, for worker tasks - pub redis: Option>, + /// An optional connection pool for Queue, for worker tasks + pub queue: Option>, /// Configuration settings for the application pub config: Config, /// An optional email sender component that can be used to send email. diff --git a/src/boot.rs b/src/boot.rs index c0d4f54a5..e1289c692 100644 --- a/src/boot.rs +++ b/src/boot.rs @@ -200,13 +200,11 @@ pub async fn create_context(environment: &Environment) -> Result( let app_context = create_context::(environment).await?; db::converge::(&app_context.db, &app_context.config.database).await?; - if let Some(pool) = &app_context.redis { - redis::converge(pool, &app_context.config.redis).await?; + if let Some(pool) = &app_context.queue { + redis::converge(pool, &app_context.config.queue).await?; } run_app::(&mode, app_context).await @@ -305,9 +303,9 @@ fn create_processor(app_context: &AppContext) -> Result { queues = ?queues, "registering queues (merged config and default)" ); - let mut p = if let Some(redis) = &app_context.redis { + let mut p = if let Some(queue) = &app_context.queue { Processor::new( - redis.clone(), + queue.clone(), DEFAULT_QUEUES .iter() .map(ToString::to_string) @@ -315,7 +313,7 @@ fn create_processor(app_context: &AppContext) -> Result { ) } else { return Err(Error::Message( - "redis is missing, cannot initialize workers".to_string(), + "queue is missing, cannot initialize workers".to_string(), )); }; @@ -351,7 +349,7 @@ fn create_mailer(config: &config::Mailer) -> Result> { // TODO: Refactor to eliminate unwrapping and instead return an appropriate // error type. pub async fn connect_redis(config: &Config) -> Option> { - if let Some(redis) = &config.redis { + if let Some(redis) = &config.queue { let manager = RedisConnectionManager::new(redis.uri.clone()).unwrap(); let redis = Pool::builder().build(manager).await.unwrap(); Some(redis) diff --git a/src/config.rs b/src/config.rs index d1b00bd0c..1fbe49ddf 100644 --- a/src/config.rs +++ b/src/config.rs @@ -51,7 +51,7 @@ pub struct Config { pub server: Server, #[cfg(feature = "with-db")] pub database: Database, - pub redis: Option, + pub queue: Option, pub auth: Option, #[serde(default)] pub workers: Workers, diff --git a/src/controller/health.rs b/src/controller/health.rs index 9509cf66a..3829a3d4c 100644 --- a/src/controller/health.rs +++ b/src/controller/health.rs @@ -24,7 +24,7 @@ async fn health(State(ctx): State) -> Result { false } }; - if let Some(pool) = ctx.redis { + if let Some(pool) = ctx.queue { if let Err(error) = redis::ping(&pool).await { tracing::error!(err.msg = %error, err.detail = ?error, "health_redis_ping_error"); is_ok = false; diff --git a/src/worker.rs b/src/worker.rs index f42eaace0..4ca2e6ccb 100644 --- a/src/worker.rs +++ b/src/worker.rs @@ -35,12 +35,12 @@ where async fn perform_later(ctx: &AppContext, args: T) -> Result<()> { match &ctx.config.workers.mode { WorkerMode::BackgroundQueue => { - if let Some(redis) = &ctx.redis { - Self::perform_async(redis, args).await.unwrap(); + if let Some(queue) = &ctx.queue { + Self::perform_async(queue, args).await.unwrap(); } else { error!( error.msg = - "worker mode requested but no redis connection supplied, skipping job", + "worker mode requested but no queue connection supplied, skipping job", "worker_error" ); } From 6e448ffc73dca642ac32e15dcb3b13f37c57aae9 Mon Sep 17 00:00:00 2001 From: Elad Kaplan Date: Mon, 27 May 2024 11:39:17 +0300 Subject: [PATCH 19/23] order the valiation results --- src/validation.rs | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/validation.rs b/src/validation.rs index 537eeb648..af20ab9c7 100644 --- a/src/validation.rs +++ b/src/validation.rs @@ -84,10 +84,10 @@ impl From for DbErr { #[cfg(feature = "with-db")] #[must_use] pub fn into_db_error(errors: &ModelValidationErrors) -> sea_orm::DbErr { - use std::collections::HashMap; + use std::collections::BTreeMap; let errors = &errors.0; - let error_data: HashMap> = errors + let error_data: BTreeMap> = errors .field_errors() .iter() .map(|(field, field_errors)| { From 77306dc341297cede1996a9f3fb92ce3e96ad2ea Mon Sep 17 00:00:00 2001 From: Elad Kaplan Date: Mon, 27 May 2024 14:24:49 +0300 Subject: [PATCH 20/23] update docs --- docs-site/content/docs/getting-started/deployment.md | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/docs-site/content/docs/getting-started/deployment.md b/docs-site/content/docs/getting-started/deployment.md index 77e6aa959..b1e1cab69 100644 --- a/docs-site/content/docs/getting-started/deployment.md +++ b/docs-site/content/docs/getting-started/deployment.md @@ -113,10 +113,10 @@ logger: ``` -- Redis: +- Queue: ```yaml - redis: + queue: # Redis connection URI uri: {{get_env(name="REDIS_URL", default="redis://127.0.0.1")}} # Dangerously flush all data in Redis on startup. dangerous operation, make sure that you using this flag only on dev environments or test mode From 7246c028f8eace9d0e958ded989138c51d747bf1 Mon Sep 17 00:00:00 2001 From: Elad Kaplan Date: Mon, 27 May 2024 14:32:03 +0300 Subject: [PATCH 21/23] update docs --- CONTRIBUTING.md | 16 +++++++++++++--- 1 file changed, 13 insertions(+), 3 deletions(-) diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md index 4dcc0fa4c..c40fa52a7 100644 --- a/CONTRIBUTING.md +++ b/CONTRIBUTING.md @@ -64,9 +64,19 @@ In case of cli changes we snapshot the binary commands. in case of changes run t LOCO_CI_MODE=true TRYCMD=overwrite cargo test ``` -## Running Docs website -The documentation website based on [zola](https://www.getzola.org/), and you can see the docs [here](./docs-site/). -then cd to `docs-site` and run `zola serve` +## Docs + +The documentation consists of two main components: + ++ The [loco.rs website](https://loco.rs) with its source code available [here](./docs-site/). ++ RustDocs. + +To reduce duplication in documentation and examples, we use [snipdoc](https://github.com/kaplanelad/snipdoc). As part of our CI process, we ensure that the documentation remains consistent. + +Updating the Documentation ++ Download [snipdoc](https://github.com/kaplanelad/snipdoc). ++ Create the snippet in the [yaml file](./snipdoc.yml) or inline the code. ++ Run `snipdoc run`. ## Open A Pull Request From c5eec19f325e44b3d69c2c6c8e9a85e7e18de03c Mon Sep 17 00:00:00 2001 From: Elad Kaplan Date: Mon, 27 May 2024 14:38:57 +0300 Subject: [PATCH 22/23] Update CHANGELOG.md --- CHANGELOG.md | 3 +++ 1 file changed, 3 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index f6cfa0678..77e95abc5 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,6 +1,9 @@ # Changelog ## vNext +* **Braking changes** In the application configuration setting `redis`, change to `queue`. [https://github.com/loco-rs/loco/pull/590](https://github.com/loco-rs/loco/pull/590) +* **Braking changes** The storage webhook was changed to `after_context` see example [here](https://github.com/loco-rs/loco/pull/570/files#diff-5534e8826fb82e5c7f2587d270a51b48009341e79889d1504e6b63b2f0b652bdR83). [https://github.com/loco-rs/loco/pull/570](https://github.com/loco-rs/loco/pull/570) +* Adding Cache to app content. [https://github.com/loco-rs/loco/pull/570](https://github.com/loco-rs/loco/pull/570) * Apply a layer to a specific handler using `layer` method. [https://github.com/loco-rs/loco/pull/554](https://github.com/loco-rs/loco/pull/554) * Add the debug macro to the templates to improve the errors. [https://github.com/loco-rs/loco/pull/547](https://github.com/loco-rs/loco/pull/547) * Opentelemetry initializer. [https://github.com/loco-rs/loco/pull/531](https://github.com/loco-rs/loco/pull/531) From 9179d3dbcc769ddaa6a106a313e8b1376d26d48f Mon Sep 17 00:00:00 2001 From: Elad Kaplan Date: Mon, 27 May 2024 15:00:53 +0300 Subject: [PATCH 23/23] more changes --- docs-site/content/docs/getting-started/cli.md | 2 +- docs-site/content/docs/getting-started/config.md | 2 +- .../content/docs/getting-started/deployment.md | 14 +++++++------- examples/demo/config/development.yaml | 14 +++++++------- snipdoc.yml | 16 ++-------------- 5 files changed, 18 insertions(+), 30 deletions(-) diff --git a/docs-site/content/docs/getting-started/cli.md b/docs-site/content/docs/getting-started/cli.md index 4d7c4a7bf..873760991 100644 --- a/docs-site/content/docs/getting-started/cli.md +++ b/docs-site/content/docs/getting-started/cli.md @@ -29,7 +29,7 @@ myapp Now `cd` into your app, set up a convenience `rr` alias and try out the various commands: - + ```sh cargo loco --help ``` diff --git a/docs-site/content/docs/getting-started/config.md b/docs-site/content/docs/getting-started/config.md index 4c460cb44..70f9cd305 100644 --- a/docs-site/content/docs/getting-started/config.md +++ b/docs-site/content/docs/getting-started/config.md @@ -70,7 +70,7 @@ LOCO_ENV=qa cargo loco start ## Settings The configuration files contain knobs to set up your Loco app. You can also have your custom settings, with the `settings:` section. in `config/development.yaml` add the `settings:` section - + ```yaml settings: allow_list: diff --git a/docs-site/content/docs/getting-started/deployment.md b/docs-site/content/docs/getting-started/deployment.md index b1e1cab69..bd127be30 100644 --- a/docs-site/content/docs/getting-started/deployment.md +++ b/docs-site/content/docs/getting-started/deployment.md @@ -17,7 +17,7 @@ Deployment is super simple in Loco, and this is why this guide is super short. A To deploy, build your production binary for your relevant server architecture: - + ```sh cargo build --release ``` @@ -35,7 +35,7 @@ There are a few configuration sections that are important to review and set acco - Logger: - + ```yaml # Application logging configuration logger: @@ -55,7 +55,7 @@ logger: - Server: - + ```yaml server: # Port on which the server will listen. the server binding is 0.0.0.0:{PORT} @@ -68,7 +68,7 @@ logger: - Database: - + ```yaml database: # Database connection URI @@ -94,7 +94,7 @@ logger: - Mailer: - + ```yaml mailer: # SMTP mailer configuration. @@ -114,7 +114,7 @@ logger: - Queue: - + ```yaml queue: # Redis connection URI @@ -125,7 +125,7 @@ logger: - JWT secret: - + ```yaml auth: # JWT authentication diff --git a/examples/demo/config/development.yaml b/examples/demo/config/development.yaml index 64a258cc9..ceb527184 100644 --- a/examples/demo/config/development.yaml +++ b/examples/demo/config/development.yaml @@ -1,13 +1,13 @@ # Loco configuration file documentation -# +# settings: allow_list: - google.com - apple.com # -# +# # Application logging configuration logger: # Enable or disable logging. @@ -24,7 +24,7 @@ logger: # # Web server configuration -# +# server: # Port on which the server will listen. the server binding is 0.0.0.0:{PORT} port: {{get_env(name="NODE_PORT", default=3000)}} @@ -86,7 +86,7 @@ workers: mode: BackgroundQueue # Mailer Configuration. -# +# mailer: # SMTP mailer configuration. smtp: @@ -112,7 +112,7 @@ mailer: # Database Configuration -# +# database: # Database connection URI uri: {{get_env(name="DATABASE_URL", default="postgres://loco:loco@localhost:5432/loco_app")}} @@ -135,7 +135,7 @@ database: # # Queue Configuration -# +# queue: # Redis connection URI uri: {{get_env(name="REDIS_URL", default="redis://127.0.0.1")}} @@ -144,7 +144,7 @@ queue: # # Authentication Configuration -# +# auth: # JWT authentication jwt: diff --git a/snipdoc.yml b/snipdoc.yml index 1a7f61b14..0883e2855 100644 --- a/snipdoc.yml +++ b/snipdoc.yml @@ -1,21 +1,18 @@ snippets: description: content: ๐Ÿš‚ Loco is Rust on Rails. - kind: Yaml path: ./snipdoc.yml - loco-help-command: + help-command: content: |- ```sh cargo loco --help ``` - kind: Yaml path: ./snipdoc.yml - loco-build-command: + build-command: content: |- ```sh cargo build --release ``` - kind: Yaml path: ./snipdoc.yml quick-installation-command: content: |- @@ -23,7 +20,6 @@ snippets: cargo install loco-cli cargo install sea-orm-cli # Only when DB is needed ``` - kind: Yaml path: ./snipdoc.yml loco-cli-new-from-template: content: |- @@ -35,7 +31,6 @@ snippets: ๐Ÿš‚ Loco app generated successfully in: myapp ``` - kind: Yaml path: ./snipdoc.yml postgres-run-docker-command: content: |- @@ -46,28 +41,24 @@ snippets: -e POSTGRES_PASSWORD="loco" \ postgres:15.3-alpine ``` - kind: Yaml path: ./snipdoc.yml redis-run-docker-command: content: |- ```sh docker run -p 6379:6379 -d redis redis-server ``` - kind: Yaml path: ./snipdoc.yml starting-the-server-command: content: |- ```sh cargo loco start ``` - kind: Yaml path: ./snipdoc.yml starting-the-server-command-with-environment-env-var: content: |- ```sh LOCO_ENV=qa cargo loco start ``` - kind: Yaml path: ./snipdoc.yml starting-the-server-command-with-output: content: |- @@ -92,7 +83,6 @@ snippets: listening on port 3000 ``` - kind: Yaml path: ./snipdoc.yml doctor-command: content: |- @@ -104,7 +94,6 @@ snippets: โœ… DB connection: success โœ… Redis connection: success ``` - kind: Yaml path: ./snipdoc.yml generate-deployment-command: content: |- @@ -120,5 +109,4 @@ snippets: skipped (exists): "dockerfile" added: ".dockerignore" ``` - kind: Yaml path: ./snipdoc.yml