| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269 |
- // Copyright The OpenTelemetry Authors
- // SPDX-License-Identifier: Apache-2.0
- #include <gtest/gtest.h>
- #include <stdint.h>
- #include <map>
- #include <string>
- #include <utility>
- #include <vector>
- #include "opentelemetry/common/key_value_iterable_view.h"
- #include "opentelemetry/nostd/span.h"
- #include "opentelemetry/sdk/trace/sampler.h"
- #include "opentelemetry/sdk/trace/samplers/trace_id_ratio.h"
- #include "opentelemetry/trace/span_context.h"
- #include "opentelemetry/trace/span_context_kv_iterable_view.h"
- #include "opentelemetry/trace/span_id.h"
- #include "opentelemetry/trace/span_metadata.h"
- #include "opentelemetry/trace/trace_flags.h"
- #include "opentelemetry/trace/trace_id.h"
- #include "src/common/random.h"
- using opentelemetry::sdk::common::Random;
- using opentelemetry::sdk::trace::Decision;
- using opentelemetry::sdk::trace::TraceIdRatioBasedSampler;
- namespace trace_api = opentelemetry::trace;
- namespace common = opentelemetry::common;
- namespace
- {
- /*
- * Helper function for running TraceIdBased sampler tests.
- * Given a span context, sampler, and number of iterations this function
- * will return the number of RECORD_AND_SAMPLE decision based on randomly
- * generated traces.
- *
- * @param context a required valid span context
- * @param sampler a required valid sampler
- * @param iterations a required number specifying the number of times to
- * generate a random trace_id and check if it should sample using the provided
- * provider and context
- */
- int RunShouldSampleCountDecision(trace_api::SpanContext &context,
- TraceIdRatioBasedSampler &sampler,
- int iterations)
- {
- int actual_count = 0;
- trace_api::SpanKind span_kind = trace_api::SpanKind::kInternal;
- using M = std::map<std::string, int>;
- M m1 = {{}};
- using L = std::vector<std::pair<trace_api::SpanContext, std::map<std::string, std::string>>>;
- L l1 = {{trace_api::SpanContext(false, false), {}}, {trace_api::SpanContext(false, false), {}}};
- common::KeyValueIterableView<M> view{m1};
- trace_api::SpanContextKeyValueIterableView<L> links{l1};
- for (int i = 0; i < iterations; ++i)
- {
- uint8_t buf[16] = {0};
- Random::GenerateRandomBuffer(buf);
- trace_api::TraceId trace_id(buf);
- auto result = sampler.ShouldSample(context, trace_id, "", span_kind, view, links);
- if (result.decision == Decision::RECORD_AND_SAMPLE)
- {
- ++actual_count;
- }
- }
- return actual_count;
- }
- } // namespace
- TEST(TraceIdRatioBasedSampler, ShouldSampleWithoutContext)
- {
- trace_api::TraceId invalid_trace_id;
- trace_api::SpanKind span_kind = trace_api::SpanKind::kInternal;
- using M = std::map<std::string, int>;
- M m1 = {{}};
- using L = std::vector<std::pair<trace_api::SpanContext, std::map<std::string, std::string>>>;
- L l1 = {{trace_api::SpanContext(false, false), {}}, {trace_api::SpanContext(false, false), {}}};
- common::KeyValueIterableView<M> view{m1};
- trace_api::SpanContextKeyValueIterableView<L> links{l1};
- TraceIdRatioBasedSampler s1(0.01);
- auto sampling_result = s1.ShouldSample(trace_api::SpanContext::GetInvalid(), invalid_trace_id, "",
- span_kind, view, links);
- ASSERT_EQ(Decision::RECORD_AND_SAMPLE, sampling_result.decision);
- ASSERT_EQ(nullptr, sampling_result.attributes);
- constexpr uint8_t buf[] = {0, 0, 0, 0, 0, 0, 0, 0x80, 0, 0, 0, 0, 0, 0, 0, 0};
- trace_api::TraceId valid_trace_id(buf);
- sampling_result = s1.ShouldSample(trace_api::SpanContext::GetInvalid(), valid_trace_id, "",
- span_kind, view, links);
- ASSERT_EQ(Decision::DROP, sampling_result.decision);
- ASSERT_EQ(nullptr, sampling_result.attributes);
- TraceIdRatioBasedSampler s2(0.50000001);
- sampling_result = s2.ShouldSample(trace_api::SpanContext::GetInvalid(), valid_trace_id, "",
- span_kind, view, links);
- ASSERT_EQ(Decision::RECORD_AND_SAMPLE, sampling_result.decision);
- ASSERT_EQ(nullptr, sampling_result.attributes);
- TraceIdRatioBasedSampler s3(0.49999999);
- sampling_result = s3.ShouldSample(trace_api::SpanContext::GetInvalid(), valid_trace_id, "",
- span_kind, view, links);
- ASSERT_EQ(Decision::DROP, sampling_result.decision);
- ASSERT_EQ(nullptr, sampling_result.attributes);
- TraceIdRatioBasedSampler s4(0.50000000);
- sampling_result = s4.ShouldSample(trace_api::SpanContext::GetInvalid(), valid_trace_id, "",
- span_kind, view, links);
- ASSERT_EQ(Decision::RECORD_AND_SAMPLE, sampling_result.decision);
- ASSERT_EQ(nullptr, sampling_result.attributes);
- }
- TEST(TraceIdRatioBasedSampler, ShouldSampleWithContext)
- {
- uint8_t trace_id_buffer[trace_api::TraceId::kSize] = {1};
- trace_api::TraceId trace_id{trace_id_buffer};
- uint8_t span_id_buffer[trace_api::SpanId::kSize] = {1};
- trace_api::SpanId span_id{span_id_buffer};
- trace_api::SpanKind span_kind = trace_api::SpanKind::kInternal;
- trace_api::SpanContext c1(trace_id, span_id, trace_api::TraceFlags{0}, false);
- trace_api::SpanContext c2(trace_id, span_id, trace_api::TraceFlags{1}, false);
- trace_api::SpanContext c3(trace_id, span_id, trace_api::TraceFlags{0}, true);
- trace_api::SpanContext c4(trace_id, span_id, trace_api::TraceFlags{1}, true);
- using M = std::map<std::string, int>;
- M m1 = {{}};
- using L = std::vector<std::pair<trace_api::SpanContext, std::map<std::string, std::string>>>;
- L l1 = {{trace_api::SpanContext(false, false), {}}, {trace_api::SpanContext(false, false), {}}};
- common::KeyValueIterableView<M> view{m1};
- trace_api::SpanContextKeyValueIterableView<L> links{l1};
- TraceIdRatioBasedSampler s1(0.01);
- auto sampling_result = s1.ShouldSample(c1, trace_id, "", span_kind, view, links);
- ASSERT_EQ(Decision::RECORD_AND_SAMPLE, sampling_result.decision);
- ASSERT_EQ(nullptr, sampling_result.attributes);
- sampling_result = s1.ShouldSample(c2, trace_id, "", span_kind, view, links);
- ASSERT_EQ(Decision::RECORD_AND_SAMPLE, sampling_result.decision);
- ASSERT_EQ(nullptr, sampling_result.attributes);
- sampling_result = s1.ShouldSample(c3, trace_id, "", span_kind, view, links);
- ASSERT_EQ(Decision::RECORD_AND_SAMPLE, sampling_result.decision);
- ASSERT_EQ(nullptr, sampling_result.attributes);
- sampling_result = s1.ShouldSample(c4, trace_id, "", span_kind, view, links);
- ASSERT_EQ(Decision::RECORD_AND_SAMPLE, sampling_result.decision);
- ASSERT_EQ(nullptr, sampling_result.attributes);
- }
- TEST(TraceIdRatioBasedSampler, TraceIdRatioBasedSamplerHalf)
- {
- double ratio = 0.5;
- int iterations = 100000;
- int expected_count = static_cast<int>(iterations * ratio);
- int variance = static_cast<int>(iterations * 0.01);
- trace_api::SpanContext c(true, true);
- TraceIdRatioBasedSampler s(ratio);
- int actual_count = RunShouldSampleCountDecision(c, s, iterations);
- ASSERT_TRUE(actual_count < (expected_count + variance));
- ASSERT_TRUE(actual_count > (expected_count - variance));
- }
- TEST(TraceIdRatioBasedSampler, TraceIdRatioBasedSamplerOnePercent)
- {
- double ratio = 0.01;
- int iterations = 100000;
- int expected_count = static_cast<int>(iterations * ratio);
- int variance = static_cast<int>(iterations * 0.01);
- trace_api::SpanContext c(true, true);
- TraceIdRatioBasedSampler s(ratio);
- int actual_count = RunShouldSampleCountDecision(c, s, iterations);
- ASSERT_TRUE(actual_count < (expected_count + variance));
- ASSERT_TRUE(actual_count > (expected_count - variance));
- }
- TEST(TraceIdRatioBasedSampler, TraceIdRatioBasedSamplerAll)
- {
- double ratio = 1.0;
- int iterations = 100000;
- int expected_count = static_cast<int>(iterations * ratio);
- trace_api::SpanContext c(true, true);
- TraceIdRatioBasedSampler s(ratio);
- int actual_count = RunShouldSampleCountDecision(c, s, iterations);
- ASSERT_EQ(actual_count, expected_count);
- }
- TEST(TraceIdRatioBasedSampler, TraceIdRatioBasedSamplerNone)
- {
- double ratio = 0.0;
- int iterations = 100000;
- int expected_count = static_cast<int>(iterations * ratio);
- trace_api::SpanContext c(true, true);
- TraceIdRatioBasedSampler s(ratio);
- int actual_count = RunShouldSampleCountDecision(c, s, iterations);
- ASSERT_EQ(actual_count, expected_count);
- }
- TEST(TraceIdRatioBasedSampler, GetDescription)
- {
- TraceIdRatioBasedSampler s1(0.01);
- ASSERT_EQ("TraceIdRatioBasedSampler{0.010000}", s1.GetDescription());
- TraceIdRatioBasedSampler s2(0.00);
- ASSERT_EQ("TraceIdRatioBasedSampler{0.000000}", s2.GetDescription());
- TraceIdRatioBasedSampler s3(1.00);
- ASSERT_EQ("TraceIdRatioBasedSampler{1.000000}", s3.GetDescription());
- TraceIdRatioBasedSampler s4(0.102030405);
- ASSERT_EQ("TraceIdRatioBasedSampler{0.102030}", s4.GetDescription());
- TraceIdRatioBasedSampler s5(3.00);
- ASSERT_EQ("TraceIdRatioBasedSampler{1.000000}", s5.GetDescription());
- TraceIdRatioBasedSampler s6(-3.00);
- ASSERT_EQ("TraceIdRatioBasedSampler{0.000000}", s6.GetDescription());
- TraceIdRatioBasedSampler s7(1.00000000001);
- ASSERT_EQ("TraceIdRatioBasedSampler{1.000000}", s7.GetDescription());
- TraceIdRatioBasedSampler s8(-1.00000000001);
- ASSERT_EQ("TraceIdRatioBasedSampler{0.000000}", s8.GetDescription());
- TraceIdRatioBasedSampler s9(0.50);
- ASSERT_EQ("TraceIdRatioBasedSampler{0.500000}", s9.GetDescription());
- }
|