| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // Copyright 2014 The go-ethereum Authors | 
					
						
							| 
									
										
										
										
											2015-07-22 18:48:40 +02:00
										 |  |  | // This file is part of the go-ethereum library. | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // | 
					
						
							| 
									
										
										
										
											2015-07-23 18:35:11 +02:00
										 |  |  | // The go-ethereum library is free software: you can redistribute it and/or modify | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // it under the terms of the GNU Lesser General Public License as published by | 
					
						
							|  |  |  | // the Free Software Foundation, either version 3 of the License, or | 
					
						
							|  |  |  | // (at your option) any later version. | 
					
						
							|  |  |  | // | 
					
						
							| 
									
										
										
										
											2015-07-22 18:48:40 +02:00
										 |  |  | // The go-ethereum library is distributed in the hope that it will be useful, | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							| 
									
										
										
										
											2015-07-22 18:48:40 +02:00
										 |  |  | // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // GNU Lesser General Public License for more details. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // You should have received a copy of the GNU Lesser General Public License | 
					
						
							| 
									
										
										
										
											2015-07-22 18:48:40 +02:00
										 |  |  | // along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>. | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-31 14:43:14 +01:00
										 |  |  | package state | 
					
						
							| 
									
										
										
										
											2014-08-06 09:53:00 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"encoding/json" | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-16 11:27:38 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/common" | 
					
						
							| 
									
										
										
										
											2019-06-24 16:16:44 +02:00
										 |  |  | 	"github.com/ethereum/go-ethereum/common/hexutil" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/log" | 
					
						
							| 
									
										
										
										
											2016-09-22 21:04:58 +02:00
										 |  |  | 	"github.com/ethereum/go-ethereum/rlp" | 
					
						
							| 
									
										
										
										
											2017-04-18 13:37:10 +02:00
										 |  |  | 	"github.com/ethereum/go-ethereum/trie" | 
					
						
							| 
									
										
										
										
											2014-08-06 09:53:00 +02:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-24 16:16:44 +02:00
										 |  |  | // DumpAccount represents an account in the state | 
					
						
							| 
									
										
										
										
											2016-09-22 21:04:58 +02:00
										 |  |  | type DumpAccount struct { | 
					
						
							| 
									
										
										
										
											2019-06-24 16:16:44 +02:00
										 |  |  | 	Balance   string                 `json:"balance"` | 
					
						
							|  |  |  | 	Nonce     uint64                 `json:"nonce"` | 
					
						
							|  |  |  | 	Root      string                 `json:"root"` | 
					
						
							|  |  |  | 	CodeHash  string                 `json:"codeHash"` | 
					
						
							|  |  |  | 	Code      string                 `json:"code,omitempty"` | 
					
						
							|  |  |  | 	Storage   map[common.Hash]string `json:"storage,omitempty"` | 
					
						
							|  |  |  | 	Address   *common.Address        `json:"address,omitempty"` // Address only present in iterative (line-by-line) mode | 
					
						
							|  |  |  | 	SecureKey hexutil.Bytes          `json:"key,omitempty"`     // If we don't have address, we can output the key | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-06 09:53:00 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-24 16:16:44 +02:00
										 |  |  | // Dump represents the full dump in a collected format, as one large map | 
					
						
							| 
									
										
										
										
											2016-09-22 21:04:58 +02:00
										 |  |  | type Dump struct { | 
					
						
							| 
									
										
										
										
											2019-06-24 16:16:44 +02:00
										 |  |  | 	Root     string                         `json:"root"` | 
					
						
							|  |  |  | 	Accounts map[common.Address]DumpAccount `json:"accounts"` | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // iterativeDump is a 'collector'-implementation which dump output line-by-line iteratively | 
					
						
							|  |  |  | type iterativeDump json.Encoder | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Collector interface which the state trie calls during iteration | 
					
						
							|  |  |  | type collector interface { | 
					
						
							|  |  |  | 	onRoot(common.Hash) | 
					
						
							|  |  |  | 	onAccount(common.Address, DumpAccount) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (self *Dump) onRoot(root common.Hash) { | 
					
						
							|  |  |  | 	self.Root = fmt.Sprintf("%x", root) | 
					
						
							| 
									
										
										
										
											2014-08-06 09:53:00 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-24 16:16:44 +02:00
										 |  |  | func (self *Dump) onAccount(addr common.Address, account DumpAccount) { | 
					
						
							|  |  |  | 	self.Accounts[addr] = account | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (self iterativeDump) onAccount(addr common.Address, account DumpAccount) { | 
					
						
							|  |  |  | 	dumpAccount := &DumpAccount{ | 
					
						
							|  |  |  | 		Balance:   account.Balance, | 
					
						
							|  |  |  | 		Nonce:     account.Nonce, | 
					
						
							|  |  |  | 		Root:      account.Root, | 
					
						
							|  |  |  | 		CodeHash:  account.CodeHash, | 
					
						
							|  |  |  | 		Code:      account.Code, | 
					
						
							|  |  |  | 		Storage:   account.Storage, | 
					
						
							|  |  |  | 		SecureKey: account.SecureKey, | 
					
						
							|  |  |  | 		Address:   nil, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if addr != (common.Address{}) { | 
					
						
							|  |  |  | 		dumpAccount.Address = &addr | 
					
						
							| 
									
										
										
										
											2014-08-06 09:53:00 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-06-24 16:16:44 +02:00
										 |  |  | 	(*json.Encoder)(&self).Encode(dumpAccount) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | func (self iterativeDump) onRoot(root common.Hash) { | 
					
						
							|  |  |  | 	(*json.Encoder)(&self).Encode(struct { | 
					
						
							|  |  |  | 		Root common.Hash `json:"root"` | 
					
						
							|  |  |  | 	}{root}) | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2014-08-06 09:53:00 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-24 16:16:44 +02:00
										 |  |  | func (self *StateDB) dump(c collector, excludeCode, excludeStorage, excludeMissingPreimages bool) { | 
					
						
							|  |  |  | 	emptyAddress := (common.Address{}) | 
					
						
							|  |  |  | 	missingPreimages := 0 | 
					
						
							|  |  |  | 	c.onRoot(self.trie.Hash()) | 
					
						
							| 
									
										
										
										
											2017-04-13 14:41:24 +02:00
										 |  |  | 	it := trie.NewIterator(self.trie.NodeIterator(nil)) | 
					
						
							| 
									
										
										
										
											2014-12-23 18:35:36 +01:00
										 |  |  | 	for it.Next() { | 
					
						
							| 
									
										
										
										
											2016-09-22 21:04:58 +02:00
										 |  |  | 		var data Account | 
					
						
							|  |  |  | 		if err := rlp.DecodeBytes(it.Value, &data); err != nil { | 
					
						
							| 
									
										
										
										
											2016-04-15 11:16:56 +02:00
										 |  |  | 			panic(err) | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-06-24 16:16:44 +02:00
										 |  |  | 		addr := common.BytesToAddress(self.trie.GetKey(it.Key)) | 
					
						
							|  |  |  | 		obj := newObject(nil, addr, data) | 
					
						
							| 
									
										
										
										
											2016-09-22 21:04:58 +02:00
										 |  |  | 		account := DumpAccount{ | 
					
						
							|  |  |  | 			Balance:  data.Balance.String(), | 
					
						
							|  |  |  | 			Nonce:    data.Nonce, | 
					
						
							|  |  |  | 			Root:     common.Bytes2Hex(data.Root[:]), | 
					
						
							|  |  |  | 			CodeHash: common.Bytes2Hex(data.CodeHash), | 
					
						
							| 
									
										
										
										
											2016-04-15 11:16:56 +02:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-06-24 16:16:44 +02:00
										 |  |  | 		if emptyAddress == addr { | 
					
						
							|  |  |  | 			// Preimage missing | 
					
						
							|  |  |  | 			missingPreimages++ | 
					
						
							|  |  |  | 			if excludeMissingPreimages { | 
					
						
							|  |  |  | 				continue | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			account.SecureKey = it.Key | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if !excludeCode { | 
					
						
							|  |  |  | 			account.Code = common.Bytes2Hex(obj.Code(self.db)) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if !excludeStorage { | 
					
						
							|  |  |  | 			account.Storage = make(map[common.Hash]string) | 
					
						
							|  |  |  | 			storageIt := trie.NewIterator(obj.getTrie(self.db).NodeIterator(nil)) | 
					
						
							|  |  |  | 			for storageIt.Next() { | 
					
						
							| 
									
										
										
										
											2019-08-12 22:14:40 +08:00
										 |  |  | 				_, content, _, err := rlp.Split(storageIt.Value) | 
					
						
							|  |  |  | 				if err != nil { | 
					
						
							|  |  |  | 					log.Error("Failed to decode the value returned by iterator", "error", err) | 
					
						
							|  |  |  | 					continue | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				account.Storage[common.BytesToHash(self.trie.GetKey(storageIt.Key))] = common.Bytes2Hex(content) | 
					
						
							| 
									
										
										
										
											2019-06-24 16:16:44 +02:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2014-12-23 18:35:36 +01:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-06-24 16:16:44 +02:00
										 |  |  | 		c.onAccount(addr, account) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if missingPreimages > 0 { | 
					
						
							|  |  |  | 		log.Warn("Dump incomplete due to missing preimages", "missing", missingPreimages) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // RawDump returns the entire state an a single large object | 
					
						
							|  |  |  | func (self *StateDB) RawDump(excludeCode, excludeStorage, excludeMissingPreimages bool) Dump { | 
					
						
							|  |  |  | 	dump := &Dump{ | 
					
						
							|  |  |  | 		Accounts: make(map[common.Address]DumpAccount), | 
					
						
							| 
									
										
										
										
											2014-12-23 18:35:36 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-06-24 16:16:44 +02:00
										 |  |  | 	self.dump(dump, excludeCode, excludeStorage, excludeMissingPreimages) | 
					
						
							|  |  |  | 	return *dump | 
					
						
							| 
									
										
										
										
											2015-02-28 19:15:57 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2014-08-06 09:53:00 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-24 16:16:44 +02:00
										 |  |  | // Dump returns a JSON string representing the entire state as a single json-object | 
					
						
							|  |  |  | func (self *StateDB) Dump(excludeCode, excludeStorage, excludeMissingPreimages bool) []byte { | 
					
						
							|  |  |  | 	dump := self.RawDump(excludeCode, excludeStorage, excludeMissingPreimages) | 
					
						
							|  |  |  | 	json, err := json.MarshalIndent(dump, "", "    ") | 
					
						
							| 
									
										
										
										
											2014-08-06 09:53:00 +02:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		fmt.Println("dump err", err) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-08-17 12:42:32 +02:00
										 |  |  | 	return json | 
					
						
							| 
									
										
										
										
											2014-08-06 09:53:00 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-06-24 16:16:44 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | // IterativeDump dumps out accounts as json-objects, delimited by linebreaks on stdout | 
					
						
							|  |  |  | func (self *StateDB) IterativeDump(excludeCode, excludeStorage, excludeMissingPreimages bool, output *json.Encoder) { | 
					
						
							|  |  |  | 	self.dump(iterativeDump(*output), excludeCode, excludeStorage, excludeMissingPreimages) | 
					
						
							|  |  |  | } |