|
|
/*
* =========================================================================================
* Copyright © 2013 the kamon project <http://kamon.io/>
*
* 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, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
* either express or implied. See the License for the specific language governing permissions
* and limitations under the License.
* =========================================================================================
*/
package kamon.metric.instrument
import java.nio.LongBuffer
import kamon.metric.instrument.Histogram.DynamicRange
import org.scalatest.{Matchers, WordSpec}
import scala.util.Random
class HistogramSpec extends WordSpec with Matchers {
"a Histogram" should {
"allow record values within the configured range" in new HistogramFixture {
histogram.record(1000)
histogram.record(5000, count = 100)
histogram.record(10000)
}
"not fail when recording values higher than the highest trackable value" in new HistogramFixture {
histogram.record(Long.MaxValue)
}
"reset all recorded levels to zero after a snapshot collection" in new HistogramFixture {
histogram.record(100)
histogram.record(200)
histogram.record(300)
takeSnapshot().numberOfMeasurements should be(3)
takeSnapshot().numberOfMeasurements should be(0)
}
"produce a snapshot" which {
"supports min, max, percentile, sum, numberOfMeasurements and recordsIterator operations" in new HistogramFixture {
histogram.record(100)
histogram.record(200, count = 200)
histogram.record(300)
histogram.record(900)
val snapshot = takeSnapshot()
snapshot.min should equal(100L +- 1L)
snapshot.max should equal(900L +- 9L)
snapshot.percentile(50.0D) should be(200)
snapshot.percentile(99.5D) should be(300)
snapshot.percentile(99.9D) should be(900)
snapshot.sum should be(41300)
snapshot.numberOfMeasurements should be(203)
val records = snapshot.recordsIterator.map(r ⇒ r.level → r.count).toSeq
records.size should be(4)
records(0) should be(100 → 1)
records(1) should be(200 → 200)
records(2) should be(300 → 1)
records(3) should be(900 → 1)
}
"can be scaled" in new HistogramFixture {
histogram.record(100)
histogram.record(200, count = 200)
histogram.record(300)
histogram.record(900)
val snapshot = takeSnapshot().scale(Time.Seconds, Time.Milliseconds)
snapshot.min should equal(100000L +- 1000L)
snapshot.max should equal(900000L +- 9000L)
snapshot.percentile(50.0D) should be(200000)
snapshot.percentile(99.5D) should be(300000)
snapshot.percentile(99.9D) should be(900000)
snapshot.sum should be(41300000)
snapshot.numberOfMeasurements should be(203)
val records = snapshot.recordsIterator.map(r ⇒ r.level → r.count).toSeq
records.size should be(4)
records(0) should be(100000 → 1)
records(1) should be(200000 → 200)
records(2) should be(300000 → 1)
records(3) should be(900000 → 1)
}
"can be merged with another snapshot" in new MultipleHistogramFixture {
val random = new Random(System.nanoTime())
for (repetitions ← 1 to 1000) {
// Put some values on A and Control
for (_ ← 1 to 1000) {
val newRecording = random.nextInt(100000)
controlHistogram.record(newRecording)
histogramA.record(newRecording)
}
// Put some values on B and Control
for (_ ← 1 to 2000) {
val newRecording = random.nextInt(100000)
controlHistogram.record(newRecording)
histogramB.record(newRecording)
}
val controlSnapshot = takeSnapshotFrom(controlHistogram)
val histogramASnapshot = takeSnapshotFrom(histogramA)
val histogramBSnapshot = takeSnapshotFrom(histogramB)
assertEquals(controlSnapshot, histogramASnapshot.merge(histogramBSnapshot, collectionContext))
assertEquals(controlSnapshot, histogramBSnapshot.merge(histogramASnapshot, collectionContext))
}
}
}
}
trait HistogramFixture {
val collectionContext = new CollectionContext {
val buffer: LongBuffer = LongBuffer.allocate(10000)
}
val histogram = Histogram(DynamicRange(1, 100000, 2))
def takeSnapshot(): Histogram.Snapshot = histogram.collect(collectionContext)
}
trait MultipleHistogramFixture {
val collectionContext = new CollectionContext {
val buffer: LongBuffer = LongBuffer.allocate(10000)
}
val controlHistogram = Histogram(DynamicRange(1, 100000, 2))
val histogramA = Histogram(DynamicRange(1, 100000, 2))
val histogramB = Histogram(DynamicRange(1, 100000, 2))
def takeSnapshotFrom(histogram: Histogram): InstrumentSnapshot = histogram.collect(collectionContext)
def assertEquals(left: InstrumentSnapshot, right: InstrumentSnapshot): Unit = {
val leftSnapshot = left.asInstanceOf[Histogram.Snapshot]
val rightSnapshot = right.asInstanceOf[Histogram.Snapshot]
leftSnapshot.numberOfMeasurements should equal(rightSnapshot.numberOfMeasurements)
leftSnapshot.min should equal(rightSnapshot.min)
leftSnapshot.max should equal(rightSnapshot.max)
leftSnapshot.recordsIterator.toStream should contain theSameElementsAs (rightSnapshot.recordsIterator.toStream)
}
}
}
|