github.com/letterj/go-ethereum@v1.8.22-0.20190204142846-520024dfd689/les/peer.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 "errors" 22 "fmt" 23 "math/big" 24 "sync" 25 "time" 26 27 "github.com/ethereum/go-ethereum/common" 28 "github.com/ethereum/go-ethereum/core/types" 29 "github.com/ethereum/go-ethereum/eth" 30 "github.com/ethereum/go-ethereum/les/flowcontrol" 31 "github.com/ethereum/go-ethereum/light" 32 "github.com/ethereum/go-ethereum/p2p" 33 "github.com/ethereum/go-ethereum/rlp" 34 ) 35 36 var ( 37 errClosed = errors.New("peer set is closed") 38 errAlreadyRegistered = errors.New("peer is already registered") 39 errNotRegistered = errors.New("peer is not registered") 40 errInvalidHelpTrieReq = errors.New("invalid help trie request") 41 ) 42 43 const maxResponseErrors = 50 // number of invalid responses tolerated (makes the protocol less brittle but still avoids spam) 44 45 const ( 46 announceTypeNone = iota 47 announceTypeSimple 48 announceTypeSigned 49 ) 50 51 type peer struct { 52 *p2p.Peer 53 54 rw p2p.MsgReadWriter 55 56 version int // Protocol version negotiated 57 network uint64 // Network ID being on 58 59 announceType uint64 60 61 id string 62 63 headInfo *announceData 64 lock sync.RWMutex 65 66 announceChn chan announceData 67 sendQueue *execQueue 68 69 poolEntry *poolEntry 70 hasBlock func(common.Hash, uint64, bool) bool 71 responseErrors int 72 73 fcClient *flowcontrol.ClientNode // nil if the peer is server only 74 fcServer *flowcontrol.ServerNode // nil if the peer is client only 75 fcServerParams *flowcontrol.ServerParams 76 fcCosts requestCostTable 77 78 isTrusted bool 79 isOnlyAnnounce bool 80 } 81 82 func newPeer(version int, network uint64, isTrusted bool, p *p2p.Peer, rw p2p.MsgReadWriter) *peer { 83 id := p.ID() 84 85 return &peer{ 86 Peer: p, 87 rw: rw, 88 version: version, 89 network: network, 90 id: fmt.Sprintf("%x", id[:8]), 91 announceChn: make(chan announceData, 20), 92 isTrusted: isTrusted, 93 } 94 } 95 96 func (p *peer) canQueue() bool { 97 return p.sendQueue.canQueue() 98 } 99 100 func (p *peer) queueSend(f func()) { 101 p.sendQueue.queue(f) 102 } 103 104 // Info gathers and returns a collection of metadata known about a peer. 105 func (p *peer) Info() *eth.PeerInfo { 106 return ð.PeerInfo{ 107 Version: p.version, 108 Difficulty: p.Td(), 109 Head: fmt.Sprintf("%x", p.Head()), 110 } 111 } 112 113 // Head retrieves a copy of the current head (most recent) hash of the peer. 114 func (p *peer) Head() (hash common.Hash) { 115 p.lock.RLock() 116 defer p.lock.RUnlock() 117 118 copy(hash[:], p.headInfo.Hash[:]) 119 return hash 120 } 121 122 func (p *peer) HeadAndTd() (hash common.Hash, td *big.Int) { 123 p.lock.RLock() 124 defer p.lock.RUnlock() 125 126 copy(hash[:], p.headInfo.Hash[:]) 127 return hash, p.headInfo.Td 128 } 129 130 func (p *peer) headBlockInfo() blockInfo { 131 p.lock.RLock() 132 defer p.lock.RUnlock() 133 134 return blockInfo{Hash: p.headInfo.Hash, Number: p.headInfo.Number, Td: p.headInfo.Td} 135 } 136 137 // Td retrieves the current total difficulty of a peer. 138 func (p *peer) Td() *big.Int { 139 p.lock.RLock() 140 defer p.lock.RUnlock() 141 142 return new(big.Int).Set(p.headInfo.Td) 143 } 144 145 // waitBefore implements distPeer interface 146 func (p *peer) waitBefore(maxCost uint64) (time.Duration, float64) { 147 return p.fcServer.CanSend(maxCost) 148 } 149 150 func sendRequest(w p2p.MsgWriter, msgcode, reqID, cost uint64, data interface{}) error { 151 type req struct { 152 ReqID uint64 153 Data interface{} 154 } 155 return p2p.Send(w, msgcode, req{reqID, data}) 156 } 157 158 func sendResponse(w p2p.MsgWriter, msgcode, reqID, bv uint64, data interface{}) error { 159 type resp struct { 160 ReqID, BV uint64 161 Data interface{} 162 } 163 return p2p.Send(w, msgcode, resp{reqID, bv, data}) 164 } 165 166 func (p *peer) GetRequestCost(msgcode uint64, amount int) uint64 { 167 p.lock.RLock() 168 defer p.lock.RUnlock() 169 170 cost := p.fcCosts[msgcode].baseCost + p.fcCosts[msgcode].reqCost*uint64(amount) 171 if cost > p.fcServerParams.BufLimit { 172 cost = p.fcServerParams.BufLimit 173 } 174 return cost 175 } 176 177 // HasBlock checks if the peer has a given block 178 func (p *peer) HasBlock(hash common.Hash, number uint64, hasState bool) bool { 179 p.lock.RLock() 180 hasBlock := p.hasBlock 181 p.lock.RUnlock() 182 return hasBlock != nil && hasBlock(hash, number, hasState) 183 } 184 185 // SendAnnounce announces the availability of a number of blocks through 186 // a hash notification. 187 func (p *peer) SendAnnounce(request announceData) error { 188 return p2p.Send(p.rw, AnnounceMsg, request) 189 } 190 191 // SendBlockHeaders sends a batch of block headers to the remote peer. 192 func (p *peer) SendBlockHeaders(reqID, bv uint64, headers []*types.Header) error { 193 return sendResponse(p.rw, BlockHeadersMsg, reqID, bv, headers) 194 } 195 196 // SendBlockBodiesRLP sends a batch of block contents to the remote peer from 197 // an already RLP encoded format. 198 func (p *peer) SendBlockBodiesRLP(reqID, bv uint64, bodies []rlp.RawValue) error { 199 return sendResponse(p.rw, BlockBodiesMsg, reqID, bv, bodies) 200 } 201 202 // SendCodeRLP sends a batch of arbitrary internal data, corresponding to the 203 // hashes requested. 204 func (p *peer) SendCode(reqID, bv uint64, data [][]byte) error { 205 return sendResponse(p.rw, CodeMsg, reqID, bv, data) 206 } 207 208 // SendReceiptsRLP sends a batch of transaction receipts, corresponding to the 209 // ones requested from an already RLP encoded format. 210 func (p *peer) SendReceiptsRLP(reqID, bv uint64, receipts []rlp.RawValue) error { 211 return sendResponse(p.rw, ReceiptsMsg, reqID, bv, receipts) 212 } 213 214 // SendProofs sends a batch of legacy LES/1 merkle proofs, corresponding to the ones requested. 215 func (p *peer) SendProofs(reqID, bv uint64, proofs proofsData) error { 216 return sendResponse(p.rw, ProofsV1Msg, reqID, bv, proofs) 217 } 218 219 // SendProofsV2 sends a batch of merkle proofs, corresponding to the ones requested. 220 func (p *peer) SendProofsV2(reqID, bv uint64, proofs light.NodeList) error { 221 return sendResponse(p.rw, ProofsV2Msg, reqID, bv, proofs) 222 } 223 224 // SendHeaderProofs sends a batch of legacy LES/1 header proofs, corresponding to the ones requested. 225 func (p *peer) SendHeaderProofs(reqID, bv uint64, proofs []ChtResp) error { 226 return sendResponse(p.rw, HeaderProofsMsg, reqID, bv, proofs) 227 } 228 229 // SendHelperTrieProofs sends a batch of HelperTrie proofs, corresponding to the ones requested. 230 func (p *peer) SendHelperTrieProofs(reqID, bv uint64, resp HelperTrieResps) error { 231 return sendResponse(p.rw, HelperTrieProofsMsg, reqID, bv, resp) 232 } 233 234 // SendTxStatus sends a batch of transaction status records, corresponding to the ones requested. 235 func (p *peer) SendTxStatus(reqID, bv uint64, stats []txStatus) error { 236 return sendResponse(p.rw, TxStatusMsg, reqID, bv, stats) 237 } 238 239 // RequestHeadersByHash fetches a batch of blocks' headers corresponding to the 240 // specified header query, based on the hash of an origin block. 241 func (p *peer) RequestHeadersByHash(reqID, cost uint64, origin common.Hash, amount int, skip int, reverse bool) error { 242 p.Log().Debug("Fetching batch of headers", "count", amount, "fromhash", origin, "skip", skip, "reverse", reverse) 243 return sendRequest(p.rw, GetBlockHeadersMsg, reqID, cost, &getBlockHeadersData{Origin: hashOrNumber{Hash: origin}, Amount: uint64(amount), Skip: uint64(skip), Reverse: reverse}) 244 } 245 246 // RequestHeadersByNumber fetches a batch of blocks' headers corresponding to the 247 // specified header query, based on the number of an origin block. 248 func (p *peer) RequestHeadersByNumber(reqID, cost, origin uint64, amount int, skip int, reverse bool) error { 249 p.Log().Debug("Fetching batch of headers", "count", amount, "fromnum", origin, "skip", skip, "reverse", reverse) 250 return sendRequest(p.rw, GetBlockHeadersMsg, reqID, cost, &getBlockHeadersData{Origin: hashOrNumber{Number: origin}, Amount: uint64(amount), Skip: uint64(skip), Reverse: reverse}) 251 } 252 253 // RequestBodies fetches a batch of blocks' bodies corresponding to the hashes 254 // specified. 255 func (p *peer) RequestBodies(reqID, cost uint64, hashes []common.Hash) error { 256 p.Log().Debug("Fetching batch of block bodies", "count", len(hashes)) 257 return sendRequest(p.rw, GetBlockBodiesMsg, reqID, cost, hashes) 258 } 259 260 // RequestCode fetches a batch of arbitrary data from a node's known state 261 // data, corresponding to the specified hashes. 262 func (p *peer) RequestCode(reqID, cost uint64, reqs []CodeReq) error { 263 p.Log().Debug("Fetching batch of codes", "count", len(reqs)) 264 return sendRequest(p.rw, GetCodeMsg, reqID, cost, reqs) 265 } 266 267 // RequestReceipts fetches a batch of transaction receipts from a remote node. 268 func (p *peer) RequestReceipts(reqID, cost uint64, hashes []common.Hash) error { 269 p.Log().Debug("Fetching batch of receipts", "count", len(hashes)) 270 return sendRequest(p.rw, GetReceiptsMsg, reqID, cost, hashes) 271 } 272 273 // RequestProofs fetches a batch of merkle proofs from a remote node. 274 func (p *peer) RequestProofs(reqID, cost uint64, reqs []ProofReq) error { 275 p.Log().Debug("Fetching batch of proofs", "count", len(reqs)) 276 switch p.version { 277 case lpv1: 278 return sendRequest(p.rw, GetProofsV1Msg, reqID, cost, reqs) 279 case lpv2: 280 return sendRequest(p.rw, GetProofsV2Msg, reqID, cost, reqs) 281 default: 282 panic(nil) 283 } 284 } 285 286 // RequestHelperTrieProofs fetches a batch of HelperTrie merkle proofs from a remote node. 287 func (p *peer) RequestHelperTrieProofs(reqID, cost uint64, data interface{}) error { 288 switch p.version { 289 case lpv1: 290 reqs, ok := data.([]ChtReq) 291 if !ok { 292 return errInvalidHelpTrieReq 293 } 294 p.Log().Debug("Fetching batch of header proofs", "count", len(reqs)) 295 return sendRequest(p.rw, GetHeaderProofsMsg, reqID, cost, reqs) 296 case lpv2: 297 reqs, ok := data.([]HelperTrieReq) 298 if !ok { 299 return errInvalidHelpTrieReq 300 } 301 p.Log().Debug("Fetching batch of HelperTrie proofs", "count", len(reqs)) 302 return sendRequest(p.rw, GetHelperTrieProofsMsg, reqID, cost, reqs) 303 default: 304 panic(nil) 305 } 306 } 307 308 // RequestTxStatus fetches a batch of transaction status records from a remote node. 309 func (p *peer) RequestTxStatus(reqID, cost uint64, txHashes []common.Hash) error { 310 p.Log().Debug("Requesting transaction status", "count", len(txHashes)) 311 return sendRequest(p.rw, GetTxStatusMsg, reqID, cost, txHashes) 312 } 313 314 // SendTxStatus sends a batch of transactions to be added to the remote transaction pool. 315 func (p *peer) SendTxs(reqID, cost uint64, txs types.Transactions) error { 316 p.Log().Debug("Fetching batch of transactions", "count", len(txs)) 317 switch p.version { 318 case lpv1: 319 return p2p.Send(p.rw, SendTxMsg, txs) // old message format does not include reqID 320 case lpv2: 321 return sendRequest(p.rw, SendTxV2Msg, reqID, cost, txs) 322 default: 323 panic(nil) 324 } 325 } 326 327 type keyValueEntry struct { 328 Key string 329 Value rlp.RawValue 330 } 331 type keyValueList []keyValueEntry 332 type keyValueMap map[string]rlp.RawValue 333 334 func (l keyValueList) add(key string, val interface{}) keyValueList { 335 var entry keyValueEntry 336 entry.Key = key 337 if val == nil { 338 val = uint64(0) 339 } 340 enc, err := rlp.EncodeToBytes(val) 341 if err == nil { 342 entry.Value = enc 343 } 344 return append(l, entry) 345 } 346 347 func (l keyValueList) decode() keyValueMap { 348 m := make(keyValueMap) 349 for _, entry := range l { 350 m[entry.Key] = entry.Value 351 } 352 return m 353 } 354 355 func (m keyValueMap) get(key string, val interface{}) error { 356 enc, ok := m[key] 357 if !ok { 358 return errResp(ErrMissingKey, "%s", key) 359 } 360 if val == nil { 361 return nil 362 } 363 return rlp.DecodeBytes(enc, val) 364 } 365 366 func (p *peer) sendReceiveHandshake(sendList keyValueList) (keyValueList, error) { 367 // Send out own handshake in a new thread 368 errc := make(chan error, 1) 369 go func() { 370 errc <- p2p.Send(p.rw, StatusMsg, sendList) 371 }() 372 // In the mean time retrieve the remote status message 373 msg, err := p.rw.ReadMsg() 374 if err != nil { 375 return nil, err 376 } 377 if msg.Code != StatusMsg { 378 return nil, errResp(ErrNoStatusMsg, "first msg has code %x (!= %x)", msg.Code, StatusMsg) 379 } 380 if msg.Size > ProtocolMaxMsgSize { 381 return nil, errResp(ErrMsgTooLarge, "%v > %v", msg.Size, ProtocolMaxMsgSize) 382 } 383 // Decode the handshake 384 var recvList keyValueList 385 if err := msg.Decode(&recvList); err != nil { 386 return nil, errResp(ErrDecode, "msg %v: %v", msg, err) 387 } 388 if err := <-errc; err != nil { 389 return nil, err 390 } 391 return recvList, nil 392 } 393 394 // Handshake executes the les protocol handshake, negotiating version number, 395 // network IDs, difficulties, head and genesis blocks. 396 func (p *peer) Handshake(td *big.Int, head common.Hash, headNum uint64, genesis common.Hash, server *LesServer) error { 397 p.lock.Lock() 398 defer p.lock.Unlock() 399 400 var send keyValueList 401 send = send.add("protocolVersion", uint64(p.version)) 402 send = send.add("networkId", p.network) 403 send = send.add("headTd", td) 404 send = send.add("headHash", head) 405 send = send.add("headNum", headNum) 406 send = send.add("genesisHash", genesis) 407 if server != nil { 408 if !server.onlyAnnounce { 409 //only announce server. It sends only announse requests 410 send = send.add("serveHeaders", nil) 411 send = send.add("serveChainSince", uint64(0)) 412 send = send.add("serveStateSince", uint64(0)) 413 send = send.add("txRelay", nil) 414 } 415 send = send.add("flowControl/BL", server.defParams.BufLimit) 416 send = send.add("flowControl/MRR", server.defParams.MinRecharge) 417 list := server.fcCostStats.getCurrentList() 418 send = send.add("flowControl/MRC", list) 419 p.fcCosts = list.decode() 420 } else { 421 //on client node 422 p.announceType = announceTypeSimple 423 if p.isTrusted { 424 p.announceType = announceTypeSigned 425 } 426 send = send.add("announceType", p.announceType) 427 } 428 429 recvList, err := p.sendReceiveHandshake(send) 430 if err != nil { 431 return err 432 } 433 434 recv := recvList.decode() 435 436 var rGenesis, rHash common.Hash 437 var rVersion, rNetwork, rNum uint64 438 var rTd *big.Int 439 440 if err := recv.get("protocolVersion", &rVersion); err != nil { 441 return err 442 } 443 if err := recv.get("networkId", &rNetwork); err != nil { 444 return err 445 } 446 if err := recv.get("headTd", &rTd); err != nil { 447 return err 448 } 449 if err := recv.get("headHash", &rHash); err != nil { 450 return err 451 } 452 if err := recv.get("headNum", &rNum); err != nil { 453 return err 454 } 455 if err := recv.get("genesisHash", &rGenesis); err != nil { 456 return err 457 } 458 459 if rGenesis != genesis { 460 return errResp(ErrGenesisBlockMismatch, "%x (!= %x)", rGenesis[:8], genesis[:8]) 461 } 462 if rNetwork != p.network { 463 return errResp(ErrNetworkIdMismatch, "%d (!= %d)", rNetwork, p.network) 464 } 465 if int(rVersion) != p.version { 466 return errResp(ErrProtocolVersionMismatch, "%d (!= %d)", rVersion, p.version) 467 } 468 469 if server != nil { 470 // until we have a proper peer connectivity API, allow LES connection to other servers 471 /*if recv.get("serveStateSince", nil) == nil { 472 return errResp(ErrUselessPeer, "wanted client, got server") 473 }*/ 474 if recv.get("announceType", &p.announceType) != nil { 475 //set default announceType on server side 476 p.announceType = announceTypeSimple 477 } 478 p.fcClient = flowcontrol.NewClientNode(server.fcManager, server.defParams) 479 } else { 480 //mark OnlyAnnounce server if "serveHeaders", "serveChainSince", "serveStateSince" or "txRelay" fields don't exist 481 if recv.get("serveChainSince", nil) != nil { 482 p.isOnlyAnnounce = true 483 } 484 if recv.get("serveStateSince", nil) != nil { 485 p.isOnlyAnnounce = true 486 } 487 if recv.get("txRelay", nil) != nil { 488 p.isOnlyAnnounce = true 489 } 490 491 if p.isOnlyAnnounce && !p.isTrusted { 492 return errResp(ErrUselessPeer, "peer cannot serve requests") 493 } 494 495 params := &flowcontrol.ServerParams{} 496 if err := recv.get("flowControl/BL", ¶ms.BufLimit); err != nil { 497 return err 498 } 499 if err := recv.get("flowControl/MRR", ¶ms.MinRecharge); err != nil { 500 return err 501 } 502 var MRC RequestCostList 503 if err := recv.get("flowControl/MRC", &MRC); err != nil { 504 return err 505 } 506 p.fcServerParams = params 507 p.fcServer = flowcontrol.NewServerNode(params) 508 p.fcCosts = MRC.decode() 509 } 510 p.headInfo = &announceData{Td: rTd, Hash: rHash, Number: rNum} 511 return nil 512 } 513 514 // String implements fmt.Stringer. 515 func (p *peer) String() string { 516 return fmt.Sprintf("Peer %s [%s]", p.id, 517 fmt.Sprintf("les/%d", p.version), 518 ) 519 } 520 521 // peerSetNotify is a callback interface to notify services about added or 522 // removed peers 523 type peerSetNotify interface { 524 registerPeer(*peer) 525 unregisterPeer(*peer) 526 } 527 528 // peerSet represents the collection of active peers currently participating in 529 // the Light Ethereum sub-protocol. 530 type peerSet struct { 531 peers map[string]*peer 532 lock sync.RWMutex 533 notifyList []peerSetNotify 534 closed bool 535 } 536 537 // newPeerSet creates a new peer set to track the active participants. 538 func newPeerSet() *peerSet { 539 return &peerSet{ 540 peers: make(map[string]*peer), 541 } 542 } 543 544 // notify adds a service to be notified about added or removed peers 545 func (ps *peerSet) notify(n peerSetNotify) { 546 ps.lock.Lock() 547 ps.notifyList = append(ps.notifyList, n) 548 peers := make([]*peer, 0, len(ps.peers)) 549 for _, p := range ps.peers { 550 peers = append(peers, p) 551 } 552 ps.lock.Unlock() 553 554 for _, p := range peers { 555 n.registerPeer(p) 556 } 557 } 558 559 // Register injects a new peer into the working set, or returns an error if the 560 // peer is already known. 561 func (ps *peerSet) Register(p *peer) error { 562 ps.lock.Lock() 563 if ps.closed { 564 ps.lock.Unlock() 565 return errClosed 566 } 567 if _, ok := ps.peers[p.id]; ok { 568 ps.lock.Unlock() 569 return errAlreadyRegistered 570 } 571 ps.peers[p.id] = p 572 p.sendQueue = newExecQueue(100) 573 peers := make([]peerSetNotify, len(ps.notifyList)) 574 copy(peers, ps.notifyList) 575 ps.lock.Unlock() 576 577 for _, n := range peers { 578 n.registerPeer(p) 579 } 580 return nil 581 } 582 583 // Unregister removes a remote peer from the active set, disabling any further 584 // actions to/from that particular entity. It also initiates disconnection at the networking layer. 585 func (ps *peerSet) Unregister(id string) error { 586 ps.lock.Lock() 587 if p, ok := ps.peers[id]; !ok { 588 ps.lock.Unlock() 589 return errNotRegistered 590 } else { 591 delete(ps.peers, id) 592 peers := make([]peerSetNotify, len(ps.notifyList)) 593 copy(peers, ps.notifyList) 594 ps.lock.Unlock() 595 596 for _, n := range peers { 597 n.unregisterPeer(p) 598 } 599 600 p.sendQueue.quit() 601 p.Peer.Disconnect(p2p.DiscUselessPeer) 602 603 return nil 604 } 605 } 606 607 // AllPeerIDs returns a list of all registered peer IDs 608 func (ps *peerSet) AllPeerIDs() []string { 609 ps.lock.RLock() 610 defer ps.lock.RUnlock() 611 612 res := make([]string, len(ps.peers)) 613 idx := 0 614 for id := range ps.peers { 615 res[idx] = id 616 idx++ 617 } 618 return res 619 } 620 621 // Peer retrieves the registered peer with the given id. 622 func (ps *peerSet) Peer(id string) *peer { 623 ps.lock.RLock() 624 defer ps.lock.RUnlock() 625 626 return ps.peers[id] 627 } 628 629 // Len returns if the current number of peers in the set. 630 func (ps *peerSet) Len() int { 631 ps.lock.RLock() 632 defer ps.lock.RUnlock() 633 634 return len(ps.peers) 635 } 636 637 // BestPeer retrieves the known peer with the currently highest total difficulty. 638 func (ps *peerSet) BestPeer() *peer { 639 ps.lock.RLock() 640 defer ps.lock.RUnlock() 641 642 var ( 643 bestPeer *peer 644 bestTd *big.Int 645 ) 646 for _, p := range ps.peers { 647 if td := p.Td(); bestPeer == nil || td.Cmp(bestTd) > 0 { 648 bestPeer, bestTd = p, td 649 } 650 } 651 return bestPeer 652 } 653 654 // AllPeers returns all peers in a list 655 func (ps *peerSet) AllPeers() []*peer { 656 ps.lock.RLock() 657 defer ps.lock.RUnlock() 658 659 list := make([]*peer, len(ps.peers)) 660 i := 0 661 for _, peer := range ps.peers { 662 list[i] = peer 663 i++ 664 } 665 return list 666 } 667 668 // Close disconnects all peers. 669 // No new peers can be registered after Close has returned. 670 func (ps *peerSet) Close() { 671 ps.lock.Lock() 672 defer ps.lock.Unlock() 673 674 for _, p := range ps.peers { 675 p.Disconnect(p2p.DiscQuitting) 676 } 677 ps.closed = true 678 }