2021-02-02 22:33:14 +00:00
|
|
|
# Chronos Test Suite
|
|
|
|
# (c) Copyright 2021-Present
|
|
|
|
# Status Research & Development GmbH
|
|
|
|
#
|
|
|
|
# Licensed under either of
|
|
|
|
# Apache License, version 2.0, (LICENSE-APACHEv2)
|
|
|
|
# MIT license (LICENSE-MIT)
|
exception tracking (#166)
* exception tracking
This PR adds minimal exception tracking to chronos, moving the goalpost
one step further.
In particular, it becomes invalid to raise exceptions from `callSoon`
callbacks: this is critical for writing correct error handling because
there's no reasonable way that a user of chronos can possibly _reason_
about exceptions coming out of there: the event loop will be in an
indeterminite state when the loop is executing an _random_ callback.
As expected, there are several issues in the error handling of chronos:
in particular, it will end up in an inconsistent internal state whenever
the selector loop operations fail, because the internal state update
functions are not written in an exception-safe way. This PR turns this
into a Defect, which probably is not the optimal way of handling things
- expect more work to be done here.
Some API have no way of reporting back errors to callers - for example,
when something fails in the accept loop, there's not much it can do, and
no way to report it back to the user of the API - this has been fixed
with the new accept flow - the old one should be deprecated.
Finally, there is information loss in the API: in composite operations
like `poll` and `waitFor` there's no way to differentiate internal
errors from user-level errors originating from callbacks.
* store `CatchableError` in future
* annotate proc's with correct raises information
* `selectors2` to avoid non-CatchableError IOSelectorsException
* `$` should never raise
* remove unnecessary gcsafe annotations
* fix exceptions leaking out of timer waits
* fix some imports
* functions must signal raising the union of all exceptions across all
platforms to enable cross-platform code
* switch to unittest2
* add `selectors2` which supercedes the std library version and fixes
several exception handling issues in there
* fixes
* docs, platform-independent eh specifiers for some functions
* add feature flag for strict exception mode
also bump version to 3.0.0 - _most_ existing code should be compatible
with this version of exception handling but some things might need
fixing - callbacks, existing raises specifications etc.
* fix AsyncCheck for non-void T
2021-03-24 09:08:33 +00:00
|
|
|
import std/[strutils, algorithm, strutils]
|
|
|
|
import unittest2
|
2021-05-17 19:39:24 +00:00
|
|
|
import ../chronos, ../chronos/apps/http/httpserver,
|
|
|
|
../chronos/apps/http/httpcommon
|
2021-03-05 22:22:32 +00:00
|
|
|
import stew/base10
|
2021-02-02 22:33:14 +00:00
|
|
|
|
2023-03-31 05:35:04 +00:00
|
|
|
{.used.}
|
2021-02-03 10:47:03 +00:00
|
|
|
|
2021-02-02 22:33:14 +00:00
|
|
|
suite "HTTP server testing suite":
|
2021-02-18 12:08:21 +00:00
|
|
|
type
|
|
|
|
TooBigTest = enum
|
|
|
|
GetBodyTest, ConsumeBodyTest, PostUrlTest, PostMultipartTest
|
|
|
|
|
2021-02-02 22:33:14 +00:00
|
|
|
proc httpClient(address: TransportAddress,
|
|
|
|
data: string): Future[string] {.async.} =
|
|
|
|
var transp: StreamTransport
|
|
|
|
try:
|
|
|
|
transp = await connect(address)
|
|
|
|
if len(data) > 0:
|
|
|
|
let wres {.used.} = await transp.write(data)
|
|
|
|
var rres = await transp.read()
|
2021-02-18 12:08:21 +00:00
|
|
|
return bytesToString(rres)
|
2021-02-02 22:33:14 +00:00
|
|
|
except CatchableError:
|
|
|
|
return "EXCEPTION"
|
|
|
|
finally:
|
|
|
|
if not(isNil(transp)):
|
|
|
|
await closeWait(transp)
|
|
|
|
|
2021-02-10 08:43:05 +00:00
|
|
|
proc testTooBigBodyChunked(address: TransportAddress,
|
2021-02-18 12:08:21 +00:00
|
|
|
operation: TooBigTest): Future[bool] {.async.} =
|
2021-02-10 08:43:05 +00:00
|
|
|
var serverRes = false
|
2021-02-18 12:08:21 +00:00
|
|
|
proc process(r: RequestFence): Future[HttpResponseRef] {.
|
2021-02-10 08:43:05 +00:00
|
|
|
async.} =
|
|
|
|
if r.isOk():
|
|
|
|
let request = r.get()
|
|
|
|
try:
|
2021-02-18 12:08:21 +00:00
|
|
|
case operation
|
|
|
|
of GetBodyTest:
|
2021-02-10 08:43:05 +00:00
|
|
|
let body {.used.} = await request.getBody()
|
2021-02-18 12:08:21 +00:00
|
|
|
of ConsumeBodyTest:
|
2021-02-10 08:43:05 +00:00
|
|
|
await request.consumeBody()
|
2021-02-18 12:08:21 +00:00
|
|
|
of PostUrlTest:
|
2021-02-10 08:43:05 +00:00
|
|
|
let ptable {.used.} = await request.post()
|
2021-02-18 12:08:21 +00:00
|
|
|
of PostMultipartTest:
|
2021-02-10 08:43:05 +00:00
|
|
|
let ptable {.used.} = await request.post()
|
|
|
|
except HttpCriticalError as exc:
|
|
|
|
if exc.code == Http413:
|
|
|
|
serverRes = true
|
|
|
|
# Reraising exception, because processor should properly handle it.
|
|
|
|
raise exc
|
|
|
|
else:
|
|
|
|
return dumbResponse()
|
|
|
|
|
|
|
|
let socketFlags = {ServerFlags.TcpNoDelay, ServerFlags.ReuseAddr}
|
|
|
|
let res = HttpServerRef.new(address, process,
|
|
|
|
maxRequestBodySize = 10,
|
|
|
|
socketFlags = socketFlags)
|
|
|
|
if res.isErr():
|
|
|
|
return false
|
|
|
|
|
|
|
|
let server = res.get()
|
|
|
|
server.start()
|
|
|
|
|
|
|
|
let request =
|
2021-02-18 12:08:21 +00:00
|
|
|
case operation
|
|
|
|
of GetBodyTest, ConsumeBodyTest, PostUrlTest:
|
2021-02-10 08:43:05 +00:00
|
|
|
"POST / HTTP/1.0\r\n" &
|
|
|
|
"Content-Type: application/x-www-form-urlencoded\r\n" &
|
|
|
|
"Transfer-Encoding: chunked\r\n" &
|
|
|
|
"Cookie: 2\r\n\r\n" &
|
|
|
|
"5\r\na=a&b\r\n5\r\n=b&c=\r\n4\r\nc&d=\r\n4\r\n%D0%\r\n" &
|
|
|
|
"2\r\n9F\r\n0\r\n\r\n"
|
2021-02-18 12:08:21 +00:00
|
|
|
of PostMultipartTest:
|
2021-02-10 08:43:05 +00:00
|
|
|
"POST / HTTP/1.0\r\n" &
|
|
|
|
"Host: 127.0.0.1:30080\r\n" &
|
|
|
|
"Transfer-Encoding: chunked\r\n" &
|
|
|
|
"Content-Type: multipart/form-data; boundary=f98f0\r\n\r\n" &
|
|
|
|
"3b\r\n--f98f0\r\nContent-Disposition: form-data; name=\"key1\"" &
|
|
|
|
"\r\n\r\nA\r\n\r\n" &
|
|
|
|
"3b\r\n--f98f0\r\nContent-Disposition: form-data; name=\"key2\"" &
|
|
|
|
"\r\n\r\nB\r\n\r\n" &
|
|
|
|
"3b\r\n--f98f0\r\nContent-Disposition: form-data; name=\"key3\"" &
|
|
|
|
"\r\n\r\nC\r\n\r\n" &
|
|
|
|
"b\r\n--f98f0--\r\n\r\n" &
|
|
|
|
"0\r\n\r\n"
|
|
|
|
|
|
|
|
let data = await httpClient(address, request)
|
|
|
|
await server.stop()
|
|
|
|
await server.closeWait()
|
|
|
|
return serverRes and (data.startsWith("HTTP/1.1 413"))
|
|
|
|
|
2021-02-02 22:33:14 +00:00
|
|
|
test "Request headers timeout test":
|
|
|
|
proc testTimeout(address: TransportAddress): Future[bool] {.async.} =
|
|
|
|
var serverRes = false
|
2021-02-18 12:08:21 +00:00
|
|
|
proc process(r: RequestFence): Future[HttpResponseRef] {.
|
2021-02-02 22:33:14 +00:00
|
|
|
async.} =
|
|
|
|
if r.isOk():
|
|
|
|
let request = r.get()
|
|
|
|
return await request.respond(Http200, "TEST_OK", HttpTable.init())
|
|
|
|
else:
|
2021-12-10 09:55:25 +00:00
|
|
|
if r.error().error == HttpServerError.TimeoutError:
|
2021-02-02 22:33:14 +00:00
|
|
|
serverRes = true
|
|
|
|
return dumbResponse()
|
|
|
|
|
|
|
|
let socketFlags = {ServerFlags.TcpNoDelay, ServerFlags.ReuseAddr}
|
|
|
|
let res = HttpServerRef.new(address, process, socketFlags = socketFlags,
|
|
|
|
httpHeadersTimeout = 100.milliseconds)
|
|
|
|
if res.isErr():
|
|
|
|
return false
|
|
|
|
|
|
|
|
let server = res.get()
|
|
|
|
server.start()
|
|
|
|
|
|
|
|
let data = await httpClient(address, "")
|
|
|
|
await server.stop()
|
2021-02-10 08:43:05 +00:00
|
|
|
await server.closeWait()
|
2021-02-02 22:33:14 +00:00
|
|
|
return serverRes and (data.startsWith("HTTP/1.1 408"))
|
|
|
|
|
|
|
|
check waitFor(testTimeout(initTAddress("127.0.0.1:30080"))) == true
|
|
|
|
|
|
|
|
test "Empty headers test":
|
|
|
|
proc testEmpty(address: TransportAddress): Future[bool] {.async.} =
|
|
|
|
var serverRes = false
|
2021-02-18 12:08:21 +00:00
|
|
|
proc process(r: RequestFence): Future[HttpResponseRef] {.
|
2021-02-02 22:33:14 +00:00
|
|
|
async.} =
|
|
|
|
if r.isOk():
|
|
|
|
let request = r.get()
|
|
|
|
return await request.respond(Http200, "TEST_OK", HttpTable.init())
|
|
|
|
else:
|
2021-12-10 09:55:25 +00:00
|
|
|
if r.error().error == HttpServerError.CriticalError:
|
2021-02-02 22:33:14 +00:00
|
|
|
serverRes = true
|
|
|
|
return dumbResponse()
|
|
|
|
|
|
|
|
let socketFlags = {ServerFlags.TcpNoDelay, ServerFlags.ReuseAddr}
|
|
|
|
let res = HttpServerRef.new(address, process, socketFlags = socketFlags)
|
|
|
|
if res.isErr():
|
|
|
|
return false
|
|
|
|
|
|
|
|
let server = res.get()
|
|
|
|
server.start()
|
|
|
|
|
|
|
|
let data = await httpClient(address, "\r\n\r\n")
|
|
|
|
await server.stop()
|
2021-02-10 08:43:05 +00:00
|
|
|
await server.closeWait()
|
2021-02-02 22:33:14 +00:00
|
|
|
return serverRes and (data.startsWith("HTTP/1.1 400"))
|
|
|
|
|
|
|
|
check waitFor(testEmpty(initTAddress("127.0.0.1:30080"))) == true
|
|
|
|
|
|
|
|
test "Too big headers test":
|
|
|
|
proc testTooBig(address: TransportAddress): Future[bool] {.async.} =
|
|
|
|
var serverRes = false
|
2021-02-18 12:08:21 +00:00
|
|
|
proc process(r: RequestFence): Future[HttpResponseRef] {.
|
2021-02-02 22:33:14 +00:00
|
|
|
async.} =
|
|
|
|
if r.isOk():
|
|
|
|
let request = r.get()
|
|
|
|
return await request.respond(Http200, "TEST_OK", HttpTable.init())
|
|
|
|
else:
|
2021-12-10 09:55:25 +00:00
|
|
|
if r.error().error == HttpServerError.CriticalError:
|
2021-02-02 22:33:14 +00:00
|
|
|
serverRes = true
|
|
|
|
return dumbResponse()
|
|
|
|
|
|
|
|
let socketFlags = {ServerFlags.TcpNoDelay, ServerFlags.ReuseAddr}
|
|
|
|
let res = HttpServerRef.new(address, process,
|
|
|
|
maxHeadersSize = 10,
|
|
|
|
socketFlags = socketFlags)
|
|
|
|
if res.isErr():
|
|
|
|
return false
|
|
|
|
|
|
|
|
let server = res.get()
|
|
|
|
server.start()
|
|
|
|
|
|
|
|
let data = await httpClient(address, "GET / HTTP/1.1\r\n\r\n")
|
|
|
|
await server.stop()
|
2021-02-10 08:43:05 +00:00
|
|
|
await server.closeWait()
|
2021-02-21 15:34:55 +00:00
|
|
|
return serverRes and (data.startsWith("HTTP/1.1 431"))
|
2021-02-02 22:33:14 +00:00
|
|
|
|
|
|
|
check waitFor(testTooBig(initTAddress("127.0.0.1:30080"))) == true
|
|
|
|
|
2021-02-10 08:43:05 +00:00
|
|
|
test "Too big request body test (content-length)":
|
|
|
|
proc testTooBigBody(address: TransportAddress): Future[bool] {.async.} =
|
|
|
|
var serverRes = false
|
2021-02-18 12:08:21 +00:00
|
|
|
proc process(r: RequestFence): Future[HttpResponseRef] {.
|
2021-02-10 08:43:05 +00:00
|
|
|
async.} =
|
|
|
|
if r.isOk():
|
|
|
|
discard
|
|
|
|
else:
|
2021-12-10 09:55:25 +00:00
|
|
|
if r.error().error == HttpServerError.CriticalError:
|
2021-02-10 08:43:05 +00:00
|
|
|
serverRes = true
|
|
|
|
return dumbResponse()
|
|
|
|
|
|
|
|
let socketFlags = {ServerFlags.TcpNoDelay, ServerFlags.ReuseAddr}
|
|
|
|
let res = HttpServerRef.new(address, process,
|
|
|
|
maxRequestBodySize = 10,
|
|
|
|
socketFlags = socketFlags)
|
|
|
|
if res.isErr():
|
|
|
|
return false
|
|
|
|
|
|
|
|
let server = res.get()
|
|
|
|
server.start()
|
|
|
|
|
|
|
|
let request = "GET / HTTP/1.1\r\nContent-Length: 20\r\n\r\n"
|
|
|
|
let data = await httpClient(address, request)
|
|
|
|
await server.stop()
|
|
|
|
await server.closeWait()
|
|
|
|
return serverRes and (data.startsWith("HTTP/1.1 413"))
|
|
|
|
|
|
|
|
check waitFor(testTooBigBody(initTAddress("127.0.0.1:30080"))) == true
|
|
|
|
|
|
|
|
test "Too big request body test (getBody()/chunked encoding)":
|
|
|
|
check:
|
2021-02-18 12:08:21 +00:00
|
|
|
waitFor(testTooBigBodyChunked(initTAddress("127.0.0.1:30080"),
|
|
|
|
GetBodyTest)) == true
|
2021-02-10 08:43:05 +00:00
|
|
|
|
|
|
|
test "Too big request body test (consumeBody()/chunked encoding)":
|
|
|
|
check:
|
2021-02-18 12:08:21 +00:00
|
|
|
waitFor(testTooBigBodyChunked(initTAddress("127.0.0.1:30080"),
|
|
|
|
ConsumeBodyTest)) == true
|
2021-02-10 08:43:05 +00:00
|
|
|
|
|
|
|
test "Too big request body test (post()/urlencoded/chunked encoding)":
|
|
|
|
check:
|
2021-02-18 12:08:21 +00:00
|
|
|
waitFor(testTooBigBodyChunked(initTAddress("127.0.0.1:30080"),
|
|
|
|
PostUrlTest)) == true
|
2021-02-10 08:43:05 +00:00
|
|
|
|
|
|
|
test "Too big request body test (post()/multipart/chunked encoding)":
|
|
|
|
check:
|
2021-02-18 12:08:21 +00:00
|
|
|
waitFor(testTooBigBodyChunked(initTAddress("127.0.0.1:30080"),
|
|
|
|
PostMultipartTest)) == true
|
2021-02-10 08:43:05 +00:00
|
|
|
|
2021-02-02 22:33:14 +00:00
|
|
|
test "Query arguments test":
|
|
|
|
proc testQuery(address: TransportAddress): Future[bool] {.async.} =
|
|
|
|
var serverRes = false
|
2021-02-18 12:08:21 +00:00
|
|
|
proc process(r: RequestFence): Future[HttpResponseRef] {.
|
2021-02-02 22:33:14 +00:00
|
|
|
async.} =
|
|
|
|
if r.isOk():
|
|
|
|
let request = r.get()
|
|
|
|
var kres = newSeq[string]()
|
|
|
|
for k, v in request.query.stringItems():
|
|
|
|
kres.add(k & ":" & v)
|
|
|
|
sort(kres)
|
|
|
|
serverRes = true
|
|
|
|
return await request.respond(Http200, "TEST_OK:" & kres.join(":"),
|
|
|
|
HttpTable.init())
|
|
|
|
else:
|
|
|
|
serverRes = false
|
|
|
|
return dumbResponse()
|
|
|
|
|
|
|
|
let socketFlags = {ServerFlags.TcpNoDelay, ServerFlags.ReuseAddr}
|
|
|
|
let res = HttpServerRef.new(address, process,
|
|
|
|
socketFlags = socketFlags)
|
|
|
|
if res.isErr():
|
|
|
|
return false
|
|
|
|
|
|
|
|
let server = res.get()
|
|
|
|
server.start()
|
|
|
|
|
|
|
|
let data1 = await httpClient(address,
|
|
|
|
"GET /?a=1&a=2&b=3&c=4 HTTP/1.0\r\n\r\n")
|
|
|
|
let data2 = await httpClient(address,
|
|
|
|
"GET /?a=%D0%9F&%D0%A4=%D0%91&b=%D0%A6&c=%D0%AE HTTP/1.0\r\n\r\n")
|
|
|
|
await server.stop()
|
2021-02-10 08:43:05 +00:00
|
|
|
await server.closeWait()
|
2021-02-02 22:33:14 +00:00
|
|
|
let r = serverRes and
|
|
|
|
(data1.find("TEST_OK:a:1:a:2:b:3:c:4") >= 0) and
|
|
|
|
(data2.find("TEST_OK:a:П:b:Ц:c:Ю:Ф:Б") >= 0)
|
|
|
|
return r
|
|
|
|
|
|
|
|
check waitFor(testQuery(initTAddress("127.0.0.1:30080"))) == true
|
|
|
|
|
|
|
|
test "Headers test":
|
|
|
|
proc testHeaders(address: TransportAddress): Future[bool] {.async.} =
|
|
|
|
var serverRes = false
|
2021-02-18 12:08:21 +00:00
|
|
|
proc process(r: RequestFence): Future[HttpResponseRef] {.
|
2021-02-02 22:33:14 +00:00
|
|
|
async.} =
|
|
|
|
if r.isOk():
|
|
|
|
let request = r.get()
|
|
|
|
var kres = newSeq[string]()
|
|
|
|
for k, v in request.headers.stringItems():
|
|
|
|
kres.add(k & ":" & v)
|
|
|
|
sort(kres)
|
|
|
|
serverRes = true
|
|
|
|
return await request.respond(Http200, "TEST_OK:" & kres.join(":"),
|
|
|
|
HttpTable.init())
|
|
|
|
else:
|
|
|
|
serverRes = false
|
|
|
|
return dumbResponse()
|
|
|
|
|
|
|
|
let socketFlags = {ServerFlags.TcpNoDelay, ServerFlags.ReuseAddr}
|
|
|
|
let res = HttpServerRef.new(address, process,
|
|
|
|
socketFlags = socketFlags)
|
|
|
|
if res.isErr():
|
|
|
|
return false
|
|
|
|
|
|
|
|
let server = res.get()
|
|
|
|
server.start()
|
|
|
|
|
|
|
|
let message =
|
|
|
|
"GET / HTTP/1.0\r\n" &
|
|
|
|
"Host: www.google.com\r\n" &
|
|
|
|
"Content-Type: text/html\r\n" &
|
|
|
|
"Expect: 100-continue\r\n" &
|
|
|
|
"Cookie: 1\r\n" &
|
|
|
|
"Cookie: 2\r\n\r\n"
|
|
|
|
let expect = "TEST_OK:content-type:text/html:cookie:1:cookie:2" &
|
|
|
|
":expect:100-continue:host:www.google.com"
|
|
|
|
let data = await httpClient(address, message)
|
|
|
|
await server.stop()
|
2021-02-10 08:43:05 +00:00
|
|
|
await server.closeWait()
|
2021-02-02 22:33:14 +00:00
|
|
|
return serverRes and (data.find(expect) >= 0)
|
|
|
|
|
|
|
|
check waitFor(testHeaders(initTAddress("127.0.0.1:30080"))) == true
|
|
|
|
|
2021-02-10 08:43:05 +00:00
|
|
|
test "POST arguments (urlencoded/content-length) test":
|
2021-02-02 22:33:14 +00:00
|
|
|
proc testPostUrl(address: TransportAddress): Future[bool] {.async.} =
|
|
|
|
var serverRes = false
|
2021-02-18 12:08:21 +00:00
|
|
|
proc process(r: RequestFence): Future[HttpResponseRef] {.
|
2021-02-02 22:33:14 +00:00
|
|
|
async.} =
|
|
|
|
if r.isOk():
|
|
|
|
var kres = newSeq[string]()
|
|
|
|
let request = r.get()
|
|
|
|
if request.meth in PostMethods:
|
|
|
|
let post = await request.post()
|
|
|
|
for k, v in post.stringItems():
|
|
|
|
kres.add(k & ":" & v)
|
|
|
|
sort(kres)
|
|
|
|
serverRes = true
|
|
|
|
return await request.respond(Http200, "TEST_OK:" & kres.join(":"),
|
|
|
|
HttpTable.init())
|
|
|
|
else:
|
|
|
|
serverRes = false
|
|
|
|
return dumbResponse()
|
|
|
|
|
|
|
|
let socketFlags = {ServerFlags.TcpNoDelay, ServerFlags.ReuseAddr}
|
|
|
|
let res = HttpServerRef.new(address, process,
|
|
|
|
socketFlags = socketFlags)
|
|
|
|
if res.isErr():
|
|
|
|
return false
|
|
|
|
|
|
|
|
let server = res.get()
|
|
|
|
server.start()
|
|
|
|
|
|
|
|
let message =
|
|
|
|
"POST / HTTP/1.0\r\n" &
|
|
|
|
"Content-Type: application/x-www-form-urlencoded\r\n" &
|
2021-03-05 22:22:32 +00:00
|
|
|
"Content-Length: 20\r\n" &
|
2021-02-02 22:33:14 +00:00
|
|
|
"Cookie: 2\r\n\r\n" &
|
|
|
|
"a=a&b=b&c=c&d=%D0%9F"
|
|
|
|
let data = await httpClient(address, message)
|
|
|
|
let expect = "TEST_OK:a:a:b:b:c:c:d:П"
|
|
|
|
await server.stop()
|
2021-02-10 08:43:05 +00:00
|
|
|
await server.closeWait()
|
2021-02-02 22:33:14 +00:00
|
|
|
return serverRes and (data.find(expect) >= 0)
|
|
|
|
|
|
|
|
check waitFor(testPostUrl(initTAddress("127.0.0.1:30080"))) == true
|
|
|
|
|
2021-02-10 08:43:05 +00:00
|
|
|
test "POST arguments (urlencoded/chunked encoding) test":
|
|
|
|
proc testPostUrl2(address: TransportAddress): Future[bool] {.async.} =
|
|
|
|
var serverRes = false
|
2021-02-18 12:08:21 +00:00
|
|
|
proc process(r: RequestFence): Future[HttpResponseRef] {.
|
2021-02-10 08:43:05 +00:00
|
|
|
async.} =
|
|
|
|
if r.isOk():
|
|
|
|
var kres = newSeq[string]()
|
|
|
|
let request = r.get()
|
|
|
|
if request.meth in PostMethods:
|
|
|
|
let post = await request.post()
|
|
|
|
for k, v in post.stringItems():
|
|
|
|
kres.add(k & ":" & v)
|
|
|
|
sort(kres)
|
|
|
|
serverRes = true
|
|
|
|
return await request.respond(Http200, "TEST_OK:" & kres.join(":"),
|
|
|
|
HttpTable.init())
|
|
|
|
else:
|
|
|
|
serverRes = false
|
|
|
|
return dumbResponse()
|
|
|
|
|
|
|
|
let socketFlags = {ServerFlags.TcpNoDelay, ServerFlags.ReuseAddr}
|
|
|
|
let res = HttpServerRef.new(address, process,
|
|
|
|
socketFlags = socketFlags)
|
|
|
|
if res.isErr():
|
|
|
|
return false
|
|
|
|
|
|
|
|
let server = res.get()
|
|
|
|
server.start()
|
|
|
|
|
|
|
|
let message =
|
|
|
|
"POST / HTTP/1.0\r\n" &
|
|
|
|
"Content-Type: application/x-www-form-urlencoded\r\n" &
|
|
|
|
"Transfer-Encoding: chunked\r\n" &
|
|
|
|
"Cookie: 2\r\n\r\n" &
|
|
|
|
"5\r\na=a&b\r\n5\r\n=b&c=\r\n4\r\nc&d=\r\n4\r\n%D0%\r\n" &
|
|
|
|
"2\r\n9F\r\n0\r\n\r\n"
|
|
|
|
let data = await httpClient(address, message)
|
|
|
|
let expect = "TEST_OK:a:a:b:b:c:c:d:П"
|
|
|
|
await server.stop()
|
|
|
|
await server.closeWait()
|
|
|
|
return serverRes and (data.find(expect) >= 0)
|
|
|
|
|
|
|
|
check waitFor(testPostUrl2(initTAddress("127.0.0.1:30080"))) == true
|
|
|
|
|
|
|
|
test "POST arguments (multipart/content-length) test":
|
2021-02-02 22:33:14 +00:00
|
|
|
proc testPostMultipart(address: TransportAddress): Future[bool] {.async.} =
|
|
|
|
var serverRes = false
|
2021-02-18 12:08:21 +00:00
|
|
|
proc process(r: RequestFence): Future[HttpResponseRef] {.
|
2021-02-02 22:33:14 +00:00
|
|
|
async.} =
|
|
|
|
if r.isOk():
|
|
|
|
var kres = newSeq[string]()
|
|
|
|
let request = r.get()
|
|
|
|
if request.meth in PostMethods:
|
|
|
|
let post = await request.post()
|
|
|
|
for k, v in post.stringItems():
|
|
|
|
kres.add(k & ":" & v)
|
|
|
|
sort(kres)
|
|
|
|
serverRes = true
|
|
|
|
return await request.respond(Http200, "TEST_OK:" & kres.join(":"),
|
|
|
|
HttpTable.init())
|
|
|
|
else:
|
|
|
|
serverRes = false
|
|
|
|
return dumbResponse()
|
|
|
|
|
|
|
|
let socketFlags = {ServerFlags.TcpNoDelay, ServerFlags.ReuseAddr}
|
|
|
|
let res = HttpServerRef.new(address, process,
|
|
|
|
socketFlags = socketFlags)
|
|
|
|
if res.isErr():
|
|
|
|
return false
|
|
|
|
|
|
|
|
let server = res.get()
|
|
|
|
server.start()
|
|
|
|
|
|
|
|
let message =
|
|
|
|
"POST / HTTP/1.0\r\n" &
|
|
|
|
"Host: 127.0.0.1:30080\r\n" &
|
|
|
|
"User-Agent: curl/7.55.1\r\n" &
|
|
|
|
"Accept: */*\r\n" &
|
|
|
|
"Content-Length: 343\r\n" &
|
|
|
|
"Content-Type: multipart/form-data; " &
|
|
|
|
"boundary=------------------------ab5706ba6f80b795\r\n\r\n" &
|
|
|
|
"--------------------------ab5706ba6f80b795\r\n" &
|
|
|
|
"Content-Disposition: form-data; name=\"key1\"\r\n\r\n" &
|
|
|
|
"value1\r\n" &
|
|
|
|
"--------------------------ab5706ba6f80b795\r\n" &
|
|
|
|
"Content-Disposition: form-data; name=\"key2\"\r\n\r\n" &
|
|
|
|
"value2\r\n" &
|
|
|
|
"--------------------------ab5706ba6f80b795\r\n" &
|
|
|
|
"Content-Disposition: form-data; name=\"key2\"\r\n\r\n" &
|
|
|
|
"value4\r\n" &
|
|
|
|
"--------------------------ab5706ba6f80b795--\r\n"
|
|
|
|
let data = await httpClient(address, message)
|
|
|
|
let expect = "TEST_OK:key1:value1:key2:value2:key2:value4"
|
|
|
|
await server.stop()
|
2021-02-10 08:43:05 +00:00
|
|
|
await server.closeWait()
|
2021-02-02 22:33:14 +00:00
|
|
|
return serverRes and (data.find(expect) >= 0)
|
|
|
|
|
|
|
|
check waitFor(testPostMultipart(initTAddress("127.0.0.1:30080"))) == true
|
|
|
|
|
2021-02-10 08:43:05 +00:00
|
|
|
test "POST arguments (multipart/chunked encoding) test":
|
2021-02-02 22:33:14 +00:00
|
|
|
proc testPostMultipart2(address: TransportAddress): Future[bool] {.async.} =
|
|
|
|
var serverRes = false
|
2021-02-18 12:08:21 +00:00
|
|
|
proc process(r: RequestFence): Future[HttpResponseRef] {.
|
2021-02-02 22:33:14 +00:00
|
|
|
async.} =
|
|
|
|
if r.isOk():
|
|
|
|
var kres = newSeq[string]()
|
|
|
|
let request = r.get()
|
|
|
|
if request.meth in PostMethods:
|
|
|
|
let post = await request.post()
|
|
|
|
for k, v in post.stringItems():
|
|
|
|
kres.add(k & ":" & v)
|
|
|
|
sort(kres)
|
|
|
|
serverRes = true
|
|
|
|
return await request.respond(Http200, "TEST_OK:" & kres.join(":"),
|
|
|
|
HttpTable.init())
|
|
|
|
else:
|
|
|
|
serverRes = false
|
|
|
|
return dumbResponse()
|
|
|
|
|
|
|
|
let socketFlags = {ServerFlags.TcpNoDelay, ServerFlags.ReuseAddr}
|
|
|
|
let res = HttpServerRef.new(address, process,
|
|
|
|
socketFlags = socketFlags)
|
|
|
|
if res.isErr():
|
|
|
|
return false
|
|
|
|
|
|
|
|
let server = res.get()
|
|
|
|
server.start()
|
|
|
|
|
|
|
|
let message =
|
|
|
|
"POST / HTTP/1.0\r\n" &
|
|
|
|
"Host: 127.0.0.1:30080\r\n" &
|
|
|
|
"Transfer-Encoding: chunked\r\n" &
|
|
|
|
"Content-Type: multipart/form-data; boundary=---" &
|
|
|
|
"---------------------f98f0e32c55fa2ae\r\n\r\n" &
|
|
|
|
"271\r\n" &
|
|
|
|
"--------------------------f98f0e32c55fa2ae\r\n" &
|
|
|
|
"Content-Disposition: form-data; name=\"key1\"\r\n\r\n" &
|
|
|
|
"AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA" &
|
|
|
|
"AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA\r\n" &
|
|
|
|
"--------------------------f98f0e32c55fa2ae\r\n" &
|
|
|
|
"Content-Disposition: form-data; name=\"key2\"\r\n\r\n" &
|
|
|
|
"BBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBB" &
|
|
|
|
"BBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBB\r\n" &
|
|
|
|
"--------------------------f98f0e32c55fa2ae\r\n" &
|
|
|
|
"Content-Disposition: form-data; name=\"key2\"\r\n\r\n" &
|
|
|
|
"CCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCC" &
|
|
|
|
"CCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCC\r\n" &
|
|
|
|
"--------------------------f98f0e32c55fa2ae--\r\n" &
|
|
|
|
"\r\n0\r\n\r\n"
|
|
|
|
|
|
|
|
let data = await httpClient(address, message)
|
|
|
|
let expect = "TEST_OK:key1:AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA" &
|
|
|
|
"AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA" &
|
|
|
|
"AAAAA:key2:BBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBB" &
|
|
|
|
"BBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBB" &
|
|
|
|
"BBB:key2:CCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCC" &
|
|
|
|
"CCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCC"
|
|
|
|
await server.stop()
|
2021-02-10 08:43:05 +00:00
|
|
|
await server.closeWait()
|
2021-02-02 22:33:14 +00:00
|
|
|
return serverRes and (data.find(expect) >= 0)
|
|
|
|
|
|
|
|
check waitFor(testPostMultipart2(initTAddress("127.0.0.1:30080"))) == true
|
|
|
|
|
2021-02-19 12:07:20 +00:00
|
|
|
test "drop() connections test":
|
|
|
|
const ClientsCount = 10
|
|
|
|
|
|
|
|
proc testHTTPdrop(address: TransportAddress): Future[bool] {.async.} =
|
|
|
|
var eventWait = newAsyncEvent()
|
|
|
|
var eventContinue = newAsyncEvent()
|
|
|
|
var count = 0
|
|
|
|
|
|
|
|
proc process(r: RequestFence): Future[HttpResponseRef] {.async.} =
|
|
|
|
if r.isOk():
|
|
|
|
let request = r.get()
|
|
|
|
inc(count)
|
|
|
|
if count == ClientsCount:
|
|
|
|
eventWait.fire()
|
|
|
|
await eventContinue.wait()
|
|
|
|
return await request.respond(Http404, "", HttpTable.init())
|
|
|
|
else:
|
|
|
|
return dumbResponse()
|
|
|
|
|
|
|
|
let socketFlags = {ServerFlags.TcpNoDelay, ServerFlags.ReuseAddr}
|
|
|
|
let res = HttpServerRef.new(address, process,
|
|
|
|
socketFlags = socketFlags,
|
|
|
|
maxConnections = 100)
|
|
|
|
if res.isErr():
|
|
|
|
return false
|
|
|
|
|
|
|
|
let server = res.get()
|
|
|
|
server.start()
|
|
|
|
|
|
|
|
var clients: seq[Future[string]]
|
|
|
|
let message = "GET / HTTP/1.0\r\nHost: https://127.0.0.1:80\r\n\r\n"
|
|
|
|
for i in 0 ..< ClientsCount:
|
|
|
|
var clientFut = httpClient(address, message)
|
|
|
|
if clientFut.finished():
|
|
|
|
return false
|
|
|
|
clients.add(clientFut)
|
|
|
|
# Waiting for all clients to connect to the server
|
|
|
|
await eventWait.wait()
|
|
|
|
# Dropping
|
|
|
|
await server.closeWait()
|
|
|
|
# We are firing second event to unblock client loops, but this loops
|
|
|
|
# must be already cancelled.
|
|
|
|
eventContinue.fire()
|
|
|
|
# Now all clients should be dropped
|
|
|
|
discard await allFutures(clients).withTimeout(1.seconds)
|
|
|
|
for item in clients:
|
|
|
|
if item.read() != "":
|
|
|
|
return false
|
|
|
|
return true
|
|
|
|
|
|
|
|
check waitFor(testHTTPdrop(initTAddress("127.0.0.1:30080"))) == true
|
2021-02-10 10:37:56 +00:00
|
|
|
|
|
|
|
test "Content-Type multipart boundary test":
|
|
|
|
const AllowedCharacters = {
|
|
|
|
'a' .. 'z', 'A' .. 'Z', '0' .. '9',
|
|
|
|
'\'', '(', ')', '+', '_', ',', '-', '.' ,'/', ':', '=', '?'
|
|
|
|
}
|
|
|
|
|
|
|
|
const FailureVectors = [
|
|
|
|
"",
|
|
|
|
"multipart/byteranges; boundary=A",
|
|
|
|
"multipart/form-data;",
|
|
|
|
"multipart/form-data; boundary",
|
|
|
|
"multipart/form-data; boundary=",
|
|
|
|
"multipart/form-data; boundaryMore=A",
|
|
|
|
"multipart/form-data; charset=UTF-8; boundary",
|
|
|
|
"multipart/form-data; charset=UTF-8; boundary=",
|
|
|
|
"multipart/form-data; charset=UTF-8; boundary =",
|
|
|
|
"multipart/form-data; charset=UTF-8; boundary= ",
|
|
|
|
"multipart/form-data; charset=UTF-8; boundaryMore=",
|
|
|
|
"multipart/form-data; charset=UTF-8; boundaryMore=A",
|
|
|
|
"multipart/form-data; charset=UTF-8; boundaryMore=AAAAAAAAAAAAAAAAAAAA" &
|
|
|
|
"AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA"
|
|
|
|
]
|
|
|
|
|
|
|
|
const SuccessVectors = [
|
|
|
|
("multipart/form-data; boundary=A", "A"),
|
|
|
|
("multipart/form-data; charset=UTF-8; boundary=B", "B"),
|
|
|
|
("multipart/form-data; charset=UTF-8; boundary=--------------------" &
|
|
|
|
"--------------------------------------------------", "-----------" &
|
|
|
|
"-----------------------------------------------------------"),
|
|
|
|
("multipart/form-data; boundary=--------------------" &
|
|
|
|
"--------------------------------------------------", "-----------" &
|
|
|
|
"-----------------------------------------------------------"),
|
|
|
|
("multipart/form-data; boundary=--------------------" &
|
|
|
|
"--------------------------------------------------; charset=UTF-8",
|
|
|
|
"-----------------------------------------------------------------" &
|
|
|
|
"-----"),
|
2022-08-05 16:59:26 +00:00
|
|
|
("multipart/form-data; boundary=\"ABCDEFGHIJKLMNOPQRST" &
|
|
|
|
"UVWXYZabcdefghijklmnopqrstuvwxyz0123456789'()+_,-.\"; charset=UTF-8",
|
2021-02-10 10:37:56 +00:00
|
|
|
"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789'()" &
|
|
|
|
"+_,-."),
|
2022-08-05 16:59:26 +00:00
|
|
|
("multipart/form-data; boundary=\"ABCDEFGHIJKLMNOPQRST" &
|
|
|
|
"UVWXYZabcdefghijklmnopqrstuvwxyz0123456789'()+?=:/\"; charset=UTF-8",
|
2021-02-10 10:37:56 +00:00
|
|
|
"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789'()" &
|
|
|
|
"+?=:/"),
|
2022-08-05 16:59:26 +00:00
|
|
|
("multipart/form-data; charset=UTF-8; boundary=\"ABCDEFGHIJKLMNOPQRST" &
|
|
|
|
"UVWXYZabcdefghijklmnopqrstuvwxyz0123456789'()+_,-.\"",
|
2021-02-10 10:37:56 +00:00
|
|
|
"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789'()" &
|
|
|
|
"+_,-."),
|
2022-08-05 16:59:26 +00:00
|
|
|
("multipart/form-data; charset=UTF-8; boundary=\"ABCDEFGHIJKLMNOPQRST" &
|
|
|
|
"UVWXYZabcdefghijklmnopqrstuvwxyz0123456789'()+?=:/\"",
|
2021-02-10 10:37:56 +00:00
|
|
|
"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789'()" &
|
2022-08-05 16:59:26 +00:00
|
|
|
"+?=:/"),
|
|
|
|
("multipart/form-data; charset=UTF-8; boundary=0123456789ABCDEFGHIJKL" &
|
|
|
|
"MNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz+-",
|
|
|
|
"0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz+-"),
|
|
|
|
("multipart/form-data; boundary=0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZa" &
|
|
|
|
"bcdefghijklmnopqrstuvwxyz+-; charset=UTF-8",
|
|
|
|
"0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz+-")
|
2021-02-10 10:37:56 +00:00
|
|
|
]
|
2022-08-05 16:59:26 +00:00
|
|
|
proc performCheck(ch: openArray[string]): HttpResult[string] =
|
|
|
|
let cdata = ? getContentType(ch)
|
|
|
|
if cdata.mediaType != MediaType.init("multipart/form-data"):
|
|
|
|
return err("Invalid media type")
|
|
|
|
getMultipartBoundary(cdata)
|
2021-02-10 10:37:56 +00:00
|
|
|
|
|
|
|
for i in 0 ..< 256:
|
2022-08-05 16:59:26 +00:00
|
|
|
let boundary = "multipart/form-data; boundary=\"" & $char(i) & "\""
|
2021-02-10 10:37:56 +00:00
|
|
|
if char(i) in AllowedCharacters:
|
2022-08-05 16:59:26 +00:00
|
|
|
check performCheck([boundary]).isOk()
|
2021-02-10 10:37:56 +00:00
|
|
|
else:
|
2022-08-05 16:59:26 +00:00
|
|
|
check performCheck([boundary]).isErr()
|
2021-02-10 10:37:56 +00:00
|
|
|
|
|
|
|
check:
|
2022-08-05 16:59:26 +00:00
|
|
|
performCheck([]).isErr()
|
|
|
|
performCheck(["multipart/form-data; boundary=A",
|
|
|
|
"multipart/form-data; boundary=B"]).isErr()
|
2021-02-10 10:37:56 +00:00
|
|
|
for item in FailureVectors:
|
2022-08-05 16:59:26 +00:00
|
|
|
check performCheck([item]).isErr()
|
2021-02-10 10:37:56 +00:00
|
|
|
for item in SuccessVectors:
|
2022-08-05 16:59:26 +00:00
|
|
|
let res = performCheck([item[0]])
|
2021-02-10 10:37:56 +00:00
|
|
|
check:
|
|
|
|
res.isOk()
|
|
|
|
item[1] == res.get()
|
|
|
|
|
2021-02-10 11:11:27 +00:00
|
|
|
test "HttpTable integer parser test":
|
|
|
|
const TestVectors = [
|
|
|
|
("", 0'u64), ("0", 0'u64), ("-0", 0'u64), ("0-", 0'u64),
|
|
|
|
("01", 1'u64), ("001", 1'u64), ("0000000000001", 1'u64),
|
|
|
|
("18446744073709551615", 0xFFFF_FFFF_FFFF_FFFF'u64),
|
2021-03-05 22:22:32 +00:00
|
|
|
("18446744073709551616", 0'u64),
|
|
|
|
("99999999999999999999", 0'u64),
|
|
|
|
("999999999999999999999999999999999999", 0'u64),
|
2021-02-10 11:11:27 +00:00
|
|
|
("FFFFFFFFFFFFFFFF", 0'u64),
|
2021-03-05 22:22:32 +00:00
|
|
|
("0123456789ABCDEF", 0'u64)
|
2021-02-10 11:11:27 +00:00
|
|
|
]
|
|
|
|
for i in 0 ..< 256:
|
2021-03-05 22:22:32 +00:00
|
|
|
let res = Base10.decode(uint64, [char(i)])
|
2021-02-10 11:11:27 +00:00
|
|
|
if char(i) in {'0' .. '9'}:
|
2021-03-05 22:22:32 +00:00
|
|
|
check:
|
|
|
|
res.isOk()
|
|
|
|
res.get() == uint64(i - ord('0'))
|
2021-02-10 11:11:27 +00:00
|
|
|
else:
|
2021-03-05 22:22:32 +00:00
|
|
|
check res.isErr()
|
|
|
|
|
2021-02-10 11:11:27 +00:00
|
|
|
for item in TestVectors:
|
2021-03-05 22:22:32 +00:00
|
|
|
var ht = HttpTable.init([("test", item[0])])
|
|
|
|
let value = ht.getInt("test")
|
|
|
|
check value == item[1]
|
2021-02-10 11:11:27 +00:00
|
|
|
|
2021-02-18 12:08:21 +00:00
|
|
|
test "HttpTable behavior test":
|
|
|
|
var table1 = HttpTable.init()
|
|
|
|
var table2 = HttpTable.init([("Header1", "value1"), ("Header2", "value2")])
|
|
|
|
check:
|
|
|
|
table1.isEmpty() == true
|
|
|
|
table2.isEmpty() == false
|
|
|
|
|
|
|
|
table1.add("Header1", "value1")
|
|
|
|
table1.add("Header2", "value2")
|
|
|
|
table1.add("HEADER2", "VALUE3")
|
|
|
|
check:
|
|
|
|
table1.getList("HeAdEr2") == @["value2", "VALUE3"]
|
|
|
|
table1.getString("HeAdEr2") == "value2,VALUE3"
|
|
|
|
table2.getString("HEADER1") == "value1"
|
|
|
|
table1.count("HEADER2") == 2
|
|
|
|
table1.count("HEADER1") == 1
|
|
|
|
table1.getLastString("HEADER1") == "value1"
|
|
|
|
table1.getLastString("HEADER2") == "VALUE3"
|
|
|
|
"header1" in table1 == true
|
|
|
|
"HEADER1" in table1 == true
|
|
|
|
"header2" in table1 == true
|
|
|
|
"HEADER2" in table1 == true
|
|
|
|
"HEADER3" in table1 == false
|
|
|
|
|
|
|
|
var
|
|
|
|
data1: seq[tuple[key: string, value: string]]
|
|
|
|
data2: seq[tuple[key: string, value: seq[string]]]
|
|
|
|
for key, value in table1.stringItems(true):
|
|
|
|
data1.add((key, value))
|
|
|
|
for key, value in table1.items(true):
|
|
|
|
data2.add((key, value))
|
|
|
|
|
|
|
|
check:
|
|
|
|
data1 == @[("Header2", "value2"), ("Header2", "VALUE3"),
|
|
|
|
("Header1", "value1")]
|
|
|
|
data2 == @[("Header2", @["value2", "VALUE3"]),
|
|
|
|
("Header1", @["value1"])]
|
|
|
|
|
|
|
|
table1.set("header2", "value4")
|
|
|
|
check:
|
|
|
|
table1.getList("header2") == @["value4"]
|
|
|
|
table1.getString("header2") == "value4"
|
|
|
|
table1.count("header2") == 1
|
|
|
|
table1.getLastString("header2") == "value4"
|
|
|
|
|
2021-02-10 15:10:50 +00:00
|
|
|
test "getTransferEncoding() test":
|
|
|
|
var encodings = [
|
|
|
|
"chunked", "compress", "deflate", "gzip", "identity", "x-gzip"
|
|
|
|
]
|
|
|
|
|
|
|
|
const FlagsVectors = [
|
|
|
|
{
|
|
|
|
TransferEncodingFlags.Identity, TransferEncodingFlags.Chunked,
|
|
|
|
TransferEncodingFlags.Compress, TransferEncodingFlags.Deflate,
|
|
|
|
TransferEncodingFlags.Gzip
|
|
|
|
},
|
|
|
|
{
|
|
|
|
TransferEncodingFlags.Identity, TransferEncodingFlags.Compress,
|
|
|
|
TransferEncodingFlags.Deflate, TransferEncodingFlags.Gzip
|
|
|
|
},
|
|
|
|
{
|
|
|
|
TransferEncodingFlags.Identity, TransferEncodingFlags.Deflate,
|
|
|
|
TransferEncodingFlags.Gzip
|
|
|
|
},
|
|
|
|
{ TransferEncodingFlags.Identity, TransferEncodingFlags.Gzip },
|
|
|
|
{ TransferEncodingFlags.Identity, TransferEncodingFlags.Gzip },
|
|
|
|
{ TransferEncodingFlags.Gzip },
|
|
|
|
{ TransferEncodingFlags.Identity }
|
|
|
|
]
|
|
|
|
|
|
|
|
for i in 0 ..< 7:
|
|
|
|
var checkEncodings = @encodings
|
|
|
|
if i - 1 >= 0:
|
|
|
|
for k in 0 .. (i - 1):
|
|
|
|
checkEncodings.delete(0)
|
|
|
|
|
|
|
|
while nextPermutation(checkEncodings):
|
|
|
|
let res1 = getTransferEncoding([checkEncodings.join(", ")])
|
|
|
|
let res2 = getTransferEncoding([checkEncodings.join(",")])
|
|
|
|
let res3 = getTransferEncoding([checkEncodings.join("")])
|
|
|
|
let res4 = getTransferEncoding([checkEncodings.join(" ")])
|
|
|
|
let res5 = getTransferEncoding([checkEncodings.join(" , ")])
|
|
|
|
check:
|
|
|
|
res1.isOk()
|
|
|
|
res1.get() == FlagsVectors[i]
|
|
|
|
res2.isOk()
|
|
|
|
res2.get() == FlagsVectors[i]
|
|
|
|
res3.isErr()
|
2021-02-17 00:03:12 +00:00
|
|
|
res4.isErr()
|
2021-02-10 15:10:50 +00:00
|
|
|
res5.isOk()
|
|
|
|
res5.get() == FlagsVectors[i]
|
|
|
|
|
|
|
|
check:
|
|
|
|
getTransferEncoding([]).tryGet() == { TransferEncodingFlags.Identity }
|
|
|
|
getTransferEncoding(["", ""]).tryGet() ==
|
|
|
|
{ TransferEncodingFlags.Identity }
|
|
|
|
|
|
|
|
test "getContentEncoding() test":
|
|
|
|
var encodings = [
|
|
|
|
"br", "compress", "deflate", "gzip", "identity", "x-gzip"
|
|
|
|
]
|
|
|
|
|
|
|
|
const FlagsVectors = [
|
|
|
|
{
|
|
|
|
ContentEncodingFlags.Identity, ContentEncodingFlags.Br,
|
|
|
|
ContentEncodingFlags.Compress, ContentEncodingFlags.Deflate,
|
|
|
|
ContentEncodingFlags.Gzip
|
|
|
|
},
|
|
|
|
{
|
|
|
|
ContentEncodingFlags.Identity, ContentEncodingFlags.Compress,
|
|
|
|
ContentEncodingFlags.Deflate, ContentEncodingFlags.Gzip
|
|
|
|
},
|
|
|
|
{
|
|
|
|
ContentEncodingFlags.Identity, ContentEncodingFlags.Deflate,
|
|
|
|
ContentEncodingFlags.Gzip
|
|
|
|
},
|
|
|
|
{ ContentEncodingFlags.Identity, ContentEncodingFlags.Gzip },
|
|
|
|
{ ContentEncodingFlags.Identity, ContentEncodingFlags.Gzip },
|
|
|
|
{ ContentEncodingFlags.Gzip },
|
|
|
|
{ ContentEncodingFlags.Identity }
|
|
|
|
]
|
|
|
|
|
|
|
|
for i in 0 ..< 7:
|
|
|
|
var checkEncodings = @encodings
|
|
|
|
if i - 1 >= 0:
|
|
|
|
for k in 0 .. (i - 1):
|
|
|
|
checkEncodings.delete(0)
|
|
|
|
|
|
|
|
while nextPermutation(checkEncodings):
|
|
|
|
let res1 = getContentEncoding([checkEncodings.join(", ")])
|
|
|
|
let res2 = getContentEncoding([checkEncodings.join(",")])
|
|
|
|
let res3 = getContentEncoding([checkEncodings.join("")])
|
|
|
|
let res4 = getContentEncoding([checkEncodings.join(" ")])
|
|
|
|
let res5 = getContentEncoding([checkEncodings.join(" , ")])
|
|
|
|
check:
|
|
|
|
res1.isOk()
|
|
|
|
res1.get() == FlagsVectors[i]
|
|
|
|
res2.isOk()
|
|
|
|
res2.get() == FlagsVectors[i]
|
|
|
|
res3.isErr()
|
2021-02-17 00:03:12 +00:00
|
|
|
res4.isErr()
|
2021-02-10 15:10:50 +00:00
|
|
|
res5.isOk()
|
|
|
|
res5.get() == FlagsVectors[i]
|
|
|
|
|
|
|
|
check:
|
|
|
|
getContentEncoding([]).tryGet() == { ContentEncodingFlags.Identity }
|
|
|
|
getContentEncoding(["", ""]).tryGet() == { ContentEncodingFlags.Identity }
|
|
|
|
|
2021-05-17 19:39:24 +00:00
|
|
|
test "queryParams() test":
|
|
|
|
const Vectors = [
|
|
|
|
("id=1&id=2&id=3&id=4", {}, "id:1,id:2,id:3,id:4"),
|
|
|
|
("id=1,2,3,4", {}, "id:1,2,3,4"),
|
|
|
|
("id=1%2C2%2C3%2C4", {}, "id:1,2,3,4"),
|
|
|
|
("id=", {}, "id:"),
|
|
|
|
("id=&id=", {}, "id:,id:"),
|
|
|
|
("id=1&id=2&id=3&id=4", {QueryParamsFlag.CommaSeparatedArray},
|
|
|
|
"id:1,id:2,id:3,id:4"),
|
|
|
|
("id=1,2,3,4", {QueryParamsFlag.CommaSeparatedArray},
|
|
|
|
"id:1,id:2,id:3,id:4"),
|
|
|
|
("id=1%2C2%2C3%2C4", {QueryParamsFlag.CommaSeparatedArray},
|
|
|
|
"id:1,id:2,id:3,id:4"),
|
|
|
|
("id=", {QueryParamsFlag.CommaSeparatedArray}, "id:"),
|
|
|
|
("id=&id=", {QueryParamsFlag.CommaSeparatedArray}, "id:,id:"),
|
|
|
|
("id=,", {QueryParamsFlag.CommaSeparatedArray}, "id:,id:"),
|
|
|
|
("id=,,", {QueryParamsFlag.CommaSeparatedArray}, "id:,id:,id:"),
|
|
|
|
("id=1&id=2&id=3,4,5,6&id=7%2C8%2C9%2C10",
|
|
|
|
{QueryParamsFlag.CommaSeparatedArray},
|
|
|
|
"id:1,id:2,id:3,id:4,id:5,id:6,id:7,id:8,id:9,id:10")
|
|
|
|
]
|
|
|
|
|
|
|
|
proc toString(ht: HttpTable): string =
|
|
|
|
var res: seq[string]
|
|
|
|
for key, value in ht.items():
|
|
|
|
for item in value:
|
|
|
|
res.add(key & ":" & item)
|
|
|
|
res.join(",")
|
|
|
|
|
|
|
|
for vector in Vectors:
|
|
|
|
var table = HttpTable.init()
|
|
|
|
for key, value in queryParams(vector[0], vector[1]):
|
|
|
|
table.add(key, value)
|
|
|
|
check toString(table) == vector[2]
|
|
|
|
|
2022-08-06 10:53:40 +00:00
|
|
|
test "preferredContentType() test":
|
2022-01-21 15:16:59 +00:00
|
|
|
const
|
|
|
|
jsonMediaType = MediaType.init("application/json")
|
|
|
|
sszMediaType = MediaType.init("application/octet-stream")
|
|
|
|
plainTextMediaType = MediaType.init("text/plain")
|
|
|
|
imageMediaType = MediaType.init("image/jpg")
|
|
|
|
|
2021-07-28 14:08:38 +00:00
|
|
|
proc createRequest(acceptHeader: string): HttpRequestRef =
|
|
|
|
let headers = HttpTable.init([("accept", acceptHeader)])
|
|
|
|
HttpRequestRef(headers: headers)
|
|
|
|
|
|
|
|
proc createRequest(): HttpRequestRef =
|
|
|
|
HttpRequestRef(headers: HttpTable.init())
|
|
|
|
|
2022-01-21 15:16:59 +00:00
|
|
|
var singleHeader = @[
|
2021-07-28 14:08:38 +00:00
|
|
|
(
|
2022-01-21 15:16:59 +00:00
|
|
|
createRequest("application/json"),
|
2021-07-28 14:08:38 +00:00
|
|
|
@[
|
2022-01-21 15:16:59 +00:00
|
|
|
"application/json"
|
|
|
|
]
|
|
|
|
)
|
|
|
|
]
|
|
|
|
|
|
|
|
var complexHeaders = @[
|
|
|
|
(
|
|
|
|
createRequest(),
|
|
|
|
@[
|
|
|
|
"*/*",
|
|
|
|
"application/json",
|
2021-07-28 14:08:38 +00:00
|
|
|
"application/octet-stream",
|
2022-01-21 15:16:59 +00:00
|
|
|
"application/json",
|
2021-07-28 14:08:38 +00:00
|
|
|
"application/octet-stream",
|
|
|
|
"application/json",
|
2022-01-21 15:16:59 +00:00
|
|
|
"image/jpg"
|
2021-07-28 14:08:38 +00:00
|
|
|
]
|
|
|
|
),
|
|
|
|
(
|
|
|
|
createRequest(""),
|
|
|
|
@[
|
|
|
|
"*/*",
|
2022-01-21 15:16:59 +00:00
|
|
|
"application/json",
|
|
|
|
"application/octet-stream",
|
|
|
|
"application/json",
|
|
|
|
"application/octet-stream",
|
|
|
|
"application/json",
|
|
|
|
"image/jpg"
|
|
|
|
]
|
|
|
|
),
|
|
|
|
(
|
|
|
|
createRequest("application/json, application/octet-stream"),
|
|
|
|
@[
|
|
|
|
"application/json",
|
|
|
|
"application/json",
|
|
|
|
"application/octet-stream",
|
|
|
|
"application/json",
|
|
|
|
"application/octet-stream",
|
2021-07-28 14:08:38 +00:00
|
|
|
"application/json",
|
|
|
|
"application/json"
|
|
|
|
]
|
|
|
|
),
|
|
|
|
(
|
2022-01-21 15:16:59 +00:00
|
|
|
createRequest("application/octet-stream, application/json"),
|
|
|
|
@[
|
|
|
|
"application/octet-stream",
|
|
|
|
"application/json",
|
|
|
|
"application/octet-stream",
|
|
|
|
"application/json",
|
|
|
|
"application/octet-stream",
|
|
|
|
"application/json",
|
|
|
|
"application/json"
|
|
|
|
]
|
|
|
|
),
|
|
|
|
(
|
|
|
|
createRequest("application/json;q=0.9, application/octet-stream"),
|
|
|
|
@[
|
|
|
|
"application/octet-stream",
|
|
|
|
"application/json",
|
|
|
|
"application/octet-stream",
|
|
|
|
"application/octet-stream",
|
|
|
|
"application/octet-stream",
|
|
|
|
"application/octet-stream",
|
|
|
|
"application/octet-stream"
|
|
|
|
]
|
|
|
|
),
|
|
|
|
(
|
|
|
|
createRequest("application/json, application/octet-stream;q=0.9"),
|
|
|
|
@[
|
|
|
|
"application/json",
|
|
|
|
"application/json",
|
|
|
|
"application/octet-stream",
|
|
|
|
"application/json",
|
|
|
|
"application/json",
|
|
|
|
"application/json",
|
|
|
|
"application/json"
|
|
|
|
]
|
|
|
|
),
|
|
|
|
(
|
|
|
|
createRequest("application/json;q=0.9, application/octet-stream;q=0.8"),
|
|
|
|
@[
|
|
|
|
"application/json",
|
|
|
|
"application/json",
|
|
|
|
"application/octet-stream",
|
|
|
|
"application/json",
|
|
|
|
"application/json",
|
|
|
|
"application/json",
|
|
|
|
"application/json"
|
|
|
|
]
|
|
|
|
),
|
|
|
|
(
|
|
|
|
createRequest("application/json;q=0.8, application/octet-stream;q=0.9"),
|
|
|
|
@[
|
|
|
|
"application/octet-stream",
|
|
|
|
"application/json",
|
|
|
|
"application/octet-stream",
|
|
|
|
"application/octet-stream",
|
|
|
|
"application/octet-stream",
|
|
|
|
"application/octet-stream",
|
|
|
|
"application/octet-stream"
|
|
|
|
]
|
|
|
|
),
|
|
|
|
(
|
|
|
|
createRequest("text/plain, application/octet-stream, application/json"),
|
|
|
|
@[
|
|
|
|
"text/plain",
|
|
|
|
"application/json",
|
|
|
|
"application/octet-stream",
|
|
|
|
"application/json",
|
|
|
|
"application/octet-stream",
|
|
|
|
"application/json",
|
|
|
|
"application/json"
|
|
|
|
]
|
|
|
|
),
|
|
|
|
(
|
2022-08-06 10:53:40 +00:00
|
|
|
createRequest("text/plain, application/json;q=0.8, " &
|
|
|
|
"application/octet-stream;q=0.8"),
|
2022-01-21 15:16:59 +00:00
|
|
|
@[
|
|
|
|
"text/plain",
|
|
|
|
"application/json",
|
|
|
|
"application/octet-stream",
|
|
|
|
"application/json",
|
|
|
|
"application/octet-stream",
|
|
|
|
"text/plain",
|
|
|
|
"text/plain"
|
|
|
|
]
|
|
|
|
),
|
|
|
|
(
|
2022-08-06 10:53:40 +00:00
|
|
|
createRequest("text/plain, application/json;q=0.8, " &
|
|
|
|
"application/octet-stream;q=0.5"),
|
2022-01-21 15:16:59 +00:00
|
|
|
@[
|
|
|
|
"text/plain",
|
|
|
|
"application/json",
|
|
|
|
"application/octet-stream",
|
|
|
|
"application/json",
|
|
|
|
"application/json",
|
|
|
|
"text/plain",
|
|
|
|
"text/plain"
|
|
|
|
]
|
|
|
|
),
|
|
|
|
(
|
2022-08-06 10:53:40 +00:00
|
|
|
createRequest("text/plain;q=0.8, application/json, " &
|
|
|
|
"application/octet-stream;q=0.8"),
|
2022-01-21 15:16:59 +00:00
|
|
|
@[
|
|
|
|
"application/json",
|
|
|
|
"application/json",
|
|
|
|
"application/octet-stream",
|
|
|
|
"application/json",
|
|
|
|
"application/json",
|
|
|
|
"application/json",
|
|
|
|
"application/json"
|
|
|
|
]
|
|
|
|
),
|
|
|
|
(
|
2022-08-06 10:53:40 +00:00
|
|
|
createRequest("text/*, application/json;q=0.8, " &
|
|
|
|
"application/octet-stream;q=0.8"),
|
2022-01-21 15:16:59 +00:00
|
|
|
@[
|
|
|
|
"text/*",
|
|
|
|
"application/json",
|
|
|
|
"application/octet-stream",
|
|
|
|
"application/json",
|
|
|
|
"application/octet-stream",
|
|
|
|
"text/plain",
|
|
|
|
"text/plain"
|
|
|
|
]
|
|
|
|
),
|
|
|
|
(
|
2022-08-06 10:53:40 +00:00
|
|
|
createRequest("text/*, application/json;q=0.8, " &
|
|
|
|
"application/octet-stream;q=0.5"),
|
2022-01-21 15:16:59 +00:00
|
|
|
@[
|
|
|
|
"text/*",
|
|
|
|
"application/json",
|
|
|
|
"application/octet-stream",
|
|
|
|
"application/json",
|
|
|
|
"application/json",
|
|
|
|
"text/plain",
|
|
|
|
"text/plain"
|
|
|
|
]
|
|
|
|
),
|
2022-08-06 10:53:40 +00:00
|
|
|
(createRequest("image/jpg, text/plain, application/octet-stream, " &
|
|
|
|
"application/json"),
|
2022-01-21 15:16:59 +00:00
|
|
|
@[
|
|
|
|
"image/jpg",
|
|
|
|
"application/json",
|
|
|
|
"application/octet-stream",
|
|
|
|
"application/json",
|
|
|
|
"application/octet-stream",
|
|
|
|
"application/json",
|
|
|
|
"image/jpg"
|
|
|
|
]
|
|
|
|
),
|
2022-08-06 10:53:40 +00:00
|
|
|
(createRequest("image/jpg;q=1, text/plain;q=0.2, " &
|
|
|
|
"application/octet-stream;q=0.2, " &
|
|
|
|
"application/json;q=0.2"),
|
2022-01-21 15:16:59 +00:00
|
|
|
@[
|
|
|
|
"image/jpg",
|
|
|
|
"application/json",
|
|
|
|
"application/octet-stream",
|
|
|
|
"application/json",
|
|
|
|
"application/octet-stream",
|
|
|
|
"application/json",
|
|
|
|
"image/jpg"
|
|
|
|
]
|
|
|
|
),
|
|
|
|
(
|
2022-08-06 10:53:40 +00:00
|
|
|
createRequest("*/*, application/json;q=0.8, " &
|
|
|
|
"application/octet-stream;q=0.5"),
|
2021-07-28 14:08:38 +00:00
|
|
|
@[
|
|
|
|
"*/*",
|
2022-01-21 15:16:59 +00:00
|
|
|
"application/json",
|
|
|
|
"application/octet-stream",
|
|
|
|
"application/json",
|
|
|
|
"application/octet-stream",
|
|
|
|
"application/json",
|
|
|
|
"image/jpg"
|
|
|
|
]
|
|
|
|
),
|
|
|
|
(
|
|
|
|
createRequest("*/*"),
|
|
|
|
@[
|
2021-07-28 14:08:38 +00:00
|
|
|
"*/*",
|
|
|
|
"application/json",
|
2022-01-21 15:16:59 +00:00
|
|
|
"application/octet-stream",
|
|
|
|
"application/json",
|
|
|
|
"application/octet-stream",
|
|
|
|
"application/json",
|
|
|
|
"image/jpg"
|
|
|
|
]
|
|
|
|
),
|
|
|
|
(
|
|
|
|
createRequest("application/*"),
|
|
|
|
@[
|
|
|
|
"application/*",
|
|
|
|
"application/json",
|
|
|
|
"application/octet-stream",
|
|
|
|
"application/json",
|
|
|
|
"application/octet-stream",
|
|
|
|
"application/json",
|
2021-07-28 14:08:38 +00:00
|
|
|
"application/json"
|
|
|
|
]
|
|
|
|
)
|
|
|
|
]
|
|
|
|
|
2022-01-21 15:16:59 +00:00
|
|
|
for req in singleHeader:
|
|
|
|
check $req[0].preferredContentMediaType() == req[1][0]
|
|
|
|
let r0 = req[0].preferredContentType()
|
|
|
|
let r1 = req[0].preferredContentType(jsonMediaType)
|
|
|
|
let r2 = req[0].preferredContentType(sszMediaType)
|
|
|
|
let r3 = req[0].preferredContentType(jsonMediaType,
|
|
|
|
sszMediaType)
|
|
|
|
let r4 = req[0].preferredContentType(sszMediaType,
|
|
|
|
jsonMediaType)
|
|
|
|
let r5 = req[0].preferredContentType(jsonMediaType,
|
|
|
|
sszMediaType,
|
|
|
|
plainTextMediaType)
|
|
|
|
let r6 = req[0].preferredContentType(imageMediaType,
|
|
|
|
jsonMediaType,
|
|
|
|
sszMediaType,
|
|
|
|
plainTextMediaType)
|
|
|
|
check:
|
|
|
|
r0.isOk() == true
|
|
|
|
r1.isOk() == true
|
|
|
|
r2.isErr() == true
|
|
|
|
r3.isOk() == true
|
|
|
|
r4.isOk() == true
|
|
|
|
r5.isOk() == true
|
|
|
|
r6.isOk() == true
|
|
|
|
r0.get() == MediaType.init(req[1][0])
|
|
|
|
r1.get() == MediaType.init(req[1][0])
|
|
|
|
r3.get() == MediaType.init(req[1][0])
|
|
|
|
r4.get() == MediaType.init(req[1][0])
|
|
|
|
r5.get() == MediaType.init(req[1][0])
|
|
|
|
r6.get() == MediaType.init(req[1][0])
|
|
|
|
|
|
|
|
for req in complexHeaders:
|
2021-07-28 14:08:38 +00:00
|
|
|
let r0 = req[0].preferredContentType()
|
2022-01-21 15:16:59 +00:00
|
|
|
let r1 = req[0].preferredContentType(jsonMediaType)
|
|
|
|
let r2 = req[0].preferredContentType(sszMediaType)
|
|
|
|
let r3 = req[0].preferredContentType(jsonMediaType,
|
|
|
|
sszMediaType)
|
|
|
|
let r4 = req[0].preferredContentType(sszMediaType,
|
|
|
|
jsonMediaType)
|
|
|
|
let r5 = req[0].preferredContentType(jsonMediaType,
|
|
|
|
sszMediaType,
|
|
|
|
plainTextMediaType)
|
|
|
|
let r6 = req[0].preferredContentType(imageMediaType,
|
|
|
|
jsonMediaType,
|
|
|
|
sszMediaType,
|
|
|
|
plainTextMediaType)
|
2021-07-28 14:08:38 +00:00
|
|
|
check:
|
|
|
|
r0.isOk() == true
|
|
|
|
r1.isOk() == true
|
|
|
|
r2.isOk() == true
|
2022-01-21 15:16:59 +00:00
|
|
|
r3.isOk() == true
|
|
|
|
r4.isOk() == true
|
|
|
|
r5.isOk() == true
|
|
|
|
r6.isOk() == true
|
|
|
|
r0.get() == MediaType.init(req[1][0])
|
|
|
|
r1.get() == MediaType.init(req[1][1])
|
|
|
|
r2.get() == MediaType.init(req[1][2])
|
|
|
|
r3.get() == MediaType.init(req[1][3])
|
|
|
|
r4.get() == MediaType.init(req[1][4])
|
|
|
|
r5.get() == MediaType.init(req[1][5])
|
|
|
|
r6.get() == MediaType.init(req[1][6])
|
2021-07-28 14:08:38 +00:00
|
|
|
|
2021-09-14 17:32:58 +00:00
|
|
|
test "SSE server-side events stream test":
|
|
|
|
proc testPostMultipart2(address: TransportAddress): Future[bool] {.async.} =
|
|
|
|
var serverRes = false
|
|
|
|
proc process(r: RequestFence): Future[HttpResponseRef] {.
|
|
|
|
async.} =
|
|
|
|
if r.isOk():
|
|
|
|
let request = r.get()
|
|
|
|
let response = request.getResponse()
|
|
|
|
await response.prepareSSE()
|
|
|
|
await response.send("event: event1\r\ndata: data1\r\n\r\n")
|
|
|
|
await response.send("event: event2\r\ndata: data2\r\n\r\n")
|
|
|
|
await response.sendEvent("event3", "data3")
|
|
|
|
await response.sendEvent("event4", "data4")
|
|
|
|
await response.send("data: data5\r\n\r\n")
|
|
|
|
await response.sendEvent("", "data6")
|
|
|
|
await response.finish()
|
|
|
|
serverRes = true
|
|
|
|
return response
|
|
|
|
else:
|
|
|
|
serverRes = false
|
|
|
|
return dumbResponse()
|
|
|
|
|
|
|
|
let socketFlags = {ServerFlags.TcpNoDelay, ServerFlags.ReuseAddr}
|
|
|
|
let res = HttpServerRef.new(address, process,
|
|
|
|
socketFlags = socketFlags)
|
|
|
|
if res.isErr():
|
|
|
|
return false
|
|
|
|
|
|
|
|
let server = res.get()
|
|
|
|
server.start()
|
|
|
|
|
|
|
|
let message =
|
|
|
|
"GET / HTTP/1.1\r\n" &
|
|
|
|
"Host: 127.0.0.1:30080\r\n" &
|
|
|
|
"Accept: text/event-stream\r\n" &
|
|
|
|
"\r\n"
|
|
|
|
|
|
|
|
let data = await httpClient(address, message)
|
|
|
|
let expect = "event: event1\r\ndata: data1\r\n\r\n" &
|
|
|
|
"event: event2\r\ndata: data2\r\n\r\n" &
|
|
|
|
"event: event3\r\ndata: data3\r\n\r\n" &
|
|
|
|
"event: event4\r\ndata: data4\r\n\r\n" &
|
|
|
|
"data: data5\r\n\r\n" &
|
|
|
|
"data: data6\r\n\r\n"
|
|
|
|
await server.stop()
|
|
|
|
await server.closeWait()
|
|
|
|
return serverRes and (data.find(expect) >= 0)
|
|
|
|
|
|
|
|
check waitFor(testPostMultipart2(initTAddress("127.0.0.1:30080"))) == true
|
|
|
|
|
|
|
|
|
2021-02-02 22:33:14 +00:00
|
|
|
test "Leaks test":
|
|
|
|
check:
|
|
|
|
getTracker("async.stream.reader").isLeaked() == false
|
|
|
|
getTracker("async.stream.writer").isLeaked() == false
|
|
|
|
getTracker("stream.server").isLeaked() == false
|
|
|
|
getTracker("stream.transport").isLeaked() == false
|