Allow both prefix & suffix (#57)
This commit is contained in:
parent
46d8883e3e
commit
4637b5df9d
6 changed files with 112 additions and 81 deletions
17
src/App.vue
17
src/App.vue
|
@ -33,7 +33,8 @@
|
||||||
<!--Statistics-->
|
<!--Statistics-->
|
||||||
<div class="col-md-6">
|
<div class="col-md-6">
|
||||||
<statistics
|
<statistics
|
||||||
:hex="input.hex"
|
:prefix="input.prefix"
|
||||||
|
:suffix="input.suffix"
|
||||||
:checksum="input.checksum"
|
:checksum="input.checksum"
|
||||||
:status="status"
|
:status="status"
|
||||||
:first-tick="firstTick"
|
:first-tick="firstTick"
|
||||||
|
@ -83,7 +84,7 @@
|
||||||
threads: 4,
|
threads: 4,
|
||||||
cores: 0,
|
cores: 0,
|
||||||
result: { address: '', privateKey: '' },
|
result: { address: '', privateKey: '' },
|
||||||
input: { hex: '', checksum: true, suffix: false },
|
input: { prefix: '', suffix: '', checksum: true },
|
||||||
firstTick: null,
|
firstTick: null,
|
||||||
error: null,
|
error: null,
|
||||||
};
|
};
|
||||||
|
@ -99,15 +100,15 @@
|
||||||
setInput: function (inputType, value) {
|
setInput: function (inputType, value) {
|
||||||
// eslint-disable-next-line default-case
|
// eslint-disable-next-line default-case
|
||||||
switch (inputType) {
|
switch (inputType) {
|
||||||
case 'hex':
|
case 'prefix':
|
||||||
this.input.hex = value;
|
this.input.prefix = value;
|
||||||
break;
|
|
||||||
case 'checksum':
|
|
||||||
this.input.checksum = value;
|
|
||||||
break;
|
break;
|
||||||
case 'suffix':
|
case 'suffix':
|
||||||
this.input.suffix = value;
|
this.input.suffix = value;
|
||||||
break;
|
break;
|
||||||
|
case 'checksum':
|
||||||
|
this.input.checksum = value;
|
||||||
|
break;
|
||||||
case 'threads':
|
case 'threads':
|
||||||
this.threads = value;
|
this.threads = value;
|
||||||
}
|
}
|
||||||
|
@ -256,7 +257,7 @@
|
||||||
worker.terminate();
|
worker.terminate();
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
const input = { checksum: true, hex: 'f'.repeat(5), suffix: false };
|
const input = { checksum: true, prefix: 'f'.repeat(5), suffix: '' };
|
||||||
console.info('Starting benchmark with 1 core...');
|
console.info('Starting benchmark with 1 core...');
|
||||||
worker.postMessage(input);
|
worker.postMessage(input);
|
||||||
},
|
},
|
||||||
|
|
|
@ -27,35 +27,42 @@ const getRandomWallet = () => {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Check if a wallet respects the input constraints
|
* Check if a wallet respects the input constraints
|
||||||
* @param address
|
* @param address - Wallet address
|
||||||
* @param input
|
* @param prefix - Prefix chosen by the user
|
||||||
* @param isChecksum
|
* @param suffix - Suffix chosen by the user
|
||||||
* @param isSuffix
|
* @param isChecksum - Is the input case-sensitive
|
||||||
* @returns {boolean}
|
* @returns {boolean}
|
||||||
*/
|
*/
|
||||||
const isValidVanityAddress = (address, input, isChecksum, isSuffix) => {
|
const isValidVanityAddress = (address, prefix, suffix, isChecksum) => {
|
||||||
const subStr = isSuffix ? address.substr(40 - input.length) : address.substr(0, input.length);
|
const addressPrefix = address.substring(0, prefix.length);
|
||||||
|
const addressSuffix = address.substring(40 - suffix.length);
|
||||||
|
|
||||||
if (!isChecksum) {
|
if (!isChecksum) {
|
||||||
return input === subStr;
|
return prefix === addressPrefix && suffix === addressSuffix;
|
||||||
}
|
}
|
||||||
if (input.toLowerCase() !== subStr) {
|
if (prefix.toLowerCase() !== addressPrefix || suffix.toLowerCase() !== addressSuffix) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
return isValidChecksum(address, input, isSuffix);
|
return isValidChecksum(address, prefix, suffix);
|
||||||
};
|
};
|
||||||
|
|
||||||
const isValidChecksum = (address, input, isSuffix) => {
|
const isValidChecksum = (address, prefix, suffix) => {
|
||||||
const hash = keccak('keccak256').update(address).digest().toString('hex');
|
const hash = keccak('keccak256').update(address).digest().toString('hex');
|
||||||
const shift = isSuffix ? 40 - input.length : 0;
|
|
||||||
|
|
||||||
for (let i = 0; i < input.length; i++) {
|
for (let i = 0; i < prefix.length; i++) {
|
||||||
const j = i + shift;
|
if (prefix[i] !== (parseInt(hash[i], 16) >= 8 ? address[i].toUpperCase() : address[i])) {
|
||||||
if (input[i] !== (parseInt(hash[j], 16) >= 8 ? address[j].toUpperCase() : address[j])) {
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
for (let i = 0; i < suffix.length; i++) {
|
||||||
|
const j = i + 40 - suffix.length;
|
||||||
|
if (suffix[i] !== (parseInt(hash[j], 16) >= 8 ? address[j].toUpperCase() : address[j])) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -70,18 +77,20 @@ const toChecksumAddress = (address) => {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Generate a lot of wallets until one satisfies the input constraints
|
* Generate a lot of wallets until one satisfies the input constraints
|
||||||
* @param input - String chosen by the user
|
* @param prefix - Prefix chosen by the user
|
||||||
|
* @param suffix - Suffix chosen by the user
|
||||||
* @param isChecksum - Is the input case-sensitive
|
* @param isChecksum - Is the input case-sensitive
|
||||||
* @param isSuffix - Is it a suffix, or a prefix
|
|
||||||
* @param cb - Callback called after x attempts, or when an address if found
|
* @param cb - Callback called after x attempts, or when an address if found
|
||||||
* @returns
|
* @returns
|
||||||
*/
|
*/
|
||||||
const getVanityWallet = (input, isChecksum, isSuffix, cb) => {
|
const getVanityWallet = (prefix, suffix, isChecksum, cb) => {
|
||||||
input = isChecksum ? input : input.toLowerCase();
|
|
||||||
let wallet = getRandomWallet();
|
let wallet = getRandomWallet();
|
||||||
let attempts = 1;
|
let attempts = 1;
|
||||||
|
|
||||||
while (!isValidVanityAddress(wallet.address, input, isChecksum, isSuffix)) {
|
const pre = isChecksum ? prefix : prefix.toLowerCase();
|
||||||
|
const suf = isChecksum ? suffix : suffix.toLowerCase();
|
||||||
|
|
||||||
|
while (!isValidVanityAddress(wallet.address, pre, suf, isChecksum)) {
|
||||||
if (attempts >= step) {
|
if (attempts >= step) {
|
||||||
cb({ attempts });
|
cb({ attempts });
|
||||||
attempts = 0;
|
attempts = 0;
|
||||||
|
@ -95,7 +104,7 @@ const getVanityWallet = (input, isChecksum, isSuffix, cb) => {
|
||||||
onmessage = function (event) {
|
onmessage = function (event) {
|
||||||
const input = event.data;
|
const input = event.data;
|
||||||
try {
|
try {
|
||||||
getVanityWallet(input.hex, input.checksum, input.suffix, (message) => postMessage(message));
|
getVanityWallet(input.prefix, input.suffix, input.checksum, (message) => postMessage(message));
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
self.postMessage({ error: err.toString() });
|
self.postMessage({ error: err.toString() });
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,15 +1,32 @@
|
||||||
<template>
|
<template>
|
||||||
<div class="panel" id="input-panel">
|
<div class="panel" id="input-panel">
|
||||||
<form :class="{ error: inputError }" @submit.prevent="startGen">
|
<form @submit.prevent="startGen">
|
||||||
<div class="error-text">Numbers and letters from A to F only</div>
|
<div class="error-text" v-if="inputError">Numbers and letters from A to F only</div>
|
||||||
<input
|
|
||||||
type="text"
|
<div class="row">
|
||||||
class="text-input-large"
|
<div class="col-12 col-sm-6 col-md-12 col-lg-6">
|
||||||
id="input"
|
<input
|
||||||
:placeholder="suffix ? 'Suffix' : 'Prefix'"
|
:class="{ error: prefixError }"
|
||||||
v-model="hex"
|
type="text"
|
||||||
:disabled="running"
|
class="text-input-large"
|
||||||
/>
|
id="input"
|
||||||
|
placeholder="Prefix"
|
||||||
|
v-model="prefix"
|
||||||
|
:disabled="running"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="col-12 col-sm-6 col-md-12 col-lg-6">
|
||||||
|
<input
|
||||||
|
:class="{ error: suffixError }"
|
||||||
|
type="text"
|
||||||
|
class="text-input-large"
|
||||||
|
id="input"
|
||||||
|
placeholder="Suffix"
|
||||||
|
v-model="suffix"
|
||||||
|
:disabled="running"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
<div class="row justify-content-center hide-render">
|
<div class="row justify-content-center hide-render">
|
||||||
<div class="spinner">
|
<div class="spinner">
|
||||||
<div></div>
|
<div></div>
|
||||||
|
@ -20,31 +37,22 @@
|
||||||
</div>
|
</div>
|
||||||
<div class="example hide-prerender">
|
<div class="example hide-prerender">
|
||||||
E.g.
|
E.g.
|
||||||
<span class="monospace">
|
<span v-if="inputError" class="monospace">N/A</span>
|
||||||
|
<span v-else class="monospace">
|
||||||
0x<!--
|
0x<!--
|
||||||
--><b v-if="!suffix" v-text="example.chosen"></b
|
--><b v-if="example.prefix" v-text="example.prefix"></b
|
||||||
><!--
|
><!--
|
||||||
--><span v-text="example.random"></span
|
--><span v-text="example.random"></span
|
||||||
><!--
|
><!--
|
||||||
--><b v-if="suffix" v-text="example.chosen"></b>
|
--><b v-if="example.suffix" v-text="example.suffix"></b>
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
<div class="row controls hide-prerender">
|
<div class="controls hide-prerender">
|
||||||
<div class="col-12 col-sm-6 col-md-12 col-lg-6">
|
<label class="checkbox">
|
||||||
<label class="checkbox">
|
<input type="checkbox" name="checkbox" checked="" v-model="checksum" :disabled="running" />
|
||||||
<input type="checkbox" name="checkbox" checked="" v-model="checksum" :disabled="running" />
|
<i class="left"> </i>
|
||||||
<i class="left"> </i>
|
Case-sensitive
|
||||||
Case-sensitive
|
</label>
|
||||||
</label>
|
|
||||||
</div>
|
|
||||||
<div class="col-12 col-sm-6 col-md-12 col-lg-6">
|
|
||||||
<span>Prefix</span>
|
|
||||||
<label class="switch">
|
|
||||||
<input type="checkbox" v-model="suffix" :disabled="running" />
|
|
||||||
<span class="slider"></span>
|
|
||||||
</label>
|
|
||||||
<span>Suffix</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
<div class="threads hide-prerender">
|
<div class="threads hide-prerender">
|
||||||
<input
|
<input
|
||||||
|
@ -105,26 +113,33 @@
|
||||||
data: function () {
|
data: function () {
|
||||||
return {
|
return {
|
||||||
threads: this.$props.cores || 4,
|
threads: this.$props.cores || 4,
|
||||||
hex: '',
|
prefix: '',
|
||||||
|
suffix: '',
|
||||||
checksum: true,
|
checksum: true,
|
||||||
suffix: false,
|
|
||||||
error: false,
|
error: false,
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
|
prefixError: function () {
|
||||||
|
return !isValidHex(this.prefix);
|
||||||
|
},
|
||||||
|
suffixError: function () {
|
||||||
|
return !isValidHex(this.suffix);
|
||||||
|
},
|
||||||
inputError: function () {
|
inputError: function () {
|
||||||
return !isValidHex(this.hex);
|
return this.prefixError || this.suffixError;
|
||||||
},
|
},
|
||||||
example: function () {
|
example: function () {
|
||||||
if (this.inputError) {
|
if (this.inputError) {
|
||||||
return 'N/A';
|
return null;
|
||||||
}
|
}
|
||||||
const chosen = this.checksum ? this.hex : mixCase(this.hex);
|
const prefix = this.checksum ? this.prefix : mixCase(this.prefix);
|
||||||
|
const suffix = this.checksum ? this.suffix : mixCase(this.suffix);
|
||||||
let random = '';
|
let random = '';
|
||||||
for (let i = 0; i < 40 - this.hex.length; i++) {
|
for (let i = 0; i < 40 - this.prefix.length - this.suffix.length; i++) {
|
||||||
random += mixCase(Math.floor(Math.random() * 16).toString(16));
|
random += mixCase(Math.floor(Math.random() * 16).toString(16));
|
||||||
}
|
}
|
||||||
return { random, chosen };
|
return { random, prefix, suffix };
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
|
@ -138,15 +153,15 @@
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
watch: {
|
watch: {
|
||||||
hex: function () {
|
prefix: function () {
|
||||||
this.$emit('input-change', 'hex', this.hex);
|
this.$emit('input-change', 'prefix', this.prefix);
|
||||||
},
|
|
||||||
checksum: function () {
|
|
||||||
this.$emit('input-change', 'checksum', this.checksum);
|
|
||||||
},
|
},
|
||||||
suffix: function () {
|
suffix: function () {
|
||||||
this.$emit('input-change', 'suffix', this.suffix);
|
this.$emit('input-change', 'suffix', this.suffix);
|
||||||
},
|
},
|
||||||
|
checksum: function () {
|
||||||
|
this.$emit('input-change', 'checksum', this.checksum);
|
||||||
|
},
|
||||||
threads: function () {
|
threads: function () {
|
||||||
this.$emit('input-change', 'threads', this.threads);
|
this.$emit('input-change', 'threads', this.threads);
|
||||||
},
|
},
|
||||||
|
@ -160,15 +175,11 @@
|
||||||
min-height: 280px
|
min-height: 280px
|
||||||
|
|
||||||
.error-text
|
.error-text
|
||||||
display: none
|
|
||||||
font-size: 14px
|
font-size: 14px
|
||||||
color: $error
|
color: $error
|
||||||
|
|
||||||
.error
|
input.error
|
||||||
input[type="text"]
|
border: 1px solid $error
|
||||||
border: 1px solid $error
|
|
||||||
.error-text
|
|
||||||
display: block
|
|
||||||
|
|
||||||
.example
|
.example
|
||||||
font-size: 14px
|
font-size: 14px
|
||||||
|
|
|
@ -16,7 +16,7 @@
|
||||||
</div>
|
</div>
|
||||||
<div class="col-lg-2 col-12">
|
<div class="col-lg-2 col-12">
|
||||||
<button data-remodal-target="modal" class="save button-large" :disabled="!privateKey">
|
<button data-remodal-target="modal" class="save button-large" :disabled="!privateKey">
|
||||||
<i class="icon-lock"></i> Save
|
<i class="icon-lock"></i>Save
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -74,6 +74,10 @@
|
||||||
|
|
||||||
.save
|
.save
|
||||||
margin-top: 30px
|
margin-top: 30px
|
||||||
|
i
|
||||||
|
margin-right: 8px
|
||||||
|
top: 2px
|
||||||
|
position: relative
|
||||||
|
|
||||||
@media screen and (min-width: 992px)
|
@media screen and (min-width: 992px)
|
||||||
.save
|
.save
|
||||||
|
|
|
@ -134,6 +134,7 @@
|
||||||
margin-bottom: 45px
|
margin-bottom: 45px
|
||||||
.remodal-close
|
.remodal-close
|
||||||
outline: none
|
outline: none
|
||||||
|
margin: 8px
|
||||||
&:before
|
&:before
|
||||||
font-size: 2em
|
font-size: 2em
|
||||||
&:hover
|
&:hover
|
||||||
|
|
|
@ -25,7 +25,8 @@
|
||||||
<script>
|
<script>
|
||||||
import humanizeDuration from 'humanize-duration';
|
import humanizeDuration from 'humanize-duration';
|
||||||
|
|
||||||
const computeDifficulty = function (pattern, isChecksum) {
|
const computeDifficulty = function (prefix, suffix, isChecksum) {
|
||||||
|
const pattern = prefix + suffix;
|
||||||
const ret = Math.pow(16, pattern.length);
|
const ret = Math.pow(16, pattern.length);
|
||||||
return isChecksum ? ret * Math.pow(2, pattern.replace(/[^a-f]/gi, '').length) : ret;
|
return isChecksum ? ret * Math.pow(2, pattern.replace(/[^a-f]/gi, '').length) : ret;
|
||||||
};
|
};
|
||||||
|
@ -46,13 +47,17 @@
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
props: {
|
props: {
|
||||||
hex: String,
|
prefix: String,
|
||||||
|
suffix: String,
|
||||||
checksum: Boolean,
|
checksum: Boolean,
|
||||||
status: String,
|
status: String,
|
||||||
firstTick: {},
|
firstTick: {},
|
||||||
},
|
},
|
||||||
watch: {
|
watch: {
|
||||||
hex() {
|
prefix() {
|
||||||
|
this.count = 0;
|
||||||
|
},
|
||||||
|
suffix() {
|
||||||
this.count = 0;
|
this.count = 0;
|
||||||
},
|
},
|
||||||
checksum() {
|
checksum() {
|
||||||
|
@ -61,10 +66,10 @@
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
inputError: function () {
|
inputError: function () {
|
||||||
return !isValidHex(this.hex);
|
return !isValidHex(this.prefix) || !isValidHex(this.suffix);
|
||||||
},
|
},
|
||||||
difficulty: function () {
|
difficulty: function () {
|
||||||
return this.inputError ? 'N/A' : computeDifficulty(this.hex, this.checksum);
|
return this.inputError ? 'N/A' : computeDifficulty(this.prefix, this.suffix, this.checksum);
|
||||||
},
|
},
|
||||||
probability50() {
|
probability50() {
|
||||||
return this.inputError ? 0 : Math.floor(Math.log(0.5) / Math.log(1 - 1 / this.difficulty));
|
return this.inputError ? 0 : Math.floor(Math.log(0.5) / Math.log(1 - 1 / this.difficulty));
|
||||||
|
|
Loading…
Add table
Reference in a new issue