Add routes for creating and deleting favorites

This commit is contained in:
Reimar 2024-08-16 12:26:26 +02:00
parent f82ba23c5d
commit bb44d2bc56
Signed by: Reimar
GPG Key ID: 93549FA07F0AE268

View File

@ -1,10 +1,11 @@
mod auth; mod auth;
mod models; mod models;
use actix_web::{get, Responder, HttpResponse, HttpServer, App, web}; use actix_web::{get, post, delete, Responder, HttpResponse, HttpServer, App, web};
use std::sync::{Mutex, MutexGuard, Arc}; use std::sync::{Mutex, MutexGuard, Arc};
use auth::AuthorizedUser; use auth::AuthorizedUser;
use models::Favorite; use models::Favorite;
use serde::Deserialize;
mod embedded { mod embedded {
use refinery::embed_migrations; use refinery::embed_migrations;
@ -51,6 +52,49 @@ fn get_favorites(db: MutexGuard<'_, rusqlite::Connection>, user_id: String) -> O
) )
} }
#[derive(Deserialize)]
struct CreateFavoriteRequest {
lat: f64,
lng: f64,
}
#[post("/favorites")]
async fn create_favorite(auth: AuthorizedUser, data: web::Data<AppData>, input: web::Json<CreateFavoriteRequest>) -> impl Responder {
let db = data.database.lock().unwrap();
match db.execute(
"INSERT INTO favorites (user_id, lat, lng) VALUES (:user_id, :lat, :lng)",
&[(":user_id", &auth.user_id), (":lat", &input.lat.to_string()), (":lng", &input.lng.to_string())]
) {
Ok(_) => HttpResponse::Created(),
Err(_) => HttpResponse::InternalServerError(),
}
}
#[delete("/favorites/{favorite}")]
async fn delete_favorite(auth: AuthorizedUser, data:web::Data<AppData>, path: web::Path<usize>) -> impl Responder {
let db = data.database.lock().unwrap();
let favorite_id = path.into_inner();
let params = &[(":id", &favorite_id.to_string())];
let result = db.query_row("SELECT * FROM favorites WHERE id = :id LIMIT 1", params, |row| Favorite::from_row(row));
if result.is_err() {
return HttpResponse::InternalServerError().finish();
}
let favorite = result.unwrap();
if favorite.user_id != auth.user_id {
return HttpResponse::Forbidden().body("Cannot remove favorite that you did not create");
}
match db.execute("DELETE FROM favorites WHERE id = :id", params) {
Ok(_) => HttpResponse::NoContent().finish(),
Err(_) => HttpResponse::InternalServerError().finish(),
}
}
#[actix_web::main] #[actix_web::main]
async fn main() -> std::io::Result<()> { async fn main() -> std::io::Result<()> {
let _ = dotenvy::dotenv(); let _ = dotenvy::dotenv();
@ -81,6 +125,8 @@ async fn main() -> std::io::Result<()> {
.service(healthcheck) .service(healthcheck)
.service(authorized) .service(authorized)
.service(favorites) .service(favorites)
.service(create_favorite)
.service(delete_favorite)
}) })
.bind(("0.0.0.0", port))? .bind(("0.0.0.0", port))?
.run() .run()