Skip to content

Commit

Permalink
C#: Ignore invalid enum string values in JSON, when invalid fields ar…
Browse files Browse the repository at this point in the history
…e being ignored.

PiperOrigin-RevId: 605310357
  • Loading branch information
protobuf-github-bot authored and copybara-github committed Feb 8, 2024
1 parent 959e5df commit 55e50ba
Show file tree
Hide file tree
Showing 3 changed files with 115 additions and 33 deletions.
6 changes: 0 additions & 6 deletions conformance/failure_list_csharp.txt
Original file line number Diff line number Diff line change
Expand Up @@ -13,9 +13,3 @@ Required.Proto3.JsonInput.OneofFieldNullSecond.JsonOutput
Required.Proto3.JsonInput.OneofFieldNullSecond.ProtobufOutput
Recommended.Proto3.ValueRejectInfNumberValue.JsonOutput
Recommended.Proto3.ValueRejectNanNumberValue.JsonOutput
Recommended.Proto2.JsonInput.IgnoreUnknownEnumStringValueInMapValue.ProtobufOutput
Recommended.Proto2.JsonInput.IgnoreUnknownEnumStringValueInOptionalField.ProtobufOutput
Recommended.Proto2.JsonInput.IgnoreUnknownEnumStringValueInRepeatedField.ProtobufOutput
Recommended.Proto3.JsonInput.IgnoreUnknownEnumStringValueInMapValue.ProtobufOutput
Recommended.Proto3.JsonInput.IgnoreUnknownEnumStringValueInOptionalField.ProtobufOutput
Recommended.Proto3.JsonInput.IgnoreUnknownEnumStringValueInRepeatedField.ProtobufOutput
47 changes: 47 additions & 0 deletions csharp/src/Google.Protobuf.Test/JsonParserTest.cs
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@
using NUnit.Framework;
using ProtobufTestMessages.Proto2;
using ProtobufTestMessages.Proto3;
using ProtobufUnittest;
using System;
using UnitTest.Issues.TestProtos;

Expand Down Expand Up @@ -924,6 +925,52 @@ public void Enum_Invalid(string value)
Assert.Throws<InvalidProtocolBufferException>(() => TestAllTypes.Parser.ParseJson(json));
}

[Test]
public void Enum_InvalidString_IgnoreUnknownFields()
{
// When ignoring unknown fields, invalid enum value strings are ignored too.
// This test uses TestProto3Optional so we can check we're not just setting the field to the 0 value.
var parser = new JsonParser(JsonParser.Settings.Default.WithIgnoreUnknownFields(true));
string json = "{ \"optionalNestedEnum\": \"NOT_A_VALID_VALUE\" }";
var parsed = parser.Parse<TestProto3Optional>(json);
Assert.IsFalse(parsed.HasOptionalNestedEnum);
}

[Test]
public void RepeatedEnum_InvalidString_IgnoreUnknownFields()
{
// When ignoring unknown fields, invalid enum value strings are ignored too.
// For a repeated field, the value is removed entirely.
var parser = new JsonParser(JsonParser.Settings.Default.WithIgnoreUnknownFields(true));
string json = "{ \"repeatedForeignEnum\": [ \"FOREIGN_FOO\", \"NOT_A_VALID_VALUE\", \"FOREIGN_BAR\" ] }";
var parsed = parser.Parse<TestAllTypes>(json);
var expected = new[] { TestProtos.ForeignEnum.ForeignFoo, TestProtos.ForeignEnum.ForeignBar };
Assert.AreEqual(expected, parsed.RepeatedForeignEnum);
}

[Test]
public void EnumValuedMap_InvalidString_IgnoreUnknownFields()
{
// When ignoring unknown fields, invalid enum value strings are ignored too.
// For a map field, the entry is removed entirely.
var parser = new JsonParser(JsonParser.Settings.Default.WithIgnoreUnknownFields(true));
string json = "{ \"mapInt32Enum\": { \"1\": \"MAP_ENUM_BAR\", \"2\": \"NOT_A_VALID_VALUE\" } }";
var parsed = parser.Parse<TestMap>(json);
Assert.AreEqual(1, parsed.MapInt32Enum.Count);
Assert.AreEqual(MapEnum.Bar, parsed.MapInt32Enum[1]);
Assert.False(parsed.MapInt32Enum.ContainsKey(2));
}

[Test]
public void Enum_InvalidNumber_IgnoreUnknownFields()
{
// Even when ignoring unknown fields, fail for non-integer numeric values, because
// they could *never* be valid.
var parser = new JsonParser(JsonParser.Settings.Default.WithIgnoreUnknownFields(true));
string json = "{ \"singleForeignEnum\": 5.5 }";
Assert.Throws<InvalidProtocolBufferException>(() => parser.Parse<TestAllTypes>(json));
}

[Test]
public void OneofDuplicate_Invalid()
{
Expand Down
95 changes: 68 additions & 27 deletions csharp/src/Google.Protobuf/JsonParser.cs
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
#region Copyright notice and license
#region Copyright notice and license
// Protocol Buffers - Google's data interchange format
// Copyright 2015 Google Inc. All rights reserved.
//
Expand Down Expand Up @@ -219,8 +219,10 @@ private void MergeField(IMessage message, FieldDescriptor field, JsonTokenizer t
}
else
{
var value = ParseSingleValue(field, tokenizer);
field.Accessor.SetValue(message, value);
if (TryParseSingleValue(field, tokenizer, out var value))
{
field.Accessor.SetValue(message, value);
}
}
}

