Support multithreading using web workers

This commit is contained in:
Boris Kubiak 2018-01-04 20:37:07 +01:00
parent 0dada27d1c
commit 04cf5746a9
3 changed files with 92 additions and 52 deletions

File diff suppressed because one or more lines are too long

View file

@ -1,17 +1,34 @@
/* eslint-env browser */ /* eslint-env browser */
/* global vanity:false, Vue:false */ /* global Vue:false */
/**
* Check if a string is valid hexadecimal
* @param hex
* @returns {boolean}
*/
const isValidHex = hex => hex.length ? /^[0-9A-F]+$/g.test(hex.toUpperCase()) : 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);
};
// eslint-disable-next-line no-new // eslint-disable-next-line no-new
new Vue({ new Vue({
el: '#app', el: '#app',
data: { data: {
count: 0, count: 0,
lastTick: null,
firstTick: null, firstTick: null,
running: false, running: false,
step: 250, step: 500,
speed: '0 addr/s', speed: '0 addr/s',
status: 'Waiting', status: 'Waiting',
workers: [],
threads: 4,
result: { result: {
address: '', address: '',
privateKey: '' privateKey: ''
@ -24,21 +41,26 @@ new Vue({
computed: { computed: {
inputError() { inputError() {
return !vanity.isValidHex(this.input.prefix); return !isValidHex(this.input.prefix);
}, },
difficulty() { difficulty() {
return this.inputError ? 'N/A' : vanity.computeDifficulty(this.input.prefix, this.input.checksum); return this.inputError ? 'N/A' : computeDifficulty(this.input.prefix, this.input.checksum);
}, },
probability() { probability() {
return Math.round(10000 * vanity.computeProbability(this.difficulty, this.count)) / 100; return Math.round(10000 * computeProbability(this.difficulty, this.count)) / 100;
}
},
watch: {
threads() {
if (!this.running) {
this.initWorkers();
}
} }
}, },
methods: { methods: {
incrementCounter(incr) { incrementCounter(incr) {
this.count += incr; this.count += incr;
const currentTick = performance.now(); this.speed = incr > 0 ? Math.floor(1000 * this.count / (performance.now() - this.firstTick)) + ' addr/s' : '0 addr/s';
this.speed = incr > 0 ? Math.floor(1000 * incr / (currentTick - this.lastTick)) + ' addr/s' : '0 addr/s';
this.lastTick = currentTick;
}, },
displayResult(result) { displayResult(result) {
@ -46,7 +68,6 @@ new Vue({
this.result.address = result.address; this.result.address = result.address;
this.result.privateKey = result.privKey; this.result.privateKey = result.privKey;
this.status = 'Address found'; this.status = 'Address found';
this.speed = Math.floor(1000 * this.count / (performance.now() - this.firstTick)) + ' addr/s';
}, },
clearResult() { clearResult() {
@ -54,35 +75,74 @@ new Vue({
this.result.privateKey = ''; this.result.privateKey = '';
}, },
generate() { /**
const add = vanity.getVanityWallet(this.input.prefix, this.input.checksum, this.step); * Create missing workers, remove the unwanted ones.
*/
initWorkers() {
const self = this;
if (this.workers.length === this.threads) {
return;
}
// Remove unwanted workers
if (this.workers.length > this.threads) {
for (let w = this.threads - 1; w < this.workers.length; w++) {
this.workers[w].terminate();
}
this.workers = this.workers.slice(0, this.threads);
return;
}
// Create workers
for (let w = this.workers.length; w < this.threads; w++) {
this.workers[w] = new Worker('js/bundle.js');
this.workers[w].onmessage = function (event) {
self.parseWorkerMessage(event.data, w);
};
}
},
parseWorkerMessage(add, w) {
if (add !== null) { if (add !== null) {
this.running = false; this.stopGen();
return this.displayResult(add); return this.displayResult(add);
} }
this.incrementCounter(this.step); this.incrementCounter(this.step);
if (!this.running) { this.workers[w].postMessage({input: this.input, step: this.step});
this.status = 'Stopped';
return;
}
// Use setTimeout to let the browser render
setTimeout(() => this.generate(), 0);
}, },
startGen() { startGen() {
this.firstTick = performance.now(); if (!window.Worker) {
console.error('Web workers are not supported');
return;
}
this.incrementCounter(-this.count); this.incrementCounter(-this.count);
this.clearResult(); this.clearResult();
this.running = true; this.running = true;
setTimeout(() => this.generate(), 0); for (let w = 0; w < this.workers.length; w++) {
this.workers[w].postMessage({input: this.input, step: this.step});
}
this.status = 'Running';
this.firstTick = performance.now();
}, },
stopGen() { stopGen() {
this.running = false; this.running = false;
this.status = 'Stopped';
for (let i = 0; i < this.workers.length; i++) {
this.workers[i].terminate();
}
this.workers = [];
this.initWorkers();
} }
},
created() {
this.initWorkers();
} }
}); });

View file

@ -1,10 +1,8 @@
/* eslint-env worker */
const ethUtils = require('ethereumjs-util'); const ethUtils = require('ethereumjs-util');
const randomBytes = require('randombytes'); const randomBytes = require('randombytes');
const ERRORS = {
invalidHex: 'Invalid hex input'
};
/** /**
* Create a wallet from a random private key * Create a wallet from a random private key
* @returns {{address: string, privKey: string}} * @returns {{address: string, privKey: string}}
@ -17,13 +15,6 @@ const getRandomWallet = () => {
}; };
}; };
/**
* 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 * Check if a wallet respects the input constraints
* @param wallet * @param wallet
@ -50,15 +41,6 @@ const isValidVanityWallet = (wallet, input, isChecksum) => {
return true; 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 * Generate a lot of wallets until one satisfies the input constraints
* @param input * @param input
@ -67,10 +49,6 @@ const computeProbability = (difficulty, attempts) => {
* @returns * @returns
*/ */
const getVanityWallet = (input, isChecksum, max) => { const getVanityWallet = (input, isChecksum, max) => {
input = input || '';
if (!isValidHex(input)) {
throw new Error(ERRORS.invalidHex);
}
input = isChecksum ? input : input.toLowerCase(); input = isChecksum ? input : input.toLowerCase();
let _wallet = getRandomWallet(); let _wallet = getRandomWallet();
let attempts = 1; let attempts = 1;
@ -88,9 +66,11 @@ const getVanityWallet = (input, isChecksum, max) => {
return _wallet; return _wallet;
}; };
module.exports = { onmessage = function (event) {
getVanityWallet, const data = event.data;
computeDifficulty, postMessage(getVanityWallet(data.input.prefix, data.input.checksum, data.step));
computeProbability, };
isValidHex
module.exports = {
onmessage
}; };