|
|
const Verifier = artifacts.require("../../contracts/Verifier"); const Miksi = artifacts.require("../../contracts/Miksi.sol");
const chai = require("chai"); const expect = chai.expect; const truffleAssert = require('truffle-assertions');
const fs = require("fs"); const { groth } = require('snarkjs'); const { stringifyBigInts, unstringifyBigInts } = require('ffjavascript').utils; const WitnessCalculatorBuilder = require("circom_runtime").WitnessCalculatorBuilder; const circomlib = require("circomlib");
contract("miksi", (accounts) => {
const { 0: owner, 1: addr1, // used for the deposit
2: addr2, // used for the withdraw
3: addr3, } = accounts;
let insVerifier; let insMiksi;
before(async () => {
insVerifier = await Verifier.new(); insMiksi = await Miksi.new(insVerifier.address); });
it("miksi flow", async () => { const secret = "123456789";
const coinCode = "0"; // refearing to ETH
const ethAmount = '0.5'; const amount = web3.utils.toWei(ethAmount, 'ether');
let balance_wei = await web3.eth.getBalance(addr1); console.log("Balance at " + addr1, web3.utils.fromWei(balance_wei, 'ether')); expect(balance_wei).to.be.equal('100000000000000000000');
const poseidon = circomlib.poseidon.createHash(6, 8, 57); const commitment = poseidon([coinCode, amount, secret]).toString();
// deposit
console.log("Deposit of " + ethAmount + " ETH from " + addr1); await insMiksi.deposit(coinCode, commitment, {from: addr1, value: amount});
balance_wei = await web3.eth.getBalance(addr1); console.log("Balance at " + addr1, web3.utils.fromWei(balance_wei, 'ether')); expect(balance_wei).to.be.equal('99499107180000000000');
// getDeposit data
const res = await insMiksi.getDeposit(commitment); expect(res[0].toString()).to.be.equal(coinCode); expect(res[1].toString()).to.be.equal(amount);
// calculate witness
const wasm = await fs.promises.readFile("./build/withdraw.wasm"); const input = unstringifyBigInts({ "coinCode": coinCode, "amount": amount, "commitment": commitment, "secret": secret, "address": addr2 }); const options = {}; console.log("Calculate witness"); const wc = await WitnessCalculatorBuilder(wasm, options); const w = await wc.calculateWitness(input); const witness = unstringifyBigInts(stringifyBigInts(w));
// generate zkproof of commitment using snarkjs (as is a test)
const provingKey = unstringifyBigInts(JSON.parse(fs.readFileSync("./build/proving_key.json", "utf8")));
console.log("Generate zkSNARK proof"); const {proof, publicSignals} = groth.genProof(provingKey, witness);
// withdraw
console.log("Withdraw of " + ethAmount + " ETH to " + addr2); let resW = await insMiksi.withdraw( commitment, addr2, [proof.pi_a[0].toString(), proof.pi_a[1].toString()], [ [proof.pi_b[0][1].toString(), proof.pi_b[0][0].toString()], [proof.pi_b[1][1].toString(), proof.pi_b[1][0].toString()] ], [proof.pi_c[0].toString(), proof.pi_c[1].toString()] ); // console.log("resW", resW);
balance_wei = await web3.eth.getBalance(addr2); console.log("Balance at " + addr2, web3.utils.fromWei(balance_wei, 'ether')); expect(balance_wei).to.be.equal('100500000000000000000');
console.log("Try to reuse the zkproof and expect revert"); await truffleAssert.fails( insMiksi.withdraw( commitment, addr2, [proof.pi_a[0].toString(), proof.pi_a[1].toString()], [ [proof.pi_b[0][1].toString(), proof.pi_b[0][0].toString()], [proof.pi_b[1][1].toString(), proof.pi_b[1][0].toString()] ], [proof.pi_c[0].toString(), proof.pi_c[1].toString()] ), truffleAssert.ErrorType.REVERT, "deposit already withdrawed" ); balance_wei = await web3.eth.getBalance(addr2); expect(balance_wei).to.be.equal('100500000000000000000'); }); });
|