2018-08-07 02:58:07 +00:00
|
|
|
#[macro_use] extern crate rocket;
|
2016-08-02 02:07:36 +00:00
|
|
|
#[macro_use] extern crate diesel;
|
2018-09-11 19:35:33 +00:00
|
|
|
#[macro_use] extern crate diesel_migrations;
|
|
|
|
#[macro_use] extern crate log;
|
2018-10-09 11:16:57 +00:00
|
|
|
#[macro_use] extern crate rocket_contrib;
|
2016-08-02 02:07:36 +00:00
|
|
|
|
|
|
|
mod task;
|
2017-05-26 23:44:53 +00:00
|
|
|
#[cfg(test)] mod tests;
|
2016-07-16 04:09:08 +00:00
|
|
|
|
2017-05-26 23:44:53 +00:00
|
|
|
use rocket::Rocket;
|
2018-09-11 19:35:33 +00:00
|
|
|
use rocket::fairing::AdHoc;
|
2016-11-02 17:49:06 +00:00
|
|
|
use rocket::request::{Form, FlashMessage};
|
2016-09-12 01:57:04 +00:00
|
|
|
use rocket::response::{Flash, Redirect};
|
2018-10-09 11:16:57 +00:00
|
|
|
use rocket_contrib::{templates::Template, serve::StaticFiles};
|
2018-07-21 22:11:08 +00:00
|
|
|
use diesel::SqliteConnection;
|
2017-02-03 01:38:36 +00:00
|
|
|
|
2019-06-13 02:41:29 +00:00
|
|
|
use crate::task::{Task, Todo};
|
2016-08-02 02:07:36 +00:00
|
|
|
|
2018-09-11 19:35:33 +00:00
|
|
|
// 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!();
|
|
|
|
|
2018-07-21 22:11:08 +00:00
|
|
|
#[database("sqlite_database")]
|
|
|
|
pub struct DbConn(SqliteConnection);
|
|
|
|
|
2020-07-19 16:12:32 +00:00
|
|
|
#[derive(Debug, serde::Serialize)]
|
2020-06-08 19:11:34 +00:00
|
|
|
struct Context<'a> {
|
|
|
|
msg: Option<(&'a str, &'a str)>,
|
|
|
|
tasks: Vec<Task>
|
|
|
|
}
|
2016-07-16 04:09:08 +00:00
|
|
|
|
2020-02-13 04:11:51 +00:00
|
|
|
impl<'a> Context<'a> {
|
|
|
|
pub fn err(conn: &DbConn, msg: &'a str) -> Context<'a> {
|
2020-06-08 19:18:29 +00:00
|
|
|
Context { msg: Some(("error", msg)), tasks: Task::all(conn).unwrap_or_default() }
|
2016-09-22 11:12:07 +00:00
|
|
|
}
|
|
|
|
|
2020-02-13 04:11:51 +00:00
|
|
|
pub fn raw(conn: &DbConn, msg: Option<(&'a str, &'a str)>) -> Context<'a> {
|
|
|
|
match Task::all(conn) {
|
2020-06-08 19:18:29 +00:00
|
|
|
Ok(tasks) => Context { msg, tasks },
|
2020-06-08 19:11:34 +00:00
|
|
|
Err(e) => {
|
|
|
|
error_!("DB Task::all() error: {}", e);
|
|
|
|
Context {
|
|
|
|
msg: Some(("error", "Couldn't access the task database.")),
|
|
|
|
tasks: vec![]
|
|
|
|
}
|
2020-02-13 04:11:51 +00:00
|
|
|
}
|
|
|
|
}
|
2016-09-22 11:12:07 +00:00
|
|
|
}
|
2016-07-16 04:09:08 +00:00
|
|
|
}
|
|
|
|
|
2016-10-12 07:14:42 +00:00
|
|
|
#[post("/", data = "<todo_form>")]
|
2018-07-21 22:11:08 +00:00
|
|
|
fn new(todo_form: Form<Todo>, conn: DbConn) -> Flash<Redirect> {
|
2016-10-12 07:14:42 +00:00
|
|
|
let todo = todo_form.into_inner();
|
2016-08-02 02:07:36 +00:00
|
|
|
if todo.description.is_empty() {
|
2016-10-12 07:14:42 +00:00
|
|
|
Flash::error(Redirect::to("/"), "Description cannot be empty.")
|
2020-02-13 04:11:51 +00:00
|
|
|
} else if let Err(e) = Task::insert(todo, &conn) {
|
2020-06-08 19:11:34 +00:00
|
|
|
error_!("DB insertion error: {}", e);
|
|
|
|
Flash::error(Redirect::to("/"), "Todo could not be inserted due an internal error.")
|
2016-08-02 02:07:36 +00:00
|
|
|
} else {
|
2020-02-13 04:11:51 +00:00
|
|
|
Flash::success(Redirect::to("/"), "Todo successfully added.")
|
2016-08-02 02:07:36 +00:00
|
|
|
}
|
|
|
|
}
|
2016-07-16 04:09:08 +00:00
|
|
|
|
2016-09-25 09:26:15 +00:00
|
|
|
#[put("/<id>")]
|
2018-07-21 22:11:08 +00:00
|
|
|
fn toggle(id: i32, conn: DbConn) -> Result<Redirect, Template> {
|
2020-06-08 19:11:34 +00:00
|
|
|
Task::toggle_with_id(id, &conn)
|
|
|
|
.map(|_| Redirect::to("/"))
|
|
|
|
.map_err(|e| {
|
|
|
|
error_!("DB toggle({}) error: {}", id, e);
|
|
|
|
Template::render("index", Context::err(&conn, "Failed to toggle task."))
|
|
|
|
})
|
2016-07-16 04:09:08 +00:00
|
|
|
}
|
|
|
|
|
2016-09-25 09:26:15 +00:00
|
|
|
#[delete("/<id>")]
|
2018-07-21 22:11:08 +00:00
|
|
|
fn delete(id: i32, conn: DbConn) -> Result<Flash<Redirect>, Template> {
|
2020-06-08 19:11:34 +00:00
|
|
|
Task::delete_with_id(id, &conn)
|
|
|
|
.map(|_| Flash::success(Redirect::to("/"), "Todo was deleted."))
|
|
|
|
.map_err(|e| {
|
|
|
|
error_!("DB deletion({}) error: {}", id, e);
|
|
|
|
Template::render("index", Context::err(&conn, "Failed to delete task."))
|
|
|
|
})
|
2016-07-16 04:09:08 +00:00
|
|
|
}
|
|
|
|
|
2016-09-04 11:06:28 +00:00
|
|
|
#[get("/")]
|
2019-06-13 02:41:29 +00:00
|
|
|
fn index(msg: Option<FlashMessage<'_, '_>>, conn: DbConn) -> Template {
|
2020-06-08 19:11:34 +00:00
|
|
|
Template::render("index", match msg {
|
2017-02-03 01:38:36 +00:00
|
|
|
Some(ref msg) => Context::raw(&conn, Some((msg.name(), msg.msg()))),
|
|
|
|
None => Context::raw(&conn, None),
|
2016-09-22 11:12:07 +00:00
|
|
|
})
|
2016-07-16 04:09:08 +00:00
|
|
|
}
|
|
|
|
|
2020-06-14 15:57:53 +00:00
|
|
|
async fn run_db_migrations(mut rocket: Rocket) -> Result<Rocket, Rocket> {
|
|
|
|
let conn = DbConn::get_one(rocket.inspect().await).expect("database connection");
|
2018-12-30 21:52:08 +00:00
|
|
|
match embedded_migrations::run(&*conn) {
|
|
|
|
Ok(()) => Ok(rocket),
|
|
|
|
Err(e) => {
|
|
|
|
error!("Failed to run database migrations: {:?}", e);
|
|
|
|
Err(rocket)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-22 23:10:02 +00:00
|
|
|
#[launch]
|
2018-12-30 21:52:08 +00:00
|
|
|
fn rocket() -> Rocket {
|
|
|
|
rocket::ignite()
|
2018-07-21 22:11:08 +00:00
|
|
|
.attach(DbConn::fairing())
|
2018-12-30 21:52:08 +00:00
|
|
|
.attach(AdHoc::on_attach("Database Migrations", run_db_migrations))
|
2018-08-24 21:00:36 +00:00
|
|
|
.mount("/", StaticFiles::from("static/"))
|
|
|
|
.mount("/", routes![index])
|
2018-07-30 01:24:25 +00:00
|
|
|
.mount("/todo", routes![new, toggle, delete])
|
2018-12-30 21:52:08 +00:00
|
|
|
.attach(Template::fairing())
|
2017-05-26 23:44:53 +00:00
|
|
|
}
|