mirror of
https://github.com/libp2p/go-libp2p-peerstore.git
synced 2025-01-16 02:50:07 +08:00
fix multiple TTL bugs
The first fix independently extends the address expiration time and the address TTL: By example: * We have an address with a TTL of 4s that will expire in 1s. * We update it with a TTL of 3s. Before this change: * We end up with an address with a TTL of 3s that will expire in 3s. After this change: * We end up with an address with a TTL of 4s that will expire in 3s. --- The second fix prevents the in-memory addressbook from announcing existing addresses every time their TTLs get updated. --- The third fix correctly updates TTLs for existing addresses in the on-disk addressbook. This fixes https://github.com/libp2p/go-libp2p-identify/issues/2
This commit is contained in:
parent
ad0faef7c7
commit
2fb6d7a48f
@ -347,11 +347,21 @@ Outer:
|
|||||||
for _, have := range pr.Addrs {
|
for _, have := range pr.Addrs {
|
||||||
if incoming.Equal(have.Addr) {
|
if incoming.Equal(have.Addr) {
|
||||||
existed[i] = true
|
existed[i] = true
|
||||||
if mode == ttlExtend && have.Expiry > newExp {
|
switch mode {
|
||||||
// if we're only extending TTLs but the addr already has a longer one, we skip it.
|
case ttlOverride:
|
||||||
continue Outer
|
have.Ttl = int64(ttl)
|
||||||
}
|
|
||||||
have.Expiry = newExp
|
have.Expiry = newExp
|
||||||
|
case ttlExtend:
|
||||||
|
if int64(ttl) > have.Ttl {
|
||||||
|
have.Ttl = int64(ttl)
|
||||||
|
}
|
||||||
|
if newExp > have.Expiry {
|
||||||
|
have.Expiry = newExp
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
panic("BUG: unimplemented ttl mode")
|
||||||
|
}
|
||||||
|
|
||||||
// we found the address, and addresses cannot be duplicate,
|
// we found the address, and addresses cannot be duplicate,
|
||||||
// so let's move on to the next.
|
// so let's move on to the next.
|
||||||
continue Outer
|
continue Outer
|
||||||
|
@ -132,7 +132,7 @@ func (mab *memoryAddrBook) AddAddr(p peer.ID, addr ma.Multiaddr, ttl time.Durati
|
|||||||
|
|
||||||
// AddAddrs gives memoryAddrBook addresses to use, with a given ttl
|
// AddAddrs gives memoryAddrBook addresses to use, with a given ttl
|
||||||
// (time-to-live), after which the address is no longer valid.
|
// (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
|
// This function never reduces the TTL or expiration of an address.
|
||||||
func (mab *memoryAddrBook) AddAddrs(p peer.ID, addrs []ma.Multiaddr, ttl time.Duration) {
|
func (mab *memoryAddrBook) AddAddrs(p peer.ID, addrs []ma.Multiaddr, ttl time.Duration) {
|
||||||
// if ttl is zero, exit. nothing to do.
|
// if ttl is zero, exit. nothing to do.
|
||||||
if ttl <= 0 {
|
if ttl <= 0 {
|
||||||
@ -156,10 +156,19 @@ func (mab *memoryAddrBook) AddAddrs(p peer.ID, addrs []ma.Multiaddr, ttl time.Du
|
|||||||
}
|
}
|
||||||
asBytes := addr.Bytes()
|
asBytes := addr.Bytes()
|
||||||
a, found := amap[string(asBytes)] // won't allocate.
|
a, found := amap[string(asBytes)] // won't allocate.
|
||||||
if !found || exp.After(a.Expires) {
|
if !found {
|
||||||
|
// not found, save and announce it.
|
||||||
amap[string(asBytes)] = &expiringAddr{Addr: addr, Expires: exp, TTL: ttl}
|
amap[string(asBytes)] = &expiringAddr{Addr: addr, Expires: exp, TTL: ttl}
|
||||||
|
|
||||||
mab.subManager.BroadcastAddr(p, addr)
|
mab.subManager.BroadcastAddr(p, addr)
|
||||||
|
} else {
|
||||||
|
// Update expiration/TTL independently.
|
||||||
|
// We never want to reduce either.
|
||||||
|
if ttl > a.TTL {
|
||||||
|
a.TTL = ttl
|
||||||
|
}
|
||||||
|
if exp.After(a.Expires) {
|
||||||
|
a.Expires = exp
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -86,9 +86,13 @@ func testAddAddress(ab pstore.AddrBook) func(*testing.T) {
|
|||||||
// after the initial TTL has expired, check that only the third address is present.
|
// after the initial TTL has expired, check that only the third address is present.
|
||||||
time.Sleep(1200 * time.Millisecond)
|
time.Sleep(1200 * time.Millisecond)
|
||||||
AssertAddressesEqual(t, addrs[2:], ab.Addrs(id))
|
AssertAddressesEqual(t, addrs[2:], ab.Addrs(id))
|
||||||
|
|
||||||
|
// make sure we actually set the TTL
|
||||||
|
ab.UpdateAddrs(id, time.Hour, 0)
|
||||||
|
AssertAddressesEqual(t, nil, ab.Addrs(id))
|
||||||
})
|
})
|
||||||
|
|
||||||
t.Run("adding an existing address with an earlier expiration is noop", func(t *testing.T) {
|
t.Run("adding an existing address with an earlier expiration never reduces the expiration", func(t *testing.T) {
|
||||||
id := GeneratePeerIDs(1)[0]
|
id := GeneratePeerIDs(1)[0]
|
||||||
addrs := GenerateAddrs(3)
|
addrs := GenerateAddrs(3)
|
||||||
|
|
||||||
@ -102,6 +106,27 @@ func testAddAddress(ab pstore.AddrBook) func(*testing.T) {
|
|||||||
time.Sleep(2100 * time.Millisecond)
|
time.Sleep(2100 * time.Millisecond)
|
||||||
AssertAddressesEqual(t, addrs, ab.Addrs(id))
|
AssertAddressesEqual(t, addrs, ab.Addrs(id))
|
||||||
})
|
})
|
||||||
|
|
||||||
|
t.Run("adding an existing address with an earlier expiration never reduces the TTL", func(t *testing.T) {
|
||||||
|
id := GeneratePeerIDs(1)[0]
|
||||||
|
addrs := GenerateAddrs(1)
|
||||||
|
|
||||||
|
ab.AddAddrs(id, addrs, 4*time.Second)
|
||||||
|
// 4 seconds left
|
||||||
|
time.Sleep(3 * time.Second)
|
||||||
|
// 1 second left
|
||||||
|
ab.AddAddrs(id, addrs, 3*time.Second)
|
||||||
|
// 3 seconds left
|
||||||
|
time.Sleep(2)
|
||||||
|
// 1 seconds left.
|
||||||
|
|
||||||
|
// We still have the address.
|
||||||
|
AssertAddressesEqual(t, addrs, ab.Addrs(id))
|
||||||
|
|
||||||
|
// The TTL wasn't reduced
|
||||||
|
ab.UpdateAddrs(id, 4*time.Second, 0)
|
||||||
|
AssertAddressesEqual(t, nil, ab.Addrs(id))
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user