cmd/geth, eth/fetcher: initial metrics support
Conflicts: cmd/geth/admin.go
This commit is contained in:
		
							
								
								
									
										960
									
								
								cmd/geth/admin.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										960
									
								
								cmd/geth/admin.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,960 @@ | |||||||
|  | package main | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"encoding/json" | ||||||
|  | 	"errors" | ||||||
|  | 	"fmt" | ||||||
|  | 	"math/big" | ||||||
|  | 	"strconv" | ||||||
|  | 	"time" | ||||||
|  |  | ||||||
|  | 	"github.com/rcrowley/go-metrics" | ||||||
|  |  | ||||||
|  | 	"github.com/ethereum/ethash" | ||||||
|  | 	"github.com/ethereum/go-ethereum/accounts" | ||||||
|  | 	"github.com/ethereum/go-ethereum/cmd/utils" | ||||||
|  | 	"github.com/ethereum/go-ethereum/common" | ||||||
|  | 	"github.com/ethereum/go-ethereum/common/compiler" | ||||||
|  | 	"github.com/ethereum/go-ethereum/common/natspec" | ||||||
|  | 	"github.com/ethereum/go-ethereum/common/resolver" | ||||||
|  | 	"github.com/ethereum/go-ethereum/core/state" | ||||||
|  | 	"github.com/ethereum/go-ethereum/core/types" | ||||||
|  | 	"github.com/ethereum/go-ethereum/core/vm" | ||||||
|  | 	"github.com/ethereum/go-ethereum/crypto" | ||||||
|  | 	"github.com/ethereum/go-ethereum/logger/glog" | ||||||
|  | 	"github.com/ethereum/go-ethereum/rlp" | ||||||
|  | 	"github.com/ethereum/go-ethereum/rpc" | ||||||
|  | 	"github.com/ethereum/go-ethereum/xeth" | ||||||
|  | 	"github.com/robertkrimen/otto" | ||||||
|  | 	"gopkg.in/fatih/set.v0" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | /* | ||||||
|  | node admin bindings | ||||||
|  | */ | ||||||
|  |  | ||||||
|  | func (js *jsre) adminBindings() { | ||||||
|  | 	ethO, _ := js.re.Get("eth") | ||||||
|  | 	eth := ethO.Object() | ||||||
|  | 	eth.Set("pendingTransactions", js.pendingTransactions) | ||||||
|  | 	eth.Set("resend", js.resend) | ||||||
|  | 	eth.Set("sign", js.sign) | ||||||
|  |  | ||||||
|  | 	js.re.Set("admin", struct{}{}) | ||||||
|  | 	t, _ := js.re.Get("admin") | ||||||
|  | 	admin := t.Object() | ||||||
|  | 	admin.Set("addPeer", js.addPeer) | ||||||
|  | 	admin.Set("startRPC", js.startRPC) | ||||||
|  | 	admin.Set("stopRPC", js.stopRPC) | ||||||
|  | 	admin.Set("nodeInfo", js.nodeInfo) | ||||||
|  | 	admin.Set("peers", js.peers) | ||||||
|  | 	admin.Set("newAccount", js.newAccount) | ||||||
|  | 	admin.Set("unlock", js.unlock) | ||||||
|  | 	admin.Set("import", js.importChain) | ||||||
|  | 	admin.Set("export", js.exportChain) | ||||||
|  | 	admin.Set("verbosity", js.verbosity) | ||||||
|  | 	admin.Set("progress", js.syncProgress) | ||||||
|  | 	admin.Set("setSolc", js.setSolc) | ||||||
|  |  | ||||||
|  | 	admin.Set("contractInfo", struct{}{}) | ||||||
|  | 	t, _ = admin.Get("contractInfo") | ||||||
|  | 	cinfo := t.Object() | ||||||
|  | 	// newRegistry officially not documented temporary option | ||||||
|  | 	cinfo.Set("start", js.startNatSpec) | ||||||
|  | 	cinfo.Set("stop", js.stopNatSpec) | ||||||
|  | 	cinfo.Set("newRegistry", js.newRegistry) | ||||||
|  | 	cinfo.Set("get", js.getContractInfo) | ||||||
|  | 	cinfo.Set("register", js.register) | ||||||
|  | 	cinfo.Set("registerUrl", js.registerUrl) | ||||||
|  | 	// cinfo.Set("verify", js.verify) | ||||||
|  |  | ||||||
|  | 	admin.Set("miner", struct{}{}) | ||||||
|  | 	t, _ = admin.Get("miner") | ||||||
|  | 	miner := t.Object() | ||||||
|  | 	miner.Set("start", js.startMining) | ||||||
|  | 	miner.Set("stop", js.stopMining) | ||||||
|  | 	miner.Set("hashrate", js.hashrate) | ||||||
|  | 	miner.Set("setExtra", js.setExtra) | ||||||
|  | 	miner.Set("setGasPrice", js.setGasPrice) | ||||||
|  | 	miner.Set("startAutoDAG", js.startAutoDAG) | ||||||
|  | 	miner.Set("stopAutoDAG", js.stopAutoDAG) | ||||||
|  | 	miner.Set("makeDAG", js.makeDAG) | ||||||
|  |  | ||||||
|  | 	admin.Set("txPool", struct{}{}) | ||||||
|  | 	t, _ = admin.Get("txPool") | ||||||
|  | 	txPool := t.Object() | ||||||
|  | 	txPool.Set("pending", js.allPendingTransactions) | ||||||
|  | 	txPool.Set("queued", js.allQueuedTransactions) | ||||||
|  |  | ||||||
|  | 	admin.Set("debug", struct{}{}) | ||||||
|  | 	t, _ = admin.Get("debug") | ||||||
|  | 	debug := t.Object() | ||||||
|  | 	js.re.Set("sleep", js.sleep) | ||||||
|  | 	debug.Set("backtrace", js.backtrace) | ||||||
|  | 	debug.Set("printBlock", js.printBlock) | ||||||
|  | 	debug.Set("dumpBlock", js.dumpBlock) | ||||||
|  | 	debug.Set("getBlockRlp", js.getBlockRlp) | ||||||
|  | 	debug.Set("setHead", js.setHead) | ||||||
|  | 	debug.Set("processBlock", js.debugBlock) | ||||||
|  | 	debug.Set("seedhash", js.seedHash) | ||||||
|  | 	debug.Set("insertBlock", js.insertBlockRlp) | ||||||
|  | 	// undocumented temporary | ||||||
|  | 	debug.Set("waitForBlocks", js.waitForBlocks) | ||||||
|  |  | ||||||
|  | 	admin.Set("metrics", js.metrics) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // generic helper to getBlock by Number/Height or Hex depending on autodetected input | ||||||
|  | // if argument is missing the current block is returned | ||||||
|  | // if block is not found or there is problem with decoding | ||||||
|  | // the appropriate value is returned and block is guaranteed to be nil | ||||||
|  | func (js *jsre) getBlock(call otto.FunctionCall) (*types.Block, error) { | ||||||
|  | 	var block *types.Block | ||||||
|  | 	if len(call.ArgumentList) > 0 { | ||||||
|  | 		if call.Argument(0).IsNumber() { | ||||||
|  | 			num, _ := call.Argument(0).ToInteger() | ||||||
|  | 			block = js.ethereum.ChainManager().GetBlockByNumber(uint64(num)) | ||||||
|  | 		} else if call.Argument(0).IsString() { | ||||||
|  | 			hash, _ := call.Argument(0).ToString() | ||||||
|  | 			block = js.ethereum.ChainManager().GetBlock(common.HexToHash(hash)) | ||||||
|  | 		} else { | ||||||
|  | 			return nil, errors.New("invalid argument for dump. Either hex string or number") | ||||||
|  | 		} | ||||||
|  | 	} else { | ||||||
|  | 		block = js.ethereum.ChainManager().CurrentBlock() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if block == nil { | ||||||
|  | 		return nil, errors.New("block not found") | ||||||
|  | 	} | ||||||
|  | 	return block, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) seedHash(call otto.FunctionCall) otto.Value { | ||||||
|  | 	if len(call.ArgumentList) > 0 { | ||||||
|  | 		if call.Argument(0).IsNumber() { | ||||||
|  | 			num, _ := call.Argument(0).ToInteger() | ||||||
|  | 			hash, err := ethash.GetSeedHash(uint64(num)) | ||||||
|  | 			if err != nil { | ||||||
|  | 				fmt.Println(err) | ||||||
|  | 				return otto.UndefinedValue() | ||||||
|  | 			} | ||||||
|  | 			v, _ := call.Otto.ToValue(fmt.Sprintf("0x%x", hash)) | ||||||
|  | 			return v | ||||||
|  | 		} else { | ||||||
|  | 			fmt.Println("arg not a number") | ||||||
|  | 		} | ||||||
|  | 	} else { | ||||||
|  | 		fmt.Println("requires number argument") | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return otto.UndefinedValue() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) allPendingTransactions(call otto.FunctionCall) otto.Value { | ||||||
|  | 	txs := js.ethereum.TxPool().GetTransactions() | ||||||
|  |  | ||||||
|  | 	ltxs := make([]*tx, len(txs)) | ||||||
|  | 	for i, tx := range txs { | ||||||
|  | 		// no need to check err | ||||||
|  | 		ltxs[i] = newTx(tx) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	v, _ := call.Otto.ToValue(ltxs) | ||||||
|  | 	return v | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) allQueuedTransactions(call otto.FunctionCall) otto.Value { | ||||||
|  | 	txs := js.ethereum.TxPool().GetQueuedTransactions() | ||||||
|  |  | ||||||
|  | 	ltxs := make([]*tx, len(txs)) | ||||||
|  | 	for i, tx := range txs { | ||||||
|  | 		// no need to check err | ||||||
|  | 		ltxs[i] = newTx(tx) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	v, _ := call.Otto.ToValue(ltxs) | ||||||
|  | 	return v | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) pendingTransactions(call otto.FunctionCall) otto.Value { | ||||||
|  | 	txs := js.ethereum.TxPool().GetTransactions() | ||||||
|  |  | ||||||
|  | 	// grab the accounts from the account manager. This will help with determening which | ||||||
|  | 	// transactions should be returned. | ||||||
|  | 	accounts, err := js.ethereum.AccountManager().Accounts() | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.UndefinedValue() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Add the accouns to a new set | ||||||
|  | 	accountSet := set.New() | ||||||
|  | 	for _, account := range accounts { | ||||||
|  | 		accountSet.Add(account.Address) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	//ltxs := make([]*tx, len(txs)) | ||||||
|  | 	var ltxs []*tx | ||||||
|  | 	for _, tx := range txs { | ||||||
|  | 		if from, _ := tx.From(); accountSet.Has(from) { | ||||||
|  | 			ltxs = append(ltxs, newTx(tx)) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	v, _ := call.Otto.ToValue(ltxs) | ||||||
|  | 	return v | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) resend(call otto.FunctionCall) otto.Value { | ||||||
|  | 	if len(call.ArgumentList) == 0 { | ||||||
|  | 		fmt.Println("first argument must be a transaction") | ||||||
|  | 		return otto.FalseValue() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	v, err := call.Argument(0).Export() | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.FalseValue() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if tx, ok := v.(*tx); ok { | ||||||
|  | 		gl, gp := tx.GasLimit, tx.GasPrice | ||||||
|  | 		if len(call.ArgumentList) > 1 { | ||||||
|  | 			gp = call.Argument(1).String() | ||||||
|  | 		} | ||||||
|  | 		if len(call.ArgumentList) > 2 { | ||||||
|  | 			gl = call.Argument(2).String() | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		ret, err := js.xeth.Transact(tx.From, tx.To, tx.Nonce, tx.Value, gl, gp, tx.Data) | ||||||
|  | 		if err != nil { | ||||||
|  | 			fmt.Println(err) | ||||||
|  | 			return otto.FalseValue() | ||||||
|  | 		} | ||||||
|  | 		js.ethereum.TxPool().RemoveTransactions(types.Transactions{tx.tx}) | ||||||
|  |  | ||||||
|  | 		v, _ := call.Otto.ToValue(ret) | ||||||
|  | 		return v | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	fmt.Println("first argument must be a transaction") | ||||||
|  | 	return otto.FalseValue() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) sign(call otto.FunctionCall) otto.Value { | ||||||
|  | 	if len(call.ArgumentList) != 2 { | ||||||
|  | 		fmt.Println("requires 2 arguments: eth.sign(signer, data)") | ||||||
|  | 		return otto.UndefinedValue() | ||||||
|  | 	} | ||||||
|  | 	signer, err := call.Argument(0).ToString() | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.UndefinedValue() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	data, err := call.Argument(1).ToString() | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.UndefinedValue() | ||||||
|  | 	} | ||||||
|  | 	signed, err := js.xeth.Sign(signer, data, false) | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.UndefinedValue() | ||||||
|  | 	} | ||||||
|  | 	v, _ := call.Otto.ToValue(signed) | ||||||
|  | 	return v | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) debugBlock(call otto.FunctionCall) otto.Value { | ||||||
|  | 	block, err := js.getBlock(call) | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.UndefinedValue() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	tstart := time.Now() | ||||||
|  | 	old := vm.Debug | ||||||
|  |  | ||||||
|  | 	if len(call.ArgumentList) > 1 { | ||||||
|  | 		vm.Debug, _ = call.Argument(1).ToBoolean() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	_, err = js.ethereum.BlockProcessor().RetryProcess(block) | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		r, _ := call.Otto.ToValue(map[string]interface{}{"success": false, "time": time.Since(tstart).Seconds()}) | ||||||
|  | 		return r | ||||||
|  | 	} | ||||||
|  | 	vm.Debug = old | ||||||
|  |  | ||||||
|  | 	r, _ := call.Otto.ToValue(map[string]interface{}{"success": true, "time": time.Since(tstart).Seconds()}) | ||||||
|  | 	return r | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) insertBlockRlp(call otto.FunctionCall) otto.Value { | ||||||
|  | 	tstart := time.Now() | ||||||
|  |  | ||||||
|  | 	var block types.Block | ||||||
|  | 	if call.Argument(0).IsString() { | ||||||
|  | 		blockRlp, _ := call.Argument(0).ToString() | ||||||
|  | 		err := rlp.DecodeBytes(common.Hex2Bytes(blockRlp), &block) | ||||||
|  | 		if err != nil { | ||||||
|  | 			fmt.Println(err) | ||||||
|  | 			return otto.UndefinedValue() | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	old := vm.Debug | ||||||
|  | 	vm.Debug = true | ||||||
|  | 	_, err := js.ethereum.BlockProcessor().RetryProcess(&block) | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		r, _ := call.Otto.ToValue(map[string]interface{}{"success": false, "time": time.Since(tstart).Seconds()}) | ||||||
|  | 		return r | ||||||
|  | 	} | ||||||
|  | 	vm.Debug = old | ||||||
|  |  | ||||||
|  | 	r, _ := call.Otto.ToValue(map[string]interface{}{"success": true, "time": time.Since(tstart).Seconds()}) | ||||||
|  | 	return r | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) setHead(call otto.FunctionCall) otto.Value { | ||||||
|  | 	block, err := js.getBlock(call) | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.UndefinedValue() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	js.ethereum.ChainManager().SetHead(block) | ||||||
|  | 	return otto.UndefinedValue() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) syncProgress(call otto.FunctionCall) otto.Value { | ||||||
|  | 	pending, cached, importing, eta := js.ethereum.Downloader().Stats() | ||||||
|  | 	v, _ := call.Otto.ToValue(map[string]interface{}{ | ||||||
|  | 		"pending":   pending, | ||||||
|  | 		"cached":    cached, | ||||||
|  | 		"importing": importing, | ||||||
|  | 		"estimate":  (eta / time.Second * time.Second).String(), | ||||||
|  | 	}) | ||||||
|  | 	return v | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) getBlockRlp(call otto.FunctionCall) otto.Value { | ||||||
|  | 	block, err := js.getBlock(call) | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.UndefinedValue() | ||||||
|  | 	} | ||||||
|  | 	encoded, _ := rlp.EncodeToBytes(block) | ||||||
|  | 	v, _ := call.Otto.ToValue(fmt.Sprintf("%x", encoded)) | ||||||
|  | 	return v | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) setExtra(call otto.FunctionCall) otto.Value { | ||||||
|  | 	extra, err := call.Argument(0).ToString() | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.UndefinedValue() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if len(extra) > 1024 { | ||||||
|  | 		fmt.Println("error: cannot exceed 1024 bytes") | ||||||
|  | 		return otto.UndefinedValue() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	js.ethereum.Miner().SetExtra([]byte(extra)) | ||||||
|  | 	return otto.UndefinedValue() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) setGasPrice(call otto.FunctionCall) otto.Value { | ||||||
|  | 	gasPrice, err := call.Argument(0).ToString() | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.UndefinedValue() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	js.ethereum.Miner().SetGasPrice(common.String2Big(gasPrice)) | ||||||
|  | 	return otto.UndefinedValue() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) hashrate(call otto.FunctionCall) otto.Value { | ||||||
|  | 	v, _ := call.Otto.ToValue(js.ethereum.Miner().HashRate()) | ||||||
|  | 	return v | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) makeDAG(call otto.FunctionCall) otto.Value { | ||||||
|  | 	blockNumber, err := call.Argument(1).ToInteger() | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.FalseValue() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	err = ethash.MakeDAG(uint64(blockNumber), "") | ||||||
|  | 	if err != nil { | ||||||
|  | 		return otto.FalseValue() | ||||||
|  | 	} | ||||||
|  | 	return otto.TrueValue() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) startAutoDAG(otto.FunctionCall) otto.Value { | ||||||
|  | 	js.ethereum.StartAutoDAG() | ||||||
|  | 	return otto.TrueValue() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) stopAutoDAG(otto.FunctionCall) otto.Value { | ||||||
|  | 	js.ethereum.StopAutoDAG() | ||||||
|  | 	return otto.TrueValue() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) backtrace(call otto.FunctionCall) otto.Value { | ||||||
|  | 	tracestr, err := call.Argument(0).ToString() | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.UndefinedValue() | ||||||
|  | 	} | ||||||
|  | 	glog.GetTraceLocation().Set(tracestr) | ||||||
|  |  | ||||||
|  | 	return otto.UndefinedValue() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) verbosity(call otto.FunctionCall) otto.Value { | ||||||
|  | 	v, err := call.Argument(0).ToInteger() | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.UndefinedValue() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	glog.SetV(int(v)) | ||||||
|  | 	return otto.UndefinedValue() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) startMining(call otto.FunctionCall) otto.Value { | ||||||
|  | 	var ( | ||||||
|  | 		threads int64 | ||||||
|  | 		err     error | ||||||
|  | 	) | ||||||
|  |  | ||||||
|  | 	if len(call.ArgumentList) > 0 { | ||||||
|  | 		threads, err = call.Argument(0).ToInteger() | ||||||
|  | 		if err != nil { | ||||||
|  | 			fmt.Println(err) | ||||||
|  | 			return otto.FalseValue() | ||||||
|  | 		} | ||||||
|  | 	} else { | ||||||
|  | 		threads = int64(js.ethereum.MinerThreads) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// switch on DAG autogeneration when miner starts | ||||||
|  | 	js.ethereum.StartAutoDAG() | ||||||
|  |  | ||||||
|  | 	err = js.ethereum.StartMining(int(threads)) | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.FalseValue() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return otto.TrueValue() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) stopMining(call otto.FunctionCall) otto.Value { | ||||||
|  | 	js.ethereum.StopMining() | ||||||
|  | 	js.ethereum.StopAutoDAG() | ||||||
|  | 	return otto.TrueValue() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) startRPC(call otto.FunctionCall) otto.Value { | ||||||
|  | 	addr, err := call.Argument(0).ToString() | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.FalseValue() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	port, err := call.Argument(1).ToInteger() | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.FalseValue() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	corsDomain := js.corsDomain | ||||||
|  | 	if len(call.ArgumentList) > 2 { | ||||||
|  | 		corsDomain, err = call.Argument(2).ToString() | ||||||
|  | 		if err != nil { | ||||||
|  | 			fmt.Println(err) | ||||||
|  | 			return otto.FalseValue() | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	config := rpc.RpcConfig{ | ||||||
|  | 		ListenAddress: addr, | ||||||
|  | 		ListenPort:    uint(port), | ||||||
|  | 		CorsDomain:    corsDomain, | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	xeth := xeth.New(js.ethereum, nil) | ||||||
|  | 	err = rpc.Start(xeth, config) | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.FalseValue() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return otto.TrueValue() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) stopRPC(call otto.FunctionCall) otto.Value { | ||||||
|  | 	if rpc.Stop() == nil { | ||||||
|  | 		return otto.TrueValue() | ||||||
|  | 	} | ||||||
|  | 	return otto.FalseValue() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) addPeer(call otto.FunctionCall) otto.Value { | ||||||
|  | 	nodeURL, err := call.Argument(0).ToString() | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.FalseValue() | ||||||
|  | 	} | ||||||
|  | 	err = js.ethereum.AddPeer(nodeURL) | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.FalseValue() | ||||||
|  | 	} | ||||||
|  | 	return otto.TrueValue() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) unlock(call otto.FunctionCall) otto.Value { | ||||||
|  | 	addr, err := call.Argument(0).ToString() | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.FalseValue() | ||||||
|  | 	} | ||||||
|  | 	seconds, err := call.Argument(2).ToInteger() | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.FalseValue() | ||||||
|  | 	} | ||||||
|  | 	if seconds == 0 { | ||||||
|  | 		seconds = accounts.DefaultAccountUnlockDuration | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	arg := call.Argument(1) | ||||||
|  | 	var passphrase string | ||||||
|  | 	if arg.IsUndefined() { | ||||||
|  | 		fmt.Println("Please enter a passphrase now.") | ||||||
|  | 		passphrase, err = utils.PromptPassword("Passphrase: ", true) | ||||||
|  | 		if err != nil { | ||||||
|  | 			fmt.Println(err) | ||||||
|  | 			return otto.FalseValue() | ||||||
|  | 		} | ||||||
|  | 	} else { | ||||||
|  | 		passphrase, err = arg.ToString() | ||||||
|  | 		if err != nil { | ||||||
|  | 			fmt.Println(err) | ||||||
|  | 			return otto.FalseValue() | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	am := js.ethereum.AccountManager() | ||||||
|  | 	err = am.TimedUnlock(common.HexToAddress(addr), passphrase, time.Duration(seconds)*time.Second) | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Printf("Unlock account failed '%v'\n", err) | ||||||
|  | 		return otto.FalseValue() | ||||||
|  | 	} | ||||||
|  | 	return otto.TrueValue() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) newAccount(call otto.FunctionCall) otto.Value { | ||||||
|  | 	arg := call.Argument(0) | ||||||
|  | 	var passphrase string | ||||||
|  | 	if arg.IsUndefined() { | ||||||
|  | 		fmt.Println("The new account will be encrypted with a passphrase.") | ||||||
|  | 		fmt.Println("Please enter a passphrase now.") | ||||||
|  | 		auth, err := utils.PromptPassword("Passphrase: ", true) | ||||||
|  | 		if err != nil { | ||||||
|  | 			fmt.Println(err) | ||||||
|  | 			return otto.FalseValue() | ||||||
|  | 		} | ||||||
|  | 		confirm, err := utils.PromptPassword("Repeat Passphrase: ", false) | ||||||
|  | 		if err != nil { | ||||||
|  | 			fmt.Println(err) | ||||||
|  | 			return otto.FalseValue() | ||||||
|  | 		} | ||||||
|  | 		if auth != confirm { | ||||||
|  | 			fmt.Println("Passphrases did not match.") | ||||||
|  | 			return otto.FalseValue() | ||||||
|  | 		} | ||||||
|  | 		passphrase = auth | ||||||
|  | 	} else { | ||||||
|  | 		var err error | ||||||
|  | 		passphrase, err = arg.ToString() | ||||||
|  | 		if err != nil { | ||||||
|  | 			fmt.Println(err) | ||||||
|  | 			return otto.FalseValue() | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	acct, err := js.ethereum.AccountManager().NewAccount(passphrase) | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Printf("Could not create the account: %v", err) | ||||||
|  | 		return otto.UndefinedValue() | ||||||
|  | 	} | ||||||
|  | 	v, _ := call.Otto.ToValue(acct.Address.Hex()) | ||||||
|  | 	return v | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) nodeInfo(call otto.FunctionCall) otto.Value { | ||||||
|  | 	v, _ := call.Otto.ToValue(js.ethereum.NodeInfo()) | ||||||
|  | 	return v | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) peers(call otto.FunctionCall) otto.Value { | ||||||
|  | 	v, _ := call.Otto.ToValue(js.ethereum.PeersInfo()) | ||||||
|  | 	return v | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) importChain(call otto.FunctionCall) otto.Value { | ||||||
|  | 	if len(call.ArgumentList) == 0 { | ||||||
|  | 		fmt.Println("require file name. admin.importChain(filename)") | ||||||
|  | 		return otto.FalseValue() | ||||||
|  | 	} | ||||||
|  | 	fn, err := call.Argument(0).ToString() | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.FalseValue() | ||||||
|  | 	} | ||||||
|  | 	if err := utils.ImportChain(js.ethereum.ChainManager(), fn); err != nil { | ||||||
|  | 		fmt.Println("Import error: ", err) | ||||||
|  | 		return otto.FalseValue() | ||||||
|  | 	} | ||||||
|  | 	return otto.TrueValue() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) exportChain(call otto.FunctionCall) otto.Value { | ||||||
|  | 	if len(call.ArgumentList) == 0 { | ||||||
|  | 		fmt.Println("require file name: admin.exportChain(filename)") | ||||||
|  | 		return otto.FalseValue() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	fn, err := call.Argument(0).ToString() | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.FalseValue() | ||||||
|  | 	} | ||||||
|  | 	if err := utils.ExportChain(js.ethereum.ChainManager(), fn); err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.FalseValue() | ||||||
|  | 	} | ||||||
|  | 	return otto.TrueValue() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) printBlock(call otto.FunctionCall) otto.Value { | ||||||
|  | 	block, err := js.getBlock(call) | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.UndefinedValue() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	fmt.Println(block) | ||||||
|  |  | ||||||
|  | 	return otto.UndefinedValue() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) dumpBlock(call otto.FunctionCall) otto.Value { | ||||||
|  | 	block, err := js.getBlock(call) | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.UndefinedValue() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	statedb := state.New(block.Root(), js.ethereum.StateDb()) | ||||||
|  | 	dump := statedb.RawDump() | ||||||
|  | 	v, _ := call.Otto.ToValue(dump) | ||||||
|  | 	return v | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) waitForBlocks(call otto.FunctionCall) otto.Value { | ||||||
|  | 	if len(call.ArgumentList) > 2 { | ||||||
|  | 		fmt.Println("requires 0, 1 or 2 arguments: admin.debug.waitForBlock(minHeight, timeout)") | ||||||
|  | 		return otto.FalseValue() | ||||||
|  | 	} | ||||||
|  | 	var n, timeout int64 | ||||||
|  | 	var timer <-chan time.Time | ||||||
|  | 	var height *big.Int | ||||||
|  | 	var err error | ||||||
|  | 	args := len(call.ArgumentList) | ||||||
|  | 	if args == 2 { | ||||||
|  | 		timeout, err = call.Argument(1).ToInteger() | ||||||
|  | 		if err != nil { | ||||||
|  | 			fmt.Println(err) | ||||||
|  | 			return otto.UndefinedValue() | ||||||
|  | 		} | ||||||
|  | 		timer = time.NewTimer(time.Duration(timeout) * time.Second).C | ||||||
|  | 	} | ||||||
|  | 	if args >= 1 { | ||||||
|  | 		n, err = call.Argument(0).ToInteger() | ||||||
|  | 		if err != nil { | ||||||
|  | 			fmt.Println(err) | ||||||
|  | 			return otto.UndefinedValue() | ||||||
|  | 		} | ||||||
|  | 		height = big.NewInt(n) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if args == 0 { | ||||||
|  | 		height = js.xeth.CurrentBlock().Number() | ||||||
|  | 		height.Add(height, common.Big1) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	wait := js.wait | ||||||
|  | 	js.wait <- height | ||||||
|  | 	select { | ||||||
|  | 	case <-timer: | ||||||
|  | 		// if times out make sure the xeth loop does not block | ||||||
|  | 		go func() { | ||||||
|  | 			select { | ||||||
|  | 			case wait <- nil: | ||||||
|  | 			case <-wait: | ||||||
|  | 			} | ||||||
|  | 		}() | ||||||
|  | 		return otto.UndefinedValue() | ||||||
|  | 	case height = <-wait: | ||||||
|  | 	} | ||||||
|  | 	v, _ := call.Otto.ToValue(height.Uint64()) | ||||||
|  | 	return v | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) metrics(call otto.FunctionCall) otto.Value { | ||||||
|  | 	// Iterate over all the metrics, and just dump for now | ||||||
|  | 	counters := make(map[string]interface{}) | ||||||
|  | 	metrics.DefaultRegistry.Each(func(name string, metric interface{}) { | ||||||
|  | 		switch metric := metric.(type) { | ||||||
|  | 		case metrics.Meter: | ||||||
|  | 			counters[name+"( 1 min)"] = int(metric.Rate1() * 60) | ||||||
|  | 			counters[name+"( 5 min)"] = int(metric.Rate5() * 300) | ||||||
|  | 			counters[name+"(15 min)"] = int(metric.Rate15() * 900) | ||||||
|  |  | ||||||
|  | 		default: | ||||||
|  | 			counters[name] = "Unknown metric type" | ||||||
|  | 		} | ||||||
|  | 	}) | ||||||
|  | 	// Flatten the counters into some metrics and return | ||||||
|  | 	v, _ := call.Otto.ToValue(counters) | ||||||
|  | 	return v | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) sleep(call otto.FunctionCall) otto.Value { | ||||||
|  | 	sec, err := call.Argument(0).ToInteger() | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.FalseValue() | ||||||
|  | 	} | ||||||
|  | 	time.Sleep(time.Duration(sec) * time.Second) | ||||||
|  | 	return otto.UndefinedValue() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) setSolc(call otto.FunctionCall) otto.Value { | ||||||
|  | 	if len(call.ArgumentList) != 1 { | ||||||
|  | 		fmt.Println("needs 1 argument: admin.contractInfo.setSolc(solcPath)") | ||||||
|  | 		return otto.FalseValue() | ||||||
|  | 	} | ||||||
|  | 	solcPath, err := call.Argument(0).ToString() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return otto.FalseValue() | ||||||
|  | 	} | ||||||
|  | 	solc, err := js.xeth.SetSolc(solcPath) | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.FalseValue() | ||||||
|  | 	} | ||||||
|  | 	fmt.Println(solc.Info()) | ||||||
|  | 	return otto.TrueValue() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) register(call otto.FunctionCall) otto.Value { | ||||||
|  | 	if len(call.ArgumentList) != 4 { | ||||||
|  | 		fmt.Println("requires 4 arguments: admin.contractInfo.register(fromaddress, contractaddress, contract, filename)") | ||||||
|  | 		return otto.UndefinedValue() | ||||||
|  | 	} | ||||||
|  | 	sender, err := call.Argument(0).ToString() | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.UndefinedValue() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	address, err := call.Argument(1).ToString() | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.UndefinedValue() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	raw, err := call.Argument(2).Export() | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.UndefinedValue() | ||||||
|  | 	} | ||||||
|  | 	jsonraw, err := json.Marshal(raw) | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.UndefinedValue() | ||||||
|  | 	} | ||||||
|  | 	var contract compiler.Contract | ||||||
|  | 	err = json.Unmarshal(jsonraw, &contract) | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.UndefinedValue() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	filename, err := call.Argument(3).ToString() | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.UndefinedValue() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	contenthash, err := compiler.ExtractInfo(&contract, filename) | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.UndefinedValue() | ||||||
|  | 	} | ||||||
|  | 	// sender and contract address are passed as hex strings | ||||||
|  | 	codeb := js.xeth.CodeAtBytes(address) | ||||||
|  | 	codehash := common.BytesToHash(crypto.Sha3(codeb)) | ||||||
|  |  | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.UndefinedValue() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	registry := resolver.New(js.xeth) | ||||||
|  |  | ||||||
|  | 	_, err = registry.RegisterContentHash(common.HexToAddress(sender), codehash, contenthash) | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.UndefinedValue() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	v, _ := call.Otto.ToValue(contenthash.Hex()) | ||||||
|  | 	return v | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) registerUrl(call otto.FunctionCall) otto.Value { | ||||||
|  | 	if len(call.ArgumentList) != 3 { | ||||||
|  | 		fmt.Println("requires 3 arguments: admin.contractInfo.register(fromaddress, contenthash, filename)") | ||||||
|  | 		return otto.FalseValue() | ||||||
|  | 	} | ||||||
|  | 	sender, err := call.Argument(0).ToString() | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.FalseValue() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	contenthash, err := call.Argument(1).ToString() | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.FalseValue() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	url, err := call.Argument(2).ToString() | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.FalseValue() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	registry := resolver.New(js.xeth) | ||||||
|  |  | ||||||
|  | 	_, err = registry.RegisterUrl(common.HexToAddress(sender), common.HexToHash(contenthash), url) | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.FalseValue() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return otto.TrueValue() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) getContractInfo(call otto.FunctionCall) otto.Value { | ||||||
|  | 	if len(call.ArgumentList) != 1 { | ||||||
|  | 		fmt.Println("requires 1 argument: admin.contractInfo.register(contractaddress)") | ||||||
|  | 		return otto.FalseValue() | ||||||
|  | 	} | ||||||
|  | 	addr, err := call.Argument(0).ToString() | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.FalseValue() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	infoDoc, err := natspec.FetchDocsForContract(addr, js.xeth, ds) | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.UndefinedValue() | ||||||
|  | 	} | ||||||
|  | 	var info compiler.ContractInfo | ||||||
|  | 	err = json.Unmarshal(infoDoc, &info) | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.UndefinedValue() | ||||||
|  | 	} | ||||||
|  | 	v, _ := call.Otto.ToValue(info) | ||||||
|  | 	return v | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) startNatSpec(call otto.FunctionCall) otto.Value { | ||||||
|  | 	js.ethereum.NatSpec = true | ||||||
|  | 	return otto.TrueValue() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) stopNatSpec(call otto.FunctionCall) otto.Value { | ||||||
|  | 	js.ethereum.NatSpec = false | ||||||
|  | 	return otto.TrueValue() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (js *jsre) newRegistry(call otto.FunctionCall) otto.Value { | ||||||
|  |  | ||||||
|  | 	if len(call.ArgumentList) != 1 { | ||||||
|  | 		fmt.Println("requires 1 argument: admin.contractInfo.newRegistry(adminaddress)") | ||||||
|  | 		return otto.FalseValue() | ||||||
|  | 	} | ||||||
|  | 	addr, err := call.Argument(0).ToString() | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.FalseValue() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	registry := resolver.New(js.xeth) | ||||||
|  | 	err = registry.CreateContracts(common.HexToAddress(addr)) | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Println(err) | ||||||
|  | 		return otto.FalseValue() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return otto.TrueValue() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // internal transaction type which will allow us to resend transactions  using `eth.resend` | ||||||
|  | type tx struct { | ||||||
|  | 	tx *types.Transaction | ||||||
|  |  | ||||||
|  | 	To       string | ||||||
|  | 	From     string | ||||||
|  | 	Nonce    string | ||||||
|  | 	Value    string | ||||||
|  | 	Data     string | ||||||
|  | 	GasLimit string | ||||||
|  | 	GasPrice string | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func newTx(t *types.Transaction) *tx { | ||||||
|  | 	from, _ := t.From() | ||||||
|  | 	var to string | ||||||
|  | 	if t := t.To(); t != nil { | ||||||
|  | 		to = t.Hex() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return &tx{ | ||||||
|  | 		tx:       t, | ||||||
|  | 		To:       to, | ||||||
|  | 		From:     from.Hex(), | ||||||
|  | 		Value:    t.Amount.String(), | ||||||
|  | 		Nonce:    strconv.Itoa(int(t.Nonce())), | ||||||
|  | 		Data:     "0x" + common.Bytes2Hex(t.Data()), | ||||||
|  | 		GasLimit: t.GasLimit.String(), | ||||||
|  | 		GasPrice: t.GasPrice().String(), | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @@ -7,6 +7,8 @@ import ( | |||||||
| 	"math/rand" | 	"math/rand" | ||||||
| 	"time" | 	"time" | ||||||
|  |  | ||||||
|  | 	"github.com/rcrowley/go-metrics" | ||||||
|  |  | ||||||
| 	"github.com/ethereum/go-ethereum/common" | 	"github.com/ethereum/go-ethereum/common" | ||||||
| 	"github.com/ethereum/go-ethereum/core/types" | 	"github.com/ethereum/go-ethereum/core/types" | ||||||
| 	"github.com/ethereum/go-ethereum/logger" | 	"github.com/ethereum/go-ethereum/logger" | ||||||
| @@ -96,6 +98,11 @@ type Fetcher struct { | |||||||
| 	// Testing hooks | 	// Testing hooks | ||||||
| 	fetchingHook func([]common.Hash) // Method to call upon starting a block fetch | 	fetchingHook func([]common.Hash) // Method to call upon starting a block fetch | ||||||
| 	importedHook func(*types.Block)  // Method to call upon successful block import | 	importedHook func(*types.Block)  // Method to call upon successful block import | ||||||
|  |  | ||||||
|  | 	// Runtime metrics | ||||||
|  | 	announceStats  metrics.Meter | ||||||
|  | 	broadcastStats metrics.Meter | ||||||
|  | 	discardStats   metrics.Meter | ||||||
| } | } | ||||||
|  |  | ||||||
| // New creates a block fetcher to retrieve blocks based on hash announcements. | // New creates a block fetcher to retrieve blocks based on hash announcements. | ||||||
| @@ -118,6 +125,9 @@ func New(getBlock blockRetrievalFn, validateBlock blockValidatorFn, broadcastBlo | |||||||
| 		chainHeight:    chainHeight, | 		chainHeight:    chainHeight, | ||||||
| 		insertChain:    insertChain, | 		insertChain:    insertChain, | ||||||
| 		dropPeer:       dropPeer, | 		dropPeer:       dropPeer, | ||||||
|  | 		announceStats:  metrics.GetOrRegisterMeter("eth/Announced Blocks", metrics.DefaultRegistry), | ||||||
|  | 		broadcastStats: metrics.GetOrRegisterMeter("eth/Propagated Blocks", metrics.DefaultRegistry), | ||||||
|  | 		discardStats:   metrics.GetOrRegisterMeter("eth/Discarded Blocks", metrics.DefaultRegistry), | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -229,6 +239,8 @@ func (f *Fetcher) loop() { | |||||||
|  |  | ||||||
| 		case notification := <-f.notify: | 		case notification := <-f.notify: | ||||||
| 			// A block was announced, make sure the peer isn't DOSing us | 			// A block was announced, make sure the peer isn't DOSing us | ||||||
|  | 			f.announceStats.Mark(1) | ||||||
|  |  | ||||||
| 			count := f.announces[notification.origin] + 1 | 			count := f.announces[notification.origin] + 1 | ||||||
| 			if count > hashLimit { | 			if count > hashLimit { | ||||||
| 				glog.V(logger.Debug).Infof("Peer %s: exceeded outstanding announces (%d)", notification.origin, hashLimit) | 				glog.V(logger.Debug).Infof("Peer %s: exceeded outstanding announces (%d)", notification.origin, hashLimit) | ||||||
| @@ -246,6 +258,7 @@ func (f *Fetcher) loop() { | |||||||
|  |  | ||||||
| 		case op := <-f.inject: | 		case op := <-f.inject: | ||||||
| 			// A direct block insertion was requested, try and fill any pending gaps | 			// A direct block insertion was requested, try and fill any pending gaps | ||||||
|  | 			f.broadcastStats.Mark(1) | ||||||
| 			f.enqueue(op.origin, op.block) | 			f.enqueue(op.origin, op.block) | ||||||
|  |  | ||||||
| 		case hash := <-f.done: | 		case hash := <-f.done: | ||||||
| @@ -364,6 +377,7 @@ func (f *Fetcher) enqueue(peer string, block *types.Block) { | |||||||
| 	// Discard any past or too distant blocks | 	// Discard any past or too distant blocks | ||||||
| 	if dist := int64(block.NumberU64()) - int64(f.chainHeight()); dist < -maxUncleDist || dist > maxQueueDist { | 	if dist := int64(block.NumberU64()) - int64(f.chainHeight()); dist < -maxUncleDist || dist > maxQueueDist { | ||||||
| 		glog.V(logger.Debug).Infof("Peer %s: discarded block #%d [%x], distance %d", peer, block.NumberU64(), hash.Bytes()[:4], dist) | 		glog.V(logger.Debug).Infof("Peer %s: discarded block #%d [%x], distance %d", peer, block.NumberU64(), hash.Bytes()[:4], dist) | ||||||
|  | 		f.discardStats.Mark(1) | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| 	// Schedule the block for future importing | 	// Schedule the block for future importing | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user