nim-chronos/chronos/apps/http/httpdebug.nim
Eugene Kabanov 6c2ea67512
Unroll defers and remove breaks. (#440)
* Unpack `finally/defer` blocks and introduce explicit cleaning of objects.
Add request query to debug information.

* Unroll one more loop to avoid `break`.
Add test for query debug string.

* Fix cancellation behavior.

* Address review comments.
2023-08-09 10:57:49 +03:00

130 lines
3.8 KiB
Nim

#
# Chronos HTTP/S server implementation
# (c) Copyright 2021-Present
# Status Research & Development GmbH
#
# Licensed under either of
# Apache License, version 2.0, (LICENSE-APACHEv2)
# MIT license (LICENSE-MIT)
import std/tables
import stew/results
import ../../timer
import httpserver, shttpserver
from httpclient import HttpClientScheme
from httpcommon import HttpState
from ../../osdefs import SocketHandle
from ../../transports/common import TransportAddress, ServerFlags
export HttpClientScheme, SocketHandle, TransportAddress, ServerFlags, HttpState
{.push raises: [].}
type
ConnectionType* {.pure.} = enum
NonSecure, Secure
ConnectionState* {.pure.} = enum
Accepted, Alive, Closing, Closed
ServerConnectionInfo* = object
handle*: SocketHandle
connectionType*: ConnectionType
connectionState*: ConnectionState
query*: Opt[string]
remoteAddress*: Opt[TransportAddress]
localAddress*: Opt[TransportAddress]
acceptMoment*: Moment
createMoment*: Opt[Moment]
ServerInfo* = object
connectionType*: ConnectionType
address*: TransportAddress
state*: HttpServerState
maxConnections*: int
backlogSize*: int
baseUri*: Uri
serverIdent*: string
flags*: set[HttpServerFlags]
socketFlags*: set[ServerFlags]
headersTimeout*: Duration
bufferSize*: int
maxHeadersSize*: int
maxRequestBodySize*: int
proc getConnectionType*(
server: HttpServerRef | SecureHttpServerRef): ConnectionType =
when server is SecureHttpServerRef:
ConnectionType.Secure
else:
if HttpServerFlags.Secure in server.flags:
ConnectionType.Secure
else:
ConnectionType.NonSecure
proc getServerInfo*(server: HttpServerRef|SecureHttpServerRef): ServerInfo =
ServerInfo(
connectionType: server.getConnectionType(),
address: server.address,
state: server.state(),
maxConnections: server.maxConnections,
backlogSize: server.backlogSize,
baseUri: server.baseUri,
serverIdent: server.serverIdent,
flags: server.flags,
socketFlags: server.socketFlags,
headersTimeout: server.headersTimeout,
bufferSize: server.bufferSize,
maxHeadersSize: server.maxHeadersSize,
maxRequestBodySize: server.maxRequestBodySize
)
proc getConnectionState*(holder: HttpConnectionHolderRef): ConnectionState =
if not(isNil(holder.connection)):
case holder.connection.state
of HttpState.Alive: ConnectionState.Alive
of HttpState.Closing: ConnectionState.Closing
of HttpState.Closed: ConnectionState.Closed
else:
ConnectionState.Accepted
proc getQueryString*(holder: HttpConnectionHolderRef): Opt[string] =
if not(isNil(holder.connection)):
holder.connection.currentRawQuery
else:
Opt.none(string)
proc init*(t: typedesc[ServerConnectionInfo],
holder: HttpConnectionHolderRef): ServerConnectionInfo =
let
localAddress =
try:
Opt.some(holder.transp.localAddress())
except CatchableError:
Opt.none(TransportAddress)
remoteAddress =
try:
Opt.some(holder.transp.remoteAddress())
except CatchableError:
Opt.none(TransportAddress)
queryString = holder.getQueryString()
ServerConnectionInfo(
handle: SocketHandle(holder.transp.fd),
connectionType: holder.server.getConnectionType(),
connectionState: holder.getConnectionState(),
remoteAddress: remoteAddress,
localAddress: localAddress,
acceptMoment: holder.acceptMoment,
query: queryString,
createMoment:
if not(isNil(holder.connection)):
Opt.some(holder.connection.createMoment)
else:
Opt.none(Moment)
)
proc getConnections*(server: HttpServerRef): seq[ServerConnectionInfo] =
var res: seq[ServerConnectionInfo]
for holder in server.connections.values():
res.add(ServerConnectionInfo.init(holder))
res