| 
									
										
										
										
											2015-11-30 13:34:19 +01:00
										 |  |  | // Copyright 2015 The go-ethereum Authors | 
					
						
							|  |  |  | // This file is part of the go-ethereum library. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // The go-ethereum library is free software: you can redistribute it and/or modify | 
					
						
							|  |  |  | // 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. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // The go-ethereum library is distributed in the hope that it will be useful, | 
					
						
							|  |  |  | // but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  | // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | 
					
						
							|  |  |  | // GNU Lesser General Public License for more details. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // You should have received a copy of the GNU Lesser General Public License | 
					
						
							|  |  |  | // along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Package light implements on-demand retrieval capable state and chain objects | 
					
						
							|  |  |  | // for the Ethereum Light Client. | 
					
						
							|  |  |  | package light | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2016-10-14 05:47:09 +02:00
										 |  |  | 	"math/big" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-30 13:34:19 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/common" | 
					
						
							| 
									
										
										
										
											2016-10-14 05:47:09 +02:00
										 |  |  | 	"github.com/ethereum/go-ethereum/core" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/core/types" | 
					
						
							| 
									
										
										
										
											2015-11-30 13:34:19 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/crypto" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/ethdb" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/rlp" | 
					
						
							|  |  |  | 	"golang.org/x/net/context" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-14 05:47:09 +02:00
										 |  |  | // NoOdr is the default context passed to an ODR capable function when the ODR | 
					
						
							|  |  |  | // service is not required. | 
					
						
							|  |  |  | var NoOdr = context.Background() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // OdrBackend is an interface to a backend service that handles ODR retrievals | 
					
						
							| 
									
										
										
										
											2015-11-30 13:34:19 +01:00
										 |  |  | type OdrBackend interface { | 
					
						
							|  |  |  | 	Database() ethdb.Database | 
					
						
							|  |  |  | 	Retrieve(ctx context.Context, req OdrRequest) error | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // OdrRequest is an interface for retrieval requests | 
					
						
							|  |  |  | type OdrRequest interface { | 
					
						
							|  |  |  | 	StoreResult(db ethdb.Database) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-14 05:47:09 +02:00
										 |  |  | // TrieID identifies a state or account storage trie | 
					
						
							|  |  |  | type TrieID struct { | 
					
						
							|  |  |  | 	BlockHash, Root common.Hash | 
					
						
							| 
									
										
										
										
											2016-12-10 06:50:36 +01:00
										 |  |  | 	BlockNumber     uint64 | 
					
						
							| 
									
										
										
										
											2016-10-14 05:47:09 +02:00
										 |  |  | 	AccKey          []byte | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // StateTrieID returns a TrieID for a state trie belonging to a certain block | 
					
						
							|  |  |  | // header. | 
					
						
							|  |  |  | func StateTrieID(header *types.Header) *TrieID { | 
					
						
							|  |  |  | 	return &TrieID{ | 
					
						
							| 
									
										
										
										
											2016-12-10 06:50:36 +01:00
										 |  |  | 		BlockHash:   header.Hash(), | 
					
						
							|  |  |  | 		BlockNumber: header.Number.Uint64(), | 
					
						
							|  |  |  | 		AccKey:      nil, | 
					
						
							|  |  |  | 		Root:        header.Root, | 
					
						
							| 
									
										
										
										
											2016-10-14 05:47:09 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // StorageTrieID returns a TrieID for a contract storage trie at a given account | 
					
						
							|  |  |  | // of a given state trie. It also requires the root hash of the trie for | 
					
						
							|  |  |  | // checking Merkle proofs. | 
					
						
							|  |  |  | func StorageTrieID(state *TrieID, addr common.Address, root common.Hash) *TrieID { | 
					
						
							|  |  |  | 	return &TrieID{ | 
					
						
							| 
									
										
										
										
											2016-12-10 06:50:36 +01:00
										 |  |  | 		BlockHash:   state.BlockHash, | 
					
						
							|  |  |  | 		BlockNumber: state.BlockNumber, | 
					
						
							|  |  |  | 		AccKey:      crypto.Keccak256(addr[:]), | 
					
						
							|  |  |  | 		Root:        root, | 
					
						
							| 
									
										
										
										
											2016-10-14 05:47:09 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-30 13:34:19 +01:00
										 |  |  | // TrieRequest is the ODR request type for state/storage trie entries | 
					
						
							|  |  |  | type TrieRequest struct { | 
					
						
							|  |  |  | 	OdrRequest | 
					
						
							| 
									
										
										
										
											2016-10-14 05:47:09 +02:00
										 |  |  | 	Id    *TrieID | 
					
						
							|  |  |  | 	Key   []byte | 
					
						
							|  |  |  | 	Proof []rlp.RawValue | 
					
						
							| 
									
										
										
										
											2015-11-30 13:34:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // StoreResult stores the retrieved data in local database | 
					
						
							|  |  |  | func (req *TrieRequest) StoreResult(db ethdb.Database) { | 
					
						
							| 
									
										
										
										
											2016-10-14 05:47:09 +02:00
										 |  |  | 	storeProof(db, req.Proof) | 
					
						
							| 
									
										
										
										
											2015-11-30 13:34:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // storeProof stores the new trie nodes obtained from a merkle proof in the database | 
					
						
							|  |  |  | func storeProof(db ethdb.Database, proof []rlp.RawValue) { | 
					
						
							|  |  |  | 	for _, buf := range proof { | 
					
						
							| 
									
										
										
										
											2016-02-21 18:40:27 +00:00
										 |  |  | 		hash := crypto.Keccak256(buf) | 
					
						
							| 
									
										
										
										
											2015-11-30 13:34:19 +01:00
										 |  |  | 		val, _ := db.Get(hash) | 
					
						
							|  |  |  | 		if val == nil { | 
					
						
							|  |  |  | 			db.Put(hash, buf) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-14 05:47:09 +02:00
										 |  |  | // CodeRequest is the ODR request type for retrieving contract code | 
					
						
							|  |  |  | type CodeRequest struct { | 
					
						
							| 
									
										
										
										
											2015-11-30 13:34:19 +01:00
										 |  |  | 	OdrRequest | 
					
						
							| 
									
										
										
										
											2016-10-14 05:47:09 +02:00
										 |  |  | 	Id   *TrieID | 
					
						
							|  |  |  | 	Hash common.Hash | 
					
						
							|  |  |  | 	Data []byte | 
					
						
							| 
									
										
										
										
											2015-11-30 13:34:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-14 05:47:09 +02:00
										 |  |  | // StoreResult stores the retrieved data in local database | 
					
						
							|  |  |  | func (req *CodeRequest) StoreResult(db ethdb.Database) { | 
					
						
							|  |  |  | 	db.Put(req.Hash[:], req.Data) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // BlockRequest is the ODR request type for retrieving block bodies | 
					
						
							|  |  |  | type BlockRequest struct { | 
					
						
							|  |  |  | 	OdrRequest | 
					
						
							|  |  |  | 	Hash   common.Hash | 
					
						
							|  |  |  | 	Number uint64 | 
					
						
							|  |  |  | 	Rlp    []byte | 
					
						
							| 
									
										
										
										
											2015-11-30 13:34:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // StoreResult stores the retrieved data in local database | 
					
						
							| 
									
										
										
										
											2016-10-14 05:47:09 +02:00
										 |  |  | func (req *BlockRequest) StoreResult(db ethdb.Database) { | 
					
						
							|  |  |  | 	core.WriteBodyRLP(db, req.Hash, req.Number, req.Rlp) | 
					
						
							| 
									
										
										
										
											2015-11-30 13:34:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-14 05:47:09 +02:00
										 |  |  | // ReceiptsRequest is the ODR request type for retrieving block bodies | 
					
						
							|  |  |  | type ReceiptsRequest struct { | 
					
						
							|  |  |  | 	OdrRequest | 
					
						
							|  |  |  | 	Hash     common.Hash | 
					
						
							|  |  |  | 	Number   uint64 | 
					
						
							|  |  |  | 	Receipts types.Receipts | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2015-11-30 13:34:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-14 05:47:09 +02:00
										 |  |  | // StoreResult stores the retrieved data in local database | 
					
						
							|  |  |  | func (req *ReceiptsRequest) StoreResult(db ethdb.Database) { | 
					
						
							|  |  |  | 	core.WriteBlockReceipts(db, req.Hash, req.Number, req.Receipts) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // TrieRequest is the ODR request type for state/storage trie entries | 
					
						
							|  |  |  | type ChtRequest struct { | 
					
						
							|  |  |  | 	OdrRequest | 
					
						
							|  |  |  | 	ChtNum, BlockNum uint64 | 
					
						
							|  |  |  | 	ChtRoot          common.Hash | 
					
						
							|  |  |  | 	Header           *types.Header | 
					
						
							|  |  |  | 	Td               *big.Int | 
					
						
							|  |  |  | 	Proof            []rlp.RawValue | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // StoreResult stores the retrieved data in local database | 
					
						
							|  |  |  | func (req *ChtRequest) StoreResult(db ethdb.Database) { | 
					
						
							|  |  |  | 	// if there is a canonical hash, there is a header too | 
					
						
							|  |  |  | 	core.WriteHeader(db, req.Header) | 
					
						
							|  |  |  | 	hash, num := req.Header.Hash(), req.Header.Number.Uint64() | 
					
						
							|  |  |  | 	core.WriteTd(db, hash, num, req.Td) | 
					
						
							|  |  |  | 	core.WriteCanonicalHash(db, hash, num) | 
					
						
							|  |  |  | 	//storeProof(db, req.Proof) | 
					
						
							| 
									
										
										
										
											2015-11-30 13:34:19 +01:00
										 |  |  | } |