const fs = require('fs') const Web3 = require('web3') const { toBN } = require('web3-utils') const MerkleTree = require('fixed-merkle-tree') const Redis = require('ioredis') const { GasPriceOracle } = require('gas-price-oracle') const tornadoABI = require('../abis/tornadoABI.json') const miningABI = require('../abis/mining.abi.json') const swapABI = require('../abis/swap.abi.json') const { queue } = require('./queue') const { poseidonHash2, jobType } = require('./utils') const { rpcUrl, redisUrl, privateKey, updateConfig, swapAddress, minerAddress } = require('../config') const { TxManager } = require('tx-manager') const { Controller } = require('tornado-cash-anonymity-mining') let web3 let currentTx let currentJob let tree let txManager let controller const redis = new Redis(redisUrl) const redisSubscribe = new Redis(redisUrl) const gasPriceOracle = new GasPriceOracle({ defaultRpc: rpcUrl }) const status = Object.freeze({ ACCEPTED: 'ACCEPTED', SENT: 'SENT', MINED: 'MINED', CONFIRMED: 'CONFIRMED', }) async function fetchTree() { console.log('got tree update') const elements = await redis.get('tree:elements') const convert = (_, val) => (typeof val === 'string' ? toBN(val) : val) tree = MerkleTree.deserialize(JSON.parse(elements, convert), poseidonHash2) if (currentTx && currentJob && ['miningReward', 'miningWithdraw'].includes(currentJob.data.type)) { const { proof, args } = currentJob.data if (toBN(args.account.inputRoot).eq(toBN(tree.root()))) { return } const update = await controller.treeUpdate(args.account.outputCommitment, tree) const instance = new web3.eth.Contract(miningABI, minerAddress) const data = currentJob.data.type === 'miningReward' ? instance.methods.reward(proof, args, update.proof, update.args).encodeABI() : instance.methods.withdraw(proof, args, update.proof, update.args).encodeABI() currentTx = await currentTx.replace({ to: minerAddress, data, }) console.log('replaced pending tx') } } async function start() { web3 = new Web3(rpcUrl) txManager = new TxManager({ privateKey, rpcUrl }) updateConfig({ rewardAccount: txManager.address }) redisSubscribe.subscribe('treeUpdate', fetchTree) await fetchTree() const provingKeys = { treeUpdateCircuit: require('../keys/TreeUpdate.json'), treeUpdateProvingKey: fs.readFileSync('./keys/TreeUpdate_proving_key.bin').buffer, } controller = new Controller({ provingKeys }) await controller.init() queue.process(process) console.log('Worker started') } function checkFee({ data, type }) { if (type === jobType.TORNADO_WITHDRAW) { return checkTornadoFee(data) } return checkMiningFee(data) } async function checkTornadoFee({ args, contract }) { console.log('args, contract', args, contract) const { fast } = await gasPriceOracle.gasPrices() console.log('fast', fast) } async function checkMiningFee({ args }) { const swap = new web3.eth.Contract(swapABI, swapAddress) const TornAmount = await swap.methods.getExpectedReturn(args.fee).call() console.log('TornAmount', TornAmount) // todo: use desired torn/eth rate and compute the same way as in tornado } // may be this looks better // const isTornadoWithdraw = type === jobType.TORNADO_WITHDRAW // const ABI = isTornadoWithdraw ? tornadoABI : miningABI // const contractAddress = isTornadoWithdraw ? data.contract : minerAddress // const value = isTornadoWithdraw ? data.args[5] : 0 // refund function getTxObject({ data, type }) { let ABI, contractAddress, value = type === jobType.TORNADO_WITHDRAW ? [tornadoABI, data.contract, data.args[5]] : [miningABI, minerAddress, 0] const method = type !== jobType.MINING_REWARD ? 'withdraw' : 'reward' const contract = new web3.eth.Contract(ABI, contractAddress) const calldata = contract.methods[method](data.proof, ...data.args).encodeABI() return { value, to: contract, data: calldata, } } async function process(job) { if (!jobType[job.data.type]) { throw new Error(`Unknown job type: ${job.data.type}`) } await updateStatus(status.ACCEPTED) currentJob = job console.log(`Start processing a new ${job.data.type} job #${job.id}`) await checkFee(job) if (job.data.type !== jobType.TORNADO_WITHDRAW) { // precheck if root is up to date } currentTx = await txManager.createTx(getTxObject(job)) try { await currentTx .send() .on('transactionHash', txHash => { updateTxHash(txHash) updateStatus(status.SENT) }) .on('mined', receipt => { console.log('Mined in block', receipt.blockNumber) updateStatus(status.MINED) }) .on('confirmations', updateConfirmations) await updateStatus(status.CONFIRMED) } catch (e) { console.error('Revert', e) throw new Error(`Revert by smart contract ${e.message}`) } } async function updateTxHash(txHash) { console.log(`A new successfully sent tx ${txHash}`) currentJob.data.txHash = txHash await currentJob.update(currentJob.data) } async function updateConfirmations(confirmations) { console.log(`Confirmations count ${confirmations}`) currentJob.data.confirmations = confirmations await currentJob.update(currentJob.data) } async function updateStatus(status) { console.log(`Job status updated ${status}`) currentJob.data.status = status await currentJob.update(currentJob.data) } module.exports = { start, process }