merge direct_call_net_stats branch

This commit is contained in:
2022-01-12 11:35:55 +02:00
5 changed files with 404 additions and 410 deletions

578
Cargo.lock generated

File diff suppressed because it is too large Load Diff

View File

@ -41,6 +41,8 @@ log = "0.4"
nest = "1.0.0" nest = "1.0.0"
openssl = { version = "0.10", features = ["vendored"] } openssl = { version = "0.10", features = ["vendored"] }
peach-lib = { path = "../peach-lib" } peach-lib = { path = "../peach-lib" }
peach-network = { path = "../peach-network", features = ["serde_support"] }
peach-stats = { path = "../peach-stats", features = ["serde_support"] }
percent-encoding = "2.1.0" percent-encoding = "2.1.0"
regex = "1" regex = "1"
rocket = { version = "0.5.0-rc.1", features = ["json", "secrets"] } rocket = { version = "0.5.0-rc.1", features = ["json", "secrets"] }

View File

@ -3,8 +3,8 @@ use rocket::{
get, post, get, post,
request::FlashMessage, request::FlashMessage,
response::{Flash, Redirect}, response::{Flash, Redirect},
serde::json::Value,
}; };
use rocket_dyn_templates::Template; use rocket_dyn_templates::Template;
use serde::Serialize; use serde::Serialize;
use std::{ use std::{
@ -12,13 +12,16 @@ use std::{
process::{Command, Output}, process::{Command, Output},
}; };
use peach_lib::config_manager::load_peach_config; use peach_lib::{
use peach_lib::stats_client::{CpuStatPercentages, DiskUsage, LoadAverage, MemStat}; config_manager::load_peach_config, dyndns_client, network_client, oled_client, sbot_client,
use peach_lib::{dyndns_client, network_client, oled_client, sbot_client, stats_client}; };
use peach_stats::{
stats,
stats::{CpuStatPercentages, DiskUsage, LoadAverage, MemStat},
};
use crate::routes::authentication::Authenticated; use crate::routes::authentication::Authenticated;
use crate::utils::build_json_response; use crate::utils::build_json_response;
use rocket::serde::json::Value;
// HELPERS AND ROUTES FOR /status // HELPERS AND ROUTES FOR /status
@ -34,7 +37,6 @@ pub struct StatusContext {
pub mem_stats: Option<MemStat>, pub mem_stats: Option<MemStat>,
pub network_ping: String, pub network_ping: String,
pub oled_ping: String, pub oled_ping: String,
pub stats_ping: String,
pub dyndns_enabled: bool, pub dyndns_enabled: bool,
pub dyndns_is_online: bool, pub dyndns_is_online: bool,
pub config_is_valid: bool, pub config_is_valid: bool,
@ -46,9 +48,12 @@ pub struct StatusContext {
impl StatusContext { impl StatusContext {
pub fn build() -> StatusContext { pub fn build() -> StatusContext {
// convert result to Option<CpuStatPercentages>, discard any error // convert result to Option<CpuStatPercentages>, discard any error
let cpu_stat_percent = stats_client::cpu_stats_percent().ok(); let cpu_stat_percent = stats::cpu_stats_percent().ok();
let load_average = stats_client::load_average().ok(); let load_average = stats::load_average().ok();
let mem_stats = stats_client::mem_stats().ok(); let mem_stats = stats::mem_stats().ok();
// TODO: add `wpa_supplicant_status` to peach_network to replace this ping call
// instead of: "is the network json-rpc server running?", we want to ask:
// "is the wpa_supplicant systemd service functioning correctly?"
let network_ping = match network_client::ping() { let network_ping = match network_client::ping() {
Ok(_) => "ONLINE".to_string(), Ok(_) => "ONLINE".to_string(),
Err(_) => "OFFLINE".to_string(), Err(_) => "OFFLINE".to_string(),
@ -57,22 +62,21 @@ impl StatusContext {
Ok(_) => "ONLINE".to_string(), Ok(_) => "ONLINE".to_string(),
Err(_) => "OFFLINE".to_string(), Err(_) => "OFFLINE".to_string(),
}; };
let stats_ping = match stats_client::ping() {
Ok(_) => "ONLINE".to_string(), let uptime = match stats::uptime() {
Err(_) => "OFFLINE".to_string(), Ok(secs) => {
}; let uptime_mins = secs / 60;
let uptime = match stats_client::uptime() { uptime_mins.to_string()
Ok(mins) => mins, }
Err(_) => "Unavailable".to_string(), Err(_) => "Unavailable".to_string(),
}; };
// parse the uptime string to a signed integer (for math)
let uptime_parsed = uptime.parse::<i32>().ok();
// serialize disk usage data into Vec<DiskUsage> // serialize disk usage data into Vec<DiskUsage>
let disk_usage_stats = match stats_client::disk_usage() { let disk_usage_stats = match stats::disk_usage() {
Ok(disks) => { Ok(disks) => disks,
let partitions: Vec<DiskUsage> = serde_json::from_str(disks.as_str())
.expect("Failed to deserialize disk_usage response");
partitions
}
Err(_) => Vec::new(), Err(_) => Vec::new(),
}; };
@ -84,9 +88,6 @@ impl StatusContext {
} }
} }
// parse the uptime string to a signed integer (for math)
let uptime_parsed = uptime.parse::<i32>().ok();
// dyndns_is_online & config_is_valid // dyndns_is_online & config_is_valid
let dyndns_enabled: bool; let dyndns_enabled: bool;
let dyndns_is_online: bool; let dyndns_is_online: bool;
@ -139,7 +140,6 @@ impl StatusContext {
mem_stats, mem_stats,
network_ping, network_ping,
oled_ping, oled_ping,
stats_ping,
dyndns_enabled, dyndns_enabled,
dyndns_is_online, dyndns_is_online,
config_is_valid, config_is_valid,

View File

@ -2,25 +2,77 @@ use rocket::{get, request::FlashMessage};
use rocket_dyn_templates::Template; use rocket_dyn_templates::Template;
use serde::Serialize; use serde::Serialize;
use peach_lib::network_client; use peach_network::{
use peach_lib::stats_client::Traffic; network,
network::{Status, Traffic},
};
use crate::routes::authentication::Authenticated; use crate::routes::authentication::Authenticated;
// HELPERS AND ROUTES FOR /status/network // HELPERS AND ROUTES FOR /status/network
#[derive(Debug, Serialize)]
pub struct IfaceTraffic {
pub rx: u64,
pub rx_unit: Option<String>,
pub tx: u64,
pub tx_unit: Option<String>,
}
impl IfaceTraffic {
fn default() -> Self {
IfaceTraffic {
rx: 0,
rx_unit: None,
tx: 0,
tx_unit: None,
}
}
}
fn convert_traffic(traffic: Traffic) -> Option<IfaceTraffic> {
let mut t = IfaceTraffic::default();
// modify traffic values & assign measurement units
// based on received and transmitted values.
// if received > 999 MB, convert it to GB
if traffic.received > 1_047_527_424 {
t.rx = traffic.received / 1_073_741_824;
t.rx_unit = Some("GB".to_string());
} else if traffic.received > 0 {
// otherwise, convert it to MB
t.rx = (traffic.received / 1024) / 1024;
t.rx_unit = Some("MB".to_string());
} else {
t.rx = 0;
t.rx_unit = Some("MB".to_string());
}
if traffic.transmitted > 1_047_527_424 {
t.tx = traffic.transmitted / 1_073_741_824;
t.tx_unit = Some("GB".to_string());
} else if traffic.transmitted > 0 {
t.tx = (traffic.transmitted / 1024) / 1024;
t.tx_unit = Some("MB".to_string());
} else {
t.tx = 0;
t.tx_unit = Some("MB".to_string());
}
Some(t)
}
#[derive(Debug, Serialize)] #[derive(Debug, Serialize)]
pub struct NetworkContext { pub struct NetworkContext {
pub ap_ip: String, pub ap_ip: String,
pub ap_ssid: String, pub ap_ssid: String,
pub ap_state: String, pub ap_state: String,
pub ap_traffic: Option<Traffic>, pub ap_traffic: Option<IfaceTraffic>,
pub wlan_ip: String, pub wlan_ip: String,
pub wlan_rssi: Option<String>, pub wlan_rssi: Option<String>,
pub wlan_ssid: String, pub wlan_ssid: String,
pub wlan_state: String, pub wlan_state: String,
pub wlan_status: String, pub wlan_status: Option<Status>,
pub wlan_traffic: Option<Traffic>, pub wlan_traffic: Option<IfaceTraffic>,
pub flash_name: Option<String>, pub flash_name: Option<String>,
pub flash_msg: Option<String>, pub flash_msg: Option<String>,
// page title for header in navbar // page title for header in navbar
@ -31,101 +83,47 @@ pub struct NetworkContext {
impl NetworkContext { impl NetworkContext {
pub fn build() -> NetworkContext { pub fn build() -> NetworkContext {
let ap_ip = match network_client::ip("ap0") { let ap_ip = match network::ip("ap0") {
Ok(ip) => ip, Ok(Some(ip)) => ip,
Err(_) => "x.x.x.x".to_string(), _ => "x.x.x.x".to_string(),
}; };
let ap_ssid = match network_client::ssid("ap0") { let ap_ssid = match network::ssid("ap0") {
Ok(ssid) => ssid, Ok(Some(ssid)) => ssid,
Err(_) => "Not currently activated".to_string(), _ => "Not currently activated".to_string(),
}; };
let ap_state = match network_client::state("ap0") { let ap_state = match network::state("ap0") {
Ok(state) => state, Ok(Some(state)) => state,
Err(_) => "Interface unavailable".to_string(), _ => "Interface unavailable".to_string(),
}; };
let ap_traffic = match network_client::traffic("ap0") { let ap_traffic = match network::traffic("ap0") {
Ok(traffic) => { // convert bytes to mb or gb and add appropriate units
let mut t = traffic; Ok(Some(traffic)) => convert_traffic(traffic),
// modify traffic values & assign measurement unit _ => None,
// based on received and transmitted values
// if received > 999 MB, convert it to GB
if t.received > 1_047_527_424 {
t.received /= 1_073_741_824;
t.rx_unit = Some("GB".to_string());
} else if t.received > 0 {
// otherwise, convert it to MB
t.received = (t.received / 1024) / 1024;
t.rx_unit = Some("MB".to_string());
} else {
t.received = 0;
t.rx_unit = Some("MB".to_string());
}
if t.transmitted > 1_047_527_424 {
t.transmitted /= 1_073_741_824;
t.tx_unit = Some("GB".to_string());
} else if t.transmitted > 0 {
t.transmitted = (t.transmitted / 1024) / 1024;
t.tx_unit = Some("MB".to_string());
} else {
t.transmitted = 0;
t.tx_unit = Some("MB".to_string());
}
Some(t)
}
Err(_) => None,
}; };
let wlan_ip = match network_client::ip("wlan0") { let wlan_ip = match network::ip("wlan0") {
Ok(ip) => ip, Ok(Some(ip)) => ip,
Err(_) => "x.x.x.x".to_string(), _ => "x.x.x.x".to_string(),
}; };
let wlan_rssi = match network_client::rssi_percent("wlan0") { let wlan_rssi = match network::rssi_percent("wlan0") {
Ok(rssi) => Some(rssi), Ok(rssi) => rssi,
Err(_) => None, _ => None,
}; };
let wlan_ssid = match network_client::ssid("wlan0") { let wlan_ssid = match network::ssid("wlan0") {
Ok(ssid) => ssid, Ok(Some(ssid)) => ssid,
Err(_) => "Not connected".to_string(), _ => "Not connected".to_string(),
}; };
let wlan_state = match network_client::state("wlan0") { let wlan_state = match network::state("wlan0") {
Ok(state) => state, Ok(Some(state)) => state,
Err(_) => "Interface unavailable".to_string(), _ => "Interface unavailable".to_string(),
}; };
let wlan_status = match network_client::status("wlan0") { let wlan_status = match network::status("wlan0") {
Ok(status) => status, Ok(status) => status,
Err(_) => "Interface unavailable".to_string(), _ => None,
}; };
let wlan_traffic = match network_client::traffic("wlan0") { let wlan_traffic = match network::traffic("wlan0") {
Ok(traffic) => { // convert bytes to mb or gb and add appropriate units
let mut t = traffic; Ok(Some(traffic)) => convert_traffic(traffic),
// modify traffic values & assign measurement unit _ => None,
// based on received and transmitted values
// if received > 999 MB, convert it to GB
if t.received > 1_047_527_424 {
t.received /= 1_073_741_824;
t.rx_unit = Some("GB".to_string());
} else if t.received > 0 {
// otherwise, convert it to MB
t.received = (t.received / 1024) / 1024;
t.rx_unit = Some("MB".to_string());
} else {
t.received = 0;
t.rx_unit = Some("MB".to_string());
}
if t.transmitted > 1_047_527_424 {
t.transmitted /= 1_073_741_824;
t.tx_unit = Some("GB".to_string());
} else if t.transmitted > 0 {
t.transmitted = (t.transmitted / 1024) / 1024;
t.tx_unit = Some("MB".to_string());
} else {
t.transmitted = 0;
t.tx_unit = Some("MB".to_string());
}
Some(t)
}
Err(_) => None,
}; };
NetworkContext { NetworkContext {

View File

@ -42,11 +42,11 @@
</div> </div>
</div> </div>
<!-- PEACH-STATS STATUS STACK --> <!-- PEACH-STATS STATUS STACK -->
<div class="stack capsule{% if stats_ping == "ONLINE" %} success-border{% else %} warning-border{% endif %}"> <div class="stack capsule success-border">
<img id="statsIcon" class="icon{% if stats_ping == "OFFLINE" %} icon-inactive{% endif %} icon-medium" alt="Stats" title="System statistics microservice status" src="/icons/chart.svg"> <img id="statsIcon" class="icon icon-medium" alt="Stats" title="System statistics microservice status" src="/icons/chart.svg">
<div class="stack" style="padding-top: 0.5rem;"> <div class="stack" style="padding-top: 0.5rem;">
<label class="label-small font-near-black">Statistics</label> <label class="label-small font-near-black">Statistics</label>
<label class="label-small font-near-black">{{ stats_ping }}</label> <label class="label-small font-near-black">AVAILABLE</label>
</div> </div>
</div> </div>
{# Display status for dynsdns, config & sbot #} {# Display status for dynsdns, config & sbot #}