Rocket/examples/todo/src/main.rs

120 lines
3.8 KiB
Rust
Raw Normal View History

#[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 rocket_contrib;
mod task;
2017-05-26 23:44:53 +00:00
#[cfg(test)] mod tests;
use std::fmt::Display;
2017-05-26 23:44:53 +00:00
use rocket::Rocket;
use rocket::fairing::AdHoc;
2016-11-02 17:49:06 +00:00
use rocket::request::{Form, FlashMessage};
use rocket::response::{Flash, Redirect};
use rocket_contrib::{templates::Template, serve::{StaticFiles, crate_relative}};
use diesel::SqliteConnection;
2017-02-03 01:38:36 +00:00
2019-06-13 02:41:29 +00:00
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, serde::Serialize)]
struct Context {
msg: Option<(String, String)>,
2020-06-08 19:11:34 +00:00
tasks: Vec<Task>
}
impl Context {
pub async fn err<M: Display>(conn: &DbConn, msg: M) -> Context {
Context {
msg: Some(("error".into(), msg.to_string())),
tasks: Task::all(conn).await.unwrap_or_default()
}
}
pub async fn raw(conn: &DbConn, msg: Option<(String, String)>) -> Context {
match Task::all(conn).await {
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".into(), "Fail to access database.".into())),
2020-06-08 19:11:34 +00:00
tasks: vec![]
}
}
}
}
}
#[post("/", data = "<todo_form>")]
async fn new(todo_form: Form<Todo>, conn: DbConn) -> Flash<Redirect> {
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 {
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.")
} else {
Flash::success(Redirect::to("/"), "Todo successfully added.")
}
}
#[put("/<id>")]
async fn toggle(id: i32, conn: DbConn) -> Result<Redirect, Template> {
match Task::toggle_with_id(id, &conn).await {
Ok(_) => Ok(Redirect::to("/")),
Err(e) => {
2020-06-08 19:11:34 +00:00
error_!("DB toggle({}) error: {}", id, e);
Err(Template::render("index", Context::err(&conn, "Failed to toggle task.").await))
}
}
}
#[delete("/<id>")]
async fn delete(id: i32, conn: DbConn) -> Result<Flash<Redirect>, Template> {
match Task::delete_with_id(id, &conn).await {
Ok(_) => Ok(Flash::success(Redirect::to("/"), "Todo was deleted.")),
Err(e) => {
2020-06-08 19:11:34 +00:00
error_!("DB deletion({}) error: {}", id, e);
Err(Template::render("index", Context::err(&conn, "Failed to delete task.").await))
}
}
}
2016-09-04 11:06:28 +00:00
#[get("/")]
async fn index(msg: Option<FlashMessage<'_, '_>>, conn: DbConn) -> Template {
let msg = msg.map(|m| (m.name().to_string(), m.msg().to_string()));
Template::render("index", Context::raw(&conn, msg).await)
}
async fn run_db_migrations(mut rocket: Rocket) -> Result<Rocket, Rocket> {
DbConn::get_one(rocket.inspect().await).await
.expect("database connection")
.run(|c| match embedded_migrations::run(c) {
Ok(()) => Ok(rocket),
Err(e) => {
error!("Failed to run database migrations: {:?}", e);
Err(rocket)
}
}).await
2018-12-30 21:52:08 +00:00
}
#[launch]
2018-12-30 21:52:08 +00:00
fn rocket() -> Rocket {
rocket::ignite()
.attach(DbConn::fairing())
2018-12-30 21:52:08 +00:00
.attach(AdHoc::on_attach("Database Migrations", run_db_migrations))
.mount("/", StaticFiles::from(crate_relative!("/static")))
2018-08-24 21:00:36 +00:00
.mount("/", routes![index])
.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
}