Skip to content

Commit e7ab498

Browse files
authored
feat!: ErrorType as enum, add ErrorMessage string (#72)
1 parent 1f7e4cb commit e7ab498

9 files changed

+177
-87
lines changed

src/OpenFeatureSDK/Constant/ErrorType.cs

+11-1
Original file line numberDiff line numberDiff line change
@@ -36,6 +36,16 @@ public enum ErrorType
3636
/// <summary>
3737
/// Abnormal execution of the provider
3838
/// </summary>
39-
[Description("GENERAL")] General
39+
[Description("GENERAL")] General,
40+
41+
/// <summary>
42+
/// Context does not satisfy provider requirements.
43+
/// </summary>
44+
[Description("INVALID_CONTEXT")] InvalidContext,
45+
46+
/// <summary>
47+
/// Context does not contain a targeting key and the provider requires one.
48+
/// </summary>
49+
[Description("TARGETING_KEY_MISSING")] TargetingKeyMissing,
4050
}
4151
}
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,5 @@
11
using System;
22
using OpenFeatureSDK.Constant;
3-
using OpenFeatureSDK.Extension;
43

54
namespace OpenFeatureSDK.Error
65
{
@@ -11,9 +10,9 @@ namespace OpenFeatureSDK.Error
1110
public class FeatureProviderException : Exception
1211
{
1312
/// <summary>
14-
/// Description of error that occured when evaluating a flag
13+
/// Error that occurred during evaluation
1514
/// </summary>
16-
public string ErrorDescription { get; }
15+
public ErrorType ErrorType { get; }
1716

1817
/// <summary>
1918
/// Initialize a new instance of the <see cref="FeatureProviderException"/> class
@@ -24,19 +23,7 @@ public class FeatureProviderException : Exception
2423
public FeatureProviderException(ErrorType errorType, string message = null, Exception innerException = null)
2524
: base(message, innerException)
2625
{
27-
this.ErrorDescription = errorType.GetDescription();
28-
}
29-
30-
/// <summary>
31-
/// Initialize a new instance of the <see cref="FeatureProviderException"/> class
32-
/// </summary>
33-
/// <param name="errorCode">A string representation describing the error that occured</param>
34-
/// <param name="message">Exception message</param>
35-
/// <param name="innerException">Optional inner exception</param>
36-
public FeatureProviderException(string errorCode, string message = null, Exception innerException = null)
37-
: base(message, innerException)
38-
{
39-
this.ErrorDescription = errorCode;
26+
this.ErrorType = errorType;
4027
}
4128
}
4229
}

src/OpenFeatureSDK/Extension/ResolutionDetailsExtensions.cs

+2-1
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,8 @@ internal static class ResolutionDetailsExtensions
66
{
77
public static FlagEvaluationDetails<T> ToFlagEvaluationDetails<T>(this ResolutionDetails<T> details)
88
{
9-
return new FlagEvaluationDetails<T>(details.FlagKey, details.Value, details.ErrorType, details.Reason, details.Variant);
9+
return new FlagEvaluationDetails<T>(details.FlagKey, details.Value, details.ErrorType, details.Reason,
10+
details.Variant, details.ErrorMessage);
1011
}
1112
}
1213
}

src/OpenFeatureSDK/Model/FlagEvalusationDetails.cs src/OpenFeatureSDK/Model/FlagEvaluationDetails.cs

+14-20
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,4 @@
11
using OpenFeatureSDK.Constant;
2-
using OpenFeatureSDK.Extension;
32

43
namespace OpenFeatureSDK.Model
54
{
@@ -23,7 +22,16 @@ public class FlagEvaluationDetails<T>
2322
/// <summary>
2423
/// Error that occurred during evaluation
2524
/// </summary>
26-
public string ErrorType { get; }
25+
public ErrorType ErrorType { get; }
26+
27+
/// <summary>
28+
/// Message containing additional details about an error.
29+
/// <para>
30+
/// Will be <see langword="null" /> if there is no error or if the provider didn't provide any additional error
31+
/// details.
32+
/// </para>
33+
/// </summary>
34+
public string ErrorMessage { get; }
2735

2836
/// <summary>
2937
/// Describes the reason for the outcome of the evaluation process
@@ -45,30 +53,16 @@ public class FlagEvaluationDetails<T>
4553
/// <param name="errorType">Error</param>
4654
/// <param name="reason">Reason</param>
4755
/// <param name="variant">Variant</param>
48-
public FlagEvaluationDetails(string flagKey, T value, ErrorType errorType, string reason, string variant)
49-
{
50-
this.Value = value;
51-
this.FlagKey = flagKey;
52-
this.ErrorType = errorType.GetDescription();
53-
this.Reason = reason;
54-
this.Variant = variant;
55-
}
56-
57-
/// <summary>
58-
/// Initializes a new instance of the <see cref="FlagEvaluationDetails{T}"/> class.
59-
/// </summary>
60-
/// <param name="flagKey">Feature flag key</param>
61-
/// <param name="value">Evaluated value</param>
62-
/// <param name="errorType">Error</param>
63-
/// <param name="reason">Reason</param>
64-
/// <param name="variant">Variant</param>
65-
public FlagEvaluationDetails(string flagKey, T value, string errorType, string reason, string variant)
56+
/// <param name="errorMessage">Error message</param>
57+
public FlagEvaluationDetails(string flagKey, T value, ErrorType errorType, string reason, string variant,
58+
string errorMessage = null)
6659
{
6760
this.Value = value;
6861
this.FlagKey = flagKey;
6962
this.ErrorType = errorType;
7063
this.Reason = reason;
7164
this.Variant = variant;
65+
this.ErrorMessage = errorMessage;
7266
}
7367
}
7468
}

