composite_propagator_test.cc 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152
  1. // Copyright The OpenTelemetry Authors
  2. // SPDX-License-Identifier: Apache-2.0
  3. #include <gtest/gtest.h>
  4. #include <stdint.h>
  5. #include <algorithm>
  6. #include <map>
  7. #include <string>
  8. #include <utility>
  9. #include <vector>
  10. #include "opentelemetry/context/context.h"
  11. #include "opentelemetry/context/propagation/composite_propagator.h"
  12. #include "opentelemetry/context/propagation/text_map_propagator.h"
  13. #include "opentelemetry/context/runtime_context.h"
  14. #include "opentelemetry/nostd/function_ref.h"
  15. #include "opentelemetry/nostd/shared_ptr.h"
  16. #include "opentelemetry/nostd/span.h"
  17. #include "opentelemetry/nostd/string_view.h"
  18. #include "opentelemetry/nostd/variant.h"
  19. #include "opentelemetry/trace/default_span.h"
  20. #include "opentelemetry/trace/propagation/b3_propagator.h"
  21. #include "opentelemetry/trace/propagation/http_trace_context.h"
  22. #include "opentelemetry/trace/scope.h"
  23. #include "opentelemetry/trace/span.h"
  24. #include "opentelemetry/trace/span_context.h"
  25. #include "opentelemetry/trace/span_id.h"
  26. #include "opentelemetry/trace/span_metadata.h"
  27. #include "opentelemetry/trace/trace_flags.h"
  28. #include "opentelemetry/trace/trace_id.h"
  29. using namespace opentelemetry;
  30. template <typename T>
  31. static std::string Hex(const T &id_item)
  32. {
  33. char buf[T::kSize * 2];
  34. id_item.ToLowerBase16(buf);
  35. return std::string(buf, sizeof(buf));
  36. }
  37. class TextMapCarrierTest : public context::propagation::TextMapCarrier
  38. {
  39. public:
  40. nostd::string_view Get(nostd::string_view key) const noexcept override
  41. {
  42. auto it = headers_.find(std::string(key));
  43. if (it != headers_.end())
  44. {
  45. return nostd::string_view(it->second);
  46. }
  47. return "";
  48. }
  49. void Set(nostd::string_view key, nostd::string_view value) noexcept override
  50. {
  51. headers_[std::string(key)] = std::string(value);
  52. }
  53. std::map<std::string, std::string> headers_;
  54. };
  55. class CompositePropagatorTest : public ::testing::Test
  56. {
  57. public:
  58. CompositePropagatorTest()
  59. {
  60. std::vector<std::unique_ptr<context::propagation::TextMapPropagator>> propogator_list = {};
  61. std::unique_ptr<context::propagation::TextMapPropagator> w3c_propogator(
  62. new trace::propagation::HttpTraceContext());
  63. std::unique_ptr<context::propagation::TextMapPropagator> b3_propogator(
  64. new trace::propagation::B3Propagator());
  65. propogator_list.push_back(std::move(w3c_propogator));
  66. propogator_list.push_back(std::move(b3_propogator));
  67. composite_propagator_ =
  68. new context::propagation::CompositePropagator(std::move(propogator_list));
  69. }
  70. ~CompositePropagatorTest() override { delete composite_propagator_; }
  71. protected:
  72. context::propagation::CompositePropagator *composite_propagator_;
  73. };
  74. TEST_F(CompositePropagatorTest, Extract)
  75. {
  76. TextMapCarrierTest carrier;
  77. carrier.headers_ = {
  78. {"traceparent", "00-4bf92f3577b34da6a3ce929d0e0e4736-0102030405060708-01"},
  79. {"b3", "80f198ee56343ba864fe8b2a57d3eff7-e457b5a2e4d86bd1-1-05e3ac9a4f6e3b90"}};
  80. context::Context ctx1 = context::Context{};
  81. context::Context ctx2 = composite_propagator_->Extract(carrier, ctx1);
  82. auto ctx2_span = ctx2.GetValue(trace::kSpanKey);
  83. EXPECT_TRUE(nostd::holds_alternative<nostd::shared_ptr<trace::Span>>(ctx2_span));
  84. auto span = nostd::get<nostd::shared_ptr<trace::Span>>(ctx2_span);
  85. // confirm last propagator in composite propagator list (B3 here) wins for same key
  86. // ("active_span" here).
  87. EXPECT_EQ(Hex(span->GetContext().trace_id()), "80f198ee56343ba864fe8b2a57d3eff7");
  88. EXPECT_EQ(Hex(span->GetContext().span_id()), "e457b5a2e4d86bd1");
  89. EXPECT_EQ(span->GetContext().IsSampled(), true);
  90. EXPECT_EQ(span->GetContext().IsRemote(), true);
  91. // Now check that last propagator does not win if there is no header for it
  92. carrier.headers_ = {{"traceparent", "00-4bf92f3577b34da6a3ce929d0e0e4736-0102030405060708-00"}};
  93. ctx1 = context::Context{};
  94. ctx2 = composite_propagator_->Extract(carrier, ctx1);
  95. ctx2_span = ctx2.GetValue(trace::kSpanKey);
  96. EXPECT_TRUE(nostd::holds_alternative<nostd::shared_ptr<trace::Span>>(ctx2_span));
  97. span = nostd::get<nostd::shared_ptr<trace::Span>>(ctx2_span);
  98. // Here the first propagator (W3C) wins
  99. EXPECT_EQ(Hex(span->GetContext().trace_id()), "4bf92f3577b34da6a3ce929d0e0e4736");
  100. EXPECT_EQ(Hex(span->GetContext().span_id()), "0102030405060708");
  101. EXPECT_EQ(span->GetContext().IsSampled(), false);
  102. EXPECT_EQ(span->GetContext().IsRemote(), true);
  103. }
  104. TEST_F(CompositePropagatorTest, Inject)
  105. {
  106. TextMapCarrierTest carrier;
  107. constexpr uint8_t buf_span[] = {1, 2, 3, 4, 5, 6, 7, 8};
  108. constexpr uint8_t buf_trace[] = {1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16};
  109. trace::SpanContext span_context{trace::TraceId{buf_trace}, trace::SpanId{buf_span},
  110. trace::TraceFlags{true}, false};
  111. nostd::shared_ptr<trace::Span> sp{new trace::DefaultSpan{span_context}};
  112. // Set `sp` as the currently active span, which must be used by `Inject`.
  113. trace::Scope scoped_span{sp};
  114. composite_propagator_->Inject(carrier, context::RuntimeContext::GetCurrent());
  115. EXPECT_EQ(carrier.headers_["traceparent"],
  116. "00-0102030405060708090a0b0c0d0e0f10-0102030405060708-01");
  117. EXPECT_EQ(carrier.headers_["b3"], "0102030405060708090a0b0c0d0e0f10-0102030405060708-1");
  118. std::vector<std::string> fields;
  119. composite_propagator_->Fields([&fields](nostd::string_view field) {
  120. fields.push_back(field.data());
  121. return true;
  122. });
  123. EXPECT_EQ(fields.size(), 3);
  124. EXPECT_EQ(fields[0], trace::propagation::kTraceParent);
  125. EXPECT_EQ(fields[1], trace::propagation::kTraceState);
  126. EXPECT_EQ(fields[2], trace::propagation::kB3CombinedHeader);
  127. }