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 ripple
', function(done) {
413 selectText: "XRP - Ripple",
414 firstAddress: "rLTFnqbmCVPGx6VfaygdtuKWJgcN4v1zRS",
415 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",
417 testNetwork(done, params);
419 it('Allows selection
of dogecoin
', function(done) {
421 selectText: "DOGE - Dogecoin",
422 firstAddress: "DPQH2AtuzkVSG6ovjKk4jbUmZ6iXLpgbJA",
424 testNetwork(done, params);
426 it('Allows selection
of denarius
', function(done) {
428 selectText: "DNR - Denarius",
429 firstAddress: "DFdFMVUMzU9xX88EywXvAGwjiwpxyh9vKb",
431 testNetwork(done, params);
433 it('Allows selection
of shadowcash
', function(done) {
435 selectText: "SDC - ShadowCash",
436 firstAddress: "SiSZtfYAXEFvMm3XM8hmtkGDyViRwErtCG",
438 testNetwork(done, params);
440 it('Allows selection
of shadowcash testnet
', function(done) {
442 selectText: "SDC - ShadowCash Testnet",
443 firstAddress: "tM2EDpVKaTiEg2NZg3yKg8eqjLr55BErHe",
445 testNetwork(done, params);
447 it('Allows selection
of viacoin
', function(done) {
449 selectText: "VIA - Viacoin",
450 firstAddress: "Vq9Eq4N5SQnjqZvxtxzo7hZPW5XnyJsmXT",
452 testNetwork(done, params);
454 it('Allows selection
of viacoin testnet
', function(done) {
456 selectText: "VIA - Viacoin Testnet",
457 firstAddress: "tM2EDpVKaTiEg2NZg3yKg8eqjLr55BErHe",
459 testNetwork(done, params);
461 it('Allows selection
of jumbucks
', function(done) {
463 selectText: "JBS - Jumbucks",
464 firstAddress: "JLEXccwDXADK4RxBPkRez7mqsHVoJBEUew",
466 testNetwork(done, params);
468 it('Allows selection
of clam
', function(done) {
470 selectText: "CLAM - Clams",
471 firstAddress: "xCp4sakjVx4pUAZ6cBCtuin8Ddb6U1sk9y",
473 testNetwork(done, params);
475 it('Allows selection
of crown
', function(done) {
477 selectText: "CRW - Crown",
478 firstAddress: "18pWSwSUAQdiwMHUfFZB1fM2xue9X1FqE5",
480 testNetwork(done, params);
482 it('Allows selection
of dash
', function(done) {
484 selectText: "DASH - Dash",
485 firstAddress: "XdbhtMuGsPSkE6bPdNTHoFSszQKmK4S5LT",
487 testNetwork(done, params);
489 it('Allows selection
of dash testnet
', function(done) {
491 selectText: "DASH - Dash Testnet",
492 firstAddress: "yaR52EN4oojdJfBgzWJTymC4uuCLPT29Gw",
494 testNetwork(done, params);
496 it('Allows selection
of game
', function(done) {
498 selectText: "GAME - GameCredits",
499 firstAddress: "GSMY9bAp36cMR4zyT4uGVS7GFjpdXbao5Q",
501 testNetwork(done, params);
503 it('Allows selection
of komodo
', function(done) {
505 selectText: "KMD - Komodo",
506 firstAddress: "RMPPzJwAjPVZZAwJvXivHJGGjdCx6WBD2t",
508 testNetwork(done, params);
510 it('Allows selection
of namecoin
', function(done) {
512 selectText: "NMC - Namecoin",
513 firstAddress: "Mw2vK2Bvex1yYtYF6sfbEg2YGoUc98YUD2",
515 testNetwork(done, params);
517 it('Allows selection
of onixcoin
', function(done) {
519 selectText: "ONX - Onixcoin",
520 firstAddress: "XGwMqddeKjT3ddgX73QokjVbCL3aK6Yxfk",
522 testNetwork(done, params);
524 it('Allows selection
of peercoin
', function(done) {
526 selectText: "PPC - Peercoin",
527 firstAddress: "PVAiioTaK2eDHSEo3tppT9AVdBYqxRTBAm",
529 testNetwork(done, params);
531 it('Allows selection
of ethereum
', function(done) {
533 selectText: "ETH - Ethereum",
534 firstAddress: "0xe5815d5902Ad612d49283DEdEc02100Bd44C2772",
536 testNetwork(done, params);
537 // TODO test private key and public key
539 it('Allows selection
of slimcoin
', function(done) {
541 selectText: "SLM - Slimcoin",
542 firstAddress: "SNzPi1CafHFm3WWjRo43aMgiaEEj3ogjww",
544 testNetwork(done, params);
546 it('Allows selection
of slimcoin testnet
', function(done) {
548 selectText: "SLM - Slimcoin Testnet",
549 firstAddress: "n3nMgWufTek5QQAr6uwMhg5xbzj8xqc4Dq",
551 testNetwork(done, params);
553 it('Allows selection
of bitcoin cash
', function(done) {
555 selectText: "BCH - Bitcoin Cash",
556 firstAddress: "bitcoincash:qzlquk7w4hkudxypl4fgv8x279r754dkvur7jpcsps",
558 testNetwork(done, params);
560 it('Allows selection
of myriadcoin
', function(done) {
562 selectText: "XMY - Myriadcoin",
563 firstAddress: "MJEswvRR46wh9BoiVj9DzKYMBkCramhoBV",
565 testNetwork(done, params);
567 it('Allows selection
of pivx
', function(done) {
569 selectText: "PIVX - PIVX",
570 firstAddress: "DBxgT7faCuno7jmtKuu6KWCiwqsVPqh1tS",
572 testNetwork(done, params);
574 it('Allows selection
of pivx testnet
', function(done) {
576 selectText: "PIVX - PIVX Testnet",
577 firstAddress: "yB5U384n6dGkVE3by5y9VdvHHPwPg68fQj",
579 testNetwork(done, params);
581 it('Allows selection
of maza
', function(done) {
583 selectText: "MAZA - Maza",
584 firstAddress: "MGW4Bmi2NEm4PxSjgeFwhP9vg18JHoRnfw",
586 testNetwork(done, params);
588 it('Allows selection
of fujicoin
', function(done) {
590 selectText: "FJC - Fujicoin",
591 firstAddress: "FgiaLpG7C99DyR4WnPxXedRVHXSfKzUDhF",
593 testNetwork(done, params);
595 it('Allows selection
of nubits
', function(done) {
597 selectText: "USNBT - NuBits",
598 firstAddress: "BLxkabXuZSJSdesLD7KxZdqovd4YwyBTU6",
600 testNetwork(done, params);
602 it('Allows selection
of bitcoin gold
', function(done) {
604 selectText: "BTG - Bitcoin Gold",
605 firstAddress: "GdDqug4WUsn5syNbSTHatNn4XnuwZtzedx",
607 testNetwork(done, params);
609 it('Allows selection
of monacoin
', function(done) {
611 selectText: "MONA - Monacoin",
612 firstAddress: "MKMiMr7MyjDKjJbCBzgF6u4ByqTS4NkRB1",
614 testNetwork(done, params);
616 it('Allows selection
of AXE
', function(done) {
618 selectText: "AXE - Axe",
619 firstAddress: "PScwtLUyPiGrqtKXrHF37DGETLXLZdw4up",
621 testNetwork(done, params);
623 it('Allows selection
of BlackCoin
', function(done) {
625 selectText: "BLK - BlackCoin",
626 firstAddress: "B5MznAKwj7uQ42vDz3w4onhBXPcqhTwJ9z",
628 testNetwork(done, params);
630 it('Allows selection
of Neblio
', function(done) {
632 selectText: "NEBL - Neblio",
633 firstAddress: "NefkeEEvhusbHMmTRrxx7H9wFnUXd8qQsE",
635 testNetwork(done, params);
637 it('Allows selection
of Beetlecoin
', function(done) {
639 selectText: "BEET - Beetlecoin",
640 firstAddress: "BVmtbEsGrjpknprmpHFq26z4kYHJUFHE71",
642 testNetwork(done, params);
644 it('Allows selection
of Adcoin
', function(done) {
646 selectText: "ACC - Adcoin",
647 firstAddress: "AcEDM6V5sF4kFHC76MJjjfProtS5Sw2qcd",
649 testNetwork(done, params);
651 it('Allows selection
of Asiacoin
', function(done) {
653 selectText: "AC - Asiacoin",
654 firstAddress: "ALupuEEz7kJjQTAvmtcBMBVuEjPa7GqZzE",
656 testNetwork(done, params);
658 it('Allows selection
of Auroracoin
', function(done) {
660 selectText: "AUR - Auroracoin",
661 firstAddress: "ANuraS6F4Jpi413FEnavjYkKYJJRHkgYCm",
663 testNetwork(done, params);
665 it('Allows selection
of Bata
', function(done) {
667 selectText: "BTA - Bata",
668 firstAddress: "BGxBdNeYPtF3GCuTtZBPQdFxCkdBYSF3fj",
670 testNetwork(done, params);
672 it('Allows selection
of Belacoin
', function(done) {
674 selectText: "BELA - Belacoin",
675 firstAddress: "BEeetqpNffdzeknSpNmQp5KAFh2KK1Qx7S",
677 testNetwork(done, params);
679 it('Allows selection
of Bitcoin Atom
', function(done) {
681 selectText: "BCA - Bitcoin Atom",
682 firstAddress: "AMy6qMbJeC4zsGRL6iWszmeCdQH65fgfih",
684 testNetwork(done, params);
686 it('Allows selection
of Bitcoinplus
', function(done) {
688 selectText: "XBC - Bitcoinplus",
689 firstAddress: "B7FSynZoDbEwTCSgsXq9nJ5ue8owYLVL8r",
691 testNetwork(done, params);
693 it('Allows selection
of Bitcoin Private
', function(done) {
695 selectText: "BTCP - Bitcoin Private",
696 firstAddress: "b1M3PbiXXyN6Hdivdw5rJv5VKpLjPzhm4jM",
698 testNetwork(done, params);
700 it('Allows selection
of Bitcoinz
', function(done) {
702 selectText: "BTCZ - Bitcoinz",
703 firstAddress: "t1X2YQoxs8cYRo2oaBYgVEwW5QNjCC59NYc",
705 testNetwork(done, params);
707 it('Allows selection
of BitCloud
', function(done) {
709 selectText: "BTDX - BitCloud",
710 firstAddress: "BHbWitXCNgTf1BhsRDNMP186EeibuzmrBi",
712 testNetwork(done, params);
714 it('Allows selection
of Bitcore
', function(done) {
716 selectText: "BTX - Bitcore",
717 firstAddress: "2Rgp5Znhpy34TK4QmPkfCiYs9r4KovfTH9",
719 testNetwork(done, params);
721 it('Allows selection
of Bitsend
', function(done) {
723 selectText: "BSD - Bitsend",
724 firstAddress: "iBPk7LYjDun3EPk7CRR8UUmnPoceVc1bp2",
726 testNetwork(done, params);
728 it('Allows selection
of Britcoin
', function(done) {
730 selectText: "BRIT - Britcoin",
731 firstAddress: "B6Aue4J2XLs1f1dtD4H1SHYFfh4XrmEbrw",
733 testNetwork(done, params);
735 it('Allows selection
of Canadaecoin
', function(done) {
737 selectText: "CDN - Canadaecoin",
738 firstAddress: "CanAyCfd5Rj2CQVfaoAmvDUZunPM5W1AEQ",
740 testNetwork(done, params);
742 it('Allows selection
of Cannacoin
', function(done) {
744 selectText: "CCN - Cannacoin",
745 firstAddress: "CYjW8xWB43g6krLJTmmrPk1PonoQX7h9Qd",
747 testNetwork(done, params);
749 it('Allows selection
of Clubcoin
', function(done) {
751 selectText: "CLUB - Clubcoin",
752 firstAddress: "CHMDEXN4sihpSVX4GyAa2hZ62shnby7uyN",
754 testNetwork(done, params);
756 it('Allows selection
of Compcoin
', function(done) {
758 selectText: "CMP - Compcoin",
759 firstAddress: "CLshtw3zhxkseBJS46UF12v3AFy9Dx7JVv",
761 testNetwork(done, params);
763 it('Allows selection
of Crave
', function(done) {
765 selectText: "CRAVE - Crave",
766 firstAddress: "VCYJeti6uKMNBFKCL7eP96UwuFWYHM7c85",
768 testNetwork(done, params);
770 it('Allows selection
of Defcoin
', function(done) {
772 selectText: "DFC - Defcoin",
773 firstAddress: "D8swcgyaaFUrXZU3ATwbgy16buCpWqbG1M",
775 testNetwork(done, params);
777 it('Allows selection
of Diamond
', function(done) {
779 selectText: "DMD - Diamond",
780 firstAddress: "dJnrVbLL9UPjdaVRz2C8VpqHZknqAqjLek",
782 testNetwork(done, params);
784 it('Allows selection
of Digibyte
', function(done) {
786 selectText: "DGB - Digibyte",
787 firstAddress: "D85Rp9jwLtMdmP6wGjTiqHBdVQLST3YCEq",
789 testNetwork(done, params);
791 it('Allows selection
of Digitalcoin
', function(done) {
793 selectText: "DGC - Digitalcoin",
794 firstAddress: "DKw4UGKEAZWweDNEbBFNQx4EM8x1mpUdia",
796 testNetwork(done, params);
798 it('Allows selection
of Ecoin
', function(done) {
800 selectText: "ECN - Ecoin",
801 firstAddress: "e6WFPLG5gcXyF7cESFteH1hE2XSmowW5yB",
803 testNetwork(done, params);
805 it('Allows selection
of Edrcoin
', function(done) {
807 selectText: "EDRC - Edrcoin",
808 firstAddress: "eh1nUJsvgKPFv6ebMBfcwJ299GMCpjeZUG",
810 testNetwork(done, params);
812 it('Allows selection
of Egulden
', function(done) {
814 selectText: "EFL - Egulden",
815 firstAddress: "Lg66yt55R7edRM58cDhKzXik2kFme3viX7",
817 testNetwork(done, params);
819 it('Allows selection
of Einsteinium
', function(done) {
821 selectText: "EMC2 - Einsteinium",
822 firstAddress: "EVAABm9hXKHk2MpVMbwNakRubFnNha5m8m",
824 testNetwork(done, params);
826 it('Allows selection
of Europecoin
', function(done) {
828 selectText: "ERC - Europecoin",
829 firstAddress: "ESA2YwPYntAoaPrE8Fm5qkKRtkcwLcwD6R",
831 testNetwork(done, params);
833 it('Allows selection
of Exclusivecoin
', function(done) {
835 selectText: "EXCL - Exclusivecoin",
836 firstAddress: "EbUa6m8UZW6nTxsYZD2FsDjkadKbp5M6JT",
838 testNetwork(done, params);
840 it('Allows selection
of Feathercoin
', function(done) {
842 selectText: "FTC - Feathercoin",
843 firstAddress: "6gDdjAMoSgQaW8UhqK3oboHs6ftGAroKkM",
845 testNetwork(done, params);
847 it('Allows selection
of Firstcoin
', function(done) {
849 selectText: "FRST - Firstcoin",
850 firstAddress: "FJN9GzfMm7Q8R4DJwK1H9F6A1GTghvFiMJ",
852 testNetwork(done, params);
854 it('Allows selection
of Flashcoin
', function(done) {
856 selectText: "FLASH - Flashcoin",
857 firstAddress: "UWfpf5LfMmLxZYooEb2EyvWhZ8NG7EZDRt",
859 testNetwork(done, params);
861 it('Allows selection
of GCRCoin
', function(done) {
863 selectText: "GCR - GCRCoin",
864 firstAddress: "GJjF5cLwyXLacpuvXAVksxGxKvHDjx58d6",
866 testNetwork(done, params);
868 it('Allows selection
of Gobyte
', function(done) {
870 selectText: "GBX - Gobyte",
871 firstAddress: "GS813Ys2brkmvSUw1rUqGPm2HqQVDHJRyA",
873 testNetwork(done, params);
875 it('Allows selection
of Gridcoin
', function(done) {
877 selectText: "GRC - Gridcoin",
878 firstAddress: "SGrWbBPvobgqKRF8td1Kdc9vbRY7MJ78Y9",
880 testNetwork(done, params);
882 it('Allows selection
of Gulden
', function(done) {
884 selectText: "NLG - Gulden",
885 firstAddress: "GcDP7cNEc33MPPdTFNJ8pZc6VMZJ2CbKxY",
887 testNetwork(done, params);
889 it('Allows selection
of Helleniccoin
', function(done) {
891 selectText: "HNC - Helleniccoin",
892 firstAddress: "LbHEKe5H72zp9G1fuWNiiNePTUfJb88915",
894 testNetwork(done, params);
896 it('Allows selection
of Hempcoin
', function(done) {
898 selectText: "THC - Hempcoin",
899 firstAddress: "H8sdWbZyJV4gyXyHtLXDaNnAuUDhK5mfTV",
901 testNetwork(done, params);
903 it('Allows selection
of Insane
', function(done) {
905 selectText: "INSN - Insane",
906 firstAddress: "iMPqEJMiXWuxC9U2NVinCCMr4t72h58EWx",
908 testNetwork(done, params);
910 it('Allows selection
of Iop
', function(done) {
912 selectText: "IOP - Iop",
913 firstAddress: "pGKQmcaPf95Ur5o6oHK4qdiZ52p1yaTvq1",
915 testNetwork(done, params);
917 it('Allows selection
of Ixcoin
', function(done) {
919 selectText: "IXC - Ixcoin",
920 firstAddress: "xgE9bTZ6YypT3E6ByzkTt31Hq68E9BqywH",
922 testNetwork(done, params);
924 it('Allows selection
of Kobocoin
', function(done) {
926 selectText: "KOBO - Kobocoin",
927 firstAddress: "FTVoNJETXDAM8x7MnmdE8RwWndSr9PQWhy",
929 testNetwork(done, params);
931 it('Allows selection
of Landcoin
', function(done) {
933 selectText: "LDCN - Landcoin",
934 firstAddress: "LLvLwNjG1aJcn1RS4W4GJUbv8fNaRATG7c",
936 testNetwork(done, params);
938 it('Allows selection
of Library Credits
', function(done) {
940 selectText: "LBC - Library Credits",
941 firstAddress: "bQJEQrHDJyHdqycB32uysh1SWn8Ln8LMdg",
943 testNetwork(done, params);
945 it('Allows selection
of Linx
', function(done) {
947 selectText: "LINX - Linx",
948 firstAddress: "XGWQ3cb3LGUB3VnHmj6xYSMgnokNbf6dyk",
950 testNetwork(done, params);
952 it('Allows selection
of Litecoincash
', function(done) {
954 selectText: "LCC - Litecoincash",
955 firstAddress: "Ce5n7fjUuQPLutJ4W5nCCfQLKdKLE1mv9A",
957 testNetwork(done, params);
959 it('Allows selection
of Lynx
', function(done) {
961 selectText: "LYNX - Lynx",
962 firstAddress: "KUeY3ZdZkg96p4W98pj1JjygCFU1XqWdw3",
964 testNetwork(done, params);
966 it('Allows selection
of Megacoin
', function(done) {
968 selectText: "MEC - Megacoin",
969 firstAddress: "MDfAj9CzkC1HpcUiVGnHp8yKTa7WXgu8AY",
971 testNetwork(done, params);
973 it('Allows selection
of Minexcoin
', function(done) {
975 selectText: "MNX - Minexcoin",
976 firstAddress: "XC1VnyJVfiMDwWgFtAHDp41cgY3AHk3dJT",
978 testNetwork(done, params);
980 it('Allows selection
of Navcoin
', function(done) {
982 selectText: "NAV - Navcoin",
983 firstAddress: "NTQVTPK3NWSQLKoffkiQw99T8PifkF1Y2U",
985 testNetwork(done, params);
987 it('Allows selection
of Neoscoin
', function(done) {
989 selectText: "NEOS - Neoscoin",
990 firstAddress: "NgATz6QbQNXvayHQ4CpZayugb9HeaPDdby",
992 testNetwork(done, params);
994 it('Allows selection
of Neurocoin
', function(done) {
996 selectText: "NRO - Neurocoin",
997 firstAddress: "NVdYErQ3mFpDuF5DquW9WMiT7sLc8ufFTn",
999 testNetwork(done, params);
1001 it('Allows selection
of Newyorkc
', function(done) {
1003 selectText: "NYC - Newyorkc",
1004 firstAddress: "RSVMfyH1fKfy3puADJEhut2vfkRyon6imm",
1006 testNetwork(done, params);
1008 it('Allows selection
of Novacoin
', function(done) {
1010 selectText: "NVC - Novacoin",
1011 firstAddress: "4JRvUmxcKCJmaMXZyvRoSS1cmG2XvnZfHN",
1013 testNetwork(done, params);
1015 it('Allows selection
of Nushares
', function(done) {
1017 selectText: "NSR - Nushares",
1018 firstAddress: "SecjXzU3c7EecdT7EbC4vvmbdtBBokWh6J",
1020 testNetwork(done, params);
1022 it('Allows selection
of Okcash
', function(done) {
1024 selectText: "OK - Okcash",
1025 firstAddress: "PV4Qp1TUYuGv4TqVtLZtqvrsWWRycfx1Yi",
1027 testNetwork(done, params);
1029 it('Allows selection
of Omnicore
', function(done) {
1031 selectText: "OMNI - Omnicore",
1032 firstAddress: "1Q1t3gonjCT3rW38TsTsCvgSc3hh7zBGbi",
1034 testNetwork(done, params);
1036 it('Allows selection
of Pesobit
', function(done) {
1038 selectText: "PSB - Pesobit",
1039 firstAddress: "PDePsF7ALyXP7JaywokdYiRTDtKa14MAr1",
1041 testNetwork(done, params);
1043 it('Allows selection
of Pinkcoin
', function(done) {
1045 selectText: "PINK - Pinkcoin",
1046 firstAddress: "2TgjYQffjbzUHJghNaVbdsjHbRwruC3yzC",
1048 testNetwork(done, params);
1050 it('Allows selection
of POSWcoin
', function(done) {
1052 selectText: "POSW - POSWcoin",
1053 firstAddress: "PNxewmZoPnGBvoEbH6hgQZCK1igDiBCdgC",
1055 testNetwork(done, params);
1057 it('Allows selection
of Potcoin
', function(done) {
1059 selectText: "POT - Potcoin",
1060 firstAddress: "PEo7Vg2ctXgpP4vuLPeY9aGJtZotyrmiHc",
1062 testNetwork(done, params);
1064 it('Allows selection
of Putincoin
', function(done) {
1066 selectText: "PUT - Putincoin",
1067 firstAddress: "PViWnfr2uFtovd6e7joM49C94CsGSnqJis",
1069 testNetwork(done, params);
1071 it('Allows selection
of Reddcoin
', function(done) {
1073 selectText: "RDD - Reddcoin",
1074 firstAddress: "RtgRvXMBng1y51ftteveFqwNfyRG18HpxQ",
1076 testNetwork(done, params);
1078 it('Allows selection
of RevolutionVR
', function(done) {
1080 selectText: "RVR - RevolutionVR",
1081 firstAddress: "VXeeoP2jkzZnMFxtc66ZBZK1NHN5QJnnjL",
1083 testNetwork(done, params);
1085 it('Allows selection
of Rubycoin
', function(done) {
1087 selectText: "RBY - Rubycoin",
1088 firstAddress: "RV76JDtjTs11JdMDRToYn6CHecMRPLnKS6",
1090 testNetwork(done, params);
1092 it('Allows selection
of Salus
', function(done) {
1094 selectText: "SLS - Salus",
1095 firstAddress: "SNzPi1CafHFm3WWjRo43aMgiaEEj3ogjww",
1097 testNetwork(done, params);
1099 it('Allows selection
of Smileycoin
', function(done) {
1101 selectText: "SMLY - Smileycoin",
1102 firstAddress: "BEZVnEBCAyFByrgKpwAgYgtvP4rKAd9Sj2",
1104 testNetwork(done, params);
1106 it('Allows selection
of Solarcoin
', function(done) {
1108 selectText: "SLR - Solarcoin",
1109 firstAddress: "8LZ13HbnjtaMJWSvvVFNTLf71zFfDrhwLu",
1111 testNetwork(done, params);
1113 it('Allows selection
of stash
', function(done) {
1115 selectText: "STASH - Stash",
1116 firstAddress: "XxwAsWB7REDKmAvHA85SbEZQQtpxeUDxS3",
1118 testNetwork(done, params);
1120 it('Allows selection
of stash testnet
', function(done) {
1122 selectText: "STASH - Stash Testnet",
1123 firstAddress: "yWQCTSkUst7ddYuebKsqa1kSoXEjpCkGKR",
1125 testNetwork(done, params);
1127 it('Allows selection
of Stratis
', function(done) {
1129 selectText: "STRAT - Stratis",
1130 firstAddress: "ScfJnq3QDhKgDMEds6sqUE1ot6ShfhmXXq",
1132 testNetwork(done, params);
1134 it('Allows selection
of Stratis Test
', function(done) {
1136 selectText: "TSTRAT - Stratis Testnet",
1137 firstAddress: "TRLWm3dye4FRrDWouwYUSUZP96xb76mBE3",
1139 testNetwork(done, params);
1141 it('Allows selection
of Syscoin
', function(done) {
1143 selectText: "SYS - Syscoin",
1144 firstAddress: "SZwJi42Pst3VAMomyK5DG4157WM5ofRmSj",
1146 testNetwork(done, params);
1148 it('Allows selection
of Toa
', function(done) {
1150 selectText: "TOA - Toa",
1151 firstAddress: "TSe1QAnUwQzUfbBusDzRJ9URttrRGKoNKF",
1153 testNetwork(done, params);
1155 it('Allows selection
of Ultimatesecurecash
', function(done) {
1157 selectText: "USC - Ultimatesecurecash",
1158 firstAddress: "UPyLAZU2Che5fiy7Ed8xVJFmXAUhitA4ug",
1160 testNetwork(done, params);
1162 it('Allows selection
of Unobtanium
', function(done) {
1164 selectText: "UNO - Unobtanium",
1165 firstAddress: "uUBMPVMXrR6qhqornJqKTWgr8L69vihSL9",
1167 testNetwork(done, params);
1169 it('Allows selection
of Vcash
', function(done) {
1171 selectText: "XVC - Vcash",
1172 firstAddress: "VuL53MSY6KjvAjKSeRkh3NDnKykacDVeps",
1174 testNetwork(done, params);
1176 it('Allows selection
of Verge
', function(done) {
1178 selectText: "XVG - Verge",
1179 firstAddress: "DCrVuGkMjLJpTGgwAgv9AcMdeb1nkWbjZA",
1181 testNetwork(done, params);
1183 it('Allows selection
of Vertcoin
', function(done) {
1185 selectText: "VTC - Vertcoin",
1186 firstAddress: "Vf6koGuiWdXQfx8tNqxoNeEDxh4xh5cxsG",
1188 testNetwork(done, params);
1190 it('Allows selection
of Vivo
', function(done) {
1192 selectText: "VIVO - Vivo",
1193 firstAddress: "VFmBwuXXGhJe7MarQG2GfzHMFebRHgfSpB",
1195 testNetwork(done, params);
1197 it('Allows selection
of Vpncoin
', function(done) {
1199 selectText: "VASH - Vpncoin",
1200 firstAddress: "VoEmH1qXC4TsSgBAStR21QYetwnFqbqCx9",
1202 testNetwork(done, params);
1204 it('Allows selection
of Whitecoin
', function(done) {
1206 selectText: "XWC - Whitecoin",
1207 firstAddress: "WcSwCAUqrSgeSYbsaS3SSWWhsx8KRYTFDR",
1209 testNetwork(done, params);
1211 it('Allows selection
of Wincoin
', function(done) {
1213 selectText: "WC - Wincoin",
1214 firstAddress: "WaDVCESMGgyKgNESdn3u43NnwmGSkZED3Z",
1216 testNetwork(done, params);
1218 it('Allows selection
of Zcoin
', function(done) {
1220 selectText: "XZC - Zcoin",
1221 firstAddress: "a6VcMdP4XgAA9Tr7xNszmPG5FZpfRf17Cq",
1223 testNetwork(done, params);
1225 it('Allows selection
of Zcash
', function(done) {
1227 selectText: "ZEC - Zcash",
1228 firstAddress: "t1Sz8AneMcVuzUg3tPJ8et5AS5LFJ7K2EF9",
1230 testNetwork(done, params);
1232 it('Allows selection
of Zclassic
', function(done) {
1234 selectText: "ZCL - Zclassic",
1235 firstAddress: "t1TBMxTvVJRybUbMLGWq8H4A8F4VUL7czEc",
1237 testNetwork(done, params);
1239 it('Allows selection
of Zencash
', function(done) {
1241 selectText: "ZEN - Zencash",
1242 firstAddress: "znWh9XASyW2dZq5tck84wFjiwuqVysi7q3p",
1244 testNetwork(done, params);
1246 it('Allows selection
of Energi
', function(done) {
1248 selectText: "NRG - Energi",
1249 firstAddress: "EejRy4t4nidzhGGzkJUgFP3z4HYBjhTsRt",
1251 testNetwork(done, params);
1253 it('Allows selection
of Ethereum Classic
', function(done) {
1255 selectText: "ETC - Ethereum Classic",
1256 firstAddress: "0x3c05e5556693808367afB62eF3b63e35d6eD249A",
1258 testNetwork(done, params);
1260 it('Allows selection
of Pirl
', function(done) {
1262 selectText: "PIRL - Pirl",
1263 firstAddress: "0xe77FC0723dA122B5025CA79193c28563eB47e776",
1265 testNetwork(done, params);
1267 it('Allows selection
of MIX
', function(done) {
1269 selectText: "MIX - MIX",
1270 firstAddress: "0x98BC5e63aeb6A4e82d72850d20710F07E29A29F1",
1272 testNetwork(done, params);
1274 it('Allows selection
of Musicoin
', function(done) {
1276 selectText: "MUSIC - Musicoin",
1277 firstAddress: "0xDc060e4A0b0313ea83Cf6B3A39B9db2D29004897",
1279 testNetwork(done, params);
1281 it('Allows selection
of Poa
', function(done) {
1283 selectText: "POA - Poa",
1284 firstAddress: "0x53aF28d754e106210C3d0467Dd581eaf7e3C5e60",
1286 testNetwork(done, params);
1288 it('Allows selection
of Expanse
', function(done) {
1290 selectText: "EXP - Expanse",
1291 firstAddress: "0xf57FeAbf26582b6E3E666559d3B1Cc6fB2b2c5F6",
1293 testNetwork(done, params);
1295 it('Allows selection
of Callisto
', function(done) {
1297 selectText: "CLO - Callisto",
1298 firstAddress: "0x4f9364F7420B317266C51Dc8eB979717D4dE3f4E",
1300 testNetwork(done, params);
1302 it('Allows selection
of HUSH
', function(done) {
1304 selectText: "HUSH - Hush",
1305 firstAddress: "t1g6rLXUnJaiJuu4q4zmJjoa9Gk4fwKpiuA",
1307 testNetwork(done, params);
1309 it('Allows selection
of ExchangeCoin
', function(done) {
1311 selectText: "EXCC - ExchangeCoin",
1312 firstAddress: "22txYKpFN5fwGwdSs2UBf7ywewbLM92YqK7E",
1314 testNetwork(done, params);
1316 it('Allows selection
of Artax
', function(done) {
1318 selectText: "XAX - Artax",
1319 firstAddress: "AYxaQPY7XLidG31V7F3yNzwxPYpYzRqG4q",
1321 testNetwork(done, params);
1323 it('Allows selection
of BitcoinGreen
', function(done) {
1325 selectText: "BITG - Bitcoin Green",
1326 firstAddress: "GeNGm9SkEfwbsws3UrrUSE2sJeyWYjzraY",
1328 testNetwork(done, params);
1330 it('Allows selection
of ANON
', function(done) {
1332 selectText: "ANON - ANON",
1333 firstAddress: "AnU6pijpEeUZFWSTyM2qTqZQn996Zq1Xard",
1335 testNetwork(done, params);
1337 it('Allows selection
of ProjectCoin
', function(done) {
1339 selectText: "PRJ - ProjectCoin",
1340 firstAddress: "PXZG97saRseSCftfe1mcFmfAA7pf6qBbaz",
1342 testNetwork(done, params);
1344 it('Allows selection
of Phore
', function(done) {
1346 selectText: "PHR - Phore",
1347 firstAddress: "PJThxpoXAG6hqrmdeQQbVDX4TJtFTMMymC",
1349 testNetwork(done, params);
1351 it('Allows selection
of Blocknode
', function(done) {
1353 selectText: "BND - Blocknode",
1354 firstAddress: "BG8xZSAur2jYLG9VXt8dYfkKxxeR7w9bSe",
1356 testNetwork(done, params);
1358 it('Allows selection
of Blocknode Testnet
', function(done) {
1360 selectText: "tBND - Blocknode Testnet",
1361 firstAddress: "bSptsFyDktFSKpWveRywJsDoJA2TC6qfHv",
1363 testNetwork(done, params);
1366 // BIP39 seed is set from phrase
1367 it('Sets the bip39 seed
from the prhase
', function(done) {
1368 driver.findElement(By.css('.phrase
'))
1369 .sendKeys('abandon abandon ability
');
1370 driver.sleep(generateDelay).then(function() {
1371 driver.findElement(By.css('.seed
'))
1372 .getAttribute("value")
1373 .then(function(seed) {
1374 expect(seed).toBe("20da140d3dd1df8713cefcc4d54ce0e445b4151027a1ab567b832f6da5fcc5afc1c3a3f199ab78b8e0ab4652efd7f414ac2c9a3b81bceb879a70f377aa0a58f3");
1380 // BIP32 root key is set from phrase
1381 it('Sets the bip39 root key
from the prhase
', function(done) {
1382 driver.findElement(By.css('.phrase
'))
1383 .sendKeys('abandon abandon ability
');
1384 driver.sleep(generateDelay).then(function() {
1385 driver.findElement(By.css('.root
-key
'))
1386 .getAttribute("value")
1387 .then(function(seed) {
1388 expect(seed).toBe("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
1394 // Tabs show correct addresses when changed
1395 it('Shows the correct address when tab is changed
', function(done) {
1396 driver.findElement(By.css('.phrase
'))
1397 .sendKeys('abandon abandon ability
');
1398 driver.sleep(generateDelay).then(function() {
1399 driver.findElement(By.css('#bip32
-tab a
'))
1401 driver.sleep(generateDelay).then(function() {
1402 getFirstAddress(function(address) {
1403 expect(address).toBe("17uQ7s2izWPwBmEVFikTmZUjbBKWYdJchz");
1410 // BIP44 derivation path is shown
1411 it('Shows the derivation path
for bip44 tab
', function(done) {
1412 driver.findElement(By.css('.phrase
'))
1413 .sendKeys('abandon abandon ability
');
1414 driver.sleep(generateDelay).then(function() {
1415 driver.findElement(By.css('#bip44
.path
'))
1416 .getAttribute("value")
1417 .then(function(path) {
1418 expect(path).toBe("m/44'/0'/0'/0");
1424 // BIP44 extended private key is shown
1425 it('Shows the extended private key for bip44 tab', function(done) {
1426 driver.findElement(By.css('.phrase'))
1427 .sendKeys('abandon abandon ability');
1428 driver.sleep(generateDelay).then(function() {
1429 driver.findElement(By.css('.extended-priv-key'))
1430 .getAttribute("value
")
1431 .then(function(path) {
1432 expect(path).toBe("xprvA2DxxvPZcyRvYgZMGS53nadR32mVDeCyqQYyFhrCVbJNjPoxMeVf7QT5g7mQASbTf9Kp4cryvcXnu2qurjWKcrdsr91jXymdCDNxKgLFKJG
");
1438 // BIP44 extended public key is shown
1439 it('Shows the extended public key for bip44 tab', function(done) {
1440 driver.findElement(By.css('.phrase'))
1441 .sendKeys('abandon abandon ability');
1442 driver.sleep(generateDelay).then(function() {
1443 driver.findElement(By.css('.extended-pub-key'))
1444 .getAttribute("value
")
1445 .then(function(path) {
1446 expect(path).toBe("xpub6FDKNRvTTLzDmAdpNTc49ia9b4byd6vqCdUa46Fp3vqMcC96uBoufCmZXQLiN5AK3iSCJMhf9gT2sxkpyaPepRuA7W3MujV5tGmF5VfbueM
");
1452 // BIP44 account field changes address list
1453 it('Changes the address list if bip44 account is changed', function(done) {
1454 driver.findElement(By.css('#bip44 .account'))
1456 driver.findElement(By.css('.phrase'))
1457 .sendKeys('abandon abandon ability');
1458 driver.sleep(generateDelay).then(function() {
1459 getFirstAddress(function(address) {
1460 expect(address).toBe("1Nq2Wmu726XHCuGhctEtGmhxo3wzk5wZ1H
");
1466 // BIP44 change field changes address list
1467 it('Changes the address list if bip44 change is changed', function(done) {
1468 driver.findElement(By.css('#bip44 .change'))
1470 driver.findElement(By.css('.phrase'))
1471 .sendKeys('abandon abandon ability');
1472 driver.sleep(generateDelay).then(function() {
1473 getFirstAddress(function(address) {
1474 expect(address).toBe("1KAGfWgqfVbSSXY56fNQ7YnhyKuoskHtYo
");
1480 // BIP32 derivation path can be set
1481 it('Can use a custom bip32 derivation path', function(done) {
1482 driver.findElement(By.css('#bip32-tab a'))
1484 driver.findElement(By.css('#bip32 .path'))
1486 driver.findElement(By.css('#bip32 .path'))
1488 driver.findElement(By.css('.phrase'))
1489 .sendKeys('abandon abandon ability');
1490 driver.sleep(generateDelay).then(function() {
1491 getFirstAddress(function(address) {
1492 expect(address).toBe("16pYQQdLD1hH4hwTGLXBaZ9Teboi1AGL8L
");
1498 // BIP32 can use hardened derivation paths
1499 it('Can use a hardened derivation paths', function(done) {
1500 driver.findElement(By.css('#bip32-tab a'))
1502 driver.findElement(By.css('#bip32 .path'))
1504 driver.findElement(By.css('#bip32 .path'))
1506 driver.findElement(By.css('.phrase
'))
1507 .sendKeys('abandon abandon ability
');
1508 driver.sleep(generateDelay).then(function() {
1509 getFirstAddress(function(address) {
1510 expect(address).toBe("14aXZeprXAE3UUKQc4ihvwBvww2LuEoHo4");
1516 // BIP32 extended private key is shown
1517 it('Shows the BIP32 extended
private key
', function(done) {
1518 driver.findElement(By.css('#bip32
-tab a
'))
1520 driver.findElement(By.css('.phrase
'))
1521 .sendKeys('abandon abandon ability
');
1522 driver.sleep(generateDelay).then(function() {
1523 driver.findElement(By.css('.extended
-priv
-key
'))
1524 .getAttribute("value")
1525 .then(function(privKey) {
1526 expect(privKey).toBe("xprv9va99uTVE5aLiutUVLTyfxfe8v8aaXjSQ1XxZbK6SezYVuikA9MnjQVTA8rQHpNA5LKvyQBpLiHbBQiiccKiBDs7eRmBogsvq3THFeLHYbe");
1532 // BIP32 extended public key is shown
1533 it('Shows the BIP32 extended
public key
', function(done) {
1534 driver.findElement(By.css('#bip32
-tab a
'))
1536 driver.findElement(By.css('.phrase
'))
1537 .sendKeys('abandon abandon ability
');
1538 driver.sleep(generateDelay).then(function() {
1539 driver.findElement(By.css('.extended
-pub
-key
'))
1540 .getAttribute("value")
1541 .then(function(pubKey) {
1542 expect(pubKey).toBe("xpub69ZVZQzP4T8dwPxwbMzz36cNgwy4yzTHmETZMyihzzXXNi3thgg3HCow1RtY252wdw5rS8369xKnraN5Q93y3FkFfJp2XEHWUrkyXsjS93P");
1548 // Derivation path is shown in table
1549 it('Shows the derivation path
in the table
', function(done) {
1550 driver.findElement(By.css('.phrase
'))
1551 .sendKeys('abandon abandon ability
');
1552 driver.sleep(generateDelay).then(function() {
1553 getFirstPath(function(path) {
1554 expect(path).toBe("m/44'/0'/0'/0/0");
1560 // Derivation path for address can be hardened
1561 it('Can derive hardened addresses', function(done) {
1562 driver.findElement(By.css('#bip32-tab a'))
1564 driver.executeScript(function() {
1565 $(".hardened
-addresses
").prop("checked
", true);
1567 driver.findElement(By.css('.phrase'))
1568 .sendKeys('abandon abandon ability');
1569 driver.sleep(generateDelay).then(function() {
1570 getFirstAddress(function(address) {
1571 expect(address).toBe("18exLzUv7kfpiXRzmCjFDoC9qwNLFyvwyd
");
1577 // Derivation path visibility can be toggled
1578 it('Can toggle visibility of the derivation path column', function(done) {
1579 driver.findElement(By.css('.phrase'))
1580 .sendKeys('abandon abandon ability');
1581 driver.sleep(generateDelay).then(function() {
1582 driver.findElement(By.css('.index-toggle'))
1584 testColumnValuesAreInvisible(done, "index
");
1589 it('Shows the address in the table', function(done) {
1590 driver.findElement(By.css('.phrase'))
1591 .sendKeys('abandon abandon ability');
1592 driver.sleep(generateDelay).then(function() {
1593 getFirstAddress(function(address) {
1594 expect(address).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug
");
1600 // Addresses are shown in order of derivation path
1601 it('Shows the address in order of derivation path', function(done) {
1602 driver.findElement(By.css('.phrase'))
1603 .sendKeys('abandon abandon ability');
1604 driver.sleep(generateDelay).then(function() {
1605 testRowsAreInCorrectOrder(done);
1609 // Address visibility can be toggled
1610 it('Can toggle visibility of the address column', function(done) {
1611 driver.findElement(By.css('.phrase'))
1612 .sendKeys('abandon abandon ability');
1613 driver.sleep(generateDelay).then(function() {
1614 driver.findElement(By.css('.address-toggle'))
1616 testColumnValuesAreInvisible(done, "address
");
1620 // Public key is shown in table
1621 it('Shows the public key in the table', function(done) {
1622 driver.findElement(By.css('.phrase'))
1623 .sendKeys('abandon abandon ability');
1624 driver.sleep(generateDelay).then(function() {
1625 driver.findElements(By.css('.pubkey'))
1626 .then(function(els) {
1628 .then(function(pubkey) {
1629 expect(pubkey).toBe("033f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3
");
1636 // Public key visibility can be toggled
1637 it('Can toggle visibility of the public key column', function(done) {
1638 driver.findElement(By.css('.phrase'))
1639 .sendKeys('abandon abandon ability');
1640 driver.sleep(generateDelay).then(function() {
1641 driver.findElement(By.css('.public-key-toggle'))
1643 testColumnValuesAreInvisible(done, "pubkey
");
1647 // Private key is shown in table
1648 it('Shows the private key in the table', function(done) {
1649 driver.findElement(By.css('.phrase'))
1650 .sendKeys('abandon abandon ability');
1651 driver.sleep(generateDelay).then(function() {
1652 driver.findElements(By.css('.privkey'))
1653 .then(function(els) {
1655 .then(function(pubkey) {
1656 expect(pubkey).toBe("L26cVSpWFkJ6aQkPkKmTzLqTdLJ923e6CzrVh9cmx21QHsoUmrEE
");
1663 // Private key visibility can be toggled
1664 it('Can toggle visibility of the private key column', function(done) {
1665 driver.findElement(By.css('.phrase'))
1666 .sendKeys('abandon abandon ability');
1667 driver.sleep(generateDelay).then(function() {
1668 driver.findElement(By.css('.private-key-toggle'))
1670 testColumnValuesAreInvisible(done, "privkey
");
1674 // More addresses can be generated
1675 it('Can generate more rows in the table', function(done) {
1676 driver.findElement(By.css('.phrase'))
1677 .sendKeys('abandon abandon ability');
1678 driver.sleep(generateDelay).then(function() {
1679 driver.findElement(By.css('.more'))
1681 driver.sleep(generateDelay).then(function() {
1682 driver.findElements(By.css('.address'))
1683 .then(function(els) {
1684 expect(els.length).toBe(40);
1691 // A custom number of additional addresses can be generated
1692 it('Can generate more rows in the table', function(done) {
1693 driver.findElement(By.css('.phrase'))
1694 .sendKeys('abandon abandon ability');
1695 driver.sleep(generateDelay).then(function() {
1696 driver.findElement(By.css('.rows-to-add'))
1698 driver.findElement(By.css('.rows-to-add'))
1700 driver.findElement(By.css('.more'))
1702 driver.sleep(generateDelay).then(function() {
1703 driver.findElements(By.css('.address'))
1704 .then(function(els) {
1705 expect(els.length).toBe(21);
1712 // Additional addresses are shown in order of derivation path
1713 it('Shows additional addresses in order of derivation path', function(done) {
1714 driver.findElement(By.css('.phrase'))
1715 .sendKeys('abandon abandon ability');
1716 driver.sleep(generateDelay).then(function() {
1717 driver.findElement(By.css('.more'))
1719 driver.sleep(generateDelay).then(function() {
1720 testRowsAreInCorrectOrder(done);
1725 // BIP32 root key can be set by the user
1726 it('Allows the user to set the BIP32 root key', function(done) {
1727 driver.findElement(By.css('.root-key'))
1728 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1729 driver.sleep(generateDelay).then(function() {
1730 getFirstAddress(function(address) {
1731 expect(address).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug
");
1737 // Setting BIP32 root key clears the existing phrase, passphrase and seed
1738 // TODO this doesn't work in selenium with chrome
1739 it('Confirms the existing phrase should be cleared', function(done) {
1740 if (browser == "chrome
") {
1741 pending("Selenium
+ Chrome headless bug
for alert
, see
https://stackoverflow.com/q/45242264");
1743 driver
.findElement(By
.css('.phrase'))
1744 .sendKeys('A non-blank but invalid value');
1745 driver
.findElement(By
.css('.root-key'))
1746 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1747 driver
.switchTo().alert().accept();
1748 driver
.findElement(By
.css('.phrase'))
1749 .getAttribute("value").then(function(value
) {
1750 expect(value
).toBe("");
1755 // Clearing of phrase, passphrase and seed can be cancelled by user
1756 // TODO this doesn't work in selenium with chrome
1757 it('Allows the clearing of the phrase to be cancelled', function(done
) {
1758 if (browser
== "chrome") {
1759 pending("Selenium + Chrome headless bug for alert, see https://stackoverflow.com/q/45242264");
1761 driver
.findElement(By
.css('.phrase'))
1762 .sendKeys('abandon abandon ability');
1763 driver
.sleep(generateDelay
).then(function() {
1764 driver
.findElement(By
.css('.root-key'))
1766 driver
.findElement(By
.css('.root-key'))
1768 driver
.switchTo().alert().dismiss();
1769 driver
.findElement(By
.css('.phrase'))
1770 .getAttribute("value").then(function(value
) {
1771 expect(value
).toBe("abandon abandon ability");
1777 // Custom BIP32 root key is used when changing the derivation path
1778 it('Can set derivation path for root key instead of phrase', function(done
) {
1779 driver
.findElement(By
.css('#bip44 .account'))
1781 driver
.findElement(By
.css('.root-key'))
1782 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1783 driver
.sleep(generateDelay
).then(function() {
1784 getFirstAddress(function(address
) {
1785 expect(address
).toBe("1Nq2Wmu726XHCuGhctEtGmhxo3wzk5wZ1H");
1791 // Incorrect mnemonic shows error
1792 it('Shows an error for incorrect mnemonic', function(done
) {
1793 driver
.findElement(By
.css('.phrase'))
1794 .sendKeys('abandon abandon abandon');
1795 driver
.sleep(feedbackDelay
).then(function() {
1796 driver
.findElement(By
.css('.feedback'))
1798 .then(function(feedback
) {
1799 expect(feedback
).toBe("Invalid mnemonic");
1805 // Incorrect word shows suggested replacement
1806 it('Shows word suggestion for incorrect word', function(done
) {
1807 driver
.findElement(By
.css('.phrase'))
1808 .sendKeys('abandon abandon abiliti');
1809 driver
.sleep(feedbackDelay
).then(function() {
1810 driver
.findElement(By
.css('.feedback'))
1812 .then(function(feedback
) {
1813 var msg
= "abiliti not in wordlist, did you mean ability?";
1814 expect(feedback
).toBe(msg
);
1820 // Github pull request 48
1821 // First four letters of word shows that word, not closest
1822 // since first four letters gives unique word in BIP39 wordlist
1823 // eg ille should show illegal, not idle
1824 it('Shows word suggestion based on first four chars', function(done
) {
1825 driver
.findElement(By
.css('.phrase'))
1827 driver
.sleep(feedbackDelay
).then(function() {
1828 driver
.findElement(By
.css('.feedback'))
1830 .then(function(feedback
) {
1831 var msg
= "ille not in wordlist, did you mean illegal?";
1832 expect(feedback
).toBe(msg
);
1838 // Incorrect BIP32 root key shows error
1839 it('Shows error for incorrect root key', function(done
) {
1840 driver
.findElement(By
.css('.root-key'))
1841 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpj');
1842 driver
.sleep(feedbackDelay
).then(function() {
1843 driver
.findElement(By
.css('.feedback'))
1845 .then(function(feedback
) {
1846 var msg
= "Invalid root key";
1847 expect(feedback
).toBe(msg
);
1853 // Derivation path not starting with m shows error
1854 it('Shows error for derivation path not starting with m', function(done
) {
1855 driver
.findElement(By
.css('#bip32-tab a'))
1857 driver
.findElement(By
.css('#bip32 .path'))
1859 driver
.findElement(By
.css('#bip32 .path'))
1861 driver
.findElement(By
.css('.phrase'))
1862 .sendKeys('abandon abandon ability');
1863 driver
.sleep(feedbackDelay
).then(function() {
1864 driver
.findElement(By
.css('.feedback'))
1866 .then(function(feedback
) {
1867 var msg
= "First character must be 'm'";
1868 expect(feedback
).toBe(msg
);
1874 // Derivation path containing invalid characters shows useful error
1875 it('Shows error for derivation path not starting with m', function(done
) {
1876 driver
.findElement(By
.css('#bip32-tab a'))
1878 driver
.findElement(By
.css('#bip32 .path'))
1880 driver
.findElement(By
.css('#bip32 .path'))
1881 .sendKeys('m/1/0wrong1/1');
1882 driver
.findElement(By
.css('.phrase'))
1883 .sendKeys('abandon abandon ability');
1884 driver
.sleep(feedbackDelay
).then(function() {
1885 driver
.findElement(By
.css('.feedback'))
1887 .then(function(feedback
) {
1888 var msg
= "Invalid characters 0wrong1 found at depth 2";
1889 expect(feedback
).toBe(msg
);
1895 // Github Issue 11: Default word length is 15
1896 // https://github.com/iancoleman/bip39/issues/11
1897 it('Sets the default word length to 15', function(done
) {
1898 driver
.findElement(By
.css('.strength'))
1899 .getAttribute("value")
1900 .then(function(strength
) {
1901 expect(strength
).toBe("15");
1906 // Github Issue 12: Generate more rows with private keys hidden
1907 // https://github.com/iancoleman/bip39/issues/12
1908 it('Sets the correct hidden column state on new rows', function(done
) {
1909 driver
.findElement(By
.css('.phrase'))
1910 .sendKeys("abandon abandon ability");
1911 driver
.sleep(generateDelay
).then(function() {
1912 driver
.findElement(By
.css('.private-key-toggle'))
1914 driver
.findElement(By
.css('.more'))
1916 driver
.sleep(generateDelay
).then(function() {
1917 driver
.findElements(By
.css('.privkey'))
1918 .then(function(els
) {
1919 expect(els
.length
).toBe(40);
1921 testColumnValuesAreInvisible(done
, "privkey");
1926 // Github Issue 19: Mnemonic is not sensitive to whitespace
1927 // https://github.com/iancoleman/bip39/issues/19
1928 it('Ignores excess whitespace in the mnemonic', function(done
) {
1929 var doublespace
= " ";
1930 var mnemonic
= "urge cat" + doublespace
+ "bid";
1931 driver
.findElement(By
.css('.phrase'))
1932 .sendKeys(mnemonic
);
1933 driver
.sleep(generateDelay
).then(function() {
1934 driver
.findElement(By
.css('.root-key'))
1935 .getAttribute("value")
1936 .then(function(seed
) {
1937 expect(seed
).toBe("xprv9s21ZrQH143K3isaZsWbKVoTtbvd34Y1ZGRugGdMeBGbM3AgBVzTH159mj1cbbtYSJtQr65w6L5xy5L9SFC7c9VJZWHxgAzpj4mun5LhrbC");
1943 // Github Issue 23: Part 1: Use correct derivation path when changing tabs
1944 // https://github.com/iancoleman/bip39/issues/23
1945 it('Uses the correct derivation path when changing tabs', function(done
) {
1946 // 1) and 2) set the phrase
1947 driver
.findElement(By
.css('.phrase'))
1948 .sendKeys("abandon abandon ability");
1949 driver
.sleep(generateDelay
).then(function() {
1950 // 3) select bip32 tab
1951 driver
.findElement(By
.css('#bip32-tab a'))
1953 driver
.sleep(generateDelay
).then(function() {
1954 // 4) switch from bitcoin to litecoin
1955 selectNetwork("LTC - Litecoin");
1956 driver
.sleep(generateDelay
).then(function() {
1957 // 5) Check address is displayed correctly
1958 getFirstAddress(function(address
) {
1959 expect(address
).toBe("LS8MP5LZ5AdzSZveRrjm3aYVoPgnfFh5T5");
1960 // 5) Check derivation path is displayed correctly
1961 getFirstPath(function(path
) {
1962 expect(path
).toBe("m/0/0");
1971 // Github Issue 23 Part 2: Coin selection in derivation path
1972 // https://github.com/iancoleman/bip39/issues/23#issuecomment-238011920
1973 it('Uses the correct derivation path when changing coins', function(done
) {
1975 driver
.findElement(By
.css('.phrase'))
1976 .sendKeys("abandon abandon ability");
1977 driver
.sleep(generateDelay
).then(function() {
1978 // switch from bitcoin to clam
1979 selectNetwork("CLAM - Clams");
1980 driver
.sleep(generateDelay
).then(function() {
1981 // check derivation path is displayed correctly
1982 getFirstPath(function(path
) {
1983 expect(path
).toBe("m/44'/23'/0'/0/0");
1990 // Github Issue 26: When using a Root key derrived altcoins are incorrect
1991 // https://github.com/iancoleman/bip39/issues/26
1992 it('Uses the correct derivation for altcoins with root keys', function(done
) {
1993 // 1) 2) and 3) set the root key
1994 driver
.findElement(By
.css('.root-key'))
1995 .sendKeys("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
1996 driver
.sleep(generateDelay
).then(function() {
1997 // 4) switch from bitcoin to viacoin
1998 selectNetwork("VIA - Viacoin");
1999 driver
.sleep(generateDelay
).then(function() {
2000 // 5) ensure the derived address is correct
2001 getFirstAddress(function(address
) {
2002 expect(address
).toBe("Vq9Eq4N5SQnjqZvxtxzo7hZPW5XnyJsmXT");
2009 // Selecting a language with no existing phrase should generate a phrase in
2011 it('Generate a random phrase when language is selected and no current phrase', function(done
) {
2012 driver
.findElement(By
.css("a[href='#japanese']"))
2014 driver
.sleep(generateDelay
).then(function() {
2015 driver
.findElement(By
.css(".phrase"))
2016 .getAttribute("value").then(function(phrase
) {
2017 expect(phrase
.search(/[a-z]/)).toBe(-1);
2018 expect(phrase
.length
).toBeGreaterThan(0);
2024 // Selecting a language with existing phrase should update the phrase to use
2026 it('Updates existing phrases when the language is changed', function(done
) {
2027 driver
.findElement(By
.css(".phrase"))
2028 .sendKeys("abandon abandon ability");
2029 driver
.sleep(generateDelay
).then(function() {
2030 driver
.findElement(By
.css("a[href='#italian']"))
2032 driver
.sleep(generateDelay
).then(function() {
2033 driver
.findElement(By
.css(".phrase"))
2034 .getAttribute("value").then(function(phrase
) {
2035 // Check only the language changes, not the phrase
2036 expect(phrase
).toBe("abaco abaco abbaglio");
2037 getFirstAddress(function(address
) {
2038 // Check the address is correct
2039 expect(address
).toBe("1Dz5TgDhdki9spa6xbPFbBqv5sjMrx3xgV");
2047 // Suggested replacement for erroneous word in non-English language
2048 it('Shows word suggestion for incorrect word in non-English language', function(done
) {
2049 driver
.findElement(By
.css('.phrase'))
2050 .sendKeys('abaco abaco zbbaglio');
2051 driver
.sleep(feedbackDelay
).then(function() {
2052 driver
.findElement(By
.css('.feedback'))
2054 .then(function(feedback
) {
2055 var msg
= "zbbaglio not in wordlist, did you mean abbaglio?";
2056 expect(feedback
).toBe(msg
);
2062 // Japanese word does not break across lines.
2064 // https://github.com/bitcoin/bips/blob/master/bip-0039/bip-0039-wordlists.md#japanese
2065 it('Does not break Japanese words across lines', function(done
) {
2066 driver
.findElement(By
.css('.phrase'))
2067 .getCssValue("word-break")
2068 .then(function(value
) {
2069 expect(value
).toBe("keep-all");
2074 // Language can be specified at page load using hash value in url
2075 it('Can set the language from the url hash', function(done
) {
2076 driver
.get(url
+ "#japanese").then(function() {
2077 driver
.findElement(By
.css('.generate')).click();
2078 driver
.sleep(generateDelay
).then(function() {
2079 driver
.findElement(By
.css(".phrase"))
2080 .getAttribute("value").then(function(phrase
) {
2081 expect(phrase
.search(/[a-z]/)).toBe(-1);
2082 expect(phrase
.length
).toBeGreaterThan(0);
2089 // Entropy can be entered by the user
2090 it('Allows entropy to be entered', function(done
) {
2091 driver
.findElement(By
.css('.use-entropy'))
2093 driver
.findElement(By
.css('.entropy'))
2094 .sendKeys('00000000 00000000 00000000 00000000');
2095 driver
.sleep(generateDelay
).then(function() {
2096 driver
.findElement(By
.css(".phrase"))
2097 .getAttribute("value").then(function(phrase
) {
2098 expect(phrase
).toBe("abandon abandon ability");
2099 getFirstAddress(function(address
) {
2100 expect(address
).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug");
2107 // A warning about entropy is shown to the user, with additional information
2108 it('Shows a warning about using entropy', function(done
) {
2109 driver
.findElement(By
.css('.use-entropy'))
2111 driver
.findElement(By
.css('.entropy-container'))
2113 .then(function(containerText
) {
2114 var warning
= "mnemonic may be insecure";
2115 expect(containerText
).toContain(warning
);
2116 driver
.findElement(By
.css('#entropy-notes'))
2117 .findElement(By
.xpath("parent::*"))
2119 .then(function(notesText
) {
2120 var detail
= "flipping a fair coin, rolling a fair dice, noise measurements etc";
2121 expect(notesText
).toContain(detail
);
2127 // The types of entropy available are described to the user
2128 it('Shows the types of entropy available', function(done
) {
2129 driver
.findElement(By
.css('.entropy'))
2130 .getAttribute("placeholder")
2131 .then(function(placeholderText
) {
2140 for (var i
=0; i
<options
.length
; i
++) {
2141 var option
= options
[i
];
2142 expect(placeholderText
).toContain(option
);
2148 // The actual entropy used is shown to the user
2149 it('Shows the actual entropy used', function(done
) {
2150 driver
.findElement(By
.css('.use-entropy'))
2152 driver
.findElement(By
.css('.entropy'))
2153 .sendKeys('Not A Very Good Entropy Source At All');
2154 driver
.sleep(generateDelay
).then(function() {
2155 driver
.findElement(By
.css('.entropy-container'))
2157 .then(function(text
) {
2158 expect(text
).toMatch(/Filtered Entropy
\s
+AedEceAA
/);
2164 // Binary entropy can be entered
2165 it('Allows binary entropy to be entered', function(done
) {
2166 testEntropyType(done
, "01", "binary");
2169 // Base 6 entropy can be entered
2170 it('Allows base 6 entropy to be entered', function(done
) {
2171 testEntropyType(done
, "012345", "base 6");
2174 // Base 6 dice entropy can be entered
2175 it('Allows base 6 dice entropy to be entered', function(done
) {
2176 testEntropyType(done
, "123456", "base 6 (dice)");
2179 // Base 10 entropy can be entered
2180 it('Allows base 10 entropy to be entered', function(done
) {
2181 testEntropyType(done
, "789", "base 10");
2184 // Hexadecimal entropy can be entered
2185 it('Allows hexadecimal entropy to be entered', function(done
) {
2186 testEntropyType(done
, "abcdef", "hexadecimal");
2189 // Dice entropy value is shown as the converted base 6 value
2190 // ie 123456 is converted to 123450
2191 it('Shows dice entropy as base 6', function(done
) {
2192 driver
.findElement(By
.css('.use-entropy'))
2194 driver
.findElement(By
.css('.entropy'))
2195 .sendKeys("123456");
2196 driver
.sleep(generateDelay
).then(function() {
2197 driver
.findElement(By
.css('.entropy-container'))
2199 .then(function(text
) {
2200 expect(text
).toMatch(/Filtered Entropy
\s
+123450/);
2206 // The number of bits of entropy accumulated is shown
2207 it("Shows the number of bits of entropy for 20 bits of binary", function(done
) {
2208 testEntropyBits(done
, "0000 0000 0000 0000 0000", "20");
2210 it("Shows the number of bits of entropy for 1 bit of binary", function(done
) {
2211 testEntropyBits(done
, "0", "1");
2213 it("Shows the number of bits of entropy for 4 bits of binary", function(done
) {
2214 testEntropyBits(done
, "0000", "4");
2216 it("Shows the number of bits of entropy for 1 character of base 6 (dice)", function(done
) {
2217 // 6 in card is 0 in base 6, 0 in base 6 is 2.6 bits (rounded down to 2 bits)
2218 testEntropyBits(done
, "6", "2");
2220 it("Shows the number of bits of entropy for 1 character of base 10 with 3 bits", function(done
) {
2221 // 7 in base 10 is 111 in base 2, no leading zeros
2222 testEntropyBits(done
, "7", "3");
2224 it("Shows the number of bits of entropy for 1 character of base 10 with 4 bis", function(done
) {
2225 testEntropyBits(done
, "8", "4");
2227 it("Shows the number of bits of entropy for 1 character of hex", function(done
) {
2228 testEntropyBits(done
, "F", "4");
2230 it("Shows the number of bits of entropy for 2 characters of base 10", function(done
) {
2231 testEntropyBits(done
, "29", "6");
2233 it("Shows the number of bits of entropy for 2 characters of hex", function(done
) {
2234 testEntropyBits(done
, "0A", "8");
2236 it("Shows the number of bits of entropy for 2 characters of hex with 3 leading zeros", function(done
) {
2237 // hex is always multiple of 4 bits of entropy
2238 testEntropyBits(done
, "1A", "8");
2240 it("Shows the number of bits of entropy for 2 characters of hex with 2 leading zeros", function(done
) {
2241 testEntropyBits(done
, "2A", "8");
2243 it("Shows the number of bits of entropy for 2 characters of hex with 1 leading zero", function(done
) {
2244 testEntropyBits(done
, "4A", "8");
2246 it("Shows the number of bits of entropy for 2 characters of hex with no leading zeros", function(done
) {
2247 testEntropyBits(done
, "8A", "8");
2249 it("Shows the number of bits of entropy for 2 characters of hex starting with F", function(done
) {
2250 testEntropyBits(done
, "FA", "8");
2252 it("Shows the number of bits of entropy for 4 characters of hex with leading zeros", function(done
) {
2253 testEntropyBits(done
, "000A", "16");
2255 it("Shows the number of bits of entropy for 4 characters of base 6", function(done
) {
2256 testEntropyBits(done
, "5555", "11");
2258 it("Shows the number of bits of entropy for 4 characters of base 6 dice", function(done
) {
2259 // uses dice, so entropy is actually 0000 in base 6, which is 4 lots of
2260 // 2.58 bits, which is 10.32 bits (rounded down to 10 bits)
2261 testEntropyBits(done
, "6666", "10");
2263 it("Shows the number of bits of entropy for 4 charactes of base 10", function(done
) {
2264 // Uses base 10, which is 4 lots of 3.32 bits, which is 13.3 bits (rounded
2266 testEntropyBits(done
, "2227", "13");
2268 it("Shows the number of bits of entropy for 4 characters of hex with 2 leading zeros", function(done
) {
2269 testEntropyBits(done
, "222F", "16");
2271 it("Shows the number of bits of entropy for 4 characters of hex starting with F", function(done
) {
2272 testEntropyBits(done
, "FFFF", "16");
2274 it("Shows the number of bits of entropy for 10 characters of base 10", function(done
) {
2275 // 10 events at 3.32 bits per event
2276 testEntropyBits(done
, "0000101017", "33");
2278 it("Shows the number of bits of entropy for a full deck of cards", function(done
) {
2279 // cards are not replaced, so a full deck is not 52^52 entropy which is 296
2280 // bits, it's 52!, which is 225 bits
2281 testEntropyBits(done
, "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks", "225");
2284 it("Shows details about the entered entropy", function(done
) {
2285 testEntropyFeedback(done
,
2289 type: "hexadecimal",
2293 strength: "less than a second",
2297 it("Shows details about the entered entropy", function(done
) {
2298 testEntropyFeedback(done
,
2300 entropy: "AAAAAAAA",
2301 filtered: "AAAAAAAA",
2302 type: "hexadecimal",
2306 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2310 it("Shows details about the entered entropy", function(done
) {
2311 testEntropyFeedback(done
,
2313 entropy: "AAAAAAAA B",
2314 filtered: "AAAAAAAAB",
2315 type: "hexadecimal",
2319 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2323 it("Shows details about the entered entropy", function(done
) {
2324 testEntropyFeedback(done
,
2326 entropy: "AAAAAAAA BBBBBBBB",
2327 filtered: "AAAAAAAABBBBBBBB",
2328 type: "hexadecimal",
2332 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2336 it("Shows details about the entered entropy", function(done
) {
2337 testEntropyFeedback(done
,
2339 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC",
2340 filtered: "AAAAAAAABBBBBBBBCCCCCCCC",
2341 type: "hexadecimal",
2345 strength: "less than a second",
2349 it("Shows details about the entered entropy", function(done
) {
2350 testEntropyFeedback(done
,
2352 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDD",
2353 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDD",
2354 type: "hexadecimal",
2358 strength: "2 minutes",
2362 it("Shows details about the entered entropy", function(done
) {
2363 testEntropyFeedback(done
,
2365 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA",
2366 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDA",
2367 type: "hexadecimal",
2375 it("Shows details about the entered entropy", function(done
) {
2376 testEntropyFeedback(done
,
2378 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA EEEEEEEE",
2379 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDAEEEEEEEE",
2380 type: "hexadecimal",
2384 strength: "3 years",
2388 it("Shows details about the entered entropy", function(done
) {
2389 testEntropyFeedback(done
,
2391 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA EEEEEEEE FFFFFFFF",
2392 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDAEEEEEEEEFFFFFFFF",
2393 type: "hexadecimal",
2397 strength: "centuries",
2401 it("Shows details about the entered entropy", function(done
) {
2402 testEntropyFeedback(done
,
2409 strength: "less than a second",
2413 it("Shows details about the entered entropy", function(done
) {
2414 testEntropyFeedback(done
,
2416 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2417 type: "card (full deck)",
2421 strength: "centuries",
2425 it("Shows details about the entered entropy", function(done
) {
2426 testEntropyFeedback(done
,
2428 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks3d",
2429 type: "card (full deck, 1 duplicate: 3d)",
2433 strength: "centuries",
2437 it("Shows details about the entered entropy", function(done
) {
2438 testEntropyFeedback(done
,
2440 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqs3d4d",
2441 type: "card (2 duplicates: 3d 4d, 1 missing: KS)",
2445 strength: "centuries",
2449 it("Shows details about the entered entropy", function(done
) {
2450 testEntropyFeedback(done
,
2452 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqs3d4d5d6d",
2453 type: "card (4 duplicates: 3d 4d 5d..., 1 missing: KS)",
2457 strength: "centuries",
2461 it("Shows details about the entered entropy", function(done
) {
2462 testEntropyFeedback(done
,
2463 // Next test was throwing uncaught error in zxcvbn
2464 // Also tests 451 bits, ie Math.log2(52!)*2 = 225.58 * 2
2466 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsksac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2467 type: "card (full deck, 52 duplicates: ac 2c 3c...)",
2471 strength: "centuries",
2475 it("Shows details about the entered entropy", function(done
) {
2476 testEntropyFeedback(done
,
2477 // Case insensitivity to duplicate cards
2480 type: "card (1 duplicate: AS)",
2484 strength: "less than a second",
2488 it("Shows details about the entered entropy", function(done
) {
2489 testEntropyFeedback(done
,
2492 type: "card (1 duplicate: as)",
2496 strength: "less than a second",
2500 it("Shows details about the entered entropy", function(done
) {
2501 testEntropyFeedback(done
,
2502 // Missing cards are detected
2504 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2505 type: "card (1 missing: 9C)",
2509 strength: "centuries",
2513 it("Shows details about the entered entropy", function(done
) {
2514 testEntropyFeedback(done
,
2516 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2517 type: "card (2 missing: 9C 5D)",
2521 strength: "centuries",
2525 it("Shows details about the entered entropy", function(done
) {
2526 testEntropyFeedback(done
,
2528 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d7d8d9dtdjd kdah2h3h 5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2529 type: "card (4 missing: 9C 5D QD...)",
2533 strength: "centuries",
2537 it("Shows details about the entered entropy", function(done
) {
2538 testEntropyFeedback(done
,
2539 // More than six missing cards does not show message
2541 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d 8d9d jd kdah2h3h 5h6h7h8h9hthjhqhkh 2s3s4s5s6s7s8s9stsjsqsks",
2546 strength: "centuries",
2550 it("Shows details about the entered entropy", function(done
) {
2551 testEntropyFeedback(done
,
2552 // Multiple decks of cards increases bits per event
2557 bitsPerEvent: "4.34",
2561 it("Shows details about the entered entropy", function(done
) {
2562 testEntropyFeedback(done
,
2567 bitsPerEvent: "4.80",
2571 it("Shows details about the entered entropy", function(done
) {
2572 testEntropyFeedback(done
,
2577 bitsPerEvent: "5.01",
2581 it("Shows details about the entered entropy", function(done
) {
2582 testEntropyFeedback(done
,
2584 entropy: "3d3d3d3d",
2587 bitsPerEvent: "5.14",
2591 it("Shows details about the entered entropy", function(done
) {
2592 testEntropyFeedback(done
,
2594 entropy: "3d3d3d3d3d",
2597 bitsPerEvent: "5.22",
2601 it("Shows details about the entered entropy", function(done
) {
2602 testEntropyFeedback(done
,
2604 entropy: "3d3d3d3d3d3d",
2607 bitsPerEvent: "5.28",
2611 it("Shows details about the entered entropy", function(done
) {
2612 testEntropyFeedback(done
,
2614 entropy: "3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d",
2617 bitsPerEvent: "5.59",
2618 strength: 'less than a second - Repeats like "abcabcabc" are only slightly harder to guess than "abc"',
2623 // Entropy is truncated from the left
2624 it('Truncates entropy from the left', function(done
) {
2625 // Truncate from left means 0000 is removed from the start
2626 // which gives mnemonic 'avocado zoo zone'
2627 // not 1111 removed from the end
2628 // which gives the mnemonic 'abstract zoo zoo'
2629 var entropy
= "00000000 00000000 00000000 00000000";
2630 entropy
+= "11111111 11111111 11111111 1111"; // Missing last byte
2631 driver
.findElement(By
.css('.use-entropy'))
2633 driver
.findElement(By
.css('.entropy'))
2635 driver
.sleep(generateDelay
).then(function() {
2636 driver
.findElement(By
.css(".phrase"))
2637 .getAttribute("value").then(function(phrase
) {
2638 expect(phrase
).toBe("avocado zoo zone");
2644 // Very large entropy results in very long mnemonics
2645 it('Converts very long entropy to very long mnemonics', function(done
) {
2647 for (var i
=0; i
<33; i
++) {
2648 entropy
+= "AAAAAAAA"; // 3 words * 33 iterations = 99 words
2650 driver
.findElement(By
.css('.use-entropy'))
2652 driver
.findElement(By
.css('.entropy'))
2654 driver
.sleep(generateDelay
).then(function() {
2655 driver
.findElement(By
.css(".phrase"))
2656 .getAttribute("value").then(function(phrase
) {
2657 var wordCount
= phrase
.split(/\s+/g).length
;
2658 expect(wordCount
).toBe(99);
2664 // Is compatible with bip32jp entropy
2665 // https://bip32jp.github.io/english/index.html
2667 // Is incompatible with:
2669 it('Is compatible with bip32jp.github.io', function(done
) {
2670 var entropy
= "543210543210543210543210543210543210543210543210543210543210543210543210543210543210543210543210543";
2671 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";
2672 driver
.findElement(By
.css('.use-entropy'))
2674 driver
.findElement(By
.css('.entropy'))
2676 driver
.sleep(generateDelay
).then(function() {
2677 driver
.findElement(By
.css(".phrase"))
2678 .getAttribute("value").then(function(phrase
) {
2679 expect(phrase
).toBe(expectedPhrase
);
2685 // Blank entropy does not generate mnemonic or addresses
2686 it('Does not generate mnemonic for blank entropy', function(done
) {
2687 driver
.findElement(By
.css('.use-entropy'))
2689 driver
.findElement(By
.css('.entropy'))
2691 // check there is no mnemonic
2692 driver
.sleep(generateDelay
).then(function() {
2693 driver
.findElement(By
.css(".phrase"))
2694 .getAttribute("value").then(function(phrase
) {
2695 expect(phrase
).toBe("");
2696 // check there is no mnemonic
2697 driver
.findElements(By
.css(".address"))
2698 .then(function(addresses
) {
2699 expect(addresses
.length
).toBe(0);
2700 // Check the feedback says 'blank entropy'
2701 driver
.findElement(By
.css(".feedback"))
2703 .then(function(feedbackText
) {
2704 expect(feedbackText
).toBe("Blank entropy");
2712 // Mnemonic length can be selected even for weak entropy
2713 it('Allows selection of mnemonic length even for weak entropy', function(done
) {
2714 driver
.findElement(By
.css('.use-entropy'))
2716 driver
.executeScript(function() {
2717 $(".mnemonic-length").val("18").trigger("change");
2719 driver
.findElement(By
.css('.entropy'))
2720 .sendKeys("012345");
2721 driver
.sleep(generateDelay
).then(function() {
2722 driver
.findElement(By
.css(".phrase"))
2723 .getAttribute("value").then(function(phrase
) {
2724 var wordCount
= phrase
.split(/\s+/g).length
;
2725 expect(wordCount
).toBe(18);
2732 // https://github.com/iancoleman/bip39/issues/33
2733 // Final cards should contribute entropy
2734 it('Uses as much entropy as possible for the mnemonic', function(done
) {
2735 driver
.findElement(By
.css('.use-entropy'))
2737 driver
.findElement(By
.css('.entropy'))
2738 .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");
2739 driver
.sleep(generateDelay
).then(function() {
2741 driver
.findElement(By
.css(".phrase"))
2742 .getAttribute("value").then(function(originalPhrase
) {
2743 // Set the last 12 cards to be AS
2744 driver
.findElement(By
.css('.entropy'))
2746 driver
.findElement(By
.css('.entropy'))
2747 .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");
2748 driver
.sleep(generateDelay
).then(function() {
2750 driver
.findElement(By
.css(".phrase"))
2751 .getAttribute("value").then(function(newPhrase
) {
2752 expect(originalPhrase
).not
.toEqual(newPhrase
);
2761 // https://github.com/iancoleman/bip39/issues/35
2763 // TODO this doesn't work in selenium with firefox
2764 // see https://stackoverflow.com/q/40360223
2765 it('Shows a qr code on hover for the phrase', function(done
) {
2766 if (browser
== "firefox") {
2767 pending("Selenium + Firefox bug for mouseMove, see https://stackoverflow.com/q/40360223");
2769 // generate a random mnemonic
2770 var generateEl
= driver
.findElement(By
.css('.generate'));
2772 // toggle qr to show (hidden by default)
2773 var phraseEl
= driver
.findElement(By
.css(".phrase"));
2775 var rootKeyEl
= driver
.findElement(By
.css(".root-key"));
2776 driver
.sleep(generateDelay
).then(function() {
2777 // hover over the root key
2778 driver
.actions().mouseMove(rootKeyEl
).perform().then(function() {
2779 // check the qr code shows
2780 driver
.executeScript(function() {
2781 return $(".qr-container").find("canvas").length
> 0;
2783 .then(function(qrShowing
) {
2784 expect(qrShowing
).toBe(true);
2785 // hover away from the phrase
2786 driver
.actions().mouseMove(generateEl
).perform().then(function() {;
2787 // check the qr code hides
2788 driver
.executeScript(function() {
2789 return $(".qr-container").find("canvas").length
== 0;
2791 .then(function(qrHidden
) {
2792 expect(qrHidden
).toBe(true);
2801 // BIP44 account extendend private key is shown
2802 // github issue 37 - compatibility with electrum
2803 it('Shows the bip44 account extended private key', function(done
) {
2804 driver
.findElement(By
.css(".phrase"))
2805 .sendKeys("abandon abandon ability");
2806 driver
.sleep(generateDelay
).then(function() {
2807 driver
.findElement(By
.css("#bip44 .account-xprv"))
2808 .getAttribute("value")
2809 .then(function(xprv
) {
2810 expect(xprv
).toBe("xprv9yzrnt4zWVJUr1k2VxSPy9ettKz5PpeDMgaVG7UKedhqnw1tDkxP2UyYNhuNSumk2sLE5ctwKZs9vwjsq3e1vo9egCK6CzP87H2cVYXpfwQ");
2816 // BIP44 account extendend public key is shown
2817 // github issue 37 - compatibility with electrum
2818 it('Shows the bip44 account extended public key', function(done
) {
2819 driver
.findElement(By
.css(".phrase"))
2820 .sendKeys("abandon abandon ability");
2821 driver
.sleep(generateDelay
).then(function() {
2822 driver
.findElement(By
.css("#bip44 .account-xpub"))
2823 .getAttribute("value")
2824 .then(function(xprv
) {
2825 expect(xprv
).toBe("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2832 // BIP32 root key can be set as an xpub
2833 it('Generates addresses from xpub as bip32 root key', function(done
) {
2834 driver
.findElement(By
.css('#bip32-tab a'))
2836 // set xpub for account 0 of bip44 for 'abandon abandon ability'
2837 driver
.findElement(By
.css("#root-key"))
2838 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2839 driver
.sleep(generateDelay
).then(function() {
2840 // check the addresses are generated
2841 getFirstAddress(function(address
) {
2842 expect(address
).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug");
2843 // check the xprv key is not set
2844 driver
.findElement(By
.css(".extended-priv-key"))
2845 .getAttribute("value")
2846 .then(function(xprv
) {
2847 expect(xprv
).toBe("NA");
2848 // check the private key is not set
2849 driver
.findElements(By
.css(".privkey"))
2850 .then(function(els
) {
2853 .then(function(privkey
) {
2854 expect(xprv
).toBe("NA");
2864 // xpub for bip32 root key will not work with hardened derivation paths
2865 it('Shows error for hardened derivation paths with xpub root key', function(done
) {
2866 // set xpub for account 0 of bip44 for 'abandon abandon ability'
2867 driver
.findElement(By
.css("#root-key"))
2868 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2869 driver
.sleep(feedbackDelay
).then(function() {
2870 // Check feedback is correct
2871 driver
.findElement(By
.css('.feedback'))
2873 .then(function(feedback
) {
2874 var msg
= "Hardened derivation path is invalid with xpub key";
2875 expect(feedback
).toBe(msg
);
2876 // Check no addresses are shown
2877 driver
.findElements(By
.css('.addresses tr'))
2878 .then(function(rows
) {
2879 expect(rows
.length
).toBe(0);
2887 // no root key shows feedback
2888 it('Shows feedback for no root key', function(done
) {
2889 // set xpub for account 0 of bip44 for 'abandon abandon ability'
2890 driver
.findElement(By
.css('#bip32-tab a'))
2892 driver
.sleep(feedbackDelay
).then(function() {
2893 // Check feedback is correct
2894 driver
.findElement(By
.css('.feedback'))
2896 .then(function(feedback
) {
2897 expect(feedback
).toBe("Invalid root key");
2904 // display error switching tabs while addresses are generating
2905 it('Can change details while old addresses are still being generated', function(done
) {
2906 // Set to generate 199 more addresses.
2907 // This will take a long time allowing a new set of addresses to be
2908 // generated midway through this lot.
2909 // The newly generated addresses should not include any from the old set.
2910 // Any more than 199 will show an alert which needs to be accepted.
2911 driver
.findElement(By
.css('.rows-to-add'))
2913 driver
.findElement(By
.css('.rows-to-add'))
2916 driver
.findElement(By
.css('.phrase'))
2917 .sendKeys("abandon abandon ability");
2918 driver
.sleep(generateDelay
).then(function() {
2919 // change tabs which should cancel the previous generating
2920 driver
.findElement(By
.css('.rows-to-add'))
2922 driver
.findElement(By
.css('.rows-to-add'))
2924 driver
.findElement(By
.css('#bip32-tab a'))
2926 driver
.sleep(generateDelay
).then(function() {
2927 driver
.findElements(By
.css('.index'))
2928 .then(function(els
) {
2929 // check the derivation paths have the right quantity
2930 expect(els
.length
).toBe(20);
2931 // check the derivation paths are in order
2932 testRowsAreInCorrectOrder(done
);
2936 }, generateDelay
+ 10000);
2939 // padding for binary should give length with multiple of 256
2940 // hashed entropy 1111 is length 252, so requires 4 leading zeros
2941 // prior to issue 49 it would only generate 2 leading zeros, ie missing 2
2942 it('Pads hashed entropy with leading zeros', function(done
) {
2943 driver
.findElement(By
.css('.use-entropy'))
2945 driver
.executeScript(function() {
2946 $(".mnemonic-length").val("15").trigger("change");
2948 driver
.findElement(By
.css('.entropy'))
2950 driver
.sleep(generateDelay
).then(function() {
2951 driver
.findElement(By
.css('.phrase'))
2952 .getAttribute("value")
2953 .then(function(phrase
) {
2954 expect(phrase
).toBe("avocado valid quantum cross link predict excuse edit street able flame large galaxy ginger nuclear");
2960 // Github pull request 55
2961 // https://github.com/iancoleman/bip39/pull/55
2963 it('Can set the derivation path on bip32 tab for bitcoincore', function(done
) {
2964 testClientSelect(done
, {
2966 bip32path: "m/0'/0'",
2967 useHardenedAddresses: "true",
2970 it('Can set the derivation path on bip32 tab for multibit', function(done
) {
2971 testClientSelect(done
, {
2973 bip32path: "m/0'/0",
2974 useHardenedAddresses: null,
2977 it('Can set the derivation path on bip32 tab for coinomi/ledger', function(done
) {
2978 testClientSelect(done
, {
2980 bip32path: "m/44'/0'/0'",
2981 useHardenedAddresses: null,
2986 // https://github.com/iancoleman/bip39/issues/58
2987 // bip32 derivation is correct, does not drop leading zeros
2989 // https://medium.com/@alexberegszaszi/why-do-my-bip32-wallets-disagree-6f3254cc5846
2990 it('Retains leading zeros for bip32 derivation', function(done
) {
2991 driver
.findElement(By
.css(".phrase"))
2992 .sendKeys("fruit wave dwarf banana earth journey tattoo true farm silk olive fence");
2993 driver
.findElement(By
.css(".passphrase"))
2994 .sendKeys("banana");
2995 driver
.sleep(generateDelay
).then(function() {
2996 getFirstAddress(function(address
) {
2997 // Note that bitcore generates an incorrect address
2998 // 13EuKhffWkBE2KUwcbkbELZb1MpzbimJ3Y
2999 // see the medium.com link above for more details
3000 expect(address
).toBe("17rxURoF96VhmkcEGCj5LNQkmN9HVhWb7F");
3007 // Japanese mnemonics generate incorrect bip32 seed
3008 // BIP39 seed is set from phrase
3009 it('Generates correct seed for Japanese mnemonics', function(done
) {
3010 driver
.findElement(By
.css(".phrase"))
3011 .sendKeys("あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あおぞら");
3012 driver
.findElement(By
.css(".passphrase"))
3013 .sendKeys("メートルガバヴァぱばぐゞちぢ十人十色");
3014 driver
.sleep(generateDelay
).then(function() {
3015 driver
.findElement(By
.css(".seed"))
3016 .getAttribute("value")
3017 .then(function(seed
) {
3018 expect(seed
).toBe("a262d6fb6122ecf45be09c50492b31f92e9beb7d9a845987a02cefda57a15f9c467a17872029a9e92299b5cbdf306e3a0ee620245cbd508959b6cb7ca637bd55");
3024 // BIP49 official test vectors
3025 // https://github.com/bitcoin/bips/blob/master/bip-0049.mediawiki#test-vectors
3026 it('Generates BIP49 addresses matching the official test vectors', function(done
) {
3027 driver
.findElement(By
.css('#bip49-tab a'))
3029 selectNetwork("BTC - Bitcoin Testnet");
3030 driver
.findElement(By
.css(".phrase"))
3031 .sendKeys("abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about");
3032 driver
.sleep(generateDelay
).then(function() {
3033 getFirstAddress(function(address
) {
3034 expect(address
).toBe("2Mww8dCYPUpKHofjgcXcBCEGmniw9CoaiD2");
3040 // BIP49 derivation path is shown
3041 it('Shows the bip49 derivation path', function(done
) {
3042 driver
.findElement(By
.css('#bip49-tab a'))
3044 driver
.findElement(By
.css(".phrase"))
3045 .sendKeys("abandon abandon ability");
3046 driver
.sleep(generateDelay
).then(function() {
3047 driver
.findElement(By
.css('#bip49 .path'))
3048 .getAttribute("value")
3049 .then(function(path
) {
3050 expect(path
).toBe("m/49'/0'/0'/0");
3056 // BIP49 extended private key is shown
3057 it('Shows the bip49 extended private key', function(done
) {
3058 driver
.findElement(By
.css('#bip49-tab a'))
3060 driver
.findElement(By
.css(".phrase"))
3061 .sendKeys("abandon abandon ability");
3062 driver
.sleep(generateDelay
).then(function() {
3063 driver
.findElement(By
.css('.extended-priv-key'))
3064 .getAttribute("value")
3065 .then(function(xprv
) {
3066 expect(xprv
).toBe("yprvALYB4DYRG6CzzVgzQZwwqjAA2wjBGC3iEd7KYYScpoDdmf75qMRWZWxoFcRXBJjgEXdFqJ9vDRGRLJQsrL22Su5jMbNFeM9vetaGVqy9Qy2");
3072 // BIP49 extended public key is shown
3073 it('Shows the bip49 extended public key', function(done
) {
3074 driver
.findElement(By
.css('#bip49-tab a'))
3076 driver
.findElement(By
.css(".phrase"))
3077 .sendKeys("abandon abandon ability");
3078 driver
.sleep(generateDelay
).then(function() {
3079 driver
.findElement(By
.css('.extended-pub-key'))
3080 .getAttribute("value")
3081 .then(function(xprv
) {
3082 expect(xprv
).toBe("ypub6ZXXTj5K6TmJCymTWbUxCs6tayZffemZbr2vLvrEP8kceTSENtjm7KHH6thvAKxVar9fGe8rgsPEX369zURLZ68b4f7Vexz7RuXsjQ69YDt");
3088 // BIP49 account field changes address list
3089 it('Can set the bip49 account field', function(done
) {
3090 driver
.findElement(By
.css('#bip49-tab a'))
3092 driver
.findElement(By
.css('#bip49 .account'))
3094 driver
.findElement(By
.css('#bip49 .account'))
3096 driver
.findElement(By
.css(".phrase"))
3097 .sendKeys("abandon abandon ability");
3098 driver
.sleep(generateDelay
).then(function() {
3099 getFirstAddress(function(address
) {
3100 expect(address
).toBe("381wg1GGN4rP88rNC9v7QWsiww63yLVPsn");
3106 // BIP49 change field changes address list
3107 it('Can set the bip49 change field', function(done
) {
3108 driver
.findElement(By
.css('#bip49-tab a'))
3110 driver
.findElement(By
.css('#bip49 .change'))
3112 driver
.findElement(By
.css('#bip49 .change'))
3114 driver
.findElement(By
.css(".phrase"))
3115 .sendKeys("abandon abandon ability");
3116 driver
.sleep(generateDelay
).then(function() {
3117 getFirstAddress(function(address
) {
3118 expect(address
).toBe("3PEM7MiKed5konBoN66PQhK8r3hjGhy9dT");
3124 // BIP49 account extendend private key is shown
3125 it('Shows the bip49 account extended private key', function(done
) {
3126 driver
.findElement(By
.css('#bip49-tab a'))
3128 driver
.findElement(By
.css(".phrase"))
3129 .sendKeys("abandon abandon ability");
3130 driver
.sleep(generateDelay
).then(function() {
3131 driver
.findElement(By
.css('#bip49 .account-xprv'))
3132 .getAttribute("value")
3133 .then(function(xprv
) {
3134 expect(xprv
).toBe("yprvAHtB1M5Wp675aLzFy9TJYK2mSsLkg6mcBRh5DZTR7L4EnYSmYPaL63KFA4ycg1PngW5LfkmejxzosCs17TKZMpRFKc3z5SJar6QAKaFcaZL");
3140 // BIP49 account extendend public key is shown
3141 it('Shows the bip49 account extended public key', function(done
) {
3142 driver
.findElement(By
.css('#bip49-tab a'))
3144 driver
.findElement(By
.css(".phrase"))
3145 .sendKeys("abandon abandon ability");
3146 driver
.sleep(generateDelay
).then(function() {
3147 driver
.findElement(By
.css('#bip49 .account-xpub'))
3148 .getAttribute("value")
3149 .then(function(xprv
) {
3150 expect(xprv
).toBe("ypub6WsXQrcQeTfNnq4j5AzJuSyVzuBF5ZVTYecg1ws2ffbDfLmv5vtadqdj1NgR6C6gufMpMfJpHxvb6JEQKvETVNWCRanNedfJtnTchZiJtsL");
3156 // Test selecting coin where bip49 is unavailable (eg CLAM)
3157 it('Shows an error on bip49 tab for coins without bip49', function(done
) {
3158 driver
.findElement(By
.css('#bip49-tab a'))
3160 driver
.findElement(By
.css(".phrase"))
3161 .sendKeys("abandon abandon ability");
3162 driver
.sleep(generateDelay
).then(function() {
3163 selectNetwork("CLAM - Clams");
3164 // bip49 available is hidden
3165 driver
.findElement(By
.css('#bip49 .available'))
3166 .getAttribute("class")
3167 .then(function(classes
) {
3168 expect(classes
).toContain("hidden");
3169 // bip49 unavailable is shown
3170 driver
.findElement(By
.css('#bip49 .unavailable'))
3171 .getAttribute("class")
3172 .then(function(classes
) {
3173 expect(classes
).not
.toContain("hidden");
3174 // check there are no addresses shown
3175 driver
.findElements(By
.css('.addresses tr'))
3176 .then(function(rows
) {
3177 expect(rows
.length
).toBe(0);
3178 // check the derived private key is blank
3179 driver
.findElement(By
.css('.extended-priv-key'))
3180 .getAttribute("value")
3181 .then(function(xprv
) {
3182 expect(xprv
).toBe('');
3183 // check the derived public key is blank
3184 driver
.findElement(By
.css('.extended-pub-key'))
3185 .getAttribute("value")
3186 .then(function(xpub
) {
3187 expect(xpub
).toBe('');
3198 // Cleared mnemonic and root key still allows addresses to be generated
3199 // https://github.com/iancoleman/bip39/issues/43
3200 it('Clears old root keys from memory when mnemonic is cleared', function(done
) {
3202 driver
.findElement(By
.css(".phrase"))
3203 .sendKeys("abandon abandon ability");
3204 driver
.sleep(generateDelay
).then(function() {
3205 // clear the mnemonic and root key
3206 // using selenium .clear() doesn't seem to trigger the 'input' event
3207 // so clear it using keys instead
3208 driver
.findElement(By
.css('.phrase'))
3209 .sendKeys(Key
.CONTROL
,"a");
3210 driver
.findElement(By
.css('.phrase'))
3211 .sendKeys(Key
.DELETE
);
3212 driver
.findElement(By
.css('.root-key'))
3213 .sendKeys(Key
.CONTROL
,"a");
3214 driver
.findElement(By
.css('.root-key'))
3215 .sendKeys(Key
.DELETE
);
3216 driver
.sleep(generateDelay
).then(function() {
3217 // try to generate more addresses
3218 driver
.findElement(By
.css('.more'))
3220 driver
.sleep(generateDelay
).then(function() {
3221 driver
.findElements(By
.css(".addresses tr"))
3222 .then(function(els
) {
3223 // check there are no addresses shown
3224 expect(els
.length
).toBe(0);
3233 // error trying to generate addresses from xpub with hardened derivation
3234 it('Shows error for hardened addresses with xpub root key', function(done
) {
3235 driver
.findElement(By
.css('#bip32-tab a'))
3237 driver
.executeScript(function() {
3238 $(".hardened-addresses").prop("checked", true);
3240 // set xpub for account 0 of bip44 for 'abandon abandon ability'
3241 driver
.findElement(By
.css("#root-key"))
3242 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
3243 driver
.sleep(feedbackDelay
).then(function() {
3244 // Check feedback is correct
3245 driver
.findElement(By
.css('.feedback'))
3247 .then(function(feedback
) {
3248 var msg
= "Hardened derivation path is invalid with xpub key";
3249 expect(feedback
).toBe(msg
);
3255 // Litecoin uses ltub by default, and can optionally be set to xprv
3257 // https://github.com/iancoleman/bip39/issues/96
3258 // Issue with extended keys on Litecoin
3259 it('Uses ltub by default for litecoin, but can be set to xprv', function(done
) {
3260 driver
.findElement(By
.css('.phrase'))
3261 .sendKeys("abandon abandon ability");
3262 selectNetwork("LTC - Litecoin");
3263 driver
.sleep(generateDelay
).then(function() {
3264 // check the extended key is generated correctly
3265 driver
.findElement(By
.css('.root-key'))
3266 .getAttribute("value")
3267 .then(function(rootKey
) {
3268 expect(rootKey
).toBe("Ltpv71G8qDifUiNesiPqf6h5V6eQ8ic77oxQiYtawiACjBEx3sTXNR2HGDGnHETYxESjqkMLFBkKhWVq67ey1B2MKQXannUqNy1RZVHbmrEjnEU");
3269 // set litecoin to use ltub
3270 driver
.executeScript(function() {
3271 $(".litecoin-use-ltub").prop("checked", false);
3272 $(".litecoin-use-ltub").trigger("change");
3274 driver
.sleep(generateDelay
).then(function() {
3275 driver
.findElement(By
.css('.root-key'))
3276 .getAttribute("value")
3277 .then(function(rootKey
) {
3278 expect(rootKey
).toBe("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
3287 // https://github.com/iancoleman/bip39/issues/99#issuecomment-327094159
3288 // "warn me emphatically when they have detected invalid input" to the entropy field
3289 // A warning is shown when entropy is filtered and discarded
3290 it('Warns when entropy is filtered and discarded', function(done
) {
3291 driver
.findElement(By
.css('.use-entropy'))
3293 // set entropy to have no filtered content
3294 driver
.findElement(By
.css('.entropy'))
3295 .sendKeys("00000000 00000000 00000000 00000000");
3296 driver
.sleep(generateDelay
).then(function() {
3297 // check the filter warning does not show
3298 driver
.findElement(By
.css('.entropy-container .filter-warning'))
3299 .getAttribute("class")
3300 .then(function(classes
) {
3301 expect(classes
).toContain("hidden");
3302 // set entropy to have some filtered content
3303 driver
.findElement(By
.css('.entropy'))
3304 .sendKeys("10000000 zxcvbn 00000000 00000000 00000000");
3305 driver
.sleep(entropyFeedbackDelay
).then(function() {
3306 // check the filter warning shows
3307 driver
.findElement(By
.css('.entropy-container .filter-warning'))
3308 .getAttribute("class")
3309 .then(function(classes
) {
3310 expect(classes
).not
.toContain("hidden");
3318 // Bitcoin Cash address can be set to use cashaddr format
3319 it('Can use cashaddr format for bitcoin cash addresses', function(done
) {
3320 driver
.executeScript(function() {
3321 $(".use-bch-cashaddr-addresses").prop("checked", true);
3323 driver
.findElement(By
.css('.phrase'))
3324 .sendKeys("abandon abandon ability");
3325 selectNetwork("BCH - Bitcoin Cash");
3326 driver
.sleep(generateDelay
).then(function() {
3327 getFirstAddress(function(address
) {
3328 expect(address
).toBe("bitcoincash:qzlquk7w4hkudxypl4fgv8x279r754dkvur7jpcsps");
3334 // Bitcoin Cash address can be set to use bitpay format
3335 it('Can use bitpay format for bitcoin cash addresses', function(done
) {
3336 driver
.executeScript(function() {
3337 $(".use-bch-bitpay-addresses").prop("checked", true);
3339 driver
.findElement(By
.css('.phrase'))
3340 .sendKeys("abandon abandon ability");
3341 selectNetwork("BCH - Bitcoin Cash");
3342 driver
.sleep(generateDelay
).then(function() {
3343 getFirstAddress(function(address
) {
3344 expect(address
).toBe("CZnpA9HPmvhuhLLPWJP8rNDpLUYXy1LXFk");
3350 // Bitcoin Cash address can be set to use legacy format
3351 it('Can use legacy format for bitcoin cash addresses', function(done
) {
3352 driver
.executeScript(function() {
3353 $(".use-bch-legacy-addresses").prop("checked", true);
3355 driver
.findElement(By
.css('.phrase'))
3356 .sendKeys("abandon abandon ability");
3357 selectNetwork("BCH - Bitcoin Cash");
3358 driver
.sleep(generateDelay
).then(function() {
3359 getFirstAddress(function(address
) {
3360 expect(address
).toBe("1JKvb6wKtsjNoCRxpZ4DGrbniML7z5U16A");
3366 // End of tests ported from old suit, so no more comments above each test now
3368 it('Can generate more addresses from a custom index', function(done
) {
3369 var expectedIndexes
= [
3370 0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,
3371 40,41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59
3373 driver
.findElement(By
.css('.phrase'))
3374 .sendKeys("abandon abandon ability");
3375 driver
.sleep(generateDelay
).then(function() {
3376 // Set start of next lot of rows to be from index 40
3377 // which means indexes 20-39 will not be in the table.
3378 driver
.findElement(By
.css('.more-rows-start-index'))
3380 driver
.findElement(By
.css('.more'))
3382 driver
.sleep(generateDelay
).then(function() {
3383 // Check actual indexes in the table match the expected pattern
3384 driver
.findElements(By
.css(".index"))
3385 .then(function(els
) {
3386 expect(els
.length
).toBe(expectedIndexes
.length
);
3387 var testRowAtIndex = function(i
) {
3388 if (i
>= expectedIndexes
.length
) {
3393 .then(function(actualPath
) {
3394 var noHardened
= actualPath
.replace(/'/g, "");
3395 var pathBits = noHardened.split("/")
3396 var lastBit = pathBits[pathBits.length-1];
3397 var actualIndex = parseInt(lastBit);
3398 var expectedIndex = expectedIndexes[i];
3399 expect(actualIndex).toBe(expectedIndex);
3400 testRowAtIndex(i+1);
3410 it('Can generate BIP141 addresses
with P2WPKH
-in-P2SH semanitcs
', function(done) {
3411 // Sourced from BIP49 official test specs
3412 driver.findElement(By.css('#bip141
-tab a
'))
3414 driver.findElement(By.css('.bip141
-path
'))
3416 driver.findElement(By.css('.bip141
-path
'))
3417 .sendKeys("m/49'/1'/0'/0");
3418 selectNetwork("BTC
- Bitcoin Testnet
");
3419 driver.findElement(By.css(".phrase
"))
3420 .sendKeys("abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about
");
3421 driver.sleep(generateDelay).then(function() {
3422 getFirstAddress(function(address) {
3423 expect(address).toBe("2Mww8dCYPUpKHofjgcXcBCEGmniw9CoaiD2
");
3429 it('Can generate BIP141 addresses with P2WPKH semanitcs', function(done) {
3430 // This result tested against bitcoinjs-lib test spec for segwit address
3431 // using the first private key of this mnemonic and default path m/0
3432 // https://github.com/bitcoinjs/bitcoinjs-lib/blob/9c8503cab0c6c30a95127042703bc18e8d28c76d/test/integration/addresses.js#L50
3433 // so whilst not directly comparable, substituting the private key produces
3434 // identical results between this tool and the bitcoinjs-lib test.
3435 // Private key generated is:
3436 // L3L8Nu9whawPBNLGtFqDhKut9DKKfG3CQoysupT7BimqVCZsLFNP
3437 driver.findElement(By.css('#bip141-tab a'))
3440 driver.executeScript(function() {
3441 $(".bip141
-semantics option
[selected
]").removeAttr("selected
");
3442 $(".bip141
-semantics option
").filter(function(i,e) {
3443 return $(e).html() == "P2WPKH
";
3444 }).prop("selected
", true);
3445 $(".bip141
-semantics
").trigger("change
");
3447 driver.findElement(By.css(".phrase
"))
3448 .sendKeys("abandon abandon ability
");
3449 driver.sleep(generateDelay).then(function() {
3450 getFirstAddress(function(address) {
3451 expect(address).toBe("bc1qfwu6a5a3evygrk8zvdxxvz4547lmpyx5vsfxe9
");
3457 it('Shows the entropy used by the PRNG when clicking generate', function(done) {
3458 driver.findElement(By.css('.generate')).click();
3459 driver.sleep(generateDelay).then(function() {
3460 driver.findElement(By.css('.entropy'))
3461 .getAttribute("value
")
3462 .then(function(entropy) {
3463 expect(entropy).not.toBe("");
3469 it('Shows the index of each word in the mnemonic', function(done) {
3470 driver.findElement(By.css('.phrase'))
3471 .sendKeys("abandon abandon ability
");
3472 driver.sleep(generateDelay).then(function() {
3473 driver.findElement(By.css('.use-entropy'))
3475 driver.findElement(By.css('.word-indexes'))
3477 .then(function(indexes) {
3478 expect(indexes).toBe("0, 0, 1");
3484 it('Shows the derivation path for bip84 tab', function(done) {
3485 driver.findElement(By.css('#bip84-tab a'))
3487 driver.findElement(By.css('.phrase'))
3488 .sendKeys('abandon abandon ability');
3489 driver.sleep(generateDelay).then(function() {
3490 driver.findElement(By.css('#bip84 .path'))
3491 .getAttribute("value
")
3492 .then(function(path) {
3493 expect(path).toBe("m
/84'/0'/0'/0");
3499 it('Shows the extended private key for bip84 tab', function(done) {
3500 driver.findElement(By.css('#bip84-tab a'))
3502 driver.findElement(By.css('.phrase'))
3503 .sendKeys('abandon abandon ability');
3504 driver.sleep(generateDelay).then(function() {
3505 driver.findElement(By.css('.extended-priv-key'))
3506 .getAttribute("value
")
3507 .then(function(path) {
3508 expect(path).toBe("zprvAev3RKrZ3QVKiUFCfdeMRen1BPDJgdNt1XpxiDy8acSs4kkAGTCvq7HeRYRNNpo8EtEjCFQBWavJwtCUR29y4TUCH4X5RXMcyq48uN8y9BP
");
3514 it('Shows the extended public key for bip84 tab', function(done) {
3515 driver.findElement(By.css('#bip84-tab a'))
3517 driver.findElement(By.css('.phrase'))
3518 .sendKeys('abandon abandon ability');
3519 driver.sleep(generateDelay).then(function() {
3520 driver.findElement(By.css('.extended-pub-key'))
3521 .getAttribute("value
")
3522 .then(function(path) {
3523 expect(path).toBe("zpub6suPpqPSsn3cvxKfmfBMnnijjR3o666jNkkZWcNk8wyqwZ5JozXBNuc8Gs7DB3uLwTDvGVTspVEAUQcEjKF3pZHgywVbubdTqbXTUg7usyx
");
3529 it('Changes the address list if bip84 account is changed', function(done) {
3530 driver.findElement(By.css('#bip84-tab a'))
3532 driver.findElement(By.css('#bip84 .account'))
3534 driver.findElement(By.css('.phrase'))
3535 .sendKeys('abandon abandon ability');
3536 driver.sleep(generateDelay).then(function() {
3537 getFirstAddress(function(address) {
3538 expect(address).toBe("bc1qp7vv669t2fy965jdzvqwrraana89ctd5ewc662
");
3544 it('Changes the address list if bip84 change is changed', function(done) {
3545 driver.findElement(By.css('#bip84-tab a'))
3547 driver.findElement(By.css('#bip84 .change'))
3549 driver.findElement(By.css('.phrase'))
3550 .sendKeys('abandon abandon ability');
3551 driver.sleep(generateDelay).then(function() {
3552 getFirstAddress(function(address) {
3553 expect(address).toBe("bc1qr39vj6rh06ff05m53uxq8uazehwhccswylhrs2
");
3559 it('Passes the official BIP84 test spec for rootpriv', function(done) {
3560 driver.findElement(By.css('#bip84-tab a'))
3562 driver.findElement(By.css('.phrase'))
3563 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3564 driver.sleep(generateDelay).then(function() {
3565 driver.findElement(By.css(".root
-key
"))
3566 .getAttribute("value
")
3567 .then(function(rootKey) {
3568 expect(rootKey).toBe("zprvAWgYBBk7JR8Gjrh4UJQ2uJdG1r3WNRRfURiABBE3RvMXYSrRJL62XuezvGdPvG6GFBZduosCc1YP5wixPox7zhZLfiUm8aunE96BBa4Kei5
");
3574 it('Passes the official BIP84 test spec for account 0 xprv', function(done) {
3575 driver.findElement(By.css('#bip84-tab a'))
3577 driver.findElement(By.css('.phrase'))
3578 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3579 driver.sleep(generateDelay).then(function() {
3580 driver.findElement(By.css("#bip84
.account
-xprv
"))
3581 .getAttribute("value
")
3582 .then(function(rootKey) {
3583 expect(rootKey).toBe("zprvAdG4iTXWBoARxkkzNpNh8r6Qag3irQB8PzEMkAFeTRXxHpbF9z4QgEvBRmfvqWvGp42t42nvgGpNgYSJA9iefm1yYNZKEm7z6qUWCroSQnE
");
3589 it('Passes the official BIP84 test spec for account 0 xpub', function(done) {
3590 driver.findElement(By.css('#bip84-tab a'))
3592 driver.findElement(By.css('.phrase'))
3593 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3594 driver.sleep(generateDelay).then(function() {
3595 driver.findElement(By.css("#bip84
.account
-xpub
"))
3596 .getAttribute("value
")
3597 .then(function(rootKey) {
3598 expect(rootKey).toBe("zpub6rFR7y4Q2AijBEqTUquhVz398htDFrtymD9xYYfG1m4wAcvPhXNfE3EfH1r1ADqtfSdVCToUG868RvUUkgDKf31mGDtKsAYz2oz2AGutZYs
");
3604 it('Passes the official BIP84 test spec for account 0 first address', function(done) {
3605 driver.findElement(By.css('#bip84-tab a'))
3607 driver.findElement(By.css('.phrase'))
3608 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3609 driver.sleep(generateDelay).then(function() {
3610 getFirstAddress(function(address) {
3611 expect(address).toBe("bc1qcr8te4kr609gcawutmrza0j4xv80jy8z306fyu
");
3617 it('Passes the official BIP84 test spec for account 0 first change address', function(done) {
3618 driver.findElement(By.css('#bip84-tab a'))
3620 driver.findElement(By.css('.phrase'))
3621 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3622 driver.findElement(By.css('#bip84 .change'))
3624 driver.sleep(generateDelay).then(function() {
3625 getFirstAddress(function(address) {
3626 expect(address).toBe("bc1q8c6fshw2dlwun7ekn9qwf37cu2rn755upcp6el
");
3632 it('Can display the table as csv', function(done) {
3633 var headings = "path
,address
,public key
,private key
";
3634 var row1 = "m
/44'/0'/0'/0/0,1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug
,033f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3
,L26cVSpWFkJ6aQkPkKmTzLqTdLJ923e6CzrVh9cmx21QHsoUmrEE
";
3635 var row20 = "m
/44'/0'/0'/0/19,1KhBy28XLAciXnnRvm71PvQJaETyrxGV55
,02b4b3e396434d8cdd20c03ac4aaa07387784d5d867b75987f516f5705ee68cb3a
,L4GrDrjReMsCAu5DkLXn79jSb95qR7Zfx7eshybCQZ1qL32MXJab
";
3636 driver.findElement(By.css('.phrase'))
3637 .sendKeys('abandon abandon ability');
3638 driver.sleep(generateDelay).then(function() {
3639 driver.findElement(By.css('.csv'))
3640 .getAttribute("value
")
3641 .then(function(csv) {
3642 expect(csv).toContain(headings);
3643 expect(csv).toContain(row1);
3644 expect(csv).toContain(row20);
3650 it('LeftPads ethereum keys that are less than 32 bytes', function(done) {
3651 // see https://github.com/iancoleman/bip39/issues/155
3652 selectNetwork("ETH
- Ethereum
");
3653 driver.findElement(By.css('#bip32-tab a'))
3655 driver.findElement(By.css('#bip32-path'))
3657 driver.findElement(By.css('#bip32-path'))
3658 .sendKeys("m
/44'/60'/0'");
3659 driver.findElement(By.css('.phrase'))
3660 .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');
3661 driver.sleep(generateDelay).then(function() {
3662 getFirstAddress(function(address) {
3663 expect(address).toBe("0x8943E785B4a5714FC87a3aFAad1eB1FeB602B118");
3669 it('Can encrypt private keys using BIP38', function(done) {
3670 // see https://github.com/iancoleman/bip39/issues/140
3671 driver.executeScript(function() {
3672 $(".use-bip38
").prop("checked
", true);
3674 driver.findElement(By.css('.bip38-password'))
3675 .sendKeys('bip38password');
3676 driver.findElement(By.css('.rows-to-add'))
3678 driver.findElement(By.css('.rows-to-add'))
3680 driver.findElement(By.css('.phrase'))
3681 .sendKeys('abandon abandon ability');
3682 driver.sleep(bip38delay).then(function() {
3684 getFirstRowValue(function(address) {
3685 expect(address).toBe("1NCvSdumA3ngMM9c4aqU56AM6rqXddfuXB
");
3687 getFirstRowValue(function(pubkey) {
3688 expect(pubkey).toBe("043f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3884a74447ea901729b1e73a999b7520e7cb55b4120e6432c64153ccab8a848e1
");
3690 getFirstRowValue(function(privkey) {
3691 expect(privkey).toBe("6PRNRiFnj1RoR3sXhymdCvoZCgnUHQpfupNdKkFbWJkwWQEKesWt1EDMDM
");
3697 }, bip38delay + 5000);
3699 it('Shows the checksum for the entropy', function(done) {
3700 driver.findElement(By.css('.use-entropy'))
3702 driver.findElement(By.css('.entropy'))
3703 .sendKeys("00000000000000000000000000000000");
3704 driver.sleep(generateDelay).then(function() {
3705 driver.findElement(By.css('.checksum'))
3707 .then(function(text) {
3708 expect(text).toBe("1");
3714 it('Shows the checksum for the entropy with the correct groupings', function(done) {
3715 driver.findElement(By.css('.use-entropy'))
3717 // create a checksum of 20 bits, which spans multiple words
3718 driver.findElement(By.css('.entropy'))
3719 .sendKeys("F000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
");
3720 driver.sleep(generateDelay).then(function() {
3721 driver.findElement(By.css('.checksum'))
3723 .then(function(text) {
3724 // first group is 9 bits, second group is 11
3725 expect(text).toBe("011010111 01110000110");
3731 it('Uses vprv for bitcoin testnet p2wpkh', function(done) {
3732 selectNetwork("BTC
- Bitcoin Testnet
");
3733 driver.findElement(By.css('#bip84-tab a'))
3735 driver.findElement(By.css('.phrase'))
3736 .sendKeys('abandon abandon ability');
3737 driver.sleep(generateDelay).then(function() {
3738 driver.findElement(By.css('.root-key'))
3739 .getAttribute("value
")
3740 .then(function(path) {
3741 expect(path).toBe("vprv9DMUxX4ShgxML9N2YV5CvWEebWrM9aJ5ULpbRRyzyWu6vs4BzTvbfFFrH41N5hVi7MYSfiugd765L3JmAfDM5po36Y8ouCKRDeYQwByCmS7
");
3747 it('Shows a warning if generating weak mnemonics', function(done) {
3748 driver.executeScript(function() {
3749 $(".strength option
[selected
]").removeAttr("selected
");
3750 $(".strength option
[value
=6]").prop("selected
", true);
3751 $(".strength
").trigger("change
");
3753 driver.findElement(By.css(".generate
-container
.warning
"))
3754 .getAttribute("class")
3755 .then(function(classes) {
3756 expect(classes).not.toContain("hidden
");
3761 it('Does not show a warning if generating strong mnemonics', function(done) {
3762 driver.executeScript(function() {
3763 $(".strength option
[selected
]").removeAttr("selected
");
3764 $(".strength option
[value
=12]").prop("selected
", true);
3766 driver.findElement(By.css(".generate
-container
.warning
"))
3767 .getAttribute("class")
3768 .then(function(classes) {
3769 expect(classes).toContain("hidden
");
3774 it('Shows a warning if overriding weak entropy with longer mnemonics', function(done) {
3775 driver.findElement(By.css('.use-entropy'))
3777 driver.findElement(By.css('.entropy'))
3778 .sendKeys("0123456789abcdef
"); // 6 words
3779 driver.executeScript(function() {
3780 $(".mnemonic
-length
").val("12").trigger("change
");
3782 driver.findElement(By.css(".weak
-entropy
-override
-warning
"))
3783 .getAttribute("class")
3784 .then(function(classes) {
3785 expect(classes).not.toContain("hidden
");
3790 it('Does not show a warning if entropy is stronger than mnemonic length', function(done) {
3791 driver.findElement(By.css('.use-entropy'))
3793 driver.findElement(By.css('.entropy'))
3794 .sendKeys("0123456789abcdef0123456789abcdef0123456789abcdef
"); // 18 words
3795 driver.executeScript(function() {
3796 $(".mnemonic
-length
").val("12").trigger("change
");
3798 driver.findElement(By.css(".weak
-entropy
-override
-warning
"))
3799 .getAttribute("class")
3800 .then(function(classes) {
3801 expect(classes).toContain("hidden
");
3806 it('Shows litecoin BIP49 addresses', function(done) {
3807 driver.findElement(By.css('.phrase'))
3808 .sendKeys('abandon abandon ability');
3809 selectNetwork("LTC
- Litecoin
");
3810 driver.findElement(By.css('#bip49-tab a'))
3812 // bip49 addresses are shown
3813 driver.sleep(generateDelay).then(function() {
3814 driver.findElement(By.css('#bip49 .available'))
3815 .getAttribute("class")
3816 .then(function(classes) {
3817 expect(classes).not.toContain("hidden
");
3818 // check first address
3819 getFirstAddress(function(address) {
3820 expect(address).toBe("MFwLPhsXoBuSLL8cLmW9uK6tChkzduV8qN
");
3827 it('Can use root keys to generate segwit table rows', function(done) {
3828 // segwit uses ypub / zpub instead of xpub but the root key should still
3829 // be valid regardless of the encoding used to import that key.
3830 // Maybe this breaks the reason for the different extended key prefixes, but
3831 // since the parsed root key is used behind the scenes anyhow this should be
3833 driver.findElement(By.css('#root-key'))
3834 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
3835 driver.findElement(By.css('#bip49-tab a'))
3837 // bip49 addresses are shown
3838 driver.sleep(generateDelay).then(function() {
3839 getFirstAddress(function(address) {
3840 expect(address).toBe("3QG2Y9AA4xZ846gKHZqNf7mvVKbLqMKxr2
");