package geth /* #include #include extern bool StatusServiceSignalEvent(const char *jsonEvent); */ import "C" import ( "bytes" "io" "os" "path" "path/filepath" "strconv" "sync" "time" "github.com/ethereum/go-ethereum/accounts" "github.com/ethereum/go-ethereum/common" "github.com/ethereum/go-ethereum/logger" "github.com/ethereum/go-ethereum/logger/glog" ) var muPrepareTestNode sync.Mutex const ( TestDataDir = "../.ethereumtest" TestNodeSyncSeconds = 120 ) type NodeNotificationHandler func(jsonEvent string) var notificationHandler NodeNotificationHandler = func(jsonEvent string) { // internal signal handler (used in tests) glog.V(logger.Info).Infof("notification received (default notification handler): %s\n", jsonEvent) } func SetDefaultNodeNotificationHandler(fn NodeNotificationHandler) { notificationHandler = fn } //export NotifyNode func NotifyNode(jsonEvent *C.char) { notificationHandler(C.GoString(jsonEvent)) } // export TriggerTestSignal func TriggerTestSignal() { C.StatusServiceSignalEvent(C.CString(`{"answer": 42}`)) } func CopyFile(dst, src string) error { s, err := os.Open(src) if err != nil { return err } defer s.Close() d, err := os.Create(dst) if err != nil { return err } defer d.Close() if _, err := io.Copy(d, s); err != nil { return err } return nil } // LoadFromFile is usefull for loading test data, from testdata/filename into a variable func LoadFromFile(filename string) string { f, err := os.Open(filename) if err != nil { return "" } buf := bytes.NewBuffer(nil) io.Copy(buf, f) f.Close() return string(buf.Bytes()) } func PrepareTestNode() (err error) { muPrepareTestNode.Lock() defer muPrepareTestNode.Unlock() manager := NodeManagerInstance() if manager.NodeInited() { return nil } syncRequired := false if _, err := os.Stat(filepath.Join(TestDataDir, "testnet")); os.IsNotExist(err) { syncRequired = true } // prepare node directory dataDir, err := PreprocessDataDir(TestDataDir) if err != nil { glog.V(logger.Warn).Infoln("make node failed:", err) return err } // import test account (with test ether on it) dst := filepath.Join(TestDataDir, "testnet", "keystore", "test-account.pk") if _, err := os.Stat(dst); os.IsNotExist(err) { err = CopyFile(dst, filepath.Join("../data", "test-account.pk")) if err != nil { glog.V(logger.Warn).Infof("cannot copy test account PK: %v", err) return err } } // start geth node and wait for it to initialize // internally once.Do() is used, so call below is thread-safe CreateAndRunNode(dataDir, 8546) // to avoid conflicts with running react-native app, run on different port manager = NodeManagerInstance() if !manager.NodeInited() { panic(ErrInvalidGethNode) } if service, err := manager.RPCClient(); err != nil || service == nil { panic(ErrInvalidGethNode) } if service, err := manager.WhisperService(); err != nil || service == nil { panic(ErrInvalidGethNode) } if service, err := manager.LightEthereumService(); err != nil || service == nil { panic(ErrInvalidGethNode) } if syncRequired { glog.V(logger.Warn).Infof("Sync is required, it will take %d seconds", TestNodeSyncSeconds) time.Sleep(TestNodeSyncSeconds * time.Second) // LES syncs headers, so that we are up do date when it is done } else { time.Sleep(5 * time.Second) } return nil } func RemoveTestNode() { err := os.RemoveAll(TestDataDir) if err != nil { glog.V(logger.Warn).Infof("could not clean up temporary datadir") } } func PreprocessDataDir(dataDir string) (string, error) { testDataDir := path.Join(dataDir, "testnet", "keystore") if _, err := os.Stat(testDataDir); os.IsNotExist(err) { if err := os.MkdirAll(testDataDir, 0755); err != nil { return dataDir, ErrDataDirPreprocessingFailed } } // copy over static peer nodes list (LES auto-discovery is not stable yet) dst := filepath.Join(dataDir, "testnet", "static-nodes.json") if _, err := os.Stat(dst); os.IsNotExist(err) { src := filepath.Join("../data", "static-nodes.json") if err := CopyFile(dst, src); err != nil { return dataDir, err } } return dataDir, nil } // PanicAfter throws panic() after waitSeconds, unless abort channel receives notification func PanicAfter(waitSeconds time.Duration, abort chan struct{}, desc string) { go func() { select { case <-abort: return case <-time.After(waitSeconds): panic("whatever you were doing takes toooo long: " + desc) } }() } func FromAddress(accountAddress string) common.Address { accountManager, err := NodeManagerInstance().AccountManager() if err != nil { return common.Address{} } from, err := ParseAccountString(accountManager, accountAddress) if err != nil { return common.Address{} } return from.Address } func ToAddress(accountAddress string) *common.Address { accountManager, err := NodeManagerInstance().AccountManager() if err != nil { return nil } to, err := ParseAccountString(accountManager, accountAddress) if err != nil { return nil } return &to.Address } // parseAccount parses hex encoded string or key index in the accounts key store // and converts it to an internal account representation. func ParseAccountString(accman *accounts.Manager, account string) (accounts.Account, error) { // valid address, convert to account if common.IsHexAddress(account) { return accounts.Account{Address: common.HexToAddress(account)}, nil } // valid key index, return account referenced by that key index, err := strconv.Atoi(account) if err != nil { return accounts.Account{}, ErrInvalidAccountAddressOrKey } return accman.AccountByIndex(index) }