| 
									
										
										
										
											2014-06-16 00:51:21 +02:00
										 |  |  | // Package ethwire provides low level access to the Ethereum network and allows | 
					
						
							|  |  |  | // you to broadcast data over the network. | 
					
						
							| 
									
										
										
										
											2014-02-14 23:56:09 +01:00
										 |  |  | package ethwire | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"bytes" | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							|  |  |  | 	"net" | 
					
						
							|  |  |  | 	"time" | 
					
						
							| 
									
										
										
										
											2014-08-11 16:23:38 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/ethereum/eth-go/ethutil" | 
					
						
							| 
									
										
										
										
											2014-02-14 23:56:09 +01:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-16 00:51:21 +02:00
										 |  |  | // Connection interface describing the methods required to implement the wire protocol. | 
					
						
							|  |  |  | type Conn interface { | 
					
						
							|  |  |  | 	Write(typ MsgType, v ...interface{}) error | 
					
						
							|  |  |  | 	Read() *Msg | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2014-02-14 23:56:09 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-16 00:51:21 +02:00
										 |  |  | // The magic token which should be the first 4 bytes of every message and can be used as separator between messages. | 
					
						
							| 
									
										
										
										
											2014-02-14 23:56:09 +01:00
										 |  |  | var MagicToken = []byte{34, 64, 8, 145} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type MsgType byte | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const ( | 
					
						
							| 
									
										
										
										
											2014-02-18 17:24:44 -05:00
										 |  |  | 	// 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-09-10 11:22:19 +02:00
										 |  |  | 	MsgHandshakeTy = 0x00 | 
					
						
							|  |  |  | 	MsgDiscTy      = 0x01 | 
					
						
							|  |  |  | 	MsgPingTy      = 0x02 | 
					
						
							|  |  |  | 	MsgPongTy      = 0x03 | 
					
						
							|  |  |  | 	MsgGetPeersTy  = 0x04 | 
					
						
							|  |  |  | 	MsgPeersTy     = 0x05 | 
					
						
							| 
									
										
										
										
											2014-02-14 23:56:09 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-10 11:22:19 +02:00
										 |  |  | 	MsgStatusTy         = 0x10 | 
					
						
							|  |  |  | 	MsgGetTxsTy         = 0x11 | 
					
						
							|  |  |  | 	MsgTxTy             = 0x12 | 
					
						
							|  |  |  | 	MsgGetBlockHashesTy = 0x13 | 
					
						
							|  |  |  | 	MsgBlockHashesTy    = 0x14 | 
					
						
							|  |  |  | 	MsgGetBlocksTy      = 0x15 | 
					
						
							|  |  |  | 	MsgBlockTy          = 0x16 | 
					
						
							| 
									
										
										
										
											2014-02-14 23:56:09 +01:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var msgTypeToString = map[MsgType]string{ | 
					
						
							| 
									
										
										
										
											2014-08-21 14:47:58 +02:00
										 |  |  | 	MsgHandshakeTy:      "Handshake", | 
					
						
							|  |  |  | 	MsgDiscTy:           "Disconnect", | 
					
						
							|  |  |  | 	MsgPingTy:           "Ping", | 
					
						
							|  |  |  | 	MsgPongTy:           "Pong", | 
					
						
							|  |  |  | 	MsgGetPeersTy:       "Get peers", | 
					
						
							| 
									
										
										
										
											2014-09-10 11:39:11 +02:00
										 |  |  | 	MsgStatusTy:         "Status", | 
					
						
							| 
									
										
										
										
											2014-08-21 14:47:58 +02:00
										 |  |  | 	MsgPeersTy:          "Peers", | 
					
						
							|  |  |  | 	MsgTxTy:             "Transactions", | 
					
						
							|  |  |  | 	MsgBlockTy:          "Blocks", | 
					
						
							|  |  |  | 	MsgGetTxsTy:         "Get Txs", | 
					
						
							|  |  |  | 	MsgGetBlockHashesTy: "Get block hashes", | 
					
						
							|  |  |  | 	MsgBlockHashesTy:    "Block hashes", | 
					
						
							|  |  |  | 	MsgGetBlocksTy:      "Get blocks", | 
					
						
							| 
									
										
										
										
											2014-02-14 23:56:09 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (mt MsgType) String() string { | 
					
						
							|  |  |  | 	return msgTypeToString[mt] | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type Msg struct { | 
					
						
							|  |  |  | 	Type MsgType // Specifies how the encoded data should be interpreted | 
					
						
							|  |  |  | 	//Data []byte | 
					
						
							|  |  |  | 	Data *ethutil.Value | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func NewMessage(msgType MsgType, data interface{}) *Msg { | 
					
						
							|  |  |  | 	return &Msg{ | 
					
						
							|  |  |  | 		Type: msgType, | 
					
						
							|  |  |  | 		Data: ethutil.NewValue(data), | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-16 00:51:21 +02:00
										 |  |  | type Messages []*Msg | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // The basic message reader waits for data on the given connection, decoding | 
					
						
							|  |  |  | // and doing a few sanity checks such as if there's a data type and | 
					
						
							|  |  |  | // unmarhals the given data | 
					
						
							| 
									
										
										
										
											2014-09-16 16:06:38 +02:00
										 |  |  | func ReadMessages(conn net.Conn) (msgs []*Msg, err error) { | 
					
						
							| 
									
										
										
										
											2014-06-16 00:51:21 +02:00
										 |  |  | 	// The recovering function in case anything goes horribly wrong | 
					
						
							|  |  |  | 	defer func() { | 
					
						
							|  |  |  | 		if r := recover(); r != nil { | 
					
						
							|  |  |  | 			err = fmt.Errorf("ethwire.ReadMessage error: %v", r) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	}() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-16 16:06:38 +02:00
										 |  |  | 	var ( | 
					
						
							|  |  |  | 		buff      []byte | 
					
						
							|  |  |  | 		messages  [][]byte | 
					
						
							|  |  |  | 		msgLength int | 
					
						
							|  |  |  | 	) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-16 00:51:21 +02:00
										 |  |  | 	for { | 
					
						
							|  |  |  | 		// Give buffering some time | 
					
						
							| 
									
										
										
										
											2014-09-17 15:57:32 +02:00
										 |  |  | 		conn.SetReadDeadline(time.Now().Add(5 * time.Millisecond)) | 
					
						
							| 
									
										
										
										
											2014-06-16 00:51:21 +02:00
										 |  |  | 		// Create a new temporarily buffer | 
					
						
							|  |  |  | 		b := make([]byte, 1440) | 
					
						
							| 
									
										
										
										
											2014-09-16 16:06:38 +02:00
										 |  |  | 		n, _ := conn.Read(b) | 
					
						
							| 
									
										
										
										
											2014-06-16 00:51:21 +02:00
										 |  |  | 		if err != nil && n == 0 { | 
					
						
							|  |  |  | 			if err.Error() != "EOF" { | 
					
						
							|  |  |  | 				fmt.Println("err now", err) | 
					
						
							| 
									
										
										
										
											2014-09-16 16:06:38 +02:00
										 |  |  | 				return nil, err | 
					
						
							| 
									
										
										
										
											2014-06-16 00:51:21 +02:00
										 |  |  | 			} else { | 
					
						
							|  |  |  | 				break | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-16 16:19:48 +02:00
										 |  |  | 		if n == 0 && len(buff) == 0 { | 
					
						
							| 
									
										
										
										
											2014-09-17 15:57:32 +02:00
										 |  |  | 			// If there's nothing on the wire wait for a bit | 
					
						
							| 
									
										
										
										
											2014-09-22 18:15:10 +02:00
										 |  |  | 			time.Sleep(200 * time.Millisecond) | 
					
						
							| 
									
										
										
										
											2014-09-17 15:57:32 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-16 16:06:38 +02:00
										 |  |  | 			continue | 
					
						
							| 
									
										
										
										
											2014-06-16 00:51:21 +02:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2014-02-14 23:56:09 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-16 16:19:48 +02:00
										 |  |  | 		buff = append(buff, b[:n]...) | 
					
						
							| 
									
										
										
										
											2014-09-16 16:06:38 +02:00
										 |  |  | 		if msgLength == 0 { | 
					
						
							|  |  |  | 			// Check if the received 4 first bytes are the magic token | 
					
						
							| 
									
										
										
										
											2014-09-16 16:19:48 +02:00
										 |  |  | 			if bytes.Compare(MagicToken, buff[:4]) != 0 { | 
					
						
							|  |  |  | 				return nil, fmt.Errorf("MagicToken mismatch. Received %v", buff[:4]) | 
					
						
							| 
									
										
										
										
											2014-09-16 16:06:38 +02:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2014-02-14 23:56:09 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-16 16:06:38 +02:00
										 |  |  | 			// Read the length of the message | 
					
						
							| 
									
										
										
										
											2014-09-16 16:19:48 +02:00
										 |  |  | 			msgLength = int(ethutil.BytesToNumber(buff[4:8])) | 
					
						
							| 
									
										
										
										
											2014-02-14 23:56:09 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-16 16:19:48 +02:00
										 |  |  | 			// Remove the token and length | 
					
						
							|  |  |  | 			buff = buff[8:] | 
					
						
							| 
									
										
										
										
											2014-09-16 16:06:38 +02:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2014-02-14 23:56:09 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-16 16:06:38 +02:00
										 |  |  | 		if len(buff) >= msgLength { | 
					
						
							|  |  |  | 			messages = append(messages, buff[:msgLength]) | 
					
						
							|  |  |  | 			buff = buff[msgLength:] | 
					
						
							|  |  |  | 			msgLength = 0 | 
					
						
							| 
									
										
										
										
											2014-02-14 23:56:09 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-16 16:06:38 +02:00
										 |  |  | 			if len(buff) == 0 { | 
					
						
							| 
									
										
										
										
											2014-02-14 23:56:09 +01:00
										 |  |  | 				break | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-16 16:06:38 +02:00
										 |  |  | 	for _, m := range messages { | 
					
						
							|  |  |  | 		decoder := ethutil.NewValueFromBytes(m) | 
					
						
							|  |  |  | 		// Type of message | 
					
						
							|  |  |  | 		t := decoder.Get(0).Uint() | 
					
						
							|  |  |  | 		// Actual data | 
					
						
							|  |  |  | 		d := decoder.SliceFrom(1) | 
					
						
							| 
									
										
										
										
											2014-02-14 23:56:09 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-16 16:06:38 +02:00
										 |  |  | 		msgs = append(msgs, &Msg{Type: MsgType(t), Data: d}) | 
					
						
							| 
									
										
										
										
											2014-02-14 23:56:09 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // The basic message writer takes care of writing data over the given | 
					
						
							|  |  |  | // connection and does some basic error checking | 
					
						
							|  |  |  | func WriteMessage(conn net.Conn, msg *Msg) error { | 
					
						
							|  |  |  | 	var pack []byte | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Encode the type and the (RLP encoded) data for sending over the wire | 
					
						
							|  |  |  | 	encoded := ethutil.NewValue(append([]interface{}{byte(msg.Type)}, msg.Data.Slice()...)).Encode() | 
					
						
							|  |  |  | 	payloadLength := ethutil.NumberToBytes(uint32(len(encoded)), 32) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Write magic token and payload length (first 8 bytes) | 
					
						
							|  |  |  | 	pack = append(MagicToken, payloadLength...) | 
					
						
							|  |  |  | 	pack = append(pack, encoded...) | 
					
						
							|  |  |  | 	//fmt.Printf("payload %v (%v) %q\n", msg.Type, conn.RemoteAddr(), encoded) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Write to the connection | 
					
						
							|  |  |  | 	_, err := conn.Write(pack) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } |