Skip to content

Commit f5fc1dd

Browse files
authored
feat: add support for eventing (#166)
Signed-off-by: Florian Bacher <[email protected]>
1 parent d0c25af commit f5fc1dd

14 files changed

+1054
-11
lines changed

README.md

+38-1
Original file line numberDiff line numberDiff line change
@@ -96,7 +96,7 @@ public async Task Example()
9696
|| [Hooks](#hooks) | Add functionality to various stages of the flag evaluation life-cycle. |
9797
|| [Logging](#logging) | Integrate with popular logging packages. |
9898
|| [Named clients](#named-clients) | Utilize multiple providers in a single application. |
99-
| | [Eventing](#eventing) | React to state changes in the provider or flag management system. |
99+
| | [Eventing](#eventing) | React to state changes in the provider or flag management system. |
100100
|| [Shutdown](#shutdown) | Gracefully clean up a provider during application shutdown. |
101101
|| [Extending](#extending) | Extend OpenFeature with custom providers and hooks. |
102102

@@ -167,6 +167,43 @@ client.AddHooks(new ExampleClientHook());
167167
var value = await client.GetBooleanValue("boolFlag", false, context, new FlagEvaluationOptions(new ExampleInvocationHook()));
168168
```
169169

170+
### Eventing
171+
172+
Events allow you to react to state changes in the provider or underlying flag management system, such as flag definition changes,
173+
provider readiness, or error conditions.
174+
Initialization events (`PROVIDER_READY` on success, `PROVIDER_ERROR` on failure) are dispatched for every provider.
175+
Some providers support additional events, such as `PROVIDER_CONFIGURATION_CHANGED`.
176+
177+
Please refer to the documentation of the provider you're using to see what events are supported.
178+
179+
Example usage of an Event handler:
180+
181+
```csharp
182+
public static void EventHandler(ProviderEventPayload eventDetails)
183+
{
184+
Console.WriteLine(eventDetails.Type);
185+
}
186+
```
187+
188+
```csharp
189+
EventHandlerDelegate callback = EventHandler;
190+
// add an implementation of the EventHandlerDelegate for the PROVIDER_READY event
191+
Api.Instance.AddHandler(ProviderEventTypes.ProviderReady, callback);
192+
```
193+
194+
It is also possible to register an event handler for a specific client, as in the following example:
195+
196+
```csharp
197+
EventHandlerDelegate callback = EventHandler;
198+
199+
var myClient = Api.Instance.GetClient("my-client");
200+
201+
var provider = new ExampleProvider();
202+
await Api.Instance.SetProvider(myClient.GetMetadata().Name, provider);
203+
204+
myClient.AddHandler(ProviderEventTypes.ProviderReady, callback);
205+
```
206+
170207
### Logging
171208

172209
The .NET SDK uses Microsoft.Extensions.Logging. See the [manual](https://learn.microsoft.com/en-us/dotnet/core/extensions/logging?tabs=command-line) for complete documentation.

build/Common.props

+1
Original file line numberDiff line numberDiff line change
@@ -25,5 +25,6 @@
2525

2626
<ItemGroup>
2727
<PackageReference Include="System.Collections.Immutable" Version="[1.7.1, 8.0.0)" />
28+
<PackageReference Include="System.Threading.Channels" Version="[6.0.0, 8.0.0)" />
2829
</ItemGroup>
2930
</Project>

src/OpenFeature/Api.cs

+28-1
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@
44
using System.Threading;
55
using System.Threading.Tasks;
66
using Microsoft.Extensions.Logging;
7+
using OpenFeature.Constant;
78
using OpenFeature.Model;
89

910
namespace OpenFeature
@@ -13,7 +14,7 @@ namespace OpenFeature
1314
/// In the absence of a provider the evaluation API uses the "No-op provider", which simply returns the supplied default flag value.
1415
/// </summary>
1516
/// <seealso href="https://github.com/open-feature/spec/blob/v0.5.2/specification/sections/01-flag-evaluation.md#1-flag-evaluation-api"/>
16-
public sealed class Api
17+
public sealed class Api : IEventBus
1718
{
1819
private EvaluationContext _evaluationContext = EvaluationContext.Empty;
1920
private readonly ProviderRepository _repository = new ProviderRepository();
@@ -22,6 +23,8 @@ public sealed class Api
2223
/// The reader/writer locks are not disposed because the singleton instance should never be disposed.
2324
private readonly ReaderWriterLockSlim _evaluationContextLock = new ReaderWriterLockSlim();
2425

26+
internal readonly EventExecutor EventExecutor = new EventExecutor();
27+
2528

2629
/// <summary>
2730
/// Singleton instance of Api
@@ -42,6 +45,7 @@ private Api() { }
4245
/// <param name="featureProvider">Implementation of <see cref="FeatureProvider"/></param>
4346
public async Task SetProvider(FeatureProvider featureProvider)
4447
{
48+
this.EventExecutor.RegisterDefaultFeatureProvider(featureProvider);
4549
await this._repository.SetProvider(featureProvider, this.GetContext()).ConfigureAwait(false);
4650
}
4751

@@ -54,6 +58,7 @@ public async Task SetProvider(FeatureProvider featureProvider)
5458
/// <param name="featureProvider">Implementation of <see cref="FeatureProvider"/></param>
5559
public async Task SetProvider(string clientName, FeatureProvider featureProvider)
5660
{
61+
this.EventExecutor.RegisterClientFeatureProvider(clientName, featureProvider);
5762
await this._repository.SetProvider(clientName, featureProvider, this.GetContext()).ConfigureAwait(false);
5863
}
5964

@@ -201,6 +206,28 @@ public EvaluationContext GetContext()
201206
public async Task Shutdown()
202207
{
203208
await this._repository.Shutdown().ConfigureAwait(false);
209+
await this.EventExecutor.Shutdown().ConfigureAwait(false);
210+
}
211+
212+
/// <inheritdoc />
213+
public void AddHandler(ProviderEventTypes type, EventHandlerDelegate handler)
214+
{
215+
this.EventExecutor.AddApiLevelHandler(type, handler);
216+
}
217+
218+
/// <inheritdoc />
219+
public void RemoveHandler(ProviderEventTypes type, EventHandlerDelegate handler)
220+
{
221+
this.EventExecutor.RemoveApiLevelHandler(type, handler);
222+
}
223+
224+
/// <summary>
225+
/// Sets the logger for the API
226+
/// </summary>
227+
/// <param name="logger">The logger to be used</param>
228+
public void SetLogger(ILogger logger)
229+
{
230+
this.EventExecutor.Logger = logger;
204231
}
205232
}
206233
}

src/OpenFeature/Constant/EventType.cs

+25
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
namespace OpenFeature.Constant
2+
{
3+
/// <summary>
4+
/// The ProviderEventTypes enum represents the available event types of a provider.
5+
/// </summary>
6+
public enum ProviderEventTypes
7+
{
8+
/// <summary>
9+
/// ProviderReady should be emitted by a provider upon completing its initialisation.
10+
/// </summary>
11+
ProviderReady,
12+
/// <summary>
13+
/// ProviderError should be emitted by a provider upon encountering an error.
14+
/// </summary>
15+
ProviderError,
16+
/// <summary>
17+
/// ProviderConfigurationChanged should be emitted by a provider when a flag configuration has been changed.
18+
/// </summary>
19+
ProviderConfigurationChanged,
20+
/// <summary>
21+
/// ProviderStale should be emitted by a provider when it goes into the stale state.
22+
/// </summary>
23+
ProviderStale
24+
}
25+
}

0 commit comments

Comments
 (0)