vanity-eth/js/vanity.js
2017-12-26 19:46:19 +01:00

96 lines
2.3 KiB
JavaScript

const ethUtils = require('ethereumjs-util');
const randomBytes = require('randombytes');
const ERRORS = {
invalidHex: 'Invalid hex input'
};
/**
* 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 string is valid hexadecimal
* @param hex
* @returns {boolean}
*/
const isValidHex = hex => hex.length ? /^[0-9A-F]+$/g.test(hex.toUpperCase()) : true;
/**
* Check if a wallet respects the input constraints
* @param wallet
* @param input
* @param isChecksum
* @returns {boolean}
*/
const isValidVanityWallet = (wallet, input, isChecksum) => {
if (!isChecksum) {
return wallet.address.substr(2, input.length) === input;
}
const address = ethUtils.stripHexPrefix(wallet.address).toLowerCase();
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;
};
const computeDifficulty = (pattern, isChecksum) => {
const ret = Math.pow(16, pattern.length);
return isChecksum ? (ret * Math.pow(2, pattern.replace(/[^a-f]/gi, '').length)) : ret;
};
const computeProbability = (difficulty, attempts) => {
return 1 - Math.pow((difficulty - 1) / difficulty, attempts);
};
/**
* Generate a lot of wallets until one satisfies the input constraints
* @param input
* @param isChecksum
* @param max - Stop the generation after <max> attempts. Set to 0 for unlimited
* @returns
*/
const getVanityWallet = (input, isChecksum, max) => {
input = input || '';
if (!isValidHex(input)) {
throw new Error(ERRORS.invalidHex);
}
input = isChecksum ? input : input.toLowerCase();
let _wallet = getRandomWallet();
let attempts = 1;
while (!isValidVanityWallet(_wallet, input, isChecksum)) {
_wallet = getRandomWallet();
attempts++;
if (max && attempts >= max) {
return null;
}
}
_wallet.address = ethUtils.toChecksumAddress(_wallet.address);
_wallet.attempts = attempts;
return _wallet;
};
module.exports = {
getVanityWallet,
computeDifficulty,
computeProbability,
isValidHex
};