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);
1324 // BIP39 seed is set from phrase
1325 it('Sets the bip39 seed
from the prhase
', function(done) {
1326 driver.findElement(By.css('.phrase
'))
1327 .sendKeys('abandon abandon ability
');
1328 driver.sleep(generateDelay).then(function() {
1329 driver.findElement(By.css('.seed
'))
1330 .getAttribute("value")
1331 .then(function(seed) {
1332 expect(seed).toBe("20da140d3dd1df8713cefcc4d54ce0e445b4151027a1ab567b832f6da5fcc5afc1c3a3f199ab78b8e0ab4652efd7f414ac2c9a3b81bceb879a70f377aa0a58f3");
1338 // BIP32 root key is set from phrase
1339 it('Sets the bip39 root key
from the prhase
', function(done) {
1340 driver.findElement(By.css('.phrase
'))
1341 .sendKeys('abandon abandon ability
');
1342 driver.sleep(generateDelay).then(function() {
1343 driver.findElement(By.css('.root
-key
'))
1344 .getAttribute("value")
1345 .then(function(seed) {
1346 expect(seed).toBe("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
1352 // Tabs show correct addresses when changed
1353 it('Shows the correct address when tab is changed
', function(done) {
1354 driver.findElement(By.css('.phrase
'))
1355 .sendKeys('abandon abandon ability
');
1356 driver.sleep(generateDelay).then(function() {
1357 driver.findElement(By.css('#bip32
-tab a
'))
1359 driver.sleep(generateDelay).then(function() {
1360 getFirstAddress(function(address) {
1361 expect(address).toBe("17uQ7s2izWPwBmEVFikTmZUjbBKWYdJchz");
1368 // BIP44 derivation path is shown
1369 it('Shows the derivation path
for bip44 tab
', function(done) {
1370 driver.findElement(By.css('.phrase
'))
1371 .sendKeys('abandon abandon ability
');
1372 driver.sleep(generateDelay).then(function() {
1373 driver.findElement(By.css('#bip44
.path
'))
1374 .getAttribute("value")
1375 .then(function(path) {
1376 expect(path).toBe("m/44'/0'/0'/0");
1382 // BIP44 extended private key is shown
1383 it('Shows the extended private key for bip44 tab', function(done) {
1384 driver.findElement(By.css('.phrase'))
1385 .sendKeys('abandon abandon ability');
1386 driver.sleep(generateDelay).then(function() {
1387 driver.findElement(By.css('.extended-priv-key'))
1388 .getAttribute("value
")
1389 .then(function(path) {
1390 expect(path).toBe("xprvA2DxxvPZcyRvYgZMGS53nadR32mVDeCyqQYyFhrCVbJNjPoxMeVf7QT5g7mQASbTf9Kp4cryvcXnu2qurjWKcrdsr91jXymdCDNxKgLFKJG
");
1396 // BIP44 extended public key is shown
1397 it('Shows the extended public key for bip44 tab', function(done) {
1398 driver.findElement(By.css('.phrase'))
1399 .sendKeys('abandon abandon ability');
1400 driver.sleep(generateDelay).then(function() {
1401 driver.findElement(By.css('.extended-pub-key'))
1402 .getAttribute("value
")
1403 .then(function(path) {
1404 expect(path).toBe("xpub6FDKNRvTTLzDmAdpNTc49ia9b4byd6vqCdUa46Fp3vqMcC96uBoufCmZXQLiN5AK3iSCJMhf9gT2sxkpyaPepRuA7W3MujV5tGmF5VfbueM
");
1410 // BIP44 account field changes address list
1411 it('Changes the address list if bip44 account is changed', function(done) {
1412 driver.findElement(By.css('#bip44 .account'))
1414 driver.findElement(By.css('.phrase'))
1415 .sendKeys('abandon abandon ability');
1416 driver.sleep(generateDelay).then(function() {
1417 getFirstAddress(function(address) {
1418 expect(address).toBe("1Nq2Wmu726XHCuGhctEtGmhxo3wzk5wZ1H
");
1424 // BIP44 change field changes address list
1425 it('Changes the address list if bip44 change is changed', function(done) {
1426 driver.findElement(By.css('#bip44 .change'))
1428 driver.findElement(By.css('.phrase'))
1429 .sendKeys('abandon abandon ability');
1430 driver.sleep(generateDelay).then(function() {
1431 getFirstAddress(function(address) {
1432 expect(address).toBe("1KAGfWgqfVbSSXY56fNQ7YnhyKuoskHtYo
");
1438 // BIP32 derivation path can be set
1439 it('Can use a custom bip32 derivation path', function(done) {
1440 driver.findElement(By.css('#bip32-tab a'))
1442 driver.findElement(By.css('#bip32 .path'))
1444 driver.findElement(By.css('#bip32 .path'))
1446 driver.findElement(By.css('.phrase'))
1447 .sendKeys('abandon abandon ability');
1448 driver.sleep(generateDelay).then(function() {
1449 getFirstAddress(function(address) {
1450 expect(address).toBe("16pYQQdLD1hH4hwTGLXBaZ9Teboi1AGL8L
");
1456 // BIP32 can use hardened derivation paths
1457 it('Can use a hardened derivation paths', function(done) {
1458 driver.findElement(By.css('#bip32-tab a'))
1460 driver.findElement(By.css('#bip32 .path'))
1462 driver.findElement(By.css('#bip32 .path'))
1464 driver.findElement(By.css('.phrase
'))
1465 .sendKeys('abandon abandon ability
');
1466 driver.sleep(generateDelay).then(function() {
1467 getFirstAddress(function(address) {
1468 expect(address).toBe("14aXZeprXAE3UUKQc4ihvwBvww2LuEoHo4");
1474 // BIP32 extended private key is shown
1475 it('Shows the BIP32 extended
private key
', function(done) {
1476 driver.findElement(By.css('#bip32
-tab a
'))
1478 driver.findElement(By.css('.phrase
'))
1479 .sendKeys('abandon abandon ability
');
1480 driver.sleep(generateDelay).then(function() {
1481 driver.findElement(By.css('.extended
-priv
-key
'))
1482 .getAttribute("value")
1483 .then(function(privKey) {
1484 expect(privKey).toBe("xprv9va99uTVE5aLiutUVLTyfxfe8v8aaXjSQ1XxZbK6SezYVuikA9MnjQVTA8rQHpNA5LKvyQBpLiHbBQiiccKiBDs7eRmBogsvq3THFeLHYbe");
1490 // BIP32 extended public key is shown
1491 it('Shows the BIP32 extended
public key
', function(done) {
1492 driver.findElement(By.css('#bip32
-tab a
'))
1494 driver.findElement(By.css('.phrase
'))
1495 .sendKeys('abandon abandon ability
');
1496 driver.sleep(generateDelay).then(function() {
1497 driver.findElement(By.css('.extended
-pub
-key
'))
1498 .getAttribute("value")
1499 .then(function(pubKey) {
1500 expect(pubKey).toBe("xpub69ZVZQzP4T8dwPxwbMzz36cNgwy4yzTHmETZMyihzzXXNi3thgg3HCow1RtY252wdw5rS8369xKnraN5Q93y3FkFfJp2XEHWUrkyXsjS93P");
1506 // Derivation path is shown in table
1507 it('Shows the derivation path
in the table
', function(done) {
1508 driver.findElement(By.css('.phrase
'))
1509 .sendKeys('abandon abandon ability
');
1510 driver.sleep(generateDelay).then(function() {
1511 getFirstPath(function(path) {
1512 expect(path).toBe("m/44'/0'/0'/0/0");
1518 // Derivation path for address can be hardened
1519 it('Can derive hardened addresses', function(done) {
1520 driver.findElement(By.css('#bip32-tab a'))
1522 driver.executeScript(function() {
1523 $(".hardened
-addresses
").prop("checked
", true);
1525 driver.findElement(By.css('.phrase'))
1526 .sendKeys('abandon abandon ability');
1527 driver.sleep(generateDelay).then(function() {
1528 getFirstAddress(function(address) {
1529 expect(address).toBe("18exLzUv7kfpiXRzmCjFDoC9qwNLFyvwyd
");
1535 // Derivation path visibility can be toggled
1536 it('Can toggle visibility of the derivation path column', function(done) {
1537 driver.findElement(By.css('.phrase'))
1538 .sendKeys('abandon abandon ability');
1539 driver.sleep(generateDelay).then(function() {
1540 driver.findElement(By.css('.index-toggle'))
1542 testColumnValuesAreInvisible(done, "index
");
1547 it('Shows the address in the table', function(done) {
1548 driver.findElement(By.css('.phrase'))
1549 .sendKeys('abandon abandon ability');
1550 driver.sleep(generateDelay).then(function() {
1551 getFirstAddress(function(address) {
1552 expect(address).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug
");
1558 // Addresses are shown in order of derivation path
1559 it('Shows the address in order of derivation path', function(done) {
1560 driver.findElement(By.css('.phrase'))
1561 .sendKeys('abandon abandon ability');
1562 driver.sleep(generateDelay).then(function() {
1563 testRowsAreInCorrectOrder(done);
1567 // Address visibility can be toggled
1568 it('Can toggle visibility of the address column', function(done) {
1569 driver.findElement(By.css('.phrase'))
1570 .sendKeys('abandon abandon ability');
1571 driver.sleep(generateDelay).then(function() {
1572 driver.findElement(By.css('.address-toggle'))
1574 testColumnValuesAreInvisible(done, "address
");
1578 // Public key is shown in table
1579 it('Shows the public key in the table', function(done) {
1580 driver.findElement(By.css('.phrase'))
1581 .sendKeys('abandon abandon ability');
1582 driver.sleep(generateDelay).then(function() {
1583 driver.findElements(By.css('.pubkey'))
1584 .then(function(els) {
1586 .then(function(pubkey) {
1587 expect(pubkey).toBe("033f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3
");
1594 // Public key visibility can be toggled
1595 it('Can toggle visibility of the public key column', function(done) {
1596 driver.findElement(By.css('.phrase'))
1597 .sendKeys('abandon abandon ability');
1598 driver.sleep(generateDelay).then(function() {
1599 driver.findElement(By.css('.public-key-toggle'))
1601 testColumnValuesAreInvisible(done, "pubkey
");
1605 // Private key is shown in table
1606 it('Shows the private key in the table', function(done) {
1607 driver.findElement(By.css('.phrase'))
1608 .sendKeys('abandon abandon ability');
1609 driver.sleep(generateDelay).then(function() {
1610 driver.findElements(By.css('.privkey'))
1611 .then(function(els) {
1613 .then(function(pubkey) {
1614 expect(pubkey).toBe("L26cVSpWFkJ6aQkPkKmTzLqTdLJ923e6CzrVh9cmx21QHsoUmrEE
");
1621 // Private key visibility can be toggled
1622 it('Can toggle visibility of the private key column', function(done) {
1623 driver.findElement(By.css('.phrase'))
1624 .sendKeys('abandon abandon ability');
1625 driver.sleep(generateDelay).then(function() {
1626 driver.findElement(By.css('.private-key-toggle'))
1628 testColumnValuesAreInvisible(done, "privkey
");
1632 // More addresses can be generated
1633 it('Can generate more rows in the table', function(done) {
1634 driver.findElement(By.css('.phrase'))
1635 .sendKeys('abandon abandon ability');
1636 driver.sleep(generateDelay).then(function() {
1637 driver.findElement(By.css('.more'))
1639 driver.sleep(generateDelay).then(function() {
1640 driver.findElements(By.css('.address'))
1641 .then(function(els) {
1642 expect(els.length).toBe(40);
1649 // A custom number of additional addresses can be generated
1650 it('Can generate more rows in the table', function(done) {
1651 driver.findElement(By.css('.phrase'))
1652 .sendKeys('abandon abandon ability');
1653 driver.sleep(generateDelay).then(function() {
1654 driver.findElement(By.css('.rows-to-add'))
1656 driver.findElement(By.css('.rows-to-add'))
1658 driver.findElement(By.css('.more'))
1660 driver.sleep(generateDelay).then(function() {
1661 driver.findElements(By.css('.address'))
1662 .then(function(els) {
1663 expect(els.length).toBe(21);
1670 // Additional addresses are shown in order of derivation path
1671 it('Shows additional addresses in order of derivation path', function(done) {
1672 driver.findElement(By.css('.phrase'))
1673 .sendKeys('abandon abandon ability');
1674 driver.sleep(generateDelay).then(function() {
1675 driver.findElement(By.css('.more'))
1677 driver.sleep(generateDelay).then(function() {
1678 testRowsAreInCorrectOrder(done);
1683 // BIP32 root key can be set by the user
1684 it('Allows the user to set the BIP32 root key', function(done) {
1685 driver.findElement(By.css('.root-key'))
1686 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1687 driver.sleep(generateDelay).then(function() {
1688 getFirstAddress(function(address) {
1689 expect(address).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug
");
1695 // Setting BIP32 root key clears the existing phrase, passphrase and seed
1696 // TODO this doesn't work in selenium with chrome
1697 it('Confirms the existing phrase should be cleared', function(done) {
1698 if (browser == "chrome
") {
1699 pending("Selenium
+ Chrome headless bug
for alert
, see
https://stackoverflow.com/q/45242264");
1701 driver
.findElement(By
.css('.phrase'))
1702 .sendKeys('A non-blank but invalid value');
1703 driver
.findElement(By
.css('.root-key'))
1704 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1705 driver
.switchTo().alert().accept();
1706 driver
.findElement(By
.css('.phrase'))
1707 .getAttribute("value").then(function(value
) {
1708 expect(value
).toBe("");
1713 // Clearing of phrase, passphrase and seed can be cancelled by user
1714 // TODO this doesn't work in selenium with chrome
1715 it('Allows the clearing of the phrase to be cancelled', function(done
) {
1716 if (browser
== "chrome") {
1717 pending("Selenium + Chrome headless bug for alert, see https://stackoverflow.com/q/45242264");
1719 driver
.findElement(By
.css('.phrase'))
1720 .sendKeys('abandon abandon ability');
1721 driver
.sleep(generateDelay
).then(function() {
1722 driver
.findElement(By
.css('.root-key'))
1724 driver
.findElement(By
.css('.root-key'))
1726 driver
.switchTo().alert().dismiss();
1727 driver
.findElement(By
.css('.phrase'))
1728 .getAttribute("value").then(function(value
) {
1729 expect(value
).toBe("abandon abandon ability");
1735 // Custom BIP32 root key is used when changing the derivation path
1736 it('Can set derivation path for root key instead of phrase', function(done
) {
1737 driver
.findElement(By
.css('#bip44 .account'))
1739 driver
.findElement(By
.css('.root-key'))
1740 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1741 driver
.sleep(generateDelay
).then(function() {
1742 getFirstAddress(function(address
) {
1743 expect(address
).toBe("1Nq2Wmu726XHCuGhctEtGmhxo3wzk5wZ1H");
1749 // Incorrect mnemonic shows error
1750 it('Shows an error for incorrect mnemonic', function(done
) {
1751 driver
.findElement(By
.css('.phrase'))
1752 .sendKeys('abandon abandon abandon');
1753 driver
.sleep(feedbackDelay
).then(function() {
1754 driver
.findElement(By
.css('.feedback'))
1756 .then(function(feedback
) {
1757 expect(feedback
).toBe("Invalid mnemonic");
1763 // Incorrect word shows suggested replacement
1764 it('Shows word suggestion for incorrect word', function(done
) {
1765 driver
.findElement(By
.css('.phrase'))
1766 .sendKeys('abandon abandon abiliti');
1767 driver
.sleep(feedbackDelay
).then(function() {
1768 driver
.findElement(By
.css('.feedback'))
1770 .then(function(feedback
) {
1771 var msg
= "abiliti not in wordlist, did you mean ability?";
1772 expect(feedback
).toBe(msg
);
1778 // Github pull request 48
1779 // First four letters of word shows that word, not closest
1780 // since first four letters gives unique word in BIP39 wordlist
1781 // eg ille should show illegal, not idle
1782 it('Shows word suggestion based on first four chars', function(done
) {
1783 driver
.findElement(By
.css('.phrase'))
1785 driver
.sleep(feedbackDelay
).then(function() {
1786 driver
.findElement(By
.css('.feedback'))
1788 .then(function(feedback
) {
1789 var msg
= "ille not in wordlist, did you mean illegal?";
1790 expect(feedback
).toBe(msg
);
1796 // Incorrect BIP32 root key shows error
1797 it('Shows error for incorrect root key', function(done
) {
1798 driver
.findElement(By
.css('.root-key'))
1799 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpj');
1800 driver
.sleep(feedbackDelay
).then(function() {
1801 driver
.findElement(By
.css('.feedback'))
1803 .then(function(feedback
) {
1804 var msg
= "Invalid root key";
1805 expect(feedback
).toBe(msg
);
1811 // Derivation path not starting with m shows error
1812 it('Shows error for derivation path not starting with m', function(done
) {
1813 driver
.findElement(By
.css('#bip32-tab a'))
1815 driver
.findElement(By
.css('#bip32 .path'))
1817 driver
.findElement(By
.css('#bip32 .path'))
1819 driver
.findElement(By
.css('.phrase'))
1820 .sendKeys('abandon abandon ability');
1821 driver
.sleep(feedbackDelay
).then(function() {
1822 driver
.findElement(By
.css('.feedback'))
1824 .then(function(feedback
) {
1825 var msg
= "First character must be 'm'";
1826 expect(feedback
).toBe(msg
);
1832 // Derivation path containing invalid characters shows useful error
1833 it('Shows error for derivation path not starting with m', function(done
) {
1834 driver
.findElement(By
.css('#bip32-tab a'))
1836 driver
.findElement(By
.css('#bip32 .path'))
1838 driver
.findElement(By
.css('#bip32 .path'))
1839 .sendKeys('m/1/0wrong1/1');
1840 driver
.findElement(By
.css('.phrase'))
1841 .sendKeys('abandon abandon ability');
1842 driver
.sleep(feedbackDelay
).then(function() {
1843 driver
.findElement(By
.css('.feedback'))
1845 .then(function(feedback
) {
1846 var msg
= "Invalid characters 0wrong1 found at depth 2";
1847 expect(feedback
).toBe(msg
);
1853 // Github Issue 11: Default word length is 15
1854 // https://github.com/iancoleman/bip39/issues/11
1855 it('Sets the default word length to 15', function(done
) {
1856 driver
.findElement(By
.css('.strength'))
1857 .getAttribute("value")
1858 .then(function(strength
) {
1859 expect(strength
).toBe("15");
1864 // Github Issue 12: Generate more rows with private keys hidden
1865 // https://github.com/iancoleman/bip39/issues/12
1866 it('Sets the correct hidden column state on new rows', function(done
) {
1867 driver
.findElement(By
.css('.phrase'))
1868 .sendKeys("abandon abandon ability");
1869 driver
.sleep(generateDelay
).then(function() {
1870 driver
.findElement(By
.css('.private-key-toggle'))
1872 driver
.findElement(By
.css('.more'))
1874 driver
.sleep(generateDelay
).then(function() {
1875 driver
.findElements(By
.css('.privkey'))
1876 .then(function(els
) {
1877 expect(els
.length
).toBe(40);
1879 testColumnValuesAreInvisible(done
, "privkey");
1884 // Github Issue 19: Mnemonic is not sensitive to whitespace
1885 // https://github.com/iancoleman/bip39/issues/19
1886 it('Ignores excess whitespace in the mnemonic', function(done
) {
1887 var doublespace
= " ";
1888 var mnemonic
= "urge cat" + doublespace
+ "bid";
1889 driver
.findElement(By
.css('.phrase'))
1890 .sendKeys(mnemonic
);
1891 driver
.sleep(generateDelay
).then(function() {
1892 driver
.findElement(By
.css('.root-key'))
1893 .getAttribute("value")
1894 .then(function(seed
) {
1895 expect(seed
).toBe("xprv9s21ZrQH143K3isaZsWbKVoTtbvd34Y1ZGRugGdMeBGbM3AgBVzTH159mj1cbbtYSJtQr65w6L5xy5L9SFC7c9VJZWHxgAzpj4mun5LhrbC");
1901 // Github Issue 23: Part 1: Use correct derivation path when changing tabs
1902 // https://github.com/iancoleman/bip39/issues/23
1903 it('Uses the correct derivation path when changing tabs', function(done
) {
1904 // 1) and 2) set the phrase
1905 driver
.findElement(By
.css('.phrase'))
1906 .sendKeys("abandon abandon ability");
1907 driver
.sleep(generateDelay
).then(function() {
1908 // 3) select bip32 tab
1909 driver
.findElement(By
.css('#bip32-tab a'))
1911 driver
.sleep(generateDelay
).then(function() {
1912 // 4) switch from bitcoin to litecoin
1913 selectNetwork("LTC - Litecoin");
1914 driver
.sleep(generateDelay
).then(function() {
1915 // 5) Check address is displayed correctly
1916 getFirstAddress(function(address
) {
1917 expect(address
).toBe("LS8MP5LZ5AdzSZveRrjm3aYVoPgnfFh5T5");
1918 // 5) Check derivation path is displayed correctly
1919 getFirstPath(function(path
) {
1920 expect(path
).toBe("m/0/0");
1929 // Github Issue 23 Part 2: Coin selection in derivation path
1930 // https://github.com/iancoleman/bip39/issues/23#issuecomment-238011920
1931 it('Uses the correct derivation path when changing coins', function(done
) {
1933 driver
.findElement(By
.css('.phrase'))
1934 .sendKeys("abandon abandon ability");
1935 driver
.sleep(generateDelay
).then(function() {
1936 // switch from bitcoin to clam
1937 selectNetwork("CLAM - Clams");
1938 driver
.sleep(generateDelay
).then(function() {
1939 // check derivation path is displayed correctly
1940 getFirstPath(function(path
) {
1941 expect(path
).toBe("m/44'/23'/0'/0/0");
1948 // Github Issue 26: When using a Root key derrived altcoins are incorrect
1949 // https://github.com/iancoleman/bip39/issues/26
1950 it('Uses the correct derivation for altcoins with root keys', function(done
) {
1951 // 1) 2) and 3) set the root key
1952 driver
.findElement(By
.css('.root-key'))
1953 .sendKeys("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
1954 driver
.sleep(generateDelay
).then(function() {
1955 // 4) switch from bitcoin to viacoin
1956 selectNetwork("VIA - Viacoin");
1957 driver
.sleep(generateDelay
).then(function() {
1958 // 5) ensure the derived address is correct
1959 getFirstAddress(function(address
) {
1960 expect(address
).toBe("Vq9Eq4N5SQnjqZvxtxzo7hZPW5XnyJsmXT");
1967 // Selecting a language with no existing phrase should generate a phrase in
1969 it('Generate a random phrase when language is selected and no current phrase', function(done
) {
1970 driver
.findElement(By
.css("a[href='#japanese']"))
1972 driver
.sleep(generateDelay
).then(function() {
1973 driver
.findElement(By
.css(".phrase"))
1974 .getAttribute("value").then(function(phrase
) {
1975 expect(phrase
.search(/[a-z]/)).toBe(-1);
1976 expect(phrase
.length
).toBeGreaterThan(0);
1982 // Selecting a language with existing phrase should update the phrase to use
1984 it('Updates existing phrases when the language is changed', function(done
) {
1985 driver
.findElement(By
.css(".phrase"))
1986 .sendKeys("abandon abandon ability");
1987 driver
.sleep(generateDelay
).then(function() {
1988 driver
.findElement(By
.css("a[href='#italian']"))
1990 driver
.sleep(generateDelay
).then(function() {
1991 driver
.findElement(By
.css(".phrase"))
1992 .getAttribute("value").then(function(phrase
) {
1993 // Check only the language changes, not the phrase
1994 expect(phrase
).toBe("abaco abaco abbaglio");
1995 getFirstAddress(function(address
) {
1996 // Check the address is correct
1997 expect(address
).toBe("1Dz5TgDhdki9spa6xbPFbBqv5sjMrx3xgV");
2005 // Suggested replacement for erroneous word in non-English language
2006 it('Shows word suggestion for incorrect word in non-English language', function(done
) {
2007 driver
.findElement(By
.css('.phrase'))
2008 .sendKeys('abaco abaco zbbaglio');
2009 driver
.sleep(feedbackDelay
).then(function() {
2010 driver
.findElement(By
.css('.feedback'))
2012 .then(function(feedback
) {
2013 var msg
= "zbbaglio not in wordlist, did you mean abbaglio?";
2014 expect(feedback
).toBe(msg
);
2020 // Japanese word does not break across lines.
2022 // https://github.com/bitcoin/bips/blob/master/bip-0039/bip-0039-wordlists.md#japanese
2023 it('Does not break Japanese words across lines', function(done
) {
2024 driver
.findElement(By
.css('.phrase'))
2025 .getCssValue("word-break")
2026 .then(function(value
) {
2027 expect(value
).toBe("keep-all");
2032 // Language can be specified at page load using hash value in url
2033 it('Can set the language from the url hash', function(done
) {
2034 driver
.get(url
+ "#japanese").then(function() {
2035 driver
.findElement(By
.css('.generate')).click();
2036 driver
.sleep(generateDelay
).then(function() {
2037 driver
.findElement(By
.css(".phrase"))
2038 .getAttribute("value").then(function(phrase
) {
2039 expect(phrase
.search(/[a-z]/)).toBe(-1);
2040 expect(phrase
.length
).toBeGreaterThan(0);
2047 // Entropy can be entered by the user
2048 it('Allows entropy to be entered', function(done
) {
2049 driver
.findElement(By
.css('.use-entropy'))
2051 driver
.findElement(By
.css('.entropy'))
2052 .sendKeys('00000000 00000000 00000000 00000000');
2053 driver
.sleep(generateDelay
).then(function() {
2054 driver
.findElement(By
.css(".phrase"))
2055 .getAttribute("value").then(function(phrase
) {
2056 expect(phrase
).toBe("abandon abandon ability");
2057 getFirstAddress(function(address
) {
2058 expect(address
).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug");
2065 // A warning about entropy is shown to the user, with additional information
2066 it('Shows a warning about using entropy', function(done
) {
2067 driver
.findElement(By
.css('.use-entropy'))
2069 driver
.findElement(By
.css('.entropy-container'))
2071 .then(function(containerText
) {
2072 var warning
= "mnemonic may be insecure";
2073 expect(containerText
).toContain(warning
);
2074 driver
.findElement(By
.css('#entropy-notes'))
2075 .findElement(By
.xpath("parent::*"))
2077 .then(function(notesText
) {
2078 var detail
= "flipping a fair coin, rolling a fair dice, noise measurements etc";
2079 expect(notesText
).toContain(detail
);
2085 // The types of entropy available are described to the user
2086 it('Shows the types of entropy available', function(done
) {
2087 driver
.findElement(By
.css('.entropy'))
2088 .getAttribute("placeholder")
2089 .then(function(placeholderText
) {
2098 for (var i
=0; i
<options
.length
; i
++) {
2099 var option
= options
[i
];
2100 expect(placeholderText
).toContain(option
);
2106 // The actual entropy used is shown to the user
2107 it('Shows the actual entropy used', function(done
) {
2108 driver
.findElement(By
.css('.use-entropy'))
2110 driver
.findElement(By
.css('.entropy'))
2111 .sendKeys('Not A Very Good Entropy Source At All');
2112 driver
.sleep(generateDelay
).then(function() {
2113 driver
.findElement(By
.css('.entropy-container'))
2115 .then(function(text
) {
2116 expect(text
).toMatch(/Filtered Entropy
\s
+AedEceAA
/);
2122 // Binary entropy can be entered
2123 it('Allows binary entropy to be entered', function(done
) {
2124 testEntropyType(done
, "01", "binary");
2127 // Base 6 entropy can be entered
2128 it('Allows base 6 entropy to be entered', function(done
) {
2129 testEntropyType(done
, "012345", "base 6");
2132 // Base 6 dice entropy can be entered
2133 it('Allows base 6 dice entropy to be entered', function(done
) {
2134 testEntropyType(done
, "123456", "base 6 (dice)");
2137 // Base 10 entropy can be entered
2138 it('Allows base 10 entropy to be entered', function(done
) {
2139 testEntropyType(done
, "789", "base 10");
2142 // Hexadecimal entropy can be entered
2143 it('Allows hexadecimal entropy to be entered', function(done
) {
2144 testEntropyType(done
, "abcdef", "hexadecimal");
2147 // Dice entropy value is shown as the converted base 6 value
2148 // ie 123456 is converted to 123450
2149 it('Shows dice entropy as base 6', function(done
) {
2150 driver
.findElement(By
.css('.use-entropy'))
2152 driver
.findElement(By
.css('.entropy'))
2153 .sendKeys("123456");
2154 driver
.sleep(generateDelay
).then(function() {
2155 driver
.findElement(By
.css('.entropy-container'))
2157 .then(function(text
) {
2158 expect(text
).toMatch(/Filtered Entropy
\s
+123450/);
2164 // The number of bits of entropy accumulated is shown
2165 it("Shows the number of bits of entropy for 20 bits of binary", function(done
) {
2166 testEntropyBits(done
, "0000 0000 0000 0000 0000", "20");
2168 it("Shows the number of bits of entropy for 1 bit of binary", function(done
) {
2169 testEntropyBits(done
, "0", "1");
2171 it("Shows the number of bits of entropy for 4 bits of binary", function(done
) {
2172 testEntropyBits(done
, "0000", "4");
2174 it("Shows the number of bits of entropy for 1 character of base 6 (dice)", function(done
) {
2175 // 6 in card is 0 in base 6, 0 in base 6 is 2.6 bits (rounded down to 2 bits)
2176 testEntropyBits(done
, "6", "2");
2178 it("Shows the number of bits of entropy for 1 character of base 10 with 3 bits", function(done
) {
2179 // 7 in base 10 is 111 in base 2, no leading zeros
2180 testEntropyBits(done
, "7", "3");
2182 it("Shows the number of bits of entropy for 1 character of base 10 with 4 bis", function(done
) {
2183 testEntropyBits(done
, "8", "4");
2185 it("Shows the number of bits of entropy for 1 character of hex", function(done
) {
2186 testEntropyBits(done
, "F", "4");
2188 it("Shows the number of bits of entropy for 2 characters of base 10", function(done
) {
2189 testEntropyBits(done
, "29", "6");
2191 it("Shows the number of bits of entropy for 2 characters of hex", function(done
) {
2192 testEntropyBits(done
, "0A", "8");
2194 it("Shows the number of bits of entropy for 2 characters of hex with 3 leading zeros", function(done
) {
2195 // hex is always multiple of 4 bits of entropy
2196 testEntropyBits(done
, "1A", "8");
2198 it("Shows the number of bits of entropy for 2 characters of hex with 2 leading zeros", function(done
) {
2199 testEntropyBits(done
, "2A", "8");
2201 it("Shows the number of bits of entropy for 2 characters of hex with 1 leading zero", function(done
) {
2202 testEntropyBits(done
, "4A", "8");
2204 it("Shows the number of bits of entropy for 2 characters of hex with no leading zeros", function(done
) {
2205 testEntropyBits(done
, "8A", "8");
2207 it("Shows the number of bits of entropy for 2 characters of hex starting with F", function(done
) {
2208 testEntropyBits(done
, "FA", "8");
2210 it("Shows the number of bits of entropy for 4 characters of hex with leading zeros", function(done
) {
2211 testEntropyBits(done
, "000A", "16");
2213 it("Shows the number of bits of entropy for 4 characters of base 6", function(done
) {
2214 testEntropyBits(done
, "5555", "11");
2216 it("Shows the number of bits of entropy for 4 characters of base 6 dice", function(done
) {
2217 // uses dice, so entropy is actually 0000 in base 6, which is 4 lots of
2218 // 2.58 bits, which is 10.32 bits (rounded down to 10 bits)
2219 testEntropyBits(done
, "6666", "10");
2221 it("Shows the number of bits of entropy for 4 charactes of base 10", function(done
) {
2222 // Uses base 10, which is 4 lots of 3.32 bits, which is 13.3 bits (rounded
2224 testEntropyBits(done
, "2227", "13");
2226 it("Shows the number of bits of entropy for 4 characters of hex with 2 leading zeros", function(done
) {
2227 testEntropyBits(done
, "222F", "16");
2229 it("Shows the number of bits of entropy for 4 characters of hex starting with F", function(done
) {
2230 testEntropyBits(done
, "FFFF", "16");
2232 it("Shows the number of bits of entropy for 10 characters of base 10", function(done
) {
2233 // 10 events at 3.32 bits per event
2234 testEntropyBits(done
, "0000101017", "33");
2236 it("Shows the number of bits of entropy for a full deck of cards", function(done
) {
2237 // cards are not replaced, so a full deck is not 52^52 entropy which is 296
2238 // bits, it's 52!, which is 225 bits
2239 testEntropyBits(done
, "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks", "225");
2242 it("Shows details about the entered entropy", function(done
) {
2243 testEntropyFeedback(done
,
2247 type: "hexadecimal",
2251 strength: "less than a second",
2255 it("Shows details about the entered entropy", function(done
) {
2256 testEntropyFeedback(done
,
2258 entropy: "AAAAAAAA",
2259 filtered: "AAAAAAAA",
2260 type: "hexadecimal",
2264 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2268 it("Shows details about the entered entropy", function(done
) {
2269 testEntropyFeedback(done
,
2271 entropy: "AAAAAAAA B",
2272 filtered: "AAAAAAAAB",
2273 type: "hexadecimal",
2277 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2281 it("Shows details about the entered entropy", function(done
) {
2282 testEntropyFeedback(done
,
2284 entropy: "AAAAAAAA BBBBBBBB",
2285 filtered: "AAAAAAAABBBBBBBB",
2286 type: "hexadecimal",
2290 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2294 it("Shows details about the entered entropy", function(done
) {
2295 testEntropyFeedback(done
,
2297 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC",
2298 filtered: "AAAAAAAABBBBBBBBCCCCCCCC",
2299 type: "hexadecimal",
2303 strength: "less than a second",
2307 it("Shows details about the entered entropy", function(done
) {
2308 testEntropyFeedback(done
,
2310 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDD",
2311 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDD",
2312 type: "hexadecimal",
2316 strength: "2 minutes",
2320 it("Shows details about the entered entropy", function(done
) {
2321 testEntropyFeedback(done
,
2323 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA",
2324 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDA",
2325 type: "hexadecimal",
2333 it("Shows details about the entered entropy", function(done
) {
2334 testEntropyFeedback(done
,
2336 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA EEEEEEEE",
2337 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDAEEEEEEEE",
2338 type: "hexadecimal",
2342 strength: "3 years",
2346 it("Shows details about the entered entropy", function(done
) {
2347 testEntropyFeedback(done
,
2349 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA EEEEEEEE FFFFFFFF",
2350 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDAEEEEEEEEFFFFFFFF",
2351 type: "hexadecimal",
2355 strength: "centuries",
2359 it("Shows details about the entered entropy", function(done
) {
2360 testEntropyFeedback(done
,
2367 strength: "less than a second",
2371 it("Shows details about the entered entropy", function(done
) {
2372 testEntropyFeedback(done
,
2374 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2375 type: "card (full deck)",
2379 strength: "centuries",
2383 it("Shows details about the entered entropy", function(done
) {
2384 testEntropyFeedback(done
,
2386 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks3d",
2387 type: "card (full deck, 1 duplicate: 3d)",
2391 strength: "centuries",
2395 it("Shows details about the entered entropy", function(done
) {
2396 testEntropyFeedback(done
,
2398 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqs3d4d",
2399 type: "card (2 duplicates: 3d 4d, 1 missing: KS)",
2403 strength: "centuries",
2407 it("Shows details about the entered entropy", function(done
) {
2408 testEntropyFeedback(done
,
2410 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqs3d4d5d6d",
2411 type: "card (4 duplicates: 3d 4d 5d..., 1 missing: KS)",
2415 strength: "centuries",
2419 it("Shows details about the entered entropy", function(done
) {
2420 testEntropyFeedback(done
,
2421 // Next test was throwing uncaught error in zxcvbn
2422 // Also tests 451 bits, ie Math.log2(52!)*2 = 225.58 * 2
2424 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsksac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2425 type: "card (full deck, 52 duplicates: ac 2c 3c...)",
2429 strength: "centuries",
2433 it("Shows details about the entered entropy", function(done
) {
2434 testEntropyFeedback(done
,
2435 // Case insensitivity to duplicate cards
2438 type: "card (1 duplicate: AS)",
2442 strength: "less than a second",
2446 it("Shows details about the entered entropy", function(done
) {
2447 testEntropyFeedback(done
,
2450 type: "card (1 duplicate: as)",
2454 strength: "less than a second",
2458 it("Shows details about the entered entropy", function(done
) {
2459 testEntropyFeedback(done
,
2460 // Missing cards are detected
2462 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2463 type: "card (1 missing: 9C)",
2467 strength: "centuries",
2471 it("Shows details about the entered entropy", function(done
) {
2472 testEntropyFeedback(done
,
2474 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2475 type: "card (2 missing: 9C 5D)",
2479 strength: "centuries",
2483 it("Shows details about the entered entropy", function(done
) {
2484 testEntropyFeedback(done
,
2486 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d7d8d9dtdjd kdah2h3h 5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2487 type: "card (4 missing: 9C 5D QD...)",
2491 strength: "centuries",
2495 it("Shows details about the entered entropy", function(done
) {
2496 testEntropyFeedback(done
,
2497 // More than six missing cards does not show message
2499 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d 8d9d jd kdah2h3h 5h6h7h8h9hthjhqhkh 2s3s4s5s6s7s8s9stsjsqsks",
2504 strength: "centuries",
2508 it("Shows details about the entered entropy", function(done
) {
2509 testEntropyFeedback(done
,
2510 // Multiple decks of cards increases bits per event
2515 bitsPerEvent: "4.34",
2519 it("Shows details about the entered entropy", function(done
) {
2520 testEntropyFeedback(done
,
2525 bitsPerEvent: "4.80",
2529 it("Shows details about the entered entropy", function(done
) {
2530 testEntropyFeedback(done
,
2535 bitsPerEvent: "5.01",
2539 it("Shows details about the entered entropy", function(done
) {
2540 testEntropyFeedback(done
,
2542 entropy: "3d3d3d3d",
2545 bitsPerEvent: "5.14",
2549 it("Shows details about the entered entropy", function(done
) {
2550 testEntropyFeedback(done
,
2552 entropy: "3d3d3d3d3d",
2555 bitsPerEvent: "5.22",
2559 it("Shows details about the entered entropy", function(done
) {
2560 testEntropyFeedback(done
,
2562 entropy: "3d3d3d3d3d3d",
2565 bitsPerEvent: "5.28",
2569 it("Shows details about the entered entropy", function(done
) {
2570 testEntropyFeedback(done
,
2572 entropy: "3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d",
2575 bitsPerEvent: "5.59",
2576 strength: 'less than a second - Repeats like "abcabcabc" are only slightly harder to guess than "abc"',
2581 // Entropy is truncated from the left
2582 it('Truncates entropy from the left', function(done
) {
2583 // Truncate from left means 0000 is removed from the start
2584 // which gives mnemonic 'avocado zoo zone'
2585 // not 1111 removed from the end
2586 // which gives the mnemonic 'abstract zoo zoo'
2587 var entropy
= "00000000 00000000 00000000 00000000";
2588 entropy
+= "11111111 11111111 11111111 1111"; // Missing last byte
2589 driver
.findElement(By
.css('.use-entropy'))
2591 driver
.findElement(By
.css('.entropy'))
2593 driver
.sleep(generateDelay
).then(function() {
2594 driver
.findElement(By
.css(".phrase"))
2595 .getAttribute("value").then(function(phrase
) {
2596 expect(phrase
).toBe("avocado zoo zone");
2602 // Very large entropy results in very long mnemonics
2603 it('Converts very long entropy to very long mnemonics', function(done
) {
2605 for (var i
=0; i
<33; i
++) {
2606 entropy
+= "AAAAAAAA"; // 3 words * 33 iterations = 99 words
2608 driver
.findElement(By
.css('.use-entropy'))
2610 driver
.findElement(By
.css('.entropy'))
2612 driver
.sleep(generateDelay
).then(function() {
2613 driver
.findElement(By
.css(".phrase"))
2614 .getAttribute("value").then(function(phrase
) {
2615 var wordCount
= phrase
.split(/\s+/g).length
;
2616 expect(wordCount
).toBe(99);
2622 // Is compatible with bip32jp entropy
2623 // https://bip32jp.github.io/english/index.html
2625 // Is incompatible with:
2627 it('Is compatible with bip32jp.github.io', function(done
) {
2628 var entropy
= "543210543210543210543210543210543210543210543210543210543210543210543210543210543210543210543210543";
2629 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";
2630 driver
.findElement(By
.css('.use-entropy'))
2632 driver
.findElement(By
.css('.entropy'))
2634 driver
.sleep(generateDelay
).then(function() {
2635 driver
.findElement(By
.css(".phrase"))
2636 .getAttribute("value").then(function(phrase
) {
2637 expect(phrase
).toBe(expectedPhrase
);
2643 // Blank entropy does not generate mnemonic or addresses
2644 it('Does not generate mnemonic for blank entropy', function(done
) {
2645 driver
.findElement(By
.css('.use-entropy'))
2647 driver
.findElement(By
.css('.entropy'))
2649 // check there is no mnemonic
2650 driver
.sleep(generateDelay
).then(function() {
2651 driver
.findElement(By
.css(".phrase"))
2652 .getAttribute("value").then(function(phrase
) {
2653 expect(phrase
).toBe("");
2654 // check there is no mnemonic
2655 driver
.findElements(By
.css(".address"))
2656 .then(function(addresses
) {
2657 expect(addresses
.length
).toBe(0);
2658 // Check the feedback says 'blank entropy'
2659 driver
.findElement(By
.css(".feedback"))
2661 .then(function(feedbackText
) {
2662 expect(feedbackText
).toBe("Blank entropy");
2670 // Mnemonic length can be selected even for weak entropy
2671 it('Allows selection of mnemonic length even for weak entropy', function(done
) {
2672 driver
.findElement(By
.css('.use-entropy'))
2674 driver
.executeScript(function() {
2675 $(".mnemonic-length").val("18").trigger("change");
2677 driver
.findElement(By
.css('.entropy'))
2678 .sendKeys("012345");
2679 driver
.sleep(generateDelay
).then(function() {
2680 driver
.findElement(By
.css(".phrase"))
2681 .getAttribute("value").then(function(phrase
) {
2682 var wordCount
= phrase
.split(/\s+/g).length
;
2683 expect(wordCount
).toBe(18);
2690 // https://github.com/iancoleman/bip39/issues/33
2691 // Final cards should contribute entropy
2692 it('Uses as much entropy as possible for the mnemonic', function(done
) {
2693 driver
.findElement(By
.css('.use-entropy'))
2695 driver
.findElement(By
.css('.entropy'))
2696 .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");
2697 driver
.sleep(generateDelay
).then(function() {
2699 driver
.findElement(By
.css(".phrase"))
2700 .getAttribute("value").then(function(originalPhrase
) {
2701 // Set the last 12 cards to be AS
2702 driver
.findElement(By
.css('.entropy'))
2704 driver
.findElement(By
.css('.entropy'))
2705 .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");
2706 driver
.sleep(generateDelay
).then(function() {
2708 driver
.findElement(By
.css(".phrase"))
2709 .getAttribute("value").then(function(newPhrase
) {
2710 expect(originalPhrase
).not
.toEqual(newPhrase
);
2719 // https://github.com/iancoleman/bip39/issues/35
2721 // TODO this doesn't work in selenium with firefox
2722 // see https://stackoverflow.com/q/40360223
2723 it('Shows a qr code on hover for the phrase', function(done
) {
2724 if (browser
== "firefox") {
2725 pending("Selenium + Firefox bug for mouseMove, see https://stackoverflow.com/q/40360223");
2727 // generate a random mnemonic
2728 var generateEl
= driver
.findElement(By
.css('.generate'));
2730 // toggle qr to show (hidden by default)
2731 var phraseEl
= driver
.findElement(By
.css(".phrase"));
2733 var rootKeyEl
= driver
.findElement(By
.css(".root-key"));
2734 driver
.sleep(generateDelay
).then(function() {
2735 // hover over the root key
2736 driver
.actions().mouseMove(rootKeyEl
).perform().then(function() {
2737 // check the qr code shows
2738 driver
.executeScript(function() {
2739 return $(".qr-container").find("canvas").length
> 0;
2741 .then(function(qrShowing
) {
2742 expect(qrShowing
).toBe(true);
2743 // hover away from the phrase
2744 driver
.actions().mouseMove(generateEl
).perform().then(function() {;
2745 // check the qr code hides
2746 driver
.executeScript(function() {
2747 return $(".qr-container").find("canvas").length
== 0;
2749 .then(function(qrHidden
) {
2750 expect(qrHidden
).toBe(true);
2759 // BIP44 account extendend private key is shown
2760 // github issue 37 - compatibility with electrum
2761 it('Shows the bip44 account extended private key', function(done
) {
2762 driver
.findElement(By
.css(".phrase"))
2763 .sendKeys("abandon abandon ability");
2764 driver
.sleep(generateDelay
).then(function() {
2765 driver
.findElement(By
.css("#bip44 .account-xprv"))
2766 .getAttribute("value")
2767 .then(function(xprv
) {
2768 expect(xprv
).toBe("xprv9yzrnt4zWVJUr1k2VxSPy9ettKz5PpeDMgaVG7UKedhqnw1tDkxP2UyYNhuNSumk2sLE5ctwKZs9vwjsq3e1vo9egCK6CzP87H2cVYXpfwQ");
2774 // BIP44 account extendend public key is shown
2775 // github issue 37 - compatibility with electrum
2776 it('Shows the bip44 account extended public key', function(done
) {
2777 driver
.findElement(By
.css(".phrase"))
2778 .sendKeys("abandon abandon ability");
2779 driver
.sleep(generateDelay
).then(function() {
2780 driver
.findElement(By
.css("#bip44 .account-xpub"))
2781 .getAttribute("value")
2782 .then(function(xprv
) {
2783 expect(xprv
).toBe("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2790 // BIP32 root key can be set as an xpub
2791 it('Generates addresses from xpub as bip32 root key', function(done
) {
2792 driver
.findElement(By
.css('#bip32-tab a'))
2794 // set xpub for account 0 of bip44 for 'abandon abandon ability'
2795 driver
.findElement(By
.css("#root-key"))
2796 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2797 driver
.sleep(generateDelay
).then(function() {
2798 // check the addresses are generated
2799 getFirstAddress(function(address
) {
2800 expect(address
).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug");
2801 // check the xprv key is not set
2802 driver
.findElement(By
.css(".extended-priv-key"))
2803 .getAttribute("value")
2804 .then(function(xprv
) {
2805 expect(xprv
).toBe("NA");
2806 // check the private key is not set
2807 driver
.findElements(By
.css(".privkey"))
2808 .then(function(els
) {
2811 .then(function(privkey
) {
2812 expect(xprv
).toBe("NA");
2822 // xpub for bip32 root key will not work with hardened derivation paths
2823 it('Shows error for hardened derivation paths with xpub root key', function(done
) {
2824 // set xpub for account 0 of bip44 for 'abandon abandon ability'
2825 driver
.findElement(By
.css("#root-key"))
2826 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2827 driver
.sleep(feedbackDelay
).then(function() {
2828 // Check feedback is correct
2829 driver
.findElement(By
.css('.feedback'))
2831 .then(function(feedback
) {
2832 var msg
= "Hardened derivation path is invalid with xpub key";
2833 expect(feedback
).toBe(msg
);
2834 // Check no addresses are shown
2835 driver
.findElements(By
.css('.addresses tr'))
2836 .then(function(rows
) {
2837 expect(rows
.length
).toBe(0);
2845 // no root key shows feedback
2846 it('Shows feedback for no root key', function(done
) {
2847 // set xpub for account 0 of bip44 for 'abandon abandon ability'
2848 driver
.findElement(By
.css('#bip32-tab a'))
2850 driver
.sleep(feedbackDelay
).then(function() {
2851 // Check feedback is correct
2852 driver
.findElement(By
.css('.feedback'))
2854 .then(function(feedback
) {
2855 expect(feedback
).toBe("Invalid root key");
2862 // display error switching tabs while addresses are generating
2863 it('Can change details while old addresses are still being generated', function(done
) {
2864 // Set to generate 199 more addresses.
2865 // This will take a long time allowing a new set of addresses to be
2866 // generated midway through this lot.
2867 // The newly generated addresses should not include any from the old set.
2868 // Any more than 199 will show an alert which needs to be accepted.
2869 driver
.findElement(By
.css('.rows-to-add'))
2871 driver
.findElement(By
.css('.rows-to-add'))
2874 driver
.findElement(By
.css('.phrase'))
2875 .sendKeys("abandon abandon ability");
2876 driver
.sleep(generateDelay
).then(function() {
2877 // change tabs which should cancel the previous generating
2878 driver
.findElement(By
.css('.rows-to-add'))
2880 driver
.findElement(By
.css('.rows-to-add'))
2882 driver
.findElement(By
.css('#bip32-tab a'))
2884 driver
.sleep(generateDelay
).then(function() {
2885 driver
.findElements(By
.css('.index'))
2886 .then(function(els
) {
2887 // check the derivation paths have the right quantity
2888 expect(els
.length
).toBe(20);
2889 // check the derivation paths are in order
2890 testRowsAreInCorrectOrder(done
);
2894 }, generateDelay
+ 5000);
2897 // padding for binary should give length with multiple of 256
2898 // hashed entropy 1111 is length 252, so requires 4 leading zeros
2899 // prior to issue 49 it would only generate 2 leading zeros, ie missing 2
2900 it('Pads hashed entropy with leading zeros', function(done
) {
2901 driver
.findElement(By
.css('.use-entropy'))
2903 driver
.executeScript(function() {
2904 $(".mnemonic-length").val("15").trigger("change");
2906 driver
.findElement(By
.css('.entropy'))
2908 driver
.sleep(generateDelay
).then(function() {
2909 driver
.findElement(By
.css('.phrase'))
2910 .getAttribute("value")
2911 .then(function(phrase
) {
2912 expect(phrase
).toBe("avocado valid quantum cross link predict excuse edit street able flame large galaxy ginger nuclear");
2918 // Github pull request 55
2919 // https://github.com/iancoleman/bip39/pull/55
2921 it('Can set the derivation path on bip32 tab for bitcoincore', function(done
) {
2922 testClientSelect(done
, {
2924 bip32path: "m/0'/0'",
2925 useHardenedAddresses: "true",
2928 it('Can set the derivation path on bip32 tab for multibit', function(done
) {
2929 testClientSelect(done
, {
2931 bip32path: "m/0'/0",
2932 useHardenedAddresses: null,
2935 it('Can set the derivation path on bip32 tab for coinomi/ledger', function(done
) {
2936 testClientSelect(done
, {
2938 bip32path: "m/44'/0'/0'",
2939 useHardenedAddresses: null,
2944 // https://github.com/iancoleman/bip39/issues/58
2945 // bip32 derivation is correct, does not drop leading zeros
2947 // https://medium.com/@alexberegszaszi/why-do-my-bip32-wallets-disagree-6f3254cc5846
2948 it('Retains leading zeros for bip32 derivation', function(done
) {
2949 driver
.findElement(By
.css(".phrase"))
2950 .sendKeys("fruit wave dwarf banana earth journey tattoo true farm silk olive fence");
2951 driver
.findElement(By
.css(".passphrase"))
2952 .sendKeys("banana");
2953 driver
.sleep(generateDelay
).then(function() {
2954 getFirstAddress(function(address
) {
2955 // Note that bitcore generates an incorrect address
2956 // 13EuKhffWkBE2KUwcbkbELZb1MpzbimJ3Y
2957 // see the medium.com link above for more details
2958 expect(address
).toBe("17rxURoF96VhmkcEGCj5LNQkmN9HVhWb7F");
2965 // Japanese mnemonics generate incorrect bip32 seed
2966 // BIP39 seed is set from phrase
2967 it('Generates correct seed for Japanese mnemonics', function(done
) {
2968 driver
.findElement(By
.css(".phrase"))
2969 .sendKeys("あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あおぞら");
2970 driver
.findElement(By
.css(".passphrase"))
2971 .sendKeys("メートルガバヴァぱばぐゞちぢ十人十色");
2972 driver
.sleep(generateDelay
).then(function() {
2973 driver
.findElement(By
.css(".seed"))
2974 .getAttribute("value")
2975 .then(function(seed
) {
2976 expect(seed
).toBe("a262d6fb6122ecf45be09c50492b31f92e9beb7d9a845987a02cefda57a15f9c467a17872029a9e92299b5cbdf306e3a0ee620245cbd508959b6cb7ca637bd55");
2982 // BIP49 official test vectors
2983 // https://github.com/bitcoin/bips/blob/master/bip-0049.mediawiki#test-vectors
2984 it('Generates BIP49 addresses matching the official test vectors', function(done
) {
2985 driver
.findElement(By
.css('#bip49-tab a'))
2987 selectNetwork("BTC - Bitcoin Testnet");
2988 driver
.findElement(By
.css(".phrase"))
2989 .sendKeys("abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about");
2990 driver
.sleep(generateDelay
).then(function() {
2991 getFirstAddress(function(address
) {
2992 expect(address
).toBe("2Mww8dCYPUpKHofjgcXcBCEGmniw9CoaiD2");
2998 // BIP49 derivation path is shown
2999 it('Shows the bip49 derivation path', function(done
) {
3000 driver
.findElement(By
.css('#bip49-tab a'))
3002 driver
.findElement(By
.css(".phrase"))
3003 .sendKeys("abandon abandon ability");
3004 driver
.sleep(generateDelay
).then(function() {
3005 driver
.findElement(By
.css('#bip49 .path'))
3006 .getAttribute("value")
3007 .then(function(path
) {
3008 expect(path
).toBe("m/49'/0'/0'/0");
3014 // BIP49 extended private key is shown
3015 it('Shows the bip49 extended private key', function(done
) {
3016 driver
.findElement(By
.css('#bip49-tab a'))
3018 driver
.findElement(By
.css(".phrase"))
3019 .sendKeys("abandon abandon ability");
3020 driver
.sleep(generateDelay
).then(function() {
3021 driver
.findElement(By
.css('.extended-priv-key'))
3022 .getAttribute("value")
3023 .then(function(xprv
) {
3024 expect(xprv
).toBe("yprvALYB4DYRG6CzzVgzQZwwqjAA2wjBGC3iEd7KYYScpoDdmf75qMRWZWxoFcRXBJjgEXdFqJ9vDRGRLJQsrL22Su5jMbNFeM9vetaGVqy9Qy2");
3030 // BIP49 extended public key is shown
3031 it('Shows the bip49 extended public key', function(done
) {
3032 driver
.findElement(By
.css('#bip49-tab a'))
3034 driver
.findElement(By
.css(".phrase"))
3035 .sendKeys("abandon abandon ability");
3036 driver
.sleep(generateDelay
).then(function() {
3037 driver
.findElement(By
.css('.extended-pub-key'))
3038 .getAttribute("value")
3039 .then(function(xprv
) {
3040 expect(xprv
).toBe("ypub6ZXXTj5K6TmJCymTWbUxCs6tayZffemZbr2vLvrEP8kceTSENtjm7KHH6thvAKxVar9fGe8rgsPEX369zURLZ68b4f7Vexz7RuXsjQ69YDt");
3046 // BIP49 account field changes address list
3047 it('Can set the bip49 account field', function(done
) {
3048 driver
.findElement(By
.css('#bip49-tab a'))
3050 driver
.findElement(By
.css('#bip49 .account'))
3052 driver
.findElement(By
.css('#bip49 .account'))
3054 driver
.findElement(By
.css(".phrase"))
3055 .sendKeys("abandon abandon ability");
3056 driver
.sleep(generateDelay
).then(function() {
3057 getFirstAddress(function(address
) {
3058 expect(address
).toBe("381wg1GGN4rP88rNC9v7QWsiww63yLVPsn");
3064 // BIP49 change field changes address list
3065 it('Can set the bip49 change field', function(done
) {
3066 driver
.findElement(By
.css('#bip49-tab a'))
3068 driver
.findElement(By
.css('#bip49 .change'))
3070 driver
.findElement(By
.css('#bip49 .change'))
3072 driver
.findElement(By
.css(".phrase"))
3073 .sendKeys("abandon abandon ability");
3074 driver
.sleep(generateDelay
).then(function() {
3075 getFirstAddress(function(address
) {
3076 expect(address
).toBe("3PEM7MiKed5konBoN66PQhK8r3hjGhy9dT");
3082 // BIP49 account extendend private key is shown
3083 it('Shows the bip49 account extended private key', function(done
) {
3084 driver
.findElement(By
.css('#bip49-tab a'))
3086 driver
.findElement(By
.css(".phrase"))
3087 .sendKeys("abandon abandon ability");
3088 driver
.sleep(generateDelay
).then(function() {
3089 driver
.findElement(By
.css('#bip49 .account-xprv'))
3090 .getAttribute("value")
3091 .then(function(xprv
) {
3092 expect(xprv
).toBe("yprvAHtB1M5Wp675aLzFy9TJYK2mSsLkg6mcBRh5DZTR7L4EnYSmYPaL63KFA4ycg1PngW5LfkmejxzosCs17TKZMpRFKc3z5SJar6QAKaFcaZL");
3098 // BIP49 account extendend public key is shown
3099 it('Shows the bip49 account extended public key', function(done
) {
3100 driver
.findElement(By
.css('#bip49-tab a'))
3102 driver
.findElement(By
.css(".phrase"))
3103 .sendKeys("abandon abandon ability");
3104 driver
.sleep(generateDelay
).then(function() {
3105 driver
.findElement(By
.css('#bip49 .account-xpub'))
3106 .getAttribute("value")
3107 .then(function(xprv
) {
3108 expect(xprv
).toBe("ypub6WsXQrcQeTfNnq4j5AzJuSyVzuBF5ZVTYecg1ws2ffbDfLmv5vtadqdj1NgR6C6gufMpMfJpHxvb6JEQKvETVNWCRanNedfJtnTchZiJtsL");
3114 // Test selecting coin where bip49 is unavailable (eg CLAM)
3115 it('Shows an error on bip49 tab for coins without bip49', function(done
) {
3116 driver
.findElement(By
.css('#bip49-tab a'))
3118 driver
.findElement(By
.css(".phrase"))
3119 .sendKeys("abandon abandon ability");
3120 driver
.sleep(generateDelay
).then(function() {
3121 selectNetwork("CLAM - Clams");
3122 // bip49 available is hidden
3123 driver
.findElement(By
.css('#bip49 .available'))
3124 .getAttribute("class")
3125 .then(function(classes
) {
3126 expect(classes
).toContain("hidden");
3127 // bip49 unavailable is shown
3128 driver
.findElement(By
.css('#bip49 .unavailable'))
3129 .getAttribute("class")
3130 .then(function(classes
) {
3131 expect(classes
).not
.toContain("hidden");
3132 // check there are no addresses shown
3133 driver
.findElements(By
.css('.addresses tr'))
3134 .then(function(rows
) {
3135 expect(rows
.length
).toBe(0);
3136 // check the derived private key is blank
3137 driver
.findElement(By
.css('.extended-priv-key'))
3138 .getAttribute("value")
3139 .then(function(xprv
) {
3140 expect(xprv
).toBe('');
3141 // check the derived public key is blank
3142 driver
.findElement(By
.css('.extended-pub-key'))
3143 .getAttribute("value")
3144 .then(function(xpub
) {
3145 expect(xpub
).toBe('');
3156 // Cleared mnemonic and root key still allows addresses to be generated
3157 // https://github.com/iancoleman/bip39/issues/43
3158 it('Clears old root keys from memory when mnemonic is cleared', function(done
) {
3160 driver
.findElement(By
.css(".phrase"))
3161 .sendKeys("abandon abandon ability");
3162 driver
.sleep(generateDelay
).then(function() {
3163 // clear the mnemonic and root key
3164 // using selenium .clear() doesn't seem to trigger the 'input' event
3165 // so clear it using keys instead
3166 driver
.findElement(By
.css('.phrase'))
3167 .sendKeys(Key
.CONTROL
,"a");
3168 driver
.findElement(By
.css('.phrase'))
3169 .sendKeys(Key
.DELETE
);
3170 driver
.findElement(By
.css('.root-key'))
3171 .sendKeys(Key
.CONTROL
,"a");
3172 driver
.findElement(By
.css('.root-key'))
3173 .sendKeys(Key
.DELETE
);
3174 driver
.sleep(generateDelay
).then(function() {
3175 // try to generate more addresses
3176 driver
.findElement(By
.css('.more'))
3178 driver
.sleep(generateDelay
).then(function() {
3179 driver
.findElements(By
.css(".addresses tr"))
3180 .then(function(els
) {
3181 // check there are no addresses shown
3182 expect(els
.length
).toBe(0);
3191 // error trying to generate addresses from xpub with hardened derivation
3192 it('Shows error for hardened addresses with xpub root key', function(done
) {
3193 driver
.findElement(By
.css('#bip32-tab a'))
3195 driver
.executeScript(function() {
3196 $(".hardened-addresses").prop("checked", true);
3198 // set xpub for account 0 of bip44 for 'abandon abandon ability'
3199 driver
.findElement(By
.css("#root-key"))
3200 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
3201 driver
.sleep(feedbackDelay
).then(function() {
3202 // Check feedback is correct
3203 driver
.findElement(By
.css('.feedback'))
3205 .then(function(feedback
) {
3206 var msg
= "Hardened derivation path is invalid with xpub key";
3207 expect(feedback
).toBe(msg
);
3213 // Litecoin uses ltub by default, and can optionally be set to xprv
3215 // https://github.com/iancoleman/bip39/issues/96
3216 // Issue with extended keys on Litecoin
3217 it('Uses ltub by default for litecoin, but can be set to xprv', function(done
) {
3218 driver
.findElement(By
.css('.phrase'))
3219 .sendKeys("abandon abandon ability");
3220 selectNetwork("LTC - Litecoin");
3221 driver
.sleep(generateDelay
).then(function() {
3222 // check the extended key is generated correctly
3223 driver
.findElement(By
.css('.root-key'))
3224 .getAttribute("value")
3225 .then(function(rootKey
) {
3226 expect(rootKey
).toBe("Ltpv71G8qDifUiNesiPqf6h5V6eQ8ic77oxQiYtawiACjBEx3sTXNR2HGDGnHETYxESjqkMLFBkKhWVq67ey1B2MKQXannUqNy1RZVHbmrEjnEU");
3227 // set litecoin to use ltub
3228 driver
.executeScript(function() {
3229 $(".litecoin-use-ltub").prop("checked", false);
3230 $(".litecoin-use-ltub").trigger("change");
3232 driver
.sleep(generateDelay
).then(function() {
3233 driver
.findElement(By
.css('.root-key'))
3234 .getAttribute("value")
3235 .then(function(rootKey
) {
3236 expect(rootKey
).toBe("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
3245 // https://github.com/iancoleman/bip39/issues/99#issuecomment-327094159
3246 // "warn me emphatically when they have detected invalid input" to the entropy field
3247 // A warning is shown when entropy is filtered and discarded
3248 it('Warns when entropy is filtered and discarded', function(done
) {
3249 driver
.findElement(By
.css('.use-entropy'))
3251 // set entropy to have no filtered content
3252 driver
.findElement(By
.css('.entropy'))
3253 .sendKeys("00000000 00000000 00000000 00000000");
3254 driver
.sleep(generateDelay
).then(function() {
3255 // check the filter warning does not show
3256 driver
.findElement(By
.css('.entropy-container .filter-warning'))
3257 .getAttribute("class")
3258 .then(function(classes
) {
3259 expect(classes
).toContain("hidden");
3260 // set entropy to have some filtered content
3261 driver
.findElement(By
.css('.entropy'))
3262 .sendKeys("10000000 zxcvbn 00000000 00000000 00000000");
3263 driver
.sleep(entropyFeedbackDelay
).then(function() {
3264 // check the filter warning shows
3265 driver
.findElement(By
.css('.entropy-container .filter-warning'))
3266 .getAttribute("class")
3267 .then(function(classes
) {
3268 expect(classes
).not
.toContain("hidden");
3276 // Bitcoin Cash address can be set to use cashaddr format
3277 it('Can use cashaddr format for bitcoin cash addresses', function(done
) {
3278 driver
.executeScript(function() {
3279 $(".use-bch-cashaddr-addresses").prop("checked", true);
3281 driver
.findElement(By
.css('.phrase'))
3282 .sendKeys("abandon abandon ability");
3283 selectNetwork("BCH - Bitcoin Cash");
3284 driver
.sleep(generateDelay
).then(function() {
3285 getFirstAddress(function(address
) {
3286 expect(address
).toBe("bitcoincash:qzlquk7w4hkudxypl4fgv8x279r754dkvur7jpcsps");
3292 // Bitcoin Cash address can be set to use bitpay format
3293 it('Can use bitpay format for bitcoin cash addresses', function(done
) {
3294 driver
.executeScript(function() {
3295 $(".use-bch-bitpay-addresses").prop("checked", true);
3297 driver
.findElement(By
.css('.phrase'))
3298 .sendKeys("abandon abandon ability");
3299 selectNetwork("BCH - Bitcoin Cash");
3300 driver
.sleep(generateDelay
).then(function() {
3301 getFirstAddress(function(address
) {
3302 expect(address
).toBe("CZnpA9HPmvhuhLLPWJP8rNDpLUYXy1LXFk");
3308 // Bitcoin Cash address can be set to use legacy format
3309 it('Can use legacy format for bitcoin cash addresses', function(done
) {
3310 driver
.executeScript(function() {
3311 $(".use-bch-legacy-addresses").prop("checked", true);
3313 driver
.findElement(By
.css('.phrase'))
3314 .sendKeys("abandon abandon ability");
3315 selectNetwork("BCH - Bitcoin Cash");
3316 driver
.sleep(generateDelay
).then(function() {
3317 getFirstAddress(function(address
) {
3318 expect(address
).toBe("1JKvb6wKtsjNoCRxpZ4DGrbniML7z5U16A");
3324 // End of tests ported from old suit, so no more comments above each test now
3326 it('Can generate more addresses from a custom index', function(done
) {
3327 var expectedIndexes
= [
3328 0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,
3329 40,41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59
3331 driver
.findElement(By
.css('.phrase'))
3332 .sendKeys("abandon abandon ability");
3333 driver
.sleep(generateDelay
).then(function() {
3334 // Set start of next lot of rows to be from index 40
3335 // which means indexes 20-39 will not be in the table.
3336 driver
.findElement(By
.css('.more-rows-start-index'))
3338 driver
.findElement(By
.css('.more'))
3340 driver
.sleep(generateDelay
).then(function() {
3341 // Check actual indexes in the table match the expected pattern
3342 driver
.findElements(By
.css(".index"))
3343 .then(function(els
) {
3344 expect(els
.length
).toBe(expectedIndexes
.length
);
3345 var testRowAtIndex = function(i
) {
3346 if (i
>= expectedIndexes
.length
) {
3351 .then(function(actualPath
) {
3352 var noHardened
= actualPath
.replace(/'/g, "");
3353 var pathBits = noHardened.split("/")
3354 var lastBit = pathBits[pathBits.length-1];
3355 var actualIndex = parseInt(lastBit);
3356 var expectedIndex = expectedIndexes[i];
3357 expect(actualIndex).toBe(expectedIndex);
3358 testRowAtIndex(i+1);
3368 it('Can generate BIP141 addresses
with P2WPKH
-in-P2SH semanitcs
', function(done) {
3369 // Sourced from BIP49 official test specs
3370 driver.findElement(By.css('#bip141
-tab a
'))
3372 driver.findElement(By.css('.bip141
-path
'))
3374 driver.findElement(By.css('.bip141
-path
'))
3375 .sendKeys("m/49'/1'/0'/0");
3376 selectNetwork("BTC
- Bitcoin Testnet
");
3377 driver.findElement(By.css(".phrase
"))
3378 .sendKeys("abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about
");
3379 driver.sleep(generateDelay).then(function() {
3380 getFirstAddress(function(address) {
3381 expect(address).toBe("2Mww8dCYPUpKHofjgcXcBCEGmniw9CoaiD2
");
3387 it('Can generate BIP141 addresses with P2WPKH semanitcs', function(done) {
3388 // This result tested against bitcoinjs-lib test spec for segwit address
3389 // using the first private key of this mnemonic and default path m/0
3390 // https://github.com/bitcoinjs/bitcoinjs-lib/blob/9c8503cab0c6c30a95127042703bc18e8d28c76d/test/integration/addresses.js#L50
3391 // so whilst not directly comparable, substituting the private key produces
3392 // identical results between this tool and the bitcoinjs-lib test.
3393 // Private key generated is:
3394 // L3L8Nu9whawPBNLGtFqDhKut9DKKfG3CQoysupT7BimqVCZsLFNP
3395 driver.findElement(By.css('#bip141-tab a'))
3398 driver.executeScript(function() {
3399 $(".bip141
-semantics option
[selected
]").removeAttr("selected
");
3400 $(".bip141
-semantics option
").filter(function(i,e) {
3401 return $(e).html() == "P2WPKH
";
3402 }).prop("selected
", true);
3403 $(".bip141
-semantics
").trigger("change
");
3405 driver.findElement(By.css(".phrase
"))
3406 .sendKeys("abandon abandon ability
");
3407 driver.sleep(generateDelay).then(function() {
3408 getFirstAddress(function(address) {
3409 expect(address).toBe("bc1qfwu6a5a3evygrk8zvdxxvz4547lmpyx5vsfxe9
");
3415 it('Shows the entropy used by the PRNG when clicking generate', function(done) {
3416 driver.findElement(By.css('.generate')).click();
3417 driver.sleep(generateDelay).then(function() {
3418 driver.findElement(By.css('.entropy'))
3419 .getAttribute("value
")
3420 .then(function(entropy) {
3421 expect(entropy).not.toBe("");
3427 it('Shows the index of each word in the mnemonic', function(done) {
3428 driver.findElement(By.css('.phrase'))
3429 .sendKeys("abandon abandon ability
");
3430 driver.sleep(generateDelay).then(function() {
3431 driver.findElement(By.css('.use-entropy'))
3433 driver.findElement(By.css('.word-indexes'))
3435 .then(function(indexes) {
3436 expect(indexes).toBe("0, 0, 1");
3442 it('Shows the derivation path for bip84 tab', function(done) {
3443 driver.findElement(By.css('#bip84-tab a'))
3445 driver.findElement(By.css('.phrase'))
3446 .sendKeys('abandon abandon ability');
3447 driver.sleep(generateDelay).then(function() {
3448 driver.findElement(By.css('#bip84 .path'))
3449 .getAttribute("value
")
3450 .then(function(path) {
3451 expect(path).toBe("m
/84'/0'/0'/0");
3457 it('Shows the extended private key for bip84 tab', function(done) {
3458 driver.findElement(By.css('#bip84-tab a'))
3460 driver.findElement(By.css('.phrase'))
3461 .sendKeys('abandon abandon ability');
3462 driver.sleep(generateDelay).then(function() {
3463 driver.findElement(By.css('.extended-priv-key'))
3464 .getAttribute("value
")
3465 .then(function(path) {
3466 expect(path).toBe("zprvAev3RKrZ3QVKiUFCfdeMRen1BPDJgdNt1XpxiDy8acSs4kkAGTCvq7HeRYRNNpo8EtEjCFQBWavJwtCUR29y4TUCH4X5RXMcyq48uN8y9BP
");
3472 it('Shows the extended public key for bip84 tab', function(done) {
3473 driver.findElement(By.css('#bip84-tab a'))
3475 driver.findElement(By.css('.phrase'))
3476 .sendKeys('abandon abandon ability');
3477 driver.sleep(generateDelay).then(function() {
3478 driver.findElement(By.css('.extended-pub-key'))
3479 .getAttribute("value
")
3480 .then(function(path) {
3481 expect(path).toBe("zpub6suPpqPSsn3cvxKfmfBMnnijjR3o666jNkkZWcNk8wyqwZ5JozXBNuc8Gs7DB3uLwTDvGVTspVEAUQcEjKF3pZHgywVbubdTqbXTUg7usyx
");
3487 it('Changes the address list if bip84 account is changed', function(done) {
3488 driver.findElement(By.css('#bip84-tab a'))
3490 driver.findElement(By.css('#bip84 .account'))
3492 driver.findElement(By.css('.phrase'))
3493 .sendKeys('abandon abandon ability');
3494 driver.sleep(generateDelay).then(function() {
3495 getFirstAddress(function(address) {
3496 expect(address).toBe("bc1qp7vv669t2fy965jdzvqwrraana89ctd5ewc662
");
3502 it('Changes the address list if bip84 change is changed', function(done) {
3503 driver.findElement(By.css('#bip84-tab a'))
3505 driver.findElement(By.css('#bip84 .change'))
3507 driver.findElement(By.css('.phrase'))
3508 .sendKeys('abandon abandon ability');
3509 driver.sleep(generateDelay).then(function() {
3510 getFirstAddress(function(address) {
3511 expect(address).toBe("bc1qr39vj6rh06ff05m53uxq8uazehwhccswylhrs2
");
3517 it('Passes the official BIP84 test spec for rootpriv', function(done) {
3518 driver.findElement(By.css('#bip84-tab a'))
3520 driver.findElement(By.css('.phrase'))
3521 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3522 driver.sleep(generateDelay).then(function() {
3523 driver.findElement(By.css(".root
-key
"))
3524 .getAttribute("value
")
3525 .then(function(rootKey) {
3526 expect(rootKey).toBe("zprvAWgYBBk7JR8Gjrh4UJQ2uJdG1r3WNRRfURiABBE3RvMXYSrRJL62XuezvGdPvG6GFBZduosCc1YP5wixPox7zhZLfiUm8aunE96BBa4Kei5
");
3532 it('Passes the official BIP84 test spec for account 0 xprv', function(done) {
3533 driver.findElement(By.css('#bip84-tab a'))
3535 driver.findElement(By.css('.phrase'))
3536 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3537 driver.sleep(generateDelay).then(function() {
3538 driver.findElement(By.css("#bip84
.account
-xprv
"))
3539 .getAttribute("value
")
3540 .then(function(rootKey) {
3541 expect(rootKey).toBe("zprvAdG4iTXWBoARxkkzNpNh8r6Qag3irQB8PzEMkAFeTRXxHpbF9z4QgEvBRmfvqWvGp42t42nvgGpNgYSJA9iefm1yYNZKEm7z6qUWCroSQnE
");
3547 it('Passes the official BIP84 test spec for account 0 xpub', function(done) {
3548 driver.findElement(By.css('#bip84-tab a'))
3550 driver.findElement(By.css('.phrase'))
3551 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3552 driver.sleep(generateDelay).then(function() {
3553 driver.findElement(By.css("#bip84
.account
-xpub
"))
3554 .getAttribute("value
")
3555 .then(function(rootKey) {
3556 expect(rootKey).toBe("zpub6rFR7y4Q2AijBEqTUquhVz398htDFrtymD9xYYfG1m4wAcvPhXNfE3EfH1r1ADqtfSdVCToUG868RvUUkgDKf31mGDtKsAYz2oz2AGutZYs
");
3562 it('Passes the official BIP84 test spec for account 0 first address', function(done) {
3563 driver.findElement(By.css('#bip84-tab a'))
3565 driver.findElement(By.css('.phrase'))
3566 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3567 driver.sleep(generateDelay).then(function() {
3568 getFirstAddress(function(address) {
3569 expect(address).toBe("bc1qcr8te4kr609gcawutmrza0j4xv80jy8z306fyu
");
3575 it('Passes the official BIP84 test spec for account 0 first change address', function(done) {
3576 driver.findElement(By.css('#bip84-tab a'))
3578 driver.findElement(By.css('.phrase'))
3579 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3580 driver.findElement(By.css('#bip84 .change'))
3582 driver.sleep(generateDelay).then(function() {
3583 getFirstAddress(function(address) {
3584 expect(address).toBe("bc1q8c6fshw2dlwun7ekn9qwf37cu2rn755upcp6el
");
3590 it('Can display the table as csv', function(done) {
3591 var headings = "path
,address
,public key
,private key
";
3592 var row1 = "m
/44'/0'/0'/0/0,1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug
,033f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3
,L26cVSpWFkJ6aQkPkKmTzLqTdLJ923e6CzrVh9cmx21QHsoUmrEE
";
3593 var row20 = "m
/44'/0'/0'/0/19,1KhBy28XLAciXnnRvm71PvQJaETyrxGV55
,02b4b3e396434d8cdd20c03ac4aaa07387784d5d867b75987f516f5705ee68cb3a
,L4GrDrjReMsCAu5DkLXn79jSb95qR7Zfx7eshybCQZ1qL32MXJab
";
3594 driver.findElement(By.css('.phrase'))
3595 .sendKeys('abandon abandon ability');
3596 driver.sleep(generateDelay).then(function() {
3597 driver.findElement(By.css('.csv'))
3598 .getAttribute("value
")
3599 .then(function(csv) {
3600 expect(csv).toContain(headings);
3601 expect(csv).toContain(row1);
3602 expect(csv).toContain(row20);
3608 it('LeftPads ethereum keys that are less than 32 bytes', function(done) {
3609 // see https://github.com/iancoleman/bip39/issues/155
3610 selectNetwork("ETH
- Ethereum
");
3611 driver.findElement(By.css('#bip32-tab a'))
3613 driver.findElement(By.css('#bip32-path'))
3615 driver.findElement(By.css('#bip32-path'))
3616 .sendKeys("m
/44'/60'/0'");
3617 driver.findElement(By.css('.phrase'))
3618 .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');
3619 driver.sleep(generateDelay).then(function() {
3620 getFirstAddress(function(address) {
3621 expect(address).toBe("0x8943E785B4a5714FC87a3aFAad1eB1FeB602B118");
3627 it('Can encrypt private keys using BIP38', function(done) {
3628 // see https://github.com/iancoleman/bip39/issues/140
3629 driver.executeScript(function() {
3630 $(".use-bip38
").prop("checked
", true);
3632 driver.findElement(By.css('.bip38-password'))
3633 .sendKeys('bip38password');
3634 driver.findElement(By.css('.rows-to-add'))
3636 driver.findElement(By.css('.rows-to-add'))
3638 driver.findElement(By.css('.phrase'))
3639 .sendKeys('abandon abandon ability');
3640 driver.sleep(bip38delay).then(function() {
3642 getFirstRowValue(function(address) {
3643 expect(address).toBe("1NCvSdumA3ngMM9c4aqU56AM6rqXddfuXB
");
3645 getFirstRowValue(function(pubkey) {
3646 expect(pubkey).toBe("043f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3884a74447ea901729b1e73a999b7520e7cb55b4120e6432c64153ccab8a848e1
");
3648 getFirstRowValue(function(privkey) {
3649 expect(privkey).toBe("6PRNRiFnj1RoR3sXhymdCvoZCgnUHQpfupNdKkFbWJkwWQEKesWt1EDMDM
");
3655 }, bip38delay + 5000);
3657 it('Shows the checksum for the entropy', function(done) {
3658 driver.findElement(By.css('.use-entropy'))
3660 driver.findElement(By.css('.entropy'))
3661 .sendKeys("00000000000000000000000000000000");
3662 driver.sleep(generateDelay).then(function() {
3663 driver.findElement(By.css('.checksum'))
3665 .then(function(text) {
3666 expect(text).toBe("1");
3672 it('Shows the checksum for the entropy with the correct groupings', function(done) {
3673 driver.findElement(By.css('.use-entropy'))
3675 // create a checksum of 20 bits, which spans multiple words
3676 driver.findElement(By.css('.entropy'))
3677 .sendKeys("F000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
");
3678 driver.sleep(generateDelay).then(function() {
3679 driver.findElement(By.css('.checksum'))
3681 .then(function(text) {
3682 // first group is 9 bits, second group is 11
3683 expect(text).toBe("011010111 01110000110");
3689 it('Uses vprv for bitcoin testnet p2wpkh', function(done) {
3690 selectNetwork("BTC
- Bitcoin Testnet
");
3691 driver.findElement(By.css('#bip84-tab a'))
3693 driver.findElement(By.css('.phrase'))
3694 .sendKeys('abandon abandon ability');
3695 driver.sleep(generateDelay).then(function() {
3696 driver.findElement(By.css('.root-key'))
3697 .getAttribute("value
")
3698 .then(function(path) {
3699 expect(path).toBe("vprv9DMUxX4ShgxML9N2YV5CvWEebWrM9aJ5ULpbRRyzyWu6vs4BzTvbfFFrH41N5hVi7MYSfiugd765L3JmAfDM5po36Y8ouCKRDeYQwByCmS7
");
3705 it('Shows a warning if generating weak mnemonics', function(done) {
3706 driver.executeScript(function() {
3707 $(".strength option
[selected
]").removeAttr("selected
");
3708 $(".strength option
[value
=6]").prop("selected
", true);
3709 $(".strength
").trigger("change
");
3711 driver.findElement(By.css(".generate
-container
.warning
"))
3712 .getAttribute("class")
3713 .then(function(classes) {
3714 expect(classes).not.toContain("hidden
");
3719 it('Does not show a warning if generating strong mnemonics', function(done) {
3720 driver.executeScript(function() {
3721 $(".strength option
[selected
]").removeAttr("selected
");
3722 $(".strength option
[value
=12]").prop("selected
", true);
3724 driver.findElement(By.css(".generate
-container
.warning
"))
3725 .getAttribute("class")
3726 .then(function(classes) {
3727 expect(classes).toContain("hidden
");
3732 it('Shows a warning if overriding weak entropy with longer mnemonics', function(done) {
3733 driver.findElement(By.css('.use-entropy'))
3735 driver.findElement(By.css('.entropy'))
3736 .sendKeys("0123456789abcdef
"); // 6 words
3737 driver.executeScript(function() {
3738 $(".mnemonic
-length
").val("12").trigger("change
");
3740 driver.findElement(By.css(".weak
-entropy
-override
-warning
"))
3741 .getAttribute("class")
3742 .then(function(classes) {
3743 expect(classes).not.toContain("hidden
");
3748 it('Does not show a warning if entropy is stronger than mnemonic length', function(done) {
3749 driver.findElement(By.css('.use-entropy'))
3751 driver.findElement(By.css('.entropy'))
3752 .sendKeys("0123456789abcdef0123456789abcdef0123456789abcdef
"); // 18 words
3753 driver.executeScript(function() {
3754 $(".mnemonic
-length
").val("12").trigger("change
");
3756 driver.findElement(By.css(".weak
-entropy
-override
-warning
"))
3757 .getAttribute("class")
3758 .then(function(classes) {
3759 expect(classes).toContain("hidden
");
3764 it('Shows litecoin BIP49 addresses', function(done) {
3765 driver.findElement(By.css('.phrase'))
3766 .sendKeys('abandon abandon ability');
3767 selectNetwork("LTC
- Litecoin
");
3768 driver.findElement(By.css('#bip49-tab a'))
3770 // bip49 addresses are shown
3771 driver.sleep(generateDelay).then(function() {
3772 driver.findElement(By.css('#bip49 .available'))
3773 .getAttribute("class")
3774 .then(function(classes) {
3775 expect(classes).not.toContain("hidden
");
3776 // check first address
3777 getFirstAddress(function(address) {
3778 expect(address).toBe("MFwLPhsXoBuSLL8cLmW9uK6tChkzduV8qN
");
3785 it('Can use root keys to generate segwit table rows', function(done) {
3786 // segwit uses ypub / zpub instead of xpub but the root key should still
3787 // be valid regardless of the encoding used to import that key.
3788 // Maybe this breaks the reason for the different extended key prefixes, but
3789 // since the parsed root key is used behind the scenes anyhow this should be
3791 driver.findElement(By.css('#root-key'))
3792 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
3793 driver.findElement(By.css('#bip49-tab a'))
3795 // bip49 addresses are shown
3796 driver.sleep(generateDelay).then(function() {
3797 getFirstAddress(function(address) {
3798 expect(address).toBe("3QG2Y9AA4xZ846gKHZqNf7mvVKbLqMKxr2
");