go-ethereum/peer.go

825 lines
22 KiB
Go
Raw Normal View History

2014-01-23 21:14:01 +02:00
package eth
import (
2014-02-13 16:12:16 +02:00
"bytes"
"container/list"
2014-02-10 12:45:08 +02:00
"fmt"
2014-01-23 21:14:01 +02:00
"net"
"strconv"
2014-01-31 21:01:28 +02:00
"strings"
2014-01-23 21:14:01 +02:00
"sync/atomic"
"time"
2014-07-26 12:24:44 +03:00
"github.com/ethereum/eth-go/ethchain"
"github.com/ethereum/eth-go/ethlog"
"github.com/ethereum/eth-go/ethutil"
"github.com/ethereum/eth-go/ethwire"
2014-01-23 21:14:01 +02:00
)
var peerlogger = ethlog.NewLogger("PEER")
2014-01-23 21:14:01 +02:00
const (
// The size of the output buffer for writing messages
outputBufferSize = 50
2014-03-03 12:34:04 +02:00
// Current protocol version
2014-07-07 17:06:09 +03:00
ProtocolVersion = 23
// Interval for ping/pong message
2014-06-15 01:04:18 +03:00
pingPongTimer = 2 * time.Second
2014-01-23 21:14:01 +02:00
)
2014-02-02 21:00:09 +02:00
type DiscReason byte
const (
// Values are given explicitly instead of by iota because these values are
// defined by the wire protocol spec; it is easier for humans to ensure
// correctness when values are explicit.
2014-02-02 21:00:09 +02:00
DiscReRequested = 0x00
DiscReTcpSysErr = 0x01
DiscBadProto = 0x02
DiscBadPeer = 0x03
DiscTooManyPeers = 0x04
2014-02-10 12:36:49 +02:00
DiscConnDup = 0x05
DiscGenesisErr = 0x06
DiscProtoErr = 0x07
2014-07-01 21:32:47 +03:00
DiscQuitting = 0x08
2014-02-02 21:00:09 +02:00
)
var discReasonToString = []string{
2014-07-01 21:32:47 +03:00
"requested",
"TCP sys error",
"bad protocol",
"useless peer",
"too many peers",
"already connected",
"wrong genesis block",
"incompatible network",
"quitting",
2014-02-02 21:00:09 +02:00
}
func (d DiscReason) String() string {
if len(discReasonToString) < int(d) {
2014-02-02 21:00:09 +02:00
return "Unknown"
}
return discReasonToString[d]
}
// Peer capabilities
2014-01-31 21:01:28 +02:00
type Caps byte
const (
CapPeerDiscTy = 1 << iota
CapTxTy
CapChainTy
2014-02-02 17:15:39 +02:00
2014-02-02 21:00:09 +02:00
CapDefault = CapChainTy | CapTxTy | CapPeerDiscTy
2014-01-31 21:01:28 +02:00
)
var capsToString = map[Caps]string{
2014-02-02 21:00:09 +02:00
CapPeerDiscTy: "Peer discovery",
CapTxTy: "Transaction relaying",
CapChainTy: "Block chain relaying",
2014-01-31 21:01:28 +02:00
}
2014-02-06 14:27:57 +02:00
func (c Caps) IsCap(cap Caps) bool {
return c&cap > 0
}
2014-01-31 21:01:28 +02:00
func (c Caps) String() string {
var caps []string
2014-02-06 14:27:57 +02:00
if c.IsCap(CapPeerDiscTy) {
2014-02-02 21:00:09 +02:00
caps = append(caps, capsToString[CapPeerDiscTy])
2014-01-31 21:01:28 +02:00
}
2014-02-06 14:27:57 +02:00
if c.IsCap(CapChainTy) {
2014-01-31 21:01:28 +02:00
caps = append(caps, capsToString[CapChainTy])
}
2014-02-06 14:27:57 +02:00
if c.IsCap(CapTxTy) {
2014-01-31 21:01:28 +02:00
caps = append(caps, capsToString[CapTxTy])
}
return strings.Join(caps, " | ")
}
2014-01-23 21:14:01 +02:00
type Peer struct {
// Ethereum interface
ethereum *Ethereum
// Net connection
conn net.Conn
// Output queue which is used to communicate and handle messages
outputQueue chan *ethwire.Msg
// Quit channel
quit chan bool
// Determines whether it's an inbound or outbound peer
inbound bool
// Flag for checking the peer's connectivity state
connected int32
disconnect int32
// Last known message send
lastSend time.Time
// Indicated whether a verack has been send or not
// This flag is used by writeMessage to check if messages are allowed
// to be send or not. If no version is known all messages are ignored.
versionKnown bool
// Last received pong message
lastPong int64
lastBlockReceived time.Time
2014-01-31 00:48:52 +02:00
host []byte
2014-01-31 21:01:28 +02:00
port uint16
caps Caps
2014-06-18 11:39:42 +03:00
// This peer's public key
pubkey []byte
2014-02-10 02:09:12 +02:00
// Indicated whether the node is catching up or not
2014-03-21 16:06:23 +02:00
catchingUp bool
diverted bool
2014-03-21 16:06:23 +02:00
blocksRequested int
2014-02-13 16:12:16 +02:00
2014-06-02 16:20:27 +03:00
version string
2014-06-03 11:42:55 +03:00
// We use this to give some kind of pingtime to a node, not very accurate, could be improved.
pingTime time.Duration
pingStartTime time.Time
lastRequestedBlock *ethchain.Block
2014-01-23 21:14:01 +02:00
}
func NewPeer(conn net.Conn, ethereum *Ethereum, inbound bool) *Peer {
pubkey := ethereum.KeyManager().PublicKey()[1:]
2014-02-13 16:12:16 +02:00
2014-01-23 21:14:01 +02:00
return &Peer{
2014-03-21 16:06:23 +02:00
outputQueue: make(chan *ethwire.Msg, outputBufferSize),
quit: make(chan bool),
ethereum: ethereum,
conn: conn,
inbound: inbound,
disconnect: 0,
connected: 1,
port: 30303,
pubkey: pubkey,
blocksRequested: 10,
caps: ethereum.ServerCaps(),
version: ethereum.ClientIdentity().String(),
2014-01-23 21:14:01 +02:00
}
}
2014-02-02 17:15:39 +02:00
func NewOutboundPeer(addr string, ethereum *Ethereum, caps Caps) *Peer {
2014-01-23 21:14:01 +02:00
p := &Peer{
outputQueue: make(chan *ethwire.Msg, outputBufferSize),
quit: make(chan bool),
ethereum: ethereum,
inbound: false,
connected: 0,
disconnect: 0,
2014-02-02 17:15:39 +02:00
caps: caps,
version: ethereum.ClientIdentity().String(),
2014-01-23 21:14:01 +02:00
}
// Set up the connection in another goroutine so we don't block the main thread
go func() {
2014-07-24 13:30:04 +03:00
conn, err := p.Connect(addr)
2014-01-23 21:14:01 +02:00
if err != nil {
2014-07-24 13:25:41 +03:00
peerlogger.Debugln("Connection to peer failed. Giving up.", err)
2014-01-23 21:14:01 +02:00
p.Stop()
2014-01-27 16:34:50 +02:00
return
2014-01-23 21:14:01 +02:00
}
p.conn = conn
// Atomically set the connection state
atomic.StoreInt32(&p.connected, 1)
atomic.StoreInt32(&p.disconnect, 0)
p.Start()
2014-01-23 21:14:01 +02:00
}()
return p
}
2014-07-24 13:30:04 +03:00
func (self *Peer) Connect(addr string) (conn net.Conn, err error) {
2014-07-26 12:24:44 +03:00
const maxTries = 3
for attempts := 0; attempts < maxTries; attempts++ {
2014-07-24 13:30:04 +03:00
conn, err = net.DialTimeout("tcp", addr, 10*time.Second)
if err != nil {
2014-07-26 12:24:44 +03:00
//peerlogger.Debugf("Peer connection failed. Retrying (%d/%d) (%s)\n", attempts+1, maxTries, addr)
time.Sleep(time.Duration(attempts*20) * time.Second)
2014-07-24 13:30:04 +03:00
continue
}
// Success
return
}
return
}
2014-06-02 16:20:27 +03:00
// Getters
2014-06-03 11:42:55 +03:00
func (p *Peer) PingTime() string {
return p.pingTime.String()
}
2014-06-02 16:20:27 +03:00
func (p *Peer) Inbound() bool {
return p.inbound
}
func (p *Peer) LastSend() time.Time {
return p.lastSend
}
func (p *Peer) LastPong() int64 {
return p.lastPong
}
func (p *Peer) Host() []byte {
return p.host
}
func (p *Peer) Port() uint16 {
return p.port
}
func (p *Peer) Version() string {
return p.version
}
func (p *Peer) Connected() *int32 {
return &p.connected
}
// Setters
func (p *Peer) SetVersion(version string) {
p.version = version
}
2014-01-23 21:14:01 +02:00
// Outputs any RLP encoded data to the peer
func (p *Peer) QueueMessage(msg *ethwire.Msg) {
if atomic.LoadInt32(&p.connected) != 1 {
return
}
2014-01-23 21:14:01 +02:00
p.outputQueue <- msg
}
func (p *Peer) writeMessage(msg *ethwire.Msg) {
// Ignore the write if we're not connected
if atomic.LoadInt32(&p.connected) != 1 {
return
}
if !p.versionKnown {
switch msg.Type {
case ethwire.MsgHandshakeTy: // Ok
default: // Anything but ack is allowed
return
}
}
2014-07-10 16:03:26 +03:00
peerlogger.DebugDetailf("(%v) <= %v %v\n", p.conn.RemoteAddr(), msg.Type, msg.Data)
2014-06-14 16:44:13 +03:00
2014-01-23 21:14:01 +02:00
err := ethwire.WriteMessage(p.conn, msg)
if err != nil {
peerlogger.Debugln(" Can't send message:", err)
2014-01-23 21:14:01 +02:00
// Stop the client if there was an error writing to it
p.Stop()
return
}
}
// Outbound message handler. Outbound messages are handled here
func (p *Peer) HandleOutbound() {
// The ping timer. Makes sure that every 2 minutes a ping is send to the peer
pingTimer := time.NewTicker(pingPongTimer)
2014-02-02 21:06:37 +02:00
serviceTimer := time.NewTicker(5 * time.Minute)
2014-02-11 19:46:28 +02:00
2014-01-23 21:14:01 +02:00
out:
for {
select {
// Main message queue. All outbound messages are processed through here
case msg := <-p.outputQueue:
p.writeMessage(msg)
p.lastSend = time.Now()
// Ping timer
2014-02-02 17:15:39 +02:00
case <-pingTimer.C:
timeSince := time.Since(time.Unix(p.lastPong, 0))
2014-06-15 01:04:18 +03:00
if !p.pingStartTime.IsZero() && p.lastPong != 0 && timeSince > (pingPongTimer+30*time.Second) {
peerlogger.Infof("Peer did not respond to latest pong fast enough, it took %s, disconnecting.\n", timeSince)
p.Stop()
return
}
2014-01-24 18:48:21 +02:00
p.writeMessage(ethwire.NewMessage(ethwire.MsgPingTy, ""))
2014-06-03 11:42:55 +03:00
p.pingStartTime = time.Now()
2014-01-23 21:14:01 +02:00
2014-02-02 17:15:39 +02:00
// Service timer takes care of peer broadcasting, transaction
// posting or block posting
case <-serviceTimer.C:
2014-02-02 21:00:09 +02:00
if p.caps&CapPeerDiscTy > 0 {
2014-02-02 17:15:39 +02:00
msg := p.peersMessage()
p.ethereum.BroadcastMsg(msg)
}
2014-01-23 21:14:01 +02:00
case <-p.quit:
2014-02-02 17:15:39 +02:00
// Break out of the for loop if a quit message is posted
2014-01-23 21:14:01 +02:00
break out
}
}
clean:
// This loop is for draining the output queue and anybody waiting for us
for {
select {
case <-p.outputQueue:
// TODO
default:
break clean
}
}
}
// Inbound handler. Inbound messages are received here and passed to the appropriate methods
func (p *Peer) HandleInbound() {
for atomic.LoadInt32(&p.disconnect) == 0 {
2014-06-03 11:42:55 +03:00
2014-02-10 02:09:12 +02:00
// HMM?
2014-07-15 21:36:11 +03:00
time.Sleep(50 * time.Millisecond)
2014-01-23 21:14:01 +02:00
// Wait for a message from the peer
2014-01-31 00:48:52 +02:00
msgs, err := ethwire.ReadMessages(p.conn)
if err != nil {
peerlogger.Debugln(err)
}
for _, msg := range msgs {
2014-07-10 16:03:26 +03:00
peerlogger.DebugDetailf("(%v) => %v %v\n", p.conn.RemoteAddr(), msg.Type, msg.Data)
2014-06-14 16:44:13 +03:00
2014-07-18 12:57:44 +03:00
nextMsg:
2014-01-31 00:48:52 +02:00
switch msg.Type {
case ethwire.MsgHandshakeTy:
// Version message
p.handleHandshake(msg)
2014-01-31 01:56:32 +02:00
2014-02-06 14:27:57 +02:00
if p.caps.IsCap(CapPeerDiscTy) {
p.QueueMessage(ethwire.NewMessage(ethwire.MsgGetPeersTy, ""))
}
2014-01-31 00:48:52 +02:00
case ethwire.MsgDiscTy:
p.Stop()
peerlogger.Infoln("Disconnect peer:", DiscReason(msg.Data.Get(0).Uint()))
2014-01-31 00:48:52 +02:00
case ethwire.MsgPingTy:
// Respond back with pong
p.QueueMessage(ethwire.NewMessage(ethwire.MsgPongTy, ""))
case ethwire.MsgPongTy:
// If we received a pong back from a peer we set the
// last pong so the peer handler knows this peer is still
// active.
p.lastPong = time.Now().Unix()
2014-06-03 11:42:55 +03:00
p.pingTime = time.Now().Sub(p.pingStartTime)
2014-01-31 00:48:52 +02:00
case ethwire.MsgBlockTy:
// Get all blocks and process them
2014-02-13 16:12:16 +02:00
var block, lastBlock *ethchain.Block
var err error
2014-03-21 16:06:23 +02:00
// Make sure we are actually receiving anything
if msg.Data.Len()-1 > 1 && p.diverted {
// We requested blocks and now we need to make sure we have a common ancestor somewhere in these blocks so we can find
// common ground to start syncing from
2014-03-21 16:06:23 +02:00
lastBlock = ethchain.NewBlockFromRlpValue(msg.Data.Get(msg.Data.Len() - 1))
if p.lastRequestedBlock != nil && bytes.Compare(lastBlock.Hash(), p.lastRequestedBlock.Hash()) == 0 {
p.catchingUp = false
continue
}
p.lastRequestedBlock = lastBlock
peerlogger.Infof("Last block: %x. Checking if we have it locally.\n", lastBlock.Hash())
2014-03-21 16:06:23 +02:00
for i := msg.Data.Len() - 1; i >= 0; i-- {
block = ethchain.NewBlockFromRlpValue(msg.Data.Get(i))
// Do we have this block on our chain? If so we can continue
2014-04-30 18:43:48 +03:00
if !p.ethereum.StateManager().BlockChain().HasBlock(block.Hash()) {
2014-03-21 16:06:23 +02:00
// We don't have this block, but we do have a block with the same prevHash, diversion time!
if p.ethereum.StateManager().BlockChain().HasBlockWithPrevHash(block.PrevHash) {
p.diverted = false
if !p.ethereum.StateManager().BlockChain().FindCanonicalChainFromMsg(msg, block.PrevHash) {
p.SyncWithPeerToLastKnown()
2014-07-18 12:57:44 +03:00
break nextMsg
}
break
2014-03-21 16:06:23 +02:00
}
}
}
if !p.ethereum.StateManager().BlockChain().HasBlock(lastBlock.Hash()) {
// If we can't find a common ancenstor we need to request more blocks.
// FIXME: At one point this won't scale anymore since we are not asking for an offset
// we just keep increasing the amount of blocks.
p.blocksRequested = p.blocksRequested * 2
peerlogger.Infof("No common ancestor found, requesting %d more blocks.\n", p.blocksRequested)
p.FindCommonParentBlock()
2014-07-18 12:57:44 +03:00
break nextMsg
}
2014-07-18 12:57:44 +03:00
p.catchingUp = false
2014-03-21 16:06:23 +02:00
}
2014-02-13 16:12:16 +02:00
for i := msg.Data.Len() - 1; i >= 0; i-- {
block = ethchain.NewBlockFromRlpValue(msg.Data.Get(i))
2014-03-17 11:33:03 +02:00
2014-05-21 12:42:20 +03:00
err = p.ethereum.StateManager().Process(block, false)
2014-01-31 00:48:52 +02:00
if err != nil {
2014-02-16 21:33:23 +02:00
if ethutil.Config.Debug {
peerlogger.Infof("Block %x failed\n", block.Hash())
peerlogger.Infof("%v\n", err)
peerlogger.Debugln(block)
2014-02-16 21:33:23 +02:00
}
2014-02-13 16:12:16 +02:00
break
} else {
lastBlock = block
2014-01-31 00:48:52 +02:00
}
p.lastBlockReceived = time.Now()
2014-01-25 18:13:33 +02:00
}
2014-02-10 12:20:42 +02:00
2014-07-18 14:00:22 +03:00
if msg.Data.Len() <= 1 {
// Set catching up to false if
// the peer has nothing left to give
p.catchingUp = false
}
2014-02-13 16:12:16 +02:00
if err != nil {
// If the parent is unknown try to catch up with this peer
if ethchain.IsParentErr(err) {
2014-07-13 18:45:39 +03:00
/*
b := ethchain.NewBlockFromRlpValue(msg.Data.Get(0))
peerlogger.Infof("Attempting to catch (%x). Parent known\n", b.Hash())
p.catchingUp = false
p.CatchupWithPeer(b.Hash())
peerlogger.Infoln(b)
*/
peerlogger.Infoln("Attempting to catch. Parent known")
2014-02-13 16:12:16 +02:00
p.catchingUp = false
2014-03-21 16:06:23 +02:00
p.CatchupWithPeer(p.ethereum.BlockChain().CurrentBlock.Hash())
2014-02-15 00:56:09 +02:00
} else if ethchain.IsValidationErr(err) {
fmt.Println("Err:", err)
2014-05-14 21:35:23 +03:00
p.catchingUp = false
2014-02-13 16:12:16 +02:00
}
} else {
// If we're catching up, try to catch up further.
if p.catchingUp && msg.Data.Len() > 1 {
if lastBlock != nil {
2014-02-13 16:12:16 +02:00
blockInfo := lastBlock.BlockInfo()
peerlogger.DebugDetailf("Synced chain to #%d %x %x\n", blockInfo.Number, lastBlock.Hash(), blockInfo.Hash)
2014-02-13 16:12:16 +02:00
}
2014-05-14 21:35:23 +03:00
2014-02-13 16:12:16 +02:00
p.catchingUp = false
2014-05-14 21:35:23 +03:00
hash := p.ethereum.BlockChain().CurrentBlock.Hash()
p.CatchupWithPeer(hash)
2014-02-10 12:36:49 +02:00
}
2014-02-10 12:20:42 +02:00
}
2014-01-31 00:48:52 +02:00
case ethwire.MsgTxTy:
// If the message was a transaction queue the transaction
// in the TxPool where it will undergo validation and
// processing when a new block is found
2014-02-13 16:12:16 +02:00
for i := 0; i < msg.Data.Len(); i++ {
tx := ethchain.NewTransactionFromValue(msg.Data.Get(i))
p.ethereum.TxPool().QueueTransaction(tx)
2014-01-23 21:14:01 +02:00
}
2014-01-31 00:48:52 +02:00
case ethwire.MsgGetPeersTy:
// Peer asked for list of connected peers
p.pushPeers()
case ethwire.MsgPeersTy:
// Received a list of peers (probably because MsgGetPeersTy was send)
2014-07-24 13:04:15 +03:00
data := msg.Data
// Create new list of possible peers for the ethereum to process
peers := make([]string, data.Len())
// Parse each possible peer
for i := 0; i < data.Len(); i++ {
value := data.Get(i)
peers[i] = unpackAddr(value.Get(0), value.Get(1).Uint())
}
2014-01-31 00:48:52 +02:00
2014-07-24 13:04:15 +03:00
// Connect to the list of peers
p.ethereum.ProcessPeerList(peers)
2014-01-31 00:48:52 +02:00
case ethwire.MsgGetChainTy:
var parent *ethchain.Block
// Length minus one since the very last element in the array is a count
2014-02-13 16:12:16 +02:00
l := msg.Data.Len() - 1
2014-01-31 00:48:52 +02:00
// Ignore empty get chains
2014-02-06 14:27:57 +02:00
if l == 0 {
2014-01-28 16:35:44 +02:00
break
2014-01-27 16:34:50 +02:00
}
2014-01-31 00:48:52 +02:00
// Amount of parents in the canonical chain
2014-02-08 22:02:42 +02:00
//amountOfBlocks := msg.Data.Get(l).AsUint()
amountOfBlocks := uint64(100)
2014-03-21 16:06:23 +02:00
2014-01-31 00:48:52 +02:00
// Check each SHA block hash from the message and determine whether
// the SHA is in the database
for i := 0; i < l; i++ {
2014-03-21 16:06:23 +02:00
if data := msg.Data.Get(i).Bytes(); p.ethereum.StateManager().BlockChain().HasBlock(data) {
parent = p.ethereum.BlockChain().GetBlock(data)
2014-01-31 00:48:52 +02:00
break
}
}
// If a parent is found send back a reply
if parent != nil {
peerlogger.DebugDetailf("Found canonical block, returning chain from: %x ", parent.Hash())
chain := p.ethereum.BlockChain().GetChainFromHash(parent.Hash(), amountOfBlocks)
2014-04-29 13:36:27 +03:00
if len(chain) > 0 {
//peerlogger.Debugf("Returning %d blocks: %x ", len(chain), parent.Hash())
2014-04-29 13:36:27 +03:00
p.QueueMessage(ethwire.NewMessage(ethwire.MsgBlockTy, chain))
2014-05-14 21:35:23 +03:00
} else {
2014-05-14 21:36:21 +03:00
p.QueueMessage(ethwire.NewMessage(ethwire.MsgBlockTy, []interface{}{}))
2014-04-29 13:36:27 +03:00
}
2014-05-14 21:35:23 +03:00
2014-01-31 00:48:52 +02:00
} else {
//peerlogger.Debugf("Could not find a similar block")
2014-01-31 00:48:52 +02:00
// If no blocks are found we send back a reply with msg not in chain
// and the last hash from get chain
if l > 0 {
lastHash := msg.Data.Get(l - 1)
//log.Printf("Sending not in chain with hash %x\n", lastHash.AsRaw())
p.QueueMessage(ethwire.NewMessage(ethwire.MsgNotInChainTy, []interface{}{lastHash.Raw()}))
}
2014-01-31 00:48:52 +02:00
}
case ethwire.MsgNotInChainTy:
peerlogger.DebugDetailf("Not in chain: %x\n", msg.Data.Get(0).Bytes())
if p.diverted == true {
// If were already looking for a common parent and we get here again we need to go deeper
p.blocksRequested = p.blocksRequested * 2
}
p.diverted = true
p.catchingUp = false
p.FindCommonParentBlock()
case ethwire.MsgGetTxsTy:
// Get the current transactions of the pool
txs := p.ethereum.TxPool().CurrentTransactions()
// Get the RlpData values from the txs
txsInterface := make([]interface{}, len(txs))
for i, tx := range txs {
txsInterface[i] = tx.RlpData()
}
// Broadcast it back to the peer
p.QueueMessage(ethwire.NewMessage(ethwire.MsgTxTy, txsInterface))
2014-01-25 18:13:33 +02:00
2014-01-31 00:48:52 +02:00
// Unofficial but fun nonetheless
case ethwire.MsgTalkTy:
peerlogger.Infoln("%v says: %s\n", p.conn.RemoteAddr(), msg.Data.Str())
2014-01-31 00:48:52 +02:00
}
2014-01-23 21:14:01 +02:00
}
}
p.Stop()
}
// General update method
func (self *Peer) update() {
serviceTimer := time.NewTicker(5 * time.Second)
out:
for {
select {
case <-serviceTimer.C:
if time.Since(self.lastBlockReceived) > 10*time.Second {
self.catchingUp = false
}
case <-self.quit:
break out
}
}
serviceTimer.Stop()
}
func (p *Peer) Start() {
2014-01-31 21:01:28 +02:00
peerHost, peerPort, _ := net.SplitHostPort(p.conn.LocalAddr().String())
servHost, servPort, _ := net.SplitHostPort(p.conn.RemoteAddr().String())
2014-01-31 01:56:32 +02:00
2014-01-31 21:01:28 +02:00
if p.inbound {
p.host, p.port = packAddr(peerHost, peerPort)
} else {
p.host, p.port = packAddr(servHost, servPort)
2014-01-31 01:56:32 +02:00
}
err := p.pushHandshake()
if err != nil {
peerlogger.Debugln("Peer can't send outbound version ack", err)
2014-01-31 01:56:32 +02:00
p.Stop()
return
2014-01-23 21:14:01 +02:00
}
go p.HandleOutbound()
// Run the inbound handler in a new goroutine
go p.HandleInbound()
// Run the general update handler
go p.update()
2014-01-31 00:48:52 +02:00
2014-06-03 11:42:55 +03:00
// Wait a few seconds for startup and then ask for an initial ping
time.Sleep(2 * time.Second)
p.writeMessage(ethwire.NewMessage(ethwire.MsgPingTy, ""))
p.pingStartTime = time.Now()
2014-01-23 21:14:01 +02:00
}
func (p *Peer) Stop() {
if atomic.AddInt32(&p.disconnect, 1) != 1 {
return
}
close(p.quit)
if atomic.LoadInt32(&p.connected) != 0 {
2014-01-25 18:13:33 +02:00
p.writeMessage(ethwire.NewMessage(ethwire.MsgDiscTy, ""))
2014-01-23 21:14:01 +02:00
p.conn.Close()
}
// Pre-emptively remove the peer; don't wait for reaping. We already know it's dead if we are here
p.ethereum.RemovePeer(p)
2014-01-23 21:14:01 +02:00
}
func (p *Peer) pushHandshake() error {
pubkey := p.ethereum.KeyManager().PublicKey()
msg := ethwire.NewMessage(ethwire.MsgHandshakeTy, []interface{}{
uint32(ProtocolVersion), uint32(0), []byte(p.version), byte(p.caps), p.port, pubkey[1:],
})
2014-01-23 21:14:01 +02:00
p.QueueMessage(msg)
2014-01-23 21:14:01 +02:00
return nil
}
2014-02-02 17:15:39 +02:00
func (p *Peer) peersMessage() *ethwire.Msg {
outPeers := make([]interface{}, len(p.ethereum.InOutPeers()))
2014-01-23 21:14:01 +02:00
// Serialise each peer
for i, peer := range p.ethereum.InOutPeers() {
// Don't return localhost as valid peer
if !net.ParseIP(peer.conn.RemoteAddr().String()).IsLoopback() {
outPeers[i] = peer.RlpData()
}
2014-01-23 21:14:01 +02:00
}
2014-02-02 17:15:39 +02:00
// Return the message to the peer with the known list of connected clients
return ethwire.NewMessage(ethwire.MsgPeersTy, outPeers)
}
2014-01-23 21:14:01 +02:00
2014-02-02 17:15:39 +02:00
// Pushes the list of outbound peers to the client when requested
func (p *Peer) pushPeers() {
p.QueueMessage(p.peersMessage())
2014-01-23 21:14:01 +02:00
}
func (p *Peer) handleHandshake(msg *ethwire.Msg) {
c := msg.Data
// Set pubkey
p.pubkey = c.Get(5).Bytes()
if p.pubkey == nil {
peerlogger.Warnln("Pubkey required, not supplied in handshake.")
p.Stop()
return
}
usedPub := 0
// This peer is already added to the peerlist so we expect to find a double pubkey at least once
eachPeer(p.ethereum.Peers(), func(peer *Peer, e *list.Element) {
if bytes.Compare(p.pubkey, peer.pubkey) == 0 {
usedPub++
}
})
if usedPub > 0 {
peerlogger.Debugf("Pubkey %x found more then once. Already connected to client.", p.pubkey)
p.Stop()
return
}
2014-03-03 12:34:04 +02:00
if c.Get(0).Uint() != ProtocolVersion {
peerlogger.Debugf("Invalid peer version. Require protocol: %d. Received: %d\n", ProtocolVersion, c.Get(0).Uint())
p.Stop()
return
}
2014-02-10 02:09:12 +02:00
// [PROTOCOL_VERSION, NETWORK_ID, CLIENT_ID, CAPS, PORT, PUBKEY]
2014-01-23 21:14:01 +02:00
p.versionKnown = true
// If this is an inbound connection send an ack back
if p.inbound {
2014-02-13 16:12:16 +02:00
p.port = uint16(c.Get(4).Uint())
2014-01-23 21:14:01 +02:00
2014-02-10 02:09:12 +02:00
// Self connect detection
pubkey := p.ethereum.KeyManager().PublicKey()
if bytes.Compare(pubkey, p.pubkey) == 0 {
2014-02-13 16:12:16 +02:00
p.Stop()
2014-02-10 00:58:59 +02:00
2014-02-13 16:12:16 +02:00
return
}
}
2014-02-10 00:58:59 +02:00
2014-02-13 16:12:16 +02:00
// Set the peer's caps
p.caps = Caps(c.Get(3).Byte())
2014-06-02 16:20:27 +03:00
2014-02-13 16:12:16 +02:00
// Get a reference to the peers version
2014-06-02 16:20:27 +03:00
versionString := c.Get(2).Str()
if len(versionString) > 0 {
p.SetVersion(c.Get(2).Str())
}
2014-01-31 00:48:52 +02:00
p.ethereum.PushPeer(p)
p.ethereum.reactor.Post("peerList", p.ethereum.Peers())
ethlogger.Infof("Added peer (%s) %d / %d\n", p.conn.RemoteAddr(), p.ethereum.Peers().Len(), p.ethereum.MaxPeers)
/*
// Catch up with the connected peer
if !p.ethereum.IsUpToDate() {
peerlogger.Debugln("Already syncing up with a peer; sleeping")
time.Sleep(10 * time.Second)
}
*/
p.SyncWithPeerToLastKnown()
peerlogger.Debugln(p)
2014-02-13 16:12:16 +02:00
}
func (p *Peer) String() string {
var strBoundType string
if p.inbound {
strBoundType = "inbound"
} else {
strBoundType = "outbound"
}
var strConnectType string
if atomic.LoadInt32(&p.disconnect) == 0 {
strConnectType = "connected"
} else {
strConnectType = "disconnected"
2014-01-23 21:14:01 +02:00
}
2014-01-31 00:48:52 +02:00
2014-06-02 16:20:27 +03:00
return fmt.Sprintf("[%s] (%s) %v %s [%s]", strConnectType, strBoundType, p.conn.RemoteAddr(), p.version, p.caps)
2014-01-31 01:56:32 +02:00
2014-01-31 21:01:28 +02:00
}
func (p *Peer) SyncWithPeerToLastKnown() {
p.catchingUp = false
p.CatchupWithPeer(p.ethereum.BlockChain().CurrentBlock.Hash())
}
2014-03-21 16:06:23 +02:00
func (p *Peer) FindCommonParentBlock() {
if p.catchingUp {
return
}
2014-03-21 16:06:23 +02:00
p.catchingUp = true
if p.blocksRequested == 0 {
p.blocksRequested = 20
}
blocks := p.ethereum.BlockChain().GetChain(p.ethereum.BlockChain().CurrentBlock.Hash(), p.blocksRequested)
2014-03-21 16:06:23 +02:00
var hashes []interface{}
for _, block := range blocks {
hashes = append(hashes, block.Hash())
2014-03-21 16:06:23 +02:00
}
2014-01-31 01:56:32 +02:00
msgInfo := append(hashes, uint64(len(hashes)))
peerlogger.DebugDetailf("Asking for block from %x (%d total) from %s\n", p.ethereum.BlockChain().CurrentBlock.Hash(), len(hashes), p.conn.RemoteAddr().String())
msg := ethwire.NewMessage(ethwire.MsgGetChainTy, msgInfo)
p.QueueMessage(msg)
}
2014-03-21 16:06:23 +02:00
func (p *Peer) CatchupWithPeer(blockHash []byte) {
2014-02-10 02:09:12 +02:00
if !p.catchingUp {
// Make sure nobody else is catching up when you want to do this
2014-02-10 02:09:12 +02:00
p.catchingUp = true
2014-07-14 01:37:41 +03:00
msg := ethwire.NewMessage(ethwire.MsgGetChainTy, []interface{}{blockHash, uint64(30)})
2014-02-10 02:09:12 +02:00
p.QueueMessage(msg)
peerlogger.DebugDetailf("Requesting blockchain %x... from peer %s\n", p.ethereum.BlockChain().CurrentBlock.Hash()[:4], p.conn.RemoteAddr())
2014-05-21 13:39:15 +03:00
msg = ethwire.NewMessage(ethwire.MsgGetTxsTy, []interface{}{})
p.QueueMessage(msg)
2014-02-10 02:09:12 +02:00
}
}
2014-01-31 21:01:28 +02:00
func (p *Peer) RlpData() []interface{} {
return []interface{}{p.host, p.port, p.pubkey}
2014-01-23 21:14:01 +02:00
}
2014-05-01 23:13:59 +03:00
func packAddr(address, port string) ([]byte, uint16) {
2014-05-01 23:13:59 +03:00
addr := strings.Split(address, ".")
a, _ := strconv.Atoi(addr[0])
b, _ := strconv.Atoi(addr[1])
c, _ := strconv.Atoi(addr[2])
d, _ := strconv.Atoi(addr[3])
host := []byte{byte(a), byte(b), byte(c), byte(d)}
2014-05-01 23:13:59 +03:00
prt, _ := strconv.Atoi(port)
return host, uint16(prt)
}
func unpackAddr(value *ethutil.Value, p uint64) string {
byts := value.Bytes()
a := strconv.Itoa(int(byts[0]))
b := strconv.Itoa(int(byts[1]))
c := strconv.Itoa(int(byts[2]))
d := strconv.Itoa(int(byts[3]))
2014-05-01 23:13:59 +03:00
host := strings.Join([]string{a, b, c, d}, ".")
port := strconv.Itoa(int(p))
return net.JoinHostPort(host, port)
}