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 Bitcore
', function(done) {
709 selectText: "BTX - Bitcore",
710 firstAddress: "2Rgp5Znhpy34TK4QmPkfCiYs9r4KovfTH9",
712 testNetwork(done, params);
714 it('Allows selection
of Bitsend
', function(done) {
716 selectText: "BSD - Bitsend",
717 firstAddress: "iBPk7LYjDun3EPk7CRR8UUmnPoceVc1bp2",
719 testNetwork(done, params);
721 it('Allows selection
of Britcoin
', function(done) {
723 selectText: "BRIT - Britcoin",
724 firstAddress: "B6Aue4J2XLs1f1dtD4H1SHYFfh4XrmEbrw",
726 testNetwork(done, params);
728 it('Allows selection
of Canadaecoin
', function(done) {
730 selectText: "CDN - Canadaecoin",
731 firstAddress: "CanAyCfd5Rj2CQVfaoAmvDUZunPM5W1AEQ",
733 testNetwork(done, params);
735 it('Allows selection
of Cannacoin
', function(done) {
737 selectText: "CCN - Cannacoin",
738 firstAddress: "CYjW8xWB43g6krLJTmmrPk1PonoQX7h9Qd",
740 testNetwork(done, params);
742 it('Allows selection
of Clubcoin
', function(done) {
744 selectText: "CLUB - Clubcoin",
745 firstAddress: "CHMDEXN4sihpSVX4GyAa2hZ62shnby7uyN",
747 testNetwork(done, params);
749 it('Allows selection
of Compcoin
', function(done) {
751 selectText: "CMP - Compcoin",
752 firstAddress: "CLshtw3zhxkseBJS46UF12v3AFy9Dx7JVv",
754 testNetwork(done, params);
756 it('Allows selection
of Crave
', function(done) {
758 selectText: "CRAVE - Crave",
759 firstAddress: "VCYJeti6uKMNBFKCL7eP96UwuFWYHM7c85",
761 testNetwork(done, params);
763 it('Allows selection
of Defcoin
', function(done) {
765 selectText: "DFC - Defcoin",
766 firstAddress: "D8swcgyaaFUrXZU3ATwbgy16buCpWqbG1M",
768 testNetwork(done, params);
770 it('Allows selection
of Diamond
', function(done) {
772 selectText: "DMD - Diamond",
773 firstAddress: "dJnrVbLL9UPjdaVRz2C8VpqHZknqAqjLek",
775 testNetwork(done, params);
777 it('Allows selection
of Digibyte
', function(done) {
779 selectText: "DGB - Digibyte",
780 firstAddress: "D85Rp9jwLtMdmP6wGjTiqHBdVQLST3YCEq",
782 testNetwork(done, params);
784 it('Allows selection
of Digitalcoin
', function(done) {
786 selectText: "DGC - Digitalcoin",
787 firstAddress: "DKw4UGKEAZWweDNEbBFNQx4EM8x1mpUdia",
789 testNetwork(done, params);
791 it('Allows selection
of Ecoin
', function(done) {
793 selectText: "ECN - Ecoin",
794 firstAddress: "e6WFPLG5gcXyF7cESFteH1hE2XSmowW5yB",
796 testNetwork(done, params);
798 it('Allows selection
of Edrcoin
', function(done) {
800 selectText: "EDRC - Edrcoin",
801 firstAddress: "eh1nUJsvgKPFv6ebMBfcwJ299GMCpjeZUG",
803 testNetwork(done, params);
805 it('Allows selection
of Egulden
', function(done) {
807 selectText: "EFL - Egulden",
808 firstAddress: "Lg66yt55R7edRM58cDhKzXik2kFme3viX7",
810 testNetwork(done, params);
812 it('Allows selection
of Einsteinium
', function(done) {
814 selectText: "EMC2 - Einsteinium",
815 firstAddress: "EVAABm9hXKHk2MpVMbwNakRubFnNha5m8m",
817 testNetwork(done, params);
819 it('Allows selection
of Europecoin
', function(done) {
821 selectText: "ERC - Europecoin",
822 firstAddress: "ESA2YwPYntAoaPrE8Fm5qkKRtkcwLcwD6R",
824 testNetwork(done, params);
826 it('Allows selection
of Exclusivecoin
', function(done) {
828 selectText: "EXCL - Exclusivecoin",
829 firstAddress: "EbUa6m8UZW6nTxsYZD2FsDjkadKbp5M6JT",
831 testNetwork(done, params);
833 it('Allows selection
of Feathercoin
', function(done) {
835 selectText: "FTC - Feathercoin",
836 firstAddress: "6gDdjAMoSgQaW8UhqK3oboHs6ftGAroKkM",
838 testNetwork(done, params);
840 it('Allows selection
of Firstcoin
', function(done) {
842 selectText: "FRST - Firstcoin",
843 firstAddress: "FJN9GzfMm7Q8R4DJwK1H9F6A1GTghvFiMJ",
845 testNetwork(done, params);
847 it('Allows selection
of Flashcoin
', function(done) {
849 selectText: "FLASH - Flashcoin",
850 firstAddress: "UWfpf5LfMmLxZYooEb2EyvWhZ8NG7EZDRt",
852 testNetwork(done, params);
854 it('Allows selection
of GCRCoin
', function(done) {
856 selectText: "GCR - GCRCoin",
857 firstAddress: "GJjF5cLwyXLacpuvXAVksxGxKvHDjx58d6",
859 testNetwork(done, params);
861 it('Allows selection
of Gobyte
', function(done) {
863 selectText: "GBX - Gobyte",
864 firstAddress: "GS813Ys2brkmvSUw1rUqGPm2HqQVDHJRyA",
866 testNetwork(done, params);
868 it('Allows selection
of Gridcoin
', function(done) {
870 selectText: "GRC - Gridcoin",
871 firstAddress: "SGrWbBPvobgqKRF8td1Kdc9vbRY7MJ78Y9",
873 testNetwork(done, params);
875 it('Allows selection
of Gulden
', function(done) {
877 selectText: "NLG - Gulden",
878 firstAddress: "GcDP7cNEc33MPPdTFNJ8pZc6VMZJ2CbKxY",
880 testNetwork(done, params);
882 it('Allows selection
of Helleniccoin
', function(done) {
884 selectText: "HNC - Helleniccoin",
885 firstAddress: "LbHEKe5H72zp9G1fuWNiiNePTUfJb88915",
887 testNetwork(done, params);
889 it('Allows selection
of Hempcoin
', function(done) {
891 selectText: "THC - Hempcoin",
892 firstAddress: "H8sdWbZyJV4gyXyHtLXDaNnAuUDhK5mfTV",
894 testNetwork(done, params);
896 it('Allows selection
of Insane
', function(done) {
898 selectText: "INSN - Insane",
899 firstAddress: "iMPqEJMiXWuxC9U2NVinCCMr4t72h58EWx",
901 testNetwork(done, params);
903 it('Allows selection
of Iop
', function(done) {
905 selectText: "IOP - Iop",
906 firstAddress: "pGKQmcaPf95Ur5o6oHK4qdiZ52p1yaTvq1",
908 testNetwork(done, params);
910 it('Allows selection
of Ixcoin
', function(done) {
912 selectText: "IXC - Ixcoin",
913 firstAddress: "xgE9bTZ6YypT3E6ByzkTt31Hq68E9BqywH",
915 testNetwork(done, params);
917 it('Allows selection
of Kobocoin
', function(done) {
919 selectText: "KOBO - Kobocoin",
920 firstAddress: "FTVoNJETXDAM8x7MnmdE8RwWndSr9PQWhy",
922 testNetwork(done, params);
924 it('Allows selection
of Landcoin
', function(done) {
926 selectText: "LDCN - Landcoin",
927 firstAddress: "LLvLwNjG1aJcn1RS4W4GJUbv8fNaRATG7c",
929 testNetwork(done, params);
931 it('Allows selection
of Library Credits
', function(done) {
933 selectText: "LBC - Library Credits",
934 firstAddress: "bQJEQrHDJyHdqycB32uysh1SWn8Ln8LMdg",
936 testNetwork(done, params);
938 it('Allows selection
of Linx
', function(done) {
940 selectText: "LINX - Linx",
941 firstAddress: "XGWQ3cb3LGUB3VnHmj6xYSMgnokNbf6dyk",
943 testNetwork(done, params);
945 it('Allows selection
of Litecoincash
', function(done) {
947 selectText: "LCC - Litecoincash",
948 firstAddress: "Ce5n7fjUuQPLutJ4W5nCCfQLKdKLE1mv9A",
950 testNetwork(done, params);
952 it('Allows selection
of Lynx
', function(done) {
954 selectText: "LYNX - Lynx",
955 firstAddress: "KUeY3ZdZkg96p4W98pj1JjygCFU1XqWdw3",
957 testNetwork(done, params);
959 it('Allows selection
of Minexcoin
', function(done) {
961 selectText: "MNX - Minexcoin",
962 firstAddress: "XC1VnyJVfiMDwWgFtAHDp41cgY3AHk3dJT",
964 testNetwork(done, params);
966 it('Allows selection
of Navcoin
', function(done) {
968 selectText: "NAV - Navcoin",
969 firstAddress: "NTQVTPK3NWSQLKoffkiQw99T8PifkF1Y2U",
971 testNetwork(done, params);
973 it('Allows selection
of Neoscoin
', function(done) {
975 selectText: "NEOS - Neoscoin",
976 firstAddress: "NgATz6QbQNXvayHQ4CpZayugb9HeaPDdby",
978 testNetwork(done, params);
980 it('Allows selection
of Neurocoin
', function(done) {
982 selectText: "NRO - Neurocoin",
983 firstAddress: "NVdYErQ3mFpDuF5DquW9WMiT7sLc8ufFTn",
985 testNetwork(done, params);
987 it('Allows selection
of Newyorkc
', function(done) {
989 selectText: "NYC - Newyorkc",
990 firstAddress: "RSVMfyH1fKfy3puADJEhut2vfkRyon6imm",
992 testNetwork(done, params);
994 it('Allows selection
of Novacoin
', function(done) {
996 selectText: "NVC - Novacoin",
997 firstAddress: "4JRvUmxcKCJmaMXZyvRoSS1cmG2XvnZfHN",
999 testNetwork(done, params);
1001 it('Allows selection
of Nushares
', function(done) {
1003 selectText: "NSR - Nushares",
1004 firstAddress: "SecjXzU3c7EecdT7EbC4vvmbdtBBokWh6J",
1006 testNetwork(done, params);
1008 it('Allows selection
of Okcash
', function(done) {
1010 selectText: "OK - Okcash",
1011 firstAddress: "PV4Qp1TUYuGv4TqVtLZtqvrsWWRycfx1Yi",
1013 testNetwork(done, params);
1015 it('Allows selection
of Omnicore
', function(done) {
1017 selectText: "OMNI - Omnicore",
1018 firstAddress: "1Q1t3gonjCT3rW38TsTsCvgSc3hh7zBGbi",
1020 testNetwork(done, params);
1022 it('Allows selection
of Pesobit
', function(done) {
1024 selectText: "PSB - Pesobit",
1025 firstAddress: "PDePsF7ALyXP7JaywokdYiRTDtKa14MAr1",
1027 testNetwork(done, params);
1029 it('Allows selection
of Pinkcoin
', function(done) {
1031 selectText: "PINK - Pinkcoin",
1032 firstAddress: "2TgjYQffjbzUHJghNaVbdsjHbRwruC3yzC",
1034 testNetwork(done, params);
1036 it('Allows selection
of POSWcoin
', function(done) {
1038 selectText: "POSW - POSWcoin",
1039 firstAddress: "PNxewmZoPnGBvoEbH6hgQZCK1igDiBCdgC",
1041 testNetwork(done, params);
1043 it('Allows selection
of Potcoin
', function(done) {
1045 selectText: "POT - Potcoin",
1046 firstAddress: "PEo7Vg2ctXgpP4vuLPeY9aGJtZotyrmiHc",
1048 testNetwork(done, params);
1050 it('Allows selection
of Putincoin
', function(done) {
1052 selectText: "PUT - Putincoin",
1053 firstAddress: "PViWnfr2uFtovd6e7joM49C94CsGSnqJis",
1055 testNetwork(done, params);
1057 it('Allows selection
of Reddcoin
', function(done) {
1059 selectText: "RDD - Reddcoin",
1060 firstAddress: "RtgRvXMBng1y51ftteveFqwNfyRG18HpxQ",
1062 testNetwork(done, params);
1064 it('Allows selection
of RevolutionVR
', function(done) {
1066 selectText: "RVR - RevolutionVR",
1067 firstAddress: "VXeeoP2jkzZnMFxtc66ZBZK1NHN5QJnnjL",
1069 testNetwork(done, params);
1071 it('Allows selection
of Rubycoin
', function(done) {
1073 selectText: "RBY - Rubycoin",
1074 firstAddress: "RV76JDtjTs11JdMDRToYn6CHecMRPLnKS6",
1076 testNetwork(done, params);
1078 it('Allows selection
of Salus
', function(done) {
1080 selectText: "SLS - Salus",
1081 firstAddress: "SgdYBmVytcW2aCYitdegwkUcCU7RSqYokB",
1083 it('Allows selection
of Smileycoin
', function(done) {
1085 selectText: "SMLY - Smileycoin",
1086 firstAddress: "BEZVnEBCAyFByrgKpwAgYgtvP4rKAd9Sj2",
1088 testNetwork(done, params);
1090 it('Allows selection
of Solarcoin
', function(done) {
1092 selectText: "SLR - Solarcoin",
1093 firstAddress: "8LZ13HbnjtaMJWSvvVFNTLf71zFfDrhwLu",
1095 testNetwork(done, params);
1097 it('Allows selection
of stash
', function(done) {
1099 selectText: "STASH - Stash",
1100 firstAddress: "XxwAsWB7REDKmAvHA85SbEZQQtpxeUDxS3",
1102 testNetwork(done, params);
1104 it('Allows selection
of stash testnet
', function(done) {
1106 selectText: "STASH - Stash Testnet",
1107 firstAddress: "YdbhtMuGsPSkE6bPdNTHoFSszQKmK4S5LT",
1109 testNetwork(done, params);
1111 it('Allows selection
of Stratis
', function(done) {
1113 selectText: "STRAT - Stratis",
1114 firstAddress: "ScfJnq3QDhKgDMEds6sqUE1ot6ShfhmXXq",
1116 testNetwork(done, params);
1118 it('Allows selection
of Stratis Test
', function(done) {
1120 selectText: "TSTRAT - Stratis Testnet",
1121 firstAddress: "TRLWm3dye4FRrDWouwYUSUZP96xb76mBE3",
1123 testNetwork(done, params);
1125 it('Allows selection
of Syscoin
', function(done) {
1127 selectText: "SYS - Syscoin",
1128 firstAddress: "SZwJi42Pst3VAMomyK5DG4157WM5ofRmSj",
1130 testNetwork(done, params);
1132 it('Allows selection
of Toa
', function(done) {
1134 selectText: "TOA - Toa",
1135 firstAddress: "TSe1QAnUwQzUfbBusDzRJ9URttrRGKoNKF",
1137 testNetwork(done, params);
1139 it('Allows selection
of Ultimatesecurecash
', function(done) {
1141 selectText: "USC - Ultimatesecurecash",
1142 firstAddress: "UPyLAZU2Che5fiy7Ed8xVJFmXAUhitA4ug",
1144 testNetwork(done, params);
1146 it('Allows selection
of Unobtanium
', function(done) {
1148 selectText: "UNO - Unobtanium",
1149 firstAddress: "uUBMPVMXrR6qhqornJqKTWgr8L69vihSL9",
1151 testNetwork(done, params);
1153 it('Allows selection
of Vcash
', function(done) {
1155 selectText: "XVC - Vcash",
1156 firstAddress: "VuL53MSY6KjvAjKSeRkh3NDnKykacDVeps",
1158 testNetwork(done, params);
1160 it('Allows selection
of Verge
', function(done) {
1162 selectText: "XVG - Verge",
1163 firstAddress: "DCrVuGkMjLJpTGgwAgv9AcMdeb1nkWbjZA",
1165 testNetwork(done, params);
1167 it('Allows selection
of Vertcoin
', function(done) {
1169 selectText: "VTC - Vertcoin",
1170 firstAddress: "Vf6koGuiWdXQfx8tNqxoNeEDxh4xh5cxsG",
1172 testNetwork(done, params);
1174 it('Allows selection
of Vivo
', function(done) {
1176 selectText: "VIVO - Vivo",
1177 firstAddress: "VFmBwuXXGhJe7MarQG2GfzHMFebRHgfSpB",
1179 testNetwork(done, params);
1181 it('Allows selection
of Vpncoin
', function(done) {
1183 selectText: "VASH - Vpncoin",
1184 firstAddress: "VoEmH1qXC4TsSgBAStR21QYetwnFqbqCx9",
1186 testNetwork(done, params);
1188 it('Allows selection
of Whitecoin
', function(done) {
1190 selectText: "XWC - Whitecoin",
1191 firstAddress: "WcSwCAUqrSgeSYbsaS3SSWWhsx8KRYTFDR",
1193 testNetwork(done, params);
1195 it('Allows selection
of Wincoin
', function(done) {
1197 selectText: "WC - Wincoin",
1198 firstAddress: "WaDVCESMGgyKgNESdn3u43NnwmGSkZED3Z",
1200 testNetwork(done, params);
1202 it('Allows selection
of Zcoin
', function(done) {
1204 selectText: "XZC - Zcoin",
1205 firstAddress: "a6VcMdP4XgAA9Tr7xNszmPG5FZpfRf17Cq",
1207 testNetwork(done, params);
1209 it('Allows selection
of Zcash
', function(done) {
1211 selectText: "ZEC - Zcash",
1212 firstAddress: "t1Sz8AneMcVuzUg3tPJ8et5AS5LFJ7K2EF9",
1214 testNetwork(done, params);
1216 it('Allows selection
of Zclassic
', function(done) {
1218 selectText: "ZCL - Zclassic",
1219 firstAddress: "t1TBMxTvVJRybUbMLGWq8H4A8F4VUL7czEc",
1221 testNetwork(done, params);
1223 it('Allows selection
of Zencash
', function(done) {
1225 selectText: "ZEN - Zencash",
1226 firstAddress: "znWh9XASyW2dZq5tck84wFjiwuqVysi7q3p",
1228 testNetwork(done, params);
1230 it('Allows selection
of Energi
', function(done) {
1232 selectText: "NRG - Energi",
1233 firstAddress: "EejRy4t4nidzhGGzkJUgFP3z4HYBjhTsRt",
1235 testNetwork(done, params);
1237 it('Allows selection
of Ethereum Classic
', function(done) {
1239 selectText: "ETC - Ethereum Classic",
1240 firstAddress: "0x3c05e5556693808367afB62eF3b63e35d6eD249A",
1242 testNetwork(done, params);
1244 it('Allows selection
of Pirl
', function(done) {
1246 selectText: "PIRL - Pirl",
1247 firstAddress: "0xe77FC0723dA122B5025CA79193c28563eB47e776",
1249 testNetwork(done, params);
1251 it('Allows selection
of MIX
', function(done) {
1253 selectText: "MIX - MIX",
1254 firstAddress: "0x98BC5e63aeb6A4e82d72850d20710F07E29A29F1",
1256 testNetwork(done, params);
1258 it('Allows selection
of Musicoin
', function(done) {
1260 selectText: "MUSIC - Musicoin",
1261 firstAddress: "0xDc060e4A0b0313ea83Cf6B3A39B9db2D29004897",
1263 testNetwork(done, params);
1265 it('Allows selection
of Poa
', function(done) {
1267 selectText: "POA - Poa",
1268 firstAddress: "0x53aF28d754e106210C3d0467Dd581eaf7e3C5e60",
1270 testNetwork(done, params);
1272 it('Allows selection
of Expanse
', function(done) {
1274 selectText: "EXP - Expanse",
1275 firstAddress: "0xf57FeAbf26582b6E3E666559d3B1Cc6fB2b2c5F6",
1277 testNetwork(done, params);
1279 it('Allows selection
of Callisto
', function(done) {
1281 selectText: "CLO - Callisto",
1282 firstAddress: "0x4f9364F7420B317266C51Dc8eB979717D4dE3f4E",
1284 testNetwork(done, params);
1286 it('Allows selection
of HUSH
', function(done) {
1288 selectText: "HUSH - Hush",
1289 firstAddress: "t1g6rLXUnJaiJuu4q4zmJjoa9Gk4fwKpiuA",
1291 testNetwork(done, params);
1293 it('Allows selection
of ExchangeCoin
', function(done) {
1295 selectText: "EXCC - ExchangeCoin",
1296 firstAddress: "22txYKpFN5fwGwdSs2UBf7ywewbLM92YqK7E",
1298 testNetwork(done, params);
1302 // BIP39 seed is set from phrase
1303 it('Sets the bip39 seed
from the prhase
', function(done) {
1304 driver.findElement(By.css('.phrase
'))
1305 .sendKeys('abandon abandon ability
');
1306 driver.sleep(generateDelay).then(function() {
1307 driver.findElement(By.css('.seed
'))
1308 .getAttribute("value")
1309 .then(function(seed) {
1310 expect(seed).toBe("20da140d3dd1df8713cefcc4d54ce0e445b4151027a1ab567b832f6da5fcc5afc1c3a3f199ab78b8e0ab4652efd7f414ac2c9a3b81bceb879a70f377aa0a58f3");
1316 // BIP32 root key is set from phrase
1317 it('Sets the bip39 root key
from the prhase
', function(done) {
1318 driver.findElement(By.css('.phrase
'))
1319 .sendKeys('abandon abandon ability
');
1320 driver.sleep(generateDelay).then(function() {
1321 driver.findElement(By.css('.root
-key
'))
1322 .getAttribute("value")
1323 .then(function(seed) {
1324 expect(seed).toBe("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
1330 // Tabs show correct addresses when changed
1331 it('Shows the correct address when tab is changed
', function(done) {
1332 driver.findElement(By.css('.phrase
'))
1333 .sendKeys('abandon abandon ability
');
1334 driver.sleep(generateDelay).then(function() {
1335 driver.findElement(By.css('#bip32
-tab a
'))
1337 driver.sleep(generateDelay).then(function() {
1338 getFirstAddress(function(address) {
1339 expect(address).toBe("17uQ7s2izWPwBmEVFikTmZUjbBKWYdJchz");
1346 // BIP44 derivation path is shown
1347 it('Shows the derivation path
for bip44 tab
', function(done) {
1348 driver.findElement(By.css('.phrase
'))
1349 .sendKeys('abandon abandon ability
');
1350 driver.sleep(generateDelay).then(function() {
1351 driver.findElement(By.css('#bip44
.path
'))
1352 .getAttribute("value")
1353 .then(function(path) {
1354 expect(path).toBe("m/44'/0'/0'/0");
1360 // BIP44 extended private key is shown
1361 it('Shows the extended private key for bip44 tab', function(done) {
1362 driver.findElement(By.css('.phrase'))
1363 .sendKeys('abandon abandon ability');
1364 driver.sleep(generateDelay).then(function() {
1365 driver.findElement(By.css('.extended-priv-key'))
1366 .getAttribute("value
")
1367 .then(function(path) {
1368 expect(path).toBe("xprvA2DxxvPZcyRvYgZMGS53nadR32mVDeCyqQYyFhrCVbJNjPoxMeVf7QT5g7mQASbTf9Kp4cryvcXnu2qurjWKcrdsr91jXymdCDNxKgLFKJG
");
1374 // BIP44 extended public key is shown
1375 it('Shows the extended public key for bip44 tab', function(done) {
1376 driver.findElement(By.css('.phrase'))
1377 .sendKeys('abandon abandon ability');
1378 driver.sleep(generateDelay).then(function() {
1379 driver.findElement(By.css('.extended-pub-key'))
1380 .getAttribute("value
")
1381 .then(function(path) {
1382 expect(path).toBe("xpub6FDKNRvTTLzDmAdpNTc49ia9b4byd6vqCdUa46Fp3vqMcC96uBoufCmZXQLiN5AK3iSCJMhf9gT2sxkpyaPepRuA7W3MujV5tGmF5VfbueM
");
1388 // BIP44 account field changes address list
1389 it('Changes the address list if bip44 account is changed', function(done) {
1390 driver.findElement(By.css('#bip44 .account'))
1392 driver.findElement(By.css('.phrase'))
1393 .sendKeys('abandon abandon ability');
1394 driver.sleep(generateDelay).then(function() {
1395 getFirstAddress(function(address) {
1396 expect(address).toBe("1Nq2Wmu726XHCuGhctEtGmhxo3wzk5wZ1H
");
1402 // BIP44 change field changes address list
1403 it('Changes the address list if bip44 change is changed', function(done) {
1404 driver.findElement(By.css('#bip44 .change'))
1406 driver.findElement(By.css('.phrase'))
1407 .sendKeys('abandon abandon ability');
1408 driver.sleep(generateDelay).then(function() {
1409 getFirstAddress(function(address) {
1410 expect(address).toBe("1KAGfWgqfVbSSXY56fNQ7YnhyKuoskHtYo
");
1416 // BIP32 derivation path can be set
1417 it('Can use a custom bip32 derivation path', function(done) {
1418 driver.findElement(By.css('#bip32-tab a'))
1420 driver.findElement(By.css('#bip32 .path'))
1422 driver.findElement(By.css('#bip32 .path'))
1424 driver.findElement(By.css('.phrase'))
1425 .sendKeys('abandon abandon ability');
1426 driver.sleep(generateDelay).then(function() {
1427 getFirstAddress(function(address) {
1428 expect(address).toBe("16pYQQdLD1hH4hwTGLXBaZ9Teboi1AGL8L
");
1434 // BIP32 can use hardened derivation paths
1435 it('Can use a hardened derivation paths', function(done) {
1436 driver.findElement(By.css('#bip32-tab a'))
1438 driver.findElement(By.css('#bip32 .path'))
1440 driver.findElement(By.css('#bip32 .path'))
1442 driver.findElement(By.css('.phrase
'))
1443 .sendKeys('abandon abandon ability
');
1444 driver.sleep(generateDelay).then(function() {
1445 getFirstAddress(function(address) {
1446 expect(address).toBe("14aXZeprXAE3UUKQc4ihvwBvww2LuEoHo4");
1452 // BIP32 extended private key is shown
1453 it('Shows the BIP32 extended
private key
', function(done) {
1454 driver.findElement(By.css('#bip32
-tab a
'))
1456 driver.findElement(By.css('.phrase
'))
1457 .sendKeys('abandon abandon ability
');
1458 driver.sleep(generateDelay).then(function() {
1459 driver.findElement(By.css('.extended
-priv
-key
'))
1460 .getAttribute("value")
1461 .then(function(privKey) {
1462 expect(privKey).toBe("xprv9va99uTVE5aLiutUVLTyfxfe8v8aaXjSQ1XxZbK6SezYVuikA9MnjQVTA8rQHpNA5LKvyQBpLiHbBQiiccKiBDs7eRmBogsvq3THFeLHYbe");
1468 // BIP32 extended public key is shown
1469 it('Shows the BIP32 extended
public key
', function(done) {
1470 driver.findElement(By.css('#bip32
-tab a
'))
1472 driver.findElement(By.css('.phrase
'))
1473 .sendKeys('abandon abandon ability
');
1474 driver.sleep(generateDelay).then(function() {
1475 driver.findElement(By.css('.extended
-pub
-key
'))
1476 .getAttribute("value")
1477 .then(function(pubKey) {
1478 expect(pubKey).toBe("xpub69ZVZQzP4T8dwPxwbMzz36cNgwy4yzTHmETZMyihzzXXNi3thgg3HCow1RtY252wdw5rS8369xKnraN5Q93y3FkFfJp2XEHWUrkyXsjS93P");
1484 // Derivation path is shown in table
1485 it('Shows the derivation path
in the table
', function(done) {
1486 driver.findElement(By.css('.phrase
'))
1487 .sendKeys('abandon abandon ability
');
1488 driver.sleep(generateDelay).then(function() {
1489 getFirstPath(function(path) {
1490 expect(path).toBe("m/44'/0'/0'/0/0");
1496 // Derivation path for address can be hardened
1497 it('Can derive hardened addresses', function(done) {
1498 driver.findElement(By.css('#bip32-tab a'))
1500 driver.executeScript(function() {
1501 $(".hardened
-addresses
").prop("checked
", true);
1503 driver.findElement(By.css('.phrase'))
1504 .sendKeys('abandon abandon ability');
1505 driver.sleep(generateDelay).then(function() {
1506 getFirstAddress(function(address) {
1507 expect(address).toBe("18exLzUv7kfpiXRzmCjFDoC9qwNLFyvwyd
");
1513 // Derivation path visibility can be toggled
1514 it('Can toggle visibility of the derivation path column', function(done) {
1515 driver.findElement(By.css('.phrase'))
1516 .sendKeys('abandon abandon ability');
1517 driver.sleep(generateDelay).then(function() {
1518 driver.findElement(By.css('.index-toggle'))
1520 testColumnValuesAreInvisible(done, "index
");
1525 it('Shows the address in the table', function(done) {
1526 driver.findElement(By.css('.phrase'))
1527 .sendKeys('abandon abandon ability');
1528 driver.sleep(generateDelay).then(function() {
1529 getFirstAddress(function(address) {
1530 expect(address).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug
");
1536 // Addresses are shown in order of derivation path
1537 it('Shows the address in order of derivation path', function(done) {
1538 driver.findElement(By.css('.phrase'))
1539 .sendKeys('abandon abandon ability');
1540 driver.sleep(generateDelay).then(function() {
1541 testRowsAreInCorrectOrder(done);
1545 // Address visibility can be toggled
1546 it('Can toggle visibility of the address column', function(done) {
1547 driver.findElement(By.css('.phrase'))
1548 .sendKeys('abandon abandon ability');
1549 driver.sleep(generateDelay).then(function() {
1550 driver.findElement(By.css('.address-toggle'))
1552 testColumnValuesAreInvisible(done, "address
");
1556 // Public key is shown in table
1557 it('Shows the public key in the table', function(done) {
1558 driver.findElement(By.css('.phrase'))
1559 .sendKeys('abandon abandon ability');
1560 driver.sleep(generateDelay).then(function() {
1561 driver.findElements(By.css('.pubkey'))
1562 .then(function(els) {
1564 .then(function(pubkey) {
1565 expect(pubkey).toBe("033f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3
");
1572 // Public key visibility can be toggled
1573 it('Can toggle visibility of the public key column', function(done) {
1574 driver.findElement(By.css('.phrase'))
1575 .sendKeys('abandon abandon ability');
1576 driver.sleep(generateDelay).then(function() {
1577 driver.findElement(By.css('.public-key-toggle'))
1579 testColumnValuesAreInvisible(done, "pubkey
");
1583 // Private key is shown in table
1584 it('Shows the private key in the table', function(done) {
1585 driver.findElement(By.css('.phrase'))
1586 .sendKeys('abandon abandon ability');
1587 driver.sleep(generateDelay).then(function() {
1588 driver.findElements(By.css('.privkey'))
1589 .then(function(els) {
1591 .then(function(pubkey) {
1592 expect(pubkey).toBe("L26cVSpWFkJ6aQkPkKmTzLqTdLJ923e6CzrVh9cmx21QHsoUmrEE
");
1599 // Private key visibility can be toggled
1600 it('Can toggle visibility of the private key column', function(done) {
1601 driver.findElement(By.css('.phrase'))
1602 .sendKeys('abandon abandon ability');
1603 driver.sleep(generateDelay).then(function() {
1604 driver.findElement(By.css('.private-key-toggle'))
1606 testColumnValuesAreInvisible(done, "privkey
");
1610 // More addresses can be generated
1611 it('Can generate more rows in the table', function(done) {
1612 driver.findElement(By.css('.phrase'))
1613 .sendKeys('abandon abandon ability');
1614 driver.sleep(generateDelay).then(function() {
1615 driver.findElement(By.css('.more'))
1617 driver.sleep(generateDelay).then(function() {
1618 driver.findElements(By.css('.address'))
1619 .then(function(els) {
1620 expect(els.length).toBe(40);
1627 // A custom number of additional addresses can be generated
1628 it('Can generate more rows in the table', function(done) {
1629 driver.findElement(By.css('.phrase'))
1630 .sendKeys('abandon abandon ability');
1631 driver.sleep(generateDelay).then(function() {
1632 driver.findElement(By.css('.rows-to-add'))
1634 driver.findElement(By.css('.rows-to-add'))
1636 driver.findElement(By.css('.more'))
1638 driver.sleep(generateDelay).then(function() {
1639 driver.findElements(By.css('.address'))
1640 .then(function(els) {
1641 expect(els.length).toBe(21);
1648 // Additional addresses are shown in order of derivation path
1649 it('Shows additional addresses in order of derivation path', function(done) {
1650 driver.findElement(By.css('.phrase'))
1651 .sendKeys('abandon abandon ability');
1652 driver.sleep(generateDelay).then(function() {
1653 driver.findElement(By.css('.more'))
1655 driver.sleep(generateDelay).then(function() {
1656 testRowsAreInCorrectOrder(done);
1661 // BIP32 root key can be set by the user
1662 it('Allows the user to set the BIP32 root key', function(done) {
1663 driver.findElement(By.css('.root-key'))
1664 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1665 driver.sleep(generateDelay).then(function() {
1666 getFirstAddress(function(address) {
1667 expect(address).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug
");
1673 // Setting BIP32 root key clears the existing phrase, passphrase and seed
1674 // TODO this doesn't work in selenium with chrome
1675 it('Confirms the existing phrase should be cleared', function(done) {
1676 if (browser == "chrome
") {
1677 pending("Selenium
+ Chrome headless bug
for alert
, see
https://stackoverflow.com/q/45242264");
1679 driver
.findElement(By
.css('.phrase'))
1680 .sendKeys('A non-blank but invalid value');
1681 driver
.findElement(By
.css('.root-key'))
1682 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1683 driver
.switchTo().alert().accept();
1684 driver
.findElement(By
.css('.phrase'))
1685 .getAttribute("value").then(function(value
) {
1686 expect(value
).toBe("");
1691 // Clearing of phrase, passphrase and seed can be cancelled by user
1692 // TODO this doesn't work in selenium with chrome
1693 it('Allows the clearing of the phrase to be cancelled', function(done
) {
1694 if (browser
== "chrome") {
1695 pending("Selenium + Chrome headless bug for alert, see https://stackoverflow.com/q/45242264");
1697 driver
.findElement(By
.css('.phrase'))
1698 .sendKeys('abandon abandon ability');
1699 driver
.sleep(generateDelay
).then(function() {
1700 driver
.findElement(By
.css('.root-key'))
1702 driver
.findElement(By
.css('.root-key'))
1704 driver
.switchTo().alert().dismiss();
1705 driver
.findElement(By
.css('.phrase'))
1706 .getAttribute("value").then(function(value
) {
1707 expect(value
).toBe("abandon abandon ability");
1713 // Custom BIP32 root key is used when changing the derivation path
1714 it('Can set derivation path for root key instead of phrase', function(done
) {
1715 driver
.findElement(By
.css('#bip44 .account'))
1717 driver
.findElement(By
.css('.root-key'))
1718 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1719 driver
.sleep(generateDelay
).then(function() {
1720 getFirstAddress(function(address
) {
1721 expect(address
).toBe("1Nq2Wmu726XHCuGhctEtGmhxo3wzk5wZ1H");
1727 // Incorrect mnemonic shows error
1728 it('Shows an error for incorrect mnemonic', function(done
) {
1729 driver
.findElement(By
.css('.phrase'))
1730 .sendKeys('abandon abandon abandon');
1731 driver
.sleep(feedbackDelay
).then(function() {
1732 driver
.findElement(By
.css('.feedback'))
1734 .then(function(feedback
) {
1735 expect(feedback
).toBe("Invalid mnemonic");
1741 // Incorrect word shows suggested replacement
1742 it('Shows word suggestion for incorrect word', function(done
) {
1743 driver
.findElement(By
.css('.phrase'))
1744 .sendKeys('abandon abandon abiliti');
1745 driver
.sleep(feedbackDelay
).then(function() {
1746 driver
.findElement(By
.css('.feedback'))
1748 .then(function(feedback
) {
1749 var msg
= "abiliti not in wordlist, did you mean ability?";
1750 expect(feedback
).toBe(msg
);
1756 // Github pull request 48
1757 // First four letters of word shows that word, not closest
1758 // since first four letters gives unique word in BIP39 wordlist
1759 // eg ille should show illegal, not idle
1760 it('Shows word suggestion based on first four chars', function(done
) {
1761 driver
.findElement(By
.css('.phrase'))
1763 driver
.sleep(feedbackDelay
).then(function() {
1764 driver
.findElement(By
.css('.feedback'))
1766 .then(function(feedback
) {
1767 var msg
= "ille not in wordlist, did you mean illegal?";
1768 expect(feedback
).toBe(msg
);
1774 // Incorrect BIP32 root key shows error
1775 it('Shows error for incorrect root key', function(done
) {
1776 driver
.findElement(By
.css('.root-key'))
1777 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpj');
1778 driver
.sleep(feedbackDelay
).then(function() {
1779 driver
.findElement(By
.css('.feedback'))
1781 .then(function(feedback
) {
1782 var msg
= "Invalid root key";
1783 expect(feedback
).toBe(msg
);
1789 // Derivation path not starting with m shows error
1790 it('Shows error for derivation path not starting with m', function(done
) {
1791 driver
.findElement(By
.css('#bip32-tab a'))
1793 driver
.findElement(By
.css('#bip32 .path'))
1795 driver
.findElement(By
.css('#bip32 .path'))
1797 driver
.findElement(By
.css('.phrase'))
1798 .sendKeys('abandon abandon ability');
1799 driver
.sleep(feedbackDelay
).then(function() {
1800 driver
.findElement(By
.css('.feedback'))
1802 .then(function(feedback
) {
1803 var msg
= "First character must be 'm'";
1804 expect(feedback
).toBe(msg
);
1810 // Derivation path containing invalid characters shows useful error
1811 it('Shows error for derivation path not starting with m', function(done
) {
1812 driver
.findElement(By
.css('#bip32-tab a'))
1814 driver
.findElement(By
.css('#bip32 .path'))
1816 driver
.findElement(By
.css('#bip32 .path'))
1817 .sendKeys('m/1/0wrong1/1');
1818 driver
.findElement(By
.css('.phrase'))
1819 .sendKeys('abandon abandon ability');
1820 driver
.sleep(feedbackDelay
).then(function() {
1821 driver
.findElement(By
.css('.feedback'))
1823 .then(function(feedback
) {
1824 var msg
= "Invalid characters 0wrong1 found at depth 2";
1825 expect(feedback
).toBe(msg
);
1831 // Github Issue 11: Default word length is 15
1832 // https://github.com/iancoleman/bip39/issues/11
1833 it('Sets the default word length to 15', function(done
) {
1834 driver
.findElement(By
.css('.strength'))
1835 .getAttribute("value")
1836 .then(function(strength
) {
1837 expect(strength
).toBe("15");
1842 // Github Issue 12: Generate more rows with private keys hidden
1843 // https://github.com/iancoleman/bip39/issues/12
1844 it('Sets the correct hidden column state on new rows', function(done
) {
1845 driver
.findElement(By
.css('.phrase'))
1846 .sendKeys("abandon abandon ability");
1847 driver
.sleep(generateDelay
).then(function() {
1848 driver
.findElement(By
.css('.private-key-toggle'))
1850 driver
.findElement(By
.css('.more'))
1852 driver
.sleep(generateDelay
).then(function() {
1853 driver
.findElements(By
.css('.privkey'))
1854 .then(function(els
) {
1855 expect(els
.length
).toBe(40);
1857 testColumnValuesAreInvisible(done
, "privkey");
1862 // Github Issue 19: Mnemonic is not sensitive to whitespace
1863 // https://github.com/iancoleman/bip39/issues/19
1864 it('Ignores excess whitespace in the mnemonic', function(done
) {
1865 var doublespace
= " ";
1866 var mnemonic
= "urge cat" + doublespace
+ "bid";
1867 driver
.findElement(By
.css('.phrase'))
1868 .sendKeys(mnemonic
);
1869 driver
.sleep(generateDelay
).then(function() {
1870 driver
.findElement(By
.css('.root-key'))
1871 .getAttribute("value")
1872 .then(function(seed
) {
1873 expect(seed
).toBe("xprv9s21ZrQH143K3isaZsWbKVoTtbvd34Y1ZGRugGdMeBGbM3AgBVzTH159mj1cbbtYSJtQr65w6L5xy5L9SFC7c9VJZWHxgAzpj4mun5LhrbC");
1879 // Github Issue 23: Part 1: Use correct derivation path when changing tabs
1880 // https://github.com/iancoleman/bip39/issues/23
1881 it('Uses the correct derivation path when changing tabs', function(done
) {
1882 // 1) and 2) set the phrase
1883 driver
.findElement(By
.css('.phrase'))
1884 .sendKeys("abandon abandon ability");
1885 driver
.sleep(generateDelay
).then(function() {
1886 // 3) select bip32 tab
1887 driver
.findElement(By
.css('#bip32-tab a'))
1889 driver
.sleep(generateDelay
).then(function() {
1890 // 4) switch from bitcoin to litecoin
1891 selectNetwork("LTC - Litecoin");
1892 driver
.sleep(generateDelay
).then(function() {
1893 // 5) Check address is displayed correctly
1894 getFirstAddress(function(address
) {
1895 expect(address
).toBe("LS8MP5LZ5AdzSZveRrjm3aYVoPgnfFh5T5");
1896 // 5) Check derivation path is displayed correctly
1897 getFirstPath(function(path
) {
1898 expect(path
).toBe("m/0/0");
1907 // Github Issue 23 Part 2: Coin selection in derivation path
1908 // https://github.com/iancoleman/bip39/issues/23#issuecomment-238011920
1909 it('Uses the correct derivation path when changing coins', function(done
) {
1911 driver
.findElement(By
.css('.phrase'))
1912 .sendKeys("abandon abandon ability");
1913 driver
.sleep(generateDelay
).then(function() {
1914 // switch from bitcoin to clam
1915 selectNetwork("CLAM - Clams");
1916 driver
.sleep(generateDelay
).then(function() {
1917 // check derivation path is displayed correctly
1918 getFirstPath(function(path
) {
1919 expect(path
).toBe("m/44'/23'/0'/0/0");
1926 // Github Issue 26: When using a Root key derrived altcoins are incorrect
1927 // https://github.com/iancoleman/bip39/issues/26
1928 it('Uses the correct derivation for altcoins with root keys', function(done
) {
1929 // 1) 2) and 3) set the root key
1930 driver
.findElement(By
.css('.root-key'))
1931 .sendKeys("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
1932 driver
.sleep(generateDelay
).then(function() {
1933 // 4) switch from bitcoin to viacoin
1934 selectNetwork("VIA - Viacoin");
1935 driver
.sleep(generateDelay
).then(function() {
1936 // 5) ensure the derived address is correct
1937 getFirstAddress(function(address
) {
1938 expect(address
).toBe("Vq9Eq4N5SQnjqZvxtxzo7hZPW5XnyJsmXT");
1945 // Selecting a language with no existing phrase should generate a phrase in
1947 it('Generate a random phrase when language is selected and no current phrase', function(done
) {
1948 driver
.findElement(By
.css("a[href='#japanese']"))
1950 driver
.sleep(generateDelay
).then(function() {
1951 driver
.findElement(By
.css(".phrase"))
1952 .getAttribute("value").then(function(phrase
) {
1953 expect(phrase
.search(/[a-z]/)).toBe(-1);
1954 expect(phrase
.length
).toBeGreaterThan(0);
1960 // Selecting a language with existing phrase should update the phrase to use
1962 it('Updates existing phrases when the language is changed', function(done
) {
1963 driver
.findElement(By
.css(".phrase"))
1964 .sendKeys("abandon abandon ability");
1965 driver
.sleep(generateDelay
).then(function() {
1966 driver
.findElement(By
.css("a[href='#italian']"))
1968 driver
.sleep(generateDelay
).then(function() {
1969 driver
.findElement(By
.css(".phrase"))
1970 .getAttribute("value").then(function(phrase
) {
1971 // Check only the language changes, not the phrase
1972 expect(phrase
).toBe("abaco abaco abbaglio");
1973 getFirstAddress(function(address
) {
1974 // Check the address is correct
1975 expect(address
).toBe("1Dz5TgDhdki9spa6xbPFbBqv5sjMrx3xgV");
1983 // Suggested replacement for erroneous word in non-English language
1984 it('Shows word suggestion for incorrect word in non-English language', function(done
) {
1985 driver
.findElement(By
.css('.phrase'))
1986 .sendKeys('abaco abaco zbbaglio');
1987 driver
.sleep(feedbackDelay
).then(function() {
1988 driver
.findElement(By
.css('.feedback'))
1990 .then(function(feedback
) {
1991 var msg
= "zbbaglio not in wordlist, did you mean abbaglio?";
1992 expect(feedback
).toBe(msg
);
1998 // Japanese word does not break across lines.
2000 // https://github.com/bitcoin/bips/blob/master/bip-0039/bip-0039-wordlists.md#japanese
2001 it('Does not break Japanese words across lines', function(done
) {
2002 driver
.findElement(By
.css('.phrase'))
2003 .getCssValue("word-break")
2004 .then(function(value
) {
2005 expect(value
).toBe("keep-all");
2010 // Language can be specified at page load using hash value in url
2011 it('Can set the language from the url hash', function(done
) {
2012 driver
.get(url
+ "#japanese").then(function() {
2013 driver
.findElement(By
.css('.generate')).click();
2014 driver
.sleep(generateDelay
).then(function() {
2015 driver
.findElement(By
.css(".phrase"))
2016 .getAttribute("value").then(function(phrase
) {
2017 expect(phrase
.search(/[a-z]/)).toBe(-1);
2018 expect(phrase
.length
).toBeGreaterThan(0);
2025 // Entropy can be entered by the user
2026 it('Allows entropy to be entered', function(done
) {
2027 driver
.findElement(By
.css('.use-entropy'))
2029 driver
.findElement(By
.css('.entropy'))
2030 .sendKeys('00000000 00000000 00000000 00000000');
2031 driver
.sleep(generateDelay
).then(function() {
2032 driver
.findElement(By
.css(".phrase"))
2033 .getAttribute("value").then(function(phrase
) {
2034 expect(phrase
).toBe("abandon abandon ability");
2035 getFirstAddress(function(address
) {
2036 expect(address
).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug");
2043 // A warning about entropy is shown to the user, with additional information
2044 it('Shows a warning about using entropy', function(done
) {
2045 driver
.findElement(By
.css('.use-entropy'))
2047 driver
.findElement(By
.css('.entropy-container'))
2049 .then(function(containerText
) {
2050 var warning
= "mnemonic may be insecure";
2051 expect(containerText
).toContain(warning
);
2052 driver
.findElement(By
.css('#entropy-notes'))
2053 .findElement(By
.xpath("parent::*"))
2055 .then(function(notesText
) {
2056 var detail
= "flipping a fair coin, rolling a fair dice, noise measurements etc";
2057 expect(notesText
).toContain(detail
);
2063 // The types of entropy available are described to the user
2064 it('Shows the types of entropy available', function(done
) {
2065 driver
.findElement(By
.css('.entropy'))
2066 .getAttribute("placeholder")
2067 .then(function(placeholderText
) {
2076 for (var i
=0; i
<options
.length
; i
++) {
2077 var option
= options
[i
];
2078 expect(placeholderText
).toContain(option
);
2084 // The actual entropy used is shown to the user
2085 it('Shows the actual entropy used', function(done
) {
2086 driver
.findElement(By
.css('.use-entropy'))
2088 driver
.findElement(By
.css('.entropy'))
2089 .sendKeys('Not A Very Good Entropy Source At All');
2090 driver
.sleep(generateDelay
).then(function() {
2091 driver
.findElement(By
.css('.entropy-container'))
2093 .then(function(text
) {
2094 expect(text
).toMatch(/Filtered Entropy
\s
+AedEceAA
/);
2100 // Binary entropy can be entered
2101 it('Allows binary entropy to be entered', function(done
) {
2102 testEntropyType(done
, "01", "binary");
2105 // Base 6 entropy can be entered
2106 it('Allows base 6 entropy to be entered', function(done
) {
2107 testEntropyType(done
, "012345", "base 6");
2110 // Base 6 dice entropy can be entered
2111 it('Allows base 6 dice entropy to be entered', function(done
) {
2112 testEntropyType(done
, "123456", "base 6 (dice)");
2115 // Base 10 entropy can be entered
2116 it('Allows base 10 entropy to be entered', function(done
) {
2117 testEntropyType(done
, "789", "base 10");
2120 // Hexadecimal entropy can be entered
2121 it('Allows hexadecimal entropy to be entered', function(done
) {
2122 testEntropyType(done
, "abcdef", "hexadecimal");
2125 // Dice entropy value is shown as the converted base 6 value
2126 // ie 123456 is converted to 123450
2127 it('Shows dice entropy as base 6', function(done
) {
2128 driver
.findElement(By
.css('.use-entropy'))
2130 driver
.findElement(By
.css('.entropy'))
2131 .sendKeys("123456");
2132 driver
.sleep(generateDelay
).then(function() {
2133 driver
.findElement(By
.css('.entropy-container'))
2135 .then(function(text
) {
2136 expect(text
).toMatch(/Filtered Entropy
\s
+123450/);
2142 // The number of bits of entropy accumulated is shown
2143 it("Shows the number of bits of entropy for 20 bits of binary", function(done
) {
2144 testEntropyBits(done
, "0000 0000 0000 0000 0000", "20");
2146 it("Shows the number of bits of entropy for 1 bit of binary", function(done
) {
2147 testEntropyBits(done
, "0", "1");
2149 it("Shows the number of bits of entropy for 4 bits of binary", function(done
) {
2150 testEntropyBits(done
, "0000", "4");
2152 it("Shows the number of bits of entropy for 1 character of base 6 (dice)", function(done
) {
2153 // 6 in card is 0 in base 6, 0 in base 6 is 2.6 bits (rounded down to 2 bits)
2154 testEntropyBits(done
, "6", "2");
2156 it("Shows the number of bits of entropy for 1 character of base 10 with 3 bits", function(done
) {
2157 // 7 in base 10 is 111 in base 2, no leading zeros
2158 testEntropyBits(done
, "7", "3");
2160 it("Shows the number of bits of entropy for 1 character of base 10 with 4 bis", function(done
) {
2161 testEntropyBits(done
, "8", "4");
2163 it("Shows the number of bits of entropy for 1 character of hex", function(done
) {
2164 testEntropyBits(done
, "F", "4");
2166 it("Shows the number of bits of entropy for 2 characters of base 10", function(done
) {
2167 testEntropyBits(done
, "29", "6");
2169 it("Shows the number of bits of entropy for 2 characters of hex", function(done
) {
2170 testEntropyBits(done
, "0A", "8");
2172 it("Shows the number of bits of entropy for 2 characters of hex with 3 leading zeros", function(done
) {
2173 // hex is always multiple of 4 bits of entropy
2174 testEntropyBits(done
, "1A", "8");
2176 it("Shows the number of bits of entropy for 2 characters of hex with 2 leading zeros", function(done
) {
2177 testEntropyBits(done
, "2A", "8");
2179 it("Shows the number of bits of entropy for 2 characters of hex with 1 leading zero", function(done
) {
2180 testEntropyBits(done
, "4A", "8");
2182 it("Shows the number of bits of entropy for 2 characters of hex with no leading zeros", function(done
) {
2183 testEntropyBits(done
, "8A", "8");
2185 it("Shows the number of bits of entropy for 2 characters of hex starting with F", function(done
) {
2186 testEntropyBits(done
, "FA", "8");
2188 it("Shows the number of bits of entropy for 4 characters of hex with leading zeros", function(done
) {
2189 testEntropyBits(done
, "000A", "16");
2191 it("Shows the number of bits of entropy for 4 characters of base 6", function(done
) {
2192 testEntropyBits(done
, "5555", "11");
2194 it("Shows the number of bits of entropy for 4 characters of base 6 dice", function(done
) {
2195 // uses dice, so entropy is actually 0000 in base 6, which is 4 lots of
2196 // 2.58 bits, which is 10.32 bits (rounded down to 10 bits)
2197 testEntropyBits(done
, "6666", "10");
2199 it("Shows the number of bits of entropy for 4 charactes of base 10", function(done
) {
2200 // Uses base 10, which is 4 lots of 3.32 bits, which is 13.3 bits (rounded
2202 testEntropyBits(done
, "2227", "13");
2204 it("Shows the number of bits of entropy for 4 characters of hex with 2 leading zeros", function(done
) {
2205 testEntropyBits(done
, "222F", "16");
2207 it("Shows the number of bits of entropy for 4 characters of hex starting with F", function(done
) {
2208 testEntropyBits(done
, "FFFF", "16");
2210 it("Shows the number of bits of entropy for 10 characters of base 10", function(done
) {
2211 // 10 events at 3.32 bits per event
2212 testEntropyBits(done
, "0000101017", "33");
2214 it("Shows the number of bits of entropy for a full deck of cards", function(done
) {
2215 // cards are not replaced, so a full deck is not 52^52 entropy which is 296
2216 // bits, it's 52!, which is 225 bits
2217 testEntropyBits(done
, "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks", "225");
2220 it("Shows details about the entered entropy", function(done
) {
2221 testEntropyFeedback(done
,
2225 type: "hexadecimal",
2229 strength: "less than a second",
2233 it("Shows details about the entered entropy", function(done
) {
2234 testEntropyFeedback(done
,
2236 entropy: "AAAAAAAA",
2237 filtered: "AAAAAAAA",
2238 type: "hexadecimal",
2242 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2246 it("Shows details about the entered entropy", function(done
) {
2247 testEntropyFeedback(done
,
2249 entropy: "AAAAAAAA B",
2250 filtered: "AAAAAAAAB",
2251 type: "hexadecimal",
2255 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2259 it("Shows details about the entered entropy", function(done
) {
2260 testEntropyFeedback(done
,
2262 entropy: "AAAAAAAA BBBBBBBB",
2263 filtered: "AAAAAAAABBBBBBBB",
2264 type: "hexadecimal",
2268 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2272 it("Shows details about the entered entropy", function(done
) {
2273 testEntropyFeedback(done
,
2275 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC",
2276 filtered: "AAAAAAAABBBBBBBBCCCCCCCC",
2277 type: "hexadecimal",
2281 strength: "less than a second",
2285 it("Shows details about the entered entropy", function(done
) {
2286 testEntropyFeedback(done
,
2288 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDD",
2289 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDD",
2290 type: "hexadecimal",
2294 strength: "2 minutes",
2298 it("Shows details about the entered entropy", function(done
) {
2299 testEntropyFeedback(done
,
2301 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA",
2302 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDA",
2303 type: "hexadecimal",
2311 it("Shows details about the entered entropy", function(done
) {
2312 testEntropyFeedback(done
,
2314 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA EEEEEEEE",
2315 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDAEEEEEEEE",
2316 type: "hexadecimal",
2320 strength: "3 years",
2324 it("Shows details about the entered entropy", function(done
) {
2325 testEntropyFeedback(done
,
2327 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA EEEEEEEE FFFFFFFF",
2328 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDAEEEEEEEEFFFFFFFF",
2329 type: "hexadecimal",
2333 strength: "centuries",
2337 it("Shows details about the entered entropy", function(done
) {
2338 testEntropyFeedback(done
,
2345 strength: "less than a second",
2349 it("Shows details about the entered entropy", function(done
) {
2350 testEntropyFeedback(done
,
2352 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2353 type: "card (full deck)",
2357 strength: "centuries",
2361 it("Shows details about the entered entropy", function(done
) {
2362 testEntropyFeedback(done
,
2364 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks3d",
2365 type: "card (full deck, 1 duplicate: 3d)",
2369 strength: "centuries",
2373 it("Shows details about the entered entropy", function(done
) {
2374 testEntropyFeedback(done
,
2376 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqs3d4d",
2377 type: "card (2 duplicates: 3d 4d, 1 missing: KS)",
2381 strength: "centuries",
2385 it("Shows details about the entered entropy", function(done
) {
2386 testEntropyFeedback(done
,
2388 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqs3d4d5d6d",
2389 type: "card (4 duplicates: 3d 4d 5d..., 1 missing: KS)",
2393 strength: "centuries",
2397 it("Shows details about the entered entropy", function(done
) {
2398 testEntropyFeedback(done
,
2399 // Next test was throwing uncaught error in zxcvbn
2400 // Also tests 451 bits, ie Math.log2(52!)*2 = 225.58 * 2
2402 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsksac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2403 type: "card (full deck, 52 duplicates: ac 2c 3c...)",
2407 strength: "centuries",
2411 it("Shows details about the entered entropy", function(done
) {
2412 testEntropyFeedback(done
,
2413 // Case insensitivity to duplicate cards
2416 type: "card (1 duplicate: AS)",
2420 strength: "less than a second",
2424 it("Shows details about the entered entropy", function(done
) {
2425 testEntropyFeedback(done
,
2428 type: "card (1 duplicate: as)",
2432 strength: "less than a second",
2436 it("Shows details about the entered entropy", function(done
) {
2437 testEntropyFeedback(done
,
2438 // Missing cards are detected
2440 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2441 type: "card (1 missing: 9C)",
2445 strength: "centuries",
2449 it("Shows details about the entered entropy", function(done
) {
2450 testEntropyFeedback(done
,
2452 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2453 type: "card (2 missing: 9C 5D)",
2457 strength: "centuries",
2461 it("Shows details about the entered entropy", function(done
) {
2462 testEntropyFeedback(done
,
2464 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d7d8d9dtdjd kdah2h3h 5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2465 type: "card (4 missing: 9C 5D QD...)",
2469 strength: "centuries",
2473 it("Shows details about the entered entropy", function(done
) {
2474 testEntropyFeedback(done
,
2475 // More than six missing cards does not show message
2477 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d 8d9d jd kdah2h3h 5h6h7h8h9hthjhqhkh 2s3s4s5s6s7s8s9stsjsqsks",
2482 strength: "centuries",
2486 it("Shows details about the entered entropy", function(done
) {
2487 testEntropyFeedback(done
,
2488 // Multiple decks of cards increases bits per event
2493 bitsPerEvent: "4.34",
2497 it("Shows details about the entered entropy", function(done
) {
2498 testEntropyFeedback(done
,
2503 bitsPerEvent: "4.80",
2507 it("Shows details about the entered entropy", function(done
) {
2508 testEntropyFeedback(done
,
2513 bitsPerEvent: "5.01",
2517 it("Shows details about the entered entropy", function(done
) {
2518 testEntropyFeedback(done
,
2520 entropy: "3d3d3d3d",
2523 bitsPerEvent: "5.14",
2527 it("Shows details about the entered entropy", function(done
) {
2528 testEntropyFeedback(done
,
2530 entropy: "3d3d3d3d3d",
2533 bitsPerEvent: "5.22",
2537 it("Shows details about the entered entropy", function(done
) {
2538 testEntropyFeedback(done
,
2540 entropy: "3d3d3d3d3d3d",
2543 bitsPerEvent: "5.28",
2547 it("Shows details about the entered entropy", function(done
) {
2548 testEntropyFeedback(done
,
2550 entropy: "3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d",
2553 bitsPerEvent: "5.59",
2554 strength: 'less than a second - Repeats like "abcabcabc" are only slightly harder to guess than "abc"',
2559 // Entropy is truncated from the left
2560 it('Truncates entropy from the left', function(done
) {
2561 // Truncate from left means 0000 is removed from the start
2562 // which gives mnemonic 'avocado zoo zone'
2563 // not 1111 removed from the end
2564 // which gives the mnemonic 'abstract zoo zoo'
2565 var entropy
= "00000000 00000000 00000000 00000000";
2566 entropy
+= "11111111 11111111 11111111 1111"; // Missing last byte
2567 driver
.findElement(By
.css('.use-entropy'))
2569 driver
.findElement(By
.css('.entropy'))
2571 driver
.sleep(generateDelay
).then(function() {
2572 driver
.findElement(By
.css(".phrase"))
2573 .getAttribute("value").then(function(phrase
) {
2574 expect(phrase
).toBe("avocado zoo zone");
2580 // Very large entropy results in very long mnemonics
2581 it('Converts very long entropy to very long mnemonics', function(done
) {
2583 for (var i
=0; i
<33; i
++) {
2584 entropy
+= "AAAAAAAA"; // 3 words * 33 iterations = 99 words
2586 driver
.findElement(By
.css('.use-entropy'))
2588 driver
.findElement(By
.css('.entropy'))
2590 driver
.sleep(generateDelay
).then(function() {
2591 driver
.findElement(By
.css(".phrase"))
2592 .getAttribute("value").then(function(phrase
) {
2593 var wordCount
= phrase
.split(/\s+/g).length
;
2594 expect(wordCount
).toBe(99);
2600 // Is compatible with bip32jp entropy
2601 // https://bip32jp.github.io/english/index.html
2603 // Is incompatible with:
2605 it('Is compatible with bip32jp.github.io', function(done
) {
2606 var entropy
= "543210543210543210543210543210543210543210543210543210543210543210543210543210543210543210543210543";
2607 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";
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 expect(phrase
).toBe(expectedPhrase
);
2621 // Blank entropy does not generate mnemonic or addresses
2622 it('Does not generate mnemonic for blank entropy', function(done
) {
2623 driver
.findElement(By
.css('.use-entropy'))
2625 driver
.findElement(By
.css('.entropy'))
2627 // check there is no mnemonic
2628 driver
.sleep(generateDelay
).then(function() {
2629 driver
.findElement(By
.css(".phrase"))
2630 .getAttribute("value").then(function(phrase
) {
2631 expect(phrase
).toBe("");
2632 // check there is no mnemonic
2633 driver
.findElements(By
.css(".address"))
2634 .then(function(addresses
) {
2635 expect(addresses
.length
).toBe(0);
2636 // Check the feedback says 'blank entropy'
2637 driver
.findElement(By
.css(".feedback"))
2639 .then(function(feedbackText
) {
2640 expect(feedbackText
).toBe("Blank entropy");
2648 // Mnemonic length can be selected even for weak entropy
2649 it('Allows selection of mnemonic length even for weak entropy', function(done
) {
2650 driver
.findElement(By
.css('.use-entropy'))
2652 driver
.executeScript(function() {
2653 $(".mnemonic-length").val("18").trigger("change");
2655 driver
.findElement(By
.css('.entropy'))
2656 .sendKeys("012345");
2657 driver
.sleep(generateDelay
).then(function() {
2658 driver
.findElement(By
.css(".phrase"))
2659 .getAttribute("value").then(function(phrase
) {
2660 var wordCount
= phrase
.split(/\s+/g).length
;
2661 expect(wordCount
).toBe(18);
2668 // https://github.com/iancoleman/bip39/issues/33
2669 // Final cards should contribute entropy
2670 it('Uses as much entropy as possible for the mnemonic', function(done
) {
2671 driver
.findElement(By
.css('.use-entropy'))
2673 driver
.findElement(By
.css('.entropy'))
2674 .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");
2675 driver
.sleep(generateDelay
).then(function() {
2677 driver
.findElement(By
.css(".phrase"))
2678 .getAttribute("value").then(function(originalPhrase
) {
2679 // Set the last 12 cards to be AS
2680 driver
.findElement(By
.css('.entropy'))
2682 driver
.findElement(By
.css('.entropy'))
2683 .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");
2684 driver
.sleep(generateDelay
).then(function() {
2686 driver
.findElement(By
.css(".phrase"))
2687 .getAttribute("value").then(function(newPhrase
) {
2688 expect(originalPhrase
).not
.toEqual(newPhrase
);
2697 // https://github.com/iancoleman/bip39/issues/35
2699 // TODO this doesn't work in selenium with firefox
2700 // see https://stackoverflow.com/q/40360223
2701 it('Shows a qr code on hover for the phrase', function(done
) {
2702 if (browser
== "firefox") {
2703 pending("Selenium + Firefox bug for mouseMove, see https://stackoverflow.com/q/40360223");
2705 // generate a random mnemonic
2706 var generateEl
= driver
.findElement(By
.css('.generate'));
2708 // toggle qr to show (hidden by default)
2709 var phraseEl
= driver
.findElement(By
.css(".phrase"));
2711 var rootKeyEl
= driver
.findElement(By
.css(".root-key"));
2712 driver
.sleep(generateDelay
).then(function() {
2713 // hover over the root key
2714 driver
.actions().mouseMove(rootKeyEl
).perform().then(function() {
2715 // check the qr code shows
2716 driver
.executeScript(function() {
2717 return $(".qr-container").find("canvas").length
> 0;
2719 .then(function(qrShowing
) {
2720 expect(qrShowing
).toBe(true);
2721 // hover away from the phrase
2722 driver
.actions().mouseMove(generateEl
).perform().then(function() {;
2723 // check the qr code hides
2724 driver
.executeScript(function() {
2725 return $(".qr-container").find("canvas").length
== 0;
2727 .then(function(qrHidden
) {
2728 expect(qrHidden
).toBe(true);
2737 // BIP44 account extendend private key is shown
2738 // github issue 37 - compatibility with electrum
2739 it('Shows the bip44 account extended private key', function(done
) {
2740 driver
.findElement(By
.css(".phrase"))
2741 .sendKeys("abandon abandon ability");
2742 driver
.sleep(generateDelay
).then(function() {
2743 driver
.findElement(By
.css("#bip44 .account-xprv"))
2744 .getAttribute("value")
2745 .then(function(xprv
) {
2746 expect(xprv
).toBe("xprv9yzrnt4zWVJUr1k2VxSPy9ettKz5PpeDMgaVG7UKedhqnw1tDkxP2UyYNhuNSumk2sLE5ctwKZs9vwjsq3e1vo9egCK6CzP87H2cVYXpfwQ");
2752 // BIP44 account extendend public key is shown
2753 // github issue 37 - compatibility with electrum
2754 it('Shows the bip44 account extended public key', function(done
) {
2755 driver
.findElement(By
.css(".phrase"))
2756 .sendKeys("abandon abandon ability");
2757 driver
.sleep(generateDelay
).then(function() {
2758 driver
.findElement(By
.css("#bip44 .account-xpub"))
2759 .getAttribute("value")
2760 .then(function(xprv
) {
2761 expect(xprv
).toBe("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2768 // BIP32 root key can be set as an xpub
2769 it('Generates addresses from xpub as bip32 root key', function(done
) {
2770 driver
.findElement(By
.css('#bip32-tab a'))
2772 // set xpub for account 0 of bip44 for 'abandon abandon ability'
2773 driver
.findElement(By
.css("#root-key"))
2774 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2775 driver
.sleep(generateDelay
).then(function() {
2776 // check the addresses are generated
2777 getFirstAddress(function(address
) {
2778 expect(address
).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug");
2779 // check the xprv key is not set
2780 driver
.findElement(By
.css(".extended-priv-key"))
2781 .getAttribute("value")
2782 .then(function(xprv
) {
2783 expect(xprv
).toBe("NA");
2784 // check the private key is not set
2785 driver
.findElements(By
.css(".privkey"))
2786 .then(function(els
) {
2789 .then(function(privkey
) {
2790 expect(xprv
).toBe("NA");
2800 // xpub for bip32 root key will not work with hardened derivation paths
2801 it('Shows error for hardened derivation paths with xpub root key', function(done
) {
2802 // set xpub for account 0 of bip44 for 'abandon abandon ability'
2803 driver
.findElement(By
.css("#root-key"))
2804 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2805 driver
.sleep(feedbackDelay
).then(function() {
2806 // Check feedback is correct
2807 driver
.findElement(By
.css('.feedback'))
2809 .then(function(feedback
) {
2810 var msg
= "Hardened derivation path is invalid with xpub key";
2811 expect(feedback
).toBe(msg
);
2812 // Check no addresses are shown
2813 driver
.findElements(By
.css('.addresses tr'))
2814 .then(function(rows
) {
2815 expect(rows
.length
).toBe(0);
2823 // no root key shows feedback
2824 it('Shows feedback for no root key', function(done
) {
2825 // set xpub for account 0 of bip44 for 'abandon abandon ability'
2826 driver
.findElement(By
.css('#bip32-tab a'))
2828 driver
.sleep(feedbackDelay
).then(function() {
2829 // Check feedback is correct
2830 driver
.findElement(By
.css('.feedback'))
2832 .then(function(feedback
) {
2833 expect(feedback
).toBe("Invalid root key");
2840 // display error switching tabs while addresses are generating
2841 it('Can change details while old addresses are still being generated', function(done
) {
2842 // Set to generate 199 more addresses.
2843 // This will take a long time allowing a new set of addresses to be
2844 // generated midway through this lot.
2845 // The newly generated addresses should not include any from the old set.
2846 // Any more than 199 will show an alert which needs to be accepted.
2847 driver
.findElement(By
.css('.rows-to-add'))
2849 driver
.findElement(By
.css('.rows-to-add'))
2852 driver
.findElement(By
.css('.phrase'))
2853 .sendKeys("abandon abandon ability");
2854 driver
.sleep(generateDelay
).then(function() {
2855 // change tabs which should cancel the previous generating
2856 driver
.findElement(By
.css('.rows-to-add'))
2858 driver
.findElement(By
.css('.rows-to-add'))
2860 driver
.findElement(By
.css('#bip32-tab a'))
2862 driver
.sleep(generateDelay
).then(function() {
2863 driver
.findElements(By
.css('.index'))
2864 .then(function(els
) {
2865 // check the derivation paths have the right quantity
2866 expect(els
.length
).toBe(20);
2867 // check the derivation paths are in order
2868 testRowsAreInCorrectOrder(done
);
2872 }, generateDelay
+ 5000);
2875 // padding for binary should give length with multiple of 256
2876 // hashed entropy 1111 is length 252, so requires 4 leading zeros
2877 // prior to issue 49 it would only generate 2 leading zeros, ie missing 2
2878 it('Pads hashed entropy with leading zeros', function(done
) {
2879 driver
.findElement(By
.css('.use-entropy'))
2881 driver
.executeScript(function() {
2882 $(".mnemonic-length").val("15").trigger("change");
2884 driver
.findElement(By
.css('.entropy'))
2886 driver
.sleep(generateDelay
).then(function() {
2887 driver
.findElement(By
.css('.phrase'))
2888 .getAttribute("value")
2889 .then(function(phrase
) {
2890 expect(phrase
).toBe("avocado valid quantum cross link predict excuse edit street able flame large galaxy ginger nuclear");
2896 // Github pull request 55
2897 // https://github.com/iancoleman/bip39/pull/55
2899 it('Can set the derivation path on bip32 tab for bitcoincore', function(done
) {
2900 testClientSelect(done
, {
2902 bip32path: "m/0'/0'",
2903 useHardenedAddresses: "true",
2906 it('Can set the derivation path on bip32 tab for multibit', function(done
) {
2907 testClientSelect(done
, {
2909 bip32path: "m/0'/0",
2910 useHardenedAddresses: null,
2913 it('Can set the derivation path on bip32 tab for coinomi/ledger', function(done
) {
2914 testClientSelect(done
, {
2916 bip32path: "m/44'/0'/0'",
2917 useHardenedAddresses: null,
2922 // https://github.com/iancoleman/bip39/issues/58
2923 // bip32 derivation is correct, does not drop leading zeros
2925 // https://medium.com/@alexberegszaszi/why-do-my-bip32-wallets-disagree-6f3254cc5846
2926 it('Retains leading zeros for bip32 derivation', function(done
) {
2927 driver
.findElement(By
.css(".phrase"))
2928 .sendKeys("fruit wave dwarf banana earth journey tattoo true farm silk olive fence");
2929 driver
.findElement(By
.css(".passphrase"))
2930 .sendKeys("banana");
2931 driver
.sleep(generateDelay
).then(function() {
2932 getFirstAddress(function(address
) {
2933 // Note that bitcore generates an incorrect address
2934 // 13EuKhffWkBE2KUwcbkbELZb1MpzbimJ3Y
2935 // see the medium.com link above for more details
2936 expect(address
).toBe("17rxURoF96VhmkcEGCj5LNQkmN9HVhWb7F");
2943 // Japanese mnemonics generate incorrect bip32 seed
2944 // BIP39 seed is set from phrase
2945 it('Generates correct seed for Japanese mnemonics', function(done
) {
2946 driver
.findElement(By
.css(".phrase"))
2947 .sendKeys("あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あおぞら");
2948 driver
.findElement(By
.css(".passphrase"))
2949 .sendKeys("メートルガバヴァぱばぐゞちぢ十人十色");
2950 driver
.sleep(generateDelay
).then(function() {
2951 driver
.findElement(By
.css(".seed"))
2952 .getAttribute("value")
2953 .then(function(seed
) {
2954 expect(seed
).toBe("a262d6fb6122ecf45be09c50492b31f92e9beb7d9a845987a02cefda57a15f9c467a17872029a9e92299b5cbdf306e3a0ee620245cbd508959b6cb7ca637bd55");
2960 // BIP49 official test vectors
2961 // https://github.com/bitcoin/bips/blob/master/bip-0049.mediawiki#test-vectors
2962 it('Generates BIP49 addresses matching the official test vectors', function(done
) {
2963 driver
.findElement(By
.css('#bip49-tab a'))
2965 selectNetwork("BTC - Bitcoin Testnet");
2966 driver
.findElement(By
.css(".phrase"))
2967 .sendKeys("abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about");
2968 driver
.sleep(generateDelay
).then(function() {
2969 getFirstAddress(function(address
) {
2970 expect(address
).toBe("2Mww8dCYPUpKHofjgcXcBCEGmniw9CoaiD2");
2976 // BIP49 derivation path is shown
2977 it('Shows the bip49 derivation path', function(done
) {
2978 driver
.findElement(By
.css('#bip49-tab a'))
2980 driver
.findElement(By
.css(".phrase"))
2981 .sendKeys("abandon abandon ability");
2982 driver
.sleep(generateDelay
).then(function() {
2983 driver
.findElement(By
.css('#bip49 .path'))
2984 .getAttribute("value")
2985 .then(function(path
) {
2986 expect(path
).toBe("m/49'/0'/0'/0");
2992 // BIP49 extended private key is shown
2993 it('Shows the bip49 extended private key', function(done
) {
2994 driver
.findElement(By
.css('#bip49-tab a'))
2996 driver
.findElement(By
.css(".phrase"))
2997 .sendKeys("abandon abandon ability");
2998 driver
.sleep(generateDelay
).then(function() {
2999 driver
.findElement(By
.css('.extended-priv-key'))
3000 .getAttribute("value")
3001 .then(function(xprv
) {
3002 expect(xprv
).toBe("yprvALYB4DYRG6CzzVgzQZwwqjAA2wjBGC3iEd7KYYScpoDdmf75qMRWZWxoFcRXBJjgEXdFqJ9vDRGRLJQsrL22Su5jMbNFeM9vetaGVqy9Qy2");
3008 // BIP49 extended public key is shown
3009 it('Shows the bip49 extended public key', function(done
) {
3010 driver
.findElement(By
.css('#bip49-tab a'))
3012 driver
.findElement(By
.css(".phrase"))
3013 .sendKeys("abandon abandon ability");
3014 driver
.sleep(generateDelay
).then(function() {
3015 driver
.findElement(By
.css('.extended-pub-key'))
3016 .getAttribute("value")
3017 .then(function(xprv
) {
3018 expect(xprv
).toBe("ypub6ZXXTj5K6TmJCymTWbUxCs6tayZffemZbr2vLvrEP8kceTSENtjm7KHH6thvAKxVar9fGe8rgsPEX369zURLZ68b4f7Vexz7RuXsjQ69YDt");
3024 // BIP49 account field changes address list
3025 it('Can set the bip49 account field', function(done
) {
3026 driver
.findElement(By
.css('#bip49-tab a'))
3028 driver
.findElement(By
.css('#bip49 .account'))
3030 driver
.findElement(By
.css('#bip49 .account'))
3032 driver
.findElement(By
.css(".phrase"))
3033 .sendKeys("abandon abandon ability");
3034 driver
.sleep(generateDelay
).then(function() {
3035 getFirstAddress(function(address
) {
3036 expect(address
).toBe("381wg1GGN4rP88rNC9v7QWsiww63yLVPsn");
3042 // BIP49 change field changes address list
3043 it('Can set the bip49 change field', function(done
) {
3044 driver
.findElement(By
.css('#bip49-tab a'))
3046 driver
.findElement(By
.css('#bip49 .change'))
3048 driver
.findElement(By
.css('#bip49 .change'))
3050 driver
.findElement(By
.css(".phrase"))
3051 .sendKeys("abandon abandon ability");
3052 driver
.sleep(generateDelay
).then(function() {
3053 getFirstAddress(function(address
) {
3054 expect(address
).toBe("3PEM7MiKed5konBoN66PQhK8r3hjGhy9dT");
3060 // BIP49 account extendend private key is shown
3061 it('Shows the bip49 account extended private key', function(done
) {
3062 driver
.findElement(By
.css('#bip49-tab a'))
3064 driver
.findElement(By
.css(".phrase"))
3065 .sendKeys("abandon abandon ability");
3066 driver
.sleep(generateDelay
).then(function() {
3067 driver
.findElement(By
.css('#bip49 .account-xprv'))
3068 .getAttribute("value")
3069 .then(function(xprv
) {
3070 expect(xprv
).toBe("yprvAHtB1M5Wp675aLzFy9TJYK2mSsLkg6mcBRh5DZTR7L4EnYSmYPaL63KFA4ycg1PngW5LfkmejxzosCs17TKZMpRFKc3z5SJar6QAKaFcaZL");
3076 // BIP49 account extendend public key is shown
3077 it('Shows the bip49 account extended public key', function(done
) {
3078 driver
.findElement(By
.css('#bip49-tab a'))
3080 driver
.findElement(By
.css(".phrase"))
3081 .sendKeys("abandon abandon ability");
3082 driver
.sleep(generateDelay
).then(function() {
3083 driver
.findElement(By
.css('#bip49 .account-xpub'))
3084 .getAttribute("value")
3085 .then(function(xprv
) {
3086 expect(xprv
).toBe("ypub6WsXQrcQeTfNnq4j5AzJuSyVzuBF5ZVTYecg1ws2ffbDfLmv5vtadqdj1NgR6C6gufMpMfJpHxvb6JEQKvETVNWCRanNedfJtnTchZiJtsL");
3092 // Test selecting coin where bip49 is unavailable (eg CLAM)
3093 it('Shows an error on bip49 tab for coins without bip49', function(done
) {
3094 driver
.findElement(By
.css('#bip49-tab a'))
3096 driver
.findElement(By
.css(".phrase"))
3097 .sendKeys("abandon abandon ability");
3098 driver
.sleep(generateDelay
).then(function() {
3099 selectNetwork("CLAM - Clams");
3100 // bip49 available is hidden
3101 driver
.findElement(By
.css('#bip49 .available'))
3102 .getAttribute("class")
3103 .then(function(classes
) {
3104 expect(classes
).toContain("hidden");
3105 // bip49 unavailable is shown
3106 driver
.findElement(By
.css('#bip49 .unavailable'))
3107 .getAttribute("class")
3108 .then(function(classes
) {
3109 expect(classes
).not
.toContain("hidden");
3110 // check there are no addresses shown
3111 driver
.findElements(By
.css('.addresses tr'))
3112 .then(function(rows
) {
3113 expect(rows
.length
).toBe(0);
3114 // check the derived private key is blank
3115 driver
.findElement(By
.css('.extended-priv-key'))
3116 .getAttribute("value")
3117 .then(function(xprv
) {
3118 expect(xprv
).toBe('');
3119 // check the derived public key is blank
3120 driver
.findElement(By
.css('.extended-pub-key'))
3121 .getAttribute("value")
3122 .then(function(xpub
) {
3123 expect(xpub
).toBe('');
3134 // Cleared mnemonic and root key still allows addresses to be generated
3135 // https://github.com/iancoleman/bip39/issues/43
3136 it('Clears old root keys from memory when mnemonic is cleared', function(done
) {
3138 driver
.findElement(By
.css(".phrase"))
3139 .sendKeys("abandon abandon ability");
3140 driver
.sleep(generateDelay
).then(function() {
3141 // clear the mnemonic and root key
3142 // using selenium .clear() doesn't seem to trigger the 'input' event
3143 // so clear it using keys instead
3144 driver
.findElement(By
.css('.phrase'))
3145 .sendKeys(Key
.CONTROL
,"a");
3146 driver
.findElement(By
.css('.phrase'))
3147 .sendKeys(Key
.DELETE
);
3148 driver
.findElement(By
.css('.root-key'))
3149 .sendKeys(Key
.CONTROL
,"a");
3150 driver
.findElement(By
.css('.root-key'))
3151 .sendKeys(Key
.DELETE
);
3152 driver
.sleep(generateDelay
).then(function() {
3153 // try to generate more addresses
3154 driver
.findElement(By
.css('.more'))
3156 driver
.sleep(generateDelay
).then(function() {
3157 driver
.findElements(By
.css(".addresses tr"))
3158 .then(function(els
) {
3159 // check there are no addresses shown
3160 expect(els
.length
).toBe(0);
3169 // error trying to generate addresses from xpub with hardened derivation
3170 it('Shows error for hardened addresses with xpub root key', function(done
) {
3171 driver
.findElement(By
.css('#bip32-tab a'))
3173 driver
.executeScript(function() {
3174 $(".hardened-addresses").prop("checked", true);
3176 // set xpub for account 0 of bip44 for 'abandon abandon ability'
3177 driver
.findElement(By
.css("#root-key"))
3178 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
3179 driver
.sleep(feedbackDelay
).then(function() {
3180 // Check feedback is correct
3181 driver
.findElement(By
.css('.feedback'))
3183 .then(function(feedback
) {
3184 var msg
= "Hardened derivation path is invalid with xpub key";
3185 expect(feedback
).toBe(msg
);
3191 // Litecoin uses ltub by default, and can optionally be set to xprv
3193 // https://github.com/iancoleman/bip39/issues/96
3194 // Issue with extended keys on Litecoin
3195 it('Uses ltub by default for litecoin, but can be set to xprv', function(done
) {
3196 driver
.findElement(By
.css('.phrase'))
3197 .sendKeys("abandon abandon ability");
3198 selectNetwork("LTC - Litecoin");
3199 driver
.sleep(generateDelay
).then(function() {
3200 // check the extended key is generated correctly
3201 driver
.findElement(By
.css('.root-key'))
3202 .getAttribute("value")
3203 .then(function(rootKey
) {
3204 expect(rootKey
).toBe("Ltpv71G8qDifUiNesiPqf6h5V6eQ8ic77oxQiYtawiACjBEx3sTXNR2HGDGnHETYxESjqkMLFBkKhWVq67ey1B2MKQXannUqNy1RZVHbmrEjnEU");
3205 // set litecoin to use ltub
3206 driver
.executeScript(function() {
3207 $(".litecoin-use-ltub").prop("checked", false);
3208 $(".litecoin-use-ltub").trigger("change");
3210 driver
.sleep(generateDelay
).then(function() {
3211 driver
.findElement(By
.css('.root-key'))
3212 .getAttribute("value")
3213 .then(function(rootKey
) {
3214 expect(rootKey
).toBe("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
3223 // https://github.com/iancoleman/bip39/issues/99#issuecomment-327094159
3224 // "warn me emphatically when they have detected invalid input" to the entropy field
3225 // A warning is shown when entropy is filtered and discarded
3226 it('Warns when entropy is filtered and discarded', function(done
) {
3227 driver
.findElement(By
.css('.use-entropy'))
3229 // set entropy to have no filtered content
3230 driver
.findElement(By
.css('.entropy'))
3231 .sendKeys("00000000 00000000 00000000 00000000");
3232 driver
.sleep(generateDelay
).then(function() {
3233 // check the filter warning does not show
3234 driver
.findElement(By
.css('.entropy-container .filter-warning'))
3235 .getAttribute("class")
3236 .then(function(classes
) {
3237 expect(classes
).toContain("hidden");
3238 // set entropy to have some filtered content
3239 driver
.findElement(By
.css('.entropy'))
3240 .sendKeys("10000000 zxcvbn 00000000 00000000 00000000");
3241 driver
.sleep(entropyFeedbackDelay
).then(function() {
3242 // check the filter warning shows
3243 driver
.findElement(By
.css('.entropy-container .filter-warning'))
3244 .getAttribute("class")
3245 .then(function(classes
) {
3246 expect(classes
).not
.toContain("hidden");
3254 // Bitcoin Cash address can be set to use cashaddr format
3255 it('Can use cashaddr format for bitcoin cash addresses', function(done
) {
3256 driver
.executeScript(function() {
3257 $(".use-bch-cashaddr-addresses").prop("checked", true);
3259 driver
.findElement(By
.css('.phrase'))
3260 .sendKeys("abandon abandon ability");
3261 selectNetwork("BCH - Bitcoin Cash");
3262 driver
.sleep(generateDelay
).then(function() {
3263 getFirstAddress(function(address
) {
3264 expect(address
).toBe("bitcoincash:qzlquk7w4hkudxypl4fgv8x279r754dkvur7jpcsps");
3270 // Bitcoin Cash address can be set to use bitpay format
3271 it('Can use bitpay format for bitcoin cash addresses', function(done
) {
3272 driver
.executeScript(function() {
3273 $(".use-bch-bitpay-addresses").prop("checked", true);
3275 driver
.findElement(By
.css('.phrase'))
3276 .sendKeys("abandon abandon ability");
3277 selectNetwork("BCH - Bitcoin Cash");
3278 driver
.sleep(generateDelay
).then(function() {
3279 getFirstAddress(function(address
) {
3280 expect(address
).toBe("CZnpA9HPmvhuhLLPWJP8rNDpLUYXy1LXFk");
3286 // Bitcoin Cash address can be set to use legacy format
3287 it('Can use legacy format for bitcoin cash addresses', function(done
) {
3288 driver
.executeScript(function() {
3289 $(".use-bch-legacy-addresses").prop("checked", true);
3291 driver
.findElement(By
.css('.phrase'))
3292 .sendKeys("abandon abandon ability");
3293 selectNetwork("BCH - Bitcoin Cash");
3294 driver
.sleep(generateDelay
).then(function() {
3295 getFirstAddress(function(address
) {
3296 expect(address
).toBe("1JKvb6wKtsjNoCRxpZ4DGrbniML7z5U16A");
3302 // End of tests ported from old suit, so no more comments above each test now
3304 it('Can generate more addresses from a custom index', function(done
) {
3305 var expectedIndexes
= [
3306 0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,
3307 40,41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59
3309 driver
.findElement(By
.css('.phrase'))
3310 .sendKeys("abandon abandon ability");
3311 driver
.sleep(generateDelay
).then(function() {
3312 // Set start of next lot of rows to be from index 40
3313 // which means indexes 20-39 will not be in the table.
3314 driver
.findElement(By
.css('.more-rows-start-index'))
3316 driver
.findElement(By
.css('.more'))
3318 driver
.sleep(generateDelay
).then(function() {
3319 // Check actual indexes in the table match the expected pattern
3320 driver
.findElements(By
.css(".index"))
3321 .then(function(els
) {
3322 expect(els
.length
).toBe(expectedIndexes
.length
);
3323 var testRowAtIndex = function(i
) {
3324 if (i
>= expectedIndexes
.length
) {
3329 .then(function(actualPath
) {
3330 var noHardened
= actualPath
.replace(/'/g, "");
3331 var pathBits = noHardened.split("/")
3332 var lastBit = pathBits[pathBits.length-1];
3333 var actualIndex = parseInt(lastBit);
3334 var expectedIndex = expectedIndexes[i];
3335 expect(actualIndex).toBe(expectedIndex);
3336 testRowAtIndex(i+1);
3346 it('Can generate BIP141 addresses
with P2WPKH
-in-P2SH semanitcs
', function(done) {
3347 // Sourced from BIP49 official test specs
3348 driver.findElement(By.css('#bip141
-tab a
'))
3350 driver.findElement(By.css('.bip141
-path
'))
3352 driver.findElement(By.css('.bip141
-path
'))
3353 .sendKeys("m/49'/1'/0'/0");
3354 selectNetwork("BTC
- Bitcoin Testnet
");
3355 driver.findElement(By.css(".phrase
"))
3356 .sendKeys("abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about
");
3357 driver.sleep(generateDelay).then(function() {
3358 getFirstAddress(function(address) {
3359 expect(address).toBe("2Mww8dCYPUpKHofjgcXcBCEGmniw9CoaiD2
");
3365 it('Can generate BIP141 addresses with P2WPKH semanitcs', function(done) {
3366 // This result tested against bitcoinjs-lib test spec for segwit address
3367 // using the first private key of this mnemonic and default path m/0
3368 // https://github.com/bitcoinjs/bitcoinjs-lib/blob/9c8503cab0c6c30a95127042703bc18e8d28c76d/test/integration/addresses.js#L50
3369 // so whilst not directly comparable, substituting the private key produces
3370 // identical results between this tool and the bitcoinjs-lib test.
3371 // Private key generated is:
3372 // L3L8Nu9whawPBNLGtFqDhKut9DKKfG3CQoysupT7BimqVCZsLFNP
3373 driver.findElement(By.css('#bip141-tab a'))
3376 driver.executeScript(function() {
3377 $(".bip141
-semantics option
[selected
]").removeAttr("selected
");
3378 $(".bip141
-semantics option
").filter(function(i,e) {
3379 return $(e).html() == "P2WPKH
";
3380 }).prop("selected
", true);
3381 $(".bip141
-semantics
").trigger("change
");
3383 driver.findElement(By.css(".phrase
"))
3384 .sendKeys("abandon abandon ability
");
3385 driver.sleep(generateDelay).then(function() {
3386 getFirstAddress(function(address) {
3387 expect(address).toBe("bc1qfwu6a5a3evygrk8zvdxxvz4547lmpyx5vsfxe9
");
3393 it('Shows the entropy used by the PRNG when clicking generate', function(done) {
3394 driver.findElement(By.css('.generate')).click();
3395 driver.sleep(generateDelay).then(function() {
3396 driver.findElement(By.css('.entropy'))
3397 .getAttribute("value
")
3398 .then(function(entropy) {
3399 expect(entropy).not.toBe("");
3405 it('Shows the index of each word in the mnemonic', function(done) {
3406 driver.findElement(By.css('.phrase'))
3407 .sendKeys("abandon abandon ability
");
3408 driver.sleep(generateDelay).then(function() {
3409 driver.findElement(By.css('.use-entropy'))
3411 driver.findElement(By.css('.word-indexes'))
3413 .then(function(indexes) {
3414 expect(indexes).toBe("0, 0, 1");
3420 it('Shows the derivation path for bip84 tab', function(done) {
3421 driver.findElement(By.css('#bip84-tab a'))
3423 driver.findElement(By.css('.phrase'))
3424 .sendKeys('abandon abandon ability');
3425 driver.sleep(generateDelay).then(function() {
3426 driver.findElement(By.css('#bip84 .path'))
3427 .getAttribute("value
")
3428 .then(function(path) {
3429 expect(path).toBe("m
/84'/0'/0'/0");
3435 it('Shows the extended private key for bip84 tab', function(done) {
3436 driver.findElement(By.css('#bip84-tab a'))
3438 driver.findElement(By.css('.phrase'))
3439 .sendKeys('abandon abandon ability');
3440 driver.sleep(generateDelay).then(function() {
3441 driver.findElement(By.css('.extended-priv-key'))
3442 .getAttribute("value
")
3443 .then(function(path) {
3444 expect(path).toBe("zprvAev3RKrZ3QVKiUFCfdeMRen1BPDJgdNt1XpxiDy8acSs4kkAGTCvq7HeRYRNNpo8EtEjCFQBWavJwtCUR29y4TUCH4X5RXMcyq48uN8y9BP
");
3450 it('Shows the extended public key for bip84 tab', function(done) {
3451 driver.findElement(By.css('#bip84-tab a'))
3453 driver.findElement(By.css('.phrase'))
3454 .sendKeys('abandon abandon ability');
3455 driver.sleep(generateDelay).then(function() {
3456 driver.findElement(By.css('.extended-pub-key'))
3457 .getAttribute("value
")
3458 .then(function(path) {
3459 expect(path).toBe("zpub6suPpqPSsn3cvxKfmfBMnnijjR3o666jNkkZWcNk8wyqwZ5JozXBNuc8Gs7DB3uLwTDvGVTspVEAUQcEjKF3pZHgywVbubdTqbXTUg7usyx
");
3465 it('Changes the address list if bip84 account is changed', function(done) {
3466 driver.findElement(By.css('#bip84-tab a'))
3468 driver.findElement(By.css('#bip84 .account'))
3470 driver.findElement(By.css('.phrase'))
3471 .sendKeys('abandon abandon ability');
3472 driver.sleep(generateDelay).then(function() {
3473 getFirstAddress(function(address) {
3474 expect(address).toBe("bc1qp7vv669t2fy965jdzvqwrraana89ctd5ewc662
");
3480 it('Changes the address list if bip84 change is changed', function(done) {
3481 driver.findElement(By.css('#bip84-tab a'))
3483 driver.findElement(By.css('#bip84 .change'))
3485 driver.findElement(By.css('.phrase'))
3486 .sendKeys('abandon abandon ability');
3487 driver.sleep(generateDelay).then(function() {
3488 getFirstAddress(function(address) {
3489 expect(address).toBe("bc1qr39vj6rh06ff05m53uxq8uazehwhccswylhrs2
");
3495 it('Passes the official BIP84 test spec for rootpriv', function(done) {
3496 driver.findElement(By.css('#bip84-tab a'))
3498 driver.findElement(By.css('.phrase'))
3499 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3500 driver.sleep(generateDelay).then(function() {
3501 driver.findElement(By.css(".root
-key
"))
3502 .getAttribute("value
")
3503 .then(function(rootKey) {
3504 expect(rootKey).toBe("zprvAWgYBBk7JR8Gjrh4UJQ2uJdG1r3WNRRfURiABBE3RvMXYSrRJL62XuezvGdPvG6GFBZduosCc1YP5wixPox7zhZLfiUm8aunE96BBa4Kei5
");
3510 it('Passes the official BIP84 test spec for account 0 xprv', function(done) {
3511 driver.findElement(By.css('#bip84-tab a'))
3513 driver.findElement(By.css('.phrase'))
3514 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3515 driver.sleep(generateDelay).then(function() {
3516 driver.findElement(By.css("#bip84
.account
-xprv
"))
3517 .getAttribute("value
")
3518 .then(function(rootKey) {
3519 expect(rootKey).toBe("zprvAdG4iTXWBoARxkkzNpNh8r6Qag3irQB8PzEMkAFeTRXxHpbF9z4QgEvBRmfvqWvGp42t42nvgGpNgYSJA9iefm1yYNZKEm7z6qUWCroSQnE
");
3525 it('Passes the official BIP84 test spec for account 0 xpub', function(done) {
3526 driver.findElement(By.css('#bip84-tab a'))
3528 driver.findElement(By.css('.phrase'))
3529 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3530 driver.sleep(generateDelay).then(function() {
3531 driver.findElement(By.css("#bip84
.account
-xpub
"))
3532 .getAttribute("value
")
3533 .then(function(rootKey) {
3534 expect(rootKey).toBe("zpub6rFR7y4Q2AijBEqTUquhVz398htDFrtymD9xYYfG1m4wAcvPhXNfE3EfH1r1ADqtfSdVCToUG868RvUUkgDKf31mGDtKsAYz2oz2AGutZYs
");
3540 it('Passes the official BIP84 test spec for account 0 first address', function(done) {
3541 driver.findElement(By.css('#bip84-tab a'))
3543 driver.findElement(By.css('.phrase'))
3544 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3545 driver.sleep(generateDelay).then(function() {
3546 getFirstAddress(function(address) {
3547 expect(address).toBe("bc1qcr8te4kr609gcawutmrza0j4xv80jy8z306fyu
");
3553 it('Passes the official BIP84 test spec for account 0 first change address', function(done) {
3554 driver.findElement(By.css('#bip84-tab a'))
3556 driver.findElement(By.css('.phrase'))
3557 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3558 driver.findElement(By.css('#bip84 .change'))
3560 driver.sleep(generateDelay).then(function() {
3561 getFirstAddress(function(address) {
3562 expect(address).toBe("bc1q8c6fshw2dlwun7ekn9qwf37cu2rn755upcp6el
");
3568 it('Can display the table as csv', function(done) {
3569 var headings = "path
,address
,public key
,private key
";
3570 var row1 = "m
/44'/0'/0'/0/0,1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug
,033f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3
,L26cVSpWFkJ6aQkPkKmTzLqTdLJ923e6CzrVh9cmx21QHsoUmrEE
";
3571 var row20 = "m
/44'/0'/0'/0/19,1KhBy28XLAciXnnRvm71PvQJaETyrxGV55
,02b4b3e396434d8cdd20c03ac4aaa07387784d5d867b75987f516f5705ee68cb3a
,L4GrDrjReMsCAu5DkLXn79jSb95qR7Zfx7eshybCQZ1qL32MXJab
";
3572 driver.findElement(By.css('.phrase'))
3573 .sendKeys('abandon abandon ability');
3574 driver.sleep(generateDelay).then(function() {
3575 driver.findElement(By.css('.csv'))
3576 .getAttribute("value
")
3577 .then(function(csv) {
3578 expect(csv).toContain(headings);
3579 expect(csv).toContain(row1);
3580 expect(csv).toContain(row20);
3586 it('LeftPads ethereum keys that are less than 32 bytes', function(done) {
3587 // see https://github.com/iancoleman/bip39/issues/155
3588 selectNetwork("ETH
- Ethereum
");
3589 driver.findElement(By.css('#bip32-tab a'))
3591 driver.findElement(By.css('#bip32-path'))
3593 driver.findElement(By.css('#bip32-path'))
3594 .sendKeys("m
/44'/60'/0'");
3595 driver.findElement(By.css('.phrase'))
3596 .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');
3597 driver.sleep(generateDelay).then(function() {
3598 getFirstAddress(function(address) {
3599 expect(address).toBe("0x8943E785B4a5714FC87a3aFAad1eB1FeB602B118");
3605 it('Can encrypt private keys using BIP38', function(done) {
3606 // see https://github.com/iancoleman/bip39/issues/140
3607 driver.executeScript(function() {
3608 $(".use-bip38
").prop("checked
", true);
3610 driver.findElement(By.css('.bip38-password'))
3611 .sendKeys('bip38password');
3612 driver.findElement(By.css('.rows-to-add'))
3614 driver.findElement(By.css('.rows-to-add'))
3616 driver.findElement(By.css('.phrase'))
3617 .sendKeys('abandon abandon ability');
3618 driver.sleep(bip38delay).then(function() {
3620 getFirstRowValue(function(address) {
3621 expect(address).toBe("1NCvSdumA3ngMM9c4aqU56AM6rqXddfuXB
");
3623 getFirstRowValue(function(pubkey) {
3624 expect(pubkey).toBe("043f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3884a74447ea901729b1e73a999b7520e7cb55b4120e6432c64153ccab8a848e1
");
3626 getFirstRowValue(function(privkey) {
3627 expect(privkey).toBe("6PRNRiFnj1RoR3sXhymdCvoZCgnUHQpfupNdKkFbWJkwWQEKesWt1EDMDM
");
3633 }, bip38delay + 5000);
3635 it('Shows the checksum for the entropy', function(done) {
3636 driver.findElement(By.css('.use-entropy'))
3638 driver.findElement(By.css('.entropy'))
3639 .sendKeys("00000000000000000000000000000000");
3640 driver.sleep(generateDelay).then(function() {
3641 driver.findElement(By.css('.checksum'))
3643 .then(function(text) {
3644 expect(text).toBe("1");
3650 it('Shows the checksum for the entropy with the correct groupings', function(done) {
3651 driver.findElement(By.css('.use-entropy'))
3653 // create a checksum of 20 bits, which spans multiple words
3654 driver.findElement(By.css('.entropy'))
3655 .sendKeys("F000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
");
3656 driver.sleep(generateDelay).then(function() {
3657 driver.findElement(By.css('.checksum'))
3659 .then(function(text) {
3660 // first group is 9 bits, second group is 11
3661 expect(text).toBe("011010111 01110000110");
3667 it('Uses vprv for bitcoin testnet p2wpkh', function(done) {
3668 selectNetwork("BTC
- Bitcoin Testnet
");
3669 driver.findElement(By.css('#bip84-tab a'))
3671 driver.findElement(By.css('.phrase'))
3672 .sendKeys('abandon abandon ability');
3673 driver.sleep(generateDelay).then(function() {
3674 driver.findElement(By.css('.root-key'))
3675 .getAttribute("value
")
3676 .then(function(path) {
3677 expect(path).toBe("vprv9DMUxX4ShgxML9N2YV5CvWEebWrM9aJ5ULpbRRyzyWu6vs4BzTvbfFFrH41N5hVi7MYSfiugd765L3JmAfDM5po36Y8ouCKRDeYQwByCmS7
");
3683 it('Shows a warning if generating weak mnemonics', function(done) {
3684 driver.executeScript(function() {
3685 $(".strength option
[selected
]").removeAttr("selected
");
3686 $(".strength option
[value
=6]").prop("selected
", true);
3687 $(".strength
").trigger("change
");
3689 driver.findElement(By.css(".generate
-container
.warning
"))
3690 .getAttribute("class")
3691 .then(function(classes) {
3692 expect(classes).not.toContain("hidden
");
3697 it('Does not show a warning if generating strong mnemonics', function(done) {
3698 driver.executeScript(function() {
3699 $(".strength option
[selected
]").removeAttr("selected
");
3700 $(".strength option
[value
=12]").prop("selected
", true);
3702 driver.findElement(By.css(".generate
-container
.warning
"))
3703 .getAttribute("class")
3704 .then(function(classes) {
3705 expect(classes).toContain("hidden
");
3710 it('Shows a warning if overriding weak entropy with longer mnemonics', function(done) {
3711 driver.findElement(By.css('.use-entropy'))
3713 driver.findElement(By.css('.entropy'))
3714 .sendKeys("0123456789abcdef
"); // 6 words
3715 driver.executeScript(function() {
3716 $(".mnemonic
-length
").val("12").trigger("change
");
3718 driver.findElement(By.css(".weak
-entropy
-override
-warning
"))
3719 .getAttribute("class")
3720 .then(function(classes) {
3721 expect(classes).not.toContain("hidden
");
3726 it('Does not show a warning if entropy is stronger than mnemonic length', function(done) {
3727 driver.findElement(By.css('.use-entropy'))
3729 driver.findElement(By.css('.entropy'))
3730 .sendKeys("0123456789abcdef0123456789abcdef0123456789abcdef
"); // 18 words
3731 driver.executeScript(function() {
3732 $(".mnemonic
-length
").val("12").trigger("change
");
3734 driver.findElement(By.css(".weak
-entropy
-override
-warning
"))
3735 .getAttribute("class")
3736 .then(function(classes) {
3737 expect(classes).toContain("hidden
");
3742 it('Shows litecoin BIP49 addresses', function(done) {
3743 driver.findElement(By.css('.phrase'))
3744 .sendKeys('abandon abandon ability');
3745 selectNetwork("LTC
- Litecoin
");
3746 driver.findElement(By.css('#bip49-tab a'))
3748 // bip49 addresses are shown
3749 driver.sleep(generateDelay).then(function() {
3750 driver.findElement(By.css('#bip49 .available'))
3751 .getAttribute("class")
3752 .then(function(classes) {
3753 expect(classes).not.toContain("hidden
");
3754 // check first address
3755 getFirstAddress(function(address) {
3756 expect(address).toBe("MFwLPhsXoBuSLL8cLmW9uK6tChkzduV8qN
");
3763 it('Can use root keys to generate segwit table rows', function(done) {
3764 // segwit uses ypub / zpub instead of xpub but the root key should still
3765 // be valid regardless of the encoding used to import that key.
3766 // Maybe this breaks the reason for the different extended key prefixes, but
3767 // since the parsed root key is used behind the scenes anyhow this should be
3769 driver.findElement(By.css('#root-key'))
3770 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
3771 driver.findElement(By.css('#bip49-tab a'))
3773 // bip49 addresses are shown
3774 driver.sleep(generateDelay).then(function() {
3775 getFirstAddress(function(address) {
3776 expect(address).toBe("3QG2Y9AA4xZ846gKHZqNf7mvVKbLqMKxr2
");