#![feature(proc_macro_hygiene)] #[macro_use] extern crate rocket; #[macro_use] extern crate diesel; #[macro_use] extern crate diesel_migrations; #[macro_use] extern crate log; #[macro_use] extern crate serde_derive; #[macro_use] extern crate rocket_contrib; mod task; #[cfg(test)] mod tests; use rocket::Rocket; use rocket::fairing::AdHoc; use rocket::request::{Form, FlashMessage}; use rocket::response::{Flash, Redirect}; use rocket_contrib::{templates::Template, serve::StaticFiles}; use diesel::SqliteConnection; use crate::task::{Task, Todo}; // This macro from `diesel_migrations` defines an `embedded_migrations` module // containing a function named `run`. This allows the example to be run and // tested without any outside setup of the database. embed_migrations!(); #[database("sqlite_database")] pub struct DbConn(SqliteConnection); #[derive(Debug, Serialize)] struct Context<'a, 'b>{ msg: Option<(&'a str, &'b str)>, tasks: Vec } impl<'a, 'b> Context<'a, 'b> { pub fn err(conn: &DbConn, msg: &'a str) -> Context<'static, 'a> { Context{msg: Some(("error", msg)), tasks: Task::all(conn)} } pub fn raw(conn: &DbConn, msg: Option<(&'a str, &'b str)>) -> Context<'a, 'b> { Context{msg: msg, tasks: Task::all(conn)} } } #[post("/", data = "")] 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 Task::insert(todo, &conn) { Flash::success(Redirect::to("/"), "Todo successfully added.") } else { Flash::error(Redirect::to("/"), "Whoops! The server failed.") } } #[put("/")] fn toggle(id: i32, conn: DbConn) -> Result { if Task::toggle_with_id(id, &conn) { Ok(Redirect::to("/")) } else { Err(Template::render("index", &Context::err(&conn, "Couldn't toggle task."))) } } #[delete("/")] fn delete(id: i32, conn: DbConn) -> Result, Template> { if Task::delete_with_id(id, &conn) { Ok(Flash::success(Redirect::to("/"), "Todo was deleted.")) } else { Err(Template::render("index", &Context::err(&conn, "Couldn't delete task."))) } } #[get("/")] fn index(msg: Option>, conn: DbConn) -> Template { Template::render("index", &match msg { Some(ref msg) => Context::raw(&conn, Some((msg.name(), msg.msg()))), None => Context::raw(&conn, None), }) } fn run_db_migrations(rocket: Rocket) -> Result { let conn = DbConn::get_one(&rocket).expect("database connection"); match embedded_migrations::run(&*conn) { Ok(()) => Ok(rocket), Err(e) => { error!("Failed to run database migrations: {:?}", e); Err(rocket) } } } fn rocket() -> Rocket { rocket::ignite() .attach(DbConn::fairing()) .attach(AdHoc::on_attach("Database Migrations", run_db_migrations)) .mount("/", StaticFiles::from("static/")) .mount("/", routes![index]) .mount("/todo", routes![new, toggle, delete]) .attach(Template::fairing()) } fn main() { rocket().launch(); }