vanity-eth/js/vanity.js
2018-01-16 20:35:22 +01:00

84 lines
1.9 KiB
JavaScript

/* eslint-env worker */
const ethUtils = require('ethereumjs-util');
const randomBytes = require('randombytes');
const step = 500;
/**
* Create a wallet from a random private key
* @returns {{address: string, privKey: string}}
*/
const getRandomWallet = () => {
const randbytes = randomBytes(32);
return {
address: '0x' + ethUtils.privateToAddress(randbytes).toString('hex'),
privKey: randbytes.toString('hex')
};
};
/**
* Check if a wallet respects the input constraints
* @param wallet
* @param input
* @param isChecksum
* @returns {boolean}
*/
const isValidVanityWallet = (wallet, input, isChecksum) => {
if (input !== wallet.address.substr(2, input.length)) {
return false;
}
if (!isChecksum) {
return true;
}
const address = wallet.address.substr(2);
const hash = ethUtils.sha3(address).toString('hex');
for (let i = 0; i < input.length; i++) {
if (parseInt(hash[i], 16) >= 8) {
if (address[i].toUpperCase() !== input[i]) {
return false;
}
} else if (address[i] !== input[i]) {
return false;
}
}
return true;
};
/**
* Generate a lot of wallets until one satisfies the input constraints
* @param input
* @param isChecksum
* @param cb - Callback called after x attempts, or when an address if found
* @returns
*/
const getVanityWallet = (input, isChecksum, cb) => {
input = isChecksum ? input : input.toLowerCase();
let wallet = getRandomWallet();
let attempts = 1;
while (!isValidVanityWallet(wallet, input, isChecksum)) {
if (attempts >= step) {
cb({attempts});
attempts = 0;
}
wallet = getRandomWallet();
attempts++;
}
cb({address: ethUtils.toChecksumAddress(wallet.address), privKey: wallet.privKey, attempts});
};
onmessage = function (event) {
const input = event.data;
try {
getVanityWallet(input.prefix, input.checksum, message => postMessage(message));
} catch (err) {
postMessage({error: err.toString()});
}
};
module.exports = {
onmessage
};