| 
									
										
										
										
											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/>. | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											2015-01-25 14:50:43 -06:00
										 |  |  | package rpchttp | 
					
						
							| 
									
										
										
										
											2014-10-21 13:24:48 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							|  |  |  | 	"net" | 
					
						
							| 
									
										
										
										
											2015-01-12 23:25:29 -06:00
										 |  |  | 	"net/http" | 
					
						
							| 
									
										
										
										
											2014-10-21 13:24:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-31 12:56:05 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/logger" | 
					
						
							| 
									
										
										
										
											2015-01-25 14:50:43 -06:00
										 |  |  | 	"github.com/ethereum/go-ethereum/rpc" | 
					
						
							| 
									
										
										
										
											2014-10-31 14:30:08 +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 rpchttplogger = logger.NewLogger("RPC-HTTP") | 
					
						
							|  |  |  | var JSON rpc.JsonWrapper | 
					
						
							| 
									
										
										
										
											2014-10-21 13:24:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-28 18:35:49 +01:00
										 |  |  | func NewRpcHttpServer(pipe *xeth.XEth, port int) (*RpcHttpServer, error) { | 
					
						
							| 
									
										
										
										
											2015-02-15 02:26:30 +01:00
										 |  |  | 	sport := fmt.Sprintf("127.0.0.1:%d", port) | 
					
						
							| 
									
										
										
										
											2015-01-25 14:50:43 -06:00
										 |  |  | 	l, err := net.Listen("tcp", sport) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return &RpcHttpServer{ | 
					
						
							|  |  |  | 		listener: l, | 
					
						
							|  |  |  | 		quit:     make(chan bool), | 
					
						
							|  |  |  | 		pipe:     pipe, | 
					
						
							| 
									
										
										
										
											2015-01-27 12:26:30 -06:00
										 |  |  | 		port:     port, | 
					
						
							| 
									
										
										
										
											2015-01-25 14:50:43 -06:00
										 |  |  | 	}, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type RpcHttpServer struct { | 
					
						
							| 
									
										
										
										
											2014-10-21 13:24:48 +02:00
										 |  |  | 	quit     chan bool | 
					
						
							|  |  |  | 	listener net.Listener | 
					
						
							| 
									
										
										
										
											2015-01-28 18:35:49 +01:00
										 |  |  | 	pipe     *xeth.XEth | 
					
						
							| 
									
										
										
										
											2015-01-27 12:26:30 -06:00
										 |  |  | 	port     int | 
					
						
							| 
									
										
										
										
											2014-10-21 13:24:48 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-25 14:50:43 -06:00
										 |  |  | func (s *RpcHttpServer) exitHandler() { | 
					
						
							| 
									
										
										
										
											2014-10-21 13:24:48 +02:00
										 |  |  | out: | 
					
						
							|  |  |  | 	for { | 
					
						
							|  |  |  | 		select { | 
					
						
							|  |  |  | 		case <-s.quit: | 
					
						
							|  |  |  | 			s.listener.Close() | 
					
						
							|  |  |  | 			break out | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-25 14:50:43 -06:00
										 |  |  | 	rpchttplogger.Infoln("Shutdown RPC-HTTP server") | 
					
						
							| 
									
										
										
										
											2014-10-21 13:24:48 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-25 14:50:43 -06:00
										 |  |  | func (s *RpcHttpServer) Stop() { | 
					
						
							| 
									
										
										
										
											2014-10-21 13:24:48 +02:00
										 |  |  | 	close(s.quit) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-25 14:50:43 -06:00
										 |  |  | func (s *RpcHttpServer) Start() { | 
					
						
							| 
									
										
										
										
											2015-01-27 12:26:30 -06:00
										 |  |  | 	rpchttplogger.Infof("Starting RPC-HTTP server on port %d", s.port) | 
					
						
							| 
									
										
										
										
											2014-10-21 13:24:48 +02:00
										 |  |  | 	go s.exitHandler() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-25 14:50:43 -06:00
										 |  |  | 	api := rpc.NewEthereumApi(s.pipe) | 
					
						
							|  |  |  | 	h := s.apiHandler(api) | 
					
						
							| 
									
										
										
										
											2015-01-12 23:25:29 -06:00
										 |  |  | 	http.Handle("/", h) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	err := http.Serve(s.listener, nil) | 
					
						
							| 
									
										
										
										
											2015-01-13 09:37:08 -06:00
										 |  |  | 	// FIX Complains on shutdown due to listner already being closed | 
					
						
							| 
									
										
										
										
											2015-01-12 23:25:29 -06:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2015-01-25 14:50:43 -06:00
										 |  |  | 		rpchttplogger.Errorln("Error on RPC-HTTP interface:", err) | 
					
						
							| 
									
										
										
										
											2014-10-21 13:24:48 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-27 14:34:01 -06:00
										 |  |  | func (s *RpcHttpServer) apiHandler(api *rpc.EthereumApi) http.Handler { | 
					
						
							| 
									
										
										
										
											2015-02-03 17:29:29 -06:00
										 |  |  | 	var jsonrpcver string = "2.0" | 
					
						
							| 
									
										
										
										
											2015-01-12 23:25:29 -06:00
										 |  |  | 	fn := func(w http.ResponseWriter, req *http.Request) { | 
					
						
							| 
									
										
										
										
											2015-01-28 18:14:28 +01:00
										 |  |  | 		w.Header().Set("Access-Control-Allow-Origin", "*") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-04 15:05:47 -08:00
										 |  |  | 		rpchttplogger.DebugDetailln("Handling request") | 
					
						
							| 
									
										
										
										
											2015-01-12 23:25:29 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		reqParsed, reqerr := JSON.ParseRequestBody(req) | 
					
						
							|  |  |  | 		if reqerr != nil { | 
					
						
							| 
									
										
										
										
											2015-02-24 19:22:08 +01:00
										 |  |  | 			jsonerr := &rpc.RpcErrorObject{-32700, "Error: Could not parse request"} | 
					
						
							| 
									
										
										
										
											2015-02-03 17:29:29 -06:00
										 |  |  | 			JSON.Send(w, &rpc.RpcErrorResponse{JsonRpc: jsonrpcver, ID: nil, Error: jsonerr}) | 
					
						
							| 
									
										
										
										
											2015-01-12 23:25:29 -06:00
										 |  |  | 			return | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		var response interface{} | 
					
						
							| 
									
										
										
										
											2015-01-27 14:34:01 -06:00
										 |  |  | 		reserr := api.GetRequestReply(&reqParsed, &response) | 
					
						
							| 
									
										
										
										
											2015-01-12 23:25:29 -06:00
										 |  |  | 		if reserr != nil { | 
					
						
							| 
									
										
										
										
											2015-02-03 17:29:29 -06:00
										 |  |  | 			rpchttplogger.Warnln(reserr) | 
					
						
							|  |  |  | 			jsonerr := &rpc.RpcErrorObject{-32603, reserr.Error()} | 
					
						
							| 
									
										
										
										
											2015-02-11 11:56:29 +01:00
										 |  |  | 			JSON.Send(w, &rpc.RpcErrorResponse{JsonRpc: jsonrpcver, ID: reqParsed.ID, Error: jsonerr}) | 
					
						
							| 
									
										
										
										
											2015-01-12 23:25:29 -06:00
										 |  |  | 			return | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-07 17:03:33 +01:00
										 |  |  | 		rpchttplogger.DebugDetailf("Generated response: %T %s", response, response) | 
					
						
							| 
									
										
										
										
											2015-02-03 17:29:29 -06:00
										 |  |  | 		JSON.Send(w, &rpc.RpcSuccessResponse{JsonRpc: jsonrpcver, ID: reqParsed.ID, Result: response}) | 
					
						
							| 
									
										
										
										
											2015-01-12 23:25:29 -06:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return http.HandlerFunc(fn) | 
					
						
							|  |  |  | } |