nim-datastore/datastore/tieredds.nim

102 lines
2.1 KiB
Nim
Raw Normal View History

import std/sequtils
import pkg/chronos
2022-05-11 10:50:05 -05:00
import pkg/questionable
import pkg/questionable/results
from pkg/stew/results as stewResults import get, isErr
2022-05-11 10:50:05 -05:00
import pkg/upraises
import ./datastore
export datastore
push: {.upraises: [].}
type
TieredDatastore* = ref object of Datastore
stores: seq[Datastore]
proc new*(
T: type TieredDatastore,
stores: varargs[Datastore]): ?!T =
if stores.len == 0:
failure "stores must contain at least one Datastore"
else:
success T(stores: @stores)
proc stores*(self: TieredDatastore): seq[Datastore] =
self.stores
method contains*(
self: TieredDatastore,
key: Key): Future[?!bool] {.async, locks: "unknown".} =
2022-05-11 10:50:05 -05:00
for store in self.stores:
let
containsRes = await store.contains(key)
2022-05-11 10:50:05 -05:00
if containsRes.isErr: return containsRes
if containsRes.get == true: return success true
return success false
2022-05-11 10:50:05 -05:00
method delete*(
self: TieredDatastore,
key: Key): Future[?!void] {.async, locks: "unknown".} =
2022-05-11 10:50:05 -05:00
let
pending = await allFinished(self.stores.mapIt(it.delete(key)))
for fut in pending:
if fut.read().isErr: return fut.read()
2022-05-11 10:50:05 -05:00
return success()
2022-05-11 10:50:05 -05:00
method get*(
self: TieredDatastore,
key: Key): Future[?!(?seq[byte])] {.async, locks: "unknown".} =
2022-05-11 10:50:05 -05:00
var
bytesOpt: ?seq[byte]
for store in self.stores:
let
getRes = await store.get(key)
if getRes.isErr: return getRes
bytesOpt = getRes.get
2022-05-11 10:50:05 -05:00
# put found data into stores logically in front of the current store
if bytes =? bytesOpt:
for s in self.stores:
if s == store: break
let
putRes = await s.put(key, bytes)
if putRes.isErr: return failure putRes.error.msg
2022-05-11 10:50:05 -05:00
break
return success bytesOpt
2022-05-11 10:50:05 -05:00
method put*(
self: TieredDatastore,
key: Key,
data: seq[byte]): Future[?!void] {.async, locks: "unknown".} =
2022-05-11 10:50:05 -05:00
let
pending = await allFinished(self.stores.mapIt(it.put(key, data)))
for fut in pending:
if fut.read().isErr: return fut.read()
2022-05-11 10:50:05 -05:00
return success()
2022-05-11 10:50:05 -05:00
# method query*(
# self: TieredDatastore,
# query: ...): Future[?!(?...)] {.async, locks: "unknown".} =
2022-05-11 10:50:05 -05:00
#
# return success ....some