nwaku/waku/waku_api/rest/server.nim

210 lines
6.1 KiB
Nim

when (NimMajor, NimMinor) < (1, 4):
{.push raises: [Defect].}
else:
{.push raises: [].}
import
stew/results,
stew/shims/net,
chronicles,
chronos,
chronos/apps/http/httpserver,
presto,
presto/middleware,
presto/servercommon
import ./origin_handler
type
RestServerResult*[T] = Result[T, string]
WakuRestServer* = object of RootObj
router*: RestRouter
httpServer*: HttpServerRef
WakuRestServerRef* = ref WakuRestServer
### Configuration
type RestServerConf* = object
cacheSize*: Natural
## \
## The maximum number of recently accessed states that are kept in \
## memory. Speeds up requests obtaining information for consecutive
## slots or epochs.
cacheTtl*: Natural
## \
## The number of seconds to keep recently accessed states in memory
requestTimeout*: Natural
## \
## The number of seconds to wait until complete REST request will be received
maxRequestBodySize*: Natural
## \
## Maximum size of REST request body (kilobytes)
maxRequestHeadersSize*: Natural
## \
## Maximum size of REST request headers (kilobytes)
proc default*(T: type RestServerConf): T =
RestServerConf(
cacheSize: 3,
cacheTtl: 60,
requestTimeout: 0,
maxRequestBodySize: 16_384,
maxRequestHeadersSize: 64,
)
### Initialization
proc new*(
t: typedesc[WakuRestServerRef],
router: RestRouter,
address: TransportAddress,
serverIdent: string = PrestoIdent,
serverFlags = {HttpServerFlags.NotifyDisconnect},
socketFlags: set[ServerFlags] = {ReuseAddr},
serverUri = Uri(),
maxConnections: int = -1,
backlogSize: int = DefaultBacklogSize,
bufferSize: int = 4096,
httpHeadersTimeout = 10.seconds,
maxHeadersSize: int = 8192,
maxRequestBodySize: int = 1_048_576,
requestErrorHandler: RestRequestErrorHandler = nil,
dualstack = DualStackType.Auto,
allowedOrigin: Option[string] = none(string),
): RestServerResult[WakuRestServerRef] =
var server = WakuRestServerRef(router: router)
let restMiddleware = RestServerMiddlewareRef.new(
router = server.router, errorHandler = requestErrorHandler
)
let originHandlerMiddleware = OriginHandlerMiddlewareRef.new(allowedOrigin)
let middlewares = [originHandlerMiddleware, restMiddleware]
## This must be empty and needed only to confirm original initialization requirements of
## the RestHttpServer now combining old and new middleware approach.
proc defaultProcessCallback(
rf: RequestFence
): Future[HttpResponseRef] {.async: (raises: [CancelledError]).} =
discard
let sres = HttpServerRef.new(
address,
defaultProcessCallback,
serverFlags,
socketFlags,
serverUri,
serverIdent,
maxConnections,
bufferSize,
backlogSize,
httpHeadersTimeout,
maxHeadersSize,
maxRequestBodySize,
dualstack = dualstack,
middlewares = middlewares,
)
if sres.isOk():
server.httpServer = sres.get()
ok(server)
else:
err(sres.error)
proc getRouter(): RestRouter =
# TODO: Review this `validate` method. Check in nim-presto what is this used for.
proc validate(pattern: string, value: string): int =
## This is rough validation procedure which should be simple and fast,
## because it will be used for query routing.
if pattern.startsWith("{") and pattern.endsWith("}"): 0 else: 1
# disable allowed origin handling by presto, we add our own handling as middleware
RestRouter.init(validate, allowedOrigin = none(string))
proc init*(
T: type WakuRestServerRef,
ip: IpAddress,
port: Port,
allowedOrigin = none(string),
conf = RestServerConf.default(),
requestErrorHandler: RestRequestErrorHandler = nil,
): RestServerResult[T] =
let address = initTAddress(ip, port)
let serverFlags =
{HttpServerFlags.QueryCommaSeparatedArray, HttpServerFlags.NotifyDisconnect}
let
headersTimeout =
if conf.requestTimeout == 0:
chronos.InfiniteDuration
else:
seconds(int64(conf.requestTimeout))
maxHeadersSize = conf.maxRequestHeadersSize * 1024
maxRequestBodySize = conf.maxRequestBodySize * 1024
let router = getRouter()
try:
return WakuRestServerRef.new(
router,
address,
serverFlags = serverFlags,
httpHeadersTimeout = headersTimeout,
maxHeadersSize = maxHeadersSize,
maxRequestBodySize = maxRequestBodySize,
requestErrorHandler = requestErrorHandler,
allowedOrigin = allowedOrigin,
)
except CatchableError:
return err(getCurrentExceptionMsg())
proc newRestHttpServer*(
ip: IpAddress,
port: Port,
allowedOrigin = none(string),
conf = RestServerConf.default(),
requestErrorHandler: RestRequestErrorHandler = nil,
): RestServerResult[WakuRestServerRef] =
WakuRestServerRef.init(ip, port, allowedOrigin, conf, requestErrorHandler)
proc localAddress*(rs: WakuRestServerRef): TransportAddress =
## Returns `rs` bound local socket address.
rs.httpServer.instance.localAddress()
proc state*(rs: WakuRestServerRef): RestServerState =
## Returns current REST server's state.
case rs.httpServer.state
of HttpServerState.ServerClosed: RestServerState.Closed
of HttpServerState.ServerStopped: RestServerState.Stopped
of HttpServerState.ServerRunning: RestServerState.Running
proc start*(rs: WakuRestServerRef) =
## Starts REST server.
rs.httpServer.start()
notice "REST service started", address = $rs.localAddress()
proc stop*(rs: WakuRestServerRef) {.async: (raises: []).} =
## Stop REST server from accepting new connections.
await rs.httpServer.stop()
notice "REST service stopped", address = $rs.localAddress()
proc drop*(rs: WakuRestServerRef): Future[void] {.async: (raw: true, raises: []).} =
## Drop all pending connections.
rs.httpServer.drop()
proc closeWait*(rs: WakuRestServerRef) {.async: (raises: []).} =
## Stop REST server and drop all the pending connections.
await rs.httpServer.closeWait()
notice "REST service closed", address = $rs.localAddress()
proc join*(
rs: WakuRestServerRef
): Future[void] {.async: (raw: true, raises: [CancelledError]).} =
## Wait until REST server will not be closed.
rs.httpServer.join()