241 lines
6.9 KiB
Nim
241 lines
6.9 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.
|
|
|
|
when (NimMajor, NimMinor) < (1, 4):
|
|
{.push raises: [Defect].}
|
|
else:
|
|
{.push raises: [].}
|
|
|
|
import std/[tables, sequtils]
|
|
import pkg/[chronos, chronicles, metrics]
|
|
|
|
import ../upgrademngrs/upgrade,
|
|
../muxers/muxer
|
|
|
|
export Upgrade
|
|
|
|
logScope:
|
|
topics = "libp2p muxedupgrade"
|
|
|
|
type
|
|
MuxedUpgrade* = ref object of Upgrade
|
|
muxers*: seq[MuxerProvider]
|
|
streamHandler*: StreamHandler
|
|
|
|
proc getMuxerByCodec(self: MuxedUpgrade, muxerName: string): MuxerProvider =
|
|
for m in self.muxers:
|
|
if muxerName in m.codecs:
|
|
return m
|
|
|
|
proc identify*(
|
|
self: MuxedUpgrade,
|
|
muxer: Muxer) {.async, gcsafe.} =
|
|
# new stream for identify
|
|
var stream = await muxer.newStream()
|
|
if stream == nil:
|
|
return
|
|
|
|
try:
|
|
await self.identify(stream)
|
|
when defined(libp2p_agents_metrics):
|
|
muxer.connection.shortAgent = stream.shortAgent
|
|
finally:
|
|
await stream.closeWithEOF()
|
|
|
|
proc mux*(
|
|
self: MuxedUpgrade,
|
|
conn: Connection): Future[Muxer] {.async, gcsafe.} =
|
|
## mux outgoing connection
|
|
|
|
trace "Muxing connection", conn
|
|
if self.muxers.len == 0:
|
|
warn "no muxers registered, skipping upgrade flow", conn
|
|
return
|
|
|
|
let muxerName = await self.ms.select(conn, self.muxers.mapIt(it.codec))
|
|
if muxerName.len == 0 or muxerName == "na":
|
|
debug "no muxer available, early exit", conn
|
|
return
|
|
|
|
trace "Found a muxer", conn, muxerName
|
|
|
|
# create new muxer for connection
|
|
let muxer = self.getMuxerByCodec(muxerName).newMuxer(conn)
|
|
|
|
# install stream handler
|
|
muxer.streamHandler = self.streamHandler
|
|
|
|
self.connManager.storeConn(conn)
|
|
|
|
# store it in muxed connections if we have a peer for it
|
|
self.connManager.storeMuxer(muxer, muxer.handle()) # store muxer and start read loop
|
|
|
|
try:
|
|
await self.identify(muxer)
|
|
except CatchableError as exc:
|
|
# Identify is non-essential, though if it fails, it might indicate that
|
|
# the connection was closed already - this will be picked up by the read
|
|
# loop
|
|
debug "Could not identify connection", conn, msg = exc.msg
|
|
|
|
return muxer
|
|
|
|
method upgradeOutgoing*(
|
|
self: MuxedUpgrade,
|
|
conn: Connection,
|
|
peerId: Opt[PeerId]): Future[Connection] {.async, gcsafe.} =
|
|
trace "Upgrading outgoing connection", conn
|
|
|
|
let sconn = await self.secure(conn, peerId) # secure the connection
|
|
if isNil(sconn):
|
|
raise newException(UpgradeFailedError,
|
|
"unable to secure connection, stopping upgrade")
|
|
|
|
let muxer = await self.mux(sconn) # mux it if possible
|
|
if muxer == nil:
|
|
# TODO this might be relaxed in the future
|
|
raise newException(UpgradeFailedError,
|
|
"a muxer is required for outgoing connections")
|
|
|
|
when defined(libp2p_agents_metrics):
|
|
conn.shortAgent = muxer.connection.shortAgent
|
|
|
|
if sconn.closed():
|
|
await sconn.close()
|
|
raise newException(UpgradeFailedError,
|
|
"Connection closed or missing peer info, stopping upgrade")
|
|
|
|
trace "Upgraded outgoing connection", conn, sconn
|
|
|
|
return sconn
|
|
|
|
method upgradeIncoming*(
|
|
self: MuxedUpgrade,
|
|
incomingConn: Connection) {.async, gcsafe.} = # noraises
|
|
trace "Upgrading incoming connection", incomingConn
|
|
let ms = MultistreamSelect.new()
|
|
|
|
# secure incoming connections
|
|
proc securedHandler(conn: Connection,
|
|
proto: string)
|
|
{.async, gcsafe, closure.} =
|
|
trace "Starting secure handler", conn
|
|
let secure = self.secureManagers.filterIt(it.codec == proto)[0]
|
|
|
|
var cconn = conn
|
|
try:
|
|
var sconn = await secure.secure(cconn, false, Opt.none(PeerId))
|
|
if isNil(sconn):
|
|
return
|
|
|
|
cconn = sconn
|
|
# add the muxer
|
|
for muxer in self.muxers:
|
|
ms.addHandler(muxer.codecs, muxer)
|
|
|
|
# handle subsequent secure requests
|
|
await ms.handle(cconn)
|
|
except CatchableError as exc:
|
|
debug "Exception in secure handler during incoming upgrade", msg = exc.msg, conn
|
|
if not cconn.isUpgraded:
|
|
cconn.upgrade(exc)
|
|
finally:
|
|
if not isNil(cconn):
|
|
await cconn.close()
|
|
|
|
trace "Stopped secure handler", conn
|
|
|
|
try:
|
|
if (await ms.select(incomingConn)): # just handshake
|
|
# add the secure handlers
|
|
for k in self.secureManagers:
|
|
ms.addHandler(k.codec, securedHandler)
|
|
|
|
# handle un-secured connections
|
|
# we handshaked above, set this ms handler as active
|
|
await ms.handle(incomingConn, active = true)
|
|
except CatchableError as exc:
|
|
debug "Exception upgrading incoming", exc = exc.msg
|
|
if not incomingConn.isUpgraded:
|
|
incomingConn.upgrade(exc)
|
|
finally:
|
|
if not isNil(incomingConn):
|
|
await incomingConn.close()
|
|
|
|
proc muxerHandler(
|
|
self: MuxedUpgrade,
|
|
muxer: Muxer) {.async, gcsafe.} =
|
|
let
|
|
conn = muxer.connection
|
|
|
|
# store incoming connection
|
|
self.connManager.storeConn(conn)
|
|
|
|
# store muxer and muxed connection
|
|
self.connManager.storeMuxer(muxer)
|
|
|
|
try:
|
|
await self.identify(muxer)
|
|
when defined(libp2p_agents_metrics):
|
|
#TODO Passing data between layers is a pain
|
|
if muxer.connection of SecureConn:
|
|
let secureConn = (SecureConn)muxer.connection
|
|
secureConn.stream.shortAgent = muxer.connection.shortAgent
|
|
except IdentifyError as exc:
|
|
# Identify is non-essential, though if it fails, it might indicate that
|
|
# the connection was closed already - this will be picked up by the read
|
|
# loop
|
|
debug "Could not identify connection", conn, msg = exc.msg
|
|
except LPStreamClosedError as exc:
|
|
debug "Identify stream closed", conn, msg = exc.msg
|
|
except LPStreamEOFError as exc:
|
|
debug "Identify stream EOF", conn, msg = exc.msg
|
|
except CancelledError as exc:
|
|
await muxer.close()
|
|
raise exc
|
|
except CatchableError as exc:
|
|
await muxer.close()
|
|
trace "Exception in muxer handler", conn, msg = exc.msg
|
|
|
|
proc new*(
|
|
T: type MuxedUpgrade,
|
|
identity: Identify,
|
|
muxers: seq[MuxerProvider],
|
|
secureManagers: openArray[Secure] = [],
|
|
connManager: ConnManager,
|
|
ms: MultistreamSelect): T =
|
|
|
|
let upgrader = T(
|
|
identity: identity,
|
|
muxers: muxers,
|
|
secureManagers: @secureManagers,
|
|
connManager: connManager,
|
|
ms: ms)
|
|
|
|
upgrader.streamHandler = proc(conn: Connection)
|
|
{.async, gcsafe, raises: [Defect].} =
|
|
trace "Starting stream handler", conn
|
|
try:
|
|
await upgrader.ms.handle(conn) # handle incoming connection
|
|
except CancelledError as exc:
|
|
raise exc
|
|
except CatchableError as exc:
|
|
trace "exception in stream handler", conn, msg = exc.msg
|
|
finally:
|
|
await conn.closeWithEOF()
|
|
trace "Stream handler done", conn
|
|
|
|
for _, val in muxers:
|
|
val.streamHandler = upgrader.streamHandler
|
|
val.muxerHandler = proc(muxer: Muxer): Future[void]
|
|
{.raises: [Defect].} =
|
|
upgrader.muxerHandler(muxer)
|
|
|
|
return upgrader
|