1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
|
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;
#[axum_macros::debug_handler]
pub async fn start(
State(state): State<Arc<crate::AppState>>,
headers: HeaderMap,
Json(payload): Json<Payload>,
) -> Result<Json<Value>, Error> {
info!("POST request");
let secret = headers.get("authorization");
let authorized = auth(&state.config, secret).map_err(Error::Auth)?;
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
.map_err(Error::DB)?;
info!("starting {}", device.id);
let bind_addr = "0.0.0.0:0";
let _ = send_packet(
&bind_addr.parse().map_err(Error::IpParse)?,
&device.broadcast_addr.parse().map_err(Error::IpParse)?,
&create_buffer(&device.mac)?,
)?;
let dev_id = device.id.clone();
let uuid = if payload.ping.is_some_and(|ping| ping) {
let mut uuid: Option<String> = 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_genc = uuid_gen.clone();
tokio::spawn(async move {
debug!("init ping service");
state.ping_map.insert(
uuid_gen.clone(),
PingValue {
ip: device.ip.clone(),
online: false,
},
);
crate::services::ping::spawn(
state.ping_send.clone(),
&state.config,
device,
uuid_gen.clone(),
&state.ping_map,
&state.db,
)
.await;
});
Some(uuid_genc)
} else {
None
};
Ok(Json(json!(Response {
id: dev_id,
boot: true,
uuid
})))
} else {
Err(Error::Generic)
}
}
#[derive(Deserialize)]
pub struct Payload {
id: String,
ping: Option<bool>,
}
#[derive(Serialize)]
struct Response {
id: String,
boot: bool,
uuid: Option<String>,
}
|