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);
1300 it('Allows selection
of Artax
', function(done) {
1302 selectText: "XAX - Artax",
1303 firstAddress: "AYxaQPY7XLidG31V7F3yNzwxPYpYzRqG4q",
1305 testNetwork(done, params);
1309 // BIP39 seed is set from phrase
1310 it('Sets the bip39 seed
from the prhase
', function(done) {
1311 driver.findElement(By.css('.phrase
'))
1312 .sendKeys('abandon abandon ability
');
1313 driver.sleep(generateDelay).then(function() {
1314 driver.findElement(By.css('.seed
'))
1315 .getAttribute("value")
1316 .then(function(seed) {
1317 expect(seed).toBe("20da140d3dd1df8713cefcc4d54ce0e445b4151027a1ab567b832f6da5fcc5afc1c3a3f199ab78b8e0ab4652efd7f414ac2c9a3b81bceb879a70f377aa0a58f3");
1323 // BIP32 root key is set from phrase
1324 it('Sets the bip39 root key
from the prhase
', function(done) {
1325 driver.findElement(By.css('.phrase
'))
1326 .sendKeys('abandon abandon ability
');
1327 driver.sleep(generateDelay).then(function() {
1328 driver.findElement(By.css('.root
-key
'))
1329 .getAttribute("value")
1330 .then(function(seed) {
1331 expect(seed).toBe("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
1337 // Tabs show correct addresses when changed
1338 it('Shows the correct address when tab is changed
', function(done) {
1339 driver.findElement(By.css('.phrase
'))
1340 .sendKeys('abandon abandon ability
');
1341 driver.sleep(generateDelay).then(function() {
1342 driver.findElement(By.css('#bip32
-tab a
'))
1344 driver.sleep(generateDelay).then(function() {
1345 getFirstAddress(function(address) {
1346 expect(address).toBe("17uQ7s2izWPwBmEVFikTmZUjbBKWYdJchz");
1353 // BIP44 derivation path is shown
1354 it('Shows the derivation path
for bip44 tab
', function(done) {
1355 driver.findElement(By.css('.phrase
'))
1356 .sendKeys('abandon abandon ability
');
1357 driver.sleep(generateDelay).then(function() {
1358 driver.findElement(By.css('#bip44
.path
'))
1359 .getAttribute("value")
1360 .then(function(path) {
1361 expect(path).toBe("m/44'/0'/0'/0");
1367 // BIP44 extended private key is shown
1368 it('Shows the extended private key for bip44 tab', function(done) {
1369 driver.findElement(By.css('.phrase'))
1370 .sendKeys('abandon abandon ability');
1371 driver.sleep(generateDelay).then(function() {
1372 driver.findElement(By.css('.extended-priv-key'))
1373 .getAttribute("value
")
1374 .then(function(path) {
1375 expect(path).toBe("xprvA2DxxvPZcyRvYgZMGS53nadR32mVDeCyqQYyFhrCVbJNjPoxMeVf7QT5g7mQASbTf9Kp4cryvcXnu2qurjWKcrdsr91jXymdCDNxKgLFKJG
");
1381 // BIP44 extended public key is shown
1382 it('Shows the extended public key for bip44 tab', function(done) {
1383 driver.findElement(By.css('.phrase'))
1384 .sendKeys('abandon abandon ability');
1385 driver.sleep(generateDelay).then(function() {
1386 driver.findElement(By.css('.extended-pub-key'))
1387 .getAttribute("value
")
1388 .then(function(path) {
1389 expect(path).toBe("xpub6FDKNRvTTLzDmAdpNTc49ia9b4byd6vqCdUa46Fp3vqMcC96uBoufCmZXQLiN5AK3iSCJMhf9gT2sxkpyaPepRuA7W3MujV5tGmF5VfbueM
");
1395 // BIP44 account field changes address list
1396 it('Changes the address list if bip44 account is changed', function(done) {
1397 driver.findElement(By.css('#bip44 .account'))
1399 driver.findElement(By.css('.phrase'))
1400 .sendKeys('abandon abandon ability');
1401 driver.sleep(generateDelay).then(function() {
1402 getFirstAddress(function(address) {
1403 expect(address).toBe("1Nq2Wmu726XHCuGhctEtGmhxo3wzk5wZ1H
");
1409 // BIP44 change field changes address list
1410 it('Changes the address list if bip44 change is changed', function(done) {
1411 driver.findElement(By.css('#bip44 .change'))
1413 driver.findElement(By.css('.phrase'))
1414 .sendKeys('abandon abandon ability');
1415 driver.sleep(generateDelay).then(function() {
1416 getFirstAddress(function(address) {
1417 expect(address).toBe("1KAGfWgqfVbSSXY56fNQ7YnhyKuoskHtYo
");
1423 // BIP32 derivation path can be set
1424 it('Can use a custom bip32 derivation path', function(done) {
1425 driver.findElement(By.css('#bip32-tab a'))
1427 driver.findElement(By.css('#bip32 .path'))
1429 driver.findElement(By.css('#bip32 .path'))
1431 driver.findElement(By.css('.phrase'))
1432 .sendKeys('abandon abandon ability');
1433 driver.sleep(generateDelay).then(function() {
1434 getFirstAddress(function(address) {
1435 expect(address).toBe("16pYQQdLD1hH4hwTGLXBaZ9Teboi1AGL8L
");
1441 // BIP32 can use hardened derivation paths
1442 it('Can use a hardened derivation paths', function(done) {
1443 driver.findElement(By.css('#bip32-tab a'))
1445 driver.findElement(By.css('#bip32 .path'))
1447 driver.findElement(By.css('#bip32 .path'))
1449 driver.findElement(By.css('.phrase
'))
1450 .sendKeys('abandon abandon ability
');
1451 driver.sleep(generateDelay).then(function() {
1452 getFirstAddress(function(address) {
1453 expect(address).toBe("14aXZeprXAE3UUKQc4ihvwBvww2LuEoHo4");
1459 // BIP32 extended private key is shown
1460 it('Shows the BIP32 extended
private key
', function(done) {
1461 driver.findElement(By.css('#bip32
-tab a
'))
1463 driver.findElement(By.css('.phrase
'))
1464 .sendKeys('abandon abandon ability
');
1465 driver.sleep(generateDelay).then(function() {
1466 driver.findElement(By.css('.extended
-priv
-key
'))
1467 .getAttribute("value")
1468 .then(function(privKey) {
1469 expect(privKey).toBe("xprv9va99uTVE5aLiutUVLTyfxfe8v8aaXjSQ1XxZbK6SezYVuikA9MnjQVTA8rQHpNA5LKvyQBpLiHbBQiiccKiBDs7eRmBogsvq3THFeLHYbe");
1475 // BIP32 extended public key is shown
1476 it('Shows the BIP32 extended
public key
', function(done) {
1477 driver.findElement(By.css('#bip32
-tab a
'))
1479 driver.findElement(By.css('.phrase
'))
1480 .sendKeys('abandon abandon ability
');
1481 driver.sleep(generateDelay).then(function() {
1482 driver.findElement(By.css('.extended
-pub
-key
'))
1483 .getAttribute("value")
1484 .then(function(pubKey) {
1485 expect(pubKey).toBe("xpub69ZVZQzP4T8dwPxwbMzz36cNgwy4yzTHmETZMyihzzXXNi3thgg3HCow1RtY252wdw5rS8369xKnraN5Q93y3FkFfJp2XEHWUrkyXsjS93P");
1491 // Derivation path is shown in table
1492 it('Shows the derivation path
in the table
', function(done) {
1493 driver.findElement(By.css('.phrase
'))
1494 .sendKeys('abandon abandon ability
');
1495 driver.sleep(generateDelay).then(function() {
1496 getFirstPath(function(path) {
1497 expect(path).toBe("m/44'/0'/0'/0/0");
1503 // Derivation path for address can be hardened
1504 it('Can derive hardened addresses', function(done) {
1505 driver.findElement(By.css('#bip32-tab a'))
1507 driver.executeScript(function() {
1508 $(".hardened
-addresses
").prop("checked
", true);
1510 driver.findElement(By.css('.phrase'))
1511 .sendKeys('abandon abandon ability');
1512 driver.sleep(generateDelay).then(function() {
1513 getFirstAddress(function(address) {
1514 expect(address).toBe("18exLzUv7kfpiXRzmCjFDoC9qwNLFyvwyd
");
1520 // Derivation path visibility can be toggled
1521 it('Can toggle visibility of the derivation path column', function(done) {
1522 driver.findElement(By.css('.phrase'))
1523 .sendKeys('abandon abandon ability');
1524 driver.sleep(generateDelay).then(function() {
1525 driver.findElement(By.css('.index-toggle'))
1527 testColumnValuesAreInvisible(done, "index
");
1532 it('Shows the address in the table', function(done) {
1533 driver.findElement(By.css('.phrase'))
1534 .sendKeys('abandon abandon ability');
1535 driver.sleep(generateDelay).then(function() {
1536 getFirstAddress(function(address) {
1537 expect(address).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug
");
1543 // Addresses are shown in order of derivation path
1544 it('Shows the address in order of derivation path', function(done) {
1545 driver.findElement(By.css('.phrase'))
1546 .sendKeys('abandon abandon ability');
1547 driver.sleep(generateDelay).then(function() {
1548 testRowsAreInCorrectOrder(done);
1552 // Address visibility can be toggled
1553 it('Can toggle visibility of the address column', function(done) {
1554 driver.findElement(By.css('.phrase'))
1555 .sendKeys('abandon abandon ability');
1556 driver.sleep(generateDelay).then(function() {
1557 driver.findElement(By.css('.address-toggle'))
1559 testColumnValuesAreInvisible(done, "address
");
1563 // Public key is shown in table
1564 it('Shows the public key in the table', function(done) {
1565 driver.findElement(By.css('.phrase'))
1566 .sendKeys('abandon abandon ability');
1567 driver.sleep(generateDelay).then(function() {
1568 driver.findElements(By.css('.pubkey'))
1569 .then(function(els) {
1571 .then(function(pubkey) {
1572 expect(pubkey).toBe("033f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3
");
1579 // Public key visibility can be toggled
1580 it('Can toggle visibility of the public key column', function(done) {
1581 driver.findElement(By.css('.phrase'))
1582 .sendKeys('abandon abandon ability');
1583 driver.sleep(generateDelay).then(function() {
1584 driver.findElement(By.css('.public-key-toggle'))
1586 testColumnValuesAreInvisible(done, "pubkey
");
1590 // Private key is shown in table
1591 it('Shows the private key in the table', function(done) {
1592 driver.findElement(By.css('.phrase'))
1593 .sendKeys('abandon abandon ability');
1594 driver.sleep(generateDelay).then(function() {
1595 driver.findElements(By.css('.privkey'))
1596 .then(function(els) {
1598 .then(function(pubkey) {
1599 expect(pubkey).toBe("L26cVSpWFkJ6aQkPkKmTzLqTdLJ923e6CzrVh9cmx21QHsoUmrEE
");
1606 // Private key visibility can be toggled
1607 it('Can toggle visibility of the private key column', function(done) {
1608 driver.findElement(By.css('.phrase'))
1609 .sendKeys('abandon abandon ability');
1610 driver.sleep(generateDelay).then(function() {
1611 driver.findElement(By.css('.private-key-toggle'))
1613 testColumnValuesAreInvisible(done, "privkey
");
1617 // More addresses can be generated
1618 it('Can generate more rows in the table', function(done) {
1619 driver.findElement(By.css('.phrase'))
1620 .sendKeys('abandon abandon ability');
1621 driver.sleep(generateDelay).then(function() {
1622 driver.findElement(By.css('.more'))
1624 driver.sleep(generateDelay).then(function() {
1625 driver.findElements(By.css('.address'))
1626 .then(function(els) {
1627 expect(els.length).toBe(40);
1634 // A custom number of additional addresses can be generated
1635 it('Can generate more rows in the table', function(done) {
1636 driver.findElement(By.css('.phrase'))
1637 .sendKeys('abandon abandon ability');
1638 driver.sleep(generateDelay).then(function() {
1639 driver.findElement(By.css('.rows-to-add'))
1641 driver.findElement(By.css('.rows-to-add'))
1643 driver.findElement(By.css('.more'))
1645 driver.sleep(generateDelay).then(function() {
1646 driver.findElements(By.css('.address'))
1647 .then(function(els) {
1648 expect(els.length).toBe(21);
1655 // Additional addresses are shown in order of derivation path
1656 it('Shows additional addresses in order of derivation path', function(done) {
1657 driver.findElement(By.css('.phrase'))
1658 .sendKeys('abandon abandon ability');
1659 driver.sleep(generateDelay).then(function() {
1660 driver.findElement(By.css('.more'))
1662 driver.sleep(generateDelay).then(function() {
1663 testRowsAreInCorrectOrder(done);
1668 // BIP32 root key can be set by the user
1669 it('Allows the user to set the BIP32 root key', function(done) {
1670 driver.findElement(By.css('.root-key'))
1671 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1672 driver.sleep(generateDelay).then(function() {
1673 getFirstAddress(function(address) {
1674 expect(address).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug
");
1680 // Setting BIP32 root key clears the existing phrase, passphrase and seed
1681 // TODO this doesn't work in selenium with chrome
1682 it('Confirms the existing phrase should be cleared', function(done) {
1683 if (browser == "chrome
") {
1684 pending("Selenium
+ Chrome headless bug
for alert
, see
https://stackoverflow.com/q/45242264");
1686 driver
.findElement(By
.css('.phrase'))
1687 .sendKeys('A non-blank but invalid value');
1688 driver
.findElement(By
.css('.root-key'))
1689 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1690 driver
.switchTo().alert().accept();
1691 driver
.findElement(By
.css('.phrase'))
1692 .getAttribute("value").then(function(value
) {
1693 expect(value
).toBe("");
1698 // Clearing of phrase, passphrase and seed can be cancelled by user
1699 // TODO this doesn't work in selenium with chrome
1700 it('Allows the clearing of the phrase to be cancelled', function(done
) {
1701 if (browser
== "chrome") {
1702 pending("Selenium + Chrome headless bug for alert, see https://stackoverflow.com/q/45242264");
1704 driver
.findElement(By
.css('.phrase'))
1705 .sendKeys('abandon abandon ability');
1706 driver
.sleep(generateDelay
).then(function() {
1707 driver
.findElement(By
.css('.root-key'))
1709 driver
.findElement(By
.css('.root-key'))
1711 driver
.switchTo().alert().dismiss();
1712 driver
.findElement(By
.css('.phrase'))
1713 .getAttribute("value").then(function(value
) {
1714 expect(value
).toBe("abandon abandon ability");
1720 // Custom BIP32 root key is used when changing the derivation path
1721 it('Can set derivation path for root key instead of phrase', function(done
) {
1722 driver
.findElement(By
.css('#bip44 .account'))
1724 driver
.findElement(By
.css('.root-key'))
1725 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1726 driver
.sleep(generateDelay
).then(function() {
1727 getFirstAddress(function(address
) {
1728 expect(address
).toBe("1Nq2Wmu726XHCuGhctEtGmhxo3wzk5wZ1H");
1734 // Incorrect mnemonic shows error
1735 it('Shows an error for incorrect mnemonic', function(done
) {
1736 driver
.findElement(By
.css('.phrase'))
1737 .sendKeys('abandon abandon abandon');
1738 driver
.sleep(feedbackDelay
).then(function() {
1739 driver
.findElement(By
.css('.feedback'))
1741 .then(function(feedback
) {
1742 expect(feedback
).toBe("Invalid mnemonic");
1748 // Incorrect word shows suggested replacement
1749 it('Shows word suggestion for incorrect word', function(done
) {
1750 driver
.findElement(By
.css('.phrase'))
1751 .sendKeys('abandon abandon abiliti');
1752 driver
.sleep(feedbackDelay
).then(function() {
1753 driver
.findElement(By
.css('.feedback'))
1755 .then(function(feedback
) {
1756 var msg
= "abiliti not in wordlist, did you mean ability?";
1757 expect(feedback
).toBe(msg
);
1763 // Github pull request 48
1764 // First four letters of word shows that word, not closest
1765 // since first four letters gives unique word in BIP39 wordlist
1766 // eg ille should show illegal, not idle
1767 it('Shows word suggestion based on first four chars', function(done
) {
1768 driver
.findElement(By
.css('.phrase'))
1770 driver
.sleep(feedbackDelay
).then(function() {
1771 driver
.findElement(By
.css('.feedback'))
1773 .then(function(feedback
) {
1774 var msg
= "ille not in wordlist, did you mean illegal?";
1775 expect(feedback
).toBe(msg
);
1781 // Incorrect BIP32 root key shows error
1782 it('Shows error for incorrect root key', function(done
) {
1783 driver
.findElement(By
.css('.root-key'))
1784 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpj');
1785 driver
.sleep(feedbackDelay
).then(function() {
1786 driver
.findElement(By
.css('.feedback'))
1788 .then(function(feedback
) {
1789 var msg
= "Invalid root key";
1790 expect(feedback
).toBe(msg
);
1796 // Derivation path not starting with m shows error
1797 it('Shows error for derivation path not starting with m', function(done
) {
1798 driver
.findElement(By
.css('#bip32-tab a'))
1800 driver
.findElement(By
.css('#bip32 .path'))
1802 driver
.findElement(By
.css('#bip32 .path'))
1804 driver
.findElement(By
.css('.phrase'))
1805 .sendKeys('abandon abandon ability');
1806 driver
.sleep(feedbackDelay
).then(function() {
1807 driver
.findElement(By
.css('.feedback'))
1809 .then(function(feedback
) {
1810 var msg
= "First character must be 'm'";
1811 expect(feedback
).toBe(msg
);
1817 // Derivation path containing invalid characters shows useful error
1818 it('Shows error for derivation path not starting with m', function(done
) {
1819 driver
.findElement(By
.css('#bip32-tab a'))
1821 driver
.findElement(By
.css('#bip32 .path'))
1823 driver
.findElement(By
.css('#bip32 .path'))
1824 .sendKeys('m/1/0wrong1/1');
1825 driver
.findElement(By
.css('.phrase'))
1826 .sendKeys('abandon abandon ability');
1827 driver
.sleep(feedbackDelay
).then(function() {
1828 driver
.findElement(By
.css('.feedback'))
1830 .then(function(feedback
) {
1831 var msg
= "Invalid characters 0wrong1 found at depth 2";
1832 expect(feedback
).toBe(msg
);
1838 // Github Issue 11: Default word length is 15
1839 // https://github.com/iancoleman/bip39/issues/11
1840 it('Sets the default word length to 15', function(done
) {
1841 driver
.findElement(By
.css('.strength'))
1842 .getAttribute("value")
1843 .then(function(strength
) {
1844 expect(strength
).toBe("15");
1849 // Github Issue 12: Generate more rows with private keys hidden
1850 // https://github.com/iancoleman/bip39/issues/12
1851 it('Sets the correct hidden column state on new rows', function(done
) {
1852 driver
.findElement(By
.css('.phrase'))
1853 .sendKeys("abandon abandon ability");
1854 driver
.sleep(generateDelay
).then(function() {
1855 driver
.findElement(By
.css('.private-key-toggle'))
1857 driver
.findElement(By
.css('.more'))
1859 driver
.sleep(generateDelay
).then(function() {
1860 driver
.findElements(By
.css('.privkey'))
1861 .then(function(els
) {
1862 expect(els
.length
).toBe(40);
1864 testColumnValuesAreInvisible(done
, "privkey");
1869 // Github Issue 19: Mnemonic is not sensitive to whitespace
1870 // https://github.com/iancoleman/bip39/issues/19
1871 it('Ignores excess whitespace in the mnemonic', function(done
) {
1872 var doublespace
= " ";
1873 var mnemonic
= "urge cat" + doublespace
+ "bid";
1874 driver
.findElement(By
.css('.phrase'))
1875 .sendKeys(mnemonic
);
1876 driver
.sleep(generateDelay
).then(function() {
1877 driver
.findElement(By
.css('.root-key'))
1878 .getAttribute("value")
1879 .then(function(seed
) {
1880 expect(seed
).toBe("xprv9s21ZrQH143K3isaZsWbKVoTtbvd34Y1ZGRugGdMeBGbM3AgBVzTH159mj1cbbtYSJtQr65w6L5xy5L9SFC7c9VJZWHxgAzpj4mun5LhrbC");
1886 // Github Issue 23: Part 1: Use correct derivation path when changing tabs
1887 // https://github.com/iancoleman/bip39/issues/23
1888 it('Uses the correct derivation path when changing tabs', function(done
) {
1889 // 1) and 2) set the phrase
1890 driver
.findElement(By
.css('.phrase'))
1891 .sendKeys("abandon abandon ability");
1892 driver
.sleep(generateDelay
).then(function() {
1893 // 3) select bip32 tab
1894 driver
.findElement(By
.css('#bip32-tab a'))
1896 driver
.sleep(generateDelay
).then(function() {
1897 // 4) switch from bitcoin to litecoin
1898 selectNetwork("LTC - Litecoin");
1899 driver
.sleep(generateDelay
).then(function() {
1900 // 5) Check address is displayed correctly
1901 getFirstAddress(function(address
) {
1902 expect(address
).toBe("LS8MP5LZ5AdzSZveRrjm3aYVoPgnfFh5T5");
1903 // 5) Check derivation path is displayed correctly
1904 getFirstPath(function(path
) {
1905 expect(path
).toBe("m/0/0");
1914 // Github Issue 23 Part 2: Coin selection in derivation path
1915 // https://github.com/iancoleman/bip39/issues/23#issuecomment-238011920
1916 it('Uses the correct derivation path when changing coins', function(done
) {
1918 driver
.findElement(By
.css('.phrase'))
1919 .sendKeys("abandon abandon ability");
1920 driver
.sleep(generateDelay
).then(function() {
1921 // switch from bitcoin to clam
1922 selectNetwork("CLAM - Clams");
1923 driver
.sleep(generateDelay
).then(function() {
1924 // check derivation path is displayed correctly
1925 getFirstPath(function(path
) {
1926 expect(path
).toBe("m/44'/23'/0'/0/0");
1933 // Github Issue 26: When using a Root key derrived altcoins are incorrect
1934 // https://github.com/iancoleman/bip39/issues/26
1935 it('Uses the correct derivation for altcoins with root keys', function(done
) {
1936 // 1) 2) and 3) set the root key
1937 driver
.findElement(By
.css('.root-key'))
1938 .sendKeys("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
1939 driver
.sleep(generateDelay
).then(function() {
1940 // 4) switch from bitcoin to viacoin
1941 selectNetwork("VIA - Viacoin");
1942 driver
.sleep(generateDelay
).then(function() {
1943 // 5) ensure the derived address is correct
1944 getFirstAddress(function(address
) {
1945 expect(address
).toBe("Vq9Eq4N5SQnjqZvxtxzo7hZPW5XnyJsmXT");
1952 // Selecting a language with no existing phrase should generate a phrase in
1954 it('Generate a random phrase when language is selected and no current phrase', function(done
) {
1955 driver
.findElement(By
.css("a[href='#japanese']"))
1957 driver
.sleep(generateDelay
).then(function() {
1958 driver
.findElement(By
.css(".phrase"))
1959 .getAttribute("value").then(function(phrase
) {
1960 expect(phrase
.search(/[a-z]/)).toBe(-1);
1961 expect(phrase
.length
).toBeGreaterThan(0);
1967 // Selecting a language with existing phrase should update the phrase to use
1969 it('Updates existing phrases when the language is changed', function(done
) {
1970 driver
.findElement(By
.css(".phrase"))
1971 .sendKeys("abandon abandon ability");
1972 driver
.sleep(generateDelay
).then(function() {
1973 driver
.findElement(By
.css("a[href='#italian']"))
1975 driver
.sleep(generateDelay
).then(function() {
1976 driver
.findElement(By
.css(".phrase"))
1977 .getAttribute("value").then(function(phrase
) {
1978 // Check only the language changes, not the phrase
1979 expect(phrase
).toBe("abaco abaco abbaglio");
1980 getFirstAddress(function(address
) {
1981 // Check the address is correct
1982 expect(address
).toBe("1Dz5TgDhdki9spa6xbPFbBqv5sjMrx3xgV");
1990 // Suggested replacement for erroneous word in non-English language
1991 it('Shows word suggestion for incorrect word in non-English language', function(done
) {
1992 driver
.findElement(By
.css('.phrase'))
1993 .sendKeys('abaco abaco zbbaglio');
1994 driver
.sleep(feedbackDelay
).then(function() {
1995 driver
.findElement(By
.css('.feedback'))
1997 .then(function(feedback
) {
1998 var msg
= "zbbaglio not in wordlist, did you mean abbaglio?";
1999 expect(feedback
).toBe(msg
);
2005 // Japanese word does not break across lines.
2007 // https://github.com/bitcoin/bips/blob/master/bip-0039/bip-0039-wordlists.md#japanese
2008 it('Does not break Japanese words across lines', function(done
) {
2009 driver
.findElement(By
.css('.phrase'))
2010 .getCssValue("word-break")
2011 .then(function(value
) {
2012 expect(value
).toBe("keep-all");
2017 // Language can be specified at page load using hash value in url
2018 it('Can set the language from the url hash', function(done
) {
2019 driver
.get(url
+ "#japanese").then(function() {
2020 driver
.findElement(By
.css('.generate')).click();
2021 driver
.sleep(generateDelay
).then(function() {
2022 driver
.findElement(By
.css(".phrase"))
2023 .getAttribute("value").then(function(phrase
) {
2024 expect(phrase
.search(/[a-z]/)).toBe(-1);
2025 expect(phrase
.length
).toBeGreaterThan(0);
2032 // Entropy can be entered by the user
2033 it('Allows entropy to be entered', function(done
) {
2034 driver
.findElement(By
.css('.use-entropy'))
2036 driver
.findElement(By
.css('.entropy'))
2037 .sendKeys('00000000 00000000 00000000 00000000');
2038 driver
.sleep(generateDelay
).then(function() {
2039 driver
.findElement(By
.css(".phrase"))
2040 .getAttribute("value").then(function(phrase
) {
2041 expect(phrase
).toBe("abandon abandon ability");
2042 getFirstAddress(function(address
) {
2043 expect(address
).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug");
2050 // A warning about entropy is shown to the user, with additional information
2051 it('Shows a warning about using entropy', function(done
) {
2052 driver
.findElement(By
.css('.use-entropy'))
2054 driver
.findElement(By
.css('.entropy-container'))
2056 .then(function(containerText
) {
2057 var warning
= "mnemonic may be insecure";
2058 expect(containerText
).toContain(warning
);
2059 driver
.findElement(By
.css('#entropy-notes'))
2060 .findElement(By
.xpath("parent::*"))
2062 .then(function(notesText
) {
2063 var detail
= "flipping a fair coin, rolling a fair dice, noise measurements etc";
2064 expect(notesText
).toContain(detail
);
2070 // The types of entropy available are described to the user
2071 it('Shows the types of entropy available', function(done
) {
2072 driver
.findElement(By
.css('.entropy'))
2073 .getAttribute("placeholder")
2074 .then(function(placeholderText
) {
2083 for (var i
=0; i
<options
.length
; i
++) {
2084 var option
= options
[i
];
2085 expect(placeholderText
).toContain(option
);
2091 // The actual entropy used is shown to the user
2092 it('Shows the actual entropy used', function(done
) {
2093 driver
.findElement(By
.css('.use-entropy'))
2095 driver
.findElement(By
.css('.entropy'))
2096 .sendKeys('Not A Very Good Entropy Source At All');
2097 driver
.sleep(generateDelay
).then(function() {
2098 driver
.findElement(By
.css('.entropy-container'))
2100 .then(function(text
) {
2101 expect(text
).toMatch(/Filtered Entropy
\s
+AedEceAA
/);
2107 // Binary entropy can be entered
2108 it('Allows binary entropy to be entered', function(done
) {
2109 testEntropyType(done
, "01", "binary");
2112 // Base 6 entropy can be entered
2113 it('Allows base 6 entropy to be entered', function(done
) {
2114 testEntropyType(done
, "012345", "base 6");
2117 // Base 6 dice entropy can be entered
2118 it('Allows base 6 dice entropy to be entered', function(done
) {
2119 testEntropyType(done
, "123456", "base 6 (dice)");
2122 // Base 10 entropy can be entered
2123 it('Allows base 10 entropy to be entered', function(done
) {
2124 testEntropyType(done
, "789", "base 10");
2127 // Hexadecimal entropy can be entered
2128 it('Allows hexadecimal entropy to be entered', function(done
) {
2129 testEntropyType(done
, "abcdef", "hexadecimal");
2132 // Dice entropy value is shown as the converted base 6 value
2133 // ie 123456 is converted to 123450
2134 it('Shows dice entropy as base 6', function(done
) {
2135 driver
.findElement(By
.css('.use-entropy'))
2137 driver
.findElement(By
.css('.entropy'))
2138 .sendKeys("123456");
2139 driver
.sleep(generateDelay
).then(function() {
2140 driver
.findElement(By
.css('.entropy-container'))
2142 .then(function(text
) {
2143 expect(text
).toMatch(/Filtered Entropy
\s
+123450/);
2149 // The number of bits of entropy accumulated is shown
2150 it("Shows the number of bits of entropy for 20 bits of binary", function(done
) {
2151 testEntropyBits(done
, "0000 0000 0000 0000 0000", "20");
2153 it("Shows the number of bits of entropy for 1 bit of binary", function(done
) {
2154 testEntropyBits(done
, "0", "1");
2156 it("Shows the number of bits of entropy for 4 bits of binary", function(done
) {
2157 testEntropyBits(done
, "0000", "4");
2159 it("Shows the number of bits of entropy for 1 character of base 6 (dice)", function(done
) {
2160 // 6 in card is 0 in base 6, 0 in base 6 is 2.6 bits (rounded down to 2 bits)
2161 testEntropyBits(done
, "6", "2");
2163 it("Shows the number of bits of entropy for 1 character of base 10 with 3 bits", function(done
) {
2164 // 7 in base 10 is 111 in base 2, no leading zeros
2165 testEntropyBits(done
, "7", "3");
2167 it("Shows the number of bits of entropy for 1 character of base 10 with 4 bis", function(done
) {
2168 testEntropyBits(done
, "8", "4");
2170 it("Shows the number of bits of entropy for 1 character of hex", function(done
) {
2171 testEntropyBits(done
, "F", "4");
2173 it("Shows the number of bits of entropy for 2 characters of base 10", function(done
) {
2174 testEntropyBits(done
, "29", "6");
2176 it("Shows the number of bits of entropy for 2 characters of hex", function(done
) {
2177 testEntropyBits(done
, "0A", "8");
2179 it("Shows the number of bits of entropy for 2 characters of hex with 3 leading zeros", function(done
) {
2180 // hex is always multiple of 4 bits of entropy
2181 testEntropyBits(done
, "1A", "8");
2183 it("Shows the number of bits of entropy for 2 characters of hex with 2 leading zeros", function(done
) {
2184 testEntropyBits(done
, "2A", "8");
2186 it("Shows the number of bits of entropy for 2 characters of hex with 1 leading zero", function(done
) {
2187 testEntropyBits(done
, "4A", "8");
2189 it("Shows the number of bits of entropy for 2 characters of hex with no leading zeros", function(done
) {
2190 testEntropyBits(done
, "8A", "8");
2192 it("Shows the number of bits of entropy for 2 characters of hex starting with F", function(done
) {
2193 testEntropyBits(done
, "FA", "8");
2195 it("Shows the number of bits of entropy for 4 characters of hex with leading zeros", function(done
) {
2196 testEntropyBits(done
, "000A", "16");
2198 it("Shows the number of bits of entropy for 4 characters of base 6", function(done
) {
2199 testEntropyBits(done
, "5555", "11");
2201 it("Shows the number of bits of entropy for 4 characters of base 6 dice", function(done
) {
2202 // uses dice, so entropy is actually 0000 in base 6, which is 4 lots of
2203 // 2.58 bits, which is 10.32 bits (rounded down to 10 bits)
2204 testEntropyBits(done
, "6666", "10");
2206 it("Shows the number of bits of entropy for 4 charactes of base 10", function(done
) {
2207 // Uses base 10, which is 4 lots of 3.32 bits, which is 13.3 bits (rounded
2209 testEntropyBits(done
, "2227", "13");
2211 it("Shows the number of bits of entropy for 4 characters of hex with 2 leading zeros", function(done
) {
2212 testEntropyBits(done
, "222F", "16");
2214 it("Shows the number of bits of entropy for 4 characters of hex starting with F", function(done
) {
2215 testEntropyBits(done
, "FFFF", "16");
2217 it("Shows the number of bits of entropy for 10 characters of base 10", function(done
) {
2218 // 10 events at 3.32 bits per event
2219 testEntropyBits(done
, "0000101017", "33");
2221 it("Shows the number of bits of entropy for a full deck of cards", function(done
) {
2222 // cards are not replaced, so a full deck is not 52^52 entropy which is 296
2223 // bits, it's 52!, which is 225 bits
2224 testEntropyBits(done
, "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks", "225");
2227 it("Shows details about the entered entropy", function(done
) {
2228 testEntropyFeedback(done
,
2232 type: "hexadecimal",
2236 strength: "less than a second",
2240 it("Shows details about the entered entropy", function(done
) {
2241 testEntropyFeedback(done
,
2243 entropy: "AAAAAAAA",
2244 filtered: "AAAAAAAA",
2245 type: "hexadecimal",
2249 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2253 it("Shows details about the entered entropy", function(done
) {
2254 testEntropyFeedback(done
,
2256 entropy: "AAAAAAAA B",
2257 filtered: "AAAAAAAAB",
2258 type: "hexadecimal",
2262 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2266 it("Shows details about the entered entropy", function(done
) {
2267 testEntropyFeedback(done
,
2269 entropy: "AAAAAAAA BBBBBBBB",
2270 filtered: "AAAAAAAABBBBBBBB",
2271 type: "hexadecimal",
2275 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2279 it("Shows details about the entered entropy", function(done
) {
2280 testEntropyFeedback(done
,
2282 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC",
2283 filtered: "AAAAAAAABBBBBBBBCCCCCCCC",
2284 type: "hexadecimal",
2288 strength: "less than a second",
2292 it("Shows details about the entered entropy", function(done
) {
2293 testEntropyFeedback(done
,
2295 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDD",
2296 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDD",
2297 type: "hexadecimal",
2301 strength: "2 minutes",
2305 it("Shows details about the entered entropy", function(done
) {
2306 testEntropyFeedback(done
,
2308 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA",
2309 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDA",
2310 type: "hexadecimal",
2318 it("Shows details about the entered entropy", function(done
) {
2319 testEntropyFeedback(done
,
2321 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA EEEEEEEE",
2322 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDAEEEEEEEE",
2323 type: "hexadecimal",
2327 strength: "3 years",
2331 it("Shows details about the entered entropy", function(done
) {
2332 testEntropyFeedback(done
,
2334 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA EEEEEEEE FFFFFFFF",
2335 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDAEEEEEEEEFFFFFFFF",
2336 type: "hexadecimal",
2340 strength: "centuries",
2344 it("Shows details about the entered entropy", function(done
) {
2345 testEntropyFeedback(done
,
2352 strength: "less than a second",
2356 it("Shows details about the entered entropy", function(done
) {
2357 testEntropyFeedback(done
,
2359 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2360 type: "card (full deck)",
2364 strength: "centuries",
2368 it("Shows details about the entered entropy", function(done
) {
2369 testEntropyFeedback(done
,
2371 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks3d",
2372 type: "card (full deck, 1 duplicate: 3d)",
2376 strength: "centuries",
2380 it("Shows details about the entered entropy", function(done
) {
2381 testEntropyFeedback(done
,
2383 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqs3d4d",
2384 type: "card (2 duplicates: 3d 4d, 1 missing: KS)",
2388 strength: "centuries",
2392 it("Shows details about the entered entropy", function(done
) {
2393 testEntropyFeedback(done
,
2395 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqs3d4d5d6d",
2396 type: "card (4 duplicates: 3d 4d 5d..., 1 missing: KS)",
2400 strength: "centuries",
2404 it("Shows details about the entered entropy", function(done
) {
2405 testEntropyFeedback(done
,
2406 // Next test was throwing uncaught error in zxcvbn
2407 // Also tests 451 bits, ie Math.log2(52!)*2 = 225.58 * 2
2409 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsksac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2410 type: "card (full deck, 52 duplicates: ac 2c 3c...)",
2414 strength: "centuries",
2418 it("Shows details about the entered entropy", function(done
) {
2419 testEntropyFeedback(done
,
2420 // Case insensitivity to duplicate cards
2423 type: "card (1 duplicate: AS)",
2427 strength: "less than a second",
2431 it("Shows details about the entered entropy", function(done
) {
2432 testEntropyFeedback(done
,
2435 type: "card (1 duplicate: as)",
2439 strength: "less than a second",
2443 it("Shows details about the entered entropy", function(done
) {
2444 testEntropyFeedback(done
,
2445 // Missing cards are detected
2447 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2448 type: "card (1 missing: 9C)",
2452 strength: "centuries",
2456 it("Shows details about the entered entropy", function(done
) {
2457 testEntropyFeedback(done
,
2459 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2460 type: "card (2 missing: 9C 5D)",
2464 strength: "centuries",
2468 it("Shows details about the entered entropy", function(done
) {
2469 testEntropyFeedback(done
,
2471 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d7d8d9dtdjd kdah2h3h 5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2472 type: "card (4 missing: 9C 5D QD...)",
2476 strength: "centuries",
2480 it("Shows details about the entered entropy", function(done
) {
2481 testEntropyFeedback(done
,
2482 // More than six missing cards does not show message
2484 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d 8d9d jd kdah2h3h 5h6h7h8h9hthjhqhkh 2s3s4s5s6s7s8s9stsjsqsks",
2489 strength: "centuries",
2493 it("Shows details about the entered entropy", function(done
) {
2494 testEntropyFeedback(done
,
2495 // Multiple decks of cards increases bits per event
2500 bitsPerEvent: "4.34",
2504 it("Shows details about the entered entropy", function(done
) {
2505 testEntropyFeedback(done
,
2510 bitsPerEvent: "4.80",
2514 it("Shows details about the entered entropy", function(done
) {
2515 testEntropyFeedback(done
,
2520 bitsPerEvent: "5.01",
2524 it("Shows details about the entered entropy", function(done
) {
2525 testEntropyFeedback(done
,
2527 entropy: "3d3d3d3d",
2530 bitsPerEvent: "5.14",
2534 it("Shows details about the entered entropy", function(done
) {
2535 testEntropyFeedback(done
,
2537 entropy: "3d3d3d3d3d",
2540 bitsPerEvent: "5.22",
2544 it("Shows details about the entered entropy", function(done
) {
2545 testEntropyFeedback(done
,
2547 entropy: "3d3d3d3d3d3d",
2550 bitsPerEvent: "5.28",
2554 it("Shows details about the entered entropy", function(done
) {
2555 testEntropyFeedback(done
,
2557 entropy: "3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d",
2560 bitsPerEvent: "5.59",
2561 strength: 'less than a second - Repeats like "abcabcabc" are only slightly harder to guess than "abc"',
2566 // Entropy is truncated from the left
2567 it('Truncates entropy from the left', function(done
) {
2568 // Truncate from left means 0000 is removed from the start
2569 // which gives mnemonic 'avocado zoo zone'
2570 // not 1111 removed from the end
2571 // which gives the mnemonic 'abstract zoo zoo'
2572 var entropy
= "00000000 00000000 00000000 00000000";
2573 entropy
+= "11111111 11111111 11111111 1111"; // Missing last byte
2574 driver
.findElement(By
.css('.use-entropy'))
2576 driver
.findElement(By
.css('.entropy'))
2578 driver
.sleep(generateDelay
).then(function() {
2579 driver
.findElement(By
.css(".phrase"))
2580 .getAttribute("value").then(function(phrase
) {
2581 expect(phrase
).toBe("avocado zoo zone");
2587 // Very large entropy results in very long mnemonics
2588 it('Converts very long entropy to very long mnemonics', function(done
) {
2590 for (var i
=0; i
<33; i
++) {
2591 entropy
+= "AAAAAAAA"; // 3 words * 33 iterations = 99 words
2593 driver
.findElement(By
.css('.use-entropy'))
2595 driver
.findElement(By
.css('.entropy'))
2597 driver
.sleep(generateDelay
).then(function() {
2598 driver
.findElement(By
.css(".phrase"))
2599 .getAttribute("value").then(function(phrase
) {
2600 var wordCount
= phrase
.split(/\s+/g).length
;
2601 expect(wordCount
).toBe(99);
2607 // Is compatible with bip32jp entropy
2608 // https://bip32jp.github.io/english/index.html
2610 // Is incompatible with:
2612 it('Is compatible with bip32jp.github.io', function(done
) {
2613 var entropy
= "543210543210543210543210543210543210543210543210543210543210543210543210543210543210543210543210543";
2614 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";
2615 driver
.findElement(By
.css('.use-entropy'))
2617 driver
.findElement(By
.css('.entropy'))
2619 driver
.sleep(generateDelay
).then(function() {
2620 driver
.findElement(By
.css(".phrase"))
2621 .getAttribute("value").then(function(phrase
) {
2622 expect(phrase
).toBe(expectedPhrase
);
2628 // Blank entropy does not generate mnemonic or addresses
2629 it('Does not generate mnemonic for blank entropy', function(done
) {
2630 driver
.findElement(By
.css('.use-entropy'))
2632 driver
.findElement(By
.css('.entropy'))
2634 // check there is no mnemonic
2635 driver
.sleep(generateDelay
).then(function() {
2636 driver
.findElement(By
.css(".phrase"))
2637 .getAttribute("value").then(function(phrase
) {
2638 expect(phrase
).toBe("");
2639 // check there is no mnemonic
2640 driver
.findElements(By
.css(".address"))
2641 .then(function(addresses
) {
2642 expect(addresses
.length
).toBe(0);
2643 // Check the feedback says 'blank entropy'
2644 driver
.findElement(By
.css(".feedback"))
2646 .then(function(feedbackText
) {
2647 expect(feedbackText
).toBe("Blank entropy");
2655 // Mnemonic length can be selected even for weak entropy
2656 it('Allows selection of mnemonic length even for weak entropy', function(done
) {
2657 driver
.findElement(By
.css('.use-entropy'))
2659 driver
.executeScript(function() {
2660 $(".mnemonic-length").val("18").trigger("change");
2662 driver
.findElement(By
.css('.entropy'))
2663 .sendKeys("012345");
2664 driver
.sleep(generateDelay
).then(function() {
2665 driver
.findElement(By
.css(".phrase"))
2666 .getAttribute("value").then(function(phrase
) {
2667 var wordCount
= phrase
.split(/\s+/g).length
;
2668 expect(wordCount
).toBe(18);
2675 // https://github.com/iancoleman/bip39/issues/33
2676 // Final cards should contribute entropy
2677 it('Uses as much entropy as possible for the mnemonic', function(done
) {
2678 driver
.findElement(By
.css('.use-entropy'))
2680 driver
.findElement(By
.css('.entropy'))
2681 .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");
2682 driver
.sleep(generateDelay
).then(function() {
2684 driver
.findElement(By
.css(".phrase"))
2685 .getAttribute("value").then(function(originalPhrase
) {
2686 // Set the last 12 cards to be AS
2687 driver
.findElement(By
.css('.entropy'))
2689 driver
.findElement(By
.css('.entropy'))
2690 .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");
2691 driver
.sleep(generateDelay
).then(function() {
2693 driver
.findElement(By
.css(".phrase"))
2694 .getAttribute("value").then(function(newPhrase
) {
2695 expect(originalPhrase
).not
.toEqual(newPhrase
);
2704 // https://github.com/iancoleman/bip39/issues/35
2706 // TODO this doesn't work in selenium with firefox
2707 // see https://stackoverflow.com/q/40360223
2708 it('Shows a qr code on hover for the phrase', function(done
) {
2709 if (browser
== "firefox") {
2710 pending("Selenium + Firefox bug for mouseMove, see https://stackoverflow.com/q/40360223");
2712 // generate a random mnemonic
2713 var generateEl
= driver
.findElement(By
.css('.generate'));
2715 // toggle qr to show (hidden by default)
2716 var phraseEl
= driver
.findElement(By
.css(".phrase"));
2718 var rootKeyEl
= driver
.findElement(By
.css(".root-key"));
2719 driver
.sleep(generateDelay
).then(function() {
2720 // hover over the root key
2721 driver
.actions().mouseMove(rootKeyEl
).perform().then(function() {
2722 // check the qr code shows
2723 driver
.executeScript(function() {
2724 return $(".qr-container").find("canvas").length
> 0;
2726 .then(function(qrShowing
) {
2727 expect(qrShowing
).toBe(true);
2728 // hover away from the phrase
2729 driver
.actions().mouseMove(generateEl
).perform().then(function() {;
2730 // check the qr code hides
2731 driver
.executeScript(function() {
2732 return $(".qr-container").find("canvas").length
== 0;
2734 .then(function(qrHidden
) {
2735 expect(qrHidden
).toBe(true);
2744 // BIP44 account extendend private key is shown
2745 // github issue 37 - compatibility with electrum
2746 it('Shows the bip44 account extended private key', function(done
) {
2747 driver
.findElement(By
.css(".phrase"))
2748 .sendKeys("abandon abandon ability");
2749 driver
.sleep(generateDelay
).then(function() {
2750 driver
.findElement(By
.css("#bip44 .account-xprv"))
2751 .getAttribute("value")
2752 .then(function(xprv
) {
2753 expect(xprv
).toBe("xprv9yzrnt4zWVJUr1k2VxSPy9ettKz5PpeDMgaVG7UKedhqnw1tDkxP2UyYNhuNSumk2sLE5ctwKZs9vwjsq3e1vo9egCK6CzP87H2cVYXpfwQ");
2759 // BIP44 account extendend public key is shown
2760 // github issue 37 - compatibility with electrum
2761 it('Shows the bip44 account extended public key', function(done
) {
2762 driver
.findElement(By
.css(".phrase"))
2763 .sendKeys("abandon abandon ability");
2764 driver
.sleep(generateDelay
).then(function() {
2765 driver
.findElement(By
.css("#bip44 .account-xpub"))
2766 .getAttribute("value")
2767 .then(function(xprv
) {
2768 expect(xprv
).toBe("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2775 // BIP32 root key can be set as an xpub
2776 it('Generates addresses from xpub as bip32 root key', function(done
) {
2777 driver
.findElement(By
.css('#bip32-tab a'))
2779 // set xpub for account 0 of bip44 for 'abandon abandon ability'
2780 driver
.findElement(By
.css("#root-key"))
2781 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2782 driver
.sleep(generateDelay
).then(function() {
2783 // check the addresses are generated
2784 getFirstAddress(function(address
) {
2785 expect(address
).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug");
2786 // check the xprv key is not set
2787 driver
.findElement(By
.css(".extended-priv-key"))
2788 .getAttribute("value")
2789 .then(function(xprv
) {
2790 expect(xprv
).toBe("NA");
2791 // check the private key is not set
2792 driver
.findElements(By
.css(".privkey"))
2793 .then(function(els
) {
2796 .then(function(privkey
) {
2797 expect(xprv
).toBe("NA");
2807 // xpub for bip32 root key will not work with hardened derivation paths
2808 it('Shows error for hardened derivation paths with xpub root key', function(done
) {
2809 // set xpub for account 0 of bip44 for 'abandon abandon ability'
2810 driver
.findElement(By
.css("#root-key"))
2811 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2812 driver
.sleep(feedbackDelay
).then(function() {
2813 // Check feedback is correct
2814 driver
.findElement(By
.css('.feedback'))
2816 .then(function(feedback
) {
2817 var msg
= "Hardened derivation path is invalid with xpub key";
2818 expect(feedback
).toBe(msg
);
2819 // Check no addresses are shown
2820 driver
.findElements(By
.css('.addresses tr'))
2821 .then(function(rows
) {
2822 expect(rows
.length
).toBe(0);
2830 // no root key shows feedback
2831 it('Shows feedback for no root key', function(done
) {
2832 // set xpub for account 0 of bip44 for 'abandon abandon ability'
2833 driver
.findElement(By
.css('#bip32-tab a'))
2835 driver
.sleep(feedbackDelay
).then(function() {
2836 // Check feedback is correct
2837 driver
.findElement(By
.css('.feedback'))
2839 .then(function(feedback
) {
2840 expect(feedback
).toBe("Invalid root key");
2847 // display error switching tabs while addresses are generating
2848 it('Can change details while old addresses are still being generated', function(done
) {
2849 // Set to generate 199 more addresses.
2850 // This will take a long time allowing a new set of addresses to be
2851 // generated midway through this lot.
2852 // The newly generated addresses should not include any from the old set.
2853 // Any more than 199 will show an alert which needs to be accepted.
2854 driver
.findElement(By
.css('.rows-to-add'))
2856 driver
.findElement(By
.css('.rows-to-add'))
2859 driver
.findElement(By
.css('.phrase'))
2860 .sendKeys("abandon abandon ability");
2861 driver
.sleep(generateDelay
).then(function() {
2862 // change tabs which should cancel the previous generating
2863 driver
.findElement(By
.css('.rows-to-add'))
2865 driver
.findElement(By
.css('.rows-to-add'))
2867 driver
.findElement(By
.css('#bip32-tab a'))
2869 driver
.sleep(generateDelay
).then(function() {
2870 driver
.findElements(By
.css('.index'))
2871 .then(function(els
) {
2872 // check the derivation paths have the right quantity
2873 expect(els
.length
).toBe(20);
2874 // check the derivation paths are in order
2875 testRowsAreInCorrectOrder(done
);
2879 }, generateDelay
+ 5000);
2882 // padding for binary should give length with multiple of 256
2883 // hashed entropy 1111 is length 252, so requires 4 leading zeros
2884 // prior to issue 49 it would only generate 2 leading zeros, ie missing 2
2885 it('Pads hashed entropy with leading zeros', function(done
) {
2886 driver
.findElement(By
.css('.use-entropy'))
2888 driver
.executeScript(function() {
2889 $(".mnemonic-length").val("15").trigger("change");
2891 driver
.findElement(By
.css('.entropy'))
2893 driver
.sleep(generateDelay
).then(function() {
2894 driver
.findElement(By
.css('.phrase'))
2895 .getAttribute("value")
2896 .then(function(phrase
) {
2897 expect(phrase
).toBe("avocado valid quantum cross link predict excuse edit street able flame large galaxy ginger nuclear");
2903 // Github pull request 55
2904 // https://github.com/iancoleman/bip39/pull/55
2906 it('Can set the derivation path on bip32 tab for bitcoincore', function(done
) {
2907 testClientSelect(done
, {
2909 bip32path: "m/0'/0'",
2910 useHardenedAddresses: "true",
2913 it('Can set the derivation path on bip32 tab for multibit', function(done
) {
2914 testClientSelect(done
, {
2916 bip32path: "m/0'/0",
2917 useHardenedAddresses: null,
2920 it('Can set the derivation path on bip32 tab for coinomi/ledger', function(done
) {
2921 testClientSelect(done
, {
2923 bip32path: "m/44'/0'/0'",
2924 useHardenedAddresses: null,
2929 // https://github.com/iancoleman/bip39/issues/58
2930 // bip32 derivation is correct, does not drop leading zeros
2932 // https://medium.com/@alexberegszaszi/why-do-my-bip32-wallets-disagree-6f3254cc5846
2933 it('Retains leading zeros for bip32 derivation', function(done
) {
2934 driver
.findElement(By
.css(".phrase"))
2935 .sendKeys("fruit wave dwarf banana earth journey tattoo true farm silk olive fence");
2936 driver
.findElement(By
.css(".passphrase"))
2937 .sendKeys("banana");
2938 driver
.sleep(generateDelay
).then(function() {
2939 getFirstAddress(function(address
) {
2940 // Note that bitcore generates an incorrect address
2941 // 13EuKhffWkBE2KUwcbkbELZb1MpzbimJ3Y
2942 // see the medium.com link above for more details
2943 expect(address
).toBe("17rxURoF96VhmkcEGCj5LNQkmN9HVhWb7F");
2950 // Japanese mnemonics generate incorrect bip32 seed
2951 // BIP39 seed is set from phrase
2952 it('Generates correct seed for Japanese mnemonics', function(done
) {
2953 driver
.findElement(By
.css(".phrase"))
2954 .sendKeys("あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あおぞら");
2955 driver
.findElement(By
.css(".passphrase"))
2956 .sendKeys("メートルガバヴァぱばぐゞちぢ十人十色");
2957 driver
.sleep(generateDelay
).then(function() {
2958 driver
.findElement(By
.css(".seed"))
2959 .getAttribute("value")
2960 .then(function(seed
) {
2961 expect(seed
).toBe("a262d6fb6122ecf45be09c50492b31f92e9beb7d9a845987a02cefda57a15f9c467a17872029a9e92299b5cbdf306e3a0ee620245cbd508959b6cb7ca637bd55");
2967 // BIP49 official test vectors
2968 // https://github.com/bitcoin/bips/blob/master/bip-0049.mediawiki#test-vectors
2969 it('Generates BIP49 addresses matching the official test vectors', function(done
) {
2970 driver
.findElement(By
.css('#bip49-tab a'))
2972 selectNetwork("BTC - Bitcoin Testnet");
2973 driver
.findElement(By
.css(".phrase"))
2974 .sendKeys("abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about");
2975 driver
.sleep(generateDelay
).then(function() {
2976 getFirstAddress(function(address
) {
2977 expect(address
).toBe("2Mww8dCYPUpKHofjgcXcBCEGmniw9CoaiD2");
2983 // BIP49 derivation path is shown
2984 it('Shows the bip49 derivation path', function(done
) {
2985 driver
.findElement(By
.css('#bip49-tab a'))
2987 driver
.findElement(By
.css(".phrase"))
2988 .sendKeys("abandon abandon ability");
2989 driver
.sleep(generateDelay
).then(function() {
2990 driver
.findElement(By
.css('#bip49 .path'))
2991 .getAttribute("value")
2992 .then(function(path
) {
2993 expect(path
).toBe("m/49'/0'/0'/0");
2999 // BIP49 extended private key is shown
3000 it('Shows the bip49 extended private key', function(done
) {
3001 driver
.findElement(By
.css('#bip49-tab a'))
3003 driver
.findElement(By
.css(".phrase"))
3004 .sendKeys("abandon abandon ability");
3005 driver
.sleep(generateDelay
).then(function() {
3006 driver
.findElement(By
.css('.extended-priv-key'))
3007 .getAttribute("value")
3008 .then(function(xprv
) {
3009 expect(xprv
).toBe("yprvALYB4DYRG6CzzVgzQZwwqjAA2wjBGC3iEd7KYYScpoDdmf75qMRWZWxoFcRXBJjgEXdFqJ9vDRGRLJQsrL22Su5jMbNFeM9vetaGVqy9Qy2");
3015 // BIP49 extended public key is shown
3016 it('Shows the bip49 extended public key', function(done
) {
3017 driver
.findElement(By
.css('#bip49-tab a'))
3019 driver
.findElement(By
.css(".phrase"))
3020 .sendKeys("abandon abandon ability");
3021 driver
.sleep(generateDelay
).then(function() {
3022 driver
.findElement(By
.css('.extended-pub-key'))
3023 .getAttribute("value")
3024 .then(function(xprv
) {
3025 expect(xprv
).toBe("ypub6ZXXTj5K6TmJCymTWbUxCs6tayZffemZbr2vLvrEP8kceTSENtjm7KHH6thvAKxVar9fGe8rgsPEX369zURLZ68b4f7Vexz7RuXsjQ69YDt");
3031 // BIP49 account field changes address list
3032 it('Can set the bip49 account field', function(done
) {
3033 driver
.findElement(By
.css('#bip49-tab a'))
3035 driver
.findElement(By
.css('#bip49 .account'))
3037 driver
.findElement(By
.css('#bip49 .account'))
3039 driver
.findElement(By
.css(".phrase"))
3040 .sendKeys("abandon abandon ability");
3041 driver
.sleep(generateDelay
).then(function() {
3042 getFirstAddress(function(address
) {
3043 expect(address
).toBe("381wg1GGN4rP88rNC9v7QWsiww63yLVPsn");
3049 // BIP49 change field changes address list
3050 it('Can set the bip49 change field', function(done
) {
3051 driver
.findElement(By
.css('#bip49-tab a'))
3053 driver
.findElement(By
.css('#bip49 .change'))
3055 driver
.findElement(By
.css('#bip49 .change'))
3057 driver
.findElement(By
.css(".phrase"))
3058 .sendKeys("abandon abandon ability");
3059 driver
.sleep(generateDelay
).then(function() {
3060 getFirstAddress(function(address
) {
3061 expect(address
).toBe("3PEM7MiKed5konBoN66PQhK8r3hjGhy9dT");
3067 // BIP49 account extendend private key is shown
3068 it('Shows the bip49 account extended private key', function(done
) {
3069 driver
.findElement(By
.css('#bip49-tab a'))
3071 driver
.findElement(By
.css(".phrase"))
3072 .sendKeys("abandon abandon ability");
3073 driver
.sleep(generateDelay
).then(function() {
3074 driver
.findElement(By
.css('#bip49 .account-xprv'))
3075 .getAttribute("value")
3076 .then(function(xprv
) {
3077 expect(xprv
).toBe("yprvAHtB1M5Wp675aLzFy9TJYK2mSsLkg6mcBRh5DZTR7L4EnYSmYPaL63KFA4ycg1PngW5LfkmejxzosCs17TKZMpRFKc3z5SJar6QAKaFcaZL");
3083 // BIP49 account extendend public key is shown
3084 it('Shows the bip49 account extended public key', function(done
) {
3085 driver
.findElement(By
.css('#bip49-tab a'))
3087 driver
.findElement(By
.css(".phrase"))
3088 .sendKeys("abandon abandon ability");
3089 driver
.sleep(generateDelay
).then(function() {
3090 driver
.findElement(By
.css('#bip49 .account-xpub'))
3091 .getAttribute("value")
3092 .then(function(xprv
) {
3093 expect(xprv
).toBe("ypub6WsXQrcQeTfNnq4j5AzJuSyVzuBF5ZVTYecg1ws2ffbDfLmv5vtadqdj1NgR6C6gufMpMfJpHxvb6JEQKvETVNWCRanNedfJtnTchZiJtsL");
3099 // Test selecting coin where bip49 is unavailable (eg CLAM)
3100 it('Shows an error on bip49 tab for coins without bip49', function(done
) {
3101 driver
.findElement(By
.css('#bip49-tab a'))
3103 driver
.findElement(By
.css(".phrase"))
3104 .sendKeys("abandon abandon ability");
3105 driver
.sleep(generateDelay
).then(function() {
3106 selectNetwork("CLAM - Clams");
3107 // bip49 available is hidden
3108 driver
.findElement(By
.css('#bip49 .available'))
3109 .getAttribute("class")
3110 .then(function(classes
) {
3111 expect(classes
).toContain("hidden");
3112 // bip49 unavailable is shown
3113 driver
.findElement(By
.css('#bip49 .unavailable'))
3114 .getAttribute("class")
3115 .then(function(classes
) {
3116 expect(classes
).not
.toContain("hidden");
3117 // check there are no addresses shown
3118 driver
.findElements(By
.css('.addresses tr'))
3119 .then(function(rows
) {
3120 expect(rows
.length
).toBe(0);
3121 // check the derived private key is blank
3122 driver
.findElement(By
.css('.extended-priv-key'))
3123 .getAttribute("value")
3124 .then(function(xprv
) {
3125 expect(xprv
).toBe('');
3126 // check the derived public key is blank
3127 driver
.findElement(By
.css('.extended-pub-key'))
3128 .getAttribute("value")
3129 .then(function(xpub
) {
3130 expect(xpub
).toBe('');
3141 // Cleared mnemonic and root key still allows addresses to be generated
3142 // https://github.com/iancoleman/bip39/issues/43
3143 it('Clears old root keys from memory when mnemonic is cleared', function(done
) {
3145 driver
.findElement(By
.css(".phrase"))
3146 .sendKeys("abandon abandon ability");
3147 driver
.sleep(generateDelay
).then(function() {
3148 // clear the mnemonic and root key
3149 // using selenium .clear() doesn't seem to trigger the 'input' event
3150 // so clear it using keys instead
3151 driver
.findElement(By
.css('.phrase'))
3152 .sendKeys(Key
.CONTROL
,"a");
3153 driver
.findElement(By
.css('.phrase'))
3154 .sendKeys(Key
.DELETE
);
3155 driver
.findElement(By
.css('.root-key'))
3156 .sendKeys(Key
.CONTROL
,"a");
3157 driver
.findElement(By
.css('.root-key'))
3158 .sendKeys(Key
.DELETE
);
3159 driver
.sleep(generateDelay
).then(function() {
3160 // try to generate more addresses
3161 driver
.findElement(By
.css('.more'))
3163 driver
.sleep(generateDelay
).then(function() {
3164 driver
.findElements(By
.css(".addresses tr"))
3165 .then(function(els
) {
3166 // check there are no addresses shown
3167 expect(els
.length
).toBe(0);
3176 // error trying to generate addresses from xpub with hardened derivation
3177 it('Shows error for hardened addresses with xpub root key', function(done
) {
3178 driver
.findElement(By
.css('#bip32-tab a'))
3180 driver
.executeScript(function() {
3181 $(".hardened-addresses").prop("checked", true);
3183 // set xpub for account 0 of bip44 for 'abandon abandon ability'
3184 driver
.findElement(By
.css("#root-key"))
3185 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
3186 driver
.sleep(feedbackDelay
).then(function() {
3187 // Check feedback is correct
3188 driver
.findElement(By
.css('.feedback'))
3190 .then(function(feedback
) {
3191 var msg
= "Hardened derivation path is invalid with xpub key";
3192 expect(feedback
).toBe(msg
);
3198 // Litecoin uses ltub by default, and can optionally be set to xprv
3200 // https://github.com/iancoleman/bip39/issues/96
3201 // Issue with extended keys on Litecoin
3202 it('Uses ltub by default for litecoin, but can be set to xprv', function(done
) {
3203 driver
.findElement(By
.css('.phrase'))
3204 .sendKeys("abandon abandon ability");
3205 selectNetwork("LTC - Litecoin");
3206 driver
.sleep(generateDelay
).then(function() {
3207 // check the extended key is generated correctly
3208 driver
.findElement(By
.css('.root-key'))
3209 .getAttribute("value")
3210 .then(function(rootKey
) {
3211 expect(rootKey
).toBe("Ltpv71G8qDifUiNesiPqf6h5V6eQ8ic77oxQiYtawiACjBEx3sTXNR2HGDGnHETYxESjqkMLFBkKhWVq67ey1B2MKQXannUqNy1RZVHbmrEjnEU");
3212 // set litecoin to use ltub
3213 driver
.executeScript(function() {
3214 $(".litecoin-use-ltub").prop("checked", false);
3215 $(".litecoin-use-ltub").trigger("change");
3217 driver
.sleep(generateDelay
).then(function() {
3218 driver
.findElement(By
.css('.root-key'))
3219 .getAttribute("value")
3220 .then(function(rootKey
) {
3221 expect(rootKey
).toBe("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
3230 // https://github.com/iancoleman/bip39/issues/99#issuecomment-327094159
3231 // "warn me emphatically when they have detected invalid input" to the entropy field
3232 // A warning is shown when entropy is filtered and discarded
3233 it('Warns when entropy is filtered and discarded', function(done
) {
3234 driver
.findElement(By
.css('.use-entropy'))
3236 // set entropy to have no filtered content
3237 driver
.findElement(By
.css('.entropy'))
3238 .sendKeys("00000000 00000000 00000000 00000000");
3239 driver
.sleep(generateDelay
).then(function() {
3240 // check the filter warning does not show
3241 driver
.findElement(By
.css('.entropy-container .filter-warning'))
3242 .getAttribute("class")
3243 .then(function(classes
) {
3244 expect(classes
).toContain("hidden");
3245 // set entropy to have some filtered content
3246 driver
.findElement(By
.css('.entropy'))
3247 .sendKeys("10000000 zxcvbn 00000000 00000000 00000000");
3248 driver
.sleep(entropyFeedbackDelay
).then(function() {
3249 // check the filter warning shows
3250 driver
.findElement(By
.css('.entropy-container .filter-warning'))
3251 .getAttribute("class")
3252 .then(function(classes
) {
3253 expect(classes
).not
.toContain("hidden");
3261 // Bitcoin Cash address can be set to use cashaddr format
3262 it('Can use cashaddr format for bitcoin cash addresses', function(done
) {
3263 driver
.executeScript(function() {
3264 $(".use-bch-cashaddr-addresses").prop("checked", true);
3266 driver
.findElement(By
.css('.phrase'))
3267 .sendKeys("abandon abandon ability");
3268 selectNetwork("BCH - Bitcoin Cash");
3269 driver
.sleep(generateDelay
).then(function() {
3270 getFirstAddress(function(address
) {
3271 expect(address
).toBe("bitcoincash:qzlquk7w4hkudxypl4fgv8x279r754dkvur7jpcsps");
3277 // Bitcoin Cash address can be set to use bitpay format
3278 it('Can use bitpay format for bitcoin cash addresses', function(done
) {
3279 driver
.executeScript(function() {
3280 $(".use-bch-bitpay-addresses").prop("checked", true);
3282 driver
.findElement(By
.css('.phrase'))
3283 .sendKeys("abandon abandon ability");
3284 selectNetwork("BCH - Bitcoin Cash");
3285 driver
.sleep(generateDelay
).then(function() {
3286 getFirstAddress(function(address
) {
3287 expect(address
).toBe("CZnpA9HPmvhuhLLPWJP8rNDpLUYXy1LXFk");
3293 // Bitcoin Cash address can be set to use legacy format
3294 it('Can use legacy format for bitcoin cash addresses', function(done
) {
3295 driver
.executeScript(function() {
3296 $(".use-bch-legacy-addresses").prop("checked", true);
3298 driver
.findElement(By
.css('.phrase'))
3299 .sendKeys("abandon abandon ability");
3300 selectNetwork("BCH - Bitcoin Cash");
3301 driver
.sleep(generateDelay
).then(function() {
3302 getFirstAddress(function(address
) {
3303 expect(address
).toBe("1JKvb6wKtsjNoCRxpZ4DGrbniML7z5U16A");
3309 // End of tests ported from old suit, so no more comments above each test now
3311 it('Can generate more addresses from a custom index', function(done
) {
3312 var expectedIndexes
= [
3313 0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,
3314 40,41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59
3316 driver
.findElement(By
.css('.phrase'))
3317 .sendKeys("abandon abandon ability");
3318 driver
.sleep(generateDelay
).then(function() {
3319 // Set start of next lot of rows to be from index 40
3320 // which means indexes 20-39 will not be in the table.
3321 driver
.findElement(By
.css('.more-rows-start-index'))
3323 driver
.findElement(By
.css('.more'))
3325 driver
.sleep(generateDelay
).then(function() {
3326 // Check actual indexes in the table match the expected pattern
3327 driver
.findElements(By
.css(".index"))
3328 .then(function(els
) {
3329 expect(els
.length
).toBe(expectedIndexes
.length
);
3330 var testRowAtIndex = function(i
) {
3331 if (i
>= expectedIndexes
.length
) {
3336 .then(function(actualPath
) {
3337 var noHardened
= actualPath
.replace(/'/g, "");
3338 var pathBits = noHardened.split("/")
3339 var lastBit = pathBits[pathBits.length-1];
3340 var actualIndex = parseInt(lastBit);
3341 var expectedIndex = expectedIndexes[i];
3342 expect(actualIndex).toBe(expectedIndex);
3343 testRowAtIndex(i+1);
3353 it('Can generate BIP141 addresses
with P2WPKH
-in-P2SH semanitcs
', function(done) {
3354 // Sourced from BIP49 official test specs
3355 driver.findElement(By.css('#bip141
-tab a
'))
3357 driver.findElement(By.css('.bip141
-path
'))
3359 driver.findElement(By.css('.bip141
-path
'))
3360 .sendKeys("m/49'/1'/0'/0");
3361 selectNetwork("BTC
- Bitcoin Testnet
");
3362 driver.findElement(By.css(".phrase
"))
3363 .sendKeys("abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about
");
3364 driver.sleep(generateDelay).then(function() {
3365 getFirstAddress(function(address) {
3366 expect(address).toBe("2Mww8dCYPUpKHofjgcXcBCEGmniw9CoaiD2
");
3372 it('Can generate BIP141 addresses with P2WPKH semanitcs', function(done) {
3373 // This result tested against bitcoinjs-lib test spec for segwit address
3374 // using the first private key of this mnemonic and default path m/0
3375 // https://github.com/bitcoinjs/bitcoinjs-lib/blob/9c8503cab0c6c30a95127042703bc18e8d28c76d/test/integration/addresses.js#L50
3376 // so whilst not directly comparable, substituting the private key produces
3377 // identical results between this tool and the bitcoinjs-lib test.
3378 // Private key generated is:
3379 // L3L8Nu9whawPBNLGtFqDhKut9DKKfG3CQoysupT7BimqVCZsLFNP
3380 driver.findElement(By.css('#bip141-tab a'))
3383 driver.executeScript(function() {
3384 $(".bip141
-semantics option
[selected
]").removeAttr("selected
");
3385 $(".bip141
-semantics option
").filter(function(i,e) {
3386 return $(e).html() == "P2WPKH
";
3387 }).prop("selected
", true);
3388 $(".bip141
-semantics
").trigger("change
");
3390 driver.findElement(By.css(".phrase
"))
3391 .sendKeys("abandon abandon ability
");
3392 driver.sleep(generateDelay).then(function() {
3393 getFirstAddress(function(address) {
3394 expect(address).toBe("bc1qfwu6a5a3evygrk8zvdxxvz4547lmpyx5vsfxe9
");
3400 it('Shows the entropy used by the PRNG when clicking generate', function(done) {
3401 driver.findElement(By.css('.generate')).click();
3402 driver.sleep(generateDelay).then(function() {
3403 driver.findElement(By.css('.entropy'))
3404 .getAttribute("value
")
3405 .then(function(entropy) {
3406 expect(entropy).not.toBe("");
3412 it('Shows the index of each word in the mnemonic', function(done) {
3413 driver.findElement(By.css('.phrase'))
3414 .sendKeys("abandon abandon ability
");
3415 driver.sleep(generateDelay).then(function() {
3416 driver.findElement(By.css('.use-entropy'))
3418 driver.findElement(By.css('.word-indexes'))
3420 .then(function(indexes) {
3421 expect(indexes).toBe("0, 0, 1");
3427 it('Shows the derivation path for bip84 tab', function(done) {
3428 driver.findElement(By.css('#bip84-tab a'))
3430 driver.findElement(By.css('.phrase'))
3431 .sendKeys('abandon abandon ability');
3432 driver.sleep(generateDelay).then(function() {
3433 driver.findElement(By.css('#bip84 .path'))
3434 .getAttribute("value
")
3435 .then(function(path) {
3436 expect(path).toBe("m
/84'/0'/0'/0");
3442 it('Shows the extended private key for bip84 tab', function(done) {
3443 driver.findElement(By.css('#bip84-tab a'))
3445 driver.findElement(By.css('.phrase'))
3446 .sendKeys('abandon abandon ability');
3447 driver.sleep(generateDelay).then(function() {
3448 driver.findElement(By.css('.extended-priv-key'))
3449 .getAttribute("value
")
3450 .then(function(path) {
3451 expect(path).toBe("zprvAev3RKrZ3QVKiUFCfdeMRen1BPDJgdNt1XpxiDy8acSs4kkAGTCvq7HeRYRNNpo8EtEjCFQBWavJwtCUR29y4TUCH4X5RXMcyq48uN8y9BP
");
3457 it('Shows the extended public key for bip84 tab', function(done) {
3458 driver.findElement(By.css('#bip84-tab a'))
3460 driver.findElement(By.css('.phrase'))
3461 .sendKeys('abandon abandon ability');
3462 driver.sleep(generateDelay).then(function() {
3463 driver.findElement(By.css('.extended-pub-key'))
3464 .getAttribute("value
")
3465 .then(function(path) {
3466 expect(path).toBe("zpub6suPpqPSsn3cvxKfmfBMnnijjR3o666jNkkZWcNk8wyqwZ5JozXBNuc8Gs7DB3uLwTDvGVTspVEAUQcEjKF3pZHgywVbubdTqbXTUg7usyx
");
3472 it('Changes the address list if bip84 account is changed', function(done) {
3473 driver.findElement(By.css('#bip84-tab a'))
3475 driver.findElement(By.css('#bip84 .account'))
3477 driver.findElement(By.css('.phrase'))
3478 .sendKeys('abandon abandon ability');
3479 driver.sleep(generateDelay).then(function() {
3480 getFirstAddress(function(address) {
3481 expect(address).toBe("bc1qp7vv669t2fy965jdzvqwrraana89ctd5ewc662
");
3487 it('Changes the address list if bip84 change is changed', function(done) {
3488 driver.findElement(By.css('#bip84-tab a'))
3490 driver.findElement(By.css('#bip84 .change'))
3492 driver.findElement(By.css('.phrase'))
3493 .sendKeys('abandon abandon ability');
3494 driver.sleep(generateDelay).then(function() {
3495 getFirstAddress(function(address) {
3496 expect(address).toBe("bc1qr39vj6rh06ff05m53uxq8uazehwhccswylhrs2
");
3502 it('Passes the official BIP84 test spec for rootpriv', function(done) {
3503 driver.findElement(By.css('#bip84-tab a'))
3505 driver.findElement(By.css('.phrase'))
3506 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3507 driver.sleep(generateDelay).then(function() {
3508 driver.findElement(By.css(".root
-key
"))
3509 .getAttribute("value
")
3510 .then(function(rootKey) {
3511 expect(rootKey).toBe("zprvAWgYBBk7JR8Gjrh4UJQ2uJdG1r3WNRRfURiABBE3RvMXYSrRJL62XuezvGdPvG6GFBZduosCc1YP5wixPox7zhZLfiUm8aunE96BBa4Kei5
");
3517 it('Passes the official BIP84 test spec for account 0 xprv', function(done) {
3518 driver.findElement(By.css('#bip84-tab a'))
3520 driver.findElement(By.css('.phrase'))
3521 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3522 driver.sleep(generateDelay).then(function() {
3523 driver.findElement(By.css("#bip84
.account
-xprv
"))
3524 .getAttribute("value
")
3525 .then(function(rootKey) {
3526 expect(rootKey).toBe("zprvAdG4iTXWBoARxkkzNpNh8r6Qag3irQB8PzEMkAFeTRXxHpbF9z4QgEvBRmfvqWvGp42t42nvgGpNgYSJA9iefm1yYNZKEm7z6qUWCroSQnE
");
3532 it('Passes the official BIP84 test spec for account 0 xpub', function(done) {
3533 driver.findElement(By.css('#bip84-tab a'))
3535 driver.findElement(By.css('.phrase'))
3536 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3537 driver.sleep(generateDelay).then(function() {
3538 driver.findElement(By.css("#bip84
.account
-xpub
"))
3539 .getAttribute("value
")
3540 .then(function(rootKey) {
3541 expect(rootKey).toBe("zpub6rFR7y4Q2AijBEqTUquhVz398htDFrtymD9xYYfG1m4wAcvPhXNfE3EfH1r1ADqtfSdVCToUG868RvUUkgDKf31mGDtKsAYz2oz2AGutZYs
");
3547 it('Passes the official BIP84 test spec for account 0 first address', function(done) {
3548 driver.findElement(By.css('#bip84-tab a'))
3550 driver.findElement(By.css('.phrase'))
3551 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3552 driver.sleep(generateDelay).then(function() {
3553 getFirstAddress(function(address) {
3554 expect(address).toBe("bc1qcr8te4kr609gcawutmrza0j4xv80jy8z306fyu
");
3560 it('Passes the official BIP84 test spec for account 0 first change address', function(done) {
3561 driver.findElement(By.css('#bip84-tab a'))
3563 driver.findElement(By.css('.phrase'))
3564 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3565 driver.findElement(By.css('#bip84 .change'))
3567 driver.sleep(generateDelay).then(function() {
3568 getFirstAddress(function(address) {
3569 expect(address).toBe("bc1q8c6fshw2dlwun7ekn9qwf37cu2rn755upcp6el
");
3575 it('Can display the table as csv', function(done) {
3576 var headings = "path
,address
,public key
,private key
";
3577 var row1 = "m
/44'/0'/0'/0/0,1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug
,033f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3
,L26cVSpWFkJ6aQkPkKmTzLqTdLJ923e6CzrVh9cmx21QHsoUmrEE
";
3578 var row20 = "m
/44'/0'/0'/0/19,1KhBy28XLAciXnnRvm71PvQJaETyrxGV55
,02b4b3e396434d8cdd20c03ac4aaa07387784d5d867b75987f516f5705ee68cb3a
,L4GrDrjReMsCAu5DkLXn79jSb95qR7Zfx7eshybCQZ1qL32MXJab
";
3579 driver.findElement(By.css('.phrase'))
3580 .sendKeys('abandon abandon ability');
3581 driver.sleep(generateDelay).then(function() {
3582 driver.findElement(By.css('.csv'))
3583 .getAttribute("value
")
3584 .then(function(csv) {
3585 expect(csv).toContain(headings);
3586 expect(csv).toContain(row1);
3587 expect(csv).toContain(row20);
3593 it('LeftPads ethereum keys that are less than 32 bytes', function(done) {
3594 // see https://github.com/iancoleman/bip39/issues/155
3595 selectNetwork("ETH
- Ethereum
");
3596 driver.findElement(By.css('#bip32-tab a'))
3598 driver.findElement(By.css('#bip32-path'))
3600 driver.findElement(By.css('#bip32-path'))
3601 .sendKeys("m
/44'/60'/0'");
3602 driver.findElement(By.css('.phrase'))
3603 .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');
3604 driver.sleep(generateDelay).then(function() {
3605 getFirstAddress(function(address) {
3606 expect(address).toBe("0x8943E785B4a5714FC87a3aFAad1eB1FeB602B118");
3612 it('Can encrypt private keys using BIP38', function(done) {
3613 // see https://github.com/iancoleman/bip39/issues/140
3614 driver.executeScript(function() {
3615 $(".use-bip38
").prop("checked
", true);
3617 driver.findElement(By.css('.bip38-password'))
3618 .sendKeys('bip38password');
3619 driver.findElement(By.css('.rows-to-add'))
3621 driver.findElement(By.css('.rows-to-add'))
3623 driver.findElement(By.css('.phrase'))
3624 .sendKeys('abandon abandon ability');
3625 driver.sleep(bip38delay).then(function() {
3627 getFirstRowValue(function(address) {
3628 expect(address).toBe("1NCvSdumA3ngMM9c4aqU56AM6rqXddfuXB
");
3630 getFirstRowValue(function(pubkey) {
3631 expect(pubkey).toBe("043f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3884a74447ea901729b1e73a999b7520e7cb55b4120e6432c64153ccab8a848e1
");
3633 getFirstRowValue(function(privkey) {
3634 expect(privkey).toBe("6PRNRiFnj1RoR3sXhymdCvoZCgnUHQpfupNdKkFbWJkwWQEKesWt1EDMDM
");
3640 }, bip38delay + 5000);
3642 it('Shows the checksum for the entropy', function(done) {
3643 driver.findElement(By.css('.use-entropy'))
3645 driver.findElement(By.css('.entropy'))
3646 .sendKeys("00000000000000000000000000000000");
3647 driver.sleep(generateDelay).then(function() {
3648 driver.findElement(By.css('.checksum'))
3650 .then(function(text) {
3651 expect(text).toBe("1");
3657 it('Shows the checksum for the entropy with the correct groupings', function(done) {
3658 driver.findElement(By.css('.use-entropy'))
3660 // create a checksum of 20 bits, which spans multiple words
3661 driver.findElement(By.css('.entropy'))
3662 .sendKeys("F000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
");
3663 driver.sleep(generateDelay).then(function() {
3664 driver.findElement(By.css('.checksum'))
3666 .then(function(text) {
3667 // first group is 9 bits, second group is 11
3668 expect(text).toBe("011010111 01110000110");
3674 it('Uses vprv for bitcoin testnet p2wpkh', function(done) {
3675 selectNetwork("BTC
- Bitcoin Testnet
");
3676 driver.findElement(By.css('#bip84-tab a'))
3678 driver.findElement(By.css('.phrase'))
3679 .sendKeys('abandon abandon ability');
3680 driver.sleep(generateDelay).then(function() {
3681 driver.findElement(By.css('.root-key'))
3682 .getAttribute("value
")
3683 .then(function(path) {
3684 expect(path).toBe("vprv9DMUxX4ShgxML9N2YV5CvWEebWrM9aJ5ULpbRRyzyWu6vs4BzTvbfFFrH41N5hVi7MYSfiugd765L3JmAfDM5po36Y8ouCKRDeYQwByCmS7
");
3690 it('Shows a warning if generating weak mnemonics', function(done) {
3691 driver.executeScript(function() {
3692 $(".strength option
[selected
]").removeAttr("selected
");
3693 $(".strength option
[value
=6]").prop("selected
", true);
3694 $(".strength
").trigger("change
");
3696 driver.findElement(By.css(".generate
-container
.warning
"))
3697 .getAttribute("class")
3698 .then(function(classes) {
3699 expect(classes).not.toContain("hidden
");
3704 it('Does not show a warning if generating strong mnemonics', function(done) {
3705 driver.executeScript(function() {
3706 $(".strength option
[selected
]").removeAttr("selected
");
3707 $(".strength option
[value
=12]").prop("selected
", true);
3709 driver.findElement(By.css(".generate
-container
.warning
"))
3710 .getAttribute("class")
3711 .then(function(classes) {
3712 expect(classes).toContain("hidden
");
3717 it('Shows a warning if overriding weak entropy with longer mnemonics', function(done) {
3718 driver.findElement(By.css('.use-entropy'))
3720 driver.findElement(By.css('.entropy'))
3721 .sendKeys("0123456789abcdef
"); // 6 words
3722 driver.executeScript(function() {
3723 $(".mnemonic
-length
").val("12").trigger("change
");
3725 driver.findElement(By.css(".weak
-entropy
-override
-warning
"))
3726 .getAttribute("class")
3727 .then(function(classes) {
3728 expect(classes).not.toContain("hidden
");
3733 it('Does not show a warning if entropy is stronger than mnemonic length', function(done) {
3734 driver.findElement(By.css('.use-entropy'))
3736 driver.findElement(By.css('.entropy'))
3737 .sendKeys("0123456789abcdef0123456789abcdef0123456789abcdef
"); // 18 words
3738 driver.executeScript(function() {
3739 $(".mnemonic
-length
").val("12").trigger("change
");
3741 driver.findElement(By.css(".weak
-entropy
-override
-warning
"))
3742 .getAttribute("class")
3743 .then(function(classes) {
3744 expect(classes).toContain("hidden
");
3749 it('Shows litecoin BIP49 addresses', function(done) {
3750 driver.findElement(By.css('.phrase'))
3751 .sendKeys('abandon abandon ability');
3752 selectNetwork("LTC
- Litecoin
");
3753 driver.findElement(By.css('#bip49-tab a'))
3755 // bip49 addresses are shown
3756 driver.sleep(generateDelay).then(function() {
3757 driver.findElement(By.css('#bip49 .available'))
3758 .getAttribute("class")
3759 .then(function(classes) {
3760 expect(classes).not.toContain("hidden
");
3761 // check first address
3762 getFirstAddress(function(address) {
3763 expect(address).toBe("MFwLPhsXoBuSLL8cLmW9uK6tChkzduV8qN
");
3770 it('Can use root keys to generate segwit table rows', function(done) {
3771 // segwit uses ypub / zpub instead of xpub but the root key should still
3772 // be valid regardless of the encoding used to import that key.
3773 // Maybe this breaks the reason for the different extended key prefixes, but
3774 // since the parsed root key is used behind the scenes anyhow this should be
3776 driver.findElement(By.css('#root-key'))
3777 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
3778 driver.findElement(By.css('#bip49-tab a'))
3780 // bip49 addresses are shown
3781 driver.sleep(generateDelay).then(function() {
3782 getFirstAddress(function(address) {
3783 expect(address).toBe("3QG2Y9AA4xZ846gKHZqNf7mvVKbLqMKxr2
");