Skip to content

Commit

Permalink
Fix duration parsing error and missing test checks (#23)
Browse files Browse the repository at this point in the history
And disable "\n" string tests for now (see
go-yaml/yaml#1004)

Previously, negative durations with nanos were producing invalid values,
which wasn't caught because the round trip tests didn't actually check
the results of the diff :-(.
  • Loading branch information
Alfus authored Nov 18, 2023
1 parent 3a6cbb9 commit c0ac034
Show file tree
Hide file tree
Showing 3 changed files with 96 additions and 38 deletions.
6 changes: 5 additions & 1 deletion decode.go
Original file line number Diff line number Diff line change
Expand Up @@ -705,7 +705,11 @@ func parseDuration(txt string, duration *durationpb.Duration) error {
return errors.New("too many fractional second digits")
}
nanos *= int64(math.Pow10(power))
duration.Nanos = int32(nanos)
if duration.GetSeconds() >= 0 {
duration.Nanos = int32(nanos)
} else { // Sign must be consistent.
duration.Nanos = -int32(nanos)
}
default:
return errors.New("invalid duration: too many '.' characters")
}
Expand Down
6 changes: 4 additions & 2 deletions internal/protoyamltest/combine.go
Original file line number Diff line number Diff line change
Expand Up @@ -294,7 +294,9 @@ func interestingEnumValues(enum protoreflect.EnumDescriptor) []protoreflect.Enum
for i := 0; i < values.Len(); i++ {
result = append(result, values.Get(i).Number())
}
result = append(result, 0, -1, -1>>1)
if enum.FullName() != "google.protobuf.NullValue" {
result = append(result, 0, -1, math.MaxInt32, math.MinInt32)
}
return result
}

Expand Down Expand Up @@ -373,7 +375,7 @@ func interestingStrings() []string {
"",
// Whitespace
" ",
"\n",
// "\n", TODO: Uncomment once https://github.com/go-yaml/yaml/issues/1004 is fixed
"\t",
"\r",
// Nonprintable
Expand Down
122 changes: 87 additions & 35 deletions protoyaml_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,7 @@ import (
"google.golang.org/protobuf/types/known/durationpb"
"google.golang.org/protobuf/types/known/structpb"
"google.golang.org/protobuf/types/known/timestamppb"
"gopkg.in/yaml.v3"
)

func TestParseFieldPath(t *testing.T) {
Expand Down Expand Up @@ -84,42 +85,44 @@ func testDynamicRoundTrip(t *testing.T, desc protoreflect.MessageDescriptor, dat

func TestDuration_DynamicWithNanos(t *testing.T) {
t.Parallel()
val := &durationpb.Duration{
Seconds: 3600,
Nanos: 1010,
}
data, err := Marshal(val)
if err != nil {
t.Fatal(err)
}
actual := &durationpb.Duration{}
if err := Unmarshal(data, actual); err != nil {
t.Fatal(err)
}
if !reflect.DeepEqual(val, actual) {
t.Fatalf("Expected %v, got %v", val, actual)
for _, testCase := range []*durationpb.Duration{
{Seconds: 3600, Nanos: 1001},
{Seconds: -3600, Nanos: -1001},
} {
data, err := Marshal(testCase)
if err != nil {
t.Fatal(err)
}
actual := &durationpb.Duration{}
if err := Unmarshal(data, actual); err != nil {
t.Fatal(err)
}
if !reflect.DeepEqual(testCase, actual) {
t.Fatalf("Expected %v, got %v", testCase, actual)
}
testDynamicRoundTrip(t, testCase.ProtoReflect().Descriptor(), data)
}
testDynamicRoundTrip(t, val.ProtoReflect().Descriptor(), data)
}

func TestTimestamp_DynamicWithNanos(t *testing.T) {
t.Parallel()
val := &timestamppb.Timestamp{
Seconds: 3600,
Nanos: 1001,
}
data, err := Marshal(val)
if err != nil {
t.Fatal(err)
}
actual := &timestamppb.Timestamp{}
if err := Unmarshal(data, actual); err != nil {
t.Fatal(err)
}
if !reflect.DeepEqual(val, actual) {
t.Fatalf("Expected %v, got %v", val, actual)
for _, testCase := range []*timestamppb.Timestamp{
{Seconds: 3600, Nanos: 1001},
{Seconds: -3600, Nanos: 1001},
} {
data, err := Marshal(testCase)
if err != nil {
t.Fatal(err)
}
actual := &timestamppb.Timestamp{}
if err := Unmarshal(data, actual); err != nil {
t.Fatal(err)
}
if !reflect.DeepEqual(testCase, actual) {
t.Fatalf("Expected %v, got %v", testCase, actual)
}
testDynamicRoundTrip(t, testCase.ProtoReflect().Descriptor(), data)
}
testDynamicRoundTrip(t, val.ProtoReflect().Descriptor(), data)
}

func testValueDynamicRoundTrip(t *testing.T, data string) {
Expand Down Expand Up @@ -180,6 +183,50 @@ func TestListValue_Dynamic(t *testing.T) {
}
}

// See https://github.com/go-yaml/yaml/issues/1004
//
// Update combinatorial tests to include this case again when fixed.
func TestYamlNewLineMaps(t *testing.T) {
t.Parallel()
value := map[string]string{
"\n": "\n",
}
data, err := yaml.Marshal(value)
if err != nil {
t.Fatal(err)
}
// Should be `"\n": "\n"`, but its garbled.
if string(data) != "? |4+\n: |4+\n" {
t.Fatalf("Expected garbled output, got %s", string(data))
}
}

// Test that non-zero null values don't round trip.
func TestNullValueEnum(t *testing.T) {
t.Parallel()
data, err := Marshal(&proto3.TestAllTypes{
RepeatedNullValue: []structpb.NullValue{
1,
},
})
if err != nil {
t.Fatal(err)
}
if string(data) != "repeatedNullValue:\n - null\n" {
t.Fatalf("Expected %#v, got %#v", "repeatedNullValue:\n - null\n", string(data))
}
actual := &proto3.TestAllTypes{}
if err := Unmarshal(data, actual); err != nil {
t.Fatal(err)
}
if len(actual.GetRepeatedNullValue()) != 1 {
t.Fatalf("Expected 1, got %d", len(actual.GetRepeatedNullValue()))
}
if actual.GetRepeatedNullValue()[0] != structpb.NullValue_NULL_VALUE {
t.Fatalf("Expected %v, got %v", structpb.NullValue_NULL_VALUE, actual.GetRepeatedNullValue()[0])
}
}

func TestCombinatorial(t *testing.T) {
t.Parallel()
cases := protoyamltest.InterestingTestValues()
Expand Down Expand Up @@ -210,14 +257,17 @@ func TestFuzz(t *testing.T) {
if err != nil {
t.Fatal(err)
}
cmp.Diff(msg, roundTrip, protocmp.Transform(),
diff := cmp.Diff(msg, roundTrip, protocmp.Transform(),
cmp.Comparer(func(x, y float32) bool {
return fmt.Sprintf("%f", x) == fmt.Sprintf("%f", y)
}),
cmp.Comparer(func(x, y float64) bool {
return fmt.Sprintf("%f", x) == fmt.Sprintf("%f", y)
}),
)
if diff != "" {
t.Fatal(diff)
}
})
}
}
Expand All @@ -229,9 +279,8 @@ func testRoundTrip(t *testing.T, testCase *proto3.TestAllTypes) {
})
t.Run("Alt", func(t *testing.T) {
testRoundTripOption(t, testCase, MarshalOptions{
UseProtoNames: true,
UseEnumNumbers: true,
EmitUnpopulated: true,
UseProtoNames: true,
UseEnumNumbers: true,
})
})
}
Expand Down Expand Up @@ -273,11 +322,14 @@ func testUnmarshal(t *testing.T, testCase *proto3.TestAllTypes, data []byte) {
}

// Compare the two messages
cmp.Diff(testCase, roundTrip, protocmp.Transform(),
diff := cmp.Diff(testCase, roundTrip, protocmp.Transform(),
cmp.Comparer(func(x, y float32) bool {
return fmt.Sprintf("%f", x) == fmt.Sprintf("%f", y)
}),
cmp.Comparer(func(x, y float64) bool {
return fmt.Sprintf("%f", x) == fmt.Sprintf("%f", y)
}))
if diff != "" {
t.Fatal(diff)
}
}

0 comments on commit c0ac034

Please sign in to comment.