added debug API
This commit is contained in:
parent
faab931ce1
commit
09d0d55fc5
@ -4,9 +4,10 @@ import "github.com/ethereum/go-ethereum/rpc/shared"
|
|||||||
|
|
||||||
const (
|
const (
|
||||||
// List with all API's which are offered over the IPC interface by default
|
// List with all API's which are offered over the IPC interface by default
|
||||||
DefaultIpcApis = "eth,miner,net,web3"
|
DefaultIpcApis = "debug,eth,miner,net,web3"
|
||||||
|
|
||||||
EthApiName = "eth"
|
EthApiName = "eth"
|
||||||
|
DebugApiName = "debug"
|
||||||
MergedApiName = "merged"
|
MergedApiName = "merged"
|
||||||
MinerApiName = "miner"
|
MinerApiName = "miner"
|
||||||
NetApiName = "net"
|
NetApiName = "net"
|
||||||
|
169
rpc/api/debug.go
Normal file
169
rpc/api/debug.go
Normal file
@ -0,0 +1,169 @@
|
|||||||
|
package api
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"github.com/ethereum/ethash"
|
||||||
|
"github.com/ethereum/go-ethereum/core/state"
|
||||||
|
"github.com/ethereum/go-ethereum/core/vm"
|
||||||
|
"github.com/ethereum/go-ethereum/eth"
|
||||||
|
"github.com/ethereum/go-ethereum/rlp"
|
||||||
|
"github.com/ethereum/go-ethereum/rpc/codec"
|
||||||
|
"github.com/ethereum/go-ethereum/rpc/shared"
|
||||||
|
"github.com/ethereum/go-ethereum/xeth"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
DebugVersion = "1.0.0"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
// mapping between methods and handlers
|
||||||
|
DebugMapping = map[string]debughandler{
|
||||||
|
"debug_dumpBlock": (*DebugApi).DumpBlock,
|
||||||
|
"debug_getBlockRlp": (*DebugApi).GetBlockRlp,
|
||||||
|
"debug_printBlock": (*DebugApi).PrintBlock,
|
||||||
|
"debug_processBlock": (*DebugApi).ProcessBlock,
|
||||||
|
"debug_seedHash": (*DebugApi).SeedHash,
|
||||||
|
"debug_setHead": (*DebugApi).SetHead,
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
// debug callback handler
|
||||||
|
type debughandler func(*DebugApi, *shared.Request) (interface{}, error)
|
||||||
|
|
||||||
|
// admin api provider
|
||||||
|
type DebugApi struct {
|
||||||
|
xeth *xeth.XEth
|
||||||
|
ethereum *eth.Ethereum
|
||||||
|
methods map[string]debughandler
|
||||||
|
codec codec.ApiCoder
|
||||||
|
}
|
||||||
|
|
||||||
|
// create a new debug api instance
|
||||||
|
func NewDebugApi(xeth *xeth.XEth, ethereum *eth.Ethereum, coder codec.Codec) *DebugApi {
|
||||||
|
return &DebugApi{
|
||||||
|
xeth: xeth,
|
||||||
|
ethereum: ethereum,
|
||||||
|
methods: DebugMapping,
|
||||||
|
codec: coder.New(nil),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// collection with supported methods
|
||||||
|
func (self *DebugApi) Methods() []string {
|
||||||
|
methods := make([]string, len(self.methods))
|
||||||
|
i := 0
|
||||||
|
for k := range self.methods {
|
||||||
|
methods[i] = k
|
||||||
|
i++
|
||||||
|
}
|
||||||
|
return methods
|
||||||
|
}
|
||||||
|
|
||||||
|
// Execute given request
|
||||||
|
func (self *DebugApi) Execute(req *shared.Request) (interface{}, error) {
|
||||||
|
if callback, ok := self.methods[req.Method]; ok {
|
||||||
|
return callback(self, req)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil, &shared.NotImplementedError{req.Method}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (self *DebugApi) Name() string {
|
||||||
|
return DebugApiName
|
||||||
|
}
|
||||||
|
|
||||||
|
func (self *DebugApi) PrintBlock(req *shared.Request) (interface{}, error) {
|
||||||
|
args := new(BlockNumArg)
|
||||||
|
if err := self.codec.Decode(req.Params, &args); err != nil {
|
||||||
|
return nil, shared.NewDecodeParamError(err.Error())
|
||||||
|
}
|
||||||
|
|
||||||
|
block := self.xeth.EthBlockByNumber(args.BlockNumber)
|
||||||
|
return fmt.Sprintf("%s", block), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (self *DebugApi) DumpBlock(req *shared.Request) (interface{}, error) {
|
||||||
|
args := new(BlockNumArg)
|
||||||
|
if err := self.codec.Decode(req.Params, &args); err != nil {
|
||||||
|
return nil, shared.NewDecodeParamError(err.Error())
|
||||||
|
}
|
||||||
|
|
||||||
|
block := self.xeth.EthBlockByNumber(args.BlockNumber)
|
||||||
|
if block == nil {
|
||||||
|
return nil, fmt.Errorf("block #%d not found", args.BlockNumber)
|
||||||
|
}
|
||||||
|
|
||||||
|
stateDb := state.New(block.Root(), self.ethereum.StateDb())
|
||||||
|
if stateDb == nil {
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return stateDb.Dump(), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (self *DebugApi) GetBlockRlp(req *shared.Request) (interface{}, error) {
|
||||||
|
args := new(BlockNumArg)
|
||||||
|
if err := self.codec.Decode(req.Params, &args); err != nil {
|
||||||
|
return nil, shared.NewDecodeParamError(err.Error())
|
||||||
|
}
|
||||||
|
|
||||||
|
block := self.xeth.EthBlockByNumber(args.BlockNumber)
|
||||||
|
if block == nil {
|
||||||
|
return nil, fmt.Errorf("block #%d not found", args.BlockNumber)
|
||||||
|
}
|
||||||
|
encoded, err := rlp.EncodeToBytes(block)
|
||||||
|
return fmt.Sprintf("%x", encoded), err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (self *DebugApi) SetHead(req *shared.Request) (interface{}, error) {
|
||||||
|
args := new(BlockNumArg)
|
||||||
|
if err := self.codec.Decode(req.Params, &args); err != nil {
|
||||||
|
return nil, shared.NewDecodeParamError(err.Error())
|
||||||
|
}
|
||||||
|
|
||||||
|
block := self.xeth.EthBlockByNumber(args.BlockNumber)
|
||||||
|
if block == nil {
|
||||||
|
return nil, fmt.Errorf("block #%d not found", args.BlockNumber)
|
||||||
|
}
|
||||||
|
|
||||||
|
self.ethereum.ChainManager().SetHead(block)
|
||||||
|
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (self *DebugApi) ProcessBlock(req *shared.Request) (interface{}, error) {
|
||||||
|
args := new(BlockNumArg)
|
||||||
|
if err := self.codec.Decode(req.Params, &args); err != nil {
|
||||||
|
return nil, shared.NewDecodeParamError(err.Error())
|
||||||
|
}
|
||||||
|
|
||||||
|
block := self.xeth.EthBlockByNumber(args.BlockNumber)
|
||||||
|
if block == nil {
|
||||||
|
return nil, fmt.Errorf("block #%d not found", args.BlockNumber)
|
||||||
|
}
|
||||||
|
|
||||||
|
old := vm.Debug
|
||||||
|
defer func() { vm.Debug = old }()
|
||||||
|
vm.Debug = true
|
||||||
|
|
||||||
|
_, err := self.ethereum.BlockProcessor().RetryProcess(block)
|
||||||
|
if err == nil {
|
||||||
|
return true, nil
|
||||||
|
}
|
||||||
|
return false, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (self *DebugApi) SeedHash(req *shared.Request) (interface{}, error) {
|
||||||
|
args := new(BlockNumArg)
|
||||||
|
if err := self.codec.Decode(req.Params, &args); err != nil {
|
||||||
|
return nil, shared.NewDecodeParamError(err.Error())
|
||||||
|
}
|
||||||
|
|
||||||
|
if hash, err := ethash.GetSeedHash(uint64(args.BlockNumber)); err == nil {
|
||||||
|
return fmt.Sprintf("0x%x", hash), nil
|
||||||
|
} else {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
47
rpc/api/debug_args.go
Normal file
47
rpc/api/debug_args.go
Normal file
@ -0,0 +1,47 @@
|
|||||||
|
package api
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
|
"math/big"
|
||||||
|
|
||||||
|
"github.com/ethereum/go-ethereum/rpc/shared"
|
||||||
|
)
|
||||||
|
|
||||||
|
type WaitForBlockArgs struct {
|
||||||
|
MinHeight int
|
||||||
|
Timeout int // in seconds
|
||||||
|
}
|
||||||
|
|
||||||
|
func (args *WaitForBlockArgs) UnmarshalJSON(b []byte) (err error) {
|
||||||
|
var obj []interface{}
|
||||||
|
if err := json.Unmarshal(b, &obj); err != nil {
|
||||||
|
return shared.NewDecodeParamError(err.Error())
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(obj) > 2 {
|
||||||
|
return fmt.Errorf("waitForArgs needs 0, 1, 2 arguments")
|
||||||
|
}
|
||||||
|
|
||||||
|
// default values when not provided
|
||||||
|
args.MinHeight = -1
|
||||||
|
args.Timeout = -1
|
||||||
|
|
||||||
|
if len(obj) >= 1 {
|
||||||
|
var minHeight *big.Int
|
||||||
|
if minHeight, err = numString(obj[0]); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
args.MinHeight = int(minHeight.Int64())
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(obj) >= 2 {
|
||||||
|
timeout, err := numString(obj[1])
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
args.Timeout = int(timeout.Int64())
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
48
rpc/api/debug_js.go
Normal file
48
rpc/api/debug_js.go
Normal file
@ -0,0 +1,48 @@
|
|||||||
|
package api
|
||||||
|
|
||||||
|
const Debug_JS = `
|
||||||
|
web3.extend({
|
||||||
|
property: 'debug',
|
||||||
|
methods:
|
||||||
|
[
|
||||||
|
new web3.extend.Method({
|
||||||
|
name: 'printBlock',
|
||||||
|
call: 'debug_printBlock',
|
||||||
|
params: 1,
|
||||||
|
inputFormatter: [web3.extend.formatters.formatInputInt],
|
||||||
|
outputFormatter: web3.extend.formatters.formatOutputString
|
||||||
|
}),
|
||||||
|
new web3.extend.Method({
|
||||||
|
name: 'getBlockRlp',
|
||||||
|
call: 'debug_getBlockRlp',
|
||||||
|
params: 1,
|
||||||
|
inputFormatter: [web3.extend.formatters.formatInputInt],
|
||||||
|
outputFormatter: web3.extend.formatters.formatOutputString
|
||||||
|
}),
|
||||||
|
new web3.extend.Method({
|
||||||
|
name: 'setHead',
|
||||||
|
call: 'debug_setHead',
|
||||||
|
params: 1,
|
||||||
|
inputFormatter: [web3.extend.formatters.formatInputInt],
|
||||||
|
outputFormatter: web3.extend.formatters.formatOutputBool
|
||||||
|
}),
|
||||||
|
new web3.extend.Method({
|
||||||
|
name: 'processBlock',
|
||||||
|
call: 'debug_processBlock',
|
||||||
|
params: 1,
|
||||||
|
inputFormatter: [web3.extend.formatters.formatInputInt],
|
||||||
|
outputFormatter: function(obj) { return obj; }
|
||||||
|
}),
|
||||||
|
new web3.extend.Method({
|
||||||
|
name: 'seedHash',
|
||||||
|
call: 'debug_seedHash',
|
||||||
|
params: 1,
|
||||||
|
inputFormatter: [web3.extend.formatters.formatInputInt],
|
||||||
|
outputFormatter: web3.extend.formatters.formatOutputString
|
||||||
|
})
|
||||||
|
],
|
||||||
|
properties:
|
||||||
|
[
|
||||||
|
]
|
||||||
|
});
|
||||||
|
`
|
@ -21,6 +21,8 @@ func ParseApiString(apistr string, codec codec.Codec, xeth *xeth.XEth, eth *eth.
|
|||||||
|
|
||||||
for i, name := range names {
|
for i, name := range names {
|
||||||
switch strings.ToLower(strings.TrimSpace(name)) {
|
switch strings.ToLower(strings.TrimSpace(name)) {
|
||||||
|
case DebugApiName:
|
||||||
|
apis[i] = NewDebugApi(xeth, eth, codec)
|
||||||
case EthApiName:
|
case EthApiName:
|
||||||
apis[i] = NewEthApi(xeth, codec)
|
apis[i] = NewEthApi(xeth, codec)
|
||||||
case MinerApiName:
|
case MinerApiName:
|
||||||
@ -39,6 +41,8 @@ func ParseApiString(apistr string, codec codec.Codec, xeth *xeth.XEth, eth *eth.
|
|||||||
|
|
||||||
func Javascript(name string) string {
|
func Javascript(name string) string {
|
||||||
switch strings.ToLower(strings.TrimSpace(name)) {
|
switch strings.ToLower(strings.TrimSpace(name)) {
|
||||||
|
case DebugApiName:
|
||||||
|
return Debug_JS
|
||||||
case MinerApiName:
|
case MinerApiName:
|
||||||
return Miner_JS
|
return Miner_JS
|
||||||
case NetApiName:
|
case NetApiName:
|
||||||
|
@ -3,9 +3,9 @@
|
|||||||
package comms
|
package comms
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"io"
|
"io"
|
||||||
"net"
|
"net"
|
||||||
"os"
|
"os"
|
||||||
|
|
||||||
"github.com/ethereum/go-ethereum/logger"
|
"github.com/ethereum/go-ethereum/logger"
|
||||||
"github.com/ethereum/go-ethereum/logger/glog"
|
"github.com/ethereum/go-ethereum/logger/glog"
|
||||||
@ -71,7 +71,7 @@ func startIpc(cfg IpcConfig, codec codec.Codec, api api.EthereumApi) error {
|
|||||||
os.Remove(cfg.Endpoint)
|
os.Remove(cfg.Endpoint)
|
||||||
}()
|
}()
|
||||||
|
|
||||||
glog.V(logger.Info).Infof("IPC service started (%s)\n", cfg.Endpoint)
|
glog.V(logger.Info).Infof("IPC service started (%s)\n", cfg.Endpoint)
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
@ -3,16 +3,16 @@
|
|||||||
package comms
|
package comms
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"net"
|
"net"
|
||||||
"os"
|
"os"
|
||||||
"sync"
|
"sync"
|
||||||
"syscall"
|
"syscall"
|
||||||
"time"
|
"time"
|
||||||
"unsafe"
|
"unsafe"
|
||||||
|
|
||||||
"github.com/ethereum/go-ethereum/logger"
|
"github.com/ethereum/go-ethereum/logger"
|
||||||
"github.com/ethereum/go-ethereum/logger/glog"
|
"github.com/ethereum/go-ethereum/logger/glog"
|
||||||
"github.com/ethereum/go-ethereum/rpc/api"
|
"github.com/ethereum/go-ethereum/rpc/api"
|
||||||
"github.com/ethereum/go-ethereum/rpc/codec"
|
"github.com/ethereum/go-ethereum/rpc/codec"
|
||||||
@ -135,7 +135,6 @@ func getOverlappedResult(handle syscall.Handle, overlapped *syscall.Overlapped,
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
const (
|
const (
|
||||||
// openMode
|
// openMode
|
||||||
pipe_access_duplex = 0x3
|
pipe_access_duplex = 0x3
|
||||||
@ -636,10 +635,8 @@ func timeout(addr string) PipeError {
|
|||||||
return PipeError{fmt.Sprintf("Pipe IO timed out waiting for '%s'", addr), true}
|
return PipeError{fmt.Sprintf("Pipe IO timed out waiting for '%s'", addr), true}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
func newIpcClient(cfg IpcConfig, codec codec.Codec) (*ipcClient, error) {
|
func newIpcClient(cfg IpcConfig, codec codec.Codec) (*ipcClient, error) {
|
||||||
c, err := Dial(cfg.Endpoint)
|
c, err := Dial(cfg.Endpoint)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@ -648,15 +645,15 @@ func newIpcClient(cfg IpcConfig, codec codec.Codec) (*ipcClient, error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func startIpc(cfg IpcConfig, codec codec.Codec, api api.EthereumApi) error {
|
func startIpc(cfg IpcConfig, codec codec.Codec, api api.EthereumApi) error {
|
||||||
os.Remove(cfg.Endpoint) // in case it still exists from a previous run
|
os.Remove(cfg.Endpoint) // in case it still exists from a previous run
|
||||||
|
|
||||||
l, err := Listen(cfg.Endpoint)
|
l, err := Listen(cfg.Endpoint)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
os.Chmod(cfg.Endpoint, 0600)
|
os.Chmod(cfg.Endpoint, 0600)
|
||||||
|
|
||||||
go func() {
|
go func() {
|
||||||
for {
|
for {
|
||||||
conn, err := l.Accept()
|
conn, err := l.Accept()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -691,9 +688,9 @@ func startIpc(cfg IpcConfig, codec codec.Codec, api api.EthereumApi) error {
|
|||||||
}
|
}
|
||||||
}(conn)
|
}(conn)
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
|
|
||||||
glog.V(logger.Info).Infof("IPC service started (%s)\n", cfg.Endpoint)
|
glog.V(logger.Info).Infof("IPC service started (%s)\n", cfg.Endpoint)
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
11
rpc/jeth.go
11
rpc/jeth.go
@ -4,12 +4,13 @@ import (
|
|||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
|
||||||
"github.com/ethereum/go-ethereum/jsre"
|
|
||||||
"github.com/robertkrimen/otto"
|
|
||||||
"github.com/ethereum/go-ethereum/rpc/comms"
|
|
||||||
"github.com/ethereum/go-ethereum/rpc/codec"
|
|
||||||
"github.com/ethereum/go-ethereum/rpc/shared"
|
|
||||||
"reflect"
|
"reflect"
|
||||||
|
|
||||||
|
"github.com/ethereum/go-ethereum/jsre"
|
||||||
|
"github.com/ethereum/go-ethereum/rpc/codec"
|
||||||
|
"github.com/ethereum/go-ethereum/rpc/comms"
|
||||||
|
"github.com/ethereum/go-ethereum/rpc/shared"
|
||||||
|
"github.com/robertkrimen/otto"
|
||||||
)
|
)
|
||||||
|
|
||||||
type Jeth struct {
|
type Jeth struct {
|
||||||
|
@ -2,6 +2,7 @@ package shared
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
|
|
||||||
"github.com/ethereum/go-ethereum/logger"
|
"github.com/ethereum/go-ethereum/logger"
|
||||||
"github.com/ethereum/go-ethereum/logger/glog"
|
"github.com/ethereum/go-ethereum/logger/glog"
|
||||||
)
|
)
|
||||||
@ -45,15 +46,15 @@ func NewRpcResponse(id interface{}, jsonrpcver string, reply interface{}, err er
|
|||||||
var response interface{}
|
var response interface{}
|
||||||
|
|
||||||
switch err.(type) {
|
switch err.(type) {
|
||||||
case nil:
|
case nil:
|
||||||
response = &SuccessResponse{Jsonrpc: jsonrpcver, Id: id, Result: reply}
|
response = &SuccessResponse{Jsonrpc: jsonrpcver, Id: id, Result: reply}
|
||||||
case *NotImplementedError:
|
case *NotImplementedError:
|
||||||
jsonerr := &ErrorObject{-32601, err.Error()}
|
jsonerr := &ErrorObject{-32601, err.Error()}
|
||||||
response = &ErrorResponse{Jsonrpc: jsonrpcver, Id: id, Error: jsonerr}
|
response = &ErrorResponse{Jsonrpc: jsonrpcver, Id: id, Error: jsonerr}
|
||||||
case *DecodeParamError, *InsufficientParamsError, *ValidationError, *InvalidTypeError:
|
case *DecodeParamError, *InsufficientParamsError, *ValidationError, *InvalidTypeError:
|
||||||
jsonerr := &ErrorObject{-32602, err.Error()}
|
jsonerr := &ErrorObject{-32602, err.Error()}
|
||||||
response = &ErrorResponse{Jsonrpc: jsonrpcver, Id: id, Error: jsonerr}
|
response = &ErrorResponse{Jsonrpc: jsonrpcver, Id: id, Error: jsonerr}
|
||||||
default:
|
default:
|
||||||
jsonerr := &ErrorObject{-32603, err.Error()}
|
jsonerr := &ErrorObject{-32603, err.Error()}
|
||||||
response = &ErrorResponse{Jsonrpc: jsonrpcver, Id: id, Error: jsonerr}
|
response = &ErrorResponse{Jsonrpc: jsonrpcver, Id: id, Error: jsonerr}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user