mirror of https://github.com/waku-org/nwaku.git
163 lines
4.2 KiB
Nim
163 lines
4.2 KiB
Nim
|
## TimedMap
|
||
|
## ===========
|
||
|
## Inspired by nim-libp2p's TimedCache class. This is using the same approach to prune
|
||
|
## untouched items from the map where the set timeout duration is reached.
|
||
|
## But unlike TimedCache this TimedMap is capable to hold and return any type of value for a key.
|
||
|
##
|
||
|
## - `mgetOrPut` proc is similar to std/tables, but will renew the timeout for the key.
|
||
|
## - For non-renewal check use `contains` proc.
|
||
|
## - `expire` proc will remove all items that have expired.
|
||
|
##
|
||
|
## Choose your initial timeout for your needs to control the size of the map.
|
||
|
|
||
|
{.push raises: [].}
|
||
|
|
||
|
import std/[hashes, sets]
|
||
|
import chronos/timer, results
|
||
|
import libp2p/utility
|
||
|
|
||
|
export results
|
||
|
|
||
|
type
|
||
|
TimedEntry[K, V] = ref object of RootObj
|
||
|
key: K
|
||
|
value: V
|
||
|
addedAt: Moment
|
||
|
expiresAt: Moment
|
||
|
next, prev: TimedEntry[K, V]
|
||
|
|
||
|
TimedMap*[K, V] = object of RootObj
|
||
|
head, tail: TimedEntry[K, V] # nim linked list doesn't allow inserting at pos
|
||
|
entries: HashSet[TimedEntry[K, V]]
|
||
|
timeout: Duration
|
||
|
|
||
|
func `==`*[K, V](a, b: TimedEntry[K, V]): bool =
|
||
|
if isNil(a) == isNil(b):
|
||
|
isNil(a) or a.key == b.key
|
||
|
else:
|
||
|
false
|
||
|
|
||
|
func hash*(a: TimedEntry): Hash =
|
||
|
if isNil(a):
|
||
|
default(Hash)
|
||
|
else:
|
||
|
hash(a[].key)
|
||
|
|
||
|
func `$`*[T](a: T): string =
|
||
|
if isNil(a):
|
||
|
"nil"
|
||
|
|
||
|
return $a
|
||
|
|
||
|
func `$`*[K, V](a: TimedEntry[K, V]): string =
|
||
|
if isNil(a):
|
||
|
return "nil"
|
||
|
|
||
|
return
|
||
|
"TimedEntry: key:" & $a.key & ", val:" & $a.value & ", addedAt:" & $a.addedAt &
|
||
|
", expiresAt:" & $a.expiresAt
|
||
|
|
||
|
func expire*(t: var TimedMap, now: Moment = Moment.now()) =
|
||
|
while t.head != nil and t.head.expiresAt <= now:
|
||
|
t.entries.excl(t.head)
|
||
|
t.head.prev = nil
|
||
|
t.head = t.head.next
|
||
|
if t.head == nil:
|
||
|
t.tail = nil
|
||
|
|
||
|
func del[K, V](t: var TimedMap[K, V], key: K): Opt[TimedEntry[K, V]] =
|
||
|
# Removes existing key from cache, returning the previous item if present
|
||
|
let tmp = TimedEntry[K, V](key: key)
|
||
|
if tmp in t.entries:
|
||
|
let item =
|
||
|
try:
|
||
|
t.entries[tmp] # use the shared instance in the set
|
||
|
except KeyError:
|
||
|
raiseAssert "just checked"
|
||
|
t.entries.excl(item)
|
||
|
|
||
|
if t.head == item:
|
||
|
t.head = item.next
|
||
|
if t.tail == item:
|
||
|
t.tail = item.prev
|
||
|
|
||
|
if item.next != nil:
|
||
|
item.next.prev = item.prev
|
||
|
if item.prev != nil:
|
||
|
item.prev.next = item.next
|
||
|
Opt.some(item)
|
||
|
else:
|
||
|
Opt.none(TimedEntry[K, V])
|
||
|
|
||
|
func remove*[K, V](t: var TimedMap[K, V], key: K): Opt[V] =
|
||
|
# Removes existing key from cache, returning the previous value if present
|
||
|
# public version of del without exporting TimedEntry
|
||
|
let deleted = t.del(key)
|
||
|
if deleted.isSome():
|
||
|
Opt.some(deleted[].value)
|
||
|
else:
|
||
|
Opt.none(V)
|
||
|
|
||
|
proc mgetOrPut*[K, V](t: var TimedMap[K, V], k: K, v: V, now = Moment.now()): var V =
|
||
|
# Puts k in cache, returning true if the item was already present and false
|
||
|
# otherwise. If the item was already present, its expiry timer will be
|
||
|
# refreshed.
|
||
|
t.expire(now)
|
||
|
|
||
|
let
|
||
|
previous = t.del(k) # Refresh existing item
|
||
|
addedAt =
|
||
|
if previous.isSome():
|
||
|
previous[].addedAt
|
||
|
else:
|
||
|
now
|
||
|
value =
|
||
|
if previous.isSome():
|
||
|
previous[].value
|
||
|
else:
|
||
|
v
|
||
|
|
||
|
let node =
|
||
|
TimedEntry[K, V](key: k, value: value, addedAt: addedAt, expiresAt: now + t.timeout)
|
||
|
if t.head == nil:
|
||
|
t.tail = node
|
||
|
t.head = t.tail
|
||
|
else:
|
||
|
# search from tail because typically that's where we add when now grows
|
||
|
var cur = t.tail
|
||
|
while cur != nil and node.expiresAt < cur.expiresAt:
|
||
|
cur = cur.prev
|
||
|
|
||
|
if cur == nil:
|
||
|
node.next = t.head
|
||
|
t.head.prev = node
|
||
|
t.head = node
|
||
|
else:
|
||
|
node.prev = cur
|
||
|
node.next = cur.next
|
||
|
cur.next = node
|
||
|
if cur == t.tail:
|
||
|
t.tail = node
|
||
|
|
||
|
t.entries.incl(node)
|
||
|
|
||
|
return node.value
|
||
|
|
||
|
func contains*[K, V](t: TimedMap[K, V], k: K): bool =
|
||
|
let tmp = TimedEntry[K, V](key: k)
|
||
|
tmp in t.entries
|
||
|
|
||
|
func addedAt*[K, V](t: var TimedMap[K, V], k: K): Moment =
|
||
|
let tmp = TimedEntry[K, V](key: k)
|
||
|
try:
|
||
|
if tmp in t.entries: # raising is slow
|
||
|
# Use shared instance from entries
|
||
|
return t.entries[tmp][].addedAt
|
||
|
except KeyError:
|
||
|
raiseAssert "just checked"
|
||
|
|
||
|
default(Moment)
|
||
|
|
||
|
func init*[K, V](T: type TimedMap[K, V], timeout: Duration): T =
|
||
|
T(timeout: timeout)
|