go-libp2p-peerstore/pstoremem/addr_book.go

340 lines
7.4 KiB
Go
Raw Normal View History

2018-09-05 01:07:44 +08:00
package pstoremem
2015-10-01 06:42:55 +08:00
import (
2016-10-05 07:19:16 +08:00
"context"
"sort"
2015-10-01 06:42:55 +08:00
"sync"
"time"
2018-09-02 19:10:55 +08:00
logging "github.com/ipfs/go-log"
2018-09-08 01:46:23 +08:00
peer "github.com/libp2p/go-libp2p-peer"
2018-09-02 19:10:55 +08:00
ma "github.com/multiformats/go-multiaddr"
2018-08-30 23:24:09 +08:00
pstore "github.com/libp2p/go-libp2p-peerstore"
2018-09-08 01:46:23 +08:00
addr "github.com/libp2p/go-libp2p-peerstore/addr"
2015-10-01 06:42:55 +08:00
)
var log = logging.Logger("peerstore")
2015-10-01 06:42:55 +08:00
type expiringAddr struct {
Addr ma.Multiaddr
TTL time.Duration
Expires time.Time
2015-10-01 06:42:55 +08:00
}
func (e *expiringAddr) ExpiredBy(t time.Time) bool {
return t.After(e.Expires)
2015-10-01 06:42:55 +08:00
}
var _ pstore.AddrBook = (*memoryAddrBook)(nil)
// memoryAddrBook manages addresses.
type memoryAddrBook struct {
addrmu sync.Mutex
addrs map[peer.ID]map[string]expiringAddr
subManager *AddrSubManager
2015-10-01 06:42:55 +08:00
}
func NewAddrBook() pstore.AddrBook {
return &memoryAddrBook{
addrs: make(map[peer.ID]map[string]expiringAddr),
subManager: NewAddrSubManager(),
}
2015-10-01 06:42:55 +08:00
}
func (mab *memoryAddrBook) PeersWithAddrs() peer.IDSlice {
mab.addrmu.Lock()
defer mab.addrmu.Unlock()
if mab.addrs == nil {
2015-10-01 06:42:55 +08:00
return nil
}
pids := make(peer.IDSlice, 0, len(mab.addrs))
for pid := range mab.addrs {
2015-10-01 06:42:55 +08:00
pids = append(pids, pid)
}
return pids
}
// AddAddr calls AddAddrs(p, []ma.Multiaddr{addr}, ttl)
func (mab *memoryAddrBook) AddAddr(p peer.ID, addr ma.Multiaddr, ttl time.Duration) {
mab.AddAddrs(p, []ma.Multiaddr{addr}, ttl)
2015-10-01 06:42:55 +08:00
}
// AddAddrs gives memoryAddrBook addresses to use, with a given ttl
2015-10-01 06:42:55 +08:00
// (time-to-live), after which the address is no longer valid.
// If the manager has a longer TTL, the operation is a no-op for that address
func (mab *memoryAddrBook) AddAddrs(p peer.ID, addrs []ma.Multiaddr, ttl time.Duration) {
mab.addrmu.Lock()
defer mab.addrmu.Unlock()
2015-10-01 06:42:55 +08:00
// if ttl is zero, exit. nothing to do.
if ttl <= 0 {
return
}
amap := mab.addrs[p]
if amap == nil {
amap = make(map[string]expiringAddr, len(addrs))
mab.addrs[p] = amap
2015-10-01 06:42:55 +08:00
}
exp := time.Now().Add(ttl)
for _, addr := range addrs {
if addr == nil {
log.Warningf("was passed nil multiaddr for %s", p)
continue
}
addrstr := string(addr.Bytes())
2015-10-01 06:42:55 +08:00
a, found := amap[addrstr]
if !found || exp.After(a.Expires) {
amap[addrstr] = expiringAddr{Addr: addr, Expires: exp, TTL: ttl}
mab.subManager.BroadcastAddr(p, addr)
2015-10-01 06:42:55 +08:00
}
}
}
// SetAddr calls mgr.SetAddrs(p, addr, ttl)
func (mab *memoryAddrBook) SetAddr(p peer.ID, addr ma.Multiaddr, ttl time.Duration) {
mab.SetAddrs(p, []ma.Multiaddr{addr}, ttl)
2015-10-01 06:42:55 +08:00
}
// SetAddrs sets the ttl on addresses. This clears any TTL there previously.
// This is used when we receive the best estimate of the validity of an address.
func (mab *memoryAddrBook) SetAddrs(p peer.ID, addrs []ma.Multiaddr, ttl time.Duration) {
mab.addrmu.Lock()
defer mab.addrmu.Unlock()
2015-10-01 06:42:55 +08:00
amap := mab.addrs[p]
if amap == nil {
amap = make(map[string]expiringAddr, len(addrs))
mab.addrs[p] = amap
2015-10-01 06:42:55 +08:00
}
exp := time.Now().Add(ttl)
for _, addr := range addrs {
if addr == nil {
log.Warningf("was passed nil multiaddr for %s", p)
continue
}
2015-10-01 06:42:55 +08:00
// re-set all of them for new ttl.
addrstr := string(addr.Bytes())
2015-10-01 06:42:55 +08:00
if ttl > 0 {
amap[addrstr] = expiringAddr{Addr: addr, Expires: exp, TTL: ttl}
mab.subManager.BroadcastAddr(p, addr)
2015-10-01 06:42:55 +08:00
} else {
delete(amap, addrstr)
2015-10-01 06:42:55 +08:00
}
}
}
// UpdateAddrs updates the addresses associated with the given peer that have
// the given oldTTL to have the given newTTL.
func (mab *memoryAddrBook) UpdateAddrs(p peer.ID, oldTTL time.Duration, newTTL time.Duration) {
mab.addrmu.Lock()
defer mab.addrmu.Unlock()
if mab.addrs == nil {
return
}
amap, found := mab.addrs[p]
if !found {
return
}
exp := time.Now().Add(newTTL)
for k, addr := range amap {
if oldTTL == addr.TTL {
addr.TTL = newTTL
addr.Expires = exp
amap[k] = addr
}
}
}
2015-10-01 06:42:55 +08:00
// Addresses returns all known (and valid) addresses for a given
func (mab *memoryAddrBook) Addrs(p peer.ID) []ma.Multiaddr {
mab.addrmu.Lock()
defer mab.addrmu.Unlock()
2015-10-01 06:42:55 +08:00
// not initialized? nothing to give.
if mab.addrs == nil {
2015-10-01 06:42:55 +08:00
return nil
}
amap, found := mab.addrs[p]
2015-10-01 06:42:55 +08:00
if !found {
return nil
}
now := time.Now()
good := make([]ma.Multiaddr, 0, len(amap))
for k, m := range amap {
if !m.ExpiredBy(now) {
2015-10-01 06:42:55 +08:00
good = append(good, m.Addr)
} else {
delete(amap, k)
2015-10-01 06:42:55 +08:00
}
}
return good
}
2018-06-16 01:54:04 +08:00
// ClearAddrs removes all previously stored addresses
func (mab *memoryAddrBook) ClearAddrs(p peer.ID) {
mab.addrmu.Lock()
defer mab.addrmu.Unlock()
2015-10-01 06:42:55 +08:00
delete(mab.addrs, p)
2015-10-01 06:42:55 +08:00
}
2018-06-16 01:54:04 +08:00
// AddrStream returns a channel on which all new addresses discovered for a
// given peer ID will be published.
func (mab *memoryAddrBook) AddrStream(ctx context.Context, p peer.ID) <-chan ma.Multiaddr {
mab.addrmu.Lock()
defer mab.addrmu.Unlock()
2018-06-15 04:17:50 +08:00
baseaddrslice := mab.addrs[p]
initial := make([]ma.Multiaddr, 0, len(baseaddrslice))
for _, a := range baseaddrslice {
initial = append(initial, a.Addr)
}
return mab.subManager.AddrStream(ctx, p, initial)
}
type addrSub struct {
pubch chan ma.Multiaddr
lk sync.Mutex
buffer []ma.Multiaddr
ctx context.Context
}
func (s *addrSub) pubAddr(a ma.Multiaddr) {
select {
case s.pubch <- a:
case <-s.ctx.Done():
}
}
2018-06-16 01:54:04 +08:00
// An abstracted, pub-sub manager for address streams. Extracted from
// memoryAddrBook in order to support additional implementations.
type AddrSubManager struct {
mu sync.RWMutex
subs map[peer.ID][]*addrSub
}
2018-06-16 01:54:04 +08:00
// NewAddrSubManager initializes an AddrSubManager.
func NewAddrSubManager() *AddrSubManager {
return &AddrSubManager{
subs: make(map[peer.ID][]*addrSub),
}
}
2018-06-16 01:54:04 +08:00
// Used internally by the address stream coroutine to remove a subscription
// from the manager.
func (mgr *AddrSubManager) removeSub(p peer.ID, s *addrSub) {
mgr.mu.Lock()
defer mgr.mu.Unlock()
2018-08-31 20:23:02 +08:00
subs := mgr.subs[p]
if len(subs) == 1 {
if subs[0] != s {
return
}
delete(mgr.subs, p)
return
}
for i, v := range subs {
if v == s {
subs[i] = subs[len(subs)-1]
subs[len(subs)-1] = nil
mgr.subs[p] = subs[:len(subs)-1]
return
}
}
}
2018-06-16 01:54:04 +08:00
// BroadcastAddr broadcasts a new address to all subscribed streams.
func (mgr *AddrSubManager) BroadcastAddr(p peer.ID, addr ma.Multiaddr) {
2018-06-16 01:54:04 +08:00
mgr.mu.RLock()
defer mgr.mu.RUnlock()
2018-06-16 01:54:04 +08:00
if subs, ok := mgr.subs[p]; ok {
for _, sub := range subs {
sub.pubAddr(addr)
}
}
}
2018-06-16 01:54:04 +08:00
// AddrStream creates a new subscription for a given peer ID, pre-populating the
// channel with any addresses we might already have on file.
func (mgr *AddrSubManager) AddrStream(ctx context.Context, p peer.ID, initial []ma.Multiaddr) <-chan ma.Multiaddr {
sub := &addrSub{pubch: make(chan ma.Multiaddr), ctx: ctx}
out := make(chan ma.Multiaddr)
2018-06-16 05:33:18 +08:00
mgr.mu.Lock()
2018-06-15 04:17:50 +08:00
if _, ok := mgr.subs[p]; ok {
mgr.subs[p] = append(mgr.subs[p], sub)
} else {
mgr.subs[p] = []*addrSub{sub}
}
2018-06-16 05:33:18 +08:00
mgr.mu.Unlock()
sort.Sort(addr.AddrList(initial))
go func(buffer []ma.Multiaddr) {
defer close(out)
2017-12-06 09:37:20 +08:00
sent := make(map[string]bool, len(buffer))
2016-04-21 02:25:03 +08:00
var outch chan ma.Multiaddr
for _, a := range buffer {
2017-12-06 09:38:43 +08:00
sent[string(a.Bytes())] = true
}
var next ma.Multiaddr
if len(buffer) > 0 {
next = buffer[0]
buffer = buffer[1:]
2016-04-21 02:25:03 +08:00
outch = out
}
for {
select {
case outch <- next:
if len(buffer) > 0 {
next = buffer[0]
buffer = buffer[1:]
} else {
outch = nil
next = nil
}
case naddr := <-sub.pubch:
2017-12-06 09:38:43 +08:00
if sent[string(naddr.Bytes())] {
continue
}
2017-12-06 09:38:43 +08:00
sent[string(naddr.Bytes())] = true
if next == nil {
next = naddr
outch = out
} else {
buffer = append(buffer, naddr)
}
case <-ctx.Done():
mgr.removeSub(p, sub)
return
}
}
}(initial)
return out
}