You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
389 lines
11 KiB
Go
389 lines
11 KiB
Go
// Protocol Buffers for Go with Gadgets
|
|
//
|
|
// Copyright (c) 2018, The GoGo Authors. All rights reserved.
|
|
// http://github.com/gogo/protobuf
|
|
//
|
|
// Redistribution and use in source and binary forms, with or without
|
|
// modification, are permitted provided that the following conditions are
|
|
// met:
|
|
//
|
|
// * Redistributions of source code must retain the above copyright
|
|
// notice, this list of conditions and the following disclaimer.
|
|
// * Redistributions in binary form must reproduce the above
|
|
// copyright notice, this list of conditions and the following disclaimer
|
|
// in the documentation and/or other materials provided with the
|
|
// distribution.
|
|
//
|
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
|
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
package proto
|
|
|
|
import (
|
|
"reflect"
|
|
"time"
|
|
)
|
|
|
|
// makeMessageRefMarshaler differs a bit from makeMessageMarshaler
|
|
// It marshal a message T instead of a *T
|
|
func makeMessageRefMarshaler(u *marshalInfo) (sizer, marshaler) {
|
|
return func(ptr pointer, tagsize int) int {
|
|
siz := u.size(ptr)
|
|
return siz + SizeVarint(uint64(siz)) + tagsize
|
|
},
|
|
func(b []byte, ptr pointer, wiretag uint64, deterministic bool) ([]byte, error) {
|
|
b = appendVarint(b, wiretag)
|
|
siz := u.cachedsize(ptr)
|
|
b = appendVarint(b, uint64(siz))
|
|
return u.marshal(b, ptr, deterministic)
|
|
}
|
|
}
|
|
|
|
// makeMessageRefSliceMarshaler differs quite a lot from makeMessageSliceMarshaler
|
|
// It marshals a slice of messages []T instead of []*T
|
|
func makeMessageRefSliceMarshaler(u *marshalInfo) (sizer, marshaler) {
|
|
return func(ptr pointer, tagsize int) int {
|
|
s := ptr.getSlice(u.typ)
|
|
n := 0
|
|
for i := 0; i < s.Len(); i++ {
|
|
elem := s.Index(i)
|
|
e := elem.Interface()
|
|
v := toAddrPointer(&e, false)
|
|
siz := u.size(v)
|
|
n += siz + SizeVarint(uint64(siz)) + tagsize
|
|
}
|
|
return n
|
|
},
|
|
func(b []byte, ptr pointer, wiretag uint64, deterministic bool) ([]byte, error) {
|
|
s := ptr.getSlice(u.typ)
|
|
var err, errreq error
|
|
for i := 0; i < s.Len(); i++ {
|
|
elem := s.Index(i)
|
|
e := elem.Interface()
|
|
v := toAddrPointer(&e, false)
|
|
b = appendVarint(b, wiretag)
|
|
siz := u.size(v)
|
|
b = appendVarint(b, uint64(siz))
|
|
b, err = u.marshal(b, v, deterministic)
|
|
|
|
if err != nil {
|
|
if _, ok := err.(*RequiredNotSetError); ok {
|
|
// Required field in submessage is not set.
|
|
// We record the error but keep going, to give a complete marshaling.
|
|
if errreq == nil {
|
|
errreq = err
|
|
}
|
|
continue
|
|
}
|
|
if err == ErrNil {
|
|
err = errRepeatedHasNil
|
|
}
|
|
return b, err
|
|
}
|
|
}
|
|
|
|
return b, errreq
|
|
}
|
|
}
|
|
|
|
func makeCustomPtrMarshaler(u *marshalInfo) (sizer, marshaler) {
|
|
return func(ptr pointer, tagsize int) int {
|
|
if ptr.isNil() {
|
|
return 0
|
|
}
|
|
m := ptr.asPointerTo(reflect.PtrTo(u.typ)).Elem().Interface().(custom)
|
|
siz := m.Size()
|
|
return tagsize + SizeVarint(uint64(siz)) + siz
|
|
}, func(b []byte, ptr pointer, wiretag uint64, deterministic bool) ([]byte, error) {
|
|
if ptr.isNil() {
|
|
return b, nil
|
|
}
|
|
m := ptr.asPointerTo(reflect.PtrTo(u.typ)).Elem().Interface().(custom)
|
|
siz := m.Size()
|
|
buf, err := m.Marshal()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
b = appendVarint(b, wiretag)
|
|
b = appendVarint(b, uint64(siz))
|
|
b = append(b, buf...)
|
|
return b, nil
|
|
}
|
|
}
|
|
|
|
func makeCustomMarshaler(u *marshalInfo) (sizer, marshaler) {
|
|
return func(ptr pointer, tagsize int) int {
|
|
m := ptr.asPointerTo(u.typ).Interface().(custom)
|
|
siz := m.Size()
|
|
return tagsize + SizeVarint(uint64(siz)) + siz
|
|
}, func(b []byte, ptr pointer, wiretag uint64, deterministic bool) ([]byte, error) {
|
|
m := ptr.asPointerTo(u.typ).Interface().(custom)
|
|
siz := m.Size()
|
|
buf, err := m.Marshal()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
b = appendVarint(b, wiretag)
|
|
b = appendVarint(b, uint64(siz))
|
|
b = append(b, buf...)
|
|
return b, nil
|
|
}
|
|
}
|
|
|
|
func makeTimeMarshaler(u *marshalInfo) (sizer, marshaler) {
|
|
return func(ptr pointer, tagsize int) int {
|
|
t := ptr.asPointerTo(u.typ).Interface().(*time.Time)
|
|
ts, err := timestampProto(*t)
|
|
if err != nil {
|
|
return 0
|
|
}
|
|
siz := Size(ts)
|
|
return tagsize + SizeVarint(uint64(siz)) + siz
|
|
}, func(b []byte, ptr pointer, wiretag uint64, deterministic bool) ([]byte, error) {
|
|
t := ptr.asPointerTo(u.typ).Interface().(*time.Time)
|
|
ts, err := timestampProto(*t)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
buf, err := Marshal(ts)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
b = appendVarint(b, wiretag)
|
|
b = appendVarint(b, uint64(len(buf)))
|
|
b = append(b, buf...)
|
|
return b, nil
|
|
}
|
|
}
|
|
|
|
func makeTimePtrMarshaler(u *marshalInfo) (sizer, marshaler) {
|
|
return func(ptr pointer, tagsize int) int {
|
|
if ptr.isNil() {
|
|
return 0
|
|
}
|
|
t := ptr.asPointerTo(reflect.PtrTo(u.typ)).Elem().Interface().(*time.Time)
|
|
ts, err := timestampProto(*t)
|
|
if err != nil {
|
|
return 0
|
|
}
|
|
siz := Size(ts)
|
|
return tagsize + SizeVarint(uint64(siz)) + siz
|
|
}, func(b []byte, ptr pointer, wiretag uint64, deterministic bool) ([]byte, error) {
|
|
if ptr.isNil() {
|
|
return b, nil
|
|
}
|
|
t := ptr.asPointerTo(reflect.PtrTo(u.typ)).Elem().Interface().(*time.Time)
|
|
ts, err := timestampProto(*t)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
buf, err := Marshal(ts)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
b = appendVarint(b, wiretag)
|
|
b = appendVarint(b, uint64(len(buf)))
|
|
b = append(b, buf...)
|
|
return b, nil
|
|
}
|
|
}
|
|
|
|
func makeTimeSliceMarshaler(u *marshalInfo) (sizer, marshaler) {
|
|
return func(ptr pointer, tagsize int) int {
|
|
s := ptr.getSlice(u.typ)
|
|
n := 0
|
|
for i := 0; i < s.Len(); i++ {
|
|
elem := s.Index(i)
|
|
t := elem.Interface().(time.Time)
|
|
ts, err := timestampProto(t)
|
|
if err != nil {
|
|
return 0
|
|
}
|
|
siz := Size(ts)
|
|
n += siz + SizeVarint(uint64(siz)) + tagsize
|
|
}
|
|
return n
|
|
},
|
|
func(b []byte, ptr pointer, wiretag uint64, deterministic bool) ([]byte, error) {
|
|
s := ptr.getSlice(u.typ)
|
|
for i := 0; i < s.Len(); i++ {
|
|
elem := s.Index(i)
|
|
t := elem.Interface().(time.Time)
|
|
ts, err := timestampProto(t)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
siz := Size(ts)
|
|
buf, err := Marshal(ts)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
b = appendVarint(b, wiretag)
|
|
b = appendVarint(b, uint64(siz))
|
|
b = append(b, buf...)
|
|
}
|
|
|
|
return b, nil
|
|
}
|
|
}
|
|
|
|
func makeTimePtrSliceMarshaler(u *marshalInfo) (sizer, marshaler) {
|
|
return func(ptr pointer, tagsize int) int {
|
|
s := ptr.getSlice(reflect.PtrTo(u.typ))
|
|
n := 0
|
|
for i := 0; i < s.Len(); i++ {
|
|
elem := s.Index(i)
|
|
t := elem.Interface().(*time.Time)
|
|
ts, err := timestampProto(*t)
|
|
if err != nil {
|
|
return 0
|
|
}
|
|
siz := Size(ts)
|
|
n += siz + SizeVarint(uint64(siz)) + tagsize
|
|
}
|
|
return n
|
|
},
|
|
func(b []byte, ptr pointer, wiretag uint64, deterministic bool) ([]byte, error) {
|
|
s := ptr.getSlice(reflect.PtrTo(u.typ))
|
|
for i := 0; i < s.Len(); i++ {
|
|
elem := s.Index(i)
|
|
t := elem.Interface().(*time.Time)
|
|
ts, err := timestampProto(*t)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
siz := Size(ts)
|
|
buf, err := Marshal(ts)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
b = appendVarint(b, wiretag)
|
|
b = appendVarint(b, uint64(siz))
|
|
b = append(b, buf...)
|
|
}
|
|
|
|
return b, nil
|
|
}
|
|
}
|
|
|
|
func makeDurationMarshaler(u *marshalInfo) (sizer, marshaler) {
|
|
return func(ptr pointer, tagsize int) int {
|
|
d := ptr.asPointerTo(u.typ).Interface().(*time.Duration)
|
|
dur := durationProto(*d)
|
|
siz := Size(dur)
|
|
return tagsize + SizeVarint(uint64(siz)) + siz
|
|
}, func(b []byte, ptr pointer, wiretag uint64, deterministic bool) ([]byte, error) {
|
|
d := ptr.asPointerTo(u.typ).Interface().(*time.Duration)
|
|
dur := durationProto(*d)
|
|
buf, err := Marshal(dur)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
b = appendVarint(b, wiretag)
|
|
b = appendVarint(b, uint64(len(buf)))
|
|
b = append(b, buf...)
|
|
return b, nil
|
|
}
|
|
}
|
|
|
|
func makeDurationPtrMarshaler(u *marshalInfo) (sizer, marshaler) {
|
|
return func(ptr pointer, tagsize int) int {
|
|
if ptr.isNil() {
|
|
return 0
|
|
}
|
|
d := ptr.asPointerTo(reflect.PtrTo(u.typ)).Elem().Interface().(*time.Duration)
|
|
dur := durationProto(*d)
|
|
siz := Size(dur)
|
|
return tagsize + SizeVarint(uint64(siz)) + siz
|
|
}, func(b []byte, ptr pointer, wiretag uint64, deterministic bool) ([]byte, error) {
|
|
if ptr.isNil() {
|
|
return b, nil
|
|
}
|
|
d := ptr.asPointerTo(reflect.PtrTo(u.typ)).Elem().Interface().(*time.Duration)
|
|
dur := durationProto(*d)
|
|
buf, err := Marshal(dur)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
b = appendVarint(b, wiretag)
|
|
b = appendVarint(b, uint64(len(buf)))
|
|
b = append(b, buf...)
|
|
return b, nil
|
|
}
|
|
}
|
|
|
|
func makeDurationSliceMarshaler(u *marshalInfo) (sizer, marshaler) {
|
|
return func(ptr pointer, tagsize int) int {
|
|
s := ptr.getSlice(u.typ)
|
|
n := 0
|
|
for i := 0; i < s.Len(); i++ {
|
|
elem := s.Index(i)
|
|
d := elem.Interface().(time.Duration)
|
|
dur := durationProto(d)
|
|
siz := Size(dur)
|
|
n += siz + SizeVarint(uint64(siz)) + tagsize
|
|
}
|
|
return n
|
|
},
|
|
func(b []byte, ptr pointer, wiretag uint64, deterministic bool) ([]byte, error) {
|
|
s := ptr.getSlice(u.typ)
|
|
for i := 0; i < s.Len(); i++ {
|
|
elem := s.Index(i)
|
|
d := elem.Interface().(time.Duration)
|
|
dur := durationProto(d)
|
|
siz := Size(dur)
|
|
buf, err := Marshal(dur)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
b = appendVarint(b, wiretag)
|
|
b = appendVarint(b, uint64(siz))
|
|
b = append(b, buf...)
|
|
}
|
|
|
|
return b, nil
|
|
}
|
|
}
|
|
|
|
func makeDurationPtrSliceMarshaler(u *marshalInfo) (sizer, marshaler) {
|
|
return func(ptr pointer, tagsize int) int {
|
|
s := ptr.getSlice(reflect.PtrTo(u.typ))
|
|
n := 0
|
|
for i := 0; i < s.Len(); i++ {
|
|
elem := s.Index(i)
|
|
d := elem.Interface().(*time.Duration)
|
|
dur := durationProto(*d)
|
|
siz := Size(dur)
|
|
n += siz + SizeVarint(uint64(siz)) + tagsize
|
|
}
|
|
return n
|
|
},
|
|
func(b []byte, ptr pointer, wiretag uint64, deterministic bool) ([]byte, error) {
|
|
s := ptr.getSlice(reflect.PtrTo(u.typ))
|
|
for i := 0; i < s.Len(); i++ {
|
|
elem := s.Index(i)
|
|
d := elem.Interface().(*time.Duration)
|
|
dur := durationProto(*d)
|
|
siz := Size(dur)
|
|
buf, err := Marshal(dur)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
b = appendVarint(b, wiretag)
|
|
b = appendVarint(b, uint64(siz))
|
|
b = append(b, buf...)
|
|
}
|
|
|
|
return b, nil
|
|
}
|
|
}
|