use crate::auth::auth; use crate::db::Device; use crate::error::Error; use crate::services::ping::Value as PingValue; use crate::wol::{create_buffer, send_packet}; use axum::extract::State; use axum::http::HeaderMap; use axum::Json; use serde::{Deserialize, Serialize}; use serde_json::{json, Value}; use std::sync::Arc; use tracing::{debug, info}; use uuid::Uuid; pub async fn start( State(state): State>, headers: HeaderMap, Json(payload): Json, ) -> Result, Error> { info!("POST request"); let secret = headers.get("authorization"); let authorized = matches!(auth(&state.config, secret)?, crate::auth::Response::Success); if authorized { 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?; info!("starting {}", device.id); let bind_addr = "0.0.0.0:0"; let _ = send_packet( bind_addr, &device.broadcast_addr, &create_buffer(&device.mac.to_string())?, )?; let dev_id = device.id.clone(); let uuid = if payload.ping.is_some_and(|ping| ping) { Some(setup_ping(state, device)) } else { None }; Ok(Json(json!(Response { id: dev_id, boot: true, uuid }))) } else { Err(Error::Generic) } } fn setup_ping(state: Arc, device: Device) -> String { let mut uuid: Option = None; for (key, value) in state.ping_map.clone() { if value.ip == device.ip { debug!("service already exists"); uuid = Some(key); break; } } let uuid_gen = match uuid { Some(u) => u, None => Uuid::new_v4().to_string(), }; let uuid_ret = uuid_gen.clone(); debug!("init ping service"); state.ping_map.insert( uuid_gen.clone(), PingValue { ip: device.ip, online: false, }, ); tokio::spawn(async move { crate::services::ping::spawn( state.ping_send.clone(), &state.config, device, uuid_gen, &state.ping_map, &state.db, ) .await; }); uuid_ret } #[derive(Deserialize)] pub struct Payload { id: String, ping: Option, } #[derive(Serialize)] struct Response { id: String, boot: bool, uuid: Option, }