mirror of https://github.com/waku-org/nwaku.git
210 lines
6.1 KiB
Nim
210 lines
6.1 KiB
Nim
when (NimMajor, NimMinor) < (1, 4):
|
|
{.push raises: [Defect].}
|
|
else:
|
|
{.push raises: [].}
|
|
|
|
import std/net
|
|
import
|
|
stew/results,
|
|
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()
|