2 // cd /path/to/repo/tests
3 // jasmine spec/tests.js
9 // see https://developer.mozilla.org/en-US/docs/Mozilla/Firefox/Headless_mode#Automated_testing_with_headless_mode
11 // USER SPECIFIED OPTIONS
12 var browser
= process
.env
.BROWSER
; //"firefox"; // or "chrome"
14 console
.log("Browser can be set via environment variable, eg");
15 console
.log("BROWSER=firefox jasmine spec/tests.js");
16 console
.log("Options for BROWSER are firefox chrome");
17 console
.log("Using default browser: chrome");
21 console
.log("Using browser: " + browser
);
26 var webdriver
= require('selenium-webdriver');
27 var By
= webdriver
.By
;
28 var Key
= webdriver
.Key
;
29 var until
= webdriver
.until
;
33 var generateDelay
= 1500;
34 var feedbackDelay
= 500;
35 var entropyFeedbackDelay
= 500;
36 var bip38delay
= 15000;
38 // url uses file:// scheme
39 var path
= require('path')
40 var parentDir
= path
.resolve(process
.cwd(), '..', 'src', 'index.html');
41 var url
= "file://" + parentDir
;
42 if (browser
== "firefox") {
43 // TODO loading local html in firefox is broken
44 console
.log("Loading local html in firefox is broken, see https://stackoverflow.com/q/46367054");
45 console
.log("You must run a server in this case, ie do this:");
46 console
.log("$ cd /path/to/bip39/src");
47 console
.log("$ python -m http.server");
48 url
= "http://localhost:8000";
51 // Variables dependent on specific browser selection
53 if (browser
== "firefox") {
54 var firefox
= require('selenium-webdriver/firefox');
55 var binary
= new firefox
.Binary(firefox
.Channel
.NIGHTLY
);
56 binary
.addArguments("-headless");
57 newDriver = function() {
58 return new webdriver
.Builder()
59 .forBrowser('firefox')
60 .setFirefoxOptions(new firefox
.Options().setBinary(binary
))
64 else if (browser
== "chrome") {
65 var chrome
= require('selenium-webdriver/chrome');
66 newDriver = function() {
67 return new webdriver
.Builder()
69 .setChromeOptions(new chrome
.Options().addArguments("headless"))
76 function testNetwork(done
, params
, comparePub
= false) {
77 var phrase
= params
.phrase
|| 'abandon abandon ability';
78 driver
.findElement(By
.css('.phrase'))
80 selectNetwork(params
.selectText
);
81 driver
.sleep(generateDelay
).then(function() {
83 getFirstAddress(function(address
) {
84 expect(address
).toBe(params
.firstAddress
);
88 getFirstPublicKey(function(pubkey
) {
89 expect(pubkey
).toBe(params
.firstPubKey
);
96 function getFirstRowValue(handler
, selector
) {
97 driver
.findElements(By
.css(selector
))
104 function getFirstAddress(handler
) {
105 getFirstRowValue(handler
, ".address");
108 function getFirstPublicKey(handler
) {
109 getFirstRowValue(handler
, ".pubkey");
112 function getFirstPath(handler
) {
113 getFirstRowValue(handler
, ".index");
116 function testColumnValuesAreInvisible(done
, columnClassName
) {
117 var selector
= "." + columnClassName
+ " span";
118 driver
.findElements(By
.css(selector
))
119 .then(function(els
) {
120 els
[0].getAttribute("class")
121 .then(function(classes
) {
122 expect(classes
).toContain("invisible");
128 function testRowsAreInCorrectOrder(done
) {
129 driver
.findElements(By
.css('.index'))
130 .then(function(els
) {
131 var testRowAtIndex = function(i
) {
132 if (i
>= els
.length
) {
137 .then(function(actualPath
) {
138 var noHardened
= actualPath
.replace(/'/g, "");
139 var pathBits = noHardened.split("/")
140 var lastBit = pathBits[pathBits.length-1];
141 var actualIndex = parseInt(lastBit);
142 expect(actualIndex).toBe(i);
151 function selectNetwork(name) {
152 driver.executeScript(function() {
153 var selectText = arguments[0];
154 $(".network option[selected]").removeAttr("selected");
155 $(".network option").filter(function(i,e) {
156 return $(e).html() == selectText;
157 }).prop("selected", true);
158 $(".network").trigger("change");
162 function testEntropyType(done, entropyText, entropyTypeUnsafe) {
163 // entropy type is compiled into regexp so needs escaping
164 // see https://stackoverflow.com/a/2593661
165 var entropyType = (entropyTypeUnsafe+'').replace(/[.?*+^$[\]\\(){}|-]/g, "\\$&");
166 driver.findElement(By.css('.use-entropy
'))
168 driver.findElement(By.css('.entropy
'))
169 .sendKeys(entropyText);
170 driver.sleep(generateDelay).then(function() {
171 driver.findElement(By.css('.entropy
-container
'))
173 .then(function(text) {
174 var re = new RegExp("Entropy Type\\s+" + entropyType);
175 expect(text).toMatch(re);
181 function testEntropyBits(done, entropyText, entropyBits) {
182 driver.findElement(By.css('.use-entropy
'))
184 driver.findElement(By.css('.entropy
'))
185 .sendKeys(entropyText);
186 driver.sleep(generateDelay).then(function() {
187 driver.findElement(By.css('.entropy
-container
'))
189 .then(function(text) {
190 var re = new RegExp("Total Bits\\s+" + entropyBits);
191 expect(text).toMatch(re);
197 function testEntropyFeedback(done, entropyDetail) {
198 // entropy type is compiled into regexp so needs escaping
199 // see https://stackoverflow.com/a/2593661
200 if ("type" in entropyDetail) {
201 entropyDetail.type = (entropyDetail.type+'').replace(/[.?*+^$[\]\\(){}|-]/g, "\\$&");
203 driver.findElement(By.css('.use-entropy
'))
205 driver.findElement(By.css('.entropy
'))
206 .sendKeys(entropyDetail.entropy);
207 driver.sleep(entropyFeedbackDelay).then(function() {
208 driver.findElement(By.css('.entropy
-container
'))
210 .then(function(text) {
211 driver.findElement(By.css('.phrase
'))
212 .getAttribute("value")
213 .then(function(phrase) {
214 if ("filtered" in entropyDetail) {
215 var key = "Filtered Entropy";
216 var value = entropyDetail.filtered;
217 var reText = key + "\\s+" + value;
218 var re = new RegExp(reText);
219 expect(text).toMatch(re);
221 if ("type" in entropyDetail) {
222 var key = "Entropy Type";
223 var value = entropyDetail.type;
224 var reText = key + "\\s+" + value;
225 var re = new RegExp(reText);
226 expect(text).toMatch(re);
228 if ("events" in entropyDetail) {
229 var key = "Event Count";
230 var value = entropyDetail.events;
231 var reText = key + "\\s+" + value;
232 var re = new RegExp(reText);
233 expect(text).toMatch(re);
235 if ("bits" in entropyDetail) {
236 var key = "Total Bits";
237 var value = entropyDetail.bits;
238 var reText = key + "\\s+" + value;
239 var re = new RegExp(reText);
240 expect(text).toMatch(re);
242 if ("bitsPerEvent" in entropyDetail) {
243 var key = "Bits Per Event";
244 var value = entropyDetail.bitsPerEvent;
245 var reText = key + "\\s+" + value;
246 var re = new RegExp(reText);
247 expect(text).toMatch(re);
249 if ("words" in entropyDetail) {
250 var actualWords = phrase.split(/\s+/)
251 .filter(function(w) { return w.length > 0 })
253 expect(actualWords).toBe(entropyDetail.words);
255 if ("strength" in entropyDetail) {
256 var key = "Time To Crack";
257 var value = entropyDetail.strength;
258 var reText = key + "\\s+" + value;
259 var re = new RegExp(reText);
260 expect(text).toMatch(re);
268 function testClientSelect(done, params) {
269 // set mnemonic and select bip32 tab
270 driver.findElement(By.css('#bip32
-tab a
'))
272 driver.findElement(By.css('.phrase
'))
273 .sendKeys("abandon abandon ability");
274 driver.sleep(generateDelay).then(function() {
276 // set bip32 client to bitcoin core
277 driver.executeScript(function() {
278 $("#bip32-client").val(arguments[0]).trigger("change");
279 }, params.selectValue);
280 driver.sleep(generateDelay).then(function() {
281 // check the derivation path is correct
282 driver.findElement(By.css("#bip32-path"))
283 .getAttribute("value")
284 .then(function(path) {
285 expect(path).toBe(params.bip32path);
286 // check hardened addresses is selected
287 driver.findElement(By.css(".hardened-addresses"))
288 .getAttribute("checked")
289 .then(function(isChecked) {
290 expect(isChecked).toBe(params.useHardenedAddresses);
291 // check input is readonly
292 driver.findElement(By.css("#bip32-path"))
293 .getAttribute("readonly")
294 .then(function(isReadonly) {
295 expect(isReadonly).toBe("true");
306 describe('BIP39 Tool Tests
', function() {
308 beforeEach(function(done) {
309 driver = newDriver();
310 driver.get(url).then(done);
313 // Close the website after each test is run (so that it is opened fresh each time)
314 afterEach(function(done) {
315 driver.quit().then(done);
320 // Page initially loads with blank phrase
321 it('Should load the page
', function(done) {
322 driver.findElement(By.css('.phrase
'))
323 .getAttribute('value
').then(function(value) {
324 expect(value).toBe('');
330 it('Should have text on the page
', function(done) {
331 driver.findElement(By.css('body
'))
333 .then(function(text) {
334 var textToFind = "You can enter an existing BIP39 mnemonic";
335 expect(text).toContain(textToFind);
340 // Entering mnemonic generates addresses
341 it('Should have a list
of addresses
', function(done) {
342 driver.findElement(By.css('.phrase
'))
343 .sendKeys('abandon abandon ability
');
344 driver.sleep(generateDelay).then(function() {
345 driver.findElements(By.css('.address
'))
346 .then(function(els) {
347 expect(els.length).toBe(20);
353 // Generate button generates random mnemonic
354 it('Should be able to generate a random mnemonic
', function(done) {
355 // initial phrase is blank
356 driver.findElement(By.css('.phrase
'))
357 .getAttribute("value")
358 .then(function(phrase) {
359 expect(phrase.length).toBe(0);
361 driver.findElement(By.css('.generate
')).click();
362 driver.sleep(generateDelay).then(function() {
363 // new phrase is not blank
364 driver.findElement(By.css('.phrase
'))
365 .getAttribute("value")
366 .then(function(phrase) {
367 expect(phrase.length).toBeGreaterThan(0);
374 // Mnemonic length can be customized
375 it('Should allow custom length mnemonics
', function(done) {
377 driver.executeScript(function() {
378 $(".strength option[selected]").removeAttr("selected");
379 $(".strength option[value=6]").prop("selected", true);
381 driver.findElement(By.css('.generate
')).click();
382 driver.sleep(generateDelay).then(function() {
383 driver.findElement(By.css('.phrase
'))
384 .getAttribute("value")
385 .then(function(phrase) {
386 var words = phrase.split(" ");
387 expect(words.length).toBe(6);
393 // Passphrase can be set
394 it('Allows a passphrase to be
set', function(done) {
395 driver.findElement(By.css('.phrase
'))
396 .sendKeys('abandon abandon ability
');
397 driver.findElement(By.css('.passphrase
'))
398 .sendKeys('secure_passphrase
');
399 driver.sleep(generateDelay).then(function() {
400 getFirstAddress(function(address) {
401 expect(address).toBe("15pJzUWPGzR7avffV9nY5by4PSgSKG9rba");
407 // Network can be set to networks other than bitcoin
408 it('Allows selection
of bitcoin testnet
', function(done) {
410 selectText: "BTC - Bitcoin Testnet",
411 firstAddress: "mucaU5iiDaJDb69BHLeDv8JFfGiyg2nJKi",
413 testNetwork(done, params);
415 it('Allows selection
of litecoin
', function(done) {
417 selectText: "LTC - Litecoin",
418 firstAddress: "LQ4XU8RX2ULPmPq9FcUHdVmPVchP9nwXdn",
420 testNetwork(done, params);
422 it('Allows selection
of litecoin testnet
', function(done) {
424 selectText: "LTCt - Litecoin Testnet",
425 firstAddress: "mucaU5iiDaJDb69BHLeDv8JFfGiyg2nJKi",
427 testNetwork(done, params);
429 it('Allows selection
of ripple
', function(done) {
431 selectText: "XRP - Ripple",
432 firstAddress: "rLTFnqbmCVPGx6VfaygdtuKWJgcN4v1zRS",
433 phrase: "ill clump only blind unit burden thing track silver cloth review awake useful craft whale all satisfy else trophy sunset walk vanish hope valve",
435 testNetwork(done, params);
437 it('Allows selection
of casinocoin
', function(done) {
439 selectText: "CSC - CasinoCoin",
440 firstAddress: "c3P5EUb27Pzk9dcGt4s7zQDQj4sC6Y81mT",
441 phrase: "ill clump only blind unit burden thing track silver cloth review awake useful craft whale all satisfy else trophy sunset walk vanish hope valve",
443 testNetwork(done, params);
445 it('Allows selection
of dogecoin
', function(done) {
447 selectText: "DOGE - Dogecoin",
448 firstAddress: "DPQH2AtuzkVSG6ovjKk4jbUmZ6iXLpgbJA",
450 testNetwork(done, params);
452 it('Allows selection
of dogecoin testnet
', function(done) {
454 selectText: "DOGEt - Dogecoin Testnet",
455 firstAddress: "niHnSJKHdwDyDxRMLBJrtNqpvHEsAFWe6B",
457 testNetwork(done, params);
459 it('Allows selection
of denarius
', function(done) {
461 selectText: "DNR - Denarius",
462 firstAddress: "DFdFMVUMzU9xX88EywXvAGwjiwpxyh9vKb",
464 testNetwork(done, params);
466 it('Allows selection
of shadowcash
', function(done) {
468 selectText: "SDC - ShadowCash",
469 firstAddress: "SiSZtfYAXEFvMm3XM8hmtkGDyViRwErtCG",
471 testNetwork(done, params);
473 it('Allows selection
of shadowcash testnet
', function(done) {
475 selectText: "SDC - ShadowCash Testnet",
476 firstAddress: "tM2EDpVKaTiEg2NZg3yKg8eqjLr55BErHe",
478 testNetwork(done, params);
480 it('Allows selection
of viacoin
', function(done) {
482 selectText: "VIA - Viacoin",
483 firstAddress: "Vq9Eq4N5SQnjqZvxtxzo7hZPW5XnyJsmXT",
485 testNetwork(done, params);
487 it('Allows selection
of viacoin testnet
', function(done) {
489 selectText: "VIA - Viacoin Testnet",
490 firstAddress: "tM2EDpVKaTiEg2NZg3yKg8eqjLr55BErHe",
492 testNetwork(done, params);
494 it('Allows selection
of jumbucks
', function(done) {
496 selectText: "JBS - Jumbucks",
497 firstAddress: "JLEXccwDXADK4RxBPkRez7mqsHVoJBEUew",
499 testNetwork(done, params);
501 it('Allows selection
of clam
', function(done) {
503 selectText: "CLAM - Clams",
504 firstAddress: "xCp4sakjVx4pUAZ6cBCtuin8Ddb6U1sk9y",
506 testNetwork(done, params);
508 it('Allows selection
of crown
', function(done) {
510 selectText: "CRW - Crown (Legacy)",
511 firstAddress: "18pWSwSUAQdiwMHUfFZB1fM2xue9X1FqE5",
513 testNetwork(done, params);
515 it('Allows selection
of crown
', function(done) {
517 selectText: "CRW - Crown",
518 firstAddress: "CRWKnVmVhvH1KWTYe6sq8xV4dFGcFpBEEkPQ",
520 testNetwork(done, params);
522 it('Allows selection
of dash
', function(done) {
524 selectText: "DASH - Dash",
525 firstAddress: "XdbhtMuGsPSkE6bPdNTHoFSszQKmK4S5LT",
527 testNetwork(done, params);
529 it('Allows selection
of dash testnet
', function(done) {
531 selectText: "DASH - Dash Testnet",
532 firstAddress: "yaR52EN4oojdJfBgzWJTymC4uuCLPT29Gw",
534 testNetwork(done, params);
536 it('Allows selection
of game
', function(done) {
538 selectText: "GAME - GameCredits",
539 firstAddress: "GSMY9bAp36cMR4zyT4uGVS7GFjpdXbao5Q",
541 testNetwork(done, params);
543 it('Allows selection
of komodo
', function(done) {
545 selectText: "KMD - Komodo",
546 firstAddress: "RMPPzJwAjPVZZAwJvXivHJGGjdCx6WBD2t",
548 testNetwork(done, params);
550 it('Allows selection
of namecoin
', function(done) {
552 selectText: "NMC - Namecoin",
553 firstAddress: "Mw2vK2Bvex1yYtYF6sfbEg2YGoUc98YUD2",
555 testNetwork(done, params);
557 it('Allows selection
of onixcoin
', function(done) {
559 selectText: "ONX - Onixcoin",
560 firstAddress: "XGwMqddeKjT3ddgX73QokjVbCL3aK6Yxfk",
562 testNetwork(done, params);
564 it('Allows selection
of lkrcoin
', function(done) {
566 selectText: "LKR - Lkrcoin",
567 firstAddress: "LfbT296e7AEEnn4bYDbL535Nd8P9g98CdJ",
569 testNetwork(done, params);
571 it('Allows selection
of bolivarcoin
', function(done) {
573 selectText: "BOLI - Bolivarcoin",
574 firstAddress: "bbKzCAUR7hZ3nqfffy7VgrSz8LmAP3S5mK",
576 testNetwork(done, params);
578 it('Allows selection
of peercoin
', function(done) {
580 selectText: "PPC - Peercoin",
581 firstAddress: "PVAiioTaK2eDHSEo3tppT9AVdBYqxRTBAm",
583 testNetwork(done, params);
585 it('Allows selection
of ethereum
', function(done) {
587 selectText: "ETH - Ethereum",
588 firstAddress: "0xe5815d5902Ad612d49283DEdEc02100Bd44C2772",
590 testNetwork(done, params);
591 // TODO test private key and public key
593 it('Allows selection
of slimcoin
', function(done) {
595 selectText: "SLM - Slimcoin",
596 firstAddress: "SNzPi1CafHFm3WWjRo43aMgiaEEj3ogjww",
598 testNetwork(done, params);
600 it('Allows selection
of slimcoin testnet
', function(done) {
602 selectText: "SLM - Slimcoin Testnet",
603 firstAddress: "n3nMgWufTek5QQAr6uwMhg5xbzj8xqc4Dq",
605 testNetwork(done, params);
607 it('Allows selection
of bitcoin cash
', function(done) {
609 selectText: "BCH - Bitcoin Cash",
610 firstAddress: "bitcoincash:qzlquk7w4hkudxypl4fgv8x279r754dkvur7jpcsps",
612 testNetwork(done, params);
615 it('Allows selection
of simpleledger(SLP
)', function(done) {
617 selectText: "SLP - Simple Ledger Protocol",
618 firstAddress: "simpleledger:qrtffz6ajfsn74gpur7y3epjquz42pvww5acewqmre",
620 testNetwork(done, params);
623 it('Allows selection
of myriadcoin
', function(done) {
625 selectText: "XMY - Myriadcoin",
626 firstAddress: "MJEswvRR46wh9BoiVj9DzKYMBkCramhoBV",
628 testNetwork(done, params);
630 it('Allows selection
of pivx
', function(done) {
632 selectText: "PIVX - PIVX",
633 firstAddress: "DBxgT7faCuno7jmtKuu6KWCiwqsVPqh1tS",
635 testNetwork(done, params);
637 it('Allows selection
of pivx testnet
', function(done) {
639 selectText: "PIVX - PIVX Testnet",
640 firstAddress: "yB5U384n6dGkVE3by5y9VdvHHPwPg68fQj",
642 testNetwork(done, params);
644 it('Allows selection
of maza
', function(done) {
646 selectText: "MAZA - Maza",
647 firstAddress: "MGW4Bmi2NEm4PxSjgeFwhP9vg18JHoRnfw",
649 testNetwork(done, params);
651 it('Allows selection
of FIX
', function(done) {
653 selectText: "FIX - FIX",
654 firstAddress: "FS5MEU8fs5dUvsaSCSusV8RQtC8j2h3JEh",
656 testNetwork(done, params);
658 it('Allows selection
of FIX testnet
', function(done) {
660 selectText: "FIX - FIX Testnet",
661 firstAddress: "XpnU1HHdNG5YxvG9Rez4wjmidchxqnZaNa",
663 testNetwork(done, params);
665 it('Allows selection
of fujicoin
', function(done) {
667 selectText: "FJC - Fujicoin",
668 firstAddress: "FgiaLpG7C99DyR4WnPxXedRVHXSfKzUDhF",
670 testNetwork(done, params);
672 it('Allows selection
of nubits
', function(done) {
674 selectText: "USNBT - NuBits",
675 firstAddress: "BLxkabXuZSJSdesLD7KxZdqovd4YwyBTU6",
677 testNetwork(done, params);
679 it('Allows selection
of bitcoin gold
', function(done) {
681 selectText: "BTG - Bitcoin Gold",
682 firstAddress: "GdDqug4WUsn5syNbSTHatNn4XnuwZtzedx",
684 testNetwork(done, params);
686 it('Allows selection
of monacoin
', function(done) {
688 selectText: "MONA - Monacoin",
689 firstAddress: "MKMiMr7MyjDKjJbCBzgF6u4ByqTS4NkRB1",
691 testNetwork(done, params);
693 it('Allows selection
of AXE
', function(done) {
695 selectText: "AXE - Axe",
696 firstAddress: "PScwtLUyPiGrqtKXrHF37DGETLXLZdw4up",
698 testNetwork(done, params);
700 it('Allows selection
of BlackCoin
', function(done) {
702 selectText: "BLK - BlackCoin",
703 firstAddress: "B5MznAKwj7uQ42vDz3w4onhBXPcqhTwJ9z",
705 testNetwork(done, params);
707 it('Allows selection
of Neblio
', function(done) {
709 selectText: "NEBL - Neblio",
710 firstAddress: "NefkeEEvhusbHMmTRrxx7H9wFnUXd8qQsE",
712 testNetwork(done, params);
714 it('Allows selection
of Beetlecoin
', function(done) {
716 selectText: "BEET - Beetlecoin",
717 firstAddress: "BVmtbEsGrjpknprmpHFq26z4kYHJUFHE71",
719 testNetwork(done, params);
721 it('Allows selection
of Adcoin
', function(done) {
723 selectText: "ACC - Adcoin",
724 firstAddress: "AcEDM6V5sF4kFHC76MJjjfProtS5Sw2qcd",
726 testNetwork(done, params);
728 it('Allows selection
of Asiacoin
', function(done) {
730 selectText: "AC - Asiacoin",
731 firstAddress: "ALupuEEz7kJjQTAvmtcBMBVuEjPa7GqZzE",
733 testNetwork(done, params);
735 it('Allows selection
of Auroracoin
', function(done) {
737 selectText: "AUR - Auroracoin",
738 firstAddress: "ANuraS6F4Jpi413FEnavjYkKYJJRHkgYCm",
740 testNetwork(done, params);
742 it('Allows selection
of Bata
', function(done) {
744 selectText: "BTA - Bata",
745 firstAddress: "BGxBdNeYPtF3GCuTtZBPQdFxCkdBYSF3fj",
747 testNetwork(done, params);
749 it('Allows selection
of Belacoin
', function(done) {
751 selectText: "BELA - Belacoin",
752 firstAddress: "BEeetqpNffdzeknSpNmQp5KAFh2KK1Qx7S",
754 testNetwork(done, params);
756 it('Allows selection
of Bitcoin Atom
', function(done) {
758 selectText: "BCA - Bitcoin Atom",
759 firstAddress: "AMy6qMbJeC4zsGRL6iWszmeCdQH65fgfih",
761 testNetwork(done, params);
763 it('Allows selection
of Bitcoinplus
', function(done) {
765 selectText: "XBC - Bitcoinplus",
766 firstAddress: "B7FSynZoDbEwTCSgsXq9nJ5ue8owYLVL8r",
768 testNetwork(done, params);
770 it('Allows selection
of Bitcoin Private
', function(done) {
772 selectText: "BTCP - Bitcoin Private",
773 firstAddress: "b1M3PbiXXyN6Hdivdw5rJv5VKpLjPzhm4jM",
775 testNetwork(done, params);
777 it('Allows selection
of Bitcoinz
', function(done) {
779 selectText: "BTCZ - Bitcoinz",
780 firstAddress: "t1X2YQoxs8cYRo2oaBYgVEwW5QNjCC59NYc",
782 testNetwork(done, params);
784 it('Allows selection
of BitCloud
', function(done) {
786 selectText: "BTDX - BitCloud",
787 firstAddress: "BHbWitXCNgTf1BhsRDNMP186EeibuzmrBi",
789 testNetwork(done, params);
791 it('Allows selection
of Bitcore
', function(done) {
793 selectText: "BTX - Bitcore",
794 firstAddress: "2Rgp5Znhpy34TK4QmPkfCiYs9r4KovfTH9",
796 testNetwork(done, params);
798 it('Allows selection
of Bitsend
', function(done) {
800 selectText: "BSD - Bitsend",
801 firstAddress: "iBPk7LYjDun3EPk7CRR8UUmnPoceVc1bp2",
803 testNetwork(done, params);
805 it('Allows selection
of Britcoin
', function(done) {
807 selectText: "BRIT - Britcoin",
808 firstAddress: "B6Aue4J2XLs1f1dtD4H1SHYFfh4XrmEbrw",
810 testNetwork(done, params);
812 it('Allows selection
of Canadaecoin
', function(done) {
814 selectText: "CDN - Canadaecoin",
815 firstAddress: "CanAyCfd5Rj2CQVfaoAmvDUZunPM5W1AEQ",
817 testNetwork(done, params);
819 it('Allows selection
of Cannacoin
', function(done) {
821 selectText: "CCN - Cannacoin",
822 firstAddress: "CYjW8xWB43g6krLJTmmrPk1PonoQX7h9Qd",
824 testNetwork(done, params);
826 it('Allows selection
of Clubcoin
', function(done) {
828 selectText: "CLUB - Clubcoin",
829 firstAddress: "CHMDEXN4sihpSVX4GyAa2hZ62shnby7uyN",
831 testNetwork(done, params);
833 it('Allows selection
of Compcoin
', function(done) {
835 selectText: "CMP - Compcoin",
836 firstAddress: "CLshtw3zhxkseBJS46UF12v3AFy9Dx7JVv",
838 testNetwork(done, params);
840 it('Allows selection
of CPUchain
', function(done) {
842 selectText: "CPU - CPUchain",
843 firstAddress: "CWWkTPkNRdpTDSfPw7gxUt9cEaC5PSsP3Y",
845 testNetwork(done, params);
847 it('Allows selection
of Crave
', function(done) {
849 selectText: "CRAVE - Crave",
850 firstAddress: "VCYJeti6uKMNBFKCL7eP96UwuFWYHM7c85",
852 testNetwork(done, params);
854 it('Allows selection
of Defcoin
', function(done) {
856 selectText: "DFC - Defcoin",
857 firstAddress: "D8swcgyaaFUrXZU3ATwbgy16buCpWqbG1M",
859 testNetwork(done, params);
861 it('Allows selection
of Diamond
', function(done) {
863 selectText: "DMD - Diamond",
864 firstAddress: "dJnrVbLL9UPjdaVRz2C8VpqHZknqAqjLek",
866 testNetwork(done, params);
868 it('Allows selection
of Digibyte
', function(done) {
870 selectText: "DGB - Digibyte",
871 firstAddress: "D85Rp9jwLtMdmP6wGjTiqHBdVQLST3YCEq",
873 testNetwork(done, params);
875 it('Allows selection
of Digitalcoin
', function(done) {
877 selectText: "DGC - Digitalcoin",
878 firstAddress: "DKw4UGKEAZWweDNEbBFNQx4EM8x1mpUdia",
880 testNetwork(done, params);
882 it('Allows selection
of Ecoin
', function(done) {
884 selectText: "ECN - Ecoin",
885 firstAddress: "e6WFPLG5gcXyF7cESFteH1hE2XSmowW5yB",
887 testNetwork(done, params);
889 it('Allows selection
of Edrcoin
', function(done) {
891 selectText: "EDRC - Edrcoin",
892 firstAddress: "eh1nUJsvgKPFv6ebMBfcwJ299GMCpjeZUG",
894 testNetwork(done, params);
896 it('Allows selection
of Egulden
', function(done) {
898 selectText: "EFL - Egulden",
899 firstAddress: "Lg66yt55R7edRM58cDhKzXik2kFme3viX7",
901 testNetwork(done, params);
903 it('Allows selection
of Einsteinium
', function(done) {
905 selectText: "EMC2 - Einsteinium",
906 firstAddress: "EVAABm9hXKHk2MpVMbwNakRubFnNha5m8m",
908 testNetwork(done, params);
910 it('Allows selection
of EOSIO
', function(done) {
912 selectText: "EOS - EOSIO",
913 firstPubKey: "EOS692VJTBK3Rmw93onNnpnZ8ZtmE9PdxjDStArvbyzoe11QUTNoy",
915 testNetwork(done, params, true);
917 it('Allows selection
of Europecoin
', function(done) {
919 selectText: "ERC - Europecoin",
920 firstAddress: "ESA2YwPYntAoaPrE8Fm5qkKRtkcwLcwD6R",
922 testNetwork(done, params);
924 it('Allows selection
of Exclusivecoin
', function(done) {
926 selectText: "EXCL - Exclusivecoin",
927 firstAddress: "EbUa6m8UZW6nTxsYZD2FsDjkadKbp5M6JT",
929 testNetwork(done, params);
931 it('Allows selection
of Feathercoin
', function(done) {
933 selectText: "FTC - Feathercoin",
934 firstAddress: "6gDdjAMoSgQaW8UhqK3oboHs6ftGAroKkM",
936 testNetwork(done, params);
938 it('Allows selection
of Firstcoin
', function(done) {
940 selectText: "FRST - Firstcoin",
941 firstAddress: "FJN9GzfMm7Q8R4DJwK1H9F6A1GTghvFiMJ",
943 testNetwork(done, params);
945 it('Allows selection
of Flashcoin
', function(done) {
947 selectText: "FLASH - Flashcoin",
948 firstAddress: "UWfpf5LfMmLxZYooEb2EyvWhZ8NG7EZDRt",
950 testNetwork(done, params);
952 it('Allows selection
of GCRCoin
', function(done) {
954 selectText: "GCR - GCRCoin",
955 firstAddress: "GJjF5cLwyXLacpuvXAVksxGxKvHDjx58d6",
957 testNetwork(done, params);
959 it('Allows selection
of Gobyte
', function(done) {
961 selectText: "GBX - Gobyte",
962 firstAddress: "GS813Ys2brkmvSUw1rUqGPm2HqQVDHJRyA",
964 testNetwork(done, params);
966 it('Allows selection
of Gridcoin
', function(done) {
968 selectText: "GRC - Gridcoin",
969 firstAddress: "SGrWbBPvobgqKRF8td1Kdc9vbRY7MJ78Y9",
971 testNetwork(done, params);
973 it('Allows selection
of Gulden
', function(done) {
975 selectText: "NLG - Gulden",
976 firstAddress: "GcDP7cNEc33MPPdTFNJ8pZc6VMZJ2CbKxY",
978 testNetwork(done, params);
980 it('Allows selection
of Helleniccoin
', function(done) {
982 selectText: "HNC - Helleniccoin",
983 firstAddress: "LbHEKe5H72zp9G1fuWNiiNePTUfJb88915",
985 testNetwork(done, params);
987 it('Allows selection
of Hempcoin
', function(done) {
989 selectText: "THC - Hempcoin",
990 firstAddress: "H8sdWbZyJV4gyXyHtLXDaNnAuUDhK5mfTV",
992 testNetwork(done, params);
994 it('Allows selection
of Insane
', function(done) {
996 selectText: "INSN - Insane",
997 firstAddress: "iMPqEJMiXWuxC9U2NVinCCMr4t72h58EWx",
999 testNetwork(done, params);
1001 it('Allows selection
of Iop
', function(done) {
1003 selectText: "IOP - Iop",
1004 firstAddress: "pGKQmcaPf95Ur5o6oHK4qdiZ52p1yaTvq1",
1006 testNetwork(done, params);
1008 it('Allows selection
of Ixcoin
', function(done) {
1010 selectText: "IXC - Ixcoin",
1011 firstAddress: "xgE9bTZ6YypT3E6ByzkTt31Hq68E9BqywH",
1013 testNetwork(done, params);
1015 it('Allows selection
of Kobocoin
', function(done) {
1017 selectText: "KOBO - Kobocoin",
1018 firstAddress: "FTVoNJETXDAM8x7MnmdE8RwWndSr9PQWhy",
1020 testNetwork(done, params);
1022 it('Allows selection
of Landcoin
', function(done) {
1024 selectText: "LDCN - Landcoin",
1025 firstAddress: "LLvLwNjG1aJcn1RS4W4GJUbv8fNaRATG7c",
1027 testNetwork(done, params);
1029 it('Allows selection
of Library Credits
', function(done) {
1031 selectText: "LBC - Library Credits",
1032 firstAddress: "bQJEQrHDJyHdqycB32uysh1SWn8Ln8LMdg",
1034 testNetwork(done, params);
1036 it('Allows selection
of Linx
', function(done) {
1038 selectText: "LINX - Linx",
1039 firstAddress: "XGWQ3cb3LGUB3VnHmj6xYSMgnokNbf6dyk",
1041 testNetwork(done, params);
1043 it('Allows selection
of Litecoincash
', function(done) {
1045 selectText: "LCC - Litecoincash",
1046 firstAddress: "Ce5n7fjUuQPLutJ4W5nCCfQLKdKLE1mv9A",
1048 testNetwork(done, params);
1050 it('Allows selection
of Lynx
', function(done) {
1052 selectText: "LYNX - Lynx",
1053 firstAddress: "KUeY3ZdZkg96p4W98pj1JjygCFU1XqWdw3",
1055 testNetwork(done, params);
1057 it('Allows selection
of Megacoin
', function(done) {
1059 selectText: "MEC - Megacoin",
1060 firstAddress: "MDfAj9CzkC1HpcUiVGnHp8yKTa7WXgu8AY",
1062 testNetwork(done, params);
1064 it('Allows selection
of Minexcoin
', function(done) {
1066 selectText: "MNX - Minexcoin",
1067 firstAddress: "XC1VnyJVfiMDwWgFtAHDp41cgY3AHk3dJT",
1069 testNetwork(done, params);
1071 it('Allows selection
of Navcoin
', function(done) {
1073 selectText: "NAV - Navcoin",
1074 firstAddress: "NTQVTPK3NWSQLKoffkiQw99T8PifkF1Y2U",
1076 testNetwork(done, params);
1078 it('Allows selection
of Nebulas
', function(done) {
1080 selectText: "NAS - Nebulas",
1081 firstAddress: "n1PbK61DGBfDoDusLw621G6sVSMfLLHdfnm",
1083 testNetwork(done, params);
1085 it('Allows selection
of Neoscoin
', function(done) {
1087 selectText: "NEOS - Neoscoin",
1088 firstAddress: "NgATz6QbQNXvayHQ4CpZayugb9HeaPDdby",
1090 testNetwork(done, params);
1092 it('Allows selection
of Nix
', function(done) {
1094 selectText: "NIX - NIX Platform",
1095 firstAddress: "GgcNW2SQQXB4LWHRQTHKkQF3GzXNSLqS8u",
1097 testNetwork(done, params);
1099 it('Allows selection
of Neurocoin
', function(done) {
1101 selectText: "NRO - Neurocoin",
1102 firstAddress: "NVdYErQ3mFpDuF5DquW9WMiT7sLc8ufFTn",
1104 testNetwork(done, params);
1106 it('Allows selection
of Newyorkc
', function(done) {
1108 selectText: "NYC - Newyorkc",
1109 firstAddress: "RSVMfyH1fKfy3puADJEhut2vfkRyon6imm",
1111 testNetwork(done, params);
1113 it('Allows selection
of Novacoin
', function(done) {
1115 selectText: "NVC - Novacoin",
1116 firstAddress: "4JRvUmxcKCJmaMXZyvRoSS1cmG2XvnZfHN",
1118 testNetwork(done, params);
1120 it('Allows selection
of Nushares
', function(done) {
1122 selectText: "NSR - Nushares",
1123 firstAddress: "SecjXzU3c7EecdT7EbC4vvmbdtBBokWh6J",
1125 testNetwork(done, params);
1127 it('Allows selection
of Okcash
', function(done) {
1129 selectText: "OK - Okcash",
1130 firstAddress: "PV4Qp1TUYuGv4TqVtLZtqvrsWWRycfx1Yi",
1132 testNetwork(done, params);
1134 it('Allows selection
of Omnicore
', function(done) {
1136 selectText: "OMNI - Omnicore",
1137 firstAddress: "1Q1t3gonjCT3rW38TsTsCvgSc3hh7zBGbi",
1139 testNetwork(done, params);
1141 it('Allows selection
of DeepOnion
', function(done) {
1143 selectText: "ONION - DeepOnion",
1144 firstAddress: "DYREY7XCFXVqJ3x5UuN43k2JwD2s1kif48",
1146 testNetwork(done, params);
1148 it('Allows selection
of Pesobit
', function(done) {
1150 selectText: "PSB - Pesobit",
1151 firstAddress: "PDePsF7ALyXP7JaywokdYiRTDtKa14MAr1",
1153 testNetwork(done, params);
1155 it('Allows selection
of Pinkcoin
', function(done) {
1157 selectText: "PINK - Pinkcoin",
1158 firstAddress: "2TgjYQffjbzUHJghNaVbdsjHbRwruC3yzC",
1160 testNetwork(done, params);
1162 it('Allows selection
of POSWcoin
', function(done) {
1164 selectText: "POSW - POSWcoin",
1165 firstAddress: "PNxewmZoPnGBvoEbH6hgQZCK1igDiBCdgC",
1167 testNetwork(done, params);
1169 it('Allows selection
of Potcoin
', function(done) {
1171 selectText: "POT - Potcoin",
1172 firstAddress: "PEo7Vg2ctXgpP4vuLPeY9aGJtZotyrmiHc",
1174 testNetwork(done, params);
1176 it('Allows selection
of Putincoin
', function(done) {
1178 selectText: "PUT - Putincoin",
1179 firstAddress: "PViWnfr2uFtovd6e7joM49C94CsGSnqJis",
1181 testNetwork(done, params);
1183 it('Allows selection
of Ravencoin
', function(done) {
1185 selectText: "RVN - Ravencoin",
1186 firstAddress: "RBuDoVNnzvFsEcX8XKPm8ic4mgiCzjUCNk",
1188 testNetwork(done, params);
1190 it('Allows selection
of Reddcoin
', function(done) {
1192 selectText: "RDD - Reddcoin",
1193 firstAddress: "RtgRvXMBng1y51ftteveFqwNfyRG18HpxQ",
1195 testNetwork(done, params);
1197 it('Allows selection
of RevolutionVR
', function(done) {
1199 selectText: "RVR - RevolutionVR",
1200 firstAddress: "VXeeoP2jkzZnMFxtc66ZBZK1NHN5QJnnjL",
1202 testNetwork(done, params);
1204 it('Allows selection
of Rubycoin
', function(done) {
1206 selectText: "RBY - Rubycoin",
1207 firstAddress: "RV76JDtjTs11JdMDRToYn6CHecMRPLnKS6",
1209 testNetwork(done, params);
1211 it('Allows selection
of Salus
', function(done) {
1213 selectText: "SLS - Salus",
1214 firstAddress: "SNzPi1CafHFm3WWjRo43aMgiaEEj3ogjww",
1216 testNetwork(done, params);
1218 it('Allows selection
of Smileycoin
', function(done) {
1220 selectText: "SMLY - Smileycoin",
1221 firstAddress: "BEZVnEBCAyFByrgKpwAgYgtvP4rKAd9Sj2",
1223 testNetwork(done, params);
1225 it('Allows selection
of Solarcoin
', function(done) {
1227 selectText: "SLR - Solarcoin",
1228 firstAddress: "8LZ13HbnjtaMJWSvvVFNTLf71zFfDrhwLu",
1230 testNetwork(done, params);
1232 it('Allows selection
of stash
', function(done) {
1234 selectText: "STASH - Stash",
1235 firstAddress: "XxwAsWB7REDKmAvHA85SbEZQQtpxeUDxS3",
1237 testNetwork(done, params);
1239 it('Allows selection
of stash testnet
', function(done) {
1241 selectText: "STASH - Stash Testnet",
1242 firstAddress: "yWQCTSkUst7ddYuebKsqa1kSoXEjpCkGKR",
1244 testNetwork(done, params);
1246 it('Allows selection
of Stratis
', function(done) {
1248 selectText: "STRAT - Stratis",
1249 firstAddress: "ScfJnq3QDhKgDMEds6sqUE1ot6ShfhmXXq",
1251 testNetwork(done, params);
1253 it('Allows selection
of Stratis Test
', function(done) {
1255 selectText: "TSTRAT - Stratis Testnet",
1256 firstAddress: "TRLWm3dye4FRrDWouwYUSUZP96xb76mBE3",
1258 testNetwork(done, params);
1260 it('Allows selection
of Syscoin
', function(done) {
1262 selectText: "SYS - Syscoin",
1263 firstAddress: "SZwJi42Pst3VAMomyK5DG4157WM5ofRmSj",
1265 testNetwork(done, params);
1267 it('Allows selection
of Toa
', function(done) {
1269 selectText: "TOA - Toa",
1270 firstAddress: "TSe1QAnUwQzUfbBusDzRJ9URttrRGKoNKF",
1272 testNetwork(done, params);
1274 it('Allows selection
of TWINS
', function(done) {
1276 selectText: "TWINS - TWINS",
1277 firstAddress: "WPpJnfLLubNmF7HLNxg8d8zH5haxn4wri8",
1279 testNetwork(done, params);
1281 it('Allows selection
of TWINS testnet
', function(done) {
1283 selectText: "TWINS - TWINS Testnet",
1284 firstAddress: "XpnU1HHdNG5YxvG9Rez4wjmidchxqnZaNa",
1286 testNetwork(done, params);
1288 it('Allows selection
of Ultimatesecurecash
', function(done) {
1290 selectText: "USC - Ultimatesecurecash",
1291 firstAddress: "UPyLAZU2Che5fiy7Ed8xVJFmXAUhitA4ug",
1293 testNetwork(done, params);
1295 it('Allows selection
of Unobtanium
', function(done) {
1297 selectText: "UNO - Unobtanium",
1298 firstAddress: "uUBMPVMXrR6qhqornJqKTWgr8L69vihSL9",
1300 testNetwork(done, params);
1302 it('Allows selection
of Vcash
', function(done) {
1304 selectText: "XVC - Vcash",
1305 firstAddress: "VuL53MSY6KjvAjKSeRkh3NDnKykacDVeps",
1307 testNetwork(done, params);
1309 it('Allows selection
of Verge
', function(done) {
1311 selectText: "XVG - Verge",
1312 firstAddress: "DCrVuGkMjLJpTGgwAgv9AcMdeb1nkWbjZA",
1314 testNetwork(done, params);
1316 it('Allows selection
of Vertcoin
', function(done) {
1318 selectText: "VTC - Vertcoin",
1319 firstAddress: "Vf6koGuiWdXQfx8tNqxoNeEDxh4xh5cxsG",
1321 testNetwork(done, params);
1323 it('Allows selection
of Vivo
', function(done) {
1325 selectText: "VIVO - Vivo",
1326 firstAddress: "VFmBwuXXGhJe7MarQG2GfzHMFebRHgfSpB",
1328 testNetwork(done, params);
1330 it('Allows selection
of Vpncoin
', function(done) {
1332 selectText: "VASH - Vpncoin",
1333 firstAddress: "VoEmH1qXC4TsSgBAStR21QYetwnFqbqCx9",
1335 testNetwork(done, params);
1337 it('Allows selection
of VeChain
', function(done) {
1339 selectText: "VET - VeChain",
1340 firstAddress: "0xdba55B1B6070f3a733D5eDFf35F0da4A00E455F2",
1342 testNetwork(done, params);
1344 it('Allows selection
of Whitecoin
', function(done) {
1346 selectText: "XWC - Whitecoin",
1347 firstAddress: "WcSwCAUqrSgeSYbsaS3SSWWhsx8KRYTFDR",
1349 testNetwork(done, params);
1351 it('Allows selection
of Wincoin
', function(done) {
1353 selectText: "WC - Wincoin",
1354 firstAddress: "WaDVCESMGgyKgNESdn3u43NnwmGSkZED3Z",
1356 testNetwork(done, params);
1358 it('Allows selection
of Zcoin
', function(done) {
1360 selectText: "XZC - Zcoin",
1361 firstAddress: "a6VcMdP4XgAA9Tr7xNszmPG5FZpfRf17Cq",
1363 testNetwork(done, params);
1365 it('Allows selection
of Zcash
', function(done) {
1367 selectText: "ZEC - Zcash",
1368 firstAddress: "t1Sz8AneMcVuzUg3tPJ8et5AS5LFJ7K2EF9",
1370 testNetwork(done, params);
1372 it('Allows selection
of Zclassic
', function(done) {
1374 selectText: "ZCL - Zclassic",
1375 firstAddress: "t1TBMxTvVJRybUbMLGWq8H4A8F4VUL7czEc",
1377 testNetwork(done, params);
1379 it('Allows selection
of Horizen
', function(done) {
1381 selectText: "ZEN - Horizen",
1382 firstAddress: "znWh9XASyW2dZq5tck84wFjiwuqVysi7q3p",
1384 testNetwork(done, params);
1386 it('Allows selection
of Energi
', function(done) {
1388 selectText: "NRG - Energi",
1389 firstAddress: "EejRy4t4nidzhGGzkJUgFP3z4HYBjhTsRt",
1391 testNetwork(done, params);
1393 it('Allows selection
of Ethereum Classic
', function(done) {
1395 selectText: "ETC - Ethereum Classic",
1396 firstAddress: "0x3c05e5556693808367afB62eF3b63e35d6eD249A",
1398 testNetwork(done, params);
1400 it('Allows selection
of Pirl
', function(done) {
1402 selectText: "PIRL - Pirl",
1403 firstAddress: "0xe77FC0723dA122B5025CA79193c28563eB47e776",
1405 testNetwork(done, params);
1407 it('Allows selection
of MIX
', function(done) {
1409 selectText: "MIX - MIX",
1410 firstAddress: "0x98BC5e63aeb6A4e82d72850d20710F07E29A29F1",
1412 testNetwork(done, params);
1414 it('Allows selection
of Musicoin
', function(done) {
1416 selectText: "MUSIC - Musicoin",
1417 firstAddress: "0xDc060e4A0b0313ea83Cf6B3A39B9db2D29004897",
1419 testNetwork(done, params);
1421 it('Allows selection
of Poa
', function(done) {
1423 selectText: "POA - Poa",
1424 firstAddress: "0x53aF28d754e106210C3d0467Dd581eaf7e3C5e60",
1426 testNetwork(done, params);
1428 it('Allows selection
of Expanse
', function(done) {
1430 selectText: "EXP - Expanse",
1431 firstAddress: "0xf57FeAbf26582b6E3E666559d3B1Cc6fB2b2c5F6",
1433 testNetwork(done, params);
1435 it('Allows selection
of Callisto
', function(done) {
1437 selectText: "CLO - Callisto",
1438 firstAddress: "0x4f9364F7420B317266C51Dc8eB979717D4dE3f4E",
1440 testNetwork(done, params);
1442 it('Allows selection
of HUSH
', function(done) {
1444 selectText: "HUSH - Hush (Legacy)",
1445 firstAddress: "t1g6rLXUnJaiJuu4q4zmJjoa9Gk4fwKpiuA",
1447 testNetwork(done, params);
1449 it('Allows selection
of HUSH3
', function(done) {
1451 selectText: "HUSH - Hush3",
1452 firstAddress: "RXWSQhwvw5jHPGP8bjwJhWoRnMLBnuPDKD",
1454 testNetwork(done, params);
1456 it('Allows selection
of ExchangeCoin
', function(done) {
1458 selectText: "EXCC - ExchangeCoin",
1459 firstAddress: "22txYKpFN5fwGwdSs2UBf7ywewbLM92YqK7E",
1461 testNetwork(done, params);
1463 it('Allows selection
of Artax
', function(done) {
1465 selectText: "XAX - Artax",
1466 firstAddress: "AYxaQPY7XLidG31V7F3yNzwxPYpYzRqG4q",
1468 testNetwork(done, params);
1470 it('Allows selection
of BitcoinGreen
', function(done) {
1472 selectText: "BITG - Bitcoin Green",
1473 firstAddress: "GeNGm9SkEfwbsws3UrrUSE2sJeyWYjzraY",
1475 testNetwork(done, params);
1477 it('Allows selection
of ANON
', function(done) {
1479 selectText: "ANON - ANON",
1480 firstAddress: "AnU6pijpEeUZFWSTyM2qTqZQn996Zq1Xard",
1482 testNetwork(done, params);
1484 it('Allows selection
of ProjectCoin
', function(done) {
1486 selectText: "PRJ - ProjectCoin",
1487 firstAddress: "PXZG97saRseSCftfe1mcFmfAA7pf6qBbaz",
1489 testNetwork(done, params);
1491 it('Allows selection
of Phore
', function(done) {
1493 selectText: "PHR - Phore",
1494 firstAddress: "PJThxpoXAG6hqrmdeQQbVDX4TJtFTMMymC",
1496 testNetwork(done, params);
1498 it('Allows selection
of Safecoin
', function(done) {
1500 selectText: "SAFE - Safecoin",
1501 firstAddress: "RtxHpnhJz6RY8k9owP3ua5QWraunmewB1G",
1503 testNetwork(done, params);
1505 it('Allows selection
of Blocknode
', function(done) {
1507 selectText: "BND - Blocknode",
1508 firstAddress: "BG8xZSAur2jYLG9VXt8dYfkKxxeR7w9bSe",
1510 testNetwork(done, params);
1512 it('Allows selection
of Blocknode Testnet
', function(done) {
1514 selectText: "tBND - Blocknode Testnet",
1515 firstAddress: "bSptsFyDktFSKpWveRywJsDoJA2TC6qfHv",
1517 testNetwork(done, params);
1519 it('Allows selection
of LitecoinZ
', function(done) {
1521 selectText: "LTZ - LitecoinZ",
1522 firstAddress: "L1VTXju7hLgKV4T7fGXS9sKsnm2gmtRCmyw",
1524 testNetwork(done, params);
1526 it('Allows selection
of BlockStamp
', function(done) {
1528 selectText: "BST - BlockStamp",
1529 firstAddress: "15gypKtim4cVTj137ApfryG17RkvSbPazZ",
1531 testNetwork(done, params);
1533 it('Allows selection
of DEXON
', function(done) {
1535 selectText: "DXN - DEXON",
1536 firstAddress: "0x136a58788033E028CCd740FbDec6734358DB56Ec",
1538 testNetwork(done, params);
1540 it('Allows selection
of Ellaism
', function(done) {
1542 selectText: "ELLA - Ellaism",
1543 firstAddress: "0xa8B0BeA09eeBc41062308546a01d6E544277e2Ca",
1545 testNetwork(done, params);
1547 it('Allows selection
of Ethersocial Network
', function(done) {
1549 selectText: "ESN - Ethersocial Network",
1550 firstAddress: "0x6EE99Be2A0C7F887a71e21C8608ACF0aa0D2b767",
1552 testNetwork(done, params);
1554 it('Allows selection
of Stellar
', function(done) {
1556 selectText: "XLM - Stellar",
1557 firstAddress: "GCUK3NYYUXA2QGN6KU5RR36WAKN3Y5EANZV65XNAWN4XM4CHQ3G4DMO2",
1559 testNetwork(done, params);
1561 it('Allows selection
of Wagerr
', function(done) {
1563 selectText: "WGR - Wagerr",
1564 firstAddress: "WYiVgQU39VcQxcnacoCiaZHZZLjDCJoS95",
1566 testNetwork(done, params);
1569 // BIP39 seed is set from phrase
1570 it('Sets the bip39 seed
from the prhase
', function(done) {
1571 driver.findElement(By.css('.phrase
'))
1572 .sendKeys('abandon abandon ability
');
1573 driver.sleep(generateDelay).then(function() {
1574 driver.findElement(By.css('.seed
'))
1575 .getAttribute("value")
1576 .then(function(seed) {
1577 expect(seed).toBe("20da140d3dd1df8713cefcc4d54ce0e445b4151027a1ab567b832f6da5fcc5afc1c3a3f199ab78b8e0ab4652efd7f414ac2c9a3b81bceb879a70f377aa0a58f3");
1583 // BIP32 root key is set from phrase
1584 it('Sets the bip39 root key
from the prhase
', function(done) {
1585 driver.findElement(By.css('.phrase
'))
1586 .sendKeys('abandon abandon ability
');
1587 driver.sleep(generateDelay).then(function() {
1588 driver.findElement(By.css('.root
-key
'))
1589 .getAttribute("value")
1590 .then(function(seed) {
1591 expect(seed).toBe("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
1597 // Tabs show correct addresses when changed
1598 it('Shows the correct address when tab is changed
', function(done) {
1599 driver.findElement(By.css('.phrase
'))
1600 .sendKeys('abandon abandon ability
');
1601 driver.sleep(generateDelay).then(function() {
1602 driver.findElement(By.css('#bip32
-tab a
'))
1604 driver.sleep(generateDelay).then(function() {
1605 getFirstAddress(function(address) {
1606 expect(address).toBe("17uQ7s2izWPwBmEVFikTmZUjbBKWYdJchz");
1613 // BIP44 derivation path is shown
1614 it('Shows the derivation path
for bip44 tab
', function(done) {
1615 driver.findElement(By.css('.phrase
'))
1616 .sendKeys('abandon abandon ability
');
1617 driver.sleep(generateDelay).then(function() {
1618 driver.findElement(By.css('#bip44
.path
'))
1619 .getAttribute("value")
1620 .then(function(path) {
1621 expect(path).toBe("m/44'/0'/0'/0");
1627 // BIP44 extended private key is shown
1628 it('Shows the extended private key for bip44 tab', function(done) {
1629 driver.findElement(By.css('.phrase'))
1630 .sendKeys('abandon abandon ability');
1631 driver.sleep(generateDelay).then(function() {
1632 driver.findElement(By.css('.extended-priv-key'))
1633 .getAttribute("value
")
1634 .then(function(path) {
1635 expect(path).toBe("xprvA2DxxvPZcyRvYgZMGS53nadR32mVDeCyqQYyFhrCVbJNjPoxMeVf7QT5g7mQASbTf9Kp4cryvcXnu2qurjWKcrdsr91jXymdCDNxKgLFKJG
");
1641 // BIP44 extended public key is shown
1642 it('Shows the extended public key for bip44 tab', function(done) {
1643 driver.findElement(By.css('.phrase'))
1644 .sendKeys('abandon abandon ability');
1645 driver.sleep(generateDelay).then(function() {
1646 driver.findElement(By.css('.extended-pub-key'))
1647 .getAttribute("value
")
1648 .then(function(path) {
1649 expect(path).toBe("xpub6FDKNRvTTLzDmAdpNTc49ia9b4byd6vqCdUa46Fp3vqMcC96uBoufCmZXQLiN5AK3iSCJMhf9gT2sxkpyaPepRuA7W3MujV5tGmF5VfbueM
");
1655 // BIP44 account field changes address list
1656 it('Changes the address list if bip44 account is changed', function(done) {
1657 driver.findElement(By.css('#bip44 .account'))
1659 driver.findElement(By.css('.phrase'))
1660 .sendKeys('abandon abandon ability');
1661 driver.sleep(generateDelay).then(function() {
1662 getFirstAddress(function(address) {
1663 expect(address).toBe("1Nq2Wmu726XHCuGhctEtGmhxo3wzk5wZ1H
");
1669 // BIP44 change field changes address list
1670 it('Changes the address list if bip44 change is changed', function(done) {
1671 driver.findElement(By.css('#bip44 .change'))
1673 driver.findElement(By.css('.phrase'))
1674 .sendKeys('abandon abandon ability');
1675 driver.sleep(generateDelay).then(function() {
1676 getFirstAddress(function(address) {
1677 expect(address).toBe("1KAGfWgqfVbSSXY56fNQ7YnhyKuoskHtYo
");
1683 // BIP32 derivation path can be set
1684 it('Can use a custom bip32 derivation path', function(done) {
1685 driver.findElement(By.css('#bip32-tab a'))
1687 driver.findElement(By.css('#bip32 .path'))
1689 driver.findElement(By.css('#bip32 .path'))
1691 driver.findElement(By.css('.phrase'))
1692 .sendKeys('abandon abandon ability');
1693 driver.sleep(generateDelay).then(function() {
1694 getFirstAddress(function(address) {
1695 expect(address).toBe("16pYQQdLD1hH4hwTGLXBaZ9Teboi1AGL8L
");
1701 // BIP32 can use hardened derivation paths
1702 it('Can use a hardened derivation paths', function(done) {
1703 driver.findElement(By.css('#bip32-tab a'))
1705 driver.findElement(By.css('#bip32 .path'))
1707 driver.findElement(By.css('#bip32 .path'))
1709 driver.findElement(By.css('.phrase
'))
1710 .sendKeys('abandon abandon ability
');
1711 driver.sleep(generateDelay).then(function() {
1712 getFirstAddress(function(address) {
1713 expect(address).toBe("14aXZeprXAE3UUKQc4ihvwBvww2LuEoHo4");
1719 // BIP32 extended private key is shown
1720 it('Shows the BIP32 extended
private key
', function(done) {
1721 driver.findElement(By.css('#bip32
-tab a
'))
1723 driver.findElement(By.css('.phrase
'))
1724 .sendKeys('abandon abandon ability
');
1725 driver.sleep(generateDelay).then(function() {
1726 driver.findElement(By.css('.extended
-priv
-key
'))
1727 .getAttribute("value")
1728 .then(function(privKey) {
1729 expect(privKey).toBe("xprv9va99uTVE5aLiutUVLTyfxfe8v8aaXjSQ1XxZbK6SezYVuikA9MnjQVTA8rQHpNA5LKvyQBpLiHbBQiiccKiBDs7eRmBogsvq3THFeLHYbe");
1735 // BIP32 extended public key is shown
1736 it('Shows the BIP32 extended
public key
', function(done) {
1737 driver.findElement(By.css('#bip32
-tab a
'))
1739 driver.findElement(By.css('.phrase
'))
1740 .sendKeys('abandon abandon ability
');
1741 driver.sleep(generateDelay).then(function() {
1742 driver.findElement(By.css('.extended
-pub
-key
'))
1743 .getAttribute("value")
1744 .then(function(pubKey) {
1745 expect(pubKey).toBe("xpub69ZVZQzP4T8dwPxwbMzz36cNgwy4yzTHmETZMyihzzXXNi3thgg3HCow1RtY252wdw5rS8369xKnraN5Q93y3FkFfJp2XEHWUrkyXsjS93P");
1751 // Derivation path is shown in table
1752 it('Shows the derivation path
in the table
', function(done) {
1753 driver.findElement(By.css('.phrase
'))
1754 .sendKeys('abandon abandon ability
');
1755 driver.sleep(generateDelay).then(function() {
1756 getFirstPath(function(path) {
1757 expect(path).toBe("m/44'/0'/0'/0/0");
1763 // Derivation path for address can be hardened
1764 it('Can derive hardened addresses', function(done) {
1765 driver.findElement(By.css('#bip32-tab a'))
1767 driver.executeScript(function() {
1768 $(".hardened
-addresses
").prop("checked
", true);
1770 driver.findElement(By.css('.phrase'))
1771 .sendKeys('abandon abandon ability');
1772 driver.sleep(generateDelay).then(function() {
1773 getFirstAddress(function(address) {
1774 expect(address).toBe("18exLzUv7kfpiXRzmCjFDoC9qwNLFyvwyd
");
1780 // Derivation path visibility can be toggled
1781 it('Can toggle visibility of the derivation path column', function(done) {
1782 driver.findElement(By.css('.phrase'))
1783 .sendKeys('abandon abandon ability');
1784 driver.sleep(generateDelay).then(function() {
1785 driver.findElement(By.css('.index-toggle'))
1787 testColumnValuesAreInvisible(done, "index
");
1792 it('Shows the address in the table', function(done) {
1793 driver.findElement(By.css('.phrase'))
1794 .sendKeys('abandon abandon ability');
1795 driver.sleep(generateDelay).then(function() {
1796 getFirstAddress(function(address) {
1797 expect(address).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug
");
1803 // Addresses are shown in order of derivation path
1804 it('Shows the address in order of derivation path', function(done) {
1805 driver.findElement(By.css('.phrase'))
1806 .sendKeys('abandon abandon ability');
1807 driver.sleep(generateDelay).then(function() {
1808 testRowsAreInCorrectOrder(done);
1812 // Address visibility can be toggled
1813 it('Can toggle visibility of the address column', function(done) {
1814 driver.findElement(By.css('.phrase'))
1815 .sendKeys('abandon abandon ability');
1816 driver.sleep(generateDelay).then(function() {
1817 driver.findElement(By.css('.address-toggle'))
1819 testColumnValuesAreInvisible(done, "address
");
1823 // Public key is shown in table
1824 it('Shows the public key in the table', function(done) {
1825 driver.findElement(By.css('.phrase'))
1826 .sendKeys('abandon abandon ability');
1827 driver.sleep(generateDelay).then(function() {
1828 driver.findElements(By.css('.pubkey'))
1829 .then(function(els) {
1831 .then(function(pubkey) {
1832 expect(pubkey).toBe("033f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3
");
1839 // Public key visibility can be toggled
1840 it('Can toggle visibility of the public key column', function(done) {
1841 driver.findElement(By.css('.phrase'))
1842 .sendKeys('abandon abandon ability');
1843 driver.sleep(generateDelay).then(function() {
1844 driver.findElement(By.css('.public-key-toggle'))
1846 testColumnValuesAreInvisible(done, "pubkey
");
1850 // Private key is shown in table
1851 it('Shows the private key in the table', function(done) {
1852 driver.findElement(By.css('.phrase'))
1853 .sendKeys('abandon abandon ability');
1854 driver.sleep(generateDelay).then(function() {
1855 driver.findElements(By.css('.privkey'))
1856 .then(function(els) {
1858 .then(function(pubkey) {
1859 expect(pubkey).toBe("L26cVSpWFkJ6aQkPkKmTzLqTdLJ923e6CzrVh9cmx21QHsoUmrEE
");
1866 // Private key visibility can be toggled
1867 it('Can toggle visibility of the private key column', function(done) {
1868 driver.findElement(By.css('.phrase'))
1869 .sendKeys('abandon abandon ability');
1870 driver.sleep(generateDelay).then(function() {
1871 driver.findElement(By.css('.private-key-toggle'))
1873 testColumnValuesAreInvisible(done, "privkey
");
1877 // More addresses can be generated
1878 it('Can generate more rows in the table', function(done) {
1879 driver.findElement(By.css('.phrase'))
1880 .sendKeys('abandon abandon ability');
1881 driver.sleep(generateDelay).then(function() {
1882 driver.findElement(By.css('.more'))
1884 driver.sleep(generateDelay).then(function() {
1885 driver.findElements(By.css('.address'))
1886 .then(function(els) {
1887 expect(els.length).toBe(40);
1894 // A custom number of additional addresses can be generated
1895 it('Can generate more rows in the table', function(done) {
1896 driver.findElement(By.css('.phrase'))
1897 .sendKeys('abandon abandon ability');
1898 driver.sleep(generateDelay).then(function() {
1899 driver.findElement(By.css('.rows-to-add'))
1901 driver.findElement(By.css('.rows-to-add'))
1903 driver.findElement(By.css('.more'))
1905 driver.sleep(generateDelay).then(function() {
1906 driver.findElements(By.css('.address'))
1907 .then(function(els) {
1908 expect(els.length).toBe(21);
1915 // Additional addresses are shown in order of derivation path
1916 it('Shows additional addresses in order of derivation path', function(done) {
1917 driver.findElement(By.css('.phrase'))
1918 .sendKeys('abandon abandon ability');
1919 driver.sleep(generateDelay).then(function() {
1920 driver.findElement(By.css('.more'))
1922 driver.sleep(generateDelay).then(function() {
1923 testRowsAreInCorrectOrder(done);
1928 // BIP32 root key can be set by the user
1929 it('Allows the user to set the BIP32 root key', function(done) {
1930 driver.findElement(By.css('.root-key'))
1931 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1932 driver.sleep(generateDelay).then(function() {
1933 getFirstAddress(function(address) {
1934 expect(address).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug
");
1940 // Setting BIP32 root key clears the existing phrase, passphrase and seed
1941 // TODO this doesn't work in selenium with chrome
1942 it('Confirms the existing phrase should be cleared', function(done) {
1943 if (browser == "chrome
") {
1944 pending("Selenium
+ Chrome headless bug
for alert
, see
https://stackoverflow.com/q/45242264");
1946 driver
.findElement(By
.css('.phrase'))
1947 .sendKeys('A non-blank but invalid value');
1948 driver
.findElement(By
.css('.root-key'))
1949 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1950 driver
.switchTo().alert().accept();
1951 driver
.findElement(By
.css('.phrase'))
1952 .getAttribute("value").then(function(value
) {
1953 expect(value
).toBe("");
1958 // Clearing of phrase, passphrase and seed can be cancelled by user
1959 // TODO this doesn't work in selenium with chrome
1960 it('Allows the clearing of the phrase to be cancelled', function(done
) {
1961 if (browser
== "chrome") {
1962 pending("Selenium + Chrome headless bug for alert, see https://stackoverflow.com/q/45242264");
1964 driver
.findElement(By
.css('.phrase'))
1965 .sendKeys('abandon abandon ability');
1966 driver
.sleep(generateDelay
).then(function() {
1967 driver
.findElement(By
.css('.root-key'))
1969 driver
.findElement(By
.css('.root-key'))
1971 driver
.switchTo().alert().dismiss();
1972 driver
.findElement(By
.css('.phrase'))
1973 .getAttribute("value").then(function(value
) {
1974 expect(value
).toBe("abandon abandon ability");
1980 // Custom BIP32 root key is used when changing the derivation path
1981 it('Can set derivation path for root key instead of phrase', function(done
) {
1982 driver
.findElement(By
.css('#bip44 .account'))
1984 driver
.findElement(By
.css('.root-key'))
1985 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1986 driver
.sleep(generateDelay
).then(function() {
1987 getFirstAddress(function(address
) {
1988 expect(address
).toBe("1Nq2Wmu726XHCuGhctEtGmhxo3wzk5wZ1H");
1994 // Incorrect mnemonic shows error
1995 it('Shows an error for incorrect mnemonic', function(done
) {
1996 driver
.findElement(By
.css('.phrase'))
1997 .sendKeys('abandon abandon abandon');
1998 driver
.sleep(feedbackDelay
).then(function() {
1999 driver
.findElement(By
.css('.feedback'))
2001 .then(function(feedback
) {
2002 expect(feedback
).toBe("Invalid mnemonic");
2008 // Incorrect word shows suggested replacement
2009 it('Shows word suggestion for incorrect word', function(done
) {
2010 driver
.findElement(By
.css('.phrase'))
2011 .sendKeys('abandon abandon abiliti');
2012 driver
.sleep(feedbackDelay
).then(function() {
2013 driver
.findElement(By
.css('.feedback'))
2015 .then(function(feedback
) {
2016 var msg
= "abiliti not in wordlist, did you mean ability?";
2017 expect(feedback
).toBe(msg
);
2023 // Github pull request 48
2024 // First four letters of word shows that word, not closest
2025 // since first four letters gives unique word in BIP39 wordlist
2026 // eg ille should show illegal, not idle
2027 it('Shows word suggestion based on first four chars', function(done
) {
2028 driver
.findElement(By
.css('.phrase'))
2030 driver
.sleep(feedbackDelay
).then(function() {
2031 driver
.findElement(By
.css('.feedback'))
2033 .then(function(feedback
) {
2034 var msg
= "ille not in wordlist, did you mean illegal?";
2035 expect(feedback
).toBe(msg
);
2041 // Incorrect BIP32 root key shows error
2042 it('Shows error for incorrect root key', function(done
) {
2043 driver
.findElement(By
.css('.root-key'))
2044 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpj');
2045 driver
.sleep(feedbackDelay
).then(function() {
2046 driver
.findElement(By
.css('.feedback'))
2048 .then(function(feedback
) {
2049 var msg
= "Invalid root key";
2050 expect(feedback
).toBe(msg
);
2056 // Derivation path not starting with m shows error
2057 it('Shows error for derivation path not starting with m', function(done
) {
2058 driver
.findElement(By
.css('#bip32-tab a'))
2060 driver
.findElement(By
.css('#bip32 .path'))
2062 driver
.findElement(By
.css('#bip32 .path'))
2064 driver
.findElement(By
.css('.phrase'))
2065 .sendKeys('abandon abandon ability');
2066 driver
.sleep(feedbackDelay
).then(function() {
2067 driver
.findElement(By
.css('.feedback'))
2069 .then(function(feedback
) {
2070 var msg
= "First character must be 'm'";
2071 expect(feedback
).toBe(msg
);
2077 // Derivation path containing invalid characters shows useful error
2078 it('Shows error for derivation path not starting with m', function(done
) {
2079 driver
.findElement(By
.css('#bip32-tab a'))
2081 driver
.findElement(By
.css('#bip32 .path'))
2083 driver
.findElement(By
.css('#bip32 .path'))
2084 .sendKeys('m/1/0wrong1/1');
2085 driver
.findElement(By
.css('.phrase'))
2086 .sendKeys('abandon abandon ability');
2087 driver
.sleep(feedbackDelay
).then(function() {
2088 driver
.findElement(By
.css('.feedback'))
2090 .then(function(feedback
) {
2091 var msg
= "Invalid characters 0wrong1 found at depth 2";
2092 expect(feedback
).toBe(msg
);
2098 // Github Issue 11: Default word length is 15
2099 // https://github.com/iancoleman/bip39/issues/11
2100 it('Sets the default word length to 15', function(done
) {
2101 driver
.findElement(By
.css('.strength'))
2102 .getAttribute("value")
2103 .then(function(strength
) {
2104 expect(strength
).toBe("15");
2109 // Github Issue 12: Generate more rows with private keys hidden
2110 // https://github.com/iancoleman/bip39/issues/12
2111 it('Sets the correct hidden column state on new rows', function(done
) {
2112 driver
.findElement(By
.css('.phrase'))
2113 .sendKeys("abandon abandon ability");
2114 driver
.sleep(generateDelay
).then(function() {
2115 driver
.findElement(By
.css('.private-key-toggle'))
2117 driver
.findElement(By
.css('.more'))
2119 driver
.sleep(generateDelay
).then(function() {
2120 driver
.findElements(By
.css('.privkey'))
2121 .then(function(els
) {
2122 expect(els
.length
).toBe(40);
2124 testColumnValuesAreInvisible(done
, "privkey");
2129 // Github Issue 19: Mnemonic is not sensitive to whitespace
2130 // https://github.com/iancoleman/bip39/issues/19
2131 it('Ignores excess whitespace in the mnemonic', function(done
) {
2132 var doublespace
= " ";
2133 var mnemonic
= "urge cat" + doublespace
+ "bid";
2134 driver
.findElement(By
.css('.phrase'))
2135 .sendKeys(mnemonic
);
2136 driver
.sleep(generateDelay
).then(function() {
2137 driver
.findElement(By
.css('.root-key'))
2138 .getAttribute("value")
2139 .then(function(seed
) {
2140 expect(seed
).toBe("xprv9s21ZrQH143K3isaZsWbKVoTtbvd34Y1ZGRugGdMeBGbM3AgBVzTH159mj1cbbtYSJtQr65w6L5xy5L9SFC7c9VJZWHxgAzpj4mun5LhrbC");
2146 // Github Issue 23: Part 1: Use correct derivation path when changing tabs
2147 // https://github.com/iancoleman/bip39/issues/23
2148 it('Uses the correct derivation path when changing tabs', function(done
) {
2149 // 1) and 2) set the phrase
2150 driver
.findElement(By
.css('.phrase'))
2151 .sendKeys("abandon abandon ability");
2152 driver
.sleep(generateDelay
).then(function() {
2153 // 3) select bip32 tab
2154 driver
.findElement(By
.css('#bip32-tab a'))
2156 driver
.sleep(generateDelay
).then(function() {
2157 // 4) switch from bitcoin to litecoin
2158 selectNetwork("LTC - Litecoin");
2159 driver
.sleep(generateDelay
).then(function() {
2160 // 5) Check address is displayed correctly
2161 getFirstAddress(function(address
) {
2162 expect(address
).toBe("LS8MP5LZ5AdzSZveRrjm3aYVoPgnfFh5T5");
2163 // 5) Check derivation path is displayed correctly
2164 getFirstPath(function(path
) {
2165 expect(path
).toBe("m/0/0");
2174 // Github Issue 23 Part 2: Coin selection in derivation path
2175 // https://github.com/iancoleman/bip39/issues/23#issuecomment-238011920
2176 it('Uses the correct derivation path when changing coins', function(done
) {
2178 driver
.findElement(By
.css('.phrase'))
2179 .sendKeys("abandon abandon ability");
2180 driver
.sleep(generateDelay
).then(function() {
2181 // switch from bitcoin to clam
2182 selectNetwork("CLAM - Clams");
2183 driver
.sleep(generateDelay
).then(function() {
2184 // check derivation path is displayed correctly
2185 getFirstPath(function(path
) {
2186 expect(path
).toBe("m/44'/23'/0'/0/0");
2193 // Github Issue 26: When using a Root key derrived altcoins are incorrect
2194 // https://github.com/iancoleman/bip39/issues/26
2195 it('Uses the correct derivation for altcoins with root keys', function(done
) {
2196 // 1) 2) and 3) set the root key
2197 driver
.findElement(By
.css('.root-key'))
2198 .sendKeys("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
2199 driver
.sleep(generateDelay
).then(function() {
2200 // 4) switch from bitcoin to viacoin
2201 selectNetwork("VIA - Viacoin");
2202 driver
.sleep(generateDelay
).then(function() {
2203 // 5) ensure the derived address is correct
2204 getFirstAddress(function(address
) {
2205 expect(address
).toBe("Vq9Eq4N5SQnjqZvxtxzo7hZPW5XnyJsmXT");
2212 // Selecting a language with no existing phrase should generate a phrase in
2214 it('Generate a random phrase when language is selected and no current phrase', function(done
) {
2215 driver
.findElement(By
.css("a[href='#japanese']"))
2217 driver
.sleep(generateDelay
).then(function() {
2218 driver
.findElement(By
.css(".phrase"))
2219 .getAttribute("value").then(function(phrase
) {
2220 expect(phrase
.search(/[a-z]/)).toBe(-1);
2221 expect(phrase
.length
).toBeGreaterThan(0);
2227 // Selecting a language with existing phrase should update the phrase to use
2229 it('Updates existing phrases when the language is changed', function(done
) {
2230 driver
.findElement(By
.css(".phrase"))
2231 .sendKeys("abandon abandon ability");
2232 driver
.sleep(generateDelay
).then(function() {
2233 driver
.findElement(By
.css("a[href='#italian']"))
2235 driver
.sleep(generateDelay
).then(function() {
2236 driver
.findElement(By
.css(".phrase"))
2237 .getAttribute("value").then(function(phrase
) {
2238 // Check only the language changes, not the phrase
2239 expect(phrase
).toBe("abaco abaco abbaglio");
2240 getFirstAddress(function(address
) {
2241 // Check the address is correct
2242 expect(address
).toBe("1Dz5TgDhdki9spa6xbPFbBqv5sjMrx3xgV");
2250 // Suggested replacement for erroneous word in non-English language
2251 it('Shows word suggestion for incorrect word in non-English language', function(done
) {
2252 driver
.findElement(By
.css('.phrase'))
2253 .sendKeys('abaco abaco zbbaglio');
2254 driver
.sleep(feedbackDelay
).then(function() {
2255 driver
.findElement(By
.css('.feedback'))
2257 .then(function(feedback
) {
2258 var msg
= "zbbaglio not in wordlist, did you mean abbaglio?";
2259 expect(feedback
).toBe(msg
);
2265 // Japanese word does not break across lines.
2267 // https://github.com/bitcoin/bips/blob/master/bip-0039/bip-0039-wordlists.md#japanese
2268 it('Does not break Japanese words across lines', function(done
) {
2269 driver
.findElement(By
.css('.phrase'))
2270 .getCssValue("word-break")
2271 .then(function(value
) {
2272 expect(value
).toBe("keep-all");
2277 // Language can be specified at page load using hash value in url
2278 it('Can set the language from the url hash', function(done
) {
2279 driver
.get(url
+ "#japanese").then(function() {
2280 driver
.findElement(By
.css('.generate')).click();
2281 driver
.sleep(generateDelay
).then(function() {
2282 driver
.findElement(By
.css(".phrase"))
2283 .getAttribute("value").then(function(phrase
) {
2284 expect(phrase
.search(/[a-z]/)).toBe(-1);
2285 expect(phrase
.length
).toBeGreaterThan(0);
2292 // Entropy can be entered by the user
2293 it('Allows entropy to be entered', function(done
) {
2294 driver
.findElement(By
.css('.use-entropy'))
2296 driver
.findElement(By
.css('.entropy'))
2297 .sendKeys('00000000 00000000 00000000 00000000');
2298 driver
.sleep(generateDelay
).then(function() {
2299 driver
.findElement(By
.css(".phrase"))
2300 .getAttribute("value").then(function(phrase
) {
2301 expect(phrase
).toBe("abandon abandon ability");
2302 getFirstAddress(function(address
) {
2303 expect(address
).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug");
2310 // A warning about entropy is shown to the user, with additional information
2311 it('Shows a warning about using entropy', function(done
) {
2312 driver
.findElement(By
.css('.use-entropy'))
2314 driver
.findElement(By
.css('.entropy-container'))
2316 .then(function(containerText
) {
2317 var warning
= "mnemonic may be insecure";
2318 expect(containerText
).toContain(warning
);
2319 driver
.findElement(By
.css('#entropy-notes'))
2320 .findElement(By
.xpath("parent::*"))
2322 .then(function(notesText
) {
2323 var detail
= "flipping a fair coin, rolling a fair dice, noise measurements etc";
2324 expect(notesText
).toContain(detail
);
2330 // The types of entropy available are described to the user
2331 it('Shows the types of entropy available', function(done
) {
2332 driver
.findElement(By
.css('.entropy'))
2333 .getAttribute("placeholder")
2334 .then(function(placeholderText
) {
2343 for (var i
=0; i
<options
.length
; i
++) {
2344 var option
= options
[i
];
2345 expect(placeholderText
).toContain(option
);
2351 // The actual entropy used is shown to the user
2352 it('Shows the actual entropy used', function(done
) {
2353 driver
.findElement(By
.css('.use-entropy'))
2355 driver
.findElement(By
.css('.entropy'))
2356 .sendKeys('Not A Very Good Entropy Source At All');
2357 driver
.sleep(generateDelay
).then(function() {
2358 driver
.findElement(By
.css('.entropy-container'))
2360 .then(function(text
) {
2361 expect(text
).toMatch(/Filtered Entropy
\s
+AedEceAA
/);
2367 // Binary entropy can be entered
2368 it('Allows binary entropy to be entered', function(done
) {
2369 testEntropyType(done
, "01", "binary");
2372 // Base 6 entropy can be entered
2373 it('Allows base 6 entropy to be entered', function(done
) {
2374 testEntropyType(done
, "012345", "base 6");
2377 // Base 6 dice entropy can be entered
2378 it('Allows base 6 dice entropy to be entered', function(done
) {
2379 testEntropyType(done
, "123456", "base 6 (dice)");
2382 // Base 10 entropy can be entered
2383 it('Allows base 10 entropy to be entered', function(done
) {
2384 testEntropyType(done
, "789", "base 10");
2387 // Hexadecimal entropy can be entered
2388 it('Allows hexadecimal entropy to be entered', function(done
) {
2389 testEntropyType(done
, "abcdef", "hexadecimal");
2392 // Dice entropy value is shown as the converted base 6 value
2393 // ie 123456 is converted to 123450
2394 it('Shows dice entropy as base 6', function(done
) {
2395 driver
.findElement(By
.css('.use-entropy'))
2397 driver
.findElement(By
.css('.entropy'))
2398 .sendKeys("123456");
2399 driver
.sleep(generateDelay
).then(function() {
2400 driver
.findElement(By
.css('.entropy-container'))
2402 .then(function(text
) {
2403 expect(text
).toMatch(/Filtered Entropy
\s
+123450/);
2409 // The number of bits of entropy accumulated is shown
2410 it("Shows the number of bits of entropy for 20 bits of binary", function(done
) {
2411 testEntropyBits(done
, "0000 0000 0000 0000 0000", "20");
2413 it("Shows the number of bits of entropy for 1 bit of binary", function(done
) {
2414 testEntropyBits(done
, "0", "1");
2416 it("Shows the number of bits of entropy for 4 bits of binary", function(done
) {
2417 testEntropyBits(done
, "0000", "4");
2419 it("Shows the number of bits of entropy for 1 character of base 6 (dice)", function(done
) {
2420 // 6 in card is 0 in base 6, 0 in base 6 is 2.6 bits (rounded down to 2 bits)
2421 testEntropyBits(done
, "6", "2");
2423 it("Shows the number of bits of entropy for 1 character of base 10 with 3 bits", function(done
) {
2424 // 7 in base 10 is 111 in base 2, no leading zeros
2425 testEntropyBits(done
, "7", "3");
2427 it("Shows the number of bits of entropy for 1 character of base 10 with 4 bis", function(done
) {
2428 testEntropyBits(done
, "8", "4");
2430 it("Shows the number of bits of entropy for 1 character of hex", function(done
) {
2431 testEntropyBits(done
, "F", "4");
2433 it("Shows the number of bits of entropy for 2 characters of base 10", function(done
) {
2434 testEntropyBits(done
, "29", "6");
2436 it("Shows the number of bits of entropy for 2 characters of hex", function(done
) {
2437 testEntropyBits(done
, "0A", "8");
2439 it("Shows the number of bits of entropy for 2 characters of hex with 3 leading zeros", function(done
) {
2440 // hex is always multiple of 4 bits of entropy
2441 testEntropyBits(done
, "1A", "8");
2443 it("Shows the number of bits of entropy for 2 characters of hex with 2 leading zeros", function(done
) {
2444 testEntropyBits(done
, "2A", "8");
2446 it("Shows the number of bits of entropy for 2 characters of hex with 1 leading zero", function(done
) {
2447 testEntropyBits(done
, "4A", "8");
2449 it("Shows the number of bits of entropy for 2 characters of hex with no leading zeros", function(done
) {
2450 testEntropyBits(done
, "8A", "8");
2452 it("Shows the number of bits of entropy for 2 characters of hex starting with F", function(done
) {
2453 testEntropyBits(done
, "FA", "8");
2455 it("Shows the number of bits of entropy for 4 characters of hex with leading zeros", function(done
) {
2456 testEntropyBits(done
, "000A", "16");
2458 it("Shows the number of bits of entropy for 4 characters of base 6", function(done
) {
2459 testEntropyBits(done
, "5555", "11");
2461 it("Shows the number of bits of entropy for 4 characters of base 6 dice", function(done
) {
2462 // uses dice, so entropy is actually 0000 in base 6, which is 4 lots of
2463 // 2.58 bits, which is 10.32 bits (rounded down to 10 bits)
2464 testEntropyBits(done
, "6666", "10");
2466 it("Shows the number of bits of entropy for 4 charactes of base 10", function(done
) {
2467 // Uses base 10, which is 4 lots of 3.32 bits, which is 13.3 bits (rounded
2469 testEntropyBits(done
, "2227", "13");
2471 it("Shows the number of bits of entropy for 4 characters of hex with 2 leading zeros", function(done
) {
2472 testEntropyBits(done
, "222F", "16");
2474 it("Shows the number of bits of entropy for 4 characters of hex starting with F", function(done
) {
2475 testEntropyBits(done
, "FFFF", "16");
2477 it("Shows the number of bits of entropy for 10 characters of base 10", function(done
) {
2478 // 10 events at 3.32 bits per event
2479 testEntropyBits(done
, "0000101017", "33");
2481 it("Shows the number of bits of entropy for a full deck of cards", function(done
) {
2482 // cards are not replaced, so a full deck is not 52^52 entropy which is 296
2483 // bits, it's 52!, which is 225 bits
2484 testEntropyBits(done
, "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks", "225");
2487 it("Shows details about the entered entropy", function(done
) {
2488 testEntropyFeedback(done
,
2492 type: "hexadecimal",
2496 strength: "less than a second",
2500 it("Shows details about the entered entropy", function(done
) {
2501 testEntropyFeedback(done
,
2503 entropy: "AAAAAAAA",
2504 filtered: "AAAAAAAA",
2505 type: "hexadecimal",
2509 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2513 it("Shows details about the entered entropy", function(done
) {
2514 testEntropyFeedback(done
,
2516 entropy: "AAAAAAAA B",
2517 filtered: "AAAAAAAAB",
2518 type: "hexadecimal",
2522 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2526 it("Shows details about the entered entropy", function(done
) {
2527 testEntropyFeedback(done
,
2529 entropy: "AAAAAAAA BBBBBBBB",
2530 filtered: "AAAAAAAABBBBBBBB",
2531 type: "hexadecimal",
2535 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2539 it("Shows details about the entered entropy", function(done
) {
2540 testEntropyFeedback(done
,
2542 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC",
2543 filtered: "AAAAAAAABBBBBBBBCCCCCCCC",
2544 type: "hexadecimal",
2548 strength: "less than a second",
2552 it("Shows details about the entered entropy", function(done
) {
2553 testEntropyFeedback(done
,
2555 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDD",
2556 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDD",
2557 type: "hexadecimal",
2561 strength: "2 minutes",
2565 it("Shows details about the entered entropy", function(done
) {
2566 testEntropyFeedback(done
,
2568 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA",
2569 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDA",
2570 type: "hexadecimal",
2578 it("Shows details about the entered entropy", function(done
) {
2579 testEntropyFeedback(done
,
2581 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA EEEEEEEE",
2582 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDAEEEEEEEE",
2583 type: "hexadecimal",
2587 strength: "3 years",
2591 it("Shows details about the entered entropy", function(done
) {
2592 testEntropyFeedback(done
,
2594 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA EEEEEEEE FFFFFFFF",
2595 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDAEEEEEEEEFFFFFFFF",
2596 type: "hexadecimal",
2600 strength: "centuries",
2604 it("Shows details about the entered entropy", function(done
) {
2605 testEntropyFeedback(done
,
2612 strength: "less than a second",
2616 it("Shows details about the entered entropy", function(done
) {
2617 testEntropyFeedback(done
,
2619 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2620 type: "card (full deck)",
2624 strength: "centuries",
2628 it("Shows details about the entered entropy", function(done
) {
2629 testEntropyFeedback(done
,
2631 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks3d",
2632 type: "card (full deck, 1 duplicate: 3d)",
2636 strength: "centuries",
2640 it("Shows details about the entered entropy", function(done
) {
2641 testEntropyFeedback(done
,
2643 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqs3d4d",
2644 type: "card (2 duplicates: 3d 4d, 1 missing: KS)",
2648 strength: "centuries",
2652 it("Shows details about the entered entropy", function(done
) {
2653 testEntropyFeedback(done
,
2655 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqs3d4d5d6d",
2656 type: "card (4 duplicates: 3d 4d 5d..., 1 missing: KS)",
2660 strength: "centuries",
2664 it("Shows details about the entered entropy", function(done
) {
2665 testEntropyFeedback(done
,
2666 // Next test was throwing uncaught error in zxcvbn
2667 // Also tests 451 bits, ie Math.log2(52!)*2 = 225.58 * 2
2669 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsksac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2670 type: "card (full deck, 52 duplicates: ac 2c 3c...)",
2674 strength: "centuries",
2678 it("Shows details about the entered entropy", function(done
) {
2679 testEntropyFeedback(done
,
2680 // Case insensitivity to duplicate cards
2683 type: "card (1 duplicate: AS)",
2687 strength: "less than a second",
2691 it("Shows details about the entered entropy", function(done
) {
2692 testEntropyFeedback(done
,
2695 type: "card (1 duplicate: as)",
2699 strength: "less than a second",
2703 it("Shows details about the entered entropy", function(done
) {
2704 testEntropyFeedback(done
,
2705 // Missing cards are detected
2707 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2708 type: "card (1 missing: 9C)",
2712 strength: "centuries",
2716 it("Shows details about the entered entropy", function(done
) {
2717 testEntropyFeedback(done
,
2719 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2720 type: "card (2 missing: 9C 5D)",
2724 strength: "centuries",
2728 it("Shows details about the entered entropy", function(done
) {
2729 testEntropyFeedback(done
,
2731 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d7d8d9dtdjd kdah2h3h 5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2732 type: "card (4 missing: 9C 5D QD...)",
2736 strength: "centuries",
2740 it("Shows details about the entered entropy", function(done
) {
2741 testEntropyFeedback(done
,
2742 // More than six missing cards does not show message
2744 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d 8d9d jd kdah2h3h 5h6h7h8h9hthjhqhkh 2s3s4s5s6s7s8s9stsjsqsks",
2749 strength: "centuries",
2753 it("Shows details about the entered entropy", function(done
) {
2754 testEntropyFeedback(done
,
2755 // Multiple decks of cards increases bits per event
2760 bitsPerEvent: "4.34",
2764 it("Shows details about the entered entropy", function(done
) {
2765 testEntropyFeedback(done
,
2770 bitsPerEvent: "4.80",
2774 it("Shows details about the entered entropy", function(done
) {
2775 testEntropyFeedback(done
,
2780 bitsPerEvent: "5.01",
2784 it("Shows details about the entered entropy", function(done
) {
2785 testEntropyFeedback(done
,
2787 entropy: "3d3d3d3d",
2790 bitsPerEvent: "5.14",
2794 it("Shows details about the entered entropy", function(done
) {
2795 testEntropyFeedback(done
,
2797 entropy: "3d3d3d3d3d",
2800 bitsPerEvent: "5.22",
2804 it("Shows details about the entered entropy", function(done
) {
2805 testEntropyFeedback(done
,
2807 entropy: "3d3d3d3d3d3d",
2810 bitsPerEvent: "5.28",
2814 it("Shows details about the entered entropy", function(done
) {
2815 testEntropyFeedback(done
,
2817 entropy: "3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d",
2820 bitsPerEvent: "5.59",
2821 strength: 'less than a second - Repeats like "abcabcabc" are only slightly harder to guess than "abc"',
2826 // Entropy is truncated from the left
2827 it('Truncates entropy from the left', function(done
) {
2828 // Truncate from left means 0000 is removed from the start
2829 // which gives mnemonic 'avocado zoo zone'
2830 // not 1111 removed from the end
2831 // which gives the mnemonic 'abstract zoo zoo'
2832 var entropy
= "00000000 00000000 00000000 00000000";
2833 entropy
+= "11111111 11111111 11111111 1111"; // Missing last byte
2834 driver
.findElement(By
.css('.use-entropy'))
2836 driver
.findElement(By
.css('.entropy'))
2838 driver
.sleep(generateDelay
).then(function() {
2839 driver
.findElement(By
.css(".phrase"))
2840 .getAttribute("value").then(function(phrase
) {
2841 expect(phrase
).toBe("avocado zoo zone");
2847 // Very large entropy results in very long mnemonics
2848 it('Converts very long entropy to very long mnemonics', function(done
) {
2850 for (var i
=0; i
<33; i
++) {
2851 entropy
+= "AAAAAAAA"; // 3 words * 33 iterations = 99 words
2853 driver
.findElement(By
.css('.use-entropy'))
2855 driver
.findElement(By
.css('.entropy'))
2857 driver
.sleep(generateDelay
).then(function() {
2858 driver
.findElement(By
.css(".phrase"))
2859 .getAttribute("value").then(function(phrase
) {
2860 var wordCount
= phrase
.split(/\s+/g).length
;
2861 expect(wordCount
).toBe(99);
2867 // Is compatible with bip32jp entropy
2868 // https://bip32jp.github.io/english/index.html
2870 // Is incompatible with:
2872 it('Is compatible with bip32jp.github.io', function(done
) {
2873 var entropy
= "543210543210543210543210543210543210543210543210543210543210543210543210543210543210543210543210543";
2874 var expectedPhrase
= "train then jungle barely whip fiber purpose puppy eagle cloud clump hospital robot brave balcony utility detect estate old green desk skill multiply virus";
2875 driver
.findElement(By
.css('.use-entropy'))
2877 driver
.findElement(By
.css('.entropy'))
2879 driver
.sleep(generateDelay
).then(function() {
2880 driver
.findElement(By
.css(".phrase"))
2881 .getAttribute("value").then(function(phrase
) {
2882 expect(phrase
).toBe(expectedPhrase
);
2888 // Blank entropy does not generate mnemonic or addresses
2889 it('Does not generate mnemonic for blank entropy', function(done
) {
2890 driver
.findElement(By
.css('.use-entropy'))
2892 driver
.findElement(By
.css('.entropy'))
2894 // check there is no mnemonic
2895 driver
.sleep(generateDelay
).then(function() {
2896 driver
.findElement(By
.css(".phrase"))
2897 .getAttribute("value").then(function(phrase
) {
2898 expect(phrase
).toBe("");
2899 // check there is no mnemonic
2900 driver
.findElements(By
.css(".address"))
2901 .then(function(addresses
) {
2902 expect(addresses
.length
).toBe(0);
2903 // Check the feedback says 'blank entropy'
2904 driver
.findElement(By
.css(".feedback"))
2906 .then(function(feedbackText
) {
2907 expect(feedbackText
).toBe("Blank entropy");
2915 // Mnemonic length can be selected even for weak entropy
2916 it('Allows selection of mnemonic length even for weak entropy', function(done
) {
2917 driver
.findElement(By
.css('.use-entropy'))
2919 driver
.executeScript(function() {
2920 $(".mnemonic-length").val("18").trigger("change");
2922 driver
.findElement(By
.css('.entropy'))
2923 .sendKeys("012345");
2924 driver
.sleep(generateDelay
).then(function() {
2925 driver
.findElement(By
.css(".phrase"))
2926 .getAttribute("value").then(function(phrase
) {
2927 var wordCount
= phrase
.split(/\s+/g).length
;
2928 expect(wordCount
).toBe(18);
2935 // https://github.com/iancoleman/bip39/issues/33
2936 // Final cards should contribute entropy
2937 it('Uses as much entropy as possible for the mnemonic', function(done
) {
2938 driver
.findElement(By
.css('.use-entropy'))
2940 driver
.findElement(By
.css('.entropy'))
2941 .sendKeys("7S 9H 9S QH 8C KS AS 7D 7C QD 4S 4D TC 2D 5S JS 3D 8S 8H 4C 3C AC 3S QC 9C JC 7H AD TD JD 6D KH 5C QS 2S 6S 6H JH KD 9D-6C TS TH 4H KC 5H 2H AH 2C 8D 3H 5D");
2942 driver
.sleep(generateDelay
).then(function() {
2944 driver
.findElement(By
.css(".phrase"))
2945 .getAttribute("value").then(function(originalPhrase
) {
2946 // Set the last 12 cards to be AS
2947 driver
.findElement(By
.css('.entropy'))
2949 driver
.findElement(By
.css('.entropy'))
2950 .sendKeys("7S 9H 9S QH 8C KS AS 7D 7C QD 4S 4D TC 2D 5S JS 3D 8S 8H 4C 3C AC 3S QC 9C JC 7H AD TD JD 6D KH 5C QS 2S 6S 6H JH KD 9D-AS AS AS AS AS AS AS AS AS AS AS AS");
2951 driver
.sleep(generateDelay
).then(function() {
2953 driver
.findElement(By
.css(".phrase"))
2954 .getAttribute("value").then(function(newPhrase
) {
2955 expect(originalPhrase
).not
.toEqual(newPhrase
);
2964 // https://github.com/iancoleman/bip39/issues/35
2966 // TODO this doesn't work in selenium with firefox
2967 // see https://stackoverflow.com/q/40360223
2968 it('Shows a qr code on hover for the phrase', function(done
) {
2969 if (browser
== "firefox") {
2970 pending("Selenium + Firefox bug for mouseMove, see https://stackoverflow.com/q/40360223");
2972 // generate a random mnemonic
2973 var generateEl
= driver
.findElement(By
.css('.generate'));
2975 // toggle qr to show (hidden by default)
2976 var phraseEl
= driver
.findElement(By
.css(".phrase"));
2978 var rootKeyEl
= driver
.findElement(By
.css(".root-key"));
2979 driver
.sleep(generateDelay
).then(function() {
2980 // hover over the root key
2981 driver
.actions().mouseMove(rootKeyEl
).perform().then(function() {
2982 // check the qr code shows
2983 driver
.executeScript(function() {
2984 return $(".qr-container").find("canvas").length
> 0;
2986 .then(function(qrShowing
) {
2987 expect(qrShowing
).toBe(true);
2988 // hover away from the phrase
2989 driver
.actions().mouseMove(generateEl
).perform().then(function() {;
2990 // check the qr code hides
2991 driver
.executeScript(function() {
2992 return $(".qr-container").find("canvas").length
== 0;
2994 .then(function(qrHidden
) {
2995 expect(qrHidden
).toBe(true);
3004 // BIP44 account extendend private key is shown
3005 // github issue 37 - compatibility with electrum
3006 it('Shows the bip44 account extended private key', function(done
) {
3007 driver
.findElement(By
.css(".phrase"))
3008 .sendKeys("abandon abandon ability");
3009 driver
.sleep(generateDelay
).then(function() {
3010 driver
.findElement(By
.css("#bip44 .account-xprv"))
3011 .getAttribute("value")
3012 .then(function(xprv
) {
3013 expect(xprv
).toBe("xprv9yzrnt4zWVJUr1k2VxSPy9ettKz5PpeDMgaVG7UKedhqnw1tDkxP2UyYNhuNSumk2sLE5ctwKZs9vwjsq3e1vo9egCK6CzP87H2cVYXpfwQ");
3019 // BIP44 account extendend public key is shown
3020 // github issue 37 - compatibility with electrum
3021 it('Shows the bip44 account extended public key', function(done
) {
3022 driver
.findElement(By
.css(".phrase"))
3023 .sendKeys("abandon abandon ability");
3024 driver
.sleep(generateDelay
).then(function() {
3025 driver
.findElement(By
.css("#bip44 .account-xpub"))
3026 .getAttribute("value")
3027 .then(function(xprv
) {
3028 expect(xprv
).toBe("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
3035 // BIP32 root key can be set as an xpub
3036 it('Generates addresses from xpub as bip32 root key', function(done
) {
3037 driver
.findElement(By
.css('#bip32-tab a'))
3039 // set xpub for account 0 of bip44 for 'abandon abandon ability'
3040 driver
.findElement(By
.css("#root-key"))
3041 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
3042 driver
.sleep(generateDelay
).then(function() {
3043 // check the addresses are generated
3044 getFirstAddress(function(address
) {
3045 expect(address
).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug");
3046 // check the xprv key is not set
3047 driver
.findElement(By
.css(".extended-priv-key"))
3048 .getAttribute("value")
3049 .then(function(xprv
) {
3050 expect(xprv
).toBe("NA");
3051 // check the private key is not set
3052 driver
.findElements(By
.css(".privkey"))
3053 .then(function(els
) {
3056 .then(function(privkey
) {
3057 expect(xprv
).toBe("NA");
3067 // xpub for bip32 root key will not work with hardened derivation paths
3068 it('Shows error for hardened derivation paths with xpub root key', function(done
) {
3069 // set xpub for account 0 of bip44 for 'abandon abandon ability'
3070 driver
.findElement(By
.css("#root-key"))
3071 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
3072 driver
.sleep(feedbackDelay
).then(function() {
3073 // Check feedback is correct
3074 driver
.findElement(By
.css('.feedback'))
3076 .then(function(feedback
) {
3077 var msg
= "Hardened derivation path is invalid with xpub key";
3078 expect(feedback
).toBe(msg
);
3079 // Check no addresses are shown
3080 driver
.findElements(By
.css('.addresses tr'))
3081 .then(function(rows
) {
3082 expect(rows
.length
).toBe(0);
3090 // no root key shows feedback
3091 it('Shows feedback for no root key', function(done
) {
3092 // set xpub for account 0 of bip44 for 'abandon abandon ability'
3093 driver
.findElement(By
.css('#bip32-tab a'))
3095 driver
.sleep(feedbackDelay
).then(function() {
3096 // Check feedback is correct
3097 driver
.findElement(By
.css('.feedback'))
3099 .then(function(feedback
) {
3100 expect(feedback
).toBe("Invalid root key");
3107 // display error switching tabs while addresses are generating
3108 it('Can change details while old addresses are still being generated', function(done
) {
3109 // Set to generate 199 more addresses.
3110 // This will take a long time allowing a new set of addresses to be
3111 // generated midway through this lot.
3112 // The newly generated addresses should not include any from the old set.
3113 // Any more than 199 will show an alert which needs to be accepted.
3114 driver
.findElement(By
.css('.rows-to-add'))
3116 driver
.findElement(By
.css('.rows-to-add'))
3119 driver
.findElement(By
.css('.phrase'))
3120 .sendKeys("abandon abandon ability");
3121 driver
.sleep(generateDelay
).then(function() {
3122 // change tabs which should cancel the previous generating
3123 driver
.findElement(By
.css('.rows-to-add'))
3125 driver
.findElement(By
.css('.rows-to-add'))
3127 driver
.findElement(By
.css('#bip32-tab a'))
3129 driver
.sleep(generateDelay
).then(function() {
3130 driver
.findElements(By
.css('.index'))
3131 .then(function(els
) {
3132 // check the derivation paths have the right quantity
3133 expect(els
.length
).toBe(20);
3134 // check the derivation paths are in order
3135 testRowsAreInCorrectOrder(done
);
3139 }, generateDelay
+ 10000);
3142 // padding for binary should give length with multiple of 256
3143 // hashed entropy 1111 is length 252, so requires 4 leading zeros
3144 // prior to issue 49 it would only generate 2 leading zeros, ie missing 2
3145 it('Pads hashed entropy with leading zeros', function(done
) {
3146 driver
.findElement(By
.css('.use-entropy'))
3148 driver
.executeScript(function() {
3149 $(".mnemonic-length").val("15").trigger("change");
3151 driver
.findElement(By
.css('.entropy'))
3153 driver
.sleep(generateDelay
).then(function() {
3154 driver
.findElement(By
.css('.phrase'))
3155 .getAttribute("value")
3156 .then(function(phrase
) {
3157 expect(phrase
).toBe("avocado valid quantum cross link predict excuse edit street able flame large galaxy ginger nuclear");
3163 // Github pull request 55
3164 // https://github.com/iancoleman/bip39/pull/55
3166 it('Can set the derivation path on bip32 tab for bitcoincore', function(done
) {
3167 testClientSelect(done
, {
3169 bip32path: "m/0'/0'",
3170 useHardenedAddresses: "true",
3173 it('Can set the derivation path on bip32 tab for multibit', function(done
) {
3174 testClientSelect(done
, {
3176 bip32path: "m/0'/0",
3177 useHardenedAddresses: null,
3180 it('Can set the derivation path on bip32 tab for coinomi/ledger', function(done
) {
3181 testClientSelect(done
, {
3183 bip32path: "m/44'/0'/0'",
3184 useHardenedAddresses: null,
3189 // https://github.com/iancoleman/bip39/issues/58
3190 // bip32 derivation is correct, does not drop leading zeros
3192 // https://medium.com/@alexberegszaszi/why-do-my-bip32-wallets-disagree-6f3254cc5846
3193 it('Retains leading zeros for bip32 derivation', function(done
) {
3194 driver
.findElement(By
.css(".phrase"))
3195 .sendKeys("fruit wave dwarf banana earth journey tattoo true farm silk olive fence");
3196 driver
.findElement(By
.css(".passphrase"))
3197 .sendKeys("banana");
3198 driver
.sleep(generateDelay
).then(function() {
3199 getFirstAddress(function(address
) {
3200 // Note that bitcore generates an incorrect address
3201 // 13EuKhffWkBE2KUwcbkbELZb1MpzbimJ3Y
3202 // see the medium.com link above for more details
3203 expect(address
).toBe("17rxURoF96VhmkcEGCj5LNQkmN9HVhWb7F");
3210 // Japanese mnemonics generate incorrect bip32 seed
3211 // BIP39 seed is set from phrase
3212 it('Generates correct seed for Japanese mnemonics', function(done
) {
3213 driver
.findElement(By
.css(".phrase"))
3214 .sendKeys("あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あおぞら");
3215 driver
.findElement(By
.css(".passphrase"))
3216 .sendKeys("メートルガバヴァぱばぐゞちぢ十人十色");
3217 driver
.sleep(generateDelay
).then(function() {
3218 driver
.findElement(By
.css(".seed"))
3219 .getAttribute("value")
3220 .then(function(seed
) {
3221 expect(seed
).toBe("a262d6fb6122ecf45be09c50492b31f92e9beb7d9a845987a02cefda57a15f9c467a17872029a9e92299b5cbdf306e3a0ee620245cbd508959b6cb7ca637bd55");
3227 // BIP49 official test vectors
3228 // https://github.com/bitcoin/bips/blob/master/bip-0049.mediawiki#test-vectors
3229 it('Generates BIP49 addresses matching the official test vectors', function(done
) {
3230 driver
.findElement(By
.css('#bip49-tab a'))
3232 selectNetwork("BTC - Bitcoin Testnet");
3233 driver
.findElement(By
.css(".phrase"))
3234 .sendKeys("abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about");
3235 driver
.sleep(generateDelay
).then(function() {
3236 getFirstAddress(function(address
) {
3237 expect(address
).toBe("2Mww8dCYPUpKHofjgcXcBCEGmniw9CoaiD2");
3243 // BIP49 derivation path is shown
3244 it('Shows the bip49 derivation path', function(done
) {
3245 driver
.findElement(By
.css('#bip49-tab a'))
3247 driver
.findElement(By
.css(".phrase"))
3248 .sendKeys("abandon abandon ability");
3249 driver
.sleep(generateDelay
).then(function() {
3250 driver
.findElement(By
.css('#bip49 .path'))
3251 .getAttribute("value")
3252 .then(function(path
) {
3253 expect(path
).toBe("m/49'/0'/0'/0");
3259 // BIP49 extended private key is shown
3260 it('Shows the bip49 extended private key', function(done
) {
3261 driver
.findElement(By
.css('#bip49-tab a'))
3263 driver
.findElement(By
.css(".phrase"))
3264 .sendKeys("abandon abandon ability");
3265 driver
.sleep(generateDelay
).then(function() {
3266 driver
.findElement(By
.css('.extended-priv-key'))
3267 .getAttribute("value")
3268 .then(function(xprv
) {
3269 expect(xprv
).toBe("yprvALYB4DYRG6CzzVgzQZwwqjAA2wjBGC3iEd7KYYScpoDdmf75qMRWZWxoFcRXBJjgEXdFqJ9vDRGRLJQsrL22Su5jMbNFeM9vetaGVqy9Qy2");
3275 // BIP49 extended public key is shown
3276 it('Shows the bip49 extended public key', function(done
) {
3277 driver
.findElement(By
.css('#bip49-tab a'))
3279 driver
.findElement(By
.css(".phrase"))
3280 .sendKeys("abandon abandon ability");
3281 driver
.sleep(generateDelay
).then(function() {
3282 driver
.findElement(By
.css('.extended-pub-key'))
3283 .getAttribute("value")
3284 .then(function(xprv
) {
3285 expect(xprv
).toBe("ypub6ZXXTj5K6TmJCymTWbUxCs6tayZffemZbr2vLvrEP8kceTSENtjm7KHH6thvAKxVar9fGe8rgsPEX369zURLZ68b4f7Vexz7RuXsjQ69YDt");
3291 // BIP49 account field changes address list
3292 it('Can set the bip49 account field', function(done
) {
3293 driver
.findElement(By
.css('#bip49-tab a'))
3295 driver
.findElement(By
.css('#bip49 .account'))
3297 driver
.findElement(By
.css('#bip49 .account'))
3299 driver
.findElement(By
.css(".phrase"))
3300 .sendKeys("abandon abandon ability");
3301 driver
.sleep(generateDelay
).then(function() {
3302 getFirstAddress(function(address
) {
3303 expect(address
).toBe("381wg1GGN4rP88rNC9v7QWsiww63yLVPsn");
3309 // BIP49 change field changes address list
3310 it('Can set the bip49 change field', function(done
) {
3311 driver
.findElement(By
.css('#bip49-tab a'))
3313 driver
.findElement(By
.css('#bip49 .change'))
3315 driver
.findElement(By
.css('#bip49 .change'))
3317 driver
.findElement(By
.css(".phrase"))
3318 .sendKeys("abandon abandon ability");
3319 driver
.sleep(generateDelay
).then(function() {
3320 getFirstAddress(function(address
) {
3321 expect(address
).toBe("3PEM7MiKed5konBoN66PQhK8r3hjGhy9dT");
3327 // BIP49 account extendend private key is shown
3328 it('Shows the bip49 account extended private key', function(done
) {
3329 driver
.findElement(By
.css('#bip49-tab a'))
3331 driver
.findElement(By
.css(".phrase"))
3332 .sendKeys("abandon abandon ability");
3333 driver
.sleep(generateDelay
).then(function() {
3334 driver
.findElement(By
.css('#bip49 .account-xprv'))
3335 .getAttribute("value")
3336 .then(function(xprv
) {
3337 expect(xprv
).toBe("yprvAHtB1M5Wp675aLzFy9TJYK2mSsLkg6mcBRh5DZTR7L4EnYSmYPaL63KFA4ycg1PngW5LfkmejxzosCs17TKZMpRFKc3z5SJar6QAKaFcaZL");
3343 // BIP49 account extendend public key is shown
3344 it('Shows the bip49 account extended public key', function(done
) {
3345 driver
.findElement(By
.css('#bip49-tab a'))
3347 driver
.findElement(By
.css(".phrase"))
3348 .sendKeys("abandon abandon ability");
3349 driver
.sleep(generateDelay
).then(function() {
3350 driver
.findElement(By
.css('#bip49 .account-xpub'))
3351 .getAttribute("value")
3352 .then(function(xprv
) {
3353 expect(xprv
).toBe("ypub6WsXQrcQeTfNnq4j5AzJuSyVzuBF5ZVTYecg1ws2ffbDfLmv5vtadqdj1NgR6C6gufMpMfJpHxvb6JEQKvETVNWCRanNedfJtnTchZiJtsL");
3359 // Test selecting coin where bip49 is unavailable (eg CLAM)
3360 it('Shows an error on bip49 tab for coins without bip49', function(done
) {
3361 driver
.findElement(By
.css('#bip49-tab a'))
3363 driver
.findElement(By
.css(".phrase"))
3364 .sendKeys("abandon abandon ability");
3365 driver
.sleep(generateDelay
).then(function() {
3366 selectNetwork("CLAM - Clams");
3367 // bip49 available is hidden
3368 driver
.findElement(By
.css('#bip49 .available'))
3369 .getAttribute("class")
3370 .then(function(classes
) {
3371 expect(classes
).toContain("hidden");
3372 // bip49 unavailable is shown
3373 driver
.findElement(By
.css('#bip49 .unavailable'))
3374 .getAttribute("class")
3375 .then(function(classes
) {
3376 expect(classes
).not
.toContain("hidden");
3377 // check there are no addresses shown
3378 driver
.findElements(By
.css('.addresses tr'))
3379 .then(function(rows
) {
3380 expect(rows
.length
).toBe(0);
3381 // check the derived private key is blank
3382 driver
.findElement(By
.css('.extended-priv-key'))
3383 .getAttribute("value")
3384 .then(function(xprv
) {
3385 expect(xprv
).toBe('');
3386 // check the derived public key is blank
3387 driver
.findElement(By
.css('.extended-pub-key'))
3388 .getAttribute("value")
3389 .then(function(xpub
) {
3390 expect(xpub
).toBe('');
3401 // Cleared mnemonic and root key still allows addresses to be generated
3402 // https://github.com/iancoleman/bip39/issues/43
3403 it('Clears old root keys from memory when mnemonic is cleared', function(done
) {
3405 driver
.findElement(By
.css(".phrase"))
3406 .sendKeys("abandon abandon ability");
3407 driver
.sleep(generateDelay
).then(function() {
3408 // clear the mnemonic and root key
3409 // using selenium .clear() doesn't seem to trigger the 'input' event
3410 // so clear it using keys instead
3411 driver
.findElement(By
.css('.phrase'))
3412 .sendKeys(Key
.CONTROL
,"a");
3413 driver
.findElement(By
.css('.phrase'))
3414 .sendKeys(Key
.DELETE
);
3415 driver
.findElement(By
.css('.root-key'))
3416 .sendKeys(Key
.CONTROL
,"a");
3417 driver
.findElement(By
.css('.root-key'))
3418 .sendKeys(Key
.DELETE
);
3419 driver
.sleep(generateDelay
).then(function() {
3420 // try to generate more addresses
3421 driver
.findElement(By
.css('.more'))
3423 driver
.sleep(generateDelay
).then(function() {
3424 driver
.findElements(By
.css(".addresses tr"))
3425 .then(function(els
) {
3426 // check there are no addresses shown
3427 expect(els
.length
).toBe(0);
3436 // error trying to generate addresses from xpub with hardened derivation
3437 it('Shows error for hardened addresses with xpub root key', function(done
) {
3438 driver
.findElement(By
.css('#bip32-tab a'))
3440 driver
.executeScript(function() {
3441 $(".hardened-addresses").prop("checked", true);
3443 // set xpub for account 0 of bip44 for 'abandon abandon ability'
3444 driver
.findElement(By
.css("#root-key"))
3445 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
3446 driver
.sleep(feedbackDelay
).then(function() {
3447 // Check feedback is correct
3448 driver
.findElement(By
.css('.feedback'))
3450 .then(function(feedback
) {
3451 var msg
= "Hardened derivation path is invalid with xpub key";
3452 expect(feedback
).toBe(msg
);
3458 // Litecoin uses ltub by default, and can optionally be set to xprv
3460 // https://github.com/iancoleman/bip39/issues/96
3461 // Issue with extended keys on Litecoin
3462 it('Uses ltub by default for litecoin, but can be set to xprv', function(done
) {
3463 driver
.findElement(By
.css('.phrase'))
3464 .sendKeys("abandon abandon ability");
3465 selectNetwork("LTC - Litecoin");
3466 driver
.sleep(generateDelay
).then(function() {
3467 // check the extended key is generated correctly
3468 driver
.findElement(By
.css('.root-key'))
3469 .getAttribute("value")
3470 .then(function(rootKey
) {
3471 expect(rootKey
).toBe("Ltpv71G8qDifUiNesiPqf6h5V6eQ8ic77oxQiYtawiACjBEx3sTXNR2HGDGnHETYxESjqkMLFBkKhWVq67ey1B2MKQXannUqNy1RZVHbmrEjnEU");
3472 // set litecoin to use ltub
3473 driver
.executeScript(function() {
3474 $(".litecoin-use-ltub").prop("checked", false);
3475 $(".litecoin-use-ltub").trigger("change");
3477 driver
.sleep(generateDelay
).then(function() {
3478 driver
.findElement(By
.css('.root-key'))
3479 .getAttribute("value")
3480 .then(function(rootKey
) {
3481 expect(rootKey
).toBe("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
3490 // https://github.com/iancoleman/bip39/issues/99#issuecomment-327094159
3491 // "warn me emphatically when they have detected invalid input" to the entropy field
3492 // A warning is shown when entropy is filtered and discarded
3493 it('Warns when entropy is filtered and discarded', function(done
) {
3494 driver
.findElement(By
.css('.use-entropy'))
3496 // set entropy to have no filtered content
3497 driver
.findElement(By
.css('.entropy'))
3498 .sendKeys("00000000 00000000 00000000 00000000");
3499 driver
.sleep(generateDelay
).then(function() {
3500 // check the filter warning does not show
3501 driver
.findElement(By
.css('.entropy-container .filter-warning'))
3502 .getAttribute("class")
3503 .then(function(classes
) {
3504 expect(classes
).toContain("hidden");
3505 // set entropy to have some filtered content
3506 driver
.findElement(By
.css('.entropy'))
3507 .sendKeys("10000000 zxcvbn 00000000 00000000 00000000");
3508 driver
.sleep(entropyFeedbackDelay
).then(function() {
3509 // check the filter warning shows
3510 driver
.findElement(By
.css('.entropy-container .filter-warning'))
3511 .getAttribute("class")
3512 .then(function(classes
) {
3513 expect(classes
).not
.toContain("hidden");
3521 // Bitcoin Cash address can be set to use cashaddr format
3522 it('Can use cashaddr format for bitcoin cash addresses', function(done
) {
3523 driver
.executeScript(function() {
3524 $(".use-bch-cashaddr-addresses").prop("checked", true);
3526 driver
.findElement(By
.css('.phrase'))
3527 .sendKeys("abandon abandon ability");
3528 selectNetwork("BCH - Bitcoin Cash");
3529 driver
.sleep(generateDelay
).then(function() {
3530 getFirstAddress(function(address
) {
3531 expect(address
).toBe("bitcoincash:qzlquk7w4hkudxypl4fgv8x279r754dkvur7jpcsps");
3537 // Bitcoin Cash address can be set to use bitpay format
3538 it('Can use bitpay format for bitcoin cash addresses', function(done
) {
3539 driver
.executeScript(function() {
3540 $(".use-bch-bitpay-addresses").prop("checked", true);
3542 driver
.findElement(By
.css('.phrase'))
3543 .sendKeys("abandon abandon ability");
3544 selectNetwork("BCH - Bitcoin Cash");
3545 driver
.sleep(generateDelay
).then(function() {
3546 getFirstAddress(function(address
) {
3547 expect(address
).toBe("CZnpA9HPmvhuhLLPWJP8rNDpLUYXy1LXFk");
3553 // Bitcoin Cash address can be set to use legacy format
3554 it('Can use legacy format for bitcoin cash addresses', function(done
) {
3555 driver
.executeScript(function() {
3556 $(".use-bch-legacy-addresses").prop("checked", true);
3558 driver
.findElement(By
.css('.phrase'))
3559 .sendKeys("abandon abandon ability");
3560 selectNetwork("BCH - Bitcoin Cash");
3561 driver
.sleep(generateDelay
).then(function() {
3562 getFirstAddress(function(address
) {
3563 expect(address
).toBe("1JKvb6wKtsjNoCRxpZ4DGrbniML7z5U16A");
3569 // End of tests ported from old suit, so no more comments above each test now
3571 it('Can generate more addresses from a custom index', function(done
) {
3572 var expectedIndexes
= [
3573 0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,
3574 40,41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59
3576 driver
.findElement(By
.css('.phrase'))
3577 .sendKeys("abandon abandon ability");
3578 driver
.sleep(generateDelay
).then(function() {
3579 // Set start of next lot of rows to be from index 40
3580 // which means indexes 20-39 will not be in the table.
3581 driver
.findElement(By
.css('.more-rows-start-index'))
3583 driver
.findElement(By
.css('.more'))
3585 driver
.sleep(generateDelay
).then(function() {
3586 // Check actual indexes in the table match the expected pattern
3587 driver
.findElements(By
.css(".index"))
3588 .then(function(els
) {
3589 expect(els
.length
).toBe(expectedIndexes
.length
);
3590 var testRowAtIndex = function(i
) {
3591 if (i
>= expectedIndexes
.length
) {
3596 .then(function(actualPath
) {
3597 var noHardened
= actualPath
.replace(/'/g, "");
3598 var pathBits = noHardened.split("/")
3599 var lastBit = pathBits[pathBits.length-1];
3600 var actualIndex = parseInt(lastBit);
3601 var expectedIndex = expectedIndexes[i];
3602 expect(actualIndex).toBe(expectedIndex);
3603 testRowAtIndex(i+1);
3613 it('Can generate BIP141 addresses
with P2WPKH
-in-P2SH semanitcs
', function(done) {
3614 // Sourced from BIP49 official test specs
3615 driver.findElement(By.css('#bip141
-tab a
'))
3617 driver.findElement(By.css('.bip141
-path
'))
3619 driver.findElement(By.css('.bip141
-path
'))
3620 .sendKeys("m/49'/1'/0'/0");
3621 selectNetwork("BTC
- Bitcoin Testnet
");
3622 driver.findElement(By.css(".phrase
"))
3623 .sendKeys("abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about
");
3624 driver.sleep(generateDelay).then(function() {
3625 getFirstAddress(function(address) {
3626 expect(address).toBe("2Mww8dCYPUpKHofjgcXcBCEGmniw9CoaiD2
");
3632 it('Can generate BIP141 addresses with P2WPKH semanitcs', function(done) {
3633 // This result tested against bitcoinjs-lib test spec for segwit address
3634 // using the first private key of this mnemonic and default path m/0
3635 // https://github.com/bitcoinjs/bitcoinjs-lib/blob/9c8503cab0c6c30a95127042703bc18e8d28c76d/test/integration/addresses.js#L50
3636 // so whilst not directly comparable, substituting the private key produces
3637 // identical results between this tool and the bitcoinjs-lib test.
3638 // Private key generated is:
3639 // L3L8Nu9whawPBNLGtFqDhKut9DKKfG3CQoysupT7BimqVCZsLFNP
3640 driver.findElement(By.css('#bip141-tab a'))
3643 driver.executeScript(function() {
3644 $(".bip141
-semantics option
[selected
]").removeAttr("selected
");
3645 $(".bip141
-semantics option
").filter(function(i,e) {
3646 return $(e).html() == "P2WPKH
";
3647 }).prop("selected
", true);
3648 $(".bip141
-semantics
").trigger("change
");
3650 driver.findElement(By.css(".phrase
"))
3651 .sendKeys("abandon abandon ability
");
3652 driver.sleep(generateDelay).then(function() {
3653 getFirstAddress(function(address) {
3654 expect(address).toBe("bc1qfwu6a5a3evygrk8zvdxxvz4547lmpyx5vsfxe9
");
3660 it('Shows the entropy used by the PRNG when clicking generate', function(done) {
3661 driver.findElement(By.css('.generate')).click();
3662 driver.sleep(generateDelay).then(function() {
3663 driver.findElement(By.css('.entropy'))
3664 .getAttribute("value
")
3665 .then(function(entropy) {
3666 expect(entropy).not.toBe("");
3672 it('Shows the index of each word in the mnemonic', function(done) {
3673 driver.findElement(By.css('.phrase'))
3674 .sendKeys("abandon abandon ability
");
3675 driver.sleep(generateDelay).then(function() {
3676 driver.findElement(By.css('.use-entropy'))
3678 driver.findElement(By.css('.word-indexes'))
3680 .then(function(indexes) {
3681 expect(indexes).toBe("0, 0, 1");
3687 it('Shows the derivation path for bip84 tab', function(done) {
3688 driver.findElement(By.css('#bip84-tab a'))
3690 driver.findElement(By.css('.phrase'))
3691 .sendKeys('abandon abandon ability');
3692 driver.sleep(generateDelay).then(function() {
3693 driver.findElement(By.css('#bip84 .path'))
3694 .getAttribute("value
")
3695 .then(function(path) {
3696 expect(path).toBe("m
/84'/0'/0'/0");
3702 it('Shows the extended private key for bip84 tab', function(done) {
3703 driver.findElement(By.css('#bip84-tab a'))
3705 driver.findElement(By.css('.phrase'))
3706 .sendKeys('abandon abandon ability');
3707 driver.sleep(generateDelay).then(function() {
3708 driver.findElement(By.css('.extended-priv-key'))
3709 .getAttribute("value
")
3710 .then(function(path) {
3711 expect(path).toBe("zprvAev3RKrZ3QVKiUFCfdeMRen1BPDJgdNt1XpxiDy8acSs4kkAGTCvq7HeRYRNNpo8EtEjCFQBWavJwtCUR29y4TUCH4X5RXMcyq48uN8y9BP
");
3717 it('Shows the extended public key for bip84 tab', function(done) {
3718 driver.findElement(By.css('#bip84-tab a'))
3720 driver.findElement(By.css('.phrase'))
3721 .sendKeys('abandon abandon ability');
3722 driver.sleep(generateDelay).then(function() {
3723 driver.findElement(By.css('.extended-pub-key'))
3724 .getAttribute("value
")
3725 .then(function(path) {
3726 expect(path).toBe("zpub6suPpqPSsn3cvxKfmfBMnnijjR3o666jNkkZWcNk8wyqwZ5JozXBNuc8Gs7DB3uLwTDvGVTspVEAUQcEjKF3pZHgywVbubdTqbXTUg7usyx
");
3732 it('Changes the address list if bip84 account is changed', function(done) {
3733 driver.findElement(By.css('#bip84-tab a'))
3735 driver.findElement(By.css('#bip84 .account'))
3737 driver.findElement(By.css('.phrase'))
3738 .sendKeys('abandon abandon ability');
3739 driver.sleep(generateDelay).then(function() {
3740 getFirstAddress(function(address) {
3741 expect(address).toBe("bc1qp7vv669t2fy965jdzvqwrraana89ctd5ewc662
");
3747 it('Changes the address list if bip84 change is changed', function(done) {
3748 driver.findElement(By.css('#bip84-tab a'))
3750 driver.findElement(By.css('#bip84 .change'))
3752 driver.findElement(By.css('.phrase'))
3753 .sendKeys('abandon abandon ability');
3754 driver.sleep(generateDelay).then(function() {
3755 getFirstAddress(function(address) {
3756 expect(address).toBe("bc1qr39vj6rh06ff05m53uxq8uazehwhccswylhrs2
");
3762 it('Passes the official BIP84 test spec for rootpriv', function(done) {
3763 driver.findElement(By.css('#bip84-tab a'))
3765 driver.findElement(By.css('.phrase'))
3766 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3767 driver.sleep(generateDelay).then(function() {
3768 driver.findElement(By.css(".root
-key
"))
3769 .getAttribute("value
")
3770 .then(function(rootKey) {
3771 expect(rootKey).toBe("zprvAWgYBBk7JR8Gjrh4UJQ2uJdG1r3WNRRfURiABBE3RvMXYSrRJL62XuezvGdPvG6GFBZduosCc1YP5wixPox7zhZLfiUm8aunE96BBa4Kei5
");
3777 it('Passes the official BIP84 test spec for account 0 xprv', function(done) {
3778 driver.findElement(By.css('#bip84-tab a'))
3780 driver.findElement(By.css('.phrase'))
3781 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3782 driver.sleep(generateDelay).then(function() {
3783 driver.findElement(By.css("#bip84
.account
-xprv
"))
3784 .getAttribute("value
")
3785 .then(function(rootKey) {
3786 expect(rootKey).toBe("zprvAdG4iTXWBoARxkkzNpNh8r6Qag3irQB8PzEMkAFeTRXxHpbF9z4QgEvBRmfvqWvGp42t42nvgGpNgYSJA9iefm1yYNZKEm7z6qUWCroSQnE
");
3792 it('Passes the official BIP84 test spec for account 0 xpub', function(done) {
3793 driver.findElement(By.css('#bip84-tab a'))
3795 driver.findElement(By.css('.phrase'))
3796 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3797 driver.sleep(generateDelay).then(function() {
3798 driver.findElement(By.css("#bip84
.account
-xpub
"))
3799 .getAttribute("value
")
3800 .then(function(rootKey) {
3801 expect(rootKey).toBe("zpub6rFR7y4Q2AijBEqTUquhVz398htDFrtymD9xYYfG1m4wAcvPhXNfE3EfH1r1ADqtfSdVCToUG868RvUUkgDKf31mGDtKsAYz2oz2AGutZYs
");
3807 it('Passes the official BIP84 test spec for account 0 first address', function(done) {
3808 driver.findElement(By.css('#bip84-tab a'))
3810 driver.findElement(By.css('.phrase'))
3811 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3812 driver.sleep(generateDelay).then(function() {
3813 getFirstAddress(function(address) {
3814 expect(address).toBe("bc1qcr8te4kr609gcawutmrza0j4xv80jy8z306fyu
");
3820 it('Passes the official BIP84 test spec for account 0 first change address', function(done) {
3821 driver.findElement(By.css('#bip84-tab a'))
3823 driver.findElement(By.css('.phrase'))
3824 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3825 driver.findElement(By.css('#bip84 .change'))
3827 driver.sleep(generateDelay).then(function() {
3828 getFirstAddress(function(address) {
3829 expect(address).toBe("bc1q8c6fshw2dlwun7ekn9qwf37cu2rn755upcp6el
");
3835 it('Can display the table as csv', function(done) {
3836 var headings = "path
,address
,public key
,private key
";
3837 var row1 = "m
/44'/0'/0'/0/0,1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug
,033f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3
,L26cVSpWFkJ6aQkPkKmTzLqTdLJ923e6CzrVh9cmx21QHsoUmrEE
";
3838 var row20 = "m
/44'/0'/0'/0/19,1KhBy28XLAciXnnRvm71PvQJaETyrxGV55
,02b4b3e396434d8cdd20c03ac4aaa07387784d5d867b75987f516f5705ee68cb3a
,L4GrDrjReMsCAu5DkLXn79jSb95qR7Zfx7eshybCQZ1qL32MXJab
";
3839 driver.findElement(By.css('.phrase'))
3840 .sendKeys('abandon abandon ability');
3841 driver.sleep(generateDelay).then(function() {
3842 driver.findElement(By.css('.csv'))
3843 .getAttribute("value
")
3844 .then(function(csv) {
3845 expect(csv).toContain(headings);
3846 expect(csv).toContain(row1);
3847 expect(csv).toContain(row20);
3853 it('LeftPads ethereum keys that are less than 32 bytes', function(done) {
3854 // see https://github.com/iancoleman/bip39/issues/155
3855 selectNetwork("ETH
- Ethereum
");
3856 driver.findElement(By.css('#bip32-tab a'))
3858 driver.findElement(By.css('#bip32-path'))
3860 driver.findElement(By.css('#bip32-path'))
3861 .sendKeys("m
/44'/60'/0'");
3862 driver.findElement(By.css('.phrase'))
3863 .sendKeys('scout sort custom elite radar rare vivid thing trophy gesture cover snake change narrow kite list nation sustain buffalo erode open balance system young');
3864 driver.sleep(generateDelay).then(function() {
3865 getFirstAddress(function(address) {
3866 expect(address).toBe("0x8943E785B4a5714FC87a3aFAad1eB1FeB602B118");
3872 it('Can encrypt private keys using BIP38', function(done) {
3873 // see https://github.com/iancoleman/bip39/issues/140
3874 driver.executeScript(function() {
3875 $(".use-bip38
").prop("checked
", true);
3877 driver.findElement(By.css('.bip38-password'))
3878 .sendKeys('bip38password');
3879 driver.findElement(By.css('.rows-to-add'))
3881 driver.findElement(By.css('.rows-to-add'))
3883 driver.findElement(By.css('.phrase'))
3884 .sendKeys('abandon abandon ability');
3885 driver.sleep(bip38delay).then(function() {
3887 getFirstRowValue(function(address) {
3888 expect(address).toBe("1NCvSdumA3ngMM9c4aqU56AM6rqXddfuXB
");
3890 getFirstRowValue(function(pubkey) {
3891 expect(pubkey).toBe("043f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3884a74447ea901729b1e73a999b7520e7cb55b4120e6432c64153ccab8a848e1
");
3893 getFirstRowValue(function(privkey) {
3894 expect(privkey).toBe("6PRNRiFnj1RoR3sXhymdCvoZCgnUHQpfupNdKkFbWJkwWQEKesWt1EDMDM
");
3900 }, bip38delay + 5000);
3902 it('Shows the checksum for the entropy', function(done) {
3903 driver.findElement(By.css('.use-entropy'))
3905 driver.findElement(By.css('.entropy'))
3906 .sendKeys("00000000000000000000000000000000");
3907 driver.sleep(generateDelay).then(function() {
3908 driver.findElement(By.css('.checksum'))
3910 .then(function(text) {
3911 expect(text).toBe("1");
3917 it('Shows the checksum for the entropy with the correct groupings', function(done) {
3918 driver.findElement(By.css('.use-entropy'))
3920 // create a checksum of 20 bits, which spans multiple words
3921 driver.findElement(By.css('.entropy'))
3922 .sendKeys("F000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
");
3923 driver.sleep(generateDelay).then(function() {
3924 driver.findElement(By.css('.checksum'))
3926 .then(function(text) {
3927 // first group is 9 bits, second group is 11
3928 expect(text).toBe("011010111 01110000110");
3934 it('Uses vprv for bitcoin testnet p2wpkh', function(done) {
3935 selectNetwork("BTC
- Bitcoin Testnet
");
3936 driver.findElement(By.css('#bip84-tab a'))
3938 driver.findElement(By.css('.phrase'))
3939 .sendKeys('abandon abandon ability');
3940 driver.sleep(generateDelay).then(function() {
3941 driver.findElement(By.css('.root-key'))
3942 .getAttribute("value
")
3943 .then(function(path) {
3944 expect(path).toBe("vprv9DMUxX4ShgxML9N2YV5CvWEebWrM9aJ5ULpbRRyzyWu6vs4BzTvbfFFrH41N5hVi7MYSfiugd765L3JmAfDM5po36Y8ouCKRDeYQwByCmS7
");
3950 it('Shows a warning if generating weak mnemonics', function(done) {
3951 driver.executeScript(function() {
3952 $(".strength option
[selected
]").removeAttr("selected
");
3953 $(".strength option
[value
=6]").prop("selected
", true);
3954 $(".strength
").trigger("change
");
3956 driver.findElement(By.css(".generate
-container
.warning
"))
3957 .getAttribute("class")
3958 .then(function(classes) {
3959 expect(classes).not.toContain("hidden
");
3964 it('Does not show a warning if generating strong mnemonics', function(done) {
3965 driver.executeScript(function() {
3966 $(".strength option
[selected
]").removeAttr("selected
");
3967 $(".strength option
[value
=12]").prop("selected
", true);
3969 driver.findElement(By.css(".generate
-container
.warning
"))
3970 .getAttribute("class")
3971 .then(function(classes) {
3972 expect(classes).toContain("hidden
");
3977 it('Shows a warning if overriding weak entropy with longer mnemonics', function(done) {
3978 driver.findElement(By.css('.use-entropy'))
3980 driver.findElement(By.css('.entropy'))
3981 .sendKeys("0123456789abcdef
"); // 6 words
3982 driver.executeScript(function() {
3983 $(".mnemonic
-length
").val("12").trigger("change
");
3985 driver.findElement(By.css(".weak
-entropy
-override
-warning
"))
3986 .getAttribute("class")
3987 .then(function(classes) {
3988 expect(classes).not.toContain("hidden
");
3993 it('Does not show a warning if entropy is stronger than mnemonic length', function(done) {
3994 driver.findElement(By.css('.use-entropy'))
3996 driver.findElement(By.css('.entropy'))
3997 .sendKeys("0123456789abcdef0123456789abcdef0123456789abcdef
"); // 18 words
3998 driver.executeScript(function() {
3999 $(".mnemonic
-length
").val("12").trigger("change
");
4001 driver.findElement(By.css(".weak
-entropy
-override
-warning
"))
4002 .getAttribute("class")
4003 .then(function(classes) {
4004 expect(classes).toContain("hidden
");
4009 it('Shows litecoin BIP49 addresses', function(done) {
4010 driver.findElement(By.css('.phrase'))
4011 .sendKeys('abandon abandon ability');
4012 selectNetwork("LTC
- Litecoin
");
4013 driver.findElement(By.css('#bip49-tab a'))
4015 // bip49 addresses are shown
4016 driver.sleep(generateDelay).then(function() {
4017 driver.findElement(By.css('#bip49 .available'))
4018 .getAttribute("class")
4019 .then(function(classes) {
4020 expect(classes).not.toContain("hidden
");
4021 // check first address
4022 getFirstAddress(function(address) {
4023 expect(address).toBe("MFwLPhsXoBuSLL8cLmW9uK6tChkzduV8qN
");
4030 it('Can use root keys to generate segwit table rows', function(done) {
4031 // segwit uses ypub / zpub instead of xpub but the root key should still
4032 // be valid regardless of the encoding used to import that key.
4033 // Maybe this breaks the reason for the different extended key prefixes, but
4034 // since the parsed root key is used behind the scenes anyhow this should be
4036 driver.findElement(By.css('#root-key'))
4037 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
4038 driver.findElement(By.css('#bip49-tab a'))
4040 // bip49 addresses are shown
4041 driver.sleep(generateDelay).then(function() {
4042 getFirstAddress(function(address) {
4043 expect(address).toBe("3QG2Y9AA4xZ846gKHZqNf7mvVKbLqMKxr2
");