mirror of
https://github.com/codex-storage/nim-libp2p.git
synced 2025-01-14 13:04:50 +00:00
Restore interop with Lighthouse by preventing concurrent meshsub dials
This commit is contained in:
parent
833a5b8e57
commit
60122a044c
@ -46,6 +46,7 @@ type
|
|||||||
recvdRpcCache: TimedCache[string] # cache for already received messages
|
recvdRpcCache: TimedCache[string] # cache for already received messages
|
||||||
observers*: ref seq[PubSubObserver] # ref as in smart_ptr
|
observers*: ref seq[PubSubObserver] # ref as in smart_ptr
|
||||||
subscribed*: bool # are we subscribed to this peer
|
subscribed*: bool # are we subscribed to this peer
|
||||||
|
dialLock: AsyncLock
|
||||||
|
|
||||||
RPCHandler* = proc(peer: PubSubPeer, msg: seq[RPCMsg]): Future[void] {.gcsafe.}
|
RPCHandler* = proc(peer: PubSubPeer, msg: seq[RPCMsg]): Future[void] {.gcsafe.}
|
||||||
|
|
||||||
@ -141,32 +142,37 @@ proc getSendConn(p: PubSubPeer): Future[Connection] {.async.} =
|
|||||||
# Careful, p.sendConn might change after here!
|
# Careful, p.sendConn might change after here!
|
||||||
await current.close() # TODO this might be unnecessary
|
await current.close() # TODO this might be unnecessary
|
||||||
|
|
||||||
|
try:
|
||||||
|
# Testing has demonstrated that when we perform concurrent meshsub dials
|
||||||
|
# and later close one of them, other implementations such as rust-libp2p
|
||||||
|
# become deaf to our messages (potentially due to the clean-up associated
|
||||||
|
# with closing connections). To prevent this, we use a lock that ensures
|
||||||
|
# that only a single dial will be performed for each peer:
|
||||||
|
await p.dialLock.acquire()
|
||||||
|
|
||||||
|
# Another concurrent dial may have populated p.sendConn
|
||||||
|
if p.sendConn != nil:
|
||||||
|
let current = p.sendConn
|
||||||
|
if not current.isNil:
|
||||||
|
if not (current.closed() or current.atEof):
|
||||||
|
# The existing send connection looks like it might work - reuse it
|
||||||
|
trace "Reusing existing connection", oid = $current.oid
|
||||||
|
return current
|
||||||
|
|
||||||
# Grab a new send connection
|
# Grab a new send connection
|
||||||
let newConn = await p.switch.dial(p.peerId, p.codec) # ...and here
|
let newConn = await p.switch.dial(p.peerId, p.codec) # ...and here
|
||||||
if newConn == nil:
|
if newConn.isNil:
|
||||||
return p.sendConn # A concurrent attempt perhaps succeeded?
|
return nil
|
||||||
|
|
||||||
# Because of the awaits above, a concurrent `getSendConn` call might have
|
|
||||||
# set up a send connection already. We cannot take a lock here because
|
|
||||||
# it might block the reading of data from mplex which will cause its
|
|
||||||
# backpressure handling to stop reading from the socket and thus prevent the
|
|
||||||
# channel negotiation from finishing
|
|
||||||
if p.sendConn != nil and not(p.sendConn.closed or p.sendConn.atEof):
|
|
||||||
let current = p.sendConn
|
|
||||||
# Either the new or the old connection could potentially be closed - it's
|
|
||||||
# slightly easier to sequence the closing of the new connection because the
|
|
||||||
# old one might still be in use.
|
|
||||||
trace "Closing redundant connection",
|
|
||||||
oid = $current.oid, newConn = $newConn.oid
|
|
||||||
await newConn.close()
|
|
||||||
return current
|
|
||||||
|
|
||||||
trace "Caching new send connection", oid = $newConn.oid
|
trace "Caching new send connection", oid = $newConn.oid
|
||||||
p.sendConn = newConn
|
p.sendConn = newConn
|
||||||
asyncCheck p.handle(newConn) # start a read loop on the new connection
|
asyncCheck p.handle(newConn) # start a read loop on the new connection
|
||||||
|
|
||||||
return newConn
|
return newConn
|
||||||
|
|
||||||
|
finally:
|
||||||
|
if p.dialLock.locked:
|
||||||
|
p.dialLock.release()
|
||||||
|
|
||||||
proc send*(
|
proc send*(
|
||||||
p: PubSubPeer,
|
p: PubSubPeer,
|
||||||
msg: RPCMsg,
|
msg: RPCMsg,
|
||||||
@ -207,11 +213,11 @@ proc send*(
|
|||||||
if conn == nil:
|
if conn == nil:
|
||||||
debug "Couldn't get send connection, dropping message"
|
debug "Couldn't get send connection, dropping message"
|
||||||
return
|
return
|
||||||
trace "sending encoded msgs to peer"
|
trace "sending encoded msgs to peer", connId = $conn.oid
|
||||||
await conn.writeLp(encoded).wait(timeout)
|
await conn.writeLp(encoded).wait(timeout)
|
||||||
|
|
||||||
p.sentRpcCache.put(digest)
|
p.sentRpcCache.put(digest)
|
||||||
trace "sent pubsub message to remote"
|
trace "sent pubsub message to remote", connId = $conn.oid
|
||||||
|
|
||||||
when defined(libp2p_expensive_metrics):
|
when defined(libp2p_expensive_metrics):
|
||||||
for x in mm.messages:
|
for x in mm.messages:
|
||||||
@ -240,3 +246,4 @@ proc newPubSubPeer*(peerId: PeerID,
|
|||||||
result.peerId = peerId
|
result.peerId = peerId
|
||||||
result.sentRpcCache = newTimedCache[string](2.minutes)
|
result.sentRpcCache = newTimedCache[string](2.minutes)
|
||||||
result.recvdRpcCache = newTimedCache[string](2.minutes)
|
result.recvdRpcCache = newTimedCache[string](2.minutes)
|
||||||
|
result.dialLock = newAsyncLock()
|
||||||
|
Loading…
x
Reference in New Issue
Block a user