| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // Copyright 2015 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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-08 10:41:04 +02:00
										 |  |  | // +build darwin dragonfly freebsd linux nacl netbsd openbsd solaris | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | package comms | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"net" | 
					
						
							| 
									
										
										
										
											2015-06-09 09:48:18 +02:00
										 |  |  | 	"os" | 
					
						
							| 
									
										
										
										
											2015-06-08 10:41:04 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/logger" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/logger/glog" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/rpc/codec" | 
					
						
							| 
									
										
										
										
											2015-06-22 12:47:32 +02:00
										 |  |  | 	"github.com/ethereum/go-ethereum/rpc/shared" | 
					
						
							| 
									
										
										
										
											2015-08-24 12:22:12 +02:00
										 |  |  | 	"github.com/ethereum/go-ethereum/rpc/useragent" | 
					
						
							| 
									
										
										
										
											2015-06-08 10:41:04 +02:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func newIpcClient(cfg IpcConfig, codec codec.Codec) (*ipcClient, error) { | 
					
						
							|  |  |  | 	c, err := net.DialUnix("unix", nil, &net.UnixAddr{cfg.Endpoint, "unix"}) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-24 12:22:12 +02:00
										 |  |  | 	coder := codec.New(c) | 
					
						
							|  |  |  | 	msg := shared.Request{ | 
					
						
							|  |  |  | 		Id:      0, | 
					
						
							|  |  |  | 		Method:  useragent.EnableUserAgentMethod, | 
					
						
							|  |  |  | 		Jsonrpc: shared.JsonRpcVersion, | 
					
						
							|  |  |  | 		Params:  []byte("[]"), | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	coder.WriteResponse(msg) | 
					
						
							|  |  |  | 	coder.Recv() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return &ipcClient{cfg.Endpoint, c, codec, coder}, nil | 
					
						
							| 
									
										
										
										
											2015-06-18 18:23:13 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (self *ipcClient) reconnect() error { | 
					
						
							|  |  |  | 	self.coder.Close() | 
					
						
							|  |  |  | 	c, err := net.DialUnix("unix", nil, &net.UnixAddr{self.endpoint, "unix"}) | 
					
						
							|  |  |  | 	if err == nil { | 
					
						
							|  |  |  | 		self.coder = self.codec.New(c) | 
					
						
							| 
									
										
										
										
											2015-08-24 12:22:12 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		msg := shared.Request{ | 
					
						
							|  |  |  | 			Id:      0, | 
					
						
							|  |  |  | 			Method:  useragent.EnableUserAgentMethod, | 
					
						
							|  |  |  | 			Jsonrpc: shared.JsonRpcVersion, | 
					
						
							|  |  |  | 			Params:  []byte("[]"), | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		self.coder.WriteResponse(msg) | 
					
						
							|  |  |  | 		self.coder.Recv() | 
					
						
							| 
									
										
										
										
											2015-06-18 18:23:13 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return err | 
					
						
							| 
									
										
										
										
											2015-06-08 10:41:04 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-07 09:56:49 +02:00
										 |  |  | func startIpc(cfg IpcConfig, codec codec.Codec, initializer func(conn net.Conn) (shared.EthereumApi, error)) error { | 
					
						
							| 
									
										
										
										
											2015-06-08 10:41:04 +02:00
										 |  |  | 	os.Remove(cfg.Endpoint) // in case it still exists from a previous run | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-19 21:46:01 +02:00
										 |  |  | 	l, err := net.ListenUnix("unix", &net.UnixAddr{Name: cfg.Endpoint, Net: "unix"}) | 
					
						
							| 
									
										
										
										
											2015-06-08 10:41:04 +02:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	os.Chmod(cfg.Endpoint, 0600) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	go func() { | 
					
						
							|  |  |  | 		for { | 
					
						
							| 
									
										
										
										
											2015-08-19 21:46:01 +02:00
										 |  |  | 			conn, err := l.AcceptUnix() | 
					
						
							| 
									
										
										
										
											2015-06-08 10:41:04 +02:00
										 |  |  | 			if err != nil { | 
					
						
							|  |  |  | 				glog.V(logger.Error).Infof("Error accepting ipc connection - %v\n", err) | 
					
						
							|  |  |  | 				continue | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-25 15:54:16 +02:00
										 |  |  | 			id := newIpcConnId() | 
					
						
							|  |  |  | 			glog.V(logger.Debug).Infof("New IPC connection with id %06d started\n", id) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-07 09:56:49 +02:00
										 |  |  | 			api, err := initializer(conn) | 
					
						
							|  |  |  | 			if err != nil { | 
					
						
							|  |  |  | 				glog.V(logger.Error).Infof("Unable to initialize IPC connection - %v\n", err) | 
					
						
							|  |  |  | 				conn.Close() | 
					
						
							|  |  |  | 				continue | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-25 15:54:16 +02:00
										 |  |  | 			go handle(id, conn, api, codec) | 
					
						
							| 
									
										
										
										
											2015-06-08 10:41:04 +02:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		os.Remove(cfg.Endpoint) | 
					
						
							|  |  |  | 	}() | 
					
						
							| 
									
										
										
										
											2015-06-09 09:48:18 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	glog.V(logger.Info).Infof("IPC service started (%s)\n", cfg.Endpoint) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-08 10:41:04 +02:00
										 |  |  | 	return nil | 
					
						
							|  |  |  | } |