| 
									
										
										
										
											2015-11-05 23:57:57 +02:00
										 |  |  | // Copyright 2015 The go-ethereum Authors | 
					
						
							|  |  |  | // This file is part of the go-ethereum library. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // The go-ethereum library is free software: you can redistribute it and/or modify | 
					
						
							|  |  |  | // 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. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // The go-ethereum library 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 Lesser General Public License for more details. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // You should have received a copy of the GNU Lesser General Public License | 
					
						
							|  |  |  | // along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | package node | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"crypto/ecdsa" | 
					
						
							|  |  |  | 	"encoding/json" | 
					
						
							| 
									
										
										
										
											2016-02-05 13:45:36 +02:00
										 |  |  | 	"fmt" | 
					
						
							| 
									
										
										
										
											2015-11-05 23:57:57 +02:00
										 |  |  | 	"io/ioutil" | 
					
						
							|  |  |  | 	"net" | 
					
						
							|  |  |  | 	"os" | 
					
						
							|  |  |  | 	"path/filepath" | 
					
						
							| 
									
										
										
										
											2016-02-02 19:06:43 +02:00
										 |  |  | 	"runtime" | 
					
						
							|  |  |  | 	"strings" | 
					
						
							| 
									
										
										
										
											2015-11-05 23:57:57 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-02 19:06:43 +02:00
										 |  |  | 	"github.com/ethereum/go-ethereum/common" | 
					
						
							| 
									
										
										
										
											2015-11-05 23:57:57 +02:00
										 |  |  | 	"github.com/ethereum/go-ethereum/crypto" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/logger" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/logger/glog" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/p2p/discover" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/p2p/nat" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var ( | 
					
						
							|  |  |  | 	datadirPrivateKey   = "nodekey"            // Path within the datadir to the node's private key | 
					
						
							|  |  |  | 	datadirStaticNodes  = "static-nodes.json"  // Path within the datadir to the static node list | 
					
						
							|  |  |  | 	datadirTrustedNodes = "trusted-nodes.json" // Path within the datadir to the trusted node list | 
					
						
							|  |  |  | 	datadirNodeDatabase = "nodes"              // Path within the datadir to store the node infos | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Config represents a small collection of configuration values to fine tune the | 
					
						
							|  |  |  | // P2P network layer of a protocol stack. These values can be further extended by | 
					
						
							|  |  |  | // all registered services. | 
					
						
							|  |  |  | type Config struct { | 
					
						
							|  |  |  | 	// DataDir is the file system folder the node should use for any data storage | 
					
						
							|  |  |  | 	// requirements. The configured data directory will not be directly shared with | 
					
						
							|  |  |  | 	// registered services, instead those can use utility methods to create/access | 
					
						
							|  |  |  | 	// databases or flat files. This enables ephemeral nodes which can fully reside | 
					
						
							|  |  |  | 	// in memory. | 
					
						
							|  |  |  | 	DataDir string | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-09 13:24:42 +02:00
										 |  |  | 	// IPCPath is the requested location to place the IPC endpoint. If the path is | 
					
						
							| 
									
										
										
										
											2016-02-02 19:06:43 +02:00
										 |  |  | 	// a simple file name, it is placed inside the data directory (or on the root | 
					
						
							|  |  |  | 	// pipe path on Windows), whereas if it's a resolvable path name (absolute or | 
					
						
							|  |  |  | 	// relative), then that specific path is enforced. An empty path disables IPC. | 
					
						
							| 
									
										
										
										
											2016-02-09 13:24:42 +02:00
										 |  |  | 	IPCPath string | 
					
						
							| 
									
										
										
										
											2016-02-02 19:06:43 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-05 23:57:57 +02:00
										 |  |  | 	// This field should be a valid secp256k1 private key that will be used for both | 
					
						
							|  |  |  | 	// remote peer identification as well as network traffic encryption. If no key | 
					
						
							|  |  |  | 	// is configured, the preset one is loaded from the data dir, generating it if | 
					
						
							|  |  |  | 	// needed. | 
					
						
							|  |  |  | 	PrivateKey *ecdsa.PrivateKey | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Name sets the node name of this server. Use common.MakeName to create a name | 
					
						
							|  |  |  | 	// that follows existing conventions. | 
					
						
							|  |  |  | 	Name string | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// NoDiscovery specifies whether the peer discovery mechanism should be started | 
					
						
							|  |  |  | 	// or not. Disabling is usually useful for protocol debugging (manual topology). | 
					
						
							|  |  |  | 	NoDiscovery bool | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Bootstrap nodes used to establish connectivity with the rest of the network. | 
					
						
							|  |  |  | 	BootstrapNodes []*discover.Node | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Network interface address on which the node should listen for inbound peers. | 
					
						
							|  |  |  | 	ListenAddr string | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// If set to a non-nil value, the given NAT port mapper is used to make the | 
					
						
							|  |  |  | 	// listening port available to the Internet. | 
					
						
							|  |  |  | 	NAT nat.Interface | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// If Dialer is set to a non-nil value, the given Dialer is used to dial outbound | 
					
						
							|  |  |  | 	// peer connections. | 
					
						
							|  |  |  | 	Dialer *net.Dialer | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// If NoDial is true, the node will not dial any peers. | 
					
						
							|  |  |  | 	NoDial bool | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// MaxPeers is the maximum number of peers that can be connected. If this is | 
					
						
							|  |  |  | 	// set to zero, then only the configured static and trusted peers can connect. | 
					
						
							|  |  |  | 	MaxPeers int | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// MaxPendingPeers is the maximum number of peers that can be pending in the | 
					
						
							|  |  |  | 	// handshake phase, counted separately for inbound and outbound connections. | 
					
						
							|  |  |  | 	// Zero defaults to preset values. | 
					
						
							|  |  |  | 	MaxPendingPeers int | 
					
						
							| 
									
										
										
										
											2016-02-05 13:45:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-09 13:24:42 +02:00
										 |  |  | 	// HTTPHost is the host interface on which to start the HTTP RPC server. If this | 
					
						
							| 
									
										
										
										
											2016-02-05 13:45:36 +02:00
										 |  |  | 	// field is empty, no HTTP API endpoint will be started. | 
					
						
							| 
									
										
										
										
											2016-02-09 13:24:42 +02:00
										 |  |  | 	HTTPHost string | 
					
						
							| 
									
										
										
										
											2016-02-05 13:45:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-09 13:24:42 +02:00
										 |  |  | 	// HTTPPort is the TCP port number on which to start the HTTP RPC server. The | 
					
						
							| 
									
										
										
										
											2016-02-05 13:45:36 +02:00
										 |  |  | 	// default zero value is/ valid and will pick a port number randomly (useful | 
					
						
							|  |  |  | 	// for ephemeral nodes). | 
					
						
							| 
									
										
										
										
											2016-02-09 13:24:42 +02:00
										 |  |  | 	HTTPPort int | 
					
						
							| 
									
										
										
										
											2016-02-05 13:45:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-09 13:24:42 +02:00
										 |  |  | 	// HTTPCors is the Cross-Origin Resource Sharing header to send to requesting | 
					
						
							| 
									
										
										
										
											2016-02-05 13:45:36 +02:00
										 |  |  | 	// clients. Please be aware that CORS is a browser enforced security, it's fully | 
					
						
							|  |  |  | 	// useless for custom HTTP clients. | 
					
						
							| 
									
										
										
										
											2016-02-09 13:24:42 +02:00
										 |  |  | 	HTTPCors string | 
					
						
							| 
									
										
										
										
											2016-02-05 13:45:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-09 13:24:42 +02:00
										 |  |  | 	// HTTPModules is a list of API modules to expose via the HTTP RPC interface. | 
					
						
							| 
									
										
										
										
											2016-02-05 13:45:36 +02:00
										 |  |  | 	// If the module list is empty, all RPC API endpoints designated public will be | 
					
						
							|  |  |  | 	// exposed. | 
					
						
							| 
									
										
										
										
											2016-02-09 13:24:42 +02:00
										 |  |  | 	HTTPModules []string | 
					
						
							| 
									
										
										
										
											2016-02-05 15:08:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-09 13:24:42 +02:00
										 |  |  | 	// WSHost is the host interface on which to start the websocket RPC server. If | 
					
						
							| 
									
										
										
										
											2016-02-05 15:08:48 +02:00
										 |  |  | 	// this field is empty, no websocket API endpoint will be started. | 
					
						
							| 
									
										
										
										
											2016-02-09 13:24:42 +02:00
										 |  |  | 	WSHost string | 
					
						
							| 
									
										
										
										
											2016-02-05 15:08:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-09 13:24:42 +02:00
										 |  |  | 	// WSPort is the TCP port number on which to start the websocket RPC server. The | 
					
						
							| 
									
										
										
										
											2016-02-05 15:08:48 +02:00
										 |  |  | 	// default zero value is/ valid and will pick a port number randomly (useful for | 
					
						
							|  |  |  | 	// ephemeral nodes). | 
					
						
							| 
									
										
										
										
											2016-02-09 13:24:42 +02:00
										 |  |  | 	WSPort int | 
					
						
							| 
									
										
										
										
											2016-02-05 15:08:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-09 13:24:42 +02:00
										 |  |  | 	// WSDomains is the list of domain to accept websocket requests from. Please be | 
					
						
							|  |  |  | 	// aware that the server can only act upon the HTTP request the client sends and | 
					
						
							|  |  |  | 	// cannot verify the validity of the request header. | 
					
						
							|  |  |  | 	WSDomains string | 
					
						
							| 
									
										
										
										
											2016-02-05 15:08:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-09 13:24:42 +02:00
										 |  |  | 	// WSModules is a list of API modules to expose via the websocket RPC interface. | 
					
						
							| 
									
										
										
										
											2016-02-05 15:08:48 +02:00
										 |  |  | 	// If the module list is empty, all RPC API endpoints designated public will be | 
					
						
							|  |  |  | 	// exposed. | 
					
						
							| 
									
										
										
										
											2016-02-09 13:24:42 +02:00
										 |  |  | 	WSModules []string | 
					
						
							| 
									
										
										
										
											2015-11-05 23:57:57 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-09 13:24:42 +02:00
										 |  |  | // IPCEndpoint resolves an IPC endpoint based on a configured value, taking into | 
					
						
							| 
									
										
										
										
											2016-02-02 19:06:43 +02:00
										 |  |  | // account the set data folders as well as the designated platform we're currently | 
					
						
							|  |  |  | // running on. | 
					
						
							| 
									
										
										
										
											2016-02-09 13:24:42 +02:00
										 |  |  | func (c *Config) IPCEndpoint() string { | 
					
						
							| 
									
										
										
										
											2016-02-02 19:06:43 +02:00
										 |  |  | 	// Short circuit if IPC has not been enabled | 
					
						
							| 
									
										
										
										
											2016-02-09 13:24:42 +02:00
										 |  |  | 	if c.IPCPath == "" { | 
					
						
							| 
									
										
										
										
											2016-02-02 19:06:43 +02:00
										 |  |  | 		return "" | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// On windows we can only use plain top-level pipes | 
					
						
							|  |  |  | 	if runtime.GOOS == "windows" { | 
					
						
							| 
									
										
										
										
											2016-02-09 13:24:42 +02:00
										 |  |  | 		if strings.HasPrefix(c.IPCPath, `\\.\pipe\`) { | 
					
						
							|  |  |  | 			return c.IPCPath | 
					
						
							| 
									
										
										
										
											2016-02-02 19:06:43 +02:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-02-09 13:24:42 +02:00
										 |  |  | 		return `\\.\pipe\` + c.IPCPath | 
					
						
							| 
									
										
										
										
											2016-02-02 19:06:43 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	// Resolve names into the data directory full paths otherwise | 
					
						
							| 
									
										
										
										
											2016-02-09 13:24:42 +02:00
										 |  |  | 	if filepath.Base(c.IPCPath) == c.IPCPath { | 
					
						
							| 
									
										
										
										
											2016-02-02 19:06:43 +02:00
										 |  |  | 		if c.DataDir == "" { | 
					
						
							| 
									
										
										
										
											2016-02-09 13:24:42 +02:00
										 |  |  | 			return filepath.Join(os.TempDir(), c.IPCPath) | 
					
						
							| 
									
										
										
										
											2016-02-02 19:06:43 +02:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-02-09 13:24:42 +02:00
										 |  |  | 		return filepath.Join(c.DataDir, c.IPCPath) | 
					
						
							| 
									
										
										
										
											2016-02-02 19:06:43 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-02-09 13:24:42 +02:00
										 |  |  | 	return c.IPCPath | 
					
						
							| 
									
										
										
										
											2016-02-02 19:06:43 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-09 13:24:42 +02:00
										 |  |  | // DefaultIPCEndpoint returns the IPC path used by default. | 
					
						
							|  |  |  | func DefaultIPCEndpoint() string { | 
					
						
							|  |  |  | 	config := &Config{DataDir: common.DefaultDataDir(), IPCPath: common.DefaultIPCSocket} | 
					
						
							|  |  |  | 	return config.IPCEndpoint() | 
					
						
							| 
									
										
										
										
											2016-02-02 19:06:43 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-09 13:24:42 +02:00
										 |  |  | // HTTPEndpoint resolves an HTTP endpoint based on the configured host interface | 
					
						
							| 
									
										
										
										
											2016-02-05 13:45:36 +02:00
										 |  |  | // and port parameters. | 
					
						
							| 
									
										
										
										
											2016-02-09 13:24:42 +02:00
										 |  |  | func (c *Config) HTTPEndpoint() string { | 
					
						
							|  |  |  | 	if c.HTTPHost == "" { | 
					
						
							| 
									
										
										
										
											2016-02-05 13:45:36 +02:00
										 |  |  | 		return "" | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-02-09 13:24:42 +02:00
										 |  |  | 	return fmt.Sprintf("%s:%d", c.HTTPHost, c.HTTPPort) | 
					
						
							| 
									
										
										
										
											2016-02-05 13:45:36 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-09 13:24:42 +02:00
										 |  |  | // DefaultHTTPEndpoint returns the HTTP endpoint used by default. | 
					
						
							|  |  |  | func DefaultHTTPEndpoint() string { | 
					
						
							|  |  |  | 	config := &Config{HTTPHost: common.DefaultHTTPHost, HTTPPort: common.DefaultHTTPPort} | 
					
						
							|  |  |  | 	return config.HTTPEndpoint() | 
					
						
							| 
									
										
										
										
											2016-02-05 13:45:36 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-09 13:24:42 +02:00
										 |  |  | // WSEndpoint resolves an websocket endpoint based on the configured host interface | 
					
						
							| 
									
										
										
										
											2016-02-05 15:08:48 +02:00
										 |  |  | // and port parameters. | 
					
						
							| 
									
										
										
										
											2016-02-09 13:24:42 +02:00
										 |  |  | func (c *Config) WSEndpoint() string { | 
					
						
							|  |  |  | 	if c.WSHost == "" { | 
					
						
							| 
									
										
										
										
											2016-02-05 15:08:48 +02:00
										 |  |  | 		return "" | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-02-09 13:24:42 +02:00
										 |  |  | 	return fmt.Sprintf("%s:%d", c.WSHost, c.WSPort) | 
					
						
							| 
									
										
										
										
											2016-02-05 15:08:48 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-09 13:24:42 +02:00
										 |  |  | // DefaultWSEndpoint returns the websocket endpoint used by default. | 
					
						
							|  |  |  | func DefaultWSEndpoint() string { | 
					
						
							|  |  |  | 	config := &Config{WSHost: common.DefaultWSHost, WSPort: common.DefaultWSPort} | 
					
						
							|  |  |  | 	return config.WSEndpoint() | 
					
						
							| 
									
										
										
										
											2016-02-05 15:08:48 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-05 23:57:57 +02:00
										 |  |  | // NodeKey retrieves the currently configured private key of the node, checking | 
					
						
							|  |  |  | // first any manually set key, falling back to the one found in the configured | 
					
						
							|  |  |  | // data folder. If no key can be found, a new one is generated. | 
					
						
							|  |  |  | func (c *Config) NodeKey() *ecdsa.PrivateKey { | 
					
						
							|  |  |  | 	// Use any specifically configured key | 
					
						
							|  |  |  | 	if c.PrivateKey != nil { | 
					
						
							|  |  |  | 		return c.PrivateKey | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Generate ephemeral key if no datadir is being used | 
					
						
							|  |  |  | 	if c.DataDir == "" { | 
					
						
							|  |  |  | 		key, err := crypto.GenerateKey() | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			glog.Fatalf("Failed to generate ephemeral node key: %v", err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return key | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Fall back to persistent key from the data directory | 
					
						
							|  |  |  | 	keyfile := filepath.Join(c.DataDir, datadirPrivateKey) | 
					
						
							|  |  |  | 	if key, err := crypto.LoadECDSA(keyfile); err == nil { | 
					
						
							|  |  |  | 		return key | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// No persistent key found, generate and store a new one | 
					
						
							|  |  |  | 	key, err := crypto.GenerateKey() | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		glog.Fatalf("Failed to generate node key: %v", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if err := crypto.SaveECDSA(keyfile, key); err != nil { | 
					
						
							|  |  |  | 		glog.V(logger.Error).Infof("Failed to persist node key: %v", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return key | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // StaticNodes returns a list of node enode URLs configured as static nodes. | 
					
						
							|  |  |  | func (c *Config) StaticNodes() []*discover.Node { | 
					
						
							|  |  |  | 	return c.parsePersistentNodes(datadirStaticNodes) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // TrusterNodes returns a list of node enode URLs configured as trusted nodes. | 
					
						
							|  |  |  | func (c *Config) TrusterNodes() []*discover.Node { | 
					
						
							|  |  |  | 	return c.parsePersistentNodes(datadirTrustedNodes) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // parsePersistentNodes parses a list of discovery node URLs loaded from a .json | 
					
						
							|  |  |  | // file from within the data directory. | 
					
						
							|  |  |  | func (c *Config) parsePersistentNodes(file string) []*discover.Node { | 
					
						
							|  |  |  | 	// Short circuit if no node config is present | 
					
						
							|  |  |  | 	if c.DataDir == "" { | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	path := filepath.Join(c.DataDir, file) | 
					
						
							|  |  |  | 	if _, err := os.Stat(path); err != nil { | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Load the nodes from the config file | 
					
						
							|  |  |  | 	blob, err := ioutil.ReadFile(path) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		glog.V(logger.Error).Infof("Failed to access nodes: %v", err) | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	nodelist := []string{} | 
					
						
							|  |  |  | 	if err := json.Unmarshal(blob, &nodelist); err != nil { | 
					
						
							|  |  |  | 		glog.V(logger.Error).Infof("Failed to load nodes: %v", err) | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Interpret the list as a discovery node array | 
					
						
							|  |  |  | 	var nodes []*discover.Node | 
					
						
							|  |  |  | 	for _, url := range nodelist { | 
					
						
							|  |  |  | 		if url == "" { | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		node, err := discover.ParseNode(url) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			glog.V(logger.Error).Infof("Node URL %s: %v\n", url, err) | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		nodes = append(nodes, node) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return nodes | 
					
						
							|  |  |  | } |