2019-08-14 01:37:45 +00:00
|
|
|
package encoder
|
|
|
|
|
|
|
|
import (
|
2019-09-24 14:56:50 +00:00
|
|
|
"fmt"
|
2019-08-14 21:18:32 +00:00
|
|
|
"io"
|
|
|
|
|
2019-08-14 01:37:45 +00:00
|
|
|
"github.com/gogo/protobuf/proto"
|
|
|
|
"github.com/golang/snappy"
|
|
|
|
"github.com/prysmaticlabs/go-ssz"
|
|
|
|
)
|
|
|
|
|
|
|
|
var _ = NetworkEncoding(&SszNetworkEncoder{})
|
|
|
|
|
2020-04-02 23:51:54 +00:00
|
|
|
// MaxChunkSize allowed for decoding messages.
|
|
|
|
const MaxChunkSize = uint64(1 << 20) // 1Mb
|
|
|
|
|
2019-08-14 01:37:45 +00:00
|
|
|
// SszNetworkEncoder supports p2p networking encoding using SimpleSerialize
|
|
|
|
// with snappy compression (if enabled).
|
|
|
|
type SszNetworkEncoder struct {
|
|
|
|
UseSnappyCompression bool
|
|
|
|
}
|
|
|
|
|
2019-09-10 14:24:14 +00:00
|
|
|
func (e SszNetworkEncoder) doEncode(msg interface{}) ([]byte, error) {
|
2019-09-09 02:34:52 +00:00
|
|
|
b, err := ssz.Marshal(msg)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if e.UseSnappyCompression {
|
|
|
|
b = snappy.Encode(nil /*dst*/, b)
|
|
|
|
}
|
|
|
|
return b, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Encode the proto message to the io.Writer.
|
2019-09-10 14:24:14 +00:00
|
|
|
func (e SszNetworkEncoder) Encode(w io.Writer, msg interface{}) (int, error) {
|
2019-08-14 01:37:45 +00:00
|
|
|
if msg == nil {
|
2019-08-14 21:18:32 +00:00
|
|
|
return 0, nil
|
2019-08-14 01:37:45 +00:00
|
|
|
}
|
|
|
|
|
2019-09-09 02:34:52 +00:00
|
|
|
b, err := e.doEncode(msg)
|
2019-08-14 01:37:45 +00:00
|
|
|
if err != nil {
|
2019-08-14 21:18:32 +00:00
|
|
|
return 0, err
|
2019-08-14 01:37:45 +00:00
|
|
|
}
|
2019-09-09 02:34:52 +00:00
|
|
|
return w.Write(b)
|
|
|
|
}
|
|
|
|
|
|
|
|
// EncodeWithLength the proto message to the io.Writer. This encoding prefixes the byte slice with a protobuf varint
|
|
|
|
// to indicate the size of the message.
|
2019-09-10 14:24:14 +00:00
|
|
|
func (e SszNetworkEncoder) EncodeWithLength(w io.Writer, msg interface{}) (int, error) {
|
2019-09-09 02:34:52 +00:00
|
|
|
if msg == nil {
|
|
|
|
return 0, nil
|
|
|
|
}
|
|
|
|
b, err := e.doEncode(msg)
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
2019-08-14 01:37:45 +00:00
|
|
|
}
|
2019-08-14 21:18:32 +00:00
|
|
|
b = append(proto.EncodeVarint(uint64(len(b))), b...)
|
|
|
|
return w.Write(b)
|
2019-08-14 01:37:45 +00:00
|
|
|
}
|
|
|
|
|
2019-09-24 14:56:50 +00:00
|
|
|
// EncodeWithMaxLength the proto message to the io.Writer. This encoding prefixes the byte slice with a protobuf varint
|
|
|
|
// to indicate the size of the message. This checks that the encoded message isn't larger than the provided max limit.
|
|
|
|
func (e SszNetworkEncoder) EncodeWithMaxLength(w io.Writer, msg interface{}, maxSize uint64) (int, error) {
|
|
|
|
if msg == nil {
|
|
|
|
return 0, nil
|
|
|
|
}
|
|
|
|
b, err := e.doEncode(msg)
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
if uint64(len(b)) > maxSize {
|
|
|
|
return 0, fmt.Errorf("size of encoded message is %d which is larger than the provided max limit of %d", len(b), maxSize)
|
|
|
|
}
|
|
|
|
b = append(proto.EncodeVarint(uint64(len(b))), b...)
|
|
|
|
return w.Write(b)
|
|
|
|
}
|
|
|
|
|
2019-09-09 02:34:52 +00:00
|
|
|
// Decode the bytes to the protobuf message provided.
|
2019-09-10 14:24:14 +00:00
|
|
|
func (e SszNetworkEncoder) Decode(b []byte, to interface{}) error {
|
2019-09-09 02:34:52 +00:00
|
|
|
if e.UseSnappyCompression {
|
|
|
|
var err error
|
|
|
|
b, err = snappy.Decode(nil /*dst*/, b)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ssz.Unmarshal(b, to)
|
|
|
|
}
|
|
|
|
|
|
|
|
// DecodeWithLength the bytes from io.Reader to the protobuf message provided.
|
2019-09-10 14:24:14 +00:00
|
|
|
func (e SszNetworkEncoder) DecodeWithLength(r io.Reader, to interface{}) error {
|
2020-04-02 23:51:54 +00:00
|
|
|
return e.DecodeWithMaxLength(r, to, MaxChunkSize)
|
2019-09-24 14:56:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// DecodeWithMaxLength the bytes from io.Reader to the protobuf message provided.
|
|
|
|
// This checks that the decoded message isn't larger than the provided max limit.
|
|
|
|
func (e SszNetworkEncoder) DecodeWithMaxLength(r io.Reader, to interface{}, maxSize uint64) error {
|
2020-04-02 23:51:54 +00:00
|
|
|
if maxSize > MaxChunkSize {
|
|
|
|
return fmt.Errorf("maxSize %d exceeds max chunk size %d", maxSize, MaxChunkSize)
|
|
|
|
}
|
2019-09-24 14:56:50 +00:00
|
|
|
msgLen, err := readVarint(r)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2019-08-14 01:37:45 +00:00
|
|
|
}
|
2019-09-24 14:56:50 +00:00
|
|
|
if msgLen > maxSize {
|
|
|
|
return fmt.Errorf("size of decoded message is %d which is larger than the provided max limit of %d", msgLen, maxSize)
|
|
|
|
}
|
|
|
|
b := make([]byte, msgLen)
|
|
|
|
_, err = r.Read(b)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return e.Decode(b, to)
|
2019-08-14 01:37:45 +00:00
|
|
|
}
|
2019-08-14 21:18:32 +00:00
|
|
|
|
|
|
|
// ProtocolSuffix returns the appropriate suffix for protocol IDs.
|
|
|
|
func (e SszNetworkEncoder) ProtocolSuffix() string {
|
|
|
|
if e.UseSnappyCompression {
|
|
|
|
return "/ssz_snappy"
|
|
|
|
}
|
|
|
|
return "/ssz"
|
|
|
|
}
|