Release observable features by using FeatureGates, instrumented abstractions that replace if
statements.
The aim of FeatureGates is to enable reusable dashboards, tooling and automation by standardizing the way metrics and traces are emitted. These metrics and traces can then be collected and shipped via OpenTelemetry or by other means.
FeatureGates are vendor-agnostic and under the hood uses .NET metrics and .NET distributed tracing.
Install the package from NuGet.org by running;
dotnet add package FeatureGates --version <version>
Instead of surrounding your feature with an if
statement like the code below;
string myFeatureFlag = "MyCoolNewFeature";
bool IsEnabled() => myFeatureManager.Evaluate(myFeatureFlag);
if (IsEnabled())
{
// My code when IsEnabled() returns 'true'.
}
else
{
// Optional: My code when IsEnabled() returns 'false'.
}
Create a feature gate and invoke it like the following;
string myFeatureFlag = "MyCoolNewFeature";
bool IsEnabled() => myFeatureManager.Evaluate(myFeatureFlag);
var featureGate = FeatureGate.WithKey(myFeatureFlag)
.ControlledBy(IsEnabled)
.WhenOpened(() =>
{
// My code when IsEnabled() returns 'true'.
})
.WhenClosed(() =>
{
// Optional: My code when IsEnabled() returns 'false'.
});
featureGate.Invoke();
Feature gate executions are recorded every time a feature gate is invoked. A feature gate may be invoked multiple times.
The easiest way to create a feature gate is to call the static method FeatureGate.WithKey("MyFeatureGateKey")
which will return a FeatureGateBuilder
. The FeatureGateBuilder
will allow you to chain methods to help you correctly create the type of feature gate you need.
Alternatively you can instantiate the type of feature gate you need by calling its constructor. There are only four types;
FeatureGate
FeatureGateAsync
FeatureGate<TResult>
FeatureGateAsync<TResult>
Once created, feature gates are immutable.
The feature gate key is a unique string identifier that you define for each of your feature gates. Using the same name as the feature flag that controls your feature gate is recommended. If your feature flag controls more than one feature gate, adding an identifying suffix to the feature gate key for each feature gate is recommended.
Warning - Failure to give each of your feature gates in your application a unique feature gate key will result in inaccurate metrics being collected for those feature gates.
By default, feature gates are configured to record executions using a counter. If you want to capture timings for your feature, you can change the instrument type of your feature gate during its construction to a histogram. Histograms should be used cautiously as they use more memory than counters.
A feature gate can be configured to execute its WhenClosed
delegate when an uncaught exception is thrown during execution of its WhenOpened
delegate. This behavior is not enabled by default and can be enabled during construction of your feature gate.
An alternative FeatureGate
class has been provided in the child namespace FeatureGates.Static
and is intended to be used for high-performance or memory-intensive scenarios. This alternative class is static and removes the need to new
up feature gate instances by providing static methods that perform the same functionality found in the non-static versions.
Invoke this functionality by calling one of the Invoke()
or InvokeAsync()
methods on the static FeatureGate
with all the required parameters.
Feature gates are typically removed from code after a successful rollout of a feature. In the event you want to continue to record executions after the removal, you can replace your feature gate with a call to the RecordExecution()
or RecordExecutionAsync()
methods on the static FeatureGate
. These methods immediately execute your code and records its execution as an opened
feature gate execution.
Warning - Metric names are currently unstable.
Depending on the instrument type configured for a feature gate, feature gates will output one of the following metrics.
Name | Instrument Type | Unit | Description |
---|---|---|---|
feature.gate.executions |
Counter | executions | Measures the number of times a feature gate has been executed. |
feature.gate.duration |
Histogram | milliseconds | Measures the duration of feature gate executions. |
To derive a full set of RED (Rate, Error and Duration) metrics for your feature, configure your feature gate to use a histogram. Histograms should be used cautiously as they use more memory than counters. Use a counter if you do not need to record execution durations.
Warning - Metric attributes are currently unstable.
The metrics that feature gates output can be aggregated or filtered on the following dimensions.
Name | Description |
---|---|
feature.gate.key |
The unique string identifier for a feature gate. |
feature.gate.state |
Whether a feature gate was executed as opened or closed . |
feature.gate.exception |
true if an uncaught exception occurred during execution, otherwise false . |
In order to collect metrics from your feature gates, you will need to subscribe to the Meter
named FeatureGates
. If using the OpenTelemetry .NET SDK, you can do this by calling AddMeter("FeatureGates")
while building your MeterProvider
. For example;
MeterProvider meterProvider = Sdk.CreateMeterProviderBuilder()
// Other configuration
.AddMeter("FeatureGates")
// Other configuration
.Build();
For those using OpenTelemetry .NET automatic instrumentation, you can subscribe by appending FeatureGates
to the environment variable OTEL_DOTNET_AUTO_METRICS_ADDITIONAL_SOURCES
.
It is also possible to subscribe to the metrics from your feature gates by using a MeterListener.
To learn more on how to collect metrics, please checkout the metrics collection tutorial at the .NET documentation website.
Feature gates also output spans that represent feature gate executions to be included in traces. These spans will have the operation name feature.gate.execution
so that you can easily identify them in a trace.
A span is also known as an Activity in .NET.
Warning - Span attributes are currently unstable.
Spans emitted by feature gates will have the following attributes.
Name | Description |
---|---|
feature.gate.key |
The unique string identifier for a feature gate. |
feature.gate.state |
Whether a feature gate was executed as opened or closed . |
otel.status_code |
OK if the execution was successful otherwise ERROR if an uncaught exception occurred. |
otel.status_description |
A description that is only present when otel.status_code is set to ERROR . |
When an uncaught exception occurs during a feature gate execution, a span event will be added to the span with the details of the exception. The span event will have the below attributes which follow the OpenTelemetry Trace Semantic Conventions for Exceptions.
Name | Description |
---|---|
event |
The event name which will always be the value exception . |
exception.message |
The exception message. |
exception.stacktrace |
A stacktrace related to the exception. |
exception.type |
The type of the exception including its namespace. |
In order to collect spans from your feature gates, you will need to subscribe to the ActivitySource
named FeatureGates
. If using the OpenTelemetry .NET SDK, you can do this by calling AddSource("FeatureGates")
while building your TracerProvider
. For example;
TracerProvider tracerProvider = Sdk.CreateTracerProviderBuilder()
// Other configuration
.AddSource("FeatureGates")
// Other configuration
.Build();
For those using OpenTelemetry .NET automatic instrumentation, you can subscribe by appending FeatureGates
to the environment variable OTEL_DOTNET_AUTO_TRACES_ADDITIONAL_SOURCES
.
It is also possible to subscribe to spans from your feature gates by using an ActivityListener.
To learn more on how to collect spans, please checkout the trace collection tutorial at the .NET documentation website.