I love using Refit to call web APIs in a nice type-safe manner. Sometimes though, APIs don’t want to cooperate with your strongly-typed hopes. For example, you might run into an API written by a hipster in a beanie, aka a dynamic-type enthusiast. I don’t say that pejoratively. Some of my closest friends write Python and Ruby.
For example, I came across an API that returned a value like this:
{
"important": true
}
No problem, I defined a class like this to deserialize it to:
public class ImportantResponse
{
public bool Important { get; set; }
}
And life was good. Until that awful day that the API returned this:
{
"important": "What is important is subjective to the viewer."
}
Damn! This philosophy lesson broke my client. One workaround is to do this:
public class ImportantResponse
{
public JsonElement Important { get; set; }
}
It works, but it’s not great. It doesn’t communicate to the consumer that this value can only be a string
or a bool
. That’s when I remembered an old blog post from my past.
When I was the Program Manager (PM) for ASP.NET MVC, my colleague and lead developer, Eilon, wrote a blog post entitled “The String or the Cat: A New .NET Framework Library where he introduced the class StringOr<TOther>
. This class could represent a dual-state value that’s either a string or another type.
The concepts presented here are based on a thought experiment proposed by scientist Erwin Schrödinger. While an understanding of quantum physics will help to understand the new types and APIs, it is not required.
It turned out his blog post was an April Fool’s joke. But the idea stuck with me. And now, here’s a case where I need a real implementation of it. But I’m going to name mine, StringOrValue<T>
.
One nice thing about implementing this today is we can leverage modern C# features. Here’s the starting implementation:
[JsonConverter(typeof(StringOrValueConverter))]
public readonly struct StringOrValue<T> : IStringOrObject {
public StringOrValue(string stringValue) {
StringValue = stringValue;
IsString = true;
}
public StringOrValue(T value) {
Value = value;
IsValue = true;
}
public T? Value { get; }
public string? StringValue { get; }
[MemberNotNullWhen(true, nameof(StringValue))]
public bool IsString { get; }
[MemberNotNullWhen(true, nameof(Value))]
public bool IsValue { get; }
}
/// <summary>
/// Internal interface for <see cref="StringOrValue{T}"/>.
/// </summary>
/// <remarks>
/// This is here to make serialization and deserialization easy.
/// </remarks>
[JsonConverter(typeof(StringOrValueConverter))]
internal interface IStringOrObject
{
bool IsString { get; }
bool IsValue { get; }
string? StringValue { get; }
object? ObjectValue { get; }
}
We can use the MemberNotNullWhen
attribute to tell the compiler that when IsString
is true, StringValue
is not null. And when IsValue
is true, Value
is not null. That way, code like this compiles just fine without raising null warnings:
var value = new StringOrValue<string>("Hello");
if (value.IsString) {
Console.WriteLine(value.StringValue.Length);
}
and
var value = new StringOrValue<SomeType>(42);
if (value.IsValue) {
Console.WriteLine(value.ToString());
}
It also is decorated with the JsonConverter
attribute to tell the JSON serializer to use the StringOrValueConverter
class to serialize and deserialize this type. I wanted this type to Just Work™. I didn’t want consumers of this class have to bother with registering a JsonConverterFactory
for this type.
This also explains why I introduced the internal IStringOrObject
interface. We can’t implement the JsonConverter
attribute on a open generic type, so we need a non-generic interface to apply the attribute to. It also makes it easier to write the converter as you’ll see.
/// <summary>
/// Value converter for <see cref="StringOrValue{T}"/>.
/// </summary>
internal class StringOrValueConverter : JsonConverter<IStringOrObject>
{
public override bool CanConvert(Type typeToConvert)
=> typeToConvert.IsGenericType
&& typeToConvert.GetGenericTypeDefinition() == typeof(StringOrValue<>);
public override IStringOrObject Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options)
{
var targetType = typeToConvert.GetGenericArguments()[0];
if (reader.TokenType == JsonTokenType.String)
{
var stringValue = reader.GetString();
return stringValue is null
? CreateEmptyInstance(targetType)
: CreateStringInstance(targetType, stringValue);
}
var value = JsonSerializer.Deserialize(ref reader, targetType, options);
return value is null
? CreateEmptyInstance(targetType)
: CreateValueInstance(targetType, value);
}
static ConstructorInfo GetEmptyConstructor(Type targetType)
{
return typeof(StringOrValue<>)
.MakeGenericType(targetType).
GetConstructor([])
?? throw new InvalidOperationException($"No constructor found for StringOrValue<{targetType.Name}>.");
}
static ConstructorInfo GetConstructor(Type targetType, Type argumentType)
{
return typeof(StringOrValue<>)
.MakeGenericType(targetType).
GetConstructor([argumentType])
?? throw new InvalidOperationException($"No constructor found for StringOrValue<{targetType.Name}>.");
}
static IStringOrObject CreateEmptyInstance(Type targetType)
{
var ctor = GetEmptyConstructor(targetType);
return (IStringOrObject)ctor.Invoke([]);
}
static IStringOrObject CreateStringInstance(Type targetType, string value)
{
var ctor = GetConstructor(targetType, typeof(string));
return (IStringOrObject)ctor.Invoke([value]);
}
static IStringOrObject CreateValueInstance(Type targetType, object value)
{
var ctor = GetConstructor(targetType, targetType);
return (IStringOrObject)ctor.Invoke([value]);
}
public override void Write(Utf8JsonWriter writer, IStringOrObject value, JsonSerializerOptions options)
{
if (value.IsString)
{
writer.WriteStringValue(value.StringValue);
}
else if (value.IsValue)
{
JsonSerializer.Serialize(writer, value.ObjectValue, options);
}
else
{
writer.WriteNullValue();
}
}
}
In the actual implementation of StringOrValue<T>
, I implemented IEquatable<T>, IEquatable<StringOrValue<T>> and overrode the implicit operators:
public static implicit operator StringOrValue<T>(string stringValue) => new(stringValue);
public static implicit operator StringOrValue<T>(T value) => new(value);
This allows you to write code like this:
StringOrValue<int> valueAsString = "Hello";
StringOrValue<int> valueAsNumber = 42;
Assert.Equals("Hello", valueAsString);
Assert.Equals(42, valueAsNumber);
So with this implementation in place, I can go back to the original example and write this:
public class ImportantResponse
{
public StringOrValue<bool> Important { get; set; }
}
And now I can handle both cases:
var response = JsonSerializer.Deserialize<ImportantResponse>(json)
?? throw new InvalidOperationException("Deserialization failed.");
if (response.Important.IsValue) {
if (response.Important.Value) {
Console.WriteLine("It's important!");
}
else {
Console.WriteLine("It's not important.");
}
}
else {
Console.WriteLine(response.Important.StringValue);
}
It’s time to go shopping for a beanie!
Here’s the full implementation for those interested in using this in your own projects!