debug.go 4.39 KB
Newer Older
Bas van Kervel's avatar
Bas van Kervel committed
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
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 (
17
	DebugApiVersion = "1.0"
Bas van Kervel's avatar
Bas van Kervel committed
18 19 20 21 22
)

var (
	// mapping between methods and handlers
	DebugMapping = map[string]debughandler{
Bas van Kervel's avatar
Bas van Kervel committed
23 24 25 26 27 28
		"debug_dumpBlock":    (*debugApi).DumpBlock,
		"debug_getBlockRlp":  (*debugApi).GetBlockRlp,
		"debug_printBlock":   (*debugApi).PrintBlock,
		"debug_processBlock": (*debugApi).ProcessBlock,
		"debug_seedHash":     (*debugApi).SeedHash,
		"debug_setHead":      (*debugApi).SetHead,
Bas van Kervel's avatar
Bas van Kervel committed
29 30 31 32
	}
)

// debug callback handler
Bas van Kervel's avatar
Bas van Kervel committed
33
type debughandler func(*debugApi, *shared.Request) (interface{}, error)
Bas van Kervel's avatar
Bas van Kervel committed
34 35

// admin api provider
Bas van Kervel's avatar
Bas van Kervel committed
36
type debugApi struct {
Bas van Kervel's avatar
Bas van Kervel committed
37 38 39 40 41 42 43
	xeth     *xeth.XEth
	ethereum *eth.Ethereum
	methods  map[string]debughandler
	codec    codec.ApiCoder
}

// create a new debug api instance
Bas van Kervel's avatar
Bas van Kervel committed
44 45
func NewDebugApi(xeth *xeth.XEth, ethereum *eth.Ethereum, coder codec.Codec) *debugApi {
	return &debugApi{
Bas van Kervel's avatar
Bas van Kervel committed
46 47 48 49 50 51 52 53
		xeth:     xeth,
		ethereum: ethereum,
		methods:  DebugMapping,
		codec:    coder.New(nil),
	}
}

// collection with supported methods
Bas van Kervel's avatar
Bas van Kervel committed
54
func (self *debugApi) Methods() []string {
Bas van Kervel's avatar
Bas van Kervel committed
55 56 57 58 59 60 61 62 63 64
	methods := make([]string, len(self.methods))
	i := 0
	for k := range self.methods {
		methods[i] = k
		i++
	}
	return methods
}

// Execute given request
Bas van Kervel's avatar
Bas van Kervel committed
65
func (self *debugApi) Execute(req *shared.Request) (interface{}, error) {
Bas van Kervel's avatar
Bas van Kervel committed
66 67 68 69 70 71 72
	if callback, ok := self.methods[req.Method]; ok {
		return callback(self, req)
	}

	return nil, &shared.NotImplementedError{req.Method}
}

Bas van Kervel's avatar
Bas van Kervel committed
73
func (self *debugApi) Name() string {
74
	return shared.DebugApiName
Bas van Kervel's avatar
Bas van Kervel committed
75 76
}

77 78 79 80
func (self *debugApi) ApiVersion() string {
	return DebugApiVersion
}

Bas van Kervel's avatar
Bas van Kervel committed
81
func (self *debugApi) PrintBlock(req *shared.Request) (interface{}, error) {
Bas van Kervel's avatar
Bas van Kervel committed
82 83 84 85 86 87 88 89 90
	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
}

Bas van Kervel's avatar
Bas van Kervel committed
91
func (self *debugApi) DumpBlock(req *shared.Request) (interface{}, error) {
Bas van Kervel's avatar
Bas van Kervel committed
92 93 94 95 96 97 98 99 100 101 102 103 104 105 106
	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
	}

107
	return stateDb.RawDump(), nil
Bas van Kervel's avatar
Bas van Kervel committed
108 109
}

Bas van Kervel's avatar
Bas van Kervel committed
110
func (self *debugApi) GetBlockRlp(req *shared.Request) (interface{}, error) {
Bas van Kervel's avatar
Bas van Kervel committed
111 112 113 114 115 116 117 118 119 120 121 122 123
	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
}

Bas van Kervel's avatar
Bas van Kervel committed
124
func (self *debugApi) SetHead(req *shared.Request) (interface{}, error) {
Bas van Kervel's avatar
Bas van Kervel committed
125 126 127 128 129 130 131 132 133 134 135 136 137 138 139
	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
}

Bas van Kervel's avatar
Bas van Kervel committed
140
func (self *debugApi) ProcessBlock(req *shared.Request) (interface{}, error) {
Bas van Kervel's avatar
Bas van Kervel committed
141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161
	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
}

Bas van Kervel's avatar
Bas van Kervel committed
162
func (self *debugApi) SeedHash(req *shared.Request) (interface{}, error) {
Bas van Kervel's avatar
Bas van Kervel committed
163 164 165 166 167 168 169 170 171 172 173
	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
	}
}