Pulling data off KUma status page correctly
This commit is contained in:
56
src/api.rs
56
src/api.rs
@@ -1,6 +1,8 @@
|
|||||||
use serde_json::Value;
|
use serde_json::Value;
|
||||||
|
use serde::Serialize;
|
||||||
use reqwest::{self, Error};
|
use reqwest::{self, Error};
|
||||||
|
|
||||||
|
/// Target base URL to use for all KUMA API call's
|
||||||
pub const BASE_URL: &str = "https://uptime.shockrah.xyz";
|
pub const BASE_URL: &str = "https://uptime.shockrah.xyz";
|
||||||
|
|
||||||
|
|
||||||
@@ -14,21 +16,30 @@ macro_rules! endpoints {
|
|||||||
($slug:expr) => { format!("{}/api/status-page/{}", crate::api::BASE_URL, $slug) }
|
($slug:expr) => { format!("{}/api/status-page/{}", crate::api::BASE_URL, $slug) }
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
/// A single heartbeat within a monitor's latest status
|
||||||
|
#[derive(Debug, Serialize)]
|
||||||
pub struct HeartBeat {
|
pub struct HeartBeat {
|
||||||
status: i64,
|
status: i64,
|
||||||
time: String,
|
time: String,
|
||||||
msg: String,
|
msg: String,
|
||||||
|
ping: i64,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
/// A monitor which contains the recent heartbeats for that monitor
|
||||||
|
/// Heartbeat list themselves will always be populated and can be configured
|
||||||
|
/// to have a maximum size with the MAX_HEARTBEATS env variable
|
||||||
|
#[derive(Debug, Serialize)]
|
||||||
pub struct KumaMonitor {
|
pub struct KumaMonitor {
|
||||||
id: i64,
|
id: i64,
|
||||||
name: String,
|
name: String,
|
||||||
heartbeats: Option<Vec<HeartBeat>>
|
// TODO: make sure we support MAX_HEARTBEATS somehow
|
||||||
|
heartbeats: Vec<HeartBeat>
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
/// The overall look of the Kuma status page
|
||||||
|
/// For now we only have pub however we can control which we are loking at
|
||||||
|
/// with the use of the `slug` field.
|
||||||
|
#[derive(Debug, Serialize)]
|
||||||
pub struct KumaStatusPage {
|
pub struct KumaStatusPage {
|
||||||
slug: String,
|
slug: String,
|
||||||
title: String,
|
title: String,
|
||||||
@@ -37,22 +48,37 @@ pub struct KumaStatusPage {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl KumaMonitor {
|
impl KumaMonitor {
|
||||||
pub fn blank(val: &Value) -> Self {
|
pub async fn new(val: &Value) -> Result<Self, Error> {
|
||||||
Self {
|
// Populate the monitor with it's respective heartbeats at that time
|
||||||
id: val["id"].as_i64().unwrap_or(0),
|
let id = val["id"].as_i64().unwrap_or(0);
|
||||||
name: val["name"].to_string(),
|
let name = val["name"].to_string();
|
||||||
heartbeats: None
|
let response: Value = reqwest::get(heartbeat!("pub")).await?.json().await?;
|
||||||
|
if let Some(list) = &response["heartbeatList"][id.to_string()].as_array() {
|
||||||
|
let heartbeats = list.iter().map(|item| {
|
||||||
|
HeartBeat {
|
||||||
|
status: item["status"].as_i64().unwrap_or(-1),
|
||||||
|
time: item["time"].to_string().replace("\"", ""),
|
||||||
|
msg: item["msg"].to_string().replace("\"", ""),
|
||||||
|
ping: item["ping"].as_i64().unwrap_or(-1),
|
||||||
}
|
}
|
||||||
|
}).collect();
|
||||||
|
return Ok(Self { id, name, heartbeats })
|
||||||
|
}
|
||||||
|
// TODO: wtf should a default response be?
|
||||||
|
// temporary blank response for now
|
||||||
|
Ok(Self { id, name, heartbeats: vec![] })
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
impl KumaStatusPage {
|
impl KumaStatusPage {
|
||||||
fn get_monitors(json: &Value) -> Vec<KumaMonitor> {
|
async fn get_monitors(json: &Value) -> Vec<KumaMonitor> {
|
||||||
let mut monitors = vec![];
|
let mut monitors = vec![];
|
||||||
for group in json["publicGroupList"].as_array().unwrap_or(&vec![]) {
|
for group in json["publicGroupList"].as_array().unwrap_or(&vec![]) {
|
||||||
for monitor in group["monitorList"].as_array().unwrap_or(&vec![]) {
|
for monitor in group["monitorList"].as_array().unwrap_or(&vec![]) {
|
||||||
monitors.push(KumaMonitor::blank(&monitor));
|
if let Ok(mon) = KumaMonitor::new(&monitor).await {
|
||||||
|
monitors.push(mon);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return monitors;
|
return monitors;
|
||||||
@@ -62,10 +88,10 @@ impl KumaStatusPage {
|
|||||||
let endpoint = endpoints!(slug);
|
let endpoint = endpoints!(slug);
|
||||||
let resp: Value = reqwest::get(&endpoint).await?.json().await?;
|
let resp: Value = reqwest::get(&endpoint).await?.json().await?;
|
||||||
return Ok(KumaStatusPage {
|
return Ok(KumaStatusPage {
|
||||||
slug: resp["config"]["slug"].to_string(),
|
slug: resp["config"]["slug"].as_str().unwrap().into(),
|
||||||
title: resp["config"]["title"].to_string(),
|
title: resp["config"]["title"].as_str().unwrap().into(),
|
||||||
description: resp["config"]["description"].to_string(),
|
description: resp["config"]["description"].as_str().unwrap().into(),
|
||||||
monitors: KumaStatusPage::get_monitors(&resp)
|
monitors: KumaStatusPage::get_monitors(&resp).await
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,12 +3,16 @@ mod api;
|
|||||||
|
|
||||||
use reqwest::{self, Error};
|
use reqwest::{self, Error};
|
||||||
use crate::api::KumaStatusPage;
|
use crate::api::KumaStatusPage;
|
||||||
|
use serde_json;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
#[tokio::main]
|
#[tokio::main]
|
||||||
async fn main() -> Result<(), Error> {
|
async fn main() -> Result<(), Error> {
|
||||||
let page = KumaStatusPage::get("pub").await?;
|
let public = KumaStatusPage::get("pub").await?;
|
||||||
println!("{page:?}");
|
match serde_json::to_string(&public) {
|
||||||
|
Ok(result) => println!("{result}"),
|
||||||
|
Err(_) => eprintln!("bruh")
|
||||||
|
};
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user