95 lines
3.2 KiB
Nim
95 lines
3.2 KiB
Nim
# Nim-LibP2P
|
|
# Copyright (c) 2023 Status Research & Development GmbH
|
|
# Licensed under either of
|
|
# * Apache License, version 2.0, ([LICENSE-APACHE](LICENSE-APACHE))
|
|
# * MIT license ([LICENSE-MIT](LICENSE-MIT))
|
|
# at your option.
|
|
# This file may not be copied, modified, or distributed except according to
|
|
# those terms.
|
|
|
|
{.push raises: [].}
|
|
|
|
import std/[sets, sequtils]
|
|
import stew/[results, objects]
|
|
import chronos, chronicles
|
|
|
|
import core
|
|
import
|
|
../../protocol, ../../../stream/connection, ../../../switch, ../../../utils/future
|
|
|
|
export chronicles
|
|
|
|
type Dcutr* = ref object of LPProtocol
|
|
|
|
logScope:
|
|
topics = "libp2p dcutr"
|
|
|
|
proc new*(
|
|
T: typedesc[Dcutr],
|
|
switch: Switch,
|
|
connectTimeout = 15.seconds,
|
|
maxDialableAddrs = 8,
|
|
): T =
|
|
proc handleStream(stream: Connection, proto: string) {.async.} =
|
|
var peerDialableAddrs: seq[MultiAddress]
|
|
try:
|
|
let connectMsg = DcutrMsg.decode(await stream.readLp(1024))
|
|
debug "Dcutr receiver received a Connect message.", connectMsg
|
|
|
|
var ourAddrs = switch.peerStore.getMostObservedProtosAndPorts()
|
|
# likely empty when the peer is reachable
|
|
if ourAddrs.len == 0:
|
|
# this list should be the same as the peer's public addrs when it is reachable
|
|
ourAddrs =
|
|
switch.peerInfo.listenAddrs.mapIt(switch.peerStore.guessDialableAddr(it))
|
|
var ourDialableAddrs = getHolePunchableAddrs(ourAddrs)
|
|
if ourDialableAddrs.len == 0:
|
|
debug "Dcutr receiver has no supported dialable addresses. Aborting Dcutr.",
|
|
ourAddrs
|
|
return
|
|
|
|
await stream.send(MsgType.Connect, ourAddrs)
|
|
debug "Dcutr receiver has sent a Connect message back."
|
|
let syncMsg = DcutrMsg.decode(await stream.readLp(1024))
|
|
debug "Dcutr receiver has received a Sync message.", syncMsg
|
|
|
|
peerDialableAddrs = getHolePunchableAddrs(connectMsg.addrs)
|
|
if peerDialableAddrs.len == 0:
|
|
debug "Dcutr initiator has no supported dialable addresses to connect to. Aborting Dcutr.",
|
|
addrs = connectMsg.addrs
|
|
return
|
|
|
|
if peerDialableAddrs.len > maxDialableAddrs:
|
|
peerDialableAddrs = peerDialableAddrs[0 ..< maxDialableAddrs]
|
|
var futs = peerDialableAddrs.mapIt(
|
|
switch.connect(
|
|
stream.peerId,
|
|
@[it],
|
|
forceDial = true,
|
|
reuseConnection = false,
|
|
dir = Direction.Out,
|
|
)
|
|
)
|
|
try:
|
|
discard await anyCompleted(futs).wait(connectTimeout)
|
|
debug "Dcutr receiver has directly connected to the remote peer."
|
|
finally:
|
|
for fut in futs:
|
|
fut.cancel()
|
|
except CancelledError as err:
|
|
raise err
|
|
except AllFuturesFailedError as err:
|
|
debug "Dcutr receiver could not connect to the remote peer, " &
|
|
"all connect attempts failed", peerDialableAddrs, description = err.msg
|
|
except AsyncTimeoutError as err:
|
|
debug "Dcutr receiver could not connect to the remote peer, " &
|
|
"all connect attempts timed out", peerDialableAddrs, description = err.msg
|
|
except CatchableError as err:
|
|
warn "Unexpected error when Dcutr receiver tried to connect " &
|
|
"to the remote peer", description = err.msg
|
|
|
|
let self = T()
|
|
self.handler = handleStream
|
|
self.codec = DcutrCodec
|
|
self
|