# 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. ## Length Prefixed stream implementation {.push gcsafe.} {.push raises: [].} import std/oids import stew/byteutils import chronicles, chronos, metrics import ../varint, ../peerinfo, ../multiaddress, ../utility, ../errors export errors declareGauge(libp2p_open_streams, "open stream instances", labels = ["type", "dir"]) export oids logScope: topics = "libp2p lpstream" const LPStreamTrackerName* = "LPStream" Eof* = @[] type Direction* {.pure.} = enum In, Out LPStream* = ref object of RootObj closeEvent*: AsyncEvent isClosed*: bool isEof*: bool objName*: string oid*: Oid dir*: Direction closedWithEOF: bool # prevent concurrent calls LPStreamError* = object of LPError LPStreamIncompleteError* = object of LPStreamError LPStreamIncorrectDefect* = object of Defect LPStreamLimitError* = object of LPStreamError LPStreamReadError* = object of LPStreamError par*: ref CatchableError LPStreamWriteError* = object of LPStreamError par*: ref CatchableError LPStreamEOFError* = object of LPStreamError # X | Read | Write # Local close | Works | LPStreamClosedError # Remote close | LPStreamRemoteClosedError | Works # Local reset | LPStreamClosedError | LPStreamClosedError # Remote reset | LPStreamResetError | LPStreamResetError # Connection down | LPStreamConnDown | LPStreamConnDownError LPStreamResetError* = object of LPStreamEOFError LPStreamClosedError* = object of LPStreamEOFError LPStreamRemoteClosedError* = object of LPStreamEOFError LPStreamConnDownError* = object of LPStreamEOFError InvalidVarintError* = object of LPStreamError MaxSizeError* = object of LPStreamError StreamTracker* = ref object of TrackerBase opened*: uint64 closed*: uint64 proc setupStreamTracker*(name: string): StreamTracker = let tracker = new StreamTracker proc dumpTracking(): string {.gcsafe.} = return "Opened " & tracker.id & ": " & $tracker.opened & "\n" & "Closed " & tracker.id & ": " & $tracker.closed proc leakTransport(): bool {.gcsafe.} = return (tracker.opened != tracker.closed) tracker.id = name tracker.opened = 0 tracker.closed = 0 tracker.dump = dumpTracking tracker.isLeaked = leakTransport addTracker(name, tracker) return tracker proc getStreamTracker(name: string): StreamTracker {.gcsafe.} = result = cast[StreamTracker](getTracker(name)) if isNil(result): result = setupStreamTracker(name) proc newLPStreamReadError*(p: ref CatchableError): ref LPStreamReadError = var w = newException(LPStreamReadError, "Read stream failed") w.msg = w.msg & ", originated from [" & $p.name & "] " & p.msg w.par = p result = w proc newLPStreamReadError*(msg: string): ref LPStreamReadError = newException(LPStreamReadError, msg) proc newLPStreamWriteError*(p: ref CatchableError): ref LPStreamWriteError = var w = newException(LPStreamWriteError, "Write stream failed") w.msg = w.msg & ", originated from [" & $p.name & "] " & p.msg w.par = p result = w proc newLPStreamIncompleteError*(): ref LPStreamIncompleteError = result = newException(LPStreamIncompleteError, "Incomplete data received") proc newLPStreamLimitError*(): ref LPStreamLimitError = result = newException(LPStreamLimitError, "Buffer limit reached") proc newLPStreamIncorrectDefect*(m: string): ref LPStreamIncorrectDefect = result = newException(LPStreamIncorrectDefect, m) proc newLPStreamEOFError*(): ref LPStreamEOFError = result = newException(LPStreamEOFError, "Stream EOF!") proc newLPStreamResetError*(): ref LPStreamResetError = result = newException(LPStreamResetError, "Stream Reset!") proc newLPStreamClosedError*(): ref LPStreamClosedError = result = newException(LPStreamClosedError, "Stream Closed!") proc newLPStreamRemoteClosedError*(): ref LPStreamRemoteClosedError = result = newException(LPStreamRemoteClosedError, "Stream Remotely Closed!") proc newLPStreamConnDownError*( parentException: ref Exception = nil): ref LPStreamConnDownError = result = newException( LPStreamConnDownError, "Stream Underlying Connection Closed!", parentException) func shortLog*(s: LPStream): auto = if s.isNil: "LPStream(nil)" else: $s.oid chronicles.formatIt(LPStream): shortLog(it) method initStream*(s: LPStream) {.base.} = if s.objName.len == 0: s.objName = LPStreamTrackerName s.closeEvent = newAsyncEvent() s.oid = genOid() libp2p_open_streams.inc(labelValues = [s.objName, $s.dir]) inc getStreamTracker(s.objName).opened trace "Stream created", s, objName = s.objName, dir = $s.dir proc join*(s: LPStream): Future[void] {.public.} = ## Wait for the stream to be closed s.closeEvent.wait() method closed*(s: LPStream): bool {.base, public.} = s.isClosed method atEof*(s: LPStream): bool {.base, public.} = s.isEof method readOnce*( s: LPStream, pbytes: pointer, nbytes: int): Future[int] {.base, async, public.} = ## Reads whatever is available in the stream, ## up to `nbytes`. Will block if nothing is ## available doAssert(false, "not implemented!") proc readExactly*(s: LPStream, pbytes: pointer, nbytes: int): Future[void] {.async, public.} = ## Waits for `nbytes` to be available, then read ## them and return them if s.atEof: var ch: char discard await s.readOnce(addr ch, 1) raise newLPStreamEOFError() if nbytes == 0: return logScope: s nbytes = nbytes objName = s.objName var pbuffer = cast[ptr UncheckedArray[byte]](pbytes) var read = 0 while read < nbytes and not(s.atEof()): read += await s.readOnce(addr pbuffer[read], nbytes - read) if read == 0: doAssert s.atEof() trace "couldn't read all bytes, stream EOF", s, nbytes, read # Re-readOnce to raise a more specific error than EOF # Raise EOF if it doesn't raise anything(shouldn't happen) discard await s.readOnce(addr pbuffer[read], nbytes - read) warn "Read twice while at EOF" raise newLPStreamEOFError() if read < nbytes: trace "couldn't read all bytes, incomplete data", s, nbytes, read raise newLPStreamIncompleteError() proc readLine*(s: LPStream, limit = 0, sep = "\r\n"): Future[string] {.async, public.} = ## Reads up to `limit` bytes are read, or a `sep` is found # TODO replace with something that exploits buffering better var lim = if limit <= 0: -1 else: limit var state = 0 while true: var ch: char await readExactly(s, addr ch, 1) if sep[state] == ch: inc(state) if state == len(sep): break else: state = 0 if limit > 0: let missing = min(state, lim - len(result) - 1) result.add(sep[0 ..< missing]) else: result.add(sep[0 ..< state]) result.add(ch) if len(result) == lim: break proc readVarint*(conn: LPStream): Future[uint64] {.async, public.} = var buffer: array[10, byte] for i in 0.. maxLen: raise (ref MaxSizeError)(msg: "Message exceeds maximum length") if length == 0: return var res = newSeqUninitialized[byte](length) await s.readExactly(addr res[0], res.len) return res method write*(s: LPStream, msg: seq[byte]): Future[void] {.base, public.} = # Write `msg` to stream, waiting for the write to be finished doAssert(false, "not implemented!") proc writeLp*(s: LPStream, msg: openArray[byte]): Future[void] {.public.} = ## Write `msg` with a varint-encoded length prefix let vbytes = PB.toBytes(msg.len().uint64) var buf = newSeqUninitialized[byte](msg.len() + vbytes.len) buf[0..