src/OpenFeatureSDK/Model/ResolutionDetails.cs

+8-1
Original file line numberDiff line numberDiff line change
@@ -26,6 +26,11 @@ public class ResolutionDetails<T>
2626
/// </summary>
2727
public ErrorType ErrorType { get; }
2828

29+
/// <summary>
30+
/// Message containing additional details about an error.
31+
/// </summary>
32+
public string ErrorMessage { get; }
33+
2934
/// <summary>
3035
/// Describes the reason for the outcome of the evaluation process
3136
/// <see cref="Reason"/>
@@ -47,14 +52,16 @@ public class ResolutionDetails<T>
4752
/// <param name="errorType">Error</param>
4853
/// <param name="reason">Reason</param>
4954
/// <param name="variant">Variant</param>
55+
/// <param name="errorMessage">Error message</param>
5056
public ResolutionDetails(string flagKey, T value, ErrorType errorType = ErrorType.None, string reason = null,
51-
string variant = null)
57+
string variant = null, string errorMessage = null)
5258
{
5359
this.Value = value;
5460
this.FlagKey = flagKey;
5561
this.ErrorType = errorType;
5662
this.Reason = reason;
5763
this.Variant = variant;
64+
this.ErrorMessage = errorMessage;
5865
}
5966
}
6067
}

src/OpenFeatureSDK/OpenFeatureClient.cs

+3-3
Original file line numberDiff line numberDiff line change
@@ -257,9 +257,9 @@ private async Task<FlagEvaluationDetails<T>> EvaluateFlag<T>(
257257
catch (FeatureProviderException ex)
258258
{
259259
this._logger.LogError(ex, "Error while evaluating flag {FlagKey}. Error {ErrorType}", flagKey,
260-
ex.ErrorDescription);
261-
evaluation = new FlagEvaluationDetails<T>(flagKey, defaultValue, ex.ErrorDescription, Reason.Error,
262-
string.Empty);
260+
ex.ErrorType.GetDescription());
261+
evaluation = new FlagEvaluationDetails<T>(flagKey, defaultValue, ex.ErrorType, Reason.Error,
262+
string.Empty, ex.Message);
263263
await this.TriggerErrorHooks(allHooksReversed, hookContext, ex, options);
264264
}
265265
catch (Exception ex)

test/OpenFeatureSDK.Tests/FeatureProviderExceptionTests.cs

+6-5
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@
22
using FluentAssertions;
33
using OpenFeatureSDK.Constant;
44
using OpenFeatureSDK.Error;
5+
using OpenFeatureSDK.Extension;
56
using Xunit;
67

78
namespace OpenFeatureSDK.Tests
@@ -17,16 +18,16 @@ public class FeatureProviderExceptionTests
1718
public void FeatureProviderException_Should_Resolve_Description(ErrorType errorType, string errorDescription)
1819
{
1920
var ex = new FeatureProviderException(errorType);
20-
ex.ErrorDescription.Should().Be(errorDescription);
21+
ex.ErrorType.GetDescription().Should().Be(errorDescription);
2122
}
2223

2324
[Theory]
24-
[InlineData("OUT_OF_CREDIT", "Subscription has expired, please renew your subscription.")]
25-
[InlineData("Exceed quota", "User has exceeded the quota for this feature.")]
26-
public void FeatureProviderException_Should_Allow_Custom_ErrorCode_Messages(string errorCode, string message)
25+
[InlineData(ErrorType.General, "Subscription has expired, please renew your subscription.")]
26+
[InlineData(ErrorType.ProviderNotReady, "User has exceeded the quota for this feature.")]
27+
public void FeatureProviderException_Should_Allow_Custom_ErrorCode_Messages(ErrorType errorCode, string message)
2728
{
2829
var ex = new FeatureProviderException(errorCode, message, new ArgumentOutOfRangeException("flag"));
29-
ex.ErrorDescription.Should().Be(errorCode);
30+
ex.ErrorType.Should().Be(errorCode);
3031
ex.Message.Should().Be(message);
3132
ex.InnerException.Should().BeOfType<ArgumentOutOfRangeException>();
3233
}

0 commit comments

Comments
 (0)