Skip to content

Commit 24c3441

Browse files
feat!: Add support for provider shutdown and status. (#158)
Signed-off-by: Ryan Lamb <[email protected]> Co-authored-by: Todd Baert <[email protected]>
1 parent a2f70eb commit 24c3441

12 files changed

+1101
-67
lines changed

README.md

+11-6
Original file line numberDiff line numberDiff line change
@@ -72,7 +72,7 @@ dotnet add package OpenFeature
7272
public async Task Example()
7373
{
7474
// Register your feature flag provider
75-
Api.Instance.SetProvider(new InMemoryProvider());
75+
await Api.Instance.SetProvider(new InMemoryProvider());
7676

7777
// Create a new client
7878
FeatureClient client = Api.Instance.GetClient();
@@ -97,7 +97,7 @@ public async Task Example()
9797
|| [Logging](#logging) | Integrate with popular logging packages. |
9898
|| [Named clients](#named-clients) | Utilize multiple providers in a single application. |
9999
|| [Eventing](#eventing) | React to state changes in the provider or flag management system. |
100-
| | [Shutdown](#shutdown) | Gracefully clean up a provider during application shutdown. |
100+
| | [Shutdown](#shutdown) | Gracefully clean up a provider during application shutdown. |
101101
|| [Extending](#extending) | Extend OpenFeature with custom providers and hooks. |
102102

103103
<sub>Implemented: ✅ | In-progress: ⚠️ | Not implemented yet: ❌</sub>
@@ -112,7 +112,7 @@ If the provider you're looking for hasn't been created yet, see the [develop a p
112112
Once you've added a provider as a dependency, it can be registered with OpenFeature like this:
113113

114114
```csharp
115-
Api.Instance.SetProvider(new MyProvider());
115+
await Api.Instance.SetProvider(new MyProvider());
116116
```
117117

118118
In some situations, it may be beneficial to register multiple providers in the same application.
@@ -179,9 +179,9 @@ If a name has no associated provider, the global provider is used.
179179

180180
```csharp
181181
// registering the default provider
182-
Api.Instance.SetProvider(new LocalProvider());
182+
await Api.Instance.SetProvider(new LocalProvider());
183183
// registering a named provider
184-
Api.Instance.SetProvider("clientForCache", new CachedProvider());
184+
await Api.Instance.SetProvider("clientForCache", new CachedProvider());
185185

186186
// a client backed by default provider
187187
FeatureClient clientDefault = Api.Instance.GetClient();
@@ -196,7 +196,12 @@ Events are currently not supported by the .NET SDK. Progress on this feature can
196196

197197
### Shutdown
198198

199-
A shutdown handler is not yet available in the .NET SDK. Progress on this feature can be tracked [here](https://github.com/open-feature/dotnet-sdk/issues/126).
199+
The OpenFeature API provides a close function to perform a cleanup of all registered providers. This should only be called when your application is in the process of shutting down.
200+
201+
```csharp
202+
// Shut down all providers
203+
await Api.Instance.Shutdown();
204+
```
200205

201206
## Extending
202207

src/OpenFeature/Api.cs

+29-37
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@
22
using System.Collections.Generic;
33
using System.Linq;
44
using System.Threading;
5+
using System.Threading.Tasks;
56
using Microsoft.Extensions.Logging;
67
using OpenFeature.Model;
78

@@ -15,14 +16,12 @@ namespace OpenFeature
1516
public sealed class Api
1617
{
1718
private EvaluationContext _evaluationContext = EvaluationContext.Empty;
18-
private FeatureProvider _defaultProvider = new NoOpFeatureProvider();
19-
private readonly ConcurrentDictionary<string, FeatureProvider> _featureProviders =
20-
new ConcurrentDictionary<string, FeatureProvider>();
19+
private readonly ProviderRepository _repository = new ProviderRepository();
2120
private readonly ConcurrentStack<Hook> _hooks = new ConcurrentStack<Hook>();
2221

2322
/// The reader/writer locks are not disposed because the singleton instance should never be disposed.
2423
private readonly ReaderWriterLockSlim _evaluationContextLock = new ReaderWriterLockSlim();
25-
private readonly ReaderWriterLockSlim _featureProviderLock = new ReaderWriterLockSlim();
24+
2625

2726
/// <summary>
2827
/// Singleton instance of Api
@@ -36,31 +35,26 @@ static Api() { }
3635
private Api() { }
3736

3837
/// <summary>
39-
/// Sets the feature provider
38+
/// Sets the feature provider. In order to wait for the provider to be set, and initialization to complete,
39+
/// await the returned task.
4040
/// </summary>
41+
/// <remarks>The provider cannot be set to null. Attempting to set the provider to null has no effect.</remarks>
4142
/// <param name="featureProvider">Implementation of <see cref="FeatureProvider"/></param>
42-
public void SetProvider(FeatureProvider featureProvider)
43+
public async Task SetProvider(FeatureProvider featureProvider)
4344
{
44-
this._featureProviderLock.EnterWriteLock();
45-
try
46-
{
47-
this._defaultProvider = featureProvider ?? this._defaultProvider;
48-
}
49-
finally
50-
{
51-
this._featureProviderLock.ExitWriteLock();
52-
}
45+
await this._repository.SetProvider(featureProvider, this.GetContext()).ConfigureAwait(false);
5346
}
5447

48+
5549
/// <summary>
56-
/// Sets the feature provider to given clientName
50+
/// Sets the feature provider to given clientName. In order to wait for the provider to be set, and
51+
/// initialization to complete, await the returned task.
5752
/// </summary>
5853
/// <param name="clientName">Name of client</param>
5954
/// <param name="featureProvider">Implementation of <see cref="FeatureProvider"/></param>
60-
public void SetProvider(string clientName, FeatureProvider featureProvider)
55+
public async Task SetProvider(string clientName, FeatureProvider featureProvider)
6156
{
62-
this._featureProviders.AddOrUpdate(clientName, featureProvider,
63-
(key, current) => featureProvider);
57+
await this._repository.SetProvider(clientName, featureProvider, this.GetContext()).ConfigureAwait(false);
6458
}
6559

6660
/// <summary>
@@ -76,15 +70,7 @@ public void SetProvider(string clientName, FeatureProvider featureProvider)
7670
/// <returns><see cref="FeatureProvider"/></returns>
7771
public FeatureProvider GetProvider()
7872
{
79-
this._featureProviderLock.EnterReadLock();
80-
try
81-
{
82-
return this._defaultProvider;
83-
}
84-
finally
85-
{
86-
this._featureProviderLock.ExitReadLock();
87-
}
73+
return this._repository.GetProvider();
8874
}
8975

9076
/// <summary>
@@ -95,17 +81,9 @@ public FeatureProvider GetProvider()
9581
/// have a corresponding provider the default provider will be returned</returns>
9682
public FeatureProvider GetProvider(string clientName)
9783
{
98-
if (string.IsNullOrEmpty(clientName))
99-
{
100-
return this.GetProvider();
101-
}
102-
103-
return this._featureProviders.TryGetValue(clientName, out var featureProvider)
104-
? featureProvider
105-
: this.GetProvider();
84+
return this._repository.GetProvider(clientName);
10685
}
10786

108-
10987
/// <summary>
11088
/// Gets providers metadata
11189
/// <para>
@@ -210,5 +188,19 @@ public EvaluationContext GetContext()
210188
this._evaluationContextLock.ExitReadLock();
211189
}
212190
}
191+
192+
/// <summary>
193+
/// <para>
194+
/// Shut down and reset the current status of OpenFeature API.
195+
/// </para>
196+
/// <para>
197+
/// This call cleans up all active providers and attempts to shut down internal event handling mechanisms.
198+
/// Once shut down is complete, API is reset and ready to use again.
199+
/// </para>
200+
/// </summary>
201+
public async Task Shutdown()
202+
{
203+
await this._repository.Shutdown().ConfigureAwait(false);
204+
}
213205
}
214206
}
+31
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,31 @@
1+
using System.ComponentModel;
2+
3+
namespace OpenFeature.Constant
4+
{
5+
/// <summary>
6+
/// The state of the provider.
7+
/// </summary>
8+
/// <seealso href="https://github.com/open-feature/spec/blob/main/specification/sections/02-providers.md#requirement-242" />
9+
public enum ProviderStatus
10+
{
11+
/// <summary>
12+
/// The provider has not been initialized and cannot yet evaluate flags.
13+
/// </summary>
14+
[Description("NOT_READY")] NotReady,
15+
16+
/// <summary>
17+
/// The provider is ready to resolve flags.
18+
/// </summary>
19+
[Description("READY")] Ready,
20+
21+
/// <summary>
22+
/// The provider's cached state is no longer valid and may not be up-to-date with the source of truth.
23+
/// </summary>
24+
[Description("STALE")] Stale,
25+
26+
/// <summary>
27+
/// The provider is in an error state and unable to evaluate flags.
28+
/// </summary>
29+
[Description("ERROR")] Error
30+
}
31+
}

src/OpenFeature/FeatureProvider.cs

+49
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
using System.Collections.Immutable;
22
using System.Threading.Tasks;
3+
using OpenFeature.Constant;
34
using OpenFeature.Model;
45

56
namespace OpenFeature
@@ -79,5 +80,53 @@ public abstract Task<ResolutionDetails<double>> ResolveDoubleValue(string flagKe
7980
/// <returns><see cref="ResolutionDetails{T}"/></returns>
8081
public abstract Task<ResolutionDetails<Value>> ResolveStructureValue(string flagKey, Value defaultValue,
8182
EvaluationContext context = null);
83+
84+
/// <summary>
85+
/// Get the status of the provider.
86+
/// </summary>
87+
/// <returns>The current <see cref="ProviderStatus"/></returns>
88+
/// <remarks>
89+
/// If a provider does not override this method, then its status will be assumed to be
90+
/// <see cref="ProviderStatus.Ready"/>. If a provider implements this method, and supports initialization,
91+
/// then it should start in the <see cref="ProviderStatus.NotReady"/>status . If the status is
92+
/// <see cref="ProviderStatus.NotReady"/>, then the Api will call the <see cref="Initialize" /> when the
93+
/// provider is set.
94+
/// </remarks>
95+
public virtual ProviderStatus GetStatus() => ProviderStatus.Ready;
96+
97+
/// <summary>
98+
/// <para>
99+
/// This method is called before a provider is used to evaluate flags. Providers can overwrite this method,
100+
/// if they have special initialization needed prior being called for flag evaluation.
101+
/// </para>
102+
/// </summary>
103+
/// <param name="context"><see cref="EvaluationContext"/></param>
104+
/// <returns>A task that completes when the initialization process is complete.</returns>
105+
/// <remarks>
106+
/// <para>
107+
/// A provider which supports initialization should override this method as well as
108+
/// <see cref="FeatureProvider.GetStatus"/>.
109+
/// </para>
110+
/// <para>
111+
/// The provider should return <see cref="ProviderStatus.Ready"/> or <see cref="ProviderStatus.Error"/> from
112+
/// the <see cref="GetStatus"/> method after initialization is complete.
113+
/// </para>
114+
/// </remarks>
115+
public virtual Task Initialize(EvaluationContext context)
116+
{
117+
// Intentionally left blank.
118+
return Task.CompletedTask;
119+
}
120+
121+
/// <summary>
122+
/// This method is called when a new provider is about to be used to evaluate flags, or the SDK is shut down.
123+
/// Providers can overwrite this method, if they have special shutdown actions needed.
124+
/// </summary>
125+
/// <returns>A task that completes when the shutdown process is complete.</returns>
126+
public virtual Task Shutdown()
127+
{
128+
// Intentionally left blank.
129+
return Task.CompletedTask;
130+
}
82131
}
83132
}

0 commit comments

Comments
 (0)