2014-05-14 10:41:30 +00:00
|
|
|
package utils
|
|
|
|
|
|
|
|
import (
|
2014-06-26 17:41:36 +00:00
|
|
|
"fmt"
|
|
|
|
"github.com/ethereum/eth-go"
|
|
|
|
"github.com/ethereum/eth-go/ethlog"
|
|
|
|
"github.com/ethereum/eth-go/ethminer"
|
|
|
|
"github.com/ethereum/eth-go/ethpub"
|
|
|
|
"github.com/ethereum/eth-go/ethrpc"
|
|
|
|
"github.com/ethereum/eth-go/ethutil"
|
|
|
|
"io"
|
|
|
|
"log"
|
|
|
|
"os"
|
|
|
|
"os/signal"
|
|
|
|
"path"
|
|
|
|
"strings"
|
|
|
|
"time"
|
2014-05-14 10:41:30 +00:00
|
|
|
)
|
|
|
|
|
2014-06-23 11:20:59 +00:00
|
|
|
var logger = ethlog.NewLogger("CLI")
|
2014-06-26 09:47:45 +00:00
|
|
|
var interruptCallbacks = []func(os.Signal){}
|
2014-06-23 11:20:59 +00:00
|
|
|
|
2014-06-26 09:47:45 +00:00
|
|
|
// Register interrupt handlers callbacks
|
2014-06-23 11:20:59 +00:00
|
|
|
func RegisterInterrupt(cb func(os.Signal)) {
|
2014-06-26 17:41:36 +00:00
|
|
|
interruptCallbacks = append(interruptCallbacks, cb)
|
2014-06-26 09:47:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// go routine that call interrupt handlers in order of registering
|
|
|
|
func HandleInterrupt() {
|
2014-06-26 17:41:36 +00:00
|
|
|
c := make(chan os.Signal, 1)
|
|
|
|
go func() {
|
|
|
|
signal.Notify(c, os.Interrupt)
|
|
|
|
for sig := range c {
|
|
|
|
logger.Errorf("Shutting down (%v) ... \n", sig)
|
|
|
|
RunInterruptCallbacks(sig)
|
|
|
|
}
|
|
|
|
}()
|
2014-05-14 11:32:49 +00:00
|
|
|
}
|
|
|
|
|
2014-06-26 15:26:14 +00:00
|
|
|
func RunInterruptCallbacks(sig os.Signal) {
|
2014-06-26 17:41:36 +00:00
|
|
|
for _, cb := range interruptCallbacks {
|
|
|
|
cb(sig)
|
|
|
|
}
|
2014-06-26 15:26:14 +00:00
|
|
|
}
|
|
|
|
|
2014-06-23 11:20:59 +00:00
|
|
|
func AbsolutePath(Datadir string, filename string) string {
|
2014-06-26 17:41:36 +00:00
|
|
|
if path.IsAbs(filename) {
|
|
|
|
return filename
|
|
|
|
}
|
|
|
|
return path.Join(Datadir, filename)
|
|
|
|
}
|
|
|
|
|
|
|
|
func openLogFile(Datadir string, filename string) *os.File {
|
|
|
|
path := AbsolutePath(Datadir, filename)
|
|
|
|
file, err := os.OpenFile(path, os.O_RDWR|os.O_CREATE|os.O_APPEND, 0666)
|
|
|
|
if err != nil {
|
|
|
|
panic(fmt.Sprintf("error opening log file '%s': %v", filename, err))
|
|
|
|
}
|
|
|
|
return file
|
|
|
|
}
|
|
|
|
|
|
|
|
func confirm(message string) bool {
|
|
|
|
fmt.Println(message, "Are you sure? (y/n)")
|
|
|
|
var r string
|
|
|
|
fmt.Scanln(&r)
|
|
|
|
for ; ; fmt.Scanln(&r) {
|
|
|
|
if r == "n" || r == "y" {
|
|
|
|
break
|
|
|
|
} else {
|
|
|
|
fmt.Printf("Yes or no?", r)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return r == "y"
|
2014-06-23 11:20:59 +00:00
|
|
|
}
|
2014-05-21 22:25:48 +00:00
|
|
|
|
2014-06-23 11:20:59 +00:00
|
|
|
func InitDataDir(Datadir string) {
|
2014-06-26 17:41:36 +00:00
|
|
|
_, err := os.Stat(Datadir)
|
|
|
|
if err != nil {
|
|
|
|
if os.IsNotExist(err) {
|
|
|
|
fmt.Printf("Debug logging directory '%s' doesn't exist, creating it\n", Datadir)
|
|
|
|
os.Mkdir(Datadir, 0777)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func InitLogging(Datadir string, LogFile string, LogLevel int, DebugFile string) {
|
|
|
|
var writer io.Writer
|
|
|
|
if LogFile == "" {
|
|
|
|
writer = os.Stdout
|
|
|
|
} else {
|
|
|
|
writer = openLogFile(Datadir, LogFile)
|
|
|
|
}
|
|
|
|
ethlog.AddLogSystem(ethlog.NewStdLogSystem(writer, log.LstdFlags, ethlog.LogLevel(LogLevel)))
|
|
|
|
if DebugFile != "" {
|
|
|
|
writer = openLogFile(Datadir, DebugFile)
|
|
|
|
ethlog.AddLogSystem(ethlog.NewStdLogSystem(writer, log.LstdFlags, ethlog.DebugLevel))
|
|
|
|
}
|
2014-06-23 11:20:59 +00:00
|
|
|
}
|
2014-05-21 09:45:19 +00:00
|
|
|
|
2014-06-23 11:20:59 +00:00
|
|
|
func InitConfig(ConfigFile string, Datadir string, Identifier string, EnvPrefix string) {
|
2014-06-26 17:41:36 +00:00
|
|
|
InitDataDir(Datadir)
|
|
|
|
ethutil.ReadConfig(ConfigFile, Datadir, Identifier, EnvPrefix)
|
|
|
|
ethutil.Config.Set("rpcport", "700")
|
2014-06-23 11:20:59 +00:00
|
|
|
}
|
2014-05-14 11:26:15 +00:00
|
|
|
|
2014-06-23 11:20:59 +00:00
|
|
|
func exit(status int) {
|
2014-06-26 17:41:36 +00:00
|
|
|
ethlog.Flush()
|
|
|
|
os.Exit(status)
|
2014-05-14 10:41:30 +00:00
|
|
|
}
|
2014-05-21 22:25:48 +00:00
|
|
|
|
2014-06-23 11:20:59 +00:00
|
|
|
func NewEthereum(UseUPnP bool, OutboundPort string, MaxPeer int) *eth.Ethereum {
|
2014-06-26 17:41:36 +00:00
|
|
|
ethereum, err := eth.New(eth.CapDefault, UseUPnP)
|
|
|
|
if err != nil {
|
|
|
|
logger.Fatalln("eth start err:", err)
|
|
|
|
}
|
|
|
|
ethereum.Port = OutboundPort
|
|
|
|
ethereum.MaxPeers = MaxPeer
|
|
|
|
return ethereum
|
2014-06-23 11:20:59 +00:00
|
|
|
}
|
2014-05-21 22:25:48 +00:00
|
|
|
|
2014-06-23 11:20:59 +00:00
|
|
|
func StartEthereum(ethereum *eth.Ethereum, UseSeed bool) {
|
2014-06-26 17:41:36 +00:00
|
|
|
logger.Infof("Starting Ethereum v%s", ethutil.Config.Ver)
|
|
|
|
ethereum.Start(UseSeed)
|
|
|
|
RegisterInterrupt(func(sig os.Signal) {
|
|
|
|
ethereum.Stop()
|
|
|
|
ethlog.Flush()
|
|
|
|
})
|
2014-06-23 11:20:59 +00:00
|
|
|
}
|
2014-05-21 22:25:48 +00:00
|
|
|
|
2014-06-23 11:20:59 +00:00
|
|
|
func ShowGenesis(ethereum *eth.Ethereum) {
|
2014-06-26 17:41:36 +00:00
|
|
|
logger.Infoln(ethereum.BlockChain().Genesis())
|
|
|
|
exit(0)
|
2014-06-23 11:20:59 +00:00
|
|
|
}
|
2014-05-21 22:25:48 +00:00
|
|
|
|
2014-06-23 11:20:59 +00:00
|
|
|
func KeyTasks(GenAddr bool, ImportKey string, ExportKey bool, NonInteractive bool) {
|
2014-06-26 17:41:36 +00:00
|
|
|
switch {
|
|
|
|
case GenAddr:
|
|
|
|
if NonInteractive || confirm("This action overwrites your old private key.") {
|
|
|
|
CreateKeyPair(true)
|
|
|
|
}
|
|
|
|
exit(0)
|
|
|
|
case len(ImportKey) > 0:
|
|
|
|
if NonInteractive || confirm("This action overwrites your old private key.") {
|
|
|
|
// import should be from file
|
|
|
|
mnemonic := strings.Split(ImportKey, " ")
|
|
|
|
if len(mnemonic) == 24 {
|
|
|
|
logger.Infoln("Got mnemonic key, importing.")
|
|
|
|
key := ethutil.MnemonicDecode(mnemonic)
|
|
|
|
ImportPrivateKey(key)
|
|
|
|
} else if len(mnemonic) == 1 {
|
|
|
|
logger.Infoln("Got hex key, importing.")
|
|
|
|
ImportPrivateKey(ImportKey)
|
|
|
|
} else {
|
|
|
|
logger.Errorln("Did not recognise format, exiting.")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
exit(0)
|
|
|
|
case ExportKey: // this should be exporting to a filename
|
|
|
|
keyPair := ethutil.GetKeyRing().Get(0)
|
|
|
|
fmt.Printf(`
|
2014-06-23 11:20:59 +00:00
|
|
|
Generating new address and keypair.
|
|
|
|
Please keep your keys somewhere save.
|
|
|
|
|
|
|
|
++++++++++++++++ KeyRing +++++++++++++++++++
|
|
|
|
addr: %x
|
|
|
|
prvk: %x
|
|
|
|
pubk: %x
|
|
|
|
++++++++++++++++++++++++++++++++++++++++++++
|
|
|
|
save these words so you can restore your account later: %s
|
|
|
|
`, keyPair.Address(), keyPair.PrivateKey, keyPair.PublicKey)
|
|
|
|
|
2014-06-26 17:41:36 +00:00
|
|
|
exit(0)
|
|
|
|
default:
|
|
|
|
// Creates a keypair if none exists
|
|
|
|
CreateKeyPair(false)
|
|
|
|
}
|
2014-06-23 11:20:59 +00:00
|
|
|
}
|
2014-05-21 22:25:48 +00:00
|
|
|
|
2014-06-23 11:20:59 +00:00
|
|
|
func StartRpc(ethereum *eth.Ethereum, RpcPort int) {
|
2014-06-26 17:41:36 +00:00
|
|
|
var err error
|
|
|
|
ethereum.RpcServer, err = ethrpc.NewJsonRpcServer(ethpub.NewPEthereum(ethereum), RpcPort)
|
|
|
|
if err != nil {
|
|
|
|
logger.Errorf("Could not start RPC interface (port %v): %v", RpcPort, err)
|
|
|
|
} else {
|
|
|
|
go ethereum.RpcServer.Start()
|
|
|
|
}
|
2014-05-21 22:25:48 +00:00
|
|
|
}
|
|
|
|
|
2014-06-23 11:20:59 +00:00
|
|
|
var miner ethminer.Miner
|
2014-05-21 22:25:48 +00:00
|
|
|
|
2014-06-23 11:20:59 +00:00
|
|
|
func StartMining(ethereum *eth.Ethereum) bool {
|
2014-06-26 17:41:36 +00:00
|
|
|
if !ethereum.Mining {
|
|
|
|
ethereum.Mining = true
|
|
|
|
|
|
|
|
if ethutil.GetKeyRing().Len() == 0 {
|
|
|
|
logger.Errorln("No address found, can't start mining")
|
|
|
|
ethereum.Mining = false
|
|
|
|
return true //????
|
|
|
|
}
|
|
|
|
keyPair := ethutil.GetKeyRing().Get(0)
|
|
|
|
addr := keyPair.Address()
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
miner = ethminer.NewDefaultMiner(addr, ethereum)
|
|
|
|
// Give it some time to connect with peers
|
|
|
|
time.Sleep(3 * time.Second)
|
2014-06-29 08:43:56 +00:00
|
|
|
for !ethereum.IsUpToDate() == false {
|
|
|
|
time.Sleep(5 * time.Second)
|
|
|
|
}
|
|
|
|
|
2014-06-26 17:41:36 +00:00
|
|
|
logger.Infoln("Miner started")
|
|
|
|
miner := ethminer.NewDefaultMiner(addr, ethereum)
|
|
|
|
miner.Start()
|
|
|
|
}()
|
|
|
|
RegisterInterrupt(func(os.Signal) {
|
|
|
|
StopMining(ethereum)
|
|
|
|
})
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
2014-06-23 11:20:59 +00:00
|
|
|
}
|
2014-05-21 22:25:48 +00:00
|
|
|
|
2014-06-23 11:20:59 +00:00
|
|
|
func StopMining(ethereum *eth.Ethereum) bool {
|
2014-06-26 17:41:36 +00:00
|
|
|
if ethereum.Mining {
|
|
|
|
miner.Stop()
|
|
|
|
logger.Infoln("Miner stopped")
|
|
|
|
ethereum.Mining = false
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
2014-05-21 22:25:48 +00:00
|
|
|
}
|
2014-06-25 07:47:11 +00:00
|
|
|
|
|
|
|
// Replay block
|
|
|
|
func BlockDo(ethereum *eth.Ethereum, hash []byte) error {
|
|
|
|
block := ethereum.BlockChain().GetBlock(hash)
|
|
|
|
if block == nil {
|
|
|
|
return fmt.Errorf("unknown block %x", hash)
|
|
|
|
}
|
|
|
|
|
|
|
|
parent := ethereum.BlockChain().GetBlock(block.PrevHash)
|
|
|
|
|
|
|
|
_, err := ethereum.StateManager().ApplyDiff(parent.State(), parent, block)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
|
|
|
}
|