A quick, modular metrics toolkit for Rust applications of all types. Similar to popular logging frameworks, but with counters, markers, gauges and timers.
Dipstick's main attraction is the ability to send metrics to multiple customized outputs. For example, metrics could be written immediately to the log and sent over the network after a period of aggregation.
Dipstick promotes structured metrics for clean, safe code and good performance.
Dipstick builds on stable Rust with minimal dependencies.
For complete applications see the examples.
To use Dipstick in your project, add the following line to your Cargo.toml
in the [dependencies]
section:
toml
dipstick = "0.4.18"
Then add it to your code:
rust,skt-fail,no_run
let metrics = app_metrics(to_graphite("host.com:2003")?);
let counter = metrics.counter("my_counter");
counter.count(3);
Send metrics to multiple outputs:
rust,skt-fail,no_run
let _app_metrics = app_metrics((
to_stdout(),
to_statsd("localhost:8125")?.with_namespace(&["my", "app"])
));
Since instruments are decoupled from the backend, outputs can be swapped easily.
Aggregate metrics and schedule to be periodical publication in the background: ```rust,skt-run use std::time::Duration;
let appmetrics = appmetrics(aggregate(allstats, tostdout())); appmetrics.flushevery(Duration::from_secs(3)); ```
Aggregation is performed locklessly and is very fast.
Count, sum, min, max and average are tracked where they make sense.
Published statistics can be selected with presets such as all_stats
(see previous example),
summary
, average
.
For more control over published statistics, provide your own strategy:
rust,skt-run
app_metrics(aggregate(
|_kind, name, score|
match score {
ScoreType::Count(count) =>
Some((Kind::Counter, vec![name, ".per_thousand"], count / 1000)),
_ => None
},
to_log()));
Apply statistical sampling to metrics:
rust,skt-fail
let _app_metrics = app_metrics(to_statsd("server:8125")?.with_sampling_rate(0.01));
A fast random algorithm is used to pick samples.
Outputs can use sample rate to expand or format published data.
Metrics can be recorded asynchronously:
rust,skt-run
let _app_metrics = app_metrics(to_stdout()).with_async_queue(64);
The async queue uses a Rust channel and a standalone thread.
The current behavior is to block when full.
Metric definitions can be cached to make using ad-hoc metrics faster:
rust,skt-run
let app_metrics = app_metrics(to_log().with_cache(512));
app_metrics.gauge(format!("my_gauge_{}", 34)).value(44);
The preferred way is to predefine metrics, possibly in a lazy_static! block: ```rust,skt-plain
extern crate dipstick;
use dipstick::*;
lazystatic! {
pub static ref METRICS: AppMetrics
fn main() { COUNTER_A.count(11); } ```
Timers can be used multiple ways: ```rust,skt-run let appmetrics = appmetrics(tostdout()); let timer = appmetrics.timer("my_timer"); time!(timer, {/* slow code here /} ); timer.time(|| {/ slow code here */} );
let start = timer.start(); /* slow code here */ timer.stop(start);
timer.intervalus(123456); ```
Related metrics can share a namespace:
rust,skt-run
let app_metrics = app_metrics(to_stdout());
let db_metrics = app_metrics.with_prefix("database");
let _db_timer = db_metrics.timer("db_timer");
let _db_counter = db_metrics.counter("db_counter");
Dipstick is licensed under the terms of the Apache 2.0 and MIT license.