| 
									
										
										
										
											2015-01-13 09:13:43 -06:00
										 |  |  | /* | 
					
						
							|  |  |  | 	This file is part of go-ethereum | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	go-ethereum is free software: you can redistribute it and/or modify | 
					
						
							|  |  |  | 	it under the terms of the GNU General Public License as published by | 
					
						
							|  |  |  | 	the Free Software Foundation, either version 3 of the License, or | 
					
						
							|  |  |  | 	(at your option) any later version. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	go-ethereum is distributed in the hope that it will be useful, | 
					
						
							|  |  |  | 	but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  | 	MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  |   GNU General Public License for more details. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  | 	along with go-ethereum.  If not, see <http://www.gnu.org/licenses/>. | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											2014-10-21 13:24:48 +02:00
										 |  |  | package rpc | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"encoding/json" | 
					
						
							|  |  |  | 	"io" | 
					
						
							| 
									
										
										
										
											2015-01-12 23:25:29 -06:00
										 |  |  | 	"net/http" | 
					
						
							| 
									
										
										
										
											2015-02-20 12:59:54 +01:00
										 |  |  | 	"time" | 
					
						
							| 
									
										
										
										
											2015-02-04 15:05:47 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-29 11:39:59 -06:00
										 |  |  | 	"github.com/ethereum/go-ethereum/ethutil" | 
					
						
							| 
									
										
										
										
											2015-02-04 15:05:47 -08:00
										 |  |  | 	"github.com/ethereum/go-ethereum/logger" | 
					
						
							| 
									
										
										
										
											2015-01-29 11:39:59 -06:00
										 |  |  | 	"github.com/ethereum/go-ethereum/state" | 
					
						
							| 
									
										
										
										
											2015-02-20 12:59:54 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/xeth" | 
					
						
							| 
									
										
										
										
											2014-10-21 13:24:48 +02:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-25 14:50:43 -06:00
										 |  |  | var rpclogger = logger.NewLogger("RPC") | 
					
						
							| 
									
										
										
										
											2014-10-21 13:24:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-25 14:50:43 -06:00
										 |  |  | type JsonWrapper struct{} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (self JsonWrapper) Send(writer io.Writer, v interface{}) (n int, err error) { | 
					
						
							| 
									
										
										
										
											2014-10-21 13:24:48 +02:00
										 |  |  | 	var payload []byte | 
					
						
							|  |  |  | 	payload, err = json.Marshal(v) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2015-01-25 14:50:43 -06:00
										 |  |  | 		rpclogger.Fatalln("Error marshalling JSON", err) | 
					
						
							| 
									
										
										
										
											2014-10-21 13:24:48 +02:00
										 |  |  | 		return 0, err | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-02-04 15:05:47 -08:00
										 |  |  | 	rpclogger.DebugDetailf("Sending payload: %s", payload) | 
					
						
							| 
									
										
										
										
											2014-10-21 13:24:48 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return writer.Write(payload) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-25 14:50:43 -06:00
										 |  |  | func (self JsonWrapper) ParseRequestBody(req *http.Request) (RpcRequest, error) { | 
					
						
							| 
									
										
										
										
											2015-01-12 23:25:29 -06:00
										 |  |  | 	var reqParsed RpcRequest | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Convert JSON to native types | 
					
						
							|  |  |  | 	d := json.NewDecoder(req.Body) | 
					
						
							|  |  |  | 	defer req.Body.Close() | 
					
						
							|  |  |  | 	err := d.Decode(&reqParsed) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2015-01-25 14:50:43 -06:00
										 |  |  | 		rpclogger.Errorln("Error decoding JSON: ", err) | 
					
						
							| 
									
										
										
										
											2015-01-12 23:25:29 -06:00
										 |  |  | 		return reqParsed, err | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-02-11 11:56:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-25 14:50:43 -06:00
										 |  |  | 	rpclogger.DebugDetailf("Parsed request: %s", reqParsed) | 
					
						
							| 
									
										
										
										
											2015-01-12 23:25:29 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return reqParsed, nil | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2015-01-29 11:39:59 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | func toHex(b []byte) string { | 
					
						
							|  |  |  | 	return "0x" + ethutil.Bytes2Hex(b) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | func fromHex(s string) []byte { | 
					
						
							|  |  |  | 	if len(s) > 1 { | 
					
						
							|  |  |  | 		if s[0:2] == "0x" { | 
					
						
							|  |  |  | 			s = s[2:] | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return ethutil.Hex2Bytes(s) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type RpcServer interface { | 
					
						
							|  |  |  | 	Start() | 
					
						
							|  |  |  | 	Stop() | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type Log struct { | 
					
						
							|  |  |  | 	Address string   `json:"address"` | 
					
						
							| 
									
										
										
										
											2015-02-24 16:18:27 +01:00
										 |  |  | 	Topic   []string `json:"topic"` | 
					
						
							| 
									
										
										
										
											2015-01-29 11:39:59 -06:00
										 |  |  | 	Data    string   `json:"data"` | 
					
						
							| 
									
										
										
										
											2015-02-22 13:24:26 +01:00
										 |  |  | 	Number  uint64   `json:"number"` | 
					
						
							| 
									
										
										
										
											2015-01-29 11:39:59 -06:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func toLogs(logs state.Logs) (ls []Log) { | 
					
						
							|  |  |  | 	ls = make([]Log, len(logs)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for i, log := range logs { | 
					
						
							|  |  |  | 		var l Log | 
					
						
							| 
									
										
										
										
											2015-02-19 11:51:38 +01:00
										 |  |  | 		l.Topic = make([]string, len(log.Topics())) | 
					
						
							| 
									
										
										
										
											2015-01-29 11:39:59 -06:00
										 |  |  | 		l.Address = toHex(log.Address()) | 
					
						
							|  |  |  | 		l.Data = toHex(log.Data()) | 
					
						
							| 
									
										
										
										
											2015-02-22 13:24:26 +01:00
										 |  |  | 		l.Number = log.Number() | 
					
						
							| 
									
										
										
										
											2015-01-29 11:39:59 -06:00
										 |  |  | 		for j, topic := range log.Topics() { | 
					
						
							| 
									
										
										
										
											2015-02-19 11:51:38 +01:00
										 |  |  | 			l.Topic[j] = toHex(topic) | 
					
						
							| 
									
										
										
										
											2015-01-29 11:39:59 -06:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		ls[i] = l | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2015-02-20 12:59:54 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | type whisperFilter struct { | 
					
						
							|  |  |  | 	messages []xeth.WhisperMessage | 
					
						
							|  |  |  | 	timeout  time.Time | 
					
						
							| 
									
										
										
										
											2015-02-23 15:43:41 +01:00
										 |  |  | 	id       int | 
					
						
							| 
									
										
										
										
											2015-02-20 12:59:54 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (w *whisperFilter) add(msgs ...xeth.WhisperMessage) { | 
					
						
							|  |  |  | 	w.messages = append(w.messages, msgs...) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | func (w *whisperFilter) get() []xeth.WhisperMessage { | 
					
						
							|  |  |  | 	w.timeout = time.Now() | 
					
						
							|  |  |  | 	tmp := w.messages | 
					
						
							|  |  |  | 	w.messages = nil | 
					
						
							|  |  |  | 	return tmp | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type logFilter struct { | 
					
						
							|  |  |  | 	logs    state.Logs | 
					
						
							|  |  |  | 	timeout time.Time | 
					
						
							| 
									
										
										
										
											2015-02-23 15:43:41 +01:00
										 |  |  | 	id      int | 
					
						
							| 
									
										
										
										
											2015-02-20 12:59:54 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (l *logFilter) add(logs ...state.Log) { | 
					
						
							|  |  |  | 	l.logs = append(l.logs, logs...) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (l *logFilter) get() state.Logs { | 
					
						
							|  |  |  | 	l.timeout = time.Now() | 
					
						
							|  |  |  | 	tmp := l.logs | 
					
						
							|  |  |  | 	l.logs = nil | 
					
						
							|  |  |  | 	return tmp | 
					
						
							|  |  |  | } |