| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // Copyright 2014 The go-ethereum Authors | 
					
						
							| 
									
										
										
										
											2015-07-22 18:48:40 +02:00
										 |  |  | // This file is part of the go-ethereum library. | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // | 
					
						
							| 
									
										
										
										
											2015-07-23 18:35:11 +02:00
										 |  |  | // The go-ethereum library is free software: you can redistribute it and/or modify | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // it under the terms of the GNU Lesser General Public License as published by | 
					
						
							|  |  |  | // the Free Software Foundation, either version 3 of the License, or | 
					
						
							|  |  |  | // (at your option) any later version. | 
					
						
							|  |  |  | // | 
					
						
							| 
									
										
										
										
											2015-07-22 18:48:40 +02:00
										 |  |  | // The go-ethereum library is distributed in the hope that it will be useful, | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							| 
									
										
										
										
											2015-07-22 18:48:40 +02:00
										 |  |  | // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // GNU Lesser General Public License for more details. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // You should have received a copy of the GNU Lesser General Public License | 
					
						
							| 
									
										
										
										
											2015-07-22 18:48:40 +02:00
										 |  |  | // along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>. | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-05 21:14:55 +00:00
										 |  |  | package eth | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2015-07-02 19:55:18 +03:00
										 |  |  | 	"fmt" | 
					
						
							|  |  |  | 	"io" | 
					
						
							| 
									
										
										
										
											2014-12-05 21:14:55 +00:00
										 |  |  | 	"math/big" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-18 13:00:01 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/common" | 
					
						
							| 
									
										
										
										
											2014-12-05 21:14:55 +00:00
										 |  |  | 	"github.com/ethereum/go-ethereum/core/types" | 
					
						
							| 
									
										
										
										
											2015-07-02 19:55:18 +03:00
										 |  |  | 	"github.com/ethereum/go-ethereum/rlp" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Constants to match up protocol versions and messages | 
					
						
							|  |  |  | const ( | 
					
						
							|  |  |  | 	eth61 = 61 | 
					
						
							|  |  |  | 	eth62 = 62 | 
					
						
							|  |  |  | 	eth63 = 63 | 
					
						
							|  |  |  | 	eth64 = 64 | 
					
						
							| 
									
										
										
										
											2014-12-05 21:14:55 +00:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-26 16:54:27 +03:00
										 |  |  | // Supported versions of the eth protocol (first is primary). | 
					
						
							| 
									
										
										
										
											2015-08-14 17:48:26 +03:00
										 |  |  | var ProtocolVersions = []uint{eth64, eth63, eth62, eth61} | 
					
						
							| 
									
										
										
										
											2015-06-26 16:54:27 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | // Number of implemented message corresponding to different protocol versions. | 
					
						
							| 
									
										
										
										
											2015-08-14 17:48:26 +03:00
										 |  |  | var ProtocolLengths = []uint64{15, 12, 8, 9} | 
					
						
							| 
									
										
										
										
											2015-06-26 16:54:27 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-15 12:34:06 +01:00
										 |  |  | const ( | 
					
						
							| 
									
										
										
										
											2015-07-22 13:41:31 +02:00
										 |  |  | 	NetworkId          = 1 | 
					
						
							| 
									
										
										
										
											2015-06-26 16:54:27 +03:00
										 |  |  | 	ProtocolMaxMsgSize = 10 * 1024 * 1024 // Maximum cap on the size of a protocol message | 
					
						
							| 
									
										
										
										
											2014-12-15 12:34:06 +01:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // eth protocol message codes | 
					
						
							|  |  |  | const ( | 
					
						
							| 
									
										
										
										
											2015-08-14 17:48:26 +03:00
										 |  |  | 	// Protocol messages belonging to eth/61 | 
					
						
							|  |  |  | 	StatusMsg                   = 0x00 | 
					
						
							|  |  |  | 	NewBlockHashesMsg           = 0x01 | 
					
						
							|  |  |  | 	TxMsg                       = 0x02 | 
					
						
							|  |  |  | 	GetBlockHashesMsg           = 0x03 | 
					
						
							|  |  |  | 	BlockHashesMsg              = 0x04 | 
					
						
							|  |  |  | 	GetBlocksMsg                = 0x05 | 
					
						
							|  |  |  | 	BlocksMsg                   = 0x06 | 
					
						
							|  |  |  | 	NewBlockMsg                 = 0x07 | 
					
						
							| 
									
										
										
										
											2015-07-02 19:55:18 +03:00
										 |  |  | 	GetBlockHashesFromNumberMsg = 0x08 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Protocol messages belonging to eth/62 (new protocol from scratch) | 
					
						
							|  |  |  | 	// StatusMsg          = 0x00 (uncomment after eth/61 deprecation) | 
					
						
							|  |  |  | 	// NewBlockHashesMsg  = 0x01 (uncomment after eth/61 deprecation) | 
					
						
							|  |  |  | 	// TxMsg              = 0x02 (uncomment after eth/61 deprecation) | 
					
						
							|  |  |  | 	GetBlockHeadersMsg = 0x03 | 
					
						
							|  |  |  | 	BlockHeadersMsg    = 0x04 | 
					
						
							|  |  |  | 	GetBlockBodiesMsg  = 0x05 | 
					
						
							|  |  |  | 	BlockBodiesMsg     = 0x06 | 
					
						
							|  |  |  | 	// 	NewBlockMsg       = 0x07 (uncomment after eth/61 deprecation) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Protocol messages belonging to eth/63 | 
					
						
							|  |  |  | 	GetNodeDataMsg = 0x0d | 
					
						
							|  |  |  | 	NodeDataMsg    = 0x0e | 
					
						
							|  |  |  | 	GetReceiptsMsg = 0x0f | 
					
						
							|  |  |  | 	ReceiptsMsg    = 0x10 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Protocol messages belonging to eth/64 | 
					
						
							|  |  |  | 	GetAcctProofMsg     = 0x11 | 
					
						
							|  |  |  | 	GetStorageDataProof = 0x12 | 
					
						
							|  |  |  | 	Proof               = 0x13 | 
					
						
							| 
									
										
										
										
											2014-12-15 12:34:06 +01:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-18 01:11:09 +02:00
										 |  |  | type errCode int | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-25 20:06:59 +07:00
										 |  |  | const ( | 
					
						
							|  |  |  | 	ErrMsgTooLarge = iota | 
					
						
							|  |  |  | 	ErrDecode | 
					
						
							|  |  |  | 	ErrInvalidMsgCode | 
					
						
							|  |  |  | 	ErrProtocolVersionMismatch | 
					
						
							|  |  |  | 	ErrNetworkIdMismatch | 
					
						
							|  |  |  | 	ErrGenesisBlockMismatch | 
					
						
							|  |  |  | 	ErrNoStatusMsg | 
					
						
							|  |  |  | 	ErrExtraStatusMsg | 
					
						
							| 
									
										
										
										
											2015-03-19 22:46:54 +00:00
										 |  |  | 	ErrSuspendedPeer | 
					
						
							| 
									
										
										
										
											2015-02-25 20:06:59 +07:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-18 01:11:09 +02:00
										 |  |  | func (e errCode) String() string { | 
					
						
							|  |  |  | 	return errorToString[int(e)] | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // XXX change once legacy code is out | 
					
						
							| 
									
										
										
										
											2015-02-25 20:06:59 +07:00
										 |  |  | var errorToString = map[int]string{ | 
					
						
							|  |  |  | 	ErrMsgTooLarge:             "Message too long", | 
					
						
							|  |  |  | 	ErrDecode:                  "Invalid message", | 
					
						
							|  |  |  | 	ErrInvalidMsgCode:          "Invalid message code", | 
					
						
							|  |  |  | 	ErrProtocolVersionMismatch: "Protocol version mismatch", | 
					
						
							|  |  |  | 	ErrNetworkIdMismatch:       "NetworkId mismatch", | 
					
						
							|  |  |  | 	ErrGenesisBlockMismatch:    "Genesis block mismatch", | 
					
						
							|  |  |  | 	ErrNoStatusMsg:             "No status message", | 
					
						
							|  |  |  | 	ErrExtraStatusMsg:          "Extra status message", | 
					
						
							| 
									
										
										
										
											2015-03-19 22:46:54 +00:00
										 |  |  | 	ErrSuspendedPeer:           "Suspended peer", | 
					
						
							| 
									
										
										
										
											2015-02-25 20:06:59 +07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-14 18:04:50 +00:00
										 |  |  | type txPool interface { | 
					
						
							| 
									
										
										
										
											2015-06-09 17:03:07 +02:00
										 |  |  | 	// AddTransactions should add the given transactions to the pool. | 
					
						
							| 
									
										
										
										
											2014-12-09 23:55:50 +00:00
										 |  |  | 	AddTransactions([]*types.Transaction) | 
					
						
							| 
									
										
										
										
											2015-06-09 17:03:07 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// GetTransactions should return pending transactions. | 
					
						
							|  |  |  | 	// The slice should be modifiable by the caller. | 
					
						
							| 
									
										
										
										
											2015-02-04 17:28:54 -08:00
										 |  |  | 	GetTransactions() types.Transactions | 
					
						
							| 
									
										
										
										
											2014-12-14 18:04:50 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type chainManager interface { | 
					
						
							| 
									
										
										
										
											2015-03-18 13:00:01 +01:00
										 |  |  | 	GetBlockHashesFromHash(hash common.Hash, amount uint64) (hashes []common.Hash) | 
					
						
							|  |  |  | 	GetBlock(hash common.Hash) (block *types.Block) | 
					
						
							|  |  |  | 	Status() (td *big.Int, currentBlock common.Hash, genesisBlock common.Hash) | 
					
						
							| 
									
										
										
										
											2014-12-05 21:14:55 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-29 17:32:14 +03:00
										 |  |  | // statusData is the network packet for the status message. | 
					
						
							|  |  |  | type statusData struct { | 
					
						
							|  |  |  | 	ProtocolVersion uint32 | 
					
						
							|  |  |  | 	NetworkId       uint32 | 
					
						
							|  |  |  | 	TD              *big.Int | 
					
						
							|  |  |  | 	CurrentBlock    common.Hash | 
					
						
							|  |  |  | 	GenesisBlock    common.Hash | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-02 19:55:18 +03:00
										 |  |  | // newBlockHashesData is the network packet for the block announcements. | 
					
						
							|  |  |  | type newBlockHashesData []struct { | 
					
						
							|  |  |  | 	Hash   common.Hash // Hash of one particular block being announced | 
					
						
							|  |  |  | 	Number uint64      // Number of one particular block being announced | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-02 14:13:46 +03:00
										 |  |  | // getBlockHashesData is the network packet for the hash based hash retrieval. | 
					
						
							| 
									
										
										
										
											2015-06-29 17:32:14 +03:00
										 |  |  | type getBlockHashesData struct { | 
					
						
							|  |  |  | 	Hash   common.Hash | 
					
						
							|  |  |  | 	Amount uint64 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-02 14:13:46 +03:00
										 |  |  | // getBlockHashesFromNumberData is the network packet for the number based hash | 
					
						
							|  |  |  | // retrieval. | 
					
						
							| 
									
										
										
										
											2015-06-29 17:32:14 +03:00
										 |  |  | type getBlockHashesFromNumberData struct { | 
					
						
							|  |  |  | 	Number uint64 | 
					
						
							|  |  |  | 	Amount uint64 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-02 19:55:18 +03:00
										 |  |  | // getBlockHeadersData represents a block header query. | 
					
						
							|  |  |  | type getBlockHeadersData struct { | 
					
						
							|  |  |  | 	Origin  hashOrNumber // Block from which to retrieve headers | 
					
						
							|  |  |  | 	Amount  uint64       // Maximum number of headers to retrieve | 
					
						
							|  |  |  | 	Skip    uint64       // Blocks to skip between consecutive headers | 
					
						
							|  |  |  | 	Reverse bool         // Query direction (false = rising towards latest, true = falling towards genesis) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // hashOrNumber is a combined field for specifying an origin block. | 
					
						
							|  |  |  | type hashOrNumber struct { | 
					
						
							|  |  |  | 	Hash   common.Hash // Block hash from which to retrieve headers (excludes Number) | 
					
						
							|  |  |  | 	Number uint64      // Block hash from which to retrieve headers (excludes Hash) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // EncodeRLP is a specialized encoder for hashOrNumber to encode only one of the | 
					
						
							|  |  |  | // two contained union fields. | 
					
						
							|  |  |  | func (hn *hashOrNumber) EncodeRLP(w io.Writer) error { | 
					
						
							|  |  |  | 	if hn.Hash == (common.Hash{}) { | 
					
						
							|  |  |  | 		return rlp.Encode(w, hn.Number) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if hn.Number != 0 { | 
					
						
							|  |  |  | 		return fmt.Errorf("both origin hash (%x) and number (%d) provided", hn.Hash, hn.Number) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return rlp.Encode(w, hn.Hash) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // DecodeRLP is a specialized decoder for hashOrNumber to decode the contents | 
					
						
							|  |  |  | // into either a block hash or a block number. | 
					
						
							|  |  |  | func (hn *hashOrNumber) DecodeRLP(s *rlp.Stream) error { | 
					
						
							|  |  |  | 	_, size, _ := s.Kind() | 
					
						
							|  |  |  | 	origin, err := s.Raw() | 
					
						
							|  |  |  | 	if err == nil { | 
					
						
							|  |  |  | 		switch { | 
					
						
							|  |  |  | 		case size == 32: | 
					
						
							|  |  |  | 			err = rlp.DecodeBytes(origin, &hn.Hash) | 
					
						
							|  |  |  | 		case size <= 8: | 
					
						
							|  |  |  | 			err = rlp.DecodeBytes(origin, &hn.Number) | 
					
						
							|  |  |  | 		default: | 
					
						
							|  |  |  | 			err = fmt.Errorf("invalid input size %d for origin", size) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return err | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-29 17:32:14 +03:00
										 |  |  | // newBlockData is the network packet for the block propagation message. | 
					
						
							|  |  |  | type newBlockData struct { | 
					
						
							| 
									
										
										
										
											2014-12-05 21:14:55 +00:00
										 |  |  | 	Block *types.Block | 
					
						
							|  |  |  | 	TD    *big.Int | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2015-07-02 14:13:46 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-02 19:55:18 +03:00
										 |  |  | // blockBody represents the data content of a single block. | 
					
						
							|  |  |  | type blockBody struct { | 
					
						
							|  |  |  | 	Transactions []*types.Transaction // Transactions contained within a block | 
					
						
							|  |  |  | 	Uncles       []*types.Header      // Uncles contained within a block | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // blockBodiesData is the network packet for block content distribution. | 
					
						
							|  |  |  | type blockBodiesData []*blockBody | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-02 14:13:46 +03:00
										 |  |  | // nodeDataData is the network response packet for a node data retrieval. | 
					
						
							|  |  |  | type nodeDataData []struct { | 
					
						
							|  |  |  | 	Value []byte | 
					
						
							|  |  |  | } |