Skip to content

Commit d792b32

Browse files
roelofbaustindrenskitoddbaert
authored
fix: Add targeting key (#231)
Fixes #230 --------- Signed-off-by: Roelof Blom <[email protected]> Signed-off-by: Roelof Blom <[email protected]> Signed-off-by: Austin Drenski <[email protected]> Signed-off-by: Todd Baert <[email protected]> Co-authored-by: Austin Drenski <[email protected]> Co-authored-by: Todd Baert <[email protected]>
1 parent 1082094 commit d792b32

File tree

4 files changed

+82
-10
lines changed

4 files changed

+82
-10
lines changed

src/OpenFeature/Model/EvaluationContext.cs

+9-1
Original file line numberDiff line numberDiff line change
@@ -16,9 +16,11 @@ public sealed class EvaluationContext
1616
/// <summary>
1717
/// Internal constructor used by the builder.
1818
/// </summary>
19+
/// <param name="targetingKey">The targeting key</param>
1920
/// <param name="content">The content of the context.</param>
20-
internal EvaluationContext(Structure content)
21+
internal EvaluationContext(string targetingKey, Structure content)
2122
{
23+
this.TargetingKey = targetingKey;
2224
this._structure = content;
2325
}
2426

@@ -28,6 +30,7 @@ internal EvaluationContext(Structure content)
2830
private EvaluationContext()
2931
{
3032
this._structure = Structure.Empty;
33+
this.TargetingKey = string.Empty;
3134
}
3235

3336
/// <summary>
@@ -83,6 +86,11 @@ public IImmutableDictionary<string, Value> AsDictionary()
8386
/// </summary>
8487
public int Count => this._structure.Count;
8588

89+
/// <summary>
90+
/// Returns the targeting key for the context.
91+
/// </summary>
92+
public string TargetingKey { get; }
93+
8694
/// <summary>
8795
/// Return an enumerator for all values
8896
/// </summary>

src/OpenFeature/Model/EvaluationContextBuilder.cs

+31-1
Original file line numberDiff line numberDiff line change
@@ -14,11 +14,24 @@ public sealed class EvaluationContextBuilder
1414
{
1515
private readonly StructureBuilder _attributes = Structure.Builder();
1616

17+
internal string TargetingKey { get; private set; }
18+
1719
/// <summary>
1820
/// Internal to only allow direct creation by <see cref="EvaluationContext.Builder()"/>.
1921
/// </summary>
2022
internal EvaluationContextBuilder() { }
2123

24+
/// <summary>
25+
/// Set the targeting key for the context.
26+
/// </summary>
27+
/// <param name="targetingKey">The targeting key</param>
28+
/// <returns>This builder</returns>
29+
public EvaluationContextBuilder SetTargetingKey(string targetingKey)
30+
{
31+
this.TargetingKey = targetingKey;
32+
return this;
33+
}
34+
2235
/// <summary>
2336
/// Set the key to the given <see cref="Value"/>.
2437
/// </summary>
@@ -125,6 +138,23 @@ public EvaluationContextBuilder Set(string key, DateTime value)
125138
/// <returns>This builder</returns>
126139
public EvaluationContextBuilder Merge(EvaluationContext context)
127140
{
141+
string newTargetingKey = "";
142+
143+
if (!string.IsNullOrWhiteSpace(TargetingKey))
144+
{
145+
newTargetingKey = TargetingKey;
146+
}
147+
148+
if (!string.IsNullOrWhiteSpace(context.TargetingKey))
149+
{
150+
newTargetingKey = context.TargetingKey;
151+
}
152+
153+
if (!string.IsNullOrWhiteSpace(newTargetingKey))
154+
{
155+
this.TargetingKey = newTargetingKey;
156+
}
157+
128158
foreach (var kvp in context)
129159
{
130160
this.Set(kvp.Key, kvp.Value);
@@ -139,7 +169,7 @@ public EvaluationContextBuilder Merge(EvaluationContext context)
139169
/// <returns>An immutable <see cref="EvaluationContext"/></returns>
140170
public EvaluationContext Build()
141171
{
142-
return new EvaluationContext(this._attributes.Build());
172+
return new EvaluationContext(this.TargetingKey, this._attributes.Build());
143173
}
144174
}
145175
}

test/OpenFeature.E2ETests/Steps/EvaluationStepDefinitions.cs

+6-7
Original file line numberDiff line numberDiff line change
@@ -200,12 +200,11 @@ public void Giventhevariantshouldbeandthereasonshouldbe(string expectedVariant,
200200
[When(@"context contains keys ""(.*)"", ""(.*)"", ""(.*)"", ""(.*)"" with values ""(.*)"", ""(.*)"", (.*), ""(.*)""")]
201201
public void Whencontextcontainskeyswithvalues(string field1, string field2, string field3, string field4, string value1, string value2, int value3, string value4)
202202
{
203-
var attributes = ImmutableDictionary.CreateBuilder<string, Value>();
204-
attributes.Add(field1, new Value(value1));
205-
attributes.Add(field2, new Value(value2));
206-
attributes.Add(field3, new Value(value3));
207-
attributes.Add(field4, new Value(bool.Parse(value4)));
208-
this.context = new EvaluationContext(new Structure(attributes));
203+
this.context = new EvaluationContextBuilder()
204+
.Set(field1, value1)
205+
.Set(field2, value2)
206+
.Set(field3, value3)
207+
.Set(field4, bool.Parse(value4)).Build();
209208
}
210209

211210
[When(@"a flag with key ""(.*)"" is evaluated with default value ""(.*)""")]
@@ -225,7 +224,7 @@ public void Thentheresolvedstringresponseshouldbe(string expected)
225224
[Then(@"the resolved flag value is ""(.*)"" when the context is empty")]
226225
public void Giventheresolvedflagvalueiswhenthecontextisempty(string expected)
227226
{
228-
string emptyContextValue = client.GetStringValue(contextAwareFlagKey, contextAwareDefaultValue, new EvaluationContext(new Structure(ImmutableDictionary<string, Value>.Empty))).Result;
227+
string emptyContextValue = client.GetStringValue(contextAwareFlagKey, contextAwareDefaultValue, new EvaluationContextBuilder().Build()).Result;
229228
Assert.Equal(expected, emptyContextValue);
230229
}
231230

test/OpenFeature.Tests/OpenFeatureEvaluationContextTests.cs

+36-1
Original file line numberDiff line numberDiff line change
@@ -16,14 +16,42 @@ public void Should_Merge_Two_Contexts()
1616
.Set("key1", "value1");
1717
var contextBuilder2 = new EvaluationContextBuilder()
1818
.Set("key2", "value2");
19-
2019
var context1 = contextBuilder1.Merge(contextBuilder2.Build()).Build();
2120

2221
Assert.Equal(2, context1.Count);
2322
Assert.Equal("value1", context1.GetValue("key1").AsString);
2423
Assert.Equal("value2", context1.GetValue("key2").AsString);
2524
}
2625

26+
[Fact]
27+
public void Should_Change_TargetingKey_From_OverridingContext()
28+
{
29+
var contextBuilder1 = new EvaluationContextBuilder()
30+
.Set("key1", "value1")
31+
.SetTargetingKey("targeting_key");
32+
var contextBuilder2 = new EvaluationContextBuilder()
33+
.Set("key2", "value2")
34+
.SetTargetingKey("overriding_key");
35+
36+
var mergeContext = contextBuilder1.Merge(contextBuilder2.Build()).Build();
37+
38+
Assert.Equal("overriding_key", mergeContext.TargetingKey);
39+
}
40+
41+
[Fact]
42+
public void Should_Retain_TargetingKey_When_OverridingContext_TargetingKey_Value_IsEmpty()
43+
{
44+
var contextBuilder1 = new EvaluationContextBuilder()
45+
.Set("key1", "value1")
46+
.SetTargetingKey("targeting_key");
47+
var contextBuilder2 = new EvaluationContextBuilder()
48+
.Set("key2", "value2");
49+
50+
var mergeContext = contextBuilder1.Merge(contextBuilder2.Build()).Build();
51+
52+
Assert.Equal("targeting_key", mergeContext.TargetingKey);
53+
}
54+
2755
[Fact]
2856
[Specification("3.2.2", "Evaluation context MUST be merged in the order: API (global; lowest precedence) - client - invocation - before hooks (highest precedence), with duplicate values being overwritten.")]
2957
public void Should_Merge_TwoContexts_And_Override_Duplicates_With_RightHand_Context()
@@ -51,6 +79,8 @@ public void EvaluationContext_Should_All_Types()
5179
var now = fixture.Create<DateTime>();
5280
var structure = fixture.Create<Structure>();
5381
var contextBuilder = new EvaluationContextBuilder()
82+
.SetTargetingKey("targeting_key")
83+
.Set("targeting_key", "userId")
5484
.Set("key1", "value")
5585
.Set("key2", 1)
5686
.Set("key3", true)
@@ -60,6 +90,11 @@ public void EvaluationContext_Should_All_Types()
6090

6191
var context = contextBuilder.Build();
6292

93+
context.TargetingKey.Should().Be("targeting_key");
94+
var targetingKeyValue = context.GetValue(context.TargetingKey);
95+
targetingKeyValue.IsString.Should().BeTrue();
96+
targetingKeyValue.AsString.Should().Be("userId");
97+
6398
var value1 = context.GetValue("key1");
6499
value1.IsString.Should().BeTrue();
65100
value1.AsString.Should().Be("value");

0 commit comments

Comments
 (0)