Initial commit

This commit is contained in:
Connor Johnstone
2026-03-17 21:56:12 -04:00
commit 50a0ddcdbc
16 changed files with 1110 additions and 0 deletions

78
src/routes/albums.rs Normal file
View File

@@ -0,0 +1,78 @@
use actix_web::{web, HttpResponse};
use serde::Deserialize;
use shanty_db::queries;
use crate::error::ApiError;
use crate::state::AppState;
#[derive(Deserialize)]
pub struct PaginationParams {
#[serde(default = "default_limit")]
limit: u64,
#[serde(default)]
offset: u64,
}
fn default_limit() -> u64 { 50 }
#[derive(Deserialize)]
pub struct AddAlbumRequest {
artist: Option<String>,
album: Option<String>,
mbid: Option<String>,
}
pub fn configure(cfg: &mut web::ServiceConfig) {
cfg.service(
web::resource("/albums")
.route(web::get().to(list_albums))
.route(web::post().to(add_album)),
)
.service(
web::resource("/albums/{id}")
.route(web::get().to(get_album)),
);
}
async fn list_albums(
state: web::Data<AppState>,
query: web::Query<PaginationParams>,
) -> Result<HttpResponse, ApiError> {
let albums = queries::albums::list(state.db.conn(), query.limit, query.offset).await?;
Ok(HttpResponse::Ok().json(albums))
}
async fn get_album(
state: web::Data<AppState>,
path: web::Path<i32>,
) -> Result<HttpResponse, ApiError> {
let id = path.into_inner();
let album = queries::albums::get_by_id(state.db.conn(), id).await?;
let tracks = queries::tracks::get_by_album(state.db.conn(), id).await?;
Ok(HttpResponse::Ok().json(serde_json::json!({
"album": album,
"tracks": tracks,
})))
}
async fn add_album(
state: web::Data<AppState>,
body: web::Json<AddAlbumRequest>,
) -> Result<HttpResponse, ApiError> {
if body.artist.is_none() && body.album.is_none() && body.mbid.is_none() {
return Err(ApiError::BadRequest("provide artist+album or mbid".into()));
}
let summary = shanty_watch::add_album(
state.db.conn(),
body.artist.as_deref(),
body.album.as_deref(),
body.mbid.as_deref(),
&state.mb_client,
)
.await?;
Ok(HttpResponse::Ok().json(serde_json::json!({
"tracks_added": summary.tracks_added,
"tracks_already_owned": summary.tracks_already_owned,
"errors": summary.errors,
})))
}