mirror of
https://github.com/vacp2p/nim-libp2p-experimental.git
synced 2025-01-12 19:34:23 +00:00
96d4c44fec
This change modifies how the backpressure algorithm in bufferstream works - in particular, instead of working byte-by-byte, it will now work seq-by-seq. When data arrives, it usually does so in packets - in the current bufferstream, the packet is read then split into bytes which are fed one by one to the bufferstream. On the reading side, the bytes are popped of the bufferstream, again byte by byte, to satisfy `readOnce` requests - this introduces a lot of synchronization traffic because the checks for full buffer and for async event handling must be done for every byte. In this PR, a queue of length 1 is used instead - this means there will at most exist one "packet" in `pushTo`, one in the queue and one in the slush buffer that is used to store incomplete reads. * avoid byte-by-byte copy to buffer, with synchronization in-between * reuse AsyncQueue synchronization logic instead of rolling own * avoid writeHandler callback - implement `write` method instead * simplify EOF signalling by only setting EOF flag in queue reader (and reset) * remove BufferStream pipes (unused) * fixes drainBuffer deadlock when drain is called from within read loop and thus blocks draining * fix lpchannel init order
676 lines
21 KiB
Nim
676 lines
21 KiB
Nim
import unittest, strformat, strformat, random, oids
|
|
import chronos, nimcrypto/utils, chronicles, stew/byteutils
|
|
import ../libp2p/[errors,
|
|
stream/connection,
|
|
stream/bufferstream,
|
|
transports/tcptransport,
|
|
transports/transport,
|
|
multiaddress,
|
|
muxers/mplex/mplex,
|
|
muxers/mplex/coder,
|
|
muxers/mplex/types,
|
|
muxers/mplex/lpchannel,
|
|
vbuffer,
|
|
varint]
|
|
|
|
import ./helpers
|
|
|
|
{.used.}
|
|
|
|
suite "Mplex":
|
|
teardown:
|
|
for tracker in testTrackers():
|
|
# echo tracker.dump()
|
|
check tracker.isLeaked() == false
|
|
|
|
test "encode header with channel id 0":
|
|
proc testEncodeHeader() {.async.} =
|
|
proc encHandler(msg: seq[byte]) {.async.} =
|
|
check msg == fromHex("000873747265616d2031")
|
|
|
|
let conn = newBufferStream(encHandler)
|
|
await conn.writeMsg(0, MessageType.New, ("stream 1").toBytes)
|
|
await conn.close()
|
|
|
|
waitFor(testEncodeHeader())
|
|
|
|
test "encode header with channel id other than 0":
|
|
proc testEncodeHeader() {.async.} =
|
|
proc encHandler(msg: seq[byte]) {.async.} =
|
|
check msg == fromHex("88010873747265616d2031")
|
|
|
|
let conn = newBufferStream(encHandler)
|
|
await conn.writeMsg(17, MessageType.New, ("stream 1").toBytes)
|
|
await conn.close()
|
|
|
|
waitFor(testEncodeHeader())
|
|
|
|
test "encode header and body with channel id 0":
|
|
proc testEncodeHeaderBody() {.async.} =
|
|
proc encHandler(msg: seq[byte]) {.async.} =
|
|
check msg == fromHex("020873747265616d2031")
|
|
|
|
let conn = newBufferStream(encHandler)
|
|
await conn.writeMsg(0, MessageType.MsgOut, ("stream 1").toBytes)
|
|
await conn.close()
|
|
|
|
waitFor(testEncodeHeaderBody())
|
|
|
|
test "encode header and body with channel id other than 0":
|
|
proc testEncodeHeaderBody() {.async.} =
|
|
proc encHandler(msg: seq[byte]) {.async.} =
|
|
check msg == fromHex("8a010873747265616d2031")
|
|
|
|
let conn = newBufferStream(encHandler)
|
|
await conn.writeMsg(17, MessageType.MsgOut, ("stream 1").toBytes)
|
|
await conn.close()
|
|
|
|
waitFor(testEncodeHeaderBody())
|
|
|
|
test "decode header with channel id 0":
|
|
proc testDecodeHeader() {.async.} =
|
|
let stream = newBufferStream()
|
|
let conn = stream
|
|
await stream.pushTo(fromHex("000873747265616d2031"))
|
|
let msg = await conn.readMsg()
|
|
|
|
check msg.id == 0
|
|
check msg.msgType == MessageType.New
|
|
await conn.close()
|
|
|
|
waitFor(testDecodeHeader())
|
|
|
|
test "decode header and body with channel id 0":
|
|
proc testDecodeHeader() {.async.} =
|
|
let stream = newBufferStream()
|
|
let conn = stream
|
|
await stream.pushTo(fromHex("021668656C6C6F2066726F6D206368616E6E656C20302121"))
|
|
let msg = await conn.readMsg()
|
|
|
|
check msg.id == 0
|
|
check msg.msgType == MessageType.MsgOut
|
|
check string.fromBytes(msg.data) == "hello from channel 0!!"
|
|
await conn.close()
|
|
|
|
waitFor(testDecodeHeader())
|
|
|
|
test "decode header and body with channel id other than 0":
|
|
proc testDecodeHeader() {.async.} =
|
|
let stream = newBufferStream()
|
|
let conn = stream
|
|
await stream.pushTo(fromHex("8a011668656C6C6F2066726F6D206368616E6E656C20302121"))
|
|
let msg = await conn.readMsg()
|
|
|
|
check msg.id == 17
|
|
check msg.msgType == MessageType.MsgOut
|
|
check string.fromBytes(msg.data) == "hello from channel 0!!"
|
|
await conn.close()
|
|
|
|
waitFor(testDecodeHeader())
|
|
|
|
test "half closed (local close) - should close for write":
|
|
proc testClosedForWrite(): Future[bool] {.async.} =
|
|
proc writeHandler(data: seq[byte]) {.async, gcsafe.} = discard
|
|
let
|
|
conn = newBufferStream(writeHandler)
|
|
chann = LPChannel.init(1, conn, true)
|
|
await chann.close()
|
|
try:
|
|
await chann.write("Hello")
|
|
except LPStreamClosedError:
|
|
result = true
|
|
finally:
|
|
await chann.reset()
|
|
await conn.close()
|
|
|
|
check:
|
|
waitFor(testClosedForWrite()) == true
|
|
|
|
test "half closed (local close) - should allow reads until remote closes":
|
|
proc testOpenForRead(): Future[bool] {.async.} =
|
|
let
|
|
conn = newBufferStream(
|
|
proc (data: seq[byte]) {.gcsafe, async.} =
|
|
discard,
|
|
)
|
|
chann = LPChannel.init(1, conn, true)
|
|
|
|
await chann.pushTo(("Hello!").toBytes)
|
|
|
|
var data = newSeq[byte](6)
|
|
await chann.close() # closing channel
|
|
# should be able to read on local clsoe
|
|
await chann.readExactly(addr data[0], 3)
|
|
# closing remote end
|
|
let closeFut = chann.closeRemote()
|
|
# should still allow reading until buffer EOF
|
|
await chann.readExactly(addr data[3], 3)
|
|
try:
|
|
# this should fail now
|
|
await chann.readExactly(addr data[0], 3)
|
|
except LPStreamEOFError:
|
|
result = true
|
|
finally:
|
|
await chann.close()
|
|
await conn.close()
|
|
await closeFut
|
|
|
|
check:
|
|
waitFor(testOpenForRead()) == true
|
|
|
|
test "half closed (remote close) - channel should close for reading by remote":
|
|
proc testClosedForRead(): Future[bool] {.async.} =
|
|
let
|
|
conn = newBufferStream(
|
|
proc (data: seq[byte]) {.gcsafe, async.} =
|
|
discard,
|
|
)
|
|
chann = LPChannel.init(1, conn, true)
|
|
|
|
await chann.pushTo(("Hello!").toBytes)
|
|
|
|
var data = newSeq[byte](6)
|
|
await chann.readExactly(addr data[0], 3)
|
|
let closeFut = chann.closeRemote() # closing channel
|
|
let readFut = chann.readExactly(addr data[3], 3)
|
|
await all(closeFut, readFut)
|
|
try:
|
|
await chann.readExactly(addr data[0], 6) # this should fail now
|
|
except LPStreamEOFError:
|
|
result = true
|
|
finally:
|
|
await chann.close()
|
|
await conn.close()
|
|
|
|
check:
|
|
waitFor(testClosedForRead()) == true
|
|
|
|
test "half closed (remote close) - channel should allow writting on remote close":
|
|
proc testClosedForRead(): Future[bool] {.async.} =
|
|
let
|
|
testData = "Hello!".toBytes
|
|
conn = newBufferStream(
|
|
proc (data: seq[byte]) {.gcsafe, async.} =
|
|
discard
|
|
)
|
|
chann = LPChannel.init(1, conn, true)
|
|
|
|
await chann.closeRemote() # closing channel
|
|
try:
|
|
await chann.writeLp(testData)
|
|
return true
|
|
finally:
|
|
await chann.close()
|
|
await conn.close()
|
|
|
|
check:
|
|
waitFor(testClosedForRead()) == true
|
|
|
|
test "should not allow pushing data to channel when remote end closed":
|
|
proc testResetWrite(): Future[bool] {.async.} =
|
|
proc writeHandler(data: seq[byte]) {.async, gcsafe.} = discard
|
|
let
|
|
conn = newBufferStream(writeHandler)
|
|
chann = LPChannel.init(1, conn, true)
|
|
await chann.closeRemote()
|
|
try:
|
|
await chann.pushTo(@[byte(1)])
|
|
except LPStreamEOFError:
|
|
result = true
|
|
finally:
|
|
await chann.close()
|
|
await conn.close()
|
|
|
|
check:
|
|
waitFor(testResetWrite()) == true
|
|
|
|
test "reset - channel should fail reading":
|
|
proc testResetRead(): Future[bool] {.async.} =
|
|
proc writeHandler(data: seq[byte]) {.async, gcsafe.} = discard
|
|
let
|
|
conn = newBufferStream(writeHandler)
|
|
chann = LPChannel.init(1, conn, true)
|
|
|
|
await chann.reset()
|
|
var data = newSeq[byte](1)
|
|
try:
|
|
await chann.readExactly(addr data[0], 1)
|
|
check data.len == 1
|
|
except LPStreamEOFError:
|
|
result = true
|
|
finally:
|
|
await conn.close()
|
|
|
|
check:
|
|
waitFor(testResetRead()) == true
|
|
|
|
test "reset - channel should fail writing":
|
|
proc testResetWrite(): Future[bool] {.async.} =
|
|
proc writeHandler(data: seq[byte]) {.async, gcsafe.} = discard
|
|
let
|
|
conn = newBufferStream(writeHandler)
|
|
chann = LPChannel.init(1, conn, true)
|
|
await chann.reset()
|
|
try:
|
|
await chann.write(("Hello!").toBytes)
|
|
except LPStreamClosedError:
|
|
result = true
|
|
finally:
|
|
await conn.close()
|
|
|
|
check:
|
|
waitFor(testResetWrite()) == true
|
|
|
|
test "reset - channel should reset on timeout":
|
|
proc testResetWrite(): Future[bool] {.async.} =
|
|
proc writeHandler(data: seq[byte]) {.async, gcsafe.} = discard
|
|
let
|
|
conn = newBufferStream(writeHandler)
|
|
chann = LPChannel.init(
|
|
1, conn, true, timeout = 100.millis)
|
|
|
|
check await chann.closeEvent.wait().withTimeout(1.minutes)
|
|
await conn.close()
|
|
result = true
|
|
|
|
check:
|
|
waitFor(testResetWrite())
|
|
|
|
test "e2e - read/write receiver":
|
|
proc testNewStream() {.async.} =
|
|
let ma: MultiAddress = Multiaddress.init("/ip4/0.0.0.0/tcp/0").tryGet()
|
|
|
|
var done = newFuture[void]()
|
|
proc connHandler(conn: Connection) {.async, gcsafe.} =
|
|
let mplexListen = Mplex.init(conn)
|
|
mplexListen.streamHandler = proc(stream: Connection)
|
|
{.async, gcsafe.} =
|
|
let msg = await stream.readLp(1024)
|
|
check string.fromBytes(msg) == "HELLO"
|
|
await stream.close()
|
|
done.complete()
|
|
|
|
await mplexListen.handle()
|
|
await mplexListen.close()
|
|
|
|
let transport1: TcpTransport = TcpTransport.init()
|
|
let listenFut = await transport1.listen(ma, connHandler)
|
|
|
|
let transport2: TcpTransport = TcpTransport.init()
|
|
let conn = await transport2.dial(transport1.ma)
|
|
|
|
let mplexDial = Mplex.init(conn)
|
|
let mplexDialFut = mplexDial.handle()
|
|
let stream = await mplexDial.newStream()
|
|
await stream.writeLp("HELLO")
|
|
check LPChannel(stream).isOpen # not lazy
|
|
await stream.close()
|
|
|
|
await done.wait(1.seconds)
|
|
await conn.close()
|
|
await mplexDialFut.wait(1.seconds)
|
|
await allFuturesThrowing(
|
|
transport1.close(),
|
|
transport2.close())
|
|
await listenFut
|
|
|
|
waitFor(testNewStream())
|
|
|
|
test "e2e - read/write receiver lazy":
|
|
proc testNewStream() {.async.} =
|
|
let ma: MultiAddress = Multiaddress.init("/ip4/0.0.0.0/tcp/0").tryGet()
|
|
|
|
var done = newFuture[void]()
|
|
proc connHandler(conn: Connection) {.async, gcsafe.} =
|
|
let mplexListen = Mplex.init(conn)
|
|
mplexListen.streamHandler = proc(stream: Connection)
|
|
{.async, gcsafe.} =
|
|
let msg = await stream.readLp(1024)
|
|
check string.fromBytes(msg) == "HELLO"
|
|
await stream.close()
|
|
done.complete()
|
|
|
|
await mplexListen.handle()
|
|
await mplexListen.close()
|
|
|
|
let transport1: TcpTransport = TcpTransport.init()
|
|
let listenFut = await transport1.listen(ma, connHandler)
|
|
|
|
let transport2: TcpTransport = TcpTransport.init()
|
|
let conn = await transport2.dial(transport1.ma)
|
|
|
|
let mplexDial = Mplex.init(conn)
|
|
let stream = await mplexDial.newStream(lazy = true)
|
|
let mplexDialFut = mplexDial.handle()
|
|
check not LPChannel(stream).isOpen # assert lazy
|
|
await stream.writeLp("HELLO")
|
|
check LPChannel(stream).isOpen # assert lazy
|
|
await stream.close()
|
|
|
|
await done.wait(1.seconds)
|
|
await conn.close()
|
|
await mplexDialFut
|
|
await allFuturesThrowing(
|
|
transport1.close(),
|
|
transport2.close())
|
|
await listenFut
|
|
|
|
waitFor(testNewStream())
|
|
|
|
test "e2e - write fragmented":
|
|
proc testNewStream() {.async.} =
|
|
let
|
|
ma: MultiAddress = Multiaddress.init("/ip4/0.0.0.0/tcp/0").tryGet()
|
|
listenJob = newFuture[void]()
|
|
|
|
var bigseq = newSeqOfCap[uint8](MaxMsgSize * 2)
|
|
for _ in 0..<MaxMsgSize:
|
|
bigseq.add(uint8(rand(uint('A')..uint('z'))))
|
|
|
|
proc connHandler(conn: Connection) {.async, gcsafe.} =
|
|
try:
|
|
let mplexListen = Mplex.init(conn)
|
|
mplexListen.streamHandler = proc(stream: Connection)
|
|
{.async, gcsafe.} =
|
|
let msg = await stream.readLp(MaxMsgSize)
|
|
check msg == bigseq
|
|
trace "Bigseq check passed!"
|
|
await stream.close()
|
|
listenJob.complete()
|
|
|
|
await mplexListen.handle()
|
|
await sleepAsync(1.seconds) # give chronos some slack to process things
|
|
await mplexListen.close()
|
|
except CancelledError as exc:
|
|
raise exc
|
|
except CatchableError as exc:
|
|
check false
|
|
|
|
let transport1: TcpTransport = TcpTransport.init()
|
|
let listenFut = await transport1.listen(ma, connHandler)
|
|
|
|
let transport2: TcpTransport = TcpTransport.init()
|
|
let conn = await transport2.dial(transport1.ma)
|
|
|
|
let mplexDial = Mplex.init(conn)
|
|
let mplexDialFut = mplexDial.handle()
|
|
let stream = await mplexDial.newStream()
|
|
|
|
await stream.writeLp(bigseq)
|
|
try:
|
|
await listenJob.wait(10.seconds)
|
|
except AsyncTimeoutError:
|
|
check false
|
|
|
|
await stream.close()
|
|
await conn.close()
|
|
await mplexDialFut
|
|
await allFuturesThrowing(
|
|
transport1.close(),
|
|
transport2.close())
|
|
await listenFut
|
|
|
|
waitFor(testNewStream())
|
|
|
|
test "e2e - read/write initiator":
|
|
proc testNewStream() {.async.} =
|
|
let ma: MultiAddress = Multiaddress.init("/ip4/0.0.0.0/tcp/0").tryGet()
|
|
|
|
let done = newFuture[void]()
|
|
proc connHandler(conn: Connection) {.async, gcsafe.} =
|
|
let mplexListen = Mplex.init(conn)
|
|
mplexListen.streamHandler = proc(stream: Connection)
|
|
{.async, gcsafe.} =
|
|
await stream.writeLp("Hello from stream!")
|
|
await stream.close()
|
|
done.complete()
|
|
|
|
await mplexListen.handle()
|
|
await mplexListen.close()
|
|
|
|
let transport1: TcpTransport = TcpTransport.init()
|
|
let listenFut = await transport1.listen(ma, connHandler)
|
|
|
|
let transport2: TcpTransport = TcpTransport.init()
|
|
let conn = await transport2.dial(transport1.ma)
|
|
|
|
let mplexDial = Mplex.init(conn)
|
|
let mplexDialFut = mplexDial.handle()
|
|
let stream = await mplexDial.newStream("DIALER")
|
|
let msg = string.fromBytes(await stream.readLp(1024))
|
|
await stream.close()
|
|
check msg == "Hello from stream!"
|
|
|
|
await done.wait(1.seconds)
|
|
await conn.close()
|
|
await mplexDialFut
|
|
await allFuturesThrowing(
|
|
transport1.close(),
|
|
transport2.close())
|
|
await listenFut
|
|
|
|
waitFor(testNewStream())
|
|
|
|
test "e2e - multiple streams":
|
|
proc testNewStream() {.async.} =
|
|
let ma: MultiAddress = Multiaddress.init("/ip4/0.0.0.0/tcp/0").tryGet()
|
|
|
|
let done = newFuture[void]()
|
|
proc connHandler(conn: Connection) {.async, gcsafe.} =
|
|
var count = 1
|
|
let mplexListen = Mplex.init(conn)
|
|
mplexListen.streamHandler = proc(stream: Connection)
|
|
{.async, gcsafe.} =
|
|
let msg = await stream.readLp(1024)
|
|
check string.fromBytes(msg) == &"stream {count}!"
|
|
count.inc
|
|
await stream.close()
|
|
if count == 10:
|
|
done.complete()
|
|
|
|
await mplexListen.handle()
|
|
await mplexListen.close()
|
|
|
|
let transport1 = TcpTransport.init()
|
|
let listenFut = await transport1.listen(ma, connHandler)
|
|
|
|
let transport2: TcpTransport = TcpTransport.init()
|
|
let conn = await transport2.dial(transport1.ma)
|
|
|
|
let mplexDial = Mplex.init(conn)
|
|
# TODO: Reenable once half-closed is working properly
|
|
let mplexDialFut = mplexDial.handle()
|
|
for i in 1..10:
|
|
let stream = await mplexDial.newStream()
|
|
await stream.writeLp(&"stream {i}!")
|
|
await stream.close()
|
|
|
|
await done.wait(10.seconds)
|
|
await conn.close()
|
|
await mplexDialFut
|
|
await allFuturesThrowing(
|
|
transport1.close(),
|
|
transport2.close())
|
|
await listenFut
|
|
|
|
waitFor(testNewStream())
|
|
|
|
test "e2e - multiple read/write streams":
|
|
proc testNewStream() {.async.} =
|
|
let ma: MultiAddress = Multiaddress.init("/ip4/0.0.0.0/tcp/0").tryGet()
|
|
|
|
let done = newFuture[void]()
|
|
proc connHandler(conn: Connection) {.async, gcsafe.} =
|
|
var count = 1
|
|
let mplexListen = Mplex.init(conn)
|
|
mplexListen.streamHandler = proc(stream: Connection)
|
|
{.async, gcsafe.} =
|
|
let msg = await stream.readLp(1024)
|
|
check string.fromBytes(msg) == &"stream {count} from dialer!"
|
|
await stream.writeLp(&"stream {count} from listener!")
|
|
count.inc
|
|
await stream.close()
|
|
if count == 10:
|
|
done.complete()
|
|
|
|
await mplexListen.handle()
|
|
await mplexListen.close()
|
|
|
|
let transport1: TcpTransport = TcpTransport.init()
|
|
let listenFut = await transport1.listen(ma, connHandler)
|
|
|
|
let transport2: TcpTransport = TcpTransport.init()
|
|
let conn = await transport2.dial(transport1.ma)
|
|
|
|
let mplexDial = Mplex.init(conn)
|
|
let mplexDialFut = mplexDial.handle()
|
|
for i in 1..10:
|
|
let stream = await mplexDial.newStream("dialer stream")
|
|
await stream.writeLp(&"stream {i} from dialer!")
|
|
let msg = await stream.readLp(1024)
|
|
check string.fromBytes(msg) == &"stream {i} from listener!"
|
|
await stream.close()
|
|
|
|
await done.wait(5.seconds)
|
|
await conn.close()
|
|
await mplexDialFut
|
|
await allFuturesThrowing(
|
|
transport1.close(),
|
|
transport2.close())
|
|
await listenFut
|
|
|
|
waitFor(testNewStream())
|
|
|
|
test "jitter - channel should be able to handle erratic read/writes":
|
|
proc test() {.async.} =
|
|
let ma: MultiAddress = Multiaddress.init("/ip4/0.0.0.0/tcp/0").tryGet()
|
|
|
|
var complete = newFuture[void]()
|
|
const MsgSize = 1024
|
|
proc connHandler(conn: Connection) {.async, gcsafe.} =
|
|
let mplexListen = Mplex.init(conn)
|
|
mplexListen.streamHandler = proc(stream: Connection)
|
|
{.async, gcsafe.} =
|
|
try:
|
|
let msg = await stream.readLp(MsgSize)
|
|
check msg.len == MsgSize
|
|
except CatchableError as e:
|
|
echo e.msg
|
|
await stream.close()
|
|
complete.complete()
|
|
|
|
await mplexListen.handle()
|
|
await mplexListen.close()
|
|
|
|
let transport1: TcpTransport = TcpTransport.init()
|
|
let listenFut = await transport1.listen(ma, connHandler)
|
|
|
|
let transport2: TcpTransport = TcpTransport.init()
|
|
let conn = await transport2.dial(transport1.ma)
|
|
|
|
let mplexDial = Mplex.init(conn)
|
|
let mplexDialFut = mplexDial.handle()
|
|
let stream = await mplexDial.newStream()
|
|
var bigseq = newSeqOfCap[uint8](MaxMsgSize + 1)
|
|
for _ in 0..<MsgSize: # write one less than max size
|
|
bigseq.add(uint8(rand(uint('A')..uint('z'))))
|
|
|
|
## create lenght prefixed libp2p frame
|
|
var buf = initVBuffer()
|
|
buf.writeSeq(bigseq)
|
|
buf.finish()
|
|
|
|
## create mplex header
|
|
var mplexBuf = initVBuffer()
|
|
mplexBuf.writePBVarint((1.uint shl 3) or ord(MessageType.MsgOut).uint)
|
|
mplexBuf.writePBVarint(buf.buffer.len.uint) # size should be always sent
|
|
|
|
await conn.write(mplexBuf.buffer)
|
|
proc writer() {.async.} =
|
|
var sent = 0
|
|
randomize()
|
|
let total = buf.buffer.len
|
|
const min = 20
|
|
const max = 50
|
|
while sent < total:
|
|
var size = rand(min..max)
|
|
size = if size > buf.buffer.len: buf.buffer.len else: size
|
|
var send = buf.buffer[0..<size]
|
|
await conn.write(send)
|
|
sent += size
|
|
buf.buffer = buf.buffer[size..^1]
|
|
|
|
await writer()
|
|
await complete.wait(1.seconds)
|
|
|
|
await stream.close()
|
|
await conn.close()
|
|
|
|
await mplexDialFut
|
|
|
|
await allFuturesThrowing(
|
|
transport1.close(),
|
|
transport2.close())
|
|
await listenFut
|
|
|
|
waitFor(test())
|
|
|
|
test "jitter - channel should handle 1 byte read/write":
|
|
proc test() {.async.} =
|
|
let ma: MultiAddress = Multiaddress.init("/ip4/0.0.0.0/tcp/0").tryGet()
|
|
|
|
var complete = newFuture[void]()
|
|
const MsgSize = 512
|
|
proc connHandler(conn: Connection) {.async, gcsafe.} =
|
|
let mplexListen = Mplex.init(conn)
|
|
mplexListen.streamHandler = proc(stream: Connection)
|
|
{.async, gcsafe.} =
|
|
let msg = await stream.readLp(MsgSize)
|
|
check msg.len == MsgSize
|
|
await stream.close()
|
|
complete.complete()
|
|
|
|
await mplexListen.handle()
|
|
await mplexListen.close()
|
|
|
|
let transport1: TcpTransport = TcpTransport.init()
|
|
let listenFut = await transport1.listen(ma, connHandler)
|
|
|
|
let transport2: TcpTransport = TcpTransport.init()
|
|
let conn = await transport2.dial(transport1.ma)
|
|
|
|
let mplexDial = Mplex.init(conn)
|
|
let stream = await mplexDial.newStream()
|
|
let mplexDialFut = mplexDial.handle()
|
|
var bigseq = newSeqOfCap[uint8](MsgSize + 1)
|
|
for _ in 0..<MsgSize: # write one less than max size
|
|
bigseq.add(uint8(rand(uint('A')..uint('z'))))
|
|
|
|
## create lenght prefixed libp2p frame
|
|
var buf = initVBuffer()
|
|
buf.writeSeq(bigseq)
|
|
buf.finish()
|
|
|
|
## create mplex header
|
|
var mplexBuf = initVBuffer()
|
|
mplexBuf.writePBVarint((1.uint shl 3) or ord(MessageType.MsgOut).uint)
|
|
mplexBuf.writePBVarint(buf.buffer.len.uint) # size should be always sent
|
|
|
|
await conn.write(mplexBuf.buffer)
|
|
proc writer() {.async.} =
|
|
for i in buf.buffer:
|
|
await conn.write(@[i])
|
|
|
|
await writer()
|
|
|
|
await complete.wait(5.seconds)
|
|
await stream.close()
|
|
await conn.close()
|
|
await mplexDialFut
|
|
await allFuturesThrowing(
|
|
transport1.close(),
|
|
transport2.close())
|
|
await listenFut
|
|
|
|
waitFor(test())
|