mirror of
https://github.com/astaxie/beego.git
synced 2024-11-05 04:20:54 +00:00
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
|
||
|
}
|
||
|
}
|