extern crate prometheus; extern crate prometheus_static_metric; extern crate rouille; use prometheus::{Encoder, Gauge, IntGauge, IntGaugeVec, Opts, Registry, TextEncoder}; use rouille::{router, Server}; use std::{ convert::TryInto, net::{IpAddr, Ipv4Addr, SocketAddr}, sync::{ atomic::{AtomicBool, Ordering}, Arc, }, thread, time::{SystemTime, UNIX_EPOCH}, }; pub struct ServerMetrics { pub chonks_count: IntGauge, pub chunks_count: IntGauge, pub player_online: IntGauge, pub entity_count: IntGauge, pub tick_time: IntGaugeVec, pub build_info: IntGauge, pub start_time: IntGauge, pub time_of_day: Gauge, pub light_count: IntGauge, pub thread_running: Arc, pub handle: Option>, pub every_100th: i8, } impl ServerMetrics { pub fn new() -> Self { let opts = Opts::new( "player_online", "shows the number of clients connected to the server", ); let player_online = IntGauge::with_opts(opts).unwrap(); let opts = Opts::new( "entity_count", "number of all entities currently active on the server", ); let entity_count = IntGauge::with_opts(opts).unwrap(); let opts = Opts::new("veloren_build_info", "Build information") .const_label("hash", common::util::GIT_HASH) .const_label("version", ""); let build_info = IntGauge::with_opts(opts).unwrap(); let opts = Opts::new( "veloren_start_time", "start time of the server in seconds since EPOCH", ); let start_time = IntGauge::with_opts(opts).unwrap(); let opts = Opts::new("time_of_day", "ingame time in ingame-seconds"); let time_of_day = Gauge::with_opts(opts).unwrap(); let opts = Opts::new( "light_count", "number of all lights currently active on the server", ); let light_count = IntGauge::with_opts(opts).unwrap(); let opts = Opts::new( "chonks_count", "number of all chonks currently active on the server", ); let chonks_count = IntGauge::with_opts(opts).unwrap(); let opts = Opts::new( "chunks_count", "number of all chunks currently active on the server", ); let chunks_count = IntGauge::with_opts(opts).unwrap(); let vec = IntGaugeVec::new( Opts::new("tick_time", "time in ns requiered for a tick of the server"), &["period"], ) .unwrap(); let tick_time = IntGaugeVec::from(vec); let since_the_epoch = SystemTime::now() .duration_since(UNIX_EPOCH) .expect("Time went backwards"); start_time.set(since_the_epoch.as_secs().try_into().unwrap()); let registry = Registry::new(); //registry.register(Box::new(chonks_count.clone())).unwrap(); registry.register(Box::new(player_online.clone())).unwrap(); registry.register(Box::new(entity_count.clone())).unwrap(); registry.register(Box::new(build_info.clone())).unwrap(); registry.register(Box::new(start_time.clone())).unwrap(); registry.register(Box::new(time_of_day.clone())).unwrap(); //registry.register(Box::new(light_count.clone())).unwrap(); registry.register(Box::new(chonks_count.clone())).unwrap(); registry.register(Box::new(chunks_count.clone())).unwrap(); registry.register(Box::new(tick_time.clone())).unwrap(); let thread_running = Arc::new(AtomicBool::new(true)); let thread_running2 = thread_running.clone(); //TODO: make this a job let handle = Some(thread::spawn(move || { let addr = SocketAddr::new(IpAddr::V4(Ipv4Addr::new(0, 0, 0, 0)), 14005); let server = Server::new(addr, move |request| { router!(request, (GET) (/metrics) => { let encoder = TextEncoder::new(); let mut buffer = vec![]; let mf = registry.gather(); encoder.encode(&mf, &mut buffer).unwrap(); rouille::Response::text(String::from_utf8(buffer).unwrap()) }, _ => rouille::Response::empty_404() ) }) .expect("Failed to start server"); while thread_running2.load(Ordering::Relaxed) { server.poll(); } })); Self { chonks_count, chunks_count, player_online, entity_count, tick_time, build_info, start_time, time_of_day, light_count, thread_running, handle, every_100th: 0, } } pub fn is_100th_tick(&mut self) -> bool { self.every_100th += 1; if self.every_100th == 100 { self.every_100th = 0; true } else { false } } } impl Drop for ServerMetrics { fn drop(&mut self) { self.thread_running.store(false, Ordering::Relaxed); let handle = self.handle.take(); handle .unwrap() .join() .expect("Error shutting down prometheus metric exporter"); } }