2022-09-25 19:01:39 +00:00
|
|
|
/*
|
|
|
|
Copyright 2022 Erigon-Lightclient contributors
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2022-09-24 18:48:56 +00:00
|
|
|
package sentinel
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2022-09-25 00:14:10 +00:00
|
|
|
"time"
|
2022-09-24 18:48:56 +00:00
|
|
|
|
|
|
|
"github.com/ledgerwatch/erigon/cmd/lightclient/clparams"
|
2022-09-29 22:20:09 +00:00
|
|
|
"github.com/ledgerwatch/erigon/cmd/lightclient/fork"
|
|
|
|
"github.com/ledgerwatch/erigon/p2p/enode"
|
2022-09-24 18:48:56 +00:00
|
|
|
"github.com/ledgerwatch/erigon/p2p/enr"
|
|
|
|
"github.com/ledgerwatch/log/v3"
|
2022-09-26 20:34:31 +00:00
|
|
|
"github.com/libp2p/go-libp2p/core/peer"
|
2022-09-24 18:48:56 +00:00
|
|
|
"github.com/multiformats/go-multiaddr"
|
|
|
|
"github.com/pkg/errors"
|
2022-09-29 22:20:09 +00:00
|
|
|
"github.com/prysmaticlabs/go-bitfield"
|
2022-09-24 18:48:56 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func (s *Sentinel) connectWithPeer(ctx context.Context, info peer.AddrInfo) error {
|
|
|
|
if info.ID == s.host.ID() {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if s.peers.IsBadPeer(info.ID) {
|
|
|
|
return errors.New("refused to connect to bad peer")
|
|
|
|
}
|
2022-09-25 18:39:09 +00:00
|
|
|
ctxWithTimeout, cancel := context.WithTimeout(ctx, clparams.MaxDialTimeout)
|
2022-09-24 18:48:56 +00:00
|
|
|
defer cancel()
|
2022-09-25 18:39:09 +00:00
|
|
|
if err := s.host.Connect(ctxWithTimeout, info); err != nil {
|
2022-09-25 00:14:10 +00:00
|
|
|
s.peers.Penalize(info.ID)
|
2022-09-24 18:48:56 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Sentinel) connectWithAllPeers(multiAddrs []multiaddr.Multiaddr) error {
|
|
|
|
addrInfos, err := peer.AddrInfosFromP2pAddrs(multiAddrs...)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2022-09-25 18:39:09 +00:00
|
|
|
for _, peerInfo := range addrInfos {
|
|
|
|
go func(peerInfo peer.AddrInfo) {
|
|
|
|
if err := s.connectWithPeer(s.ctx, peerInfo); err != nil {
|
2022-09-24 18:48:56 +00:00
|
|
|
log.Debug("Could not connect with peer", "err", err)
|
|
|
|
}
|
2022-09-25 18:39:09 +00:00
|
|
|
}(peerInfo)
|
2022-09-24 18:48:56 +00:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
2022-09-25 00:14:10 +00:00
|
|
|
|
|
|
|
func (s *Sentinel) listenForPeers() {
|
|
|
|
iterator := s.listener.RandomNodes()
|
|
|
|
defer iterator.Close()
|
|
|
|
for {
|
2022-09-26 20:34:31 +00:00
|
|
|
|
2022-09-25 00:14:10 +00:00
|
|
|
if s.ctx.Err() != nil {
|
|
|
|
break
|
|
|
|
}
|
2022-09-25 18:39:09 +00:00
|
|
|
if s.HasTooManyPeers() {
|
2022-09-25 00:14:10 +00:00
|
|
|
log.Trace("Not looking for peers, at peer limit")
|
|
|
|
time.Sleep(100 * time.Millisecond)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
exists := iterator.Next()
|
|
|
|
if !exists {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
node := iterator.Node()
|
|
|
|
peerInfo, _, err := convertToAddrInfo(node)
|
|
|
|
if err != nil {
|
|
|
|
log.Error("Could not convert to peer info", "err", err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2022-09-25 18:39:09 +00:00
|
|
|
go func(peerInfo *peer.AddrInfo) {
|
|
|
|
if err := s.connectWithPeer(s.ctx, *peerInfo); err != nil {
|
2022-09-25 00:14:10 +00:00
|
|
|
log.Debug("Could not connect with peer", "err", err)
|
|
|
|
}
|
|
|
|
}(peerInfo)
|
|
|
|
}
|
|
|
|
}
|
2022-09-25 18:39:09 +00:00
|
|
|
|
|
|
|
func (s *Sentinel) connectToBootnodes() error {
|
|
|
|
for i := range s.cfg.DiscoverConfig.Bootnodes {
|
|
|
|
if err := s.cfg.DiscoverConfig.Bootnodes[i].Record().Load(enr.WithEntry("tcp", new(enr.TCP))); err != nil {
|
|
|
|
if !enr.IsNotFound(err) {
|
|
|
|
log.Error("Could not retrieve tcp port")
|
|
|
|
}
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
|
|
|
multiAddresses := convertToMultiAddr(s.cfg.DiscoverConfig.Bootnodes)
|
|
|
|
s.connectWithAllPeers(multiAddresses)
|
|
|
|
return nil
|
|
|
|
}
|
2022-09-29 22:20:09 +00:00
|
|
|
|
|
|
|
func (s *Sentinel) setupENR(
|
|
|
|
node *enode.LocalNode,
|
|
|
|
) (*enode.LocalNode, error) {
|
|
|
|
forkId, err := fork.ComputeForkId(s.cfg.BeaconConfig, s.cfg.GenesisConfig)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
node.Set(enr.WithEntry(s.cfg.NetworkConfig.Eth2key, forkId))
|
|
|
|
node.Set(enr.WithEntry(s.cfg.NetworkConfig.AttSubnetKey, bitfield.NewBitvector64().Bytes()))
|
|
|
|
node.Set(enr.WithEntry(s.cfg.NetworkConfig.SyncCommsSubnetKey, bitfield.Bitvector4{byte(0x00)}.Bytes()))
|
|
|
|
return node, nil
|
|
|
|
}
|