summaryrefslogtreecommitdiffstats
path: root/src/common/tracer.cc
diff options
context:
space:
mode:
Diffstat (limited to 'src/common/tracer.cc')
-rw-r--r--src/common/tracer.cc113
1 files changed, 113 insertions, 0 deletions
diff --git a/src/common/tracer.cc b/src/common/tracer.cc
new file mode 100644
index 000000000..ffabc0b20
--- /dev/null
+++ b/src/common/tracer.cc
@@ -0,0 +1,113 @@
+// -*- mode:C++; tab-width:8; c-basic-offset:2; indent-tabs-mode:t -*-
+// vim: ts=8 sw=2 smarttab
+
+#include "common/ceph_context.h"
+#include "global/global_context.h"
+#include "tracer.h"
+
+#ifdef HAVE_JAEGER
+#include "opentelemetry/sdk/trace/batch_span_processor.h"
+#include "opentelemetry/sdk/trace/tracer_provider.h"
+#include "opentelemetry/exporters/jaeger/jaeger_exporter.h"
+
+namespace tracing {
+
+const opentelemetry::nostd::shared_ptr<opentelemetry::trace::Tracer> Tracer::noop_tracer = opentelemetry::trace::Provider::GetTracerProvider()->GetTracer("no-op", OPENTELEMETRY_SDK_VERSION);
+const jspan Tracer::noop_span = noop_tracer->StartSpan("noop");
+
+using bufferlist = ceph::buffer::list;
+
+Tracer::Tracer(opentelemetry::nostd::string_view service_name) {
+ init(service_name);
+}
+
+void Tracer::init(opentelemetry::nostd::string_view service_name) {
+ if (!tracer) {
+ opentelemetry::exporter::jaeger::JaegerExporterOptions exporter_options;
+ if (g_ceph_context) {
+ exporter_options.server_port = g_ceph_context->_conf.get_val<int64_t>("jaeger_agent_port");
+ }
+ const opentelemetry::sdk::trace::BatchSpanProcessorOptions processor_options;
+ const auto jaeger_resource = opentelemetry::sdk::resource::Resource::Create(std::move(opentelemetry::sdk::resource::ResourceAttributes{{"service.name", service_name}}));
+ auto jaeger_exporter = std::unique_ptr<opentelemetry::sdk::trace::SpanExporter>(new opentelemetry::exporter::jaeger::JaegerExporter(exporter_options));
+ auto processor = std::unique_ptr<opentelemetry::sdk::trace::SpanProcessor>(new opentelemetry::sdk::trace::BatchSpanProcessor(std::move(jaeger_exporter), processor_options));
+ const auto provider = opentelemetry::nostd::shared_ptr<opentelemetry::trace::TracerProvider>(new opentelemetry::sdk::trace::TracerProvider(std::move(processor), jaeger_resource));
+ opentelemetry::trace::Provider::SetTracerProvider(provider);
+ tracer = provider->GetTracer(service_name, OPENTELEMETRY_SDK_VERSION);
+ }
+}
+
+jspan Tracer::start_trace(opentelemetry::nostd::string_view trace_name) {
+ if (is_enabled()) {
+ return tracer->StartSpan(trace_name);
+ }
+ return noop_span;
+}
+
+jspan Tracer::start_trace(opentelemetry::nostd::string_view trace_name, bool trace_is_enabled) {
+ if (trace_is_enabled) {
+ return tracer->StartSpan(trace_name);
+ }
+ return noop_tracer->StartSpan(trace_name);
+}
+
+jspan Tracer::add_span(opentelemetry::nostd::string_view span_name, const jspan& parent_span) {
+ if (is_enabled() && parent_span->IsRecording()) {
+ opentelemetry::trace::StartSpanOptions span_opts;
+ span_opts.parent = parent_span->GetContext();
+ return tracer->StartSpan(span_name, span_opts);
+ }
+ return noop_span;
+}
+
+jspan Tracer::add_span(opentelemetry::nostd::string_view span_name, const jspan_context& parent_ctx) {
+ if (is_enabled() && parent_ctx.IsValid()) {
+ opentelemetry::trace::StartSpanOptions span_opts;
+ span_opts.parent = parent_ctx;
+ return tracer->StartSpan(span_name, span_opts);
+ }
+ return noop_span;
+}
+
+bool Tracer::is_enabled() const {
+ return g_ceph_context->_conf->jaeger_tracing_enable;
+}
+
+void encode(const jspan_context& span_ctx, bufferlist& bl, uint64_t f) {
+ ENCODE_START(1, 1, bl);
+ using namespace opentelemetry;
+ using namespace trace;
+ auto is_valid = span_ctx.IsValid();
+ encode(is_valid, bl);
+ if (is_valid) {
+ encode_nohead(std::string_view(reinterpret_cast<const char*>(span_ctx.trace_id().Id().data()), TraceId::kSize), bl);
+ encode_nohead(std::string_view(reinterpret_cast<const char*>(span_ctx.span_id().Id().data()), SpanId::kSize), bl);
+ encode(span_ctx.trace_flags().flags(), bl);
+ }
+ ENCODE_FINISH(bl);
+}
+
+void decode(jspan_context& span_ctx, bufferlist::const_iterator& bl) {
+ using namespace opentelemetry;
+ using namespace trace;
+ DECODE_START(1, bl);
+ bool is_valid;
+ decode(is_valid, bl);
+ if (is_valid) {
+ std::array<uint8_t, TraceId::kSize> trace_id;
+ std::array<uint8_t, SpanId::kSize> span_id;
+ uint8_t flags;
+ decode(trace_id, bl);
+ decode(span_id, bl);
+ decode(flags, bl);
+ span_ctx = SpanContext(
+ TraceId(nostd::span<uint8_t, TraceId::kSize>(trace_id)),
+ SpanId(nostd::span<uint8_t, SpanId::kSize>(span_id)),
+ TraceFlags(flags),
+ true);
+ }
+ DECODE_FINISH(bl);
+}
+} // namespace tracing
+
+#endif // HAVE_JAEGER