As shown in the Getting Started - ASP.NET Core
Application and Getting Started -
Console Application docs, OpenTelemetry
tracing is managed by a
TracerProvider
instance configured using the TracerProviderBuilder
API.
TracerProviderBuilder
exposes various methods to configure the provider (ex:
SetSampler
, AddProcessor
, etc.) which are explained in subsequent sections
of this document. It is also common for library authors to target
TracerProviderBuilder
for extension methods which help configure SDK plug-in
components.
There are two different ways to create a TracerProviderBuilder
.
For ASP.NET
Core and
.NET Generic
host users, helper extensions are provided in the
OpenTelemetry.Extensions.Hosting
package to simplify configuration and management of the TracerProvider
.
using OpenTelemetry.Trace;
var appBuilder = WebApplication.CreateBuilder(args);
appBuilder.Services.AddOpenTelemetry()
.WithTracing(builder => builder.AddConsoleExporter());
Note The AddOpenTelemetry extension automatically starts and stops the
TracerProvider
with the host.
Sdk.CreateTracerProviderBuilder()
is provided on all runtimes to create
TracerProvider
s when either hosting is not available or multiple providers are
required.
Call Sdk.CreateTracerProviderBuilder()
to obtain a builder and then call
Build()
once configuration is done to retrieve the TracerProvider
instance.
Note Once built changes to
TracerProvider
configuration are not allowed, with the exception of adding more processors.
In most cases a single TracerProvider
is created at the application startup,
and is disposed when application shuts down.
The snippet below shows how to build a basic TracerProvider
and dispose it at
the end of the application. This will create a provider with default
configuration, and is not particularly useful. The subsequent sections shows how
to build a more useful provider.
using OpenTelemetry;
using OpenTelemetry.Trace;
var tracerProvider = Sdk.CreateTracerProviderBuilder().Build();
// ....
// Dispose at application shutdown
tracerProvider.Dispose()
TracerProvider
holds the tracing configuration, which includes the following:
- The list of
ActivitySource
s (akaTracer
s) from which traces are collected. - The list of instrumentations enabled via InstrumentationLibrary.
- The list of Processors, including exporting processors which exports traces to Exporters
- The Resource associated with the traces.
- The Sampler to be used.
ActivitySource
denotes a
Tracer
,
which is used to create activities. The SDK follows an explicit opt-in model for
listening to activity sources. i.e, by default, it listens to no sources. Every
activity source which produce telemetry must be explicitly added to the tracer
provider to start collecting traces from them.
AddSource
method on TracerProviderBuilder
can be used to add a
ActivitySource
to the provider. The name of the ActivitySource
(case-insensitive) must be the argument to this method. Multiple AddSource
can
be called to add more than one source. It also supports wildcard subscription
model as well.
It is not possible to add sources after the provider is built, by calling the
Build()
method on the TracerProviderBuilder
.
The snippet below shows how to add activity sources to the provider.
using OpenTelemetry;
using OpenTelemetry.Trace;
var tracerProvider = Sdk.CreateTracerProviderBuilder()
// The following subscribes to activities from Activity Source
// named "MyCompany.MyProduct.MyLibrary" only.
.AddSource("MyCompany.MyProduct.MyLibrary")
// The following subscribes to activities from all Activity Sources
// whose name starts with "AbcCompany.XyzProduct.".
.AddSource("AbcCompany.XyzProduct.*")
.Build();
See Program.cs for complete example.
Note A common mistake while configuring
TracerProvider
is forgetting to add allActivitySources
to the provider. It is recommended to leverage the wild card subscription model where it makes sense. For example, if your application is expecting to enable tracing from a number of libraries from a company "Abc", the you can useAddSource("Abc.*")
to enable all sources whose name starts with "Abc.".
While the OpenTelemetry API can be used to instrument any library manually,
Instrumentation
Libraries
are available for a lot of commonly used libraries. Such instrumentations can be
added to the TracerProvider
. It is not required to attach the instrumentation
to the provider, unless the life cycle of the instrumentation must be managed by
the provider. If the instrumentation must be activated/shutdown/disposed along
with the provider, then the instrumentation must be added to the provider.
Typically, the instrumentation libraries provide extension methods on
TracerProviderBuilder
to allow adding them to the TracerProvider
. Please
refer to corresponding documentation of the instrumentation library to know the
exact method name.
Follow this document to learn about the instrumentation libraries shipped from this repo and writing custom instrumentation libraries.
Processors
expose hooks for start and end processing of Activity
instances. If no
processors are configured then traces are simply dropped by the SDK. The
AddProcessor
method on TracerProviderBuilder
is provided to add a processor
to the SDK pipeline. There can be any number of processors added to the provider
and they are invoked in the same order as they are added. Unlike Sampler
and
Resource
, processors can be added to the provider even after it is built.
Exporters
expose hooks for exporting batches of completed Activity
instances (a batch
may contain a single or many records) and are called by processors. Two base
processor classes SimpleExportProcessor
& BatchExportProcessor
are provided
to support invoking exporters through the processor pipeline and implement the
standard behaviors prescribed by the OpenTelemetry specification.
Note The SDK only ever invokes processors and has no direct knowledge of any registered exporters.
The snippet below shows how to add processors to the provider before and after it is built.
using OpenTelemetry;
using OpenTelemetry.Trace;
var tracerProvider = Sdk.CreateTracerProviderBuilder()
.AddProcessor(new MyProcessor1())
.AddProcessor(new MyProcessor2()))
.Build();
// Processors can be added to provider even after it is built.
// Only those traces which are emitted after this line, will be sent to it.
tracerProvider.AddProcessor(new MyProcessor3());
Note The order of processor registration is important. Each processor added is invoked in order by the SDK. For example if a simple exporting processor is added before an enrichment processor the exported data will not contain anything added by the enrichment because it happens after the export.
Note A
TracerProvider
assumes ownership of all processors added to it. This means that the provider will call theShutdown
method on all registered processors when it is shutting down and call theDispose
method on all registered processors when it is disposed. If multiple providers are being set up in an application then separate instances of processors MUST be registered on each provider. Otherwise shutting down one provider will cause the shared processor(s) in other providers to be shut down as well which may lead to undesired results.
Processors can be used for enriching, exporting, and/or filtering telemetry.
To enrich telemetry, users may write custom processors overriding the OnStart
and/or OnEnd
methods (as needed) to implement custom logic to change the data
before it is passed to the next processor in the pipeline.
For exporting purposes, the SDK provides the following built-in processors:
-
BatchExportProcessor<T> : This is an exporting processor which batches the telemetry before sending to the configured exporter.
The following environment variables can be used to override the default values of the
BatchExportActivityProcessorOptions
.Environment variable BatchExportActivityProcessorOptions
propertyOTEL_BSP_SCHEDULE_DELAY
ScheduledDelayMilliseconds
OTEL_BSP_EXPORT_TIMEOUT
ExporterTimeoutMilliseconds
OTEL_BSP_MAX_QUEUE_SIZE
MaxQueueSize
OTEL_BSP_MAX_EXPORT_BATCH_SIZE
MaxExportBatchSizeEnvVarKey
-
SimpleExportProcessor<T> : This is an exporting processor which passes telemetry to the configured exporter immediately without any batching.
Note A special processor CompositeProcessor<T> is used by the SDK to chain multiple processors together and may be used as needed by users to define sub-pipelines.
Note The processors shipped from this SDK are generic implementations and support tracing and logging by implementing
Activity
andLogRecord
respectively.
Follow this document to learn about writing custom processors.
The snippet below shows how to add export processors to the provider before it is built.
using OpenTelemetry;
using OpenTelemetry.Trace;
var tracerProvider = Sdk.CreateTracerProviderBuilder()
.AddProcessor(new BatchActivityExportProcessor(new MyExporter1()))
.AddProcessor(new SimpleActivityExportProcessor(new MyExporter2()))
.Build();
It is also common for exporters to provide their own extensions to simplify registration. The snippet below shows how to add the JaegerExporter to the provider before it is built.
using OpenTelemetry;
using OpenTelemetry.Trace;
var tracerProvider = Sdk.CreateTracerProviderBuilder()
.AddJaegerExporter()
.Build();
Follow this document to learn about writing custom exporters.
Resource
is the immutable representation of the entity producing the telemetry. If no
Resource
is explicitly configured, the
default
resource is used to indicate the
Service.
The ConfigureResource
method on TracerProviderBuilder
can be used to
configure the resource on the provider. ConfigureResource
accepts an Action
to configure the ResourceBuilder
. Multiple calls to ConfigureResource
can be
made. When the provider is built, it builds the final Resource
combining all
the ConfigureResource
calls. There can only be a single Resource
associated
with a provider. It is not possible to change the resource builder after the
provider is built, by calling the Build()
method on the
TracerProviderBuilder
.
ResourceBuilder
offers various methods to construct resource comprising of
multiple attributes from various sources. Examples include AddTelemetrySdk()
which adds Telemetry
Sdk
resource, and AddService()
which adds
Service
resource. It also allows adding ResourceDetector
s.
Follow this document to learn about writing custom resource detectors.
The snippet below shows configuring the Resource
associated with the provider.
using OpenTelemetry;
using OpenTelemetry.Resources;
using OpenTelemetry.Trace;
var tracerProvider = Sdk.CreateTracerProviderBuilder()
.ConfigureResource(resourceBuilder => resourceBuilder.AddTelemetrySdk())
.ConfigureResource(resourceBuilder => resourceBuilder.AddService("service-name"))
.Build();
It is also possible to configure the Resource
by using following
environmental variables:
Environment variable | Description |
---|---|
OTEL_RESOURCE_ATTRIBUTES |
Key-value pairs to be used as resource attributes. See the Resource SDK specification for more details. |
OTEL_SERVICE_NAME |
Sets the value of the service.name resource attribute. If service.name is also provided in OTEL_RESOURCE_ATTRIBUTES , then OTEL_SERVICE_NAME takes precedence. |
Samplers
are used to control the noise and overhead introduced by OpenTelemetry by
reducing the number of samples of traces collected and sent to the processors.
If no sampler is explicitly configured, the default is to use
ParentBased(root=AlwaysOn)
. SetSampler
method on TracerProviderBuilder
can
be used to set sampler. Only one sampler can be associated with a provider. If
SetSampler
is called multiple times, the last one wins. Also, it is not
possible to change the sampler after the provider is built, by calling the
Build()
method on the TracerProviderBuilder
.
The snippet below shows configuring a custom sampler to the provider.
using OpenTelemetry;
using OpenTelemetry.Trace;
var tracerProvider = Sdk.CreateTracerProviderBuilder()
.SetSampler(new TraceIdRatioBasedSampler(0.25))
.Build();
Follow this document to learn about writing custom samplers.
The OpenTelemetry API exposes a method to obtain the default propagator which is no-op, by default. This SDK replaces the no-op with a composite propagator containing the Baggage Propagator and TraceContext propagator. This default propagator can be overridden with the below snippet.
using OpenTelemetry;
Sdk.SetDefaultTextMapPropagator(new MyCustomPropagator());
Note This information applies to the OpenTelemetry SDK version 1.4.0 and newer only.
The SDK implementation of TracerProviderBuilder
is backed by an
IServiceCollection
and supports a wide range of APIs to enable what is
generally known as dependency
injection.
For the below examples imagine a processor with this constructor:
public class MyCustomProcessor : BaseProcessor<Activity>
{
public MyCustomProcessor(MyCustomService myCustomService)
{
// Implementation not important
}
}
We want to inject MyCustomService
dependency into our MyCustomProcessor
instance.
To register MyCustomProcessor
and MyCustomService
we can use the
ConfigureServices
and AddProcessor
methods:
using var tracerProvider = Sdk.CreateTracerProviderBuilder()
.ConfigureServices(services =>
{
services.AddSingleton<MyCustomService>();
})
.AddProcessor<MyCustomProcessor>()
.Build();
When using the Sdk.CreateTracerProviderBuilder
method the TracerProvider
owns its own IServiceCollection
. It will only be able to see services
registered into that collection.
Note It is important to correctly manage the lifecycle of the
TracerProvider
. See Building a TracerProvider for details.
Note If you are authoring an ASP.NET Core application or using the .NET Generic Host the OpenTelemetry.Extensions.Hosting package is the recommended mechanism.
var appBuilder = WebApplication.CreateBuilder(args);
appBuilder.Services.AddSingleton<MyCustomService>();
appBuilder.Services.AddOpenTelemetry()
.WithTracing(builder => builder
.AddProcessor<MyCustomProcessor>());
When using the AddOpenTelemetry
& WithTracing
extension methods the
TracerProvider
does not own its IServiceCollection
and instead registers
into an existing collection (typically the collection used is the one managed by
the application host). The TracerProviderBuilder
will be able to access all
services registered into that collection. For lifecycle management, the
AddOpenTelemetry
registers an IHostedService
which is used to automatically start the TracerProvider
when the host starts
and the host will automatically shutdown and dispose the TracerProvider
when
it is shutdown.
Note Multiple calls to
WithTracing
will configure the sameTracerProvider
. Only a singleTraceProvider
may exist in anIServiceCollection
\IServiceProvider
.
-
AddInstrumentation<T>
: Adds instrumentation of typeT
into theTracerProvider
. -
AddInstrumentation<T>(Func<IServiceProvider, T> instrumentationFactory)
: Adds instrumentation of typeT
into theTracerProvider
using a factory function to create the instrumentation instance. -
AddProcessor<T>
: Adds a processor of typeT
(must derive fromBaseProcessor<Activity>
) into theTracerProvider
. -
AddProcessor(Func<IServiceProvider, BaseProcessor<Activity>> implementationFactory)
: Adds a processor into theTracerProvider
using a factory function to create the processor instance. -
ConfigureServices
: Registers a callback function for configuring theIServiceCollection
used by theTracerProviderBuilder
.Note
ConfigureServices
may only be called before theIServiceProvider
has been created after which point services can no longer be added. -
SetSampler<T>
: Register typeT
(must derive fromSampler
) as the sampler for theTracerProvider
. -
SetSampler(Func<IServiceProvider, Sampler> implementationFactory)
: Adds a sampler into theTracerProvider
using a factory function to create the sampler instance.
Note The factory functions accepting
IServiceProvider
may always be used regardless of how the SDK is initialized. When using an external service collection (ex:appBuilder.Services.AddOpenTelemetry()
), as is common in ASP.NET Core hosts, theIServiceProvider
will be the instance shared and managed by the host. When using "Sdk.Create" functions, as is common in .NET Framework hosts, the provider creates its ownIServiceCollection
and will build anIServiceProvider
from it to make available to extensions.
Note This information applies to the OpenTelemetry SDK version 1.4.0 and newer only.
The OpenTelemetry .NET SDK integrates with the standard configuration and options patterns provided by .NET. The configuration pattern supports building a composited view of settings from external sources and the options pattern helps use those settings to configure features by binding to simple classes.
The following sections describe how to set up configuration based on the host and OpenTelemetry API being used.
ASP.NET
Core and
.NET Generic
host users using the
OpenTelemetry.Extensions.Hosting
package do not need to do anything extra to enable IConfiguration
support. The
OpenTelemetry SDK will automatically use whatever IConfiguration
has been
supplied by the host. The host by default will load environment variables,
command-line arguments, and config files. See Configuration in
.NET for
details.
By default the Sdk.CreateTracerProviderBuilder
API will create an
IConfiguration
from environment variables. The following example shows how to
customize the IConfiguration
used by Sdk.CreateTracerProviderBuilder
for
cases where additional sources beyond environment variables are required.
// Build configuration from sources. Order is important.
var configuration = new ConfigurationBuilder()
.AddJsonFile("./myOTelSettings.json")
.AddEnvironmentVariables()
.AddCommandLine(args)
.Build();
// Set up a TracerProvider using the configuration.
var provider = Sdk.CreateTracerProviderBuilder()
.ConfigureServices(services => services.AddSingleton<IConfiguration>(configuration))
.Build();
The OpenTelemetry Specification defines specific environment variables which may be used to configure SDK implementations.
The OpenTelemetry .NET SDK will look for the environment variables defined in
the specification using IConfiguration
which means in addition to environment
variables users may also manage these settings via the command-line,
configuration files, or any other source registered with the .NET configuration
engine. This provides greater flexibility than what the specification defines.
Note Not all of the environment variables defined in the specification are supported. Consult the individual project README files for details on specific environment variable support.
As an example the OpenTelemetry Specification defines the OTEL_SERVICE_NAME
environment variable which may be used to configure the service name emitted on
telemetry by the SDK.
A traditional environment variable is set using a command like set OTEL_SERVICE_NAME=MyService
on Windows or export OTEL_SERVICE_NAME=MyService
on Linux.
That works as expected but the OpenTelemetry .NET SDK is actually looking for
the OTEL_SERVICE_NAME
key in IConfiguration
which means it may also be
configured in any configuration source registered with the
IConfigurationBuilder
used to create the final configuration for the host.
Below are two examples of configuring the OTEL_SERVICE_NAME
setting beyond
environment variables.
-
Using appsettings.json:
{ "OTEL_SERVICE_NAME": "MyService" }
-
Using command-line:
dotnet run --OTEL_SERVICE_NAME "MyService"
Note The .NET Configuration pattern is hierarchical meaning the order of registered configuration sources controls which value will seen by the SDK when it is defined in multiple sources.
Options are typically simple classes containing only properties with public
"getters" and "setters" (aka POCOs) and have "Options" at the end of the class
name. These options classes are primarily used when interacting with the
TracerProviderBuilder
to control settings and features of the different SDK
components.
Options classes can always be configured through code but users typically want to control key settings through configuration.
The following example shows how to configure JaegerExporterOptions
by binding
to an IConfiguration
section.
Json config file (usually appsettings.json):
{
"OpenTelemetry": {
"Jaeger": {
"Protocol": "UdpCompactThrift"
"AgentHost": "localhost",
"AgentPort": 6831,
"BatchExportProcessorOptions": {
"ScheduledDelayMilliseconds": 5000
}
}
}
}
Code:
var appBuilder = WebApplication.CreateBuilder(args);
appBuilder.Services.Configure<JaegerExporterOptions>(
appBuilder.Configuration.GetSection("OpenTelemetry:Jaeger"));
appBuilder.Services.AddOpenTelemetry()
.WithTracing(builder => builder.AddJaegerExporter());
The OpenTelemetry .NET SDK supports running multiple TracerProvider
s inside
the same application and it also supports registering multiple similar
components such as exporters into a single TracerProvider
. In order to allow
users to target configuration at specific components a "name" parameter is
typically supported on configuration extensions to control the options instance
used for the component being registered.
The below example shows how to configure two JaegerExporter
instances inside a
single TracerProvider
sending to different ports.
Json config file (usually appsettings.json):
{
"OpenTelemetry": {
"JaegerPrimary": {
"AgentPort": 1818
},
"JaegerSecondary": {
"AgentPort": 8818
}
}
}
Code:
var appBuilder = WebApplication.CreateBuilder(args);
appBuilder.Services.Configure<JaegerExporterOptions>(
"JaegerPrimary",
appBuilder.Configuration.GetSection("OpenTelemetry:JaegerPrimary"));
appBuilder.Services.Configure<JaegerExporterOptions>(
"JaegerSecondary",
appBuilder.Configuration.GetSection("OpenTelemetry:JaegerSecondary"));
appBuilder.Services.AddOpenTelemetry()
.WithTracing(builder => builder
.AddJaegerExporter(name: "JaegerPrimary", configure: null)
.AddJaegerExporter(name: "JaegerSecondary", configure: null));