mirror of
https://github.com/status-im/go-waku.git
synced 2025-01-12 23:04:45 +00:00
6e47bd1cf0
Filter wrapper API that takes care of managing subscriptions and multiplex them onto a single data channel to be consumed by user. Co-authored-by: Prem Chaitanya Prathi <chaitanyaprem@gmail.com>
270 lines
7.6 KiB
Go
270 lines
7.6 KiB
Go
package peerstore
|
|
|
|
import (
|
|
"errors"
|
|
"sync"
|
|
|
|
"github.com/ethereum/go-ethereum/p2p/enode"
|
|
"github.com/libp2p/go-libp2p/core/network"
|
|
"github.com/libp2p/go-libp2p/core/peer"
|
|
"github.com/libp2p/go-libp2p/core/peerstore"
|
|
"github.com/waku-org/go-waku/waku/v2/protocol"
|
|
"golang.org/x/exp/maps"
|
|
)
|
|
|
|
// Origin is used to determine how the peer is identified,
|
|
// either it is statically added or discovered via one of the discovery protocols
|
|
type Origin int64
|
|
|
|
const (
|
|
Unknown Origin = iota
|
|
Discv5
|
|
Static
|
|
PeerExchange
|
|
DNSDiscovery
|
|
Rendezvous
|
|
PeerManager
|
|
)
|
|
|
|
const peerOrigin = "origin"
|
|
const peerENR = "enr"
|
|
const peerDirection = "direction"
|
|
const peerPubSubTopics = "pubSubTopics"
|
|
const peerScore = "score"
|
|
|
|
// ConnectionFailures contains connection failure information towards all peers
|
|
type ConnectionFailures struct {
|
|
sync.RWMutex
|
|
failures map[peer.ID]int
|
|
}
|
|
|
|
// WakuPeerstoreImpl is a implementation of WakuPeerStore
|
|
type WakuPeerstoreImpl struct {
|
|
peerStore peerstore.Peerstore
|
|
connFailures ConnectionFailures
|
|
}
|
|
|
|
// WakuPeerstore is an interface for implementing WakuPeerStore
|
|
type WakuPeerstore interface {
|
|
SetOrigin(p peer.ID, origin Origin) error
|
|
Origin(p peer.ID) (Origin, error)
|
|
PeersByOrigin(origin Origin) peer.IDSlice
|
|
SetENR(p peer.ID, enr *enode.Node) error
|
|
ENR(p peer.ID) (*enode.Node, error)
|
|
AddConnFailure(p peer.AddrInfo)
|
|
ResetConnFailures(p peer.AddrInfo)
|
|
ConnFailures(p peer.AddrInfo) int
|
|
|
|
SetDirection(p peer.ID, direction network.Direction) error
|
|
Direction(p peer.ID) (network.Direction, error)
|
|
|
|
AddPubSubTopic(p peer.ID, topic string) error
|
|
RemovePubSubTopic(p peer.ID, topic string) error
|
|
PubSubTopics(p peer.ID) (protocol.TopicSet, error)
|
|
SetPubSubTopics(p peer.ID, topics []string) error
|
|
PeersByPubSubTopics(pubSubTopics []string, specificPeers ...peer.ID) peer.IDSlice
|
|
PeersByPubSubTopic(pubSubTopic string, specificPeers ...peer.ID) peer.IDSlice
|
|
|
|
SetScore(peer.ID, float64) error
|
|
Score(peer.ID) (float64, error)
|
|
}
|
|
|
|
// NewWakuPeerstore creates a new WakuPeerStore object
|
|
func NewWakuPeerstore(p peerstore.Peerstore) peerstore.Peerstore {
|
|
return &WakuPeerstoreImpl{
|
|
peerStore: p,
|
|
connFailures: ConnectionFailures{
|
|
failures: make(map[peer.ID]int),
|
|
},
|
|
}
|
|
}
|
|
|
|
// SetOrigin sets origin for a specific peer.
|
|
func (ps *WakuPeerstoreImpl) SetOrigin(p peer.ID, origin Origin) error {
|
|
return ps.peerStore.Put(p, peerOrigin, origin)
|
|
}
|
|
|
|
// Origin fetches the origin for a specific peer.
|
|
func (ps *WakuPeerstoreImpl) Origin(p peer.ID) (Origin, error) {
|
|
result, err := ps.peerStore.Get(p, peerOrigin)
|
|
if err != nil {
|
|
return Unknown, err
|
|
}
|
|
|
|
return result.(Origin), nil
|
|
}
|
|
|
|
// SetScore sets score for a specific peer.
|
|
func (ps *WakuPeerstoreImpl) SetScore(p peer.ID, score float64) error {
|
|
return ps.peerStore.Put(p, peerScore, score)
|
|
}
|
|
|
|
// Score fetches the peerScore for a specific peer.
|
|
func (ps *WakuPeerstoreImpl) Score(p peer.ID) (float64, error) {
|
|
result, err := ps.peerStore.Get(p, peerScore)
|
|
if err != nil {
|
|
return -1, err
|
|
}
|
|
|
|
return result.(float64), nil
|
|
}
|
|
|
|
// PeersByOrigin returns the list of peers for a specific origin
|
|
func (ps *WakuPeerstoreImpl) PeersByOrigin(expectedOrigin Origin) peer.IDSlice {
|
|
var result peer.IDSlice
|
|
for _, p := range ps.Peers() {
|
|
actualOrigin, err := ps.Origin(p)
|
|
if err == nil && actualOrigin == expectedOrigin {
|
|
result = append(result, p)
|
|
}
|
|
}
|
|
return result
|
|
}
|
|
|
|
// SetENR sets the ENR record a peer
|
|
func (ps *WakuPeerstoreImpl) SetENR(p peer.ID, enr *enode.Node) error {
|
|
return ps.peerStore.Put(p, peerENR, enr)
|
|
}
|
|
|
|
// ENR fetches the ENR record for a peer
|
|
func (ps *WakuPeerstoreImpl) ENR(p peer.ID) (*enode.Node, error) {
|
|
result, err := ps.peerStore.Get(p, peerENR)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return result.(*enode.Node), nil
|
|
}
|
|
|
|
// AddConnFailure increments connectionFailures for a peer
|
|
func (ps *WakuPeerstoreImpl) AddConnFailure(p peer.AddrInfo) {
|
|
ps.connFailures.Lock()
|
|
defer ps.connFailures.Unlock()
|
|
ps.connFailures.failures[p.ID]++
|
|
}
|
|
|
|
// ResetConnFailures resets connectionFailures for a peer to 0
|
|
func (ps *WakuPeerstoreImpl) ResetConnFailures(p peer.AddrInfo) {
|
|
ps.connFailures.Lock()
|
|
defer ps.connFailures.Unlock()
|
|
ps.connFailures.failures[p.ID] = 0
|
|
}
|
|
|
|
// ConnFailures fetches connectionFailures for a peer
|
|
func (ps *WakuPeerstoreImpl) ConnFailures(p peer.AddrInfo) int {
|
|
ps.connFailures.RLock()
|
|
defer ps.connFailures.RUnlock()
|
|
return ps.connFailures.failures[p.ID]
|
|
}
|
|
|
|
// SetDirection sets connection direction for a specific peer.
|
|
func (ps *WakuPeerstoreImpl) SetDirection(p peer.ID, direction network.Direction) error {
|
|
return ps.peerStore.Put(p, peerDirection, direction)
|
|
}
|
|
|
|
// Direction fetches the connection direction (Inbound or outBound) for a specific peer
|
|
func (ps *WakuPeerstoreImpl) Direction(p peer.ID) (network.Direction, error) {
|
|
result, err := ps.peerStore.Get(p, peerDirection)
|
|
if err != nil {
|
|
return network.DirUnknown, err
|
|
}
|
|
|
|
return result.(network.Direction), nil
|
|
}
|
|
|
|
// AddPubSubTopic adds a new pubSubTopic for a peer
|
|
func (ps *WakuPeerstoreImpl) AddPubSubTopic(p peer.ID, topic string) error {
|
|
existingTopics, err := ps.PubSubTopics(p)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if _, found := existingTopics[topic]; found {
|
|
return nil
|
|
}
|
|
existingTopics[topic] = struct{}{}
|
|
return ps.peerStore.Put(p, peerPubSubTopics, maps.Keys(existingTopics))
|
|
}
|
|
|
|
// RemovePubSubTopic removes a pubSubTopic from the peer
|
|
func (ps *WakuPeerstoreImpl) RemovePubSubTopic(p peer.ID, topic string) error {
|
|
existingTopics, err := ps.PubSubTopics(p)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if len(existingTopics) == 0 {
|
|
return nil
|
|
}
|
|
|
|
delete(existingTopics, topic)
|
|
|
|
err = ps.SetPubSubTopics(p, maps.Keys(existingTopics))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// SetPubSubTopics sets pubSubTopics for a peer, it also overrides existing ones that were set previously..
|
|
func (ps *WakuPeerstoreImpl) SetPubSubTopics(p peer.ID, topics []string) error {
|
|
return ps.peerStore.Put(p, peerPubSubTopics, topics)
|
|
}
|
|
|
|
// PubSubTopics fetches list of pubSubTopics for a peer
|
|
func (ps *WakuPeerstoreImpl) PubSubTopics(p peer.ID) (protocol.TopicSet, error) {
|
|
result, err := ps.peerStore.Get(p, peerPubSubTopics)
|
|
if err != nil {
|
|
if errors.Is(err, peerstore.ErrNotFound) {
|
|
return protocol.NewTopicSet(), nil
|
|
} else {
|
|
return nil, err
|
|
}
|
|
}
|
|
return protocol.NewTopicSet((result.([]string))...), nil
|
|
}
|
|
|
|
// PeersByPubSubTopic Returns list of peers that support list of pubSubTopics
|
|
// If specifiPeers are listed, filtering is done from them otherwise from all peers in peerstore
|
|
func (ps *WakuPeerstoreImpl) PeersByPubSubTopics(pubSubTopics []string, specificPeers ...peer.ID) peer.IDSlice {
|
|
if specificPeers == nil {
|
|
specificPeers = ps.Peers()
|
|
}
|
|
var result peer.IDSlice
|
|
for _, p := range specificPeers {
|
|
peerMatch := true
|
|
peerTopics, err := ps.PubSubTopics(p)
|
|
if err == nil {
|
|
for _, t := range pubSubTopics {
|
|
if _, ok := peerTopics[t]; !ok {
|
|
peerMatch = false
|
|
break
|
|
}
|
|
}
|
|
if peerMatch {
|
|
result = append(result, p)
|
|
}
|
|
} //Note: skipping a peer in case of an error as there would be others available.
|
|
}
|
|
return result
|
|
}
|
|
|
|
// PeersByPubSubTopic Returns list of peers that support a single pubSubTopic
|
|
// If specifiPeers are listed, filtering is done from them otherwise from all peers in peerstore
|
|
func (ps *WakuPeerstoreImpl) PeersByPubSubTopic(pubSubTopic string, specificPeers ...peer.ID) peer.IDSlice {
|
|
if specificPeers == nil {
|
|
specificPeers = ps.Peers()
|
|
}
|
|
var result peer.IDSlice
|
|
for _, p := range specificPeers {
|
|
topics, err := ps.PubSubTopics(p)
|
|
if err == nil {
|
|
for topic := range topics {
|
|
if topic == pubSubTopic {
|
|
result = append(result, p)
|
|
}
|
|
}
|
|
} //Note: skipping a peer in case of an error as there would be others available.
|
|
}
|
|
return result
|
|
}
|