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 Aryacoin
', function(done) {
737 selectText: "ARYA - Aryacoin",
738 firstAddress: "Abr6gX25KaU9BpwD34UfsL3A4n89NvYYSf",
740 testNetwork(done, params);
742 it('Allows selection
of Auroracoin
', function(done) {
744 selectText: "AUR - Auroracoin",
745 firstAddress: "ANuraS6F4Jpi413FEnavjYkKYJJRHkgYCm",
747 testNetwork(done, params);
749 it('Allows selection
of Bata
', function(done) {
751 selectText: "BTA - Bata",
752 firstAddress: "BGxBdNeYPtF3GCuTtZBPQdFxCkdBYSF3fj",
754 testNetwork(done, params);
756 it('Allows selection
of Belacoin
', function(done) {
758 selectText: "BELA - Belacoin",
759 firstAddress: "BEeetqpNffdzeknSpNmQp5KAFh2KK1Qx7S",
761 testNetwork(done, params);
763 it('Allows selection
of Bitcoin Atom
', function(done) {
765 selectText: "BCA - Bitcoin Atom",
766 firstAddress: "AMy6qMbJeC4zsGRL6iWszmeCdQH65fgfih",
768 testNetwork(done, params);
770 it('Allows selection
of Bitcoinplus
', function(done) {
772 selectText: "XBC - Bitcoinplus",
773 firstAddress: "B7FSynZoDbEwTCSgsXq9nJ5ue8owYLVL8r",
775 testNetwork(done, params);
777 it('Allows selection
of Bitcoin Private
', function(done) {
779 selectText: "BTCP - Bitcoin Private",
780 firstAddress: "b1M3PbiXXyN6Hdivdw5rJv5VKpLjPzhm4jM",
782 testNetwork(done, params);
784 it('Allows selection
of Bitcoin SV
', function(done) {
786 selectText: "BSV - BitcoinSV",
787 firstAddress: "1N4mgtE5yxifch9jWs7Sds6oVqxdy2t576",
789 testNetwork(done, params);
791 it('Allows selection
of Bitcoinz
', function(done) {
793 selectText: "BTCZ - Bitcoinz",
794 firstAddress: "t1X2YQoxs8cYRo2oaBYgVEwW5QNjCC59NYc",
796 testNetwork(done, params);
798 it('Allows selection
of BitCloud
', function(done) {
800 selectText: "BTDX - BitCloud",
801 firstAddress: "BHbWitXCNgTf1BhsRDNMP186EeibuzmrBi",
803 testNetwork(done, params);
805 it('Allows selection
of Bitcore
', function(done) {
807 selectText: "BTX - Bitcore",
808 firstAddress: "2Rgp5Znhpy34TK4QmPkfCiYs9r4KovfTH9",
810 testNetwork(done, params);
812 it('Allows selection
of Bitsend
', function(done) {
814 selectText: "BSD - Bitsend",
815 firstAddress: "iBPk7LYjDun3EPk7CRR8UUmnPoceVc1bp2",
817 testNetwork(done, params);
819 it('Allows selection
of Britcoin
', function(done) {
821 selectText: "BRIT - Britcoin",
822 firstAddress: "B6Aue4J2XLs1f1dtD4H1SHYFfh4XrmEbrw",
824 testNetwork(done, params);
826 it('Allows selection
of Canadaecoin
', function(done) {
828 selectText: "CDN - Canadaecoin",
829 firstAddress: "CanAyCfd5Rj2CQVfaoAmvDUZunPM5W1AEQ",
831 testNetwork(done, params);
833 it('Allows selection
of Cannacoin
', function(done) {
835 selectText: "CCN - Cannacoin",
836 firstAddress: "CYjW8xWB43g6krLJTmmrPk1PonoQX7h9Qd",
838 testNetwork(done, params);
840 it('Allows selection
of Clubcoin
', function(done) {
842 selectText: "CLUB - Clubcoin",
843 firstAddress: "CHMDEXN4sihpSVX4GyAa2hZ62shnby7uyN",
845 testNetwork(done, params);
847 it('Allows selection
of Compcoin
', function(done) {
849 selectText: "CMP - Compcoin",
850 firstAddress: "CLshtw3zhxkseBJS46UF12v3AFy9Dx7JVv",
852 testNetwork(done, params);
854 it('Allows selection
of CPUchain
', function(done) {
856 selectText: "CPU - CPUchain",
857 firstAddress: "CWSpLpW7jS4mBHJnkz3mmL5s3yQMg93zK8",
859 testNetwork(done, params);
861 it('Allows selection
of Crave
', function(done) {
863 selectText: "CRAVE - Crave",
864 firstAddress: "VCYJeti6uKMNBFKCL7eP96UwuFWYHM7c85",
866 testNetwork(done, params);
868 it('Allows selection
of Defcoin
', function(done) {
870 selectText: "DFC - Defcoin",
871 firstAddress: "D8swcgyaaFUrXZU3ATwbgy16buCpWqbG1M",
873 testNetwork(done, params);
875 it('Allows selection
of Diamond
', function(done) {
877 selectText: "DMD - Diamond",
878 firstAddress: "dJnrVbLL9UPjdaVRz2C8VpqHZknqAqjLek",
880 testNetwork(done, params);
882 it('Allows selection
of Digibyte
', function(done) {
884 selectText: "DGB - Digibyte",
885 firstAddress: "D85Rp9jwLtMdmP6wGjTiqHBdVQLST3YCEq",
887 testNetwork(done, params);
889 it('Allows selection
of Digitalcoin
', function(done) {
891 selectText: "DGC - Digitalcoin",
892 firstAddress: "DKw4UGKEAZWweDNEbBFNQx4EM8x1mpUdia",
894 testNetwork(done, params);
896 it('Allows selection
of Ecoin
', function(done) {
898 selectText: "ECN - Ecoin",
899 firstAddress: "e6WFPLG5gcXyF7cESFteH1hE2XSmowW5yB",
901 testNetwork(done, params);
903 it('Allows selection
of Edrcoin
', function(done) {
905 selectText: "EDRC - Edrcoin",
906 firstAddress: "eh1nUJsvgKPFv6ebMBfcwJ299GMCpjeZUG",
908 testNetwork(done, params);
910 it('Allows selection
of Egulden
', function(done) {
912 selectText: "EFL - Egulden",
913 firstAddress: "Lg66yt55R7edRM58cDhKzXik2kFme3viX7",
915 testNetwork(done, params);
917 it('Allows selection
of Einsteinium
', function(done) {
919 selectText: "EMC2 - Einsteinium",
920 firstAddress: "EVAABm9hXKHk2MpVMbwNakRubFnNha5m8m",
922 testNetwork(done, params);
924 it('Allows selection
of EOSIO
', function(done) {
926 selectText: "EOS - EOSIO",
927 firstPubKey: "EOS692VJTBK3Rmw93onNnpnZ8ZtmE9PdxjDStArvbyzoe11QUTNoy",
929 testNetwork(done, params, true);
931 it('Allows selection
of Europecoin
', function(done) {
933 selectText: "ERC - Europecoin",
934 firstAddress: "ESA2YwPYntAoaPrE8Fm5qkKRtkcwLcwD6R",
936 testNetwork(done, params);
938 it('Allows selection
of Exclusivecoin
', function(done) {
940 selectText: "EXCL - Exclusivecoin",
941 firstAddress: "EbUa6m8UZW6nTxsYZD2FsDjkadKbp5M6JT",
943 testNetwork(done, params);
945 it('Allows selection
of Feathercoin
', function(done) {
947 selectText: "FTC - Feathercoin",
948 firstAddress: "6gDdjAMoSgQaW8UhqK3oboHs6ftGAroKkM",
950 testNetwork(done, params);
952 it('Allows selection
of Firstcoin
', function(done) {
954 selectText: "FRST - Firstcoin",
955 firstAddress: "FJN9GzfMm7Q8R4DJwK1H9F6A1GTghvFiMJ",
957 testNetwork(done, params);
959 it('Allows selection
of Flashcoin
', function(done) {
961 selectText: "FLASH - Flashcoin",
962 firstAddress: "UWfpf5LfMmLxZYooEb2EyvWhZ8NG7EZDRt",
964 testNetwork(done, params);
966 it('Allows selection
of GCRCoin
', function(done) {
968 selectText: "GCR - GCRCoin",
969 firstAddress: "GJjF5cLwyXLacpuvXAVksxGxKvHDjx58d6",
971 testNetwork(done, params);
973 it('Allows selection
of Gobyte
', function(done) {
975 selectText: "GBX - Gobyte",
976 firstAddress: "GS813Ys2brkmvSUw1rUqGPm2HqQVDHJRyA",
978 testNetwork(done, params);
980 it('Allows selection
of Gridcoin
', function(done) {
982 selectText: "GRC - Gridcoin",
983 firstAddress: "SGrWbBPvobgqKRF8td1Kdc9vbRY7MJ78Y9",
985 testNetwork(done, params);
987 it('Allows selection
of Gulden
', function(done) {
989 selectText: "NLG - Gulden",
990 firstAddress: "GcDP7cNEc33MPPdTFNJ8pZc6VMZJ2CbKxY",
992 testNetwork(done, params);
994 it('Allows selection
of Helleniccoin
', function(done) {
996 selectText: "HNC - Helleniccoin",
997 firstAddress: "LbHEKe5H72zp9G1fuWNiiNePTUfJb88915",
999 testNetwork(done, params);
1001 it('Allows selection
of Hempcoin
', function(done) {
1003 selectText: "THC - Hempcoin",
1004 firstAddress: "H8sdWbZyJV4gyXyHtLXDaNnAuUDhK5mfTV",
1006 testNetwork(done, params);
1008 it('Allows selection
of Insane
', function(done) {
1010 selectText: "INSN - Insane",
1011 firstAddress: "iMPqEJMiXWuxC9U2NVinCCMr4t72h58EWx",
1013 testNetwork(done, params);
1015 it('Allows selection
of Iop
', function(done) {
1017 selectText: "IOP - Iop",
1018 firstAddress: "pGKQmcaPf95Ur5o6oHK4qdiZ52p1yaTvq1",
1020 testNetwork(done, params);
1022 it('Allows selection
of Ixcoin
', function(done) {
1024 selectText: "IXC - Ixcoin",
1025 firstAddress: "xgE9bTZ6YypT3E6ByzkTt31Hq68E9BqywH",
1027 testNetwork(done, params);
1029 it('Allows selection
of Kobocoin
', function(done) {
1031 selectText: "KOBO - Kobocoin",
1032 firstAddress: "FTVoNJETXDAM8x7MnmdE8RwWndSr9PQWhy",
1034 testNetwork(done, params);
1036 it('Allows selection
of Landcoin
', function(done) {
1038 selectText: "LDCN - Landcoin",
1039 firstAddress: "LLvLwNjG1aJcn1RS4W4GJUbv8fNaRATG7c",
1041 testNetwork(done, params);
1043 it('Allows selection
of Library Credits
', function(done) {
1045 selectText: "LBC - Library Credits",
1046 firstAddress: "bQJEQrHDJyHdqycB32uysh1SWn8Ln8LMdg",
1048 testNetwork(done, params);
1050 it('Allows selection
of Linx
', function(done) {
1052 selectText: "LINX - Linx",
1053 firstAddress: "XGWQ3cb3LGUB3VnHmj6xYSMgnokNbf6dyk",
1055 testNetwork(done, params);
1057 it('Allows selection
of Litecoincash
', function(done) {
1059 selectText: "LCC - Litecoincash",
1060 firstAddress: "Ce5n7fjUuQPLutJ4W5nCCfQLKdKLE1mv9A",
1062 testNetwork(done, params);
1064 it('Allows selection
of Lynx
', function(done) {
1066 selectText: "LYNX - Lynx",
1067 firstAddress: "KUeY3ZdZkg96p4W98pj1JjygCFU1XqWdw3",
1069 testNetwork(done, params);
1071 it('Allows selection
of Megacoin
', function(done) {
1073 selectText: "MEC - Megacoin",
1074 firstAddress: "MDfAj9CzkC1HpcUiVGnHp8yKTa7WXgu8AY",
1076 testNetwork(done, params);
1078 it('Allows selection
of Minexcoin
', function(done) {
1080 selectText: "MNX - Minexcoin",
1081 firstAddress: "XC1VnyJVfiMDwWgFtAHDp41cgY3AHk3dJT",
1083 testNetwork(done, params);
1085 it('Allows selection
of Navcoin
', function(done) {
1087 selectText: "NAV - Navcoin",
1088 firstAddress: "NTQVTPK3NWSQLKoffkiQw99T8PifkF1Y2U",
1090 testNetwork(done, params);
1092 it('Allows selection
of Nebulas
', function(done) {
1094 selectText: "NAS - Nebulas",
1095 firstAddress: "n1PbK61DGBfDoDusLw621G6sVSMfLLHdfnm",
1097 testNetwork(done, params);
1099 it('Allows selection
of Neoscoin
', function(done) {
1101 selectText: "NEOS - Neoscoin",
1102 firstAddress: "NgATz6QbQNXvayHQ4CpZayugb9HeaPDdby",
1104 testNetwork(done, params);
1106 it('Allows selection
of Nix
', function(done) {
1108 selectText: "NIX - NIX Platform",
1109 firstAddress: "GgcNW2SQQXB4LWHRQTHKkQF3GzXNSLqS8u",
1111 testNetwork(done, params);
1113 it('Allows selection
of Neurocoin
', function(done) {
1115 selectText: "NRO - Neurocoin",
1116 firstAddress: "NVdYErQ3mFpDuF5DquW9WMiT7sLc8ufFTn",
1118 testNetwork(done, params);
1120 it('Allows selection
of Newyorkc
', function(done) {
1122 selectText: "NYC - Newyorkc",
1123 firstAddress: "RSVMfyH1fKfy3puADJEhut2vfkRyon6imm",
1125 testNetwork(done, params);
1127 it('Allows selection
of Novacoin
', function(done) {
1129 selectText: "NVC - Novacoin",
1130 firstAddress: "4JRvUmxcKCJmaMXZyvRoSS1cmG2XvnZfHN",
1132 testNetwork(done, params);
1134 it('Allows selection
of Nushares
', function(done) {
1136 selectText: "NSR - Nushares",
1137 firstAddress: "SecjXzU3c7EecdT7EbC4vvmbdtBBokWh6J",
1139 testNetwork(done, params);
1141 it('Allows selection
of Okcash
', function(done) {
1143 selectText: "OK - Okcash",
1144 firstAddress: "PV4Qp1TUYuGv4TqVtLZtqvrsWWRycfx1Yi",
1146 testNetwork(done, params);
1148 it('Allows selection
of Omnicore
', function(done) {
1150 selectText: "OMNI - Omnicore",
1151 firstAddress: "1Q1t3gonjCT3rW38TsTsCvgSc3hh7zBGbi",
1153 testNetwork(done, params);
1155 it('Allows selection
of DeepOnion
', function(done) {
1157 selectText: "ONION - DeepOnion",
1158 firstAddress: "DYREY7XCFXVqJ3x5UuN43k2JwD2s1kif48",
1160 testNetwork(done, params);
1162 it('Allows selection
of Pesobit
', function(done) {
1164 selectText: "PSB - Pesobit",
1165 firstAddress: "PDePsF7ALyXP7JaywokdYiRTDtKa14MAr1",
1167 testNetwork(done, params);
1169 it('Allows selection
of Pinkcoin
', function(done) {
1171 selectText: "PINK - Pinkcoin",
1172 firstAddress: "2TgjYQffjbzUHJghNaVbdsjHbRwruC3yzC",
1174 testNetwork(done, params);
1176 it('Allows selection
of POSWcoin
', function(done) {
1178 selectText: "POSW - POSWcoin",
1179 firstAddress: "PNxewmZoPnGBvoEbH6hgQZCK1igDiBCdgC",
1181 testNetwork(done, params);
1183 it('Allows selection
of Potcoin
', function(done) {
1185 selectText: "POT - Potcoin",
1186 firstAddress: "PEo7Vg2ctXgpP4vuLPeY9aGJtZotyrmiHc",
1188 testNetwork(done, params);
1190 it('Allows selection
of Putincoin
', function(done) {
1192 selectText: "PUT - Putincoin",
1193 firstAddress: "PViWnfr2uFtovd6e7joM49C94CsGSnqJis",
1195 testNetwork(done, params);
1197 it('Allows selection
of Rapids
', function(done) {
1199 selectText: "RPD - Rapids",
1200 firstAddress: "Ri8XxUdZaXS5LqxmFJcFEjFinkaMbmhSUp",
1202 testNetwork(done, params);
1204 it('Allows selection
of Ravencoin
', function(done) {
1206 selectText: "RVN - Ravencoin",
1207 firstAddress: "RBuDoVNnzvFsEcX8XKPm8ic4mgiCzjUCNk",
1209 testNetwork(done, params);
1211 it('Allows selection
of Reddcoin
', function(done) {
1213 selectText: "RDD - Reddcoin",
1214 firstAddress: "RtgRvXMBng1y51ftteveFqwNfyRG18HpxQ",
1216 testNetwork(done, params);
1218 it('Allows selection
of RevolutionVR
', function(done) {
1220 selectText: "RVR - RevolutionVR",
1221 firstAddress: "VXeeoP2jkzZnMFxtc66ZBZK1NHN5QJnnjL",
1223 testNetwork(done, params);
1225 it('Allows selection
of Rubycoin
', function(done) {
1227 selectText: "RBY - Rubycoin",
1228 firstAddress: "RV76JDtjTs11JdMDRToYn6CHecMRPLnKS6",
1230 testNetwork(done, params);
1232 it('Allows selection
of Salus
', function(done) {
1234 selectText: "SLS - Salus",
1235 firstAddress: "SNzPi1CafHFm3WWjRo43aMgiaEEj3ogjww",
1237 testNetwork(done, params);
1239 it('Allows selection
of Smileycoin
', function(done) {
1241 selectText: "SMLY - Smileycoin",
1242 firstAddress: "BEZVnEBCAyFByrgKpwAgYgtvP4rKAd9Sj2",
1244 testNetwork(done, params);
1246 it('Allows selection
of Solarcoin
', function(done) {
1248 selectText: "SLR - Solarcoin",
1249 firstAddress: "8LZ13HbnjtaMJWSvvVFNTLf71zFfDrhwLu",
1251 testNetwork(done, params);
1253 it('Allows selection
of stash
', function(done) {
1255 selectText: "STASH - Stash",
1256 firstAddress: "XxwAsWB7REDKmAvHA85SbEZQQtpxeUDxS3",
1258 testNetwork(done, params);
1260 it('Allows selection
of stash testnet
', function(done) {
1262 selectText: "STASH - Stash Testnet",
1263 firstAddress: "yWQCTSkUst7ddYuebKsqa1kSoXEjpCkGKR",
1265 testNetwork(done, params);
1267 it('Allows selection
of Stratis
', function(done) {
1269 selectText: "STRAT - Stratis",
1270 firstAddress: "ScfJnq3QDhKgDMEds6sqUE1ot6ShfhmXXq",
1272 testNetwork(done, params);
1274 it('Allows selection
of Stratis Test
', function(done) {
1276 selectText: "TSTRAT - Stratis Testnet",
1277 firstAddress: "TRLWm3dye4FRrDWouwYUSUZP96xb76mBE3",
1279 testNetwork(done, params);
1281 it('Allows selection
of Syscoin
', function(done) {
1283 selectText: "SYS - Syscoin",
1284 firstAddress: "SZwJi42Pst3VAMomyK5DG4157WM5ofRmSj",
1286 testNetwork(done, params);
1288 it('Allows selection
of Toa
', function(done) {
1290 selectText: "TOA - Toa",
1291 firstAddress: "TSe1QAnUwQzUfbBusDzRJ9URttrRGKoNKF",
1293 testNetwork(done, params);
1295 it('Allows selection
of TWINS
', function(done) {
1297 selectText: "TWINS - TWINS",
1298 firstAddress: "WPpJnfLLubNmF7HLNxg8d8zH5haxn4wri8",
1300 testNetwork(done, params);
1302 it('Allows selection
of TWINS testnet
', function(done) {
1304 selectText: "TWINS - TWINS Testnet",
1305 firstAddress: "XpnU1HHdNG5YxvG9Rez4wjmidchxqnZaNa",
1307 testNetwork(done, params);
1309 it('Allows selection
of Ultimatesecurecash
', function(done) {
1311 selectText: "USC - Ultimatesecurecash",
1312 firstAddress: "UPyLAZU2Che5fiy7Ed8xVJFmXAUhitA4ug",
1314 testNetwork(done, params);
1316 it('Allows selection
of Unobtanium
', function(done) {
1318 selectText: "UNO - Unobtanium",
1319 firstAddress: "uUBMPVMXrR6qhqornJqKTWgr8L69vihSL9",
1321 testNetwork(done, params);
1323 it('Allows selection
of Vcash
', function(done) {
1325 selectText: "XVC - Vcash",
1326 firstAddress: "VuL53MSY6KjvAjKSeRkh3NDnKykacDVeps",
1328 testNetwork(done, params);
1330 it('Allows selection
of Verge
', function(done) {
1332 selectText: "XVG - Verge",
1333 firstAddress: "DCrVuGkMjLJpTGgwAgv9AcMdeb1nkWbjZA",
1335 testNetwork(done, params);
1337 it('Allows selection
of Vertcoin
', function(done) {
1339 selectText: "VTC - Vertcoin",
1340 firstAddress: "Vf6koGuiWdXQfx8tNqxoNeEDxh4xh5cxsG",
1342 testNetwork(done, params);
1344 it('Allows selection
of Vivo
', function(done) {
1346 selectText: "VIVO - Vivo",
1347 firstAddress: "VFmBwuXXGhJe7MarQG2GfzHMFebRHgfSpB",
1349 testNetwork(done, params);
1351 it('Allows selection
of Vpncoin
', function(done) {
1353 selectText: "VASH - Vpncoin",
1354 firstAddress: "VoEmH1qXC4TsSgBAStR21QYetwnFqbqCx9",
1356 testNetwork(done, params);
1358 it('Allows selection
of VeChain
', function(done) {
1360 selectText: "VET - VeChain",
1361 firstAddress: "0xdba55B1B6070f3a733D5eDFf35F0da4A00E455F2",
1363 testNetwork(done, params);
1365 it('Allows selection
of Whitecoin
', function(done) {
1367 selectText: "XWC - Whitecoin",
1368 firstAddress: "WcSwCAUqrSgeSYbsaS3SSWWhsx8KRYTFDR",
1370 testNetwork(done, params);
1372 it('Allows selection
of Wincoin
', function(done) {
1374 selectText: "WC - Wincoin",
1375 firstAddress: "WaDVCESMGgyKgNESdn3u43NnwmGSkZED3Z",
1377 testNetwork(done, params);
1379 it('Allows selection
of Zcoin
', function(done) {
1381 selectText: "XZC - Zcoin",
1382 firstAddress: "a6VcMdP4XgAA9Tr7xNszmPG5FZpfRf17Cq",
1384 testNetwork(done, params);
1386 it('Allows selection
of Zcash
', function(done) {
1388 selectText: "ZEC - Zcash",
1389 firstAddress: "t1Sz8AneMcVuzUg3tPJ8et5AS5LFJ7K2EF9",
1391 testNetwork(done, params);
1393 it('Allows selection
of Zclassic
', function(done) {
1395 selectText: "ZCL - Zclassic",
1396 firstAddress: "t1TBMxTvVJRybUbMLGWq8H4A8F4VUL7czEc",
1398 testNetwork(done, params);
1400 it('Allows selection
of Horizen
', function(done) {
1402 selectText: "ZEN - Horizen",
1403 firstAddress: "znWh9XASyW2dZq5tck84wFjiwuqVysi7q3p",
1405 testNetwork(done, params);
1407 it('Allows selection
of Energi
', function(done) {
1409 selectText: "NRG - Energi",
1410 firstAddress: "EejRy4t4nidzhGGzkJUgFP3z4HYBjhTsRt",
1412 testNetwork(done, params);
1414 it('Allows selection
of Ethereum Classic
', function(done) {
1416 selectText: "ETC - Ethereum Classic",
1417 firstAddress: "0x3c05e5556693808367afB62eF3b63e35d6eD249A",
1419 testNetwork(done, params);
1421 it('Allows selection
of Pirl
', function(done) {
1423 selectText: "PIRL - Pirl",
1424 firstAddress: "0xe77FC0723dA122B5025CA79193c28563eB47e776",
1426 testNetwork(done, params);
1428 it('Allows selection
of MIX
', function(done) {
1430 selectText: "MIX - MIX",
1431 firstAddress: "0x98BC5e63aeb6A4e82d72850d20710F07E29A29F1",
1433 testNetwork(done, params);
1435 it('Allows selection
of Monkey Project
', function(done) {
1437 selectText: "MONK - Monkey Project",
1438 firstAddress: "MnLrcnnUzKnf7TzufjRe5DLZqQJz18oYyu",
1440 testNetwork(done, params);
1443 it('Allows selection
of Musicoin
', function(done) {
1445 selectText: "MUSIC - Musicoin",
1446 firstAddress: "0xDc060e4A0b0313ea83Cf6B3A39B9db2D29004897",
1448 testNetwork(done, params);
1450 it('Allows selection
of Poa
', function(done) {
1452 selectText: "POA - Poa",
1453 firstAddress: "0x53aF28d754e106210C3d0467Dd581eaf7e3C5e60",
1455 testNetwork(done, params);
1457 it('Allows selection
of Expanse
', function(done) {
1459 selectText: "EXP - Expanse",
1460 firstAddress: "0xf57FeAbf26582b6E3E666559d3B1Cc6fB2b2c5F6",
1462 testNetwork(done, params);
1464 it('Allows selection
of Callisto
', function(done) {
1466 selectText: "CLO - Callisto",
1467 firstAddress: "0x4f9364F7420B317266C51Dc8eB979717D4dE3f4E",
1469 testNetwork(done, params);
1471 it('Allows selection
of HUSH
', function(done) {
1473 selectText: "HUSH - Hush (Legacy)",
1474 firstAddress: "t1g6rLXUnJaiJuu4q4zmJjoa9Gk4fwKpiuA",
1476 testNetwork(done, params);
1478 it('Allows selection
of HUSH3
', function(done) {
1480 selectText: "HUSH - Hush3",
1481 firstAddress: "RXWSQhwvw5jHPGP8bjwJhWoRnMLBnuPDKD",
1483 testNetwork(done, params);
1485 it('Allows selection
of ExchangeCoin
', function(done) {
1487 selectText: "EXCC - ExchangeCoin",
1488 firstAddress: "22txYKpFN5fwGwdSs2UBf7ywewbLM92YqK7E",
1490 testNetwork(done, params);
1492 it('Allows selection
of Artax
', function(done) {
1494 selectText: "XAX - Artax",
1495 firstAddress: "AYxaQPY7XLidG31V7F3yNzwxPYpYzRqG4q",
1497 testNetwork(done, params);
1499 it('Allows selection
of BitcoinGreen
', function(done) {
1501 selectText: "BITG - Bitcoin Green",
1502 firstAddress: "GeNGm9SkEfwbsws3UrrUSE2sJeyWYjzraY",
1504 testNetwork(done, params);
1506 it('Allows selection
of ANON
', function(done) {
1508 selectText: "ANON - ANON",
1509 firstAddress: "AnU6pijpEeUZFWSTyM2qTqZQn996Zq1Xard",
1511 testNetwork(done, params);
1513 it('Allows selection
of ProjectCoin
', function(done) {
1515 selectText: "PRJ - ProjectCoin",
1516 firstAddress: "PXZG97saRseSCftfe1mcFmfAA7pf6qBbaz",
1518 testNetwork(done, params);
1520 it('Allows selection
of Phore
', function(done) {
1522 selectText: "PHR - Phore",
1523 firstAddress: "PJThxpoXAG6hqrmdeQQbVDX4TJtFTMMymC",
1525 testNetwork(done, params);
1527 it('Allows selection
of Safecoin
', function(done) {
1529 selectText: "SAFE - Safecoin",
1530 firstAddress: "RtxHpnhJz6RY8k9owP3ua5QWraunmewB1G",
1532 testNetwork(done, params);
1534 it('Allows selection
of Blocknode
', function(done) {
1536 selectText: "BND - Blocknode",
1537 firstAddress: "BG8xZSAur2jYLG9VXt8dYfkKxxeR7w9bSe",
1539 testNetwork(done, params);
1541 it('Allows selection
of Blocknode Testnet
', function(done) {
1543 selectText: "tBND - Blocknode Testnet",
1544 firstAddress: "bSptsFyDktFSKpWveRywJsDoJA2TC6qfHv",
1546 testNetwork(done, params);
1548 it('Allows selection
of LitecoinZ
', function(done) {
1550 selectText: "LTZ - LitecoinZ",
1551 firstAddress: "L1VTXju7hLgKV4T7fGXS9sKsnm2gmtRCmyw",
1553 testNetwork(done, params);
1555 it('Allows selection
of BlockStamp
', function(done) {
1557 selectText: "BST - BlockStamp",
1558 firstAddress: "15gypKtim4cVTj137ApfryG17RkvSbPazZ",
1560 testNetwork(done, params);
1562 it('Allows selection
of DEXON
', function(done) {
1564 selectText: "DXN - DEXON",
1565 firstAddress: "0x136a58788033E028CCd740FbDec6734358DB56Ec",
1567 testNetwork(done, params);
1569 it('Allows selection
of Ellaism
', function(done) {
1571 selectText: "ELLA - Ellaism",
1572 firstAddress: "0xa8B0BeA09eeBc41062308546a01d6E544277e2Ca",
1574 testNetwork(done, params);
1576 it('Allows selection
of Ethersocial Network
', function(done) {
1578 selectText: "ESN - Ethersocial Network",
1579 firstAddress: "0x6EE99Be2A0C7F887a71e21C8608ACF0aa0D2b767",
1581 testNetwork(done, params);
1583 it('Allows selection
of Stellar
', function(done) {
1585 selectText: "XLM - Stellar",
1586 firstAddress: "GCUK3NYYUXA2QGN6KU5RR36WAKN3Y5EANZV65XNAWN4XM4CHQ3G4DMO2",
1588 testNetwork(done, params);
1590 it('Allows selection
of Wagerr
', function(done) {
1592 selectText: "WGR - Wagerr",
1593 firstAddress: "WYiVgQU39VcQxcnacoCiaZHZZLjDCJoS95",
1595 testNetwork(done, params);
1597 it('Allows selection
of Groestlcoin
', function(done) {
1599 selectText: "GRS - Groestlcoin",
1600 firstAddress: "FZycsFvZ1eH1hbtyjBpAgJSukVw1bN6PBN",
1602 testNetwork(done, params);
1604 it('Allows selection
of Groestlcoin Testnet
', function(done) {
1606 selectText: "GRS - Groestlcoin Testnet",
1607 firstAddress: "mucaU5iiDaJDb69BHLeDv8JFfGiygRPne9",
1609 testNetwork(done, params);
1612 // BIP39 seed is set from phrase
1613 it('Sets the bip39 seed
from the prhase
', function(done) {
1614 driver.findElement(By.css('.phrase
'))
1615 .sendKeys('abandon abandon ability
');
1616 driver.sleep(generateDelay).then(function() {
1617 driver.findElement(By.css('.seed
'))
1618 .getAttribute("value")
1619 .then(function(seed) {
1620 expect(seed).toBe("20da140d3dd1df8713cefcc4d54ce0e445b4151027a1ab567b832f6da5fcc5afc1c3a3f199ab78b8e0ab4652efd7f414ac2c9a3b81bceb879a70f377aa0a58f3");
1626 // BIP32 root key is set from phrase
1627 it('Sets the bip39 root key
from the prhase
', function(done) {
1628 driver.findElement(By.css('.phrase
'))
1629 .sendKeys('abandon abandon ability
');
1630 driver.sleep(generateDelay).then(function() {
1631 driver.findElement(By.css('.root
-key
'))
1632 .getAttribute("value")
1633 .then(function(seed) {
1634 expect(seed).toBe("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
1640 // Tabs show correct addresses when changed
1641 it('Shows the correct address when tab is changed
', function(done) {
1642 driver.findElement(By.css('.phrase
'))
1643 .sendKeys('abandon abandon ability
');
1644 driver.sleep(generateDelay).then(function() {
1645 driver.findElement(By.css('#bip32
-tab a
'))
1647 driver.sleep(generateDelay).then(function() {
1648 getFirstAddress(function(address) {
1649 expect(address).toBe("17uQ7s2izWPwBmEVFikTmZUjbBKWYdJchz");
1656 // BIP44 derivation path is shown
1657 it('Shows the derivation path
for bip44 tab
', function(done) {
1658 driver.findElement(By.css('.phrase
'))
1659 .sendKeys('abandon abandon ability
');
1660 driver.sleep(generateDelay).then(function() {
1661 driver.findElement(By.css('#bip44
.path
'))
1662 .getAttribute("value")
1663 .then(function(path) {
1664 expect(path).toBe("m/44'/0'/0'/0");
1670 // BIP44 extended private key is shown
1671 it('Shows the extended private key for bip44 tab', function(done) {
1672 driver.findElement(By.css('.phrase'))
1673 .sendKeys('abandon abandon ability');
1674 driver.sleep(generateDelay).then(function() {
1675 driver.findElement(By.css('.extended-priv-key'))
1676 .getAttribute("value
")
1677 .then(function(path) {
1678 expect(path).toBe("xprvA2DxxvPZcyRvYgZMGS53nadR32mVDeCyqQYyFhrCVbJNjPoxMeVf7QT5g7mQASbTf9Kp4cryvcXnu2qurjWKcrdsr91jXymdCDNxKgLFKJG
");
1684 // BIP44 extended public key is shown
1685 it('Shows the extended public key for bip44 tab', function(done) {
1686 driver.findElement(By.css('.phrase'))
1687 .sendKeys('abandon abandon ability');
1688 driver.sleep(generateDelay).then(function() {
1689 driver.findElement(By.css('.extended-pub-key'))
1690 .getAttribute("value
")
1691 .then(function(path) {
1692 expect(path).toBe("xpub6FDKNRvTTLzDmAdpNTc49ia9b4byd6vqCdUa46Fp3vqMcC96uBoufCmZXQLiN5AK3iSCJMhf9gT2sxkpyaPepRuA7W3MujV5tGmF5VfbueM
");
1698 // BIP44 account field changes address list
1699 it('Changes the address list if bip44 account is changed', function(done) {
1700 driver.findElement(By.css('#bip44 .account'))
1702 driver.findElement(By.css('.phrase'))
1703 .sendKeys('abandon abandon ability');
1704 driver.sleep(generateDelay).then(function() {
1705 getFirstAddress(function(address) {
1706 expect(address).toBe("1Nq2Wmu726XHCuGhctEtGmhxo3wzk5wZ1H
");
1712 // BIP44 change field changes address list
1713 it('Changes the address list if bip44 change is changed', function(done) {
1714 driver.findElement(By.css('#bip44 .change'))
1716 driver.findElement(By.css('.phrase'))
1717 .sendKeys('abandon abandon ability');
1718 driver.sleep(generateDelay).then(function() {
1719 getFirstAddress(function(address) {
1720 expect(address).toBe("1KAGfWgqfVbSSXY56fNQ7YnhyKuoskHtYo
");
1726 // BIP32 derivation path can be set
1727 it('Can use a custom bip32 derivation path', function(done) {
1728 driver.findElement(By.css('#bip32-tab a'))
1730 driver.findElement(By.css('#bip32 .path'))
1732 driver.findElement(By.css('#bip32 .path'))
1734 driver.findElement(By.css('.phrase'))
1735 .sendKeys('abandon abandon ability');
1736 driver.sleep(generateDelay).then(function() {
1737 getFirstAddress(function(address) {
1738 expect(address).toBe("16pYQQdLD1hH4hwTGLXBaZ9Teboi1AGL8L
");
1744 // BIP32 can use hardened derivation paths
1745 it('Can use a hardened derivation paths', function(done) {
1746 driver.findElement(By.css('#bip32-tab a'))
1748 driver.findElement(By.css('#bip32 .path'))
1750 driver.findElement(By.css('#bip32 .path'))
1752 driver.findElement(By.css('.phrase
'))
1753 .sendKeys('abandon abandon ability
');
1754 driver.sleep(generateDelay).then(function() {
1755 getFirstAddress(function(address) {
1756 expect(address).toBe("14aXZeprXAE3UUKQc4ihvwBvww2LuEoHo4");
1762 // BIP32 extended private key is shown
1763 it('Shows the BIP32 extended
private key
', function(done) {
1764 driver.findElement(By.css('#bip32
-tab a
'))
1766 driver.findElement(By.css('.phrase
'))
1767 .sendKeys('abandon abandon ability
');
1768 driver.sleep(generateDelay).then(function() {
1769 driver.findElement(By.css('.extended
-priv
-key
'))
1770 .getAttribute("value")
1771 .then(function(privKey) {
1772 expect(privKey).toBe("xprv9va99uTVE5aLiutUVLTyfxfe8v8aaXjSQ1XxZbK6SezYVuikA9MnjQVTA8rQHpNA5LKvyQBpLiHbBQiiccKiBDs7eRmBogsvq3THFeLHYbe");
1778 // BIP32 extended public key is shown
1779 it('Shows the BIP32 extended
public key
', function(done) {
1780 driver.findElement(By.css('#bip32
-tab a
'))
1782 driver.findElement(By.css('.phrase
'))
1783 .sendKeys('abandon abandon ability
');
1784 driver.sleep(generateDelay).then(function() {
1785 driver.findElement(By.css('.extended
-pub
-key
'))
1786 .getAttribute("value")
1787 .then(function(pubKey) {
1788 expect(pubKey).toBe("xpub69ZVZQzP4T8dwPxwbMzz36cNgwy4yzTHmETZMyihzzXXNi3thgg3HCow1RtY252wdw5rS8369xKnraN5Q93y3FkFfJp2XEHWUrkyXsjS93P");
1794 // Derivation path is shown in table
1795 it('Shows the derivation path
in the table
', function(done) {
1796 driver.findElement(By.css('.phrase
'))
1797 .sendKeys('abandon abandon ability
');
1798 driver.sleep(generateDelay).then(function() {
1799 getFirstPath(function(path) {
1800 expect(path).toBe("m/44'/0'/0'/0/0");
1806 // Derivation path for address can be hardened
1807 it('Can derive hardened addresses', function(done) {
1808 driver.findElement(By.css('#bip32-tab a'))
1810 driver.executeScript(function() {
1811 $(".hardened
-addresses
").prop("checked
", true);
1813 driver.findElement(By.css('.phrase'))
1814 .sendKeys('abandon abandon ability');
1815 driver.sleep(generateDelay).then(function() {
1816 getFirstAddress(function(address) {
1817 expect(address).toBe("18exLzUv7kfpiXRzmCjFDoC9qwNLFyvwyd
");
1823 // Derivation path visibility can be toggled
1824 it('Can toggle visibility of the derivation path column', function(done) {
1825 driver.findElement(By.css('.phrase'))
1826 .sendKeys('abandon abandon ability');
1827 driver.sleep(generateDelay).then(function() {
1828 driver.findElement(By.css('.index-toggle'))
1830 testColumnValuesAreInvisible(done, "index
");
1835 it('Shows the address in the table', function(done) {
1836 driver.findElement(By.css('.phrase'))
1837 .sendKeys('abandon abandon ability');
1838 driver.sleep(generateDelay).then(function() {
1839 getFirstAddress(function(address) {
1840 expect(address).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug
");
1846 // Addresses are shown in order of derivation path
1847 it('Shows the address in order of derivation path', function(done) {
1848 driver.findElement(By.css('.phrase'))
1849 .sendKeys('abandon abandon ability');
1850 driver.sleep(generateDelay).then(function() {
1851 testRowsAreInCorrectOrder(done);
1855 // Address visibility can be toggled
1856 it('Can toggle visibility of the address column', function(done) {
1857 driver.findElement(By.css('.phrase'))
1858 .sendKeys('abandon abandon ability');
1859 driver.sleep(generateDelay).then(function() {
1860 driver.findElement(By.css('.address-toggle'))
1862 testColumnValuesAreInvisible(done, "address
");
1866 // Public key is shown in table
1867 it('Shows the public key in the table', function(done) {
1868 driver.findElement(By.css('.phrase'))
1869 .sendKeys('abandon abandon ability');
1870 driver.sleep(generateDelay).then(function() {
1871 driver.findElements(By.css('.pubkey'))
1872 .then(function(els) {
1874 .then(function(pubkey) {
1875 expect(pubkey).toBe("033f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3
");
1882 // Public key visibility can be toggled
1883 it('Can toggle visibility of the public key column', function(done) {
1884 driver.findElement(By.css('.phrase'))
1885 .sendKeys('abandon abandon ability');
1886 driver.sleep(generateDelay).then(function() {
1887 driver.findElement(By.css('.public-key-toggle'))
1889 testColumnValuesAreInvisible(done, "pubkey
");
1893 // Private key is shown in table
1894 it('Shows the private key in the table', function(done) {
1895 driver.findElement(By.css('.phrase'))
1896 .sendKeys('abandon abandon ability');
1897 driver.sleep(generateDelay).then(function() {
1898 driver.findElements(By.css('.privkey'))
1899 .then(function(els) {
1901 .then(function(pubkey) {
1902 expect(pubkey).toBe("L26cVSpWFkJ6aQkPkKmTzLqTdLJ923e6CzrVh9cmx21QHsoUmrEE
");
1909 // Private key visibility can be toggled
1910 it('Can toggle visibility of the private key column', function(done) {
1911 driver.findElement(By.css('.phrase'))
1912 .sendKeys('abandon abandon ability');
1913 driver.sleep(generateDelay).then(function() {
1914 driver.findElement(By.css('.private-key-toggle'))
1916 testColumnValuesAreInvisible(done, "privkey
");
1920 // More addresses can be generated
1921 it('Can generate more rows in the table', function(done) {
1922 driver.findElement(By.css('.phrase'))
1923 .sendKeys('abandon abandon ability');
1924 driver.sleep(generateDelay).then(function() {
1925 driver.findElement(By.css('.more'))
1927 driver.sleep(generateDelay).then(function() {
1928 driver.findElements(By.css('.address'))
1929 .then(function(els) {
1930 expect(els.length).toBe(40);
1937 // A custom number of additional addresses can be generated
1938 it('Can generate more rows in the table', function(done) {
1939 driver.findElement(By.css('.phrase'))
1940 .sendKeys('abandon abandon ability');
1941 driver.sleep(generateDelay).then(function() {
1942 driver.findElement(By.css('.rows-to-add'))
1944 driver.findElement(By.css('.rows-to-add'))
1946 driver.findElement(By.css('.more'))
1948 driver.sleep(generateDelay).then(function() {
1949 driver.findElements(By.css('.address'))
1950 .then(function(els) {
1951 expect(els.length).toBe(21);
1958 // Additional addresses are shown in order of derivation path
1959 it('Shows additional addresses in order of derivation path', function(done) {
1960 driver.findElement(By.css('.phrase'))
1961 .sendKeys('abandon abandon ability');
1962 driver.sleep(generateDelay).then(function() {
1963 driver.findElement(By.css('.more'))
1965 driver.sleep(generateDelay).then(function() {
1966 testRowsAreInCorrectOrder(done);
1971 // BIP32 root key can be set by the user
1972 it('Allows the user to set the BIP32 root key', function(done) {
1973 driver.findElement(By.css('.root-key'))
1974 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1975 driver.sleep(generateDelay).then(function() {
1976 getFirstAddress(function(address) {
1977 expect(address).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug
");
1983 // Setting BIP32 root key clears the existing phrase, passphrase and seed
1984 // TODO this doesn't work in selenium with chrome
1985 it('Confirms the existing phrase should be cleared', function(done) {
1986 if (browser == "chrome
") {
1987 pending("Selenium
+ Chrome headless bug
for alert
, see
https://stackoverflow.com/q/45242264");
1989 driver
.findElement(By
.css('.phrase'))
1990 .sendKeys('A non-blank but invalid value');
1991 driver
.findElement(By
.css('.root-key'))
1992 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1993 driver
.switchTo().alert().accept();
1994 driver
.findElement(By
.css('.phrase'))
1995 .getAttribute("value").then(function(value
) {
1996 expect(value
).toBe("");
2001 // Clearing of phrase, passphrase and seed can be cancelled by user
2002 // TODO this doesn't work in selenium with chrome
2003 it('Allows the clearing of the phrase to be cancelled', function(done
) {
2004 if (browser
== "chrome") {
2005 pending("Selenium + Chrome headless bug for alert, see https://stackoverflow.com/q/45242264");
2007 driver
.findElement(By
.css('.phrase'))
2008 .sendKeys('abandon abandon ability');
2009 driver
.sleep(generateDelay
).then(function() {
2010 driver
.findElement(By
.css('.root-key'))
2012 driver
.findElement(By
.css('.root-key'))
2014 driver
.switchTo().alert().dismiss();
2015 driver
.findElement(By
.css('.phrase'))
2016 .getAttribute("value").then(function(value
) {
2017 expect(value
).toBe("abandon abandon ability");
2023 // Custom BIP32 root key is used when changing the derivation path
2024 it('Can set derivation path for root key instead of phrase', function(done
) {
2025 driver
.findElement(By
.css('#bip44 .account'))
2027 driver
.findElement(By
.css('.root-key'))
2028 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
2029 driver
.sleep(generateDelay
).then(function() {
2030 getFirstAddress(function(address
) {
2031 expect(address
).toBe("1Nq2Wmu726XHCuGhctEtGmhxo3wzk5wZ1H");
2037 // Incorrect mnemonic shows error
2038 it('Shows an error for incorrect mnemonic', function(done
) {
2039 driver
.findElement(By
.css('.phrase'))
2040 .sendKeys('abandon abandon abandon');
2041 driver
.sleep(feedbackDelay
).then(function() {
2042 driver
.findElement(By
.css('.feedback'))
2044 .then(function(feedback
) {
2045 expect(feedback
).toBe("Invalid mnemonic");
2051 // Incorrect word shows suggested replacement
2052 it('Shows word suggestion for incorrect word', function(done
) {
2053 driver
.findElement(By
.css('.phrase'))
2054 .sendKeys('abandon abandon abiliti');
2055 driver
.sleep(feedbackDelay
).then(function() {
2056 driver
.findElement(By
.css('.feedback'))
2058 .then(function(feedback
) {
2059 var msg
= "abiliti not in wordlist, did you mean ability?";
2060 expect(feedback
).toBe(msg
);
2066 // Github pull request 48
2067 // First four letters of word shows that word, not closest
2068 // since first four letters gives unique word in BIP39 wordlist
2069 // eg ille should show illegal, not idle
2070 it('Shows word suggestion based on first four chars', function(done
) {
2071 driver
.findElement(By
.css('.phrase'))
2073 driver
.sleep(feedbackDelay
).then(function() {
2074 driver
.findElement(By
.css('.feedback'))
2076 .then(function(feedback
) {
2077 var msg
= "ille not in wordlist, did you mean illegal?";
2078 expect(feedback
).toBe(msg
);
2084 // Incorrect BIP32 root key shows error
2085 it('Shows error for incorrect root key', function(done
) {
2086 driver
.findElement(By
.css('.root-key'))
2087 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpj');
2088 driver
.sleep(feedbackDelay
).then(function() {
2089 driver
.findElement(By
.css('.feedback'))
2091 .then(function(feedback
) {
2092 var msg
= "Invalid root key";
2093 expect(feedback
).toBe(msg
);
2099 // Derivation path not starting with m shows error
2100 it('Shows error for derivation path not starting with m', function(done
) {
2101 driver
.findElement(By
.css('#bip32-tab a'))
2103 driver
.findElement(By
.css('#bip32 .path'))
2105 driver
.findElement(By
.css('#bip32 .path'))
2107 driver
.findElement(By
.css('.phrase'))
2108 .sendKeys('abandon abandon ability');
2109 driver
.sleep(feedbackDelay
).then(function() {
2110 driver
.findElement(By
.css('.feedback'))
2112 .then(function(feedback
) {
2113 var msg
= "First character must be 'm'";
2114 expect(feedback
).toBe(msg
);
2120 // Derivation path containing invalid characters shows useful error
2121 it('Shows error for derivation path not starting with m', function(done
) {
2122 driver
.findElement(By
.css('#bip32-tab a'))
2124 driver
.findElement(By
.css('#bip32 .path'))
2126 driver
.findElement(By
.css('#bip32 .path'))
2127 .sendKeys('m/1/0wrong1/1');
2128 driver
.findElement(By
.css('.phrase'))
2129 .sendKeys('abandon abandon ability');
2130 driver
.sleep(feedbackDelay
).then(function() {
2131 driver
.findElement(By
.css('.feedback'))
2133 .then(function(feedback
) {
2134 var msg
= "Invalid characters 0wrong1 found at depth 2";
2135 expect(feedback
).toBe(msg
);
2141 // Github Issue 11: Default word length is 15
2142 // https://github.com/iancoleman/bip39/issues/11
2143 it('Sets the default word length to 15', function(done
) {
2144 driver
.findElement(By
.css('.strength'))
2145 .getAttribute("value")
2146 .then(function(strength
) {
2147 expect(strength
).toBe("15");
2152 // Github Issue 12: Generate more rows with private keys hidden
2153 // https://github.com/iancoleman/bip39/issues/12
2154 it('Sets the correct hidden column state on new rows', function(done
) {
2155 driver
.findElement(By
.css('.phrase'))
2156 .sendKeys("abandon abandon ability");
2157 driver
.sleep(generateDelay
).then(function() {
2158 driver
.findElement(By
.css('.private-key-toggle'))
2160 driver
.findElement(By
.css('.more'))
2162 driver
.sleep(generateDelay
).then(function() {
2163 driver
.findElements(By
.css('.privkey'))
2164 .then(function(els
) {
2165 expect(els
.length
).toBe(40);
2167 testColumnValuesAreInvisible(done
, "privkey");
2172 // Github Issue 19: Mnemonic is not sensitive to whitespace
2173 // https://github.com/iancoleman/bip39/issues/19
2174 it('Ignores excess whitespace in the mnemonic', function(done
) {
2175 var doublespace
= " ";
2176 var mnemonic
= "urge cat" + doublespace
+ "bid";
2177 driver
.findElement(By
.css('.phrase'))
2178 .sendKeys(mnemonic
);
2179 driver
.sleep(generateDelay
).then(function() {
2180 driver
.findElement(By
.css('.root-key'))
2181 .getAttribute("value")
2182 .then(function(seed
) {
2183 expect(seed
).toBe("xprv9s21ZrQH143K3isaZsWbKVoTtbvd34Y1ZGRugGdMeBGbM3AgBVzTH159mj1cbbtYSJtQr65w6L5xy5L9SFC7c9VJZWHxgAzpj4mun5LhrbC");
2189 // Github Issue 23: Part 1: Use correct derivation path when changing tabs
2190 // https://github.com/iancoleman/bip39/issues/23
2191 // This test was failing for default timeout of 5000ms so changed it to +10s
2192 it('Uses the correct derivation path when changing tabs', function(done
) {
2193 // 1) and 2) set the phrase
2194 driver
.findElement(By
.css('.phrase'))
2195 .sendKeys("abandon abandon ability");
2196 driver
.sleep(generateDelay
).then(function() {
2197 // 3) select bip32 tab
2198 driver
.findElement(By
.css('#bip32-tab a'))
2200 driver
.sleep(generateDelay
).then(function() {
2201 // 4) switch from bitcoin to litecoin
2202 selectNetwork("LTC - Litecoin");
2203 driver
.sleep(generateDelay
).then(function() {
2204 // 5) Check address is displayed correctly
2205 getFirstAddress(function(address
) {
2206 expect(address
).toBe("LS8MP5LZ5AdzSZveRrjm3aYVoPgnfFh5T5");
2207 // 5) Check derivation path is displayed correctly
2208 getFirstPath(function(path
) {
2209 expect(path
).toBe("m/0/0");
2216 }, generateDelay
+ 10000);
2218 // Github Issue 23 Part 2: Coin selection in derivation path
2219 // https://github.com/iancoleman/bip39/issues/23#issuecomment-238011920
2220 it('Uses the correct derivation path when changing coins', function(done
) {
2222 driver
.findElement(By
.css('.phrase'))
2223 .sendKeys("abandon abandon ability");
2224 driver
.sleep(generateDelay
).then(function() {
2225 // switch from bitcoin to clam
2226 selectNetwork("CLAM - Clams");
2227 driver
.sleep(generateDelay
).then(function() {
2228 // check derivation path is displayed correctly
2229 getFirstPath(function(path
) {
2230 expect(path
).toBe("m/44'/23'/0'/0/0");
2237 // Github Issue 26: When using a Root key derrived altcoins are incorrect
2238 // https://github.com/iancoleman/bip39/issues/26
2239 it('Uses the correct derivation for altcoins with root keys', function(done
) {
2240 // 1) 2) and 3) set the root key
2241 driver
.findElement(By
.css('.root-key'))
2242 .sendKeys("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
2243 driver
.sleep(generateDelay
).then(function() {
2244 // 4) switch from bitcoin to viacoin
2245 selectNetwork("VIA - Viacoin");
2246 driver
.sleep(generateDelay
).then(function() {
2247 // 5) ensure the derived address is correct
2248 getFirstAddress(function(address
) {
2249 expect(address
).toBe("Vq9Eq4N5SQnjqZvxtxzo7hZPW5XnyJsmXT");
2256 // Selecting a language with no existing phrase should generate a phrase in
2258 it('Generate a random phrase when language is selected and no current phrase', function(done
) {
2259 driver
.findElement(By
.css("a[href='#japanese']"))
2261 driver
.sleep(generateDelay
).then(function() {
2262 driver
.findElement(By
.css(".phrase"))
2263 .getAttribute("value").then(function(phrase
) {
2264 expect(phrase
.search(/[a-z]/)).toBe(-1);
2265 expect(phrase
.length
).toBeGreaterThan(0);
2271 // Selecting a language with existing phrase should update the phrase to use
2273 it('Updates existing phrases when the language is changed', function(done
) {
2274 driver
.findElement(By
.css(".phrase"))
2275 .sendKeys("abandon abandon ability");
2276 driver
.sleep(generateDelay
).then(function() {
2277 driver
.findElement(By
.css("a[href='#italian']"))
2279 driver
.sleep(generateDelay
).then(function() {
2280 driver
.findElement(By
.css(".phrase"))
2281 .getAttribute("value").then(function(phrase
) {
2282 // Check only the language changes, not the phrase
2283 expect(phrase
).toBe("abaco abaco abbaglio");
2284 getFirstAddress(function(address
) {
2285 // Check the address is correct
2286 expect(address
).toBe("1Dz5TgDhdki9spa6xbPFbBqv5sjMrx3xgV");
2294 // Suggested replacement for erroneous word in non-English language
2295 it('Shows word suggestion for incorrect word in non-English language', function(done
) {
2296 driver
.findElement(By
.css('.phrase'))
2297 .sendKeys('abaco abaco zbbaglio');
2298 driver
.sleep(feedbackDelay
).then(function() {
2299 driver
.findElement(By
.css('.feedback'))
2301 .then(function(feedback
) {
2302 var msg
= "zbbaglio not in wordlist, did you mean abbaglio?";
2303 expect(feedback
).toBe(msg
);
2309 // Japanese word does not break across lines.
2311 // https://github.com/bitcoin/bips/blob/master/bip-0039/bip-0039-wordlists.md#japanese
2312 it('Does not break Japanese words across lines', function(done
) {
2313 driver
.findElement(By
.css('.phrase'))
2314 .getCssValue("word-break")
2315 .then(function(value
) {
2316 expect(value
).toBe("keep-all");
2321 // Language can be specified at page load using hash value in url
2322 it('Can set the language from the url hash', function(done
) {
2323 driver
.get(url
+ "#japanese").then(function() {
2324 driver
.findElement(By
.css('.generate')).click();
2325 driver
.sleep(generateDelay
).then(function() {
2326 driver
.findElement(By
.css(".phrase"))
2327 .getAttribute("value").then(function(phrase
) {
2328 expect(phrase
.search(/[a-z]/)).toBe(-1);
2329 expect(phrase
.length
).toBeGreaterThan(0);
2336 // Entropy can be entered by the user
2337 it('Allows entropy to be entered', function(done
) {
2338 driver
.findElement(By
.css('.use-entropy'))
2340 driver
.findElement(By
.css('.entropy'))
2341 .sendKeys('00000000 00000000 00000000 00000000');
2342 driver
.sleep(generateDelay
).then(function() {
2343 driver
.findElement(By
.css(".phrase"))
2344 .getAttribute("value").then(function(phrase
) {
2345 expect(phrase
).toBe("abandon abandon ability");
2346 getFirstAddress(function(address
) {
2347 expect(address
).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug");
2354 // A warning about entropy is shown to the user, with additional information
2355 it('Shows a warning about using entropy', function(done
) {
2356 driver
.findElement(By
.css('.use-entropy'))
2358 driver
.findElement(By
.css('.entropy-container'))
2360 .then(function(containerText
) {
2361 var warning
= "mnemonic may be insecure";
2362 expect(containerText
).toContain(warning
);
2363 driver
.findElement(By
.css('#entropy-notes'))
2364 .findElement(By
.xpath("parent::*"))
2366 .then(function(notesText
) {
2367 var detail
= "flipping a fair coin, rolling a fair dice, noise measurements etc";
2368 expect(notesText
).toContain(detail
);
2374 // The types of entropy available are described to the user
2375 it('Shows the types of entropy available', function(done
) {
2376 driver
.findElement(By
.css('.entropy'))
2377 .getAttribute("placeholder")
2378 .then(function(placeholderText
) {
2387 for (var i
=0; i
<options
.length
; i
++) {
2388 var option
= options
[i
];
2389 expect(placeholderText
).toContain(option
);
2395 // The actual entropy used is shown to the user
2396 it('Shows the actual entropy used', function(done
) {
2397 driver
.findElement(By
.css('.use-entropy'))
2399 driver
.findElement(By
.css('.entropy'))
2400 .sendKeys('Not A Very Good Entropy Source At All');
2401 driver
.sleep(generateDelay
).then(function() {
2402 driver
.findElement(By
.css('.entropy-container'))
2404 .then(function(text
) {
2405 expect(text
).toMatch(/Filtered Entropy
\s
+AedEceAA
/);
2411 // Binary entropy can be entered
2412 it('Allows binary entropy to be entered', function(done
) {
2413 testEntropyType(done
, "01", "binary");
2416 // Base 6 entropy can be entered
2417 it('Allows base 6 entropy to be entered', function(done
) {
2418 testEntropyType(done
, "012345", "base 6");
2421 // Base 6 dice entropy can be entered
2422 it('Allows base 6 dice entropy to be entered', function(done
) {
2423 testEntropyType(done
, "123456", "base 6 (dice)");
2426 // Base 10 entropy can be entered
2427 it('Allows base 10 entropy to be entered', function(done
) {
2428 testEntropyType(done
, "789", "base 10");
2431 // Hexadecimal entropy can be entered
2432 it('Allows hexadecimal entropy to be entered', function(done
) {
2433 testEntropyType(done
, "abcdef", "hexadecimal");
2436 // Dice entropy value is shown as the converted base 6 value
2437 // ie 123456 is converted to 123450
2438 it('Shows dice entropy as base 6', function(done
) {
2439 driver
.findElement(By
.css('.use-entropy'))
2441 driver
.findElement(By
.css('.entropy'))
2442 .sendKeys("123456");
2443 driver
.sleep(generateDelay
).then(function() {
2444 driver
.findElement(By
.css('.entropy-container'))
2446 .then(function(text
) {
2447 expect(text
).toMatch(/Filtered Entropy
\s
+123450/);
2453 // The number of bits of entropy accumulated is shown
2454 it("Shows the number of bits of entropy for 20 bits of binary", function(done
) {
2455 testEntropyBits(done
, "0000 0000 0000 0000 0000", "20");
2457 it("Shows the number of bits of entropy for 1 bit of binary", function(done
) {
2458 testEntropyBits(done
, "0", "1");
2460 it("Shows the number of bits of entropy for 4 bits of binary", function(done
) {
2461 testEntropyBits(done
, "0000", "4");
2463 it("Shows the number of bits of entropy for 1 character of base 6 (dice)", function(done
) {
2464 // 6 in card is 0 in base 6, 0 in base 6 is 2.6 bits (rounded down to 2 bits)
2465 testEntropyBits(done
, "6", "2");
2467 it("Shows the number of bits of entropy for 1 character of base 10 with 3 bits", function(done
) {
2468 // 7 in base 10 is 111 in base 2, no leading zeros
2469 testEntropyBits(done
, "7", "3");
2471 it("Shows the number of bits of entropy for 1 character of base 10 with 4 bis", function(done
) {
2472 testEntropyBits(done
, "8", "4");
2474 it("Shows the number of bits of entropy for 1 character of hex", function(done
) {
2475 testEntropyBits(done
, "F", "4");
2477 it("Shows the number of bits of entropy for 2 characters of base 10", function(done
) {
2478 testEntropyBits(done
, "29", "6");
2480 it("Shows the number of bits of entropy for 2 characters of hex", function(done
) {
2481 testEntropyBits(done
, "0A", "8");
2483 it("Shows the number of bits of entropy for 2 characters of hex with 3 leading zeros", function(done
) {
2484 // hex is always multiple of 4 bits of entropy
2485 testEntropyBits(done
, "1A", "8");
2487 it("Shows the number of bits of entropy for 2 characters of hex with 2 leading zeros", function(done
) {
2488 testEntropyBits(done
, "2A", "8");
2490 it("Shows the number of bits of entropy for 2 characters of hex with 1 leading zero", function(done
) {
2491 testEntropyBits(done
, "4A", "8");
2493 it("Shows the number of bits of entropy for 2 characters of hex with no leading zeros", function(done
) {
2494 testEntropyBits(done
, "8A", "8");
2496 it("Shows the number of bits of entropy for 2 characters of hex starting with F", function(done
) {
2497 testEntropyBits(done
, "FA", "8");
2499 it("Shows the number of bits of entropy for 4 characters of hex with leading zeros", function(done
) {
2500 testEntropyBits(done
, "000A", "16");
2502 it("Shows the number of bits of entropy for 4 characters of base 6", function(done
) {
2503 testEntropyBits(done
, "5555", "11");
2505 it("Shows the number of bits of entropy for 4 characters of base 6 dice", function(done
) {
2506 // uses dice, so entropy is actually 0000 in base 6, which is 4 lots of
2507 // 2.58 bits, which is 10.32 bits (rounded down to 10 bits)
2508 testEntropyBits(done
, "6666", "10");
2510 it("Shows the number of bits of entropy for 4 charactes of base 10", function(done
) {
2511 // Uses base 10, which is 4 lots of 3.32 bits, which is 13.3 bits (rounded
2513 testEntropyBits(done
, "2227", "13");
2515 it("Shows the number of bits of entropy for 4 characters of hex with 2 leading zeros", function(done
) {
2516 testEntropyBits(done
, "222F", "16");
2518 it("Shows the number of bits of entropy for 4 characters of hex starting with F", function(done
) {
2519 testEntropyBits(done
, "FFFF", "16");
2521 it("Shows the number of bits of entropy for 10 characters of base 10", function(done
) {
2522 // 10 events at 3.32 bits per event
2523 testEntropyBits(done
, "0000101017", "33");
2525 it("Shows the number of bits of entropy for a full deck of cards", function(done
) {
2526 // cards are not replaced, so a full deck is not 52^52 entropy which is 296
2527 // bits, it's 52!, which is 225 bits
2528 testEntropyBits(done
, "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks", "225");
2531 it("Shows details about the entered entropy", function(done
) {
2532 testEntropyFeedback(done
,
2536 type: "hexadecimal",
2540 strength: "less than a second",
2544 it("Shows details about the entered entropy", function(done
) {
2545 testEntropyFeedback(done
,
2547 entropy: "AAAAAAAA",
2548 filtered: "AAAAAAAA",
2549 type: "hexadecimal",
2553 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2557 it("Shows details about the entered entropy", function(done
) {
2558 testEntropyFeedback(done
,
2560 entropy: "AAAAAAAA B",
2561 filtered: "AAAAAAAAB",
2562 type: "hexadecimal",
2566 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2570 it("Shows details about the entered entropy", function(done
) {
2571 testEntropyFeedback(done
,
2573 entropy: "AAAAAAAA BBBBBBBB",
2574 filtered: "AAAAAAAABBBBBBBB",
2575 type: "hexadecimal",
2579 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2583 it("Shows details about the entered entropy", function(done
) {
2584 testEntropyFeedback(done
,
2586 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC",
2587 filtered: "AAAAAAAABBBBBBBBCCCCCCCC",
2588 type: "hexadecimal",
2592 strength: "less than a second",
2596 it("Shows details about the entered entropy", function(done
) {
2597 testEntropyFeedback(done
,
2599 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDD",
2600 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDD",
2601 type: "hexadecimal",
2605 strength: "2 minutes",
2609 it("Shows details about the entered entropy", function(done
) {
2610 testEntropyFeedback(done
,
2612 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA",
2613 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDA",
2614 type: "hexadecimal",
2622 it("Shows details about the entered entropy", function(done
) {
2623 testEntropyFeedback(done
,
2625 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA EEEEEEEE",
2626 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDAEEEEEEEE",
2627 type: "hexadecimal",
2631 strength: "3 years",
2635 it("Shows details about the entered entropy", function(done
) {
2636 testEntropyFeedback(done
,
2638 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA EEEEEEEE FFFFFFFF",
2639 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDAEEEEEEEEFFFFFFFF",
2640 type: "hexadecimal",
2644 strength: "centuries",
2648 it("Shows details about the entered entropy", function(done
) {
2649 testEntropyFeedback(done
,
2656 strength: "less than a second",
2660 it("Shows details about the entered entropy", function(done
) {
2661 testEntropyFeedback(done
,
2663 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2664 type: "card (full deck)",
2668 strength: "centuries",
2672 it("Shows details about the entered entropy", function(done
) {
2673 testEntropyFeedback(done
,
2675 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks3d",
2676 type: "card (full deck, 1 duplicate: 3d)",
2680 strength: "centuries",
2684 it("Shows details about the entered entropy", function(done
) {
2685 testEntropyFeedback(done
,
2687 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqs3d4d",
2688 type: "card (2 duplicates: 3d 4d, 1 missing: KS)",
2692 strength: "centuries",
2696 it("Shows details about the entered entropy", function(done
) {
2697 testEntropyFeedback(done
,
2699 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqs3d4d5d6d",
2700 type: "card (4 duplicates: 3d 4d 5d..., 1 missing: KS)",
2704 strength: "centuries",
2708 it("Shows details about the entered entropy", function(done
) {
2709 testEntropyFeedback(done
,
2710 // Next test was throwing uncaught error in zxcvbn
2711 // Also tests 451 bits, ie Math.log2(52!)*2 = 225.58 * 2
2713 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsksac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2714 type: "card (full deck, 52 duplicates: ac 2c 3c...)",
2718 strength: "centuries",
2722 it("Shows details about the entered entropy", function(done
) {
2723 testEntropyFeedback(done
,
2724 // Case insensitivity to duplicate cards
2727 type: "card (1 duplicate: AS)",
2731 strength: "less than a second",
2735 it("Shows details about the entered entropy", function(done
) {
2736 testEntropyFeedback(done
,
2739 type: "card (1 duplicate: as)",
2743 strength: "less than a second",
2747 it("Shows details about the entered entropy", function(done
) {
2748 testEntropyFeedback(done
,
2749 // Missing cards are detected
2751 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2752 type: "card (1 missing: 9C)",
2756 strength: "centuries",
2760 it("Shows details about the entered entropy", function(done
) {
2761 testEntropyFeedback(done
,
2763 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2764 type: "card (2 missing: 9C 5D)",
2768 strength: "centuries",
2772 it("Shows details about the entered entropy", function(done
) {
2773 testEntropyFeedback(done
,
2775 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d7d8d9dtdjd kdah2h3h 5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2776 type: "card (4 missing: 9C 5D QD...)",
2780 strength: "centuries",
2784 it("Shows details about the entered entropy", function(done
) {
2785 testEntropyFeedback(done
,
2786 // More than six missing cards does not show message
2788 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d 8d9d jd kdah2h3h 5h6h7h8h9hthjhqhkh 2s3s4s5s6s7s8s9stsjsqsks",
2793 strength: "centuries",
2797 it("Shows details about the entered entropy", function(done
) {
2798 testEntropyFeedback(done
,
2799 // Multiple decks of cards increases bits per event
2804 bitsPerEvent: "4.34",
2808 it("Shows details about the entered entropy", function(done
) {
2809 testEntropyFeedback(done
,
2814 bitsPerEvent: "4.80",
2818 it("Shows details about the entered entropy", function(done
) {
2819 testEntropyFeedback(done
,
2824 bitsPerEvent: "5.01",
2828 it("Shows details about the entered entropy", function(done
) {
2829 testEntropyFeedback(done
,
2831 entropy: "3d3d3d3d",
2834 bitsPerEvent: "5.14",
2838 it("Shows details about the entered entropy", function(done
) {
2839 testEntropyFeedback(done
,
2841 entropy: "3d3d3d3d3d",
2844 bitsPerEvent: "5.22",
2848 it("Shows details about the entered entropy", function(done
) {
2849 testEntropyFeedback(done
,
2851 entropy: "3d3d3d3d3d3d",
2854 bitsPerEvent: "5.28",
2858 it("Shows details about the entered entropy", function(done
) {
2859 testEntropyFeedback(done
,
2861 entropy: "3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d",
2864 bitsPerEvent: "5.59",
2865 strength: 'less than a second - Repeats like "abcabcabc" are only slightly harder to guess than "abc"',
2870 // Entropy is truncated from the left
2871 it('Truncates entropy from the left', function(done
) {
2872 // Truncate from left means 0000 is removed from the start
2873 // which gives mnemonic 'avocado zoo zone'
2874 // not 1111 removed from the end
2875 // which gives the mnemonic 'abstract zoo zoo'
2876 var entropy
= "00000000 00000000 00000000 00000000";
2877 entropy
+= "11111111 11111111 11111111 1111"; // Missing last byte
2878 driver
.findElement(By
.css('.use-entropy'))
2880 driver
.findElement(By
.css('.entropy'))
2882 driver
.sleep(generateDelay
).then(function() {
2883 driver
.findElement(By
.css(".phrase"))
2884 .getAttribute("value").then(function(phrase
) {
2885 expect(phrase
).toBe("avocado zoo zone");
2891 // Very large entropy results in very long mnemonics
2892 it('Converts very long entropy to very long mnemonics', function(done
) {
2894 for (var i
=0; i
<33; i
++) {
2895 entropy
+= "AAAAAAAA"; // 3 words * 33 iterations = 99 words
2897 driver
.findElement(By
.css('.use-entropy'))
2899 driver
.findElement(By
.css('.entropy'))
2901 driver
.sleep(generateDelay
).then(function() {
2902 driver
.findElement(By
.css(".phrase"))
2903 .getAttribute("value").then(function(phrase
) {
2904 var wordCount
= phrase
.split(/\s+/g).length
;
2905 expect(wordCount
).toBe(99);
2911 // Is compatible with bip32jp entropy
2912 // https://bip32jp.github.io/english/index.html
2914 // Is incompatible with:
2916 it('Is compatible with bip32jp.github.io', function(done
) {
2917 var entropy
= "543210543210543210543210543210543210543210543210543210543210543210543210543210543210543210543210543";
2918 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";
2919 driver
.findElement(By
.css('.use-entropy'))
2921 driver
.findElement(By
.css('.entropy'))
2923 driver
.sleep(generateDelay
).then(function() {
2924 driver
.findElement(By
.css(".phrase"))
2925 .getAttribute("value").then(function(phrase
) {
2926 expect(phrase
).toBe(expectedPhrase
);
2932 // Blank entropy does not generate mnemonic or addresses
2933 it('Does not generate mnemonic for blank entropy', function(done
) {
2934 driver
.findElement(By
.css('.use-entropy'))
2936 driver
.findElement(By
.css('.entropy'))
2938 // check there is no mnemonic
2939 driver
.sleep(generateDelay
).then(function() {
2940 driver
.findElement(By
.css(".phrase"))
2941 .getAttribute("value").then(function(phrase
) {
2942 expect(phrase
).toBe("");
2943 // check there is no mnemonic
2944 driver
.findElements(By
.css(".address"))
2945 .then(function(addresses
) {
2946 expect(addresses
.length
).toBe(0);
2947 // Check the feedback says 'blank entropy'
2948 driver
.findElement(By
.css(".feedback"))
2950 .then(function(feedbackText
) {
2951 expect(feedbackText
).toBe("Blank entropy");
2959 // Mnemonic length can be selected even for weak entropy
2960 it('Allows selection of mnemonic length even for weak entropy', function(done
) {
2961 driver
.findElement(By
.css('.use-entropy'))
2963 driver
.executeScript(function() {
2964 $(".mnemonic-length").val("18").trigger("change");
2966 driver
.findElement(By
.css('.entropy'))
2967 .sendKeys("012345");
2968 driver
.sleep(generateDelay
).then(function() {
2969 driver
.findElement(By
.css(".phrase"))
2970 .getAttribute("value").then(function(phrase
) {
2971 var wordCount
= phrase
.split(/\s+/g).length
;
2972 expect(wordCount
).toBe(18);
2979 // https://github.com/iancoleman/bip39/issues/33
2980 // Final cards should contribute entropy
2981 it('Uses as much entropy as possible for the mnemonic', function(done
) {
2982 driver
.findElement(By
.css('.use-entropy'))
2984 driver
.findElement(By
.css('.entropy'))
2985 .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");
2986 driver
.sleep(generateDelay
).then(function() {
2988 driver
.findElement(By
.css(".phrase"))
2989 .getAttribute("value").then(function(originalPhrase
) {
2990 // Set the last 12 cards to be AS
2991 driver
.findElement(By
.css('.entropy'))
2993 driver
.findElement(By
.css('.entropy'))
2994 .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");
2995 driver
.sleep(generateDelay
).then(function() {
2997 driver
.findElement(By
.css(".phrase"))
2998 .getAttribute("value").then(function(newPhrase
) {
2999 expect(originalPhrase
).not
.toEqual(newPhrase
);
3008 // https://github.com/iancoleman/bip39/issues/35
3010 // TODO this doesn't work in selenium with firefox
3011 // see https://stackoverflow.com/q/40360223
3012 it('Shows a qr code on hover for the phrase', function(done
) {
3013 if (browser
== "firefox") {
3014 pending("Selenium + Firefox bug for mouseMove, see https://stackoverflow.com/q/40360223");
3016 // generate a random mnemonic
3017 var generateEl
= driver
.findElement(By
.css('.generate'));
3019 // toggle qr to show (hidden by default)
3020 var phraseEl
= driver
.findElement(By
.css(".phrase"));
3022 var rootKeyEl
= driver
.findElement(By
.css(".root-key"));
3023 driver
.sleep(generateDelay
).then(function() {
3024 // hover over the root key
3025 driver
.actions().mouseMove(rootKeyEl
).perform().then(function() {
3026 // check the qr code shows
3027 driver
.executeScript(function() {
3028 return $(".qr-container").find("canvas").length
> 0;
3030 .then(function(qrShowing
) {
3031 expect(qrShowing
).toBe(true);
3032 // hover away from the phrase
3033 driver
.actions().mouseMove(generateEl
).perform().then(function() {;
3034 // check the qr code hides
3035 driver
.executeScript(function() {
3036 return $(".qr-container").find("canvas").length
== 0;
3038 .then(function(qrHidden
) {
3039 expect(qrHidden
).toBe(true);
3048 // BIP44 account extendend private key is shown
3049 // github issue 37 - compatibility with electrum
3050 it('Shows the bip44 account extended private key', function(done
) {
3051 driver
.findElement(By
.css(".phrase"))
3052 .sendKeys("abandon abandon ability");
3053 driver
.sleep(generateDelay
).then(function() {
3054 driver
.findElement(By
.css("#bip44 .account-xprv"))
3055 .getAttribute("value")
3056 .then(function(xprv
) {
3057 expect(xprv
).toBe("xprv9yzrnt4zWVJUr1k2VxSPy9ettKz5PpeDMgaVG7UKedhqnw1tDkxP2UyYNhuNSumk2sLE5ctwKZs9vwjsq3e1vo9egCK6CzP87H2cVYXpfwQ");
3063 // BIP44 account extendend public key is shown
3064 // github issue 37 - compatibility with electrum
3065 it('Shows the bip44 account extended public key', function(done
) {
3066 driver
.findElement(By
.css(".phrase"))
3067 .sendKeys("abandon abandon ability");
3068 driver
.sleep(generateDelay
).then(function() {
3069 driver
.findElement(By
.css("#bip44 .account-xpub"))
3070 .getAttribute("value")
3071 .then(function(xprv
) {
3072 expect(xprv
).toBe("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
3079 // BIP32 root key can be set as an xpub
3080 it('Generates addresses from xpub as bip32 root key', function(done
) {
3081 driver
.findElement(By
.css('#bip32-tab a'))
3083 // set xpub for account 0 of bip44 for 'abandon abandon ability'
3084 driver
.findElement(By
.css("#root-key"))
3085 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
3086 driver
.sleep(generateDelay
).then(function() {
3087 // check the addresses are generated
3088 getFirstAddress(function(address
) {
3089 expect(address
).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug");
3090 // check the xprv key is not set
3091 driver
.findElement(By
.css(".extended-priv-key"))
3092 .getAttribute("value")
3093 .then(function(xprv
) {
3094 expect(xprv
).toBe("NA");
3095 // check the private key is not set
3096 driver
.findElements(By
.css(".privkey"))
3097 .then(function(els
) {
3100 .then(function(privkey
) {
3101 expect(xprv
).toBe("NA");
3111 // xpub for bip32 root key will not work with hardened derivation paths
3112 it('Shows error for hardened derivation paths with xpub root key', function(done
) {
3113 // set xpub for account 0 of bip44 for 'abandon abandon ability'
3114 driver
.findElement(By
.css("#root-key"))
3115 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
3116 driver
.sleep(feedbackDelay
).then(function() {
3117 // Check feedback is correct
3118 driver
.findElement(By
.css('.feedback'))
3120 .then(function(feedback
) {
3121 var msg
= "Hardened derivation path is invalid with xpub key";
3122 expect(feedback
).toBe(msg
);
3123 // Check no addresses are shown
3124 driver
.findElements(By
.css('.addresses tr'))
3125 .then(function(rows
) {
3126 expect(rows
.length
).toBe(0);
3134 // no root key shows feedback
3135 it('Shows feedback for no root key', function(done
) {
3136 // set xpub for account 0 of bip44 for 'abandon abandon ability'
3137 driver
.findElement(By
.css('#bip32-tab a'))
3139 driver
.sleep(feedbackDelay
).then(function() {
3140 // Check feedback is correct
3141 driver
.findElement(By
.css('.feedback'))
3143 .then(function(feedback
) {
3144 expect(feedback
).toBe("Invalid root key");
3151 // display error switching tabs while addresses are generating
3152 it('Can change details while old addresses are still being generated', function(done
) {
3153 // Set to generate 199 more addresses.
3154 // This will take a long time allowing a new set of addresses to be
3155 // generated midway through this lot.
3156 // The newly generated addresses should not include any from the old set.
3157 // Any more than 199 will show an alert which needs to be accepted.
3158 driver
.findElement(By
.css('.rows-to-add'))
3160 driver
.findElement(By
.css('.rows-to-add'))
3163 driver
.findElement(By
.css('.phrase'))
3164 .sendKeys("abandon abandon ability");
3165 driver
.sleep(generateDelay
).then(function() {
3166 // change tabs which should cancel the previous generating
3167 driver
.findElement(By
.css('.rows-to-add'))
3169 driver
.findElement(By
.css('.rows-to-add'))
3171 driver
.findElement(By
.css('#bip32-tab a'))
3173 driver
.sleep(generateDelay
).then(function() {
3174 driver
.findElements(By
.css('.index'))
3175 .then(function(els
) {
3176 // check the derivation paths have the right quantity
3177 expect(els
.length
).toBe(20);
3178 // check the derivation paths are in order
3179 testRowsAreInCorrectOrder(done
);
3183 }, generateDelay
+ 10000);
3186 // padding for binary should give length with multiple of 256
3187 // hashed entropy 1111 is length 252, so requires 4 leading zeros
3188 // prior to issue 49 it would only generate 2 leading zeros, ie missing 2
3189 it('Pads hashed entropy with leading zeros', function(done
) {
3190 driver
.findElement(By
.css('.use-entropy'))
3192 driver
.executeScript(function() {
3193 $(".mnemonic-length").val("15").trigger("change");
3195 driver
.findElement(By
.css('.entropy'))
3197 driver
.sleep(generateDelay
).then(function() {
3198 driver
.findElement(By
.css('.phrase'))
3199 .getAttribute("value")
3200 .then(function(phrase
) {
3201 expect(phrase
).toBe("avocado valid quantum cross link predict excuse edit street able flame large galaxy ginger nuclear");
3207 // Github pull request 55
3208 // https://github.com/iancoleman/bip39/pull/55
3210 it('Can set the derivation path on bip32 tab for bitcoincore', function(done
) {
3211 testClientSelect(done
, {
3213 bip32path: "m/0'/0'",
3214 useHardenedAddresses: "true",
3217 it('Can set the derivation path on bip32 tab for multibit', function(done
) {
3218 testClientSelect(done
, {
3220 bip32path: "m/0'/0",
3221 useHardenedAddresses: null,
3224 it('Can set the derivation path on bip32 tab for coinomi/ledger', function(done
) {
3225 testClientSelect(done
, {
3227 bip32path: "m/44'/0'/0'",
3228 useHardenedAddresses: null,
3233 // https://github.com/iancoleman/bip39/issues/58
3234 // bip32 derivation is correct, does not drop leading zeros
3236 // https://medium.com/@alexberegszaszi/why-do-my-bip32-wallets-disagree-6f3254cc5846
3237 it('Retains leading zeros for bip32 derivation', function(done
) {
3238 driver
.findElement(By
.css(".phrase"))
3239 .sendKeys("fruit wave dwarf banana earth journey tattoo true farm silk olive fence");
3240 driver
.findElement(By
.css(".passphrase"))
3241 .sendKeys("banana");
3242 driver
.sleep(generateDelay
).then(function() {
3243 getFirstAddress(function(address
) {
3244 // Note that bitcore generates an incorrect address
3245 // 13EuKhffWkBE2KUwcbkbELZb1MpzbimJ3Y
3246 // see the medium.com link above for more details
3247 expect(address
).toBe("17rxURoF96VhmkcEGCj5LNQkmN9HVhWb7F");
3254 // Japanese mnemonics generate incorrect bip32 seed
3255 // BIP39 seed is set from phrase
3256 it('Generates correct seed for Japanese mnemonics', function(done
) {
3257 driver
.findElement(By
.css(".phrase"))
3258 .sendKeys("あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あおぞら");
3259 driver
.findElement(By
.css(".passphrase"))
3260 .sendKeys("メートルガバヴァぱばぐゞちぢ十人十色");
3261 driver
.sleep(generateDelay
).then(function() {
3262 driver
.findElement(By
.css(".seed"))
3263 .getAttribute("value")
3264 .then(function(seed
) {
3265 expect(seed
).toBe("a262d6fb6122ecf45be09c50492b31f92e9beb7d9a845987a02cefda57a15f9c467a17872029a9e92299b5cbdf306e3a0ee620245cbd508959b6cb7ca637bd55");
3271 // BIP49 official test vectors
3272 // https://github.com/bitcoin/bips/blob/master/bip-0049.mediawiki#test-vectors
3273 it('Generates BIP49 addresses matching the official test vectors', function(done
) {
3274 driver
.findElement(By
.css('#bip49-tab a'))
3276 selectNetwork("BTC - Bitcoin Testnet");
3277 driver
.findElement(By
.css(".phrase"))
3278 .sendKeys("abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about");
3279 driver
.sleep(generateDelay
).then(function() {
3280 getFirstAddress(function(address
) {
3281 expect(address
).toBe("2Mww8dCYPUpKHofjgcXcBCEGmniw9CoaiD2");
3287 // BIP49 derivation path is shown
3288 it('Shows the bip49 derivation path', function(done
) {
3289 driver
.findElement(By
.css('#bip49-tab a'))
3291 driver
.findElement(By
.css(".phrase"))
3292 .sendKeys("abandon abandon ability");
3293 driver
.sleep(generateDelay
).then(function() {
3294 driver
.findElement(By
.css('#bip49 .path'))
3295 .getAttribute("value")
3296 .then(function(path
) {
3297 expect(path
).toBe("m/49'/0'/0'/0");
3303 // BIP49 extended private key is shown
3304 it('Shows the bip49 extended private key', function(done
) {
3305 driver
.findElement(By
.css('#bip49-tab a'))
3307 driver
.findElement(By
.css(".phrase"))
3308 .sendKeys("abandon abandon ability");
3309 driver
.sleep(generateDelay
).then(function() {
3310 driver
.findElement(By
.css('.extended-priv-key'))
3311 .getAttribute("value")
3312 .then(function(xprv
) {
3313 expect(xprv
).toBe("yprvALYB4DYRG6CzzVgzQZwwqjAA2wjBGC3iEd7KYYScpoDdmf75qMRWZWxoFcRXBJjgEXdFqJ9vDRGRLJQsrL22Su5jMbNFeM9vetaGVqy9Qy2");
3319 // BIP49 extended public key is shown
3320 it('Shows the bip49 extended public key', function(done
) {
3321 driver
.findElement(By
.css('#bip49-tab a'))
3323 driver
.findElement(By
.css(".phrase"))
3324 .sendKeys("abandon abandon ability");
3325 driver
.sleep(generateDelay
).then(function() {
3326 driver
.findElement(By
.css('.extended-pub-key'))
3327 .getAttribute("value")
3328 .then(function(xprv
) {
3329 expect(xprv
).toBe("ypub6ZXXTj5K6TmJCymTWbUxCs6tayZffemZbr2vLvrEP8kceTSENtjm7KHH6thvAKxVar9fGe8rgsPEX369zURLZ68b4f7Vexz7RuXsjQ69YDt");
3335 // BIP49 account field changes address list
3336 it('Can set the bip49 account field', function(done
) {
3337 driver
.findElement(By
.css('#bip49-tab a'))
3339 driver
.findElement(By
.css('#bip49 .account'))
3341 driver
.findElement(By
.css('#bip49 .account'))
3343 driver
.findElement(By
.css(".phrase"))
3344 .sendKeys("abandon abandon ability");
3345 driver
.sleep(generateDelay
).then(function() {
3346 getFirstAddress(function(address
) {
3347 expect(address
).toBe("381wg1GGN4rP88rNC9v7QWsiww63yLVPsn");
3353 // BIP49 change field changes address list
3354 it('Can set the bip49 change field', function(done
) {
3355 driver
.findElement(By
.css('#bip49-tab a'))
3357 driver
.findElement(By
.css('#bip49 .change'))
3359 driver
.findElement(By
.css('#bip49 .change'))
3361 driver
.findElement(By
.css(".phrase"))
3362 .sendKeys("abandon abandon ability");
3363 driver
.sleep(generateDelay
).then(function() {
3364 getFirstAddress(function(address
) {
3365 expect(address
).toBe("3PEM7MiKed5konBoN66PQhK8r3hjGhy9dT");
3371 // BIP49 account extendend private key is shown
3372 it('Shows the bip49 account extended private key', function(done
) {
3373 driver
.findElement(By
.css('#bip49-tab a'))
3375 driver
.findElement(By
.css(".phrase"))
3376 .sendKeys("abandon abandon ability");
3377 driver
.sleep(generateDelay
).then(function() {
3378 driver
.findElement(By
.css('#bip49 .account-xprv'))
3379 .getAttribute("value")
3380 .then(function(xprv
) {
3381 expect(xprv
).toBe("yprvAHtB1M5Wp675aLzFy9TJYK2mSsLkg6mcBRh5DZTR7L4EnYSmYPaL63KFA4ycg1PngW5LfkmejxzosCs17TKZMpRFKc3z5SJar6QAKaFcaZL");
3387 // BIP49 account extendend public key is shown
3388 it('Shows the bip49 account extended public key', function(done
) {
3389 driver
.findElement(By
.css('#bip49-tab a'))
3391 driver
.findElement(By
.css(".phrase"))
3392 .sendKeys("abandon abandon ability");
3393 driver
.sleep(generateDelay
).then(function() {
3394 driver
.findElement(By
.css('#bip49 .account-xpub'))
3395 .getAttribute("value")
3396 .then(function(xprv
) {
3397 expect(xprv
).toBe("ypub6WsXQrcQeTfNnq4j5AzJuSyVzuBF5ZVTYecg1ws2ffbDfLmv5vtadqdj1NgR6C6gufMpMfJpHxvb6JEQKvETVNWCRanNedfJtnTchZiJtsL");
3403 // Test selecting coin where bip49 is unavailable (eg CLAM)
3404 it('Shows an error on bip49 tab for coins without bip49', function(done
) {
3405 driver
.findElement(By
.css('#bip49-tab a'))
3407 driver
.findElement(By
.css(".phrase"))
3408 .sendKeys("abandon abandon ability");
3409 driver
.sleep(generateDelay
).then(function() {
3410 selectNetwork("CLAM - Clams");
3411 // bip49 available is hidden
3412 driver
.findElement(By
.css('#bip49 .available'))
3413 .getAttribute("class")
3414 .then(function(classes
) {
3415 expect(classes
).toContain("hidden");
3416 // bip49 unavailable is shown
3417 driver
.findElement(By
.css('#bip49 .unavailable'))
3418 .getAttribute("class")
3419 .then(function(classes
) {
3420 expect(classes
).not
.toContain("hidden");
3421 // check there are no addresses shown
3422 driver
.findElements(By
.css('.addresses tr'))
3423 .then(function(rows
) {
3424 expect(rows
.length
).toBe(0);
3425 // check the derived private key is blank
3426 driver
.findElement(By
.css('.extended-priv-key'))
3427 .getAttribute("value")
3428 .then(function(xprv
) {
3429 expect(xprv
).toBe('');
3430 // check the derived public key is blank
3431 driver
.findElement(By
.css('.extended-pub-key'))
3432 .getAttribute("value")
3433 .then(function(xpub
) {
3434 expect(xpub
).toBe('');
3445 // Cleared mnemonic and root key still allows addresses to be generated
3446 // https://github.com/iancoleman/bip39/issues/43
3447 it('Clears old root keys from memory when mnemonic is cleared', function(done
) {
3449 driver
.findElement(By
.css(".phrase"))
3450 .sendKeys("abandon abandon ability");
3451 driver
.sleep(generateDelay
).then(function() {
3452 // clear the mnemonic and root key
3453 // using selenium .clear() doesn't seem to trigger the 'input' event
3454 // so clear it using keys instead
3455 driver
.findElement(By
.css('.phrase'))
3456 .sendKeys(Key
.CONTROL
,"a");
3457 driver
.findElement(By
.css('.phrase'))
3458 .sendKeys(Key
.DELETE
);
3459 driver
.findElement(By
.css('.root-key'))
3460 .sendKeys(Key
.CONTROL
,"a");
3461 driver
.findElement(By
.css('.root-key'))
3462 .sendKeys(Key
.DELETE
);
3463 driver
.sleep(generateDelay
).then(function() {
3464 // try to generate more addresses
3465 driver
.findElement(By
.css('.more'))
3467 driver
.sleep(generateDelay
).then(function() {
3468 driver
.findElements(By
.css(".addresses tr"))
3469 .then(function(els
) {
3470 // check there are no addresses shown
3471 expect(els
.length
).toBe(0);
3480 // error trying to generate addresses from xpub with hardened derivation
3481 it('Shows error for hardened addresses with xpub root key', function(done
) {
3482 driver
.findElement(By
.css('#bip32-tab a'))
3484 driver
.executeScript(function() {
3485 $(".hardened-addresses").prop("checked", true);
3487 // set xpub for account 0 of bip44 for 'abandon abandon ability'
3488 driver
.findElement(By
.css("#root-key"))
3489 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
3490 driver
.sleep(feedbackDelay
).then(function() {
3491 // Check feedback is correct
3492 driver
.findElement(By
.css('.feedback'))
3494 .then(function(feedback
) {
3495 var msg
= "Hardened derivation path is invalid with xpub key";
3496 expect(feedback
).toBe(msg
);
3502 // Litecoin uses ltub by default, and can optionally be set to xprv
3504 // https://github.com/iancoleman/bip39/issues/96
3505 // Issue with extended keys on Litecoin
3506 it('Uses ltub by default for litecoin, but can be set to xprv', function(done
) {
3507 driver
.findElement(By
.css('.phrase'))
3508 .sendKeys("abandon abandon ability");
3509 selectNetwork("LTC - Litecoin");
3510 driver
.sleep(generateDelay
).then(function() {
3511 // check the extended key is generated correctly
3512 driver
.findElement(By
.css('.root-key'))
3513 .getAttribute("value")
3514 .then(function(rootKey
) {
3515 expect(rootKey
).toBe("Ltpv71G8qDifUiNesiPqf6h5V6eQ8ic77oxQiYtawiACjBEx3sTXNR2HGDGnHETYxESjqkMLFBkKhWVq67ey1B2MKQXannUqNy1RZVHbmrEjnEU");
3516 // set litecoin to use ltub
3517 driver
.executeScript(function() {
3518 $(".litecoin-use-ltub").prop("checked", false);
3519 $(".litecoin-use-ltub").trigger("change");
3521 driver
.sleep(generateDelay
).then(function() {
3522 driver
.findElement(By
.css('.root-key'))
3523 .getAttribute("value")
3524 .then(function(rootKey
) {
3525 expect(rootKey
).toBe("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
3534 // https://github.com/iancoleman/bip39/issues/99#issuecomment-327094159
3535 // "warn me emphatically when they have detected invalid input" to the entropy field
3536 // A warning is shown when entropy is filtered and discarded
3537 it('Warns when entropy is filtered and discarded', function(done
) {
3538 driver
.findElement(By
.css('.use-entropy'))
3540 // set entropy to have no filtered content
3541 driver
.findElement(By
.css('.entropy'))
3542 .sendKeys("00000000 00000000 00000000 00000000");
3543 driver
.sleep(generateDelay
).then(function() {
3544 // check the filter warning does not show
3545 driver
.findElement(By
.css('.entropy-container .filter-warning'))
3546 .getAttribute("class")
3547 .then(function(classes
) {
3548 expect(classes
).toContain("hidden");
3549 // set entropy to have some filtered content
3550 driver
.findElement(By
.css('.entropy'))
3551 .sendKeys("10000000 zxcvbn 00000000 00000000 00000000");
3552 driver
.sleep(entropyFeedbackDelay
).then(function() {
3553 // check the filter warning shows
3554 driver
.findElement(By
.css('.entropy-container .filter-warning'))
3555 .getAttribute("class")
3556 .then(function(classes
) {
3557 expect(classes
).not
.toContain("hidden");
3565 // Bitcoin Cash address can be set to use cashaddr format
3566 it('Can use cashaddr format for bitcoin cash addresses', function(done
) {
3567 driver
.executeScript(function() {
3568 $(".use-bch-cashaddr-addresses").prop("checked", true);
3570 driver
.findElement(By
.css('.phrase'))
3571 .sendKeys("abandon abandon ability");
3572 selectNetwork("BCH - Bitcoin Cash");
3573 driver
.sleep(generateDelay
).then(function() {
3574 getFirstAddress(function(address
) {
3575 expect(address
).toBe("bitcoincash:qzlquk7w4hkudxypl4fgv8x279r754dkvur7jpcsps");
3581 // Bitcoin Cash address can be set to use bitpay format
3582 it('Can use bitpay format for bitcoin cash addresses', function(done
) {
3583 driver
.executeScript(function() {
3584 $(".use-bch-bitpay-addresses").prop("checked", true);
3586 driver
.findElement(By
.css('.phrase'))
3587 .sendKeys("abandon abandon ability");
3588 selectNetwork("BCH - Bitcoin Cash");
3589 driver
.sleep(generateDelay
).then(function() {
3590 getFirstAddress(function(address
) {
3591 expect(address
).toBe("CZnpA9HPmvhuhLLPWJP8rNDpLUYXy1LXFk");
3597 // Bitcoin Cash address can be set to use legacy format
3598 it('Can use legacy format for bitcoin cash addresses', function(done
) {
3599 driver
.executeScript(function() {
3600 $(".use-bch-legacy-addresses").prop("checked", true);
3602 driver
.findElement(By
.css('.phrase'))
3603 .sendKeys("abandon abandon ability");
3604 selectNetwork("BCH - Bitcoin Cash");
3605 driver
.sleep(generateDelay
).then(function() {
3606 getFirstAddress(function(address
) {
3607 expect(address
).toBe("1JKvb6wKtsjNoCRxpZ4DGrbniML7z5U16A");
3613 // End of tests ported from old suit, so no more comments above each test now
3615 it('Can generate more addresses from a custom index', function(done
) {
3616 var expectedIndexes
= [
3617 0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,
3618 40,41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59
3620 driver
.findElement(By
.css('.phrase'))
3621 .sendKeys("abandon abandon ability");
3622 driver
.sleep(generateDelay
).then(function() {
3623 // Set start of next lot of rows to be from index 40
3624 // which means indexes 20-39 will not be in the table.
3625 driver
.findElement(By
.css('.more-rows-start-index'))
3627 driver
.findElement(By
.css('.more'))
3629 driver
.sleep(generateDelay
).then(function() {
3630 // Check actual indexes in the table match the expected pattern
3631 driver
.findElements(By
.css(".index"))
3632 .then(function(els
) {
3633 expect(els
.length
).toBe(expectedIndexes
.length
);
3634 var testRowAtIndex = function(i
) {
3635 if (i
>= expectedIndexes
.length
) {
3640 .then(function(actualPath
) {
3641 var noHardened
= actualPath
.replace(/'/g, "");
3642 var pathBits = noHardened.split("/")
3643 var lastBit = pathBits[pathBits.length-1];
3644 var actualIndex = parseInt(lastBit);
3645 var expectedIndex = expectedIndexes[i];
3646 expect(actualIndex).toBe(expectedIndex);
3647 testRowAtIndex(i+1);
3657 it('Can generate BIP141 addresses
with P2WPKH
-in-P2SH semanitcs
', function(done) {
3658 // Sourced from BIP49 official test specs
3659 driver.findElement(By.css('#bip141
-tab a
'))
3661 driver.findElement(By.css('.bip141
-path
'))
3663 driver.findElement(By.css('.bip141
-path
'))
3664 .sendKeys("m/49'/1'/0'/0");
3665 selectNetwork("BTC
- Bitcoin Testnet
");
3666 driver.findElement(By.css(".phrase
"))
3667 .sendKeys("abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about
");
3668 driver.sleep(generateDelay).then(function() {
3669 getFirstAddress(function(address) {
3670 expect(address).toBe("2Mww8dCYPUpKHofjgcXcBCEGmniw9CoaiD2
");
3676 it('Can generate BIP141 addresses with P2WPKH semanitcs', function(done) {
3677 // This result tested against bitcoinjs-lib test spec for segwit address
3678 // using the first private key of this mnemonic and default path m/0
3679 // https://github.com/bitcoinjs/bitcoinjs-lib/blob/9c8503cab0c6c30a95127042703bc18e8d28c76d/test/integration/addresses.js#L50
3680 // so whilst not directly comparable, substituting the private key produces
3681 // identical results between this tool and the bitcoinjs-lib test.
3682 // Private key generated is:
3683 // L3L8Nu9whawPBNLGtFqDhKut9DKKfG3CQoysupT7BimqVCZsLFNP
3684 driver.findElement(By.css('#bip141-tab a'))
3687 driver.executeScript(function() {
3688 $(".bip141
-semantics option
[selected
]").removeAttr("selected
");
3689 $(".bip141
-semantics option
").filter(function(i,e) {
3690 return $(e).html() == "P2WPKH
";
3691 }).prop("selected
", true);
3692 $(".bip141
-semantics
").trigger("change
");
3694 driver.findElement(By.css(".phrase
"))
3695 .sendKeys("abandon abandon ability
");
3696 driver.sleep(generateDelay).then(function() {
3697 getFirstAddress(function(address) {
3698 expect(address).toBe("bc1qfwu6a5a3evygrk8zvdxxvz4547lmpyx5vsfxe9
");
3704 it('Shows the entropy used by the PRNG when clicking generate', function(done) {
3705 driver.findElement(By.css('.generate')).click();
3706 driver.sleep(generateDelay).then(function() {
3707 driver.findElement(By.css('.entropy'))
3708 .getAttribute("value
")
3709 .then(function(entropy) {
3710 expect(entropy).not.toBe("");
3716 it('Shows the index of each word in the mnemonic', function(done) {
3717 driver.findElement(By.css('.phrase'))
3718 .sendKeys("abandon abandon ability
");
3719 driver.sleep(generateDelay).then(function() {
3720 driver.findElement(By.css('.use-entropy'))
3722 driver.findElement(By.css('.word-indexes'))
3724 .then(function(indexes) {
3725 expect(indexes).toBe("0, 0, 1");
3731 it('Shows the derivation path for bip84 tab', function(done) {
3732 driver.findElement(By.css('#bip84-tab a'))
3734 driver.findElement(By.css('.phrase'))
3735 .sendKeys('abandon abandon ability');
3736 driver.sleep(generateDelay).then(function() {
3737 driver.findElement(By.css('#bip84 .path'))
3738 .getAttribute("value
")
3739 .then(function(path) {
3740 expect(path).toBe("m
/84'/0'/0'/0");
3746 it('Shows the extended private key for bip84 tab', function(done) {
3747 driver.findElement(By.css('#bip84-tab a'))
3749 driver.findElement(By.css('.phrase'))
3750 .sendKeys('abandon abandon ability');
3751 driver.sleep(generateDelay).then(function() {
3752 driver.findElement(By.css('.extended-priv-key'))
3753 .getAttribute("value
")
3754 .then(function(path) {
3755 expect(path).toBe("zprvAev3RKrZ3QVKiUFCfdeMRen1BPDJgdNt1XpxiDy8acSs4kkAGTCvq7HeRYRNNpo8EtEjCFQBWavJwtCUR29y4TUCH4X5RXMcyq48uN8y9BP
");
3761 it('Shows the extended public key for bip84 tab', function(done) {
3762 driver.findElement(By.css('#bip84-tab a'))
3764 driver.findElement(By.css('.phrase'))
3765 .sendKeys('abandon abandon ability');
3766 driver.sleep(generateDelay).then(function() {
3767 driver.findElement(By.css('.extended-pub-key'))
3768 .getAttribute("value
")
3769 .then(function(path) {
3770 expect(path).toBe("zpub6suPpqPSsn3cvxKfmfBMnnijjR3o666jNkkZWcNk8wyqwZ5JozXBNuc8Gs7DB3uLwTDvGVTspVEAUQcEjKF3pZHgywVbubdTqbXTUg7usyx
");
3776 it('Changes the address list if bip84 account is changed', function(done) {
3777 driver.findElement(By.css('#bip84-tab a'))
3779 driver.findElement(By.css('#bip84 .account'))
3781 driver.findElement(By.css('.phrase'))
3782 .sendKeys('abandon abandon ability');
3783 driver.sleep(generateDelay).then(function() {
3784 getFirstAddress(function(address) {
3785 expect(address).toBe("bc1qp7vv669t2fy965jdzvqwrraana89ctd5ewc662
");
3791 it('Changes the address list if bip84 change is changed', function(done) {
3792 driver.findElement(By.css('#bip84-tab a'))
3794 driver.findElement(By.css('#bip84 .change'))
3796 driver.findElement(By.css('.phrase'))
3797 .sendKeys('abandon abandon ability');
3798 driver.sleep(generateDelay).then(function() {
3799 getFirstAddress(function(address) {
3800 expect(address).toBe("bc1qr39vj6rh06ff05m53uxq8uazehwhccswylhrs2
");
3806 it('Passes the official BIP84 test spec for rootpriv', function(done) {
3807 driver.findElement(By.css('#bip84-tab a'))
3809 driver.findElement(By.css('.phrase'))
3810 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3811 driver.sleep(generateDelay).then(function() {
3812 driver.findElement(By.css(".root
-key
"))
3813 .getAttribute("value
")
3814 .then(function(rootKey) {
3815 expect(rootKey).toBe("zprvAWgYBBk7JR8Gjrh4UJQ2uJdG1r3WNRRfURiABBE3RvMXYSrRJL62XuezvGdPvG6GFBZduosCc1YP5wixPox7zhZLfiUm8aunE96BBa4Kei5
");
3821 it('Passes the official BIP84 test spec for account 0 xprv', function(done) {
3822 driver.findElement(By.css('#bip84-tab a'))
3824 driver.findElement(By.css('.phrase'))
3825 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3826 driver.sleep(generateDelay).then(function() {
3827 driver.findElement(By.css("#bip84
.account
-xprv
"))
3828 .getAttribute("value
")
3829 .then(function(rootKey) {
3830 expect(rootKey).toBe("zprvAdG4iTXWBoARxkkzNpNh8r6Qag3irQB8PzEMkAFeTRXxHpbF9z4QgEvBRmfvqWvGp42t42nvgGpNgYSJA9iefm1yYNZKEm7z6qUWCroSQnE
");
3836 it('Passes the official BIP84 test spec for account 0 xpub', function(done) {
3837 driver.findElement(By.css('#bip84-tab a'))
3839 driver.findElement(By.css('.phrase'))
3840 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3841 driver.sleep(generateDelay).then(function() {
3842 driver.findElement(By.css("#bip84
.account
-xpub
"))
3843 .getAttribute("value
")
3844 .then(function(rootKey) {
3845 expect(rootKey).toBe("zpub6rFR7y4Q2AijBEqTUquhVz398htDFrtymD9xYYfG1m4wAcvPhXNfE3EfH1r1ADqtfSdVCToUG868RvUUkgDKf31mGDtKsAYz2oz2AGutZYs
");
3851 it('Passes the official BIP84 test spec for account 0 first address', function(done) {
3852 driver.findElement(By.css('#bip84-tab a'))
3854 driver.findElement(By.css('.phrase'))
3855 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3856 driver.sleep(generateDelay).then(function() {
3857 getFirstAddress(function(address) {
3858 expect(address).toBe("bc1qcr8te4kr609gcawutmrza0j4xv80jy8z306fyu
");
3864 it('Passes the official BIP84 test spec for account 0 first change address', function(done) {
3865 driver.findElement(By.css('#bip84-tab a'))
3867 driver.findElement(By.css('.phrase'))
3868 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3869 driver.findElement(By.css('#bip84 .change'))
3871 driver.sleep(generateDelay).then(function() {
3872 getFirstAddress(function(address) {
3873 expect(address).toBe("bc1q8c6fshw2dlwun7ekn9qwf37cu2rn755upcp6el
");
3879 it('Can display the table as csv', function(done) {
3880 var headings = "path
,address
,public key
,private key
";
3881 var row1 = "m
/44'/0'/0'/0/0,1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug
,033f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3
,L26cVSpWFkJ6aQkPkKmTzLqTdLJ923e6CzrVh9cmx21QHsoUmrEE
";
3882 var row20 = "m
/44'/0'/0'/0/19,1KhBy28XLAciXnnRvm71PvQJaETyrxGV55
,02b4b3e396434d8cdd20c03ac4aaa07387784d5d867b75987f516f5705ee68cb3a
,L4GrDrjReMsCAu5DkLXn79jSb95qR7Zfx7eshybCQZ1qL32MXJab
";
3883 driver.findElement(By.css('.phrase'))
3884 .sendKeys('abandon abandon ability');
3885 driver.sleep(generateDelay).then(function() {
3886 driver.findElement(By.css('.csv'))
3887 .getAttribute("value
")
3888 .then(function(csv) {
3889 expect(csv).toContain(headings);
3890 expect(csv).toContain(row1);
3891 expect(csv).toContain(row20);
3897 it('LeftPads ethereum keys that are less than 32 bytes', function(done) {
3898 // see https://github.com/iancoleman/bip39/issues/155
3899 selectNetwork("ETH
- Ethereum
");
3900 driver.findElement(By.css('#bip32-tab a'))
3902 driver.findElement(By.css('#bip32-path'))
3904 driver.findElement(By.css('#bip32-path'))
3905 .sendKeys("m
/44'/60'/0'");
3906 driver.findElement(By.css('.phrase'))
3907 .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');
3908 driver.sleep(generateDelay).then(function() {
3909 getFirstAddress(function(address) {
3910 expect(address).toBe("0x8943E785B4a5714FC87a3aFAad1eB1FeB602B118");
3916 it('Can encrypt private keys using BIP38', function(done) {
3917 // see https://github.com/iancoleman/bip39/issues/140
3918 driver.executeScript(function() {
3919 $(".use-bip38
").prop("checked
", true);
3921 driver.findElement(By.css('.bip38-password'))
3922 .sendKeys('bip38password');
3923 driver.findElement(By.css('.rows-to-add'))
3925 driver.findElement(By.css('.rows-to-add'))
3927 driver.findElement(By.css('.phrase'))
3928 .sendKeys('abandon abandon ability');
3929 driver.sleep(bip38delay).then(function() {
3931 getFirstRowValue(function(address) {
3932 expect(address).toBe("1NCvSdumA3ngMM9c4aqU56AM6rqXddfuXB
");
3934 getFirstRowValue(function(pubkey) {
3935 expect(pubkey).toBe("043f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3884a74447ea901729b1e73a999b7520e7cb55b4120e6432c64153ccab8a848e1
");
3937 getFirstRowValue(function(privkey) {
3938 expect(privkey).toBe("6PRNRiFnj1RoR3sXhymdCvoZCgnUHQpfupNdKkFbWJkwWQEKesWt1EDMDM
");
3944 }, bip38delay + 5000);
3946 it('Shows the checksum for the entropy', function(done) {
3947 driver.findElement(By.css('.use-entropy'))
3949 driver.findElement(By.css('.entropy'))
3950 .sendKeys("00000000000000000000000000000000");
3951 driver.sleep(generateDelay).then(function() {
3952 driver.findElement(By.css('.checksum'))
3954 .then(function(text) {
3955 expect(text).toBe("1");
3961 it('Shows the checksum for the entropy with the correct groupings', function(done) {
3962 driver.findElement(By.css('.use-entropy'))
3964 // create a checksum of 20 bits, which spans multiple words
3965 driver.findElement(By.css('.entropy'))
3966 .sendKeys("F000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
");
3967 driver.sleep(generateDelay).then(function() {
3968 driver.findElement(By.css('.checksum'))
3970 .then(function(text) {
3971 // first group is 9 bits, second group is 11
3972 expect(text).toBe("011010111 01110000110");
3978 it('Uses vprv for bitcoin testnet p2wpkh', function(done) {
3979 selectNetwork("BTC
- Bitcoin Testnet
");
3980 driver.findElement(By.css('#bip84-tab a'))
3982 driver.findElement(By.css('.phrase'))
3983 .sendKeys('abandon abandon ability');
3984 driver.sleep(generateDelay).then(function() {
3985 driver.findElement(By.css('.root-key'))
3986 .getAttribute("value
")
3987 .then(function(path) {
3988 expect(path).toBe("vprv9DMUxX4ShgxML9N2YV5CvWEebWrM9aJ5ULpbRRyzyWu6vs4BzTvbfFFrH41N5hVi7MYSfiugd765L3JmAfDM5po36Y8ouCKRDeYQwByCmS7
");
3994 it('Shows a warning if generating weak mnemonics', function(done) {
3995 driver.executeScript(function() {
3996 $(".strength option
[selected
]").removeAttr("selected
");
3997 $(".strength option
[value
=6]").prop("selected
", true);
3998 $(".strength
").trigger("change
");
4000 driver.findElement(By.css(".generate
-container
.warning
"))
4001 .getAttribute("class")
4002 .then(function(classes) {
4003 expect(classes).not.toContain("hidden
");
4008 it('Does not show a warning if generating strong mnemonics', function(done) {
4009 driver.executeScript(function() {
4010 $(".strength option
[selected
]").removeAttr("selected
");
4011 $(".strength option
[value
=12]").prop("selected
", true);
4013 driver.findElement(By.css(".generate
-container
.warning
"))
4014 .getAttribute("class")
4015 .then(function(classes) {
4016 expect(classes).toContain("hidden
");
4021 it('Shows a warning if overriding weak entropy with longer mnemonics', function(done) {
4022 driver.findElement(By.css('.use-entropy'))
4024 driver.findElement(By.css('.entropy'))
4025 .sendKeys("0123456789abcdef
"); // 6 words
4026 driver.executeScript(function() {
4027 $(".mnemonic
-length
").val("12").trigger("change
");
4029 driver.findElement(By.css(".weak
-entropy
-override
-warning
"))
4030 .getAttribute("class")
4031 .then(function(classes) {
4032 expect(classes).not.toContain("hidden
");
4037 it('Does not show a warning if entropy is stronger than mnemonic length', function(done) {
4038 driver.findElement(By.css('.use-entropy'))
4040 driver.findElement(By.css('.entropy'))
4041 .sendKeys("0123456789abcdef0123456789abcdef0123456789abcdef
"); // 18 words
4042 driver.executeScript(function() {
4043 $(".mnemonic
-length
").val("12").trigger("change
");
4045 driver.findElement(By.css(".weak
-entropy
-override
-warning
"))
4046 .getAttribute("class")
4047 .then(function(classes) {
4048 expect(classes).toContain("hidden
");
4053 it('Shows litecoin BIP49 addresses', function(done) {
4054 driver.findElement(By.css('.phrase'))
4055 .sendKeys('abandon abandon ability');
4056 selectNetwork("LTC
- Litecoin
");
4057 driver.findElement(By.css('#bip49-tab a'))
4059 // bip49 addresses are shown
4060 driver.sleep(generateDelay).then(function() {
4061 driver.findElement(By.css('#bip49 .available'))
4062 .getAttribute("class")
4063 .then(function(classes) {
4064 expect(classes).not.toContain("hidden
");
4065 // check first address
4066 getFirstAddress(function(address) {
4067 expect(address).toBe("MFwLPhsXoBuSLL8cLmW9uK6tChkzduV8qN
");
4074 it('Shows Groestlcoin BIP49 addresses', function(done) {
4075 driver.findElement(By.css('.phrase'))
4076 .sendKeys('abandon abandon ability');
4077 selectNetwork("GRS
- Groestlcoin
");
4078 driver.findElement(By.css('#bip49-tab a'))
4080 // bip49 addresses are shown
4081 driver.sleep(generateDelay).then(function() {
4082 driver.findElement(By.css('#bip49 .available'))
4083 .getAttribute("class")
4084 .then(function(classes) {
4085 expect(classes).not.toContain("hidden
");
4086 // check first address
4087 getFirstAddress(function(address) {
4088 expect(address).toBe("3HXSCZwCypLyixMsF4Z1sN49noJtrm8gnX
");
4095 it('Can use root keys to generate segwit table rows', function(done) {
4096 // segwit uses ypub / zpub instead of xpub but the root key should still
4097 // be valid regardless of the encoding used to import that key.
4098 // Maybe this breaks the reason for the different extended key prefixes, but
4099 // since the parsed root key is used behind the scenes anyhow this should be
4101 driver.findElement(By.css('#root-key'))
4102 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
4103 driver.findElement(By.css('#bip49-tab a'))
4105 // bip49 addresses are shown
4106 driver.sleep(generateDelay).then(function() {
4107 getFirstAddress(function(address) {
4108 expect(address).toBe("3QG2Y9AA4xZ846gKHZqNf7mvVKbLqMKxr2
");