use axum::headers::HeaderMap; use axum::Json; use serde::{Deserialize, Serialize}; use std::sync::Arc; use axum::extract::State; use serde_json::{json, Value}; use tracing::info; use crate::auth::auth; use crate::config::SETTINGS; use crate::wol::{create_buffer, send_packet}; use crate::db::Device; use crate::error::WebolError; pub async fn start(State(state): State>, headers: HeaderMap, Json(payload): Json) -> Result, WebolError> { info!("POST request"); let secret = headers.get("authorization"); let authorized = auth(secret).map_err(WebolError::Auth)?; if authorized { let device = sqlx::query_as!( Device, r#" SELECT id, mac, broadcast_addr FROM devices WHERE id = $1; "#, payload.id ).fetch_one(&state.db).await.map_err(WebolError::DB)?; info!("starting {}", device.id); let bind_addr = SETTINGS .get_string("bindaddr") .unwrap_or("0.0.0.0:1111".to_string()); let _ = send_packet( &bind_addr.parse().map_err(WebolError::IpParse)?, &device.broadcast_addr.parse().map_err(WebolError::IpParse)?, create_buffer(&device.mac)? )?; if payload.ping.is_some_and(|ping| ping) { tokio::spawn(async move {crate::services::ping::spawn(state.ping_send.clone()).await}); } Ok(Json(json!(StartResponse { id: device.id, boot: true }))) } else { Err(WebolError::Generic) } } #[derive(Deserialize)] pub struct StartPayload { id: String, ping: Option, } #[derive(Serialize)] struct StartResponse { id: String, boot: bool, }