Expand All @@ -241,12 +243,14 @@ private void MergeRepeatedField(IMessage message, FieldDescriptor field, JsonTok
return;
}
tokenizer.PushBack(token);
object value = ParseSingleValue(field, tokenizer);
if (value == null)
if (TryParseSingleValue(field, tokenizer, out object value))
{
throw new InvalidProtocolBufferException("Repeated field elements cannot be null");
if (value == null)
{
throw new InvalidProtocolBufferException("Repeated field elements cannot be null");
}
list.Add(value);
}
list.Add(value);
}
}

Expand Down Expand Up @@ -276,8 +280,10 @@ private void MergeMapField(IMessage message, FieldDescriptor field, JsonTokenize
return;
}
object key = ParseMapKey(keyField, token.StringValue);
object value = ParseSingleValue(valueField, tokenizer);
dictionary[key] = value ?? throw new InvalidProtocolBufferException("Map values must not be null");
if (TryParseSingleValue(valueField, tokenizer, out object value))
{
dictionary[key] = value ?? throw new InvalidProtocolBufferException("Map values must not be null");
}
}
}

Expand All @@ -293,7 +299,15 @@ private static bool IsGoogleProtobufNullValueField(FieldDescriptor field)
field.EnumType.FullName == NullValueDescriptor.FullName;
}

private object ParseSingleValue(FieldDescriptor field, JsonTokenizer tokenizer)
/// <summary>
/// Attempts to parse a single value from the JSON. When the value is completely invalid,
/// this will still throw an exception; when it's "conditionally invalid" (currently meaning
/// "when there's an unknown enum string value") the method returns false instead.
/// </summary>
/// <returns>
/// true if the value was parsed successfully; false for an ignorable parse failure.
/// </returns>
private bool TryParseSingleValue(FieldDescriptor field, JsonTokenizer tokenizer, out object value)
{
var token = tokenizer.Next();
if (token.Type == JsonToken.TokenType.Null)
Expand All @@ -302,13 +316,17 @@ private object ParseSingleValue(FieldDescriptor field, JsonTokenizer tokenizer)
// dynamically.
if (IsGoogleProtobufValueField(field))
{
return Value.ForNull();
value = Value.ForNull();
}
if (IsGoogleProtobufNullValueField(field))
else if (IsGoogleProtobufNullValueField(field))
{
return NullValue.NullValue;
value = NullValue.NullValue;
}
return null;
else
{
value = null;
}
return true;
}

var fieldType = field.FieldType;
Expand All @@ -327,7 +345,8 @@ private object ParseSingleValue(FieldDescriptor field, JsonTokenizer tokenizer)
tokenizer.PushBack(token);
IMessage subMessage = NewMessageForField(field);
Merge(subMessage, tokenizer);
return subMessage;
value = subMessage;
return true;
}
}

Expand All @@ -337,18 +356,26 @@ private object ParseSingleValue(FieldDescriptor field, JsonTokenizer tokenizer)
case JsonToken.TokenType.False:
if (fieldType == FieldType.Bool)
{
return token.Type == JsonToken.TokenType.True;
value = token.Type == JsonToken.TokenType.True;
return true;
}
// Fall through to "we don't support this type for this case"; could duplicate the behaviour of the default
// case instead, but this way we'd only need to change one place.
goto default;
case JsonToken.TokenType.StringValue:
return ParseSingleStringValue(field, token.StringValue);
if (field.FieldType != FieldType.Enum)
{
value = ParseSingleStringValue(field, token.StringValue);
return true;
}
else
{
return TryParseEnumStringValue(field, token.StringValue, out value);
}
// Note: not passing the number value itself here, as we may end up storing the string value in the token too.
case JsonToken.TokenType.Number:
return ParseSingleNumberValue(field, token);
case JsonToken.TokenType.Null:
throw new NotImplementedException("Haven't worked out what to do for null yet");
value = ParseSingleNumberValue(field, token);
return true;
default:
throw new InvalidProtocolBufferException("Unsupported JSON token type " + token.Type + " for field type " + fieldType);
}
Expand Down Expand Up @@ -694,18 +721,32 @@ private static object ParseSingleStringValue(FieldDescriptor field, string text)
ValidateInfinityAndNan(text, float.IsPositiveInfinity(f), float.IsNegativeInfinity(f), float.IsNaN(f));
return f;
case FieldType.Enum:
var enumValue = field.EnumType.FindValueByName(text);
if (enumValue == null)
{
throw new InvalidProtocolBufferException($"Invalid enum value: {text} for enum type: {field.EnumType.FullName}");
}
// Just return it as an int, and let the CLR convert it.
return enumValue.Number;
throw new InvalidOperationException($"Use TryParseEnumStringValue for enums");
default:
throw new InvalidProtocolBufferException($"Unsupported conversion from JSON string for field type {field.FieldType}");
}
}

private bool TryParseEnumStringValue(FieldDescriptor field, string text, out object value)
{
var enumValue = field.EnumType.FindValueByName(text);
if (enumValue == null)
{
if (settings.IgnoreUnknownFields)
{
value = null;
return false;
}
else
{
throw new InvalidProtocolBufferException($"Invalid enum value: {text} for enum type: {field.EnumType.FullName}");
}
}
// Just return it as an int, and let the CLR convert it.
value = enumValue.Number;
return true;
}

/// <summary>
/// Creates a new instance of the message type for the given field.
/// </summary>
Expand Down

0 comments on commit 55e50ba

Please sign in to comment.