-
Notifications
You must be signed in to change notification settings - Fork 61
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
RUM-6866 create the KnuthStableSampler
- Loading branch information
Showing
5 changed files
with
356 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
91 changes: 91 additions & 0 deletions
91
...dk-android-core/src/main/kotlin/com/datadog/android/core/sampling/DeterministicSampler.kt
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,91 @@ | ||
/* | ||
* Unless explicitly stated otherwise all files in this repository are licensed under the Apache License Version 2.0. | ||
* This product includes software developed at Datadog (https://www.datadoghq.com/). | ||
* Copyright 2016-Present Datadog, Inc. | ||
*/ | ||
|
||
package com.datadog.android.core.sampling | ||
|
||
import androidx.annotation.FloatRange | ||
import com.datadog.android.api.InternalLogger | ||
|
||
/** | ||
* [Sampler] with the given sample rate using a deterministic algorithm for a stable | ||
* sampling decision across sources. | ||
* | ||
* @param T the type of items to sample. | ||
* @param idConverter a lambda converting the input item into a stable numerical identifier | ||
* @param sampleRateProvider Provider for the sample rate value which will be called each time | ||
* the sampling decision needs to be made. All the values should be on the scale [0;100]. | ||
*/ | ||
open class DeterministicSampler<T : Any>( | ||
private val idConverter: (T) -> ULong, | ||
private val sampleRateProvider: () -> Float | ||
) : Sampler<T> { | ||
|
||
/** | ||
* Creates a new instance lof [DeterministicSampler] with the given sample rate. | ||
* | ||
* @param sampleRate Sample rate to use. | ||
*/ | ||
constructor( | ||
idConverter: (T) -> ULong, | ||
@FloatRange(from = 0.0, to = 100.0) sampleRate: Float | ||
) : this(idConverter, { sampleRate }) | ||
|
||
/** | ||
* Creates a new instance of [DeterministicSampler] with the given sample rate. | ||
* | ||
* @param sampleRate Sample rate to use. | ||
*/ | ||
constructor( | ||
idConverter: (T) -> ULong, | ||
@FloatRange(from = 0.0, to = 100.0) sampleRate: Double | ||
) : this(idConverter, sampleRate.toFloat()) | ||
|
||
/** @inheritDoc */ | ||
override fun sample(item: T): Boolean { | ||
val sampleRate = getSampleRate() | ||
if (sampleRate >= SAMPLE_ALL_RATE) { | ||
return true | ||
} else if (sampleRate <= 0f) { | ||
return false | ||
} else { | ||
val stableId = idConverter(item) | ||
val hash = stableId * SAMPLER_HASHER | ||
val threshold = (MAX_ID.toDouble() * sampleRate / SAMPLE_ALL_RATE).toULong() | ||
return hash < threshold | ||
} | ||
} | ||
|
||
/** @inheritDoc */ | ||
override fun getSampleRate(): Float { | ||
val rawSampleRate = sampleRateProvider() | ||
return if (rawSampleRate < 0f) { | ||
InternalLogger.UNBOUND.log( | ||
InternalLogger.Level.WARN, | ||
InternalLogger.Target.USER, | ||
{ "Sample rate value provided $rawSampleRate is below 0, setting it to 0." } | ||
) | ||
0f | ||
} else if (rawSampleRate > SAMPLE_ALL_RATE) { | ||
InternalLogger.UNBOUND.log( | ||
InternalLogger.Level.WARN, | ||
InternalLogger.Target.USER, | ||
{ "Sample rate value provided $rawSampleRate is above 100, setting it to 100." } | ||
) | ||
SAMPLE_ALL_RATE | ||
} else { | ||
rawSampleRate | ||
} | ||
} | ||
|
||
private companion object { | ||
const val SAMPLE_ALL_RATE = 100f | ||
|
||
// Good number for Knuth hashing (large, prime, fit in int64 for languages without uint64) | ||
private val SAMPLER_HASHER: ULong = 1111111111111111111u | ||
|
||
private val MAX_ID: ULong = 0xFFFFFFFFFFFFFFFFUL | ||
} | ||
} |
50 changes: 50 additions & 0 deletions
50
dd-sdk-android-core/src/test/java/com/datadog/trace/sampling/JavaDeterministicSampler.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,50 @@ | ||
/* | ||
* Unless explicitly stated otherwise all files in this repository are licensed under the Apache License Version 2.0. | ||
* This product includes software developed at Datadog (https://www.datadoghq.com/). | ||
* Copyright 2016-Present Datadog, Inc. | ||
*/ | ||
|
||
package com.datadog.trace.sampling; | ||
|
||
import androidx.annotation.NonNull; | ||
import androidx.annotation.Nullable; | ||
|
||
import com.datadog.android.core.sampling.Sampler; | ||
|
||
/** | ||
* This is a pseudo-duplicate of the java implementation for testing purposes only to ensure | ||
* compatibility between our generic implementation and the one in our backend agent. | ||
*/ | ||
public class JavaDeterministicSampler implements Sampler<Long> { | ||
|
||
private static final long KNUTH_FACTOR = 1111111111111111111L; | ||
|
||
private static final double MAX = Math.pow(2, 64) - 1; | ||
|
||
private final float rate; | ||
|
||
public JavaDeterministicSampler(float rate) { | ||
this.rate = rate; | ||
} | ||
|
||
@Override | ||
public boolean sample(@NonNull Long item) { | ||
return item * KNUTH_FACTOR + Long.MIN_VALUE < cutoff(rate); | ||
} | ||
|
||
@Nullable | ||
@Override | ||
public Float getSampleRate() { | ||
return rate; | ||
} | ||
|
||
private long cutoff(double rate) { | ||
if (rate < 0.5) { | ||
return (long) (rate * MAX) + Long.MIN_VALUE; | ||
} | ||
if (rate < 1.0) { | ||
return (long) ((rate * MAX) + Long.MIN_VALUE); | ||
} | ||
return Long.MAX_VALUE; | ||
} | ||
} |
200 changes: 200 additions & 0 deletions
200
...ndroid-core/src/test/kotlin/com/datadog/android/core/sampling/DeterministicSamplerTest.kt
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,200 @@ | ||
package com.datadog.android.core.sampling | ||
|
||
import com.datadog.android.utils.forge.Configurator | ||
import com.datadog.trace.sampling.JavaDeterministicSampler | ||
import fr.xgouchet.elmyr.annotation.FloatForgery | ||
import fr.xgouchet.elmyr.annotation.LongForgery | ||
import fr.xgouchet.elmyr.junit5.ForgeConfiguration | ||
import fr.xgouchet.elmyr.junit5.ForgeExtension | ||
import org.assertj.core.api.Assertions.assertThat | ||
import org.assertj.core.data.Offset | ||
import org.junit.jupiter.api.BeforeEach | ||
import org.junit.jupiter.api.RepeatedTest | ||
import org.junit.jupiter.api.Test | ||
import org.junit.jupiter.api.extension.ExtendWith | ||
import org.junit.jupiter.api.extension.Extensions | ||
import org.junit.jupiter.params.ParameterizedTest | ||
import org.junit.jupiter.params.provider.Arguments | ||
import org.junit.jupiter.params.provider.MethodSource | ||
import org.mockito.Mock | ||
import org.mockito.junit.jupiter.MockitoExtension | ||
import org.mockito.junit.jupiter.MockitoSettings | ||
import org.mockito.kotlin.doReturn | ||
import org.mockito.kotlin.whenever | ||
import org.mockito.quality.Strictness | ||
import java.util.stream.Stream | ||
|
||
@Extensions( | ||
ExtendWith(MockitoExtension::class), | ||
ExtendWith(ForgeExtension::class) | ||
) | ||
@MockitoSettings(strictness = Strictness.LENIENT) | ||
@ForgeConfiguration(Configurator::class) | ||
internal class DeterministicSamplerTest { | ||
|
||
private lateinit var testedSampler: Sampler<ULong> | ||
|
||
private var stubIdConverter: (ULong) -> ULong = { it } | ||
|
||
@Mock | ||
lateinit var mockSampleRateProvider: () -> Float | ||
|
||
@BeforeEach | ||
fun `set up`() { | ||
testedSampler = DeterministicSampler( | ||
stubIdConverter, | ||
mockSampleRateProvider | ||
) | ||
} | ||
|
||
@ParameterizedTest | ||
@MethodSource("hardcodedFixtures") | ||
fun `M return consistent results W sample() {hardcodedFixtures}`( | ||
input: Fixture, | ||
expectedDecision: Boolean | ||
) { | ||
// Given | ||
whenever(mockSampleRateProvider.invoke()) doReturn input.samplingRate | ||
|
||
// When | ||
val sampled = testedSampler.sample(input.traceId) | ||
|
||
// | ||
assertThat(sampled).isEqualTo(expectedDecision) | ||
} | ||
|
||
@RepeatedTest(128) | ||
fun `M return consistent results W sample() {java implementation}`( | ||
@LongForgery traceIds: List<Long>, | ||
@FloatForgery(min = 0f, max = 100f) fakeSampleRate: Float | ||
) { | ||
// Given | ||
whenever(mockSampleRateProvider.invoke()) doReturn fakeSampleRate | ||
val javaSampler = JavaDeterministicSampler(fakeSampleRate / 100f) | ||
|
||
// When | ||
traceIds.forEach { | ||
val result = testedSampler.sample(it.toULong()) | ||
val expectedResult = javaSampler.sample(it) | ||
|
||
assertThat(result).isEqualTo(expectedResult) | ||
} | ||
} | ||
|
||
@RepeatedTest(128) | ||
fun `the sampler will sample the values based on the fixed sample rate`( | ||
@LongForgery traceIds: List<Long>, | ||
@FloatForgery(min = 0f, max = 100f) fakeSampleRate: Float | ||
) { | ||
// Given | ||
whenever(mockSampleRateProvider.invoke()) doReturn fakeSampleRate | ||
var sampledIn = 0 | ||
|
||
// When | ||
traceIds.forEach { | ||
if (testedSampler.sample(it.toULong())) { | ||
sampledIn++ | ||
} | ||
} | ||
|
||
// Then | ||
assertThat(sampledIn.toFloat()).isCloseTo(traceIds.size * fakeSampleRate / 100f, Offset.offset(7.5f)) | ||
} | ||
|
||
@Test | ||
fun `when sample rate is 0 all values will be dropped`( | ||
@LongForgery traceIds: List<Long> | ||
) { | ||
// Given | ||
whenever(mockSampleRateProvider.invoke()) doReturn 0f | ||
var sampledIn = 0 | ||
|
||
// When | ||
traceIds.forEach { | ||
if (testedSampler.sample(it.toULong())) { | ||
sampledIn++ | ||
} | ||
} | ||
|
||
// Then | ||
assertThat(sampledIn).isEqualTo(0) | ||
} | ||
|
||
@Test | ||
fun `when sample rate is 100 all values will pass`( | ||
@LongForgery traceIds: List<Long> | ||
) { | ||
// Given | ||
whenever(mockSampleRateProvider.invoke()) doReturn 100f | ||
var sampledIn = 0 | ||
|
||
// When | ||
traceIds.forEach { | ||
if (testedSampler.sample(it.toULong())) { | ||
sampledIn++ | ||
} | ||
} | ||
|
||
// Then | ||
assertThat(sampledIn).isEqualTo(traceIds.size) | ||
} | ||
|
||
@Test | ||
fun `when sample rate is below 0 it is normalized to 0`( | ||
@FloatForgery(max = 0f) fakeSampleRate: Float | ||
) { | ||
// Given | ||
whenever(mockSampleRateProvider.invoke()) doReturn fakeSampleRate | ||
|
||
// When | ||
val effectiveSampleRate = testedSampler.getSampleRate() | ||
|
||
// Then | ||
assertThat(effectiveSampleRate).isZero | ||
} | ||
|
||
@Test | ||
fun `when sample rate is above 100 it is normalized to 100`( | ||
@FloatForgery(min = 100.01f) fakeSampleRate: Float | ||
) { | ||
// Given | ||
whenever(mockSampleRateProvider.invoke()) doReturn fakeSampleRate | ||
|
||
// When | ||
val effectiveSampleRate = testedSampler.getSampleRate() | ||
|
||
// Then | ||
assertThat(effectiveSampleRate).isEqualTo(100f) | ||
} | ||
|
||
/** | ||
* A data class is necessary to wrap the ULong, otherwise the jvm runner | ||
* converts it to Long at some point. | ||
*/ | ||
data class Fixture( | ||
val traceId: ULong, | ||
val samplingRate: Float | ||
) | ||
|
||
companion object { | ||
|
||
// Those hardcoded values ensures we are consistent with the decisions of our | ||
// Backend implementation of the knuth sampling method | ||
@Suppress("unused") | ||
@JvmStatic | ||
fun hardcodedFixtures(): Stream<Arguments> { | ||
return listOf( | ||
Arguments.of(Fixture(4815162342u, 55.9f), false), | ||
Arguments.of(Fixture(4815162342u, 56.0f), true), | ||
Arguments.of(Fixture(1415926535897932384u, 90.5f), false), | ||
Arguments.of(Fixture(1415926535897932384u, 90.6f), true), | ||
Arguments.of(Fixture(718281828459045235u, 7.4f), false), | ||
Arguments.of(Fixture(718281828459045235u, 7.5f), true), | ||
Arguments.of(Fixture(41421356237309504u, 32.1f), false), | ||
Arguments.of(Fixture(41421356237309504u, 32.2f), true), | ||
Arguments.of(Fixture(6180339887498948482u, 68.2f), false), | ||
Arguments.of(Fixture(6180339887498948482u, 68.3f), true) | ||
).stream() | ||
} | ||
} | ||
} |