Files
web/src/routes/tracks.rs
T
2026-03-24 15:58:14 -04:00

108 lines
2.9 KiB
Rust

use actix_session::Session;
use actix_web::{HttpResponse, web};
use serde::Deserialize;
use shanty_db::queries;
use crate::auth;
use crate::error::ApiError;
use crate::state::AppState;
fn default_limit() -> u64 {
50
}
#[derive(Deserialize)]
pub struct SearchParams {
q: Option<String>,
#[serde(default = "default_limit")]
limit: u64,
#[serde(default)]
offset: u64,
}
#[derive(Deserialize)]
pub struct WatchTrackRequest {
artist: Option<String>,
title: Option<String>,
mbid: Option<String>,
}
pub fn configure(cfg: &mut web::ServiceConfig) {
cfg.service(
web::resource("/tracks/watch")
.route(web::post().to(watch_track))
.route(web::delete().to(unwatch_track)),
)
.service(web::resource("/tracks").route(web::get().to(list_tracks)))
.service(web::resource("/tracks/{id}").route(web::get().to(get_track)));
}
async fn list_tracks(
state: web::Data<AppState>,
session: Session,
query: web::Query<SearchParams>,
) -> Result<HttpResponse, ApiError> {
auth::require_auth(&session)?;
let tracks = if let Some(ref q) = query.q {
queries::tracks::search(state.db.conn(), q).await?
} else {
queries::tracks::list(state.db.conn(), query.limit, query.offset).await?
};
Ok(HttpResponse::Ok().json(tracks))
}
async fn get_track(
state: web::Data<AppState>,
session: Session,
path: web::Path<i32>,
) -> Result<HttpResponse, ApiError> {
auth::require_auth(&session)?;
let id = path.into_inner();
let track = queries::tracks::get_by_id(state.db.conn(), id).await?;
Ok(HttpResponse::Ok().json(track))
}
async fn watch_track(
state: web::Data<AppState>,
session: Session,
body: web::Json<WatchTrackRequest>,
) -> Result<HttpResponse, ApiError> {
let (user_id, _, _) = auth::require_auth(&session)?;
if body.title.is_none() && body.mbid.is_none() {
return Err(ApiError::BadRequest(
"provide title or recording mbid".into(),
));
}
let entry = shanty_watch::add_track(
state.db.conn(),
body.artist.as_deref(),
body.title.as_deref(),
body.mbid.as_deref(),
&state.mb_client,
Some(user_id),
)
.await?;
Ok(HttpResponse::Ok().json(serde_json::json!({
"id": entry.id,
"status": entry.status,
"name": entry.name,
"artist_name": entry.artist_name,
})))
}
async fn unwatch_track(
state: web::Data<AppState>,
session: Session,
body: web::Json<WatchTrackRequest>,
) -> Result<HttpResponse, ApiError> {
auth::require_auth(&session)?;
let mbid = body
.mbid
.as_deref()
.ok_or_else(|| ApiError::BadRequest("provide recording mbid".into()))?;
let removed = queries::wanted::remove_by_mbid(state.db.conn(), mbid).await?;
Ok(HttpResponse::Ok().json(serde_json::json!({"removed": removed})))
}