1
0
Fork 0
mirror of https://github.com/anyproto/anytype-heart.git synced 2025-06-08 05:47:07 +09:00
anytype-heart/util/pbtypes/copy.go

231 lines
5.3 KiB
Go

package pbtypes
import (
"sync"
"github.com/gogo/protobuf/types"
"github.com/anyproto/anytype-heart/pkg/lib/logging"
"github.com/anyproto/anytype-heart/pkg/lib/pb/model"
)
var log = logging.Logger("anytype-pbtypes")
var bytesPool = &sync.Pool{
New: func() interface{} {
return []byte{}
},
}
func CopyBlock(in *model.Block) (out *model.Block) {
buf := bytesPool.Get().([]byte)
size := in.Size()
if cap(buf) < size {
buf = make([]byte, 0, size*2)
}
size, _ = in.MarshalToSizedBuffer(buf[:size])
out = &model.Block{}
_ = out.Unmarshal(buf[:size])
bytesPool.Put(buf)
return
}
func EnsureStructInited(s *types.Struct) *types.Struct {
if s == nil || s.Fields == nil {
return &types.Struct{Fields: make(map[string]*types.Value)}
}
return s
}
func CopyStruct(s *types.Struct, copyVals bool) *types.Struct {
if s == nil {
return nil
}
// this state shouldn't happen in the protobuf,
// but in Go wrapper it possible so lets copy the exact state
if s.Fields == nil {
return &types.Struct{}
}
copiedStruct := &types.Struct{
Fields: make(map[string]*types.Value, len(s.Fields)),
}
for key, value := range s.Fields {
if copyVals {
copiedStruct.Fields[key] = CopyVal(value)
} else {
copiedStruct.Fields[key] = value
}
}
return copiedStruct
}
func CopyStructFields(src *types.Struct, fields ...string) *types.Struct {
newStruct := &types.Struct{
Fields: make(map[string]*types.Value, len(fields)),
}
if src.GetFields() == nil {
return newStruct
}
for _, field := range fields {
if _, ok := src.Fields[field]; ok {
newStruct.Fields[field] = CopyVal(src.Fields[field])
}
}
return newStruct
}
func CopyVal(v *types.Value) *types.Value {
if v == nil {
return nil
}
copiedValue := &types.Value{}
switch kind := v.Kind.(type) {
case *types.Value_NullValue:
copiedValue.Kind = &types.Value_NullValue{NullValue: kind.NullValue}
case *types.Value_NumberValue:
copiedValue.Kind = &types.Value_NumberValue{NumberValue: kind.NumberValue}
case *types.Value_StringValue:
copiedValue.Kind = &types.Value_StringValue{StringValue: kind.StringValue}
case *types.Value_BoolValue:
copiedValue.Kind = &types.Value_BoolValue{BoolValue: kind.BoolValue}
case *types.Value_StructValue:
copiedValue.Kind = &types.Value_StructValue{StructValue: CopyStruct(kind.StructValue, true)}
case *types.Value_ListValue:
copiedValue.Kind = &types.Value_ListValue{ListValue: CopyListVal(kind.ListValue)}
}
return copiedValue
}
func CopyListVal(lv *types.ListValue) *types.ListValue {
if lv == nil {
return nil
}
copiedListValue := &types.ListValue{
Values: make([]*types.Value, len(lv.Values)),
}
for i, value := range lv.Values {
copiedListValue.Values[i] = CopyVal(value)
}
return copiedListValue
}
func CopyRelation(in *model.Relation) (out *model.Relation) {
if in == nil {
return nil
}
buf := bytesPool.Get().([]byte)
size := in.Size()
if cap(buf) < size {
buf = make([]byte, 0, size*2)
}
size, _ = in.MarshalToSizedBuffer(buf[:size])
out = &model.Relation{}
_ = out.Unmarshal(buf[:size])
bytesPool.Put(buf)
return out
}
func CopyLayout(in *model.Layout) (out *model.Layout) {
return &model.Layout{Id: in.Id, Name: in.Name, RequiredRelations: CopyRelations(in.RequiredRelations)}
}
func CopyObjectType(in *model.ObjectType) (out *model.ObjectType) {
if in == nil {
return nil
}
buf := bytesPool.Get().([]byte)
size := in.Size()
if cap(buf) < size {
buf = make([]byte, 0, size*2)
}
size, _ = in.MarshalToSizedBuffer(buf[:size])
out = &model.ObjectType{}
_ = out.Unmarshal(buf[:size])
bytesPool.Put(buf)
return out
}
func CopyRelations(in []*model.Relation) (out []*model.Relation) {
if in == nil {
return nil
}
buf := bytesPool.Get().([]byte)
inWrapped := model.Relations{Relations: in}
size := inWrapped.Size()
if cap(buf) < size {
buf = make([]byte, 0, size*2)
}
size, _ = inWrapped.MarshalToSizedBuffer(buf[:size])
outWrapped := &model.Relations{}
_ = outWrapped.Unmarshal(buf[:size])
bytesPool.Put(buf)
return outWrapped.Relations
}
func CopyFilter(in *model.BlockContentDataviewFilter) (out *model.BlockContentDataviewFilter) {
buf := bytesPool.Get().([]byte)
size := in.Size()
if cap(buf) < size {
buf = make([]byte, 0, size*2)
}
size, _ = in.MarshalToSizedBuffer(buf[:size])
out = &model.BlockContentDataviewFilter{}
_ = out.Unmarshal(buf[:size])
bytesPool.Put(buf)
return
}
func CopyNotification(in *model.Notification) (out *model.Notification) {
var err error
buf := bytesPool.Get().([]byte)
size := in.Size()
if cap(buf) < size {
buf = make([]byte, 0, size)
}
// nolint:errcheck
size, err = in.MarshalToSizedBuffer(buf[:size])
if err != nil {
log.Debugf("failed to marshal size buffer: %s", err)
}
out = &model.Notification{}
err = out.Unmarshal(buf[:size])
if err != nil {
log.Debugf("failed to unmarshal notification: %s", err)
}
bytesPool.Put(buf)
return
}
func CopyDevice(in *model.DeviceInfo) (out *model.DeviceInfo) {
var err error
buf := bytesPool.Get().([]byte)
size := in.Size()
if cap(buf) < size {
buf = make([]byte, 0, size)
}
// nolint:errcheck
size, err = in.MarshalToSizedBuffer(buf[:size])
if err != nil {
log.Debugf("failed to marshal size buffer: %s", err)
}
out = &model.DeviceInfo{}
err = out.Unmarshal(buf[:size])
if err != nil {
log.Debugf("failed to unmarshal deviceInfo: %s", err)
}
bytesPool.Put(buf)
return
}