mirror of
https://github.com/codex-storage/nim-codex.git
synced 2025-01-10 21:15:59 +00:00
70cbdff901
* feat: introduce LRU cache Replace `MemoryStore` with LRU caching mechanism. `lrucache` library was forked to https://github.com/status-im/lrucache.nim. Co-authored-by: Eric Mastro <eric.mastro@gmail.com> # Conflicts: # dagger/dagger.nim # dagger/stores.nim # dagger/stores/manager.nim # tests/dagger/blockexc/testengine.nim # tests/dagger/helpers/nodeutils.nim # tests/dagger/testnode.nim # tests/dagger/teststores.nim * feat: introduce --cache-size CLI option Allow for a value of `0` to disable the cache. # Conflicts: # dagger/dagger.nim * allow dynamic block size in cache allow block size to be variable/dynamic update lrucache to use updated lrucache dep Using removeLru proc, added tests * Refactor CacheStore init block Co-authored-by: Michael Bradley, Jr <michaelsbradleyjr@gmail.com>
109 lines
2.9 KiB
Nim
109 lines
2.9 KiB
Nim
import std/strutils
|
|
|
|
import pkg/chronos
|
|
import pkg/asynctest
|
|
import pkg/libp2p
|
|
import pkg/stew/byteutils
|
|
import pkg/questionable/results
|
|
import pkg/dagger/stores/cachestore
|
|
import pkg/dagger/chunker
|
|
|
|
import ../helpers
|
|
|
|
suite "Cache Store tests":
|
|
var
|
|
newBlock, newBlock1, newBlock2, newBlock3: Block
|
|
store: CacheStore
|
|
|
|
setup:
|
|
newBlock = Block.init("New Kids on the Block".toBytes()).tryGet()
|
|
newBlock1 = Block.init("1".repeat(100).toBytes()).tryGet()
|
|
newBlock2 = Block.init("2".repeat(100).toBytes()).tryGet()
|
|
newBlock3 = Block.init("3".repeat(100).toBytes()).tryGet()
|
|
store = CacheStore.new()
|
|
|
|
test "constructor":
|
|
# cache size cannot be smaller than chunk size
|
|
expect ValueError:
|
|
discard CacheStore.new(cacheSize = 1, chunkSize = 2)
|
|
|
|
store = CacheStore.new(cacheSize = 100, chunkSize = 1)
|
|
check store.currentSize == 0
|
|
store = CacheStore.new(@[newBlock1, newBlock2, newBlock3])
|
|
check store.currentSize == 300
|
|
|
|
# initial cache blocks total more than cache size, currentSize should
|
|
# never exceed max cache size
|
|
store = CacheStore.new(
|
|
blocks = @[newBlock1, newBlock2, newBlock3],
|
|
cacheSize = 200,
|
|
chunkSize = 1)
|
|
check store.currentSize == 200
|
|
|
|
# cache size cannot be less than chunks size
|
|
expect ValueError:
|
|
discard CacheStore.new(
|
|
cacheSize = 99,
|
|
chunkSize = 100)
|
|
|
|
test "putBlock":
|
|
|
|
check:
|
|
await store.putBlock(newBlock1)
|
|
newBlock1.cid in store
|
|
|
|
# block size bigger than entire cache
|
|
store = CacheStore.new(cacheSize = 99, chunkSize = 98)
|
|
check not await store.putBlock(newBlock1)
|
|
|
|
# block being added causes removal of LRU block
|
|
store = CacheStore.new(
|
|
@[newBlock1, newBlock2, newBlock3],
|
|
cacheSize = 200,
|
|
chunkSize = 1)
|
|
check:
|
|
not store.hasBlock(newBlock1.cid)
|
|
store.hasBlock(newBlock2.cid)
|
|
store.hasBlock(newBlock3.cid)
|
|
store.currentSize == newBlock2.data.len + newBlock3.data.len # 200
|
|
|
|
test "getBlock":
|
|
store = CacheStore.new(@[newBlock])
|
|
|
|
let blk = await store.getBlock(newBlock.cid)
|
|
|
|
check:
|
|
blk.isOk
|
|
blk.get == newBlock
|
|
|
|
test "fail getBlock":
|
|
let blk = await store.getBlock(newBlock.cid)
|
|
|
|
check:
|
|
blk.isErr
|
|
blk.error of system.KeyError
|
|
|
|
test "hasBlock":
|
|
let store = CacheStore.new(@[newBlock])
|
|
|
|
check store.hasBlock(newBlock.cid)
|
|
|
|
test "fail hasBlock":
|
|
check not store.hasBlock(newBlock.cid)
|
|
|
|
test "delBlock":
|
|
# empty cache
|
|
check not await store.delBlock(newBlock1.cid)
|
|
|
|
# successfully deleted
|
|
discard await store.putBlock(newBlock1)
|
|
check await store.delBlock(newBlock1.cid)
|
|
|
|
# deletes item should decrement size
|
|
store = CacheStore.new(@[newBlock1, newBlock2, newBlock3])
|
|
check:
|
|
store.currentSize == 300
|
|
await store.delBlock(newBlock2.cid)
|
|
store.currentSize == 200
|
|
newBlock2.cid notin store
|