2018-03-09 15:55:25 +00:00
|
|
|
package network
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"os"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/CityOfZion/neo-go/pkg/core"
|
|
|
|
"github.com/CityOfZion/neo-go/pkg/network/payload"
|
|
|
|
"github.com/CityOfZion/neo-go/pkg/util"
|
|
|
|
log "github.com/go-kit/kit/log"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
protoVersion = 0
|
|
|
|
)
|
|
|
|
|
|
|
|
var protoTickInterval = 5 * time.Second
|
|
|
|
|
|
|
|
// Node represents the local node.
|
|
|
|
type Node struct {
|
|
|
|
// Config fields may not be modified while the server is running.
|
|
|
|
Config
|
|
|
|
|
|
|
|
logger log.Logger
|
|
|
|
server *Server
|
|
|
|
services uint64
|
|
|
|
bc *core.Blockchain
|
|
|
|
}
|
|
|
|
|
|
|
|
func newNode(s *Server, cfg Config) *Node {
|
|
|
|
var startHash util.Uint256
|
|
|
|
if cfg.Net == ModePrivNet {
|
|
|
|
startHash = core.GenesisHashPrivNet()
|
|
|
|
}
|
2018-03-10 12:04:06 +00:00
|
|
|
if cfg.Net == ModeTestNet {
|
|
|
|
startHash = core.GenesisHashTestNet()
|
|
|
|
}
|
|
|
|
if cfg.Net == ModeMainNet {
|
|
|
|
startHash = core.GenesisHashMainNet()
|
|
|
|
}
|
2018-03-09 15:55:25 +00:00
|
|
|
|
|
|
|
bc := core.NewBlockchain(
|
|
|
|
core.NewMemoryStore(),
|
|
|
|
startHash,
|
|
|
|
)
|
|
|
|
|
|
|
|
logger := log.NewLogfmtLogger(os.Stderr)
|
|
|
|
logger = log.With(logger, "component", "node")
|
|
|
|
|
|
|
|
n := &Node{
|
2018-03-10 12:04:06 +00:00
|
|
|
Config: cfg,
|
|
|
|
server: s,
|
|
|
|
bc: bc,
|
|
|
|
logger: logger,
|
2018-03-09 15:55:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return n
|
|
|
|
}
|
|
|
|
|
|
|
|
func (n *Node) version() *payload.Version {
|
|
|
|
return payload.NewVersion(n.server.id, n.ListenTCP, n.UserAgent, 1, n.Relay)
|
|
|
|
}
|
|
|
|
|
2018-03-10 12:04:06 +00:00
|
|
|
func (n *Node) startProtocol(p Peer) {
|
|
|
|
n.logger.Log(
|
|
|
|
"event", "start protocol",
|
|
|
|
"peer", p.Endpoint(),
|
|
|
|
"userAgent", string(p.Version().UserAgent),
|
|
|
|
)
|
|
|
|
defer func() {
|
|
|
|
n.logger.Log(
|
|
|
|
"msg", "protocol stopped",
|
|
|
|
"peer", p.Endpoint(),
|
|
|
|
)
|
|
|
|
}()
|
2018-03-09 15:55:25 +00:00
|
|
|
|
2018-03-10 12:04:06 +00:00
|
|
|
timer := time.NewTimer(protoTickInterval)
|
2018-03-09 15:55:25 +00:00
|
|
|
for {
|
2018-03-10 12:04:06 +00:00
|
|
|
<-timer.C
|
2018-03-09 15:55:25 +00:00
|
|
|
select {
|
2018-03-10 12:04:06 +00:00
|
|
|
case <-p.Done():
|
|
|
|
return
|
|
|
|
default:
|
2018-03-09 15:55:25 +00:00
|
|
|
// Try to sync with the peer if his block height is higher then ours.
|
2018-03-10 12:04:06 +00:00
|
|
|
if p.Version().StartHeight > n.bc.HeaderHeight() {
|
|
|
|
n.askMoreHeaders(p)
|
2018-03-09 15:55:25 +00:00
|
|
|
}
|
|
|
|
// Only ask for more peers if the server has the capacity for it.
|
|
|
|
if n.server.hasCapacity() {
|
|
|
|
msg := NewMessage(n.Net, CMDGetAddr, nil)
|
2018-03-10 12:04:06 +00:00
|
|
|
p.Send(msg)
|
2018-03-09 15:55:25 +00:00
|
|
|
}
|
2018-03-10 12:04:06 +00:00
|
|
|
timer.Reset(protoTickInterval)
|
2018-03-09 15:55:25 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// When a peer sends out his version we reply with verack after validating
|
|
|
|
// the version.
|
2018-03-10 12:04:06 +00:00
|
|
|
func (n *Node) handleVersionCmd(version *payload.Version, p Peer) error {
|
2018-03-09 15:55:25 +00:00
|
|
|
msg := NewMessage(n.Net, CMDVerack, nil)
|
2018-03-10 12:04:06 +00:00
|
|
|
p.Send(msg)
|
2018-03-09 15:55:25 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// handleInvCmd handles the forwarded inventory received from the peer.
|
|
|
|
// We will use the getdata message to get more details about the received
|
|
|
|
// inventory.
|
|
|
|
// note: if the server has Relay on false, inventory messages are not received.
|
2018-03-10 12:04:06 +00:00
|
|
|
func (n *Node) handleInvCmd(inv *payload.Inventory, p Peer) error {
|
2018-03-09 15:55:25 +00:00
|
|
|
if !inv.Type.Valid() {
|
|
|
|
return fmt.Errorf("invalid inventory type received: %s", inv.Type)
|
|
|
|
}
|
|
|
|
if len(inv.Hashes) == 0 {
|
|
|
|
return errors.New("inventory has no hashes")
|
|
|
|
}
|
|
|
|
payload := payload.NewInventory(inv.Type, inv.Hashes)
|
2018-03-10 12:04:06 +00:00
|
|
|
p.Send(NewMessage(n.Net, CMDGetData, payload))
|
2018-03-09 15:55:25 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// handleBlockCmd processes the received block received from its peer.
|
|
|
|
func (n *Node) handleBlockCmd(block *core.Block, peer Peer) error {
|
|
|
|
n.logger.Log(
|
|
|
|
"event", "block received",
|
|
|
|
"index", block.Index,
|
|
|
|
"hash", block.Hash(),
|
|
|
|
"tx", len(block.Transactions),
|
|
|
|
)
|
|
|
|
return n.bc.AddBlock(block)
|
|
|
|
}
|
|
|
|
|
|
|
|
// After a node sends out the getaddr message its receives a list of known peers
|
|
|
|
// in the network. handleAddrCmd processes that payload.
|
|
|
|
func (n *Node) handleAddrCmd(addressList *payload.AddressList, peer Peer) error {
|
|
|
|
addrs := make([]string, len(addressList.Addrs))
|
|
|
|
for i := 0; i < len(addrs); i++ {
|
|
|
|
addrs[i] = addressList.Addrs[i].Address.String()
|
|
|
|
}
|
|
|
|
n.server.connectToPeers(addrs...)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// The handleHeadersCmd will process the received headers from its peer.
|
|
|
|
// We call this in a routine cause we may block Peers Send() for to long.
|
|
|
|
func (n *Node) handleHeadersCmd(headers *payload.Headers, peer Peer) error {
|
|
|
|
go func(headers []*core.Header) {
|
|
|
|
if err := n.bc.AddHeaders(headers...); err != nil {
|
|
|
|
n.logger.Log("msg", "failed processing headers", "err", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
// The peer will respond with a maximum of 2000 headers in one batch.
|
|
|
|
// We will ask one more batch here if needed. Eventually we will get synced
|
|
|
|
// due to the startProtocol routine that will ask headers every protoTick.
|
|
|
|
if n.bc.HeaderHeight() < peer.Version().StartHeight {
|
|
|
|
n.askMoreHeaders(peer)
|
|
|
|
}
|
|
|
|
}(headers.Hdrs)
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// askMoreHeaders will send a getheaders message to the peer.
|
|
|
|
func (n *Node) askMoreHeaders(p Peer) {
|
|
|
|
start := []util.Uint256{n.bc.CurrentHeaderHash()}
|
|
|
|
payload := payload.NewGetBlocks(start, util.Uint256{})
|
|
|
|
p.Send(NewMessage(n.Net, CMDGetHeaders, payload))
|
|
|
|
}
|
|
|
|
|
|
|
|
// blockhain implements the Noder interface.
|
|
|
|
func (n *Node) blockchain() *core.Blockchain { return n.bc }
|
|
|
|
|
2018-03-10 12:04:06 +00:00
|
|
|
func (n *Node) handleProto(msg *Message, p Peer) error {
|
|
|
|
//n.logger.Log(
|
|
|
|
// "event", "message received",
|
|
|
|
// "from", p.Endpoint(),
|
|
|
|
// "msg", msg.CommandType(),
|
|
|
|
//)
|
|
|
|
|
|
|
|
switch msg.CommandType() {
|
|
|
|
case CMDVersion:
|
|
|
|
version := msg.Payload.(*payload.Version)
|
|
|
|
return n.handleVersionCmd(version, p)
|
|
|
|
case CMDAddr:
|
|
|
|
addressList := msg.Payload.(*payload.AddressList)
|
|
|
|
return n.handleAddrCmd(addressList, p)
|
|
|
|
case CMDInv:
|
|
|
|
inventory := msg.Payload.(*payload.Inventory)
|
|
|
|
return n.handleInvCmd(inventory, p)
|
|
|
|
case CMDBlock:
|
|
|
|
block := msg.Payload.(*core.Block)
|
|
|
|
return n.handleBlockCmd(block, p)
|
|
|
|
case CMDHeaders:
|
|
|
|
headers := msg.Payload.(*payload.Headers)
|
|
|
|
return n.handleHeadersCmd(headers, p)
|
|
|
|
case CMDTX:
|
|
|
|
// tx := msg.Payload.(*transaction.Transaction)
|
|
|
|
//n.logger.Log("tx", fmt.Sprintf("%+v", tx))
|
|
|
|
return nil
|
|
|
|
case CMDVerack:
|
|
|
|
// Only start the protocol if we got the version and verack
|
|
|
|
// received.
|
|
|
|
if p.Version() != nil {
|
|
|
|
go n.startProtocol(p)
|
2018-03-09 15:55:25 +00:00
|
|
|
}
|
2018-03-10 12:04:06 +00:00
|
|
|
return nil
|
|
|
|
case CMDUnknown:
|
|
|
|
return errors.New("received non-protocol messgae")
|
2018-03-09 15:55:25 +00:00
|
|
|
}
|
2018-03-10 12:04:06 +00:00
|
|
|
return nil
|
2018-03-09 15:55:25 +00:00
|
|
|
}
|