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:
Aaryamann Challani 2023-05-18 10:42:08 +05:30 committed by GitHub
parent 665484c17b
commit 611e9539a6
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
8 changed files with 111 additions and 105 deletions

View File

@ -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

@ -1 +1 @@
Subproject commit 584c2cf4c000b391ca6b415c09d8399fde329e5c Subproject commit c2d386cb749f551541bb34c4386a3849485356f9

View File

@ -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)

View File

@ -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.} =

View File

@ -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)

View File

@ -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

View File

@ -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

View File

@ -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