2018-04-10 06:44:09 +00:00
|
|
|
package peers
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"sync"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/ethereum/go-ethereum/common/mclock"
|
|
|
|
"github.com/ethereum/go-ethereum/event"
|
|
|
|
"github.com/ethereum/go-ethereum/log"
|
|
|
|
"github.com/ethereum/go-ethereum/p2p"
|
2018-04-12 13:08:49 +00:00
|
|
|
"github.com/ethereum/go-ethereum/p2p/discover"
|
2018-04-10 06:44:09 +00:00
|
|
|
"github.com/ethereum/go-ethereum/p2p/discv5"
|
|
|
|
|
|
|
|
"github.com/status-im/status-go/geth/params"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
// ErrDiscv5NotRunning returned when pool is started but discover v5 is not running or not enabled.
|
|
|
|
ErrDiscv5NotRunning = errors.New("Discovery v5 is not running")
|
|
|
|
)
|
|
|
|
|
2018-04-13 08:34:30 +00:00
|
|
|
// PoolEvent is a type used to for peer pool events.
|
|
|
|
type PoolEvent string
|
|
|
|
|
2018-04-10 06:44:09 +00:00
|
|
|
const (
|
|
|
|
// expirationPeriod is an amount of time while peer is considered as a connectable
|
|
|
|
expirationPeriod = 60 * time.Minute
|
2018-04-12 13:08:49 +00:00
|
|
|
// discoveryRestartTimeout defines how often loop will try to start discovery server
|
|
|
|
discoveryRestartTimeout = 2 * time.Second
|
2018-04-10 06:44:09 +00:00
|
|
|
// DefaultFastSync is a recommended value for aggressive peers search.
|
|
|
|
DefaultFastSync = 3 * time.Second
|
|
|
|
// DefaultSlowSync is a recommended value for slow (background) peers search.
|
2018-05-01 15:19:11 +00:00
|
|
|
DefaultSlowSync = 30 * time.Second
|
|
|
|
// DefaultDiscV5Timeout is a timeout after which Discv5 is stopped.
|
|
|
|
DefaultDiscV5Timeout = 3 * time.Minute
|
|
|
|
// DefaultTopicFastModeTimeout is a timeout after which sync mode is switched to slow mode.
|
|
|
|
DefaultTopicFastModeTimeout = 30 * time.Second
|
2018-04-10 06:44:09 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// NewPeerPool creates instance of PeerPool
|
|
|
|
func NewPeerPool(config map[discv5.Topic]params.Limits, fastSync, slowSync time.Duration, cache *Cache, stopOnMax bool) *PeerPool {
|
|
|
|
return &PeerPool{
|
2018-05-01 15:19:11 +00:00
|
|
|
config: config,
|
|
|
|
fastSync: fastSync,
|
|
|
|
slowSync: slowSync,
|
|
|
|
discServerTimeout: DefaultDiscV5Timeout,
|
|
|
|
cache: cache,
|
|
|
|
stopOnMax: stopOnMax,
|
2018-04-10 06:44:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type peerInfo struct {
|
|
|
|
// discoveredTime last time when node was found by v5
|
|
|
|
discoveredTime mclock.AbsTime
|
2018-04-19 15:18:49 +00:00
|
|
|
// dismissed is true when our node requested a disconnect
|
|
|
|
dismissed bool
|
2018-04-10 06:44:09 +00:00
|
|
|
|
|
|
|
node *discv5.Node
|
|
|
|
}
|
|
|
|
|
|
|
|
// PeerPool manages discovered peers and connects them to p2p server
|
|
|
|
type PeerPool struct {
|
|
|
|
// config can be set only once per pool life cycle
|
2018-05-01 15:19:11 +00:00
|
|
|
config map[discv5.Topic]params.Limits
|
|
|
|
fastSync time.Duration
|
|
|
|
slowSync time.Duration
|
|
|
|
discServerTimeout time.Duration
|
|
|
|
cache *Cache
|
|
|
|
stopOnMax bool
|
2018-04-10 06:44:09 +00:00
|
|
|
|
|
|
|
mu sync.RWMutex
|
|
|
|
topics []*TopicPool
|
|
|
|
serverSubscription event.Subscription
|
2018-05-01 15:19:11 +00:00
|
|
|
events chan *p2p.PeerEvent
|
2018-04-10 06:44:09 +00:00
|
|
|
quit chan struct{}
|
2018-05-01 15:19:11 +00:00
|
|
|
timeout <-chan time.Time
|
2018-04-10 06:44:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Start creates topic pool for each topic in config and subscribes to server events.
|
|
|
|
func (p *PeerPool) Start(server *p2p.Server) error {
|
|
|
|
if server.DiscV5 == nil {
|
|
|
|
return ErrDiscv5NotRunning
|
|
|
|
}
|
|
|
|
p.mu.Lock()
|
|
|
|
defer p.mu.Unlock()
|
|
|
|
p.quit = make(chan struct{})
|
2018-05-01 15:19:11 +00:00
|
|
|
p.timeout = time.After(p.discServerTimeout)
|
2018-04-10 06:44:09 +00:00
|
|
|
p.topics = make([]*TopicPool, 0, len(p.config))
|
|
|
|
for topic, limits := range p.config {
|
|
|
|
topicPool := NewTopicPool(topic, limits, p.slowSync, p.fastSync)
|
|
|
|
if err := topicPool.StartSearch(server); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
p.topics = append(p.topics, topicPool)
|
|
|
|
}
|
|
|
|
|
2018-05-01 15:19:11 +00:00
|
|
|
// discovery must be already started when pool is started
|
|
|
|
SendDiscoveryStarted()
|
|
|
|
|
|
|
|
p.events = make(chan *p2p.PeerEvent, 20)
|
|
|
|
p.serverSubscription = server.SubscribeEvents(p.events)
|
2018-04-10 06:44:09 +00:00
|
|
|
go func() {
|
2018-05-01 15:19:11 +00:00
|
|
|
p.handleServerPeers(server, p.events)
|
2018-04-10 06:44:09 +00:00
|
|
|
}()
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-05-01 15:19:11 +00:00
|
|
|
func (p *PeerPool) startDiscovery(server *p2p.Server) error {
|
|
|
|
if server.DiscV5 != nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
ntab, err := StartDiscv5(server)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
p.mu.Lock()
|
|
|
|
server.DiscV5 = ntab
|
|
|
|
p.timeout = time.After(p.discServerTimeout)
|
|
|
|
p.mu.Unlock()
|
|
|
|
|
|
|
|
SendDiscoveryStarted()
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *PeerPool) stopDiscovery(server *p2p.Server) {
|
|
|
|
if server.DiscV5 == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
p.mu.Lock()
|
|
|
|
server.DiscV5.Close()
|
|
|
|
server.DiscV5 = nil
|
|
|
|
p.timeout = nil
|
|
|
|
p.mu.Unlock()
|
|
|
|
|
|
|
|
for _, t := range p.topics {
|
|
|
|
t.StopSearch()
|
|
|
|
}
|
|
|
|
|
|
|
|
SendDiscoveryStopped()
|
|
|
|
}
|
|
|
|
|
2018-04-12 13:08:49 +00:00
|
|
|
// restartDiscovery and search for topics that have peer count below min
|
|
|
|
func (p *PeerPool) restartDiscovery(server *p2p.Server) error {
|
|
|
|
if server.DiscV5 == nil {
|
2018-05-01 15:19:11 +00:00
|
|
|
if err := p.startDiscovery(server); err != nil {
|
2018-04-12 13:08:49 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
log.Debug("restarted discovery from peer pool")
|
|
|
|
}
|
|
|
|
for _, t := range p.topics {
|
|
|
|
if !t.BelowMin() || t.SearchRunning() {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
err := t.StartSearch(server)
|
|
|
|
if err != nil {
|
|
|
|
log.Error("search failed to start", "error", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-04-10 06:44:09 +00:00
|
|
|
// handleServerPeers watches server peer events, notifies topic pools about changes
|
|
|
|
// in the peer set and stops the discv5 if all topic pools collected enough peers.
|
2018-05-01 15:19:11 +00:00
|
|
|
//
|
|
|
|
// @TODO(adam): split it into peers and discovery management loops. This should
|
|
|
|
// simplify the whole logic and allow to remove `timeout` field from `PeerPool`.
|
2018-04-10 06:44:09 +00:00
|
|
|
func (p *PeerPool) handleServerPeers(server *p2p.Server, events <-chan *p2p.PeerEvent) {
|
2018-04-12 13:08:49 +00:00
|
|
|
var retryDiscv5 <-chan time.Time
|
|
|
|
|
2018-04-10 06:44:09 +00:00
|
|
|
for {
|
2018-05-01 15:19:11 +00:00
|
|
|
p.mu.RLock()
|
|
|
|
timeout := p.timeout
|
|
|
|
p.mu.RUnlock()
|
|
|
|
|
2018-04-10 06:44:09 +00:00
|
|
|
select {
|
|
|
|
case <-p.quit:
|
2018-05-01 15:19:11 +00:00
|
|
|
log.Debug("stopping DiscV5 because of quit", "server", server.Self())
|
|
|
|
p.stopDiscovery(server)
|
2018-04-10 06:44:09 +00:00
|
|
|
return
|
2018-05-01 15:19:11 +00:00
|
|
|
case <-timeout:
|
|
|
|
log.Debug("DiscV5 timed out", "server", server.Self())
|
|
|
|
p.stopDiscovery(server)
|
2018-04-12 13:08:49 +00:00
|
|
|
case <-retryDiscv5:
|
|
|
|
if err := p.restartDiscovery(server); err != nil {
|
|
|
|
retryDiscv5 = time.After(discoveryRestartTimeout)
|
2018-05-01 15:19:11 +00:00
|
|
|
log.Error("starting discv5 failed", "error", err, "retry", discoveryRestartTimeout)
|
2018-04-12 13:08:49 +00:00
|
|
|
}
|
2018-04-10 06:44:09 +00:00
|
|
|
case event := <-events:
|
|
|
|
switch event.Type {
|
|
|
|
case p2p.PeerEventTypeDrop:
|
2018-04-12 13:08:49 +00:00
|
|
|
log.Debug("confirm peer dropped", "ID", event.Peer)
|
2018-05-01 15:19:11 +00:00
|
|
|
if p.handleDroppedPeer(server, event.Peer) {
|
2018-04-12 13:08:49 +00:00
|
|
|
retryDiscv5 = time.After(0)
|
2018-04-10 06:44:09 +00:00
|
|
|
}
|
|
|
|
case p2p.PeerEventTypeAdd:
|
2018-05-01 15:19:11 +00:00
|
|
|
log.Debug("confirm peer added", "ID", event.Peer)
|
|
|
|
if p.stopOnMax && p.handleAddedPeer(server, event.Peer) {
|
|
|
|
log.Debug("closing discv5 connection because all topics reached max limit", "server", server.Self())
|
|
|
|
p.stopDiscovery(server)
|
|
|
|
}
|
|
|
|
default: // skip other events
|
|
|
|
continue
|
2018-04-10 06:44:09 +00:00
|
|
|
}
|
2018-04-25 07:13:59 +00:00
|
|
|
SendDiscoverySummary(server.PeersInfo())
|
2018-04-10 06:44:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-12 13:08:49 +00:00
|
|
|
// handleAddedPeer notifies all topics about added peer and return true if all topics has max limit of connections
|
|
|
|
func (p *PeerPool) handleAddedPeer(server *p2p.Server, nodeID discover.NodeID) (all bool) {
|
|
|
|
p.mu.Lock()
|
|
|
|
defer p.mu.Unlock()
|
|
|
|
all = true
|
|
|
|
for _, t := range p.topics {
|
|
|
|
t.ConfirmAdded(server, nodeID)
|
|
|
|
if p.stopOnMax && t.MaxReached() {
|
|
|
|
t.StopSearch()
|
|
|
|
} else {
|
|
|
|
all = false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return all
|
|
|
|
}
|
|
|
|
|
|
|
|
// handleDroppedPeer notifies every topic about dropped peer and returns true if any peer have connections
|
|
|
|
// below min limit
|
|
|
|
func (p *PeerPool) handleDroppedPeer(server *p2p.Server, nodeID discover.NodeID) (any bool) {
|
|
|
|
p.mu.Lock()
|
|
|
|
defer p.mu.Unlock()
|
|
|
|
for _, t := range p.topics {
|
|
|
|
confirmed := t.ConfirmDropped(server, nodeID)
|
|
|
|
if confirmed {
|
|
|
|
newPeer := t.AddPeerFromTable(server)
|
|
|
|
if newPeer != nil {
|
|
|
|
log.Debug("added peer from local table", "ID", newPeer.ID)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
log.Debug("search", "topic", t.topic, "below min", t.BelowMin())
|
|
|
|
if t.BelowMin() && !t.SearchRunning() {
|
|
|
|
any = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return any
|
|
|
|
}
|
|
|
|
|
2018-04-10 06:44:09 +00:00
|
|
|
// Stop closes pool quit channel and all channels that are watched by search queries
|
|
|
|
// and waits till all goroutines will exit.
|
|
|
|
func (p *PeerPool) Stop() {
|
|
|
|
// pool wasn't started
|
|
|
|
if p.quit == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
select {
|
|
|
|
case <-p.quit:
|
|
|
|
return
|
|
|
|
default:
|
|
|
|
log.Debug("started closing peer pool")
|
|
|
|
close(p.quit)
|
|
|
|
}
|
|
|
|
p.serverSubscription.Unsubscribe()
|
|
|
|
}
|