Commit 796e96c9 authored by Steven Allen's avatar Steven Allen

Avoid blocking in notification handlers.

This isn't the correct fix but the correct fix is probably not worth it.

introduces #70
parent 4e8a5204
package dht package dht
import ( import (
"context"
"io" "io"
"time"
inet "github.com/libp2p/go-libp2p-net" inet "github.com/libp2p/go-libp2p-net"
ma "github.com/multiformats/go-multiaddr" ma "github.com/multiformats/go-multiaddr"
mstream "github.com/multiformats/go-multistream" mstream "github.com/multiformats/go-multistream"
) )
// TODO: There is a race condition here where we could process notifications
// out-of-order and incorrectly mark some peers as DHT nodes (or not DHT nodes).
// The correct fix for this is nasty so I'm not really sure it's worth it.
// Incorrectly recording or failing to record a DHT node in the routing table
// isn't a big issue.
const dhtCheckTimeout = 10 * time.Second
// netNotifiee defines methods to be used with the IpfsDHT // netNotifiee defines methods to be used with the IpfsDHT
type netNotifiee IpfsDHT type netNotifiee IpfsDHT
...@@ -23,25 +33,36 @@ func (nn *netNotifiee) Connected(n inet.Network, v inet.Conn) { ...@@ -23,25 +33,36 @@ func (nn *netNotifiee) Connected(n inet.Network, v inet.Conn) {
default: default:
} }
// Note: We *could* just check the peerstore to see if the remote side supports the dht go func() {
// protocol, but its not clear that that information will make it into the peerstore
// by the time this notification is sent. So just to be very careful, we brute force this // Note: We *could* just check the peerstore to see if the remote side supports the dht
// and open a new stream // protocol, but its not clear that that information will make it into the peerstore
s, err := dht.host.NewStream(dht.Context(), v.RemotePeer(), ProtocolDHT, ProtocolDHTOld) // by the time this notification is sent. So just to be very careful, we brute force this
switch err { // and open a new stream
case nil:
s.Close() // TODO: There's a race condition here where the connection may
// connected fine? full dht node // not be open (and we may sit here trying to connect). I've
dht.Update(dht.Context(), v.RemotePeer()) // added a timeout but that's not really the correct fix.
case mstream.ErrNotSupported:
// Client mode only, don't bother adding them to our routing table ctx, cancel := context.WithTimeout(dht.Context(), dhtCheckTimeout)
case io.EOF: defer cancel()
// This is kindof an error, but it happens someone often so make it a warning s, err := dht.host.NewStream(ctx, v.RemotePeer(), ProtocolDHT, ProtocolDHTOld)
log.Warningf("checking dht client type: %s", err)
default: switch err {
// real error? thats odd case nil:
log.Errorf("checking dht client type: %s", err) s.Close()
} // connected fine? full dht node
dht.Update(dht.Context(), v.RemotePeer())
case mstream.ErrNotSupported:
// Client mode only, don't bother adding them to our routing table
case io.EOF:
// This is kindof an error, but it happens someone often so make it a warning
log.Warningf("checking dht client type: %s", err)
default:
// real error? thats odd
log.Errorf("checking dht client type: %s", err)
}
}()
} }
func (nn *netNotifiee) Disconnected(n inet.Network, v inet.Conn) { func (nn *netNotifiee) Disconnected(n inet.Network, v inet.Conn) {
...@@ -51,7 +72,7 @@ func (nn *netNotifiee) Disconnected(n inet.Network, v inet.Conn) { ...@@ -51,7 +72,7 @@ func (nn *netNotifiee) Disconnected(n inet.Network, v inet.Conn) {
return return
default: default:
} }
dht.routingTable.Remove(v.RemotePeer()) go dht.routingTable.Remove(v.RemotePeer())
} }
func (nn *netNotifiee) OpenedStream(n inet.Network, v inet.Stream) {} func (nn *netNotifiee) OpenedStream(n inet.Network, v inet.Stream) {}
......
Markdown is supported
0% or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment