# Chronos Test Suite # (c) Copyright 2018-Present # Status Research & Development GmbH # # Licensed under either of # Apache License, version 2.0, (LICENSE-APACHEv2) # MIT license (LICENSE-MIT) import std/[strutils, os] import unittest2 import ".."/chronos, ".."/chronos/[osdefs, oserrno] {.used.} when defined(windows): proc get_osfhandle*(fd: FileHandle): HANDLE {. importc: "_get_osfhandle", header:"".} suite "Stream Transport test suite": const ConstantMessage = "SOMEDATA" BigMessagePattern = "ABCDEFGHIJKLMNOPQRSTUVWXYZ" FilesTestName = "tests/teststream.nim" BigMessageCount = 100 ClientsCount = 5 MessagesCount = 10 MessageSize = 20 FilesCount = 10 TestsCount = 100 when defined(windows): let addresses = [ initTAddress("127.0.0.1:33335"), initTAddress(r"/LOCAL\testpipe") ] else: let addresses = [ initTAddress("127.0.0.1:33335"), initTAddress(r"/tmp/testpipe") ] let prefixes = ["[IP] ", "[UNIX] "] var markFD: int proc getCurrentFD(): int = let local = initTAddress("127.0.0.1:33334") let sock = createAsyncSocket(local.getDomain(), SockType.SOCK_DGRAM, Protocol.IPPROTO_UDP) closeSocket(sock) return int(sock) proc createBigMessage(size: int): seq[byte] = var message = "MESSAGE" result = newSeq[byte](size) for i in 0 ..< len(result): result[i] = byte(message[i mod len(message)]) proc serveClient1(server: StreamServer, transp: StreamTransport) {.async.} = while not transp.atEof(): var data = await transp.readLine() if len(data) == 0: doAssert(transp.atEof()) break doAssert(data.startsWith("REQUEST")) var numstr = data[7..^1] var num = parseInt(numstr) var ans = "ANSWER" & $num & "\r\n" var res = await transp.write(cast[pointer](addr ans[0]), len(ans)) doAssert(res == len(ans)) transp.close() await transp.join() proc serveClient2(server: StreamServer, transp: StreamTransport) {.async.} = var buffer: array[20, char] var check = "REQUEST" while not transp.atEof(): zeroMem(addr buffer[0], MessageSize) try: await transp.readExactly(addr buffer[0], MessageSize) except TransportIncompleteError: break doAssert(equalMem(addr buffer[0], addr check[0], len(check))) var numstr = "" var i = 7 while i < MessageSize and (buffer[i] in {'0'..'9'}): numstr.add(buffer[i]) inc(i) var num = parseInt(numstr) var ans = "ANSWER" & $num zeroMem(addr buffer[0], MessageSize) copyMem(addr buffer[0], addr ans[0], len(ans)) var res = await transp.write(cast[pointer](addr buffer[0]), MessageSize) doAssert(res == MessageSize) transp.close() await transp.join() proc serveClient3(server: StreamServer, transp: StreamTransport) {.async.} = var buffer: array[20, char] var check = "REQUEST" var suffixStr = "SUFFIX" var suffix = newSeq[byte](6) copyMem(addr suffix[0], addr suffixStr[0], len(suffixStr)) var counter = MessagesCount while counter > 0: zeroMem(addr buffer[0], MessageSize) var res = await transp.readUntil(addr buffer[0], MessageSize, suffix) doAssert(equalMem(addr buffer[0], addr check[0], len(check))) var numstr = "" var i = 7 while i < MessageSize and (buffer[i] in {'0'..'9'}): numstr.add(buffer[i]) inc(i) var num = parseInt(numstr) doAssert(len(numstr) < 8) var ans = "ANSWER" & $num & "SUFFIX" zeroMem(addr buffer[0], MessageSize) copyMem(addr buffer[0], addr ans[0], len(ans)) res = await transp.write(cast[pointer](addr buffer[0]), len(ans)) doAssert(res == len(ans)) dec(counter) transp.close() await transp.join() proc serveClient4(server: StreamServer, transp: StreamTransport) {.async.} = var pathname = await transp.readLine() var size = await transp.readLine() var sizeNum = parseInt(size) doAssert(sizeNum >= 0) var rbuffer = newSeq[byte](sizeNum) await transp.readExactly(addr rbuffer[0], sizeNum) var lbuffer = readFile(pathname) doAssert(len(lbuffer) == sizeNum) doAssert(equalMem(addr rbuffer[0], addr lbuffer[0], sizeNum)) var answer = "OK\r\n" var res = await transp.write(cast[pointer](addr answer[0]), len(answer)) doAssert(res == len(answer)) transp.close() await transp.join() proc serveClient7(server: StreamServer, transp: StreamTransport) {.async.} = var answer = "DONE\r\n" var expect = "" var line = await transp.readLine() doAssert(len(line) == BigMessageCount * len(BigMessagePattern)) for i in 0.. 0) name = name & "\r\n" var res = await transp.write(cast[pointer](addr name[0]), len(name)) doAssert(res == len(name)) ssize = $size & "\r\n" res = await transp.write(cast[pointer](addr ssize[0]), len(ssize)) doAssert(res == len(ssize)) var checksize = await transp.writeFile(handle, 0'u, size) doAssert(checksize == size) close(fhandle) var ans = await transp.readLine() doAssert(ans == "OK") result = 1 transp.close() await transp.join() proc swarmWorker7(address: TransportAddress): Future[int] {.async.} = var transp = await connect(address) var data = BigMessagePattern var crlf = "\r\n" for i in 0.. buffer.len: buffer.setLen(prevLen + readLength) let bytesRead = await transp.readOnce(addr buffer[prevLen], readLength) inc(prevLen, bytesRead) buffer.setLen(prevLen) doAssert(buffer == BigMessagePattern) result = 1 transp.close() await transp.join() proc test16(address: TransportAddress): Future[int] {.async.} = var server = createStreamServer(address, serveClient16, {ReuseAddr}) server.start() result = await swarmWorker16(address) server.stop() server.close() await server.join() proc testCloseTransport(address: TransportAddress): Future[int] {.async.} = proc client(server: StreamServer, transp: StreamTransport) {.async.} = discard var server = createStreamServer(address, client, {ReuseAddr}) server.start() server.stop server.close() try: await wait(server.join(), 10.seconds) result = 1 except: discard proc testWriteConnReset(address: TransportAddress): Future[int] {.async.} = var syncFut = newFuture[void]() proc client(server: StreamServer, transp: StreamTransport) {.async.} = await transp.closeWait() syncFut.complete() var n = 10 var server = createStreamServer(address, client, {ReuseAddr}) server.start() var msg = "HELLO" var ntransp = await connect(address) await syncFut while true: var res = await ntransp.write(msg) if res == 0: result = 1 break else: dec(n) if n == 0: break server.stop() await ntransp.closeWait() await server.closeWait() proc testAnyAddress(): Future[bool] {.async.} = var serverRemote, serverLocal: TransportAddress var connRemote, connLocal: TransportAddress proc serveClient(server: StreamServer, transp: StreamTransport) {.async.} = serverRemote = transp.remoteAddress() serverLocal = transp.localAddress() await transp.closeWait() server.stop() server.close() var ta = initTAddress("0.0.0.0:0") var server = createStreamServer(ta, serveClient, {ReuseAddr}) var la = server.localAddress() server.start() var connFut = connect(la) if await withTimeout(connFut, 5.seconds): var conn = connFut.read() connRemote = conn.remoteAddress() connLocal = conn.localAddress() await server.join() await conn.closeWait() result = (connRemote == serverLocal) and (connLocal == serverRemote) else: server.stop() server.close() proc testWriteReturn(address: TransportAddress): Future[bool] {.async.} = var bigMessageSize = 10 * 1024 * 1024 - 1 var finishMessage = "DONE" var cdata = newSeqOfCap[byte](bigMessageSize) proc serveClient(server: StreamServer, transp: StreamTransport) {.async.} = cdata = await transp.read(bigMessageSize) var size = await transp.write(finishMessage) doAssert(size == len(finishMessage)) await transp.closeWait() server.stop() server.close() var flag = false var server = createStreamServer(address, serveClient, {ReuseAddr}) server.start() var transp: StreamTransport try: transp = await connect(address) flag = true except: server.stop() server.close() await server.join() if flag: flag = false try: var msg = createBigMessage(bigMessageSize) var size = await transp.write(msg) var data = await transp.read() doAssert(cdata == msg) doAssert(len(data) == len(finishMessage)) doAssert(equalMem(addr data[0], addr finishMessage[0], len(data))) flag = (size == bigMessageSize) finally: await transp.closeWait() await server.join() result = flag proc testReadLine(address: TransportAddress): Future[bool] {.async.} = proc serveClient(server: StreamServer, transp: StreamTransport) {.async.} = discard await transp.write("DATA\r\r\r\r\r\n") transp.close() await transp.join() var server = createStreamServer(address, serveClient, {ReuseAddr}) server.start() try: var r1, r2, r3, r4, r5: string var t1 = await connect(address) try: r1 = await t1.readLine(4) finally: await t1.closeWait() var t2 = await connect(address) try: r2 = await t2.readLine(6) finally: await t2.closeWait() var t3 = await connect(address) try: r3 = await t3.readLine(8) finally: await t3.closeWait() var t4 = await connect(address) try: r4 = await t4.readLine(8) finally: await t4.closeWait() var t5 = await connect(address) try: r5 = await t5.readLine() finally: await t5.closeWait() doAssert(r1 == "DATA") doAssert(r2 == "DATA\r\r") doAssert(r3 == "DATA\r\r\r\r") doAssert(r4 == "DATA\r\r\r\r") doAssert(r5 == "DATA\r\r\r\r") result = true finally: server.stop() server.close() await server.join() proc readLV(transp: StreamTransport, maxLen: int): Future[seq[byte]] {.async.} = # Read length-prefixed value where length is a 32-bit integer in native # endian (don't do this at home) var valueLen = 0'u32 res: seq[byte] error: ref CatchableError proc predicate(data: openArray[byte]): tuple[consumed: int, done: bool] = if len(data) == 0: # There will be no more data, length-value incomplete error = newException(TransportIncompleteError, "LV incomplete") return (0, true) var consumed = 0 if valueLen == 0: if len(data) < 4: return (0, false) copyMem(addr valueLen, unsafeAddr data[0], sizeof(valueLen)) if valueLen == 0: return (sizeof(valueLen), true) if int(valueLen) > maxLen: error = newException(ValueError, "Size is too big") return (sizeof(valueLen), true) consumed += sizeof(valueLen) let dataLeft = len(data) - consumed count = min(dataLeft, int(valueLen) - len(res)) res.add(data.toOpenArray(consumed, count + consumed - 1)) return (consumed + count, len(res) == int(valueLen)) await transp.readMessage(predicate) if not isNil(error): raise error else: return res proc createMessage(size: uint32): seq[byte] = var message = "MESSAGE" result = newSeq[byte](int(size)) for i in 0 ..< size: result[int(i)] = byte(message[int(i) mod len(message)]) proc createLVMessage(size: uint32): seq[byte] = var message = "MESSAGE" result = newSeq[byte](sizeof(size) + int(size)) copyMem(addr result[0], unsafeAddr size, sizeof(size)) for i in 0 ..< size: result[int(i) + sizeof(size)] = byte(message[int(i) mod len(message)]) proc testReadMessage(address: TransportAddress): Future[bool] {.async.} = var state = 0 var c1, c2, c3, c4, c5, c6, c7: bool proc serveClient(server: StreamServer, transp: StreamTransport) {.async.} = if state == 0: # EOF from the beginning. state = 1 await transp.closeWait() elif state == 1: # Message has only zero-size header. var message = createLVMessage(0'u32) discard await transp.write(message) state = 2 await transp.closeWait() elif state == 2: # Message has header, but do not have any data at all. var message = createLVMessage(4'u32) message.setLen(4) discard await transp.write(message) state = 3 await transp.closeWait() elif state == 3: # Message do not have enough data for specified size in header. var message = createLVMessage(1024'u32) message.setLen(1024) discard await transp.write(message) state = 4 await transp.closeWait() elif state == 4: # Good encoded message with oversize. var message = createLVMessage(1024'u32) discard await transp.write(message) state = 5 await transp.closeWait() elif state == 5: # Good encoded message. var message = createLVMessage(1024'u32) discard await transp.write(message) state = 6 await transp.closeWait() elif state == 6: # Good encoded message with additional data. var message = createLVMessage(1024'u32) discard await transp.write(message) discard await transp.write("DONE") state = 7 await transp.closeWait() else: doAssert(false) var server = createStreamServer(address, serveClient, {ReuseAddr}) server.start() var t1 = await connect(address) try: discard await t1.readLV(2000) except TransportIncompleteError: c1 = true finally: await t1.closeWait() if not c1: server.stop() server.close() await server.join() return false var t2 = await connect(address) try: var r2 = await t2.readLV(2000) c2 = (r2 == @[]) finally: await t2.closeWait() if not c2: server.stop() server.close() await server.join() return false var t3 = await connect(address) try: discard await t3.readLV(2000) except TransportIncompleteError: c3 = true finally: await t3.closeWait() if not c3: server.stop() server.close() await server.join() return false var t4 = await connect(address) try: discard await t4.readLV(2000) except TransportIncompleteError: c4 = true finally: await t4.closeWait() if not c4: server.stop() server.close() await server.join() return false var t5 = await connect(address) try: discard await t5.readLV(1000) except ValueError: c5 = true finally: await t5.closeWait() if not c5: server.stop() server.close() await server.join() return false var t6 = await connect(address) try: var expectMsg = createMessage(1024) var r6 = await t6.readLV(2000) if len(r6) == 1024 and r6 == expectMsg: c6 = true finally: await t6.closeWait() if not c6: server.stop() server.close() await server.join() return false var t7 = await connect(address) try: var expectMsg = createMessage(1024) var expectDone = "DONE" var r7 = await t7.readLV(2000) if len(r7) == 1024 and r7 == expectMsg: var m = await t7.read(4) if len(m) == 4 and equalMem(addr m[0], addr expectDone[0], 4): c7 = true finally: await t7.closeWait() server.stop() server.close() await server.join() result = c7 proc testAccept(address: TransportAddress): Future[bool] {.async.} = var server = createStreamServer(address, flags = {ReuseAddr}) var connected = 0 var accepted = 0 proc acceptTask(server: StreamServer) {.async.} = for i in 0 ..< TestsCount: let transp = await server.accept() await transp.closeWait() inc(accepted) var acceptFut = acceptTask(server) var transp: StreamTransport try: for i in 0 ..< TestsCount: transp = await connect(address) await sleepAsync(10.milliseconds) await transp.closeWait() inc(connected) if await withTimeout(acceptFut, 5.seconds): if acceptFut.finished() and not(acceptFut.failed()): result = (connected == TestsCount) and (connected == accepted) finally: await server.closeWait() if not(isNil(transp)): await transp.closeWait() proc testAcceptClose(address: TransportAddress): Future[bool] {.async.} = var server = createStreamServer(address, flags = {ReuseAddr}) proc acceptTask(server: StreamServer) {.async.} = let transp = await server.accept() await transp.closeWait() var acceptFut = acceptTask(server) await server.closeWait() if await withTimeout(acceptFut, 5.seconds): if acceptFut.finished() and acceptFut.failed(): if acceptFut.readError() of TransportUseClosedError: result = true else: result = false when not(defined(windows)): proc testAcceptTooMany(address: TransportAddress): Future[bool] {.async.} = let maxFiles = getMaxOpenFiles() var server = createStreamServer(address, flags = {ReuseAddr}) let isock = int(server.sock) let newMaxFiles = isock + 4 setMaxOpenFiles(newMaxFiles) proc acceptTask(server: StreamServer): Future[bool] {.async.} = var transports = newSeq[StreamTransport]() try: for i in 0 ..< 3: let transp = await server.accept() transports.add(transp) except TransportTooManyError: var pending = newSeq[Future[void]]() for item in transports: pending.add(closeWait(item)) await allFutures(pending) return true var acceptFut = acceptTask(server) try: for i in 0 ..< 3: try: let transp = await connect(address) await sleepAsync(10.milliseconds) await transp.closeWait() except TransportTooManyError: break if await withTimeout(acceptFut, 5.seconds): if acceptFut.finished() and not(acceptFut.failed()): if acceptFut.read() == true: result = true finally: await server.closeWait() setMaxOpenFiles(maxFiles) proc testWriteOnClose(address: TransportAddress): Future[bool] {.async.} = var server = createStreamServer(address, flags = {ReuseAddr, NoPipeFlash}) var res = 0 proc acceptTask(server: StreamServer) {.async.} = let transp = await server.accept() var futs = newSeq[Future[int]]() var msg = createBigMessage(1024) var tries = 0 while futs.len() < TestsCount: let fut = transp.write(msg) # `write` has a fast path that puts the data in the OS socket buffer - # we'll keep writing until we get EAGAIN from the OS so that we have # data in the in-chronos queue to fail on close if not fut.completed(): futs.add(fut) else: tries += 1 if tries > 65*1024: # We've queued 64mb on the socket and it still allows writing, # something is wrong - we'll break here which will cause the test # to fail break await transp.closeWait() await sleepAsync(100.milliseconds) for i in 0 ..< len(futs): # writes may complete via fast write if futs[i].failed() and (futs[i].error of TransportUseClosedError): inc(res) await server.closeWait() var acceptFut = acceptTask(server) var transp = await connect(address) await server.join() await transp.closeWait() await acceptFut return (res == TestsCount) proc testReadOnClose(address: TransportAddress): Future[bool] {.async.} = var server = createStreamServer(address, flags = {ReuseAddr, NoPipeFlash}) var res = false proc acceptTask(server: StreamServer) {.async.} = let transp = await server.accept() var buffer = newSeq[byte](1024) var fut = transp.readOnce(addr buffer[0], len(buffer)) await transp.closeWait() await sleepAsync(100.milliseconds) if fut.failed() and (fut.error of TransportUseClosedError): res = true await server.closeWait() var acceptFut = acceptTask(server) var transp = await connect(address) await server.join() await transp.closeWait() await acceptFut return res proc testAcceptRace(address: TransportAddress): Future[bool] {.async.} = proc test1(address: TransportAddress) {.async.} = let server = createStreamServer(address, flags = {ReuseAddr}) let acceptFut = server.accept() server.close() await allFutures(acceptFut.cancelAndWait(), server.join()) proc test2(address: TransportAddress) {.async.} = let server = createStreamServer(address, flags = {ReuseAddr}) let acceptFut = server.accept() await acceptFut.cancelAndWait() server.close() await server.join() proc test3(address: TransportAddress) {.async.} = let server = createStreamServer(address, flags = {ReuseAddr}) let acceptFut = server.accept() server.stop() server.close() await allFutures(acceptFut.cancelAndWait(), server.join()) proc test4(address: TransportAddress) {.async.} = let server = createStreamServer(address, flags = {ReuseAddr}) let acceptFut = server.accept() await acceptFut.cancelAndWait() server.stop() server.close() await server.join() try: await test1(address).wait(5.seconds) await test2(address).wait(5.seconds) await test3(address).wait(5.seconds) await test4(address).wait(5.seconds) return true except AsyncTimeoutError: return false proc testPipe(): Future[bool] {.async.} = let (rfd, wfd) = createAsyncPipe() let message = createBigMessage(16384 * 1024) rtransp = fromPipe(rfd) wtransp = fromPipe(wfd) var buffer = newSeq[byte](16384 * 1024) proc writer(transp: StreamTransport): Future[int] {.async.} = let res = try: await transp.write(message) except CatchableError: -1 return res var fut {.used.} = wtransp.writer() try: await rtransp.readExactly(addr buffer[0], 16384 * 1024) except CatchableError: discard await allFutures(rtransp.closeWait(), wtransp.closeWait()) return buffer == message proc testConnectBindLocalAddress() {.async.} = let dst1 = initTAddress("127.0.0.1:33335") let dst2 = initTAddress("127.0.0.1:33336") let dst3 = initTAddress("127.0.0.1:33337") proc client(server: StreamServer, transp: StreamTransport) {.async.} = await transp.closeWait() # We use ReuseAddr here only to be able to reuse the same IP/Port when there's a TIME_WAIT socket. It's useful when # running the test multiple times or if a test ran previously used the same port. let servers = [createStreamServer(dst1, client, {ReuseAddr}), createStreamServer(dst2, client, {ReuseAddr}), createStreamServer(dst3, client, {ReusePort})] for server in servers: server.start() let ta = initTAddress("0.0.0.0:35000") # It works cause there's no active listening socket bound to ta and we are using ReuseAddr var transp1 = await connect(dst1, localAddress = ta, flags={SocketFlags.ReuseAddr}) var transp2 = await connect(dst2, localAddress = ta, flags={SocketFlags.ReuseAddr}) # It works cause even thought there's an active listening socket bound to dst3, we are using ReusePort var transp3 = await connect(dst2, localAddress = dst3, flags={SocketFlags.ReusePort}) expect(TransportOsError): var transp2 {.used.} = await connect(dst3, localAddress = ta) expect(TransportOsError): var transp3 {.used.} = await connect(dst3, localAddress = initTAddress(":::35000")) await transp1.closeWait() await transp2.closeWait() await transp3.closeWait() for server in servers: server.stop() await server.closeWait() markFD = getCurrentFD() for i in 0..