const Scalar = require("ffjavascript").Scalar;
|
|
const ZqField = require("ffjavascript").ZqField;
|
|
|
|
const Web3Utils = require("web3-utils");
|
|
const F = new ZqField(Scalar.fromString("21888242871839275222246405745257275088548364400416034343698204186575808495617"));
|
|
exports.F = F;
|
|
|
|
const SEED = "mimc";
|
|
const NROUNDS = 91;
|
|
|
|
exports.getIV = (seed) => {
|
|
if (typeof seed === "undefined") seed = SEED;
|
|
const c = Web3Utils.keccak256(seed+"_iv");
|
|
const cn = Scalar.FromString(Web3Utils.toBN(c).toString());
|
|
const iv = cn.mod(F.p);
|
|
return iv;
|
|
};
|
|
|
|
exports.getConstants = (seed, nRounds) => {
|
|
if (typeof seed === "undefined") seed = SEED;
|
|
if (typeof nRounds === "undefined") nRounds = NROUNDS;
|
|
const cts = new Array(nRounds);
|
|
let c = Web3Utils.keccak256(SEED);
|
|
for (let i=1; i<nRounds; i++) {
|
|
c = Web3Utils.keccak256(c);
|
|
|
|
const n1 = Web3Utils.toBN(c).mod(Web3Utils.toBN(F.p.toString()));
|
|
const c2 = Web3Utils.padLeft(Web3Utils.toHex(n1), 64);
|
|
cts[i] = Scalar.fromString(Web3Utils.toBN(c2).toString());
|
|
}
|
|
cts[0] = F.e(0);
|
|
return cts;
|
|
};
|
|
|
|
const cts = exports.getConstants(SEED, 91);
|
|
|
|
exports.hash = (_x_in, _k) =>{
|
|
const x_in = F.e(_x_in);
|
|
const k = F.e(_k);
|
|
let r;
|
|
for (let i=0; i<NROUNDS; i++) {
|
|
const c = cts[i];
|
|
const t = (i==0) ? F.add(x_in, k) : F.add(F.add(r, k), c);
|
|
r = F.pow(t, 7);
|
|
}
|
|
return F.add(r, k);
|
|
};
|
|
|
|
exports.multiHash = (arr, key) => {
|
|
let r;
|
|
if (typeof(key) === "undefined") {
|
|
r = F.zero;
|
|
} else {
|
|
r = key;
|
|
}
|
|
for (let i=0; i<arr.length; i++) {
|
|
r = F.add(
|
|
F.add(
|
|
r,
|
|
arr[i]
|
|
),
|
|
exports.hash(F.e(arr[i]), r)
|
|
);
|
|
}
|
|
return r;
|
|
};
|