Skip to content

Commit 70f847b

Browse files
arttonoyanchrfwow
andauthored
feat: Feature Provider Enhancements- #321 (#324)
This update focuses on enhancing the feature provider integration and testing framework, incorporating improvements to flexibility, usability, and testing capabilities. This update addresses [GitHub Issue #321](#321) in the OpenFeature .NET SDK repository. --- ### Key Enhancements 1. **Dependency Injection (DI) Enhancements:** - Improved lifecycle management for better resource handling. - Streamlined registration for feature providers, reducing configuration complexity. - Introduced the `AddProvider` extension method to simplify and adapt feature provider integration during service setup. 2. **Simplified Codebase:** - Removed `FeatureProviderFactory` logic, eliminating unnecessary complexity and improving usability. 3. **Improved InMemoryProvider:** - Enhanced the registration process for the `InMemoryProvider`, enabling smoother and more intuitive usage. 4. **Testing Improvements:** - Established a dedicated integration test project for comprehensive validation. - Improved overall test coverage, ensuring the reliability and robustness of the framework. --------- Signed-off-by: Artyom Tonoyan <[email protected]> Co-authored-by: chrfwow <[email protected]>
1 parent bf9de4e commit 70f847b

18 files changed

+616
-232
lines changed

Directory.Packages.props

+2-1
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@
99
<PackageVersion Include="Microsoft.Extensions.Hosting.Abstractions" Version="8.0.1" />
1010
<PackageVersion Include="Microsoft.Extensions.Logging.Abstractions" Version="8.0.2" />
1111
<PackageVersion Include="Microsoft.Extensions.DependencyInjection.Abstractions" Version="8.0.2" />
12+
<PackageVersion Include="Microsoft.Extensions.DependencyInjection" Version="8.0.1" />
1213
<PackageVersion Include="Microsoft.Extensions.Options" Version="8.0.2" />
1314
<PackageVersion Include="System.Collections.Immutable" Version="1.7.1" />
1415
<PackageVersion Include="System.Threading.Channels" Version="6.0.0" />
@@ -29,7 +30,7 @@
2930
<PackageVersion Include="SpecFlow.xUnit" Version="3.9.74" />
3031
<PackageVersion Include="xunit" Version="2.9.2" />
3132
<PackageVersion Include="xunit.runner.visualstudio" Version="2.8.2" />
32-
<PackageVersion Include="Microsoft.Extensions.DependencyInjection" Version="8.0.1" />
33+
<PackageVersion Include="Microsoft.AspNetCore.TestHost" Version="8.0.11" />
3334
</ItemGroup>
3435

3536
<ItemGroup Condition="'$(OS)' == 'Unix'">

OpenFeature.sln

+7
Original file line numberDiff line numberDiff line change
@@ -85,6 +85,8 @@ Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "OpenFeature.DependencyInjec
8585
EndProject
8686
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "OpenFeature.Hosting", "src\OpenFeature.Hosting\OpenFeature.Hosting.csproj", "{C99DA02A-3981-45A6-B3F8-4A1A48653DEE}"
8787
EndProject
88+
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "OpenFeature.IntegrationTests", "test\OpenFeature.IntegrationTests\OpenFeature.IntegrationTests.csproj", "{68463B47-36B4-8DB5-5D02-662C169E85B0}"
89+
EndProject
8890
Global
8991
GlobalSection(SolutionConfigurationPlatforms) = preSolution
9092
Debug|Any CPU = Debug|Any CPU
@@ -119,6 +121,10 @@ Global
119121
{C99DA02A-3981-45A6-B3F8-4A1A48653DEE}.Debug|Any CPU.Build.0 = Debug|Any CPU
120122
{C99DA02A-3981-45A6-B3F8-4A1A48653DEE}.Release|Any CPU.ActiveCfg = Release|Any CPU
121123
{C99DA02A-3981-45A6-B3F8-4A1A48653DEE}.Release|Any CPU.Build.0 = Release|Any CPU
124+
{68463B47-36B4-8DB5-5D02-662C169E85B0}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
125+
{68463B47-36B4-8DB5-5D02-662C169E85B0}.Debug|Any CPU.Build.0 = Debug|Any CPU
126+
{68463B47-36B4-8DB5-5D02-662C169E85B0}.Release|Any CPU.ActiveCfg = Release|Any CPU
127+
{68463B47-36B4-8DB5-5D02-662C169E85B0}.Release|Any CPU.Build.0 = Release|Any CPU
122128
EndGlobalSection
123129
GlobalSection(SolutionProperties) = preSolution
124130
HideSolutionNode = FALSE
@@ -137,6 +143,7 @@ Global
137143
{C5415057-2700-48B5-940A-7A10969FA639} = {C97E9975-E10A-4817-AE2C-4DD69C3C02D4}
138144
{EB35F9F6-8A79-410E-A293-9387BC4AC9A7} = {65FBA159-23E0-4CF9-881B-F78DBFF198E9}
139145
{C99DA02A-3981-45A6-B3F8-4A1A48653DEE} = {C97E9975-E10A-4817-AE2C-4DD69C3C02D4}
146+
{68463B47-36B4-8DB5-5D02-662C169E85B0} = {65FBA159-23E0-4CF9-881B-F78DBFF198E9}
140147
EndGlobalSection
141148
GlobalSection(ExtensibilityGlobals) = postSolution
142149
SolutionGuid = {41F01B78-FB06-404F-8AD0-6ED6973F948F}

README.md

+31-29
Original file line numberDiff line numberDiff line change
@@ -338,41 +338,43 @@ builder.Services.AddOpenFeature(featureBuilder => {
338338
});
339339
});
340340
```
341-
#### Creating a New Provider
342-
To integrate a custom provider, such as InMemoryProvider, you’ll need to create a factory that builds and configures the provider. This section demonstrates how to set up InMemoryProvider as a new provider with custom configuration options.
343341

344-
**Configuring InMemoryProvider as a New Provider**
345-
<br />Begin by creating a custom factory class, `InMemoryProviderFactory`, that implements `IFeatureProviderFactory`. This factory will initialize your provider with any necessary configurations.
346-
```csharp
347-
public class InMemoryProviderFactory : IFeatureProviderFactory
348-
{
349-
internal IDictionary<string, Flag>? Flags { get; set; }
350-
351-
public FeatureProvider Create() => new InMemoryProvider(Flags);
352-
}
353-
```
354-
**Adding an Extension Method to OpenFeatureBuilder**
355-
<br />To streamline the configuration process, add an extension method, `AddInMemoryProvider`, to `OpenFeatureBuilder`. This allows you to set up the provider with either a domain-scoped or a default configuration.
342+
### Registering a Custom Provider
343+
You can register a custom provider, such as `InMemoryProvider`, with OpenFeature using the `AddProvider` method. This approach allows you to dynamically resolve services or configurations during registration.
356344

357345
```csharp
358-
public static partial class FeatureBuilderExtensions
359-
{
360-
public static OpenFeatureBuilder AddInMemoryProvider(this OpenFeatureBuilder builder, Action<IDictionary<string, Flag>>? configure = null)
361-
=> builder.AddProvider<InMemoryProviderFactory>(factory => ConfigureFlags(factory, configure));
346+
services.AddOpenFeature()
347+
.AddProvider(provider =>
348+
{
349+
// Resolve services or configurations as needed
350+
var configuration = provider.GetRequiredService<IConfiguration>();
351+
var flags = new Dictionary<string, Flag>
352+
{
353+
{ "feature-key", new Flag<bool>(configuration.GetValue<bool>("FeatureFlags:Key")) }
354+
};
355+
356+
// Register a custom provider, such as InMemoryProvider
357+
return new InMemoryProvider(flags);
358+
});
359+
```
362360

363-
public static OpenFeatureBuilder AddInMemoryProvider(this OpenFeatureBuilder builder, string domain, Action<IDictionary<string, Flag>>? configure = null)
364-
=> builder.AddProvider<InMemoryProviderFactory>(domain, factory => ConfigureFlags(factory, configure));
361+
#### Adding a Domain-Scoped Provider
365362

366-
private static void ConfigureFlags(InMemoryProviderFactory factory, Action<IDictionary<string, Flag>>? configure)
367-
{
368-
if (configure == null)
369-
return;
363+
You can also register a domain-scoped custom provider, enabling configurations specific to each domain:
370364

371-
var flag = new Dictionary<string, Flag>();
372-
configure.Invoke(flag);
373-
factory.Flags = flag;
374-
}
375-
}
365+
```csharp
366+
services.AddOpenFeature()
367+
.AddProvider("my-domain", (provider, domain) =>
368+
{
369+
// Resolve services or configurations as needed for the domain
370+
var flags = new Dictionary<string, Flag>
371+
{
372+
{ $"{domain}-feature-key", new Flag<bool>(true) }
373+
};
374+
375+
// Register a domain-scoped custom provider such as InMemoryProvider
376+
return new InMemoryProvider(flags);
377+
});
376378
```
377379

378380
<!-- x-hide-in-docs-start -->

src/OpenFeature.DependencyInjection/IFeatureProviderFactory.cs

-23
This file was deleted.

0 commit comments

Comments
 (0)