forked from TrueCloudLab/neoneo-go
wip refact 2
This commit is contained in:
parent
ccaaf07dad
commit
754a473488
9 changed files with 362 additions and 177 deletions
25
pkg/network/addr_payload.go
Normal file
25
pkg/network/addr_payload.go
Normal file
|
@ -0,0 +1,25 @@
|
||||||
|
package network
|
||||||
|
|
||||||
|
import "net"
|
||||||
|
|
||||||
|
// AddrWithTimestamp payload.
|
||||||
|
type AddrWithTimestamp struct {
|
||||||
|
t uint32
|
||||||
|
services uint64
|
||||||
|
endpoint net.Addr
|
||||||
|
}
|
||||||
|
|
||||||
|
func newAddrWithTimestampFromPeer(p *Peer) AddrWithTimestamp {
|
||||||
|
return AddrWithTimestamp{
|
||||||
|
t: 1223345,
|
||||||
|
services: 1,
|
||||||
|
endpoint: p.conn.RemoteAddr(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// AddrPayload container a list of known peer addresses.
|
||||||
|
type AddrPayload []AddrWithTimestamp
|
||||||
|
|
||||||
|
func (p AddrPayload) encode() ([]byte, error) {
|
||||||
|
return nil, nil
|
||||||
|
}
|
|
@ -6,6 +6,8 @@ import (
|
||||||
"encoding/binary"
|
"encoding/binary"
|
||||||
"errors"
|
"errors"
|
||||||
"io"
|
"io"
|
||||||
|
|
||||||
|
"github.com/anthdm/neo-go/pkg/network/payload"
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
|
@ -57,7 +59,7 @@ type Message struct {
|
||||||
// hash of the payload
|
// hash of the payload
|
||||||
Checksum uint32
|
Checksum uint32
|
||||||
// Payload send with the message.
|
// Payload send with the message.
|
||||||
Payload []byte
|
Payload payload.Payloader
|
||||||
}
|
}
|
||||||
|
|
||||||
type commandType string
|
type commandType string
|
||||||
|
@ -77,19 +79,36 @@ const (
|
||||||
cmdTX = "tx"
|
cmdTX = "tx"
|
||||||
)
|
)
|
||||||
|
|
||||||
func newMessage(magic NetMode, cmd commandType, payload []byte) *Message {
|
func newMessage(magic NetMode, cmd commandType, p payload.Payloader) *Message {
|
||||||
sum := sumSHA256(sumSHA256(payload))[:4]
|
var size uint32
|
||||||
sumuint32 := binary.LittleEndian.Uint32(sum)
|
if p != nil {
|
||||||
|
size = p.Size()
|
||||||
|
}
|
||||||
return &Message{
|
return &Message{
|
||||||
Magic: magic,
|
Magic: magic,
|
||||||
Command: cmdToByteSlice(cmd),
|
Command: cmdToByteSlice(cmd),
|
||||||
Length: uint32(len(payload)),
|
Length: size,
|
||||||
Checksum: sumuint32,
|
Payload: p,
|
||||||
Payload: payload,
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TeeWriter(w io.Writer, r io.Reader) io.Writer {
|
||||||
|
return &teeWriter{w, r}
|
||||||
|
}
|
||||||
|
|
||||||
|
type teeWriter struct {
|
||||||
|
w io.Writer
|
||||||
|
r io.Reader
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *teeWriter) Write(b []byte) (n int, err error) {
|
||||||
|
n, err = w.w.Write(b)
|
||||||
|
if n > 0 {
|
||||||
|
n, err = w.r.Read(b[:n])
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
// Converts the 12 byte command slice to a commandType.
|
// Converts the 12 byte command slice to a commandType.
|
||||||
func (m *Message) commandType() commandType {
|
func (m *Message) commandType() commandType {
|
||||||
cmd := string(bytes.TrimRight(m.Command, "\x00"))
|
cmd := string(bytes.TrimRight(m.Command, "\x00"))
|
||||||
|
@ -134,140 +153,79 @@ func (m *Message) decode(r io.Reader) error {
|
||||||
m.Length = binary.LittleEndian.Uint32(buf[16:20])
|
m.Length = binary.LittleEndian.Uint32(buf[16:20])
|
||||||
m.Checksum = binary.LittleEndian.Uint32(buf[20:24])
|
m.Checksum = binary.LittleEndian.Uint32(buf[20:24])
|
||||||
|
|
||||||
payload := make([]byte, m.Length)
|
// payload is 0, so dont decode it.
|
||||||
if _, err := r.Read(payload); err != nil {
|
if m.Length == 0 {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
buffer := new(bytes.Buffer)
|
||||||
|
tr := io.TeeReader(r, buffer)
|
||||||
|
var p payload.Payloader
|
||||||
|
|
||||||
|
switch m.commandType() {
|
||||||
|
case cmdVersion:
|
||||||
|
p = &payload.Version{}
|
||||||
|
if err := p.Decode(tr); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Compare the checksum of the payload.
|
// Compare the checksum of the payload.
|
||||||
if !compareChecksum(m.Checksum, payload) {
|
if !compareChecksum(m.Checksum, buffer.Bytes()) {
|
||||||
return errors.New("checksum mismatch error")
|
return errors.New("checksum mismatch error")
|
||||||
}
|
}
|
||||||
|
|
||||||
m.Payload = payload
|
m.Payload = p
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// encode a Message to any given io.Writer.
|
// encode a Message to any given io.Writer.
|
||||||
func (m *Message) encode(w io.Writer) error {
|
func (m *Message) encode(w io.Writer) error {
|
||||||
// 24 bytes for the fixed sized fields + the length of the payload.
|
buf := make([]byte, minMessageSize)
|
||||||
buf := make([]byte, minMessageSize+m.Length)
|
pbuf := new(bytes.Buffer)
|
||||||
|
|
||||||
|
// if there is a payload fill its allocated buffer.
|
||||||
|
if m.Payload != nil {
|
||||||
|
if err := m.Payload.Encode(pbuf); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
checksum := sumSHA256(sumSHA256(pbuf.Bytes()))[:4]
|
||||||
|
m.Checksum = binary.LittleEndian.Uint32(checksum)
|
||||||
|
}
|
||||||
|
|
||||||
|
// fill the message buffer
|
||||||
binary.LittleEndian.PutUint32(buf[0:4], uint32(m.Magic))
|
binary.LittleEndian.PutUint32(buf[0:4], uint32(m.Magic))
|
||||||
copy(buf[4:16], m.Command)
|
copy(buf[4:16], m.Command)
|
||||||
binary.LittleEndian.PutUint32(buf[16:20], m.Length)
|
binary.LittleEndian.PutUint32(buf[16:20], m.Length)
|
||||||
binary.LittleEndian.PutUint32(buf[20:24], m.Checksum)
|
binary.LittleEndian.PutUint32(buf[20:24], m.Checksum)
|
||||||
copy(buf[24:len(buf)], m.Payload)
|
|
||||||
|
|
||||||
_, err := w.Write(buf)
|
// write the message
|
||||||
|
n, err := w.Write(buf)
|
||||||
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
|
||||||
|
|
||||||
func (m *Message) decodePayload() (interface{}, error) {
|
|
||||||
switch m.commandType() {
|
|
||||||
case cmdVersion:
|
|
||||||
v := &Version{}
|
|
||||||
if err := v.decode(m.Payload); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
}
|
||||||
return v, nil
|
|
||||||
|
// we need to have at least writen exactly minMessageSize bytes.
|
||||||
|
if n != minMessageSize {
|
||||||
|
return errors.New("long/short read error when encoding message")
|
||||||
}
|
}
|
||||||
return nil, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// Version payload description.
|
// write the payload if there was any
|
||||||
//
|
if pbuf.Len() > 0 {
|
||||||
// Size Field DataType Description
|
n, err = w.Write(pbuf.Bytes())
|
||||||
// ---------------------------------------------------------------------------------------------
|
if err != nil {
|
||||||
// 4 Version uint32 Version of protocol, 0 for now
|
return err
|
||||||
// 8 Services uint64 The service provided by the node is currently 1
|
|
||||||
// 4 Timestamp uint32 Current time
|
|
||||||
// 2 Port uint16 Port that the server is listening on, it's 0 if not used.
|
|
||||||
// 4 Nonce uint32 It's used to distinguish the node from public IP
|
|
||||||
// ? UserAgent varstr Client ID
|
|
||||||
// 4 StartHeight uint32 Height of block chain
|
|
||||||
// 1 Relay bool Whether to receive and forward
|
|
||||||
type Version struct {
|
|
||||||
// currently the version of the protocol is 0
|
|
||||||
Version uint32
|
|
||||||
// currently 1
|
|
||||||
Services uint64
|
|
||||||
// timestamp
|
|
||||||
Timestamp uint32
|
|
||||||
// port this server is listening on
|
|
||||||
Port uint16
|
|
||||||
// it's used to distinguish the node from public IP
|
|
||||||
Nonce uint32
|
|
||||||
// client id
|
|
||||||
UserAgent []byte // ?
|
|
||||||
// Height of the block chain
|
|
||||||
StartHeight uint32
|
|
||||||
// Whether to receive and forward
|
|
||||||
Relay bool
|
|
||||||
}
|
|
||||||
|
|
||||||
func newVersionPayload(p uint16, ua string, h uint32, r bool) *Version {
|
|
||||||
return &Version{
|
|
||||||
Version: 0,
|
|
||||||
Services: 1,
|
|
||||||
Timestamp: 12345,
|
|
||||||
Port: p,
|
|
||||||
Nonce: 19110,
|
|
||||||
UserAgent: []byte(ua),
|
|
||||||
StartHeight: 0,
|
|
||||||
Relay: r,
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
func (p *Version) decode(b []byte) error {
|
if uint32(n) != m.Payload.Size() {
|
||||||
// Fixed fields have a total of 27 bytes. We substract this size
|
return errors.New("long/short read error when encoding payload")
|
||||||
// with the total buffer length to know the length of the user agent.
|
}
|
||||||
lenUA := len(b) - 27
|
}
|
||||||
|
|
||||||
p.Version = binary.LittleEndian.Uint32(b[0:4])
|
|
||||||
p.Services = binary.LittleEndian.Uint64(b[4:12])
|
|
||||||
p.Timestamp = binary.LittleEndian.Uint32(b[12:16])
|
|
||||||
// FIXME: port's byteorder should be big endian according to the docs.
|
|
||||||
// but when connecting to the privnet docker image it's little endian.
|
|
||||||
p.Port = binary.LittleEndian.Uint16(b[16:18])
|
|
||||||
p.Nonce = binary.LittleEndian.Uint32(b[18:22])
|
|
||||||
p.UserAgent = b[22 : 22+lenUA]
|
|
||||||
curlen := 22 + lenUA
|
|
||||||
p.StartHeight = binary.LittleEndian.Uint32(b[curlen : curlen+4])
|
|
||||||
p.Relay = b[len(b)-1 : len(b)][0] == 1
|
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *Version) encode() ([]byte, error) {
|
|
||||||
// 27 bytes for the fixed size fields + the length of the user agent
|
|
||||||
// which is kinda variable, according to the docs.
|
|
||||||
buf := make([]byte, 27+len(p.UserAgent))
|
|
||||||
|
|
||||||
binary.LittleEndian.PutUint32(buf[0:4], p.Version)
|
|
||||||
binary.LittleEndian.PutUint64(buf[4:12], p.Services)
|
|
||||||
binary.LittleEndian.PutUint32(buf[12:16], p.Timestamp)
|
|
||||||
// FIXME: byte order (little / big)?
|
|
||||||
binary.LittleEndian.PutUint16(buf[16:18], p.Port)
|
|
||||||
binary.LittleEndian.PutUint32(buf[18:22], p.Nonce)
|
|
||||||
copy(buf[22:22+len(p.UserAgent)], p.UserAgent) //
|
|
||||||
curLen := 22 + len(p.UserAgent)
|
|
||||||
binary.LittleEndian.PutUint32(buf[curLen:curLen+4], p.StartHeight)
|
|
||||||
|
|
||||||
// yikes
|
|
||||||
var b []byte
|
|
||||||
if p.Relay {
|
|
||||||
b = []byte{1}
|
|
||||||
} else {
|
|
||||||
b = []byte{0}
|
|
||||||
}
|
|
||||||
|
|
||||||
copy(buf[curLen+4:len(buf)], b)
|
|
||||||
|
|
||||||
return buf, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// convert a command (string) to a byte slice filled with 0 bytes till
|
// convert a command (string) to a byte slice filled with 0 bytes till
|
||||||
// size 12.
|
// size 12.
|
||||||
func cmdToByteSlice(cmd commandType) []byte {
|
func cmdToByteSlice(cmd commandType) []byte {
|
||||||
|
|
|
@ -2,30 +2,31 @@ package network
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"encoding/binary"
|
|
||||||
"reflect"
|
"reflect"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
"github.com/anthdm/neo-go/pkg/network/payload"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestNewMessage(t *testing.T) {
|
// func TestNewMessage(t *testing.T) {
|
||||||
payload := []byte{}
|
// payload := []byte{}
|
||||||
m := newMessage(ModeTestNet, cmdVersion, payload)
|
// m := newMessage(ModeTestNet, cmdVersion, payload)
|
||||||
|
|
||||||
if have, want := m.Length, uint32(0); want != have {
|
// if have, want := m.Length, uint32(0); want != have {
|
||||||
t.Errorf("want %d have %d", want, have)
|
// t.Errorf("want %d have %d", want, have)
|
||||||
}
|
// }
|
||||||
if have, want := len(m.Command), 12; want != have {
|
// if have, want := len(m.Command), 12; want != have {
|
||||||
t.Errorf("want %d have %d", want, have)
|
// t.Errorf("want %d have %d", want, have)
|
||||||
}
|
// }
|
||||||
|
|
||||||
sum := sumSHA256(sumSHA256(payload))[:4]
|
// sum := sumSHA256(sumSHA256(payload))[:4]
|
||||||
sumuint32 := binary.LittleEndian.Uint32(sum)
|
// sumuint32 := binary.LittleEndian.Uint32(sum)
|
||||||
if have, want := m.Checksum, sumuint32; want != have {
|
// if have, want := m.Checksum, sumuint32; want != have {
|
||||||
t.Errorf("want %d have %d", want, have)
|
// t.Errorf("want %d have %d", want, have)
|
||||||
}
|
// }
|
||||||
}
|
// }
|
||||||
func TestMessageEncodeDecode(t *testing.T) {
|
func TestMessageEncodeDecode(t *testing.T) {
|
||||||
m := newMessage(ModeTestNet, cmdVersion, []byte{})
|
m := newMessage(ModeTestNet, cmdVersion, nil)
|
||||||
|
|
||||||
buf := &bytes.Buffer{}
|
buf := &bytes.Buffer{}
|
||||||
if err := m.encode(buf); err != nil {
|
if err := m.encode(buf); err != nil {
|
||||||
|
@ -48,34 +49,53 @@ func TestMessageEncodeDecode(t *testing.T) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestMessageInvalidChecksum(t *testing.T) {
|
func TestMessageEncodeDecodeWithVersion(t *testing.T) {
|
||||||
m := newMessage(ModeTestNet, cmdVersion, []byte{})
|
p := payload.NewVersion(2000, "/neo/", 0, true)
|
||||||
m.Checksum = 1337
|
m := newMessage(ModeTestNet, cmdVersion, p)
|
||||||
|
|
||||||
buf := &bytes.Buffer{}
|
buf := &bytes.Buffer{}
|
||||||
if err := m.encode(buf); err != nil {
|
if err := m.encode(buf); err != nil {
|
||||||
t.Error(err)
|
t.Error(err)
|
||||||
}
|
}
|
||||||
|
t.Log(buf.Len())
|
||||||
|
|
||||||
md := &Message{}
|
m1 := &Message{}
|
||||||
if err := md.decode(buf); err == nil {
|
if err := m1.decode(buf); err != nil {
|
||||||
t.Error("decode should failed with checkum mismatch error")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestNewVersionPayload(t *testing.T) {
|
|
||||||
ua := "/neo/0.0.1/"
|
|
||||||
p := newVersionPayload(3000, ua, 0, true)
|
|
||||||
b, err := p.encode()
|
|
||||||
if err != nil {
|
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
|
p1 := m1.Payload.(*payload.Version)
|
||||||
|
|
||||||
pd := &Version{}
|
t.Log(p1)
|
||||||
if err := pd.decode(b); err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
if !reflect.DeepEqual(p, pd) {
|
|
||||||
t.Errorf("both payloads should be equal: %v != %v", p, pd)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// func TestMessageInvalidChecksum(t *testing.T) {
|
||||||
|
// m := newMessage(ModeTestNet, cmdVersion, []byte{})
|
||||||
|
// m.Checksum = 1337
|
||||||
|
|
||||||
|
// buf := &bytes.Buffer{}
|
||||||
|
// if err := m.encode(buf); err != nil {
|
||||||
|
// t.Error(err)
|
||||||
|
// }
|
||||||
|
|
||||||
|
// md := &Message{}
|
||||||
|
// if err := md.decode(buf); err == nil {
|
||||||
|
// t.Error("decode should failed with checkum mismatch error")
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
|
||||||
|
// func TestNewVersionPayload(t *testing.T) {
|
||||||
|
// ua := "/neo/0.0.1/"
|
||||||
|
// p := newVersionPayload(3000, ua, 0, true)
|
||||||
|
// b, err := p.encode()
|
||||||
|
// if err != nil {
|
||||||
|
// t.Fatal(err)
|
||||||
|
// }
|
||||||
|
|
||||||
|
// pd := &Version{}
|
||||||
|
// if err := pd.decode(b); err != nil {
|
||||||
|
// t.Fatal(err)
|
||||||
|
// }
|
||||||
|
// if !reflect.DeepEqual(p, pd) {
|
||||||
|
// t.Errorf("both payloads should be equal: %v != %v", p, pd)
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
|
37
pkg/network/payload/addr.go
Normal file
37
pkg/network/payload/addr.go
Normal file
|
@ -0,0 +1,37 @@
|
||||||
|
package payload
|
||||||
|
|
||||||
|
import (
|
||||||
|
"io"
|
||||||
|
"net"
|
||||||
|
"unsafe"
|
||||||
|
)
|
||||||
|
|
||||||
|
// AddrWithTime payload
|
||||||
|
type AddrWithTime struct {
|
||||||
|
Timestamp uint32
|
||||||
|
Services uint64
|
||||||
|
Addr net.Addr
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *AddrWithTime) Size() uint32 {
|
||||||
|
return 4 + 8 + uint32(unsafe.Sizeof(p.Addr))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *AddrWithTime) Encode(r io.Reader) error {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *AddrWithTime) Decode(w io.Writer) error {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// AddressList is a slice of AddrWithTime.
|
||||||
|
type AddressList []*AddrWithTime
|
||||||
|
|
||||||
|
func (p AddressList) Encode(r io.Reader) error {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p AddressList) Decode(w io.Writer) error {
|
||||||
|
return nil
|
||||||
|
}
|
8
pkg/network/payload/addr_test.go
Normal file
8
pkg/network/payload/addr_test.go
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
package payload
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestNewAddrWithTime(t *testing.T) {
|
||||||
|
}
|
19
pkg/network/payload/payloader.go
Normal file
19
pkg/network/payload/payloader.go
Normal file
|
@ -0,0 +1,19 @@
|
||||||
|
package payload
|
||||||
|
|
||||||
|
import "io"
|
||||||
|
|
||||||
|
// Nothing is a safe non payload.
|
||||||
|
var Nothing = nothing{}
|
||||||
|
|
||||||
|
// Payloader ..
|
||||||
|
type Payloader interface {
|
||||||
|
Encode(io.Writer) error
|
||||||
|
Decode(io.Reader) error
|
||||||
|
Size() uint32
|
||||||
|
}
|
||||||
|
|
||||||
|
type nothing struct{}
|
||||||
|
|
||||||
|
func (p nothing) Encode(w io.Writer) error { return nil }
|
||||||
|
func (p nothing) Decode(R io.Reader) error { return nil }
|
||||||
|
func (p nothing) Size() uint32 { return 0 }
|
104
pkg/network/payload/version.go
Normal file
104
pkg/network/payload/version.go
Normal file
|
@ -0,0 +1,104 @@
|
||||||
|
package payload
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"encoding/binary"
|
||||||
|
"io"
|
||||||
|
)
|
||||||
|
|
||||||
|
const minVersionSize = 27
|
||||||
|
|
||||||
|
// Version payload.
|
||||||
|
type Version struct {
|
||||||
|
// currently the version of the protocol is 0
|
||||||
|
Version uint32
|
||||||
|
// currently 1
|
||||||
|
Services uint64
|
||||||
|
// timestamp
|
||||||
|
Timestamp uint32
|
||||||
|
// port this server is listening on
|
||||||
|
Port uint16
|
||||||
|
// it's used to distinguish the node from public IP
|
||||||
|
Nonce uint32
|
||||||
|
// client id currently 6 bytes \v/NEO:2.6.0/
|
||||||
|
UserAgent []byte
|
||||||
|
// Height of the block chain
|
||||||
|
StartHeight uint32
|
||||||
|
// Whether to receive and forward
|
||||||
|
Relay bool
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewVersion returns a pointer to a Version payload.
|
||||||
|
func NewVersion(p uint16, ua string, h uint32, r bool) *Version {
|
||||||
|
return &Version{
|
||||||
|
Version: 0,
|
||||||
|
Services: 1,
|
||||||
|
Timestamp: 12345,
|
||||||
|
Port: p,
|
||||||
|
Nonce: 19110,
|
||||||
|
UserAgent: []byte(ua),
|
||||||
|
StartHeight: 0,
|
||||||
|
Relay: r,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Size ..
|
||||||
|
func (p *Version) Size() uint32 {
|
||||||
|
n := minVersionSize + len(p.UserAgent)
|
||||||
|
return uint32(n)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Decode ..
|
||||||
|
func (p *Version) Decode(r io.Reader) error {
|
||||||
|
buf := new(bytes.Buffer)
|
||||||
|
if _, err := buf.ReadFrom(r); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
b := buf.Bytes()
|
||||||
|
// 27 bytes for the fixed size fields + the length of the user agent
|
||||||
|
// which is kinda variable, according to the docs.
|
||||||
|
lenUA := len(b) - minVersionSize
|
||||||
|
|
||||||
|
p.Version = binary.LittleEndian.Uint32(b[0:4])
|
||||||
|
p.Services = binary.LittleEndian.Uint64(b[4:12])
|
||||||
|
p.Timestamp = binary.LittleEndian.Uint32(b[12:16])
|
||||||
|
// FIXME: port's byteorder should be big endian according to the docs.
|
||||||
|
// but when connecting to the privnet docker image it's little endian.
|
||||||
|
p.Port = binary.LittleEndian.Uint16(b[16:18])
|
||||||
|
p.Nonce = binary.LittleEndian.Uint32(b[18:22])
|
||||||
|
p.UserAgent = b[22 : 22+lenUA]
|
||||||
|
curlen := 22 + lenUA
|
||||||
|
p.StartHeight = binary.LittleEndian.Uint32(b[curlen : curlen+4])
|
||||||
|
p.Relay = b[len(b)-1 : len(b)][0] == 1
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Encode ..
|
||||||
|
func (p *Version) Encode(w io.Writer) error {
|
||||||
|
buf := make([]byte, p.Size())
|
||||||
|
|
||||||
|
binary.LittleEndian.PutUint32(buf[0:4], p.Version)
|
||||||
|
binary.LittleEndian.PutUint64(buf[4:12], p.Services)
|
||||||
|
binary.LittleEndian.PutUint32(buf[12:16], p.Timestamp)
|
||||||
|
// FIXME: byte order (little / big)?
|
||||||
|
binary.LittleEndian.PutUint16(buf[16:18], p.Port)
|
||||||
|
binary.LittleEndian.PutUint32(buf[18:22], p.Nonce)
|
||||||
|
copy(buf[22:22+len(p.UserAgent)], p.UserAgent) //
|
||||||
|
curLen := 22 + len(p.UserAgent)
|
||||||
|
binary.LittleEndian.PutUint32(buf[curLen:curLen+4], p.StartHeight)
|
||||||
|
|
||||||
|
// yikes
|
||||||
|
var b []byte
|
||||||
|
if p.Relay {
|
||||||
|
b = []byte{1}
|
||||||
|
} else {
|
||||||
|
b = []byte{0}
|
||||||
|
}
|
||||||
|
|
||||||
|
copy(buf[curLen+4:len(buf)], b)
|
||||||
|
|
||||||
|
_, err := w.Write(buf)
|
||||||
|
return err
|
||||||
|
}
|
21
pkg/network/payload/version_test.go
Normal file
21
pkg/network/payload/version_test.go
Normal file
|
@ -0,0 +1,21 @@
|
||||||
|
package payload
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"reflect"
|
||||||
|
"testing"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestVersionEncodeDecode(t *testing.T) {
|
||||||
|
p := NewVersion(3000, "/NEO/", 0, true)
|
||||||
|
|
||||||
|
buf := new(bytes.Buffer)
|
||||||
|
p.Encode(buf)
|
||||||
|
|
||||||
|
pd := &Version{}
|
||||||
|
pd.Decode(buf)
|
||||||
|
|
||||||
|
if !reflect.DeepEqual(p, pd) {
|
||||||
|
t.Fatalf("expect %v to be equal to %v", p, pd)
|
||||||
|
}
|
||||||
|
}
|
|
@ -7,6 +7,8 @@ import (
|
||||||
"net"
|
"net"
|
||||||
"os"
|
"os"
|
||||||
"strconv"
|
"strconv"
|
||||||
|
|
||||||
|
"github.com/anthdm/neo-go/pkg/network/payload"
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
|
@ -165,11 +167,11 @@ func (s *Server) processMessage(msg *Message, peer *Peer) error {
|
||||||
|
|
||||||
switch msg.commandType() {
|
switch msg.commandType() {
|
||||||
case cmdVersion:
|
case cmdVersion:
|
||||||
v, err := msg.decodePayload()
|
// v, err := msg.decodePayload()
|
||||||
if err != nil {
|
// if err != nil {
|
||||||
return err
|
// return err
|
||||||
}
|
// }
|
||||||
return s.handleVersionCmd(v.(*Version), peer)
|
// return s.handleVersionCmd(v.(*Version), peer)
|
||||||
case cmdVerack:
|
case cmdVerack:
|
||||||
case cmdGetAddr:
|
case cmdGetAddr:
|
||||||
return s.handleGetAddrCmd(msg, peer)
|
return s.handleGetAddrCmd(msg, peer)
|
||||||
|
@ -192,27 +194,18 @@ func (s *Server) processMessage(msg *Message, peer *Peer) error {
|
||||||
// No further communication should been made before both sides has received
|
// No further communication should been made before both sides has received
|
||||||
// the version of eachother.
|
// the version of eachother.
|
||||||
func (s *Server) handlePeerConnected() (*Message, error) {
|
func (s *Server) handlePeerConnected() (*Message, error) {
|
||||||
payload := newVersionPayload(s.port, s.userAgent, 0, s.relay)
|
payload := payload.NewVersion(s.port, s.userAgent, 0, s.relay)
|
||||||
b, err := payload.encode()
|
msg := newMessage(s.net, cmdVersion, payload)
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
msg := newMessage(s.net, cmdVersion, b)
|
|
||||||
return msg, nil
|
return msg, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Version declares the server's version.
|
// Version declares the server's version.
|
||||||
func (s *Server) handleVersionCmd(v *Version, peer *Peer) error {
|
func (s *Server) handleVersionCmd(v *payload.Version, peer *Peer) error {
|
||||||
// TODO: check version and verify to trust that node.
|
// TODO: check version and verify to trust that node.
|
||||||
|
|
||||||
payload := newVersionPayload(s.port, s.userAgent, 0, s.relay)
|
payload := payload.NewVersion(s.port, s.userAgent, 0, s.relay)
|
||||||
b, err := payload.encode()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
// we respond with our version.
|
// we respond with our version.
|
||||||
versionMsg := newMessage(s.net, cmdVersion, b)
|
versionMsg := newMessage(s.net, cmdVersion, payload)
|
||||||
peer.send <- versionMsg
|
peer.send <- versionMsg
|
||||||
|
|
||||||
// we respond with a verack, we successfully received peer's version
|
// we respond with a verack, we successfully received peer's version
|
||||||
|
|
Loading…
Reference in a new issue