mirror of
https://github.com/siderolabs/talos.git
synced 2025-09-12 09:21:18 +02:00
Use `vtprobuf` optimized Marshal/Unmarshal methods which do not depend on reflection to reduce memory and CPU usage while using Talos API. See https://github.com/planetscale/vtprotobuf and https://vitess.io/blog/2021-06-03-a-new-protobuf-generator-for-go/ Co-authored-by: Andrey Smirnov <smirnov.andrey@gmail.com> Signed-off-by: Andrey Smirnov <smirnov.andrey@gmail.com> Signed-off-by: Alexey Palazhchenko <alexey.palazhchenko@talos-systems.com>
70 lines
1.9 KiB
Go
70 lines
1.9 KiB
Go
// This Source Code Form is subject to the terms of the Mozilla Public
|
|
// License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
// file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
|
|
package proto
|
|
|
|
import (
|
|
"fmt"
|
|
|
|
"google.golang.org/grpc/encoding"
|
|
"google.golang.org/grpc/encoding/proto"
|
|
)
|
|
|
|
// gogoMessage is the interface for gogoproto additions.
|
|
//
|
|
// We use only a subset of that interface but include additional methods
|
|
// to prevent accidental successful type assertion for unrelated types.
|
|
type gogoMessage interface {
|
|
Marshal() ([]byte, error)
|
|
MarshalTo([]byte) (int, error)
|
|
MarshalToSizedBuffer([]byte) (int, error)
|
|
Unmarshal([]byte) error
|
|
}
|
|
|
|
// Codec provides protobuf encoding.Codec.
|
|
type Codec struct{}
|
|
|
|
// Marshal implements encoding.Codec.
|
|
func (Codec) Marshal(v interface{}) ([]byte, error) {
|
|
// some third-party types (like from etcd and containerd) implement gogoMessage
|
|
if gm, ok := v.(gogoMessage); ok {
|
|
return gm.Marshal()
|
|
}
|
|
|
|
// our types implement Message (with or without vtproto additions depending on build configuration)
|
|
if m, ok := v.(Message); ok {
|
|
return Marshal(m)
|
|
}
|
|
|
|
// no types implement protobuf API v1 only, so don't check for it
|
|
|
|
return nil, fmt.Errorf("failed to marshal %T", v)
|
|
}
|
|
|
|
// Unmarshal implements encoding.Codec.
|
|
func (Codec) Unmarshal(data []byte, v interface{}) error {
|
|
// some third-party types (like from etcd and containerd) implement gogoMessage
|
|
if gm, ok := v.(gogoMessage); ok {
|
|
return gm.Unmarshal(data)
|
|
}
|
|
|
|
// our types implement Message (with or without vtproto additions depending on build configuration)
|
|
if m, ok := v.(Message); ok {
|
|
return Unmarshal(data, m)
|
|
}
|
|
|
|
// no types implement protobuf API v1 only, so don't check for it
|
|
|
|
return fmt.Errorf("failed to unmarshal %T", v)
|
|
}
|
|
|
|
// Name implements encoding.Codec.
|
|
func (Codec) Name() string {
|
|
return proto.Name // overrides google.golang.org/grpc/encoding/proto codec
|
|
}
|
|
|
|
func init() {
|
|
encoding.RegisterCodec(Codec{})
|
|
}
|