mirror of
https://github.com/waku-org/nwaku.git
synced 2025-01-27 23:27:27 +00:00
feat(rln-relay): use new atomic_operation ffi api (#1733)
* chore(rln-relay): bump zerokit * feat(rln-relay): use new atomic_operations ffi api * fix(rln-relay): static gm
This commit is contained in:
parent
665484c17b
commit
611e9539a6
@ -6,7 +6,7 @@ else:
|
|||||||
{.push raises: [].}
|
{.push raises: [].}
|
||||||
|
|
||||||
import
|
import
|
||||||
std/[options, osproc, streams, strutils, tables],
|
std/[options, osproc, streams, strutils],
|
||||||
stew/[results, byteutils],
|
stew/[results, byteutils],
|
||||||
stew/shims/net as stewNet,
|
stew/shims/net as stewNet,
|
||||||
testutils/unittests,
|
testutils/unittests,
|
||||||
@ -516,11 +516,8 @@ suite "Onchain group manager":
|
|||||||
manager.validRootBuffer.len() == 1
|
manager.validRootBuffer.len() == 1
|
||||||
|
|
||||||
# We can now simulate a chain reorg by calling backfillRootQueue
|
# We can now simulate a chain reorg by calling backfillRootQueue
|
||||||
var blockTable = default(BlockTable)
|
|
||||||
blockTable[1.uint] = @[Membership(idCommitment: credentials[4].idCommitment, index: 4.uint)]
|
|
||||||
|
|
||||||
let expectedLastRoot = manager.validRootBuffer[0]
|
let expectedLastRoot = manager.validRootBuffer[0]
|
||||||
await manager.backfillRootQueue(blockTable)
|
await manager.backfillRootQueue(1)
|
||||||
|
|
||||||
# We should now have 5 roots in the queue, and no partial buffer
|
# We should now have 5 roots in the queue, and no partial buffer
|
||||||
check:
|
check:
|
||||||
|
2
vendor/zerokit
vendored
2
vendor/zerokit
vendored
@ -1 +1 @@
|
|||||||
Subproject commit 584c2cf4c000b391ca6b415c09d8399fde329e5c
|
Subproject commit c2d386cb749f551541bb34c4386a3849485356f9
|
@ -90,6 +90,20 @@ proc serialize*(roots: seq[MerkleNode]): seq[byte] =
|
|||||||
rootsBytes = concat(rootsBytes, @root)
|
rootsBytes = concat(rootsBytes, @root)
|
||||||
return rootsBytes
|
return rootsBytes
|
||||||
|
|
||||||
|
# Serializes a sequence of MembershipIndex's
|
||||||
|
proc serialize*(memIndices: seq[MembershipIndex]): seq[byte] =
|
||||||
|
var memIndicesBytes = newSeq[byte]()
|
||||||
|
|
||||||
|
# serialize the memIndices, with its length prefixed
|
||||||
|
let len = toBytes(uint64(memIndices.len), Endianness.littleEndian)
|
||||||
|
memIndicesBytes.add(len)
|
||||||
|
|
||||||
|
for memIndex in memIndices:
|
||||||
|
let memIndexBytes = toBytes(uint64(memIndex), Endianness.littleEndian)
|
||||||
|
memIndicesBytes = concat(memIndicesBytes, @memIndexBytes)
|
||||||
|
|
||||||
|
return memIndicesBytes
|
||||||
|
|
||||||
proc toEpoch*(t: uint64): Epoch =
|
proc toEpoch*(t: uint64): Epoch =
|
||||||
## converts `t` to `Epoch` in little-endian order
|
## converts `t` to `Epoch` in little-endian order
|
||||||
let bytes = toBytes(t, Endianness.littleEndian)
|
let bytes = toBytes(t, Endianness.littleEndian)
|
||||||
|
@ -83,6 +83,10 @@ method withdraw*(g: GroupManager, identitySecretHash: IdentitySecretHash): Futur
|
|||||||
method withdrawBatch*(g: GroupManager, identitySecretHashes: seq[IdentitySecretHash]): Future[void] {.base,gcsafe.} =
|
method withdrawBatch*(g: GroupManager, identitySecretHashes: seq[IdentitySecretHash]): Future[void] {.base,gcsafe.} =
|
||||||
raise newException(CatchableError, "withdrawBatch proc for " & $g.type & " is not implemented yet")
|
raise newException(CatchableError, "withdrawBatch proc for " & $g.type & " is not implemented yet")
|
||||||
|
|
||||||
|
# This proc is used to insert and remove a set of commitments from the merkle tree
|
||||||
|
method atomicBatch*(g: GroupManager, idCommitments: seq[IDCommitment], toRemoveIndices: seq[MembershipIndex]): Future[void] {.base,gcsafe.} =
|
||||||
|
raise newException(CatchableError, "atomicBatch proc for " & $g.type & " is not implemented yet")
|
||||||
|
|
||||||
# This proc is used to set a callback that will be called when an identity commitment is withdrawn
|
# This proc is used to set a callback that will be called when an identity commitment is withdrawn
|
||||||
# The callback may be called multiple times, and should be used to for any post processing
|
# The callback may be called multiple times, and should be used to for any post processing
|
||||||
method onWithdraw*(g: GroupManager, cb: OnWithdrawCallback) {.base,gcsafe.} =
|
method onWithdraw*(g: GroupManager, cb: OnWithdrawCallback) {.base,gcsafe.} =
|
||||||
|
@ -67,25 +67,17 @@ template initializedGuard(g: OnchainGroupManager): untyped =
|
|||||||
method register*(g: OnchainGroupManager, idCommitment: IDCommitment): Future[void] {.async.} =
|
method register*(g: OnchainGroupManager, idCommitment: IDCommitment): Future[void] {.async.} =
|
||||||
initializedGuard(g)
|
initializedGuard(g)
|
||||||
|
|
||||||
let memberInserted = g.rlnInstance.insertMember(idCommitment)
|
await g.registerBatch(@[idCommitment])
|
||||||
if not memberInserted:
|
|
||||||
raise newException(ValueError,"member insertion failed")
|
|
||||||
|
|
||||||
if g.registerCb.isSome():
|
method atomicBatch*(g: OnchainGroupManager,
|
||||||
await g.registerCb.get()(@[Membership(idCommitment: idCommitment, index: g.latestIndex)])
|
idCommitments = newSeq[IDCommitment](),
|
||||||
|
toRemoveIndices = newSeq[MembershipIndex]()): Future[void] {.async.} =
|
||||||
g.validRootBuffer = g.slideRootQueue()
|
|
||||||
|
|
||||||
g.latestIndex += 1
|
|
||||||
|
|
||||||
return
|
|
||||||
|
|
||||||
method registerBatch*(g: OnchainGroupManager, idCommitments: seq[IDCommitment]): Future[void] {.async.} =
|
|
||||||
initializedGuard(g)
|
initializedGuard(g)
|
||||||
|
|
||||||
let membersInserted = g.rlnInstance.insertMembers(g.latestIndex, idCommitments)
|
let startIndex = g.latestIndex
|
||||||
if not membersInserted:
|
let operationSuccess = g.rlnInstance.atomicWrite(some(startIndex), idCommitments, toRemoveIndices)
|
||||||
raise newException(ValueError, "Failed to insert members into the merkle tree")
|
if not operationSuccess:
|
||||||
|
raise newException(ValueError, "atomic batch operation failed")
|
||||||
|
|
||||||
if g.registerCb.isSome():
|
if g.registerCb.isSome():
|
||||||
var membersSeq = newSeq[Membership]()
|
var membersSeq = newSeq[Membership]()
|
||||||
@ -100,7 +92,12 @@ method registerBatch*(g: OnchainGroupManager, idCommitments: seq[IDCommitment]):
|
|||||||
|
|
||||||
g.latestIndex += MembershipIndex(idCommitments.len())
|
g.latestIndex += MembershipIndex(idCommitments.len())
|
||||||
|
|
||||||
return
|
|
||||||
|
method registerBatch*(g: OnchainGroupManager, idCommitments: seq[IDCommitment]): Future[void] {.async.} =
|
||||||
|
initializedGuard(g)
|
||||||
|
|
||||||
|
await g.atomicBatch(idCommitments)
|
||||||
|
|
||||||
|
|
||||||
method register*(g: OnchainGroupManager, identityCredentials: IdentityCredential): Future[void] {.async.} =
|
method register*(g: OnchainGroupManager, identityCredentials: IdentityCredential): Future[void] {.async.} =
|
||||||
initializedGuard(g)
|
initializedGuard(g)
|
||||||
@ -154,7 +151,7 @@ method withdraw*(g: OnchainGroupManager, idCommitment: IDCommitment): Future[voi
|
|||||||
method withdrawBatch*(g: OnchainGroupManager, idCommitments: seq[IDCommitment]): Future[void] {.async.} =
|
method withdrawBatch*(g: OnchainGroupManager, idCommitments: seq[IDCommitment]): Future[void] {.async.} =
|
||||||
initializedGuard(g)
|
initializedGuard(g)
|
||||||
|
|
||||||
# TODO: after slashing is enabled on the contract
|
# TODO: after slashing is enabled on the contract, use atomicBatch internally
|
||||||
|
|
||||||
proc parseEvent(event: type MemberRegistered,
|
proc parseEvent(event: type MemberRegistered,
|
||||||
log: JsonNode): GroupManagerResult[Membership] =
|
log: JsonNode): GroupManagerResult[Membership] =
|
||||||
@ -178,28 +175,23 @@ proc parseEvent(event: type MemberRegistered,
|
|||||||
except CatchableError:
|
except CatchableError:
|
||||||
return err("failed to parse the data field of the MemberRegistered event")
|
return err("failed to parse the data field of the MemberRegistered event")
|
||||||
|
|
||||||
type BlockTable* = OrderedTable[BlockNumber, seq[Membership]]
|
type BlockTable* = OrderedTable[BlockNumber, seq[(Membership, bool)]]
|
||||||
|
|
||||||
proc backfillRootQueue*(g: OnchainGroupManager, blockTable: BlockTable): Future[void] {.async.} =
|
proc backfillRootQueue*(g: OnchainGroupManager, len: uint): Future[void] {.async.} =
|
||||||
if blocktable.len() > 0:
|
if len > 0:
|
||||||
for blockNumber, members in blocktable.pairs():
|
# backfill the tree's acceptable roots
|
||||||
let deletionSuccess = g.rlnInstance.removeMembers(members.mapIt(it.index))
|
for i in 0..len-1:
|
||||||
debug "deleting members to reconcile state"
|
# remove the last root
|
||||||
if not deletionSuccess:
|
g.validRoots.popLast()
|
||||||
error "failed to delete members from the tree", success=deletionSuccess
|
for i in 0..len-1:
|
||||||
raise newException(ValueError, "failed to delete member from the tree, tree is inconsistent")
|
# add the backfilled root
|
||||||
# backfill the tree's acceptable roots
|
g.validRoots.addLast(g.validRootBuffer.popLast())
|
||||||
for i in 0..blocktable.len()-1:
|
|
||||||
# remove the last root
|
|
||||||
g.validRoots.popLast()
|
|
||||||
for i in 0..blockTable.len()-1:
|
|
||||||
# add the backfilled root
|
|
||||||
g.validRoots.addLast(g.validRootBuffer.popLast())
|
|
||||||
|
|
||||||
proc insert(blockTable: var BlockTable, blockNumber: BlockNumber, member: Membership) =
|
proc insert(blockTable: var BlockTable, blockNumber: BlockNumber, member: Membership, removed: bool) =
|
||||||
if blockTable.hasKeyOrPut(blockNumber, @[member]):
|
let memberTuple = (member, removed)
|
||||||
|
if blockTable.hasKeyOrPut(blockNumber, @[memberTuple]):
|
||||||
try:
|
try:
|
||||||
blockTable[blockNumber].add(member)
|
blockTable[blockNumber].add(memberTuple)
|
||||||
except KeyError: # qed
|
except KeyError: # qed
|
||||||
error "could not insert member into block table", blockNumber=blockNumber, member=member
|
error "could not insert member into block table", blockNumber=blockNumber, member=member
|
||||||
|
|
||||||
@ -226,19 +218,18 @@ proc getRawEvents(g: OnchainGroupManager,
|
|||||||
toBlock = some(normalizedToBlock.blockId()))
|
toBlock = some(normalizedToBlock.blockId()))
|
||||||
return events
|
return events
|
||||||
|
|
||||||
proc getBlockTables(g: OnchainGroupManager,
|
proc getBlockTable(g: OnchainGroupManager,
|
||||||
fromBlock: BlockNumber,
|
fromBlock: BlockNumber,
|
||||||
toBlock: Option[BlockNumber] = none(BlockNumber)): Future[(BlockTable, BlockTable)] {.async.} =
|
toBlock: Option[BlockNumber] = none(BlockNumber)): Future[BlockTable] {.async.} =
|
||||||
initializedGuard(g)
|
initializedGuard(g)
|
||||||
|
|
||||||
var blockTable = default(BlockTable)
|
var blockTable = default(BlockTable)
|
||||||
var toRemoveBlockTable = default(BlockTable)
|
|
||||||
|
|
||||||
let events = await g.getRawEvents(fromBlock, toBlock)
|
let events = await g.getRawEvents(fromBlock, toBlock)
|
||||||
|
|
||||||
if events.len == 0:
|
if events.len == 0:
|
||||||
debug "no events found"
|
debug "no events found"
|
||||||
return (blockTable, toRemoveBlockTable)
|
return blockTable
|
||||||
|
|
||||||
for event in events:
|
for event in events:
|
||||||
let blockNumber = parseHexInt(event["blockNumber"].getStr()).uint
|
let blockNumber = parseHexInt(event["blockNumber"].getStr()).uint
|
||||||
@ -248,52 +239,45 @@ proc getBlockTables(g: OnchainGroupManager,
|
|||||||
error "failed to parse the MemberRegistered event", error=parsedEventRes.error()
|
error "failed to parse the MemberRegistered event", error=parsedEventRes.error()
|
||||||
raise newException(ValueError, "failed to parse the MemberRegistered event")
|
raise newException(ValueError, "failed to parse the MemberRegistered event")
|
||||||
let parsedEvent = parsedEventRes.get()
|
let parsedEvent = parsedEventRes.get()
|
||||||
|
blockTable.insert(blockNumber, parsedEvent, removed)
|
||||||
|
|
||||||
if removed:
|
return blockTable
|
||||||
# remove the registration from the tree, per block
|
|
||||||
warn "member removed from the tree as per canonical chain", index=parsedEvent.index
|
|
||||||
toRemoveBlockTable.insert(blockNumber, parsedEvent)
|
|
||||||
else:
|
|
||||||
blockTable.insert(blockNumber, parsedEvent)
|
|
||||||
|
|
||||||
return (blockTable, toRemoveBlockTable)
|
proc handleEvents(g: OnchainGroupManager,
|
||||||
|
blockTable: BlockTable): Future[void] {.async.} =
|
||||||
proc handleValidEvents(g: OnchainGroupManager, blockTable: BlockTable): Future[void] {.async.} =
|
|
||||||
initializedGuard(g)
|
initializedGuard(g)
|
||||||
|
|
||||||
for blockNumber, members in blockTable.pairs():
|
for blockNumber, members in blockTable.pairs():
|
||||||
let latestIndex = g.latestIndex
|
|
||||||
let startingIndex = members[0].index
|
|
||||||
try:
|
try:
|
||||||
await g.registerBatch(members.mapIt(it.idCommitment))
|
await g.atomicBatch(idCommitments = members.mapIt(it[0].idCommitment),
|
||||||
|
toRemoveIndices = members.filterIt(it[1]).mapIt(it[0].index))
|
||||||
except CatchableError:
|
except CatchableError:
|
||||||
error "failed to insert members into the tree", error=getCurrentExceptionMsg()
|
error "failed to insert members into the tree", error=getCurrentExceptionMsg()
|
||||||
raise newException(ValueError, "failed to insert members into the tree")
|
raise newException(ValueError, "failed to insert members into the tree")
|
||||||
trace "new members added to the Merkle tree", commitments=members.mapIt(it.idCommitment.inHex()) , startingIndex=startingIndex
|
trace "new members added to the Merkle tree", commitments=members.mapIt(it.idCommitment.inHex()) , startingIndex=startingIndex
|
||||||
let lastIndex = startingIndex + members.len.uint - 1
|
|
||||||
let indexGap = startingIndex - latestIndex
|
|
||||||
if not (toSeq(startingIndex..lastIndex) == members.mapIt(it.index)):
|
|
||||||
raise newException(ValueError, "membership indices are not sequential")
|
|
||||||
if indexGap != 1.uint and lastIndex != latestIndex and startingIndex != 0.uint:
|
|
||||||
warn "membership index gap, may have lost connection", lastIndex, currIndex=latestIndex, indexGap = indexGap
|
|
||||||
g.latestProcessedBlock = some(blockNumber)
|
g.latestProcessedBlock = some(blockNumber)
|
||||||
|
|
||||||
return
|
return
|
||||||
|
|
||||||
proc handleRemovedEvents(g: OnchainGroupManager, toRemoveBlockTable: BlockTable): Future[void] {.async.} =
|
proc handleRemovedEvents(g: OnchainGroupManager, blockTable: BlockTable): Future[void] {.async.} =
|
||||||
initializedGuard(g)
|
initializedGuard(g)
|
||||||
|
|
||||||
await g.backfillRootQueue(toRemoveBlockTable)
|
# count number of blocks that have been removed
|
||||||
|
var numRemovedBlocks: uint = 0
|
||||||
|
for blockNumber, members in blockTable.pairs():
|
||||||
|
if members.anyIt(it[1]):
|
||||||
|
numRemovedBlocks += 1
|
||||||
|
|
||||||
|
await g.backfillRootQueue(numRemovedBlocks)
|
||||||
|
|
||||||
proc getAndHandleEvents(g: OnchainGroupManager,
|
proc getAndHandleEvents(g: OnchainGroupManager,
|
||||||
fromBlock: BlockNumber,
|
fromBlock: BlockNumber,
|
||||||
toBlock: Option[BlockNumber] = none(BlockNumber)): Future[void] {.async.} =
|
toBlock: Option[BlockNumber] = none(BlockNumber)): Future[void] {.async.} =
|
||||||
initializedGuard(g)
|
initializedGuard(g)
|
||||||
|
|
||||||
let (validEvents, removedEvents) = await g.getBlockTables(fromBlock, toBlock)
|
let blockTable = await g.getBlockTable(fromBlock, toBlock)
|
||||||
await g.handleRemovedEvents(removedEvents)
|
await g.handleEvents(blockTable)
|
||||||
await g.handleValidEvents(validEvents)
|
await g.handleRemovedEvents(blockTable)
|
||||||
return
|
|
||||||
|
|
||||||
proc getNewHeadCallback(g: OnchainGroupManager): BlockHeaderHandler =
|
proc getNewHeadCallback(g: OnchainGroupManager): BlockHeaderHandler =
|
||||||
proc newHeadCallback(blockheader: BlockHeader) {.gcsafe.} =
|
proc newHeadCallback(blockheader: BlockHeader) {.gcsafe.} =
|
||||||
@ -435,7 +419,7 @@ method init*(g: OnchainGroupManager): Future[void] {.async.} =
|
|||||||
try:
|
try:
|
||||||
membershipFee = await contract.MEMBERSHIP_DEPOSIT().call()
|
membershipFee = await contract.MEMBERSHIP_DEPOSIT().call()
|
||||||
except CatchableError:
|
except CatchableError:
|
||||||
raise newException(ValueError, "could not get the membership deposit")
|
raise newException(ValueError, "could not get the membership deposit: {}")
|
||||||
|
|
||||||
|
|
||||||
g.ethRpc = some(ethRpc)
|
g.ethRpc = some(ethRpc)
|
||||||
|
@ -52,17 +52,8 @@ method startGroupSync*(g: StaticGroupManager): Future[void] =
|
|||||||
method register*(g: StaticGroupManager, idCommitment: IDCommitment): Future[void] {.async.} =
|
method register*(g: StaticGroupManager, idCommitment: IDCommitment): Future[void] {.async.} =
|
||||||
initializedGuard(g)
|
initializedGuard(g)
|
||||||
|
|
||||||
let memberInserted = g.rlnInstance.insertMember(idCommitment)
|
await g.registerBatch(@[idCommitment])
|
||||||
if not memberInserted:
|
|
||||||
raise newException(ValueError, "Failed to insert member into the merkle tree")
|
|
||||||
|
|
||||||
discard g.slideRootQueue()
|
|
||||||
|
|
||||||
g.latestIndex += 1
|
|
||||||
|
|
||||||
if g.registerCb.isSome():
|
|
||||||
await g.registerCb.get()(@[Membership(idCommitment: idCommitment, index: g.latestIndex)])
|
|
||||||
return
|
|
||||||
|
|
||||||
method registerBatch*(g: StaticGroupManager, idCommitments: seq[IDCommitment]): Future[void] {.async.} =
|
method registerBatch*(g: StaticGroupManager, idCommitments: seq[IDCommitment]): Future[void] {.async.} =
|
||||||
initializedGuard(g)
|
initializedGuard(g)
|
||||||
@ -74,12 +65,12 @@ method registerBatch*(g: StaticGroupManager, idCommitments: seq[IDCommitment]):
|
|||||||
if g.registerCb.isSome():
|
if g.registerCb.isSome():
|
||||||
var memberSeq = newSeq[Membership]()
|
var memberSeq = newSeq[Membership]()
|
||||||
for i in 0..<idCommitments.len():
|
for i in 0..<idCommitments.len():
|
||||||
memberSeq.add(Membership(idCommitment: idCommitments[i], index: g.latestIndex + MembershipIndex(i)))
|
memberSeq.add(Membership(idCommitment: idCommitments[i], index: g.latestIndex + MembershipIndex(i) + 1))
|
||||||
await g.registerCb.get()(memberSeq)
|
await g.registerCb.get()(memberSeq)
|
||||||
|
|
||||||
discard g.slideRootQueue()
|
discard g.slideRootQueue()
|
||||||
|
|
||||||
g.latestIndex += MembershipIndex(idCommitments.len() - 1)
|
g.latestIndex += MembershipIndex(idCommitments.len())
|
||||||
|
|
||||||
return
|
return
|
||||||
|
|
||||||
|
@ -61,10 +61,12 @@ proc init_tree_with_leaves*(ctx: ptr RLN, input_buffer: ptr Buffer): bool {.impo
|
|||||||
## leaves are set one after each other starting from index 0
|
## leaves are set one after each other starting from index 0
|
||||||
## the return bool value indicates the success or failure of the operation
|
## the return bool value indicates the success or failure of the operation
|
||||||
|
|
||||||
proc set_leaves_from*(ctx: ptr RLN, index: uint, input_buffer: ptr Buffer): bool {.importc: "set_leaves_from".}
|
proc atomic_write*(ctx: ptr RLN, index: uint, leaves_buffer: ptr Buffer, indices_buffer: ptr Buffer): bool {.importc: "atomic_operation".}
|
||||||
## sets multiple leaves in the tree stored by ctx to the value passed by input_buffer
|
## sets multiple leaves, and zeroes out indices in the tree stored by ctx to the value passed by input_buffer
|
||||||
## the input_buffer holds a serialized vector of leaves (32 bytes each)
|
## the leaves_buffer holds a serialized vector of leaves (32 bytes each)
|
||||||
## the input_buffer size is prefixed by a 8 bytes integer indicating the number of leaves
|
## the leaves_buffer size is prefixed by a 8 bytes integer indicating the number of leaves
|
||||||
|
## the indices_bufffer holds a serialized vector of indices (8 bytes each)
|
||||||
|
## the indices_buffer size is prefixed by a 8 bytes integer indicating the number of indices
|
||||||
## leaves are set one after each other starting from index `index`
|
## leaves are set one after each other starting from index `index`
|
||||||
## the return bool value indicates the success or failure of the operation
|
## the return bool value indicates the success or failure of the operation
|
||||||
|
|
||||||
|
@ -230,34 +230,48 @@ proc insertMember*(rlnInstance: ptr RLN, idComm: IDCommitment): bool =
|
|||||||
let memberAdded = update_next_member(rlnInstance, pkBufferPtr)
|
let memberAdded = update_next_member(rlnInstance, pkBufferPtr)
|
||||||
return memberAdded
|
return memberAdded
|
||||||
|
|
||||||
|
proc atomicWrite*(rlnInstance: ptr RLN,
|
||||||
|
index = none(MembershipIndex),
|
||||||
|
idComms = newSeq[IDCommitment](),
|
||||||
|
toRemoveIndices = newSeq[MembershipIndex]()): bool =
|
||||||
|
## Insert multiple members i.e., identity commitments, and remove multiple members
|
||||||
|
## returns true if the operation is successful
|
||||||
|
## returns false if the operation fails
|
||||||
|
|
||||||
|
let startIndex = if index.isNone(): MembershipIndex(0) else: index.get()
|
||||||
|
|
||||||
|
# serialize the idComms
|
||||||
|
let idCommsBytes = serialize(idComms)
|
||||||
|
var idCommsBuffer = idCommsBytes.toBuffer()
|
||||||
|
let idCommsBufferPtr = addr idCommsBuffer
|
||||||
|
|
||||||
|
# serialize the toRemoveIndices
|
||||||
|
let indicesBytes = serialize(toRemoveIndices)
|
||||||
|
var indicesBuffer = indicesBytes.toBuffer()
|
||||||
|
let indicesBufferPtr = addr indicesBuffer
|
||||||
|
|
||||||
|
let operationSuccess = atomic_write(rlnInstance,
|
||||||
|
startIndex,
|
||||||
|
idCommsBufferPtr,
|
||||||
|
indicesBufferPtr)
|
||||||
|
return operationSuccess
|
||||||
|
|
||||||
proc insertMembers*(rlnInstance: ptr RLN,
|
proc insertMembers*(rlnInstance: ptr RLN,
|
||||||
index: MembershipIndex,
|
index: MembershipIndex,
|
||||||
idComms: seq[IDCommitment]): bool =
|
idComms: seq[IDCommitment]): bool =
|
||||||
## Insert multiple members i.e., identity commitments
|
## Insert multiple members i.e., identity commitments
|
||||||
## returns true if the insertion is successful
|
## returns true if the insertion is successful
|
||||||
## returns false if any of the insertions fails
|
## returns false if any of the insertions fails
|
||||||
## Note: This proc is atomic, i.e., if any of the insertions fails, all the previous insertions are rolled back
|
## Note: This proc is atomic, i.e., if any of the insertions fails, all the previous insertions are rolled back
|
||||||
|
|
||||||
# serialize the idComms
|
return atomicWrite(rlnInstance, some(index), idComms)
|
||||||
let idCommsBytes = serialize(idComms)
|
|
||||||
|
|
||||||
var idCommsBuffer = idCommsBytes.toBuffer()
|
|
||||||
let idCommsBufferPtr = addr idCommsBuffer
|
|
||||||
# add the member to the tree
|
|
||||||
let membersAdded = set_leaves_from(rlnInstance, index, idCommsBufferPtr)
|
|
||||||
return membersAdded
|
|
||||||
|
|
||||||
proc removeMember*(rlnInstance: ptr RLN, index: MembershipIndex): bool =
|
proc removeMember*(rlnInstance: ptr RLN, index: MembershipIndex): bool =
|
||||||
let deletion_success = delete_member(rlnInstance, index)
|
let deletionSuccess = delete_member(rlnInstance, index)
|
||||||
return deletion_success
|
return deletionSuccess
|
||||||
|
|
||||||
proc removeMembers*(rlnInstance: ptr RLN, indices: seq[MembershipIndex]): bool =
|
proc removeMembers*(rlnInstance: ptr RLN, indices: seq[MembershipIndex]): bool =
|
||||||
for index in indices:
|
return atomicWrite(rlnInstance, idComms = @[], toRemoveIndices = indices)
|
||||||
let deletion_success = delete_member(rlnInstance, index)
|
|
||||||
if not deletion_success:
|
|
||||||
return false
|
|
||||||
return true
|
|
||||||
|
|
||||||
proc getMerkleRoot*(rlnInstance: ptr RLN): MerkleNodeResult =
|
proc getMerkleRoot*(rlnInstance: ptr RLN): MerkleNodeResult =
|
||||||
# read the Merkle Tree root after insertion
|
# read the Merkle Tree root after insertion
|
||||||
|
Loading…
x
Reference in New Issue
Block a user