use std::sync::Arc; use axum::extract::State; use axum::Json; use axum::http::HeaderMap; use serde::{Deserialize, Serialize}; use serde_json::{json, Value}; use tracing::{debug, info}; use crate::auth::auth; use crate::db::Device; use crate::error::Error; pub async fn get(State(state): State<Arc<crate::AppState>>, headers: HeaderMap, Json(payload): Json<GetDevicePayload>) -> Result<Json<Value>, Error> { info!("add device {}", payload.id); let secret = headers.get("authorization"); if auth(&state.config, secret).map_err(Error::Auth)? { let device = sqlx::query_as!( Device, r#" SELECT id, mac, broadcast_addr, ip, times FROM devices WHERE id = $1; "#, payload.id ).fetch_one(&state.db).await.map_err(Error::DB)?; debug!("got device {:?}", device); Ok(Json(json!(device))) } else { Err(Error::Generic) } } #[derive(Deserialize)] pub struct GetDevicePayload { id: String, } pub async fn put(State(state): State<Arc<crate::AppState>>, headers: HeaderMap, Json(payload): Json<PutDevicePayload>) -> Result<Json<Value>, Error> { info!("add device {} ({}, {}, {})", payload.id, payload.mac, payload.broadcast_addr, payload.ip); let secret = headers.get("authorization"); if auth(&state.config, secret).map_err(Error::Auth)? { sqlx::query!( r#" INSERT INTO devices (id, mac, broadcast_addr, ip) VALUES ($1, $2, $3, $4); "#, payload.id, payload.mac, payload.broadcast_addr, payload.ip ).execute(&state.db).await.map_err(Error::DB)?; Ok(Json(json!(PutDeviceResponse { success: true }))) } else { Err(Error::Generic) } } #[derive(Deserialize)] pub struct PutDevicePayload { id: String, mac: String, broadcast_addr: String, ip: String } #[derive(Serialize)] pub struct PutDeviceResponse { success: bool } pub async fn post(State(state): State<Arc<crate::AppState>>, headers: HeaderMap, Json(payload): Json<PostDevicePayload>) -> Result<Json<Value>, Error> { info!("edit device {} ({}, {}, {})", payload.id, payload.mac, payload.broadcast_addr, payload.ip); let secret = headers.get("authorization"); if auth(&state.config, secret).map_err(Error::Auth)? { let device = sqlx::query_as!( Device, r#" UPDATE devices SET mac = $1, broadcast_addr = $2, ip = $3 WHERE id = $4 RETURNING id, mac, broadcast_addr, ip, times; "#, payload.mac, payload.broadcast_addr, payload.ip, payload.id ).fetch_one(&state.db).await.map_err(Error::DB)?; Ok(Json(json!(device))) } else { Err(Error::Generic) } } #[derive(Deserialize)] pub struct PostDevicePayload { id: String, mac: String, broadcast_addr: String, ip: String, }