const chai = require("chai"); const assert = chai.assert; const fs = require("fs"); var tmp = require("tmp-promise"); const path = require("path"); const compiler = require("./compiler"); const util = require("util"); const exec = util.promisify(require("child_process").exec); const stringifyBigInts = require("./utils").stringifyBigInts; const unstringifyBigInts = require("./utils").unstringifyBigInts; const bigInt = require("big-integer"); const utils = require("./utils"); const loadR1cs = require("./r1csfile").loadR1cs; const ZqField = require("fflib").ZqField; module.exports = c_tester; async function c_tester(circomFile, _options) { tmp.setGracefulCleanup(); const dir = await tmp.dir({prefix: "circom_", unsafeCleanup: true }); const baseName = path.basename(circomFile, ".circom"); const options = Object.assign({}, _options); options.cSourceWriteStream = fs.createWriteStream(path.join(dir.path, baseName + ".cpp")); options.symWriteStream = fs.createWriteStream(path.join(dir.path, baseName + ".sym")); options.r1csFileName = path.join(dir.path, baseName + ".r1cs"); await compiler(circomFile, options); const cdir = path.join(__dirname, "..", "c"); await exec("cp" + ` ${path.join(dir.path, baseName + ".cpp")}` + " /tmp/circuit.cpp" ); await exec("g++" + ` ${path.join(cdir, "main.cpp")}` + ` ${path.join(cdir, "calcwit.cpp")}` + ` ${path.join(cdir, "utils.cpp")}` + ` ${path.join(cdir, "zqfield.cpp")}` + ` ${path.join(dir.path, baseName + ".cpp")} ` + ` -o ${path.join(dir.path, baseName)}` + ` -I ${cdir}` + " -lgmp -std=c++11 -DSANITY_CHECK" ); // console.log(dir.path); return new CTester(dir, baseName); } class CTester { constructor(dir, baseName) { this.dir=dir; this.baseName = baseName; } async release() { await this.dir.cleanup(); } async calculateWitness(input) { await fs.promises.writeFile( path.join(this.dir.path, "in.json"), JSON.stringify(stringifyBigInts(input), null, 1) ); await exec(`${path.join(this.dir.path, this.baseName)}` + ` ${path.join(this.dir.path, "in.json")}` + ` ${path.join(this.dir.path, "out.json")}` ); const resStr = await fs.promises.readFile( path.join(this.dir.path, "out.json") ); const res = unstringifyBigInts(JSON.parse(resStr)); return res; } async loadSymbols() { if (this.symbols) return; this.symbols = {}; const symsStr = await fs.promises.readFile( path.join(this.dir.path, this.baseName + ".sym"), "utf8" ); const lines = symsStr.split("\n"); for (let i=0; i ${v}`); } return lines.join("\n"); } async checkConstraints(witness) { const self = this; if (!self.constraints) await self.loadConstraints(); for (let i=0; i