165 lines
9.9 KiB
Nim
Raw Normal View History

# this module contains the Nim wrappers for the rln library https://github.com/kilic/rln/blob/3bbec368a4adc68cd5f9bfae80b17e1bbb4ef373/src/ffi.rs
when (NimMajor, NimMinor) < (1, 4):
{.push raises: [Defect].}
else:
{.push raises: [].}
import
os,
waku_rln_relay_types
const libPath = "vendor/zerokit/target/release/"
when defined(Windows):
const libName* = libPath / "rln.dll"
elif defined(Linux):
const libName* = libPath / "librln.so"
elif defined(MacOsX):
const libName* = libPath / "librln.dylib"
# all the following procedures are Nim wrappers for the functions defined in libName
Rln-relay zkp module Nim bindings (#427) * entirely replaces the prior rln header, the var variables are changed to ptr * updates the unittest of key_gen * adds test for update_next_member * updates membershipKeyGen internals and prototype * adds createRLNInstance * adds helpers methods * adds generateKeyPairBuffer * cleans up the test and adds comments * renames merkleTreeDepth to d * fixes a buf re decoding the keys into sk and pk * adds getSKPK proc * unifies key gen helper procs, adds todos * comments out the createRLNInstance * refactors the code based on the updated createRLNInstance interface * adds the test for the verify proc * fixes a variable name and replaces random key gen with the real key gen * tests a simple hash * adds get_root method * fixes the data pointer issue and adds the proof breakdown * adds rln * adds unit tests for Merkle tree * adds a sample hash test * fixes the hash bug and comments out unused part of proof gen test * cleans up the proof gent test * replaces unsafeAddr with addr * fixes an issue in key gen * updates rln submodule * fixes the verification problem * adds a failed test * replaces an old test scenario with a new one * handles createRLNInstance output * working createRLNInstance2 * refactors the code by replacing the old createRLNInstance * renames createRLNInstance2 * adds documentation and reorganizes rln.nim * replace echo with debug, renames vars, adds a bad proof test * minor * minor * edits var names * adds one more check * adds one more test to the hash * enforcing exception handling * adds pacman -Sy * removes update:true * activates update
2021-03-31 17:39:27 -07:00
{.push dynlib: libName, raises: [Defect].}
## Buffer struct is taken from
# https://github.com/celo-org/celo-threshold-bls-rs/blob/master/crates/threshold-bls-ffi/src/ffi.rs
type Buffer* = object
`ptr`*: ptr uint8
len*: uint
######################################################################
## RLN Zerokit module APIs
######################################################################
#------------------------------ Merkle Tree operations -----------------------------------------
proc update_next_member*(ctx: ptr RLN, input_buffer: ptr Buffer): bool {.importc: "set_next_leaf".}
## adds an element in the merkle tree to the next available position
## input_buffer points to the id commitment byte seq
## the return bool value indicates the success or failure of the operation
proc delete_member*(ctx: ptr RLN, index: uint): bool {.importc: "delete_leaf".}
## index is the position of the id commitment key to be deleted from the tree
## the deleted id commitment key is replaced with a zero leaf
## the return bool value indicates the success or failure of the operation
proc get_root*(ctx: ptr RLN, output_buffer: ptr Buffer): bool {.importc: "get_root".}
## get_root populates the passed pointer output_buffer with the current tree root
## the output_buffer holds the Merkle tree root of size 32 bytes
## the return bool value indicates the success or failure of the operation
proc get_merkle_proof*(ctx: ptr RLN, index: uint, output_buffer: ptr Buffer): bool {.importc: "get_proof".}
## populates the passed pointer output_buffer with the merkle proof for the leaf at position index in the tree stored by ctx
## the output_buffer holds a serialized Merkle proof (vector of 32 bytes nodes)
## the return bool value indicates the success or failure of the operation
proc set_leaf*(ctx: ptr RLN, index: uint, input_buffer: ptr Buffer): bool {.importc: "set_leaf".}
## sets the leaf at position index in the tree stored by ctx to the value passed by input_buffer
## the input_buffer holds a serialized leaf of 32 bytes
## the return bool value indicates the success or failure of the operation
feat(rln-relay): process blocks atomically (#1349) * test(rln-relay): atomic block processing * fix(rln-relay): use correct starting index * fix(rln-relay): args * fix(rln-relay): append length * fix(rln-relay): tests, remove insertMember * fix(rln-relay): camelCase, cleanup * fix(rln-relay): actually process per block * fix(rln-relay): clean up * chore(gitignore): Update .gitignore Co-authored-by: Lorenzo Delgado <lorenzo@status.im> * Update waku/v2/protocol/waku_rln_relay/waku_rln_relay_utils.nim Co-authored-by: Sanaz Taheri <35961250+staheri14@users.noreply.github.com> * Update waku/v2/protocol/waku_rln_relay/waku_rln_relay_utils.nim Co-authored-by: Sanaz Taheri <35961250+staheri14@users.noreply.github.com> * fix(rln-relay): args * fix(rln-relay): add prefix def * fix(rln-relay): make test cleaner * chore(rln-relay): apply suggestions Co-authored-by: G. <28568419+s1fr0@users.noreply.github.com> Co-authored-by: Lorenzo Delgado <lorenzo@status.im> * chore(rln-relay): add member order check * test(rln-relay): batch insert in tests * fix(rln-relay): test batching * fix(rln-relay): toSeq the HSlice * fix(rln-relay): naming * fix(rln-relay): add insertMember back * fix(rln-relay): serialize util, address review * fix(rln-relay): add atomicity desc * fix(rln-relay): inHex * fix(rln-relay): explicit proc def * fix(rln-relay): indexGap condition * fix(rln-relay): func sig * fix(rln-relay): onchain test * fix(rln-relay): use asyncSpawn vs asyncCheck * fix(rln-relay): do not explicitly insert into the index * fix(rln-relay): condition, semantics * fix(rln-relay): index must be 1 * chore(rln-relay): line br * fix(rln-relay): missing return ok(true) Co-authored-by: Lorenzo Delgado <lorenzo@status.im> Co-authored-by: Sanaz Taheri <35961250+staheri14@users.noreply.github.com> Co-authored-by: G. <28568419+s1fr0@users.noreply.github.com>
2022-11-10 22:28:31 +05:30
proc init_tree_with_leaves*(ctx: ptr RLN, input_buffer: ptr Buffer): bool {.importc: "init_tree_with_leaves".}
## sets multiple leaves 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)
feat(rln-relay): process blocks atomically (#1349) * test(rln-relay): atomic block processing * fix(rln-relay): use correct starting index * fix(rln-relay): args * fix(rln-relay): append length * fix(rln-relay): tests, remove insertMember * fix(rln-relay): camelCase, cleanup * fix(rln-relay): actually process per block * fix(rln-relay): clean up * chore(gitignore): Update .gitignore Co-authored-by: Lorenzo Delgado <lorenzo@status.im> * Update waku/v2/protocol/waku_rln_relay/waku_rln_relay_utils.nim Co-authored-by: Sanaz Taheri <35961250+staheri14@users.noreply.github.com> * Update waku/v2/protocol/waku_rln_relay/waku_rln_relay_utils.nim Co-authored-by: Sanaz Taheri <35961250+staheri14@users.noreply.github.com> * fix(rln-relay): args * fix(rln-relay): add prefix def * fix(rln-relay): make test cleaner * chore(rln-relay): apply suggestions Co-authored-by: G. <28568419+s1fr0@users.noreply.github.com> Co-authored-by: Lorenzo Delgado <lorenzo@status.im> * chore(rln-relay): add member order check * test(rln-relay): batch insert in tests * fix(rln-relay): test batching * fix(rln-relay): toSeq the HSlice * fix(rln-relay): naming * fix(rln-relay): add insertMember back * fix(rln-relay): serialize util, address review * fix(rln-relay): add atomicity desc * fix(rln-relay): inHex * fix(rln-relay): explicit proc def * fix(rln-relay): indexGap condition * fix(rln-relay): func sig * fix(rln-relay): onchain test * fix(rln-relay): use asyncSpawn vs asyncCheck * fix(rln-relay): do not explicitly insert into the index * fix(rln-relay): condition, semantics * fix(rln-relay): index must be 1 * chore(rln-relay): line br * fix(rln-relay): missing return ok(true) Co-authored-by: Lorenzo Delgado <lorenzo@status.im> Co-authored-by: Sanaz Taheri <35961250+staheri14@users.noreply.github.com> Co-authored-by: G. <28568419+s1fr0@users.noreply.github.com>
2022-11-10 22:28:31 +05:30
## the input_buffer size is prefixed by a 8 bytes integer indicating the number of leaves
## leaves are set one after each other starting from index 0
## the return bool value indicates the success or failure of the operation
feat(rln-relay): process blocks atomically (#1349) * test(rln-relay): atomic block processing * fix(rln-relay): use correct starting index * fix(rln-relay): args * fix(rln-relay): append length * fix(rln-relay): tests, remove insertMember * fix(rln-relay): camelCase, cleanup * fix(rln-relay): actually process per block * fix(rln-relay): clean up * chore(gitignore): Update .gitignore Co-authored-by: Lorenzo Delgado <lorenzo@status.im> * Update waku/v2/protocol/waku_rln_relay/waku_rln_relay_utils.nim Co-authored-by: Sanaz Taheri <35961250+staheri14@users.noreply.github.com> * Update waku/v2/protocol/waku_rln_relay/waku_rln_relay_utils.nim Co-authored-by: Sanaz Taheri <35961250+staheri14@users.noreply.github.com> * fix(rln-relay): args * fix(rln-relay): add prefix def * fix(rln-relay): make test cleaner * chore(rln-relay): apply suggestions Co-authored-by: G. <28568419+s1fr0@users.noreply.github.com> Co-authored-by: Lorenzo Delgado <lorenzo@status.im> * chore(rln-relay): add member order check * test(rln-relay): batch insert in tests * fix(rln-relay): test batching * fix(rln-relay): toSeq the HSlice * fix(rln-relay): naming * fix(rln-relay): add insertMember back * fix(rln-relay): serialize util, address review * fix(rln-relay): add atomicity desc * fix(rln-relay): inHex * fix(rln-relay): explicit proc def * fix(rln-relay): indexGap condition * fix(rln-relay): func sig * fix(rln-relay): onchain test * fix(rln-relay): use asyncSpawn vs asyncCheck * fix(rln-relay): do not explicitly insert into the index * fix(rln-relay): condition, semantics * fix(rln-relay): index must be 1 * chore(rln-relay): line br * fix(rln-relay): missing return ok(true) Co-authored-by: Lorenzo Delgado <lorenzo@status.im> Co-authored-by: Sanaz Taheri <35961250+staheri14@users.noreply.github.com> Co-authored-by: G. <28568419+s1fr0@users.noreply.github.com>
2022-11-10 22:28:31 +05:30
proc set_leaves_from*(ctx: ptr RLN, index: uint, input_buffer: ptr Buffer): bool {.importc: "set_leaves_from".}
## sets multiple leaves 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 input_buffer size is prefixed by a 8 bytes integer indicating the number of leaves
## leaves are set one after each other starting from index `index`
## the return bool value indicates the success or failure of the operation
proc reset_tree*(ctx: ptr RLN, tree_height: uint): bool {.importc: "set_tree".}
## resets the tree stored by ctx to the the empty tree (all leaves set to 0) of height tree_height
## the return bool value indicates the success or failure of the operation
#----------------------------------------------------------------------------------------------
#-------------------------------- zkSNARKs operations -----------------------------------------
proc key_gen*(ctx: ptr RLN, output_buffer: ptr Buffer): bool {.importc: "key_gen".}
## generates id key and id commitment key serialized inside output_buffer as | id_key <32 bytes>| id_commitment_key <32 bytes> |
## id commitment is the poseidon hash of the id key
## the return bool value indicates the success or failure of the operation
proc seeded_key_gen*(ctx: ptr RLN, input_buffer: ptr Buffer, output_buffer: ptr Buffer): bool {.importc: "seeded_key_gen".}
## generates id key and id commitment key serialized inside output_buffer as | id_key <32 bytes>| id_commitment_key <32 bytes> | using ChaCha20
## seeded with an arbitrary long seed serialized in input_buffer
## The input seed provided by the user is hashed using Keccak256 before being passed to ChaCha20 as seed.
## id commitment is the poseidon hash of the id key
## the return bool value indicates the success or failure of the operation
proc generate_proof*(ctx: ptr RLN,
input_buffer: ptr Buffer,
output_buffer: ptr Buffer): bool {.importc: "generate_rln_proof".}
## input_buffer has to be serialized as [ id_key<32> | id_index<8> | epoch<32> | signal_len<8> | signal<var> ]
## output_buffer holds the proof data and should be parsed as [ proof<128> | root<32> | epoch<32> | share_x<32> | share_y<32> | nullifier<32> | rln_identifier<32> ]
## integers wrapped in <> indicate value sizes in bytes
## the return bool value indicates the success or failure of the operation
proc verify*(ctx: ptr RLN,
proof_buffer: ptr Buffer,
proof_is_valid_ptr: ptr bool): bool {.importc: "verify_rln_proof".}
## proof_buffer has to be serialized as [ proof<128> | root<32> | epoch<32> | share_x<32> | share_y<32> | nullifier<32> | rln_identifier<32> | signal_len<8> | signal<var> ]
## the return bool value indicates the success or failure of the call to the verify function
## the verification of the zk proof is available in proof_is_valid_ptr, where a value of true indicates success and false a failure
proc verify_with_roots*(ctx: ptr RLN,
proof_buffer: ptr Buffer,
roots_buffer: ptr Buffer,
proof_is_valid_ptr: ptr bool): bool {.importc: "verify_with_roots".}
## proof_buffer has to be serialized as [ proof<128> | root<32> | epoch<32> | share_x<32> | share_y<32> | nullifier<32> | rln_identifier<32> | signal_len<8> | signal<var> ]
## roots_buffer contains the concatenation of 32 bytes long serializations in little endian of root values
## the return bool value indicates the success or failure of the call to the verify function
## the verification of the zk proof is available in proof_is_valid_ptr, where a value of true indicates success and false a failure
proc zk_prove*(ctx: ptr RLN,
input_buffer: ptr Buffer,
output_buffer: ptr Buffer): bool {.importc: "prove".}
## Computes the zkSNARK proof and stores it in output_buffer for input values stored in input_buffer
## input_buffer is serialized as input_data as [ id_key<32> | path_elements<Vec<32>> | identity_path_index<Vec<1>> | x<32> | epoch<32> | rln_identifier<32> ]
## output_buffer holds the proof data and should be parsed as [ proof<128> ]
## path_elements and indentity_path elements serialize a merkle proof for id_key and are vectors of elements of 32 and 1 bytes, respectively (not. Vec<>).
## x is the x coordinate of the Shamir's secret share for which the proof is computed
## epoch is the input epoch (equivalently, the nullifier)
## the return bool value indicates the success or failure of the operation
proc zk_verify*(ctx: ptr RLN,
proof_buffer: ptr Buffer,
proof_is_valid_ptr: ptr bool): bool {.importc: "verify".}
## Verifies the zkSNARK proof passed in proof_buffer
## input_buffer is serialized as input_data as [ proof<128> ]
## the verification of the zk proof is available in proof_is_valid_ptr, where a value of true indicates success and false a failure
## the return bool value indicates the success or failure of the operation
#----------------------------------------------------------------------------------------------
#-------------------------------- Common procedures -------------------------------------------
proc new_circuit*(tree_height: uint, input_buffer: ptr Buffer, ctx: ptr (ptr RLN)): bool {.importc: "new".}
## creates an instance of rln object as defined by the zerokit RLN lib
## tree_height represent the depth of the Merkle tree
## input_buffer contains a serialization of the path where the circuit resources can be found (.r1cs, .wasm, .zkey and optionally the verification_key.json)
## ctx holds the final created rln object
## the return bool value indicates the success or failure of the operation
proc new_circuit_from_data*(tree_height: uint, circom_buffer: ptr Buffer, zkey_buffer: ptr Buffer, vk_buffer: ptr Buffer, ctx: ptr (ptr RLN)): bool {.importc: "new_with_params".}
## creates an instance of rln object as defined by the zerokit RLN lib by passing the required inputs as byte arrays
## tree_height represent the depth of the Merkle tree
## circom_buffer contains the bytes read from the Circom .wasm circuit
## zkey_buffer contains the bytes read from the .zkey proving key
## vk_buffer contains the bytes read from the verification_key.json
## ctx holds the final created rln object
## the return bool value indicates the success or failure of the operation
proc hash*(ctx: ptr RLN,
input_buffer: ptr Buffer,
output_buffer: ptr Buffer): bool {.importc: "hash".}
## it hashes (sha256) the plain text supplied in inputs_buffer and then maps it to a field element
## this proc is used to map arbitrary signals to field element for the sake of proof generation
## inputs_buffer holds the hash input as a byte seq
## the hash output is generated and populated inside output_buffer
## the output_buffer contains 32 bytes hash output
feat(rln-relay): process blocks atomically (#1349) * test(rln-relay): atomic block processing * fix(rln-relay): use correct starting index * fix(rln-relay): args * fix(rln-relay): append length * fix(rln-relay): tests, remove insertMember * fix(rln-relay): camelCase, cleanup * fix(rln-relay): actually process per block * fix(rln-relay): clean up * chore(gitignore): Update .gitignore Co-authored-by: Lorenzo Delgado <lorenzo@status.im> * Update waku/v2/protocol/waku_rln_relay/waku_rln_relay_utils.nim Co-authored-by: Sanaz Taheri <35961250+staheri14@users.noreply.github.com> * Update waku/v2/protocol/waku_rln_relay/waku_rln_relay_utils.nim Co-authored-by: Sanaz Taheri <35961250+staheri14@users.noreply.github.com> * fix(rln-relay): args * fix(rln-relay): add prefix def * fix(rln-relay): make test cleaner * chore(rln-relay): apply suggestions Co-authored-by: G. <28568419+s1fr0@users.noreply.github.com> Co-authored-by: Lorenzo Delgado <lorenzo@status.im> * chore(rln-relay): add member order check * test(rln-relay): batch insert in tests * fix(rln-relay): test batching * fix(rln-relay): toSeq the HSlice * fix(rln-relay): naming * fix(rln-relay): add insertMember back * fix(rln-relay): serialize util, address review * fix(rln-relay): add atomicity desc * fix(rln-relay): inHex * fix(rln-relay): explicit proc def * fix(rln-relay): indexGap condition * fix(rln-relay): func sig * fix(rln-relay): onchain test * fix(rln-relay): use asyncSpawn vs asyncCheck * fix(rln-relay): do not explicitly insert into the index * fix(rln-relay): condition, semantics * fix(rln-relay): index must be 1 * chore(rln-relay): line br * fix(rln-relay): missing return ok(true) Co-authored-by: Lorenzo Delgado <lorenzo@status.im> Co-authored-by: Sanaz Taheri <35961250+staheri14@users.noreply.github.com> Co-authored-by: G. <28568419+s1fr0@users.noreply.github.com>
2022-11-10 22:28:31 +05:30
{.pop.}