#![allow(clippy::needless_doctest_main)] //! A wrapper around the `prometheus` crate that provides a global, `lazy_static` metrics registry //! and functions to add and use the following components (more info at //! [Prometheus docs](https://prometheus.io/docs/concepts/metric_types/)): //! //! - `Histogram`: used with `start_timer(..)` and `stop_timer(..)` to record durations (e.g., //! block processing time). //! - `IncCounter`: used to represent an ideally ever-growing, never-shrinking integer (e.g., //! number of block processing requests). //! - `IntGauge`: used to represent an varying integer (e.g., number of attestations per block). //! //! ## Important //! //! Metrics will fail if two items have the same `name`. All metrics must have a unique `name`. //! Because we use a global registry there is no namespace per crate, it's one big global space. //! //! See the [Prometheus naming best practices](https://prometheus.io/docs/practices/naming/) when //! choosing metric names. //! //! ## Example //! //! ```rust //! #[macro_use] //! extern crate lazy_static; //! use lighthouse_metrics::*; //! //! // These metrics are "magically" linked to the global registry defined in `lighthouse_metrics`. //! lazy_static! { //! pub static ref RUN_COUNT: Result = try_create_int_counter( //! "runs_total", //! "Total number of runs" //! ); //! pub static ref CURRENT_VALUE: Result = try_create_int_gauge( //! "current_value", //! "The current value" //! ); //! pub static ref RUN_TIME: Result = //! try_create_histogram("run_seconds", "Time taken (measured to high precision)"); //! } //! //! //! fn main() { //! for i in 0..100 { //! inc_counter(&RUN_COUNT); //! let timer = start_timer(&RUN_TIME); //! //! for j in 0..10 { //! set_gauge(&CURRENT_VALUE, j); //! println!("Howdy partner"); //! } //! //! stop_timer(timer); //! } //! } //! ``` use prometheus::{HistogramOpts, HistogramTimer, Opts}; pub use prometheus::{ Encoder, Gauge, Histogram, HistogramVec, IntCounter, IntGauge, IntGaugeVec, Result, TextEncoder, }; /// Collect all the metrics for reporting. pub fn gather() -> Vec { prometheus::gather() } /// Attempts to crate an `IntCounter`, returning `Err` if the registry does not accept the counter /// (potentially due to naming conflict). pub fn try_create_int_counter(name: &str, help: &str) -> Result { let opts = Opts::new(name, help); let counter = IntCounter::with_opts(opts)?; prometheus::register(Box::new(counter.clone()))?; Ok(counter) } /// Attempts to crate an `IntGauge`, returning `Err` if the registry does not accept the counter /// (potentially due to naming conflict). pub fn try_create_int_gauge(name: &str, help: &str) -> Result { let opts = Opts::new(name, help); let gauge = IntGauge::with_opts(opts)?; prometheus::register(Box::new(gauge.clone()))?; Ok(gauge) } /// Attempts to crate a `Gauge`, returning `Err` if the registry does not accept the counter /// (potentially due to naming conflict). pub fn try_create_float_gauge(name: &str, help: &str) -> Result { let opts = Opts::new(name, help); let gauge = Gauge::with_opts(opts)?; prometheus::register(Box::new(gauge.clone()))?; Ok(gauge) } /// Attempts to crate a `Histogram`, returning `Err` if the registry does not accept the counter /// (potentially due to naming conflict). pub fn try_create_histogram(name: &str, help: &str) -> Result { let opts = HistogramOpts::new(name, help); let histogram = Histogram::with_opts(opts)?; prometheus::register(Box::new(histogram.clone()))?; Ok(histogram) } /// Attempts to crate a `HistogramVec`, returning `Err` if the registry does not accept the counter /// (potentially due to naming conflict). pub fn try_create_histogram_vec( name: &str, help: &str, label_names: &[&str], ) -> Result { let opts = HistogramOpts::new(name, help); let histogram_vec = HistogramVec::new(opts, label_names)?; prometheus::register(Box::new(histogram_vec.clone()))?; Ok(histogram_vec) } /// Attempts to crate a `IntGaugeVec`, returning `Err` if the registry does not accept the gauge /// (potentially due to naming conflict). pub fn try_create_int_gauge_vec( name: &str, help: &str, label_names: &[&str], ) -> Result { let opts = Opts::new(name, help); let counter_vec = IntGaugeVec::new(opts, label_names)?; prometheus::register(Box::new(counter_vec.clone()))?; Ok(counter_vec) } pub fn get_int_gauge(int_gauge_vec: &Result, name: &[&str]) -> Option { if let Ok(int_gauge_vec) = int_gauge_vec { Some(int_gauge_vec.get_metric_with_label_values(name).ok()?) } else { None } } pub fn get_histogram(histogram_vec: &Result, name: &[&str]) -> Option { if let Ok(histogram_vec) = histogram_vec { Some(histogram_vec.get_metric_with_label_values(name).ok()?) } else { None } } /// Starts a timer for the given `Histogram`, stopping when it gets dropped or given to `stop_timer(..)`. pub fn start_timer(histogram: &Result) -> Option { if let Ok(histogram) = histogram { Some(histogram.start_timer()) } else { None } } /// Stops a timer created with `start_timer(..)`. pub fn stop_timer(timer: Option) { if let Some(t) = timer { t.observe_duration() } } pub fn inc_counter(counter: &Result) { if let Ok(counter) = counter { counter.inc(); } } pub fn inc_counter_by(counter: &Result, value: i64) { if let Ok(counter) = counter { counter.inc_by(value); } } pub fn set_gauge(gauge: &Result, value: i64) { if let Ok(gauge) = gauge { gauge.set(value); } } pub fn inc_gauge(gauge: &Result) { if let Ok(gauge) = gauge { gauge.inc(); } } pub fn dec_gauge(gauge: &Result) { if let Ok(gauge) = gauge { gauge.dec(); } } pub fn maybe_set_gauge(gauge: &Result, value_opt: Option) { if let Some(value) = value_opt { set_gauge(gauge, value) } } pub fn set_float_gauge(gauge: &Result, value: f64) { if let Ok(gauge) = gauge { gauge.set(value); } } pub fn maybe_set_float_gauge(gauge: &Result, value_opt: Option) { if let Some(value) = value_opt { set_float_gauge(gauge, value) } } /// Sets the value of a `Histogram` manually. pub fn observe(histogram: &Result, value: f64) { if let Ok(histogram) = histogram { histogram.observe(value); } }