| 
									
										
										
										
											2015-06-09 09:48:18 +02:00
										 |  |  | 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 ( | 
					
						
							| 
									
										
										
										
											2015-06-10 09:42:14 +02:00
										 |  |  | 	DebugApiVersion = "1.0" | 
					
						
							| 
									
										
										
										
											2015-06-09 09:48:18 +02:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var ( | 
					
						
							|  |  |  | 	// mapping between methods and handlers | 
					
						
							|  |  |  | 	DebugMapping = map[string]debughandler{ | 
					
						
							| 
									
										
										
										
											2015-06-09 16:06:51 +02:00
										 |  |  | 		"debug_dumpBlock":    (*debugApi).DumpBlock, | 
					
						
							|  |  |  | 		"debug_getBlockRlp":  (*debugApi).GetBlockRlp, | 
					
						
							|  |  |  | 		"debug_printBlock":   (*debugApi).PrintBlock, | 
					
						
							|  |  |  | 		"debug_processBlock": (*debugApi).ProcessBlock, | 
					
						
							|  |  |  | 		"debug_seedHash":     (*debugApi).SeedHash, | 
					
						
							|  |  |  | 		"debug_setHead":      (*debugApi).SetHead, | 
					
						
							| 
									
										
										
										
											2015-06-09 09:48:18 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // debug callback handler | 
					
						
							| 
									
										
										
										
											2015-06-09 16:06:51 +02:00
										 |  |  | type debughandler func(*debugApi, *shared.Request) (interface{}, error) | 
					
						
							| 
									
										
										
										
											2015-06-09 09:48:18 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | // admin api provider | 
					
						
							| 
									
										
										
										
											2015-06-09 16:06:51 +02:00
										 |  |  | type debugApi struct { | 
					
						
							| 
									
										
										
										
											2015-06-09 09:48:18 +02:00
										 |  |  | 	xeth     *xeth.XEth | 
					
						
							|  |  |  | 	ethereum *eth.Ethereum | 
					
						
							|  |  |  | 	methods  map[string]debughandler | 
					
						
							|  |  |  | 	codec    codec.ApiCoder | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // create a new debug api instance | 
					
						
							| 
									
										
										
										
											2015-06-09 16:06:51 +02:00
										 |  |  | func NewDebugApi(xeth *xeth.XEth, ethereum *eth.Ethereum, coder codec.Codec) *debugApi { | 
					
						
							|  |  |  | 	return &debugApi{ | 
					
						
							| 
									
										
										
										
											2015-06-09 09:48:18 +02:00
										 |  |  | 		xeth:     xeth, | 
					
						
							|  |  |  | 		ethereum: ethereum, | 
					
						
							|  |  |  | 		methods:  DebugMapping, | 
					
						
							|  |  |  | 		codec:    coder.New(nil), | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // collection with supported methods | 
					
						
							| 
									
										
										
										
											2015-06-09 16:06:51 +02:00
										 |  |  | func (self *debugApi) Methods() []string { | 
					
						
							| 
									
										
										
										
											2015-06-09 09:48:18 +02:00
										 |  |  | 	methods := make([]string, len(self.methods)) | 
					
						
							|  |  |  | 	i := 0 | 
					
						
							|  |  |  | 	for k := range self.methods { | 
					
						
							|  |  |  | 		methods[i] = k | 
					
						
							|  |  |  | 		i++ | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return methods | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Execute given request | 
					
						
							| 
									
										
										
										
											2015-06-09 16:06:51 +02:00
										 |  |  | func (self *debugApi) Execute(req *shared.Request) (interface{}, error) { | 
					
						
							| 
									
										
										
										
											2015-06-09 09:48:18 +02:00
										 |  |  | 	if callback, ok := self.methods[req.Method]; ok { | 
					
						
							|  |  |  | 		return callback(self, req) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return nil, &shared.NotImplementedError{req.Method} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-09 16:06:51 +02:00
										 |  |  | func (self *debugApi) Name() string { | 
					
						
							| 
									
										
										
										
											2015-06-09 09:48:18 +02:00
										 |  |  | 	return DebugApiName | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-10 09:42:14 +02:00
										 |  |  | func (self *debugApi) ApiVersion() string { | 
					
						
							|  |  |  | 	return DebugApiVersion | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-09 16:06:51 +02:00
										 |  |  | func (self *debugApi) PrintBlock(req *shared.Request) (interface{}, error) { | 
					
						
							| 
									
										
										
										
											2015-06-09 09:48:18 +02:00
										 |  |  | 	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 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-09 16:06:51 +02:00
										 |  |  | func (self *debugApi) DumpBlock(req *shared.Request) (interface{}, error) { | 
					
						
							| 
									
										
										
										
											2015-06-09 09:48:18 +02:00
										 |  |  | 	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 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-10 09:42:14 +02:00
										 |  |  | 	return stateDb.RawDump(), nil | 
					
						
							| 
									
										
										
										
											2015-06-09 09:48:18 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-09 16:06:51 +02:00
										 |  |  | func (self *debugApi) GetBlockRlp(req *shared.Request) (interface{}, error) { | 
					
						
							| 
									
										
										
										
											2015-06-09 09:48:18 +02:00
										 |  |  | 	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 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-09 16:06:51 +02:00
										 |  |  | func (self *debugApi) SetHead(req *shared.Request) (interface{}, error) { | 
					
						
							| 
									
										
										
										
											2015-06-09 09:48:18 +02:00
										 |  |  | 	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 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-09 16:06:51 +02:00
										 |  |  | func (self *debugApi) ProcessBlock(req *shared.Request) (interface{}, error) { | 
					
						
							| 
									
										
										
										
											2015-06-09 09:48:18 +02:00
										 |  |  | 	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 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-09 16:06:51 +02:00
										 |  |  | func (self *debugApi) SeedHash(req *shared.Request) (interface{}, error) { | 
					
						
							| 
									
										
										
										
											2015-06-09 09:48:18 +02:00
										 |  |  | 	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 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } |