90 lines
2.8 KiB
Nim
Raw Normal View History

2019-09-03 14:40:51 -06:00
## Nim-LibP2P
2019-09-24 11:48:23 -06:00
## Copyright (c) 2019 Status Research & Development GmbH
2019-09-03 14:40:51 -06:00
## 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: [Defect].}
import chronos, chronicles
2019-09-12 11:07:34 -06:00
import ../protocols/protocol,
../stream/connection,
../errors
2019-09-03 14:40:51 -06:00
logScope:
2020-12-01 11:34:27 -06:00
topics = "libp2p muxer"
const
DefaultChanTimeout* = 5.minutes
2019-09-03 14:40:51 -06:00
type
MuxerError* = object of LPError
StreamHandler* = proc(conn: Connection): Future[void] {.gcsafe, raises: [Defect].}
MuxerHandler* = proc(muxer: Muxer): Future[void] {.gcsafe, raises: [Defect].}
2019-09-03 14:40:51 -06:00
Muxer* = ref object of RootObj
streamHandler*: StreamHandler
2019-09-03 14:40:51 -06:00
connection*: Connection
2019-11-06 12:09:48 -06:00
# user provider proc that returns a constructed Muxer
MuxerConstructor* = proc(conn: Connection): Muxer {.gcsafe, closure, raises: [Defect].}
2019-11-06 12:09:48 -06:00
2019-09-03 14:40:51 -06:00
# this wraps a creator proc that knows how to make muxers
MuxerProvider* = ref object of LPProtocol
2019-11-06 12:09:48 -06:00
newMuxer*: MuxerConstructor
streamHandler*: StreamHandler # triggered every time there is a new stream, called for any muxer instance
muxerHandler*: MuxerHandler # triggered every time there is a new muxed connection created
2019-09-06 00:51:19 -06:00
func shortLog*(m: Muxer): auto = shortLog(m.connection)
chronicles.formatIt(Muxer): shortLog(it)
2019-11-06 12:09:48 -06:00
# muxer interface
method newStream*(m: Muxer, name: string = "", lazy: bool = false):
Future[Connection] {.base, async, gcsafe.} = discard
2019-09-06 00:51:19 -06:00
method close*(m: Muxer) {.base, async, gcsafe.} = discard
method handle*(m: Muxer): Future[void] {.base, async, gcsafe.} = discard
2019-09-03 14:40:51 -06:00
proc new*(
T: typedesc[MuxerProvider],
creator: MuxerConstructor,
codec: string): T {.gcsafe.} =
let muxerProvider = T(newMuxer: creator)
muxerProvider.codec = codec
muxerProvider.init()
muxerProvider
proc newMuxerProvider*(creator: MuxerConstructor, codec: string): MuxerProvider {.gcsafe, deprecated: "use MuxerProvider.new".} =
MuxerProvider.new(creator, codec)
2019-09-06 00:51:19 -06:00
2019-09-03 14:40:51 -06:00
method init(c: MuxerProvider) =
proc handler(conn: Connection, proto: string) {.async, gcsafe, closure.} =
trace "starting muxer handler", proto=proto, conn
2020-05-23 10:50:05 -06:00
try:
let
muxer = c.newMuxer(conn)
2020-05-23 10:50:05 -06:00
if not isNil(c.streamHandler):
muxer.streamHandler = c.streamHandler
2020-05-23 10:50:05 -06:00
var futs = newSeq[Future[void]]()
futs &= muxer.handle()
2020-05-23 10:50:05 -06:00
# finally await both the futures
if not isNil(c.muxerHandler):
futs &= c.muxerHandler(muxer)
checkFutures(await allFinished(futs))
except CancelledError as exc:
raise exc
2020-05-23 10:50:05 -06:00
except CatchableError as exc:
trace "exception in muxer handler", exc = exc.msg, conn, proto
finally:
await conn.close()
2019-09-03 14:40:51 -06:00
c.handler = handler