use crate::db::Device;
use crate::error::Error;
use axum::extract::{Path, State};
use axum::Json;
use mac_address::MacAddress;
use serde::Deserialize;
use serde_json::{json, Value};
use sqlx::types::ipnetwork::IpNetwork;
use std::{str::FromStr, sync::Arc};
use tracing::{debug, info};
use utoipa::ToSchema;
#[utoipa::path(
get,
path = "/device",
request_body = GetDevicePayload,
responses(
(status = 200, description = "Get `Device` information", body = [Device])
),
security(("api_key" = []))
)]
#[deprecated]
pub async fn get_payload(
State(state): State<Arc<crate::AppState>>,
Json(payload): Json<GetDevicePayload>,
) -> Result<Json<Value>, Error> {
info!("get device {}", payload.id);
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?;
debug!("got device {:?}", device);
Ok(Json(json!(device)))
}
#[utoipa::path(
get,
path = "/device/{id}",
responses(
(status = 200, description = "Get `Device` information", body = [Device])
),
params(
("id" = String, Path, description = "device id")
),
security((), ("api_key" = []))
)]
pub async fn get(
State(state): State<Arc<crate::AppState>>,
Path(path): Path<String>,
) -> Result<Json<Value>, Error> {
info!("get device from path {}", path);
let device = sqlx::query_as!(
Device,
r#"
SELECT id, mac, broadcast_addr, ip, times
FROM devices
WHERE id = $1;
"#,
path
)
.fetch_one(&state.db)
.await?;
debug!("got device {:?}", device);
Ok(Json(json!(device)))
}
#[derive(Deserialize, ToSchema)]
#[deprecated]
pub struct GetDevicePayload {
id: String,
}
#[derive(Deserialize, ToSchema)]
pub struct DevicePayload {
id: String,
mac: String,
broadcast_addr: String,
ip: String,
}
#[utoipa::path(
put,
path = "/device",
request_body = DevicePayload,
responses(
(status = 200, description = "add device to storage", body = [DeviceSchema])
),
security((), ("api_key" = []))
)]
pub async fn put(
State(state): State<Arc<crate::AppState>>,
Json(payload): Json<DevicePayload>,
) -> Result<Json<Value>, Error> {
info!(
"add device {} ({}, {}, {})",
payload.id, payload.mac, payload.broadcast_addr, payload.ip
);
let ip = IpNetwork::from_str(&payload.ip)?;
let mac = MacAddress::from_str(&payload.mac)?;
let device = sqlx::query_as!(
Device,
r#"
INSERT INTO devices (id, mac, broadcast_addr, ip)
VALUES ($1, $2, $3, $4)
RETURNING id, mac, broadcast_addr, ip, times;
"#,
payload.id,
mac,
payload.broadcast_addr,
ip
)
.fetch_one(&state.db)
.await?;
Ok(Json(json!(device)))
}
#[utoipa::path(
post,
path = "/device",
request_body = DevicePayload,
responses(
(status = 200, description = "update device in storage", body = [DeviceSchema])
),
security((), ("api_key" = []))
)]
pub async fn post(
State(state): State<Arc<crate::AppState>>,
Json(payload): Json<DevicePayload>,
) -> Result<Json<Value>, Error> {
info!(
"edit device {} ({}, {}, {})",
payload.id, payload.mac, payload.broadcast_addr, payload.ip
);
let ip = IpNetwork::from_str(&payload.ip)?;
let mac = MacAddress::from_str(&payload.mac)?;
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;
"#,
mac,
payload.broadcast_addr,
ip,
payload.id
)
.fetch_one(&state.db)
.await?;
Ok(Json(json!(device)))
}