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
) {
77 var phrase
= params
.phrase
|| 'abandon abandon ability';
78 driver
.findElement(By
.css('.phrase'))
80 selectNetwork(params
.selectText
);
81 driver
.sleep(generateDelay
).then(function() {
82 getFirstAddress(function(address
) {
83 expect(address
).toBe(params
.firstAddress
);
89 function getFirstRowValue(handler
, selector
) {
90 driver
.findElements(By
.css(selector
))
97 function getFirstAddress(handler
) {
98 getFirstRowValue(handler
, ".address");
101 function getFirstPath(handler
) {
102 getFirstRowValue(handler
, ".index");
105 function testColumnValuesAreInvisible(done
, columnClassName
) {
106 var selector
= "." + columnClassName
+ " span";
107 driver
.findElements(By
.css(selector
))
108 .then(function(els
) {
109 els
[0].getAttribute("class")
110 .then(function(classes
) {
111 expect(classes
).toContain("invisible");
117 function testRowsAreInCorrectOrder(done
) {
118 driver
.findElements(By
.css('.index'))
119 .then(function(els
) {
120 var testRowAtIndex = function(i
) {
121 if (i
>= els
.length
) {
126 .then(function(actualPath
) {
127 var noHardened
= actualPath
.replace(/'/g, "");
128 var pathBits = noHardened.split("/")
129 var lastBit = pathBits[pathBits.length-1];
130 var actualIndex = parseInt(lastBit);
131 expect(actualIndex).toBe(i);
140 function selectNetwork(name) {
141 driver.executeScript(function() {
142 var selectText = arguments[0];
143 $(".network option[selected]").removeAttr("selected");
144 $(".network option").filter(function(i,e) {
145 return $(e).html() == selectText;
146 }).prop("selected", true);
147 $(".network").trigger("change");
151 function testEntropyType(done, entropyText, entropyTypeUnsafe) {
152 // entropy type is compiled into regexp so needs escaping
153 // see https://stackoverflow.com/a/2593661
154 var entropyType = (entropyTypeUnsafe+'').replace(/[.?*+^$[\]\\(){}|-]/g, "\\$&");
155 driver.findElement(By.css('.use-entropy
'))
157 driver.findElement(By.css('.entropy
'))
158 .sendKeys(entropyText);
159 driver.sleep(generateDelay).then(function() {
160 driver.findElement(By.css('.entropy
-container
'))
162 .then(function(text) {
163 var re = new RegExp("Entropy Type\\s+" + entropyType);
164 expect(text).toMatch(re);
170 function testEntropyBits(done, entropyText, entropyBits) {
171 driver.findElement(By.css('.use-entropy
'))
173 driver.findElement(By.css('.entropy
'))
174 .sendKeys(entropyText);
175 driver.sleep(generateDelay).then(function() {
176 driver.findElement(By.css('.entropy
-container
'))
178 .then(function(text) {
179 var re = new RegExp("Total Bits\\s+" + entropyBits);
180 expect(text).toMatch(re);
186 function testEntropyFeedback(done, entropyDetail) {
187 // entropy type is compiled into regexp so needs escaping
188 // see https://stackoverflow.com/a/2593661
189 if ("type" in entropyDetail) {
190 entropyDetail.type = (entropyDetail.type+'').replace(/[.?*+^$[\]\\(){}|-]/g, "\\$&");
192 driver.findElement(By.css('.use-entropy
'))
194 driver.findElement(By.css('.entropy
'))
195 .sendKeys(entropyDetail.entropy);
196 driver.sleep(entropyFeedbackDelay).then(function() {
197 driver.findElement(By.css('.entropy
-container
'))
199 .then(function(text) {
200 driver.findElement(By.css('.phrase
'))
201 .getAttribute("value")
202 .then(function(phrase) {
203 if ("filtered" in entropyDetail) {
204 var key = "Filtered Entropy";
205 var value = entropyDetail.filtered;
206 var reText = key + "\\s+" + value;
207 var re = new RegExp(reText);
208 expect(text).toMatch(re);
210 if ("type" in entropyDetail) {
211 var key = "Entropy Type";
212 var value = entropyDetail.type;
213 var reText = key + "\\s+" + value;
214 var re = new RegExp(reText);
215 expect(text).toMatch(re);
217 if ("events" in entropyDetail) {
218 var key = "Event Count";
219 var value = entropyDetail.events;
220 var reText = key + "\\s+" + value;
221 var re = new RegExp(reText);
222 expect(text).toMatch(re);
224 if ("bits" in entropyDetail) {
225 var key = "Total Bits";
226 var value = entropyDetail.bits;
227 var reText = key + "\\s+" + value;
228 var re = new RegExp(reText);
229 expect(text).toMatch(re);
231 if ("bitsPerEvent" in entropyDetail) {
232 var key = "Bits Per Event";
233 var value = entropyDetail.bitsPerEvent;
234 var reText = key + "\\s+" + value;
235 var re = new RegExp(reText);
236 expect(text).toMatch(re);
238 if ("words" in entropyDetail) {
239 var actualWords = phrase.split(/\s+/)
240 .filter(function(w) { return w.length > 0 })
242 expect(actualWords).toBe(entropyDetail.words);
244 if ("strength" in entropyDetail) {
245 var key = "Time To Crack";
246 var value = entropyDetail.strength;
247 var reText = key + "\\s+" + value;
248 var re = new RegExp(reText);
249 expect(text).toMatch(re);
257 function testClientSelect(done, params) {
258 // set mnemonic and select bip32 tab
259 driver.findElement(By.css('#bip32
-tab a
'))
261 driver.findElement(By.css('.phrase
'))
262 .sendKeys("abandon abandon ability");
263 driver.sleep(generateDelay).then(function() {
265 // set bip32 client to bitcoin core
266 driver.executeScript(function() {
267 $("#bip32-client").val(arguments[0]).trigger("change");
268 }, params.selectValue);
269 driver.sleep(generateDelay).then(function() {
270 // check the derivation path is correct
271 driver.findElement(By.css("#bip32-path"))
272 .getAttribute("value")
273 .then(function(path) {
274 expect(path).toBe(params.bip32path);
275 // check hardened addresses is selected
276 driver.findElement(By.css(".hardened-addresses"))
277 .getAttribute("checked")
278 .then(function(isChecked) {
279 expect(isChecked).toBe(params.useHardenedAddresses);
280 // check input is readonly
281 driver.findElement(By.css("#bip32-path"))
282 .getAttribute("readonly")
283 .then(function(isReadonly) {
284 expect(isReadonly).toBe("true");
295 describe('BIP39 Tool Tests
', function() {
297 beforeEach(function(done) {
298 driver = newDriver();
299 driver.get(url).then(done);
302 // Close the website after each test is run (so that it is opened fresh each time)
303 afterEach(function(done) {
304 driver.quit().then(done);
309 // Page initially loads with blank phrase
310 it('Should load the page
', function(done) {
311 driver.findElement(By.css('.phrase
'))
312 .getAttribute('value
').then(function(value) {
313 expect(value).toBe('');
319 it('Should have text on the page
', function(done) {
320 driver.findElement(By.css('body
'))
322 .then(function(text) {
323 var textToFind = "You can enter an existing BIP39 mnemonic";
324 expect(text).toContain(textToFind);
329 // Entering mnemonic generates addresses
330 it('Should have a list
of addresses
', function(done) {
331 driver.findElement(By.css('.phrase
'))
332 .sendKeys('abandon abandon ability
');
333 driver.sleep(generateDelay).then(function() {
334 driver.findElements(By.css('.address
'))
335 .then(function(els) {
336 expect(els.length).toBe(20);
342 // Generate button generates random mnemonic
343 it('Should be able to generate a random mnemonic
', function(done) {
344 // initial phrase is blank
345 driver.findElement(By.css('.phrase
'))
346 .getAttribute("value")
347 .then(function(phrase) {
348 expect(phrase.length).toBe(0);
350 driver.findElement(By.css('.generate
')).click();
351 driver.sleep(generateDelay).then(function() {
352 // new phrase is not blank
353 driver.findElement(By.css('.phrase
'))
354 .getAttribute("value")
355 .then(function(phrase) {
356 expect(phrase.length).toBeGreaterThan(0);
363 // Mnemonic length can be customized
364 it('Should allow custom length mnemonics
', function(done) {
366 driver.executeScript(function() {
367 $(".strength option[selected]").removeAttr("selected");
368 $(".strength option[value=6]").prop("selected", true);
370 driver.findElement(By.css('.generate
')).click();
371 driver.sleep(generateDelay).then(function() {
372 driver.findElement(By.css('.phrase
'))
373 .getAttribute("value")
374 .then(function(phrase) {
375 var words = phrase.split(" ");
376 expect(words.length).toBe(6);
382 // Passphrase can be set
383 it('Allows a passphrase to be
set', function(done) {
384 driver.findElement(By.css('.phrase
'))
385 .sendKeys('abandon abandon ability
');
386 driver.findElement(By.css('.passphrase
'))
387 .sendKeys('secure_passphrase
');
388 driver.sleep(generateDelay).then(function() {
389 getFirstAddress(function(address) {
390 expect(address).toBe("15pJzUWPGzR7avffV9nY5by4PSgSKG9rba");
396 // Network can be set to networks other than bitcoin
397 it('Allows selection
of bitcoin testnet
', function(done) {
399 selectText: "BTC - Bitcoin Testnet",
400 firstAddress: "mucaU5iiDaJDb69BHLeDv8JFfGiyg2nJKi",
402 testNetwork(done, params);
404 it('Allows selection
of litecoin
', function(done) {
406 selectText: "LTC - Litecoin",
407 firstAddress: "LQ4XU8RX2ULPmPq9FcUHdVmPVchP9nwXdn",
409 testNetwork(done, params);
411 it('Allows selection
of litecoin testnet
', function(done) {
413 selectText: "LTCt - Litecoin Testnet",
414 firstAddress: "mucaU5iiDaJDb69BHLeDv8JFfGiyg2nJKi",
416 testNetwork(done, params);
418 it('Allows selection
of ripple
', function(done) {
420 selectText: "XRP - Ripple",
421 firstAddress: "rLTFnqbmCVPGx6VfaygdtuKWJgcN4v1zRS",
422 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",
424 testNetwork(done, params);
426 it('Allows selection
of dogecoin
', function(done) {
428 selectText: "DOGE - Dogecoin",
429 firstAddress: "DPQH2AtuzkVSG6ovjKk4jbUmZ6iXLpgbJA",
431 testNetwork(done, params);
433 it('Allows selection
of dogecoin testnet
', function(done) {
435 selectText: "DOGEt - Dogecoin testnet",
436 firstAddress: "niHnSJKHdwDyDxRMLBJrtNqpvHEsAFWe6B",
438 testNetwork(done, params);
440 it('Allows selection
of denarius
', function(done) {
442 selectText: "DNR - Denarius",
443 firstAddress: "DFdFMVUMzU9xX88EywXvAGwjiwpxyh9vKb",
445 testNetwork(done, params);
447 it('Allows selection
of shadowcash
', function(done) {
449 selectText: "SDC - ShadowCash",
450 firstAddress: "SiSZtfYAXEFvMm3XM8hmtkGDyViRwErtCG",
452 testNetwork(done, params);
454 it('Allows selection
of shadowcash testnet
', function(done) {
456 selectText: "SDC - ShadowCash Testnet",
457 firstAddress: "tM2EDpVKaTiEg2NZg3yKg8eqjLr55BErHe",
459 testNetwork(done, params);
461 it('Allows selection
of viacoin
', function(done) {
463 selectText: "VIA - Viacoin",
464 firstAddress: "Vq9Eq4N5SQnjqZvxtxzo7hZPW5XnyJsmXT",
466 testNetwork(done, params);
468 it('Allows selection
of viacoin testnet
', function(done) {
470 selectText: "VIA - Viacoin Testnet",
471 firstAddress: "tM2EDpVKaTiEg2NZg3yKg8eqjLr55BErHe",
473 testNetwork(done, params);
475 it('Allows selection
of jumbucks
', function(done) {
477 selectText: "JBS - Jumbucks",
478 firstAddress: "JLEXccwDXADK4RxBPkRez7mqsHVoJBEUew",
480 testNetwork(done, params);
482 it('Allows selection
of clam
', function(done) {
484 selectText: "CLAM - Clams",
485 firstAddress: "xCp4sakjVx4pUAZ6cBCtuin8Ddb6U1sk9y",
487 testNetwork(done, params);
489 it('Allows selection
of crown
', function(done) {
491 selectText: "CRW - Crown (Legacy)",
492 firstAddress: "18pWSwSUAQdiwMHUfFZB1fM2xue9X1FqE5",
494 testNetwork(done, params);
496 it('Allows selection
of crown
', function(done) {
498 selectText: "CRW - Crown",
499 firstAddress: "CRWKnVmVhvH1KWTYe6sq8xV4dFGcFpBEEkPQ",
501 testNetwork(done, params);
503 it('Allows selection
of dash
', function(done) {
505 selectText: "DASH - Dash",
506 firstAddress: "XdbhtMuGsPSkE6bPdNTHoFSszQKmK4S5LT",
508 testNetwork(done, params);
510 it('Allows selection
of dash testnet
', function(done) {
512 selectText: "DASH - Dash Testnet",
513 firstAddress: "yaR52EN4oojdJfBgzWJTymC4uuCLPT29Gw",
515 testNetwork(done, params);
517 it('Allows selection
of game
', function(done) {
519 selectText: "GAME - GameCredits",
520 firstAddress: "GSMY9bAp36cMR4zyT4uGVS7GFjpdXbao5Q",
522 testNetwork(done, params);
524 it('Allows selection
of komodo
', function(done) {
526 selectText: "KMD - Komodo",
527 firstAddress: "RMPPzJwAjPVZZAwJvXivHJGGjdCx6WBD2t",
529 testNetwork(done, params);
531 it('Allows selection
of namecoin
', function(done) {
533 selectText: "NMC - Namecoin",
534 firstAddress: "Mw2vK2Bvex1yYtYF6sfbEg2YGoUc98YUD2",
536 testNetwork(done, params);
538 it('Allows selection
of onixcoin
', function(done) {
540 selectText: "ONX - Onixcoin",
541 firstAddress: "XGwMqddeKjT3ddgX73QokjVbCL3aK6Yxfk",
543 testNetwork(done, params);
545 it('Allows selection
of lkrcoin
', function(done) {
547 selectText: "LKR - Lkrcoin",
548 firstAddress: "LfbT296e7AEEnn4bYDbL535Nd8P9g98CdJ",
550 testNetwork(done, params);
552 it('Allows selection
of bolivarcoin
', function(done) {
554 selectText: "BOLI - Bolivarcoin",
555 firstAddress: "bbKzCAUR7hZ3nqfffy7VgrSz8LmAP3S5mK",
557 testNetwork(done, params);
559 it('Allows selection
of peercoin
', function(done) {
561 selectText: "PPC - Peercoin",
562 firstAddress: "PVAiioTaK2eDHSEo3tppT9AVdBYqxRTBAm",
564 testNetwork(done, params);
566 it('Allows selection
of ethereum
', function(done) {
568 selectText: "ETH - Ethereum",
569 firstAddress: "0xe5815d5902Ad612d49283DEdEc02100Bd44C2772",
571 testNetwork(done, params);
572 // TODO test private key and public key
574 it('Allows selection
of slimcoin
', function(done) {
576 selectText: "SLM - Slimcoin",
577 firstAddress: "SNzPi1CafHFm3WWjRo43aMgiaEEj3ogjww",
579 testNetwork(done, params);
581 it('Allows selection
of slimcoin testnet
', function(done) {
583 selectText: "SLM - Slimcoin Testnet",
584 firstAddress: "n3nMgWufTek5QQAr6uwMhg5xbzj8xqc4Dq",
586 testNetwork(done, params);
588 it('Allows selection
of bitcoin cash
', function(done) {
590 selectText: "BCH - Bitcoin Cash",
591 firstAddress: "bitcoincash:qzlquk7w4hkudxypl4fgv8x279r754dkvur7jpcsps",
593 testNetwork(done, params);
596 it('Allows selection
of simpleledger(SLP
)', function(done) {
598 selectText: "SLP - Simple Ledger Protocol",
599 firstAddress: "simpleledger:qrtffz6ajfsn74gpur7y3epjquz42pvww5acewqmre",
601 testNetwork(done, params);
604 it('Allows selection
of myriadcoin
', function(done) {
606 selectText: "XMY - Myriadcoin",
607 firstAddress: "MJEswvRR46wh9BoiVj9DzKYMBkCramhoBV",
609 testNetwork(done, params);
611 it('Allows selection
of pivx
', function(done) {
613 selectText: "PIVX - PIVX",
614 firstAddress: "DBxgT7faCuno7jmtKuu6KWCiwqsVPqh1tS",
616 testNetwork(done, params);
618 it('Allows selection
of pivx testnet
', function(done) {
620 selectText: "PIVX - PIVX Testnet",
621 firstAddress: "yB5U384n6dGkVE3by5y9VdvHHPwPg68fQj",
623 testNetwork(done, params);
625 it('Allows selection
of maza
', function(done) {
627 selectText: "MAZA - Maza",
628 firstAddress: "MGW4Bmi2NEm4PxSjgeFwhP9vg18JHoRnfw",
630 testNetwork(done, params);
632 it('Allows selection
of fujicoin
', function(done) {
634 selectText: "FJC - Fujicoin",
635 firstAddress: "FgiaLpG7C99DyR4WnPxXedRVHXSfKzUDhF",
637 testNetwork(done, params);
639 it('Allows selection
of nubits
', function(done) {
641 selectText: "USNBT - NuBits",
642 firstAddress: "BLxkabXuZSJSdesLD7KxZdqovd4YwyBTU6",
644 testNetwork(done, params);
646 it('Allows selection
of bitcoin gold
', function(done) {
648 selectText: "BTG - Bitcoin Gold",
649 firstAddress: "GdDqug4WUsn5syNbSTHatNn4XnuwZtzedx",
651 testNetwork(done, params);
653 it('Allows selection
of monacoin
', function(done) {
655 selectText: "MONA - Monacoin",
656 firstAddress: "MKMiMr7MyjDKjJbCBzgF6u4ByqTS4NkRB1",
658 testNetwork(done, params);
660 it('Allows selection
of AXE
', function(done) {
662 selectText: "AXE - Axe",
663 firstAddress: "PScwtLUyPiGrqtKXrHF37DGETLXLZdw4up",
665 testNetwork(done, params);
667 it('Allows selection
of BlackCoin
', function(done) {
669 selectText: "BLK - BlackCoin",
670 firstAddress: "B5MznAKwj7uQ42vDz3w4onhBXPcqhTwJ9z",
672 testNetwork(done, params);
674 it('Allows selection
of Neblio
', function(done) {
676 selectText: "NEBL - Neblio",
677 firstAddress: "NefkeEEvhusbHMmTRrxx7H9wFnUXd8qQsE",
679 testNetwork(done, params);
681 it('Allows selection
of Beetlecoin
', function(done) {
683 selectText: "BEET - Beetlecoin",
684 firstAddress: "BVmtbEsGrjpknprmpHFq26z4kYHJUFHE71",
686 testNetwork(done, params);
688 it('Allows selection
of Adcoin
', function(done) {
690 selectText: "ACC - Adcoin",
691 firstAddress: "AcEDM6V5sF4kFHC76MJjjfProtS5Sw2qcd",
693 testNetwork(done, params);
695 it('Allows selection
of Asiacoin
', function(done) {
697 selectText: "AC - Asiacoin",
698 firstAddress: "ALupuEEz7kJjQTAvmtcBMBVuEjPa7GqZzE",
700 testNetwork(done, params);
702 it('Allows selection
of Auroracoin
', function(done) {
704 selectText: "AUR - Auroracoin",
705 firstAddress: "ANuraS6F4Jpi413FEnavjYkKYJJRHkgYCm",
707 testNetwork(done, params);
709 it('Allows selection
of Bata
', function(done) {
711 selectText: "BTA - Bata",
712 firstAddress: "BGxBdNeYPtF3GCuTtZBPQdFxCkdBYSF3fj",
714 testNetwork(done, params);
716 it('Allows selection
of Belacoin
', function(done) {
718 selectText: "BELA - Belacoin",
719 firstAddress: "BEeetqpNffdzeknSpNmQp5KAFh2KK1Qx7S",
721 testNetwork(done, params);
723 it('Allows selection
of Bitcoin Atom
', function(done) {
725 selectText: "BCA - Bitcoin Atom",
726 firstAddress: "AMy6qMbJeC4zsGRL6iWszmeCdQH65fgfih",
728 testNetwork(done, params);
730 it('Allows selection
of Bitcoinplus
', function(done) {
732 selectText: "XBC - Bitcoinplus",
733 firstAddress: "B7FSynZoDbEwTCSgsXq9nJ5ue8owYLVL8r",
735 testNetwork(done, params);
737 it('Allows selection
of Bitcoin Private
', function(done) {
739 selectText: "BTCP - Bitcoin Private",
740 firstAddress: "b1M3PbiXXyN6Hdivdw5rJv5VKpLjPzhm4jM",
742 testNetwork(done, params);
744 it('Allows selection
of Bitcoinz
', function(done) {
746 selectText: "BTCZ - Bitcoinz",
747 firstAddress: "t1X2YQoxs8cYRo2oaBYgVEwW5QNjCC59NYc",
749 testNetwork(done, params);
751 it('Allows selection
of BitCloud
', function(done) {
753 selectText: "BTDX - BitCloud",
754 firstAddress: "BHbWitXCNgTf1BhsRDNMP186EeibuzmrBi",
756 testNetwork(done, params);
758 it('Allows selection
of Bitcore
', function(done) {
760 selectText: "BTX - Bitcore",
761 firstAddress: "2Rgp5Znhpy34TK4QmPkfCiYs9r4KovfTH9",
763 testNetwork(done, params);
765 it('Allows selection
of Bitsend
', function(done) {
767 selectText: "BSD - Bitsend",
768 firstAddress: "iBPk7LYjDun3EPk7CRR8UUmnPoceVc1bp2",
770 testNetwork(done, params);
772 it('Allows selection
of Britcoin
', function(done) {
774 selectText: "BRIT - Britcoin",
775 firstAddress: "B6Aue4J2XLs1f1dtD4H1SHYFfh4XrmEbrw",
777 testNetwork(done, params);
779 it('Allows selection
of Canadaecoin
', function(done) {
781 selectText: "CDN - Canadaecoin",
782 firstAddress: "CanAyCfd5Rj2CQVfaoAmvDUZunPM5W1AEQ",
784 testNetwork(done, params);
786 it('Allows selection
of Cannacoin
', function(done) {
788 selectText: "CCN - Cannacoin",
789 firstAddress: "CYjW8xWB43g6krLJTmmrPk1PonoQX7h9Qd",
791 testNetwork(done, params);
793 it('Allows selection
of Clubcoin
', function(done) {
795 selectText: "CLUB - Clubcoin",
796 firstAddress: "CHMDEXN4sihpSVX4GyAa2hZ62shnby7uyN",
798 testNetwork(done, params);
800 it('Allows selection
of Compcoin
', function(done) {
802 selectText: "CMP - Compcoin",
803 firstAddress: "CLshtw3zhxkseBJS46UF12v3AFy9Dx7JVv",
805 testNetwork(done, params);
807 it('Allows selection
of Crave
', function(done) {
809 selectText: "CRAVE - Crave",
810 firstAddress: "VCYJeti6uKMNBFKCL7eP96UwuFWYHM7c85",
812 testNetwork(done, params);
814 it('Allows selection
of Defcoin
', function(done) {
816 selectText: "DFC - Defcoin",
817 firstAddress: "D8swcgyaaFUrXZU3ATwbgy16buCpWqbG1M",
819 testNetwork(done, params);
821 it('Allows selection
of Diamond
', function(done) {
823 selectText: "DMD - Diamond",
824 firstAddress: "dJnrVbLL9UPjdaVRz2C8VpqHZknqAqjLek",
826 testNetwork(done, params);
828 it('Allows selection
of Digibyte
', function(done) {
830 selectText: "DGB - Digibyte",
831 firstAddress: "D85Rp9jwLtMdmP6wGjTiqHBdVQLST3YCEq",
833 testNetwork(done, params);
835 it('Allows selection
of Digitalcoin
', function(done) {
837 selectText: "DGC - Digitalcoin",
838 firstAddress: "DKw4UGKEAZWweDNEbBFNQx4EM8x1mpUdia",
840 testNetwork(done, params);
842 it('Allows selection
of Ecoin
', function(done) {
844 selectText: "ECN - Ecoin",
845 firstAddress: "e6WFPLG5gcXyF7cESFteH1hE2XSmowW5yB",
847 testNetwork(done, params);
849 it('Allows selection
of Edrcoin
', function(done) {
851 selectText: "EDRC - Edrcoin",
852 firstAddress: "eh1nUJsvgKPFv6ebMBfcwJ299GMCpjeZUG",
854 testNetwork(done, params);
856 it('Allows selection
of Egulden
', function(done) {
858 selectText: "EFL - Egulden",
859 firstAddress: "Lg66yt55R7edRM58cDhKzXik2kFme3viX7",
861 testNetwork(done, params);
863 it('Allows selection
of Einsteinium
', function(done) {
865 selectText: "EMC2 - Einsteinium",
866 firstAddress: "EVAABm9hXKHk2MpVMbwNakRubFnNha5m8m",
868 testNetwork(done, params);
870 it('Allows selection
of Europecoin
', function(done) {
872 selectText: "ERC - Europecoin",
873 firstAddress: "ESA2YwPYntAoaPrE8Fm5qkKRtkcwLcwD6R",
875 testNetwork(done, params);
877 it('Allows selection
of Exclusivecoin
', function(done) {
879 selectText: "EXCL - Exclusivecoin",
880 firstAddress: "EbUa6m8UZW6nTxsYZD2FsDjkadKbp5M6JT",
882 testNetwork(done, params);
884 it('Allows selection
of Feathercoin
', function(done) {
886 selectText: "FTC - Feathercoin",
887 firstAddress: "6gDdjAMoSgQaW8UhqK3oboHs6ftGAroKkM",
889 testNetwork(done, params);
891 it('Allows selection
of Firstcoin
', function(done) {
893 selectText: "FRST - Firstcoin",
894 firstAddress: "FJN9GzfMm7Q8R4DJwK1H9F6A1GTghvFiMJ",
896 testNetwork(done, params);
898 it('Allows selection
of Flashcoin
', function(done) {
900 selectText: "FLASH - Flashcoin",
901 firstAddress: "UWfpf5LfMmLxZYooEb2EyvWhZ8NG7EZDRt",
903 testNetwork(done, params);
905 it('Allows selection
of GCRCoin
', function(done) {
907 selectText: "GCR - GCRCoin",
908 firstAddress: "GJjF5cLwyXLacpuvXAVksxGxKvHDjx58d6",
910 testNetwork(done, params);
912 it('Allows selection
of Gobyte
', function(done) {
914 selectText: "GBX - Gobyte",
915 firstAddress: "GS813Ys2brkmvSUw1rUqGPm2HqQVDHJRyA",
917 testNetwork(done, params);
919 it('Allows selection
of Gridcoin
', function(done) {
921 selectText: "GRC - Gridcoin",
922 firstAddress: "SGrWbBPvobgqKRF8td1Kdc9vbRY7MJ78Y9",
924 testNetwork(done, params);
926 it('Allows selection
of Gulden
', function(done) {
928 selectText: "NLG - Gulden",
929 firstAddress: "GcDP7cNEc33MPPdTFNJ8pZc6VMZJ2CbKxY",
931 testNetwork(done, params);
933 it('Allows selection
of Helleniccoin
', function(done) {
935 selectText: "HNC - Helleniccoin",
936 firstAddress: "LbHEKe5H72zp9G1fuWNiiNePTUfJb88915",
938 testNetwork(done, params);
940 it('Allows selection
of Hempcoin
', function(done) {
942 selectText: "THC - Hempcoin",
943 firstAddress: "H8sdWbZyJV4gyXyHtLXDaNnAuUDhK5mfTV",
945 testNetwork(done, params);
947 it('Allows selection
of Insane
', function(done) {
949 selectText: "INSN - Insane",
950 firstAddress: "iMPqEJMiXWuxC9U2NVinCCMr4t72h58EWx",
952 testNetwork(done, params);
954 it('Allows selection
of Iop
', function(done) {
956 selectText: "IOP - Iop",
957 firstAddress: "pGKQmcaPf95Ur5o6oHK4qdiZ52p1yaTvq1",
959 testNetwork(done, params);
961 it('Allows selection
of Ixcoin
', function(done) {
963 selectText: "IXC - Ixcoin",
964 firstAddress: "xgE9bTZ6YypT3E6ByzkTt31Hq68E9BqywH",
966 testNetwork(done, params);
968 it('Allows selection
of Kobocoin
', function(done) {
970 selectText: "KOBO - Kobocoin",
971 firstAddress: "FTVoNJETXDAM8x7MnmdE8RwWndSr9PQWhy",
973 testNetwork(done, params);
975 it('Allows selection
of Landcoin
', function(done) {
977 selectText: "LDCN - Landcoin",
978 firstAddress: "LLvLwNjG1aJcn1RS4W4GJUbv8fNaRATG7c",
980 testNetwork(done, params);
982 it('Allows selection
of Library Credits
', function(done) {
984 selectText: "LBC - Library Credits",
985 firstAddress: "bQJEQrHDJyHdqycB32uysh1SWn8Ln8LMdg",
987 testNetwork(done, params);
989 it('Allows selection
of Linx
', function(done) {
991 selectText: "LINX - Linx",
992 firstAddress: "XGWQ3cb3LGUB3VnHmj6xYSMgnokNbf6dyk",
994 testNetwork(done, params);
996 it('Allows selection
of Litecoincash
', function(done) {
998 selectText: "LCC - Litecoincash",
999 firstAddress: "Ce5n7fjUuQPLutJ4W5nCCfQLKdKLE1mv9A",
1001 testNetwork(done, params);
1003 it('Allows selection
of Lynx
', function(done) {
1005 selectText: "LYNX - Lynx",
1006 firstAddress: "KUeY3ZdZkg96p4W98pj1JjygCFU1XqWdw3",
1008 testNetwork(done, params);
1010 it('Allows selection
of Megacoin
', function(done) {
1012 selectText: "MEC - Megacoin",
1013 firstAddress: "MDfAj9CzkC1HpcUiVGnHp8yKTa7WXgu8AY",
1015 testNetwork(done, params);
1017 it('Allows selection
of Minexcoin
', function(done) {
1019 selectText: "MNX - Minexcoin",
1020 firstAddress: "XC1VnyJVfiMDwWgFtAHDp41cgY3AHk3dJT",
1022 testNetwork(done, params);
1024 it('Allows selection
of Navcoin
', function(done) {
1026 selectText: "NAV - Navcoin",
1027 firstAddress: "NTQVTPK3NWSQLKoffkiQw99T8PifkF1Y2U",
1029 testNetwork(done, params);
1031 it('Allows selection
of Nebulas
', function(done) {
1033 selectText: "NAS - Nebulas",
1034 firstAddress: "n1PbK61DGBfDoDusLw621G6sVSMfLLHdfnm",
1036 testNetwork(done, params);
1038 it('Allows selection
of Neoscoin
', function(done) {
1040 selectText: "NEOS - Neoscoin",
1041 firstAddress: "NgATz6QbQNXvayHQ4CpZayugb9HeaPDdby",
1043 testNetwork(done, params);
1045 it('Allows selection
of Nix
', function(done) {
1047 selectText: "NIX - NIX Platform",
1048 firstAddress: "GgcNW2SQQXB4LWHRQTHKkQF3GzXNSLqS8u",
1050 testNetwork(done, params);
1052 it('Allows selection
of Neurocoin
', function(done) {
1054 selectText: "NRO - Neurocoin",
1055 firstAddress: "NVdYErQ3mFpDuF5DquW9WMiT7sLc8ufFTn",
1057 testNetwork(done, params);
1059 it('Allows selection
of Newyorkc
', function(done) {
1061 selectText: "NYC - Newyorkc",
1062 firstAddress: "RSVMfyH1fKfy3puADJEhut2vfkRyon6imm",
1064 testNetwork(done, params);
1066 it('Allows selection
of Novacoin
', function(done) {
1068 selectText: "NVC - Novacoin",
1069 firstAddress: "4JRvUmxcKCJmaMXZyvRoSS1cmG2XvnZfHN",
1071 testNetwork(done, params);
1073 it('Allows selection
of Nushares
', function(done) {
1075 selectText: "NSR - Nushares",
1076 firstAddress: "SecjXzU3c7EecdT7EbC4vvmbdtBBokWh6J",
1078 testNetwork(done, params);
1080 it('Allows selection
of Okcash
', function(done) {
1082 selectText: "OK - Okcash",
1083 firstAddress: "PV4Qp1TUYuGv4TqVtLZtqvrsWWRycfx1Yi",
1085 testNetwork(done, params);
1087 it('Allows selection
of Omnicore
', function(done) {
1089 selectText: "OMNI - Omnicore",
1090 firstAddress: "1Q1t3gonjCT3rW38TsTsCvgSc3hh7zBGbi",
1092 testNetwork(done, params);
1094 it('Allows selection
of DeepOnion
', function(done) {
1096 selectText: "ONION - DeepOnion",
1097 firstAddress: "DVioZ2Rjc9krDf5bbHuixznSDumzvGpzVw",
1099 testNetwork(done, params);
1101 it('Allows selection
of Pesobit
', function(done) {
1103 selectText: "PSB - Pesobit",
1104 firstAddress: "PDePsF7ALyXP7JaywokdYiRTDtKa14MAr1",
1106 testNetwork(done, params);
1108 it('Allows selection
of Pinkcoin
', function(done) {
1110 selectText: "PINK - Pinkcoin",
1111 firstAddress: "2TgjYQffjbzUHJghNaVbdsjHbRwruC3yzC",
1113 testNetwork(done, params);
1115 it('Allows selection
of POSWcoin
', function(done) {
1117 selectText: "POSW - POSWcoin",
1118 firstAddress: "PNxewmZoPnGBvoEbH6hgQZCK1igDiBCdgC",
1120 testNetwork(done, params);
1122 it('Allows selection
of Potcoin
', function(done) {
1124 selectText: "POT - Potcoin",
1125 firstAddress: "PEo7Vg2ctXgpP4vuLPeY9aGJtZotyrmiHc",
1127 testNetwork(done, params);
1129 it('Allows selection
of Putincoin
', function(done) {
1131 selectText: "PUT - Putincoin",
1132 firstAddress: "PViWnfr2uFtovd6e7joM49C94CsGSnqJis",
1134 testNetwork(done, params);
1136 it('Allows selection
of Ravencoin
', function(done) {
1138 selectText: "RVN - Ravencoin",
1139 firstAddress: "RBuDoVNnzvFsEcX8XKPm8ic4mgiCzjUCNk",
1141 testNetwork(done, params);
1143 it('Allows selection
of Reddcoin
', function(done) {
1145 selectText: "RDD - Reddcoin",
1146 firstAddress: "RtgRvXMBng1y51ftteveFqwNfyRG18HpxQ",
1148 testNetwork(done, params);
1150 it('Allows selection
of RevolutionVR
', function(done) {
1152 selectText: "RVR - RevolutionVR",
1153 firstAddress: "VXeeoP2jkzZnMFxtc66ZBZK1NHN5QJnnjL",
1155 testNetwork(done, params);
1157 it('Allows selection
of Rubycoin
', function(done) {
1159 selectText: "RBY - Rubycoin",
1160 firstAddress: "RV76JDtjTs11JdMDRToYn6CHecMRPLnKS6",
1162 testNetwork(done, params);
1164 it('Allows selection
of Salus
', function(done) {
1166 selectText: "SLS - Salus",
1167 firstAddress: "SNzPi1CafHFm3WWjRo43aMgiaEEj3ogjww",
1169 testNetwork(done, params);
1171 it('Allows selection
of Smileycoin
', function(done) {
1173 selectText: "SMLY - Smileycoin",
1174 firstAddress: "BEZVnEBCAyFByrgKpwAgYgtvP4rKAd9Sj2",
1176 testNetwork(done, params);
1178 it('Allows selection
of Solarcoin
', function(done) {
1180 selectText: "SLR - Solarcoin",
1181 firstAddress: "8LZ13HbnjtaMJWSvvVFNTLf71zFfDrhwLu",
1183 testNetwork(done, params);
1185 it('Allows selection
of stash
', function(done) {
1187 selectText: "STASH - Stash",
1188 firstAddress: "XxwAsWB7REDKmAvHA85SbEZQQtpxeUDxS3",
1190 testNetwork(done, params);
1192 it('Allows selection
of stash testnet
', function(done) {
1194 selectText: "STASH - Stash Testnet",
1195 firstAddress: "yWQCTSkUst7ddYuebKsqa1kSoXEjpCkGKR",
1197 testNetwork(done, params);
1199 it('Allows selection
of Stratis
', function(done) {
1201 selectText: "STRAT - Stratis",
1202 firstAddress: "ScfJnq3QDhKgDMEds6sqUE1ot6ShfhmXXq",
1204 testNetwork(done, params);
1206 it('Allows selection
of Stratis Test
', function(done) {
1208 selectText: "TSTRAT - Stratis Testnet",
1209 firstAddress: "TRLWm3dye4FRrDWouwYUSUZP96xb76mBE3",
1211 testNetwork(done, params);
1213 it('Allows selection
of Syscoin
', function(done) {
1215 selectText: "SYS - Syscoin",
1216 firstAddress: "SZwJi42Pst3VAMomyK5DG4157WM5ofRmSj",
1218 testNetwork(done, params);
1220 it('Allows selection
of Toa
', function(done) {
1222 selectText: "TOA - Toa",
1223 firstAddress: "TSe1QAnUwQzUfbBusDzRJ9URttrRGKoNKF",
1225 testNetwork(done, params);
1227 it('Allows selection
of Ultimatesecurecash
', function(done) {
1229 selectText: "USC - Ultimatesecurecash",
1230 firstAddress: "UPyLAZU2Che5fiy7Ed8xVJFmXAUhitA4ug",
1232 testNetwork(done, params);
1234 it('Allows selection
of Unobtanium
', function(done) {
1236 selectText: "UNO - Unobtanium",
1237 firstAddress: "uUBMPVMXrR6qhqornJqKTWgr8L69vihSL9",
1239 testNetwork(done, params);
1241 it('Allows selection
of Vcash
', function(done) {
1243 selectText: "XVC - Vcash",
1244 firstAddress: "VuL53MSY6KjvAjKSeRkh3NDnKykacDVeps",
1246 testNetwork(done, params);
1248 it('Allows selection
of Verge
', function(done) {
1250 selectText: "XVG - Verge",
1251 firstAddress: "DCrVuGkMjLJpTGgwAgv9AcMdeb1nkWbjZA",
1253 testNetwork(done, params);
1255 it('Allows selection
of Vertcoin
', function(done) {
1257 selectText: "VTC - Vertcoin",
1258 firstAddress: "Vf6koGuiWdXQfx8tNqxoNeEDxh4xh5cxsG",
1260 testNetwork(done, params);
1262 it('Allows selection
of Vivo
', function(done) {
1264 selectText: "VIVO - Vivo",
1265 firstAddress: "VFmBwuXXGhJe7MarQG2GfzHMFebRHgfSpB",
1267 testNetwork(done, params);
1269 it('Allows selection
of Vpncoin
', function(done) {
1271 selectText: "VASH - Vpncoin",
1272 firstAddress: "VoEmH1qXC4TsSgBAStR21QYetwnFqbqCx9",
1274 testNetwork(done, params);
1276 it('Allows selection
of Whitecoin
', function(done) {
1278 selectText: "XWC - Whitecoin",
1279 firstAddress: "WcSwCAUqrSgeSYbsaS3SSWWhsx8KRYTFDR",
1281 testNetwork(done, params);
1283 it('Allows selection
of Wincoin
', function(done) {
1285 selectText: "WC - Wincoin",
1286 firstAddress: "WaDVCESMGgyKgNESdn3u43NnwmGSkZED3Z",
1288 testNetwork(done, params);
1290 it('Allows selection
of Zcoin
', function(done) {
1292 selectText: "XZC - Zcoin",
1293 firstAddress: "a6VcMdP4XgAA9Tr7xNszmPG5FZpfRf17Cq",
1295 testNetwork(done, params);
1297 it('Allows selection
of Zcash
', function(done) {
1299 selectText: "ZEC - Zcash",
1300 firstAddress: "t1Sz8AneMcVuzUg3tPJ8et5AS5LFJ7K2EF9",
1302 testNetwork(done, params);
1304 it('Allows selection
of Zclassic
', function(done) {
1306 selectText: "ZCL - Zclassic",
1307 firstAddress: "t1TBMxTvVJRybUbMLGWq8H4A8F4VUL7czEc",
1309 testNetwork(done, params);
1311 it('Allows selection
of Zencash
', function(done) {
1313 selectText: "ZEN - Zencash",
1314 firstAddress: "znWh9XASyW2dZq5tck84wFjiwuqVysi7q3p",
1316 testNetwork(done, params);
1318 it('Allows selection
of Energi
', function(done) {
1320 selectText: "NRG - Energi",
1321 firstAddress: "EejRy4t4nidzhGGzkJUgFP3z4HYBjhTsRt",
1323 testNetwork(done, params);
1325 it('Allows selection
of Ethereum Classic
', function(done) {
1327 selectText: "ETC - Ethereum Classic",
1328 firstAddress: "0x3c05e5556693808367afB62eF3b63e35d6eD249A",
1330 testNetwork(done, params);
1332 it('Allows selection
of Pirl
', function(done) {
1334 selectText: "PIRL - Pirl",
1335 firstAddress: "0xe77FC0723dA122B5025CA79193c28563eB47e776",
1337 testNetwork(done, params);
1339 it('Allows selection
of MIX
', function(done) {
1341 selectText: "MIX - MIX",
1342 firstAddress: "0x98BC5e63aeb6A4e82d72850d20710F07E29A29F1",
1344 testNetwork(done, params);
1346 it('Allows selection
of Musicoin
', function(done) {
1348 selectText: "MUSIC - Musicoin",
1349 firstAddress: "0xDc060e4A0b0313ea83Cf6B3A39B9db2D29004897",
1351 testNetwork(done, params);
1353 it('Allows selection
of Poa
', function(done) {
1355 selectText: "POA - Poa",
1356 firstAddress: "0x53aF28d754e106210C3d0467Dd581eaf7e3C5e60",
1358 testNetwork(done, params);
1360 it('Allows selection
of Expanse
', function(done) {
1362 selectText: "EXP - Expanse",
1363 firstAddress: "0xf57FeAbf26582b6E3E666559d3B1Cc6fB2b2c5F6",
1365 testNetwork(done, params);
1367 it('Allows selection
of Callisto
', function(done) {
1369 selectText: "CLO - Callisto",
1370 firstAddress: "0x4f9364F7420B317266C51Dc8eB979717D4dE3f4E",
1372 testNetwork(done, params);
1374 it('Allows selection
of HUSH
', function(done) {
1376 selectText: "HUSH - Hush",
1377 firstAddress: "t1g6rLXUnJaiJuu4q4zmJjoa9Gk4fwKpiuA",
1379 testNetwork(done, params);
1381 it('Allows selection
of ExchangeCoin
', function(done) {
1383 selectText: "EXCC - ExchangeCoin",
1384 firstAddress: "22txYKpFN5fwGwdSs2UBf7ywewbLM92YqK7E",
1386 testNetwork(done, params);
1388 it('Allows selection
of Artax
', function(done) {
1390 selectText: "XAX - Artax",
1391 firstAddress: "AYxaQPY7XLidG31V7F3yNzwxPYpYzRqG4q",
1393 testNetwork(done, params);
1395 it('Allows selection
of BitcoinGreen
', function(done) {
1397 selectText: "BITG - Bitcoin Green",
1398 firstAddress: "GeNGm9SkEfwbsws3UrrUSE2sJeyWYjzraY",
1400 testNetwork(done, params);
1402 it('Allows selection
of ANON
', function(done) {
1404 selectText: "ANON - ANON",
1405 firstAddress: "AnU6pijpEeUZFWSTyM2qTqZQn996Zq1Xard",
1407 testNetwork(done, params);
1409 it('Allows selection
of ProjectCoin
', function(done) {
1411 selectText: "PRJ - ProjectCoin",
1412 firstAddress: "PXZG97saRseSCftfe1mcFmfAA7pf6qBbaz",
1414 testNetwork(done, params);
1416 it('Allows selection
of Phore
', function(done) {
1418 selectText: "PHR - Phore",
1419 firstAddress: "PJThxpoXAG6hqrmdeQQbVDX4TJtFTMMymC",
1421 testNetwork(done, params);
1423 it('Allows selection
of Safecoin
', function(done) {
1425 selectText: "SAFE - Safecoin",
1426 firstAddress: "RtxHpnhJz6RY8k9owP3ua5QWraunmewB1G",
1428 testNetwork(done, params);
1430 it('Allows selection
of Blocknode
', function(done) {
1432 selectText: "BND - Blocknode",
1433 firstAddress: "BG8xZSAur2jYLG9VXt8dYfkKxxeR7w9bSe",
1435 testNetwork(done, params);
1437 it('Allows selection
of Blocknode Testnet
', function(done) {
1439 selectText: "tBND - Blocknode Testnet",
1440 firstAddress: "bSptsFyDktFSKpWveRywJsDoJA2TC6qfHv",
1442 testNetwork(done, params);
1444 it('Allows selection
of LitecoinZ
', function(done) {
1446 selectText: "LTZ - LitecoinZ",
1447 firstAddress: "L1VTXju7hLgKV4T7fGXS9sKsnm2gmtRCmyw",
1449 testNetwork(done, params);
1451 it('Allows selection
of BlockStamp
', function(done) {
1453 selectText: "BST - BlockStamp",
1454 firstAddress: "15gypKtim4cVTj137ApfryG17RkvSbPazZ",
1456 testNetwork(done, params);
1458 it('Allows selection
of DEXON
', function(done) {
1460 selectText: "DXN - DEXON",
1461 firstAddress: "0x136a58788033E028CCd740FbDec6734358DB56Ec",
1463 testNetwork(done, params);
1465 it('Allows selection
of Ellaism
', function(done) {
1467 selectText: "ELLA - Ellaism",
1468 firstAddress: "0xa8B0BeA09eeBc41062308546a01d6E544277e2Ca",
1470 testNetwork(done, params);
1472 it('Allows selection
of Ethersocial Network
', function(done) {
1474 selectText: "ESN - Ethersocial Network",
1475 firstAddress: "0x6EE99Be2A0C7F887a71e21C8608ACF0aa0D2b767",
1477 testNetwork(done, params);
1479 it('Allows selection
of Monkey Project
', function(done) {
1481 selectText: "MONK - Monkey Project",
1482 firstAddress: "MnLrcnnUzKnf7TzufjRe5DLZqQJz18oYyu",
1484 testNetwork(done, params);
1487 // BIP39 seed is set from phrase
1488 it('Sets the bip39 seed
from the prhase
', function(done) {
1489 driver.findElement(By.css('.phrase
'))
1490 .sendKeys('abandon abandon ability
');
1491 driver.sleep(generateDelay).then(function() {
1492 driver.findElement(By.css('.seed
'))
1493 .getAttribute("value")
1494 .then(function(seed) {
1495 expect(seed).toBe("20da140d3dd1df8713cefcc4d54ce0e445b4151027a1ab567b832f6da5fcc5afc1c3a3f199ab78b8e0ab4652efd7f414ac2c9a3b81bceb879a70f377aa0a58f3");
1501 // BIP32 root key is set from phrase
1502 it('Sets the bip39 root key
from the prhase
', function(done) {
1503 driver.findElement(By.css('.phrase
'))
1504 .sendKeys('abandon abandon ability
');
1505 driver.sleep(generateDelay).then(function() {
1506 driver.findElement(By.css('.root
-key
'))
1507 .getAttribute("value")
1508 .then(function(seed) {
1509 expect(seed).toBe("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
1515 // Tabs show correct addresses when changed
1516 it('Shows the correct address when tab is changed
', function(done) {
1517 driver.findElement(By.css('.phrase
'))
1518 .sendKeys('abandon abandon ability
');
1519 driver.sleep(generateDelay).then(function() {
1520 driver.findElement(By.css('#bip32
-tab a
'))
1522 driver.sleep(generateDelay).then(function() {
1523 getFirstAddress(function(address) {
1524 expect(address).toBe("17uQ7s2izWPwBmEVFikTmZUjbBKWYdJchz");
1531 // BIP44 derivation path is shown
1532 it('Shows the derivation path
for bip44 tab
', function(done) {
1533 driver.findElement(By.css('.phrase
'))
1534 .sendKeys('abandon abandon ability
');
1535 driver.sleep(generateDelay).then(function() {
1536 driver.findElement(By.css('#bip44
.path
'))
1537 .getAttribute("value")
1538 .then(function(path) {
1539 expect(path).toBe("m/44'/0'/0'/0");
1545 // BIP44 extended private key is shown
1546 it('Shows the extended private key for bip44 tab', function(done) {
1547 driver.findElement(By.css('.phrase'))
1548 .sendKeys('abandon abandon ability');
1549 driver.sleep(generateDelay).then(function() {
1550 driver.findElement(By.css('.extended-priv-key'))
1551 .getAttribute("value
")
1552 .then(function(path) {
1553 expect(path).toBe("xprvA2DxxvPZcyRvYgZMGS53nadR32mVDeCyqQYyFhrCVbJNjPoxMeVf7QT5g7mQASbTf9Kp4cryvcXnu2qurjWKcrdsr91jXymdCDNxKgLFKJG
");
1559 // BIP44 extended public key is shown
1560 it('Shows the extended public key for bip44 tab', function(done) {
1561 driver.findElement(By.css('.phrase'))
1562 .sendKeys('abandon abandon ability');
1563 driver.sleep(generateDelay).then(function() {
1564 driver.findElement(By.css('.extended-pub-key'))
1565 .getAttribute("value
")
1566 .then(function(path) {
1567 expect(path).toBe("xpub6FDKNRvTTLzDmAdpNTc49ia9b4byd6vqCdUa46Fp3vqMcC96uBoufCmZXQLiN5AK3iSCJMhf9gT2sxkpyaPepRuA7W3MujV5tGmF5VfbueM
");
1573 // BIP44 account field changes address list
1574 it('Changes the address list if bip44 account is changed', function(done) {
1575 driver.findElement(By.css('#bip44 .account'))
1577 driver.findElement(By.css('.phrase'))
1578 .sendKeys('abandon abandon ability');
1579 driver.sleep(generateDelay).then(function() {
1580 getFirstAddress(function(address) {
1581 expect(address).toBe("1Nq2Wmu726XHCuGhctEtGmhxo3wzk5wZ1H
");
1587 // BIP44 change field changes address list
1588 it('Changes the address list if bip44 change is changed', function(done) {
1589 driver.findElement(By.css('#bip44 .change'))
1591 driver.findElement(By.css('.phrase'))
1592 .sendKeys('abandon abandon ability');
1593 driver.sleep(generateDelay).then(function() {
1594 getFirstAddress(function(address) {
1595 expect(address).toBe("1KAGfWgqfVbSSXY56fNQ7YnhyKuoskHtYo
");
1601 // BIP32 derivation path can be set
1602 it('Can use a custom bip32 derivation path', function(done) {
1603 driver.findElement(By.css('#bip32-tab a'))
1605 driver.findElement(By.css('#bip32 .path'))
1607 driver.findElement(By.css('#bip32 .path'))
1609 driver.findElement(By.css('.phrase'))
1610 .sendKeys('abandon abandon ability');
1611 driver.sleep(generateDelay).then(function() {
1612 getFirstAddress(function(address) {
1613 expect(address).toBe("16pYQQdLD1hH4hwTGLXBaZ9Teboi1AGL8L
");
1619 // BIP32 can use hardened derivation paths
1620 it('Can use a hardened derivation paths', function(done) {
1621 driver.findElement(By.css('#bip32-tab a'))
1623 driver.findElement(By.css('#bip32 .path'))
1625 driver.findElement(By.css('#bip32 .path'))
1627 driver.findElement(By.css('.phrase
'))
1628 .sendKeys('abandon abandon ability
');
1629 driver.sleep(generateDelay).then(function() {
1630 getFirstAddress(function(address) {
1631 expect(address).toBe("14aXZeprXAE3UUKQc4ihvwBvww2LuEoHo4");
1637 // BIP32 extended private key is shown
1638 it('Shows the BIP32 extended
private key
', function(done) {
1639 driver.findElement(By.css('#bip32
-tab a
'))
1641 driver.findElement(By.css('.phrase
'))
1642 .sendKeys('abandon abandon ability
');
1643 driver.sleep(generateDelay).then(function() {
1644 driver.findElement(By.css('.extended
-priv
-key
'))
1645 .getAttribute("value")
1646 .then(function(privKey) {
1647 expect(privKey).toBe("xprv9va99uTVE5aLiutUVLTyfxfe8v8aaXjSQ1XxZbK6SezYVuikA9MnjQVTA8rQHpNA5LKvyQBpLiHbBQiiccKiBDs7eRmBogsvq3THFeLHYbe");
1653 // BIP32 extended public key is shown
1654 it('Shows the BIP32 extended
public key
', function(done) {
1655 driver.findElement(By.css('#bip32
-tab a
'))
1657 driver.findElement(By.css('.phrase
'))
1658 .sendKeys('abandon abandon ability
');
1659 driver.sleep(generateDelay).then(function() {
1660 driver.findElement(By.css('.extended
-pub
-key
'))
1661 .getAttribute("value")
1662 .then(function(pubKey) {
1663 expect(pubKey).toBe("xpub69ZVZQzP4T8dwPxwbMzz36cNgwy4yzTHmETZMyihzzXXNi3thgg3HCow1RtY252wdw5rS8369xKnraN5Q93y3FkFfJp2XEHWUrkyXsjS93P");
1669 // Derivation path is shown in table
1670 it('Shows the derivation path
in the table
', function(done) {
1671 driver.findElement(By.css('.phrase
'))
1672 .sendKeys('abandon abandon ability
');
1673 driver.sleep(generateDelay).then(function() {
1674 getFirstPath(function(path) {
1675 expect(path).toBe("m/44'/0'/0'/0/0");
1681 // Derivation path for address can be hardened
1682 it('Can derive hardened addresses', function(done) {
1683 driver.findElement(By.css('#bip32-tab a'))
1685 driver.executeScript(function() {
1686 $(".hardened
-addresses
").prop("checked
", true);
1688 driver.findElement(By.css('.phrase'))
1689 .sendKeys('abandon abandon ability');
1690 driver.sleep(generateDelay).then(function() {
1691 getFirstAddress(function(address) {
1692 expect(address).toBe("18exLzUv7kfpiXRzmCjFDoC9qwNLFyvwyd
");
1698 // Derivation path visibility can be toggled
1699 it('Can toggle visibility of the derivation path column', function(done) {
1700 driver.findElement(By.css('.phrase'))
1701 .sendKeys('abandon abandon ability');
1702 driver.sleep(generateDelay).then(function() {
1703 driver.findElement(By.css('.index-toggle'))
1705 testColumnValuesAreInvisible(done, "index
");
1710 it('Shows the address in the table', function(done) {
1711 driver.findElement(By.css('.phrase'))
1712 .sendKeys('abandon abandon ability');
1713 driver.sleep(generateDelay).then(function() {
1714 getFirstAddress(function(address) {
1715 expect(address).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug
");
1721 // Addresses are shown in order of derivation path
1722 it('Shows the address in order of derivation path', function(done) {
1723 driver.findElement(By.css('.phrase'))
1724 .sendKeys('abandon abandon ability');
1725 driver.sleep(generateDelay).then(function() {
1726 testRowsAreInCorrectOrder(done);
1730 // Address visibility can be toggled
1731 it('Can toggle visibility of the address column', function(done) {
1732 driver.findElement(By.css('.phrase'))
1733 .sendKeys('abandon abandon ability');
1734 driver.sleep(generateDelay).then(function() {
1735 driver.findElement(By.css('.address-toggle'))
1737 testColumnValuesAreInvisible(done, "address
");
1741 // Public key is shown in table
1742 it('Shows the public key in the table', function(done) {
1743 driver.findElement(By.css('.phrase'))
1744 .sendKeys('abandon abandon ability');
1745 driver.sleep(generateDelay).then(function() {
1746 driver.findElements(By.css('.pubkey'))
1747 .then(function(els) {
1749 .then(function(pubkey) {
1750 expect(pubkey).toBe("033f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3
");
1757 // Public key visibility can be toggled
1758 it('Can toggle visibility of the public key column', function(done) {
1759 driver.findElement(By.css('.phrase'))
1760 .sendKeys('abandon abandon ability');
1761 driver.sleep(generateDelay).then(function() {
1762 driver.findElement(By.css('.public-key-toggle'))
1764 testColumnValuesAreInvisible(done, "pubkey
");
1768 // Private key is shown in table
1769 it('Shows the private key in the table', function(done) {
1770 driver.findElement(By.css('.phrase'))
1771 .sendKeys('abandon abandon ability');
1772 driver.sleep(generateDelay).then(function() {
1773 driver.findElements(By.css('.privkey'))
1774 .then(function(els) {
1776 .then(function(pubkey) {
1777 expect(pubkey).toBe("L26cVSpWFkJ6aQkPkKmTzLqTdLJ923e6CzrVh9cmx21QHsoUmrEE
");
1784 // Private key visibility can be toggled
1785 it('Can toggle visibility of the private key column', function(done) {
1786 driver.findElement(By.css('.phrase'))
1787 .sendKeys('abandon abandon ability');
1788 driver.sleep(generateDelay).then(function() {
1789 driver.findElement(By.css('.private-key-toggle'))
1791 testColumnValuesAreInvisible(done, "privkey
");
1795 // More addresses can be generated
1796 it('Can generate more rows in the table', function(done) {
1797 driver.findElement(By.css('.phrase'))
1798 .sendKeys('abandon abandon ability');
1799 driver.sleep(generateDelay).then(function() {
1800 driver.findElement(By.css('.more'))
1802 driver.sleep(generateDelay).then(function() {
1803 driver.findElements(By.css('.address'))
1804 .then(function(els) {
1805 expect(els.length).toBe(40);
1812 // A custom number of additional addresses can be generated
1813 it('Can generate more rows in the table', function(done) {
1814 driver.findElement(By.css('.phrase'))
1815 .sendKeys('abandon abandon ability');
1816 driver.sleep(generateDelay).then(function() {
1817 driver.findElement(By.css('.rows-to-add'))
1819 driver.findElement(By.css('.rows-to-add'))
1821 driver.findElement(By.css('.more'))
1823 driver.sleep(generateDelay).then(function() {
1824 driver.findElements(By.css('.address'))
1825 .then(function(els) {
1826 expect(els.length).toBe(21);
1833 // Additional addresses are shown in order of derivation path
1834 it('Shows additional addresses in order of derivation path', function(done) {
1835 driver.findElement(By.css('.phrase'))
1836 .sendKeys('abandon abandon ability');
1837 driver.sleep(generateDelay).then(function() {
1838 driver.findElement(By.css('.more'))
1840 driver.sleep(generateDelay).then(function() {
1841 testRowsAreInCorrectOrder(done);
1846 // BIP32 root key can be set by the user
1847 it('Allows the user to set the BIP32 root key', function(done) {
1848 driver.findElement(By.css('.root-key'))
1849 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1850 driver.sleep(generateDelay).then(function() {
1851 getFirstAddress(function(address) {
1852 expect(address).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug
");
1858 // Setting BIP32 root key clears the existing phrase, passphrase and seed
1859 // TODO this doesn't work in selenium with chrome
1860 it('Confirms the existing phrase should be cleared', function(done) {
1861 if (browser == "chrome
") {
1862 pending("Selenium
+ Chrome headless bug
for alert
, see
https://stackoverflow.com/q/45242264");
1864 driver
.findElement(By
.css('.phrase'))
1865 .sendKeys('A non-blank but invalid value');
1866 driver
.findElement(By
.css('.root-key'))
1867 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1868 driver
.switchTo().alert().accept();
1869 driver
.findElement(By
.css('.phrase'))
1870 .getAttribute("value").then(function(value
) {
1871 expect(value
).toBe("");
1876 // Clearing of phrase, passphrase and seed can be cancelled by user
1877 // TODO this doesn't work in selenium with chrome
1878 it('Allows the clearing of the phrase to be cancelled', function(done
) {
1879 if (browser
== "chrome") {
1880 pending("Selenium + Chrome headless bug for alert, see https://stackoverflow.com/q/45242264");
1882 driver
.findElement(By
.css('.phrase'))
1883 .sendKeys('abandon abandon ability');
1884 driver
.sleep(generateDelay
).then(function() {
1885 driver
.findElement(By
.css('.root-key'))
1887 driver
.findElement(By
.css('.root-key'))
1889 driver
.switchTo().alert().dismiss();
1890 driver
.findElement(By
.css('.phrase'))
1891 .getAttribute("value").then(function(value
) {
1892 expect(value
).toBe("abandon abandon ability");
1898 // Custom BIP32 root key is used when changing the derivation path
1899 it('Can set derivation path for root key instead of phrase', function(done
) {
1900 driver
.findElement(By
.css('#bip44 .account'))
1902 driver
.findElement(By
.css('.root-key'))
1903 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1904 driver
.sleep(generateDelay
).then(function() {
1905 getFirstAddress(function(address
) {
1906 expect(address
).toBe("1Nq2Wmu726XHCuGhctEtGmhxo3wzk5wZ1H");
1912 // Incorrect mnemonic shows error
1913 it('Shows an error for incorrect mnemonic', function(done
) {
1914 driver
.findElement(By
.css('.phrase'))
1915 .sendKeys('abandon abandon abandon');
1916 driver
.sleep(feedbackDelay
).then(function() {
1917 driver
.findElement(By
.css('.feedback'))
1919 .then(function(feedback
) {
1920 expect(feedback
).toBe("Invalid mnemonic");
1926 // Incorrect word shows suggested replacement
1927 it('Shows word suggestion for incorrect word', function(done
) {
1928 driver
.findElement(By
.css('.phrase'))
1929 .sendKeys('abandon abandon abiliti');
1930 driver
.sleep(feedbackDelay
).then(function() {
1931 driver
.findElement(By
.css('.feedback'))
1933 .then(function(feedback
) {
1934 var msg
= "abiliti not in wordlist, did you mean ability?";
1935 expect(feedback
).toBe(msg
);
1941 // Github pull request 48
1942 // First four letters of word shows that word, not closest
1943 // since first four letters gives unique word in BIP39 wordlist
1944 // eg ille should show illegal, not idle
1945 it('Shows word suggestion based on first four chars', function(done
) {
1946 driver
.findElement(By
.css('.phrase'))
1948 driver
.sleep(feedbackDelay
).then(function() {
1949 driver
.findElement(By
.css('.feedback'))
1951 .then(function(feedback
) {
1952 var msg
= "ille not in wordlist, did you mean illegal?";
1953 expect(feedback
).toBe(msg
);
1959 // Incorrect BIP32 root key shows error
1960 it('Shows error for incorrect root key', function(done
) {
1961 driver
.findElement(By
.css('.root-key'))
1962 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpj');
1963 driver
.sleep(feedbackDelay
).then(function() {
1964 driver
.findElement(By
.css('.feedback'))
1966 .then(function(feedback
) {
1967 var msg
= "Invalid root key";
1968 expect(feedback
).toBe(msg
);
1974 // Derivation path not starting with m shows error
1975 it('Shows error for derivation path not starting with m', function(done
) {
1976 driver
.findElement(By
.css('#bip32-tab a'))
1978 driver
.findElement(By
.css('#bip32 .path'))
1980 driver
.findElement(By
.css('#bip32 .path'))
1982 driver
.findElement(By
.css('.phrase'))
1983 .sendKeys('abandon abandon ability');
1984 driver
.sleep(feedbackDelay
).then(function() {
1985 driver
.findElement(By
.css('.feedback'))
1987 .then(function(feedback
) {
1988 var msg
= "First character must be 'm'";
1989 expect(feedback
).toBe(msg
);
1995 // Derivation path containing invalid characters shows useful error
1996 it('Shows error for derivation path not starting with m', function(done
) {
1997 driver
.findElement(By
.css('#bip32-tab a'))
1999 driver
.findElement(By
.css('#bip32 .path'))
2001 driver
.findElement(By
.css('#bip32 .path'))
2002 .sendKeys('m/1/0wrong1/1');
2003 driver
.findElement(By
.css('.phrase'))
2004 .sendKeys('abandon abandon ability');
2005 driver
.sleep(feedbackDelay
).then(function() {
2006 driver
.findElement(By
.css('.feedback'))
2008 .then(function(feedback
) {
2009 var msg
= "Invalid characters 0wrong1 found at depth 2";
2010 expect(feedback
).toBe(msg
);
2016 // Github Issue 11: Default word length is 15
2017 // https://github.com/iancoleman/bip39/issues/11
2018 it('Sets the default word length to 15', function(done
) {
2019 driver
.findElement(By
.css('.strength'))
2020 .getAttribute("value")
2021 .then(function(strength
) {
2022 expect(strength
).toBe("15");
2027 // Github Issue 12: Generate more rows with private keys hidden
2028 // https://github.com/iancoleman/bip39/issues/12
2029 it('Sets the correct hidden column state on new rows', function(done
) {
2030 driver
.findElement(By
.css('.phrase'))
2031 .sendKeys("abandon abandon ability");
2032 driver
.sleep(generateDelay
).then(function() {
2033 driver
.findElement(By
.css('.private-key-toggle'))
2035 driver
.findElement(By
.css('.more'))
2037 driver
.sleep(generateDelay
).then(function() {
2038 driver
.findElements(By
.css('.privkey'))
2039 .then(function(els
) {
2040 expect(els
.length
).toBe(40);
2042 testColumnValuesAreInvisible(done
, "privkey");
2047 // Github Issue 19: Mnemonic is not sensitive to whitespace
2048 // https://github.com/iancoleman/bip39/issues/19
2049 it('Ignores excess whitespace in the mnemonic', function(done
) {
2050 var doublespace
= " ";
2051 var mnemonic
= "urge cat" + doublespace
+ "bid";
2052 driver
.findElement(By
.css('.phrase'))
2053 .sendKeys(mnemonic
);
2054 driver
.sleep(generateDelay
).then(function() {
2055 driver
.findElement(By
.css('.root-key'))
2056 .getAttribute("value")
2057 .then(function(seed
) {
2058 expect(seed
).toBe("xprv9s21ZrQH143K3isaZsWbKVoTtbvd34Y1ZGRugGdMeBGbM3AgBVzTH159mj1cbbtYSJtQr65w6L5xy5L9SFC7c9VJZWHxgAzpj4mun5LhrbC");
2064 // Github Issue 23: Part 1: Use correct derivation path when changing tabs
2065 // https://github.com/iancoleman/bip39/issues/23
2066 it('Uses the correct derivation path when changing tabs', function(done
) {
2067 // 1) and 2) set the phrase
2068 driver
.findElement(By
.css('.phrase'))
2069 .sendKeys("abandon abandon ability");
2070 driver
.sleep(generateDelay
).then(function() {
2071 // 3) select bip32 tab
2072 driver
.findElement(By
.css('#bip32-tab a'))
2074 driver
.sleep(generateDelay
).then(function() {
2075 // 4) switch from bitcoin to litecoin
2076 selectNetwork("LTC - Litecoin");
2077 driver
.sleep(generateDelay
).then(function() {
2078 // 5) Check address is displayed correctly
2079 getFirstAddress(function(address
) {
2080 expect(address
).toBe("LS8MP5LZ5AdzSZveRrjm3aYVoPgnfFh5T5");
2081 // 5) Check derivation path is displayed correctly
2082 getFirstPath(function(path
) {
2083 expect(path
).toBe("m/0/0");
2092 // Github Issue 23 Part 2: Coin selection in derivation path
2093 // https://github.com/iancoleman/bip39/issues/23#issuecomment-238011920
2094 it('Uses the correct derivation path when changing coins', function(done
) {
2096 driver
.findElement(By
.css('.phrase'))
2097 .sendKeys("abandon abandon ability");
2098 driver
.sleep(generateDelay
).then(function() {
2099 // switch from bitcoin to clam
2100 selectNetwork("CLAM - Clams");
2101 driver
.sleep(generateDelay
).then(function() {
2102 // check derivation path is displayed correctly
2103 getFirstPath(function(path
) {
2104 expect(path
).toBe("m/44'/23'/0'/0/0");
2111 // Github Issue 26: When using a Root key derrived altcoins are incorrect
2112 // https://github.com/iancoleman/bip39/issues/26
2113 it('Uses the correct derivation for altcoins with root keys', function(done
) {
2114 // 1) 2) and 3) set the root key
2115 driver
.findElement(By
.css('.root-key'))
2116 .sendKeys("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
2117 driver
.sleep(generateDelay
).then(function() {
2118 // 4) switch from bitcoin to viacoin
2119 selectNetwork("VIA - Viacoin");
2120 driver
.sleep(generateDelay
).then(function() {
2121 // 5) ensure the derived address is correct
2122 getFirstAddress(function(address
) {
2123 expect(address
).toBe("Vq9Eq4N5SQnjqZvxtxzo7hZPW5XnyJsmXT");
2130 // Selecting a language with no existing phrase should generate a phrase in
2132 it('Generate a random phrase when language is selected and no current phrase', function(done
) {
2133 driver
.findElement(By
.css("a[href='#japanese']"))
2135 driver
.sleep(generateDelay
).then(function() {
2136 driver
.findElement(By
.css(".phrase"))
2137 .getAttribute("value").then(function(phrase
) {
2138 expect(phrase
.search(/[a-z]/)).toBe(-1);
2139 expect(phrase
.length
).toBeGreaterThan(0);
2145 // Selecting a language with existing phrase should update the phrase to use
2147 it('Updates existing phrases when the language is changed', function(done
) {
2148 driver
.findElement(By
.css(".phrase"))
2149 .sendKeys("abandon abandon ability");
2150 driver
.sleep(generateDelay
).then(function() {
2151 driver
.findElement(By
.css("a[href='#italian']"))
2153 driver
.sleep(generateDelay
).then(function() {
2154 driver
.findElement(By
.css(".phrase"))
2155 .getAttribute("value").then(function(phrase
) {
2156 // Check only the language changes, not the phrase
2157 expect(phrase
).toBe("abaco abaco abbaglio");
2158 getFirstAddress(function(address
) {
2159 // Check the address is correct
2160 expect(address
).toBe("1Dz5TgDhdki9spa6xbPFbBqv5sjMrx3xgV");
2168 // Suggested replacement for erroneous word in non-English language
2169 it('Shows word suggestion for incorrect word in non-English language', function(done
) {
2170 driver
.findElement(By
.css('.phrase'))
2171 .sendKeys('abaco abaco zbbaglio');
2172 driver
.sleep(feedbackDelay
).then(function() {
2173 driver
.findElement(By
.css('.feedback'))
2175 .then(function(feedback
) {
2176 var msg
= "zbbaglio not in wordlist, did you mean abbaglio?";
2177 expect(feedback
).toBe(msg
);
2183 // Japanese word does not break across lines.
2185 // https://github.com/bitcoin/bips/blob/master/bip-0039/bip-0039-wordlists.md#japanese
2186 it('Does not break Japanese words across lines', function(done
) {
2187 driver
.findElement(By
.css('.phrase'))
2188 .getCssValue("word-break")
2189 .then(function(value
) {
2190 expect(value
).toBe("keep-all");
2195 // Language can be specified at page load using hash value in url
2196 it('Can set the language from the url hash', function(done
) {
2197 driver
.get(url
+ "#japanese").then(function() {
2198 driver
.findElement(By
.css('.generate')).click();
2199 driver
.sleep(generateDelay
).then(function() {
2200 driver
.findElement(By
.css(".phrase"))
2201 .getAttribute("value").then(function(phrase
) {
2202 expect(phrase
.search(/[a-z]/)).toBe(-1);
2203 expect(phrase
.length
).toBeGreaterThan(0);
2210 // Entropy can be entered by the user
2211 it('Allows entropy to be entered', function(done
) {
2212 driver
.findElement(By
.css('.use-entropy'))
2214 driver
.findElement(By
.css('.entropy'))
2215 .sendKeys('00000000 00000000 00000000 00000000');
2216 driver
.sleep(generateDelay
).then(function() {
2217 driver
.findElement(By
.css(".phrase"))
2218 .getAttribute("value").then(function(phrase
) {
2219 expect(phrase
).toBe("abandon abandon ability");
2220 getFirstAddress(function(address
) {
2221 expect(address
).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug");
2228 // A warning about entropy is shown to the user, with additional information
2229 it('Shows a warning about using entropy', function(done
) {
2230 driver
.findElement(By
.css('.use-entropy'))
2232 driver
.findElement(By
.css('.entropy-container'))
2234 .then(function(containerText
) {
2235 var warning
= "mnemonic may be insecure";
2236 expect(containerText
).toContain(warning
);
2237 driver
.findElement(By
.css('#entropy-notes'))
2238 .findElement(By
.xpath("parent::*"))
2240 .then(function(notesText
) {
2241 var detail
= "flipping a fair coin, rolling a fair dice, noise measurements etc";
2242 expect(notesText
).toContain(detail
);
2248 // The types of entropy available are described to the user
2249 it('Shows the types of entropy available', function(done
) {
2250 driver
.findElement(By
.css('.entropy'))
2251 .getAttribute("placeholder")
2252 .then(function(placeholderText
) {
2261 for (var i
=0; i
<options
.length
; i
++) {
2262 var option
= options
[i
];
2263 expect(placeholderText
).toContain(option
);
2269 // The actual entropy used is shown to the user
2270 it('Shows the actual entropy used', function(done
) {
2271 driver
.findElement(By
.css('.use-entropy'))
2273 driver
.findElement(By
.css('.entropy'))
2274 .sendKeys('Not A Very Good Entropy Source At All');
2275 driver
.sleep(generateDelay
).then(function() {
2276 driver
.findElement(By
.css('.entropy-container'))
2278 .then(function(text
) {
2279 expect(text
).toMatch(/Filtered Entropy
\s
+AedEceAA
/);
2285 // Binary entropy can be entered
2286 it('Allows binary entropy to be entered', function(done
) {
2287 testEntropyType(done
, "01", "binary");
2290 // Base 6 entropy can be entered
2291 it('Allows base 6 entropy to be entered', function(done
) {
2292 testEntropyType(done
, "012345", "base 6");
2295 // Base 6 dice entropy can be entered
2296 it('Allows base 6 dice entropy to be entered', function(done
) {
2297 testEntropyType(done
, "123456", "base 6 (dice)");
2300 // Base 10 entropy can be entered
2301 it('Allows base 10 entropy to be entered', function(done
) {
2302 testEntropyType(done
, "789", "base 10");
2305 // Hexadecimal entropy can be entered
2306 it('Allows hexadecimal entropy to be entered', function(done
) {
2307 testEntropyType(done
, "abcdef", "hexadecimal");
2310 // Dice entropy value is shown as the converted base 6 value
2311 // ie 123456 is converted to 123450
2312 it('Shows dice entropy as base 6', function(done
) {
2313 driver
.findElement(By
.css('.use-entropy'))
2315 driver
.findElement(By
.css('.entropy'))
2316 .sendKeys("123456");
2317 driver
.sleep(generateDelay
).then(function() {
2318 driver
.findElement(By
.css('.entropy-container'))
2320 .then(function(text
) {
2321 expect(text
).toMatch(/Filtered Entropy
\s
+123450/);
2327 // The number of bits of entropy accumulated is shown
2328 it("Shows the number of bits of entropy for 20 bits of binary", function(done
) {
2329 testEntropyBits(done
, "0000 0000 0000 0000 0000", "20");
2331 it("Shows the number of bits of entropy for 1 bit of binary", function(done
) {
2332 testEntropyBits(done
, "0", "1");
2334 it("Shows the number of bits of entropy for 4 bits of binary", function(done
) {
2335 testEntropyBits(done
, "0000", "4");
2337 it("Shows the number of bits of entropy for 1 character of base 6 (dice)", function(done
) {
2338 // 6 in card is 0 in base 6, 0 in base 6 is 2.6 bits (rounded down to 2 bits)
2339 testEntropyBits(done
, "6", "2");
2341 it("Shows the number of bits of entropy for 1 character of base 10 with 3 bits", function(done
) {
2342 // 7 in base 10 is 111 in base 2, no leading zeros
2343 testEntropyBits(done
, "7", "3");
2345 it("Shows the number of bits of entropy for 1 character of base 10 with 4 bis", function(done
) {
2346 testEntropyBits(done
, "8", "4");
2348 it("Shows the number of bits of entropy for 1 character of hex", function(done
) {
2349 testEntropyBits(done
, "F", "4");
2351 it("Shows the number of bits of entropy for 2 characters of base 10", function(done
) {
2352 testEntropyBits(done
, "29", "6");
2354 it("Shows the number of bits of entropy for 2 characters of hex", function(done
) {
2355 testEntropyBits(done
, "0A", "8");
2357 it("Shows the number of bits of entropy for 2 characters of hex with 3 leading zeros", function(done
) {
2358 // hex is always multiple of 4 bits of entropy
2359 testEntropyBits(done
, "1A", "8");
2361 it("Shows the number of bits of entropy for 2 characters of hex with 2 leading zeros", function(done
) {
2362 testEntropyBits(done
, "2A", "8");
2364 it("Shows the number of bits of entropy for 2 characters of hex with 1 leading zero", function(done
) {
2365 testEntropyBits(done
, "4A", "8");
2367 it("Shows the number of bits of entropy for 2 characters of hex with no leading zeros", function(done
) {
2368 testEntropyBits(done
, "8A", "8");
2370 it("Shows the number of bits of entropy for 2 characters of hex starting with F", function(done
) {
2371 testEntropyBits(done
, "FA", "8");
2373 it("Shows the number of bits of entropy for 4 characters of hex with leading zeros", function(done
) {
2374 testEntropyBits(done
, "000A", "16");
2376 it("Shows the number of bits of entropy for 4 characters of base 6", function(done
) {
2377 testEntropyBits(done
, "5555", "11");
2379 it("Shows the number of bits of entropy for 4 characters of base 6 dice", function(done
) {
2380 // uses dice, so entropy is actually 0000 in base 6, which is 4 lots of
2381 // 2.58 bits, which is 10.32 bits (rounded down to 10 bits)
2382 testEntropyBits(done
, "6666", "10");
2384 it("Shows the number of bits of entropy for 4 charactes of base 10", function(done
) {
2385 // Uses base 10, which is 4 lots of 3.32 bits, which is 13.3 bits (rounded
2387 testEntropyBits(done
, "2227", "13");
2389 it("Shows the number of bits of entropy for 4 characters of hex with 2 leading zeros", function(done
) {
2390 testEntropyBits(done
, "222F", "16");
2392 it("Shows the number of bits of entropy for 4 characters of hex starting with F", function(done
) {
2393 testEntropyBits(done
, "FFFF", "16");
2395 it("Shows the number of bits of entropy for 10 characters of base 10", function(done
) {
2396 // 10 events at 3.32 bits per event
2397 testEntropyBits(done
, "0000101017", "33");
2399 it("Shows the number of bits of entropy for a full deck of cards", function(done
) {
2400 // cards are not replaced, so a full deck is not 52^52 entropy which is 296
2401 // bits, it's 52!, which is 225 bits
2402 testEntropyBits(done
, "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks", "225");
2405 it("Shows details about the entered entropy", function(done
) {
2406 testEntropyFeedback(done
,
2410 type: "hexadecimal",
2414 strength: "less than a second",
2418 it("Shows details about the entered entropy", function(done
) {
2419 testEntropyFeedback(done
,
2421 entropy: "AAAAAAAA",
2422 filtered: "AAAAAAAA",
2423 type: "hexadecimal",
2427 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2431 it("Shows details about the entered entropy", function(done
) {
2432 testEntropyFeedback(done
,
2434 entropy: "AAAAAAAA B",
2435 filtered: "AAAAAAAAB",
2436 type: "hexadecimal",
2440 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2444 it("Shows details about the entered entropy", function(done
) {
2445 testEntropyFeedback(done
,
2447 entropy: "AAAAAAAA BBBBBBBB",
2448 filtered: "AAAAAAAABBBBBBBB",
2449 type: "hexadecimal",
2453 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2457 it("Shows details about the entered entropy", function(done
) {
2458 testEntropyFeedback(done
,
2460 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC",
2461 filtered: "AAAAAAAABBBBBBBBCCCCCCCC",
2462 type: "hexadecimal",
2466 strength: "less than a second",
2470 it("Shows details about the entered entropy", function(done
) {
2471 testEntropyFeedback(done
,
2473 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDD",
2474 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDD",
2475 type: "hexadecimal",
2479 strength: "2 minutes",
2483 it("Shows details about the entered entropy", function(done
) {
2484 testEntropyFeedback(done
,
2486 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA",
2487 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDA",
2488 type: "hexadecimal",
2496 it("Shows details about the entered entropy", function(done
) {
2497 testEntropyFeedback(done
,
2499 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA EEEEEEEE",
2500 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDAEEEEEEEE",
2501 type: "hexadecimal",
2505 strength: "3 years",
2509 it("Shows details about the entered entropy", function(done
) {
2510 testEntropyFeedback(done
,
2512 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA EEEEEEEE FFFFFFFF",
2513 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDAEEEEEEEEFFFFFFFF",
2514 type: "hexadecimal",
2518 strength: "centuries",
2522 it("Shows details about the entered entropy", function(done
) {
2523 testEntropyFeedback(done
,
2530 strength: "less than a second",
2534 it("Shows details about the entered entropy", function(done
) {
2535 testEntropyFeedback(done
,
2537 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2538 type: "card (full deck)",
2542 strength: "centuries",
2546 it("Shows details about the entered entropy", function(done
) {
2547 testEntropyFeedback(done
,
2549 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks3d",
2550 type: "card (full deck, 1 duplicate: 3d)",
2554 strength: "centuries",
2558 it("Shows details about the entered entropy", function(done
) {
2559 testEntropyFeedback(done
,
2561 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqs3d4d",
2562 type: "card (2 duplicates: 3d 4d, 1 missing: KS)",
2566 strength: "centuries",
2570 it("Shows details about the entered entropy", function(done
) {
2571 testEntropyFeedback(done
,
2573 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqs3d4d5d6d",
2574 type: "card (4 duplicates: 3d 4d 5d..., 1 missing: KS)",
2578 strength: "centuries",
2582 it("Shows details about the entered entropy", function(done
) {
2583 testEntropyFeedback(done
,
2584 // Next test was throwing uncaught error in zxcvbn
2585 // Also tests 451 bits, ie Math.log2(52!)*2 = 225.58 * 2
2587 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsksac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2588 type: "card (full deck, 52 duplicates: ac 2c 3c...)",
2592 strength: "centuries",
2596 it("Shows details about the entered entropy", function(done
) {
2597 testEntropyFeedback(done
,
2598 // Case insensitivity to duplicate cards
2601 type: "card (1 duplicate: AS)",
2605 strength: "less than a second",
2609 it("Shows details about the entered entropy", function(done
) {
2610 testEntropyFeedback(done
,
2613 type: "card (1 duplicate: as)",
2617 strength: "less than a second",
2621 it("Shows details about the entered entropy", function(done
) {
2622 testEntropyFeedback(done
,
2623 // Missing cards are detected
2625 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2626 type: "card (1 missing: 9C)",
2630 strength: "centuries",
2634 it("Shows details about the entered entropy", function(done
) {
2635 testEntropyFeedback(done
,
2637 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2638 type: "card (2 missing: 9C 5D)",
2642 strength: "centuries",
2646 it("Shows details about the entered entropy", function(done
) {
2647 testEntropyFeedback(done
,
2649 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d7d8d9dtdjd kdah2h3h 5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2650 type: "card (4 missing: 9C 5D QD...)",
2654 strength: "centuries",
2658 it("Shows details about the entered entropy", function(done
) {
2659 testEntropyFeedback(done
,
2660 // More than six missing cards does not show message
2662 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d 8d9d jd kdah2h3h 5h6h7h8h9hthjhqhkh 2s3s4s5s6s7s8s9stsjsqsks",
2667 strength: "centuries",
2671 it("Shows details about the entered entropy", function(done
) {
2672 testEntropyFeedback(done
,
2673 // Multiple decks of cards increases bits per event
2678 bitsPerEvent: "4.34",
2682 it("Shows details about the entered entropy", function(done
) {
2683 testEntropyFeedback(done
,
2688 bitsPerEvent: "4.80",
2692 it("Shows details about the entered entropy", function(done
) {
2693 testEntropyFeedback(done
,
2698 bitsPerEvent: "5.01",
2702 it("Shows details about the entered entropy", function(done
) {
2703 testEntropyFeedback(done
,
2705 entropy: "3d3d3d3d",
2708 bitsPerEvent: "5.14",
2712 it("Shows details about the entered entropy", function(done
) {
2713 testEntropyFeedback(done
,
2715 entropy: "3d3d3d3d3d",
2718 bitsPerEvent: "5.22",
2722 it("Shows details about the entered entropy", function(done
) {
2723 testEntropyFeedback(done
,
2725 entropy: "3d3d3d3d3d3d",
2728 bitsPerEvent: "5.28",
2732 it("Shows details about the entered entropy", function(done
) {
2733 testEntropyFeedback(done
,
2735 entropy: "3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d",
2738 bitsPerEvent: "5.59",
2739 strength: 'less than a second - Repeats like "abcabcabc" are only slightly harder to guess than "abc"',
2744 // Entropy is truncated from the left
2745 it('Truncates entropy from the left', function(done
) {
2746 // Truncate from left means 0000 is removed from the start
2747 // which gives mnemonic 'avocado zoo zone'
2748 // not 1111 removed from the end
2749 // which gives the mnemonic 'abstract zoo zoo'
2750 var entropy
= "00000000 00000000 00000000 00000000";
2751 entropy
+= "11111111 11111111 11111111 1111"; // Missing last byte
2752 driver
.findElement(By
.css('.use-entropy'))
2754 driver
.findElement(By
.css('.entropy'))
2756 driver
.sleep(generateDelay
).then(function() {
2757 driver
.findElement(By
.css(".phrase"))
2758 .getAttribute("value").then(function(phrase
) {
2759 expect(phrase
).toBe("avocado zoo zone");
2765 // Very large entropy results in very long mnemonics
2766 it('Converts very long entropy to very long mnemonics', function(done
) {
2768 for (var i
=0; i
<33; i
++) {
2769 entropy
+= "AAAAAAAA"; // 3 words * 33 iterations = 99 words
2771 driver
.findElement(By
.css('.use-entropy'))
2773 driver
.findElement(By
.css('.entropy'))
2775 driver
.sleep(generateDelay
).then(function() {
2776 driver
.findElement(By
.css(".phrase"))
2777 .getAttribute("value").then(function(phrase
) {
2778 var wordCount
= phrase
.split(/\s+/g).length
;
2779 expect(wordCount
).toBe(99);
2785 // Is compatible with bip32jp entropy
2786 // https://bip32jp.github.io/english/index.html
2788 // Is incompatible with:
2790 it('Is compatible with bip32jp.github.io', function(done
) {
2791 var entropy
= "543210543210543210543210543210543210543210543210543210543210543210543210543210543210543210543210543";
2792 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";
2793 driver
.findElement(By
.css('.use-entropy'))
2795 driver
.findElement(By
.css('.entropy'))
2797 driver
.sleep(generateDelay
).then(function() {
2798 driver
.findElement(By
.css(".phrase"))
2799 .getAttribute("value").then(function(phrase
) {
2800 expect(phrase
).toBe(expectedPhrase
);
2806 // Blank entropy does not generate mnemonic or addresses
2807 it('Does not generate mnemonic for blank entropy', function(done
) {
2808 driver
.findElement(By
.css('.use-entropy'))
2810 driver
.findElement(By
.css('.entropy'))
2812 // check there is no mnemonic
2813 driver
.sleep(generateDelay
).then(function() {
2814 driver
.findElement(By
.css(".phrase"))
2815 .getAttribute("value").then(function(phrase
) {
2816 expect(phrase
).toBe("");
2817 // check there is no mnemonic
2818 driver
.findElements(By
.css(".address"))
2819 .then(function(addresses
) {
2820 expect(addresses
.length
).toBe(0);
2821 // Check the feedback says 'blank entropy'
2822 driver
.findElement(By
.css(".feedback"))
2824 .then(function(feedbackText
) {
2825 expect(feedbackText
).toBe("Blank entropy");
2833 // Mnemonic length can be selected even for weak entropy
2834 it('Allows selection of mnemonic length even for weak entropy', function(done
) {
2835 driver
.findElement(By
.css('.use-entropy'))
2837 driver
.executeScript(function() {
2838 $(".mnemonic-length").val("18").trigger("change");
2840 driver
.findElement(By
.css('.entropy'))
2841 .sendKeys("012345");
2842 driver
.sleep(generateDelay
).then(function() {
2843 driver
.findElement(By
.css(".phrase"))
2844 .getAttribute("value").then(function(phrase
) {
2845 var wordCount
= phrase
.split(/\s+/g).length
;
2846 expect(wordCount
).toBe(18);
2853 // https://github.com/iancoleman/bip39/issues/33
2854 // Final cards should contribute entropy
2855 it('Uses as much entropy as possible for the mnemonic', function(done
) {
2856 driver
.findElement(By
.css('.use-entropy'))
2858 driver
.findElement(By
.css('.entropy'))
2859 .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");
2860 driver
.sleep(generateDelay
).then(function() {
2862 driver
.findElement(By
.css(".phrase"))
2863 .getAttribute("value").then(function(originalPhrase
) {
2864 // Set the last 12 cards to be AS
2865 driver
.findElement(By
.css('.entropy'))
2867 driver
.findElement(By
.css('.entropy'))
2868 .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");
2869 driver
.sleep(generateDelay
).then(function() {
2871 driver
.findElement(By
.css(".phrase"))
2872 .getAttribute("value").then(function(newPhrase
) {
2873 expect(originalPhrase
).not
.toEqual(newPhrase
);
2882 // https://github.com/iancoleman/bip39/issues/35
2884 // TODO this doesn't work in selenium with firefox
2885 // see https://stackoverflow.com/q/40360223
2886 it('Shows a qr code on hover for the phrase', function(done
) {
2887 if (browser
== "firefox") {
2888 pending("Selenium + Firefox bug for mouseMove, see https://stackoverflow.com/q/40360223");
2890 // generate a random mnemonic
2891 var generateEl
= driver
.findElement(By
.css('.generate'));
2893 // toggle qr to show (hidden by default)
2894 var phraseEl
= driver
.findElement(By
.css(".phrase"));
2896 var rootKeyEl
= driver
.findElement(By
.css(".root-key"));
2897 driver
.sleep(generateDelay
).then(function() {
2898 // hover over the root key
2899 driver
.actions().mouseMove(rootKeyEl
).perform().then(function() {
2900 // check the qr code shows
2901 driver
.executeScript(function() {
2902 return $(".qr-container").find("canvas").length
> 0;
2904 .then(function(qrShowing
) {
2905 expect(qrShowing
).toBe(true);
2906 // hover away from the phrase
2907 driver
.actions().mouseMove(generateEl
).perform().then(function() {;
2908 // check the qr code hides
2909 driver
.executeScript(function() {
2910 return $(".qr-container").find("canvas").length
== 0;
2912 .then(function(qrHidden
) {
2913 expect(qrHidden
).toBe(true);
2922 // BIP44 account extendend private key is shown
2923 // github issue 37 - compatibility with electrum
2924 it('Shows the bip44 account extended private key', function(done
) {
2925 driver
.findElement(By
.css(".phrase"))
2926 .sendKeys("abandon abandon ability");
2927 driver
.sleep(generateDelay
).then(function() {
2928 driver
.findElement(By
.css("#bip44 .account-xprv"))
2929 .getAttribute("value")
2930 .then(function(xprv
) {
2931 expect(xprv
).toBe("xprv9yzrnt4zWVJUr1k2VxSPy9ettKz5PpeDMgaVG7UKedhqnw1tDkxP2UyYNhuNSumk2sLE5ctwKZs9vwjsq3e1vo9egCK6CzP87H2cVYXpfwQ");
2937 // BIP44 account extendend public key is shown
2938 // github issue 37 - compatibility with electrum
2939 it('Shows the bip44 account extended public key', function(done
) {
2940 driver
.findElement(By
.css(".phrase"))
2941 .sendKeys("abandon abandon ability");
2942 driver
.sleep(generateDelay
).then(function() {
2943 driver
.findElement(By
.css("#bip44 .account-xpub"))
2944 .getAttribute("value")
2945 .then(function(xprv
) {
2946 expect(xprv
).toBe("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2953 // BIP32 root key can be set as an xpub
2954 it('Generates addresses from xpub as bip32 root key', function(done
) {
2955 driver
.findElement(By
.css('#bip32-tab a'))
2957 // set xpub for account 0 of bip44 for 'abandon abandon ability'
2958 driver
.findElement(By
.css("#root-key"))
2959 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2960 driver
.sleep(generateDelay
).then(function() {
2961 // check the addresses are generated
2962 getFirstAddress(function(address
) {
2963 expect(address
).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug");
2964 // check the xprv key is not set
2965 driver
.findElement(By
.css(".extended-priv-key"))
2966 .getAttribute("value")
2967 .then(function(xprv
) {
2968 expect(xprv
).toBe("NA");
2969 // check the private key is not set
2970 driver
.findElements(By
.css(".privkey"))
2971 .then(function(els
) {
2974 .then(function(privkey
) {
2975 expect(xprv
).toBe("NA");
2985 // xpub for bip32 root key will not work with hardened derivation paths
2986 it('Shows error for hardened derivation paths with xpub root key', function(done
) {
2987 // set xpub for account 0 of bip44 for 'abandon abandon ability'
2988 driver
.findElement(By
.css("#root-key"))
2989 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2990 driver
.sleep(feedbackDelay
).then(function() {
2991 // Check feedback is correct
2992 driver
.findElement(By
.css('.feedback'))
2994 .then(function(feedback
) {
2995 var msg
= "Hardened derivation path is invalid with xpub key";
2996 expect(feedback
).toBe(msg
);
2997 // Check no addresses are shown
2998 driver
.findElements(By
.css('.addresses tr'))
2999 .then(function(rows
) {
3000 expect(rows
.length
).toBe(0);
3008 // no root key shows feedback
3009 it('Shows feedback for no root key', function(done
) {
3010 // set xpub for account 0 of bip44 for 'abandon abandon ability'
3011 driver
.findElement(By
.css('#bip32-tab a'))
3013 driver
.sleep(feedbackDelay
).then(function() {
3014 // Check feedback is correct
3015 driver
.findElement(By
.css('.feedback'))
3017 .then(function(feedback
) {
3018 expect(feedback
).toBe("Invalid root key");
3025 // display error switching tabs while addresses are generating
3026 it('Can change details while old addresses are still being generated', function(done
) {
3027 // Set to generate 199 more addresses.
3028 // This will take a long time allowing a new set of addresses to be
3029 // generated midway through this lot.
3030 // The newly generated addresses should not include any from the old set.
3031 // Any more than 199 will show an alert which needs to be accepted.
3032 driver
.findElement(By
.css('.rows-to-add'))
3034 driver
.findElement(By
.css('.rows-to-add'))
3037 driver
.findElement(By
.css('.phrase'))
3038 .sendKeys("abandon abandon ability");
3039 driver
.sleep(generateDelay
).then(function() {
3040 // change tabs which should cancel the previous generating
3041 driver
.findElement(By
.css('.rows-to-add'))
3043 driver
.findElement(By
.css('.rows-to-add'))
3045 driver
.findElement(By
.css('#bip32-tab a'))
3047 driver
.sleep(generateDelay
).then(function() {
3048 driver
.findElements(By
.css('.index'))
3049 .then(function(els
) {
3050 // check the derivation paths have the right quantity
3051 expect(els
.length
).toBe(20);
3052 // check the derivation paths are in order
3053 testRowsAreInCorrectOrder(done
);
3057 }, generateDelay
+ 10000);
3060 // padding for binary should give length with multiple of 256
3061 // hashed entropy 1111 is length 252, so requires 4 leading zeros
3062 // prior to issue 49 it would only generate 2 leading zeros, ie missing 2
3063 it('Pads hashed entropy with leading zeros', function(done
) {
3064 driver
.findElement(By
.css('.use-entropy'))
3066 driver
.executeScript(function() {
3067 $(".mnemonic-length").val("15").trigger("change");
3069 driver
.findElement(By
.css('.entropy'))
3071 driver
.sleep(generateDelay
).then(function() {
3072 driver
.findElement(By
.css('.phrase'))
3073 .getAttribute("value")
3074 .then(function(phrase
) {
3075 expect(phrase
).toBe("avocado valid quantum cross link predict excuse edit street able flame large galaxy ginger nuclear");
3081 // Github pull request 55
3082 // https://github.com/iancoleman/bip39/pull/55
3084 it('Can set the derivation path on bip32 tab for bitcoincore', function(done
) {
3085 testClientSelect(done
, {
3087 bip32path: "m/0'/0'",
3088 useHardenedAddresses: "true",
3091 it('Can set the derivation path on bip32 tab for multibit', function(done
) {
3092 testClientSelect(done
, {
3094 bip32path: "m/0'/0",
3095 useHardenedAddresses: null,
3098 it('Can set the derivation path on bip32 tab for coinomi/ledger', function(done
) {
3099 testClientSelect(done
, {
3101 bip32path: "m/44'/0'/0'",
3102 useHardenedAddresses: null,
3107 // https://github.com/iancoleman/bip39/issues/58
3108 // bip32 derivation is correct, does not drop leading zeros
3110 // https://medium.com/@alexberegszaszi/why-do-my-bip32-wallets-disagree-6f3254cc5846
3111 it('Retains leading zeros for bip32 derivation', function(done
) {
3112 driver
.findElement(By
.css(".phrase"))
3113 .sendKeys("fruit wave dwarf banana earth journey tattoo true farm silk olive fence");
3114 driver
.findElement(By
.css(".passphrase"))
3115 .sendKeys("banana");
3116 driver
.sleep(generateDelay
).then(function() {
3117 getFirstAddress(function(address
) {
3118 // Note that bitcore generates an incorrect address
3119 // 13EuKhffWkBE2KUwcbkbELZb1MpzbimJ3Y
3120 // see the medium.com link above for more details
3121 expect(address
).toBe("17rxURoF96VhmkcEGCj5LNQkmN9HVhWb7F");
3128 // Japanese mnemonics generate incorrect bip32 seed
3129 // BIP39 seed is set from phrase
3130 it('Generates correct seed for Japanese mnemonics', function(done
) {
3131 driver
.findElement(By
.css(".phrase"))
3132 .sendKeys("あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あおぞら");
3133 driver
.findElement(By
.css(".passphrase"))
3134 .sendKeys("メートルガバヴァぱばぐゞちぢ十人十色");
3135 driver
.sleep(generateDelay
).then(function() {
3136 driver
.findElement(By
.css(".seed"))
3137 .getAttribute("value")
3138 .then(function(seed
) {
3139 expect(seed
).toBe("a262d6fb6122ecf45be09c50492b31f92e9beb7d9a845987a02cefda57a15f9c467a17872029a9e92299b5cbdf306e3a0ee620245cbd508959b6cb7ca637bd55");
3145 // BIP49 official test vectors
3146 // https://github.com/bitcoin/bips/blob/master/bip-0049.mediawiki#test-vectors
3147 it('Generates BIP49 addresses matching the official test vectors', function(done
) {
3148 driver
.findElement(By
.css('#bip49-tab a'))
3150 selectNetwork("BTC - Bitcoin Testnet");
3151 driver
.findElement(By
.css(".phrase"))
3152 .sendKeys("abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about");
3153 driver
.sleep(generateDelay
).then(function() {
3154 getFirstAddress(function(address
) {
3155 expect(address
).toBe("2Mww8dCYPUpKHofjgcXcBCEGmniw9CoaiD2");
3161 // BIP49 derivation path is shown
3162 it('Shows the bip49 derivation path', function(done
) {
3163 driver
.findElement(By
.css('#bip49-tab a'))
3165 driver
.findElement(By
.css(".phrase"))
3166 .sendKeys("abandon abandon ability");
3167 driver
.sleep(generateDelay
).then(function() {
3168 driver
.findElement(By
.css('#bip49 .path'))
3169 .getAttribute("value")
3170 .then(function(path
) {
3171 expect(path
).toBe("m/49'/0'/0'/0");
3177 // BIP49 extended private key is shown
3178 it('Shows the bip49 extended private key', function(done
) {
3179 driver
.findElement(By
.css('#bip49-tab a'))
3181 driver
.findElement(By
.css(".phrase"))
3182 .sendKeys("abandon abandon ability");
3183 driver
.sleep(generateDelay
).then(function() {
3184 driver
.findElement(By
.css('.extended-priv-key'))
3185 .getAttribute("value")
3186 .then(function(xprv
) {
3187 expect(xprv
).toBe("yprvALYB4DYRG6CzzVgzQZwwqjAA2wjBGC3iEd7KYYScpoDdmf75qMRWZWxoFcRXBJjgEXdFqJ9vDRGRLJQsrL22Su5jMbNFeM9vetaGVqy9Qy2");
3193 // BIP49 extended public key is shown
3194 it('Shows the bip49 extended public key', function(done
) {
3195 driver
.findElement(By
.css('#bip49-tab a'))
3197 driver
.findElement(By
.css(".phrase"))
3198 .sendKeys("abandon abandon ability");
3199 driver
.sleep(generateDelay
).then(function() {
3200 driver
.findElement(By
.css('.extended-pub-key'))
3201 .getAttribute("value")
3202 .then(function(xprv
) {
3203 expect(xprv
).toBe("ypub6ZXXTj5K6TmJCymTWbUxCs6tayZffemZbr2vLvrEP8kceTSENtjm7KHH6thvAKxVar9fGe8rgsPEX369zURLZ68b4f7Vexz7RuXsjQ69YDt");
3209 // BIP49 account field changes address list
3210 it('Can set the bip49 account field', function(done
) {
3211 driver
.findElement(By
.css('#bip49-tab a'))
3213 driver
.findElement(By
.css('#bip49 .account'))
3215 driver
.findElement(By
.css('#bip49 .account'))
3217 driver
.findElement(By
.css(".phrase"))
3218 .sendKeys("abandon abandon ability");
3219 driver
.sleep(generateDelay
).then(function() {
3220 getFirstAddress(function(address
) {
3221 expect(address
).toBe("381wg1GGN4rP88rNC9v7QWsiww63yLVPsn");
3227 // BIP49 change field changes address list
3228 it('Can set the bip49 change field', function(done
) {
3229 driver
.findElement(By
.css('#bip49-tab a'))
3231 driver
.findElement(By
.css('#bip49 .change'))
3233 driver
.findElement(By
.css('#bip49 .change'))
3235 driver
.findElement(By
.css(".phrase"))
3236 .sendKeys("abandon abandon ability");
3237 driver
.sleep(generateDelay
).then(function() {
3238 getFirstAddress(function(address
) {
3239 expect(address
).toBe("3PEM7MiKed5konBoN66PQhK8r3hjGhy9dT");
3245 // BIP49 account extendend private key is shown
3246 it('Shows the bip49 account extended private key', function(done
) {
3247 driver
.findElement(By
.css('#bip49-tab a'))
3249 driver
.findElement(By
.css(".phrase"))
3250 .sendKeys("abandon abandon ability");
3251 driver
.sleep(generateDelay
).then(function() {
3252 driver
.findElement(By
.css('#bip49 .account-xprv'))
3253 .getAttribute("value")
3254 .then(function(xprv
) {
3255 expect(xprv
).toBe("yprvAHtB1M5Wp675aLzFy9TJYK2mSsLkg6mcBRh5DZTR7L4EnYSmYPaL63KFA4ycg1PngW5LfkmejxzosCs17TKZMpRFKc3z5SJar6QAKaFcaZL");
3261 // BIP49 account extendend public key is shown
3262 it('Shows the bip49 account extended public key', function(done
) {
3263 driver
.findElement(By
.css('#bip49-tab a'))
3265 driver
.findElement(By
.css(".phrase"))
3266 .sendKeys("abandon abandon ability");
3267 driver
.sleep(generateDelay
).then(function() {
3268 driver
.findElement(By
.css('#bip49 .account-xpub'))
3269 .getAttribute("value")
3270 .then(function(xprv
) {
3271 expect(xprv
).toBe("ypub6WsXQrcQeTfNnq4j5AzJuSyVzuBF5ZVTYecg1ws2ffbDfLmv5vtadqdj1NgR6C6gufMpMfJpHxvb6JEQKvETVNWCRanNedfJtnTchZiJtsL");
3277 // Test selecting coin where bip49 is unavailable (eg CLAM)
3278 it('Shows an error on bip49 tab for coins without bip49', function(done
) {
3279 driver
.findElement(By
.css('#bip49-tab a'))
3281 driver
.findElement(By
.css(".phrase"))
3282 .sendKeys("abandon abandon ability");
3283 driver
.sleep(generateDelay
).then(function() {
3284 selectNetwork("CLAM - Clams");
3285 // bip49 available is hidden
3286 driver
.findElement(By
.css('#bip49 .available'))
3287 .getAttribute("class")
3288 .then(function(classes
) {
3289 expect(classes
).toContain("hidden");
3290 // bip49 unavailable is shown
3291 driver
.findElement(By
.css('#bip49 .unavailable'))
3292 .getAttribute("class")
3293 .then(function(classes
) {
3294 expect(classes
).not
.toContain("hidden");
3295 // check there are no addresses shown
3296 driver
.findElements(By
.css('.addresses tr'))
3297 .then(function(rows
) {
3298 expect(rows
.length
).toBe(0);
3299 // check the derived private key is blank
3300 driver
.findElement(By
.css('.extended-priv-key'))
3301 .getAttribute("value")
3302 .then(function(xprv
) {
3303 expect(xprv
).toBe('');
3304 // check the derived public key is blank
3305 driver
.findElement(By
.css('.extended-pub-key'))
3306 .getAttribute("value")
3307 .then(function(xpub
) {
3308 expect(xpub
).toBe('');
3319 // Cleared mnemonic and root key still allows addresses to be generated
3320 // https://github.com/iancoleman/bip39/issues/43
3321 it('Clears old root keys from memory when mnemonic is cleared', function(done
) {
3323 driver
.findElement(By
.css(".phrase"))
3324 .sendKeys("abandon abandon ability");
3325 driver
.sleep(generateDelay
).then(function() {
3326 // clear the mnemonic and root key
3327 // using selenium .clear() doesn't seem to trigger the 'input' event
3328 // so clear it using keys instead
3329 driver
.findElement(By
.css('.phrase'))
3330 .sendKeys(Key
.CONTROL
,"a");
3331 driver
.findElement(By
.css('.phrase'))
3332 .sendKeys(Key
.DELETE
);
3333 driver
.findElement(By
.css('.root-key'))
3334 .sendKeys(Key
.CONTROL
,"a");
3335 driver
.findElement(By
.css('.root-key'))
3336 .sendKeys(Key
.DELETE
);
3337 driver
.sleep(generateDelay
).then(function() {
3338 // try to generate more addresses
3339 driver
.findElement(By
.css('.more'))
3341 driver
.sleep(generateDelay
).then(function() {
3342 driver
.findElements(By
.css(".addresses tr"))
3343 .then(function(els
) {
3344 // check there are no addresses shown
3345 expect(els
.length
).toBe(0);
3354 // error trying to generate addresses from xpub with hardened derivation
3355 it('Shows error for hardened addresses with xpub root key', function(done
) {
3356 driver
.findElement(By
.css('#bip32-tab a'))
3358 driver
.executeScript(function() {
3359 $(".hardened-addresses").prop("checked", true);
3361 // set xpub for account 0 of bip44 for 'abandon abandon ability'
3362 driver
.findElement(By
.css("#root-key"))
3363 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
3364 driver
.sleep(feedbackDelay
).then(function() {
3365 // Check feedback is correct
3366 driver
.findElement(By
.css('.feedback'))
3368 .then(function(feedback
) {
3369 var msg
= "Hardened derivation path is invalid with xpub key";
3370 expect(feedback
).toBe(msg
);
3376 // Litecoin uses ltub by default, and can optionally be set to xprv
3378 // https://github.com/iancoleman/bip39/issues/96
3379 // Issue with extended keys on Litecoin
3380 it('Uses ltub by default for litecoin, but can be set to xprv', function(done
) {
3381 driver
.findElement(By
.css('.phrase'))
3382 .sendKeys("abandon abandon ability");
3383 selectNetwork("LTC - Litecoin");
3384 driver
.sleep(generateDelay
).then(function() {
3385 // check the extended key is generated correctly
3386 driver
.findElement(By
.css('.root-key'))
3387 .getAttribute("value")
3388 .then(function(rootKey
) {
3389 expect(rootKey
).toBe("Ltpv71G8qDifUiNesiPqf6h5V6eQ8ic77oxQiYtawiACjBEx3sTXNR2HGDGnHETYxESjqkMLFBkKhWVq67ey1B2MKQXannUqNy1RZVHbmrEjnEU");
3390 // set litecoin to use ltub
3391 driver
.executeScript(function() {
3392 $(".litecoin-use-ltub").prop("checked", false);
3393 $(".litecoin-use-ltub").trigger("change");
3395 driver
.sleep(generateDelay
).then(function() {
3396 driver
.findElement(By
.css('.root-key'))
3397 .getAttribute("value")
3398 .then(function(rootKey
) {
3399 expect(rootKey
).toBe("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
3408 // https://github.com/iancoleman/bip39/issues/99#issuecomment-327094159
3409 // "warn me emphatically when they have detected invalid input" to the entropy field
3410 // A warning is shown when entropy is filtered and discarded
3411 it('Warns when entropy is filtered and discarded', function(done
) {
3412 driver
.findElement(By
.css('.use-entropy'))
3414 // set entropy to have no filtered content
3415 driver
.findElement(By
.css('.entropy'))
3416 .sendKeys("00000000 00000000 00000000 00000000");
3417 driver
.sleep(generateDelay
).then(function() {
3418 // check the filter warning does not show
3419 driver
.findElement(By
.css('.entropy-container .filter-warning'))
3420 .getAttribute("class")
3421 .then(function(classes
) {
3422 expect(classes
).toContain("hidden");
3423 // set entropy to have some filtered content
3424 driver
.findElement(By
.css('.entropy'))
3425 .sendKeys("10000000 zxcvbn 00000000 00000000 00000000");
3426 driver
.sleep(entropyFeedbackDelay
).then(function() {
3427 // check the filter warning shows
3428 driver
.findElement(By
.css('.entropy-container .filter-warning'))
3429 .getAttribute("class")
3430 .then(function(classes
) {
3431 expect(classes
).not
.toContain("hidden");
3439 // Bitcoin Cash address can be set to use cashaddr format
3440 it('Can use cashaddr format for bitcoin cash addresses', function(done
) {
3441 driver
.executeScript(function() {
3442 $(".use-bch-cashaddr-addresses").prop("checked", true);
3444 driver
.findElement(By
.css('.phrase'))
3445 .sendKeys("abandon abandon ability");
3446 selectNetwork("BCH - Bitcoin Cash");
3447 driver
.sleep(generateDelay
).then(function() {
3448 getFirstAddress(function(address
) {
3449 expect(address
).toBe("bitcoincash:qzlquk7w4hkudxypl4fgv8x279r754dkvur7jpcsps");
3455 // Bitcoin Cash address can be set to use bitpay format
3456 it('Can use bitpay format for bitcoin cash addresses', function(done
) {
3457 driver
.executeScript(function() {
3458 $(".use-bch-bitpay-addresses").prop("checked", true);
3460 driver
.findElement(By
.css('.phrase'))
3461 .sendKeys("abandon abandon ability");
3462 selectNetwork("BCH - Bitcoin Cash");
3463 driver
.sleep(generateDelay
).then(function() {
3464 getFirstAddress(function(address
) {
3465 expect(address
).toBe("CZnpA9HPmvhuhLLPWJP8rNDpLUYXy1LXFk");
3471 // Bitcoin Cash address can be set to use legacy format
3472 it('Can use legacy format for bitcoin cash addresses', function(done
) {
3473 driver
.executeScript(function() {
3474 $(".use-bch-legacy-addresses").prop("checked", true);
3476 driver
.findElement(By
.css('.phrase'))
3477 .sendKeys("abandon abandon ability");
3478 selectNetwork("BCH - Bitcoin Cash");
3479 driver
.sleep(generateDelay
).then(function() {
3480 getFirstAddress(function(address
) {
3481 expect(address
).toBe("1JKvb6wKtsjNoCRxpZ4DGrbniML7z5U16A");
3487 // End of tests ported from old suit, so no more comments above each test now
3489 it('Can generate more addresses from a custom index', function(done
) {
3490 var expectedIndexes
= [
3491 0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,
3492 40,41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59
3494 driver
.findElement(By
.css('.phrase'))
3495 .sendKeys("abandon abandon ability");
3496 driver
.sleep(generateDelay
).then(function() {
3497 // Set start of next lot of rows to be from index 40
3498 // which means indexes 20-39 will not be in the table.
3499 driver
.findElement(By
.css('.more-rows-start-index'))
3501 driver
.findElement(By
.css('.more'))
3503 driver
.sleep(generateDelay
).then(function() {
3504 // Check actual indexes in the table match the expected pattern
3505 driver
.findElements(By
.css(".index"))
3506 .then(function(els
) {
3507 expect(els
.length
).toBe(expectedIndexes
.length
);
3508 var testRowAtIndex = function(i
) {
3509 if (i
>= expectedIndexes
.length
) {
3514 .then(function(actualPath
) {
3515 var noHardened
= actualPath
.replace(/'/g, "");
3516 var pathBits = noHardened.split("/")
3517 var lastBit = pathBits[pathBits.length-1];
3518 var actualIndex = parseInt(lastBit);
3519 var expectedIndex = expectedIndexes[i];
3520 expect(actualIndex).toBe(expectedIndex);
3521 testRowAtIndex(i+1);
3531 it('Can generate BIP141 addresses
with P2WPKH
-in-P2SH semanitcs
', function(done) {
3532 // Sourced from BIP49 official test specs
3533 driver.findElement(By.css('#bip141
-tab a
'))
3535 driver.findElement(By.css('.bip141
-path
'))
3537 driver.findElement(By.css('.bip141
-path
'))
3538 .sendKeys("m/49'/1'/0'/0");
3539 selectNetwork("BTC
- Bitcoin Testnet
");
3540 driver.findElement(By.css(".phrase
"))
3541 .sendKeys("abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about
");
3542 driver.sleep(generateDelay).then(function() {
3543 getFirstAddress(function(address) {
3544 expect(address).toBe("2Mww8dCYPUpKHofjgcXcBCEGmniw9CoaiD2
");
3550 it('Can generate BIP141 addresses with P2WPKH semanitcs', function(done) {
3551 // This result tested against bitcoinjs-lib test spec for segwit address
3552 // using the first private key of this mnemonic and default path m/0
3553 // https://github.com/bitcoinjs/bitcoinjs-lib/blob/9c8503cab0c6c30a95127042703bc18e8d28c76d/test/integration/addresses.js#L50
3554 // so whilst not directly comparable, substituting the private key produces
3555 // identical results between this tool and the bitcoinjs-lib test.
3556 // Private key generated is:
3557 // L3L8Nu9whawPBNLGtFqDhKut9DKKfG3CQoysupT7BimqVCZsLFNP
3558 driver.findElement(By.css('#bip141-tab a'))
3561 driver.executeScript(function() {
3562 $(".bip141
-semantics option
[selected
]").removeAttr("selected
");
3563 $(".bip141
-semantics option
").filter(function(i,e) {
3564 return $(e).html() == "P2WPKH
";
3565 }).prop("selected
", true);
3566 $(".bip141
-semantics
").trigger("change
");
3568 driver.findElement(By.css(".phrase
"))
3569 .sendKeys("abandon abandon ability
");
3570 driver.sleep(generateDelay).then(function() {
3571 getFirstAddress(function(address) {
3572 expect(address).toBe("bc1qfwu6a5a3evygrk8zvdxxvz4547lmpyx5vsfxe9
");
3578 it('Shows the entropy used by the PRNG when clicking generate', function(done) {
3579 driver.findElement(By.css('.generate')).click();
3580 driver.sleep(generateDelay).then(function() {
3581 driver.findElement(By.css('.entropy'))
3582 .getAttribute("value
")
3583 .then(function(entropy) {
3584 expect(entropy).not.toBe("");
3590 it('Shows the index of each word in the mnemonic', function(done) {
3591 driver.findElement(By.css('.phrase'))
3592 .sendKeys("abandon abandon ability
");
3593 driver.sleep(generateDelay).then(function() {
3594 driver.findElement(By.css('.use-entropy'))
3596 driver.findElement(By.css('.word-indexes'))
3598 .then(function(indexes) {
3599 expect(indexes).toBe("0, 0, 1");
3605 it('Shows the derivation path for bip84 tab', function(done) {
3606 driver.findElement(By.css('#bip84-tab a'))
3608 driver.findElement(By.css('.phrase'))
3609 .sendKeys('abandon abandon ability');
3610 driver.sleep(generateDelay).then(function() {
3611 driver.findElement(By.css('#bip84 .path'))
3612 .getAttribute("value
")
3613 .then(function(path) {
3614 expect(path).toBe("m
/84'/0'/0'/0");
3620 it('Shows the extended private key for bip84 tab', function(done) {
3621 driver.findElement(By.css('#bip84-tab a'))
3623 driver.findElement(By.css('.phrase'))
3624 .sendKeys('abandon abandon ability');
3625 driver.sleep(generateDelay).then(function() {
3626 driver.findElement(By.css('.extended-priv-key'))
3627 .getAttribute("value
")
3628 .then(function(path) {
3629 expect(path).toBe("zprvAev3RKrZ3QVKiUFCfdeMRen1BPDJgdNt1XpxiDy8acSs4kkAGTCvq7HeRYRNNpo8EtEjCFQBWavJwtCUR29y4TUCH4X5RXMcyq48uN8y9BP
");
3635 it('Shows the extended public key for bip84 tab', function(done) {
3636 driver.findElement(By.css('#bip84-tab a'))
3638 driver.findElement(By.css('.phrase'))
3639 .sendKeys('abandon abandon ability');
3640 driver.sleep(generateDelay).then(function() {
3641 driver.findElement(By.css('.extended-pub-key'))
3642 .getAttribute("value
")
3643 .then(function(path) {
3644 expect(path).toBe("zpub6suPpqPSsn3cvxKfmfBMnnijjR3o666jNkkZWcNk8wyqwZ5JozXBNuc8Gs7DB3uLwTDvGVTspVEAUQcEjKF3pZHgywVbubdTqbXTUg7usyx
");
3650 it('Changes the address list if bip84 account is changed', function(done) {
3651 driver.findElement(By.css('#bip84-tab a'))
3653 driver.findElement(By.css('#bip84 .account'))
3655 driver.findElement(By.css('.phrase'))
3656 .sendKeys('abandon abandon ability');
3657 driver.sleep(generateDelay).then(function() {
3658 getFirstAddress(function(address) {
3659 expect(address).toBe("bc1qp7vv669t2fy965jdzvqwrraana89ctd5ewc662
");
3665 it('Changes the address list if bip84 change is changed', function(done) {
3666 driver.findElement(By.css('#bip84-tab a'))
3668 driver.findElement(By.css('#bip84 .change'))
3670 driver.findElement(By.css('.phrase'))
3671 .sendKeys('abandon abandon ability');
3672 driver.sleep(generateDelay).then(function() {
3673 getFirstAddress(function(address) {
3674 expect(address).toBe("bc1qr39vj6rh06ff05m53uxq8uazehwhccswylhrs2
");
3680 it('Passes the official BIP84 test spec for rootpriv', function(done) {
3681 driver.findElement(By.css('#bip84-tab a'))
3683 driver.findElement(By.css('.phrase'))
3684 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3685 driver.sleep(generateDelay).then(function() {
3686 driver.findElement(By.css(".root
-key
"))
3687 .getAttribute("value
")
3688 .then(function(rootKey) {
3689 expect(rootKey).toBe("zprvAWgYBBk7JR8Gjrh4UJQ2uJdG1r3WNRRfURiABBE3RvMXYSrRJL62XuezvGdPvG6GFBZduosCc1YP5wixPox7zhZLfiUm8aunE96BBa4Kei5
");
3695 it('Passes the official BIP84 test spec for account 0 xprv', function(done) {
3696 driver.findElement(By.css('#bip84-tab a'))
3698 driver.findElement(By.css('.phrase'))
3699 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3700 driver.sleep(generateDelay).then(function() {
3701 driver.findElement(By.css("#bip84
.account
-xprv
"))
3702 .getAttribute("value
")
3703 .then(function(rootKey) {
3704 expect(rootKey).toBe("zprvAdG4iTXWBoARxkkzNpNh8r6Qag3irQB8PzEMkAFeTRXxHpbF9z4QgEvBRmfvqWvGp42t42nvgGpNgYSJA9iefm1yYNZKEm7z6qUWCroSQnE
");
3710 it('Passes the official BIP84 test spec for account 0 xpub', function(done) {
3711 driver.findElement(By.css('#bip84-tab a'))
3713 driver.findElement(By.css('.phrase'))
3714 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3715 driver.sleep(generateDelay).then(function() {
3716 driver.findElement(By.css("#bip84
.account
-xpub
"))
3717 .getAttribute("value
")
3718 .then(function(rootKey) {
3719 expect(rootKey).toBe("zpub6rFR7y4Q2AijBEqTUquhVz398htDFrtymD9xYYfG1m4wAcvPhXNfE3EfH1r1ADqtfSdVCToUG868RvUUkgDKf31mGDtKsAYz2oz2AGutZYs
");
3725 it('Passes the official BIP84 test spec for account 0 first address', function(done) {
3726 driver.findElement(By.css('#bip84-tab a'))
3728 driver.findElement(By.css('.phrase'))
3729 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3730 driver.sleep(generateDelay).then(function() {
3731 getFirstAddress(function(address) {
3732 expect(address).toBe("bc1qcr8te4kr609gcawutmrza0j4xv80jy8z306fyu
");
3738 it('Passes the official BIP84 test spec for account 0 first change address', function(done) {
3739 driver.findElement(By.css('#bip84-tab a'))
3741 driver.findElement(By.css('.phrase'))
3742 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3743 driver.findElement(By.css('#bip84 .change'))
3745 driver.sleep(generateDelay).then(function() {
3746 getFirstAddress(function(address) {
3747 expect(address).toBe("bc1q8c6fshw2dlwun7ekn9qwf37cu2rn755upcp6el
");
3753 it('Can display the table as csv', function(done) {
3754 var headings = "path
,address
,public key
,private key
";
3755 var row1 = "m
/44'/0'/0'/0/0,1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug
,033f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3
,L26cVSpWFkJ6aQkPkKmTzLqTdLJ923e6CzrVh9cmx21QHsoUmrEE
";
3756 var row20 = "m
/44'/0'/0'/0/19,1KhBy28XLAciXnnRvm71PvQJaETyrxGV55
,02b4b3e396434d8cdd20c03ac4aaa07387784d5d867b75987f516f5705ee68cb3a
,L4GrDrjReMsCAu5DkLXn79jSb95qR7Zfx7eshybCQZ1qL32MXJab
";
3757 driver.findElement(By.css('.phrase'))
3758 .sendKeys('abandon abandon ability');
3759 driver.sleep(generateDelay).then(function() {
3760 driver.findElement(By.css('.csv'))
3761 .getAttribute("value
")
3762 .then(function(csv) {
3763 expect(csv).toContain(headings);
3764 expect(csv).toContain(row1);
3765 expect(csv).toContain(row20);
3771 it('LeftPads ethereum keys that are less than 32 bytes', function(done) {
3772 // see https://github.com/iancoleman/bip39/issues/155
3773 selectNetwork("ETH
- Ethereum
");
3774 driver.findElement(By.css('#bip32-tab a'))
3776 driver.findElement(By.css('#bip32-path'))
3778 driver.findElement(By.css('#bip32-path'))
3779 .sendKeys("m
/44'/60'/0'");
3780 driver.findElement(By.css('.phrase'))
3781 .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');
3782 driver.sleep(generateDelay).then(function() {
3783 getFirstAddress(function(address) {
3784 expect(address).toBe("0x8943E785B4a5714FC87a3aFAad1eB1FeB602B118");
3790 it('Can encrypt private keys using BIP38', function(done) {
3791 // see https://github.com/iancoleman/bip39/issues/140
3792 driver.executeScript(function() {
3793 $(".use-bip38
").prop("checked
", true);
3795 driver.findElement(By.css('.bip38-password'))
3796 .sendKeys('bip38password');
3797 driver.findElement(By.css('.rows-to-add'))
3799 driver.findElement(By.css('.rows-to-add'))
3801 driver.findElement(By.css('.phrase'))
3802 .sendKeys('abandon abandon ability');
3803 driver.sleep(bip38delay).then(function() {
3805 getFirstRowValue(function(address) {
3806 expect(address).toBe("1NCvSdumA3ngMM9c4aqU56AM6rqXddfuXB
");
3808 getFirstRowValue(function(pubkey) {
3809 expect(pubkey).toBe("043f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3884a74447ea901729b1e73a999b7520e7cb55b4120e6432c64153ccab8a848e1
");
3811 getFirstRowValue(function(privkey) {
3812 expect(privkey).toBe("6PRNRiFnj1RoR3sXhymdCvoZCgnUHQpfupNdKkFbWJkwWQEKesWt1EDMDM
");
3818 }, bip38delay + 5000);
3820 it('Shows the checksum for the entropy', function(done) {
3821 driver.findElement(By.css('.use-entropy'))
3823 driver.findElement(By.css('.entropy'))
3824 .sendKeys("00000000000000000000000000000000");
3825 driver.sleep(generateDelay).then(function() {
3826 driver.findElement(By.css('.checksum'))
3828 .then(function(text) {
3829 expect(text).toBe("1");
3835 it('Shows the checksum for the entropy with the correct groupings', function(done) {
3836 driver.findElement(By.css('.use-entropy'))
3838 // create a checksum of 20 bits, which spans multiple words
3839 driver.findElement(By.css('.entropy'))
3840 .sendKeys("F000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
");
3841 driver.sleep(generateDelay).then(function() {
3842 driver.findElement(By.css('.checksum'))
3844 .then(function(text) {
3845 // first group is 9 bits, second group is 11
3846 expect(text).toBe("011010111 01110000110");
3852 it('Uses vprv for bitcoin testnet p2wpkh', function(done) {
3853 selectNetwork("BTC
- Bitcoin Testnet
");
3854 driver.findElement(By.css('#bip84-tab a'))
3856 driver.findElement(By.css('.phrase'))
3857 .sendKeys('abandon abandon ability');
3858 driver.sleep(generateDelay).then(function() {
3859 driver.findElement(By.css('.root-key'))
3860 .getAttribute("value
")
3861 .then(function(path) {
3862 expect(path).toBe("vprv9DMUxX4ShgxML9N2YV5CvWEebWrM9aJ5ULpbRRyzyWu6vs4BzTvbfFFrH41N5hVi7MYSfiugd765L3JmAfDM5po36Y8ouCKRDeYQwByCmS7
");
3868 it('Shows a warning if generating weak mnemonics', function(done) {
3869 driver.executeScript(function() {
3870 $(".strength option
[selected
]").removeAttr("selected
");
3871 $(".strength option
[value
=6]").prop("selected
", true);
3872 $(".strength
").trigger("change
");
3874 driver.findElement(By.css(".generate
-container
.warning
"))
3875 .getAttribute("class")
3876 .then(function(classes) {
3877 expect(classes).not.toContain("hidden
");
3882 it('Does not show a warning if generating strong mnemonics', function(done) {
3883 driver.executeScript(function() {
3884 $(".strength option
[selected
]").removeAttr("selected
");
3885 $(".strength option
[value
=12]").prop("selected
", true);
3887 driver.findElement(By.css(".generate
-container
.warning
"))
3888 .getAttribute("class")
3889 .then(function(classes) {
3890 expect(classes).toContain("hidden
");
3895 it('Shows a warning if overriding weak entropy with longer mnemonics', function(done) {
3896 driver.findElement(By.css('.use-entropy'))
3898 driver.findElement(By.css('.entropy'))
3899 .sendKeys("0123456789abcdef
"); // 6 words
3900 driver.executeScript(function() {
3901 $(".mnemonic
-length
").val("12").trigger("change
");
3903 driver.findElement(By.css(".weak
-entropy
-override
-warning
"))
3904 .getAttribute("class")
3905 .then(function(classes) {
3906 expect(classes).not.toContain("hidden
");
3911 it('Does not show a warning if entropy is stronger than mnemonic length', function(done) {
3912 driver.findElement(By.css('.use-entropy'))
3914 driver.findElement(By.css('.entropy'))
3915 .sendKeys("0123456789abcdef0123456789abcdef0123456789abcdef
"); // 18 words
3916 driver.executeScript(function() {
3917 $(".mnemonic
-length
").val("12").trigger("change
");
3919 driver.findElement(By.css(".weak
-entropy
-override
-warning
"))
3920 .getAttribute("class")
3921 .then(function(classes) {
3922 expect(classes).toContain("hidden
");
3927 it('Shows litecoin BIP49 addresses', function(done) {
3928 driver.findElement(By.css('.phrase'))
3929 .sendKeys('abandon abandon ability');
3930 selectNetwork("LTC
- Litecoin
");
3931 driver.findElement(By.css('#bip49-tab a'))
3933 // bip49 addresses are shown
3934 driver.sleep(generateDelay).then(function() {
3935 driver.findElement(By.css('#bip49 .available'))
3936 .getAttribute("class")
3937 .then(function(classes) {
3938 expect(classes).not.toContain("hidden
");
3939 // check first address
3940 getFirstAddress(function(address) {
3941 expect(address).toBe("MFwLPhsXoBuSLL8cLmW9uK6tChkzduV8qN
");
3948 it('Can use root keys to generate segwit table rows', function(done) {
3949 // segwit uses ypub / zpub instead of xpub but the root key should still
3950 // be valid regardless of the encoding used to import that key.
3951 // Maybe this breaks the reason for the different extended key prefixes, but
3952 // since the parsed root key is used behind the scenes anyhow this should be
3954 driver.findElement(By.css('#root-key'))
3955 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
3956 driver.findElement(By.css('#bip49-tab a'))
3958 // bip49 addresses are shown
3959 driver.sleep(generateDelay).then(function() {
3960 getFirstAddress(function(address) {
3961 expect(address).toBe("3QG2Y9AA4xZ846gKHZqNf7mvVKbLqMKxr2
");