plugeth/peer.go

521 lines
13 KiB
Go
Raw Normal View History

2014-01-23 19:14:01 +00:00
package eth
import (
"bytes"
2014-02-10 10:45:08 +00:00
"fmt"
2014-01-23 19:55:23 +00:00
"github.com/ethereum/ethchain-go"
2014-01-23 19:14:01 +00:00
"github.com/ethereum/ethutil-go"
"github.com/ethereum/ethwire-go"
"log"
"net"
2014-02-10 10:45:08 +00:00
"runtime"
2014-01-23 19:14:01 +00:00
"strconv"
2014-01-31 19:01:28 +00:00
"strings"
2014-01-23 19:14:01 +00:00
"sync/atomic"
"time"
)
const (
// The size of the output buffer for writing messages
outputBufferSize = 50
)
2014-02-02 19:00:09 +00:00
type DiscReason byte
const (
DiscReRequested = 0x00
DiscReTcpSysErr = 0x01
DiscBadProto = 0x02
DiscBadPeer = 0x03
DiscTooManyPeers = 0x04
2014-02-10 10:36:49 +00:00
DiscConnDup = 0x05
DiscGenesisErr = 0x06
DiscProtoErr = 0x07
2014-02-02 19:00:09 +00:00
)
var discReasonToString = []string{
"Disconnect requested",
"Disconnect TCP sys error",
2014-02-10 10:36:49 +00:00
"Disconnect bad protocol",
"Disconnect useless peer",
"Disconnect too many peers",
"Disconnect already connected",
"Disconnect wrong genesis block",
"Disconnect incompatible network",
2014-02-02 19:00:09 +00:00
}
func (d DiscReason) String() string {
if len(discReasonToString) > int(d) {
return "Unknown"
}
return discReasonToString[d]
}
// Peer capabilities
2014-01-31 19:01:28 +00:00
type Caps byte
const (
2014-02-02 19:00:09 +00:00
CapPeerDiscTy = 0x01
CapTxTy = 0x02
CapChainTy = 0x04
2014-02-02 15:15:39 +00:00
2014-02-02 19:00:09 +00:00
CapDefault = CapChainTy | CapTxTy | CapPeerDiscTy
2014-01-31 19:01:28 +00:00
)
var capsToString = map[Caps]string{
2014-02-02 19:00:09 +00:00
CapPeerDiscTy: "Peer discovery",
CapTxTy: "Transaction relaying",
CapChainTy: "Block chain relaying",
2014-01-31 19:01:28 +00:00
}
2014-02-06 12:27:57 +00:00
func (c Caps) IsCap(cap Caps) bool {
return c&cap > 0
}
2014-01-31 19:01:28 +00:00
func (c Caps) String() string {
var caps []string
2014-02-06 12:27:57 +00:00
if c.IsCap(CapPeerDiscTy) {
2014-02-02 19:00:09 +00:00
caps = append(caps, capsToString[CapPeerDiscTy])
2014-01-31 19:01:28 +00:00
}
2014-02-06 12:27:57 +00:00
if c.IsCap(CapChainTy) {
2014-01-31 19:01:28 +00:00
caps = append(caps, capsToString[CapChainTy])
}
2014-02-06 12:27:57 +00:00
if c.IsCap(CapTxTy) {
2014-01-31 19:01:28 +00:00
caps = append(caps, capsToString[CapTxTy])
}
return strings.Join(caps, " | ")
}
2014-01-23 19:14:01 +00: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
// Indicates whether a MsgGetPeersTy was requested of the peer
// this to prevent receiving false peers.
requestedPeerList bool
2014-01-30 22:48:52 +00:00
2014-02-08 20:02:42 +00:00
host []interface{}
2014-01-31 19:01:28 +00:00
port uint16
caps Caps
pubkey []byte
2014-02-10 00:09:12 +00:00
// Indicated whether the node is catching up or not
catchingUp bool
2014-01-23 19:14:01 +00:00
}
func NewPeer(conn net.Conn, ethereum *Ethereum, inbound bool) *Peer {
return &Peer{
outputQueue: make(chan *ethwire.Msg, outputBufferSize),
quit: make(chan bool),
ethereum: ethereum,
conn: conn,
inbound: inbound,
disconnect: 0,
connected: 1,
2014-01-31 19:01:28 +00:00
port: 30303,
2014-01-23 19:14:01 +00:00
}
}
2014-02-02 15:15:39 +00:00
func NewOutboundPeer(addr string, ethereum *Ethereum, caps Caps) *Peer {
2014-02-09 22:58:59 +00:00
data, _ := ethutil.Config.Db.Get([]byte("KeyRing"))
pubkey := ethutil.NewValueFromBytes(data).Get(2).Bytes()
2014-01-23 19:14:01 +00:00
p := &Peer{
outputQueue: make(chan *ethwire.Msg, outputBufferSize),
quit: make(chan bool),
ethereum: ethereum,
inbound: false,
connected: 0,
disconnect: 0,
2014-02-02 15:15:39 +00:00
caps: caps,
pubkey: pubkey,
2014-01-23 19:14:01 +00:00
}
// Set up the connection in another goroutine so we don't block the main thread
go func() {
2014-01-27 14:34:50 +00:00
conn, err := net.DialTimeout("tcp", addr, 30*time.Second)
2014-01-23 19:14:01 +00:00
if err != nil {
2014-01-27 14:34:50 +00:00
log.Println("Connection to peer failed", err)
2014-01-23 19:14:01 +00:00
p.Stop()
2014-01-27 14:34:50 +00:00
return
2014-01-23 19:14:01 +00:00
}
p.conn = conn
// Atomically set the connection state
atomic.StoreInt32(&p.connected, 1)
atomic.StoreInt32(&p.disconnect, 0)
p.Start()
2014-01-23 19:14:01 +00:00
}()
return p
}
// Outputs any RLP encoded data to the peer
func (p *Peer) QueueMessage(msg *ethwire.Msg) {
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
}
}
err := ethwire.WriteMessage(p.conn, msg)
if err != nil {
log.Println("Can't send message:", err)
// 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
2014-02-02 15:15:39 +00:00
pingTimer := time.NewTicker(2 * time.Minute)
2014-02-02 19:06:37 +00:00
serviceTimer := time.NewTicker(5 * time.Minute)
2014-01-23 19:14:01 +00: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()
2014-02-02 15:15:39 +00:00
// Ping timer sends a ping to the peer each 2 minutes
case <-pingTimer.C:
2014-01-24 16:48:21 +00:00
p.writeMessage(ethwire.NewMessage(ethwire.MsgPingTy, ""))
2014-01-23 19:14:01 +00:00
2014-02-02 15:15:39 +00:00
// Service timer takes care of peer broadcasting, transaction
// posting or block posting
case <-serviceTimer.C:
2014-02-02 19:00:09 +00:00
if p.caps&CapPeerDiscTy > 0 {
2014-02-02 15:15:39 +00:00
msg := p.peersMessage()
p.ethereum.BroadcastMsg(msg)
}
2014-01-23 19:14:01 +00:00
case <-p.quit:
2014-02-02 15:15:39 +00:00
// Break out of the for loop if a quit message is posted
2014-01-23 19:14:01 +00: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-02-10 00:09:12 +00:00
// HMM?
time.Sleep(500 * time.Millisecond)
2014-01-23 19:14:01 +00:00
// Wait for a message from the peer
2014-01-30 22:48:52 +00:00
msgs, err := ethwire.ReadMessages(p.conn)
if err != nil {
log.Println(err)
}
for _, msg := range msgs {
2014-01-30 22:48:52 +00:00
switch msg.Type {
case ethwire.MsgHandshakeTy:
// Version message
p.handleHandshake(msg)
2014-01-30 23:56:32 +00:00
2014-02-06 12:27:57 +00:00
if p.caps.IsCap(CapPeerDiscTy) {
p.QueueMessage(ethwire.NewMessage(ethwire.MsgGetPeersTy, ""))
}
2014-01-30 22:48:52 +00:00
case ethwire.MsgDiscTy:
p.Stop()
2014-02-02 19:00:09 +00:00
log.Println("Disconnect peer:", DiscReason(msg.Data.Get(0).AsUint()))
2014-01-30 22:48:52 +00: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()
case ethwire.MsgBlockTy:
// Get all blocks and process them
msg.Data = msg.Data
2014-02-10 10:36:49 +00:00
var block *ethchain.Block
2014-01-30 22:48:52 +00:00
for i := msg.Data.Length() - 1; i >= 0; i-- {
2014-02-06 12:27:57 +00:00
// FIXME
2014-02-10 10:36:49 +00:00
block = ethchain.NewBlockFromRlpValue(ethutil.NewValue(msg.Data.Get(i).AsRaw()))
2014-01-30 22:48:52 +00:00
err := p.ethereum.BlockManager.ProcessBlock(block)
if err != nil {
log.Println(err)
}
2014-01-25 16:13:33 +00:00
}
2014-02-10 10:20:42 +00:00
// If we're catching up, try to catch up further.
if p.catchingUp && msg.Data.Length() > 1 {
2014-02-10 10:36:49 +00:00
if ethutil.Config.Debug {
blockInfo := p.ethereum.BlockManager.BlockChain().BlockInfo(block)
log.Printf("Synced to block height #%d\n", blockInfo.Number)
}
2014-02-10 10:20:42 +00:00
p.catchingUp = false
p.CatchupWithPeer()
}
2014-01-30 22:48:52 +00: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
for i := 0; i < msg.Data.Length(); i++ {
2014-02-03 00:12:44 +00:00
p.ethereum.TxPool.QueueTransaction(ethchain.NewTransactionFromData(msg.Data.Get(i).Encode()))
2014-01-23 19:14:01 +00:00
}
2014-01-30 22:48:52 +00:00
case ethwire.MsgGetPeersTy:
// Flag this peer as a 'requested of new peers' this to
// prevent malicious peers being forced.
p.requestedPeerList = true
// Peer asked for list of connected peers
p.pushPeers()
case ethwire.MsgPeersTy:
// Received a list of peers (probably because MsgGetPeersTy was send)
// Only act on message if we actually requested for a peers list
2014-02-01 20:30:54 +00:00
//if p.requestedPeerList {
data := msg.Data
// Create new list of possible peers for the ethereum to process
peers := make([]string, data.Length())
// Parse each possible peer
for i := 0; i < data.Length(); i++ {
peers[i] = unpackAddr(data.Get(i).Get(0), data.Get(i).Get(1).AsUint())
}
2014-01-30 22:48:52 +00:00
2014-02-01 20:30:54 +00:00
// Connect to the list of peers
p.ethereum.ProcessPeerList(peers)
// Mark unrequested again
p.requestedPeerList = false
2014-01-23 19:14:01 +00:00
2014-02-01 20:30:54 +00:00
//}
2014-01-30 22:48:52 +00:00
case ethwire.MsgGetChainTy:
var parent *ethchain.Block
// Length minus one since the very last element in the array is a count
l := msg.Data.Length() - 1
// Ignore empty get chains
2014-02-06 12:27:57 +00:00
if l == 0 {
2014-01-28 14:35:44 +00:00
break
2014-01-27 14:34:50 +00:00
}
2014-01-30 22:48:52 +00:00
// Amount of parents in the canonical chain
2014-02-08 20:02:42 +00:00
//amountOfBlocks := msg.Data.Get(l).AsUint()
amountOfBlocks := uint64(100)
2014-01-30 22:48:52 +00:00
// Check each SHA block hash from the message and determine whether
// the SHA is in the database
for i := 0; i < l; i++ {
if data := msg.Data.Get(i).AsBytes(); p.ethereum.BlockManager.BlockChain().HasBlock(data) {
parent = p.ethereum.BlockManager.BlockChain().GetBlock(data)
break
}
}
// If a parent is found send back a reply
if parent != nil {
chain := p.ethereum.BlockManager.BlockChain().GetChainFromHash(parent.Hash(), amountOfBlocks)
2014-02-08 20:02:42 +00:00
p.QueueMessage(ethwire.NewMessage(ethwire.MsgBlockTy, chain))
2014-01-30 22:48:52 +00:00
} else {
// If no blocks are found we send back a reply with msg not in chain
// and the last hash from get chain
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.AsRaw()}))
}
case ethwire.MsgNotInChainTy:
log.Printf("Not in chain %x\n", msg.Data)
// TODO
2014-01-25 16:13:33 +00:00
2014-01-30 22:48:52 +00:00
// Unofficial but fun nonetheless
case ethwire.MsgTalkTy:
log.Printf("%v says: %s\n", p.conn.RemoteAddr(), msg.Data.AsString())
}
2014-01-23 19:14:01 +00:00
}
}
p.Stop()
}
2014-02-08 20:02:42 +00:00
func packAddr(address, port string) ([]interface{}, uint16) {
2014-01-31 19:01:28 +00: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])
2014-02-08 20:02:42 +00:00
host := []interface{}{byte(a), byte(b), byte(c), byte(d)}
2014-01-31 19:01:28 +00:00
prt, _ := strconv.Atoi(port)
return host, uint16(prt)
}
2014-02-01 20:30:54 +00:00
func unpackAddr(value *ethutil.RlpValue, p uint64) string {
a := strconv.Itoa(int(value.Get(0).AsUint()))
b := strconv.Itoa(int(value.Get(1).AsUint()))
c := strconv.Itoa(int(value.Get(2).AsUint()))
d := strconv.Itoa(int(value.Get(3).AsUint()))
2014-01-31 19:01:28 +00:00
host := strings.Join([]string{a, b, c, d}, ".")
port := strconv.Itoa(int(p))
return net.JoinHostPort(host, port)
}
func (p *Peer) Start() {
2014-01-31 19:01:28 +00:00
peerHost, peerPort, _ := net.SplitHostPort(p.conn.LocalAddr().String())
servHost, servPort, _ := net.SplitHostPort(p.conn.RemoteAddr().String())
2014-01-30 23:56:32 +00:00
2014-01-31 19:01:28 +00:00
if p.inbound {
p.host, p.port = packAddr(peerHost, peerPort)
} else {
p.host, p.port = packAddr(servHost, servPort)
2014-01-30 23:56:32 +00:00
}
err := p.pushHandshake()
if err != nil {
log.Printf("Peer can't send outbound version ack", err)
p.Stop()
return
2014-01-23 19:14:01 +00:00
}
// Run the outbound handler in a new goroutine
go p.HandleOutbound()
// Run the inbound handler in a new goroutine
go p.HandleInbound()
2014-01-30 22:48:52 +00:00
2014-01-23 19:14:01 +00: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 16:13:33 +00:00
p.writeMessage(ethwire.NewMessage(ethwire.MsgDiscTy, ""))
2014-01-23 19:14:01 +00:00
p.conn.Close()
}
}
func (p *Peer) pushHandshake() error {
2014-02-10 10:45:08 +00:00
clientId := fmt.Sprintf("/Ethereum(G) v%s/%s", ethutil.Config.Ver, runtime.GOOS)
2014-01-30 22:48:52 +00:00
msg := ethwire.NewMessage(ethwire.MsgHandshakeTy, []interface{}{
2014-02-10 10:45:08 +00:00
uint32(3), uint32(0), clientId, byte(p.caps), p.port, p.pubkey,
2014-01-30 22:48:52 +00:00
})
2014-01-23 19:14:01 +00:00
p.QueueMessage(msg)
return nil
}
2014-02-02 15:15:39 +00:00
func (p *Peer) peersMessage() *ethwire.Msg {
outPeers := make([]interface{}, len(p.ethereum.InOutPeers()))
2014-01-23 19:14:01 +00:00
// Serialise each peer
for i, peer := range p.ethereum.InOutPeers() {
2014-01-30 23:56:32 +00:00
outPeers[i] = peer.RlpData()
2014-01-23 19:14:01 +00:00
}
2014-02-02 15:15:39 +00:00
// Return the message to the peer with the known list of connected clients
return ethwire.NewMessage(ethwire.MsgPeersTy, outPeers)
}
2014-01-23 19:14:01 +00:00
2014-02-02 15:15:39 +00:00
// Pushes the list of outbound peers to the client when requested
func (p *Peer) pushPeers() {
p.QueueMessage(p.peersMessage())
2014-01-23 19:14:01 +00:00
}
func (p *Peer) handleHandshake(msg *ethwire.Msg) {
c := msg.Data
2014-02-10 00:15:14 +00:00
if c.Get(0).AsUint() != 3 {
2014-02-10 10:20:42 +00:00
log.Println("Invalid peer version. Require protocol v3")
p.Stop()
return
}
2014-02-10 00:09:12 +00:00
// [PROTOCOL_VERSION, NETWORK_ID, CLIENT_ID, CAPS, PORT, PUBKEY]
2014-01-23 19:14:01 +00:00
p.versionKnown = true
2014-01-30 22:48:52 +00:00
var istr string
2014-01-23 19:14:01 +00:00
// If this is an inbound connection send an ack back
if p.inbound {
2014-02-10 00:09:12 +00:00
p.pubkey = c.Get(5).AsBytes()
p.port = uint16(c.Get(4).AsUint())
2014-01-23 19:14:01 +00:00
2014-02-10 00:09:12 +00:00
// Self connect detection
2014-02-09 22:58:59 +00:00
data, _ := ethutil.Config.Db.Get([]byte("KeyRing"))
pubkey := ethutil.NewValueFromBytes(data).Get(2).Bytes()
if bytes.Compare(pubkey, p.pubkey) == 0 {
p.Stop()
return
}
2014-01-30 22:48:52 +00:00
istr = "inbound"
2014-01-28 14:35:44 +00:00
} else {
2014-02-10 00:09:12 +00:00
p.CatchupWithPeer()
2014-01-30 22:48:52 +00:00
istr = "outbound"
2014-01-23 19:14:01 +00:00
}
2014-01-30 22:48:52 +00:00
2014-02-10 00:09:12 +00:00
p.caps = Caps(c.Get(3).AsByte())
2014-01-30 23:56:32 +00:00
2014-01-31 19:01:28 +00:00
log.Printf("peer connect (%s) %v %s [%s]\n", istr, p.conn.RemoteAddr(), c.Get(2).AsString(), p.caps)
}
2014-01-30 23:56:32 +00:00
2014-02-10 00:09:12 +00:00
func (p *Peer) CatchupWithPeer() {
if !p.catchingUp {
p.catchingUp = true
msg := ethwire.NewMessage(ethwire.MsgGetChainTy, []interface{}{p.ethereum.BlockManager.BlockChain().CurrentBlock.Hash(), uint64(50)})
p.QueueMessage(msg)
2014-02-10 10:36:49 +00:00
log.Printf("Requesting blockchain %x...\n", p.ethereum.BlockManager.BlockChain().CurrentBlock.Hash()[:4])
2014-02-10 00:09:12 +00:00
}
}
2014-01-31 19:01:28 +00:00
func (p *Peer) RlpData() []interface{} {
return []interface{}{p.host, p.port, p.pubkey}
2014-01-23 19:14:01 +00:00
}