mirror of
https://github.com/status-im/nim-raft.git
synced 2025-01-16 08:14:13 +00:00
Refactor some stuff. Add Key-Value DB support (MDBX)
This commit is contained in:
parent
2f4095915d
commit
f1d558cae5
@ -11,13 +11,13 @@ export Database, CRUD, Collection, Transaction, Cursor, Error, Index, Collatable
|
|||||||
type
|
type
|
||||||
MDBXStoreRef* = ref object of RootObj
|
MDBXStoreRef* = ref object of RootObj
|
||||||
database* {.requiresInit.}: Database
|
database* {.requiresInit.}: Database
|
||||||
chains* {.requiresInit.}: Collection
|
raftNodesData* {.requiresInit.}: Collection
|
||||||
|
|
||||||
MDBXTransaction* = ref object of RootObj
|
MDBXTransaction* = ref object of RootObj
|
||||||
transaction: CollectionTransaction
|
transaction: CollectionTransaction
|
||||||
|
|
||||||
const
|
const
|
||||||
MaxFileSize = 1024 * 1024 * 1024 * 1024 # 1 TB (MDBX default is 400 MB)
|
MaxFileSize = 1024 * 1024 * 400 #r: 100MB (MDBX default is 400 MB)
|
||||||
|
|
||||||
# ----------------------------------------------------------------------------------------
|
# ----------------------------------------------------------------------------------------
|
||||||
# MDBX exception handling helper templates
|
# MDBX exception handling helper templates
|
||||||
@ -62,11 +62,11 @@ proc rollback*(t: MDBXTransaction): ADbTResult[void] =
|
|||||||
ok()
|
ok()
|
||||||
|
|
||||||
proc beginDbTransaction*(db: MDBXStoreRef): ADbTResult[MDBXTransaction] =
|
proc beginDbTransaction*(db: MDBXStoreRef): ADbTResult[MDBXTransaction] =
|
||||||
if db.chains != nil:
|
if db.raftNodesData != nil:
|
||||||
handleEx():
|
handleEx():
|
||||||
ok(MDBXTransaction(transaction: db.chains.beginTransaction()))
|
ok(MDBXTransaction(transaction: db.raftNodesData.beginTransaction()))
|
||||||
else:
|
else:
|
||||||
err("MDBXStoreRef.chains is nil")
|
err("MDBXStoreRef.raftNodesData is nil")
|
||||||
|
|
||||||
proc put*(t: MDBXTransaction, key, value: openArray[byte]): ADbTResult[void] =
|
proc put*(t: MDBXTransaction, key, value: openArray[byte]): ADbTResult[void] =
|
||||||
handleExEx():
|
handleExEx():
|
||||||
@ -83,13 +83,13 @@ proc del*(t: MDBXTransaction, key: openArray[byte]): ADbTResult[void] =
|
|||||||
# ----------------------------------------------------------------------------------------
|
# ----------------------------------------------------------------------------------------
|
||||||
|
|
||||||
template checkDbChainsNotNil(db: MDBXStoreRef, body: untyped) =
|
template checkDbChainsNotNil(db: MDBXStoreRef, body: untyped) =
|
||||||
## Check if db.chains is not nil and execute the body
|
## Check if db.raftNodesData is not nil and execute the body
|
||||||
## if it is not nil. Otherwise, raise an assert.
|
## if it is not nil. Otherwise, raise an assert.
|
||||||
##
|
##
|
||||||
if db.chains != nil:
|
if db.raftNodesData != nil:
|
||||||
body
|
body
|
||||||
else:
|
else:
|
||||||
raiseAssert "MDBXStoreRef.chains is nil"
|
raiseAssert "MDBXStoreRef.raftNodesData is nil"
|
||||||
|
|
||||||
template withDbSnapshot*(db: MDBXStoreRef, body: untyped) =
|
template withDbSnapshot*(db: MDBXStoreRef, body: untyped) =
|
||||||
## Create an MDBX snapshot and execute the body providing
|
## Create an MDBX snapshot and execute the body providing
|
||||||
@ -99,7 +99,7 @@ template withDbSnapshot*(db: MDBXStoreRef, body: untyped) =
|
|||||||
##
|
##
|
||||||
checkDbChainsNotNil(db):
|
checkDbChainsNotNil(db):
|
||||||
handleEx():
|
handleEx():
|
||||||
let cs {.inject.} = db.chains.beginSnapshot()
|
let cs {.inject.} = db.raftNodesData.beginSnapshot()
|
||||||
defer: cs.finish()
|
defer: cs.finish()
|
||||||
body
|
body
|
||||||
|
|
||||||
@ -110,7 +110,7 @@ template withDbTransaction*(db: MDBXStoreRef, body: untyped) =
|
|||||||
##
|
##
|
||||||
checkDbChainsNotNil(db):
|
checkDbChainsNotNil(db):
|
||||||
handleEx():
|
handleEx():
|
||||||
var dbTransaction {.inject.} = db.chains.beginTransaction()
|
var dbTransaction {.inject.} = db.raftNodesData.beginTransaction()
|
||||||
defer: dbTransaction.commit()
|
defer: dbTransaction.commit()
|
||||||
try:
|
try:
|
||||||
body
|
body
|
||||||
@ -127,8 +127,11 @@ template withDbTransaction*(db: MDBXStoreRef, body: untyped) =
|
|||||||
# ------------------------------------------------------------------------------------------
|
# ------------------------------------------------------------------------------------------
|
||||||
# MDBX KvStore interface implementation
|
# MDBX KvStore interface implementation
|
||||||
# ------------------------------------------------------------------------------------------
|
# ------------------------------------------------------------------------------------------
|
||||||
|
type
|
||||||
|
DataProc = proc(data: seq[byte])
|
||||||
|
FindProc = proc(data: seq[byte])
|
||||||
|
|
||||||
proc get*(db: MDBXStoreRef, key: openArray[byte], onData: kvstore.DataProc): Result[bool] =
|
proc get*(db: MDBXStoreRef, key: openArray[byte], onData: DataProc): Result[bool, string] =
|
||||||
if key.len <= 0:
|
if key.len <= 0:
|
||||||
return err("mdbx: key cannot be empty on get")
|
return err("mdbx: key cannot be empty on get")
|
||||||
|
|
||||||
@ -140,10 +143,10 @@ proc get*(db: MDBXStoreRef, key: openArray[byte], onData: kvstore.DataProc): Res
|
|||||||
else:
|
else:
|
||||||
return ok(false)
|
return ok(false)
|
||||||
|
|
||||||
proc find*(db: MDBXStoreRef, prefix: openArray[byte], onFind: kvstore.KeyValueProc): Result[int] =
|
proc find*(db: MDBXStoreRef, prefix: openArray[byte], onFind: FindProc): Result[int, string] =
|
||||||
raiseAssert "Unimplemented"
|
raiseAssert "Unimplemented"
|
||||||
|
|
||||||
proc put*(db: MDBXStoreRef, key, value: openArray[byte]): Result[void] =
|
proc put*(db: MDBXStoreRef, key, value: openArray[byte]): Result[void, string] =
|
||||||
if key.len <= 0:
|
if key.len <= 0:
|
||||||
return err("mdbx: key cannot be empty on get")
|
return err("mdbx: key cannot be empty on get")
|
||||||
|
|
||||||
@ -151,7 +154,7 @@ proc put*(db: MDBXStoreRef, key, value: openArray[byte]): Result[void] =
|
|||||||
dbTransaction.put(asData(key), asData(value))
|
dbTransaction.put(asData(key), asData(value))
|
||||||
ok()
|
ok()
|
||||||
|
|
||||||
proc contains*(db: MDBXStoreRef, key: openArray[byte]): Result[bool] =
|
proc contains*(db: MDBXStoreRef, key: openArray[byte]): Result[bool, string] =
|
||||||
if key.len <= 0:
|
if key.len <= 0:
|
||||||
return err("mdbx: key cannot be empty on get")
|
return err("mdbx: key cannot be empty on get")
|
||||||
|
|
||||||
@ -162,7 +165,7 @@ proc contains*(db: MDBXStoreRef, key: openArray[byte]): Result[bool] =
|
|||||||
else:
|
else:
|
||||||
return ok(false)
|
return ok(false)
|
||||||
|
|
||||||
proc del*(db: MDBXStoreRef, key: openArray[byte]): Result[bool] =
|
proc del*(db: MDBXStoreRef, key: openArray[byte]): Result[bool, string] =
|
||||||
if key.len <= 0:
|
if key.len <= 0:
|
||||||
return err("mdbx: key cannot be empty on del")
|
return err("mdbx: key cannot be empty on del")
|
||||||
|
|
||||||
@ -174,7 +177,7 @@ proc del*(db: MDBXStoreRef, key: openArray[byte]): Result[bool] =
|
|||||||
else:
|
else:
|
||||||
return ok(false)
|
return ok(false)
|
||||||
|
|
||||||
proc clear*(db: MDBXStoreRef): Result[bool] =
|
proc clear*(db: MDBXStoreRef): Result[bool, string] =
|
||||||
raiseAssert "Unimplemented"
|
raiseAssert "Unimplemented"
|
||||||
|
|
||||||
proc close*(db: MDBXStoreRef) =
|
proc close*(db: MDBXStoreRef) =
|
||||||
@ -189,21 +192,21 @@ proc close*(db: MDBXStoreRef) =
|
|||||||
# .End. MDBX KvStore interface implementation
|
# .End. MDBX KvStore interface implementation
|
||||||
# ------------------------------------------------------------------------------------------
|
# ------------------------------------------------------------------------------------------
|
||||||
|
|
||||||
proc bulkPutSortedData*[KT: ByteArray32 | ByteArray33](db: MDBXStoreRef, keys: openArray[KT], vals: openArray[seq[byte]]): KvResult[int64] =
|
# proc bulkPutSortedData*[KT: ByteArray32 | ByteArray33](db: MDBXStoreRef, keys: openArray[KT], vals: openArray[seq[byte]]): Result[int64] =
|
||||||
if keys.len <= 0:
|
# if keys.len <= 0:
|
||||||
return err("mdbx: keys cannot be empty on bulkPutSortedData")
|
# return err("mdbx: keys cannot be empty on bulkPutSortedData")
|
||||||
|
|
||||||
if keys.len != vals.len:
|
# if keys.len != vals.len:
|
||||||
return err("mdbx: keys and vals must have the same length")
|
# return err("mdbx: keys and vals must have the same length")
|
||||||
|
|
||||||
withDbTransaction(db):
|
# withDbTransaction(db):
|
||||||
for i in 0 ..< keys.len:
|
# for i in 0 ..< keys.len:
|
||||||
dbTransaction.put(asData(keys[i]), asData(vals[i]))
|
# dbTransaction.put(asData(keys[i]), asData(vals[i]))
|
||||||
return ok(0)
|
# return ok(0)
|
||||||
|
|
||||||
proc init*(
|
proc init*(
|
||||||
T: type MDBXStoreRef, basePath: string, name: string,
|
T: type MDBXStoreRef, basePath: string, name: string,
|
||||||
readOnly = false): KvResult[T] =
|
readOnly = false): Result[T, string] =
|
||||||
let
|
let
|
||||||
dataDir = basePath / name / "data"
|
dataDir = basePath / name / "data"
|
||||||
backupsDir = basePath / name / "backups" # Do we need that in case of MDBX? Should discuss this with @zah
|
backupsDir = basePath / name / "backups" # Do we need that in case of MDBX? Should discuss this with @zah
|
||||||
@ -224,5 +227,5 @@ proc init*(
|
|||||||
handleEx():
|
handleEx():
|
||||||
let
|
let
|
||||||
db = openDatabase(dataDir, flags=mdbxFlags, maxFileSize=MaxFileSize)
|
db = openDatabase(dataDir, flags=mdbxFlags, maxFileSize=MaxFileSize)
|
||||||
chains = createCollection(db, "chains", StringKeys, BlobValues)
|
raftNodesData = createCollection(db, "raftNodesData", StringKeys, BlobValues)
|
||||||
ok(T(database: db, chains: chains))
|
ok(T(database: db, raftNodesData: raftNodesData))
|
||||||
|
@ -6,8 +6,9 @@
|
|||||||
# at your option.
|
# at your option.
|
||||||
# This file may not be copied, modified, or distributed except according to
|
# This file may not be copied, modified, or distributed except according to
|
||||||
# those terms.
|
# those terms.
|
||||||
|
|
||||||
import asyncdispatch
|
import asyncdispatch
|
||||||
import std/time
|
import std/times
|
||||||
|
|
||||||
template awaitWithTimeout[T](operation: Future[T],
|
template awaitWithTimeout[T](operation: Future[T],
|
||||||
deadline: Future[void],
|
deadline: Future[void],
|
||||||
|
@ -9,34 +9,31 @@
|
|||||||
|
|
||||||
import types, protocol, log_ops
|
import types, protocol, log_ops
|
||||||
|
|
||||||
proc RaftNodeStartElection*[SmCommandType, SmStateType](node: RaftNode[SmCommandType, SmStateType]) =
|
proc RaftNodeStartElection*[SmCommandType, SmStateType](node: RaftNode[SmCommandType, SmStateType]) {.async.} =
|
||||||
var
|
|
||||||
votesFuts: seq[Future[void]]
|
|
||||||
|
|
||||||
node.state = rnsCandidate
|
node.state = rnsCandidate
|
||||||
for p in node.peers:
|
|
||||||
p.votedFor = DefaultUUID
|
|
||||||
node.votedFor = node.id
|
node.votedFor = node.id
|
||||||
|
|
||||||
for peer in node.peers:
|
for peer in node.peers:
|
||||||
votesFuts.add(node.msgSendCallback(
|
peer.hasVoted = false
|
||||||
RaftMessageRequestVote(lastLogTerm: RaftNodeLogEntry(RaftNodeLogIndexGet(node)).term, lastLogIndex: RaftNodeLogIndexGet(node), senderTerm: node.currentTerm)
|
node.votesFuts.add(node.msgSendCallback(
|
||||||
|
RaftMessageRequestVote(lastLogTerm: RaftNodeLogEntryGet(node, RaftNodeLogIndexGet(node)).value.term, lastLogIndex: RaftNodeLogIndexGet(node), senderTerm: node.currentTerm)
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
# Process votes
|
# Process votes (if any)
|
||||||
for voteFut in votesFuts:
|
for voteFut in node.votesFuts:
|
||||||
await voteFut
|
discard await voteFut
|
||||||
if voteFut.finished and not voteFut.failed:
|
if voteFut.finished and not voteFut.failed:
|
||||||
for p in node.peers:
|
for p in node.peers:
|
||||||
if p.id == voteFut.senderId:
|
debugEcho repr(voteFut)
|
||||||
if voteFut.granted:
|
# if p.id == voteFut.senderId:
|
||||||
p.votedFor = node.id
|
# p.hasVoted = voteFut.granted
|
||||||
else:
|
|
||||||
if voteFut.votedFor.initialized:
|
# node.votesFuts.clear
|
||||||
p.votedFor = voteFut.votedFor
|
|
||||||
|
|
||||||
proc RaftNodeAbortElection*[SmCommandType, SmStateType](node: RaftNode[SmCommandType, SmStateType]) =
|
proc RaftNodeAbortElection*[SmCommandType, SmStateType](node: RaftNode[SmCommandType, SmStateType]) =
|
||||||
|
for fut in node.voteFuts:
|
||||||
|
cancel(fut)
|
||||||
discard
|
discard
|
||||||
|
|
||||||
proc RaftNodeProcessRequestVote*[SmCommandType, SmStateType](node: RaftNode[SmCommandType, SmStateType], msg: RaftMessageRequestVote): Future[RaftMessageRequestVoteResponse] =
|
proc RaftNodeProcessRequestVote*[SmCommandType, SmStateType](node: RaftNode[SmCommandType, SmStateType], msg: RaftMessageRequestVote): Future[RaftMessageRequestVoteResponse] =
|
||||||
|
@ -29,8 +29,7 @@ type
|
|||||||
senderTerm*: RaftNodeTerm # Sender Raft Node Term
|
senderTerm*: RaftNodeTerm # Sender Raft Node Term
|
||||||
|
|
||||||
RaftMessageRequestVoteResponse* = ref object of RaftMessageResponseBase
|
RaftMessageRequestVoteResponse* = ref object of RaftMessageResponseBase
|
||||||
granted*: bool # Is vote granted?
|
granted*: bool # Is vote granted by the Raft node, from we requested vote?
|
||||||
votedFor*: Option[RaftNodeId] # Present if vote is not granted
|
|
||||||
|
|
||||||
RaftMessageAppendEntries*[SmCommandType] = ref object of RaftMessageBase
|
RaftMessageAppendEntries*[SmCommandType] = ref object of RaftMessageBase
|
||||||
prevLogIndex*: RaftLogIndex
|
prevLogIndex*: RaftLogIndex
|
||||||
|
@ -13,6 +13,7 @@ import types
|
|||||||
import protocol
|
import protocol
|
||||||
import consensus_module
|
import consensus_module
|
||||||
import log_ops
|
import log_ops
|
||||||
|
import ../db/kvstore_mdbx
|
||||||
|
|
||||||
export types, protocol, consensus_module, log_ops
|
export types, protocol, consensus_module, log_ops
|
||||||
|
|
||||||
@ -106,7 +107,8 @@ proc RaftNodeScheduleHeartBeatTimeout*[SmCommandType, SmStateType](node: RaftNod
|
|||||||
node.heartBeatTimeoutTimer = sleepAsync(node.heartBeatTimeout)
|
node.heartBeatTimeoutTimer = sleepAsync(node.heartBeatTimeout)
|
||||||
await node.heartBeatTimeoutTimer
|
await node.heartBeatTimeoutTimer
|
||||||
node.state = rnsCandidate # Transition to candidate state and initiate new Election
|
node.state = rnsCandidate # Transition to candidate state and initiate new Election
|
||||||
RaftNodeStartElection(node)
|
var f = RaftNodeStartElection(node)
|
||||||
|
cancel(f)
|
||||||
|
|
||||||
proc RaftNodeSendHeartBeat*[SmCommandType, SmStateType](node: RaftNode[SmCommandType, SmStateType]) =
|
proc RaftNodeSendHeartBeat*[SmCommandType, SmStateType](node: RaftNode[SmCommandType, SmStateType]) =
|
||||||
for raftPeer in node.peers:
|
for raftPeer in node.peers:
|
||||||
@ -129,9 +131,9 @@ proc RaftNodeStop*[SmCommandType, SmStateType](node: RaftNode[SmCommandType, SmS
|
|||||||
# Try to stop gracefully
|
# Try to stop gracefully
|
||||||
node.state = rnsStopped
|
node.state = rnsStopped
|
||||||
# Cancel pending timers (if any)
|
# Cancel pending timers (if any)
|
||||||
RaftNodeCancelAllTimers(node)
|
var f = RaftNodeCancelAllTimers(node)
|
||||||
|
|
||||||
proc RaftNodeStart*[SmCommandType, SmStateType](node: RaftNode[SmCommandType, SmStateType]) =
|
proc RaftNodeStart*[SmCommandType, SmStateType](node: RaftNode[SmCommandType, SmStateType]) =
|
||||||
node.state = rnsFollower
|
node.state = rnsFollower
|
||||||
asyncSpawn RaftNodeScheduleHeartBeatTimeout(node)
|
asyncSpawn RaftNodeScheduleHeartBeatTimeout(node)
|
||||||
debugEcho "StartNode: ", node.id
|
debug "Start Raft Node with ID: ", nodeid=node.id
|
@ -32,7 +32,7 @@ type
|
|||||||
RaftNodeTerm* = uint64 # Raft Node Term Type
|
RaftNodeTerm* = uint64 # Raft Node Term Type
|
||||||
RaftLogIndex* = uint64 # Raft Node Log Index Type
|
RaftLogIndex* = uint64 # Raft Node Log Index Type
|
||||||
|
|
||||||
RaftNodePeer* = object # Raft Node Peer object
|
RaftNodePeer* = ref object # Raft Node Peer object
|
||||||
id*: RaftNodeId
|
id*: RaftNodeId
|
||||||
nextIndex*: RaftLogIndex # For each peer Raft Node, index of the next log entry to send to that Node
|
nextIndex*: RaftLogIndex # For each peer Raft Node, index of the next log entry to send to that Node
|
||||||
# (initialized to leader last log index + 1)
|
# (initialized to leader last log index + 1)
|
||||||
@ -116,6 +116,8 @@ type
|
|||||||
# Raft Node Object type
|
# Raft Node Object type
|
||||||
RaftNode*[SmCommandType, SmStateType] = ref object
|
RaftNode*[SmCommandType, SmStateType] = ref object
|
||||||
# Timers
|
# Timers
|
||||||
|
votesFuts*: seq[Future[RaftMessageResponseBase]]
|
||||||
|
|
||||||
requestVotesTimeout*: int
|
requestVotesTimeout*: int
|
||||||
heartBeatTimeout*: int
|
heartBeatTimeout*: int
|
||||||
appendEntriesTimeout*: int
|
appendEntriesTimeout*: int
|
||||||
|
@ -9,7 +9,7 @@
|
|||||||
|
|
||||||
import basic_timers
|
import basic_timers
|
||||||
import basic_state_machine
|
import basic_state_machine
|
||||||
|
import std/tables
|
||||||
export raft_api
|
export raft_api
|
||||||
|
|
||||||
type
|
type
|
||||||
@ -20,9 +20,6 @@ type
|
|||||||
|
|
||||||
proc BasicRaftClusterRaftMessageSendCallbackCreate(cluster: BasicRaftCluster): RaftMessageSendCallback =
|
proc BasicRaftClusterRaftMessageSendCallbackCreate(cluster: BasicRaftCluster): RaftMessageSendCallback =
|
||||||
proc (msg: RaftMessageBase): Future[RaftMessageResponseBase] {.async, gcsafe.} =
|
proc (msg: RaftMessageBase): Future[RaftMessageResponseBase] {.async, gcsafe.} =
|
||||||
var
|
|
||||||
nodeIdx: int = -1
|
|
||||||
|
|
||||||
result = await cluster.nodes[msg.receiverId].RaftNodeMessageDeliver(msg)
|
result = await cluster.nodes[msg.receiverId].RaftNodeMessageDeliver(msg)
|
||||||
|
|
||||||
proc BasicRaftClusterStart*(cluster: BasicRaftCluster) =
|
proc BasicRaftClusterStart*(cluster: BasicRaftCluster) =
|
||||||
|
@ -22,7 +22,7 @@ proc basicClusterMain*() =
|
|||||||
nodesIds[i] = genUUID()
|
nodesIds[i] = genUUID()
|
||||||
|
|
||||||
cluster = BasicRaftClusterInit(nodesIds)
|
cluster = BasicRaftClusterInit(nodesIds)
|
||||||
check cluster.nodes.len == 5
|
# check size(cluster.nodes) == 5
|
||||||
|
|
||||||
test "Generate Random Client SmCommands Queue":
|
test "Generate Random Client SmCommands Queue":
|
||||||
discard
|
discard
|
||||||
|
Loading…
x
Reference in New Issue
Block a user