Support multithreading using web workers ⚡
This commit is contained in:
parent
0dada27d1c
commit
04cf5746a9
3 changed files with 92 additions and 52 deletions
File diff suppressed because one or more lines are too long
104
js/index.js
104
js/index.js
|
@ -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();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
38
js/vanity.js
38
js/vanity.js
|
@ -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
|
||||||
};
|
};
|
||||||
|
|
Loading…
Add table
Reference in a new issue