p2p: snappy encoding for devp2p (version bump to 5) (#15106)
* p2p: snappy encoding for devp2p (version bump to 5) * p2p: remove lazy decompression, enforce 16MB limit
This commit is contained in:
parent
2b4a5f2677
commit
2ee885958b
|
@ -32,10 +32,12 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
baseProtocolVersion = 4
|
baseProtocolVersion = 5
|
||||||
baseProtocolLength = uint64(16)
|
baseProtocolLength = uint64(16)
|
||||||
baseProtocolMaxMsgSize = 2 * 1024
|
baseProtocolMaxMsgSize = 2 * 1024
|
||||||
|
|
||||||
|
snappyProtocolVersion = 5
|
||||||
|
|
||||||
pingInterval = 15 * time.Second
|
pingInterval = 15 * time.Second
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
42
p2p/rlpx.go
42
p2p/rlpx.go
|
@ -29,6 +29,7 @@ import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"hash"
|
"hash"
|
||||||
"io"
|
"io"
|
||||||
|
"io/ioutil"
|
||||||
mrand "math/rand"
|
mrand "math/rand"
|
||||||
"net"
|
"net"
|
||||||
"sync"
|
"sync"
|
||||||
|
@ -40,6 +41,7 @@ import (
|
||||||
"github.com/ethereum/go-ethereum/crypto/sha3"
|
"github.com/ethereum/go-ethereum/crypto/sha3"
|
||||||
"github.com/ethereum/go-ethereum/p2p/discover"
|
"github.com/ethereum/go-ethereum/p2p/discover"
|
||||||
"github.com/ethereum/go-ethereum/rlp"
|
"github.com/ethereum/go-ethereum/rlp"
|
||||||
|
"github.com/golang/snappy"
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
|
@ -68,6 +70,10 @@ const (
|
||||||
discWriteTimeout = 1 * time.Second
|
discWriteTimeout = 1 * time.Second
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// errPlainMessageTooLarge is returned if a decompressed message length exceeds
|
||||||
|
// the allowed 24 bits (i.e. length >= 16MB).
|
||||||
|
var errPlainMessageTooLarge = errors.New("message length >= 16MB")
|
||||||
|
|
||||||
// rlpx is the transport protocol used by actual (non-test) connections.
|
// rlpx is the transport protocol used by actual (non-test) connections.
|
||||||
// It wraps the frame encoder with locks and read/write deadlines.
|
// It wraps the frame encoder with locks and read/write deadlines.
|
||||||
type rlpx struct {
|
type rlpx struct {
|
||||||
|
@ -127,6 +133,9 @@ func (t *rlpx) doProtoHandshake(our *protoHandshake) (their *protoHandshake, err
|
||||||
if err := <-werr; err != nil {
|
if err := <-werr; err != nil {
|
||||||
return nil, fmt.Errorf("write error: %v", err)
|
return nil, fmt.Errorf("write error: %v", err)
|
||||||
}
|
}
|
||||||
|
// If the protocol version supports Snappy encoding, upgrade immediately
|
||||||
|
t.rw.snappy = their.Version >= snappyProtocolVersion
|
||||||
|
|
||||||
return their, nil
|
return their, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -556,6 +565,8 @@ type rlpxFrameRW struct {
|
||||||
macCipher cipher.Block
|
macCipher cipher.Block
|
||||||
egressMAC hash.Hash
|
egressMAC hash.Hash
|
||||||
ingressMAC hash.Hash
|
ingressMAC hash.Hash
|
||||||
|
|
||||||
|
snappy bool
|
||||||
}
|
}
|
||||||
|
|
||||||
func newRLPXFrameRW(conn io.ReadWriter, s secrets) *rlpxFrameRW {
|
func newRLPXFrameRW(conn io.ReadWriter, s secrets) *rlpxFrameRW {
|
||||||
|
@ -583,6 +594,17 @@ func newRLPXFrameRW(conn io.ReadWriter, s secrets) *rlpxFrameRW {
|
||||||
func (rw *rlpxFrameRW) WriteMsg(msg Msg) error {
|
func (rw *rlpxFrameRW) WriteMsg(msg Msg) error {
|
||||||
ptype, _ := rlp.EncodeToBytes(msg.Code)
|
ptype, _ := rlp.EncodeToBytes(msg.Code)
|
||||||
|
|
||||||
|
// if snappy is enabled, compress message now
|
||||||
|
if rw.snappy {
|
||||||
|
if msg.Size > maxUint24 {
|
||||||
|
return errPlainMessageTooLarge
|
||||||
|
}
|
||||||
|
payload, _ := ioutil.ReadAll(msg.Payload)
|
||||||
|
payload = snappy.Encode(nil, payload)
|
||||||
|
|
||||||
|
msg.Payload = bytes.NewReader(payload)
|
||||||
|
msg.Size = uint32(len(payload))
|
||||||
|
}
|
||||||
// write header
|
// write header
|
||||||
headbuf := make([]byte, 32)
|
headbuf := make([]byte, 32)
|
||||||
fsize := uint32(len(ptype)) + msg.Size
|
fsize := uint32(len(ptype)) + msg.Size
|
||||||
|
@ -668,6 +690,26 @@ func (rw *rlpxFrameRW) ReadMsg() (msg Msg, err error) {
|
||||||
}
|
}
|
||||||
msg.Size = uint32(content.Len())
|
msg.Size = uint32(content.Len())
|
||||||
msg.Payload = content
|
msg.Payload = content
|
||||||
|
|
||||||
|
// if snappy is enabled, verify and decompress message
|
||||||
|
if rw.snappy {
|
||||||
|
payload, err := ioutil.ReadAll(msg.Payload)
|
||||||
|
if err != nil {
|
||||||
|
return msg, err
|
||||||
|
}
|
||||||
|
size, err := snappy.DecodedLen(payload)
|
||||||
|
if err != nil {
|
||||||
|
return msg, err
|
||||||
|
}
|
||||||
|
if size > int(maxUint24) {
|
||||||
|
return msg, errPlainMessageTooLarge
|
||||||
|
}
|
||||||
|
payload, err = snappy.Decode(nil, payload)
|
||||||
|
if err != nil {
|
||||||
|
return msg, err
|
||||||
|
}
|
||||||
|
msg.Size, msg.Payload = uint32(size), bytes.NewReader(payload)
|
||||||
|
}
|
||||||
return msg, nil
|
return msg, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue