// Copyright The OpenTelemetry Authors // SPDX-License-Identifier: Apache-2.0 #include #include #include #include #include #include #include #include "opentelemetry/exporters/ostream/span_exporter_factory.h" #include "opentelemetry/nostd/shared_ptr.h" #include "opentelemetry/nostd/string_view.h" #include "opentelemetry/sdk/resource/resource.h" #include "opentelemetry/sdk/trace/batch_span_processor_factory.h" #include "opentelemetry/sdk/trace/batch_span_processor_options.h" #include "opentelemetry/sdk/trace/exporter.h" #include "opentelemetry/sdk/trace/processor.h" #include "opentelemetry/sdk/trace/provider.h" #include "opentelemetry/sdk/trace/tracer_provider.h" #include "opentelemetry/sdk/trace/tracer_provider_factory.h" #include "opentelemetry/trace/provider.h" #include "opentelemetry/trace/tracer.h" #include "opentelemetry/trace/tracer_provider.h" constexpr int kNumSpans = 10; namespace trace_api = opentelemetry::trace; namespace resource = opentelemetry::sdk::resource; namespace trace_sdk = opentelemetry::sdk::trace; namespace trace_exporter = opentelemetry::exporter::trace; namespace { void InitTracer() { auto exporter = trace_exporter::OStreamSpanExporterFactory::Create(); // CONFIGURE BATCH SPAN PROCESSOR PARAMETERS trace_sdk::BatchSpanProcessorOptions options{}; // We make the queue size `KNumSpans`*2+5 because when the queue is half full, a preemptive notif // is sent to start an export call, which we want to avoid in this simple example. options.max_queue_size = kNumSpans * 2 + 5; // Time interval (in ms) between two consecutive exports. options.schedule_delay_millis = std::chrono::milliseconds(3000); // We export `kNumSpans` after every `schedule_delay_millis` milliseconds. options.max_export_batch_size = kNumSpans; resource::ResourceAttributes attributes = {{"service", "test_service"}, {"version", static_cast(1)}}; auto resource = resource::Resource::Create(attributes); auto processor = trace_sdk::BatchSpanProcessorFactory::Create(std::move(exporter), options); std::shared_ptr provider = trace_sdk::TracerProviderFactory::Create(std::move(processor), resource); // Set the global trace provider. trace_sdk::Provider::SetTracerProvider(provider); } void CleanupTracer() { std::shared_ptr none; trace_sdk::Provider::SetTracerProvider(none); } opentelemetry::nostd::shared_ptr get_tracer() { auto provider = trace_api::Provider::GetTracerProvider(); return provider->GetTracer("foo_library"); } void StartAndEndSpans() { for (int i = 1; i <= kNumSpans; ++i) { get_tracer()->StartSpan("Span " + std::to_string(i)); } } } // namespace int main(int /* argc */, char ** /* argv */) { // Removing this line will leave the default noop TracerProvider in place. InitTracer(); std::cout << "Creating first batch of " << kNumSpans << " spans and waiting 3 seconds ...\n"; StartAndEndSpans(); std::this_thread::sleep_for(std::chrono::milliseconds(3000 + 50)); // The spans should now be exported. std::cout << "....Exported!\n\n\n"; // Do the same again std::cout << "Creating second batch of " << kNumSpans << " spans and waiting 3 seconds ...\n"; StartAndEndSpans(); std::this_thread::sleep_for(std::chrono::milliseconds(3000 + 50)); std::cout << "....Exported!\n\n\n"; // Shutdown and drain queue StartAndEndSpans(); printf("Shutting down and draining queue.... \n"); std::this_thread::sleep_for(std::chrono::milliseconds(2000)); // We invoke the processor destructor // which in turn invokes the processor Shutdown(), which finally drains the queue of ALL // its spans. CleanupTracer(); return 0; }