2024-06-28 16:04:57 +05:30
|
|
|
{.push raises: [].}
|
2023-09-22 13:46:55 +02:00
|
|
|
|
|
|
|
import
|
|
|
|
std/strformat,
|
|
|
|
stew/byteutils,
|
|
|
|
chronicles,
|
|
|
|
json_serialization,
|
|
|
|
json_serialization/std/options,
|
|
|
|
presto/route,
|
|
|
|
presto/common
|
|
|
|
|
|
|
|
import
|
2024-07-09 13:14:28 +02:00
|
|
|
waku/node/peer_manager,
|
|
|
|
waku/waku_lightpush/common,
|
2023-09-22 09:36:46 -04:00
|
|
|
../../../waku_node,
|
2023-10-27 15:43:54 -04:00
|
|
|
../../handlers,
|
2023-09-22 13:46:55 +02:00
|
|
|
../serdes,
|
|
|
|
../responses,
|
2023-11-14 16:59:53 +01:00
|
|
|
../rest_serdes,
|
2023-09-22 13:46:55 +02:00
|
|
|
./types
|
|
|
|
|
|
|
|
export types
|
|
|
|
|
|
|
|
logScope:
|
|
|
|
topics = "waku node rest lightpush api"
|
|
|
|
|
|
|
|
const futTimeoutForPushRequestProcessing* = 5.seconds
|
|
|
|
|
2024-03-16 00:08:47 +01:00
|
|
|
const NoPeerNoDiscoError =
|
|
|
|
RestApiResponse.serviceUnavailable("No suitable service peer & no discovery method")
|
2023-10-27 15:43:54 -04:00
|
|
|
|
2024-03-16 00:08:47 +01:00
|
|
|
const NoPeerNoneFoundError =
|
|
|
|
RestApiResponse.serviceUnavailable("No suitable service peer & none discovered")
|
2023-10-27 15:43:54 -04:00
|
|
|
|
2024-04-26 12:42:47 +02:00
|
|
|
proc useSelfHostedLightPush(node: WakuNode): bool =
|
|
|
|
return node.wakuLightPush != nil and node.wakuLightPushClient == nil
|
|
|
|
|
2023-09-22 13:46:55 +02:00
|
|
|
#### Request handlers
|
|
|
|
|
|
|
|
const ROUTE_LIGHTPUSH* = "/lightpush/v1/message"
|
|
|
|
|
2023-10-27 15:43:54 -04:00
|
|
|
proc installLightPushRequestHandler*(
|
2024-03-16 00:08:47 +01:00
|
|
|
router: var RestRouter,
|
|
|
|
node: WakuNode,
|
|
|
|
discHandler: Option[DiscoveryHandler] = none(DiscoveryHandler),
|
|
|
|
) =
|
|
|
|
router.api(MethodPost, ROUTE_LIGHTPUSH) do(
|
|
|
|
contentBody: Option[ContentBody]
|
|
|
|
) -> RestApiResponse:
|
2023-09-22 13:46:55 +02:00
|
|
|
## Send a request to push a waku message
|
|
|
|
debug "post", ROUTE_LIGHTPUSH, contentBody
|
|
|
|
|
|
|
|
let decodedBody = decodeRequestBody[PushRequest](contentBody)
|
|
|
|
|
|
|
|
if decodedBody.isErr():
|
|
|
|
return decodedBody.error()
|
|
|
|
|
|
|
|
let req: PushRequest = decodedBody.value()
|
2023-11-14 16:59:53 +01:00
|
|
|
|
2023-10-27 15:43:54 -04:00
|
|
|
let msg = req.message.toWakuMessage().valueOr:
|
2023-11-01 11:30:53 +01:00
|
|
|
return RestApiResponse.badRequest("Invalid message: " & $error)
|
2023-09-22 13:46:55 +02:00
|
|
|
|
2024-04-26 12:42:47 +02:00
|
|
|
var peer = RemotePeerInfo.init($node.switch.peerInfo.peerId)
|
|
|
|
if useSelfHostedLightPush(node):
|
|
|
|
discard
|
|
|
|
else:
|
|
|
|
peer = node.peerManager.selectPeer(WakuLightPushCodec).valueOr:
|
|
|
|
let handler = discHandler.valueOr:
|
|
|
|
return NoPeerNoDiscoError
|
2023-10-27 15:43:54 -04:00
|
|
|
|
2024-04-26 12:42:47 +02:00
|
|
|
let peerOp = (await handler()).valueOr:
|
|
|
|
return RestApiResponse.internalServerError("No value in peerOp: " & $error)
|
2023-10-27 15:43:54 -04:00
|
|
|
|
2024-04-26 12:42:47 +02:00
|
|
|
peerOp.valueOr:
|
|
|
|
return NoPeerNoneFoundError
|
2023-09-22 13:46:55 +02:00
|
|
|
|
2023-10-27 15:43:54 -04:00
|
|
|
let subFut = node.lightpushPublish(req.pubsubTopic, msg, peer)
|
2023-09-22 13:46:55 +02:00
|
|
|
|
|
|
|
if not await subFut.withTimeout(futTimeoutForPushRequestProcessing):
|
|
|
|
error "Failed to request a message push due to timeout!"
|
|
|
|
return RestApiResponse.serviceUnavailable("Push request timed out")
|
|
|
|
|
|
|
|
if subFut.value().isErr():
|
2024-04-15 15:28:35 +02:00
|
|
|
if subFut.value().error == TooManyRequestsMessage:
|
|
|
|
return RestApiResponse.tooManyRequests("Request rate limmit reached")
|
|
|
|
|
2023-10-27 15:43:54 -04:00
|
|
|
return RestApiResponse.serviceUnavailable(
|
|
|
|
fmt("Failed to request a message push: {subFut.value().error}")
|
|
|
|
)
|
2023-09-22 13:46:55 +02:00
|
|
|
|
|
|
|
return RestApiResponse.ok()
|