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 Smileycoin
', function(done) {
1080 selectText: "SMLY - Smileycoin",
1081 firstAddress: "BEZVnEBCAyFByrgKpwAgYgtvP4rKAd9Sj2",
1083 testNetwork(done, params);
1085 it('Allows selection
of Solarcoin
', function(done) {
1087 selectText: "SLR - Solarcoin",
1088 firstAddress: "8LZ13HbnjtaMJWSvvVFNTLf71zFfDrhwLu",
1090 testNetwork(done, params);
1092 it('Allows selection
of Stratis
', function(done) {
1094 selectText: "STRAT - Stratis",
1095 firstAddress: "ScfJnq3QDhKgDMEds6sqUE1ot6ShfhmXXq",
1097 testNetwork(done, params);
1099 it('Allows selection
of Stratis Test
', function(done) {
1101 selectText: "TSTRAT - Stratis Testnet",
1102 firstAddress: "TRLWm3dye4FRrDWouwYUSUZP96xb76mBE3",
1104 testNetwork(done, params);
1106 it('Allows selection
of Syscoin
', function(done) {
1108 selectText: "SYS - Syscoin",
1109 firstAddress: "SZwJi42Pst3VAMomyK5DG4157WM5ofRmSj",
1111 testNetwork(done, params);
1113 it('Allows selection
of Toa
', function(done) {
1115 selectText: "TOA - Toa",
1116 firstAddress: "TSe1QAnUwQzUfbBusDzRJ9URttrRGKoNKF",
1118 testNetwork(done, params);
1120 it('Allows selection
of Ultimatesecurecash
', function(done) {
1122 selectText: "USC - Ultimatesecurecash",
1123 firstAddress: "UPyLAZU2Che5fiy7Ed8xVJFmXAUhitA4ug",
1125 testNetwork(done, params);
1127 it('Allows selection
of Unobtanium
', function(done) {
1129 selectText: "UNO - Unobtanium",
1130 firstAddress: "uUBMPVMXrR6qhqornJqKTWgr8L69vihSL9",
1132 testNetwork(done, params);
1134 it('Allows selection
of Vcash
', function(done) {
1136 selectText: "XVC - Vcash",
1137 firstAddress: "VuL53MSY6KjvAjKSeRkh3NDnKykacDVeps",
1139 testNetwork(done, params);
1141 it('Allows selection
of Verge
', function(done) {
1143 selectText: "XVG - Verge",
1144 firstAddress: "DCrVuGkMjLJpTGgwAgv9AcMdeb1nkWbjZA",
1146 testNetwork(done, params);
1148 it('Allows selection
of Vertcoin
', function(done) {
1150 selectText: "VTC - Vertcoin",
1151 firstAddress: "Vf6koGuiWdXQfx8tNqxoNeEDxh4xh5cxsG",
1153 testNetwork(done, params);
1155 it('Allows selection
of Vivo
', function(done) {
1157 selectText: "VIVO - Vivo",
1158 firstAddress: "VFmBwuXXGhJe7MarQG2GfzHMFebRHgfSpB",
1160 testNetwork(done, params);
1162 it('Allows selection
of Vpncoin
', function(done) {
1164 selectText: "VASH - Vpncoin",
1165 firstAddress: "VoEmH1qXC4TsSgBAStR21QYetwnFqbqCx9",
1167 testNetwork(done, params);
1169 it('Allows selection
of Whitecoin
', function(done) {
1171 selectText: "XWC - Whitecoin",
1172 firstAddress: "WcSwCAUqrSgeSYbsaS3SSWWhsx8KRYTFDR",
1174 testNetwork(done, params);
1176 it('Allows selection
of Wincoin
', function(done) {
1178 selectText: "WC - Wincoin",
1179 firstAddress: "WaDVCESMGgyKgNESdn3u43NnwmGSkZED3Z",
1181 testNetwork(done, params);
1183 it('Allows selection
of Zcoin
', function(done) {
1185 selectText: "XZC - Zcoin",
1186 firstAddress: "a6VcMdP4XgAA9Tr7xNszmPG5FZpfRf17Cq",
1188 testNetwork(done, params);
1190 it('Allows selection
of Zcash
', function(done) {
1192 selectText: "ZEC - Zcash",
1193 firstAddress: "t1Sz8AneMcVuzUg3tPJ8et5AS5LFJ7K2EF9",
1195 testNetwork(done, params);
1197 it('Allows selection
of Zclassic
', function(done) {
1199 selectText: "ZCL - Zclassic",
1200 firstAddress: "t1TBMxTvVJRybUbMLGWq8H4A8F4VUL7czEc",
1202 testNetwork(done, params);
1204 it('Allows selection
of Zencash
', function(done) {
1206 selectText: "ZEN - Zencash",
1207 firstAddress: "znWh9XASyW2dZq5tck84wFjiwuqVysi7q3p",
1209 testNetwork(done, params);
1211 it('Allows selection
of Energi
', function(done) {
1213 selectText: "NRG - Energi",
1214 firstAddress: "EejRy4t4nidzhGGzkJUgFP3z4HYBjhTsRt",
1216 testNetwork(done, params);
1218 it('Allows selection
of Ethereum Classic
', function(done) {
1220 selectText: "ETC - Ethereum Classic",
1221 firstAddress: "0x3c05e5556693808367afB62eF3b63e35d6eD249A",
1223 testNetwork(done, params);
1225 it('Allows selection
of Pirl
', function(done) {
1227 selectText: "PIRL - Pirl",
1228 firstAddress: "0xe77FC0723dA122B5025CA79193c28563eB47e776",
1230 testNetwork(done, params);
1232 it('Allows selection
of MIX
', function(done) {
1234 selectText: "MIX - MIX",
1235 firstAddress: "0x98BC5e63aeb6A4e82d72850d20710F07E29A29F1",
1237 testNetwork(done, params);
1239 it('Allows selection
of Musicoin
', function(done) {
1241 selectText: "MUSIC - Musicoin",
1242 firstAddress: "0xDc060e4A0b0313ea83Cf6B3A39B9db2D29004897",
1244 testNetwork(done, params);
1246 it('Allows selection
of Poa
', function(done) {
1248 selectText: "POA - Poa",
1249 firstAddress: "0x53aF28d754e106210C3d0467Dd581eaf7e3C5e60",
1251 testNetwork(done, params);
1253 it('Allows selection
of Expanse
', function(done) {
1255 selectText: "EXP - Expanse",
1256 firstAddress: "0xf57FeAbf26582b6E3E666559d3B1Cc6fB2b2c5F6",
1258 testNetwork(done, params);
1260 it('Allows selection
of Callisto
', function(done) {
1262 selectText: "CLO - Callisto",
1263 firstAddress: "0x4f9364F7420B317266C51Dc8eB979717D4dE3f4E",
1265 testNetwork(done, params);
1267 it('Allows selection
of HUSH
', function(done) {
1269 selectText: "HUSH - Hush",
1270 firstAddress: "t1g6rLXUnJaiJuu4q4zmJjoa9Gk4fwKpiuA",
1272 testNetwork(done, params);
1276 // BIP39 seed is set from phrase
1277 it('Sets the bip39 seed
from the prhase
', function(done) {
1278 driver.findElement(By.css('.phrase
'))
1279 .sendKeys('abandon abandon ability
');
1280 driver.sleep(generateDelay).then(function() {
1281 driver.findElement(By.css('.seed
'))
1282 .getAttribute("value")
1283 .then(function(seed) {
1284 expect(seed).toBe("20da140d3dd1df8713cefcc4d54ce0e445b4151027a1ab567b832f6da5fcc5afc1c3a3f199ab78b8e0ab4652efd7f414ac2c9a3b81bceb879a70f377aa0a58f3");
1290 // BIP32 root key is set from phrase
1291 it('Sets the bip39 root key
from the prhase
', function(done) {
1292 driver.findElement(By.css('.phrase
'))
1293 .sendKeys('abandon abandon ability
');
1294 driver.sleep(generateDelay).then(function() {
1295 driver.findElement(By.css('.root
-key
'))
1296 .getAttribute("value")
1297 .then(function(seed) {
1298 expect(seed).toBe("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
1304 // Tabs show correct addresses when changed
1305 it('Shows the correct address when tab is changed
', function(done) {
1306 driver.findElement(By.css('.phrase
'))
1307 .sendKeys('abandon abandon ability
');
1308 driver.sleep(generateDelay).then(function() {
1309 driver.findElement(By.css('#bip32
-tab a
'))
1311 driver.sleep(generateDelay).then(function() {
1312 getFirstAddress(function(address) {
1313 expect(address).toBe("17uQ7s2izWPwBmEVFikTmZUjbBKWYdJchz");
1320 // BIP44 derivation path is shown
1321 it('Shows the derivation path
for bip44 tab
', function(done) {
1322 driver.findElement(By.css('.phrase
'))
1323 .sendKeys('abandon abandon ability
');
1324 driver.sleep(generateDelay).then(function() {
1325 driver.findElement(By.css('#bip44
.path
'))
1326 .getAttribute("value")
1327 .then(function(path) {
1328 expect(path).toBe("m/44'/0'/0'/0");
1334 // BIP44 extended private key is shown
1335 it('Shows the extended private key for bip44 tab', function(done) {
1336 driver.findElement(By.css('.phrase'))
1337 .sendKeys('abandon abandon ability');
1338 driver.sleep(generateDelay).then(function() {
1339 driver.findElement(By.css('.extended-priv-key'))
1340 .getAttribute("value
")
1341 .then(function(path) {
1342 expect(path).toBe("xprvA2DxxvPZcyRvYgZMGS53nadR32mVDeCyqQYyFhrCVbJNjPoxMeVf7QT5g7mQASbTf9Kp4cryvcXnu2qurjWKcrdsr91jXymdCDNxKgLFKJG
");
1348 // BIP44 extended public key is shown
1349 it('Shows the extended public key for bip44 tab', function(done) {
1350 driver.findElement(By.css('.phrase'))
1351 .sendKeys('abandon abandon ability');
1352 driver.sleep(generateDelay).then(function() {
1353 driver.findElement(By.css('.extended-pub-key'))
1354 .getAttribute("value
")
1355 .then(function(path) {
1356 expect(path).toBe("xpub6FDKNRvTTLzDmAdpNTc49ia9b4byd6vqCdUa46Fp3vqMcC96uBoufCmZXQLiN5AK3iSCJMhf9gT2sxkpyaPepRuA7W3MujV5tGmF5VfbueM
");
1362 // BIP44 account field changes address list
1363 it('Changes the address list if bip44 account is changed', function(done) {
1364 driver.findElement(By.css('#bip44 .account'))
1366 driver.findElement(By.css('.phrase'))
1367 .sendKeys('abandon abandon ability');
1368 driver.sleep(generateDelay).then(function() {
1369 getFirstAddress(function(address) {
1370 expect(address).toBe("1Nq2Wmu726XHCuGhctEtGmhxo3wzk5wZ1H
");
1376 // BIP44 change field changes address list
1377 it('Changes the address list if bip44 change is changed', function(done) {
1378 driver.findElement(By.css('#bip44 .change'))
1380 driver.findElement(By.css('.phrase'))
1381 .sendKeys('abandon abandon ability');
1382 driver.sleep(generateDelay).then(function() {
1383 getFirstAddress(function(address) {
1384 expect(address).toBe("1KAGfWgqfVbSSXY56fNQ7YnhyKuoskHtYo
");
1390 // BIP32 derivation path can be set
1391 it('Can use a custom bip32 derivation path', function(done) {
1392 driver.findElement(By.css('#bip32-tab a'))
1394 driver.findElement(By.css('#bip32 .path'))
1396 driver.findElement(By.css('#bip32 .path'))
1398 driver.findElement(By.css('.phrase'))
1399 .sendKeys('abandon abandon ability');
1400 driver.sleep(generateDelay).then(function() {
1401 getFirstAddress(function(address) {
1402 expect(address).toBe("16pYQQdLD1hH4hwTGLXBaZ9Teboi1AGL8L
");
1408 // BIP32 can use hardened derivation paths
1409 it('Can use a hardened derivation paths', function(done) {
1410 driver.findElement(By.css('#bip32-tab a'))
1412 driver.findElement(By.css('#bip32 .path'))
1414 driver.findElement(By.css('#bip32 .path'))
1416 driver.findElement(By.css('.phrase
'))
1417 .sendKeys('abandon abandon ability
');
1418 driver.sleep(generateDelay).then(function() {
1419 getFirstAddress(function(address) {
1420 expect(address).toBe("14aXZeprXAE3UUKQc4ihvwBvww2LuEoHo4");
1426 // BIP32 extended private key is shown
1427 it('Shows the BIP32 extended
private key
', function(done) {
1428 driver.findElement(By.css('#bip32
-tab a
'))
1430 driver.findElement(By.css('.phrase
'))
1431 .sendKeys('abandon abandon ability
');
1432 driver.sleep(generateDelay).then(function() {
1433 driver.findElement(By.css('.extended
-priv
-key
'))
1434 .getAttribute("value")
1435 .then(function(privKey) {
1436 expect(privKey).toBe("xprv9va99uTVE5aLiutUVLTyfxfe8v8aaXjSQ1XxZbK6SezYVuikA9MnjQVTA8rQHpNA5LKvyQBpLiHbBQiiccKiBDs7eRmBogsvq3THFeLHYbe");
1442 // BIP32 extended public key is shown
1443 it('Shows the BIP32 extended
public key
', function(done) {
1444 driver.findElement(By.css('#bip32
-tab a
'))
1446 driver.findElement(By.css('.phrase
'))
1447 .sendKeys('abandon abandon ability
');
1448 driver.sleep(generateDelay).then(function() {
1449 driver.findElement(By.css('.extended
-pub
-key
'))
1450 .getAttribute("value")
1451 .then(function(pubKey) {
1452 expect(pubKey).toBe("xpub69ZVZQzP4T8dwPxwbMzz36cNgwy4yzTHmETZMyihzzXXNi3thgg3HCow1RtY252wdw5rS8369xKnraN5Q93y3FkFfJp2XEHWUrkyXsjS93P");
1458 // Derivation path is shown in table
1459 it('Shows the derivation path
in the table
', function(done) {
1460 driver.findElement(By.css('.phrase
'))
1461 .sendKeys('abandon abandon ability
');
1462 driver.sleep(generateDelay).then(function() {
1463 getFirstPath(function(path) {
1464 expect(path).toBe("m/44'/0'/0'/0/0");
1470 // Derivation path for address can be hardened
1471 it('Can derive hardened addresses', function(done) {
1472 driver.findElement(By.css('#bip32-tab a'))
1474 driver.executeScript(function() {
1475 $(".hardened
-addresses
").prop("checked
", true);
1477 driver.findElement(By.css('.phrase'))
1478 .sendKeys('abandon abandon ability');
1479 driver.sleep(generateDelay).then(function() {
1480 getFirstAddress(function(address) {
1481 expect(address).toBe("18exLzUv7kfpiXRzmCjFDoC9qwNLFyvwyd
");
1487 // Derivation path visibility can be toggled
1488 it('Can toggle visibility of the derivation path column', function(done) {
1489 driver.findElement(By.css('.phrase'))
1490 .sendKeys('abandon abandon ability');
1491 driver.sleep(generateDelay).then(function() {
1492 driver.findElement(By.css('.index-toggle'))
1494 testColumnValuesAreInvisible(done, "index
");
1499 it('Shows the address in the table', function(done) {
1500 driver.findElement(By.css('.phrase'))
1501 .sendKeys('abandon abandon ability');
1502 driver.sleep(generateDelay).then(function() {
1503 getFirstAddress(function(address) {
1504 expect(address).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug
");
1510 // Addresses are shown in order of derivation path
1511 it('Shows the address in order of derivation path', function(done) {
1512 driver.findElement(By.css('.phrase'))
1513 .sendKeys('abandon abandon ability');
1514 driver.sleep(generateDelay).then(function() {
1515 testRowsAreInCorrectOrder(done);
1519 // Address visibility can be toggled
1520 it('Can toggle visibility of the address column', function(done) {
1521 driver.findElement(By.css('.phrase'))
1522 .sendKeys('abandon abandon ability');
1523 driver.sleep(generateDelay).then(function() {
1524 driver.findElement(By.css('.address-toggle'))
1526 testColumnValuesAreInvisible(done, "address
");
1530 // Public key is shown in table
1531 it('Shows the public key in the table', function(done) {
1532 driver.findElement(By.css('.phrase'))
1533 .sendKeys('abandon abandon ability');
1534 driver.sleep(generateDelay).then(function() {
1535 driver.findElements(By.css('.pubkey'))
1536 .then(function(els) {
1538 .then(function(pubkey) {
1539 expect(pubkey).toBe("033f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3
");
1546 // Public key visibility can be toggled
1547 it('Can toggle visibility of the public key column', function(done) {
1548 driver.findElement(By.css('.phrase'))
1549 .sendKeys('abandon abandon ability');
1550 driver.sleep(generateDelay).then(function() {
1551 driver.findElement(By.css('.public-key-toggle'))
1553 testColumnValuesAreInvisible(done, "pubkey
");
1557 // Private key is shown in table
1558 it('Shows the private key in the table', function(done) {
1559 driver.findElement(By.css('.phrase'))
1560 .sendKeys('abandon abandon ability');
1561 driver.sleep(generateDelay).then(function() {
1562 driver.findElements(By.css('.privkey'))
1563 .then(function(els) {
1565 .then(function(pubkey) {
1566 expect(pubkey).toBe("L26cVSpWFkJ6aQkPkKmTzLqTdLJ923e6CzrVh9cmx21QHsoUmrEE
");
1573 // Private key visibility can be toggled
1574 it('Can toggle visibility of the private key column', function(done) {
1575 driver.findElement(By.css('.phrase'))
1576 .sendKeys('abandon abandon ability');
1577 driver.sleep(generateDelay).then(function() {
1578 driver.findElement(By.css('.private-key-toggle'))
1580 testColumnValuesAreInvisible(done, "privkey
");
1584 // More addresses can be generated
1585 it('Can generate more rows in the table', function(done) {
1586 driver.findElement(By.css('.phrase'))
1587 .sendKeys('abandon abandon ability');
1588 driver.sleep(generateDelay).then(function() {
1589 driver.findElement(By.css('.more'))
1591 driver.sleep(generateDelay).then(function() {
1592 driver.findElements(By.css('.address'))
1593 .then(function(els) {
1594 expect(els.length).toBe(40);
1601 // A custom number of additional addresses can be generated
1602 it('Can generate more rows in the table', function(done) {
1603 driver.findElement(By.css('.phrase'))
1604 .sendKeys('abandon abandon ability');
1605 driver.sleep(generateDelay).then(function() {
1606 driver.findElement(By.css('.rows-to-add'))
1608 driver.findElement(By.css('.rows-to-add'))
1610 driver.findElement(By.css('.more'))
1612 driver.sleep(generateDelay).then(function() {
1613 driver.findElements(By.css('.address'))
1614 .then(function(els) {
1615 expect(els.length).toBe(21);
1622 // Additional addresses are shown in order of derivation path
1623 it('Shows additional addresses in order of derivation path', function(done) {
1624 driver.findElement(By.css('.phrase'))
1625 .sendKeys('abandon abandon ability');
1626 driver.sleep(generateDelay).then(function() {
1627 driver.findElement(By.css('.more'))
1629 driver.sleep(generateDelay).then(function() {
1630 testRowsAreInCorrectOrder(done);
1635 // BIP32 root key can be set by the user
1636 it('Allows the user to set the BIP32 root key', function(done) {
1637 driver.findElement(By.css('.root-key'))
1638 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1639 driver.sleep(generateDelay).then(function() {
1640 getFirstAddress(function(address) {
1641 expect(address).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug
");
1647 // Setting BIP32 root key clears the existing phrase, passphrase and seed
1648 // TODO this doesn't work in selenium with chrome
1649 it('Confirms the existing phrase should be cleared', function(done) {
1650 if (browser == "chrome
") {
1651 pending("Selenium
+ Chrome headless bug
for alert
, see
https://stackoverflow.com/q/45242264");
1653 driver
.findElement(By
.css('.phrase'))
1654 .sendKeys('A non-blank but invalid value');
1655 driver
.findElement(By
.css('.root-key'))
1656 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1657 driver
.switchTo().alert().accept();
1658 driver
.findElement(By
.css('.phrase'))
1659 .getAttribute("value").then(function(value
) {
1660 expect(value
).toBe("");
1665 // Clearing of phrase, passphrase and seed can be cancelled by user
1666 // TODO this doesn't work in selenium with chrome
1667 it('Allows the clearing of the phrase to be cancelled', function(done
) {
1668 if (browser
== "chrome") {
1669 pending("Selenium + Chrome headless bug for alert, see https://stackoverflow.com/q/45242264");
1671 driver
.findElement(By
.css('.phrase'))
1672 .sendKeys('abandon abandon ability');
1673 driver
.sleep(generateDelay
).then(function() {
1674 driver
.findElement(By
.css('.root-key'))
1676 driver
.findElement(By
.css('.root-key'))
1678 driver
.switchTo().alert().dismiss();
1679 driver
.findElement(By
.css('.phrase'))
1680 .getAttribute("value").then(function(value
) {
1681 expect(value
).toBe("abandon abandon ability");
1687 // Custom BIP32 root key is used when changing the derivation path
1688 it('Can set derivation path for root key instead of phrase', function(done
) {
1689 driver
.findElement(By
.css('#bip44 .account'))
1691 driver
.findElement(By
.css('.root-key'))
1692 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1693 driver
.sleep(generateDelay
).then(function() {
1694 getFirstAddress(function(address
) {
1695 expect(address
).toBe("1Nq2Wmu726XHCuGhctEtGmhxo3wzk5wZ1H");
1701 // Incorrect mnemonic shows error
1702 it('Shows an error for incorrect mnemonic', function(done
) {
1703 driver
.findElement(By
.css('.phrase'))
1704 .sendKeys('abandon abandon abandon');
1705 driver
.sleep(feedbackDelay
).then(function() {
1706 driver
.findElement(By
.css('.feedback'))
1708 .then(function(feedback
) {
1709 expect(feedback
).toBe("Invalid mnemonic");
1715 // Incorrect word shows suggested replacement
1716 it('Shows word suggestion for incorrect word', function(done
) {
1717 driver
.findElement(By
.css('.phrase'))
1718 .sendKeys('abandon abandon abiliti');
1719 driver
.sleep(feedbackDelay
).then(function() {
1720 driver
.findElement(By
.css('.feedback'))
1722 .then(function(feedback
) {
1723 var msg
= "abiliti not in wordlist, did you mean ability?";
1724 expect(feedback
).toBe(msg
);
1730 // Github pull request 48
1731 // First four letters of word shows that word, not closest
1732 // since first four letters gives unique word in BIP39 wordlist
1733 // eg ille should show illegal, not idle
1734 it('Shows word suggestion based on first four chars', function(done
) {
1735 driver
.findElement(By
.css('.phrase'))
1737 driver
.sleep(feedbackDelay
).then(function() {
1738 driver
.findElement(By
.css('.feedback'))
1740 .then(function(feedback
) {
1741 var msg
= "ille not in wordlist, did you mean illegal?";
1742 expect(feedback
).toBe(msg
);
1748 // Incorrect BIP32 root key shows error
1749 it('Shows error for incorrect root key', function(done
) {
1750 driver
.findElement(By
.css('.root-key'))
1751 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpj');
1752 driver
.sleep(feedbackDelay
).then(function() {
1753 driver
.findElement(By
.css('.feedback'))
1755 .then(function(feedback
) {
1756 var msg
= "Invalid root key";
1757 expect(feedback
).toBe(msg
);
1763 // Derivation path not starting with m shows error
1764 it('Shows error for derivation path not starting with m', function(done
) {
1765 driver
.findElement(By
.css('#bip32-tab a'))
1767 driver
.findElement(By
.css('#bip32 .path'))
1769 driver
.findElement(By
.css('#bip32 .path'))
1771 driver
.findElement(By
.css('.phrase'))
1772 .sendKeys('abandon abandon ability');
1773 driver
.sleep(feedbackDelay
).then(function() {
1774 driver
.findElement(By
.css('.feedback'))
1776 .then(function(feedback
) {
1777 var msg
= "First character must be 'm'";
1778 expect(feedback
).toBe(msg
);
1784 // Derivation path containing invalid characters shows useful error
1785 it('Shows error for derivation path not starting with m', function(done
) {
1786 driver
.findElement(By
.css('#bip32-tab a'))
1788 driver
.findElement(By
.css('#bip32 .path'))
1790 driver
.findElement(By
.css('#bip32 .path'))
1791 .sendKeys('m/1/0wrong1/1');
1792 driver
.findElement(By
.css('.phrase'))
1793 .sendKeys('abandon abandon ability');
1794 driver
.sleep(feedbackDelay
).then(function() {
1795 driver
.findElement(By
.css('.feedback'))
1797 .then(function(feedback
) {
1798 var msg
= "Invalid characters 0wrong1 found at depth 2";
1799 expect(feedback
).toBe(msg
);
1805 // Github Issue 11: Default word length is 15
1806 // https://github.com/iancoleman/bip39/issues/11
1807 it('Sets the default word length to 15', function(done
) {
1808 driver
.findElement(By
.css('.strength'))
1809 .getAttribute("value")
1810 .then(function(strength
) {
1811 expect(strength
).toBe("15");
1816 // Github Issue 12: Generate more rows with private keys hidden
1817 // https://github.com/iancoleman/bip39/issues/12
1818 it('Sets the correct hidden column state on new rows', function(done
) {
1819 driver
.findElement(By
.css('.phrase'))
1820 .sendKeys("abandon abandon ability");
1821 driver
.sleep(generateDelay
).then(function() {
1822 driver
.findElement(By
.css('.private-key-toggle'))
1824 driver
.findElement(By
.css('.more'))
1826 driver
.sleep(generateDelay
).then(function() {
1827 driver
.findElements(By
.css('.privkey'))
1828 .then(function(els
) {
1829 expect(els
.length
).toBe(40);
1831 testColumnValuesAreInvisible(done
, "privkey");
1836 // Github Issue 19: Mnemonic is not sensitive to whitespace
1837 // https://github.com/iancoleman/bip39/issues/19
1838 it('Ignores excess whitespace in the mnemonic', function(done
) {
1839 var doublespace
= " ";
1840 var mnemonic
= "urge cat" + doublespace
+ "bid";
1841 driver
.findElement(By
.css('.phrase'))
1842 .sendKeys(mnemonic
);
1843 driver
.sleep(generateDelay
).then(function() {
1844 driver
.findElement(By
.css('.root-key'))
1845 .getAttribute("value")
1846 .then(function(seed
) {
1847 expect(seed
).toBe("xprv9s21ZrQH143K3isaZsWbKVoTtbvd34Y1ZGRugGdMeBGbM3AgBVzTH159mj1cbbtYSJtQr65w6L5xy5L9SFC7c9VJZWHxgAzpj4mun5LhrbC");
1853 // Github Issue 23: Part 1: Use correct derivation path when changing tabs
1854 // https://github.com/iancoleman/bip39/issues/23
1855 it('Uses the correct derivation path when changing tabs', function(done
) {
1856 // 1) and 2) set the phrase
1857 driver
.findElement(By
.css('.phrase'))
1858 .sendKeys("abandon abandon ability");
1859 driver
.sleep(generateDelay
).then(function() {
1860 // 3) select bip32 tab
1861 driver
.findElement(By
.css('#bip32-tab a'))
1863 driver
.sleep(generateDelay
).then(function() {
1864 // 4) switch from bitcoin to litecoin
1865 selectNetwork("LTC - Litecoin");
1866 driver
.sleep(generateDelay
).then(function() {
1867 // 5) Check address is displayed correctly
1868 getFirstAddress(function(address
) {
1869 expect(address
).toBe("LS8MP5LZ5AdzSZveRrjm3aYVoPgnfFh5T5");
1870 // 5) Check derivation path is displayed correctly
1871 getFirstPath(function(path
) {
1872 expect(path
).toBe("m/0/0");
1881 // Github Issue 23 Part 2: Coin selection in derivation path
1882 // https://github.com/iancoleman/bip39/issues/23#issuecomment-238011920
1883 it('Uses the correct derivation path when changing coins', function(done
) {
1885 driver
.findElement(By
.css('.phrase'))
1886 .sendKeys("abandon abandon ability");
1887 driver
.sleep(generateDelay
).then(function() {
1888 // switch from bitcoin to clam
1889 selectNetwork("CLAM - Clams");
1890 driver
.sleep(generateDelay
).then(function() {
1891 // check derivation path is displayed correctly
1892 getFirstPath(function(path
) {
1893 expect(path
).toBe("m/44'/23'/0'/0/0");
1900 // Github Issue 26: When using a Root key derrived altcoins are incorrect
1901 // https://github.com/iancoleman/bip39/issues/26
1902 it('Uses the correct derivation for altcoins with root keys', function(done
) {
1903 // 1) 2) and 3) set the root key
1904 driver
.findElement(By
.css('.root-key'))
1905 .sendKeys("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
1906 driver
.sleep(generateDelay
).then(function() {
1907 // 4) switch from bitcoin to viacoin
1908 selectNetwork("VIA - Viacoin");
1909 driver
.sleep(generateDelay
).then(function() {
1910 // 5) ensure the derived address is correct
1911 getFirstAddress(function(address
) {
1912 expect(address
).toBe("Vq9Eq4N5SQnjqZvxtxzo7hZPW5XnyJsmXT");
1919 // Selecting a language with no existing phrase should generate a phrase in
1921 it('Generate a random phrase when language is selected and no current phrase', function(done
) {
1922 driver
.findElement(By
.css("a[href='#japanese']"))
1924 driver
.sleep(generateDelay
).then(function() {
1925 driver
.findElement(By
.css(".phrase"))
1926 .getAttribute("value").then(function(phrase
) {
1927 expect(phrase
.search(/[a-z]/)).toBe(-1);
1928 expect(phrase
.length
).toBeGreaterThan(0);
1934 // Selecting a language with existing phrase should update the phrase to use
1936 it('Updates existing phrases when the language is changed', function(done
) {
1937 driver
.findElement(By
.css(".phrase"))
1938 .sendKeys("abandon abandon ability");
1939 driver
.sleep(generateDelay
).then(function() {
1940 driver
.findElement(By
.css("a[href='#italian']"))
1942 driver
.sleep(generateDelay
).then(function() {
1943 driver
.findElement(By
.css(".phrase"))
1944 .getAttribute("value").then(function(phrase
) {
1945 // Check only the language changes, not the phrase
1946 expect(phrase
).toBe("abaco abaco abbaglio");
1947 getFirstAddress(function(address
) {
1948 // Check the address is correct
1949 expect(address
).toBe("1Dz5TgDhdki9spa6xbPFbBqv5sjMrx3xgV");
1957 // Suggested replacement for erroneous word in non-English language
1958 it('Shows word suggestion for incorrect word in non-English language', function(done
) {
1959 driver
.findElement(By
.css('.phrase'))
1960 .sendKeys('abaco abaco zbbaglio');
1961 driver
.sleep(feedbackDelay
).then(function() {
1962 driver
.findElement(By
.css('.feedback'))
1964 .then(function(feedback
) {
1965 var msg
= "zbbaglio not in wordlist, did you mean abbaglio?";
1966 expect(feedback
).toBe(msg
);
1972 // Japanese word does not break across lines.
1974 // https://github.com/bitcoin/bips/blob/master/bip-0039/bip-0039-wordlists.md#japanese
1975 it('Does not break Japanese words across lines', function(done
) {
1976 driver
.findElement(By
.css('.phrase'))
1977 .getCssValue("word-break")
1978 .then(function(value
) {
1979 expect(value
).toBe("keep-all");
1984 // Language can be specified at page load using hash value in url
1985 it('Can set the language from the url hash', function(done
) {
1986 driver
.get(url
+ "#japanese").then(function() {
1987 driver
.findElement(By
.css('.generate')).click();
1988 driver
.sleep(generateDelay
).then(function() {
1989 driver
.findElement(By
.css(".phrase"))
1990 .getAttribute("value").then(function(phrase
) {
1991 expect(phrase
.search(/[a-z]/)).toBe(-1);
1992 expect(phrase
.length
).toBeGreaterThan(0);
1999 // Entropy can be entered by the user
2000 it('Allows entropy to be entered', function(done
) {
2001 driver
.findElement(By
.css('.use-entropy'))
2003 driver
.findElement(By
.css('.entropy'))
2004 .sendKeys('00000000 00000000 00000000 00000000');
2005 driver
.sleep(generateDelay
).then(function() {
2006 driver
.findElement(By
.css(".phrase"))
2007 .getAttribute("value").then(function(phrase
) {
2008 expect(phrase
).toBe("abandon abandon ability");
2009 getFirstAddress(function(address
) {
2010 expect(address
).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug");
2017 // A warning about entropy is shown to the user, with additional information
2018 it('Shows a warning about using entropy', function(done
) {
2019 driver
.findElement(By
.css('.use-entropy'))
2021 driver
.findElement(By
.css('.entropy-container'))
2023 .then(function(containerText
) {
2024 var warning
= "mnemonic may be insecure";
2025 expect(containerText
).toContain(warning
);
2026 driver
.findElement(By
.css('#entropy-notes'))
2027 .findElement(By
.xpath("parent::*"))
2029 .then(function(notesText
) {
2030 var detail
= "flipping a fair coin, rolling a fair dice, noise measurements etc";
2031 expect(notesText
).toContain(detail
);
2037 // The types of entropy available are described to the user
2038 it('Shows the types of entropy available', function(done
) {
2039 driver
.findElement(By
.css('.entropy'))
2040 .getAttribute("placeholder")
2041 .then(function(placeholderText
) {
2050 for (var i
=0; i
<options
.length
; i
++) {
2051 var option
= options
[i
];
2052 expect(placeholderText
).toContain(option
);
2058 // The actual entropy used is shown to the user
2059 it('Shows the actual entropy used', function(done
) {
2060 driver
.findElement(By
.css('.use-entropy'))
2062 driver
.findElement(By
.css('.entropy'))
2063 .sendKeys('Not A Very Good Entropy Source At All');
2064 driver
.sleep(generateDelay
).then(function() {
2065 driver
.findElement(By
.css('.entropy-container'))
2067 .then(function(text
) {
2068 expect(text
).toMatch(/Filtered Entropy
\s
+AedEceAA
/);
2074 // Binary entropy can be entered
2075 it('Allows binary entropy to be entered', function(done
) {
2076 testEntropyType(done
, "01", "binary");
2079 // Base 6 entropy can be entered
2080 it('Allows base 6 entropy to be entered', function(done
) {
2081 testEntropyType(done
, "012345", "base 6");
2084 // Base 6 dice entropy can be entered
2085 it('Allows base 6 dice entropy to be entered', function(done
) {
2086 testEntropyType(done
, "123456", "base 6 (dice)");
2089 // Base 10 entropy can be entered
2090 it('Allows base 10 entropy to be entered', function(done
) {
2091 testEntropyType(done
, "789", "base 10");
2094 // Hexadecimal entropy can be entered
2095 it('Allows hexadecimal entropy to be entered', function(done
) {
2096 testEntropyType(done
, "abcdef", "hexadecimal");
2099 // Dice entropy value is shown as the converted base 6 value
2100 // ie 123456 is converted to 123450
2101 it('Shows dice entropy as base 6', function(done
) {
2102 driver
.findElement(By
.css('.use-entropy'))
2104 driver
.findElement(By
.css('.entropy'))
2105 .sendKeys("123456");
2106 driver
.sleep(generateDelay
).then(function() {
2107 driver
.findElement(By
.css('.entropy-container'))
2109 .then(function(text
) {
2110 expect(text
).toMatch(/Filtered Entropy
\s
+123450/);
2116 // The number of bits of entropy accumulated is shown
2117 it("Shows the number of bits of entropy for 20 bits of binary", function(done
) {
2118 testEntropyBits(done
, "0000 0000 0000 0000 0000", "20");
2120 it("Shows the number of bits of entropy for 1 bit of binary", function(done
) {
2121 testEntropyBits(done
, "0", "1");
2123 it("Shows the number of bits of entropy for 4 bits of binary", function(done
) {
2124 testEntropyBits(done
, "0000", "4");
2126 it("Shows the number of bits of entropy for 1 character of base 6 (dice)", function(done
) {
2127 // 6 in card is 0 in base 6, 0 in base 6 is 2.6 bits (rounded down to 2 bits)
2128 testEntropyBits(done
, "6", "2");
2130 it("Shows the number of bits of entropy for 1 character of base 10 with 3 bits", function(done
) {
2131 // 7 in base 10 is 111 in base 2, no leading zeros
2132 testEntropyBits(done
, "7", "3");
2134 it("Shows the number of bits of entropy for 1 character of base 10 with 4 bis", function(done
) {
2135 testEntropyBits(done
, "8", "4");
2137 it("Shows the number of bits of entropy for 1 character of hex", function(done
) {
2138 testEntropyBits(done
, "F", "4");
2140 it("Shows the number of bits of entropy for 2 characters of base 10", function(done
) {
2141 testEntropyBits(done
, "29", "6");
2143 it("Shows the number of bits of entropy for 2 characters of hex", function(done
) {
2144 testEntropyBits(done
, "0A", "8");
2146 it("Shows the number of bits of entropy for 2 characters of hex with 3 leading zeros", function(done
) {
2147 // hex is always multiple of 4 bits of entropy
2148 testEntropyBits(done
, "1A", "8");
2150 it("Shows the number of bits of entropy for 2 characters of hex with 2 leading zeros", function(done
) {
2151 testEntropyBits(done
, "2A", "8");
2153 it("Shows the number of bits of entropy for 2 characters of hex with 1 leading zero", function(done
) {
2154 testEntropyBits(done
, "4A", "8");
2156 it("Shows the number of bits of entropy for 2 characters of hex with no leading zeros", function(done
) {
2157 testEntropyBits(done
, "8A", "8");
2159 it("Shows the number of bits of entropy for 2 characters of hex starting with F", function(done
) {
2160 testEntropyBits(done
, "FA", "8");
2162 it("Shows the number of bits of entropy for 4 characters of hex with leading zeros", function(done
) {
2163 testEntropyBits(done
, "000A", "16");
2165 it("Shows the number of bits of entropy for 4 characters of base 6", function(done
) {
2166 testEntropyBits(done
, "5555", "11");
2168 it("Shows the number of bits of entropy for 4 characters of base 6 dice", function(done
) {
2169 // uses dice, so entropy is actually 0000 in base 6, which is 4 lots of
2170 // 2.58 bits, which is 10.32 bits (rounded down to 10 bits)
2171 testEntropyBits(done
, "6666", "10");
2173 it("Shows the number of bits of entropy for 4 charactes of base 10", function(done
) {
2174 // Uses base 10, which is 4 lots of 3.32 bits, which is 13.3 bits (rounded
2176 testEntropyBits(done
, "2227", "13");
2178 it("Shows the number of bits of entropy for 4 characters of hex with 2 leading zeros", function(done
) {
2179 testEntropyBits(done
, "222F", "16");
2181 it("Shows the number of bits of entropy for 4 characters of hex starting with F", function(done
) {
2182 testEntropyBits(done
, "FFFF", "16");
2184 it("Shows the number of bits of entropy for 10 characters of base 10", function(done
) {
2185 // 10 events at 3.32 bits per event
2186 testEntropyBits(done
, "0000101017", "33");
2188 it("Shows the number of bits of entropy for a full deck of cards", function(done
) {
2189 // cards are not replaced, so a full deck is not 52^52 entropy which is 296
2190 // bits, it's 52!, which is 225 bits
2191 testEntropyBits(done
, "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks", "225");
2194 it("Shows details about the entered entropy", function(done
) {
2195 testEntropyFeedback(done
,
2199 type: "hexadecimal",
2203 strength: "less than a second",
2207 it("Shows details about the entered entropy", function(done
) {
2208 testEntropyFeedback(done
,
2210 entropy: "AAAAAAAA",
2211 filtered: "AAAAAAAA",
2212 type: "hexadecimal",
2216 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2220 it("Shows details about the entered entropy", function(done
) {
2221 testEntropyFeedback(done
,
2223 entropy: "AAAAAAAA B",
2224 filtered: "AAAAAAAAB",
2225 type: "hexadecimal",
2229 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2233 it("Shows details about the entered entropy", function(done
) {
2234 testEntropyFeedback(done
,
2236 entropy: "AAAAAAAA BBBBBBBB",
2237 filtered: "AAAAAAAABBBBBBBB",
2238 type: "hexadecimal",
2242 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2246 it("Shows details about the entered entropy", function(done
) {
2247 testEntropyFeedback(done
,
2249 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC",
2250 filtered: "AAAAAAAABBBBBBBBCCCCCCCC",
2251 type: "hexadecimal",
2255 strength: "less than a second",
2259 it("Shows details about the entered entropy", function(done
) {
2260 testEntropyFeedback(done
,
2262 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDD",
2263 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDD",
2264 type: "hexadecimal",
2268 strength: "2 minutes",
2272 it("Shows details about the entered entropy", function(done
) {
2273 testEntropyFeedback(done
,
2275 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA",
2276 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDA",
2277 type: "hexadecimal",
2285 it("Shows details about the entered entropy", function(done
) {
2286 testEntropyFeedback(done
,
2288 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA EEEEEEEE",
2289 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDAEEEEEEEE",
2290 type: "hexadecimal",
2294 strength: "3 years",
2298 it("Shows details about the entered entropy", function(done
) {
2299 testEntropyFeedback(done
,
2301 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA EEEEEEEE FFFFFFFF",
2302 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDAEEEEEEEEFFFFFFFF",
2303 type: "hexadecimal",
2307 strength: "centuries",
2311 it("Shows details about the entered entropy", function(done
) {
2312 testEntropyFeedback(done
,
2319 strength: "less than a second",
2323 it("Shows details about the entered entropy", function(done
) {
2324 testEntropyFeedback(done
,
2326 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2327 type: "card (full deck)",
2331 strength: "centuries",
2335 it("Shows details about the entered entropy", function(done
) {
2336 testEntropyFeedback(done
,
2338 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks3d",
2339 type: "card (full deck, 1 duplicate: 3d)",
2343 strength: "centuries",
2347 it("Shows details about the entered entropy", function(done
) {
2348 testEntropyFeedback(done
,
2350 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqs3d4d",
2351 type: "card (2 duplicates: 3d 4d, 1 missing: KS)",
2355 strength: "centuries",
2359 it("Shows details about the entered entropy", function(done
) {
2360 testEntropyFeedback(done
,
2362 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqs3d4d5d6d",
2363 type: "card (4 duplicates: 3d 4d 5d..., 1 missing: KS)",
2367 strength: "centuries",
2371 it("Shows details about the entered entropy", function(done
) {
2372 testEntropyFeedback(done
,
2373 // Next test was throwing uncaught error in zxcvbn
2374 // Also tests 451 bits, ie Math.log2(52!)*2 = 225.58 * 2
2376 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsksac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2377 type: "card (full deck, 52 duplicates: ac 2c 3c...)",
2381 strength: "centuries",
2385 it("Shows details about the entered entropy", function(done
) {
2386 testEntropyFeedback(done
,
2387 // Case insensitivity to duplicate cards
2390 type: "card (1 duplicate: AS)",
2394 strength: "less than a second",
2398 it("Shows details about the entered entropy", function(done
) {
2399 testEntropyFeedback(done
,
2402 type: "card (1 duplicate: as)",
2406 strength: "less than a second",
2410 it("Shows details about the entered entropy", function(done
) {
2411 testEntropyFeedback(done
,
2412 // Missing cards are detected
2414 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2415 type: "card (1 missing: 9C)",
2419 strength: "centuries",
2423 it("Shows details about the entered entropy", function(done
) {
2424 testEntropyFeedback(done
,
2426 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2427 type: "card (2 missing: 9C 5D)",
2431 strength: "centuries",
2435 it("Shows details about the entered entropy", function(done
) {
2436 testEntropyFeedback(done
,
2438 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d7d8d9dtdjd kdah2h3h 5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2439 type: "card (4 missing: 9C 5D QD...)",
2443 strength: "centuries",
2447 it("Shows details about the entered entropy", function(done
) {
2448 testEntropyFeedback(done
,
2449 // More than six missing cards does not show message
2451 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d 8d9d jd kdah2h3h 5h6h7h8h9hthjhqhkh 2s3s4s5s6s7s8s9stsjsqsks",
2456 strength: "centuries",
2460 it("Shows details about the entered entropy", function(done
) {
2461 testEntropyFeedback(done
,
2462 // Multiple decks of cards increases bits per event
2467 bitsPerEvent: "4.34",
2471 it("Shows details about the entered entropy", function(done
) {
2472 testEntropyFeedback(done
,
2477 bitsPerEvent: "4.80",
2481 it("Shows details about the entered entropy", function(done
) {
2482 testEntropyFeedback(done
,
2487 bitsPerEvent: "5.01",
2491 it("Shows details about the entered entropy", function(done
) {
2492 testEntropyFeedback(done
,
2494 entropy: "3d3d3d3d",
2497 bitsPerEvent: "5.14",
2501 it("Shows details about the entered entropy", function(done
) {
2502 testEntropyFeedback(done
,
2504 entropy: "3d3d3d3d3d",
2507 bitsPerEvent: "5.22",
2511 it("Shows details about the entered entropy", function(done
) {
2512 testEntropyFeedback(done
,
2514 entropy: "3d3d3d3d3d3d",
2517 bitsPerEvent: "5.28",
2521 it("Shows details about the entered entropy", function(done
) {
2522 testEntropyFeedback(done
,
2524 entropy: "3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d",
2527 bitsPerEvent: "5.59",
2528 strength: 'less than a second - Repeats like "abcabcabc" are only slightly harder to guess than "abc"',
2533 // Entropy is truncated from the left
2534 it('Truncates entropy from the left', function(done
) {
2535 // Truncate from left means 0000 is removed from the start
2536 // which gives mnemonic 'avocado zoo zone'
2537 // not 1111 removed from the end
2538 // which gives the mnemonic 'abstract zoo zoo'
2539 var entropy
= "00000000 00000000 00000000 00000000";
2540 entropy
+= "11111111 11111111 11111111 1111"; // Missing last byte
2541 driver
.findElement(By
.css('.use-entropy'))
2543 driver
.findElement(By
.css('.entropy'))
2545 driver
.sleep(generateDelay
).then(function() {
2546 driver
.findElement(By
.css(".phrase"))
2547 .getAttribute("value").then(function(phrase
) {
2548 expect(phrase
).toBe("avocado zoo zone");
2554 // Very large entropy results in very long mnemonics
2555 it('Converts very long entropy to very long mnemonics', function(done
) {
2557 for (var i
=0; i
<33; i
++) {
2558 entropy
+= "AAAAAAAA"; // 3 words * 33 iterations = 99 words
2560 driver
.findElement(By
.css('.use-entropy'))
2562 driver
.findElement(By
.css('.entropy'))
2564 driver
.sleep(generateDelay
).then(function() {
2565 driver
.findElement(By
.css(".phrase"))
2566 .getAttribute("value").then(function(phrase
) {
2567 var wordCount
= phrase
.split(/\s+/g).length
;
2568 expect(wordCount
).toBe(99);
2574 // Is compatible with bip32jp entropy
2575 // https://bip32jp.github.io/english/index.html
2577 // Is incompatible with:
2579 it('Is compatible with bip32jp.github.io', function(done
) {
2580 var entropy
= "543210543210543210543210543210543210543210543210543210543210543210543210543210543210543210543210543";
2581 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";
2582 driver
.findElement(By
.css('.use-entropy'))
2584 driver
.findElement(By
.css('.entropy'))
2586 driver
.sleep(generateDelay
).then(function() {
2587 driver
.findElement(By
.css(".phrase"))
2588 .getAttribute("value").then(function(phrase
) {
2589 expect(phrase
).toBe(expectedPhrase
);
2595 // Blank entropy does not generate mnemonic or addresses
2596 it('Does not generate mnemonic for blank entropy', function(done
) {
2597 driver
.findElement(By
.css('.use-entropy'))
2599 driver
.findElement(By
.css('.entropy'))
2601 // check there is no mnemonic
2602 driver
.sleep(generateDelay
).then(function() {
2603 driver
.findElement(By
.css(".phrase"))
2604 .getAttribute("value").then(function(phrase
) {
2605 expect(phrase
).toBe("");
2606 // check there is no mnemonic
2607 driver
.findElements(By
.css(".address"))
2608 .then(function(addresses
) {
2609 expect(addresses
.length
).toBe(0);
2610 // Check the feedback says 'blank entropy'
2611 driver
.findElement(By
.css(".feedback"))
2613 .then(function(feedbackText
) {
2614 expect(feedbackText
).toBe("Blank entropy");
2622 // Mnemonic length can be selected even for weak entropy
2623 it('Allows selection of mnemonic length even for weak entropy', function(done
) {
2624 driver
.findElement(By
.css('.use-entropy'))
2626 driver
.executeScript(function() {
2627 $(".mnemonic-length").val("18").trigger("change");
2629 driver
.findElement(By
.css('.entropy'))
2630 .sendKeys("012345");
2631 driver
.sleep(generateDelay
).then(function() {
2632 driver
.findElement(By
.css(".phrase"))
2633 .getAttribute("value").then(function(phrase
) {
2634 var wordCount
= phrase
.split(/\s+/g).length
;
2635 expect(wordCount
).toBe(18);
2642 // https://github.com/iancoleman/bip39/issues/33
2643 // Final cards should contribute entropy
2644 it('Uses as much entropy as possible for the mnemonic', function(done
) {
2645 driver
.findElement(By
.css('.use-entropy'))
2647 driver
.findElement(By
.css('.entropy'))
2648 .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");
2649 driver
.sleep(generateDelay
).then(function() {
2651 driver
.findElement(By
.css(".phrase"))
2652 .getAttribute("value").then(function(originalPhrase
) {
2653 // Set the last 12 cards to be AS
2654 driver
.findElement(By
.css('.entropy'))
2656 driver
.findElement(By
.css('.entropy'))
2657 .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");
2658 driver
.sleep(generateDelay
).then(function() {
2660 driver
.findElement(By
.css(".phrase"))
2661 .getAttribute("value").then(function(newPhrase
) {
2662 expect(originalPhrase
).not
.toEqual(newPhrase
);
2671 // https://github.com/iancoleman/bip39/issues/35
2673 // TODO this doesn't work in selenium with firefox
2674 // see https://stackoverflow.com/q/40360223
2675 it('Shows a qr code on hover for the phrase', function(done
) {
2676 if (browser
== "firefox") {
2677 pending("Selenium + Firefox bug for mouseMove, see https://stackoverflow.com/q/40360223");
2679 // generate a random mnemonic
2680 var generateEl
= driver
.findElement(By
.css('.generate'));
2682 // toggle qr to show (hidden by default)
2683 var phraseEl
= driver
.findElement(By
.css(".phrase"));
2685 var rootKeyEl
= driver
.findElement(By
.css(".root-key"));
2686 driver
.sleep(generateDelay
).then(function() {
2687 // hover over the root key
2688 driver
.actions().mouseMove(rootKeyEl
).perform().then(function() {
2689 // check the qr code shows
2690 driver
.executeScript(function() {
2691 return $(".qr-container").find("canvas").length
> 0;
2693 .then(function(qrShowing
) {
2694 expect(qrShowing
).toBe(true);
2695 // hover away from the phrase
2696 driver
.actions().mouseMove(generateEl
).perform().then(function() {;
2697 // check the qr code hides
2698 driver
.executeScript(function() {
2699 return $(".qr-container").find("canvas").length
== 0;
2701 .then(function(qrHidden
) {
2702 expect(qrHidden
).toBe(true);
2711 // BIP44 account extendend private key is shown
2712 // github issue 37 - compatibility with electrum
2713 it('Shows the bip44 account extended private key', function(done
) {
2714 driver
.findElement(By
.css(".phrase"))
2715 .sendKeys("abandon abandon ability");
2716 driver
.sleep(generateDelay
).then(function() {
2717 driver
.findElement(By
.css("#bip44 .account-xprv"))
2718 .getAttribute("value")
2719 .then(function(xprv
) {
2720 expect(xprv
).toBe("xprv9yzrnt4zWVJUr1k2VxSPy9ettKz5PpeDMgaVG7UKedhqnw1tDkxP2UyYNhuNSumk2sLE5ctwKZs9vwjsq3e1vo9egCK6CzP87H2cVYXpfwQ");
2726 // BIP44 account extendend public key is shown
2727 // github issue 37 - compatibility with electrum
2728 it('Shows the bip44 account extended public key', function(done
) {
2729 driver
.findElement(By
.css(".phrase"))
2730 .sendKeys("abandon abandon ability");
2731 driver
.sleep(generateDelay
).then(function() {
2732 driver
.findElement(By
.css("#bip44 .account-xpub"))
2733 .getAttribute("value")
2734 .then(function(xprv
) {
2735 expect(xprv
).toBe("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2742 // BIP32 root key can be set as an xpub
2743 it('Generates addresses from xpub as bip32 root key', function(done
) {
2744 driver
.findElement(By
.css('#bip32-tab a'))
2746 // set xpub for account 0 of bip44 for 'abandon abandon ability'
2747 driver
.findElement(By
.css("#root-key"))
2748 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2749 driver
.sleep(generateDelay
).then(function() {
2750 // check the addresses are generated
2751 getFirstAddress(function(address
) {
2752 expect(address
).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug");
2753 // check the xprv key is not set
2754 driver
.findElement(By
.css(".extended-priv-key"))
2755 .getAttribute("value")
2756 .then(function(xprv
) {
2757 expect(xprv
).toBe("NA");
2758 // check the private key is not set
2759 driver
.findElements(By
.css(".privkey"))
2760 .then(function(els
) {
2763 .then(function(privkey
) {
2764 expect(xprv
).toBe("NA");
2774 // xpub for bip32 root key will not work with hardened derivation paths
2775 it('Shows error for hardened derivation paths with xpub root key', function(done
) {
2776 // set xpub for account 0 of bip44 for 'abandon abandon ability'
2777 driver
.findElement(By
.css("#root-key"))
2778 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2779 driver
.sleep(feedbackDelay
).then(function() {
2780 // Check feedback is correct
2781 driver
.findElement(By
.css('.feedback'))
2783 .then(function(feedback
) {
2784 var msg
= "Hardened derivation path is invalid with xpub key";
2785 expect(feedback
).toBe(msg
);
2786 // Check no addresses are shown
2787 driver
.findElements(By
.css('.addresses tr'))
2788 .then(function(rows
) {
2789 expect(rows
.length
).toBe(0);
2797 // no root key shows feedback
2798 it('Shows feedback for no root key', function(done
) {
2799 // set xpub for account 0 of bip44 for 'abandon abandon ability'
2800 driver
.findElement(By
.css('#bip32-tab a'))
2802 driver
.sleep(feedbackDelay
).then(function() {
2803 // Check feedback is correct
2804 driver
.findElement(By
.css('.feedback'))
2806 .then(function(feedback
) {
2807 expect(feedback
).toBe("Invalid root key");
2814 // display error switching tabs while addresses are generating
2815 it('Can change details while old addresses are still being generated', function(done
) {
2816 // Set to generate 199 more addresses.
2817 // This will take a long time allowing a new set of addresses to be
2818 // generated midway through this lot.
2819 // The newly generated addresses should not include any from the old set.
2820 // Any more than 199 will show an alert which needs to be accepted.
2821 driver
.findElement(By
.css('.rows-to-add'))
2823 driver
.findElement(By
.css('.rows-to-add'))
2826 driver
.findElement(By
.css('.phrase'))
2827 .sendKeys("abandon abandon ability");
2828 driver
.sleep(generateDelay
).then(function() {
2829 // change tabs which should cancel the previous generating
2830 driver
.findElement(By
.css('.rows-to-add'))
2832 driver
.findElement(By
.css('.rows-to-add'))
2834 driver
.findElement(By
.css('#bip32-tab a'))
2836 driver
.sleep(generateDelay
).then(function() {
2837 driver
.findElements(By
.css('.index'))
2838 .then(function(els
) {
2839 // check the derivation paths have the right quantity
2840 expect(els
.length
).toBe(20);
2841 // check the derivation paths are in order
2842 testRowsAreInCorrectOrder(done
);
2846 }, generateDelay
+ 5000);
2849 // padding for binary should give length with multiple of 256
2850 // hashed entropy 1111 is length 252, so requires 4 leading zeros
2851 // prior to issue 49 it would only generate 2 leading zeros, ie missing 2
2852 it('Pads hashed entropy with leading zeros', function(done
) {
2853 driver
.findElement(By
.css('.use-entropy'))
2855 driver
.executeScript(function() {
2856 $(".mnemonic-length").val("15").trigger("change");
2858 driver
.findElement(By
.css('.entropy'))
2860 driver
.sleep(generateDelay
).then(function() {
2861 driver
.findElement(By
.css('.phrase'))
2862 .getAttribute("value")
2863 .then(function(phrase
) {
2864 expect(phrase
).toBe("avocado valid quantum cross link predict excuse edit street able flame large galaxy ginger nuclear");
2870 // Github pull request 55
2871 // https://github.com/iancoleman/bip39/pull/55
2873 it('Can set the derivation path on bip32 tab for bitcoincore', function(done
) {
2874 testClientSelect(done
, {
2876 bip32path: "m/0'/0'",
2877 useHardenedAddresses: "true",
2880 it('Can set the derivation path on bip32 tab for multibit', function(done
) {
2881 testClientSelect(done
, {
2883 bip32path: "m/0'/0",
2884 useHardenedAddresses: null,
2887 it('Can set the derivation path on bip32 tab for coinomi/ledger', function(done
) {
2888 testClientSelect(done
, {
2890 bip32path: "m/44'/0'/0'",
2891 useHardenedAddresses: null,
2896 // https://github.com/iancoleman/bip39/issues/58
2897 // bip32 derivation is correct, does not drop leading zeros
2899 // https://medium.com/@alexberegszaszi/why-do-my-bip32-wallets-disagree-6f3254cc5846
2900 it('Retains leading zeros for bip32 derivation', function(done
) {
2901 driver
.findElement(By
.css(".phrase"))
2902 .sendKeys("fruit wave dwarf banana earth journey tattoo true farm silk olive fence");
2903 driver
.findElement(By
.css(".passphrase"))
2904 .sendKeys("banana");
2905 driver
.sleep(generateDelay
).then(function() {
2906 getFirstAddress(function(address
) {
2907 // Note that bitcore generates an incorrect address
2908 // 13EuKhffWkBE2KUwcbkbELZb1MpzbimJ3Y
2909 // see the medium.com link above for more details
2910 expect(address
).toBe("17rxURoF96VhmkcEGCj5LNQkmN9HVhWb7F");
2917 // Japanese mnemonics generate incorrect bip32 seed
2918 // BIP39 seed is set from phrase
2919 it('Generates correct seed for Japanese mnemonics', function(done
) {
2920 driver
.findElement(By
.css(".phrase"))
2921 .sendKeys("あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あおぞら");
2922 driver
.findElement(By
.css(".passphrase"))
2923 .sendKeys("メートルガバヴァぱばぐゞちぢ十人十色");
2924 driver
.sleep(generateDelay
).then(function() {
2925 driver
.findElement(By
.css(".seed"))
2926 .getAttribute("value")
2927 .then(function(seed
) {
2928 expect(seed
).toBe("a262d6fb6122ecf45be09c50492b31f92e9beb7d9a845987a02cefda57a15f9c467a17872029a9e92299b5cbdf306e3a0ee620245cbd508959b6cb7ca637bd55");
2934 // BIP49 official test vectors
2935 // https://github.com/bitcoin/bips/blob/master/bip-0049.mediawiki#test-vectors
2936 it('Generates BIP49 addresses matching the official test vectors', function(done
) {
2937 driver
.findElement(By
.css('#bip49-tab a'))
2939 selectNetwork("BTC - Bitcoin Testnet");
2940 driver
.findElement(By
.css(".phrase"))
2941 .sendKeys("abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about");
2942 driver
.sleep(generateDelay
).then(function() {
2943 getFirstAddress(function(address
) {
2944 expect(address
).toBe("2Mww8dCYPUpKHofjgcXcBCEGmniw9CoaiD2");
2950 // BIP49 derivation path is shown
2951 it('Shows the bip49 derivation path', function(done
) {
2952 driver
.findElement(By
.css('#bip49-tab a'))
2954 driver
.findElement(By
.css(".phrase"))
2955 .sendKeys("abandon abandon ability");
2956 driver
.sleep(generateDelay
).then(function() {
2957 driver
.findElement(By
.css('#bip49 .path'))
2958 .getAttribute("value")
2959 .then(function(path
) {
2960 expect(path
).toBe("m/49'/0'/0'/0");
2966 // BIP49 extended private key is shown
2967 it('Shows the bip49 extended private key', function(done
) {
2968 driver
.findElement(By
.css('#bip49-tab a'))
2970 driver
.findElement(By
.css(".phrase"))
2971 .sendKeys("abandon abandon ability");
2972 driver
.sleep(generateDelay
).then(function() {
2973 driver
.findElement(By
.css('.extended-priv-key'))
2974 .getAttribute("value")
2975 .then(function(xprv
) {
2976 expect(xprv
).toBe("yprvALYB4DYRG6CzzVgzQZwwqjAA2wjBGC3iEd7KYYScpoDdmf75qMRWZWxoFcRXBJjgEXdFqJ9vDRGRLJQsrL22Su5jMbNFeM9vetaGVqy9Qy2");
2982 // BIP49 extended public key is shown
2983 it('Shows the bip49 extended public key', function(done
) {
2984 driver
.findElement(By
.css('#bip49-tab a'))
2986 driver
.findElement(By
.css(".phrase"))
2987 .sendKeys("abandon abandon ability");
2988 driver
.sleep(generateDelay
).then(function() {
2989 driver
.findElement(By
.css('.extended-pub-key'))
2990 .getAttribute("value")
2991 .then(function(xprv
) {
2992 expect(xprv
).toBe("ypub6ZXXTj5K6TmJCymTWbUxCs6tayZffemZbr2vLvrEP8kceTSENtjm7KHH6thvAKxVar9fGe8rgsPEX369zURLZ68b4f7Vexz7RuXsjQ69YDt");
2998 // BIP49 account field changes address list
2999 it('Can set the bip49 account field', function(done
) {
3000 driver
.findElement(By
.css('#bip49-tab a'))
3002 driver
.findElement(By
.css('#bip49 .account'))
3004 driver
.findElement(By
.css('#bip49 .account'))
3006 driver
.findElement(By
.css(".phrase"))
3007 .sendKeys("abandon abandon ability");
3008 driver
.sleep(generateDelay
).then(function() {
3009 getFirstAddress(function(address
) {
3010 expect(address
).toBe("381wg1GGN4rP88rNC9v7QWsiww63yLVPsn");
3016 // BIP49 change field changes address list
3017 it('Can set the bip49 change field', function(done
) {
3018 driver
.findElement(By
.css('#bip49-tab a'))
3020 driver
.findElement(By
.css('#bip49 .change'))
3022 driver
.findElement(By
.css('#bip49 .change'))
3024 driver
.findElement(By
.css(".phrase"))
3025 .sendKeys("abandon abandon ability");
3026 driver
.sleep(generateDelay
).then(function() {
3027 getFirstAddress(function(address
) {
3028 expect(address
).toBe("3PEM7MiKed5konBoN66PQhK8r3hjGhy9dT");
3034 // BIP49 account extendend private key is shown
3035 it('Shows the bip49 account extended private key', function(done
) {
3036 driver
.findElement(By
.css('#bip49-tab a'))
3038 driver
.findElement(By
.css(".phrase"))
3039 .sendKeys("abandon abandon ability");
3040 driver
.sleep(generateDelay
).then(function() {
3041 driver
.findElement(By
.css('#bip49 .account-xprv'))
3042 .getAttribute("value")
3043 .then(function(xprv
) {
3044 expect(xprv
).toBe("yprvAHtB1M5Wp675aLzFy9TJYK2mSsLkg6mcBRh5DZTR7L4EnYSmYPaL63KFA4ycg1PngW5LfkmejxzosCs17TKZMpRFKc3z5SJar6QAKaFcaZL");
3050 // BIP49 account extendend public key is shown
3051 it('Shows the bip49 account extended public key', function(done
) {
3052 driver
.findElement(By
.css('#bip49-tab a'))
3054 driver
.findElement(By
.css(".phrase"))
3055 .sendKeys("abandon abandon ability");
3056 driver
.sleep(generateDelay
).then(function() {
3057 driver
.findElement(By
.css('#bip49 .account-xpub'))
3058 .getAttribute("value")
3059 .then(function(xprv
) {
3060 expect(xprv
).toBe("ypub6WsXQrcQeTfNnq4j5AzJuSyVzuBF5ZVTYecg1ws2ffbDfLmv5vtadqdj1NgR6C6gufMpMfJpHxvb6JEQKvETVNWCRanNedfJtnTchZiJtsL");
3066 // Test selecting coin where bip49 is unavailable (eg CLAM)
3067 it('Shows an error on bip49 tab for coins without bip49', function(done
) {
3068 driver
.findElement(By
.css('#bip49-tab a'))
3070 driver
.findElement(By
.css(".phrase"))
3071 .sendKeys("abandon abandon ability");
3072 driver
.sleep(generateDelay
).then(function() {
3073 selectNetwork("CLAM - Clams");
3074 // bip49 available is hidden
3075 driver
.findElement(By
.css('#bip49 .available'))
3076 .getAttribute("class")
3077 .then(function(classes
) {
3078 expect(classes
).toContain("hidden");
3079 // bip49 unavailable is shown
3080 driver
.findElement(By
.css('#bip49 .unavailable'))
3081 .getAttribute("class")
3082 .then(function(classes
) {
3083 expect(classes
).not
.toContain("hidden");
3084 // check there are no addresses shown
3085 driver
.findElements(By
.css('.addresses tr'))
3086 .then(function(rows
) {
3087 expect(rows
.length
).toBe(0);
3088 // check the derived private key is blank
3089 driver
.findElement(By
.css('.extended-priv-key'))
3090 .getAttribute("value")
3091 .then(function(xprv
) {
3092 expect(xprv
).toBe('');
3093 // check the derived public key is blank
3094 driver
.findElement(By
.css('.extended-pub-key'))
3095 .getAttribute("value")
3096 .then(function(xpub
) {
3097 expect(xpub
).toBe('');
3108 // Cleared mnemonic and root key still allows addresses to be generated
3109 // https://github.com/iancoleman/bip39/issues/43
3110 it('Clears old root keys from memory when mnemonic is cleared', function(done
) {
3112 driver
.findElement(By
.css(".phrase"))
3113 .sendKeys("abandon abandon ability");
3114 driver
.sleep(generateDelay
).then(function() {
3115 // clear the mnemonic and root key
3116 // using selenium .clear() doesn't seem to trigger the 'input' event
3117 // so clear it using keys instead
3118 driver
.findElement(By
.css('.phrase'))
3119 .sendKeys(Key
.CONTROL
,"a");
3120 driver
.findElement(By
.css('.phrase'))
3121 .sendKeys(Key
.DELETE
);
3122 driver
.findElement(By
.css('.root-key'))
3123 .sendKeys(Key
.CONTROL
,"a");
3124 driver
.findElement(By
.css('.root-key'))
3125 .sendKeys(Key
.DELETE
);
3126 driver
.sleep(generateDelay
).then(function() {
3127 // try to generate more addresses
3128 driver
.findElement(By
.css('.more'))
3130 driver
.sleep(generateDelay
).then(function() {
3131 driver
.findElements(By
.css(".addresses tr"))
3132 .then(function(els
) {
3133 // check there are no addresses shown
3134 expect(els
.length
).toBe(0);
3143 // error trying to generate addresses from xpub with hardened derivation
3144 it('Shows error for hardened addresses with xpub root key', function(done
) {
3145 driver
.findElement(By
.css('#bip32-tab a'))
3147 driver
.executeScript(function() {
3148 $(".hardened-addresses").prop("checked", true);
3150 // set xpub for account 0 of bip44 for 'abandon abandon ability'
3151 driver
.findElement(By
.css("#root-key"))
3152 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
3153 driver
.sleep(feedbackDelay
).then(function() {
3154 // Check feedback is correct
3155 driver
.findElement(By
.css('.feedback'))
3157 .then(function(feedback
) {
3158 var msg
= "Hardened derivation path is invalid with xpub key";
3159 expect(feedback
).toBe(msg
);
3165 // Litecoin uses ltub by default, and can optionally be set to xprv
3167 // https://github.com/iancoleman/bip39/issues/96
3168 // Issue with extended keys on Litecoin
3169 it('Uses ltub by default for litecoin, but can be set to xprv', function(done
) {
3170 driver
.findElement(By
.css('.phrase'))
3171 .sendKeys("abandon abandon ability");
3172 selectNetwork("LTC - Litecoin");
3173 driver
.sleep(generateDelay
).then(function() {
3174 // check the extended key is generated correctly
3175 driver
.findElement(By
.css('.root-key'))
3176 .getAttribute("value")
3177 .then(function(rootKey
) {
3178 expect(rootKey
).toBe("Ltpv71G8qDifUiNesiPqf6h5V6eQ8ic77oxQiYtawiACjBEx3sTXNR2HGDGnHETYxESjqkMLFBkKhWVq67ey1B2MKQXannUqNy1RZVHbmrEjnEU");
3179 // set litecoin to use ltub
3180 driver
.executeScript(function() {
3181 $(".litecoin-use-ltub").prop("checked", false);
3182 $(".litecoin-use-ltub").trigger("change");
3184 driver
.sleep(generateDelay
).then(function() {
3185 driver
.findElement(By
.css('.root-key'))
3186 .getAttribute("value")
3187 .then(function(rootKey
) {
3188 expect(rootKey
).toBe("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
3197 // https://github.com/iancoleman/bip39/issues/99#issuecomment-327094159
3198 // "warn me emphatically when they have detected invalid input" to the entropy field
3199 // A warning is shown when entropy is filtered and discarded
3200 it('Warns when entropy is filtered and discarded', function(done
) {
3201 driver
.findElement(By
.css('.use-entropy'))
3203 // set entropy to have no filtered content
3204 driver
.findElement(By
.css('.entropy'))
3205 .sendKeys("00000000 00000000 00000000 00000000");
3206 driver
.sleep(generateDelay
).then(function() {
3207 // check the filter warning does not show
3208 driver
.findElement(By
.css('.entropy-container .filter-warning'))
3209 .getAttribute("class")
3210 .then(function(classes
) {
3211 expect(classes
).toContain("hidden");
3212 // set entropy to have some filtered content
3213 driver
.findElement(By
.css('.entropy'))
3214 .sendKeys("10000000 zxcvbn 00000000 00000000 00000000");
3215 driver
.sleep(entropyFeedbackDelay
).then(function() {
3216 // check the filter warning shows
3217 driver
.findElement(By
.css('.entropy-container .filter-warning'))
3218 .getAttribute("class")
3219 .then(function(classes
) {
3220 expect(classes
).not
.toContain("hidden");
3228 // Bitcoin Cash address can be set to use cashaddr format
3229 it('Can use cashaddr format for bitcoin cash addresses', function(done
) {
3230 driver
.executeScript(function() {
3231 $(".use-bch-cashaddr-addresses").prop("checked", true);
3233 driver
.findElement(By
.css('.phrase'))
3234 .sendKeys("abandon abandon ability");
3235 selectNetwork("BCH - Bitcoin Cash");
3236 driver
.sleep(generateDelay
).then(function() {
3237 getFirstAddress(function(address
) {
3238 expect(address
).toBe("bitcoincash:qzlquk7w4hkudxypl4fgv8x279r754dkvur7jpcsps");
3244 // Bitcoin Cash address can be set to use bitpay format
3245 it('Can use bitpay format for bitcoin cash addresses', function(done
) {
3246 driver
.executeScript(function() {
3247 $(".use-bch-bitpay-addresses").prop("checked", true);
3249 driver
.findElement(By
.css('.phrase'))
3250 .sendKeys("abandon abandon ability");
3251 selectNetwork("BCH - Bitcoin Cash");
3252 driver
.sleep(generateDelay
).then(function() {
3253 getFirstAddress(function(address
) {
3254 expect(address
).toBe("CZnpA9HPmvhuhLLPWJP8rNDpLUYXy1LXFk");
3260 // Bitcoin Cash address can be set to use legacy format
3261 it('Can use legacy format for bitcoin cash addresses', function(done
) {
3262 driver
.executeScript(function() {
3263 $(".use-bch-legacy-addresses").prop("checked", true);
3265 driver
.findElement(By
.css('.phrase'))
3266 .sendKeys("abandon abandon ability");
3267 selectNetwork("BCH - Bitcoin Cash");
3268 driver
.sleep(generateDelay
).then(function() {
3269 getFirstAddress(function(address
) {
3270 expect(address
).toBe("1JKvb6wKtsjNoCRxpZ4DGrbniML7z5U16A");
3276 // End of tests ported from old suit, so no more comments above each test now
3278 it('Can generate more addresses from a custom index', function(done
) {
3279 var expectedIndexes
= [
3280 0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,
3281 40,41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59
3283 driver
.findElement(By
.css('.phrase'))
3284 .sendKeys("abandon abandon ability");
3285 driver
.sleep(generateDelay
).then(function() {
3286 // Set start of next lot of rows to be from index 40
3287 // which means indexes 20-39 will not be in the table.
3288 driver
.findElement(By
.css('.more-rows-start-index'))
3290 driver
.findElement(By
.css('.more'))
3292 driver
.sleep(generateDelay
).then(function() {
3293 // Check actual indexes in the table match the expected pattern
3294 driver
.findElements(By
.css(".index"))
3295 .then(function(els
) {
3296 expect(els
.length
).toBe(expectedIndexes
.length
);
3297 var testRowAtIndex = function(i
) {
3298 if (i
>= expectedIndexes
.length
) {
3303 .then(function(actualPath
) {
3304 var noHardened
= actualPath
.replace(/'/g, "");
3305 var pathBits = noHardened.split("/")
3306 var lastBit = pathBits[pathBits.length-1];
3307 var actualIndex = parseInt(lastBit);
3308 var expectedIndex = expectedIndexes[i];
3309 expect(actualIndex).toBe(expectedIndex);
3310 testRowAtIndex(i+1);
3320 it('Can generate BIP141 addresses
with P2WPKH
-in-P2SH semanitcs
', function(done) {
3321 // Sourced from BIP49 official test specs
3322 driver.findElement(By.css('#bip141
-tab a
'))
3324 driver.findElement(By.css('.bip141
-path
'))
3326 driver.findElement(By.css('.bip141
-path
'))
3327 .sendKeys("m/49'/1'/0'/0");
3328 selectNetwork("BTC
- Bitcoin Testnet
");
3329 driver.findElement(By.css(".phrase
"))
3330 .sendKeys("abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about
");
3331 driver.sleep(generateDelay).then(function() {
3332 getFirstAddress(function(address) {
3333 expect(address).toBe("2Mww8dCYPUpKHofjgcXcBCEGmniw9CoaiD2
");
3339 it('Can generate BIP141 addresses with P2WPKH semanitcs', function(done) {
3340 // This result tested against bitcoinjs-lib test spec for segwit address
3341 // using the first private key of this mnemonic and default path m/0
3342 // https://github.com/bitcoinjs/bitcoinjs-lib/blob/9c8503cab0c6c30a95127042703bc18e8d28c76d/test/integration/addresses.js#L50
3343 // so whilst not directly comparable, substituting the private key produces
3344 // identical results between this tool and the bitcoinjs-lib test.
3345 // Private key generated is:
3346 // L3L8Nu9whawPBNLGtFqDhKut9DKKfG3CQoysupT7BimqVCZsLFNP
3347 driver.findElement(By.css('#bip141-tab a'))
3350 driver.executeScript(function() {
3351 $(".bip141
-semantics option
[selected
]").removeAttr("selected
");
3352 $(".bip141
-semantics option
").filter(function(i,e) {
3353 return $(e).html() == "P2WPKH
";
3354 }).prop("selected
", true);
3355 $(".bip141
-semantics
").trigger("change
");
3357 driver.findElement(By.css(".phrase
"))
3358 .sendKeys("abandon abandon ability
");
3359 driver.sleep(generateDelay).then(function() {
3360 getFirstAddress(function(address) {
3361 expect(address).toBe("bc1qfwu6a5a3evygrk8zvdxxvz4547lmpyx5vsfxe9
");
3367 it('Shows the entropy used by the PRNG when clicking generate', function(done) {
3368 driver.findElement(By.css('.generate')).click();
3369 driver.sleep(generateDelay).then(function() {
3370 driver.findElement(By.css('.entropy'))
3371 .getAttribute("value
")
3372 .then(function(entropy) {
3373 expect(entropy).not.toBe("");
3379 it('Shows the index of each word in the mnemonic', function(done) {
3380 driver.findElement(By.css('.phrase'))
3381 .sendKeys("abandon abandon ability
");
3382 driver.sleep(generateDelay).then(function() {
3383 driver.findElement(By.css('.use-entropy'))
3385 driver.findElement(By.css('.word-indexes'))
3387 .then(function(indexes) {
3388 expect(indexes).toBe("0, 0, 1");
3394 it('Shows the derivation path for bip84 tab', function(done) {
3395 driver.findElement(By.css('#bip84-tab a'))
3397 driver.findElement(By.css('.phrase'))
3398 .sendKeys('abandon abandon ability');
3399 driver.sleep(generateDelay).then(function() {
3400 driver.findElement(By.css('#bip84 .path'))
3401 .getAttribute("value
")
3402 .then(function(path) {
3403 expect(path).toBe("m
/84'/0'/0'/0");
3409 it('Shows the extended private key for bip84 tab', function(done) {
3410 driver.findElement(By.css('#bip84-tab a'))
3412 driver.findElement(By.css('.phrase'))
3413 .sendKeys('abandon abandon ability');
3414 driver.sleep(generateDelay).then(function() {
3415 driver.findElement(By.css('.extended-priv-key'))
3416 .getAttribute("value
")
3417 .then(function(path) {
3418 expect(path).toBe("zprvAev3RKrZ3QVKiUFCfdeMRen1BPDJgdNt1XpxiDy8acSs4kkAGTCvq7HeRYRNNpo8EtEjCFQBWavJwtCUR29y4TUCH4X5RXMcyq48uN8y9BP
");
3424 it('Shows the extended public key for bip84 tab', function(done) {
3425 driver.findElement(By.css('#bip84-tab a'))
3427 driver.findElement(By.css('.phrase'))
3428 .sendKeys('abandon abandon ability');
3429 driver.sleep(generateDelay).then(function() {
3430 driver.findElement(By.css('.extended-pub-key'))
3431 .getAttribute("value
")
3432 .then(function(path) {
3433 expect(path).toBe("zpub6suPpqPSsn3cvxKfmfBMnnijjR3o666jNkkZWcNk8wyqwZ5JozXBNuc8Gs7DB3uLwTDvGVTspVEAUQcEjKF3pZHgywVbubdTqbXTUg7usyx
");
3439 it('Changes the address list if bip84 account is changed', function(done) {
3440 driver.findElement(By.css('#bip84-tab a'))
3442 driver.findElement(By.css('#bip84 .account'))
3444 driver.findElement(By.css('.phrase'))
3445 .sendKeys('abandon abandon ability');
3446 driver.sleep(generateDelay).then(function() {
3447 getFirstAddress(function(address) {
3448 expect(address).toBe("bc1qp7vv669t2fy965jdzvqwrraana89ctd5ewc662
");
3454 it('Changes the address list if bip84 change is changed', function(done) {
3455 driver.findElement(By.css('#bip84-tab a'))
3457 driver.findElement(By.css('#bip84 .change'))
3459 driver.findElement(By.css('.phrase'))
3460 .sendKeys('abandon abandon ability');
3461 driver.sleep(generateDelay).then(function() {
3462 getFirstAddress(function(address) {
3463 expect(address).toBe("bc1qr39vj6rh06ff05m53uxq8uazehwhccswylhrs2
");
3469 it('Passes the official BIP84 test spec for rootpriv', function(done) {
3470 driver.findElement(By.css('#bip84-tab a'))
3472 driver.findElement(By.css('.phrase'))
3473 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3474 driver.sleep(generateDelay).then(function() {
3475 driver.findElement(By.css(".root
-key
"))
3476 .getAttribute("value
")
3477 .then(function(rootKey) {
3478 expect(rootKey).toBe("zprvAWgYBBk7JR8Gjrh4UJQ2uJdG1r3WNRRfURiABBE3RvMXYSrRJL62XuezvGdPvG6GFBZduosCc1YP5wixPox7zhZLfiUm8aunE96BBa4Kei5
");
3484 it('Passes the official BIP84 test spec for account 0 xprv', function(done) {
3485 driver.findElement(By.css('#bip84-tab a'))
3487 driver.findElement(By.css('.phrase'))
3488 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3489 driver.sleep(generateDelay).then(function() {
3490 driver.findElement(By.css("#bip84
.account
-xprv
"))
3491 .getAttribute("value
")
3492 .then(function(rootKey) {
3493 expect(rootKey).toBe("zprvAdG4iTXWBoARxkkzNpNh8r6Qag3irQB8PzEMkAFeTRXxHpbF9z4QgEvBRmfvqWvGp42t42nvgGpNgYSJA9iefm1yYNZKEm7z6qUWCroSQnE
");
3499 it('Passes the official BIP84 test spec for account 0 xpub', function(done) {
3500 driver.findElement(By.css('#bip84-tab a'))
3502 driver.findElement(By.css('.phrase'))
3503 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3504 driver.sleep(generateDelay).then(function() {
3505 driver.findElement(By.css("#bip84
.account
-xpub
"))
3506 .getAttribute("value
")
3507 .then(function(rootKey) {
3508 expect(rootKey).toBe("zpub6rFR7y4Q2AijBEqTUquhVz398htDFrtymD9xYYfG1m4wAcvPhXNfE3EfH1r1ADqtfSdVCToUG868RvUUkgDKf31mGDtKsAYz2oz2AGutZYs
");
3514 it('Passes the official BIP84 test spec for account 0 first address', function(done) {
3515 driver.findElement(By.css('#bip84-tab a'))
3517 driver.findElement(By.css('.phrase'))
3518 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3519 driver.sleep(generateDelay).then(function() {
3520 getFirstAddress(function(address) {
3521 expect(address).toBe("bc1qcr8te4kr609gcawutmrza0j4xv80jy8z306fyu
");
3527 it('Passes the official BIP84 test spec for account 0 first change address', function(done) {
3528 driver.findElement(By.css('#bip84-tab a'))
3530 driver.findElement(By.css('.phrase'))
3531 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3532 driver.findElement(By.css('#bip84 .change'))
3534 driver.sleep(generateDelay).then(function() {
3535 getFirstAddress(function(address) {
3536 expect(address).toBe("bc1q8c6fshw2dlwun7ekn9qwf37cu2rn755upcp6el
");
3542 it('Can display the table as csv', function(done) {
3543 var headings = "path
,address
,public key
,private key
";
3544 var row1 = "m
/44'/0'/0'/0/0,1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug
,033f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3
,L26cVSpWFkJ6aQkPkKmTzLqTdLJ923e6CzrVh9cmx21QHsoUmrEE
";
3545 var row20 = "m
/44'/0'/0'/0/19,1KhBy28XLAciXnnRvm71PvQJaETyrxGV55
,02b4b3e396434d8cdd20c03ac4aaa07387784d5d867b75987f516f5705ee68cb3a
,L4GrDrjReMsCAu5DkLXn79jSb95qR7Zfx7eshybCQZ1qL32MXJab
";
3546 driver.findElement(By.css('.phrase'))
3547 .sendKeys('abandon abandon ability');
3548 driver.sleep(generateDelay).then(function() {
3549 driver.findElement(By.css('.csv'))
3550 .getAttribute("value
")
3551 .then(function(csv) {
3552 expect(csv).toContain(headings);
3553 expect(csv).toContain(row1);
3554 expect(csv).toContain(row20);
3560 it('LeftPads ethereum keys that are less than 32 bytes', function(done) {
3561 // see https://github.com/iancoleman/bip39/issues/155
3562 selectNetwork("ETH
- Ethereum
");
3563 driver.findElement(By.css('#bip32-tab a'))
3565 driver.findElement(By.css('#bip32-path'))
3567 driver.findElement(By.css('#bip32-path'))
3568 .sendKeys("m
/44'/60'/0'");
3569 driver.findElement(By.css('.phrase'))
3570 .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');
3571 driver.sleep(generateDelay).then(function() {
3572 getFirstAddress(function(address) {
3573 expect(address).toBe("0x8943E785B4a5714FC87a3aFAad1eB1FeB602B118");
3579 it('Can encrypt private keys using BIP38', function(done) {
3580 // see https://github.com/iancoleman/bip39/issues/140
3581 driver.executeScript(function() {
3582 $(".use-bip38
").prop("checked
", true);
3584 driver.findElement(By.css('.bip38-password'))
3585 .sendKeys('bip38password');
3586 driver.findElement(By.css('.rows-to-add'))
3588 driver.findElement(By.css('.rows-to-add'))
3590 driver.findElement(By.css('.phrase'))
3591 .sendKeys('abandon abandon ability');
3592 driver.sleep(bip38delay).then(function() {
3594 getFirstRowValue(function(address) {
3595 expect(address).toBe("1NCvSdumA3ngMM9c4aqU56AM6rqXddfuXB
");
3597 getFirstRowValue(function(pubkey) {
3598 expect(pubkey).toBe("043f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3884a74447ea901729b1e73a999b7520e7cb55b4120e6432c64153ccab8a848e1
");
3600 getFirstRowValue(function(privkey) {
3601 expect(privkey).toBe("6PRNRiFnj1RoR3sXhymdCvoZCgnUHQpfupNdKkFbWJkwWQEKesWt1EDMDM
");
3607 }, bip38delay + 5000);
3609 it('Shows the checksum for the entropy', function(done) {
3610 driver.findElement(By.css('.use-entropy'))
3612 driver.findElement(By.css('.entropy'))
3613 .sendKeys("00000000000000000000000000000000");
3614 driver.sleep(generateDelay).then(function() {
3615 driver.findElement(By.css('.checksum'))
3617 .then(function(text) {
3618 expect(text).toBe("1");
3624 it('Shows the checksum for the entropy with the correct groupings', function(done) {
3625 driver.findElement(By.css('.use-entropy'))
3627 // create a checksum of 20 bits, which spans multiple words
3628 driver.findElement(By.css('.entropy'))
3629 .sendKeys("F000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
");
3630 driver.sleep(generateDelay).then(function() {
3631 driver.findElement(By.css('.checksum'))
3633 .then(function(text) {
3634 // first group is 9 bits, second group is 11
3635 expect(text).toBe("011010111 01110000110");
3641 it('Uses vprv for bitcoin testnet p2wpkh', function(done) {
3642 selectNetwork("BTC
- Bitcoin Testnet
");
3643 driver.findElement(By.css('#bip84-tab a'))
3645 driver.findElement(By.css('.phrase'))
3646 .sendKeys('abandon abandon ability');
3647 driver.sleep(generateDelay).then(function() {
3648 driver.findElement(By.css('.root-key'))
3649 .getAttribute("value
")
3650 .then(function(path) {
3651 expect(path).toBe("vprv9DMUxX4ShgxML9N2YV5CvWEebWrM9aJ5ULpbRRyzyWu6vs4BzTvbfFFrH41N5hVi7MYSfiugd765L3JmAfDM5po36Y8ouCKRDeYQwByCmS7
");
3657 it('Shows a warning if generating weak mnemonics', function(done) {
3658 driver.executeScript(function() {
3659 $(".strength option
[selected
]").removeAttr("selected
");
3660 $(".strength option
[value
=6]").prop("selected
", true);
3661 $(".strength
").trigger("change
");
3663 driver.findElement(By.css(".generate
-container
.warning
"))
3664 .getAttribute("class")
3665 .then(function(classes) {
3666 expect(classes).not.toContain("hidden
");
3671 it('Does not show a warning if generating strong mnemonics', function(done) {
3672 driver.executeScript(function() {
3673 $(".strength option
[selected
]").removeAttr("selected
");
3674 $(".strength option
[value
=12]").prop("selected
", true);
3676 driver.findElement(By.css(".generate
-container
.warning
"))
3677 .getAttribute("class")
3678 .then(function(classes) {
3679 expect(classes).toContain("hidden
");
3684 it('Shows a warning if overriding weak entropy with longer mnemonics', function(done) {
3685 driver.findElement(By.css('.use-entropy'))
3687 driver.findElement(By.css('.entropy'))
3688 .sendKeys("0123456789abcdef
"); // 6 words
3689 driver.executeScript(function() {
3690 $(".mnemonic
-length
").val("12").trigger("change
");
3692 driver.findElement(By.css(".weak
-entropy
-override
-warning
"))
3693 .getAttribute("class")
3694 .then(function(classes) {
3695 expect(classes).not.toContain("hidden
");
3700 it('Does not show a warning if entropy is stronger than mnemonic length', function(done) {
3701 driver.findElement(By.css('.use-entropy'))
3703 driver.findElement(By.css('.entropy'))
3704 .sendKeys("0123456789abcdef0123456789abcdef0123456789abcdef
"); // 18 words
3705 driver.executeScript(function() {
3706 $(".mnemonic
-length
").val("12").trigger("change
");
3708 driver.findElement(By.css(".weak
-entropy
-override
-warning
"))
3709 .getAttribute("class")
3710 .then(function(classes) {
3711 expect(classes).toContain("hidden
");
3716 it('Shows litecoin BIP49 addresses', function(done) {
3717 driver.findElement(By.css('.phrase'))
3718 .sendKeys('abandon abandon ability');
3719 selectNetwork("LTC
- Litecoin
");
3720 driver.findElement(By.css('#bip49-tab a'))
3722 // bip49 addresses are shown
3723 driver.sleep(generateDelay).then(function() {
3724 driver.findElement(By.css('#bip49 .available'))
3725 .getAttribute("class")
3726 .then(function(classes) {
3727 expect(classes).not.toContain("hidden
");
3728 // check first address
3729 getFirstAddress(function(address) {
3730 expect(address).toBe("MFwLPhsXoBuSLL8cLmW9uK6tChkzduV8qN
");
3737 it('Can use root keys to generate segwit table rows', function(done) {
3738 // segwit uses ypub / zpub instead of xpub but the root key should still
3739 // be valid regardless of the encoding used to import that key.
3740 // Maybe this breaks the reason for the different extended key prefixes, but
3741 // since the parsed root key is used behind the scenes anyhow this should be
3743 driver.findElement(By.css('#root-key'))
3744 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
3745 driver.findElement(By.css('#bip49-tab a'))
3747 // bip49 addresses are shown
3748 driver.sleep(generateDelay).then(function() {
3749 getFirstAddress(function(address) {
3750 expect(address).toBe("3QG2Y9AA4xZ846gKHZqNf7mvVKbLqMKxr2
");