Code Coverage |
||||||||||
Lines |
Functions and Methods |
Classes and Traits |
||||||||
| Total | |
88.37% |
38 / 43 |
|
60.00% |
3 / 5 |
CRAP | |
0.00% |
0 / 1 |
| HistogramMetric | |
88.37% |
38 / 43 |
|
60.00% |
3 / 5 |
16.40 | |
0.00% |
0 / 1 |
| __construct | |
84.21% |
16 / 19 |
|
0.00% |
0 / 1 |
7.19 | |||
| preloadBuckets | |
100.00% |
5 / 5 |
|
100.00% |
1 / 1 |
2 | |||
| observe | |
100.00% |
15 / 15 |
|
100.00% |
1 / 1 |
5 | |||
| setLabel | |
100.00% |
2 / 2 |
|
100.00% |
1 / 1 |
1 | |||
| fresh | |
0.00% |
0 / 2 |
|
0.00% |
0 / 1 |
2 | |||
| 1 | <?php |
| 2 | /** |
| 3 | * @license GPL-2.0-or-later |
| 4 | * @file |
| 5 | */ |
| 6 | |
| 7 | declare( strict_types=1 ); |
| 8 | |
| 9 | namespace Wikimedia\Stats\Metrics; |
| 10 | |
| 11 | use InvalidArgumentException; |
| 12 | use Wikimedia\Stats\StatsFactory; |
| 13 | |
| 14 | /** |
| 15 | * Histogram Metric Implementation |
| 16 | * |
| 17 | * Histogram Metrics are a collection of CounterMetric arranged into buckets. |
| 18 | * |
| 19 | * @author Cole White |
| 20 | * @since 1.44 |
| 21 | */ |
| 22 | class HistogramMetric { |
| 23 | |
| 24 | /** @var int Maximum number of allowed buckets. */ |
| 25 | private const MAX_BUCKETS = 10; |
| 26 | |
| 27 | private StatsFactory $statsFactory; |
| 28 | private string $name; |
| 29 | private array $buckets; |
| 30 | private array $labels = []; |
| 31 | |
| 32 | public function __construct( StatsFactory $statsFactory, string $name, array $buckets ) { |
| 33 | $this->statsFactory = $statsFactory; |
| 34 | $this->name = $name; |
| 35 | if ( !$buckets ) { |
| 36 | throw new InvalidArgumentException( "Stats: ({$name}) Histogram buckets cannot be an empty array." ); |
| 37 | } |
| 38 | $bucketCount = count( $buckets ); |
| 39 | if ( $bucketCount > self::MAX_BUCKETS ) { |
| 40 | throw new InvalidArgumentException( |
| 41 | "Stats: ({$name}) Too many buckets defined. Got:{$bucketCount}, Max:" . self::MAX_BUCKETS |
| 42 | ); |
| 43 | } |
| 44 | foreach ( $buckets as $bucket ) { |
| 45 | if ( !( is_float( $bucket ) || is_int( $bucket ) ) ) { |
| 46 | throw new InvalidArgumentException( "Stats: ({$name}) Histogram buckets can only be float or int." ); |
| 47 | } |
| 48 | } |
| 49 | $normalizedBuckets = array_unique( $buckets ); |
| 50 | sort( $normalizedBuckets, SORT_NUMERIC ); |
| 51 | if ( $buckets !== $normalizedBuckets ) { |
| 52 | throw new InvalidArgumentException( |
| 53 | "Stats: ({$name}) Histogram buckets must be unique and in order of least to greatest." |
| 54 | ); |
| 55 | } |
| 56 | $this->buckets = $buckets; |
| 57 | } |
| 58 | |
| 59 | /** |
| 60 | * Ensure every bucket has 0 as the first sample so that it is |
| 61 | * emitted for the exporter to report. |
| 62 | */ |
| 63 | private function preloadBuckets( CounterMetric $metric ): void { |
| 64 | $metric->setBucket( '+Inf' ); |
| 65 | $metric->incrementBy( 0 ); |
| 66 | foreach ( $this->buckets as $bucket ) { |
| 67 | $metric->setBucket( $bucket ); |
| 68 | $metric->incrementBy( 0 ); |
| 69 | } |
| 70 | } |
| 71 | |
| 72 | /** |
| 73 | * Increments bucket associated with the provided value. |
| 74 | */ |
| 75 | public function observe( float $value ): void { |
| 76 | $count = $this->statsFactory->getCounter( "{$this->name}_count" ); |
| 77 | $bucket = $this->statsFactory->getCounter( "{$this->name}_bucket" ); |
| 78 | $sum = $this->statsFactory->getCounter( "{$this->name}_sum" ); |
| 79 | |
| 80 | foreach ( $this->labels as $k => $v ) { |
| 81 | $count->setLabel( $k, $v ); |
| 82 | $bucket->setLabel( $k, $v ); |
| 83 | $sum->setLabel( $k, $v ); |
| 84 | } |
| 85 | |
| 86 | if ( $bucket->getSampleCount() === 0 ) { |
| 87 | $this->preloadBuckets( $bucket ); |
| 88 | } |
| 89 | |
| 90 | $bucket->setBucket( '+Inf' )->increment(); |
| 91 | foreach ( $this->buckets as $le ) { |
| 92 | if ( $value <= $le ) { |
| 93 | $bucket->setBucket( $le )->increment(); |
| 94 | } |
| 95 | } |
| 96 | |
| 97 | $count->increment(); |
| 98 | $sum->incrementBy( $value ); |
| 99 | } |
| 100 | |
| 101 | /** |
| 102 | * Adds a label $key with $value. No order is respected. |
| 103 | * |
| 104 | * @return HistogramMetric |
| 105 | */ |
| 106 | public function setLabel( string $key, string $value ): self { |
| 107 | // each metric will run its own validation logic |
| 108 | $this->labels[$key] = $value; |
| 109 | return $this; |
| 110 | } |
| 111 | |
| 112 | /** |
| 113 | * Returns metric with cleared labels. |
| 114 | * |
| 115 | * @return HistogramMetric |
| 116 | */ |
| 117 | public function fresh(): self { |
| 118 | $this->labels = []; |
| 119 | return $this; |
| 120 | } |
| 121 | } |