github.com/nnlgsakib/mind-dpos@v0.0.0-20230606105614-f3c8ca06f808/les/handler.go (about)

     1  // Copyright 2016 The go-ethereum Authors
     2  // This file is part of the go-ethereum library.
     3  //
     4  // The go-ethereum library is free software: you can redistribute it and/or modify
     5  // it under the terms of the GNU Lesser General Public License as published by
     6  // the Free Software Foundation, either version 3 of the License, or
     7  // (at your option) any later version.
     8  //
     9  // The go-ethereum library is distributed in the hope that it will be useful,
    10  // but WITHOUT ANY WARRANTY; without even the implied warranty of
    11  // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
    12  // GNU Lesser General Public License for more details.
    13  //
    14  // You should have received a copy of the GNU Lesser General Public License
    15  // along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
    16  
    17  // Package les implements the Light Ethereum Subprotocol.
    18  package les
    19  
    20  import (
    21  	"encoding/binary"
    22  	"encoding/json"
    23  	"errors"
    24  	"fmt"
    25  	"math/big"
    26  	"net"
    27  	"sync"
    28  	"time"
    29  
    30  	"github.com/TTCECO/gttc/common"
    31  	"github.com/TTCECO/gttc/consensus"
    32  	"github.com/TTCECO/gttc/core"
    33  	"github.com/TTCECO/gttc/core/rawdb"
    34  	"github.com/TTCECO/gttc/core/state"
    35  	"github.com/TTCECO/gttc/core/types"
    36  	"github.com/TTCECO/gttc/eth/downloader"
    37  	"github.com/TTCECO/gttc/ethdb"
    38  	"github.com/TTCECO/gttc/event"
    39  	"github.com/TTCECO/gttc/light"
    40  	"github.com/TTCECO/gttc/log"
    41  	"github.com/TTCECO/gttc/p2p"
    42  	"github.com/TTCECO/gttc/p2p/discover"
    43  	"github.com/TTCECO/gttc/p2p/discv5"
    44  	"github.com/TTCECO/gttc/params"
    45  	"github.com/TTCECO/gttc/rlp"
    46  	"github.com/TTCECO/gttc/trie"
    47  )
    48  
    49  const (
    50  	softResponseLimit = 2 * 1024 * 1024 // Target maximum size of returned blocks, headers or node data.
    51  	estHeaderRlpSize  = 500             // Approximate size of an RLP encoded block header
    52  
    53  	ethVersion = 63 // equivalent eth version for the downloader
    54  
    55  	MaxHeaderFetch           = 192 // Amount of block headers to be fetched per retrieval request
    56  	MaxBodyFetch             = 32  // Amount of block bodies to be fetched per retrieval request
    57  	MaxReceiptFetch          = 128 // Amount of transaction receipts to allow fetching per request
    58  	MaxCodeFetch             = 64  // Amount of contract codes to allow fetching per request
    59  	MaxProofsFetch           = 64  // Amount of merkle proofs to be fetched per retrieval request
    60  	MaxHelperTrieProofsFetch = 64  // Amount of merkle proofs to be fetched per retrieval request
    61  	MaxTxSend                = 64  // Amount of transactions to be send per request
    62  	MaxTxStatus              = 256 // Amount of transactions to queried per request
    63  
    64  	disableClientRemovePeer = false
    65  )
    66  
    67  // errIncompatibleConfig is returned if the requested protocols and configs are
    68  // not compatible (low protocol version restrictions and high requirements).
    69  var errIncompatibleConfig = errors.New("incompatible configuration")
    70  
    71  func errResp(code errCode, format string, v ...interface{}) error {
    72  	return fmt.Errorf("%v - %v", code, fmt.Sprintf(format, v...))
    73  }
    74  
    75  type BlockChain interface {
    76  	Config() *params.ChainConfig
    77  	HasHeader(hash common.Hash, number uint64) bool
    78  	GetHeader(hash common.Hash, number uint64) *types.Header
    79  	GetHeaderByHash(hash common.Hash) *types.Header
    80  	CurrentHeader() *types.Header
    81  	GetTd(hash common.Hash, number uint64) *big.Int
    82  	State() (*state.StateDB, error)
    83  	InsertHeaderChain(chain []*types.Header, checkFreq int) (int, error)
    84  	Rollback(chain []common.Hash)
    85  	GetHeaderByNumber(number uint64) *types.Header
    86  	GetBlockHashesFromHash(hash common.Hash, max uint64) []common.Hash
    87  	Genesis() *types.Block
    88  	SubscribeChainHeadEvent(ch chan<- core.ChainHeadEvent) event.Subscription
    89  }
    90  
    91  type txPool interface {
    92  	AddRemotes(txs []*types.Transaction) []error
    93  	Status(hashes []common.Hash) []core.TxStatus
    94  }
    95  
    96  type ProtocolManager struct {
    97  	lightSync   bool
    98  	txpool      txPool
    99  	txrelay     *LesTxRelay
   100  	networkId   uint64
   101  	chainConfig *params.ChainConfig
   102  	blockchain  BlockChain
   103  	chainDb     ethdb.Database
   104  	odr         *LesOdr
   105  	server      *LesServer
   106  	serverPool  *serverPool
   107  	lesTopic    discv5.Topic
   108  	reqDist     *requestDistributor
   109  	retriever   *retrieveManager
   110  
   111  	downloader *downloader.Downloader
   112  	fetcher    *lightFetcher
   113  	peers      *peerSet
   114  	maxPeers   int
   115  
   116  	SubProtocols []p2p.Protocol
   117  
   118  	eventMux *event.TypeMux
   119  
   120  	// channels for fetcher, syncer, txsyncLoop
   121  	newPeerCh   chan *peer
   122  	quitSync    chan struct{}
   123  	noMorePeers chan struct{}
   124  
   125  	// wait group is used for graceful shutdowns during downloading
   126  	// and processing
   127  	wg *sync.WaitGroup
   128  }
   129  
   130  // NewProtocolManager returns a new ethereum sub protocol manager. The Ethereum sub protocol manages peers capable
   131  // with the ethereum network.
   132  func NewProtocolManager(chainConfig *params.ChainConfig, lightSync bool, protocolVersions []uint, networkId uint64, mux *event.TypeMux, engine consensus.Engine, peers *peerSet, blockchain BlockChain, txpool txPool, chainDb ethdb.Database, odr *LesOdr, txrelay *LesTxRelay, quitSync chan struct{}, wg *sync.WaitGroup) (*ProtocolManager, error) {
   133  	// Create the protocol manager with the base fields
   134  	manager := &ProtocolManager{
   135  		lightSync:   lightSync,
   136  		eventMux:    mux,
   137  		blockchain:  blockchain,
   138  		chainConfig: chainConfig,
   139  		chainDb:     chainDb,
   140  		odr:         odr,
   141  		networkId:   networkId,
   142  		txpool:      txpool,
   143  		txrelay:     txrelay,
   144  		peers:       peers,
   145  		newPeerCh:   make(chan *peer),
   146  		quitSync:    quitSync,
   147  		wg:          wg,
   148  		noMorePeers: make(chan struct{}),
   149  	}
   150  	if odr != nil {
   151  		manager.retriever = odr.retriever
   152  		manager.reqDist = odr.retriever.dist
   153  	}
   154  
   155  	// Initiate a sub-protocol for every implemented version we can handle
   156  	manager.SubProtocols = make([]p2p.Protocol, 0, len(protocolVersions))
   157  	for _, version := range protocolVersions {
   158  		// Compatible, initialize the sub-protocol
   159  		version := version // Closure for the run
   160  		manager.SubProtocols = append(manager.SubProtocols, p2p.Protocol{
   161  			Name:    "les",
   162  			Version: version,
   163  			Length:  ProtocolLengths[version],
   164  			Run: func(p *p2p.Peer, rw p2p.MsgReadWriter) error {
   165  				var entry *poolEntry
   166  				peer := manager.newPeer(int(version), networkId, p, rw)
   167  				if manager.serverPool != nil {
   168  					addr := p.RemoteAddr().(*net.TCPAddr)
   169  					entry = manager.serverPool.connect(peer, addr.IP, uint16(addr.Port))
   170  				}
   171  				peer.poolEntry = entry
   172  				select {
   173  				case manager.newPeerCh <- peer:
   174  					manager.wg.Add(1)
   175  					defer manager.wg.Done()
   176  					err := manager.handle(peer)
   177  					if entry != nil {
   178  						manager.serverPool.disconnect(entry)
   179  					}
   180  					return err
   181  				case <-manager.quitSync:
   182  					if entry != nil {
   183  						manager.serverPool.disconnect(entry)
   184  					}
   185  					return p2p.DiscQuitting
   186  				}
   187  			},
   188  			NodeInfo: func() interface{} {
   189  				return manager.NodeInfo()
   190  			},
   191  			PeerInfo: func(id discover.NodeID) interface{} {
   192  				if p := manager.peers.Peer(fmt.Sprintf("%x", id[:8])); p != nil {
   193  					return p.Info()
   194  				}
   195  				return nil
   196  			},
   197  		})
   198  	}
   199  	if len(manager.SubProtocols) == 0 {
   200  		return nil, errIncompatibleConfig
   201  	}
   202  
   203  	removePeer := manager.removePeer
   204  	if disableClientRemovePeer {
   205  		removePeer = func(id string) {}
   206  	}
   207  
   208  	if lightSync {
   209  		manager.downloader = downloader.New(downloader.LightSync, chainDb, manager.eventMux, nil, blockchain, removePeer)
   210  		manager.peers.notify((*downloaderPeerNotify)(manager))
   211  		manager.fetcher = newLightFetcher(manager)
   212  	}
   213  
   214  	return manager, nil
   215  }
   216  
   217  // removePeer initiates disconnection from a peer by removing it from the peer set
   218  func (pm *ProtocolManager) removePeer(id string) {
   219  	pm.peers.Unregister(id)
   220  }
   221  
   222  func (pm *ProtocolManager) Start(maxPeers int) {
   223  	pm.maxPeers = maxPeers
   224  
   225  	if pm.lightSync {
   226  		go pm.syncer()
   227  	} else {
   228  		go func() {
   229  			for range pm.newPeerCh {
   230  			}
   231  		}()
   232  	}
   233  }
   234  
   235  func (pm *ProtocolManager) Stop() {
   236  	// Showing a log message. During download / process this could actually
   237  	// take between 5 to 10 seconds and therefor feedback is required.
   238  	log.Info("Stopping light TTC protocol")
   239  
   240  	// Quit the sync loop.
   241  	// After this send has completed, no new peers will be accepted.
   242  	pm.noMorePeers <- struct{}{}
   243  
   244  	close(pm.quitSync) // quits syncer, fetcher
   245  
   246  	// Disconnect existing sessions.
   247  	// This also closes the gate for any new registrations on the peer set.
   248  	// sessions which are already established but not added to pm.peers yet
   249  	// will exit when they try to register.
   250  	pm.peers.Close()
   251  
   252  	// Wait for any process action
   253  	pm.wg.Wait()
   254  
   255  	log.Info("Light TTC protocol stopped")
   256  }
   257  
   258  func (pm *ProtocolManager) newPeer(pv int, nv uint64, p *p2p.Peer, rw p2p.MsgReadWriter) *peer {
   259  	return newPeer(pv, nv, p, newMeteredMsgWriter(rw))
   260  }
   261  
   262  // handle is the callback invoked to manage the life cycle of a les peer. When
   263  // this function terminates, the peer is disconnected.
   264  func (pm *ProtocolManager) handle(p *peer) error {
   265  	// Ignore maxPeers if this is a trusted peer
   266  	if pm.peers.Len() >= pm.maxPeers && !p.Peer.Info().Network.Trusted {
   267  		return p2p.DiscTooManyPeers
   268  	}
   269  
   270  	p.Log().Debug("Light TTC peer connected", "name", p.Name())
   271  
   272  	// Execute the LES handshake
   273  	var (
   274  		genesis = pm.blockchain.Genesis()
   275  		head    = pm.blockchain.CurrentHeader()
   276  		hash    = head.Hash()
   277  		number  = head.Number.Uint64()
   278  		td      = pm.blockchain.GetTd(hash, number)
   279  	)
   280  	if err := p.Handshake(td, hash, number, genesis.Hash(), pm.server); err != nil {
   281  		p.Log().Debug("Light TTC handshake failed", "err", err)
   282  		return err
   283  	}
   284  	if rw, ok := p.rw.(*meteredMsgReadWriter); ok {
   285  		rw.Init(p.version)
   286  	}
   287  	// Register the peer locally
   288  	if err := pm.peers.Register(p); err != nil {
   289  		p.Log().Error("Light TTC peer registration failed", "err", err)
   290  		return err
   291  	}
   292  	defer func() {
   293  		if pm.server != nil && pm.server.fcManager != nil && p.fcClient != nil {
   294  			p.fcClient.Remove(pm.server.fcManager)
   295  		}
   296  		pm.removePeer(p.id)
   297  	}()
   298  	// Register the peer in the downloader. If the downloader considers it banned, we disconnect
   299  	if pm.lightSync {
   300  		p.lock.Lock()
   301  		head := p.headInfo
   302  		p.lock.Unlock()
   303  		if pm.fetcher != nil {
   304  			pm.fetcher.announce(p, head)
   305  		}
   306  
   307  		if p.poolEntry != nil {
   308  			pm.serverPool.registered(p.poolEntry)
   309  		}
   310  	}
   311  
   312  	stop := make(chan struct{})
   313  	defer close(stop)
   314  	go func() {
   315  		// new block announce loop
   316  		for {
   317  			select {
   318  			case announce := <-p.announceChn:
   319  				p.SendAnnounce(announce)
   320  			case <-stop:
   321  				return
   322  			}
   323  		}
   324  	}()
   325  
   326  	// main loop. handle incoming messages.
   327  	for {
   328  		if err := pm.handleMsg(p); err != nil {
   329  			p.Log().Debug("Light TTC message handling failed", "err", err)
   330  			return err
   331  		}
   332  	}
   333  }
   334  
   335  var reqList = []uint64{GetBlockHeadersMsg, GetBlockBodiesMsg, GetCodeMsg, GetReceiptsMsg, GetProofsV1Msg, SendTxMsg, SendTxV2Msg, GetTxStatusMsg, GetHeaderProofsMsg, GetProofsV2Msg, GetHelperTrieProofsMsg}
   336  
   337  // handleMsg is invoked whenever an inbound message is received from a remote
   338  // peer. The remote connection is torn down upon returning any error.
   339  func (pm *ProtocolManager) handleMsg(p *peer) error {
   340  	// Read the next message from the remote peer, and ensure it's fully consumed
   341  	msg, err := p.rw.ReadMsg()
   342  	if err != nil {
   343  		return err
   344  	}
   345  	p.Log().Trace("Light TTC message arrived", "code", msg.Code, "bytes", msg.Size)
   346  
   347  	costs := p.fcCosts[msg.Code]
   348  	reject := func(reqCnt, maxCnt uint64) bool {
   349  		if p.fcClient == nil || reqCnt > maxCnt {
   350  			return true
   351  		}
   352  		bufValue, _ := p.fcClient.AcceptRequest()
   353  		cost := costs.baseCost + reqCnt*costs.reqCost
   354  		if cost > pm.server.defParams.BufLimit {
   355  			cost = pm.server.defParams.BufLimit
   356  		}
   357  		if cost > bufValue {
   358  			recharge := time.Duration((cost - bufValue) * 1000000 / pm.server.defParams.MinRecharge)
   359  			p.Log().Error("Request came too early", "recharge", common.PrettyDuration(recharge))
   360  			return true
   361  		}
   362  		return false
   363  	}
   364  
   365  	if msg.Size > ProtocolMaxMsgSize {
   366  		return errResp(ErrMsgTooLarge, "%v > %v", msg.Size, ProtocolMaxMsgSize)
   367  	}
   368  	defer msg.Discard()
   369  
   370  	var deliverMsg *Msg
   371  
   372  	// Handle the message depending on its contents
   373  	switch msg.Code {
   374  	case StatusMsg:
   375  		p.Log().Trace("Received status message")
   376  		// Status messages should never arrive after the handshake
   377  		return errResp(ErrExtraStatusMsg, "uncontrolled status message")
   378  
   379  	// Block header query, collect the requested headers and reply
   380  	case AnnounceMsg:
   381  		p.Log().Trace("Received announce message")
   382  		if p.requestAnnounceType == announceTypeNone {
   383  			return errResp(ErrUnexpectedResponse, "")
   384  		}
   385  
   386  		var req announceData
   387  		if err := msg.Decode(&req); err != nil {
   388  			return errResp(ErrDecode, "%v: %v", msg, err)
   389  		}
   390  
   391  		if p.requestAnnounceType == announceTypeSigned {
   392  			if err := req.checkSignature(p.pubKey); err != nil {
   393  				p.Log().Trace("Invalid announcement signature", "err", err)
   394  				return err
   395  			}
   396  			p.Log().Trace("Valid announcement signature")
   397  		}
   398  
   399  		p.Log().Trace("Announce message content", "number", req.Number, "hash", req.Hash, "td", req.Td, "reorg", req.ReorgDepth)
   400  		if pm.fetcher != nil {
   401  			pm.fetcher.announce(p, &req)
   402  		}
   403  
   404  	case GetBlockHeadersMsg:
   405  		p.Log().Trace("Received block header request")
   406  		// Decode the complex header query
   407  		var req struct {
   408  			ReqID uint64
   409  			Query getBlockHeadersData
   410  		}
   411  		if err := msg.Decode(&req); err != nil {
   412  			return errResp(ErrDecode, "%v: %v", msg, err)
   413  		}
   414  
   415  		query := req.Query
   416  		if reject(query.Amount, MaxHeaderFetch) {
   417  			return errResp(ErrRequestRejected, "")
   418  		}
   419  
   420  		hashMode := query.Origin.Hash != (common.Hash{})
   421  
   422  		// Gather headers until the fetch or network limits is reached
   423  		var (
   424  			bytes   common.StorageSize
   425  			headers []*types.Header
   426  			unknown bool
   427  		)
   428  		for !unknown && len(headers) < int(query.Amount) && bytes < softResponseLimit {
   429  			// Retrieve the next header satisfying the query
   430  			var origin *types.Header
   431  			if hashMode {
   432  				origin = pm.blockchain.GetHeaderByHash(query.Origin.Hash)
   433  			} else {
   434  				origin = pm.blockchain.GetHeaderByNumber(query.Origin.Number)
   435  			}
   436  			if origin == nil {
   437  				break
   438  			}
   439  			number := origin.Number.Uint64()
   440  			headers = append(headers, origin)
   441  			bytes += estHeaderRlpSize
   442  
   443  			// Advance to the next header of the query
   444  			switch {
   445  			case hashMode && query.Reverse:
   446  				// Hash based traversal towards the genesis block
   447  				for i := 0; i < int(query.Skip)+1; i++ {
   448  					if header := pm.blockchain.GetHeader(query.Origin.Hash, number); header != nil {
   449  						query.Origin.Hash = header.ParentHash
   450  						number--
   451  					} else {
   452  						unknown = true
   453  						break
   454  					}
   455  				}
   456  			case hashMode && !query.Reverse:
   457  				// Hash based traversal towards the leaf block
   458  				var (
   459  					current = origin.Number.Uint64()
   460  					next    = current + query.Skip + 1
   461  				)
   462  				if next <= current {
   463  					infos, _ := json.MarshalIndent(p.Peer.Info(), "", "  ")
   464  					p.Log().Warn("GetBlockHeaders skip overflow attack", "current", current, "skip", query.Skip, "next", next, "attacker", infos)
   465  					unknown = true
   466  				} else {
   467  					if header := pm.blockchain.GetHeaderByNumber(next); header != nil {
   468  						if pm.blockchain.GetBlockHashesFromHash(header.Hash(), query.Skip+1)[query.Skip] == query.Origin.Hash {
   469  							query.Origin.Hash = header.Hash()
   470  						} else {
   471  							unknown = true
   472  						}
   473  					} else {
   474  						unknown = true
   475  					}
   476  
   477  				}
   478  			case query.Reverse:
   479  				// Number based traversal towards the genesis block
   480  				if query.Origin.Number >= query.Skip+1 {
   481  					query.Origin.Number -= query.Skip + 1
   482  				} else {
   483  					unknown = true
   484  				}
   485  
   486  			case !query.Reverse:
   487  				// Number based traversal towards the leaf block
   488  				query.Origin.Number += query.Skip + 1
   489  			}
   490  		}
   491  
   492  		bv, rcost := p.fcClient.RequestProcessed(costs.baseCost + query.Amount*costs.reqCost)
   493  		pm.server.fcCostStats.update(msg.Code, query.Amount, rcost)
   494  		return p.SendBlockHeaders(req.ReqID, bv, headers)
   495  
   496  	case BlockHeadersMsg:
   497  		if pm.downloader == nil {
   498  			return errResp(ErrUnexpectedResponse, "")
   499  		}
   500  
   501  		p.Log().Trace("Received block header response message")
   502  		// A batch of headers arrived to one of our previous requests
   503  		var resp struct {
   504  			ReqID, BV uint64
   505  			Headers   []*types.Header
   506  		}
   507  		if err := msg.Decode(&resp); err != nil {
   508  			return errResp(ErrDecode, "msg %v: %v", msg, err)
   509  		}
   510  		p.fcServer.GotReply(resp.ReqID, resp.BV)
   511  		if pm.fetcher != nil && pm.fetcher.requestedID(resp.ReqID) {
   512  			pm.fetcher.deliverHeaders(p, resp.ReqID, resp.Headers)
   513  		} else {
   514  			err := pm.downloader.DeliverHeaders(p.id, resp.Headers)
   515  			if err != nil {
   516  				log.Debug(fmt.Sprint(err))
   517  			}
   518  		}
   519  
   520  	case GetBlockBodiesMsg:
   521  		p.Log().Trace("Received block bodies request")
   522  		// Decode the retrieval message
   523  		var req struct {
   524  			ReqID  uint64
   525  			Hashes []common.Hash
   526  		}
   527  		if err := msg.Decode(&req); err != nil {
   528  			return errResp(ErrDecode, "msg %v: %v", msg, err)
   529  		}
   530  		// Gather blocks until the fetch or network limits is reached
   531  		var (
   532  			bytes  int
   533  			bodies []rlp.RawValue
   534  		)
   535  		reqCnt := len(req.Hashes)
   536  		if reject(uint64(reqCnt), MaxBodyFetch) {
   537  			return errResp(ErrRequestRejected, "")
   538  		}
   539  		for _, hash := range req.Hashes {
   540  			if bytes >= softResponseLimit {
   541  				break
   542  			}
   543  			// Retrieve the requested block body, stopping if enough was found
   544  			if number := rawdb.ReadHeaderNumber(pm.chainDb, hash); number != nil {
   545  				if data := rawdb.ReadBodyRLP(pm.chainDb, hash, *number); len(data) != 0 {
   546  					bodies = append(bodies, data)
   547  					bytes += len(data)
   548  				}
   549  			}
   550  		}
   551  		bv, rcost := p.fcClient.RequestProcessed(costs.baseCost + uint64(reqCnt)*costs.reqCost)
   552  		pm.server.fcCostStats.update(msg.Code, uint64(reqCnt), rcost)
   553  		return p.SendBlockBodiesRLP(req.ReqID, bv, bodies)
   554  
   555  	case BlockBodiesMsg:
   556  		if pm.odr == nil {
   557  			return errResp(ErrUnexpectedResponse, "")
   558  		}
   559  
   560  		p.Log().Trace("Received block bodies response")
   561  		// A batch of block bodies arrived to one of our previous requests
   562  		var resp struct {
   563  			ReqID, BV uint64
   564  			Data      []*types.Body
   565  		}
   566  		if err := msg.Decode(&resp); err != nil {
   567  			return errResp(ErrDecode, "msg %v: %v", msg, err)
   568  		}
   569  		p.fcServer.GotReply(resp.ReqID, resp.BV)
   570  		deliverMsg = &Msg{
   571  			MsgType: MsgBlockBodies,
   572  			ReqID:   resp.ReqID,
   573  			Obj:     resp.Data,
   574  		}
   575  
   576  	case GetCodeMsg:
   577  		p.Log().Trace("Received code request")
   578  		// Decode the retrieval message
   579  		var req struct {
   580  			ReqID uint64
   581  			Reqs  []CodeReq
   582  		}
   583  		if err := msg.Decode(&req); err != nil {
   584  			return errResp(ErrDecode, "msg %v: %v", msg, err)
   585  		}
   586  		// Gather state data until the fetch or network limits is reached
   587  		var (
   588  			bytes int
   589  			data  [][]byte
   590  		)
   591  		reqCnt := len(req.Reqs)
   592  		if reject(uint64(reqCnt), MaxCodeFetch) {
   593  			return errResp(ErrRequestRejected, "")
   594  		}
   595  		for _, req := range req.Reqs {
   596  			// Retrieve the requested state entry, stopping if enough was found
   597  			if number := rawdb.ReadHeaderNumber(pm.chainDb, req.BHash); number != nil {
   598  				if header := rawdb.ReadHeader(pm.chainDb, req.BHash, *number); header != nil {
   599  					statedb, err := pm.blockchain.State()
   600  					if err != nil {
   601  						continue
   602  					}
   603  					account, err := pm.getAccount(statedb, header.Root, common.BytesToHash(req.AccKey))
   604  					if err != nil {
   605  						continue
   606  					}
   607  					code, _ := statedb.Database().TrieDB().Node(common.BytesToHash(account.CodeHash))
   608  
   609  					data = append(data, code)
   610  					if bytes += len(code); bytes >= softResponseLimit {
   611  						break
   612  					}
   613  				}
   614  			}
   615  		}
   616  		bv, rcost := p.fcClient.RequestProcessed(costs.baseCost + uint64(reqCnt)*costs.reqCost)
   617  		pm.server.fcCostStats.update(msg.Code, uint64(reqCnt), rcost)
   618  		return p.SendCode(req.ReqID, bv, data)
   619  
   620  	case CodeMsg:
   621  		if pm.odr == nil {
   622  			return errResp(ErrUnexpectedResponse, "")
   623  		}
   624  
   625  		p.Log().Trace("Received code response")
   626  		// A batch of node state data arrived to one of our previous requests
   627  		var resp struct {
   628  			ReqID, BV uint64
   629  			Data      [][]byte
   630  		}
   631  		if err := msg.Decode(&resp); err != nil {
   632  			return errResp(ErrDecode, "msg %v: %v", msg, err)
   633  		}
   634  		p.fcServer.GotReply(resp.ReqID, resp.BV)
   635  		deliverMsg = &Msg{
   636  			MsgType: MsgCode,
   637  			ReqID:   resp.ReqID,
   638  			Obj:     resp.Data,
   639  		}
   640  
   641  	case GetReceiptsMsg:
   642  		p.Log().Trace("Received receipts request")
   643  		// Decode the retrieval message
   644  		var req struct {
   645  			ReqID  uint64
   646  			Hashes []common.Hash
   647  		}
   648  		if err := msg.Decode(&req); err != nil {
   649  			return errResp(ErrDecode, "msg %v: %v", msg, err)
   650  		}
   651  		// Gather state data until the fetch or network limits is reached
   652  		var (
   653  			bytes    int
   654  			receipts []rlp.RawValue
   655  		)
   656  		reqCnt := len(req.Hashes)
   657  		if reject(uint64(reqCnt), MaxReceiptFetch) {
   658  			return errResp(ErrRequestRejected, "")
   659  		}
   660  		for _, hash := range req.Hashes {
   661  			if bytes >= softResponseLimit {
   662  				break
   663  			}
   664  			// Retrieve the requested block's receipts, skipping if unknown to us
   665  			var results types.Receipts
   666  			if number := rawdb.ReadHeaderNumber(pm.chainDb, hash); number != nil {
   667  				results = rawdb.ReadReceipts(pm.chainDb, hash, *number)
   668  			}
   669  			if results == nil {
   670  				if header := pm.blockchain.GetHeaderByHash(hash); header == nil || header.ReceiptHash != types.EmptyRootHash {
   671  					continue
   672  				}
   673  			}
   674  			// If known, encode and queue for response packet
   675  			if encoded, err := rlp.EncodeToBytes(results); err != nil {
   676  				log.Error("Failed to encode receipt", "err", err)
   677  			} else {
   678  				receipts = append(receipts, encoded)
   679  				bytes += len(encoded)
   680  			}
   681  		}
   682  		bv, rcost := p.fcClient.RequestProcessed(costs.baseCost + uint64(reqCnt)*costs.reqCost)
   683  		pm.server.fcCostStats.update(msg.Code, uint64(reqCnt), rcost)
   684  		return p.SendReceiptsRLP(req.ReqID, bv, receipts)
   685  
   686  	case ReceiptsMsg:
   687  		if pm.odr == nil {
   688  			return errResp(ErrUnexpectedResponse, "")
   689  		}
   690  
   691  		p.Log().Trace("Received receipts response")
   692  		// A batch of receipts arrived to one of our previous requests
   693  		var resp struct {
   694  			ReqID, BV uint64
   695  			Receipts  []types.Receipts
   696  		}
   697  		if err := msg.Decode(&resp); err != nil {
   698  			return errResp(ErrDecode, "msg %v: %v", msg, err)
   699  		}
   700  		p.fcServer.GotReply(resp.ReqID, resp.BV)
   701  		deliverMsg = &Msg{
   702  			MsgType: MsgReceipts,
   703  			ReqID:   resp.ReqID,
   704  			Obj:     resp.Receipts,
   705  		}
   706  
   707  	case GetProofsV1Msg:
   708  		p.Log().Trace("Received proofs request")
   709  		// Decode the retrieval message
   710  		var req struct {
   711  			ReqID uint64
   712  			Reqs  []ProofReq
   713  		}
   714  		if err := msg.Decode(&req); err != nil {
   715  			return errResp(ErrDecode, "msg %v: %v", msg, err)
   716  		}
   717  		// Gather state data until the fetch or network limits is reached
   718  		var (
   719  			bytes  int
   720  			proofs proofsData
   721  		)
   722  		reqCnt := len(req.Reqs)
   723  		if reject(uint64(reqCnt), MaxProofsFetch) {
   724  			return errResp(ErrRequestRejected, "")
   725  		}
   726  		for _, req := range req.Reqs {
   727  			// Retrieve the requested state entry, stopping if enough was found
   728  			if number := rawdb.ReadHeaderNumber(pm.chainDb, req.BHash); number != nil {
   729  				if header := rawdb.ReadHeader(pm.chainDb, req.BHash, *number); header != nil {
   730  					statedb, err := pm.blockchain.State()
   731  					if err != nil {
   732  						continue
   733  					}
   734  					var trie state.Trie
   735  					if len(req.AccKey) > 0 {
   736  						account, err := pm.getAccount(statedb, header.Root, common.BytesToHash(req.AccKey))
   737  						if err != nil {
   738  							continue
   739  						}
   740  						trie, _ = statedb.Database().OpenStorageTrie(common.BytesToHash(req.AccKey), account.Root)
   741  					} else {
   742  						trie, _ = statedb.Database().OpenTrie(header.Root)
   743  					}
   744  					if trie != nil {
   745  						var proof light.NodeList
   746  						trie.Prove(req.Key, 0, &proof)
   747  
   748  						proofs = append(proofs, proof)
   749  						if bytes += proof.DataSize(); bytes >= softResponseLimit {
   750  							break
   751  						}
   752  					}
   753  				}
   754  			}
   755  		}
   756  		bv, rcost := p.fcClient.RequestProcessed(costs.baseCost + uint64(reqCnt)*costs.reqCost)
   757  		pm.server.fcCostStats.update(msg.Code, uint64(reqCnt), rcost)
   758  		return p.SendProofs(req.ReqID, bv, proofs)
   759  
   760  	case GetProofsV2Msg:
   761  		p.Log().Trace("Received les/2 proofs request")
   762  		// Decode the retrieval message
   763  		var req struct {
   764  			ReqID uint64
   765  			Reqs  []ProofReq
   766  		}
   767  		if err := msg.Decode(&req); err != nil {
   768  			return errResp(ErrDecode, "msg %v: %v", msg, err)
   769  		}
   770  		// Gather state data until the fetch or network limits is reached
   771  		var (
   772  			lastBHash common.Hash
   773  			statedb   *state.StateDB
   774  			root      common.Hash
   775  		)
   776  		reqCnt := len(req.Reqs)
   777  		if reject(uint64(reqCnt), MaxProofsFetch) {
   778  			return errResp(ErrRequestRejected, "")
   779  		}
   780  
   781  		nodes := light.NewNodeSet()
   782  
   783  		for _, req := range req.Reqs {
   784  			// Look up the state belonging to the request
   785  			if statedb == nil || req.BHash != lastBHash {
   786  				statedb, root, lastBHash = nil, common.Hash{}, req.BHash
   787  
   788  				if number := rawdb.ReadHeaderNumber(pm.chainDb, req.BHash); number != nil {
   789  					if header := rawdb.ReadHeader(pm.chainDb, req.BHash, *number); header != nil {
   790  						statedb, _ = pm.blockchain.State()
   791  						root = header.Root
   792  					}
   793  				}
   794  			}
   795  			if statedb == nil {
   796  				continue
   797  			}
   798  			// Pull the account or storage trie of the request
   799  			var trie state.Trie
   800  			if len(req.AccKey) > 0 {
   801  				account, err := pm.getAccount(statedb, root, common.BytesToHash(req.AccKey))
   802  				if err != nil {
   803  					continue
   804  				}
   805  				trie, _ = statedb.Database().OpenStorageTrie(common.BytesToHash(req.AccKey), account.Root)
   806  			} else {
   807  				trie, _ = statedb.Database().OpenTrie(root)
   808  			}
   809  			if trie == nil {
   810  				continue
   811  			}
   812  			// Prove the user's request from the account or stroage trie
   813  			trie.Prove(req.Key, req.FromLevel, nodes)
   814  			if nodes.DataSize() >= softResponseLimit {
   815  				break
   816  			}
   817  		}
   818  		bv, rcost := p.fcClient.RequestProcessed(costs.baseCost + uint64(reqCnt)*costs.reqCost)
   819  		pm.server.fcCostStats.update(msg.Code, uint64(reqCnt), rcost)
   820  		return p.SendProofsV2(req.ReqID, bv, nodes.NodeList())
   821  
   822  	case ProofsV1Msg:
   823  		if pm.odr == nil {
   824  			return errResp(ErrUnexpectedResponse, "")
   825  		}
   826  
   827  		p.Log().Trace("Received proofs response")
   828  		// A batch of merkle proofs arrived to one of our previous requests
   829  		var resp struct {
   830  			ReqID, BV uint64
   831  			Data      []light.NodeList
   832  		}
   833  		if err := msg.Decode(&resp); err != nil {
   834  			return errResp(ErrDecode, "msg %v: %v", msg, err)
   835  		}
   836  		p.fcServer.GotReply(resp.ReqID, resp.BV)
   837  		deliverMsg = &Msg{
   838  			MsgType: MsgProofsV1,
   839  			ReqID:   resp.ReqID,
   840  			Obj:     resp.Data,
   841  		}
   842  
   843  	case ProofsV2Msg:
   844  		if pm.odr == nil {
   845  			return errResp(ErrUnexpectedResponse, "")
   846  		}
   847  
   848  		p.Log().Trace("Received les/2 proofs response")
   849  		// A batch of merkle proofs arrived to one of our previous requests
   850  		var resp struct {
   851  			ReqID, BV uint64
   852  			Data      light.NodeList
   853  		}
   854  		if err := msg.Decode(&resp); err != nil {
   855  			return errResp(ErrDecode, "msg %v: %v", msg, err)
   856  		}
   857  		p.fcServer.GotReply(resp.ReqID, resp.BV)
   858  		deliverMsg = &Msg{
   859  			MsgType: MsgProofsV2,
   860  			ReqID:   resp.ReqID,
   861  			Obj:     resp.Data,
   862  		}
   863  
   864  	case GetHeaderProofsMsg:
   865  		p.Log().Trace("Received headers proof request")
   866  		// Decode the retrieval message
   867  		var req struct {
   868  			ReqID uint64
   869  			Reqs  []ChtReq
   870  		}
   871  		if err := msg.Decode(&req); err != nil {
   872  			return errResp(ErrDecode, "msg %v: %v", msg, err)
   873  		}
   874  		// Gather state data until the fetch or network limits is reached
   875  		var (
   876  			bytes  int
   877  			proofs []ChtResp
   878  		)
   879  		reqCnt := len(req.Reqs)
   880  		if reject(uint64(reqCnt), MaxHelperTrieProofsFetch) {
   881  			return errResp(ErrRequestRejected, "")
   882  		}
   883  		trieDb := trie.NewDatabase(ethdb.NewTable(pm.chainDb, light.ChtTablePrefix))
   884  		for _, req := range req.Reqs {
   885  			if header := pm.blockchain.GetHeaderByNumber(req.BlockNum); header != nil {
   886  				sectionHead := rawdb.ReadCanonicalHash(pm.chainDb, req.ChtNum*light.CHTFrequencyServer-1)
   887  				if root := light.GetChtRoot(pm.chainDb, req.ChtNum-1, sectionHead); root != (common.Hash{}) {
   888  					trie, err := trie.New(root, trieDb)
   889  					if err != nil {
   890  						continue
   891  					}
   892  					var encNumber [8]byte
   893  					binary.BigEndian.PutUint64(encNumber[:], req.BlockNum)
   894  
   895  					var proof light.NodeList
   896  					trie.Prove(encNumber[:], 0, &proof)
   897  
   898  					proofs = append(proofs, ChtResp{Header: header, Proof: proof})
   899  					if bytes += proof.DataSize() + estHeaderRlpSize; bytes >= softResponseLimit {
   900  						break
   901  					}
   902  				}
   903  			}
   904  		}
   905  		bv, rcost := p.fcClient.RequestProcessed(costs.baseCost + uint64(reqCnt)*costs.reqCost)
   906  		pm.server.fcCostStats.update(msg.Code, uint64(reqCnt), rcost)
   907  		return p.SendHeaderProofs(req.ReqID, bv, proofs)
   908  
   909  	case GetHelperTrieProofsMsg:
   910  		p.Log().Trace("Received helper trie proof request")
   911  		// Decode the retrieval message
   912  		var req struct {
   913  			ReqID uint64
   914  			Reqs  []HelperTrieReq
   915  		}
   916  		if err := msg.Decode(&req); err != nil {
   917  			return errResp(ErrDecode, "msg %v: %v", msg, err)
   918  		}
   919  		// Gather state data until the fetch or network limits is reached
   920  		var (
   921  			auxBytes int
   922  			auxData  [][]byte
   923  		)
   924  		reqCnt := len(req.Reqs)
   925  		if reject(uint64(reqCnt), MaxHelperTrieProofsFetch) {
   926  			return errResp(ErrRequestRejected, "")
   927  		}
   928  
   929  		var (
   930  			lastIdx  uint64
   931  			lastType uint
   932  			root     common.Hash
   933  			auxTrie  *trie.Trie
   934  		)
   935  		nodes := light.NewNodeSet()
   936  		for _, req := range req.Reqs {
   937  			if auxTrie == nil || req.Type != lastType || req.TrieIdx != lastIdx {
   938  				auxTrie, lastType, lastIdx = nil, req.Type, req.TrieIdx
   939  
   940  				var prefix string
   941  				if root, prefix = pm.getHelperTrie(req.Type, req.TrieIdx); root != (common.Hash{}) {
   942  					auxTrie, _ = trie.New(root, trie.NewDatabase(ethdb.NewTable(pm.chainDb, prefix)))
   943  				}
   944  			}
   945  			if req.AuxReq == auxRoot {
   946  				var data []byte
   947  				if root != (common.Hash{}) {
   948  					data = root[:]
   949  				}
   950  				auxData = append(auxData, data)
   951  				auxBytes += len(data)
   952  			} else {
   953  				if auxTrie != nil {
   954  					auxTrie.Prove(req.Key, req.FromLevel, nodes)
   955  				}
   956  				if req.AuxReq != 0 {
   957  					data := pm.getHelperTrieAuxData(req)
   958  					auxData = append(auxData, data)
   959  					auxBytes += len(data)
   960  				}
   961  			}
   962  			if nodes.DataSize()+auxBytes >= softResponseLimit {
   963  				break
   964  			}
   965  		}
   966  		bv, rcost := p.fcClient.RequestProcessed(costs.baseCost + uint64(reqCnt)*costs.reqCost)
   967  		pm.server.fcCostStats.update(msg.Code, uint64(reqCnt), rcost)
   968  		return p.SendHelperTrieProofs(req.ReqID, bv, HelperTrieResps{Proofs: nodes.NodeList(), AuxData: auxData})
   969  
   970  	case HeaderProofsMsg:
   971  		if pm.odr == nil {
   972  			return errResp(ErrUnexpectedResponse, "")
   973  		}
   974  
   975  		p.Log().Trace("Received headers proof response")
   976  		var resp struct {
   977  			ReqID, BV uint64
   978  			Data      []ChtResp
   979  		}
   980  		if err := msg.Decode(&resp); err != nil {
   981  			return errResp(ErrDecode, "msg %v: %v", msg, err)
   982  		}
   983  		p.fcServer.GotReply(resp.ReqID, resp.BV)
   984  		deliverMsg = &Msg{
   985  			MsgType: MsgHeaderProofs,
   986  			ReqID:   resp.ReqID,
   987  			Obj:     resp.Data,
   988  		}
   989  
   990  	case HelperTrieProofsMsg:
   991  		if pm.odr == nil {
   992  			return errResp(ErrUnexpectedResponse, "")
   993  		}
   994  
   995  		p.Log().Trace("Received helper trie proof response")
   996  		var resp struct {
   997  			ReqID, BV uint64
   998  			Data      HelperTrieResps
   999  		}
  1000  		if err := msg.Decode(&resp); err != nil {
  1001  			return errResp(ErrDecode, "msg %v: %v", msg, err)
  1002  		}
  1003  
  1004  		p.fcServer.GotReply(resp.ReqID, resp.BV)
  1005  		deliverMsg = &Msg{
  1006  			MsgType: MsgHelperTrieProofs,
  1007  			ReqID:   resp.ReqID,
  1008  			Obj:     resp.Data,
  1009  		}
  1010  
  1011  	case SendTxMsg:
  1012  		if pm.txpool == nil {
  1013  			return errResp(ErrRequestRejected, "")
  1014  		}
  1015  		// Transactions arrived, parse all of them and deliver to the pool
  1016  		var txs []*types.Transaction
  1017  		if err := msg.Decode(&txs); err != nil {
  1018  			return errResp(ErrDecode, "msg %v: %v", msg, err)
  1019  		}
  1020  		reqCnt := len(txs)
  1021  		if reject(uint64(reqCnt), MaxTxSend) {
  1022  			return errResp(ErrRequestRejected, "")
  1023  		}
  1024  		pm.txpool.AddRemotes(txs)
  1025  
  1026  		_, rcost := p.fcClient.RequestProcessed(costs.baseCost + uint64(reqCnt)*costs.reqCost)
  1027  		pm.server.fcCostStats.update(msg.Code, uint64(reqCnt), rcost)
  1028  
  1029  	case SendTxV2Msg:
  1030  		if pm.txpool == nil {
  1031  			return errResp(ErrRequestRejected, "")
  1032  		}
  1033  		// Transactions arrived, parse all of them and deliver to the pool
  1034  		var req struct {
  1035  			ReqID uint64
  1036  			Txs   []*types.Transaction
  1037  		}
  1038  		if err := msg.Decode(&req); err != nil {
  1039  			return errResp(ErrDecode, "msg %v: %v", msg, err)
  1040  		}
  1041  		reqCnt := len(req.Txs)
  1042  		if reject(uint64(reqCnt), MaxTxSend) {
  1043  			return errResp(ErrRequestRejected, "")
  1044  		}
  1045  
  1046  		hashes := make([]common.Hash, len(req.Txs))
  1047  		for i, tx := range req.Txs {
  1048  			hashes[i] = tx.Hash()
  1049  		}
  1050  		stats := pm.txStatus(hashes)
  1051  		for i, stat := range stats {
  1052  			if stat.Status == core.TxStatusUnknown {
  1053  				if errs := pm.txpool.AddRemotes([]*types.Transaction{req.Txs[i]}); errs[0] != nil {
  1054  					stats[i].Error = errs[0].Error()
  1055  					continue
  1056  				}
  1057  				stats[i] = pm.txStatus([]common.Hash{hashes[i]})[0]
  1058  			}
  1059  		}
  1060  
  1061  		bv, rcost := p.fcClient.RequestProcessed(costs.baseCost + uint64(reqCnt)*costs.reqCost)
  1062  		pm.server.fcCostStats.update(msg.Code, uint64(reqCnt), rcost)
  1063  
  1064  		return p.SendTxStatus(req.ReqID, bv, stats)
  1065  
  1066  	case GetTxStatusMsg:
  1067  		if pm.txpool == nil {
  1068  			return errResp(ErrUnexpectedResponse, "")
  1069  		}
  1070  		// Transactions arrived, parse all of them and deliver to the pool
  1071  		var req struct {
  1072  			ReqID  uint64
  1073  			Hashes []common.Hash
  1074  		}
  1075  		if err := msg.Decode(&req); err != nil {
  1076  			return errResp(ErrDecode, "msg %v: %v", msg, err)
  1077  		}
  1078  		reqCnt := len(req.Hashes)
  1079  		if reject(uint64(reqCnt), MaxTxStatus) {
  1080  			return errResp(ErrRequestRejected, "")
  1081  		}
  1082  		bv, rcost := p.fcClient.RequestProcessed(costs.baseCost + uint64(reqCnt)*costs.reqCost)
  1083  		pm.server.fcCostStats.update(msg.Code, uint64(reqCnt), rcost)
  1084  
  1085  		return p.SendTxStatus(req.ReqID, bv, pm.txStatus(req.Hashes))
  1086  
  1087  	case TxStatusMsg:
  1088  		if pm.odr == nil {
  1089  			return errResp(ErrUnexpectedResponse, "")
  1090  		}
  1091  
  1092  		p.Log().Trace("Received tx status response")
  1093  		var resp struct {
  1094  			ReqID, BV uint64
  1095  			Status    []txStatus
  1096  		}
  1097  		if err := msg.Decode(&resp); err != nil {
  1098  			return errResp(ErrDecode, "msg %v: %v", msg, err)
  1099  		}
  1100  
  1101  		p.fcServer.GotReply(resp.ReqID, resp.BV)
  1102  
  1103  	default:
  1104  		p.Log().Trace("Received unknown message", "code", msg.Code)
  1105  		return errResp(ErrInvalidMsgCode, "%v", msg.Code)
  1106  	}
  1107  
  1108  	if deliverMsg != nil {
  1109  		err := pm.retriever.deliver(p, deliverMsg)
  1110  		if err != nil {
  1111  			p.responseErrors++
  1112  			if p.responseErrors > maxResponseErrors {
  1113  				return err
  1114  			}
  1115  		}
  1116  	}
  1117  	return nil
  1118  }
  1119  
  1120  // getAccount retrieves an account from the state based at root.
  1121  func (pm *ProtocolManager) getAccount(statedb *state.StateDB, root, hash common.Hash) (state.Account, error) {
  1122  	trie, err := trie.New(root, statedb.Database().TrieDB())
  1123  	if err != nil {
  1124  		return state.Account{}, err
  1125  	}
  1126  	blob, err := trie.TryGet(hash[:])
  1127  	if err != nil {
  1128  		return state.Account{}, err
  1129  	}
  1130  	var account state.Account
  1131  	if err = rlp.DecodeBytes(blob, &account); err != nil {
  1132  		return state.Account{}, err
  1133  	}
  1134  	return account, nil
  1135  }
  1136  
  1137  // getHelperTrie returns the post-processed trie root for the given trie ID and section index
  1138  func (pm *ProtocolManager) getHelperTrie(id uint, idx uint64) (common.Hash, string) {
  1139  	switch id {
  1140  	case htCanonical:
  1141  		sectionHead := rawdb.ReadCanonicalHash(pm.chainDb, (idx+1)*light.CHTFrequencyClient-1)
  1142  		return light.GetChtV2Root(pm.chainDb, idx, sectionHead), light.ChtTablePrefix
  1143  	case htBloomBits:
  1144  		sectionHead := rawdb.ReadCanonicalHash(pm.chainDb, (idx+1)*light.BloomTrieFrequency-1)
  1145  		return light.GetBloomTrieRoot(pm.chainDb, idx, sectionHead), light.BloomTrieTablePrefix
  1146  	}
  1147  	return common.Hash{}, ""
  1148  }
  1149  
  1150  // getHelperTrieAuxData returns requested auxiliary data for the given HelperTrie request
  1151  func (pm *ProtocolManager) getHelperTrieAuxData(req HelperTrieReq) []byte {
  1152  	switch {
  1153  	case req.Type == htCanonical && req.AuxReq == auxHeader && len(req.Key) == 8:
  1154  		blockNum := binary.BigEndian.Uint64(req.Key)
  1155  		hash := rawdb.ReadCanonicalHash(pm.chainDb, blockNum)
  1156  		return rawdb.ReadHeaderRLP(pm.chainDb, hash, blockNum)
  1157  	}
  1158  	return nil
  1159  }
  1160  
  1161  func (pm *ProtocolManager) txStatus(hashes []common.Hash) []txStatus {
  1162  	stats := make([]txStatus, len(hashes))
  1163  	for i, stat := range pm.txpool.Status(hashes) {
  1164  		// Save the status we've got from the transaction pool
  1165  		stats[i].Status = stat
  1166  
  1167  		// If the transaction is unknown to the pool, try looking it up locally
  1168  		if stat == core.TxStatusUnknown {
  1169  			if block, number, index := rawdb.ReadTxLookupEntry(pm.chainDb, hashes[i]); block != (common.Hash{}) {
  1170  				stats[i].Status = core.TxStatusIncluded
  1171  				stats[i].Lookup = &rawdb.TxLookupEntry{BlockHash: block, BlockIndex: number, Index: index}
  1172  			}
  1173  		}
  1174  	}
  1175  	return stats
  1176  }
  1177  
  1178  // NodeInfo represents a short summary of the Ethereum sub-protocol metadata
  1179  // known about the host peer.
  1180  type NodeInfo struct {
  1181  	Network    uint64              `json:"network"`    // Ethereum network ID (1=Frontier, 2=Morden, Ropsten=3, Rinkeby=4)
  1182  	Difficulty *big.Int            `json:"difficulty"` // Total difficulty of the host's blockchain
  1183  	Genesis    common.Hash         `json:"genesis"`    // SHA3 hash of the host's genesis block
  1184  	Config     *params.ChainConfig `json:"config"`     // Chain configuration for the fork rules
  1185  	Head       common.Hash         `json:"head"`       // SHA3 hash of the host's best owned block
  1186  }
  1187  
  1188  // NodeInfo retrieves some protocol metadata about the running host node.
  1189  func (self *ProtocolManager) NodeInfo() *NodeInfo {
  1190  	head := self.blockchain.CurrentHeader()
  1191  	hash := head.Hash()
  1192  
  1193  	return &NodeInfo{
  1194  		Network:    self.networkId,
  1195  		Difficulty: self.blockchain.GetTd(hash, head.Number.Uint64()),
  1196  		Genesis:    self.blockchain.Genesis().Hash(),
  1197  		Config:     self.blockchain.Config(),
  1198  		Head:       hash,
  1199  	}
  1200  }
  1201  
  1202  // downloaderPeerNotify implements peerSetNotify
  1203  type downloaderPeerNotify ProtocolManager
  1204  
  1205  type peerConnection struct {
  1206  	manager *ProtocolManager
  1207  	peer    *peer
  1208  }
  1209  
  1210  func (pc *peerConnection) Head() (common.Hash, *big.Int) {
  1211  	return pc.peer.HeadAndTd()
  1212  }
  1213  
  1214  func (pc *peerConnection) RequestHeadersByHash(origin common.Hash, amount int, skip int, reverse bool) error {
  1215  	reqID := genReqID()
  1216  	rq := &distReq{
  1217  		getCost: func(dp distPeer) uint64 {
  1218  			peer := dp.(*peer)
  1219  			return peer.GetRequestCost(GetBlockHeadersMsg, amount)
  1220  		},
  1221  		canSend: func(dp distPeer) bool {
  1222  			return dp.(*peer) == pc.peer
  1223  		},
  1224  		request: func(dp distPeer) func() {
  1225  			peer := dp.(*peer)
  1226  			cost := peer.GetRequestCost(GetBlockHeadersMsg, amount)
  1227  			peer.fcServer.QueueRequest(reqID, cost)
  1228  			return func() { peer.RequestHeadersByHash(reqID, cost, origin, amount, skip, reverse) }
  1229  		},
  1230  	}
  1231  	_, ok := <-pc.manager.reqDist.queue(rq)
  1232  	if !ok {
  1233  		return ErrNoPeers
  1234  	}
  1235  	return nil
  1236  }
  1237  
  1238  func (pc *peerConnection) RequestHeadersByNumber(origin uint64, amount int, skip int, reverse bool) error {
  1239  	reqID := genReqID()
  1240  	rq := &distReq{
  1241  		getCost: func(dp distPeer) uint64 {
  1242  			peer := dp.(*peer)
  1243  			return peer.GetRequestCost(GetBlockHeadersMsg, amount)
  1244  		},
  1245  		canSend: func(dp distPeer) bool {
  1246  			return dp.(*peer) == pc.peer
  1247  		},
  1248  		request: func(dp distPeer) func() {
  1249  			peer := dp.(*peer)
  1250  			cost := peer.GetRequestCost(GetBlockHeadersMsg, amount)
  1251  			peer.fcServer.QueueRequest(reqID, cost)
  1252  			return func() { peer.RequestHeadersByNumber(reqID, cost, origin, amount, skip, reverse) }
  1253  		},
  1254  	}
  1255  	_, ok := <-pc.manager.reqDist.queue(rq)
  1256  	if !ok {
  1257  		return ErrNoPeers
  1258  	}
  1259  	return nil
  1260  }
  1261  
  1262  func (d *downloaderPeerNotify) registerPeer(p *peer) {
  1263  	pm := (*ProtocolManager)(d)
  1264  	pc := &peerConnection{
  1265  		manager: pm,
  1266  		peer:    p,
  1267  	}
  1268  	pm.downloader.RegisterLightPeer(p.id, ethVersion, pc)
  1269  }
  1270  
  1271  func (d *downloaderPeerNotify) unregisterPeer(p *peer) {
  1272  	pm := (*ProtocolManager)(d)
  1273  	pm.downloader.UnregisterPeer(p.id)
  1274  }