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: "BE9tnWxiR7ALgVhG8LLDi2W9pvtjzZMFoM",
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: "MHHRRPHcF8DvQpEySFF9M6fR8Qv4JH2fFC",
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: "SgdYBmVytcW2aCYitdegwkUcCU7RSqYokB",
1098 it('Allows selection
of Smileycoin
', function(done) {
1100 selectText: "SMLY - Smileycoin",
1101 firstAddress: "BEZVnEBCAyFByrgKpwAgYgtvP4rKAd9Sj2",
1103 testNetwork(done, params);
1105 it('Allows selection
of Solarcoin
', function(done) {
1107 selectText: "SLR - Solarcoin",
1108 firstAddress: "8LZ13HbnjtaMJWSvvVFNTLf71zFfDrhwLu",
1110 testNetwork(done, params);
1112 it('Allows selection
of stash
', function(done) {
1114 selectText: "STASH - Stash",
1115 firstAddress: "XxwAsWB7REDKmAvHA85SbEZQQtpxeUDxS3",
1117 testNetwork(done, params);
1119 it('Allows selection
of stash testnet
', function(done) {
1121 selectText: "STASH - Stash Testnet",
1122 firstAddress: "YdbhtMuGsPSkE6bPdNTHoFSszQKmK4S5LT",
1124 testNetwork(done, params);
1126 it('Allows selection
of Stratis
', function(done) {
1128 selectText: "STRAT - Stratis",
1129 firstAddress: "ScfJnq3QDhKgDMEds6sqUE1ot6ShfhmXXq",
1131 testNetwork(done, params);
1133 it('Allows selection
of Stratis Test
', function(done) {
1135 selectText: "TSTRAT - Stratis Testnet",
1136 firstAddress: "TRLWm3dye4FRrDWouwYUSUZP96xb76mBE3",
1138 testNetwork(done, params);
1140 it('Allows selection
of Syscoin
', function(done) {
1142 selectText: "SYS - Syscoin",
1143 firstAddress: "SZwJi42Pst3VAMomyK5DG4157WM5ofRmSj",
1145 testNetwork(done, params);
1147 it('Allows selection
of Toa
', function(done) {
1149 selectText: "TOA - Toa",
1150 firstAddress: "TSe1QAnUwQzUfbBusDzRJ9URttrRGKoNKF",
1152 testNetwork(done, params);
1154 it('Allows selection
of Ultimatesecurecash
', function(done) {
1156 selectText: "USC - Ultimatesecurecash",
1157 firstAddress: "UPyLAZU2Che5fiy7Ed8xVJFmXAUhitA4ug",
1159 testNetwork(done, params);
1161 it('Allows selection
of Unobtanium
', function(done) {
1163 selectText: "UNO - Unobtanium",
1164 firstAddress: "uUBMPVMXrR6qhqornJqKTWgr8L69vihSL9",
1166 testNetwork(done, params);
1168 it('Allows selection
of Vcash
', function(done) {
1170 selectText: "XVC - Vcash",
1171 firstAddress: "VuL53MSY6KjvAjKSeRkh3NDnKykacDVeps",
1173 testNetwork(done, params);
1175 it('Allows selection
of Verge
', function(done) {
1177 selectText: "XVG - Verge",
1178 firstAddress: "DCrVuGkMjLJpTGgwAgv9AcMdeb1nkWbjZA",
1180 testNetwork(done, params);
1182 it('Allows selection
of Vertcoin
', function(done) {
1184 selectText: "VTC - Vertcoin",
1185 firstAddress: "Vf6koGuiWdXQfx8tNqxoNeEDxh4xh5cxsG",
1187 testNetwork(done, params);
1189 it('Allows selection
of Vivo
', function(done) {
1191 selectText: "VIVO - Vivo",
1192 firstAddress: "VFmBwuXXGhJe7MarQG2GfzHMFebRHgfSpB",
1194 testNetwork(done, params);
1196 it('Allows selection
of Vpncoin
', function(done) {
1198 selectText: "VASH - Vpncoin",
1199 firstAddress: "VoEmH1qXC4TsSgBAStR21QYetwnFqbqCx9",
1201 testNetwork(done, params);
1203 it('Allows selection
of Whitecoin
', function(done) {
1205 selectText: "XWC - Whitecoin",
1206 firstAddress: "WcSwCAUqrSgeSYbsaS3SSWWhsx8KRYTFDR",
1208 testNetwork(done, params);
1210 it('Allows selection
of Wincoin
', function(done) {
1212 selectText: "WC - Wincoin",
1213 firstAddress: "WaDVCESMGgyKgNESdn3u43NnwmGSkZED3Z",
1215 testNetwork(done, params);
1217 it('Allows selection
of Zcoin
', function(done) {
1219 selectText: "XZC - Zcoin",
1220 firstAddress: "a6VcMdP4XgAA9Tr7xNszmPG5FZpfRf17Cq",
1222 testNetwork(done, params);
1224 it('Allows selection
of Zcash
', function(done) {
1226 selectText: "ZEC - Zcash",
1227 firstAddress: "t1Sz8AneMcVuzUg3tPJ8et5AS5LFJ7K2EF9",
1229 testNetwork(done, params);
1231 it('Allows selection
of Zclassic
', function(done) {
1233 selectText: "ZCL - Zclassic",
1234 firstAddress: "t1TBMxTvVJRybUbMLGWq8H4A8F4VUL7czEc",
1236 testNetwork(done, params);
1238 it('Allows selection
of Zencash
', function(done) {
1240 selectText: "ZEN - Zencash",
1241 firstAddress: "znWh9XASyW2dZq5tck84wFjiwuqVysi7q3p",
1243 testNetwork(done, params);
1245 it('Allows selection
of Energi
', function(done) {
1247 selectText: "NRG - Energi",
1248 firstAddress: "EejRy4t4nidzhGGzkJUgFP3z4HYBjhTsRt",
1250 testNetwork(done, params);
1252 it('Allows selection
of Ethereum Classic
', function(done) {
1254 selectText: "ETC - Ethereum Classic",
1255 firstAddress: "0x3c05e5556693808367afB62eF3b63e35d6eD249A",
1257 testNetwork(done, params);
1259 it('Allows selection
of Pirl
', function(done) {
1261 selectText: "PIRL - Pirl",
1262 firstAddress: "0xe77FC0723dA122B5025CA79193c28563eB47e776",
1264 testNetwork(done, params);
1266 it('Allows selection
of MIX
', function(done) {
1268 selectText: "MIX - MIX",
1269 firstAddress: "0x98BC5e63aeb6A4e82d72850d20710F07E29A29F1",
1271 testNetwork(done, params);
1273 it('Allows selection
of Musicoin
', function(done) {
1275 selectText: "MUSIC - Musicoin",
1276 firstAddress: "0xDc060e4A0b0313ea83Cf6B3A39B9db2D29004897",
1278 testNetwork(done, params);
1280 it('Allows selection
of Poa
', function(done) {
1282 selectText: "POA - Poa",
1283 firstAddress: "0x53aF28d754e106210C3d0467Dd581eaf7e3C5e60",
1285 testNetwork(done, params);
1287 it('Allows selection
of Expanse
', function(done) {
1289 selectText: "EXP - Expanse",
1290 firstAddress: "0xf57FeAbf26582b6E3E666559d3B1Cc6fB2b2c5F6",
1292 testNetwork(done, params);
1294 it('Allows selection
of Callisto
', function(done) {
1296 selectText: "CLO - Callisto",
1297 firstAddress: "0x4f9364F7420B317266C51Dc8eB979717D4dE3f4E",
1299 testNetwork(done, params);
1301 it('Allows selection
of HUSH
', function(done) {
1303 selectText: "HUSH - Hush",
1304 firstAddress: "t1g6rLXUnJaiJuu4q4zmJjoa9Gk4fwKpiuA",
1306 testNetwork(done, params);
1308 it('Allows selection
of ExchangeCoin
', function(done) {
1310 selectText: "EXCC - ExchangeCoin",
1311 firstAddress: "22txYKpFN5fwGwdSs2UBf7ywewbLM92YqK7E",
1313 testNetwork(done, params);
1315 it('Allows selection
of Artax
', function(done) {
1317 selectText: "XAX - Artax",
1318 firstAddress: "AYxaQPY7XLidG31V7F3yNzwxPYpYzRqG4q",
1320 testNetwork(done, params);
1322 it('Allows selection
of ANON
', function(done) {
1324 selectText: "ANON - ANON",
1325 firstAddress: "AnU6pijpEeUZFWSTyM2qTqZQn996Zq1Xard",
1327 testNetwork(done, params);
1331 // BIP39 seed is set from phrase
1332 it('Sets the bip39 seed
from the prhase
', function(done) {
1333 driver.findElement(By.css('.phrase
'))
1334 .sendKeys('abandon abandon ability
');
1335 driver.sleep(generateDelay).then(function() {
1336 driver.findElement(By.css('.seed
'))
1337 .getAttribute("value")
1338 .then(function(seed) {
1339 expect(seed).toBe("20da140d3dd1df8713cefcc4d54ce0e445b4151027a1ab567b832f6da5fcc5afc1c3a3f199ab78b8e0ab4652efd7f414ac2c9a3b81bceb879a70f377aa0a58f3");
1345 // BIP32 root key is set from phrase
1346 it('Sets the bip39 root key
from the prhase
', function(done) {
1347 driver.findElement(By.css('.phrase
'))
1348 .sendKeys('abandon abandon ability
');
1349 driver.sleep(generateDelay).then(function() {
1350 driver.findElement(By.css('.root
-key
'))
1351 .getAttribute("value")
1352 .then(function(seed) {
1353 expect(seed).toBe("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
1359 // Tabs show correct addresses when changed
1360 it('Shows the correct address when tab is changed
', function(done) {
1361 driver.findElement(By.css('.phrase
'))
1362 .sendKeys('abandon abandon ability
');
1363 driver.sleep(generateDelay).then(function() {
1364 driver.findElement(By.css('#bip32
-tab a
'))
1366 driver.sleep(generateDelay).then(function() {
1367 getFirstAddress(function(address) {
1368 expect(address).toBe("17uQ7s2izWPwBmEVFikTmZUjbBKWYdJchz");
1375 // BIP44 derivation path is shown
1376 it('Shows the derivation path
for bip44 tab
', function(done) {
1377 driver.findElement(By.css('.phrase
'))
1378 .sendKeys('abandon abandon ability
');
1379 driver.sleep(generateDelay).then(function() {
1380 driver.findElement(By.css('#bip44
.path
'))
1381 .getAttribute("value")
1382 .then(function(path) {
1383 expect(path).toBe("m/44'/0'/0'/0");
1389 // BIP44 extended private key is shown
1390 it('Shows the extended private key for bip44 tab', function(done) {
1391 driver.findElement(By.css('.phrase'))
1392 .sendKeys('abandon abandon ability');
1393 driver.sleep(generateDelay).then(function() {
1394 driver.findElement(By.css('.extended-priv-key'))
1395 .getAttribute("value
")
1396 .then(function(path) {
1397 expect(path).toBe("xprvA2DxxvPZcyRvYgZMGS53nadR32mVDeCyqQYyFhrCVbJNjPoxMeVf7QT5g7mQASbTf9Kp4cryvcXnu2qurjWKcrdsr91jXymdCDNxKgLFKJG
");
1403 // BIP44 extended public key is shown
1404 it('Shows the extended public key for bip44 tab', function(done) {
1405 driver.findElement(By.css('.phrase'))
1406 .sendKeys('abandon abandon ability');
1407 driver.sleep(generateDelay).then(function() {
1408 driver.findElement(By.css('.extended-pub-key'))
1409 .getAttribute("value
")
1410 .then(function(path) {
1411 expect(path).toBe("xpub6FDKNRvTTLzDmAdpNTc49ia9b4byd6vqCdUa46Fp3vqMcC96uBoufCmZXQLiN5AK3iSCJMhf9gT2sxkpyaPepRuA7W3MujV5tGmF5VfbueM
");
1417 // BIP44 account field changes address list
1418 it('Changes the address list if bip44 account is changed', function(done) {
1419 driver.findElement(By.css('#bip44 .account'))
1421 driver.findElement(By.css('.phrase'))
1422 .sendKeys('abandon abandon ability');
1423 driver.sleep(generateDelay).then(function() {
1424 getFirstAddress(function(address) {
1425 expect(address).toBe("1Nq2Wmu726XHCuGhctEtGmhxo3wzk5wZ1H
");
1431 // BIP44 change field changes address list
1432 it('Changes the address list if bip44 change is changed', function(done) {
1433 driver.findElement(By.css('#bip44 .change'))
1435 driver.findElement(By.css('.phrase'))
1436 .sendKeys('abandon abandon ability');
1437 driver.sleep(generateDelay).then(function() {
1438 getFirstAddress(function(address) {
1439 expect(address).toBe("1KAGfWgqfVbSSXY56fNQ7YnhyKuoskHtYo
");
1445 // BIP32 derivation path can be set
1446 it('Can use a custom bip32 derivation path', function(done) {
1447 driver.findElement(By.css('#bip32-tab a'))
1449 driver.findElement(By.css('#bip32 .path'))
1451 driver.findElement(By.css('#bip32 .path'))
1453 driver.findElement(By.css('.phrase'))
1454 .sendKeys('abandon abandon ability');
1455 driver.sleep(generateDelay).then(function() {
1456 getFirstAddress(function(address) {
1457 expect(address).toBe("16pYQQdLD1hH4hwTGLXBaZ9Teboi1AGL8L
");
1463 // BIP32 can use hardened derivation paths
1464 it('Can use a hardened derivation paths', function(done) {
1465 driver.findElement(By.css('#bip32-tab a'))
1467 driver.findElement(By.css('#bip32 .path'))
1469 driver.findElement(By.css('#bip32 .path'))
1471 driver.findElement(By.css('.phrase
'))
1472 .sendKeys('abandon abandon ability
');
1473 driver.sleep(generateDelay).then(function() {
1474 getFirstAddress(function(address) {
1475 expect(address).toBe("14aXZeprXAE3UUKQc4ihvwBvww2LuEoHo4");
1481 // BIP32 extended private key is shown
1482 it('Shows the BIP32 extended
private key
', function(done) {
1483 driver.findElement(By.css('#bip32
-tab a
'))
1485 driver.findElement(By.css('.phrase
'))
1486 .sendKeys('abandon abandon ability
');
1487 driver.sleep(generateDelay).then(function() {
1488 driver.findElement(By.css('.extended
-priv
-key
'))
1489 .getAttribute("value")
1490 .then(function(privKey) {
1491 expect(privKey).toBe("xprv9va99uTVE5aLiutUVLTyfxfe8v8aaXjSQ1XxZbK6SezYVuikA9MnjQVTA8rQHpNA5LKvyQBpLiHbBQiiccKiBDs7eRmBogsvq3THFeLHYbe");
1497 // BIP32 extended public key is shown
1498 it('Shows the BIP32 extended
public key
', function(done) {
1499 driver.findElement(By.css('#bip32
-tab a
'))
1501 driver.findElement(By.css('.phrase
'))
1502 .sendKeys('abandon abandon ability
');
1503 driver.sleep(generateDelay).then(function() {
1504 driver.findElement(By.css('.extended
-pub
-key
'))
1505 .getAttribute("value")
1506 .then(function(pubKey) {
1507 expect(pubKey).toBe("xpub69ZVZQzP4T8dwPxwbMzz36cNgwy4yzTHmETZMyihzzXXNi3thgg3HCow1RtY252wdw5rS8369xKnraN5Q93y3FkFfJp2XEHWUrkyXsjS93P");
1513 // Derivation path is shown in table
1514 it('Shows the derivation path
in the table
', function(done) {
1515 driver.findElement(By.css('.phrase
'))
1516 .sendKeys('abandon abandon ability
');
1517 driver.sleep(generateDelay).then(function() {
1518 getFirstPath(function(path) {
1519 expect(path).toBe("m/44'/0'/0'/0/0");
1525 // Derivation path for address can be hardened
1526 it('Can derive hardened addresses', function(done) {
1527 driver.findElement(By.css('#bip32-tab a'))
1529 driver.executeScript(function() {
1530 $(".hardened
-addresses
").prop("checked
", true);
1532 driver.findElement(By.css('.phrase'))
1533 .sendKeys('abandon abandon ability');
1534 driver.sleep(generateDelay).then(function() {
1535 getFirstAddress(function(address) {
1536 expect(address).toBe("18exLzUv7kfpiXRzmCjFDoC9qwNLFyvwyd
");
1542 // Derivation path visibility can be toggled
1543 it('Can toggle visibility of the derivation path column', function(done) {
1544 driver.findElement(By.css('.phrase'))
1545 .sendKeys('abandon abandon ability');
1546 driver.sleep(generateDelay).then(function() {
1547 driver.findElement(By.css('.index-toggle'))
1549 testColumnValuesAreInvisible(done, "index
");
1554 it('Shows the address in the table', function(done) {
1555 driver.findElement(By.css('.phrase'))
1556 .sendKeys('abandon abandon ability');
1557 driver.sleep(generateDelay).then(function() {
1558 getFirstAddress(function(address) {
1559 expect(address).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug
");
1565 // Addresses are shown in order of derivation path
1566 it('Shows the address in order of derivation path', function(done) {
1567 driver.findElement(By.css('.phrase'))
1568 .sendKeys('abandon abandon ability');
1569 driver.sleep(generateDelay).then(function() {
1570 testRowsAreInCorrectOrder(done);
1574 // Address visibility can be toggled
1575 it('Can toggle visibility of the address column', function(done) {
1576 driver.findElement(By.css('.phrase'))
1577 .sendKeys('abandon abandon ability');
1578 driver.sleep(generateDelay).then(function() {
1579 driver.findElement(By.css('.address-toggle'))
1581 testColumnValuesAreInvisible(done, "address
");
1585 // Public key is shown in table
1586 it('Shows the public key in the table', function(done) {
1587 driver.findElement(By.css('.phrase'))
1588 .sendKeys('abandon abandon ability');
1589 driver.sleep(generateDelay).then(function() {
1590 driver.findElements(By.css('.pubkey'))
1591 .then(function(els) {
1593 .then(function(pubkey) {
1594 expect(pubkey).toBe("033f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3
");
1601 // Public key visibility can be toggled
1602 it('Can toggle visibility of the public key column', function(done) {
1603 driver.findElement(By.css('.phrase'))
1604 .sendKeys('abandon abandon ability');
1605 driver.sleep(generateDelay).then(function() {
1606 driver.findElement(By.css('.public-key-toggle'))
1608 testColumnValuesAreInvisible(done, "pubkey
");
1612 // Private key is shown in table
1613 it('Shows the private key in the table', function(done) {
1614 driver.findElement(By.css('.phrase'))
1615 .sendKeys('abandon abandon ability');
1616 driver.sleep(generateDelay).then(function() {
1617 driver.findElements(By.css('.privkey'))
1618 .then(function(els) {
1620 .then(function(pubkey) {
1621 expect(pubkey).toBe("L26cVSpWFkJ6aQkPkKmTzLqTdLJ923e6CzrVh9cmx21QHsoUmrEE
");
1628 // Private key visibility can be toggled
1629 it('Can toggle visibility of the private key column', function(done) {
1630 driver.findElement(By.css('.phrase'))
1631 .sendKeys('abandon abandon ability');
1632 driver.sleep(generateDelay).then(function() {
1633 driver.findElement(By.css('.private-key-toggle'))
1635 testColumnValuesAreInvisible(done, "privkey
");
1639 // More addresses can be generated
1640 it('Can generate more rows in the table', function(done) {
1641 driver.findElement(By.css('.phrase'))
1642 .sendKeys('abandon abandon ability');
1643 driver.sleep(generateDelay).then(function() {
1644 driver.findElement(By.css('.more'))
1646 driver.sleep(generateDelay).then(function() {
1647 driver.findElements(By.css('.address'))
1648 .then(function(els) {
1649 expect(els.length).toBe(40);
1656 // A custom number of additional addresses can be generated
1657 it('Can generate more rows in the table', function(done) {
1658 driver.findElement(By.css('.phrase'))
1659 .sendKeys('abandon abandon ability');
1660 driver.sleep(generateDelay).then(function() {
1661 driver.findElement(By.css('.rows-to-add'))
1663 driver.findElement(By.css('.rows-to-add'))
1665 driver.findElement(By.css('.more'))
1667 driver.sleep(generateDelay).then(function() {
1668 driver.findElements(By.css('.address'))
1669 .then(function(els) {
1670 expect(els.length).toBe(21);
1677 // Additional addresses are shown in order of derivation path
1678 it('Shows additional addresses in order of derivation path', function(done) {
1679 driver.findElement(By.css('.phrase'))
1680 .sendKeys('abandon abandon ability');
1681 driver.sleep(generateDelay).then(function() {
1682 driver.findElement(By.css('.more'))
1684 driver.sleep(generateDelay).then(function() {
1685 testRowsAreInCorrectOrder(done);
1690 // BIP32 root key can be set by the user
1691 it('Allows the user to set the BIP32 root key', function(done) {
1692 driver.findElement(By.css('.root-key'))
1693 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1694 driver.sleep(generateDelay).then(function() {
1695 getFirstAddress(function(address) {
1696 expect(address).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug
");
1702 // Setting BIP32 root key clears the existing phrase, passphrase and seed
1703 // TODO this doesn't work in selenium with chrome
1704 it('Confirms the existing phrase should be cleared', function(done) {
1705 if (browser == "chrome
") {
1706 pending("Selenium
+ Chrome headless bug
for alert
, see
https://stackoverflow.com/q/45242264");
1708 driver
.findElement(By
.css('.phrase'))
1709 .sendKeys('A non-blank but invalid value');
1710 driver
.findElement(By
.css('.root-key'))
1711 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1712 driver
.switchTo().alert().accept();
1713 driver
.findElement(By
.css('.phrase'))
1714 .getAttribute("value").then(function(value
) {
1715 expect(value
).toBe("");
1720 // Clearing of phrase, passphrase and seed can be cancelled by user
1721 // TODO this doesn't work in selenium with chrome
1722 it('Allows the clearing of the phrase to be cancelled', function(done
) {
1723 if (browser
== "chrome") {
1724 pending("Selenium + Chrome headless bug for alert, see https://stackoverflow.com/q/45242264");
1726 driver
.findElement(By
.css('.phrase'))
1727 .sendKeys('abandon abandon ability');
1728 driver
.sleep(generateDelay
).then(function() {
1729 driver
.findElement(By
.css('.root-key'))
1731 driver
.findElement(By
.css('.root-key'))
1733 driver
.switchTo().alert().dismiss();
1734 driver
.findElement(By
.css('.phrase'))
1735 .getAttribute("value").then(function(value
) {
1736 expect(value
).toBe("abandon abandon ability");
1742 // Custom BIP32 root key is used when changing the derivation path
1743 it('Can set derivation path for root key instead of phrase', function(done
) {
1744 driver
.findElement(By
.css('#bip44 .account'))
1746 driver
.findElement(By
.css('.root-key'))
1747 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1748 driver
.sleep(generateDelay
).then(function() {
1749 getFirstAddress(function(address
) {
1750 expect(address
).toBe("1Nq2Wmu726XHCuGhctEtGmhxo3wzk5wZ1H");
1756 // Incorrect mnemonic shows error
1757 it('Shows an error for incorrect mnemonic', function(done
) {
1758 driver
.findElement(By
.css('.phrase'))
1759 .sendKeys('abandon abandon abandon');
1760 driver
.sleep(feedbackDelay
).then(function() {
1761 driver
.findElement(By
.css('.feedback'))
1763 .then(function(feedback
) {
1764 expect(feedback
).toBe("Invalid mnemonic");
1770 // Incorrect word shows suggested replacement
1771 it('Shows word suggestion for incorrect word', function(done
) {
1772 driver
.findElement(By
.css('.phrase'))
1773 .sendKeys('abandon abandon abiliti');
1774 driver
.sleep(feedbackDelay
).then(function() {
1775 driver
.findElement(By
.css('.feedback'))
1777 .then(function(feedback
) {
1778 var msg
= "abiliti not in wordlist, did you mean ability?";
1779 expect(feedback
).toBe(msg
);
1785 // Github pull request 48
1786 // First four letters of word shows that word, not closest
1787 // since first four letters gives unique word in BIP39 wordlist
1788 // eg ille should show illegal, not idle
1789 it('Shows word suggestion based on first four chars', function(done
) {
1790 driver
.findElement(By
.css('.phrase'))
1792 driver
.sleep(feedbackDelay
).then(function() {
1793 driver
.findElement(By
.css('.feedback'))
1795 .then(function(feedback
) {
1796 var msg
= "ille not in wordlist, did you mean illegal?";
1797 expect(feedback
).toBe(msg
);
1803 // Incorrect BIP32 root key shows error
1804 it('Shows error for incorrect root key', function(done
) {
1805 driver
.findElement(By
.css('.root-key'))
1806 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpj');
1807 driver
.sleep(feedbackDelay
).then(function() {
1808 driver
.findElement(By
.css('.feedback'))
1810 .then(function(feedback
) {
1811 var msg
= "Invalid root key";
1812 expect(feedback
).toBe(msg
);
1818 // Derivation path not starting with m shows error
1819 it('Shows error for derivation path not starting with m', function(done
) {
1820 driver
.findElement(By
.css('#bip32-tab a'))
1822 driver
.findElement(By
.css('#bip32 .path'))
1824 driver
.findElement(By
.css('#bip32 .path'))
1826 driver
.findElement(By
.css('.phrase'))
1827 .sendKeys('abandon abandon ability');
1828 driver
.sleep(feedbackDelay
).then(function() {
1829 driver
.findElement(By
.css('.feedback'))
1831 .then(function(feedback
) {
1832 var msg
= "First character must be 'm'";
1833 expect(feedback
).toBe(msg
);
1839 // Derivation path containing invalid characters shows useful error
1840 it('Shows error for derivation path not starting with m', function(done
) {
1841 driver
.findElement(By
.css('#bip32-tab a'))
1843 driver
.findElement(By
.css('#bip32 .path'))
1845 driver
.findElement(By
.css('#bip32 .path'))
1846 .sendKeys('m/1/0wrong1/1');
1847 driver
.findElement(By
.css('.phrase'))
1848 .sendKeys('abandon abandon ability');
1849 driver
.sleep(feedbackDelay
).then(function() {
1850 driver
.findElement(By
.css('.feedback'))
1852 .then(function(feedback
) {
1853 var msg
= "Invalid characters 0wrong1 found at depth 2";
1854 expect(feedback
).toBe(msg
);
1860 // Github Issue 11: Default word length is 15
1861 // https://github.com/iancoleman/bip39/issues/11
1862 it('Sets the default word length to 15', function(done
) {
1863 driver
.findElement(By
.css('.strength'))
1864 .getAttribute("value")
1865 .then(function(strength
) {
1866 expect(strength
).toBe("15");
1871 // Github Issue 12: Generate more rows with private keys hidden
1872 // https://github.com/iancoleman/bip39/issues/12
1873 it('Sets the correct hidden column state on new rows', function(done
) {
1874 driver
.findElement(By
.css('.phrase'))
1875 .sendKeys("abandon abandon ability");
1876 driver
.sleep(generateDelay
).then(function() {
1877 driver
.findElement(By
.css('.private-key-toggle'))
1879 driver
.findElement(By
.css('.more'))
1881 driver
.sleep(generateDelay
).then(function() {
1882 driver
.findElements(By
.css('.privkey'))
1883 .then(function(els
) {
1884 expect(els
.length
).toBe(40);
1886 testColumnValuesAreInvisible(done
, "privkey");
1891 // Github Issue 19: Mnemonic is not sensitive to whitespace
1892 // https://github.com/iancoleman/bip39/issues/19
1893 it('Ignores excess whitespace in the mnemonic', function(done
) {
1894 var doublespace
= " ";
1895 var mnemonic
= "urge cat" + doublespace
+ "bid";
1896 driver
.findElement(By
.css('.phrase'))
1897 .sendKeys(mnemonic
);
1898 driver
.sleep(generateDelay
).then(function() {
1899 driver
.findElement(By
.css('.root-key'))
1900 .getAttribute("value")
1901 .then(function(seed
) {
1902 expect(seed
).toBe("xprv9s21ZrQH143K3isaZsWbKVoTtbvd34Y1ZGRugGdMeBGbM3AgBVzTH159mj1cbbtYSJtQr65w6L5xy5L9SFC7c9VJZWHxgAzpj4mun5LhrbC");
1908 // Github Issue 23: Part 1: Use correct derivation path when changing tabs
1909 // https://github.com/iancoleman/bip39/issues/23
1910 it('Uses the correct derivation path when changing tabs', function(done
) {
1911 // 1) and 2) set the phrase
1912 driver
.findElement(By
.css('.phrase'))
1913 .sendKeys("abandon abandon ability");
1914 driver
.sleep(generateDelay
).then(function() {
1915 // 3) select bip32 tab
1916 driver
.findElement(By
.css('#bip32-tab a'))
1918 driver
.sleep(generateDelay
).then(function() {
1919 // 4) switch from bitcoin to litecoin
1920 selectNetwork("LTC - Litecoin");
1921 driver
.sleep(generateDelay
).then(function() {
1922 // 5) Check address is displayed correctly
1923 getFirstAddress(function(address
) {
1924 expect(address
).toBe("LS8MP5LZ5AdzSZveRrjm3aYVoPgnfFh5T5");
1925 // 5) Check derivation path is displayed correctly
1926 getFirstPath(function(path
) {
1927 expect(path
).toBe("m/0/0");
1936 // Github Issue 23 Part 2: Coin selection in derivation path
1937 // https://github.com/iancoleman/bip39/issues/23#issuecomment-238011920
1938 it('Uses the correct derivation path when changing coins', function(done
) {
1940 driver
.findElement(By
.css('.phrase'))
1941 .sendKeys("abandon abandon ability");
1942 driver
.sleep(generateDelay
).then(function() {
1943 // switch from bitcoin to clam
1944 selectNetwork("CLAM - Clams");
1945 driver
.sleep(generateDelay
).then(function() {
1946 // check derivation path is displayed correctly
1947 getFirstPath(function(path
) {
1948 expect(path
).toBe("m/44'/23'/0'/0/0");
1955 // Github Issue 26: When using a Root key derrived altcoins are incorrect
1956 // https://github.com/iancoleman/bip39/issues/26
1957 it('Uses the correct derivation for altcoins with root keys', function(done
) {
1958 // 1) 2) and 3) set the root key
1959 driver
.findElement(By
.css('.root-key'))
1960 .sendKeys("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
1961 driver
.sleep(generateDelay
).then(function() {
1962 // 4) switch from bitcoin to viacoin
1963 selectNetwork("VIA - Viacoin");
1964 driver
.sleep(generateDelay
).then(function() {
1965 // 5) ensure the derived address is correct
1966 getFirstAddress(function(address
) {
1967 expect(address
).toBe("Vq9Eq4N5SQnjqZvxtxzo7hZPW5XnyJsmXT");
1974 // Selecting a language with no existing phrase should generate a phrase in
1976 it('Generate a random phrase when language is selected and no current phrase', function(done
) {
1977 driver
.findElement(By
.css("a[href='#japanese']"))
1979 driver
.sleep(generateDelay
).then(function() {
1980 driver
.findElement(By
.css(".phrase"))
1981 .getAttribute("value").then(function(phrase
) {
1982 expect(phrase
.search(/[a-z]/)).toBe(-1);
1983 expect(phrase
.length
).toBeGreaterThan(0);
1989 // Selecting a language with existing phrase should update the phrase to use
1991 it('Updates existing phrases when the language is changed', function(done
) {
1992 driver
.findElement(By
.css(".phrase"))
1993 .sendKeys("abandon abandon ability");
1994 driver
.sleep(generateDelay
).then(function() {
1995 driver
.findElement(By
.css("a[href='#italian']"))
1997 driver
.sleep(generateDelay
).then(function() {
1998 driver
.findElement(By
.css(".phrase"))
1999 .getAttribute("value").then(function(phrase
) {
2000 // Check only the language changes, not the phrase
2001 expect(phrase
).toBe("abaco abaco abbaglio");
2002 getFirstAddress(function(address
) {
2003 // Check the address is correct
2004 expect(address
).toBe("1Dz5TgDhdki9spa6xbPFbBqv5sjMrx3xgV");
2012 // Suggested replacement for erroneous word in non-English language
2013 it('Shows word suggestion for incorrect word in non-English language', function(done
) {
2014 driver
.findElement(By
.css('.phrase'))
2015 .sendKeys('abaco abaco zbbaglio');
2016 driver
.sleep(feedbackDelay
).then(function() {
2017 driver
.findElement(By
.css('.feedback'))
2019 .then(function(feedback
) {
2020 var msg
= "zbbaglio not in wordlist, did you mean abbaglio?";
2021 expect(feedback
).toBe(msg
);
2027 // Japanese word does not break across lines.
2029 // https://github.com/bitcoin/bips/blob/master/bip-0039/bip-0039-wordlists.md#japanese
2030 it('Does not break Japanese words across lines', function(done
) {
2031 driver
.findElement(By
.css('.phrase'))
2032 .getCssValue("word-break")
2033 .then(function(value
) {
2034 expect(value
).toBe("keep-all");
2039 // Language can be specified at page load using hash value in url
2040 it('Can set the language from the url hash', function(done
) {
2041 driver
.get(url
+ "#japanese").then(function() {
2042 driver
.findElement(By
.css('.generate')).click();
2043 driver
.sleep(generateDelay
).then(function() {
2044 driver
.findElement(By
.css(".phrase"))
2045 .getAttribute("value").then(function(phrase
) {
2046 expect(phrase
.search(/[a-z]/)).toBe(-1);
2047 expect(phrase
.length
).toBeGreaterThan(0);
2054 // Entropy can be entered by the user
2055 it('Allows entropy to be entered', function(done
) {
2056 driver
.findElement(By
.css('.use-entropy'))
2058 driver
.findElement(By
.css('.entropy'))
2059 .sendKeys('00000000 00000000 00000000 00000000');
2060 driver
.sleep(generateDelay
).then(function() {
2061 driver
.findElement(By
.css(".phrase"))
2062 .getAttribute("value").then(function(phrase
) {
2063 expect(phrase
).toBe("abandon abandon ability");
2064 getFirstAddress(function(address
) {
2065 expect(address
).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug");
2072 // A warning about entropy is shown to the user, with additional information
2073 it('Shows a warning about using entropy', function(done
) {
2074 driver
.findElement(By
.css('.use-entropy'))
2076 driver
.findElement(By
.css('.entropy-container'))
2078 .then(function(containerText
) {
2079 var warning
= "mnemonic may be insecure";
2080 expect(containerText
).toContain(warning
);
2081 driver
.findElement(By
.css('#entropy-notes'))
2082 .findElement(By
.xpath("parent::*"))
2084 .then(function(notesText
) {
2085 var detail
= "flipping a fair coin, rolling a fair dice, noise measurements etc";
2086 expect(notesText
).toContain(detail
);
2092 // The types of entropy available are described to the user
2093 it('Shows the types of entropy available', function(done
) {
2094 driver
.findElement(By
.css('.entropy'))
2095 .getAttribute("placeholder")
2096 .then(function(placeholderText
) {
2105 for (var i
=0; i
<options
.length
; i
++) {
2106 var option
= options
[i
];
2107 expect(placeholderText
).toContain(option
);
2113 // The actual entropy used is shown to the user
2114 it('Shows the actual entropy used', function(done
) {
2115 driver
.findElement(By
.css('.use-entropy'))
2117 driver
.findElement(By
.css('.entropy'))
2118 .sendKeys('Not A Very Good Entropy Source At All');
2119 driver
.sleep(generateDelay
).then(function() {
2120 driver
.findElement(By
.css('.entropy-container'))
2122 .then(function(text
) {
2123 expect(text
).toMatch(/Filtered Entropy
\s
+AedEceAA
/);
2129 // Binary entropy can be entered
2130 it('Allows binary entropy to be entered', function(done
) {
2131 testEntropyType(done
, "01", "binary");
2134 // Base 6 entropy can be entered
2135 it('Allows base 6 entropy to be entered', function(done
) {
2136 testEntropyType(done
, "012345", "base 6");
2139 // Base 6 dice entropy can be entered
2140 it('Allows base 6 dice entropy to be entered', function(done
) {
2141 testEntropyType(done
, "123456", "base 6 (dice)");
2144 // Base 10 entropy can be entered
2145 it('Allows base 10 entropy to be entered', function(done
) {
2146 testEntropyType(done
, "789", "base 10");
2149 // Hexadecimal entropy can be entered
2150 it('Allows hexadecimal entropy to be entered', function(done
) {
2151 testEntropyType(done
, "abcdef", "hexadecimal");
2154 // Dice entropy value is shown as the converted base 6 value
2155 // ie 123456 is converted to 123450
2156 it('Shows dice entropy as base 6', function(done
) {
2157 driver
.findElement(By
.css('.use-entropy'))
2159 driver
.findElement(By
.css('.entropy'))
2160 .sendKeys("123456");
2161 driver
.sleep(generateDelay
).then(function() {
2162 driver
.findElement(By
.css('.entropy-container'))
2164 .then(function(text
) {
2165 expect(text
).toMatch(/Filtered Entropy
\s
+123450/);
2171 // The number of bits of entropy accumulated is shown
2172 it("Shows the number of bits of entropy for 20 bits of binary", function(done
) {
2173 testEntropyBits(done
, "0000 0000 0000 0000 0000", "20");
2175 it("Shows the number of bits of entropy for 1 bit of binary", function(done
) {
2176 testEntropyBits(done
, "0", "1");
2178 it("Shows the number of bits of entropy for 4 bits of binary", function(done
) {
2179 testEntropyBits(done
, "0000", "4");
2181 it("Shows the number of bits of entropy for 1 character of base 6 (dice)", function(done
) {
2182 // 6 in card is 0 in base 6, 0 in base 6 is 2.6 bits (rounded down to 2 bits)
2183 testEntropyBits(done
, "6", "2");
2185 it("Shows the number of bits of entropy for 1 character of base 10 with 3 bits", function(done
) {
2186 // 7 in base 10 is 111 in base 2, no leading zeros
2187 testEntropyBits(done
, "7", "3");
2189 it("Shows the number of bits of entropy for 1 character of base 10 with 4 bis", function(done
) {
2190 testEntropyBits(done
, "8", "4");
2192 it("Shows the number of bits of entropy for 1 character of hex", function(done
) {
2193 testEntropyBits(done
, "F", "4");
2195 it("Shows the number of bits of entropy for 2 characters of base 10", function(done
) {
2196 testEntropyBits(done
, "29", "6");
2198 it("Shows the number of bits of entropy for 2 characters of hex", function(done
) {
2199 testEntropyBits(done
, "0A", "8");
2201 it("Shows the number of bits of entropy for 2 characters of hex with 3 leading zeros", function(done
) {
2202 // hex is always multiple of 4 bits of entropy
2203 testEntropyBits(done
, "1A", "8");
2205 it("Shows the number of bits of entropy for 2 characters of hex with 2 leading zeros", function(done
) {
2206 testEntropyBits(done
, "2A", "8");
2208 it("Shows the number of bits of entropy for 2 characters of hex with 1 leading zero", function(done
) {
2209 testEntropyBits(done
, "4A", "8");
2211 it("Shows the number of bits of entropy for 2 characters of hex with no leading zeros", function(done
) {
2212 testEntropyBits(done
, "8A", "8");
2214 it("Shows the number of bits of entropy for 2 characters of hex starting with F", function(done
) {
2215 testEntropyBits(done
, "FA", "8");
2217 it("Shows the number of bits of entropy for 4 characters of hex with leading zeros", function(done
) {
2218 testEntropyBits(done
, "000A", "16");
2220 it("Shows the number of bits of entropy for 4 characters of base 6", function(done
) {
2221 testEntropyBits(done
, "5555", "11");
2223 it("Shows the number of bits of entropy for 4 characters of base 6 dice", function(done
) {
2224 // uses dice, so entropy is actually 0000 in base 6, which is 4 lots of
2225 // 2.58 bits, which is 10.32 bits (rounded down to 10 bits)
2226 testEntropyBits(done
, "6666", "10");
2228 it("Shows the number of bits of entropy for 4 charactes of base 10", function(done
) {
2229 // Uses base 10, which is 4 lots of 3.32 bits, which is 13.3 bits (rounded
2231 testEntropyBits(done
, "2227", "13");
2233 it("Shows the number of bits of entropy for 4 characters of hex with 2 leading zeros", function(done
) {
2234 testEntropyBits(done
, "222F", "16");
2236 it("Shows the number of bits of entropy for 4 characters of hex starting with F", function(done
) {
2237 testEntropyBits(done
, "FFFF", "16");
2239 it("Shows the number of bits of entropy for 10 characters of base 10", function(done
) {
2240 // 10 events at 3.32 bits per event
2241 testEntropyBits(done
, "0000101017", "33");
2243 it("Shows the number of bits of entropy for a full deck of cards", function(done
) {
2244 // cards are not replaced, so a full deck is not 52^52 entropy which is 296
2245 // bits, it's 52!, which is 225 bits
2246 testEntropyBits(done
, "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks", "225");
2249 it("Shows details about the entered entropy", function(done
) {
2250 testEntropyFeedback(done
,
2254 type: "hexadecimal",
2258 strength: "less than a second",
2262 it("Shows details about the entered entropy", function(done
) {
2263 testEntropyFeedback(done
,
2265 entropy: "AAAAAAAA",
2266 filtered: "AAAAAAAA",
2267 type: "hexadecimal",
2271 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2275 it("Shows details about the entered entropy", function(done
) {
2276 testEntropyFeedback(done
,
2278 entropy: "AAAAAAAA B",
2279 filtered: "AAAAAAAAB",
2280 type: "hexadecimal",
2284 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2288 it("Shows details about the entered entropy", function(done
) {
2289 testEntropyFeedback(done
,
2291 entropy: "AAAAAAAA BBBBBBBB",
2292 filtered: "AAAAAAAABBBBBBBB",
2293 type: "hexadecimal",
2297 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2301 it("Shows details about the entered entropy", function(done
) {
2302 testEntropyFeedback(done
,
2304 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC",
2305 filtered: "AAAAAAAABBBBBBBBCCCCCCCC",
2306 type: "hexadecimal",
2310 strength: "less than a second",
2314 it("Shows details about the entered entropy", function(done
) {
2315 testEntropyFeedback(done
,
2317 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDD",
2318 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDD",
2319 type: "hexadecimal",
2323 strength: "2 minutes",
2327 it("Shows details about the entered entropy", function(done
) {
2328 testEntropyFeedback(done
,
2330 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA",
2331 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDA",
2332 type: "hexadecimal",
2340 it("Shows details about the entered entropy", function(done
) {
2341 testEntropyFeedback(done
,
2343 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA EEEEEEEE",
2344 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDAEEEEEEEE",
2345 type: "hexadecimal",
2349 strength: "3 years",
2353 it("Shows details about the entered entropy", function(done
) {
2354 testEntropyFeedback(done
,
2356 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA EEEEEEEE FFFFFFFF",
2357 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDAEEEEEEEEFFFFFFFF",
2358 type: "hexadecimal",
2362 strength: "centuries",
2366 it("Shows details about the entered entropy", function(done
) {
2367 testEntropyFeedback(done
,
2374 strength: "less than a second",
2378 it("Shows details about the entered entropy", function(done
) {
2379 testEntropyFeedback(done
,
2381 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2382 type: "card (full deck)",
2386 strength: "centuries",
2390 it("Shows details about the entered entropy", function(done
) {
2391 testEntropyFeedback(done
,
2393 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks3d",
2394 type: "card (full deck, 1 duplicate: 3d)",
2398 strength: "centuries",
2402 it("Shows details about the entered entropy", function(done
) {
2403 testEntropyFeedback(done
,
2405 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqs3d4d",
2406 type: "card (2 duplicates: 3d 4d, 1 missing: KS)",
2410 strength: "centuries",
2414 it("Shows details about the entered entropy", function(done
) {
2415 testEntropyFeedback(done
,
2417 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqs3d4d5d6d",
2418 type: "card (4 duplicates: 3d 4d 5d..., 1 missing: KS)",
2422 strength: "centuries",
2426 it("Shows details about the entered entropy", function(done
) {
2427 testEntropyFeedback(done
,
2428 // Next test was throwing uncaught error in zxcvbn
2429 // Also tests 451 bits, ie Math.log2(52!)*2 = 225.58 * 2
2431 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsksac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2432 type: "card (full deck, 52 duplicates: ac 2c 3c...)",
2436 strength: "centuries",
2440 it("Shows details about the entered entropy", function(done
) {
2441 testEntropyFeedback(done
,
2442 // Case insensitivity to duplicate cards
2445 type: "card (1 duplicate: AS)",
2449 strength: "less than a second",
2453 it("Shows details about the entered entropy", function(done
) {
2454 testEntropyFeedback(done
,
2457 type: "card (1 duplicate: as)",
2461 strength: "less than a second",
2465 it("Shows details about the entered entropy", function(done
) {
2466 testEntropyFeedback(done
,
2467 // Missing cards are detected
2469 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2470 type: "card (1 missing: 9C)",
2474 strength: "centuries",
2478 it("Shows details about the entered entropy", function(done
) {
2479 testEntropyFeedback(done
,
2481 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2482 type: "card (2 missing: 9C 5D)",
2486 strength: "centuries",
2490 it("Shows details about the entered entropy", function(done
) {
2491 testEntropyFeedback(done
,
2493 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d7d8d9dtdjd kdah2h3h 5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2494 type: "card (4 missing: 9C 5D QD...)",
2498 strength: "centuries",
2502 it("Shows details about the entered entropy", function(done
) {
2503 testEntropyFeedback(done
,
2504 // More than six missing cards does not show message
2506 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d 8d9d jd kdah2h3h 5h6h7h8h9hthjhqhkh 2s3s4s5s6s7s8s9stsjsqsks",
2511 strength: "centuries",
2515 it("Shows details about the entered entropy", function(done
) {
2516 testEntropyFeedback(done
,
2517 // Multiple decks of cards increases bits per event
2522 bitsPerEvent: "4.34",
2526 it("Shows details about the entered entropy", function(done
) {
2527 testEntropyFeedback(done
,
2532 bitsPerEvent: "4.80",
2536 it("Shows details about the entered entropy", function(done
) {
2537 testEntropyFeedback(done
,
2542 bitsPerEvent: "5.01",
2546 it("Shows details about the entered entropy", function(done
) {
2547 testEntropyFeedback(done
,
2549 entropy: "3d3d3d3d",
2552 bitsPerEvent: "5.14",
2556 it("Shows details about the entered entropy", function(done
) {
2557 testEntropyFeedback(done
,
2559 entropy: "3d3d3d3d3d",
2562 bitsPerEvent: "5.22",
2566 it("Shows details about the entered entropy", function(done
) {
2567 testEntropyFeedback(done
,
2569 entropy: "3d3d3d3d3d3d",
2572 bitsPerEvent: "5.28",
2576 it("Shows details about the entered entropy", function(done
) {
2577 testEntropyFeedback(done
,
2579 entropy: "3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d",
2582 bitsPerEvent: "5.59",
2583 strength: 'less than a second - Repeats like "abcabcabc" are only slightly harder to guess than "abc"',
2588 // Entropy is truncated from the left
2589 it('Truncates entropy from the left', function(done
) {
2590 // Truncate from left means 0000 is removed from the start
2591 // which gives mnemonic 'avocado zoo zone'
2592 // not 1111 removed from the end
2593 // which gives the mnemonic 'abstract zoo zoo'
2594 var entropy
= "00000000 00000000 00000000 00000000";
2595 entropy
+= "11111111 11111111 11111111 1111"; // Missing last byte
2596 driver
.findElement(By
.css('.use-entropy'))
2598 driver
.findElement(By
.css('.entropy'))
2600 driver
.sleep(generateDelay
).then(function() {
2601 driver
.findElement(By
.css(".phrase"))
2602 .getAttribute("value").then(function(phrase
) {
2603 expect(phrase
).toBe("avocado zoo zone");
2609 // Very large entropy results in very long mnemonics
2610 it('Converts very long entropy to very long mnemonics', function(done
) {
2612 for (var i
=0; i
<33; i
++) {
2613 entropy
+= "AAAAAAAA"; // 3 words * 33 iterations = 99 words
2615 driver
.findElement(By
.css('.use-entropy'))
2617 driver
.findElement(By
.css('.entropy'))
2619 driver
.sleep(generateDelay
).then(function() {
2620 driver
.findElement(By
.css(".phrase"))
2621 .getAttribute("value").then(function(phrase
) {
2622 var wordCount
= phrase
.split(/\s+/g).length
;
2623 expect(wordCount
).toBe(99);
2629 // Is compatible with bip32jp entropy
2630 // https://bip32jp.github.io/english/index.html
2632 // Is incompatible with:
2634 it('Is compatible with bip32jp.github.io', function(done
) {
2635 var entropy
= "543210543210543210543210543210543210543210543210543210543210543210543210543210543210543210543210543";
2636 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";
2637 driver
.findElement(By
.css('.use-entropy'))
2639 driver
.findElement(By
.css('.entropy'))
2641 driver
.sleep(generateDelay
).then(function() {
2642 driver
.findElement(By
.css(".phrase"))
2643 .getAttribute("value").then(function(phrase
) {
2644 expect(phrase
).toBe(expectedPhrase
);
2650 // Blank entropy does not generate mnemonic or addresses
2651 it('Does not generate mnemonic for blank entropy', function(done
) {
2652 driver
.findElement(By
.css('.use-entropy'))
2654 driver
.findElement(By
.css('.entropy'))
2656 // check there is no mnemonic
2657 driver
.sleep(generateDelay
).then(function() {
2658 driver
.findElement(By
.css(".phrase"))
2659 .getAttribute("value").then(function(phrase
) {
2660 expect(phrase
).toBe("");
2661 // check there is no mnemonic
2662 driver
.findElements(By
.css(".address"))
2663 .then(function(addresses
) {
2664 expect(addresses
.length
).toBe(0);
2665 // Check the feedback says 'blank entropy'
2666 driver
.findElement(By
.css(".feedback"))
2668 .then(function(feedbackText
) {
2669 expect(feedbackText
).toBe("Blank entropy");
2677 // Mnemonic length can be selected even for weak entropy
2678 it('Allows selection of mnemonic length even for weak entropy', function(done
) {
2679 driver
.findElement(By
.css('.use-entropy'))
2681 driver
.executeScript(function() {
2682 $(".mnemonic-length").val("18").trigger("change");
2684 driver
.findElement(By
.css('.entropy'))
2685 .sendKeys("012345");
2686 driver
.sleep(generateDelay
).then(function() {
2687 driver
.findElement(By
.css(".phrase"))
2688 .getAttribute("value").then(function(phrase
) {
2689 var wordCount
= phrase
.split(/\s+/g).length
;
2690 expect(wordCount
).toBe(18);
2697 // https://github.com/iancoleman/bip39/issues/33
2698 // Final cards should contribute entropy
2699 it('Uses as much entropy as possible for the mnemonic', function(done
) {
2700 driver
.findElement(By
.css('.use-entropy'))
2702 driver
.findElement(By
.css('.entropy'))
2703 .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");
2704 driver
.sleep(generateDelay
).then(function() {
2706 driver
.findElement(By
.css(".phrase"))
2707 .getAttribute("value").then(function(originalPhrase
) {
2708 // Set the last 12 cards to be AS
2709 driver
.findElement(By
.css('.entropy'))
2711 driver
.findElement(By
.css('.entropy'))
2712 .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");
2713 driver
.sleep(generateDelay
).then(function() {
2715 driver
.findElement(By
.css(".phrase"))
2716 .getAttribute("value").then(function(newPhrase
) {
2717 expect(originalPhrase
).not
.toEqual(newPhrase
);
2726 // https://github.com/iancoleman/bip39/issues/35
2728 // TODO this doesn't work in selenium with firefox
2729 // see https://stackoverflow.com/q/40360223
2730 it('Shows a qr code on hover for the phrase', function(done
) {
2731 if (browser
== "firefox") {
2732 pending("Selenium + Firefox bug for mouseMove, see https://stackoverflow.com/q/40360223");
2734 // generate a random mnemonic
2735 var generateEl
= driver
.findElement(By
.css('.generate'));
2737 // toggle qr to show (hidden by default)
2738 var phraseEl
= driver
.findElement(By
.css(".phrase"));
2740 var rootKeyEl
= driver
.findElement(By
.css(".root-key"));
2741 driver
.sleep(generateDelay
).then(function() {
2742 // hover over the root key
2743 driver
.actions().mouseMove(rootKeyEl
).perform().then(function() {
2744 // check the qr code shows
2745 driver
.executeScript(function() {
2746 return $(".qr-container").find("canvas").length
> 0;
2748 .then(function(qrShowing
) {
2749 expect(qrShowing
).toBe(true);
2750 // hover away from the phrase
2751 driver
.actions().mouseMove(generateEl
).perform().then(function() {;
2752 // check the qr code hides
2753 driver
.executeScript(function() {
2754 return $(".qr-container").find("canvas").length
== 0;
2756 .then(function(qrHidden
) {
2757 expect(qrHidden
).toBe(true);
2766 // BIP44 account extendend private key is shown
2767 // github issue 37 - compatibility with electrum
2768 it('Shows the bip44 account extended private key', function(done
) {
2769 driver
.findElement(By
.css(".phrase"))
2770 .sendKeys("abandon abandon ability");
2771 driver
.sleep(generateDelay
).then(function() {
2772 driver
.findElement(By
.css("#bip44 .account-xprv"))
2773 .getAttribute("value")
2774 .then(function(xprv
) {
2775 expect(xprv
).toBe("xprv9yzrnt4zWVJUr1k2VxSPy9ettKz5PpeDMgaVG7UKedhqnw1tDkxP2UyYNhuNSumk2sLE5ctwKZs9vwjsq3e1vo9egCK6CzP87H2cVYXpfwQ");
2781 // BIP44 account extendend public key is shown
2782 // github issue 37 - compatibility with electrum
2783 it('Shows the bip44 account extended public key', function(done
) {
2784 driver
.findElement(By
.css(".phrase"))
2785 .sendKeys("abandon abandon ability");
2786 driver
.sleep(generateDelay
).then(function() {
2787 driver
.findElement(By
.css("#bip44 .account-xpub"))
2788 .getAttribute("value")
2789 .then(function(xprv
) {
2790 expect(xprv
).toBe("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2797 // BIP32 root key can be set as an xpub
2798 it('Generates addresses from xpub as bip32 root key', function(done
) {
2799 driver
.findElement(By
.css('#bip32-tab a'))
2801 // set xpub for account 0 of bip44 for 'abandon abandon ability'
2802 driver
.findElement(By
.css("#root-key"))
2803 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2804 driver
.sleep(generateDelay
).then(function() {
2805 // check the addresses are generated
2806 getFirstAddress(function(address
) {
2807 expect(address
).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug");
2808 // check the xprv key is not set
2809 driver
.findElement(By
.css(".extended-priv-key"))
2810 .getAttribute("value")
2811 .then(function(xprv
) {
2812 expect(xprv
).toBe("NA");
2813 // check the private key is not set
2814 driver
.findElements(By
.css(".privkey"))
2815 .then(function(els
) {
2818 .then(function(privkey
) {
2819 expect(xprv
).toBe("NA");
2829 // xpub for bip32 root key will not work with hardened derivation paths
2830 it('Shows error for hardened derivation paths with xpub root key', function(done
) {
2831 // set xpub for account 0 of bip44 for 'abandon abandon ability'
2832 driver
.findElement(By
.css("#root-key"))
2833 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2834 driver
.sleep(feedbackDelay
).then(function() {
2835 // Check feedback is correct
2836 driver
.findElement(By
.css('.feedback'))
2838 .then(function(feedback
) {
2839 var msg
= "Hardened derivation path is invalid with xpub key";
2840 expect(feedback
).toBe(msg
);
2841 // Check no addresses are shown
2842 driver
.findElements(By
.css('.addresses tr'))
2843 .then(function(rows
) {
2844 expect(rows
.length
).toBe(0);
2852 // no root key shows feedback
2853 it('Shows feedback for no root key', function(done
) {
2854 // set xpub for account 0 of bip44 for 'abandon abandon ability'
2855 driver
.findElement(By
.css('#bip32-tab a'))
2857 driver
.sleep(feedbackDelay
).then(function() {
2858 // Check feedback is correct
2859 driver
.findElement(By
.css('.feedback'))
2861 .then(function(feedback
) {
2862 expect(feedback
).toBe("Invalid root key");
2869 // display error switching tabs while addresses are generating
2870 it('Can change details while old addresses are still being generated', function(done
) {
2871 // Set to generate 199 more addresses.
2872 // This will take a long time allowing a new set of addresses to be
2873 // generated midway through this lot.
2874 // The newly generated addresses should not include any from the old set.
2875 // Any more than 199 will show an alert which needs to be accepted.
2876 driver
.findElement(By
.css('.rows-to-add'))
2878 driver
.findElement(By
.css('.rows-to-add'))
2881 driver
.findElement(By
.css('.phrase'))
2882 .sendKeys("abandon abandon ability");
2883 driver
.sleep(generateDelay
).then(function() {
2884 // change tabs which should cancel the previous generating
2885 driver
.findElement(By
.css('.rows-to-add'))
2887 driver
.findElement(By
.css('.rows-to-add'))
2889 driver
.findElement(By
.css('#bip32-tab a'))
2891 driver
.sleep(generateDelay
).then(function() {
2892 driver
.findElements(By
.css('.index'))
2893 .then(function(els
) {
2894 // check the derivation paths have the right quantity
2895 expect(els
.length
).toBe(20);
2896 // check the derivation paths are in order
2897 testRowsAreInCorrectOrder(done
);
2901 }, generateDelay
+ 5000);
2904 // padding for binary should give length with multiple of 256
2905 // hashed entropy 1111 is length 252, so requires 4 leading zeros
2906 // prior to issue 49 it would only generate 2 leading zeros, ie missing 2
2907 it('Pads hashed entropy with leading zeros', function(done
) {
2908 driver
.findElement(By
.css('.use-entropy'))
2910 driver
.executeScript(function() {
2911 $(".mnemonic-length").val("15").trigger("change");
2913 driver
.findElement(By
.css('.entropy'))
2915 driver
.sleep(generateDelay
).then(function() {
2916 driver
.findElement(By
.css('.phrase'))
2917 .getAttribute("value")
2918 .then(function(phrase
) {
2919 expect(phrase
).toBe("avocado valid quantum cross link predict excuse edit street able flame large galaxy ginger nuclear");
2925 // Github pull request 55
2926 // https://github.com/iancoleman/bip39/pull/55
2928 it('Can set the derivation path on bip32 tab for bitcoincore', function(done
) {
2929 testClientSelect(done
, {
2931 bip32path: "m/0'/0'",
2932 useHardenedAddresses: "true",
2935 it('Can set the derivation path on bip32 tab for multibit', function(done
) {
2936 testClientSelect(done
, {
2938 bip32path: "m/0'/0",
2939 useHardenedAddresses: null,
2942 it('Can set the derivation path on bip32 tab for coinomi/ledger', function(done
) {
2943 testClientSelect(done
, {
2945 bip32path: "m/44'/0'/0'",
2946 useHardenedAddresses: null,
2951 // https://github.com/iancoleman/bip39/issues/58
2952 // bip32 derivation is correct, does not drop leading zeros
2954 // https://medium.com/@alexberegszaszi/why-do-my-bip32-wallets-disagree-6f3254cc5846
2955 it('Retains leading zeros for bip32 derivation', function(done
) {
2956 driver
.findElement(By
.css(".phrase"))
2957 .sendKeys("fruit wave dwarf banana earth journey tattoo true farm silk olive fence");
2958 driver
.findElement(By
.css(".passphrase"))
2959 .sendKeys("banana");
2960 driver
.sleep(generateDelay
).then(function() {
2961 getFirstAddress(function(address
) {
2962 // Note that bitcore generates an incorrect address
2963 // 13EuKhffWkBE2KUwcbkbELZb1MpzbimJ3Y
2964 // see the medium.com link above for more details
2965 expect(address
).toBe("17rxURoF96VhmkcEGCj5LNQkmN9HVhWb7F");
2972 // Japanese mnemonics generate incorrect bip32 seed
2973 // BIP39 seed is set from phrase
2974 it('Generates correct seed for Japanese mnemonics', function(done
) {
2975 driver
.findElement(By
.css(".phrase"))
2976 .sendKeys("あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あおぞら");
2977 driver
.findElement(By
.css(".passphrase"))
2978 .sendKeys("メートルガバヴァぱばぐゞちぢ十人十色");
2979 driver
.sleep(generateDelay
).then(function() {
2980 driver
.findElement(By
.css(".seed"))
2981 .getAttribute("value")
2982 .then(function(seed
) {
2983 expect(seed
).toBe("a262d6fb6122ecf45be09c50492b31f92e9beb7d9a845987a02cefda57a15f9c467a17872029a9e92299b5cbdf306e3a0ee620245cbd508959b6cb7ca637bd55");
2989 // BIP49 official test vectors
2990 // https://github.com/bitcoin/bips/blob/master/bip-0049.mediawiki#test-vectors
2991 it('Generates BIP49 addresses matching the official test vectors', function(done
) {
2992 driver
.findElement(By
.css('#bip49-tab a'))
2994 selectNetwork("BTC - Bitcoin Testnet");
2995 driver
.findElement(By
.css(".phrase"))
2996 .sendKeys("abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about");
2997 driver
.sleep(generateDelay
).then(function() {
2998 getFirstAddress(function(address
) {
2999 expect(address
).toBe("2Mww8dCYPUpKHofjgcXcBCEGmniw9CoaiD2");
3005 // BIP49 derivation path is shown
3006 it('Shows the bip49 derivation path', function(done
) {
3007 driver
.findElement(By
.css('#bip49-tab a'))
3009 driver
.findElement(By
.css(".phrase"))
3010 .sendKeys("abandon abandon ability");
3011 driver
.sleep(generateDelay
).then(function() {
3012 driver
.findElement(By
.css('#bip49 .path'))
3013 .getAttribute("value")
3014 .then(function(path
) {
3015 expect(path
).toBe("m/49'/0'/0'/0");
3021 // BIP49 extended private key is shown
3022 it('Shows the bip49 extended private key', function(done
) {
3023 driver
.findElement(By
.css('#bip49-tab a'))
3025 driver
.findElement(By
.css(".phrase"))
3026 .sendKeys("abandon abandon ability");
3027 driver
.sleep(generateDelay
).then(function() {
3028 driver
.findElement(By
.css('.extended-priv-key'))
3029 .getAttribute("value")
3030 .then(function(xprv
) {
3031 expect(xprv
).toBe("yprvALYB4DYRG6CzzVgzQZwwqjAA2wjBGC3iEd7KYYScpoDdmf75qMRWZWxoFcRXBJjgEXdFqJ9vDRGRLJQsrL22Su5jMbNFeM9vetaGVqy9Qy2");
3037 // BIP49 extended public key is shown
3038 it('Shows the bip49 extended public key', function(done
) {
3039 driver
.findElement(By
.css('#bip49-tab a'))
3041 driver
.findElement(By
.css(".phrase"))
3042 .sendKeys("abandon abandon ability");
3043 driver
.sleep(generateDelay
).then(function() {
3044 driver
.findElement(By
.css('.extended-pub-key'))
3045 .getAttribute("value")
3046 .then(function(xprv
) {
3047 expect(xprv
).toBe("ypub6ZXXTj5K6TmJCymTWbUxCs6tayZffemZbr2vLvrEP8kceTSENtjm7KHH6thvAKxVar9fGe8rgsPEX369zURLZ68b4f7Vexz7RuXsjQ69YDt");
3053 // BIP49 account field changes address list
3054 it('Can set the bip49 account field', function(done
) {
3055 driver
.findElement(By
.css('#bip49-tab a'))
3057 driver
.findElement(By
.css('#bip49 .account'))
3059 driver
.findElement(By
.css('#bip49 .account'))
3061 driver
.findElement(By
.css(".phrase"))
3062 .sendKeys("abandon abandon ability");
3063 driver
.sleep(generateDelay
).then(function() {
3064 getFirstAddress(function(address
) {
3065 expect(address
).toBe("381wg1GGN4rP88rNC9v7QWsiww63yLVPsn");
3071 // BIP49 change field changes address list
3072 it('Can set the bip49 change field', function(done
) {
3073 driver
.findElement(By
.css('#bip49-tab a'))
3075 driver
.findElement(By
.css('#bip49 .change'))
3077 driver
.findElement(By
.css('#bip49 .change'))
3079 driver
.findElement(By
.css(".phrase"))
3080 .sendKeys("abandon abandon ability");
3081 driver
.sleep(generateDelay
).then(function() {
3082 getFirstAddress(function(address
) {
3083 expect(address
).toBe("3PEM7MiKed5konBoN66PQhK8r3hjGhy9dT");
3089 // BIP49 account extendend private key is shown
3090 it('Shows the bip49 account extended private key', function(done
) {
3091 driver
.findElement(By
.css('#bip49-tab a'))
3093 driver
.findElement(By
.css(".phrase"))
3094 .sendKeys("abandon abandon ability");
3095 driver
.sleep(generateDelay
).then(function() {
3096 driver
.findElement(By
.css('#bip49 .account-xprv'))
3097 .getAttribute("value")
3098 .then(function(xprv
) {
3099 expect(xprv
).toBe("yprvAHtB1M5Wp675aLzFy9TJYK2mSsLkg6mcBRh5DZTR7L4EnYSmYPaL63KFA4ycg1PngW5LfkmejxzosCs17TKZMpRFKc3z5SJar6QAKaFcaZL");
3105 // BIP49 account extendend public key is shown
3106 it('Shows the bip49 account extended public key', function(done
) {
3107 driver
.findElement(By
.css('#bip49-tab a'))
3109 driver
.findElement(By
.css(".phrase"))
3110 .sendKeys("abandon abandon ability");
3111 driver
.sleep(generateDelay
).then(function() {
3112 driver
.findElement(By
.css('#bip49 .account-xpub'))
3113 .getAttribute("value")
3114 .then(function(xprv
) {
3115 expect(xprv
).toBe("ypub6WsXQrcQeTfNnq4j5AzJuSyVzuBF5ZVTYecg1ws2ffbDfLmv5vtadqdj1NgR6C6gufMpMfJpHxvb6JEQKvETVNWCRanNedfJtnTchZiJtsL");
3121 // Test selecting coin where bip49 is unavailable (eg CLAM)
3122 it('Shows an error on bip49 tab for coins without bip49', function(done
) {
3123 driver
.findElement(By
.css('#bip49-tab a'))
3125 driver
.findElement(By
.css(".phrase"))
3126 .sendKeys("abandon abandon ability");
3127 driver
.sleep(generateDelay
).then(function() {
3128 selectNetwork("CLAM - Clams");
3129 // bip49 available is hidden
3130 driver
.findElement(By
.css('#bip49 .available'))
3131 .getAttribute("class")
3132 .then(function(classes
) {
3133 expect(classes
).toContain("hidden");
3134 // bip49 unavailable is shown
3135 driver
.findElement(By
.css('#bip49 .unavailable'))
3136 .getAttribute("class")
3137 .then(function(classes
) {
3138 expect(classes
).not
.toContain("hidden");
3139 // check there are no addresses shown
3140 driver
.findElements(By
.css('.addresses tr'))
3141 .then(function(rows
) {
3142 expect(rows
.length
).toBe(0);
3143 // check the derived private key is blank
3144 driver
.findElement(By
.css('.extended-priv-key'))
3145 .getAttribute("value")
3146 .then(function(xprv
) {
3147 expect(xprv
).toBe('');
3148 // check the derived public key is blank
3149 driver
.findElement(By
.css('.extended-pub-key'))
3150 .getAttribute("value")
3151 .then(function(xpub
) {
3152 expect(xpub
).toBe('');
3163 // Cleared mnemonic and root key still allows addresses to be generated
3164 // https://github.com/iancoleman/bip39/issues/43
3165 it('Clears old root keys from memory when mnemonic is cleared', function(done
) {
3167 driver
.findElement(By
.css(".phrase"))
3168 .sendKeys("abandon abandon ability");
3169 driver
.sleep(generateDelay
).then(function() {
3170 // clear the mnemonic and root key
3171 // using selenium .clear() doesn't seem to trigger the 'input' event
3172 // so clear it using keys instead
3173 driver
.findElement(By
.css('.phrase'))
3174 .sendKeys(Key
.CONTROL
,"a");
3175 driver
.findElement(By
.css('.phrase'))
3176 .sendKeys(Key
.DELETE
);
3177 driver
.findElement(By
.css('.root-key'))
3178 .sendKeys(Key
.CONTROL
,"a");
3179 driver
.findElement(By
.css('.root-key'))
3180 .sendKeys(Key
.DELETE
);
3181 driver
.sleep(generateDelay
).then(function() {
3182 // try to generate more addresses
3183 driver
.findElement(By
.css('.more'))
3185 driver
.sleep(generateDelay
).then(function() {
3186 driver
.findElements(By
.css(".addresses tr"))
3187 .then(function(els
) {
3188 // check there are no addresses shown
3189 expect(els
.length
).toBe(0);
3198 // error trying to generate addresses from xpub with hardened derivation
3199 it('Shows error for hardened addresses with xpub root key', function(done
) {
3200 driver
.findElement(By
.css('#bip32-tab a'))
3202 driver
.executeScript(function() {
3203 $(".hardened-addresses").prop("checked", true);
3205 // set xpub for account 0 of bip44 for 'abandon abandon ability'
3206 driver
.findElement(By
.css("#root-key"))
3207 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
3208 driver
.sleep(feedbackDelay
).then(function() {
3209 // Check feedback is correct
3210 driver
.findElement(By
.css('.feedback'))
3212 .then(function(feedback
) {
3213 var msg
= "Hardened derivation path is invalid with xpub key";
3214 expect(feedback
).toBe(msg
);
3220 // Litecoin uses ltub by default, and can optionally be set to xprv
3222 // https://github.com/iancoleman/bip39/issues/96
3223 // Issue with extended keys on Litecoin
3224 it('Uses ltub by default for litecoin, but can be set to xprv', function(done
) {
3225 driver
.findElement(By
.css('.phrase'))
3226 .sendKeys("abandon abandon ability");
3227 selectNetwork("LTC - Litecoin");
3228 driver
.sleep(generateDelay
).then(function() {
3229 // check the extended key is generated correctly
3230 driver
.findElement(By
.css('.root-key'))
3231 .getAttribute("value")
3232 .then(function(rootKey
) {
3233 expect(rootKey
).toBe("Ltpv71G8qDifUiNesiPqf6h5V6eQ8ic77oxQiYtawiACjBEx3sTXNR2HGDGnHETYxESjqkMLFBkKhWVq67ey1B2MKQXannUqNy1RZVHbmrEjnEU");
3234 // set litecoin to use ltub
3235 driver
.executeScript(function() {
3236 $(".litecoin-use-ltub").prop("checked", false);
3237 $(".litecoin-use-ltub").trigger("change");
3239 driver
.sleep(generateDelay
).then(function() {
3240 driver
.findElement(By
.css('.root-key'))
3241 .getAttribute("value")
3242 .then(function(rootKey
) {
3243 expect(rootKey
).toBe("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
3252 // https://github.com/iancoleman/bip39/issues/99#issuecomment-327094159
3253 // "warn me emphatically when they have detected invalid input" to the entropy field
3254 // A warning is shown when entropy is filtered and discarded
3255 it('Warns when entropy is filtered and discarded', function(done
) {
3256 driver
.findElement(By
.css('.use-entropy'))
3258 // set entropy to have no filtered content
3259 driver
.findElement(By
.css('.entropy'))
3260 .sendKeys("00000000 00000000 00000000 00000000");
3261 driver
.sleep(generateDelay
).then(function() {
3262 // check the filter warning does not show
3263 driver
.findElement(By
.css('.entropy-container .filter-warning'))
3264 .getAttribute("class")
3265 .then(function(classes
) {
3266 expect(classes
).toContain("hidden");
3267 // set entropy to have some filtered content
3268 driver
.findElement(By
.css('.entropy'))
3269 .sendKeys("10000000 zxcvbn 00000000 00000000 00000000");
3270 driver
.sleep(entropyFeedbackDelay
).then(function() {
3271 // check the filter warning shows
3272 driver
.findElement(By
.css('.entropy-container .filter-warning'))
3273 .getAttribute("class")
3274 .then(function(classes
) {
3275 expect(classes
).not
.toContain("hidden");
3283 // Bitcoin Cash address can be set to use cashaddr format
3284 it('Can use cashaddr format for bitcoin cash addresses', function(done
) {
3285 driver
.executeScript(function() {
3286 $(".use-bch-cashaddr-addresses").prop("checked", true);
3288 driver
.findElement(By
.css('.phrase'))
3289 .sendKeys("abandon abandon ability");
3290 selectNetwork("BCH - Bitcoin Cash");
3291 driver
.sleep(generateDelay
).then(function() {
3292 getFirstAddress(function(address
) {
3293 expect(address
).toBe("bitcoincash:qzlquk7w4hkudxypl4fgv8x279r754dkvur7jpcsps");
3299 // Bitcoin Cash address can be set to use bitpay format
3300 it('Can use bitpay format for bitcoin cash addresses', function(done
) {
3301 driver
.executeScript(function() {
3302 $(".use-bch-bitpay-addresses").prop("checked", true);
3304 driver
.findElement(By
.css('.phrase'))
3305 .sendKeys("abandon abandon ability");
3306 selectNetwork("BCH - Bitcoin Cash");
3307 driver
.sleep(generateDelay
).then(function() {
3308 getFirstAddress(function(address
) {
3309 expect(address
).toBe("CZnpA9HPmvhuhLLPWJP8rNDpLUYXy1LXFk");
3315 // Bitcoin Cash address can be set to use legacy format
3316 it('Can use legacy format for bitcoin cash addresses', function(done
) {
3317 driver
.executeScript(function() {
3318 $(".use-bch-legacy-addresses").prop("checked", true);
3320 driver
.findElement(By
.css('.phrase'))
3321 .sendKeys("abandon abandon ability");
3322 selectNetwork("BCH - Bitcoin Cash");
3323 driver
.sleep(generateDelay
).then(function() {
3324 getFirstAddress(function(address
) {
3325 expect(address
).toBe("1JKvb6wKtsjNoCRxpZ4DGrbniML7z5U16A");
3331 // End of tests ported from old suit, so no more comments above each test now
3333 it('Can generate more addresses from a custom index', function(done
) {
3334 var expectedIndexes
= [
3335 0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,
3336 40,41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59
3338 driver
.findElement(By
.css('.phrase'))
3339 .sendKeys("abandon abandon ability");
3340 driver
.sleep(generateDelay
).then(function() {
3341 // Set start of next lot of rows to be from index 40
3342 // which means indexes 20-39 will not be in the table.
3343 driver
.findElement(By
.css('.more-rows-start-index'))
3345 driver
.findElement(By
.css('.more'))
3347 driver
.sleep(generateDelay
).then(function() {
3348 // Check actual indexes in the table match the expected pattern
3349 driver
.findElements(By
.css(".index"))
3350 .then(function(els
) {
3351 expect(els
.length
).toBe(expectedIndexes
.length
);
3352 var testRowAtIndex = function(i
) {
3353 if (i
>= expectedIndexes
.length
) {
3358 .then(function(actualPath
) {
3359 var noHardened
= actualPath
.replace(/'/g, "");
3360 var pathBits = noHardened.split("/")
3361 var lastBit = pathBits[pathBits.length-1];
3362 var actualIndex = parseInt(lastBit);
3363 var expectedIndex = expectedIndexes[i];
3364 expect(actualIndex).toBe(expectedIndex);
3365 testRowAtIndex(i+1);
3375 it('Can generate BIP141 addresses
with P2WPKH
-in-P2SH semanitcs
', function(done) {
3376 // Sourced from BIP49 official test specs
3377 driver.findElement(By.css('#bip141
-tab a
'))
3379 driver.findElement(By.css('.bip141
-path
'))
3381 driver.findElement(By.css('.bip141
-path
'))
3382 .sendKeys("m/49'/1'/0'/0");
3383 selectNetwork("BTC
- Bitcoin Testnet
");
3384 driver.findElement(By.css(".phrase
"))
3385 .sendKeys("abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about
");
3386 driver.sleep(generateDelay).then(function() {
3387 getFirstAddress(function(address) {
3388 expect(address).toBe("2Mww8dCYPUpKHofjgcXcBCEGmniw9CoaiD2
");
3394 it('Can generate BIP141 addresses with P2WPKH semanitcs', function(done) {
3395 // This result tested against bitcoinjs-lib test spec for segwit address
3396 // using the first private key of this mnemonic and default path m/0
3397 // https://github.com/bitcoinjs/bitcoinjs-lib/blob/9c8503cab0c6c30a95127042703bc18e8d28c76d/test/integration/addresses.js#L50
3398 // so whilst not directly comparable, substituting the private key produces
3399 // identical results between this tool and the bitcoinjs-lib test.
3400 // Private key generated is:
3401 // L3L8Nu9whawPBNLGtFqDhKut9DKKfG3CQoysupT7BimqVCZsLFNP
3402 driver.findElement(By.css('#bip141-tab a'))
3405 driver.executeScript(function() {
3406 $(".bip141
-semantics option
[selected
]").removeAttr("selected
");
3407 $(".bip141
-semantics option
").filter(function(i,e) {
3408 return $(e).html() == "P2WPKH
";
3409 }).prop("selected
", true);
3410 $(".bip141
-semantics
").trigger("change
");
3412 driver.findElement(By.css(".phrase
"))
3413 .sendKeys("abandon abandon ability
");
3414 driver.sleep(generateDelay).then(function() {
3415 getFirstAddress(function(address) {
3416 expect(address).toBe("bc1qfwu6a5a3evygrk8zvdxxvz4547lmpyx5vsfxe9
");
3422 it('Shows the entropy used by the PRNG when clicking generate', function(done) {
3423 driver.findElement(By.css('.generate')).click();
3424 driver.sleep(generateDelay).then(function() {
3425 driver.findElement(By.css('.entropy'))
3426 .getAttribute("value
")
3427 .then(function(entropy) {
3428 expect(entropy).not.toBe("");
3434 it('Shows the index of each word in the mnemonic', function(done) {
3435 driver.findElement(By.css('.phrase'))
3436 .sendKeys("abandon abandon ability
");
3437 driver.sleep(generateDelay).then(function() {
3438 driver.findElement(By.css('.use-entropy'))
3440 driver.findElement(By.css('.word-indexes'))
3442 .then(function(indexes) {
3443 expect(indexes).toBe("0, 0, 1");
3449 it('Shows the derivation path for bip84 tab', function(done) {
3450 driver.findElement(By.css('#bip84-tab a'))
3452 driver.findElement(By.css('.phrase'))
3453 .sendKeys('abandon abandon ability');
3454 driver.sleep(generateDelay).then(function() {
3455 driver.findElement(By.css('#bip84 .path'))
3456 .getAttribute("value
")
3457 .then(function(path) {
3458 expect(path).toBe("m
/84'/0'/0'/0");
3464 it('Shows the extended private key for bip84 tab', function(done) {
3465 driver.findElement(By.css('#bip84-tab a'))
3467 driver.findElement(By.css('.phrase'))
3468 .sendKeys('abandon abandon ability');
3469 driver.sleep(generateDelay).then(function() {
3470 driver.findElement(By.css('.extended-priv-key'))
3471 .getAttribute("value
")
3472 .then(function(path) {
3473 expect(path).toBe("zprvAev3RKrZ3QVKiUFCfdeMRen1BPDJgdNt1XpxiDy8acSs4kkAGTCvq7HeRYRNNpo8EtEjCFQBWavJwtCUR29y4TUCH4X5RXMcyq48uN8y9BP
");
3479 it('Shows the extended public key for bip84 tab', function(done) {
3480 driver.findElement(By.css('#bip84-tab a'))
3482 driver.findElement(By.css('.phrase'))
3483 .sendKeys('abandon abandon ability');
3484 driver.sleep(generateDelay).then(function() {
3485 driver.findElement(By.css('.extended-pub-key'))
3486 .getAttribute("value
")
3487 .then(function(path) {
3488 expect(path).toBe("zpub6suPpqPSsn3cvxKfmfBMnnijjR3o666jNkkZWcNk8wyqwZ5JozXBNuc8Gs7DB3uLwTDvGVTspVEAUQcEjKF3pZHgywVbubdTqbXTUg7usyx
");
3494 it('Changes the address list if bip84 account is changed', function(done) {
3495 driver.findElement(By.css('#bip84-tab a'))
3497 driver.findElement(By.css('#bip84 .account'))
3499 driver.findElement(By.css('.phrase'))
3500 .sendKeys('abandon abandon ability');
3501 driver.sleep(generateDelay).then(function() {
3502 getFirstAddress(function(address) {
3503 expect(address).toBe("bc1qp7vv669t2fy965jdzvqwrraana89ctd5ewc662
");
3509 it('Changes the address list if bip84 change is changed', function(done) {
3510 driver.findElement(By.css('#bip84-tab a'))
3512 driver.findElement(By.css('#bip84 .change'))
3514 driver.findElement(By.css('.phrase'))
3515 .sendKeys('abandon abandon ability');
3516 driver.sleep(generateDelay).then(function() {
3517 getFirstAddress(function(address) {
3518 expect(address).toBe("bc1qr39vj6rh06ff05m53uxq8uazehwhccswylhrs2
");
3524 it('Passes the official BIP84 test spec for rootpriv', function(done) {
3525 driver.findElement(By.css('#bip84-tab a'))
3527 driver.findElement(By.css('.phrase'))
3528 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3529 driver.sleep(generateDelay).then(function() {
3530 driver.findElement(By.css(".root
-key
"))
3531 .getAttribute("value
")
3532 .then(function(rootKey) {
3533 expect(rootKey).toBe("zprvAWgYBBk7JR8Gjrh4UJQ2uJdG1r3WNRRfURiABBE3RvMXYSrRJL62XuezvGdPvG6GFBZduosCc1YP5wixPox7zhZLfiUm8aunE96BBa4Kei5
");
3539 it('Passes the official BIP84 test spec for account 0 xprv', function(done) {
3540 driver.findElement(By.css('#bip84-tab a'))
3542 driver.findElement(By.css('.phrase'))
3543 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3544 driver.sleep(generateDelay).then(function() {
3545 driver.findElement(By.css("#bip84
.account
-xprv
"))
3546 .getAttribute("value
")
3547 .then(function(rootKey) {
3548 expect(rootKey).toBe("zprvAdG4iTXWBoARxkkzNpNh8r6Qag3irQB8PzEMkAFeTRXxHpbF9z4QgEvBRmfvqWvGp42t42nvgGpNgYSJA9iefm1yYNZKEm7z6qUWCroSQnE
");
3554 it('Passes the official BIP84 test spec for account 0 xpub', function(done) {
3555 driver.findElement(By.css('#bip84-tab a'))
3557 driver.findElement(By.css('.phrase'))
3558 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3559 driver.sleep(generateDelay).then(function() {
3560 driver.findElement(By.css("#bip84
.account
-xpub
"))
3561 .getAttribute("value
")
3562 .then(function(rootKey) {
3563 expect(rootKey).toBe("zpub6rFR7y4Q2AijBEqTUquhVz398htDFrtymD9xYYfG1m4wAcvPhXNfE3EfH1r1ADqtfSdVCToUG868RvUUkgDKf31mGDtKsAYz2oz2AGutZYs
");
3569 it('Passes the official BIP84 test spec for account 0 first address', function(done) {
3570 driver.findElement(By.css('#bip84-tab a'))
3572 driver.findElement(By.css('.phrase'))
3573 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3574 driver.sleep(generateDelay).then(function() {
3575 getFirstAddress(function(address) {
3576 expect(address).toBe("bc1qcr8te4kr609gcawutmrza0j4xv80jy8z306fyu
");
3582 it('Passes the official BIP84 test spec for account 0 first change address', function(done) {
3583 driver.findElement(By.css('#bip84-tab a'))
3585 driver.findElement(By.css('.phrase'))
3586 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3587 driver.findElement(By.css('#bip84 .change'))
3589 driver.sleep(generateDelay).then(function() {
3590 getFirstAddress(function(address) {
3591 expect(address).toBe("bc1q8c6fshw2dlwun7ekn9qwf37cu2rn755upcp6el
");
3597 it('Can display the table as csv', function(done) {
3598 var headings = "path
,address
,public key
,private key
";
3599 var row1 = "m
/44'/0'/0'/0/0,1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug
,033f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3
,L26cVSpWFkJ6aQkPkKmTzLqTdLJ923e6CzrVh9cmx21QHsoUmrEE
";
3600 var row20 = "m
/44'/0'/0'/0/19,1KhBy28XLAciXnnRvm71PvQJaETyrxGV55
,02b4b3e396434d8cdd20c03ac4aaa07387784d5d867b75987f516f5705ee68cb3a
,L4GrDrjReMsCAu5DkLXn79jSb95qR7Zfx7eshybCQZ1qL32MXJab
";
3601 driver.findElement(By.css('.phrase'))
3602 .sendKeys('abandon abandon ability');
3603 driver.sleep(generateDelay).then(function() {
3604 driver.findElement(By.css('.csv'))
3605 .getAttribute("value
")
3606 .then(function(csv) {
3607 expect(csv).toContain(headings);
3608 expect(csv).toContain(row1);
3609 expect(csv).toContain(row20);
3615 it('LeftPads ethereum keys that are less than 32 bytes', function(done) {
3616 // see https://github.com/iancoleman/bip39/issues/155
3617 selectNetwork("ETH
- Ethereum
");
3618 driver.findElement(By.css('#bip32-tab a'))
3620 driver.findElement(By.css('#bip32-path'))
3622 driver.findElement(By.css('#bip32-path'))
3623 .sendKeys("m
/44'/60'/0'");
3624 driver.findElement(By.css('.phrase'))
3625 .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');
3626 driver.sleep(generateDelay).then(function() {
3627 getFirstAddress(function(address) {
3628 expect(address).toBe("0x8943E785B4a5714FC87a3aFAad1eB1FeB602B118");
3634 it('Can encrypt private keys using BIP38', function(done) {
3635 // see https://github.com/iancoleman/bip39/issues/140
3636 driver.executeScript(function() {
3637 $(".use-bip38
").prop("checked
", true);
3639 driver.findElement(By.css('.bip38-password'))
3640 .sendKeys('bip38password');
3641 driver.findElement(By.css('.rows-to-add'))
3643 driver.findElement(By.css('.rows-to-add'))
3645 driver.findElement(By.css('.phrase'))
3646 .sendKeys('abandon abandon ability');
3647 driver.sleep(bip38delay).then(function() {
3649 getFirstRowValue(function(address) {
3650 expect(address).toBe("1NCvSdumA3ngMM9c4aqU56AM6rqXddfuXB
");
3652 getFirstRowValue(function(pubkey) {
3653 expect(pubkey).toBe("043f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3884a74447ea901729b1e73a999b7520e7cb55b4120e6432c64153ccab8a848e1
");
3655 getFirstRowValue(function(privkey) {
3656 expect(privkey).toBe("6PRNRiFnj1RoR3sXhymdCvoZCgnUHQpfupNdKkFbWJkwWQEKesWt1EDMDM
");
3662 }, bip38delay + 5000);
3664 it('Shows the checksum for the entropy', function(done) {
3665 driver.findElement(By.css('.use-entropy'))
3667 driver.findElement(By.css('.entropy'))
3668 .sendKeys("00000000000000000000000000000000");
3669 driver.sleep(generateDelay).then(function() {
3670 driver.findElement(By.css('.checksum'))
3672 .then(function(text) {
3673 expect(text).toBe("1");
3679 it('Shows the checksum for the entropy with the correct groupings', function(done) {
3680 driver.findElement(By.css('.use-entropy'))
3682 // create a checksum of 20 bits, which spans multiple words
3683 driver.findElement(By.css('.entropy'))
3684 .sendKeys("F000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
");
3685 driver.sleep(generateDelay).then(function() {
3686 driver.findElement(By.css('.checksum'))
3688 .then(function(text) {
3689 // first group is 9 bits, second group is 11
3690 expect(text).toBe("011010111 01110000110");
3696 it('Uses vprv for bitcoin testnet p2wpkh', function(done) {
3697 selectNetwork("BTC
- Bitcoin Testnet
");
3698 driver.findElement(By.css('#bip84-tab a'))
3700 driver.findElement(By.css('.phrase'))
3701 .sendKeys('abandon abandon ability');
3702 driver.sleep(generateDelay).then(function() {
3703 driver.findElement(By.css('.root-key'))
3704 .getAttribute("value
")
3705 .then(function(path) {
3706 expect(path).toBe("vprv9DMUxX4ShgxML9N2YV5CvWEebWrM9aJ5ULpbRRyzyWu6vs4BzTvbfFFrH41N5hVi7MYSfiugd765L3JmAfDM5po36Y8ouCKRDeYQwByCmS7
");
3712 it('Shows a warning if generating weak mnemonics', function(done) {
3713 driver.executeScript(function() {
3714 $(".strength option
[selected
]").removeAttr("selected
");
3715 $(".strength option
[value
=6]").prop("selected
", true);
3716 $(".strength
").trigger("change
");
3718 driver.findElement(By.css(".generate
-container
.warning
"))
3719 .getAttribute("class")
3720 .then(function(classes) {
3721 expect(classes).not.toContain("hidden
");
3726 it('Does not show a warning if generating strong mnemonics', function(done) {
3727 driver.executeScript(function() {
3728 $(".strength option
[selected
]").removeAttr("selected
");
3729 $(".strength option
[value
=12]").prop("selected
", true);
3731 driver.findElement(By.css(".generate
-container
.warning
"))
3732 .getAttribute("class")
3733 .then(function(classes) {
3734 expect(classes).toContain("hidden
");
3739 it('Shows a warning if overriding weak entropy with longer mnemonics', function(done) {
3740 driver.findElement(By.css('.use-entropy'))
3742 driver.findElement(By.css('.entropy'))
3743 .sendKeys("0123456789abcdef
"); // 6 words
3744 driver.executeScript(function() {
3745 $(".mnemonic
-length
").val("12").trigger("change
");
3747 driver.findElement(By.css(".weak
-entropy
-override
-warning
"))
3748 .getAttribute("class")
3749 .then(function(classes) {
3750 expect(classes).not.toContain("hidden
");
3755 it('Does not show a warning if entropy is stronger than mnemonic length', function(done) {
3756 driver.findElement(By.css('.use-entropy'))
3758 driver.findElement(By.css('.entropy'))
3759 .sendKeys("0123456789abcdef0123456789abcdef0123456789abcdef
"); // 18 words
3760 driver.executeScript(function() {
3761 $(".mnemonic
-length
").val("12").trigger("change
");
3763 driver.findElement(By.css(".weak
-entropy
-override
-warning
"))
3764 .getAttribute("class")
3765 .then(function(classes) {
3766 expect(classes).toContain("hidden
");
3771 it('Shows litecoin BIP49 addresses', function(done) {
3772 driver.findElement(By.css('.phrase'))
3773 .sendKeys('abandon abandon ability');
3774 selectNetwork("LTC
- Litecoin
");
3775 driver.findElement(By.css('#bip49-tab a'))
3777 // bip49 addresses are shown
3778 driver.sleep(generateDelay).then(function() {
3779 driver.findElement(By.css('#bip49 .available'))
3780 .getAttribute("class")
3781 .then(function(classes) {
3782 expect(classes).not.toContain("hidden
");
3783 // check first address
3784 getFirstAddress(function(address) {
3785 expect(address).toBe("MFwLPhsXoBuSLL8cLmW9uK6tChkzduV8qN
");
3792 it('Can use root keys to generate segwit table rows', function(done) {
3793 // segwit uses ypub / zpub instead of xpub but the root key should still
3794 // be valid regardless of the encoding used to import that key.
3795 // Maybe this breaks the reason for the different extended key prefixes, but
3796 // since the parsed root key is used behind the scenes anyhow this should be
3798 driver.findElement(By.css('#root-key'))
3799 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
3800 driver.findElement(By.css('#bip49-tab a'))
3802 // bip49 addresses are shown
3803 driver.sleep(generateDelay).then(function() {
3804 getFirstAddress(function(address) {
3805 expect(address).toBe("3QG2Y9AA4xZ846gKHZqNf7mvVKbLqMKxr2
");