nim-websock/ws/http/server.nim

219 lines
6.2 KiB
Nim

{.push raises: [Defect].}
import std/uri
import pkg/[
chronos,
chronicles,
httputils]
import ./common
type
HttpAsyncCallback* = proc (request: HttpRequest):
Future[void] {.closure, gcsafe, raises: [Defect].}
HttpServer* = ref object of StreamServer
handler*: HttpAsyncCallback
TlsHttpServer* = ref object of HttpServer
tlsFlags*: set[TLSFlags]
tlsPrivateKey*: TLSPrivateKey
tlsCertificate*: TLSCertificate
minVersion*: TLSVersion
maxVersion*: TLSVersion
proc validateRequest(
stream: AsyncStreamWriter,
header: HttpRequestHeader): Future[ReqStatus] {.async.} =
## Validate Request
##
if header.meth notin {MethodGet}:
trace "GET method is only allowed", address = stream.tsource.remoteAddress()
await stream.sendError(Http405, version = header.version)
return ReqStatus.Error
var hlen = header.contentLength()
if hlen < 0 or hlen > MaxHttpRequestSize:
trace "Invalid header length", address = stream.tsource.remoteAddress()
await stream.sendError(Http413, version = header.version)
return ReqStatus.Error
return ReqStatus.Success
proc handleRequest(
server: HttpServer,
stream: AsyncStream) {.async.} =
## Process transport data to the HTTP server
##
var buffer = newSeq[byte](MaxHttpHeadersSize)
let remoteAddr = stream.reader.tsource.remoteAddress()
trace "Received connection", address = $remoteAddr
try:
let hlenfut = stream.reader.readUntil(
addr buffer[0], MaxHttpHeadersSize, sep = HeaderSep)
let ores = await withTimeout(hlenfut, HttpHeadersTimeout)
if not ores:
# Timeout
trace "Timeout expired while receiving headers", address = $remoteAddr
await stream.writer.sendError(Http408, version = HttpVersion11)
return
let hlen = hlenfut.read()
buffer.setLen(hlen)
let requestData = buffer.parseRequest()
if requestData.failed():
# Header could not be parsed
trace "Malformed header received", address = $remoteAddr
await stream.writer.sendError(Http400, version = HttpVersion11)
return
var vres = await stream.writer.validateRequest(requestData)
let hdrs =
block:
var res = HttpTable.init()
for key, value in requestData.headers():
res.add(key, value)
res
if vres == ReqStatus.ErrorFailure:
trace "Remote peer disconnected", address = $remoteAddr
return
trace "Received valid HTTP request", address = $remoteAddr
# Call the user's handler.
if server.handler != nil:
await server.handler(
HttpRequest(
headers: hdrs,
stream: stream,
uri: requestData.uri().parseUri()))
except TransportLimitError:
# size of headers exceeds `MaxHttpHeadersSize`
trace "maximum size of headers limit reached", address = $remoteAddr
await stream.writer.sendError(Http413, version = HttpVersion11)
except TransportIncompleteError:
# remote peer disconnected
trace "Remote peer disconnected", address = $remoteAddr
except TransportOsError as exc:
trace "Problems with networking", address = $remoteAddr, error = exc.msg
except CatchableError as exc:
trace "Unknown exception", address = $remoteAddr, error = exc.msg
finally:
await stream.closeWait()
proc handleConnCb(
server: StreamServer,
transp: StreamTransport) {.async.} =
let stream = AsyncStream(
reader: newAsyncStreamReader(transp),
writer: newAsyncStreamWriter(transp))
let httpServer = HttpServer(server)
await httpServer.handleRequest(stream)
proc handleTlsConnCb(
server: StreamServer,
transp: StreamTransport) {.async.} =
let tlsHttpServer = TlsHttpServer(server)
let stream = newTLSServerAsyncStream(
newAsyncStreamReader(transp),
newAsyncStreamWriter(transp),
tlsHttpServer.tlsPrivateKey,
tlsHttpServer.tlsCertificate,
minVersion = tlsHttpServer.minVersion,
maxVersion = tlsHttpServer.maxVersion,
flags = tlsHttpServer.tlsFlags)
await HttpServer(tlsHttpServer)
.handleRequest(AsyncStream(
reader: stream.reader,
writer: stream.writer))
proc create*(
_: typedesc[HttpServer],
address: TransportAddress,
handler: HttpAsyncCallback = nil,
flags: set[ServerFlags] = {}): HttpServer
{.raises: [Defect, CatchableError].} = # TODO: remove CatchableError
## Make a new HTTP Server
##
var server = HttpServer(handler: handler)
server = HttpServer(
createStreamServer(
address,
handleConnCb,
flags,
child = StreamServer(server)))
trace "Created HTTP Server", host = $address
return server
proc create*(
_: typedesc[HttpServer],
host: string,
port: Port,
handler: HttpAsyncCallback = nil,
flags: set[ServerFlags] = {}): HttpServer
{.raises: [Defect, CatchableError].} = # TODO: remove CatchableError
## Make a new HTTP Server
##
return HttpServer.create(initTAddress(host, port), handler, flags)
proc create*(
_: typedesc[TlsHttpServer],
address: TransportAddress,
tlsPrivateKey: TLSPrivateKey,
tlsCertificate: TLSCertificate,
handler: HttpAsyncCallback = nil,
flags: set[ServerFlags] = {},
tlsFlags: set[TLSFlags] = {},
tlsMinVersion = TLSVersion.TLS12,
tlsMaxVersion = TLSVersion.TLS12): TlsHttpServer
{.raises: [Defect, CatchableError].} = # TODO: remove CatchableError
var server = TlsHttpServer(
handler: handler,
tlsPrivateKey: tlsPrivateKey,
tlsCertificate: tlsCertificate,
minVersion: tlsMinVersion,
maxVersion: tlsMaxVersion)
server = TlsHttpServer(
createStreamServer(
address,
handleTlsConnCb,
flags,
child = StreamServer(server)))
trace "Created TLS HTTP Server", host = $address
return server
proc create*(
_: typedesc[TlsHttpServer],
host: string,
port: Port,
tlsPrivateKey: TLSPrivateKey,
tlsCertificate: TLSCertificate,
handler: HttpAsyncCallback = nil,
flags: set[ServerFlags] = {},
tlsFlags: set[TLSFlags] = {},
tlsMinVersion = TLSVersion.TLS12,
tlsMaxVersion = TLSVersion.TLS12): TlsHttpServer
{.raises: [Defect, CatchableError].} = # TODO: remove CatchableError
TlsHttpServer.create(
address = initTAddress(host, port),
handler = handler,
tlsPrivateKey = tlsPrivateKey,
tlsCertificate = tlsCertificate,
flags = flags,
tlsFlags = tlsFlags)