use rocket::{get, routes, serde::json::Json, State}; pub struct ServerState { primary_name: String, map: std::collections::HashMap< String, std::sync::Arc>, >, } impl ServerState { pub fn new( primary_name: String, map: std::collections::HashMap< String, std::sync::Arc>, >, ) -> Self { Self { primary_name, map } } } pub fn rocket(state: ServerState) -> rocket::Rocket { rocket::build().manage(state).mount( "/", routes![ metrics, interfaces, all_interfaces, primary_interface, interface ], ) } #[get("/interfaces")] fn interfaces(state: &State) -> Json> { Json(state.map.keys().cloned().collect()) } #[get("/interfaces/primary")] async fn primary_interface( state: &State, ) -> Result, ServerError> { let s = state .map .get(&state.primary_name) .ok_or(ServerError::InvalidPrimaryName)? .read() .await .clone(); Ok(Json(s)) } #[get("/interfaces/data")] async fn all_interfaces( state: &State, ) -> Json> { let mut data = Vec::new(); for (k, v) in &state.map { data.push((k.clone(), v.read().await.clone())); } Json(data) } #[get("/interface/")] async fn interface( name: &str, state: &State, ) -> Result, ServerError> { let data = state .map .get(name) .ok_or(ServerError::NotFound)? .read() .await .clone(); Ok(Json(data)) } #[get("/metrics")] fn metrics() -> Result { Ok( prometheus::TextEncoder::new() .encode_to_string(&prometheus::default_registry().gather())?, ) } enum ServerError { Prometheus, NotFound, InvalidPrimaryName, } impl From for ServerError { fn from(_: prometheus::Error) -> Self { Self::Prometheus } } impl<'a> rocket::response::Responder<'a, 'a> for ServerError { fn respond_to(self, _: &'a rocket::Request<'_>) -> rocket::response::Result<'a> { Err(match self { Self::Prometheus => rocket::http::Status::InternalServerError, Self::NotFound => rocket::http::Status::NotFound, Self::InvalidPrimaryName => rocket::http::Status::ServiceUnavailable, }) } }