mirror of https://github.com/waku-org/nwaku.git
279 lines
7.3 KiB
Nim
279 lines
7.3 KiB
Nim
{.push raises: [].}
|
|
|
|
import
|
|
std/[sequtils, sugar, algorithm, options],
|
|
stew/results,
|
|
chronicles,
|
|
chronos,
|
|
libp2p/protocols/pubsub
|
|
import ../waku_core
|
|
|
|
logScope:
|
|
topics = "waku node message_cache"
|
|
|
|
const DefaultMessageCacheCapacity: int = 50
|
|
|
|
type MessageCache* = ref object
|
|
pubsubTopics: seq[PubsubTopic]
|
|
contentTopics: seq[ContentTopic]
|
|
|
|
pubsubIndex: seq[tuple[pubsubIdx: int, msgIdx: int]]
|
|
contentIndex: seq[tuple[contentIdx: int, msgIdx: int]]
|
|
|
|
messages: seq[WakuMessage]
|
|
|
|
capacity: int
|
|
|
|
func `$`*(self: MessageCache): string =
|
|
"Messages: " & $self.messages.len & " \nPubsubTopics: " & $self.pubsubTopics &
|
|
" \nContentTopics: " & $self.contentTopics & " \nPubsubIndex: " & $self.pubsubIndex &
|
|
" \nContentIndex: " & $self.contentIndex
|
|
|
|
func init*(T: type MessageCache, capacity = DefaultMessageCacheCapacity): T =
|
|
MessageCache(capacity: capacity)
|
|
|
|
proc messagesCount*(self: MessageCache): int =
|
|
self.messages.len
|
|
|
|
proc pubsubTopicCount*(self: MessageCache): int =
|
|
self.pubsubTopics.len
|
|
|
|
proc contentTopicCount*(self: MessageCache): int =
|
|
self.contentTopics.len
|
|
|
|
proc pubsubSearch(self: MessageCache, pubsubTopic: PubsubTopic): Option[int] =
|
|
# Return some with the index if found none otherwise.
|
|
|
|
for i, topic in self.pubsubTopics:
|
|
if topic == pubsubTopic:
|
|
return some(i)
|
|
|
|
return none(int)
|
|
|
|
proc contentSearch(self: MessageCache, contentTopic: ContentTopic): Option[int] =
|
|
# Return some with the index if found none otherwise.
|
|
|
|
for i, topic in self.contentTopics:
|
|
if topic == contentTopic:
|
|
return some(i)
|
|
|
|
return none(int)
|
|
|
|
proc isPubsubSubscribed*(self: MessageCache, pubsubTopic: PubsubTopic): bool =
|
|
self.pubsubSearch(pubsubTopic).isSome()
|
|
|
|
proc isContentSubscribed*(self: MessageCache, contentTopic: ContentTopic): bool =
|
|
self.contentSearch(contentTopic).isSome()
|
|
|
|
proc pubsubSubscribe*(self: MessageCache, pubsubTopic: PubsubTopic) =
|
|
if self.pubsubSearch(pubsubTopic).isNone():
|
|
self.pubsubTopics.add(pubsubTopic)
|
|
|
|
proc contentSubscribe*(self: MessageCache, contentTopic: ContentTopic) =
|
|
if self.contentSearch(contentTopic).isNone():
|
|
self.contentTopics.add(contentTopic)
|
|
|
|
proc removeMessage(self: MessageCache, idx: int) =
|
|
# get last index because del() is a swap
|
|
let lastIndex = self.messages.high
|
|
|
|
self.messages.del(idx)
|
|
|
|
# update indices
|
|
var j = self.pubsubIndex.high
|
|
while j > -1:
|
|
let (pId, mId) = self.pubsubIndex[j]
|
|
|
|
if mId == idx:
|
|
self.pubsubIndex.del(j)
|
|
elif mId == lastIndex:
|
|
self.pubsubIndex[j] = (pId, idx)
|
|
|
|
dec(j)
|
|
|
|
j = self.contentIndex.high
|
|
while j > -1:
|
|
let (cId, mId) = self.contentIndex[j]
|
|
|
|
if mId == idx:
|
|
self.contentIndex.del(j)
|
|
elif mId == lastIndex:
|
|
self.contentIndex[j] = (cId, idx)
|
|
|
|
dec(j)
|
|
|
|
proc pubsubUnsubscribe*(self: MessageCache, pubsubTopic: PubsubTopic) =
|
|
let pubsubIdxOp = self.pubsubSearch(pubsubTopic)
|
|
|
|
let pubsubIdx =
|
|
if pubsubIdxOp.isSome():
|
|
pubsubIdxOp.get()
|
|
else:
|
|
return
|
|
|
|
let lastIndex = self.pubsubTopics.high
|
|
self.pubsubTopics.del(pubsubIdx)
|
|
|
|
var msgIndices = newSeq[int](0)
|
|
|
|
var j = self.pubsubIndex.high
|
|
while j > -1:
|
|
let (pId, mId) = self.pubsubIndex[j]
|
|
|
|
if pId == pubsubIdx:
|
|
# remove index for this topic
|
|
self.pubsubIndex.del(j)
|
|
msgIndices.add(mId)
|
|
elif pId == lastIndex:
|
|
# swap the index because pubsubTopics.del() is a swap
|
|
self.pubsubIndex[j] = (pubsubIdx, mId)
|
|
|
|
dec(j)
|
|
|
|
# check if messages on this pubsub topic are indexed by any content topic, if not remove them.
|
|
for mId in msgIndices.sorted(SortOrder.Descending):
|
|
if not self.contentIndex.anyIt(it.msgIdx == mId):
|
|
self.removeMessage(mId)
|
|
|
|
proc contentUnsubscribe*(self: MessageCache, contentTopic: ContentTopic) =
|
|
let contentIdxOP = self.contentSearch(contentTopic)
|
|
|
|
let contentIdx =
|
|
if contentIdxOP.isSome():
|
|
contentIdxOP.get()
|
|
else:
|
|
return
|
|
|
|
let lastIndex = self.contentTopics.high
|
|
self.contentTopics.del(contentIdx)
|
|
|
|
var msgIndices = newSeq[int](0)
|
|
|
|
var j = self.contentIndex.high
|
|
while j > -1:
|
|
let (cId, mId) = self.contentIndex[j]
|
|
|
|
if cId == contentIdx:
|
|
# remove indices for this topic
|
|
self.contentIndex.del(j)
|
|
msgIndices.add(mId)
|
|
elif cId == lastIndex:
|
|
# swap the indices because contentTopics.del() is a swap
|
|
self.contentIndex[j] = (contentIdx, mId)
|
|
|
|
dec(j)
|
|
|
|
# check if messages on this content topic are indexed by any pubsub topic, if not remove them.
|
|
for mId in msgIndices.sorted(SortOrder.Descending):
|
|
if not self.pubsubIndex.anyIt(it.msgIdx == mId):
|
|
self.removeMessage(mId)
|
|
|
|
proc reset*(self: MessageCache) =
|
|
self.messages.setLen(0)
|
|
self.pubsubTopics.setLen(0)
|
|
self.contentTopics.setLen(0)
|
|
self.pubsubIndex.setLen(0)
|
|
self.contentIndex.setLen(0)
|
|
|
|
proc addMessage*(self: MessageCache, pubsubTopic: PubsubTopic, msg: WakuMessage) =
|
|
## Idempotent message addition.
|
|
|
|
var oldestTime = int64.high
|
|
var oldestMsg = int.high
|
|
for i, message in self.messages.reversed:
|
|
if message == msg:
|
|
return
|
|
|
|
if message.timestamp < oldestTime:
|
|
oldestTime = message.timestamp
|
|
oldestMsg = i
|
|
|
|
# reverse index
|
|
oldestMsg = self.messages.high - oldestMsg
|
|
|
|
var pubsubIdxOp = self.pubsubSearch(pubsubTopic)
|
|
var contentIdxOp = self.contentSearch(msg.contentTopic)
|
|
|
|
if pubsubIdxOp.isNone() and contentIdxOp.isNone():
|
|
return
|
|
|
|
let pubsubIdx =
|
|
if pubsubIdxOp.isNone():
|
|
self.pubsubTopics.add(pubsubTopic)
|
|
self.pubsubTopics.high
|
|
else:
|
|
pubsubIdxOp.get()
|
|
|
|
let contentIdx =
|
|
if contentIdxOp.isNone():
|
|
self.contentTopics.add(msg.contentTopic)
|
|
self.contentTopics.high
|
|
else:
|
|
contentIdxOp.get()
|
|
|
|
# add the message, make space if needed
|
|
if self.messages.len >= self.capacity:
|
|
self.removeMessage(oldestMsg)
|
|
|
|
let msgIdx = self.messages.len
|
|
self.messages.add(msg)
|
|
|
|
self.pubsubIndex.add((pubsubIdx, msgIdx))
|
|
self.contentIndex.add((contentIdx, msgIdx))
|
|
|
|
proc getMessages*(
|
|
self: MessageCache, pubsubTopic: PubsubTopic, clear = false
|
|
): Result[seq[WakuMessage], string] =
|
|
## Return all messages on this pubsub topic
|
|
|
|
if self.pubsubTopics.len == 0:
|
|
return err("not subscribed to any pubsub topics")
|
|
|
|
let pubsubIdxOp = self.pubsubSearch(pubsubTopic)
|
|
let pubsubIdx =
|
|
if pubsubIdxOp.isNone:
|
|
return err("not subscribed to this pubsub topic")
|
|
else:
|
|
pubsubIdxOp.get()
|
|
|
|
let msgIndices = collect:
|
|
for (pId, mId) in self.pubsubIndex:
|
|
if pId == pubsubIdx:
|
|
mId
|
|
|
|
let messages = msgIndices.mapIt(self.messages[it])
|
|
|
|
if clear:
|
|
for idx in msgIndices.reversed:
|
|
self.removeMessage(idx)
|
|
|
|
return ok(messages)
|
|
|
|
proc getAutoMessages*(
|
|
self: MessageCache, contentTopic: ContentTopic, clear = false
|
|
): Result[seq[WakuMessage], string] =
|
|
## Return all messages on this content topic
|
|
|
|
if self.contentTopics.len == 0:
|
|
return err("not subscribed to any content topics")
|
|
|
|
let contentIdxOp = self.contentSearch(contentTopic)
|
|
let contentIdx =
|
|
if contentIdxOp.isNone():
|
|
return err("not subscribed to this content topic")
|
|
else:
|
|
contentIdxOp.get()
|
|
|
|
let msgIndices = collect:
|
|
for (cId, mId) in self.contentIndex:
|
|
if cId == contentIdx:
|
|
mId
|
|
|
|
let messages = msgIndices.mapIt(self.messages[it])
|
|
|
|
if clear:
|
|
for idx in msgIndices.sorted(SortOrder.Descending):
|
|
self.removeMessage(idx)
|
|
|
|
return ok(messages)
|