2019-12-10 02:39:50 -05:00
|
|
|
require('dotenv').config()
|
|
|
|
const Web3 = require('web3')
|
2019-12-11 07:28:48 -05:00
|
|
|
const web3 = new Web3('https://mainnet.infura.io', null, { transactionConfirmationBlocks: 1 })
|
|
|
|
const web3Kovan = new Web3('https://kovan.infura.io/v3/c7463beadf2144e68646ff049917b716', null, { transactionConfirmationBlocks: 1 })
|
2019-12-10 02:39:50 -05:00
|
|
|
const account = web3.eth.accounts.privateKeyToAccount('0x' + process.env.PRIVATE_KEY)
|
|
|
|
web3.eth.accounts.wallet.add('0x' + process.env.PRIVATE_KEY)
|
|
|
|
web3.eth.defaultAccount = account.address
|
|
|
|
const ABI = [{ 'constant':false,'inputs':[{ 'name':'_newAccount','type':'address' }],'name':'changeOperator','outputs':[],'payable':false,'stateMutability':'nonpayable','type':'function' },{ 'constant':true,'inputs':[],'name':'filled_subtrees','outputs':[{ 'name':'','type':'uint256[]' }],'payable':false,'stateMutability':'view','type':'function' },{ 'constant':true,'inputs':[{ 'name':'','type':'uint256' }],'name':'nullifierHashes','outputs':[{ 'name':'','type':'bool' }],'payable':false,'stateMutability':'view','type':'function' },{ 'constant':true,'inputs':[],'name':'verifier','outputs':[{ 'name':'','type':'address' }],'payable':false,'stateMutability':'view','type':'function' },{ 'constant':true,'inputs':[],'name':'transferValue','outputs':[{ 'name':'','type':'uint256' }],'payable':false,'stateMutability':'view','type':'function' },{ 'constant':false,'inputs':[{ 'name':'_commitments','type':'uint256[]' },{ 'name':'_nullifierHashes','type':'uint256[]' }],'name':'migrateState','outputs':[],'payable':false,'stateMutability':'nonpayable','type':'function' },{ 'constant':true,'inputs':[],'name':'roots','outputs':[{ 'name':'','type':'uint256[]' }],'payable':false,'stateMutability':'view','type':'function' },{ 'constant':false,'inputs':[{ 'name':'_verifier','type':'address' },{ 'name':'_transferValue','type':'uint256' },{ 'name':'_merkleTreeHeight','type':'uint8' },{ 'name':'_emptyElement','type':'uint256' },{ 'name':'_operator','type':'address' },{ 'name':'_filled_subtrees','type':'uint256[]' },{ 'name':'_lastRoot','type':'uint256' }],'name':'initialize','outputs':[],'payable':false,'stateMutability':'nonpayable','type':'function' },{ 'constant':true,'inputs':[{ 'name':'','type':'uint256' }],'name':'commitments','outputs':[{ 'name':'','type':'bool' }],'payable':false,'stateMutability':'view','type':'function' },{ 'constant':true,'inputs':[],'name':'zeros','outputs':[{ 'name':'','type':'uint256[]' }],'payable':false,'stateMutability':'view','type':'function' },{ 'constant':true,'inputs':[],'name':'levels','outputs':[{ 'name':'','type':'uint256' }],'payable':false,'stateMutability':'view','type':'function' },{ 'constant':false,'inputs':[{ 'name':'a','type':'uint256[2]' },{ 'name':'b','type':'uint256[2][2]' },{ 'name':'c','type':'uint256[2]' },{ 'name':'input','type':'uint256[4]' }],'name':'withdraw','outputs':[],'payable':false,'stateMutability':'nonpayable','type':'function' },{ 'constant':true,'inputs':[],'name':'operator','outputs':[{ 'name':'','type':'address' }],'payable':false,'stateMutability':'view','type':'function' },{ 'constant':true,'inputs':[],'name':'isDepositsEnabled','outputs':[{ 'name':'','type':'bool' }],'payable':false,'stateMutability':'view','type':'function' },{ 'constant':true,'inputs':[{ 'name':'nullifier','type':'uint256' }],'name':'isSpent','outputs':[{ 'name':'','type':'bool' }],'payable':false,'stateMutability':'view','type':'function' },{ 'constant':true,'inputs':[{ 'name':'left','type':'uint256' },{ 'name':'right','type':'uint256' }],'name':'hashLeftRight','outputs':[{ 'name':'mimc_hash','type':'uint256' }],'payable':false,'stateMutability':'pure','type':'function' },{ 'constant':true,'inputs':[],'name':'next_index','outputs':[{ 'name':'','type':'uint32' }],'payable':false,'stateMutability':'view','type':'function' },{ 'constant':true,'inputs':[],'name':'current_root','outputs':[{ 'name':'','type':'uint256' }],'payable':false,'stateMutability':'view','type':'function' },{ 'constant':false,'inputs':[{ 'name':'tree_levels','type':'uint256' },{ 'name':'zero_value','type':'uint256' },{ 'name':'filled_subtrees','type':'uint256[]' },{ 'name':'lastRoot','type':'uint256' }],'name':'initialize','outputs':[],'payable':false,'stateMutability':'nonpayable','type':'function' },{ 'constant':true,'inputs':[{ 'name':'root','type':'uint256' }],'name':'isKnownRoot','outputs':[{ 'name':'','type':'bool' }],'payable':false,'stateMutability':'view','type':'function' },{ 'constant':false,'inputs':[{ 'name':'commitment','type':'uint256' }],'name':'deposit','outputs':[],'payable':true,'stateMutability':'payable','type':'f
|
|
|
|
const ABIv2 = require('./build/contracts/ETHMixer.json').abi
|
|
|
|
const snarkjs = require('snarkjs')
|
|
|
|
const bigInt = snarkjs.bigInt
|
|
|
|
|
|
|
|
const { numberToHex, toWei } = require('web3-utils')
|
|
|
|
const PREVIOUS_MIXER = '0xb541fc07bC7619fD4062A54d96268525cBC6FfEF'
|
|
|
|
|
|
|
|
function toHex(number, length = 32) {
|
|
|
|
let str = number instanceof Buffer ? number.toString('hex') : bigInt(number).toString(16)
|
|
|
|
return '0x' + str.padStart(length * 2, '0')
|
|
|
|
}
|
2019-12-11 07:28:48 -05:00
|
|
|
const previousMixer = new web3Kovan.eth.Contract(ABI, PREVIOUS_MIXER)
|
|
|
|
previousMixer.deployedBlock = 14070053
|
2019-12-10 02:39:50 -05:00
|
|
|
|
|
|
|
async function loadDeposits() {
|
|
|
|
const depositEvents = await previousMixer.getPastEvents('Deposit', { fromBlock: previousMixer.deployedBlock, toBlock: 'latest' })
|
2019-12-10 10:48:11 -05:00
|
|
|
console.log('deposits length', depositEvents.length)
|
2019-12-10 02:39:50 -05:00
|
|
|
const withdrawEvents = await previousMixer.getPastEvents('Withdraw', { fromBlock: previousMixer.deployedBlock, toBlock: 'latest' })
|
2019-12-10 10:48:11 -05:00
|
|
|
console.log('withdrawEvents length', withdrawEvents.length)
|
2019-12-10 02:39:50 -05:00
|
|
|
const commitments = depositEvents
|
|
|
|
.sort((a, b) => a.returnValues.leafIndex.sub(b.returnValues.leafIndex))
|
|
|
|
.map(e => toHex(e.returnValues.commitment))
|
|
|
|
const nullifiers = withdrawEvents
|
|
|
|
.map(e => toHex(e.returnValues.nullifierHash))
|
2019-12-11 07:28:48 -05:00
|
|
|
|
|
|
|
const { root, subtrees } = await getSubtrees({ commitments })
|
|
|
|
console.log(root, subtrees)
|
|
|
|
|
2019-12-10 02:39:50 -05:00
|
|
|
// console.log(JSON.stringify({ subtrees, lastRoot, commitments, nullifiers }, null, 2))
|
2019-12-11 07:28:48 -05:00
|
|
|
return { subtrees, lastRoot: toHex(root), commitments, nullifiers }
|
|
|
|
}
|
|
|
|
|
|
|
|
async function getSubtrees({ commitments }) {
|
|
|
|
const ganache = new Web3('http://localhost:8545', null, { transactionConfirmationBlocks: 1 })
|
|
|
|
const newContractAddress = '0x1D340b022c7d83608F247a7f6fb5d96FE3450DA3'
|
|
|
|
const tempMixer = new ganache.eth.Contract(ABIv2, newContractAddress)
|
|
|
|
for(let commitment of commitments) {
|
|
|
|
await tempMixer.methods.deposit(commitment).send({ value: toWei('0.1'), from: (await ganache.eth.getAccounts())[0], gas:5e6 })
|
|
|
|
process.stdout.write('.')
|
|
|
|
}
|
|
|
|
process.stdout.write('\n')
|
|
|
|
let subtrees = []
|
|
|
|
for (let i = 0; i < process.env.MERKLE_TREE_HEIGHT; i++) {
|
|
|
|
subtrees.push(await tempMixer.methods.filledSubtrees(i).call())
|
|
|
|
}
|
|
|
|
subtrees = subtrees.map(x => toHex(x))
|
|
|
|
const lastRoot = await tempMixer.methods.getLastRoot().call()
|
|
|
|
return { subtrees, root: toHex(lastRoot) }
|
2019-12-10 02:39:50 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
async function migrateState({ subtrees, lastRoot, commitments, nullifiers, newMixer }) {
|
|
|
|
const loadBy = 100
|
|
|
|
let commitmentsToLoad
|
|
|
|
let nullifiersToLoad
|
|
|
|
await newMixer.methods.initializeTreeForMigration(subtrees, lastRoot).send({
|
|
|
|
gas: numberToHex(2500000),
|
|
|
|
gasPrice: toHex(toWei('10', 'gwei')),
|
|
|
|
from: account.address
|
|
|
|
})
|
|
|
|
for(let i=0; i < commitments.length / loadBy; i++) {
|
|
|
|
commitmentsToLoad = commitments.slice(i*loadBy, (i+1)*loadBy)
|
|
|
|
nullifiersToLoad = nullifiers.slice(i*loadBy, (i+1)*loadBy)
|
|
|
|
console.log(`Uploading commitments and nullifiers from ${i*loadBy} to ${(i+1)*loadBy}:`)
|
|
|
|
// console.log('Commitments:\n', commitmentsToLoad)
|
|
|
|
// console.log('Nullifiers:\n', nullifiersToLoad)
|
|
|
|
|
|
|
|
const tx = await newMixer.methods.migrateState(
|
|
|
|
commitmentsToLoad,
|
|
|
|
nullifiersToLoad
|
|
|
|
).send({
|
|
|
|
gas: numberToHex(6500000),
|
|
|
|
gasPrice: toHex(toWei('10', 'gwei')),
|
|
|
|
from: account.address
|
|
|
|
})
|
|
|
|
console.log('Gas used:', tx.gasUsed)
|
|
|
|
}
|
2019-12-10 10:48:11 -05:00
|
|
|
const balance = await web3.eth.getBalance(PREVIOUS_MIXER)
|
|
|
|
console.log('balance to send in wei', balance)
|
|
|
|
|
|
|
|
const finish = await newMixer.methods.finishMigration().send({
|
|
|
|
gas: numberToHex(1500000),
|
|
|
|
gasPrice: toHex(toWei('10', 'gwei')),
|
|
|
|
from: account.address,
|
|
|
|
value: balance
|
|
|
|
})
|
|
|
|
console.log('migration completed', finish.transactionHash)
|
|
|
|
|
2019-12-10 02:39:50 -05:00
|
|
|
}
|
|
|
|
async function main() {
|
|
|
|
const { subtrees, lastRoot, commitments, nullifiers } = await loadDeposits()
|
2019-12-10 10:48:11 -05:00
|
|
|
const newContractAddress = '0x9715fC42892A8cE94e7ae58E332Cc4cDa102FfDF'
|
2019-12-10 02:39:50 -05:00
|
|
|
const newMixer = new web3Kovan.eth.Contract(ABIv2, newContractAddress)
|
|
|
|
web3Kovan.eth.accounts.wallet.add('0x' + process.env.PRIVATE_KEY)
|
|
|
|
web3Kovan.eth.defaultAccount = account.address
|
2019-12-10 10:48:11 -05:00
|
|
|
console.log('commitments length ', commitments.length)
|
|
|
|
console.log('nullifiers length ', nullifiers.length)
|
2019-12-10 02:39:50 -05:00
|
|
|
await migrateState({ subtrees, lastRoot, commitments, nullifiers, newMixer })
|
|
|
|
|
|
|
|
}
|
|
|
|
main()
|