2016-12-07 21:07:08 +00:00
|
|
|
package geth
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"os"
|
|
|
|
"os/signal"
|
|
|
|
"path/filepath"
|
2016-12-11 12:19:20 +00:00
|
|
|
"strings"
|
2016-12-07 21:07:08 +00:00
|
|
|
"sync"
|
|
|
|
|
|
|
|
"github.com/ethereum/go-ethereum/accounts"
|
2017-02-27 12:52:10 +00:00
|
|
|
"github.com/ethereum/go-ethereum/accounts/keystore"
|
2016-12-07 21:07:08 +00:00
|
|
|
"github.com/ethereum/go-ethereum/common"
|
|
|
|
"github.com/ethereum/go-ethereum/les"
|
2017-05-02 14:35:37 +00:00
|
|
|
"github.com/ethereum/go-ethereum/log"
|
2016-12-11 12:19:20 +00:00
|
|
|
"github.com/ethereum/go-ethereum/node"
|
2016-12-07 21:07:08 +00:00
|
|
|
"github.com/ethereum/go-ethereum/p2p/discover"
|
|
|
|
"github.com/ethereum/go-ethereum/rpc"
|
2017-03-21 16:27:59 +00:00
|
|
|
whisper "github.com/ethereum/go-ethereum/whisper/whisperv5"
|
2017-03-14 13:17:32 +00:00
|
|
|
"github.com/status-im/status-go/geth/params"
|
2016-12-07 21:07:08 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// SelectedExtKey is a container for currently selected (logged in) account
|
|
|
|
type SelectedExtKey struct {
|
|
|
|
Address common.Address
|
2017-02-27 12:52:10 +00:00
|
|
|
AccountKey *keystore.Key
|
2016-12-07 21:07:08 +00:00
|
|
|
SubAccounts []accounts.Account
|
|
|
|
}
|
|
|
|
|
|
|
|
// NodeManager manages Status node (which abstracts contained geth node)
|
|
|
|
type NodeManager struct {
|
2016-12-11 12:19:20 +00:00
|
|
|
node *Node // reference to Status node
|
|
|
|
services *NodeServiceStack // default stack of services running on geth node
|
|
|
|
api *node.PrivateAdminAPI // exposes collection of administrative API methods
|
|
|
|
SelectedAccount *SelectedExtKey // account that was processed during the last call to SelectAccount()
|
2016-12-07 21:07:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// NodeServiceStack contains "standard" node services (which are always available)
|
|
|
|
type NodeServiceStack struct {
|
|
|
|
lightEthereum *les.LightEthereum // LES service
|
|
|
|
whisperService *whisper.Whisper // Whisper service
|
|
|
|
rpcClient *rpc.Client // RPC client
|
|
|
|
jailedRequestQueue *JailedRequestQueue // bridge via which jail notifies node of incoming requests
|
|
|
|
}
|
|
|
|
|
2017-05-03 14:24:48 +00:00
|
|
|
// errors
|
2016-12-07 21:07:08 +00:00
|
|
|
var (
|
|
|
|
ErrInvalidGethNode = errors.New("no running geth node detected")
|
|
|
|
ErrInvalidAccountManager = errors.New("could not retrieve account manager")
|
|
|
|
ErrInvalidWhisperService = errors.New("whisper service is unavailable")
|
|
|
|
ErrInvalidLightEthereumService = errors.New("can not retrieve LES service")
|
|
|
|
ErrInvalidClient = errors.New("RPC client is not properly initialized")
|
|
|
|
ErrInvalidJailedRequestQueue = errors.New("jailed request queue is not properly initialized")
|
|
|
|
ErrNodeMakeFailure = errors.New("error creating p2p node")
|
|
|
|
ErrNodeStartFailure = errors.New("error starting p2p node")
|
2017-04-12 18:26:54 +00:00
|
|
|
ErrNodeRunFailure = errors.New("error running p2p node")
|
2016-12-11 12:19:20 +00:00
|
|
|
ErrInvalidNodeAPI = errors.New("no node API connected")
|
2017-02-27 12:52:10 +00:00
|
|
|
ErrAccountKeyStoreMissing = errors.New("account key store is not set")
|
2016-12-07 21:07:08 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
nodeManagerInstance *NodeManager
|
|
|
|
createOnce sync.Once
|
|
|
|
)
|
|
|
|
|
|
|
|
// CreateAndRunNode creates and starts running Geth node locally (exposing given RPC port along the way)
|
2017-03-15 21:03:01 +00:00
|
|
|
func CreateAndRunNode(config *params.NodeConfig) error {
|
2017-01-12 17:56:36 +00:00
|
|
|
defer HaltOnPanic()
|
|
|
|
|
2017-01-26 00:39:20 +00:00
|
|
|
nodeManager := NewNodeManager(config)
|
2016-12-07 21:07:08 +00:00
|
|
|
|
|
|
|
if nodeManager.NodeInited() {
|
|
|
|
nodeManager.RunNode()
|
2017-01-23 06:25:24 +00:00
|
|
|
nodeManager.WaitNodeStarted()
|
2016-12-07 21:07:08 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return ErrNodeStartFailure
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewNodeManager makes new instance of node manager
|
2017-03-15 21:03:01 +00:00
|
|
|
func NewNodeManager(config *params.NodeConfig) *NodeManager {
|
2016-12-07 21:07:08 +00:00
|
|
|
createOnce.Do(func() {
|
|
|
|
nodeManagerInstance = &NodeManager{
|
|
|
|
services: &NodeServiceStack{
|
|
|
|
jailedRequestQueue: NewJailedRequestsQueue(),
|
|
|
|
},
|
|
|
|
}
|
2017-01-26 00:39:20 +00:00
|
|
|
nodeManagerInstance.node = MakeNode(config)
|
2016-12-07 21:07:08 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
return nodeManagerInstance
|
|
|
|
}
|
|
|
|
|
|
|
|
// NodeManagerInstance exposes node manager instance
|
|
|
|
func NodeManagerInstance() *NodeManager {
|
|
|
|
return nodeManagerInstance
|
|
|
|
}
|
|
|
|
|
|
|
|
// RunNode starts Geth node
|
|
|
|
func (m *NodeManager) RunNode() {
|
|
|
|
go func() {
|
2017-01-12 17:56:36 +00:00
|
|
|
defer HaltOnPanic()
|
|
|
|
|
2016-12-07 21:07:08 +00:00
|
|
|
m.StartNode()
|
|
|
|
|
|
|
|
if _, err := m.AccountManager(); err != nil {
|
2017-05-02 14:35:37 +00:00
|
|
|
log.Warn(ErrInvalidAccountManager.Error())
|
2016-12-07 21:07:08 +00:00
|
|
|
}
|
|
|
|
if err := m.node.geth.Service(&m.services.whisperService); err != nil {
|
2017-05-02 14:35:37 +00:00
|
|
|
log.Warn("cannot get whisper service", "error", err)
|
2016-12-07 21:07:08 +00:00
|
|
|
}
|
|
|
|
if err := m.node.geth.Service(&m.services.lightEthereum); err != nil {
|
2017-05-02 14:35:37 +00:00
|
|
|
log.Warn("cannot get light ethereum service", "error", err)
|
2016-12-07 21:07:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// setup handlers
|
2017-03-28 09:04:52 +00:00
|
|
|
if lightEthereum, err := m.LightEthereumService(); err == nil {
|
|
|
|
lightEthereum.StatusBackend.SetTransactionQueueHandler(onSendTransactionRequest)
|
|
|
|
lightEthereum.StatusBackend.SetAccountsFilterHandler(onAccountsListRequest)
|
|
|
|
lightEthereum.StatusBackend.SetTransactionReturnHandler(onSendTransactionReturn)
|
2016-12-07 21:07:08 +00:00
|
|
|
}
|
|
|
|
|
2017-03-28 09:04:52 +00:00
|
|
|
var err error
|
2016-12-07 21:07:08 +00:00
|
|
|
m.services.rpcClient, err = m.node.geth.Attach()
|
|
|
|
if err != nil {
|
2017-05-02 14:35:37 +00:00
|
|
|
log.Warn("cannot get RPC client service", "error", ErrInvalidClient)
|
2016-12-07 21:07:08 +00:00
|
|
|
}
|
|
|
|
|
2016-12-11 12:19:20 +00:00
|
|
|
// expose API
|
|
|
|
m.api = node.NewPrivateAdminAPI(m.node.geth)
|
|
|
|
|
2017-03-10 00:16:32 +00:00
|
|
|
m.PopulateStaticPeers()
|
2016-12-07 21:07:08 +00:00
|
|
|
|
|
|
|
m.onNodeStarted() // node started, notify listeners
|
|
|
|
m.node.geth.Wait()
|
2017-01-23 06:25:24 +00:00
|
|
|
|
2017-05-02 14:35:37 +00:00
|
|
|
log.Info("node stopped")
|
2016-12-07 21:07:08 +00:00
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
|
|
|
// StartNode starts running P2P node
|
|
|
|
func (m *NodeManager) StartNode() {
|
|
|
|
if m == nil || !m.NodeInited() {
|
|
|
|
panic(ErrInvalidGethNode)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := m.node.geth.Start(); err != nil {
|
|
|
|
panic(fmt.Sprintf("%v: %v", ErrNodeStartFailure, err))
|
|
|
|
}
|
|
|
|
|
2017-05-02 14:35:37 +00:00
|
|
|
if server := m.node.geth.Server(); server != nil {
|
|
|
|
if nodeInfo := server.NodeInfo(); nodeInfo != nil {
|
|
|
|
log.Info(nodeInfo.Enode)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-07 21:07:08 +00:00
|
|
|
// allow interrupting running nodes
|
|
|
|
go func() {
|
|
|
|
sigc := make(chan os.Signal, 1)
|
|
|
|
signal.Notify(sigc, os.Interrupt)
|
|
|
|
defer signal.Stop(sigc)
|
|
|
|
<-sigc
|
2017-05-02 14:35:37 +00:00
|
|
|
log.Info("Got interrupt, shutting down...")
|
2017-05-03 14:24:48 +00:00
|
|
|
go m.node.geth.Stop() // nolint: errcheck
|
2016-12-07 21:07:08 +00:00
|
|
|
for i := 3; i > 0; i-- {
|
|
|
|
<-sigc
|
|
|
|
if i > 1 {
|
2017-05-02 14:35:37 +00:00
|
|
|
log.Info(fmt.Sprintf("Already shutting down, interrupt %d more times for panic.", i-1))
|
2016-12-07 21:07:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
panic("interrupted!")
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
2017-01-23 06:25:24 +00:00
|
|
|
// StopNode stops running P2P node
|
|
|
|
func (m *NodeManager) StopNode() error {
|
|
|
|
if m == nil || !m.NodeInited() {
|
|
|
|
return ErrInvalidGethNode
|
|
|
|
}
|
|
|
|
|
2017-05-02 14:35:37 +00:00
|
|
|
if err := m.node.geth.Stop(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2017-01-23 06:25:24 +00:00
|
|
|
m.node.started = make(chan struct{})
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// RestartNode restarts P2P node
|
2016-12-11 12:19:20 +00:00
|
|
|
func (m *NodeManager) RestartNode() error {
|
|
|
|
if m == nil || !m.NodeInited() {
|
|
|
|
return ErrInvalidGethNode
|
|
|
|
}
|
|
|
|
|
2017-05-03 14:24:48 +00:00
|
|
|
if err := m.StopNode(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2017-01-23 06:25:24 +00:00
|
|
|
m.RunNode()
|
|
|
|
m.WaitNodeStarted()
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2017-01-24 18:42:55 +00:00
|
|
|
// ResumeNode resumes previously stopped P2P node
|
|
|
|
func (m *NodeManager) ResumeNode() error {
|
|
|
|
if m == nil || !m.NodeInited() {
|
|
|
|
return ErrInvalidGethNode
|
|
|
|
}
|
|
|
|
|
|
|
|
m.RunNode()
|
|
|
|
m.WaitNodeStarted()
|
|
|
|
|
2017-05-03 14:24:48 +00:00
|
|
|
return ReSelectAccount()
|
2017-01-24 18:42:55 +00:00
|
|
|
}
|
|
|
|
|
2017-01-23 06:25:24 +00:00
|
|
|
// ResetChainData purges chain data (by removing data directory). Safe to apply on running P2P node.
|
|
|
|
func (m *NodeManager) ResetChainData() error {
|
|
|
|
if m == nil || !m.NodeInited() {
|
|
|
|
return ErrInvalidGethNode
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := m.StopNode(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2017-01-26 00:39:20 +00:00
|
|
|
chainDataDir := filepath.Join(m.node.gethConfig.DataDir, m.node.gethConfig.Name, "lightchaindata")
|
2017-01-23 06:25:24 +00:00
|
|
|
if _, err := os.Stat(chainDataDir); os.IsNotExist(err) {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := os.RemoveAll(chainDataDir); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2017-05-02 14:35:37 +00:00
|
|
|
log.Info("chaindata removed", "dir", chainDataDir)
|
2017-01-23 06:25:24 +00:00
|
|
|
|
2017-05-03 14:24:48 +00:00
|
|
|
return m.ResumeNode()
|
2016-12-11 12:19:20 +00:00
|
|
|
}
|
|
|
|
|
2017-05-03 14:24:48 +00:00
|
|
|
// StartNodeRPCServer starts HTTP RPC server
|
2016-12-11 12:19:20 +00:00
|
|
|
func (m *NodeManager) StartNodeRPCServer() (bool, error) {
|
|
|
|
if m == nil || !m.NodeInited() {
|
|
|
|
return false, ErrInvalidGethNode
|
|
|
|
}
|
|
|
|
|
|
|
|
if m.api == nil {
|
|
|
|
return false, ErrInvalidNodeAPI
|
|
|
|
}
|
|
|
|
|
2017-01-26 00:39:20 +00:00
|
|
|
config := m.node.gethConfig
|
2016-12-11 12:19:20 +00:00
|
|
|
modules := strings.Join(config.HTTPModules, ",")
|
2017-05-02 14:35:37 +00:00
|
|
|
cors := strings.Join(config.HTTPCors, ",")
|
2016-12-11 12:19:20 +00:00
|
|
|
|
2017-05-02 14:35:37 +00:00
|
|
|
return m.api.StartRPC(&config.HTTPHost, &config.HTTPPort, &cors, &modules)
|
2016-12-11 12:19:20 +00:00
|
|
|
}
|
|
|
|
|
2017-05-03 14:24:48 +00:00
|
|
|
// StopNodeRPCServer stops HTTP RPC server attached to node
|
2016-12-11 12:19:20 +00:00
|
|
|
func (m *NodeManager) StopNodeRPCServer() (bool, error) {
|
|
|
|
if m == nil || !m.NodeInited() {
|
|
|
|
return false, ErrInvalidGethNode
|
|
|
|
}
|
|
|
|
|
|
|
|
if m.api == nil {
|
|
|
|
return false, ErrInvalidNodeAPI
|
|
|
|
}
|
|
|
|
|
|
|
|
return m.api.StopRPC()
|
|
|
|
}
|
|
|
|
|
2017-03-28 09:04:52 +00:00
|
|
|
// NodeInited checks whether manager has initialized node attached
|
2016-12-07 21:07:08 +00:00
|
|
|
func (m *NodeManager) NodeInited() bool {
|
|
|
|
if m == nil || !m.node.Inited() {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2017-03-28 09:04:52 +00:00
|
|
|
// Node returns attached node if it has been initialized
|
|
|
|
func (m *NodeManager) Node() *Node {
|
|
|
|
if !m.NodeInited() {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return m.node
|
|
|
|
}
|
|
|
|
|
2016-12-07 21:07:08 +00:00
|
|
|
// AccountManager exposes reference to accounts manager
|
|
|
|
func (m *NodeManager) AccountManager() (*accounts.Manager, error) {
|
|
|
|
if m == nil || !m.NodeInited() {
|
|
|
|
return nil, ErrInvalidGethNode
|
|
|
|
}
|
|
|
|
|
|
|
|
return m.node.geth.AccountManager(), nil
|
|
|
|
}
|
|
|
|
|
2017-02-27 12:52:10 +00:00
|
|
|
// AccountKeyStore exposes reference to accounts key store
|
|
|
|
func (m *NodeManager) AccountKeyStore() (*keystore.KeyStore, error) {
|
|
|
|
if m == nil || !m.NodeInited() {
|
|
|
|
return nil, ErrInvalidGethNode
|
|
|
|
}
|
|
|
|
|
|
|
|
accountManager, err := m.AccountManager()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
backends := accountManager.Backends(keystore.KeyStoreType)
|
|
|
|
if len(backends) == 0 {
|
|
|
|
return nil, ErrAccountKeyStoreMissing
|
|
|
|
}
|
|
|
|
|
|
|
|
keyStore, ok := backends[0].(*keystore.KeyStore)
|
|
|
|
if !ok {
|
|
|
|
return nil, ErrAccountKeyStoreMissing
|
|
|
|
}
|
|
|
|
|
|
|
|
return keyStore, nil
|
|
|
|
}
|
|
|
|
|
2016-12-07 21:07:08 +00:00
|
|
|
// LightEthereumService exposes LES
|
2017-05-03 14:24:48 +00:00
|
|
|
// nolint: dupl
|
2016-12-07 21:07:08 +00:00
|
|
|
func (m *NodeManager) LightEthereumService() (*les.LightEthereum, error) {
|
|
|
|
if m == nil || !m.NodeInited() {
|
|
|
|
return nil, ErrInvalidGethNode
|
|
|
|
}
|
|
|
|
|
|
|
|
if m.services.lightEthereum == nil {
|
|
|
|
return nil, ErrInvalidLightEthereumService
|
|
|
|
}
|
|
|
|
|
|
|
|
return m.services.lightEthereum, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// WhisperService exposes Whisper service
|
2017-05-03 14:24:48 +00:00
|
|
|
// nolint: dupl
|
2016-12-07 21:07:08 +00:00
|
|
|
func (m *NodeManager) WhisperService() (*whisper.Whisper, error) {
|
|
|
|
if m == nil || !m.NodeInited() {
|
|
|
|
return nil, ErrInvalidGethNode
|
|
|
|
}
|
|
|
|
|
|
|
|
if m.services.whisperService == nil {
|
|
|
|
return nil, ErrInvalidWhisperService
|
|
|
|
}
|
|
|
|
|
|
|
|
return m.services.whisperService, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// RPCClient exposes Geth's RPC client
|
2017-05-03 14:24:48 +00:00
|
|
|
// nolint: dupl
|
2016-12-07 21:07:08 +00:00
|
|
|
func (m *NodeManager) RPCClient() (*rpc.Client, error) {
|
|
|
|
if m == nil || !m.NodeInited() {
|
|
|
|
return nil, ErrInvalidGethNode
|
|
|
|
}
|
|
|
|
|
|
|
|
if m.services.rpcClient == nil {
|
|
|
|
return nil, ErrInvalidClient
|
|
|
|
}
|
|
|
|
|
|
|
|
return m.services.rpcClient, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// JailedRequestQueue exposes reference to queue of jailed requests
|
|
|
|
func (m *NodeManager) JailedRequestQueue() (*JailedRequestQueue, error) {
|
|
|
|
if m == nil || !m.NodeInited() {
|
|
|
|
return nil, ErrInvalidGethNode
|
|
|
|
}
|
|
|
|
|
|
|
|
if m.services.jailedRequestQueue == nil {
|
|
|
|
return nil, ErrInvalidJailedRequestQueue
|
|
|
|
}
|
|
|
|
|
|
|
|
return m.services.jailedRequestQueue, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// AddPeer adds new peer node
|
|
|
|
func (m *NodeManager) AddPeer(url string) (bool, error) {
|
|
|
|
if m == nil || !m.NodeInited() {
|
|
|
|
return false, ErrInvalidGethNode
|
|
|
|
}
|
|
|
|
|
|
|
|
server := m.node.geth.Server()
|
|
|
|
if server == nil {
|
|
|
|
return false, ErrInvalidGethNode
|
|
|
|
}
|
|
|
|
|
|
|
|
// Try to add the url as a static peer and return
|
|
|
|
parsedNode, err := discover.ParseNode(url)
|
|
|
|
if err != nil {
|
|
|
|
return false, fmt.Errorf("invalid enode: %v", err)
|
|
|
|
}
|
|
|
|
server.AddPeer(parsedNode)
|
|
|
|
|
|
|
|
return true, nil
|
|
|
|
}
|
|
|
|
|
2017-01-23 06:25:24 +00:00
|
|
|
// WaitNodeStarted blocks until node is started (start channel gets notified)
|
|
|
|
func (m *NodeManager) WaitNodeStarted() {
|
|
|
|
<-m.node.started // block until node is started
|
|
|
|
}
|
|
|
|
|
2016-12-07 21:07:08 +00:00
|
|
|
// onNodeStarted sends upward notification letting the app know that Geth node is ready to be used
|
|
|
|
func (m *NodeManager) onNodeStarted() {
|
|
|
|
// notify local listener
|
|
|
|
m.node.started <- struct{}{}
|
|
|
|
close(m.node.started)
|
|
|
|
|
|
|
|
// send signal up to native app
|
2016-12-18 20:36:17 +00:00
|
|
|
SendSignal(SignalEnvelope{
|
2016-12-07 21:07:08 +00:00
|
|
|
Type: EventNodeStarted,
|
|
|
|
Event: struct{}{},
|
2016-12-18 20:36:17 +00:00
|
|
|
})
|
2016-12-07 21:07:08 +00:00
|
|
|
}
|
|
|
|
|
2017-03-10 00:16:32 +00:00
|
|
|
// PopulateStaticPeers connects current node with our publicly available LES/SHH/Swarm cluster
|
|
|
|
func (m *NodeManager) PopulateStaticPeers() {
|
2017-05-12 12:04:32 +00:00
|
|
|
if !m.node.config.BootClusterConfig.Enabled {
|
2017-05-16 03:24:56 +00:00
|
|
|
log.Info("Boot cluster is disabled")
|
2017-05-12 12:04:32 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-05-11 13:20:27 +00:00
|
|
|
enodes, err := m.node.config.LoadBootClusterNodes()
|
|
|
|
if err != nil {
|
2017-05-16 03:24:56 +00:00
|
|
|
log.Warn("Can not load boot nodes", "error", err)
|
2017-05-11 13:20:27 +00:00
|
|
|
}
|
|
|
|
for _, enode := range enodes {
|
2017-05-03 14:24:48 +00:00
|
|
|
m.AddPeer(enode) // nolint: errcheck
|
2017-05-16 03:24:56 +00:00
|
|
|
log.Info("Boot node added", "enode", enode)
|
2016-12-07 21:07:08 +00:00
|
|
|
}
|
|
|
|
}
|
2016-12-15 09:49:05 +00:00
|
|
|
|
2017-05-03 14:24:48 +00:00
|
|
|
// Hex dumps address of a given extended key as hex string
|
2016-12-15 09:49:05 +00:00
|
|
|
func (k *SelectedExtKey) Hex() string {
|
|
|
|
if k == nil {
|
|
|
|
return "0x0"
|
|
|
|
}
|
|
|
|
|
|
|
|
return k.Address.Hex()
|
|
|
|
}
|