prometheus 0.13.4

Prometheus instrumentation library for Rust applications.
Documentation
// Copyright 2016 PingCAP, Inc.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
//     http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// See the License for the specific language governing permissions and
// limitations under the License.

use criterion::{criterion_group, criterion_main, Criterion};
use prometheus::{Counter, CounterVec, IntCounter, Opts};
use std::collections::HashMap;
use std::sync::{atomic, Arc};
use std::thread;

fn bench_counter_with_label_values(c: &mut Criterion) {
    let counter = CounterVec::new(
        Opts::new("benchmark_counter", "A counter to benchmark it."),
        &["one", "two", "three"],
    )
    .unwrap();
    c.bench_function("counter_with_label_values", |b| {
        b.iter(|| counter.with_label_values(&["eins", "zwei", "drei"]).inc())
    });
}

fn bench_counter_with_mapped_labels(c: &mut Criterion) {
    let counter = CounterVec::new(
        Opts::new("benchmark_counter", "A counter to benchmark it."),
        &["one", "two", "three"],
    )
    .unwrap();

    c.bench_function("counter_with_mapped_labels", |b| {
        b.iter(|| {
            let mut labels = HashMap::with_capacity(3);
            labels.insert("two", "zwei");
            labels.insert("one", "eins");
            labels.insert("three", "drei");
            counter.with(&labels).inc();
        })
    });
}

fn bench_counter_with_prepared_mapped_labels(c: &mut Criterion) {
    let counter = CounterVec::new(
        Opts::new("benchmark_counter", "A counter to benchmark it."),
        &["one", "two", "three"],
    )
    .unwrap();

    let mut labels = HashMap::with_capacity(3);
    labels.insert("two", "zwei");
    labels.insert("one", "eins");
    labels.insert("three", "drei");

    c.bench_function("counter_with_prepared_mapped_labels", |b| {
        b.iter(|| {
            counter.with(&labels).inc();
        })
    });
}

fn bench_counter_no_labels(c: &mut Criterion) {
    let counter = Counter::new("benchmark_counter", "A counter to benchmark.").unwrap();
    c.bench_function("counter_no_labels", |b| b.iter(|| counter.inc()));
}

fn bench_int_counter_no_labels(c: &mut Criterion) {
    let counter = IntCounter::new("benchmark_int_counter", "A int_counter to benchmark.").unwrap();
    c.bench_function("int_counter_no_labels", |b| b.iter(|| counter.inc()));
}

fn bench_counter_no_labels_concurrent_nop(c: &mut Criterion) {
    let signal_exit = Arc::new(atomic::AtomicBool::new(false));
    let counter = Counter::new("foo", "bar").unwrap();

    let thread_handles: Vec<_> = (0..4)
        .map(|_| {
            let signal_exit2 = signal_exit.clone();
            thread::spawn(move || {
                while !signal_exit2.load(atomic::Ordering::Relaxed) {
                    // Do nothing as the control group.
                }
            })
        })
        .collect();

    c.bench_function("counter_no_labels_concurrent_nop", |b| {
        b.iter(|| counter.inc());
    });

    // Wait for accompanying thread to exit.
    signal_exit.store(true, atomic::Ordering::Relaxed);
    for h in thread_handles {
        h.join().unwrap();
    }
}

fn bench_counter_no_labels_concurrent_write(c: &mut Criterion) {
    let signal_exit = Arc::new(atomic::AtomicBool::new(false));
    let counter = Counter::new("foo", "bar").unwrap();

    let thread_handles: Vec<_> = (0..4)
        .map(|_| {
            let signal_exit2 = signal_exit.clone();
            let counter2 = counter.clone();
            thread::spawn(move || {
                while !signal_exit2.load(atomic::Ordering::Relaxed) {
                    // Update counter concurrently as the normal group.
                    counter2.inc();
                }
            })
        })
        .collect();

    c.bench_function("counter_no_labels_concurrent_write", |b| {
        b.iter(|| counter.inc());
    });

    // Wait for accompanying thread to exit.
    signal_exit.store(true, atomic::Ordering::Relaxed);
    for h in thread_handles {
        h.join().unwrap();
    }
}

fn bench_int_counter_no_labels_concurrent_write(c: &mut Criterion) {
    let signal_exit = Arc::new(atomic::AtomicBool::new(false));
    let counter = IntCounter::new("foo", "bar").unwrap();

    let thread_handles: Vec<_> = (0..4)
        .map(|_| {
            let signal_exit2 = signal_exit.clone();
            let counter2 = counter.clone();
            thread::spawn(move || {
                while !signal_exit2.load(atomic::Ordering::Relaxed) {
                    // Update counter concurrently as the normal group.
                    counter2.inc();
                }
            })
        })
        .collect();

    c.bench_function("int_counter_no_labels_concurrent_write", |b| {
        b.iter(|| counter.inc());
    });

    // Wait for accompanying thread to exit.
    signal_exit.store(true, atomic::Ordering::Relaxed);
    for h in thread_handles {
        h.join().unwrap();
    }
}

fn bench_counter_with_label_values_concurrent_write(c: &mut Criterion) {
    let signal_exit = Arc::new(atomic::AtomicBool::new(false));
    let counter = CounterVec::new(Opts::new("foo", "bar"), &["one", "two", "three"]).unwrap();

    let thread_handles: Vec<_> = (0..4)
        .map(|_| {
            let signal_exit2 = signal_exit.clone();
            let counter2 = counter.clone();
            thread::spawn(move || {
                while !signal_exit2.load(atomic::Ordering::Relaxed) {
                    counter2.with_label_values(&["eins", "zwei", "drei"]).inc();
                }
            })
        })
        .collect();

    c.bench_function("counter_with_label_values_concurrent_write", |b| {
        b.iter(|| counter.with_label_values(&["eins", "zwei", "drei"]).inc());
    });

    // Wait for accompanying thread to exit.
    signal_exit.store(true, atomic::Ordering::Relaxed);
    for h in thread_handles {
        h.join().unwrap();
    }
}

criterion_group!(
    benches,
    bench_counter_no_labels,
    bench_counter_no_labels_concurrent_nop,
    bench_counter_no_labels_concurrent_write,
    bench_counter_with_label_values,
    bench_counter_with_label_values_concurrent_write,
    bench_counter_with_mapped_labels,
    bench_counter_with_prepared_mapped_labels,
    bench_int_counter_no_labels,
    bench_int_counter_no_labels_concurrent_write,
);
criterion_main!(benches);