2024-06-28 10:34:57 +00:00
|
|
|
{.push raises: [].}
|
2024-04-25 13:09:52 +00:00
|
|
|
|
2024-07-09 11:14:28 +00:00
|
|
|
import std/strformat, results, chronicles, uri, json_serialization, presto/route
|
2024-04-25 13:09:52 +00:00
|
|
|
import
|
|
|
|
../../../waku_core,
|
|
|
|
../../../waku_store_legacy/common,
|
|
|
|
../../../waku_store_legacy/self_req_handler,
|
|
|
|
../../../waku_node,
|
|
|
|
../../../node/peer_manager,
|
|
|
|
../../../common/paging,
|
|
|
|
../../handlers,
|
|
|
|
../responses,
|
|
|
|
../serdes,
|
|
|
|
./types
|
|
|
|
|
|
|
|
export types
|
|
|
|
|
|
|
|
logScope:
|
|
|
|
topics = "waku node rest legacy store_api"
|
|
|
|
|
|
|
|
const futTimeout* = 5.seconds # Max time to wait for futures
|
|
|
|
|
|
|
|
const NoPeerNoDiscError* =
|
|
|
|
RestApiResponse.preconditionFailed("No suitable service peer & no discovery method")
|
|
|
|
|
|
|
|
# Queries the store-node with the query parameters and
|
|
|
|
# returns a RestApiResponse that is sent back to the api client.
|
|
|
|
proc performHistoryQuery(
|
|
|
|
selfNode: WakuNode, histQuery: HistoryQuery, storePeer: RemotePeerInfo
|
|
|
|
): Future[RestApiResponse] {.async.} =
|
|
|
|
let queryFut = selfNode.query(histQuery, storePeer)
|
|
|
|
if not await queryFut.withTimeout(futTimeout):
|
|
|
|
const msg = "No history response received (timeout)"
|
|
|
|
error msg
|
|
|
|
return RestApiResponse.internalServerError(msg)
|
|
|
|
|
|
|
|
let res = queryFut.read()
|
|
|
|
if res.isErr():
|
|
|
|
const msg = "Error occurred in queryFut.read()"
|
|
|
|
error msg, error = res.error
|
|
|
|
return RestApiResponse.internalServerError(fmt("{msg} [{res.error}]"))
|
|
|
|
|
|
|
|
let storeResp = res.value.toStoreResponseRest()
|
|
|
|
let resp = RestApiResponse.jsonResponse(storeResp, status = Http200)
|
|
|
|
if resp.isErr():
|
|
|
|
const msg = "Error building the json respose"
|
|
|
|
error msg, error = resp.error
|
|
|
|
return RestApiResponse.internalServerError(fmt("{msg} [{resp.error}]"))
|
|
|
|
|
|
|
|
return resp.get()
|
|
|
|
|
|
|
|
# Converts a string time representation into an Option[Timestamp].
|
|
|
|
# Only positive time is considered a valid Timestamp in the request
|
|
|
|
proc parseTime(input: Option[string]): Result[Option[Timestamp], string] =
|
|
|
|
if input.isSome() and input.get() != "":
|
|
|
|
try:
|
|
|
|
let time = parseInt(input.get())
|
|
|
|
if time > 0:
|
|
|
|
return ok(some(Timestamp(time)))
|
|
|
|
except ValueError:
|
|
|
|
return err("Problem parsing time [" & getCurrentExceptionMsg() & "]")
|
|
|
|
|
|
|
|
return ok(none(Timestamp))
|
|
|
|
|
|
|
|
# Generates a history query cursor as per the given params
|
|
|
|
proc parseCursor(
|
|
|
|
parsedPubsubTopic: Option[string],
|
|
|
|
senderTime: Option[string],
|
|
|
|
storeTime: Option[string],
|
|
|
|
digest: Option[string],
|
|
|
|
): Result[Option[HistoryCursor], string] =
|
|
|
|
# Parse sender time
|
|
|
|
let parsedSenderTime = parseTime(senderTime)
|
|
|
|
if not parsedSenderTime.isOk():
|
|
|
|
return err(parsedSenderTime.error)
|
|
|
|
|
|
|
|
# Parse store time
|
|
|
|
let parsedStoreTime = parseTime(storeTime)
|
|
|
|
if not parsedStoreTime.isOk():
|
|
|
|
return err(parsedStoreTime.error)
|
|
|
|
|
|
|
|
# Parse message digest
|
|
|
|
let parsedMsgDigest = parseMsgDigest(digest)
|
|
|
|
if not parsedMsgDigest.isOk():
|
|
|
|
return err(parsedMsgDigest.error)
|
|
|
|
|
|
|
|
# Parse cursor information
|
|
|
|
if parsedPubsubTopic.isSome() and parsedSenderTime.value.isSome() and
|
|
|
|
parsedStoreTime.value.isSome() and parsedMsgDigest.value.isSome():
|
|
|
|
return ok(
|
|
|
|
some(
|
|
|
|
HistoryCursor(
|
|
|
|
pubsubTopic: parsedPubsubTopic.get(),
|
|
|
|
senderTime: parsedSenderTime.value.get(),
|
|
|
|
storeTime: parsedStoreTime.value.get(),
|
|
|
|
digest: parsedMsgDigest.value.get(),
|
|
|
|
)
|
|
|
|
)
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
return ok(none(HistoryCursor))
|
|
|
|
|
|
|
|
# Creates a HistoryQuery from the given params
|
|
|
|
proc createHistoryQuery(
|
|
|
|
pubsubTopic: Option[string],
|
|
|
|
contentTopics: Option[string],
|
|
|
|
senderTime: Option[string],
|
|
|
|
storeTime: Option[string],
|
|
|
|
digest: Option[string],
|
|
|
|
startTime: Option[string],
|
|
|
|
endTime: Option[string],
|
|
|
|
pageSize: Option[string],
|
|
|
|
direction: Option[string],
|
|
|
|
): Result[HistoryQuery, string] =
|
|
|
|
# Parse pubsubTopic parameter
|
|
|
|
var parsedPubsubTopic = none(string)
|
|
|
|
if pubsubTopic.isSome():
|
|
|
|
let decodedPubsubTopic = decodeUrl(pubsubTopic.get())
|
|
|
|
if decodedPubsubTopic != "":
|
|
|
|
parsedPubsubTopic = some(decodedPubsubTopic)
|
|
|
|
|
|
|
|
# Parse the content topics
|
|
|
|
var parsedContentTopics = newSeq[ContentTopic](0)
|
|
|
|
if contentTopics.isSome():
|
|
|
|
let ctList = decodeUrl(contentTopics.get())
|
|
|
|
if ctList != "":
|
|
|
|
for ct in ctList.split(','):
|
|
|
|
parsedContentTopics.add(ct)
|
|
|
|
|
|
|
|
# Parse cursor information
|
|
|
|
let parsedCursor = ?parseCursor(parsedPubsubTopic, senderTime, storeTime, digest)
|
|
|
|
|
|
|
|
# Parse page size field
|
|
|
|
var parsedPagedSize = DefaultPageSize
|
|
|
|
if pageSize.isSome() and pageSize.get() != "":
|
|
|
|
try:
|
|
|
|
parsedPagedSize = uint64(parseInt(pageSize.get()))
|
|
|
|
except CatchableError:
|
|
|
|
return err("Problem parsing page size [" & getCurrentExceptionMsg() & "]")
|
|
|
|
|
|
|
|
# Parse start time
|
|
|
|
let parsedStartTime = ?parseTime(startTime)
|
|
|
|
|
|
|
|
# Parse end time
|
|
|
|
let parsedEndTime = ?parseTime(endTime)
|
|
|
|
|
|
|
|
# Parse ascending field
|
|
|
|
var parsedDirection = default()
|
|
|
|
if direction.isSome() and direction.get() != "":
|
|
|
|
parsedDirection = direction.get().into()
|
|
|
|
|
|
|
|
return ok(
|
|
|
|
HistoryQuery(
|
|
|
|
pubsubTopic: parsedPubsubTopic,
|
|
|
|
contentTopics: parsedContentTopics,
|
|
|
|
startTime: parsedStartTime,
|
|
|
|
endTime: parsedEndTime,
|
|
|
|
direction: parsedDirection,
|
|
|
|
pageSize: parsedPagedSize,
|
|
|
|
cursor: parsedCursor,
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
# Simple type conversion. The "Option[Result[string, cstring]]"
|
|
|
|
# type is used by the nim-presto library.
|
|
|
|
proc toOpt(self: Option[Result[string, cstring]]): Option[string] =
|
|
|
|
if not self.isSome() or self.get().value == "":
|
|
|
|
return none(string)
|
|
|
|
if self.isSome() and self.get().value != "":
|
|
|
|
return some(self.get().value)
|
|
|
|
|
|
|
|
proc retrieveMsgsFromSelfNode(
|
|
|
|
self: WakuNode, histQuery: HistoryQuery
|
|
|
|
): Future[RestApiResponse] {.async.} =
|
|
|
|
## Performs a "store" request to the local node (self node.)
|
|
|
|
## Notice that this doesn't follow the regular store libp2p channel because a node
|
|
|
|
## it is not allowed to libp2p-dial a node to itself, by default.
|
|
|
|
##
|
|
|
|
|
|
|
|
let selfResp = (await self.wakuLegacyStore.handleSelfStoreRequest(histQuery)).valueOr:
|
|
|
|
return RestApiResponse.internalServerError($error)
|
|
|
|
|
|
|
|
let storeResp = selfResp.toStoreResponseRest()
|
|
|
|
let resp = RestApiResponse.jsonResponse(storeResp, status = Http200).valueOr:
|
|
|
|
const msg = "Error building the json respose"
|
2024-09-09 08:45:14 +00:00
|
|
|
let e = $error
|
|
|
|
error msg, error = e
|
|
|
|
return RestApiResponse.internalServerError(fmt("{msg} [{e}]"))
|
2024-04-25 13:09:52 +00:00
|
|
|
|
|
|
|
return resp
|
|
|
|
|
|
|
|
# Subscribes the rest handler to attend "/store/v1/messages" requests
|
|
|
|
proc installStoreApiHandlers*(
|
|
|
|
router: var RestRouter,
|
|
|
|
node: WakuNode,
|
|
|
|
discHandler: Option[DiscoveryHandler] = none(DiscoveryHandler),
|
|
|
|
) =
|
|
|
|
# Handles the store-query request according to the passed parameters
|
|
|
|
router.api(MethodGet, "/store/v1/messages") do(
|
|
|
|
peerAddr: Option[string],
|
|
|
|
pubsubTopic: Option[string],
|
|
|
|
contentTopics: Option[string],
|
|
|
|
senderTime: Option[string],
|
|
|
|
storeTime: Option[string],
|
|
|
|
digest: Option[string],
|
|
|
|
startTime: Option[string],
|
|
|
|
endTime: Option[string],
|
|
|
|
pageSize: Option[string],
|
|
|
|
ascending: Option[string]
|
|
|
|
) -> RestApiResponse:
|
|
|
|
debug "REST-GET /store/v1/messages ", peer_addr = $peerAddr
|
|
|
|
|
|
|
|
# All the GET parameters are URL-encoded (https://en.wikipedia.org/wiki/URL_encoding)
|
|
|
|
# Example:
|
|
|
|
# /store/v1/messages?peerAddr=%2Fip4%2F127.0.0.1%2Ftcp%2F60001%2Fp2p%2F16Uiu2HAmVFXtAfSj4EiR7mL2KvL4EE2wztuQgUSBoj2Jx2KeXFLN\&pubsubTopic=my-waku-topic
|
|
|
|
|
|
|
|
# Parse the rest of the parameters and create a HistoryQuery
|
|
|
|
let histQuery = createHistoryQuery(
|
|
|
|
pubsubTopic.toOpt(),
|
|
|
|
contentTopics.toOpt(),
|
|
|
|
senderTime.toOpt(),
|
|
|
|
storeTime.toOpt(),
|
|
|
|
digest.toOpt(),
|
|
|
|
startTime.toOpt(),
|
|
|
|
endTime.toOpt(),
|
|
|
|
pageSize.toOpt(),
|
|
|
|
ascending.toOpt(),
|
|
|
|
)
|
|
|
|
|
|
|
|
if not histQuery.isOk():
|
|
|
|
return RestApiResponse.badRequest(histQuery.error)
|
|
|
|
|
|
|
|
if peerAddr.isNone() and not node.wakuLegacyStore.isNil():
|
|
|
|
## The user didn't specify a peer address and self-node is configured as a store node.
|
|
|
|
## In this case we assume that the user is willing to retrieve the messages stored by
|
|
|
|
## the local/self store node.
|
|
|
|
return await node.retrieveMsgsFromSelfNode(histQuery.get())
|
|
|
|
|
|
|
|
# Parse the peer address parameter
|
|
|
|
let parsedPeerAddr = parseUrlPeerAddr(peerAddr.toOpt()).valueOr:
|
|
|
|
return RestApiResponse.badRequest(error)
|
|
|
|
|
|
|
|
let peerAddr = parsedPeerAddr.valueOr:
|
2024-05-09 18:07:49 +00:00
|
|
|
node.peerManager.selectPeer(WakuLegacyStoreCodec).valueOr:
|
2024-04-25 13:09:52 +00:00
|
|
|
let handler = discHandler.valueOr:
|
|
|
|
return NoPeerNoDiscError
|
|
|
|
|
|
|
|
let peerOp = (await handler()).valueOr:
|
|
|
|
return RestApiResponse.internalServerError($error)
|
|
|
|
|
|
|
|
peerOp.valueOr:
|
|
|
|
return RestApiResponse.preconditionFailed(
|
|
|
|
"No suitable service peer & none discovered"
|
|
|
|
)
|
|
|
|
|
|
|
|
return await node.performHistoryQuery(histQuery.value, peerAddr)
|