#[macro_use] extern crate rocket; #[macro_use] extern crate rocket_sync_db_pools; #[macro_use] extern crate diesel; #[cfg(test)] mod tests; mod task; use rocket::{Rocket, Build}; use rocket::fairing::AdHoc; use rocket::request::FlashMessage; use rocket::response::{Flash, Redirect}; use rocket::serde::Serialize; use rocket::form::Form; use rocket::fs::{FileServer, relative}; use rocket_dyn_templates::Template; use crate::task::{Task, Todo}; #[database("sqlite_database")] pub struct DbConn(diesel::SqliteConnection); #[derive(Debug, Serialize)] #[serde(crate = "rocket::serde")] struct Context { flash: Option<(String, String)>, tasks: Vec } impl Context { pub async fn err(conn: &DbConn, msg: M) -> Context { Context { flash: Some(("error".into(), msg.to_string())), tasks: Task::all(conn).await.unwrap_or_default() } } pub async fn raw(conn: &DbConn, flash: Option<(String, String)>) -> Context { match Task::all(conn).await { Ok(tasks) => Context { flash, tasks }, Err(e) => { error!("DB Task::all() error: {e}"); Context { flash: Some(("error".into(), "Fail to access database.".into())), tasks: vec![] } } } } } #[post("/", data = "")] async fn new(todo_form: Form, conn: DbConn) -> Flash { let todo = todo_form.into_inner(); if todo.description.is_empty() { Flash::error(Redirect::to("/"), "Description cannot be empty.") } else if let Err(e) = Task::insert(todo, &conn).await { error!("DB insertion error: {e}"); Flash::error(Redirect::to("/"), "Todo could not be inserted due an internal error.") } else { Flash::success(Redirect::to("/"), "Todo successfully added.") } } #[put("/")] async fn toggle(id: i32, conn: DbConn) -> Result { match Task::toggle_with_id(id, &conn).await { Ok(_) => Ok(Redirect::to("/")), Err(e) => { error!("DB toggle({id}) error: {e}"); Err(Template::render("index", Context::err(&conn, "Failed to toggle task.").await)) } } } #[delete("/")] async fn delete(id: i32, conn: DbConn) -> Result, Template> { match Task::delete_with_id(id, &conn).await { Ok(_) => Ok(Flash::success(Redirect::to("/"), "Todo was deleted.")), Err(e) => { error!("DB deletion({id}) error: {e}"); Err(Template::render("index", Context::err(&conn, "Failed to delete task.").await)) } } } #[get("/")] async fn index(flash: Option>, conn: DbConn) -> Template { let flash = flash.map(FlashMessage::into_inner); Template::render("index", Context::raw(&conn, flash).await) } async fn run_migrations(rocket: Rocket) -> Rocket { use diesel_migrations::{embed_migrations, EmbeddedMigrations, MigrationHarness}; const MIGRATIONS: EmbeddedMigrations = embed_migrations!("migrations"); DbConn::get_one(&rocket).await .expect("database connection") .run(|conn| { conn.run_pending_migrations(MIGRATIONS).expect("diesel migrations"); }) .await; rocket } #[launch] fn rocket() -> _ { rocket::build() .attach(DbConn::fairing()) .attach(Template::fairing()) .attach(AdHoc::on_ignite("Run Migrations", run_migrations)) .mount("/", FileServer::from(relative!("static"))) .mount("/", routes![index]) .mount("/todo", routes![new, toggle, delete]) }