mirror of
https://github.com/ethereum/go-ethereum.git
synced 2026-03-11 21:59:05 +00:00
Here I am adding a discv5 nodes source into the p2p dial iterator. It's an improved version of #29533. Unlike discv4, the discv5 random nodes iterator will always provide full ENRs. This means we can apply filtering to the results and will only try dialing nodes which explictly opt into the eth protocol with a matching chain. I have also removed the dial iterator from snap. We don't have an official DNS list for snap anymore, and I doubt anyone else is running one. While we could potentially filter for snap on discv5, there will be very few nodes announcing it, and the extra iterator would just stall the dialer. --------- Co-authored-by: lightclient <lightclient@protonmail.com>
80 lines
2.5 KiB
Go
80 lines
2.5 KiB
Go
// Copyright 2020 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 eth
|
|
|
|
import (
|
|
"github.com/ethereum/go-ethereum/core"
|
|
"github.com/ethereum/go-ethereum/core/forkid"
|
|
"github.com/ethereum/go-ethereum/p2p/enode"
|
|
"github.com/ethereum/go-ethereum/rlp"
|
|
)
|
|
|
|
// enrEntry is the ENR entry which advertises `eth` protocol on the discovery.
|
|
type enrEntry struct {
|
|
ForkID forkid.ID // Fork identifier per EIP-2124
|
|
|
|
// Ignore additional fields (for forward compatibility).
|
|
Rest []rlp.RawValue `rlp:"tail"`
|
|
}
|
|
|
|
// ENRKey implements enr.Entry.
|
|
func (e enrEntry) ENRKey() string {
|
|
return "eth"
|
|
}
|
|
|
|
// StartENRUpdater starts the `eth` ENR updater loop, which listens for chain
|
|
// head events and updates the requested node record whenever a fork is passed.
|
|
func StartENRUpdater(chain *core.BlockChain, ln *enode.LocalNode) {
|
|
var newHead = make(chan core.ChainHeadEvent, 10)
|
|
sub := chain.SubscribeChainHeadEvent(newHead)
|
|
|
|
go func() {
|
|
defer sub.Unsubscribe()
|
|
for {
|
|
select {
|
|
case <-newHead:
|
|
ln.Set(currentENREntry(chain))
|
|
case <-sub.Err():
|
|
// Would be nice to sync with Stop, but there is no
|
|
// good way to do that.
|
|
return
|
|
}
|
|
}
|
|
}()
|
|
}
|
|
|
|
// currentENREntry constructs an `eth` ENR entry based on the current state of the chain.
|
|
func currentENREntry(chain *core.BlockChain) *enrEntry {
|
|
head := chain.CurrentHeader()
|
|
return &enrEntry{
|
|
ForkID: forkid.NewID(chain.Config(), chain.Genesis(), head.Number.Uint64(), head.Time),
|
|
}
|
|
}
|
|
|
|
// NewNodeFilter returns a filtering function that returns whether the provided
|
|
// enode advertises a forkid compatible with the current chain.
|
|
func NewNodeFilter(chain *core.BlockChain) func(*enode.Node) bool {
|
|
filter := forkid.NewFilter(chain)
|
|
return func(n *enode.Node) bool {
|
|
var entry enrEntry
|
|
if err := n.Load(entry); err != nil {
|
|
return false
|
|
}
|
|
err := filter(entry.ForkID)
|
|
return err == nil
|
|
}
|
|
}
|