mirror of
https://github.com/codex-storage/nim-libp2p.git
synced 2025-01-27 19:25:48 +00:00
Handle when peers ask each other at the same time (#865)
This commit is contained in:
parent
382b992e00
commit
ddeb7b3bd4
@ -32,6 +32,7 @@ const
|
|||||||
|
|
||||||
type
|
type
|
||||||
TooManyConnectionsError* = object of LPError
|
TooManyConnectionsError* = object of LPError
|
||||||
|
AlreadyExpectingConnectionError* = object of LPError
|
||||||
|
|
||||||
ConnEventKind* {.pure.} = enum
|
ConnEventKind* {.pure.} = enum
|
||||||
Connected, # A connection was made and securely upgraded - there may be
|
Connected, # A connection was made and securely upgraded - there may be
|
||||||
@ -79,7 +80,7 @@ type
|
|||||||
muxed: Table[Connection, MuxerHolder]
|
muxed: Table[Connection, MuxerHolder]
|
||||||
connEvents: array[ConnEventKind, OrderedSet[ConnEventHandler]]
|
connEvents: array[ConnEventKind, OrderedSet[ConnEventHandler]]
|
||||||
peerEvents: array[PeerEventKind, OrderedSet[PeerEventHandler]]
|
peerEvents: array[PeerEventKind, OrderedSet[PeerEventHandler]]
|
||||||
expectedConnections: Table[PeerId, Future[Connection]]
|
expectedConnectionsOverLimit*: Table[(PeerId, Direction), Future[Connection]]
|
||||||
peerStore*: PeerStore
|
peerStore*: PeerStore
|
||||||
|
|
||||||
ConnectionSlot* = object
|
ConnectionSlot* = object
|
||||||
@ -221,18 +222,19 @@ proc triggerPeerEvents*(c: ConnManager,
|
|||||||
except CatchableError as exc: # handlers should not raise!
|
except CatchableError as exc: # handlers should not raise!
|
||||||
warn "Exception in triggerPeerEvents", exc = exc.msg, peer = peerId
|
warn "Exception in triggerPeerEvents", exc = exc.msg, peer = peerId
|
||||||
|
|
||||||
proc expectConnection*(c: ConnManager, p: PeerId): Future[Connection] {.async.} =
|
proc expectConnection*(c: ConnManager, p: PeerId, dir: Direction): Future[Connection] {.async.} =
|
||||||
## Wait for a peer to connect to us. This will bypass the `MaxConnectionsPerPeer`
|
## Wait for a peer to connect to us. This will bypass the `MaxConnectionsPerPeer`
|
||||||
if p in c.expectedConnections:
|
let key = (p, dir)
|
||||||
raise LPError.newException("Already expecting a connection from that peer")
|
if key in c.expectedConnectionsOverLimit:
|
||||||
|
raise newException(AlreadyExpectingConnectionError, "Already expecting an incoming connection from that peer")
|
||||||
|
|
||||||
let future = newFuture[Connection]()
|
let future = newFuture[Connection]()
|
||||||
c.expectedConnections[p] = future
|
c.expectedConnectionsOverLimit[key] = future
|
||||||
|
|
||||||
try:
|
try:
|
||||||
return await future
|
return await future
|
||||||
finally:
|
finally:
|
||||||
c.expectedConnections.del(p)
|
c.expectedConnectionsOverLimit.del(key)
|
||||||
|
|
||||||
proc contains*(c: ConnManager, conn: Connection): bool =
|
proc contains*(c: ConnManager, conn: Connection): bool =
|
||||||
## checks if a connection is being tracked by the
|
## checks if a connection is being tracked by the
|
||||||
@ -412,14 +414,16 @@ proc storeConn*(c: ConnManager, conn: Connection)
|
|||||||
let peerId = conn.peerId
|
let peerId = conn.peerId
|
||||||
|
|
||||||
# we use getOrDefault in the if below instead of [] to avoid the KeyError
|
# we use getOrDefault in the if below instead of [] to avoid the KeyError
|
||||||
if peerId in c.expectedConnections and
|
if c.conns.getOrDefault(peerId).len > c.maxConnsPerPeer:
|
||||||
not(c.expectedConnections.getOrDefault(peerId).finished):
|
let key = (peerId, conn.dir)
|
||||||
c.expectedConnections.getOrDefault(peerId).complete(conn)
|
let expectedConn = c.expectedConnectionsOverLimit.getOrDefault(key)
|
||||||
elif c.conns.getOrDefault(peerId).len > c.maxConnsPerPeer:
|
if expectedConn != nil and not expectedConn.finished:
|
||||||
debug "Too many connections for peer",
|
expectedConn.complete(conn)
|
||||||
conn, conns = c.conns.getOrDefault(peerId).len
|
else:
|
||||||
|
debug "Too many connections for peer",
|
||||||
|
conn, conns = c.conns.getOrDefault(peerId).len
|
||||||
|
|
||||||
raise newTooManyConnectionsError()
|
raise newTooManyConnectionsError()
|
||||||
|
|
||||||
c.conns.mgetOrPut(peerId, HashSet[Connection]()).incl(conn)
|
c.conns.mgetOrPut(peerId, HashSet[Connection]()).incl(conn)
|
||||||
libp2p_peers.set(c.conns.len.int64)
|
libp2p_peers.set(c.conns.len.int64)
|
||||||
@ -562,8 +566,8 @@ proc close*(c: ConnManager) {.async.} =
|
|||||||
let muxed = c.muxed
|
let muxed = c.muxed
|
||||||
c.muxed.clear()
|
c.muxed.clear()
|
||||||
|
|
||||||
let expected = c.expectedConnections
|
let expected = c.expectedConnectionsOverLimit
|
||||||
c.expectedConnections.clear()
|
c.expectedConnectionsOverLimit.clear()
|
||||||
|
|
||||||
for _, fut in expected:
|
for _, fut in expected:
|
||||||
await fut.cancelAndWait()
|
await fut.cancelAndWait()
|
||||||
|
@ -58,10 +58,14 @@ method dialMe*(self: AutonatClient, switch: Switch, pid: PeerId, addrs: seq[Mult
|
|||||||
raise newException(AutonatError, "Unexpected error when dialling: " & err.msg, err)
|
raise newException(AutonatError, "Unexpected error when dialling: " & err.msg, err)
|
||||||
|
|
||||||
# To bypass maxConnectionsPerPeer
|
# To bypass maxConnectionsPerPeer
|
||||||
let incomingConnection = switch.connManager.expectConnection(pid)
|
let incomingConnection = switch.connManager.expectConnection(pid, In)
|
||||||
|
if incomingConnection.failed() and incomingConnection.error of AlreadyExpectingConnectionError:
|
||||||
|
raise newException(AutonatError, incomingConnection.error.msg)
|
||||||
defer:
|
defer:
|
||||||
await conn.close()
|
await conn.close()
|
||||||
incomingConnection.cancel() # Safer to always try to cancel cause we aren't sure if the peer dialled us or not
|
incomingConnection.cancel() # Safer to always try to cancel cause we aren't sure if the peer dialled us or not
|
||||||
|
if incomingConnection.completed():
|
||||||
|
await (await incomingConnection).close()
|
||||||
trace "sending Dial", addrs = switch.peerInfo.addrs
|
trace "sending Dial", addrs = switch.peerInfo.addrs
|
||||||
await conn.sendDial(switch.peerInfo.peerId, switch.peerInfo.addrs)
|
await conn.sendDial(switch.peerInfo.peerId, switch.peerInfo.addrs)
|
||||||
let response = getResponseOrRaise(AutonatMsg.decode(await conn.readLp(1024)))
|
let response = getResponseOrRaise(AutonatMsg.decode(await conn.readLp(1024)))
|
||||||
|
@ -63,7 +63,10 @@ proc tryDial(autonat: Autonat, conn: Connection, addrs: seq[MultiAddress]) {.asy
|
|||||||
var futs: seq[Future[Opt[MultiAddress]]]
|
var futs: seq[Future[Opt[MultiAddress]]]
|
||||||
try:
|
try:
|
||||||
# This is to bypass the per peer max connections limit
|
# This is to bypass the per peer max connections limit
|
||||||
let outgoingConnection = autonat.switch.connManager.expectConnection(conn.peerId)
|
let outgoingConnection = autonat.switch.connManager.expectConnection(conn.peerId, Out)
|
||||||
|
if outgoingConnection.failed() and outgoingConnection.error of AlreadyExpectingConnectionError:
|
||||||
|
await conn.sendResponseError(DialRefused, outgoingConnection.error.msg)
|
||||||
|
return
|
||||||
# Safer to always try to cancel cause we aren't sure if the connection was established
|
# Safer to always try to cancel cause we aren't sure if the connection was established
|
||||||
defer: outgoingConnection.cancel()
|
defer: outgoingConnection.cancel()
|
||||||
# tryDial is to bypass the global max connections limit
|
# tryDial is to bypass the global max connections limit
|
||||||
|
@ -80,6 +80,9 @@ proc hasEnoughIncomingSlots(switch: Switch): bool =
|
|||||||
# we leave some margin instead of comparing to 0 as a peer could connect to us while we are asking for the dial back
|
# we leave some margin instead of comparing to 0 as a peer could connect to us while we are asking for the dial back
|
||||||
return switch.connManager.slotsAvailable(In) >= 2
|
return switch.connManager.slotsAvailable(In) >= 2
|
||||||
|
|
||||||
|
proc doesPeerHaveIncomingConn(switch: Switch, peerId: PeerId): bool =
|
||||||
|
return switch.connManager.selectConn(peerId, In) != nil
|
||||||
|
|
||||||
proc handleAnswer(self: AutonatService, ans: NetworkReachability) {.async.} =
|
proc handleAnswer(self: AutonatService, ans: NetworkReachability) {.async.} =
|
||||||
|
|
||||||
if ans == Unknown:
|
if ans == Unknown:
|
||||||
@ -104,6 +107,10 @@ proc handleAnswer(self: AutonatService, ans: NetworkReachability) {.async.} =
|
|||||||
proc askPeer(self: AutonatService, switch: Switch, peerId: PeerId): Future[NetworkReachability] {.async.} =
|
proc askPeer(self: AutonatService, switch: Switch, peerId: PeerId): Future[NetworkReachability] {.async.} =
|
||||||
logScope:
|
logScope:
|
||||||
peerId = $peerId
|
peerId = $peerId
|
||||||
|
|
||||||
|
if doesPeerHaveIncomingConn(switch, peerId):
|
||||||
|
return Unknown
|
||||||
|
|
||||||
if not hasEnoughIncomingSlots(switch):
|
if not hasEnoughIncomingSlots(switch):
|
||||||
debug "No incoming slots available, not asking peer", incomingSlotsAvailable=switch.connManager.slotsAvailable(In)
|
debug "No incoming slots available, not asking peer", incomingSlotsAvailable=switch.connManager.slotsAvailable(In)
|
||||||
return Unknown
|
return Unknown
|
||||||
@ -152,10 +159,7 @@ method setup*(self: AutonatService, switch: Switch): Future[bool] {.async.} =
|
|||||||
if hasBeenSetup:
|
if hasBeenSetup:
|
||||||
if self.askNewConnectedPeers:
|
if self.askNewConnectedPeers:
|
||||||
self.newConnectedPeerHandler = proc (peerId: PeerId, event: PeerEvent): Future[void] {.async.} =
|
self.newConnectedPeerHandler = proc (peerId: PeerId, event: PeerEvent): Future[void] {.async.} =
|
||||||
if switch.connManager.selectConn(peerId, In) != nil: # no need to ask an incoming peer
|
|
||||||
return
|
|
||||||
discard askPeer(self, switch, peerId)
|
discard askPeer(self, switch, peerId)
|
||||||
await self.callHandler()
|
|
||||||
switch.connManager.addPeerEventHandler(self.newConnectedPeerHandler, PeerEventKind.Joined)
|
switch.connManager.addPeerEventHandler(self.newConnectedPeerHandler, PeerEventKind.Joined)
|
||||||
if self.scheduleInterval.isSome():
|
if self.scheduleInterval.isSome():
|
||||||
self.scheduleHandle = schedule(self, switch, self.scheduleInterval.get())
|
self.scheduleHandle = schedule(self, switch, self.scheduleInterval.get())
|
||||||
|
@ -277,6 +277,94 @@ suite "Autonat Service":
|
|||||||
await allFuturesThrowing(
|
await allFuturesThrowing(
|
||||||
switch1.stop(), switch2.stop())
|
switch1.stop(), switch2.stop())
|
||||||
|
|
||||||
|
asyncTest "Must work when peers ask each other at the same time with max 1 conn per peer":
|
||||||
|
let autonatService1 = AutonatService.new(AutonatClient.new(), newRng(), some(500.millis), maxQueueSize = 3)
|
||||||
|
let autonatService2 = AutonatService.new(AutonatClient.new(), newRng(), some(500.millis), maxQueueSize = 3)
|
||||||
|
let autonatService3 = AutonatService.new(AutonatClient.new(), newRng(), some(500.millis), maxQueueSize = 3)
|
||||||
|
|
||||||
|
let switch1 = createSwitch(autonatService1, maxConnsPerPeer = 0)
|
||||||
|
let switch2 = createSwitch(autonatService2, maxConnsPerPeer = 0)
|
||||||
|
let switch3 = createSwitch(autonatService2, maxConnsPerPeer = 0)
|
||||||
|
|
||||||
|
let awaiter1 = newFuture[void]()
|
||||||
|
let awaiter2 = newFuture[void]()
|
||||||
|
let awaiter3 = newFuture[void]()
|
||||||
|
|
||||||
|
proc statusAndConfidenceHandler1(networkReachability: NetworkReachability, confidence: Option[float]) {.gcsafe, async.} =
|
||||||
|
if networkReachability == NetworkReachability.Reachable and confidence.isSome() and confidence.get() == 1:
|
||||||
|
if not awaiter1.finished:
|
||||||
|
awaiter1.complete()
|
||||||
|
|
||||||
|
proc statusAndConfidenceHandler2(networkReachability: NetworkReachability, confidence: Option[float]) {.gcsafe, async.} =
|
||||||
|
if networkReachability == NetworkReachability.Reachable and confidence.isSome() and confidence.get() == 1:
|
||||||
|
if not awaiter2.finished:
|
||||||
|
awaiter2.complete()
|
||||||
|
|
||||||
|
check autonatService1.networkReachability() == NetworkReachability.Unknown
|
||||||
|
check autonatService2.networkReachability() == NetworkReachability.Unknown
|
||||||
|
|
||||||
|
autonatService1.statusAndConfidenceHandler(statusAndConfidenceHandler1)
|
||||||
|
autonatService2.statusAndConfidenceHandler(statusAndConfidenceHandler2)
|
||||||
|
|
||||||
|
await switch1.start()
|
||||||
|
await switch2.start()
|
||||||
|
await switch3.start()
|
||||||
|
|
||||||
|
await switch1.connect(switch2.peerInfo.peerId, switch2.peerInfo.addrs)
|
||||||
|
await switch2.connect(switch1.peerInfo.peerId, switch1.peerInfo.addrs)
|
||||||
|
await switch2.connect(switch3.peerInfo.peerId, switch3.peerInfo.addrs)
|
||||||
|
|
||||||
|
await awaiter1
|
||||||
|
await awaiter2
|
||||||
|
|
||||||
|
check autonatService1.networkReachability() == NetworkReachability.Reachable
|
||||||
|
check autonatService2.networkReachability() == NetworkReachability.Reachable
|
||||||
|
check libp2p_autonat_reachability_confidence.value(["Reachable"]) == 1
|
||||||
|
|
||||||
|
await allFuturesThrowing(
|
||||||
|
switch1.stop(), switch2.stop(), switch3.stop())
|
||||||
|
|
||||||
|
asyncTest "Must work for one peer when two peers ask each other at the same time with max 1 conn per peer":
|
||||||
|
let autonatService1 = AutonatService.new(AutonatClient.new(), newRng(), some(500.millis), maxQueueSize = 3)
|
||||||
|
let autonatService2 = AutonatService.new(AutonatClient.new(), newRng(), some(500.millis), maxQueueSize = 3)
|
||||||
|
|
||||||
|
let switch1 = createSwitch(autonatService1, maxConnsPerPeer = 0)
|
||||||
|
let switch2 = createSwitch(autonatService2, maxConnsPerPeer = 0)
|
||||||
|
|
||||||
|
let awaiter1 = newFuture[void]()
|
||||||
|
|
||||||
|
proc statusAndConfidenceHandler1(networkReachability: NetworkReachability, confidence: Option[float]) {.gcsafe, async.} =
|
||||||
|
if networkReachability == NetworkReachability.Reachable and confidence.isSome() and confidence.get() == 1:
|
||||||
|
if not awaiter1.finished:
|
||||||
|
awaiter1.complete()
|
||||||
|
|
||||||
|
check autonatService1.networkReachability() == NetworkReachability.Unknown
|
||||||
|
|
||||||
|
autonatService1.statusAndConfidenceHandler(statusAndConfidenceHandler1)
|
||||||
|
|
||||||
|
await switch1.start()
|
||||||
|
await switch2.start()
|
||||||
|
|
||||||
|
await switch1.connect(switch2.peerInfo.peerId, switch2.peerInfo.addrs)
|
||||||
|
try:
|
||||||
|
# We allow a temp conn for the peer to dial us. It could use this conn to just connect to us and not dial.
|
||||||
|
# We don't care if it fails at this point or not. But this conn must be closed eventually.
|
||||||
|
# Bellow we check that there's only one connection between the peers
|
||||||
|
await switch2.connect(switch1.peerInfo.peerId, switch1.peerInfo.addrs, reuseConnection = false)
|
||||||
|
except CatchableError:
|
||||||
|
discard
|
||||||
|
|
||||||
|
await awaiter1
|
||||||
|
|
||||||
|
check autonatService1.networkReachability() == NetworkReachability.Reachable
|
||||||
|
check libp2p_autonat_reachability_confidence.value(["Reachable"]) == 1
|
||||||
|
|
||||||
|
# Make sure remote peer can't create a connection to us
|
||||||
|
check switch1.connManager.connCount(switch2.peerInfo.peerId) == 1
|
||||||
|
|
||||||
|
await allFuturesThrowing(
|
||||||
|
switch1.stop(), switch2.stop())
|
||||||
|
|
||||||
asyncTest "Must work with low maxConnections":
|
asyncTest "Must work with low maxConnections":
|
||||||
let autonatService = AutonatService.new(AutonatClient.new(), newRng(), some(1.seconds), maxQueueSize = 1)
|
let autonatService = AutonatService.new(AutonatClient.new(), newRng(), some(1.seconds), maxQueueSize = 1)
|
||||||
|
|
||||||
|
@ -208,15 +208,15 @@ suite "Connection Manager":
|
|||||||
expect TooManyConnectionsError:
|
expect TooManyConnectionsError:
|
||||||
connMngr.storeConn(conns[0])
|
connMngr.storeConn(conns[0])
|
||||||
|
|
||||||
let waitedConn1 = connMngr.expectConnection(peerId)
|
let waitedConn1 = connMngr.expectConnection(peerId, In)
|
||||||
|
|
||||||
expect LPError:
|
expect AlreadyExpectingConnectionError:
|
||||||
discard await connMngr.expectConnection(peerId)
|
discard await connMngr.expectConnection(peerId, In)
|
||||||
|
|
||||||
await waitedConn1.cancelAndWait()
|
await waitedConn1.cancelAndWait()
|
||||||
let
|
let
|
||||||
waitedConn2 = connMngr.expectConnection(peerId)
|
waitedConn2 = connMngr.expectConnection(peerId, In)
|
||||||
waitedConn3 = connMngr.expectConnection(PeerId.init(PrivateKey.random(ECDSA, (newRng())[]).tryGet()).tryGet())
|
waitedConn3 = connMngr.expectConnection(PeerId.init(PrivateKey.random(ECDSA, (newRng())[]).tryGet()).tryGet(), In)
|
||||||
conn = getConnection(peerId)
|
conn = getConnection(peerId)
|
||||||
connMngr.storeConn(conn)
|
connMngr.storeConn(conn)
|
||||||
check (await waitedConn2) == conn
|
check (await waitedConn2) == conn
|
||||||
|
Loading…
x
Reference in New Issue
Block a user