2022-07-22 22:32:37 +00:00
|
|
|
// Copyright 2022 The Go Authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
|
|
|
package protodelim_test
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bufio"
|
|
|
|
"bytes"
|
2023-06-23 12:49:52 +00:00
|
|
|
"encoding/binary"
|
2022-07-22 22:32:37 +00:00
|
|
|
"errors"
|
|
|
|
"io"
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/google/go-cmp/cmp"
|
|
|
|
"google.golang.org/protobuf/encoding/protodelim"
|
|
|
|
"google.golang.org/protobuf/encoding/protowire"
|
|
|
|
"google.golang.org/protobuf/internal/testprotos/test3"
|
|
|
|
"google.golang.org/protobuf/testing/protocmp"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestRoundTrip(t *testing.T) {
|
|
|
|
msgs := []*test3.TestAllTypes{
|
|
|
|
{SingularInt32: 1},
|
|
|
|
{SingularString: "hello"},
|
|
|
|
{RepeatedDouble: []float64{1.2, 3.4}},
|
|
|
|
{
|
|
|
|
SingularNestedMessage: &test3.TestAllTypes_NestedMessage{A: 1},
|
|
|
|
RepeatedForeignMessage: []*test3.ForeignMessage{{C: 2}, {D: 3}},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
buf := &bytes.Buffer{}
|
|
|
|
|
|
|
|
// Write all messages to buf.
|
|
|
|
for _, m := range msgs {
|
|
|
|
if n, err := protodelim.MarshalTo(buf, m); err != nil {
|
|
|
|
t.Errorf("protodelim.MarshalTo(_, %v) = %d, %v", m, n, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-02 20:07:39 +00:00
|
|
|
for _, tc := range []struct {
|
|
|
|
name string
|
|
|
|
reader protodelim.Reader
|
|
|
|
}{
|
|
|
|
{name: "defaultbuffer", reader: bufio.NewReader(bytes.NewBuffer(buf.Bytes()))},
|
|
|
|
{name: "smallbuffer", reader: bufio.NewReaderSize(bytes.NewBuffer(buf.Bytes()), 0)},
|
|
|
|
{name: "largebuffer", reader: bufio.NewReaderSize(bytes.NewBuffer(buf.Bytes()), 1<<20)},
|
|
|
|
{name: "notbufio", reader: notBufioReader{bufio.NewReader(bytes.NewBuffer(buf.Bytes()))}},
|
|
|
|
} {
|
|
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
|
|
// Read and collect messages from buf.
|
|
|
|
var got []*test3.TestAllTypes
|
|
|
|
for {
|
|
|
|
m := &test3.TestAllTypes{}
|
|
|
|
err := protodelim.UnmarshalFrom(tc.reader, m)
|
|
|
|
if errors.Is(err, io.EOF) {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("protodelim.UnmarshalFrom(_) = %v", err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
got = append(got, m)
|
|
|
|
}
|
|
|
|
|
|
|
|
want := msgs
|
|
|
|
if diff := cmp.Diff(want, got, protocmp.Transform()); diff != "" {
|
|
|
|
t.Errorf("Unmarshaler collected messages: diff -want +got = %s", diff)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Just a wrapper so that UnmarshalFrom doesn't recognize this as a bufio.Reader
|
|
|
|
type notBufioReader struct {
|
|
|
|
*bufio.Reader
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestUnmarshalFromBufioAllocations(t *testing.T) {
|
|
|
|
// Use a proto which won't require an additional allocations during unmarshalling.
|
|
|
|
// Write to buf
|
|
|
|
buf := &bytes.Buffer{}
|
|
|
|
m := &test3.TestAllTypes{SingularInt32: 1}
|
|
|
|
if n, err := protodelim.MarshalTo(buf, m); err != nil {
|
|
|
|
t.Errorf("protodelim.MarshalTo(_, %v) = %d, %v", m, n, err)
|
|
|
|
}
|
|
|
|
reader := bufio.NewReaderSize(nil, 1<<20)
|
|
|
|
got := &test3.TestAllTypes{}
|
|
|
|
|
|
|
|
allocs := testing.AllocsPerRun(5, func() {
|
|
|
|
// Read from buf.
|
|
|
|
reader.Reset(bytes.NewBuffer(buf.Bytes()))
|
|
|
|
err := protodelim.UnmarshalFrom(reader, got)
|
2022-07-22 22:32:37 +00:00
|
|
|
if err != nil {
|
2023-05-02 20:07:39 +00:00
|
|
|
t.Fatalf("protodelim.UnmarshalFrom(_) = %v", err)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
if allocs != 1 {
|
|
|
|
// bytes.NewBuffer should be the only allocation.
|
|
|
|
t.Errorf("Got %v allocs. Wanted 1", allocs)
|
|
|
|
}
|
|
|
|
|
|
|
|
if diff := cmp.Diff(m, got, protocmp.Transform()); diff != "" {
|
|
|
|
t.Errorf("Unmarshaler read: diff -want +got = %s", diff)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func BenchmarkUnmarshalFrom(b *testing.B) {
|
|
|
|
var manyInt32 []int32
|
|
|
|
for i := int32(0); i < 10000; i++ {
|
|
|
|
manyInt32 = append(manyInt32, i)
|
|
|
|
}
|
|
|
|
var msgs []*test3.TestAllTypes
|
|
|
|
for i := 0; i < 10; i++ {
|
|
|
|
msgs = append(msgs, &test3.TestAllTypes{RepeatedInt32: manyInt32})
|
|
|
|
}
|
|
|
|
|
|
|
|
buf := &bytes.Buffer{}
|
|
|
|
|
|
|
|
// Write all messages to buf.
|
|
|
|
for _, m := range msgs {
|
|
|
|
if n, err := protodelim.MarshalTo(buf, m); err != nil {
|
|
|
|
b.Errorf("protodelim.MarshalTo(_, %v) = %d, %v", m, n, err)
|
2022-07-22 22:32:37 +00:00
|
|
|
}
|
2023-05-02 20:07:39 +00:00
|
|
|
}
|
|
|
|
bufBytes := buf.Bytes()
|
|
|
|
|
|
|
|
type resetReader interface {
|
|
|
|
protodelim.Reader
|
|
|
|
Reset(io.Reader)
|
2022-07-22 22:32:37 +00:00
|
|
|
}
|
|
|
|
|
2023-05-02 20:07:39 +00:00
|
|
|
for _, tc := range []struct {
|
|
|
|
name string
|
|
|
|
reader resetReader
|
|
|
|
}{
|
|
|
|
{name: "bufio1mib", reader: bufio.NewReaderSize(nil, 1<<20)},
|
|
|
|
{name: "bufio16mib", reader: bufio.NewReaderSize(nil, 1<<24)},
|
|
|
|
{name: "notbufio1mib", reader: notBufioReader{bufio.NewReaderSize(nil, 1<<20)}},
|
|
|
|
{name: "notbufio16mib", reader: notBufioReader{bufio.NewReaderSize(nil, 1<<24)}},
|
|
|
|
} {
|
|
|
|
b.Run(tc.name, func(b *testing.B) {
|
|
|
|
b.ReportAllocs()
|
|
|
|
b.ResetTimer()
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
|
|
tc.reader.Reset(bytes.NewBuffer(bufBytes))
|
|
|
|
var got int
|
|
|
|
m := &test3.TestAllTypes{}
|
|
|
|
for {
|
|
|
|
err := protodelim.UnmarshalFrom(tc.reader, m)
|
|
|
|
if errors.Is(err, io.EOF) {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
b.Errorf("protodelim.UnmarshalFrom(_) = %v", err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
got++
|
|
|
|
}
|
|
|
|
if got != len(msgs) {
|
|
|
|
b.Errorf("Got %v messages. Wanted %v", got, len(msgs))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
2022-07-22 22:32:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestMaxSize(t *testing.T) {
|
|
|
|
in := &test3.TestAllTypes{SingularInt32: 1}
|
|
|
|
|
|
|
|
buf := &bytes.Buffer{}
|
|
|
|
|
|
|
|
if n, err := protodelim.MarshalTo(buf, in); err != nil {
|
|
|
|
t.Errorf("protodelim.MarshalTo(_, %v) = %d, %v", in, n, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
out := &test3.TestAllTypes{}
|
|
|
|
err := protodelim.UnmarshalOptions{MaxSize: 1}.UnmarshalFrom(bufio.NewReader(buf), out)
|
|
|
|
|
|
|
|
var errSize *protodelim.SizeTooLargeError
|
|
|
|
if !errors.As(err, &errSize) {
|
|
|
|
t.Errorf("protodelim.UnmarshalOptions{MaxSize: 1}.UnmarshalFrom(_, _) = %v (%T), want %T", err, err, errSize)
|
|
|
|
}
|
|
|
|
got, want := errSize, &protodelim.SizeTooLargeError{Size: 3, MaxSize: 1}
|
|
|
|
if diff := cmp.Diff(want, got); diff != "" {
|
|
|
|
t.Errorf("protodelim.UnmarshalOptions{MaxSize: 1}.UnmarshalFrom(_, _): diff -want +got = %s", diff)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestUnmarshalFrom_UnexpectedEOF(t *testing.T) {
|
|
|
|
buf := &bytes.Buffer{}
|
|
|
|
|
|
|
|
// Write a size (42), but no subsequent message.
|
|
|
|
sb := protowire.AppendVarint(nil, 42)
|
|
|
|
if _, err := buf.Write(sb); err != nil {
|
|
|
|
t.Fatalf("buf.Write(%v) = _, %v", sb, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
out := &test3.TestAllTypes{}
|
|
|
|
err := protodelim.UnmarshalFrom(bufio.NewReader(buf), out)
|
|
|
|
if got, want := err, io.ErrUnexpectedEOF; got != want {
|
|
|
|
t.Errorf("protodelim.UnmarshalFrom(size-only buf, _) = %v, want %v", got, want)
|
|
|
|
}
|
|
|
|
}
|
2023-06-23 12:49:52 +00:00
|
|
|
|
|
|
|
func TestUnmarshalFrom_PrematureHeader(t *testing.T) {
|
|
|
|
var data = []byte{128} // continuation bit set
|
|
|
|
err := protodelim.UnmarshalFrom(bytes.NewReader(data[:]), nil)
|
|
|
|
if got, want := err, io.ErrUnexpectedEOF; !errors.Is(got, want) {
|
|
|
|
t.Errorf("protodelim.UnmarshalFrom(%#v, nil) = %#v; want = %#v", data, got, want)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestUnmarshalFrom_InvalidVarint(t *testing.T) {
|
|
|
|
var data = bytes.Repeat([]byte{128}, 2*binary.MaxVarintLen64) // continuation bit set
|
|
|
|
err := protodelim.UnmarshalFrom(bytes.NewReader(data[:]), nil)
|
|
|
|
if err == nil {
|
|
|
|
t.Errorf("protodelim.UnmarshalFrom unexpectedly did not error on invalid varint")
|
|
|
|
}
|
|
|
|
}
|