2022-11-04 10:52:27 +01:00
|
|
|
when (NimMajor, NimMinor) < (1, 4):
|
|
|
|
{.push raises: [Defect].}
|
|
|
|
else:
|
|
|
|
{.push raises: [].}
|
2022-08-30 15:57:45 +02:00
|
|
|
|
|
|
|
import
|
|
|
|
std/[tables, sequtils],
|
|
|
|
stew/results,
|
|
|
|
chronicles,
|
|
|
|
chronos,
|
|
|
|
libp2p/protocols/pubsub
|
|
|
|
import
|
2023-04-19 13:29:23 +02:00
|
|
|
../waku_core
|
2022-08-30 15:57:45 +02:00
|
|
|
|
2023-04-18 15:22:10 +02:00
|
|
|
logScope:
|
2022-11-03 16:36:24 +01:00
|
|
|
topics = "waku node message_cache"
|
2022-08-30 15:57:45 +02:00
|
|
|
|
|
|
|
const DefaultMessageCacheCapacity*: uint = 30 # Max number of messages cached per topic @TODO make this configurable
|
|
|
|
|
|
|
|
|
|
|
|
type MessageCacheResult*[T] = Result[T, cstring]
|
|
|
|
|
|
|
|
type MessageCache*[K] = ref object
|
|
|
|
capacity: uint
|
|
|
|
table: Table[K, seq[WakuMessage]]
|
|
|
|
|
|
|
|
func init*[K](T: type MessageCache[K], capacity=DefaultMessageCacheCapacity): T =
|
|
|
|
MessageCache[K](
|
|
|
|
capacity: capacity,
|
2023-04-18 15:22:10 +02:00
|
|
|
table: initTable[K, seq[WakuMessage]]()
|
2022-08-30 15:57:45 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
proc isSubscribed*[K](t: MessageCache[K], topic: K): bool =
|
|
|
|
t.table.hasKey(topic)
|
|
|
|
|
|
|
|
proc subscribe*[K](t: MessageCache[K], topic: K) =
|
|
|
|
if t.isSubscribed(topic):
|
|
|
|
return
|
|
|
|
t.table[topic] = @[]
|
|
|
|
|
2023-04-18 15:22:10 +02:00
|
|
|
proc unsubscribe*[K](t: MessageCache[K], topic: K) =
|
2022-08-30 15:57:45 +02:00
|
|
|
if not t.isSubscribed(topic):
|
|
|
|
return
|
|
|
|
t.table.del(topic)
|
|
|
|
|
2023-09-14 21:28:57 +02:00
|
|
|
proc unsubscribeAll*[K](t: MessageCache[K]) =
|
|
|
|
t.table.clear()
|
2022-08-30 15:57:45 +02:00
|
|
|
|
|
|
|
proc addMessage*[K](t: MessageCache, topic: K, msg: WakuMessage) =
|
|
|
|
if not t.isSubscribed(topic):
|
|
|
|
return
|
|
|
|
|
|
|
|
# Make a copy of msgs for this topic to modify
|
|
|
|
var messages = t.table.getOrDefault(topic, @[])
|
|
|
|
|
|
|
|
if messages.len >= t.capacity.int:
|
2023-02-14 15:07:26 +02:00
|
|
|
trace "Topic cache capacity reached", topic=topic
|
2022-08-30 15:57:45 +02:00
|
|
|
# Message cache on this topic exceeds maximum. Delete oldest.
|
2023-04-18 15:22:10 +02:00
|
|
|
# TODO: this may become a bottle neck if called as the norm rather than
|
2022-08-30 15:57:45 +02:00
|
|
|
# exception when adding messages. Performance profile needed.
|
|
|
|
messages.delete(0,0)
|
2023-04-18 15:22:10 +02:00
|
|
|
|
2022-08-30 15:57:45 +02:00
|
|
|
messages.add(msg)
|
|
|
|
|
|
|
|
# Replace indexed entry with copy
|
|
|
|
t.table[topic] = messages
|
|
|
|
|
|
|
|
proc clearMessages*[K](t: MessageCache[K], topic: K) =
|
|
|
|
if not t.isSubscribed(topic):
|
|
|
|
return
|
|
|
|
t.table[topic] = @[]
|
|
|
|
|
|
|
|
proc getMessages*[K](t: MessageCache[K], topic: K, clear=false): MessageCacheResult[seq[WakuMessage]] =
|
|
|
|
if not t.isSubscribed(topic):
|
|
|
|
return err("Not subscribed to topic")
|
|
|
|
|
|
|
|
let messages = t.table.getOrDefault(topic, @[])
|
|
|
|
if clear:
|
|
|
|
t.clearMessages(topic)
|
|
|
|
|
|
|
|
ok(messages)
|