FeatureGates 1.0.0-preview.5
See the version list below for details.
dotnet add package FeatureGates --version 1.0.0-preview.5
NuGet\Install-Package FeatureGates -Version 1.0.0-preview.5
<PackageReference Include="FeatureGates" Version="1.0.0-preview.5" />
paket add FeatureGates --version 1.0.0-preview.5
#r "nuget: FeatureGates, 1.0.0-preview.5"
// Install FeatureGates as a Cake Addin #addin nuget:?package=FeatureGates&version=1.0.0-preview.5&prerelease // Install FeatureGates as a Cake Tool #tool nuget:?package=FeatureGates&version=1.0.0-preview.5&prerelease
FeatureGates for .NET
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>
Usage and Features
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.
Construction
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.
Feature Gate Key
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.
Instrument Types
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.
Fallback on Exception
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.
The Static 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.
Tidying Up
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.
Metrics
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. |
Metric Attributes
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 . |
Metric Collection
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();
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.
Traces
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.
Span Attributes
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 . |
Span Events
Warning - Span event attributes are currently unstable.
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 following details.
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. |
Span Collection
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();
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.
Product | Versions Compatible and additional computed target framework versions. |
---|---|
.NET | net6.0 is compatible. net6.0-android was computed. net6.0-ios was computed. net6.0-maccatalyst was computed. net6.0-macos was computed. net6.0-tvos was computed. net6.0-windows was computed. net7.0 was computed. net7.0-android was computed. net7.0-ios was computed. net7.0-maccatalyst was computed. net7.0-macos was computed. net7.0-tvos was computed. net7.0-windows was computed. net8.0 was computed. net8.0-android was computed. net8.0-browser was computed. net8.0-ios was computed. net8.0-maccatalyst was computed. net8.0-macos was computed. net8.0-tvos was computed. net8.0-windows was computed. |
-
net6.0
- No dependencies.
NuGet packages
This package is not used by any NuGet packages.
GitHub repositories
This package is not used by any popular GitHub repositories.
Version | Downloads | Last updated |
---|---|---|
1.0.0 | 100 | 9/18/2024 |
1.0.0-preview.5 | 128 | 9/6/2022 |
1.0.0-preview.4 | 114 | 8/27/2022 |
1.0.0-preview.3 | 119 | 8/18/2022 |
1.0.0-preview.2 | 124 | 8/12/2022 |
1.0.0-preview.1 | 109 | 7/30/2022 |