mirror of
https://github.com/status-im/go-waku.git
synced 2025-01-27 22:15:38 +00:00
6ece3c483b
- Shards in same cluster are advertised on ENR - Store().Resume() was moved to app layer - NoDefaultWakuTopic() was removed since it's the app that must determine whether it subscribes to the default waku topic - Removed `Publish` from WakuNode (not really used and easy to implement in app layer if needed)
109 lines
2.9 KiB
Go
109 lines
2.9 KiB
Go
package node
|
|
|
|
import (
|
|
"context"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/libp2p/go-libp2p/core/host"
|
|
"github.com/libp2p/go-libp2p/core/network"
|
|
"github.com/libp2p/go-libp2p/core/peer"
|
|
"github.com/libp2p/go-libp2p/p2p/protocol/ping"
|
|
"github.com/waku-org/go-waku/logging"
|
|
"go.uber.org/zap"
|
|
)
|
|
|
|
const maxAllowedPingFailures = 2
|
|
|
|
func disconnectPeers(host host.Host, logger *zap.Logger) {
|
|
logger.Warn("keep alive hasnt been executed recently. Killing all connections to peers")
|
|
for _, p := range host.Network().Peers() {
|
|
err := host.Network().ClosePeer(p)
|
|
if err != nil {
|
|
logger.Warn("while disconnecting peer", zap.Error(err))
|
|
}
|
|
}
|
|
}
|
|
|
|
// startKeepAlive creates a go routine that periodically pings connected peers.
|
|
// This is necessary because TCP connections are automatically closed due to inactivity,
|
|
// and doing a ping will avoid this (with a small bandwidth cost)
|
|
func (w *WakuNode) startKeepAlive(ctx context.Context, t time.Duration) {
|
|
defer w.wg.Done()
|
|
w.log.Info("setting up ping protocol", zap.Duration("duration", t))
|
|
ticker := time.NewTicker(t)
|
|
defer ticker.Stop()
|
|
|
|
lastTimeExecuted := w.timesource.Now()
|
|
|
|
sleepDetectionInterval := int64(t) * 3
|
|
|
|
for {
|
|
select {
|
|
case <-ticker.C:
|
|
difference := w.timesource.Now().UnixNano() - lastTimeExecuted.UnixNano()
|
|
if difference > sleepDetectionInterval {
|
|
disconnectPeers(w.host, w.log)
|
|
lastTimeExecuted = w.timesource.Now()
|
|
continue
|
|
}
|
|
|
|
// Network's peers collection,
|
|
// contains only currently active peers
|
|
pingWg := sync.WaitGroup{}
|
|
peersToPing := w.host.Network().Peers()
|
|
pingWg.Add(len(peersToPing))
|
|
for _, p := range peersToPing {
|
|
if p != w.host.ID() {
|
|
go w.pingPeer(ctx, &pingWg, p)
|
|
}
|
|
}
|
|
pingWg.Wait()
|
|
|
|
lastTimeExecuted = w.timesource.Now()
|
|
case <-ctx.Done():
|
|
w.log.Info("stopping ping protocol")
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
func (w *WakuNode) pingPeer(ctx context.Context, wg *sync.WaitGroup, peer peer.ID) {
|
|
defer wg.Done()
|
|
|
|
ctx, cancel := context.WithTimeout(ctx, 7*time.Second)
|
|
defer cancel()
|
|
|
|
logger := w.log.With(logging.HostID("peer", peer))
|
|
logger.Debug("pinging")
|
|
pr := ping.Ping(ctx, w.host, peer)
|
|
select {
|
|
case res := <-pr:
|
|
if res.Error != nil {
|
|
w.keepAliveMutex.Lock()
|
|
w.keepAliveFails[peer]++
|
|
w.keepAliveMutex.Unlock()
|
|
logger.Debug("could not ping", zap.Error(res.Error))
|
|
} else {
|
|
w.keepAliveMutex.Lock()
|
|
delete(w.keepAliveFails, peer)
|
|
w.keepAliveMutex.Unlock()
|
|
}
|
|
case <-ctx.Done():
|
|
w.keepAliveMutex.Lock()
|
|
w.keepAliveFails[peer]++
|
|
w.keepAliveMutex.Unlock()
|
|
logger.Debug("could not ping (context done)", zap.Error(ctx.Err()))
|
|
}
|
|
|
|
w.keepAliveMutex.Lock()
|
|
if w.keepAliveFails[peer] > maxAllowedPingFailures && w.host.Network().Connectedness(peer) == network.Connected {
|
|
logger.Info("disconnecting peer")
|
|
if err := w.host.Network().ClosePeer(peer); err != nil {
|
|
logger.Debug("closing conn to peer", zap.Error(err))
|
|
}
|
|
w.keepAliveFails[peer] = 0
|
|
}
|
|
w.keepAliveMutex.Unlock()
|
|
}
|