forked from TrueCloudLab/neoneo-go
network: implement getaddr sends and addr receival
Request new addresses if we're short on known addresses pool size.
This commit is contained in:
parent
2a49e68d77
commit
90e13debe4
3 changed files with 53 additions and 0 deletions
|
@ -3,6 +3,7 @@ package payload
|
||||||
import (
|
import (
|
||||||
"io"
|
"io"
|
||||||
"net"
|
"net"
|
||||||
|
"strconv"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/CityOfZion/neo-go/pkg/util"
|
"github.com/CityOfZion/neo-go/pkg/util"
|
||||||
|
@ -47,6 +48,15 @@ func (p *AddressAndTime) EncodeBinary(w io.Writer) error {
|
||||||
return bw.Err
|
return bw.Err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// IPPortString makes a string from IP and port specified.
|
||||||
|
func (p *AddressAndTime) IPPortString() string {
|
||||||
|
var netip net.IP = make(net.IP, 16)
|
||||||
|
|
||||||
|
copy(netip, p.IP[:])
|
||||||
|
port := strconv.Itoa(int(p.Port))
|
||||||
|
return netip.String() + ":" + port
|
||||||
|
}
|
||||||
|
|
||||||
// AddressList is a list with AddrAndTime.
|
// AddressList is a list with AddrAndTime.
|
||||||
type AddressList struct {
|
type AddressList struct {
|
||||||
Addrs []*AddressAndTime
|
Addrs []*AddressAndTime
|
||||||
|
|
|
@ -7,3 +7,22 @@ type Payload interface {
|
||||||
EncodeBinary(io.Writer) error
|
EncodeBinary(io.Writer) error
|
||||||
DecodeBinary(io.Reader) error
|
DecodeBinary(io.Reader) error
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// NullPayload is a dummy payload with no fields.
|
||||||
|
type NullPayload struct {
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewNullPayload returns zero-sized stub payload.
|
||||||
|
func NewNullPayload() *NullPayload {
|
||||||
|
return &NullPayload{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// DecodeBinary implements the Payload interface.
|
||||||
|
func (p *NullPayload) DecodeBinary(r io.Reader) error {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// EncodeBinary implements the Payload interface.
|
||||||
|
func (p *NullPayload) EncodeBinary(r io.Writer) error {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
|
@ -48,6 +48,7 @@ type (
|
||||||
lock sync.RWMutex
|
lock sync.RWMutex
|
||||||
peers map[Peer]bool
|
peers map[Peer]bool
|
||||||
|
|
||||||
|
addrReq chan *Message
|
||||||
register chan Peer
|
register chan Peer
|
||||||
unregister chan peerDrop
|
unregister chan peerDrop
|
||||||
quit chan struct{}
|
quit chan struct{}
|
||||||
|
@ -66,6 +67,7 @@ func NewServer(config ServerConfig, chain core.Blockchainer) *Server {
|
||||||
chain: chain,
|
chain: chain,
|
||||||
id: rand.Uint32(),
|
id: rand.Uint32(),
|
||||||
quit: make(chan struct{}),
|
quit: make(chan struct{}),
|
||||||
|
addrReq: make(chan *Message, 1),
|
||||||
register: make(chan Peer),
|
register: make(chan Peer),
|
||||||
unregister: make(chan peerDrop),
|
unregister: make(chan peerDrop),
|
||||||
peers: make(map[Peer]bool),
|
peers: make(map[Peer]bool),
|
||||||
|
@ -123,6 +125,15 @@ func (s *Server) run() {
|
||||||
if c < minPeers {
|
if c < minPeers {
|
||||||
s.discovery.RequestRemote(maxPeers - c)
|
s.discovery.RequestRemote(maxPeers - c)
|
||||||
}
|
}
|
||||||
|
if s.discovery.PoolCount() < minPoolCount {
|
||||||
|
select {
|
||||||
|
case s.addrReq <- NewMessage(s.Net, CMDGetAddr, payload.NewNullPayload()):
|
||||||
|
// sent request
|
||||||
|
default:
|
||||||
|
// we have one in the queue already that is
|
||||||
|
// gonna be served by some worker when it's ready
|
||||||
|
}
|
||||||
|
}
|
||||||
select {
|
select {
|
||||||
case <-s.quit:
|
case <-s.quit:
|
||||||
s.transport.Close()
|
s.transport.Close()
|
||||||
|
@ -184,6 +195,8 @@ func (s *Server) startProtocol(p Peer) {
|
||||||
case err := <-p.Done():
|
case err := <-p.Done():
|
||||||
s.unregister <- peerDrop{p, err}
|
s.unregister <- peerDrop{p, err}
|
||||||
return
|
return
|
||||||
|
case m := <-s.addrReq:
|
||||||
|
p.WriteMsg(m)
|
||||||
case <-timer.C:
|
case <-timer.C:
|
||||||
// Try to sync in headers and block with the peer if his block height is higher then ours.
|
// Try to sync in headers and block with the peer if his block height is higher then ours.
|
||||||
if p.Version().StartHeight > s.chain.BlockHeight() {
|
if p.Version().StartHeight > s.chain.BlockHeight() {
|
||||||
|
@ -253,6 +266,14 @@ func (s *Server) handleInvCmd(p Peer, inv *payload.Inventory) error {
|
||||||
return p.WriteMsg(NewMessage(s.Net, CMDGetData, payload))
|
return p.WriteMsg(NewMessage(s.Net, CMDGetData, payload))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// handleAddrCmd will process received addresses.
|
||||||
|
func (s *Server) handleAddrCmd(p Peer, addrs *payload.AddressList) error {
|
||||||
|
for _, a := range addrs.Addrs {
|
||||||
|
s.discovery.BackFill(a.IPPortString())
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
// requestHeaders will send a getheaders message to the peer.
|
// requestHeaders will send a getheaders message to the peer.
|
||||||
// The peer will respond with headers op to a count of 2000.
|
// The peer will respond with headers op to a count of 2000.
|
||||||
func (s *Server) requestHeaders(p Peer) {
|
func (s *Server) requestHeaders(p Peer) {
|
||||||
|
@ -292,6 +313,9 @@ func (s *Server) handleMessage(peer Peer, msg *Message) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
switch msg.CommandType() {
|
switch msg.CommandType() {
|
||||||
|
case CMDAddr:
|
||||||
|
addrs := msg.Payload.(*payload.AddressList)
|
||||||
|
return s.handleAddrCmd(peer, addrs)
|
||||||
case CMDVersion:
|
case CMDVersion:
|
||||||
version := msg.Payload.(*payload.Version)
|
version := msg.Payload.(*payload.Version)
|
||||||
return s.handleVersionCmd(peer, version)
|
return s.handleVersionCmd(peer, version)
|
||||||
|
|
Loading…
Reference in a new issue