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: "XQ4HLxUVS3egk5ff1o9e2vJFJKSSsUH3B7",
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: "1Dg18EtqhReS11e9h8khkLjWGLHVjPM2AB",
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: "1M4druAcUfkXBaAcQ4cCgCLPHChiaib6kL",
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 Syscoin
', function(done) {
1101 selectText: "SYS - Syscoin",
1102 firstAddress: "SZwJi42Pst3VAMomyK5DG4157WM5ofRmSj",
1104 testNetwork(done, params);
1106 it('Allows selection
of Toa
', function(done) {
1108 selectText: "TOA - Toa",
1109 firstAddress: "TSe1QAnUwQzUfbBusDzRJ9URttrRGKoNKF",
1111 testNetwork(done, params);
1113 it('Allows selection
of Ultimatesecurecash
', function(done) {
1115 selectText: "USC - Ultimatesecurecash",
1116 firstAddress: "UPyLAZU2Che5fiy7Ed8xVJFmXAUhitA4ug",
1118 testNetwork(done, params);
1120 it('Allows selection
of Unobtanium
', function(done) {
1122 selectText: "UNO - Unobtanium",
1123 firstAddress: "uUBMPVMXrR6qhqornJqKTWgr8L69vihSL9",
1125 testNetwork(done, params);
1127 it('Allows selection
of Vcash
', function(done) {
1129 selectText: "XVC - Vcash",
1130 firstAddress: "VuL53MSY6KjvAjKSeRkh3NDnKykacDVeps",
1132 testNetwork(done, params);
1134 it('Allows selection
of Verge
', function(done) {
1136 selectText: "XVG - Verge",
1137 firstAddress: "DCrVuGkMjLJpTGgwAgv9AcMdeb1nkWbjZA",
1139 testNetwork(done, params);
1141 it('Allows selection
of Vertcoin
', function(done) {
1143 selectText: "VTC - Vertcoin",
1144 firstAddress: "Vf6koGuiWdXQfx8tNqxoNeEDxh4xh5cxsG",
1146 testNetwork(done, params);
1148 it('Allows selection
of Vivo
', function(done) {
1150 selectText: "VIVO - Vivo",
1151 firstAddress: "VFmBwuXXGhJe7MarQG2GfzHMFebRHgfSpB",
1153 testNetwork(done, params);
1155 it('Allows selection
of Vpncoin
', function(done) {
1157 selectText: "VASH - Vpncoin",
1158 firstAddress: "VoEmH1qXC4TsSgBAStR21QYetwnFqbqCx9",
1160 testNetwork(done, params);
1162 it('Allows selection
of Whitecoin
', function(done) {
1164 selectText: "XWC - Whitecoin",
1165 firstAddress: "WcSwCAUqrSgeSYbsaS3SSWWhsx8KRYTFDR",
1167 testNetwork(done, params);
1169 it('Allows selection
of Wincoin
', function(done) {
1171 selectText: "WC - Wincoin",
1172 firstAddress: "WaDVCESMGgyKgNESdn3u43NnwmGSkZED3Z",
1174 testNetwork(done, params);
1176 it('Allows selection
of Zcoin
', function(done) {
1178 selectText: "XZC - Zcoin",
1179 firstAddress: "a6VcMdP4XgAA9Tr7xNszmPG5FZpfRf17Cq",
1181 testNetwork(done, params);
1183 it('Allows selection
of Zcash
', function(done) {
1185 selectText: "ZEC - Zcash",
1186 firstAddress: "t1Sz8AneMcVuzUg3tPJ8et5AS5LFJ7K2EF9",
1188 testNetwork(done, params);
1190 it('Allows selection
of Zclassic
', function(done) {
1192 selectText: "ZCL - Zclassic",
1193 firstAddress: "t1TBMxTvVJRybUbMLGWq8H4A8F4VUL7czEc",
1195 testNetwork(done, params);
1197 it('Allows selection
of Zencash
', function(done) {
1199 selectText: "ZEN - Zencash",
1200 firstAddress: "znWh9XASyW2dZq5tck84wFjiwuqVysi7q3p",
1202 testNetwork(done, params);
1204 it('Allows selection
of Energi
', function(done) {
1206 selectText: "NRG - Energi",
1207 firstAddress: "EejRy4t4nidzhGGzkJUgFP3z4HYBjhTsRt",
1209 testNetwork(done, params);
1213 // BIP39 seed is set from phrase
1214 it('Sets the bip39 seed
from the prhase
', function(done) {
1215 driver.findElement(By.css('.phrase
'))
1216 .sendKeys('abandon abandon ability
');
1217 driver.sleep(generateDelay).then(function() {
1218 driver.findElement(By.css('.seed
'))
1219 .getAttribute("value")
1220 .then(function(seed) {
1221 expect(seed).toBe("20da140d3dd1df8713cefcc4d54ce0e445b4151027a1ab567b832f6da5fcc5afc1c3a3f199ab78b8e0ab4652efd7f414ac2c9a3b81bceb879a70f377aa0a58f3");
1227 // BIP32 root key is set from phrase
1228 it('Sets the bip39 root key
from the prhase
', function(done) {
1229 driver.findElement(By.css('.phrase
'))
1230 .sendKeys('abandon abandon ability
');
1231 driver.sleep(generateDelay).then(function() {
1232 driver.findElement(By.css('.root
-key
'))
1233 .getAttribute("value")
1234 .then(function(seed) {
1235 expect(seed).toBe("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
1241 // Tabs show correct addresses when changed
1242 it('Shows the correct address when tab is changed
', function(done) {
1243 driver.findElement(By.css('.phrase
'))
1244 .sendKeys('abandon abandon ability
');
1245 driver.sleep(generateDelay).then(function() {
1246 driver.findElement(By.css('#bip32
-tab a
'))
1248 driver.sleep(generateDelay).then(function() {
1249 getFirstAddress(function(address) {
1250 expect(address).toBe("17uQ7s2izWPwBmEVFikTmZUjbBKWYdJchz");
1257 // BIP44 derivation path is shown
1258 it('Shows the derivation path
for bip44 tab
', function(done) {
1259 driver.findElement(By.css('.phrase
'))
1260 .sendKeys('abandon abandon ability
');
1261 driver.sleep(generateDelay).then(function() {
1262 driver.findElement(By.css('#bip44
.path
'))
1263 .getAttribute("value")
1264 .then(function(path) {
1265 expect(path).toBe("m/44'/0'/0'/0");
1271 // BIP44 extended private key is shown
1272 it('Shows the extended private key for bip44 tab', function(done) {
1273 driver.findElement(By.css('.phrase'))
1274 .sendKeys('abandon abandon ability');
1275 driver.sleep(generateDelay).then(function() {
1276 driver.findElement(By.css('.extended-priv-key'))
1277 .getAttribute("value
")
1278 .then(function(path) {
1279 expect(path).toBe("xprvA2DxxvPZcyRvYgZMGS53nadR32mVDeCyqQYyFhrCVbJNjPoxMeVf7QT5g7mQASbTf9Kp4cryvcXnu2qurjWKcrdsr91jXymdCDNxKgLFKJG
");
1285 // BIP44 extended public key is shown
1286 it('Shows the extended public key for bip44 tab', function(done) {
1287 driver.findElement(By.css('.phrase'))
1288 .sendKeys('abandon abandon ability');
1289 driver.sleep(generateDelay).then(function() {
1290 driver.findElement(By.css('.extended-pub-key'))
1291 .getAttribute("value
")
1292 .then(function(path) {
1293 expect(path).toBe("xpub6FDKNRvTTLzDmAdpNTc49ia9b4byd6vqCdUa46Fp3vqMcC96uBoufCmZXQLiN5AK3iSCJMhf9gT2sxkpyaPepRuA7W3MujV5tGmF5VfbueM
");
1299 // BIP44 account field changes address list
1300 it('Changes the address list if bip44 account is changed', function(done) {
1301 driver.findElement(By.css('#bip44 .account'))
1303 driver.findElement(By.css('.phrase'))
1304 .sendKeys('abandon abandon ability');
1305 driver.sleep(generateDelay).then(function() {
1306 getFirstAddress(function(address) {
1307 expect(address).toBe("1Nq2Wmu726XHCuGhctEtGmhxo3wzk5wZ1H
");
1313 // BIP44 change field changes address list
1314 it('Changes the address list if bip44 change is changed', function(done) {
1315 driver.findElement(By.css('#bip44 .change'))
1317 driver.findElement(By.css('.phrase'))
1318 .sendKeys('abandon abandon ability');
1319 driver.sleep(generateDelay).then(function() {
1320 getFirstAddress(function(address) {
1321 expect(address).toBe("1KAGfWgqfVbSSXY56fNQ7YnhyKuoskHtYo
");
1327 // BIP32 derivation path can be set
1328 it('Can use a custom bip32 derivation path', function(done) {
1329 driver.findElement(By.css('#bip32-tab a'))
1331 driver.findElement(By.css('#bip32 .path'))
1333 driver.findElement(By.css('#bip32 .path'))
1335 driver.findElement(By.css('.phrase'))
1336 .sendKeys('abandon abandon ability');
1337 driver.sleep(generateDelay).then(function() {
1338 getFirstAddress(function(address) {
1339 expect(address).toBe("16pYQQdLD1hH4hwTGLXBaZ9Teboi1AGL8L
");
1345 // BIP32 can use hardened derivation paths
1346 it('Can use a hardened derivation paths', function(done) {
1347 driver.findElement(By.css('#bip32-tab a'))
1349 driver.findElement(By.css('#bip32 .path'))
1351 driver.findElement(By.css('#bip32 .path'))
1353 driver.findElement(By.css('.phrase
'))
1354 .sendKeys('abandon abandon ability
');
1355 driver.sleep(generateDelay).then(function() {
1356 getFirstAddress(function(address) {
1357 expect(address).toBe("14aXZeprXAE3UUKQc4ihvwBvww2LuEoHo4");
1363 // BIP32 extended private key is shown
1364 it('Shows the BIP32 extended
private key
', function(done) {
1365 driver.findElement(By.css('#bip32
-tab a
'))
1367 driver.findElement(By.css('.phrase
'))
1368 .sendKeys('abandon abandon ability
');
1369 driver.sleep(generateDelay).then(function() {
1370 driver.findElement(By.css('.extended
-priv
-key
'))
1371 .getAttribute("value")
1372 .then(function(privKey) {
1373 expect(privKey).toBe("xprv9va99uTVE5aLiutUVLTyfxfe8v8aaXjSQ1XxZbK6SezYVuikA9MnjQVTA8rQHpNA5LKvyQBpLiHbBQiiccKiBDs7eRmBogsvq3THFeLHYbe");
1379 // BIP32 extended public key is shown
1380 it('Shows the BIP32 extended
public key
', function(done) {
1381 driver.findElement(By.css('#bip32
-tab a
'))
1383 driver.findElement(By.css('.phrase
'))
1384 .sendKeys('abandon abandon ability
');
1385 driver.sleep(generateDelay).then(function() {
1386 driver.findElement(By.css('.extended
-pub
-key
'))
1387 .getAttribute("value")
1388 .then(function(pubKey) {
1389 expect(pubKey).toBe("xpub69ZVZQzP4T8dwPxwbMzz36cNgwy4yzTHmETZMyihzzXXNi3thgg3HCow1RtY252wdw5rS8369xKnraN5Q93y3FkFfJp2XEHWUrkyXsjS93P");
1395 // Derivation path is shown in table
1396 it('Shows the derivation path
in the table
', function(done) {
1397 driver.findElement(By.css('.phrase
'))
1398 .sendKeys('abandon abandon ability
');
1399 driver.sleep(generateDelay).then(function() {
1400 getFirstPath(function(path) {
1401 expect(path).toBe("m/44'/0'/0'/0/0");
1407 // Derivation path for address can be hardened
1408 it('Can derive hardened addresses', function(done) {
1409 driver.findElement(By.css('#bip32-tab a'))
1411 driver.executeScript(function() {
1412 $(".hardened
-addresses
").prop("checked
", true);
1414 driver.findElement(By.css('.phrase'))
1415 .sendKeys('abandon abandon ability');
1416 driver.sleep(generateDelay).then(function() {
1417 getFirstAddress(function(address) {
1418 expect(address).toBe("18exLzUv7kfpiXRzmCjFDoC9qwNLFyvwyd
");
1424 // Derivation path visibility can be toggled
1425 it('Can toggle visibility of the derivation path column', function(done) {
1426 driver.findElement(By.css('.phrase'))
1427 .sendKeys('abandon abandon ability');
1428 driver.sleep(generateDelay).then(function() {
1429 driver.findElement(By.css('.index-toggle'))
1431 testColumnValuesAreInvisible(done, "index
");
1436 it('Shows the address in the table', function(done) {
1437 driver.findElement(By.css('.phrase'))
1438 .sendKeys('abandon abandon ability');
1439 driver.sleep(generateDelay).then(function() {
1440 getFirstAddress(function(address) {
1441 expect(address).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug
");
1447 // Addresses are shown in order of derivation path
1448 it('Shows the address in order of derivation path', function(done) {
1449 driver.findElement(By.css('.phrase'))
1450 .sendKeys('abandon abandon ability');
1451 driver.sleep(generateDelay).then(function() {
1452 testRowsAreInCorrectOrder(done);
1456 // Address visibility can be toggled
1457 it('Can toggle visibility of the address column', function(done) {
1458 driver.findElement(By.css('.phrase'))
1459 .sendKeys('abandon abandon ability');
1460 driver.sleep(generateDelay).then(function() {
1461 driver.findElement(By.css('.address-toggle'))
1463 testColumnValuesAreInvisible(done, "address
");
1467 // Public key is shown in table
1468 it('Shows the public key in the table', function(done) {
1469 driver.findElement(By.css('.phrase'))
1470 .sendKeys('abandon abandon ability');
1471 driver.sleep(generateDelay).then(function() {
1472 driver.findElements(By.css('.pubkey'))
1473 .then(function(els) {
1475 .then(function(pubkey) {
1476 expect(pubkey).toBe("033f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3
");
1483 // Public key visibility can be toggled
1484 it('Can toggle visibility of the public key column', function(done) {
1485 driver.findElement(By.css('.phrase'))
1486 .sendKeys('abandon abandon ability');
1487 driver.sleep(generateDelay).then(function() {
1488 driver.findElement(By.css('.public-key-toggle'))
1490 testColumnValuesAreInvisible(done, "pubkey
");
1494 // Private key is shown in table
1495 it('Shows the private key in the table', function(done) {
1496 driver.findElement(By.css('.phrase'))
1497 .sendKeys('abandon abandon ability');
1498 driver.sleep(generateDelay).then(function() {
1499 driver.findElements(By.css('.privkey'))
1500 .then(function(els) {
1502 .then(function(pubkey) {
1503 expect(pubkey).toBe("L26cVSpWFkJ6aQkPkKmTzLqTdLJ923e6CzrVh9cmx21QHsoUmrEE
");
1510 // Private key visibility can be toggled
1511 it('Can toggle visibility of the private key column', function(done) {
1512 driver.findElement(By.css('.phrase'))
1513 .sendKeys('abandon abandon ability');
1514 driver.sleep(generateDelay).then(function() {
1515 driver.findElement(By.css('.private-key-toggle'))
1517 testColumnValuesAreInvisible(done, "privkey
");
1521 // More addresses can be generated
1522 it('Can generate more rows in the table', function(done) {
1523 driver.findElement(By.css('.phrase'))
1524 .sendKeys('abandon abandon ability');
1525 driver.sleep(generateDelay).then(function() {
1526 driver.findElement(By.css('.more'))
1528 driver.sleep(generateDelay).then(function() {
1529 driver.findElements(By.css('.address'))
1530 .then(function(els) {
1531 expect(els.length).toBe(40);
1538 // A custom number of additional addresses can be generated
1539 it('Can generate more rows in the table', function(done) {
1540 driver.findElement(By.css('.phrase'))
1541 .sendKeys('abandon abandon ability');
1542 driver.sleep(generateDelay).then(function() {
1543 driver.findElement(By.css('.rows-to-add'))
1545 driver.findElement(By.css('.rows-to-add'))
1547 driver.findElement(By.css('.more'))
1549 driver.sleep(generateDelay).then(function() {
1550 driver.findElements(By.css('.address'))
1551 .then(function(els) {
1552 expect(els.length).toBe(21);
1559 // Additional addresses are shown in order of derivation path
1560 it('Shows additional addresses in order of derivation path', function(done) {
1561 driver.findElement(By.css('.phrase'))
1562 .sendKeys('abandon abandon ability');
1563 driver.sleep(generateDelay).then(function() {
1564 driver.findElement(By.css('.more'))
1566 driver.sleep(generateDelay).then(function() {
1567 testRowsAreInCorrectOrder(done);
1572 // BIP32 root key can be set by the user
1573 it('Allows the user to set the BIP32 root key', function(done) {
1574 driver.findElement(By.css('.root-key'))
1575 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1576 driver.sleep(generateDelay).then(function() {
1577 getFirstAddress(function(address) {
1578 expect(address).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug
");
1584 // Setting BIP32 root key clears the existing phrase, passphrase and seed
1585 // TODO this doesn't work in selenium with chrome
1586 it('Confirms the existing phrase should be cleared', function(done) {
1587 if (browser == "chrome
") {
1588 pending("Selenium
+ Chrome headless bug
for alert
, see
https://stackoverflow.com/q/45242264");
1590 driver
.findElement(By
.css('.phrase'))
1591 .sendKeys('A non-blank but invalid value');
1592 driver
.findElement(By
.css('.root-key'))
1593 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1594 driver
.switchTo().alert().accept();
1595 driver
.findElement(By
.css('.phrase'))
1596 .getAttribute("value").then(function(value
) {
1597 expect(value
).toBe("");
1602 // Clearing of phrase, passphrase and seed can be cancelled by user
1603 // TODO this doesn't work in selenium with chrome
1604 it('Allows the clearing of the phrase to be cancelled', function(done
) {
1605 if (browser
== "chrome") {
1606 pending("Selenium + Chrome headless bug for alert, see https://stackoverflow.com/q/45242264");
1608 driver
.findElement(By
.css('.phrase'))
1609 .sendKeys('abandon abandon ability');
1610 driver
.sleep(generateDelay
).then(function() {
1611 driver
.findElement(By
.css('.root-key'))
1613 driver
.findElement(By
.css('.root-key'))
1615 driver
.switchTo().alert().dismiss();
1616 driver
.findElement(By
.css('.phrase'))
1617 .getAttribute("value").then(function(value
) {
1618 expect(value
).toBe("abandon abandon ability");
1624 // Custom BIP32 root key is used when changing the derivation path
1625 it('Can set derivation path for root key instead of phrase', function(done
) {
1626 driver
.findElement(By
.css('#bip44 .account'))
1628 driver
.findElement(By
.css('.root-key'))
1629 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1630 driver
.sleep(generateDelay
).then(function() {
1631 getFirstAddress(function(address
) {
1632 expect(address
).toBe("1Nq2Wmu726XHCuGhctEtGmhxo3wzk5wZ1H");
1638 // Incorrect mnemonic shows error
1639 it('Shows an error for incorrect mnemonic', function(done
) {
1640 driver
.findElement(By
.css('.phrase'))
1641 .sendKeys('abandon abandon abandon');
1642 driver
.sleep(feedbackDelay
).then(function() {
1643 driver
.findElement(By
.css('.feedback'))
1645 .then(function(feedback
) {
1646 expect(feedback
).toBe("Invalid mnemonic");
1652 // Incorrect word shows suggested replacement
1653 it('Shows word suggestion for incorrect word', function(done
) {
1654 driver
.findElement(By
.css('.phrase'))
1655 .sendKeys('abandon abandon abiliti');
1656 driver
.sleep(feedbackDelay
).then(function() {
1657 driver
.findElement(By
.css('.feedback'))
1659 .then(function(feedback
) {
1660 var msg
= "abiliti not in wordlist, did you mean ability?";
1661 expect(feedback
).toBe(msg
);
1667 // Github pull request 48
1668 // First four letters of word shows that word, not closest
1669 // since first four letters gives unique word in BIP39 wordlist
1670 // eg ille should show illegal, not idle
1671 it('Shows word suggestion based on first four chars', function(done
) {
1672 driver
.findElement(By
.css('.phrase'))
1674 driver
.sleep(feedbackDelay
).then(function() {
1675 driver
.findElement(By
.css('.feedback'))
1677 .then(function(feedback
) {
1678 var msg
= "ille not in wordlist, did you mean illegal?";
1679 expect(feedback
).toBe(msg
);
1685 // Incorrect BIP32 root key shows error
1686 it('Shows error for incorrect root key', function(done
) {
1687 driver
.findElement(By
.css('.root-key'))
1688 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpj');
1689 driver
.sleep(feedbackDelay
).then(function() {
1690 driver
.findElement(By
.css('.feedback'))
1692 .then(function(feedback
) {
1693 var msg
= "Invalid root key";
1694 expect(feedback
).toBe(msg
);
1700 // Derivation path not starting with m shows error
1701 it('Shows error for derivation path not starting with m', function(done
) {
1702 driver
.findElement(By
.css('#bip32-tab a'))
1704 driver
.findElement(By
.css('#bip32 .path'))
1706 driver
.findElement(By
.css('#bip32 .path'))
1708 driver
.findElement(By
.css('.phrase'))
1709 .sendKeys('abandon abandon ability');
1710 driver
.sleep(feedbackDelay
).then(function() {
1711 driver
.findElement(By
.css('.feedback'))
1713 .then(function(feedback
) {
1714 var msg
= "First character must be 'm'";
1715 expect(feedback
).toBe(msg
);
1721 // Derivation path containing invalid characters shows useful error
1722 it('Shows error for derivation path not starting with m', function(done
) {
1723 driver
.findElement(By
.css('#bip32-tab a'))
1725 driver
.findElement(By
.css('#bip32 .path'))
1727 driver
.findElement(By
.css('#bip32 .path'))
1728 .sendKeys('m/1/0wrong1/1');
1729 driver
.findElement(By
.css('.phrase'))
1730 .sendKeys('abandon abandon ability');
1731 driver
.sleep(feedbackDelay
).then(function() {
1732 driver
.findElement(By
.css('.feedback'))
1734 .then(function(feedback
) {
1735 var msg
= "Invalid characters 0wrong1 found at depth 2";
1736 expect(feedback
).toBe(msg
);
1742 // Github Issue 11: Default word length is 15
1743 // https://github.com/iancoleman/bip39/issues/11
1744 it('Sets the default word length to 15', function(done
) {
1745 driver
.findElement(By
.css('.strength'))
1746 .getAttribute("value")
1747 .then(function(strength
) {
1748 expect(strength
).toBe("15");
1753 // Github Issue 12: Generate more rows with private keys hidden
1754 // https://github.com/iancoleman/bip39/issues/12
1755 it('Sets the correct hidden column state on new rows', function(done
) {
1756 driver
.findElement(By
.css('.phrase'))
1757 .sendKeys("abandon abandon ability");
1758 driver
.sleep(generateDelay
).then(function() {
1759 driver
.findElement(By
.css('.private-key-toggle'))
1761 driver
.findElement(By
.css('.more'))
1763 driver
.sleep(generateDelay
).then(function() {
1764 driver
.findElements(By
.css('.privkey'))
1765 .then(function(els
) {
1766 expect(els
.length
).toBe(40);
1768 testColumnValuesAreInvisible(done
, "privkey");
1773 // Github Issue 19: Mnemonic is not sensitive to whitespace
1774 // https://github.com/iancoleman/bip39/issues/19
1775 it('Ignores excess whitespace in the mnemonic', function(done
) {
1776 var doublespace
= " ";
1777 var mnemonic
= "urge cat" + doublespace
+ "bid";
1778 driver
.findElement(By
.css('.phrase'))
1779 .sendKeys(mnemonic
);
1780 driver
.sleep(generateDelay
).then(function() {
1781 driver
.findElement(By
.css('.root-key'))
1782 .getAttribute("value")
1783 .then(function(seed
) {
1784 expect(seed
).toBe("xprv9s21ZrQH143K3isaZsWbKVoTtbvd34Y1ZGRugGdMeBGbM3AgBVzTH159mj1cbbtYSJtQr65w6L5xy5L9SFC7c9VJZWHxgAzpj4mun5LhrbC");
1790 // Github Issue 23: Part 1: Use correct derivation path when changing tabs
1791 // https://github.com/iancoleman/bip39/issues/23
1792 it('Uses the correct derivation path when changing tabs', function(done
) {
1793 // 1) and 2) set the phrase
1794 driver
.findElement(By
.css('.phrase'))
1795 .sendKeys("abandon abandon ability");
1796 driver
.sleep(generateDelay
).then(function() {
1797 // 3) select bip32 tab
1798 driver
.findElement(By
.css('#bip32-tab a'))
1800 driver
.sleep(generateDelay
).then(function() {
1801 // 4) switch from bitcoin to litecoin
1802 selectNetwork("LTC - Litecoin");
1803 driver
.sleep(generateDelay
).then(function() {
1804 // 5) Check address is displayed correctly
1805 getFirstAddress(function(address
) {
1806 expect(address
).toBe("LS8MP5LZ5AdzSZveRrjm3aYVoPgnfFh5T5");
1807 // 5) Check derivation path is displayed correctly
1808 getFirstPath(function(path
) {
1809 expect(path
).toBe("m/0/0");
1818 // Github Issue 23 Part 2: Coin selection in derivation path
1819 // https://github.com/iancoleman/bip39/issues/23#issuecomment-238011920
1820 it('Uses the correct derivation path when changing coins', function(done
) {
1822 driver
.findElement(By
.css('.phrase'))
1823 .sendKeys("abandon abandon ability");
1824 driver
.sleep(generateDelay
).then(function() {
1825 // switch from bitcoin to clam
1826 selectNetwork("CLAM - Clams");
1827 driver
.sleep(generateDelay
).then(function() {
1828 // check derivation path is displayed correctly
1829 getFirstPath(function(path
) {
1830 expect(path
).toBe("m/44'/23'/0'/0/0");
1837 // Github Issue 26: When using a Root key derrived altcoins are incorrect
1838 // https://github.com/iancoleman/bip39/issues/26
1839 it('Uses the correct derivation for altcoins with root keys', function(done
) {
1840 // 1) 2) and 3) set the root key
1841 driver
.findElement(By
.css('.root-key'))
1842 .sendKeys("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
1843 driver
.sleep(generateDelay
).then(function() {
1844 // 4) switch from bitcoin to viacoin
1845 selectNetwork("VIA - Viacoin");
1846 driver
.sleep(generateDelay
).then(function() {
1847 // 5) ensure the derived address is correct
1848 getFirstAddress(function(address
) {
1849 expect(address
).toBe("Vq9Eq4N5SQnjqZvxtxzo7hZPW5XnyJsmXT");
1856 // Selecting a language with no existing phrase should generate a phrase in
1858 it('Generate a random phrase when language is selected and no current phrase', function(done
) {
1859 driver
.findElement(By
.css("a[href='#japanese']"))
1861 driver
.sleep(generateDelay
).then(function() {
1862 driver
.findElement(By
.css(".phrase"))
1863 .getAttribute("value").then(function(phrase
) {
1864 expect(phrase
.search(/[a-z]/)).toBe(-1);
1865 expect(phrase
.length
).toBeGreaterThan(0);
1871 // Selecting a language with existing phrase should update the phrase to use
1873 it('Updates existing phrases when the language is changed', function(done
) {
1874 driver
.findElement(By
.css(".phrase"))
1875 .sendKeys("abandon abandon ability");
1876 driver
.sleep(generateDelay
).then(function() {
1877 driver
.findElement(By
.css("a[href='#italian']"))
1879 driver
.sleep(generateDelay
).then(function() {
1880 driver
.findElement(By
.css(".phrase"))
1881 .getAttribute("value").then(function(phrase
) {
1882 // Check only the language changes, not the phrase
1883 expect(phrase
).toBe("abaco abaco abbaglio");
1884 getFirstAddress(function(address
) {
1885 // Check the address is correct
1886 expect(address
).toBe("1Dz5TgDhdki9spa6xbPFbBqv5sjMrx3xgV");
1894 // Suggested replacement for erroneous word in non-English language
1895 it('Shows word suggestion for incorrect word in non-English language', function(done
) {
1896 driver
.findElement(By
.css('.phrase'))
1897 .sendKeys('abaco abaco zbbaglio');
1898 driver
.sleep(feedbackDelay
).then(function() {
1899 driver
.findElement(By
.css('.feedback'))
1901 .then(function(feedback
) {
1902 var msg
= "zbbaglio not in wordlist, did you mean abbaglio?";
1903 expect(feedback
).toBe(msg
);
1909 // Japanese word does not break across lines.
1911 // https://github.com/bitcoin/bips/blob/master/bip-0039/bip-0039-wordlists.md#japanese
1912 it('Does not break Japanese words across lines', function(done
) {
1913 driver
.findElement(By
.css('.phrase'))
1914 .getCssValue("word-break")
1915 .then(function(value
) {
1916 expect(value
).toBe("keep-all");
1921 // Language can be specified at page load using hash value in url
1922 it('Can set the language from the url hash', function(done
) {
1923 driver
.get(url
+ "#japanese").then(function() {
1924 driver
.findElement(By
.css('.generate')).click();
1925 driver
.sleep(generateDelay
).then(function() {
1926 driver
.findElement(By
.css(".phrase"))
1927 .getAttribute("value").then(function(phrase
) {
1928 expect(phrase
.search(/[a-z]/)).toBe(-1);
1929 expect(phrase
.length
).toBeGreaterThan(0);
1936 // Entropy can be entered by the user
1937 it('Allows entropy to be entered', function(done
) {
1938 driver
.findElement(By
.css('.use-entropy'))
1940 driver
.findElement(By
.css('.entropy'))
1941 .sendKeys('00000000 00000000 00000000 00000000');
1942 driver
.sleep(generateDelay
).then(function() {
1943 driver
.findElement(By
.css(".phrase"))
1944 .getAttribute("value").then(function(phrase
) {
1945 expect(phrase
).toBe("abandon abandon ability");
1946 getFirstAddress(function(address
) {
1947 expect(address
).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug");
1954 // A warning about entropy is shown to the user, with additional information
1955 it('Shows a warning about using entropy', function(done
) {
1956 driver
.findElement(By
.css('.use-entropy'))
1958 driver
.findElement(By
.css('.entropy-container'))
1960 .then(function(containerText
) {
1961 var warning
= "mnemonic may be insecure";
1962 expect(containerText
).toContain(warning
);
1963 driver
.findElement(By
.css('#entropy-notes'))
1964 .findElement(By
.xpath("parent::*"))
1966 .then(function(notesText
) {
1967 var detail
= "flipping a fair coin, rolling a fair dice, noise measurements etc";
1968 expect(notesText
).toContain(detail
);
1974 // The types of entropy available are described to the user
1975 it('Shows the types of entropy available', function(done
) {
1976 driver
.findElement(By
.css('.entropy'))
1977 .getAttribute("placeholder")
1978 .then(function(placeholderText
) {
1987 for (var i
=0; i
<options
.length
; i
++) {
1988 var option
= options
[i
];
1989 expect(placeholderText
).toContain(option
);
1995 // The actual entropy used is shown to the user
1996 it('Shows the actual entropy used', function(done
) {
1997 driver
.findElement(By
.css('.use-entropy'))
1999 driver
.findElement(By
.css('.entropy'))
2000 .sendKeys('Not A Very Good Entropy Source At All');
2001 driver
.sleep(generateDelay
).then(function() {
2002 driver
.findElement(By
.css('.entropy-container'))
2004 .then(function(text
) {
2005 expect(text
).toMatch(/Filtered Entropy
\s
+AedEceAA
/);
2011 // Binary entropy can be entered
2012 it('Allows binary entropy to be entered', function(done
) {
2013 testEntropyType(done
, "01", "binary");
2016 // Base 6 entropy can be entered
2017 it('Allows base 6 entropy to be entered', function(done
) {
2018 testEntropyType(done
, "012345", "base 6");
2021 // Base 6 dice entropy can be entered
2022 it('Allows base 6 dice entropy to be entered', function(done
) {
2023 testEntropyType(done
, "123456", "base 6 (dice)");
2026 // Base 10 entropy can be entered
2027 it('Allows base 10 entropy to be entered', function(done
) {
2028 testEntropyType(done
, "789", "base 10");
2031 // Hexadecimal entropy can be entered
2032 it('Allows hexadecimal entropy to be entered', function(done
) {
2033 testEntropyType(done
, "abcdef", "hexadecimal");
2036 // Dice entropy value is shown as the converted base 6 value
2037 // ie 123456 is converted to 123450
2038 it('Shows dice entropy as base 6', function(done
) {
2039 driver
.findElement(By
.css('.use-entropy'))
2041 driver
.findElement(By
.css('.entropy'))
2042 .sendKeys("123456");
2043 driver
.sleep(generateDelay
).then(function() {
2044 driver
.findElement(By
.css('.entropy-container'))
2046 .then(function(text
) {
2047 expect(text
).toMatch(/Filtered Entropy
\s
+123450/);
2053 // The number of bits of entropy accumulated is shown
2054 it("Shows the number of bits of entropy for 20 bits of binary", function(done
) {
2055 testEntropyBits(done
, "0000 0000 0000 0000 0000", "20");
2057 it("Shows the number of bits of entropy for 1 bit of binary", function(done
) {
2058 testEntropyBits(done
, "0", "1");
2060 it("Shows the number of bits of entropy for 4 bits of binary", function(done
) {
2061 testEntropyBits(done
, "0000", "4");
2063 it("Shows the number of bits of entropy for 1 character of base 6 (dice)", function(done
) {
2064 // 6 in card is 0 in base 6, 0 in base 6 is 2.6 bits (rounded down to 2 bits)
2065 testEntropyBits(done
, "6", "2");
2067 it("Shows the number of bits of entropy for 1 character of base 10 with 3 bits", function(done
) {
2068 // 7 in base 10 is 111 in base 2, no leading zeros
2069 testEntropyBits(done
, "7", "3");
2071 it("Shows the number of bits of entropy for 1 character of base 10 with 4 bis", function(done
) {
2072 testEntropyBits(done
, "8", "4");
2074 it("Shows the number of bits of entropy for 1 character of hex", function(done
) {
2075 testEntropyBits(done
, "F", "4");
2077 it("Shows the number of bits of entropy for 2 characters of base 10", function(done
) {
2078 testEntropyBits(done
, "29", "6");
2080 it("Shows the number of bits of entropy for 2 characters of hex", function(done
) {
2081 testEntropyBits(done
, "0A", "8");
2083 it("Shows the number of bits of entropy for 2 characters of hex with 3 leading zeros", function(done
) {
2084 // hex is always multiple of 4 bits of entropy
2085 testEntropyBits(done
, "1A", "8");
2087 it("Shows the number of bits of entropy for 2 characters of hex with 2 leading zeros", function(done
) {
2088 testEntropyBits(done
, "2A", "8");
2090 it("Shows the number of bits of entropy for 2 characters of hex with 1 leading zero", function(done
) {
2091 testEntropyBits(done
, "4A", "8");
2093 it("Shows the number of bits of entropy for 2 characters of hex with no leading zeros", function(done
) {
2094 testEntropyBits(done
, "8A", "8");
2096 it("Shows the number of bits of entropy for 2 characters of hex starting with F", function(done
) {
2097 testEntropyBits(done
, "FA", "8");
2099 it("Shows the number of bits of entropy for 4 characters of hex with leading zeros", function(done
) {
2100 testEntropyBits(done
, "000A", "16");
2102 it("Shows the number of bits of entropy for 4 characters of base 6", function(done
) {
2103 testEntropyBits(done
, "5555", "11");
2105 it("Shows the number of bits of entropy for 4 characters of base 6 dice", function(done
) {
2106 // uses dice, so entropy is actually 0000 in base 6, which is 4 lots of
2107 // 2.58 bits, which is 10.32 bits (rounded down to 10 bits)
2108 testEntropyBits(done
, "6666", "10");
2110 it("Shows the number of bits of entropy for 4 charactes of base 10", function(done
) {
2111 // Uses base 10, which is 4 lots of 3.32 bits, which is 13.3 bits (rounded
2113 testEntropyBits(done
, "2227", "13");
2115 it("Shows the number of bits of entropy for 4 characters of hex with 2 leading zeros", function(done
) {
2116 testEntropyBits(done
, "222F", "16");
2118 it("Shows the number of bits of entropy for 4 characters of hex starting with F", function(done
) {
2119 testEntropyBits(done
, "FFFF", "16");
2121 it("Shows the number of bits of entropy for 10 characters of base 10", function(done
) {
2122 // 10 events at 3.32 bits per event
2123 testEntropyBits(done
, "0000101017", "33");
2125 it("Shows the number of bits of entropy for a full deck of cards", function(done
) {
2126 // cards are not replaced, so a full deck is not 52^52 entropy which is 296
2127 // bits, it's 52!, which is 225 bits
2128 testEntropyBits(done
, "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks", "225");
2131 it("Shows details about the entered entropy", function(done
) {
2132 testEntropyFeedback(done
,
2136 type: "hexadecimal",
2140 strength: "less than a second",
2144 it("Shows details about the entered entropy", function(done
) {
2145 testEntropyFeedback(done
,
2147 entropy: "AAAAAAAA",
2148 filtered: "AAAAAAAA",
2149 type: "hexadecimal",
2153 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2157 it("Shows details about the entered entropy", function(done
) {
2158 testEntropyFeedback(done
,
2160 entropy: "AAAAAAAA B",
2161 filtered: "AAAAAAAAB",
2162 type: "hexadecimal",
2166 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2170 it("Shows details about the entered entropy", function(done
) {
2171 testEntropyFeedback(done
,
2173 entropy: "AAAAAAAA BBBBBBBB",
2174 filtered: "AAAAAAAABBBBBBBB",
2175 type: "hexadecimal",
2179 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2183 it("Shows details about the entered entropy", function(done
) {
2184 testEntropyFeedback(done
,
2186 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC",
2187 filtered: "AAAAAAAABBBBBBBBCCCCCCCC",
2188 type: "hexadecimal",
2192 strength: "less than a second",
2196 it("Shows details about the entered entropy", function(done
) {
2197 testEntropyFeedback(done
,
2199 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDD",
2200 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDD",
2201 type: "hexadecimal",
2205 strength: "2 minutes",
2209 it("Shows details about the entered entropy", function(done
) {
2210 testEntropyFeedback(done
,
2212 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA",
2213 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDA",
2214 type: "hexadecimal",
2222 it("Shows details about the entered entropy", function(done
) {
2223 testEntropyFeedback(done
,
2225 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA EEEEEEEE",
2226 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDAEEEEEEEE",
2227 type: "hexadecimal",
2231 strength: "3 years",
2235 it("Shows details about the entered entropy", function(done
) {
2236 testEntropyFeedback(done
,
2238 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA EEEEEEEE FFFFFFFF",
2239 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDAEEEEEEEEFFFFFFFF",
2240 type: "hexadecimal",
2244 strength: "centuries",
2248 it("Shows details about the entered entropy", function(done
) {
2249 testEntropyFeedback(done
,
2256 strength: "less than a second",
2260 it("Shows details about the entered entropy", function(done
) {
2261 testEntropyFeedback(done
,
2263 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2264 type: "card (full deck)",
2268 strength: "centuries",
2272 it("Shows details about the entered entropy", function(done
) {
2273 testEntropyFeedback(done
,
2275 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks3d",
2276 type: "card (full deck, 1 duplicate: 3d)",
2280 strength: "centuries",
2284 it("Shows details about the entered entropy", function(done
) {
2285 testEntropyFeedback(done
,
2287 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqs3d4d",
2288 type: "card (2 duplicates: 3d 4d, 1 missing: KS)",
2292 strength: "centuries",
2296 it("Shows details about the entered entropy", function(done
) {
2297 testEntropyFeedback(done
,
2299 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqs3d4d5d6d",
2300 type: "card (4 duplicates: 3d 4d 5d..., 1 missing: KS)",
2304 strength: "centuries",
2308 it("Shows details about the entered entropy", function(done
) {
2309 testEntropyFeedback(done
,
2310 // Next test was throwing uncaught error in zxcvbn
2311 // Also tests 451 bits, ie Math.log2(52!)*2 = 225.58 * 2
2313 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsksac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2314 type: "card (full deck, 52 duplicates: ac 2c 3c...)",
2318 strength: "centuries",
2322 it("Shows details about the entered entropy", function(done
) {
2323 testEntropyFeedback(done
,
2324 // Case insensitivity to duplicate cards
2327 type: "card (1 duplicate: AS)",
2331 strength: "less than a second",
2335 it("Shows details about the entered entropy", function(done
) {
2336 testEntropyFeedback(done
,
2339 type: "card (1 duplicate: as)",
2343 strength: "less than a second",
2347 it("Shows details about the entered entropy", function(done
) {
2348 testEntropyFeedback(done
,
2349 // Missing cards are detected
2351 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2352 type: "card (1 missing: 9C)",
2356 strength: "centuries",
2360 it("Shows details about the entered entropy", function(done
) {
2361 testEntropyFeedback(done
,
2363 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2364 type: "card (2 missing: 9C 5D)",
2368 strength: "centuries",
2372 it("Shows details about the entered entropy", function(done
) {
2373 testEntropyFeedback(done
,
2375 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d7d8d9dtdjd kdah2h3h 5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2376 type: "card (4 missing: 9C 5D QD...)",
2380 strength: "centuries",
2384 it("Shows details about the entered entropy", function(done
) {
2385 testEntropyFeedback(done
,
2386 // More than six missing cards does not show message
2388 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d 8d9d jd kdah2h3h 5h6h7h8h9hthjhqhkh 2s3s4s5s6s7s8s9stsjsqsks",
2393 strength: "centuries",
2397 it("Shows details about the entered entropy", function(done
) {
2398 testEntropyFeedback(done
,
2399 // Multiple decks of cards increases bits per event
2404 bitsPerEvent: "4.34",
2408 it("Shows details about the entered entropy", function(done
) {
2409 testEntropyFeedback(done
,
2414 bitsPerEvent: "4.80",
2418 it("Shows details about the entered entropy", function(done
) {
2419 testEntropyFeedback(done
,
2424 bitsPerEvent: "5.01",
2428 it("Shows details about the entered entropy", function(done
) {
2429 testEntropyFeedback(done
,
2431 entropy: "3d3d3d3d",
2434 bitsPerEvent: "5.14",
2438 it("Shows details about the entered entropy", function(done
) {
2439 testEntropyFeedback(done
,
2441 entropy: "3d3d3d3d3d",
2444 bitsPerEvent: "5.22",
2448 it("Shows details about the entered entropy", function(done
) {
2449 testEntropyFeedback(done
,
2451 entropy: "3d3d3d3d3d3d",
2454 bitsPerEvent: "5.28",
2458 it("Shows details about the entered entropy", function(done
) {
2459 testEntropyFeedback(done
,
2461 entropy: "3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d",
2464 bitsPerEvent: "5.59",
2465 strength: 'less than a second - Repeats like "abcabcabc" are only slightly harder to guess than "abc"',
2470 // Entropy is truncated from the left
2471 it('Truncates entropy from the left', function(done
) {
2472 // Truncate from left means 0000 is removed from the start
2473 // which gives mnemonic 'avocado zoo zone'
2474 // not 1111 removed from the end
2475 // which gives the mnemonic 'abstract zoo zoo'
2476 var entropy
= "00000000 00000000 00000000 00000000";
2477 entropy
+= "11111111 11111111 11111111 1111"; // Missing last byte
2478 driver
.findElement(By
.css('.use-entropy'))
2480 driver
.findElement(By
.css('.entropy'))
2482 driver
.sleep(generateDelay
).then(function() {
2483 driver
.findElement(By
.css(".phrase"))
2484 .getAttribute("value").then(function(phrase
) {
2485 expect(phrase
).toBe("avocado zoo zone");
2491 // Very large entropy results in very long mnemonics
2492 it('Converts very long entropy to very long mnemonics', function(done
) {
2494 for (var i
=0; i
<33; i
++) {
2495 entropy
+= "AAAAAAAA"; // 3 words * 33 iterations = 99 words
2497 driver
.findElement(By
.css('.use-entropy'))
2499 driver
.findElement(By
.css('.entropy'))
2501 driver
.sleep(generateDelay
).then(function() {
2502 driver
.findElement(By
.css(".phrase"))
2503 .getAttribute("value").then(function(phrase
) {
2504 var wordCount
= phrase
.split(/\s+/g).length
;
2505 expect(wordCount
).toBe(99);
2511 // Is compatible with bip32jp entropy
2512 // https://bip32jp.github.io/english/index.html
2514 // Is incompatible with:
2516 it('Is compatible with bip32jp.github.io', function(done
) {
2517 var entropy
= "543210543210543210543210543210543210543210543210543210543210543210543210543210543210543210543210543";
2518 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";
2519 driver
.findElement(By
.css('.use-entropy'))
2521 driver
.findElement(By
.css('.entropy'))
2523 driver
.sleep(generateDelay
).then(function() {
2524 driver
.findElement(By
.css(".phrase"))
2525 .getAttribute("value").then(function(phrase
) {
2526 expect(phrase
).toBe(expectedPhrase
);
2532 // Blank entropy does not generate mnemonic or addresses
2533 it('Does not generate mnemonic for blank entropy', function(done
) {
2534 driver
.findElement(By
.css('.use-entropy'))
2536 driver
.findElement(By
.css('.entropy'))
2538 // check there is no mnemonic
2539 driver
.sleep(generateDelay
).then(function() {
2540 driver
.findElement(By
.css(".phrase"))
2541 .getAttribute("value").then(function(phrase
) {
2542 expect(phrase
).toBe("");
2543 // check there is no mnemonic
2544 driver
.findElements(By
.css(".address"))
2545 .then(function(addresses
) {
2546 expect(addresses
.length
).toBe(0);
2547 // Check the feedback says 'blank entropy'
2548 driver
.findElement(By
.css(".feedback"))
2550 .then(function(feedbackText
) {
2551 expect(feedbackText
).toBe("Blank entropy");
2559 // Mnemonic length can be selected even for weak entropy
2560 it('Allows selection of mnemonic length even for weak entropy', function(done
) {
2561 driver
.findElement(By
.css('.use-entropy'))
2563 driver
.executeScript(function() {
2564 $(".mnemonic-length").val("18").trigger("change");
2566 driver
.findElement(By
.css('.entropy'))
2567 .sendKeys("012345");
2568 driver
.sleep(generateDelay
).then(function() {
2569 driver
.findElement(By
.css(".phrase"))
2570 .getAttribute("value").then(function(phrase
) {
2571 var wordCount
= phrase
.split(/\s+/g).length
;
2572 expect(wordCount
).toBe(18);
2579 // https://github.com/iancoleman/bip39/issues/33
2580 // Final cards should contribute entropy
2581 it('Uses as much entropy as possible for the mnemonic', function(done
) {
2582 driver
.findElement(By
.css('.use-entropy'))
2584 driver
.findElement(By
.css('.entropy'))
2585 .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");
2586 driver
.sleep(generateDelay
).then(function() {
2588 driver
.findElement(By
.css(".phrase"))
2589 .getAttribute("value").then(function(originalPhrase
) {
2590 // Set the last 12 cards to be AS
2591 driver
.findElement(By
.css('.entropy'))
2593 driver
.findElement(By
.css('.entropy'))
2594 .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");
2595 driver
.sleep(generateDelay
).then(function() {
2597 driver
.findElement(By
.css(".phrase"))
2598 .getAttribute("value").then(function(newPhrase
) {
2599 expect(originalPhrase
).not
.toEqual(newPhrase
);
2608 // https://github.com/iancoleman/bip39/issues/35
2610 // TODO this doesn't work in selenium with firefox
2611 // see https://stackoverflow.com/q/40360223
2612 it('Shows a qr code on hover for the phrase', function(done
) {
2613 if (browser
== "firefox") {
2614 pending("Selenium + Firefox bug for mouseMove, see https://stackoverflow.com/q/40360223");
2616 // generate a random mnemonic
2617 var generateEl
= driver
.findElement(By
.css('.generate'));
2619 // toggle qr to show (hidden by default)
2620 var phraseEl
= driver
.findElement(By
.css(".phrase"));
2622 var rootKeyEl
= driver
.findElement(By
.css(".root-key"));
2623 driver
.sleep(generateDelay
).then(function() {
2624 // hover over the root key
2625 driver
.actions().mouseMove(rootKeyEl
).perform().then(function() {
2626 // check the qr code shows
2627 driver
.executeScript(function() {
2628 return $(".qr-container").find("canvas").length
> 0;
2630 .then(function(qrShowing
) {
2631 expect(qrShowing
).toBe(true);
2632 // hover away from the phrase
2633 driver
.actions().mouseMove(generateEl
).perform().then(function() {;
2634 // check the qr code hides
2635 driver
.executeScript(function() {
2636 return $(".qr-container").find("canvas").length
== 0;
2638 .then(function(qrHidden
) {
2639 expect(qrHidden
).toBe(true);
2648 // BIP44 account extendend private key is shown
2649 // github issue 37 - compatibility with electrum
2650 it('Shows the bip44 account extended private key', function(done
) {
2651 driver
.findElement(By
.css(".phrase"))
2652 .sendKeys("abandon abandon ability");
2653 driver
.sleep(generateDelay
).then(function() {
2654 driver
.findElement(By
.css("#bip44 .account-xprv"))
2655 .getAttribute("value")
2656 .then(function(xprv
) {
2657 expect(xprv
).toBe("xprv9yzrnt4zWVJUr1k2VxSPy9ettKz5PpeDMgaVG7UKedhqnw1tDkxP2UyYNhuNSumk2sLE5ctwKZs9vwjsq3e1vo9egCK6CzP87H2cVYXpfwQ");
2663 // BIP44 account extendend public key is shown
2664 // github issue 37 - compatibility with electrum
2665 it('Shows the bip44 account extended public key', function(done
) {
2666 driver
.findElement(By
.css(".phrase"))
2667 .sendKeys("abandon abandon ability");
2668 driver
.sleep(generateDelay
).then(function() {
2669 driver
.findElement(By
.css("#bip44 .account-xpub"))
2670 .getAttribute("value")
2671 .then(function(xprv
) {
2672 expect(xprv
).toBe("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2679 // BIP32 root key can be set as an xpub
2680 it('Generates addresses from xpub as bip32 root key', function(done
) {
2681 driver
.findElement(By
.css('#bip32-tab a'))
2683 // set xpub for account 0 of bip44 for 'abandon abandon ability'
2684 driver
.findElement(By
.css("#root-key"))
2685 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2686 driver
.sleep(generateDelay
).then(function() {
2687 // check the addresses are generated
2688 getFirstAddress(function(address
) {
2689 expect(address
).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug");
2690 // check the xprv key is not set
2691 driver
.findElement(By
.css(".extended-priv-key"))
2692 .getAttribute("value")
2693 .then(function(xprv
) {
2694 expect(xprv
).toBe("NA");
2695 // check the private key is not set
2696 driver
.findElements(By
.css(".privkey"))
2697 .then(function(els
) {
2700 .then(function(privkey
) {
2701 expect(xprv
).toBe("NA");
2711 // xpub for bip32 root key will not work with hardened derivation paths
2712 it('Shows error for hardened derivation paths with xpub root key', function(done
) {
2713 // set xpub for account 0 of bip44 for 'abandon abandon ability'
2714 driver
.findElement(By
.css("#root-key"))
2715 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2716 driver
.sleep(feedbackDelay
).then(function() {
2717 // Check feedback is correct
2718 driver
.findElement(By
.css('.feedback'))
2720 .then(function(feedback
) {
2721 var msg
= "Hardened derivation path is invalid with xpub key";
2722 expect(feedback
).toBe(msg
);
2723 // Check no addresses are shown
2724 driver
.findElements(By
.css('.addresses tr'))
2725 .then(function(rows
) {
2726 expect(rows
.length
).toBe(0);
2734 // no root key shows feedback
2735 it('Shows feedback for no root key', function(done
) {
2736 // set xpub for account 0 of bip44 for 'abandon abandon ability'
2737 driver
.findElement(By
.css('#bip32-tab a'))
2739 driver
.sleep(feedbackDelay
).then(function() {
2740 // Check feedback is correct
2741 driver
.findElement(By
.css('.feedback'))
2743 .then(function(feedback
) {
2744 expect(feedback
).toBe("Invalid root key");
2751 // display error switching tabs while addresses are generating
2752 it('Can change details while old addresses are still being generated', function(done
) {
2753 // Set to generate 199 more addresses.
2754 // This will take a long time allowing a new set of addresses to be
2755 // generated midway through this lot.
2756 // The newly generated addresses should not include any from the old set.
2757 // Any more than 199 will show an alert which needs to be accepted.
2758 driver
.findElement(By
.css('.rows-to-add'))
2760 driver
.findElement(By
.css('.rows-to-add'))
2763 driver
.findElement(By
.css('.phrase'))
2764 .sendKeys("abandon abandon ability");
2765 driver
.sleep(generateDelay
).then(function() {
2766 // change tabs which should cancel the previous generating
2767 driver
.findElement(By
.css('.rows-to-add'))
2769 driver
.findElement(By
.css('.rows-to-add'))
2771 driver
.findElement(By
.css('#bip32-tab a'))
2773 driver
.sleep(generateDelay
).then(function() {
2774 driver
.findElements(By
.css('.index'))
2775 .then(function(els
) {
2776 // check the derivation paths have the right quantity
2777 expect(els
.length
).toBe(20);
2778 // check the derivation paths are in order
2779 testRowsAreInCorrectOrder(done
);
2783 }, generateDelay
+ 5000);
2786 // padding for binary should give length with multiple of 256
2787 // hashed entropy 1111 is length 252, so requires 4 leading zeros
2788 // prior to issue 49 it would only generate 2 leading zeros, ie missing 2
2789 it('Pads hashed entropy with leading zeros', function(done
) {
2790 driver
.findElement(By
.css('.use-entropy'))
2792 driver
.executeScript(function() {
2793 $(".mnemonic-length").val("15").trigger("change");
2795 driver
.findElement(By
.css('.entropy'))
2797 driver
.sleep(generateDelay
).then(function() {
2798 driver
.findElement(By
.css('.phrase'))
2799 .getAttribute("value")
2800 .then(function(phrase
) {
2801 expect(phrase
).toBe("avocado valid quantum cross link predict excuse edit street able flame large galaxy ginger nuclear");
2807 // Github pull request 55
2808 // https://github.com/iancoleman/bip39/pull/55
2810 it('Can set the derivation path on bip32 tab for bitcoincore', function(done
) {
2811 testClientSelect(done
, {
2813 bip32path: "m/0'/0'",
2814 useHardenedAddresses: "true",
2817 it('Can set the derivation path on bip32 tab for multibit', function(done
) {
2818 testClientSelect(done
, {
2820 bip32path: "m/0'/0",
2821 useHardenedAddresses: null,
2826 // https://github.com/iancoleman/bip39/issues/58
2827 // bip32 derivation is correct, does not drop leading zeros
2829 // https://medium.com/@alexberegszaszi/why-do-my-bip32-wallets-disagree-6f3254cc5846
2830 it('Retains leading zeros for bip32 derivation', function(done
) {
2831 driver
.findElement(By
.css(".phrase"))
2832 .sendKeys("fruit wave dwarf banana earth journey tattoo true farm silk olive fence");
2833 driver
.findElement(By
.css(".passphrase"))
2834 .sendKeys("banana");
2835 driver
.sleep(generateDelay
).then(function() {
2836 getFirstAddress(function(address
) {
2837 // Note that bitcore generates an incorrect address
2838 // 13EuKhffWkBE2KUwcbkbELZb1MpzbimJ3Y
2839 // see the medium.com link above for more details
2840 expect(address
).toBe("17rxURoF96VhmkcEGCj5LNQkmN9HVhWb7F");
2847 // Japanese mnemonics generate incorrect bip32 seed
2848 // BIP39 seed is set from phrase
2849 it('Generates correct seed for Japanese mnemonics', function(done
) {
2850 driver
.findElement(By
.css(".phrase"))
2851 .sendKeys("あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あおぞら");
2852 driver
.findElement(By
.css(".passphrase"))
2853 .sendKeys("メートルガバヴァぱばぐゞちぢ十人十色");
2854 driver
.sleep(generateDelay
).then(function() {
2855 driver
.findElement(By
.css(".seed"))
2856 .getAttribute("value")
2857 .then(function(seed
) {
2858 expect(seed
).toBe("a262d6fb6122ecf45be09c50492b31f92e9beb7d9a845987a02cefda57a15f9c467a17872029a9e92299b5cbdf306e3a0ee620245cbd508959b6cb7ca637bd55");
2864 // BIP49 official test vectors
2865 // https://github.com/bitcoin/bips/blob/master/bip-0049.mediawiki#test-vectors
2866 it('Generates BIP49 addresses matching the official test vectors', function(done
) {
2867 driver
.findElement(By
.css('#bip49-tab a'))
2869 selectNetwork("BTC - Bitcoin Testnet");
2870 driver
.findElement(By
.css(".phrase"))
2871 .sendKeys("abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about");
2872 driver
.sleep(generateDelay
).then(function() {
2873 getFirstAddress(function(address
) {
2874 expect(address
).toBe("2Mww8dCYPUpKHofjgcXcBCEGmniw9CoaiD2");
2880 // BIP49 derivation path is shown
2881 it('Shows the bip49 derivation path', function(done
) {
2882 driver
.findElement(By
.css('#bip49-tab a'))
2884 driver
.findElement(By
.css(".phrase"))
2885 .sendKeys("abandon abandon ability");
2886 driver
.sleep(generateDelay
).then(function() {
2887 driver
.findElement(By
.css('#bip49 .path'))
2888 .getAttribute("value")
2889 .then(function(path
) {
2890 expect(path
).toBe("m/49'/0'/0'/0");
2896 // BIP49 extended private key is shown
2897 it('Shows the bip49 extended private key', function(done
) {
2898 driver
.findElement(By
.css('#bip49-tab a'))
2900 driver
.findElement(By
.css(".phrase"))
2901 .sendKeys("abandon abandon ability");
2902 driver
.sleep(generateDelay
).then(function() {
2903 driver
.findElement(By
.css('.extended-priv-key'))
2904 .getAttribute("value")
2905 .then(function(xprv
) {
2906 expect(xprv
).toBe("yprvALYB4DYRG6CzzVgzQZwwqjAA2wjBGC3iEd7KYYScpoDdmf75qMRWZWxoFcRXBJjgEXdFqJ9vDRGRLJQsrL22Su5jMbNFeM9vetaGVqy9Qy2");
2912 // BIP49 extended public key is shown
2913 it('Shows the bip49 extended public key', function(done
) {
2914 driver
.findElement(By
.css('#bip49-tab a'))
2916 driver
.findElement(By
.css(".phrase"))
2917 .sendKeys("abandon abandon ability");
2918 driver
.sleep(generateDelay
).then(function() {
2919 driver
.findElement(By
.css('.extended-pub-key'))
2920 .getAttribute("value")
2921 .then(function(xprv
) {
2922 expect(xprv
).toBe("ypub6ZXXTj5K6TmJCymTWbUxCs6tayZffemZbr2vLvrEP8kceTSENtjm7KHH6thvAKxVar9fGe8rgsPEX369zURLZ68b4f7Vexz7RuXsjQ69YDt");
2928 // BIP49 account field changes address list
2929 it('Can set the bip49 account field', function(done
) {
2930 driver
.findElement(By
.css('#bip49-tab a'))
2932 driver
.findElement(By
.css('#bip49 .account'))
2934 driver
.findElement(By
.css('#bip49 .account'))
2936 driver
.findElement(By
.css(".phrase"))
2937 .sendKeys("abandon abandon ability");
2938 driver
.sleep(generateDelay
).then(function() {
2939 getFirstAddress(function(address
) {
2940 expect(address
).toBe("381wg1GGN4rP88rNC9v7QWsiww63yLVPsn");
2946 // BIP49 change field changes address list
2947 it('Can set the bip49 change field', function(done
) {
2948 driver
.findElement(By
.css('#bip49-tab a'))
2950 driver
.findElement(By
.css('#bip49 .change'))
2952 driver
.findElement(By
.css('#bip49 .change'))
2954 driver
.findElement(By
.css(".phrase"))
2955 .sendKeys("abandon abandon ability");
2956 driver
.sleep(generateDelay
).then(function() {
2957 getFirstAddress(function(address
) {
2958 expect(address
).toBe("3PEM7MiKed5konBoN66PQhK8r3hjGhy9dT");
2964 // BIP49 account extendend private key is shown
2965 it('Shows the bip49 account extended private key', function(done
) {
2966 driver
.findElement(By
.css('#bip49-tab a'))
2968 driver
.findElement(By
.css(".phrase"))
2969 .sendKeys("abandon abandon ability");
2970 driver
.sleep(generateDelay
).then(function() {
2971 driver
.findElement(By
.css('#bip49 .account-xprv'))
2972 .getAttribute("value")
2973 .then(function(xprv
) {
2974 expect(xprv
).toBe("yprvAHtB1M5Wp675aLzFy9TJYK2mSsLkg6mcBRh5DZTR7L4EnYSmYPaL63KFA4ycg1PngW5LfkmejxzosCs17TKZMpRFKc3z5SJar6QAKaFcaZL");
2980 // BIP49 account extendend public key is shown
2981 it('Shows the bip49 account extended public key', function(done
) {
2982 driver
.findElement(By
.css('#bip49-tab a'))
2984 driver
.findElement(By
.css(".phrase"))
2985 .sendKeys("abandon abandon ability");
2986 driver
.sleep(generateDelay
).then(function() {
2987 driver
.findElement(By
.css('#bip49 .account-xpub'))
2988 .getAttribute("value")
2989 .then(function(xprv
) {
2990 expect(xprv
).toBe("ypub6WsXQrcQeTfNnq4j5AzJuSyVzuBF5ZVTYecg1ws2ffbDfLmv5vtadqdj1NgR6C6gufMpMfJpHxvb6JEQKvETVNWCRanNedfJtnTchZiJtsL");
2996 // Test selecting coin where bip49 is unavailable (eg CLAM)
2997 it('Shows an error on bip49 tab for coins without bip49', function(done
) {
2998 driver
.findElement(By
.css('#bip49-tab a'))
3000 driver
.findElement(By
.css(".phrase"))
3001 .sendKeys("abandon abandon ability");
3002 driver
.sleep(generateDelay
).then(function() {
3003 selectNetwork("CLAM - Clams");
3004 // bip49 available is hidden
3005 driver
.findElement(By
.css('#bip49 .available'))
3006 .getAttribute("class")
3007 .then(function(classes
) {
3008 expect(classes
).toContain("hidden");
3009 // bip49 unavailable is shown
3010 driver
.findElement(By
.css('#bip49 .unavailable'))
3011 .getAttribute("class")
3012 .then(function(classes
) {
3013 expect(classes
).not
.toContain("hidden");
3014 // check there are no addresses shown
3015 driver
.findElements(By
.css('.addresses tr'))
3016 .then(function(rows
) {
3017 expect(rows
.length
).toBe(0);
3018 // check the derived private key is blank
3019 driver
.findElement(By
.css('.extended-priv-key'))
3020 .getAttribute("value")
3021 .then(function(xprv
) {
3022 expect(xprv
).toBe('');
3023 // check the derived public key is blank
3024 driver
.findElement(By
.css('.extended-pub-key'))
3025 .getAttribute("value")
3026 .then(function(xpub
) {
3027 expect(xpub
).toBe('');
3038 // Cleared mnemonic and root key still allows addresses to be generated
3039 // https://github.com/iancoleman/bip39/issues/43
3040 it('Clears old root keys from memory when mnemonic is cleared', function(done
) {
3042 driver
.findElement(By
.css(".phrase"))
3043 .sendKeys("abandon abandon ability");
3044 driver
.sleep(generateDelay
).then(function() {
3045 // clear the mnemonic and root key
3046 // using selenium .clear() doesn't seem to trigger the 'input' event
3047 // so clear it using keys instead
3048 driver
.findElement(By
.css('.phrase'))
3049 .sendKeys(Key
.CONTROL
,"a");
3050 driver
.findElement(By
.css('.phrase'))
3051 .sendKeys(Key
.DELETE
);
3052 driver
.findElement(By
.css('.root-key'))
3053 .sendKeys(Key
.CONTROL
,"a");
3054 driver
.findElement(By
.css('.root-key'))
3055 .sendKeys(Key
.DELETE
);
3056 driver
.sleep(generateDelay
).then(function() {
3057 // try to generate more addresses
3058 driver
.findElement(By
.css('.more'))
3060 driver
.sleep(generateDelay
).then(function() {
3061 driver
.findElements(By
.css(".addresses tr"))
3062 .then(function(els
) {
3063 // check there are no addresses shown
3064 expect(els
.length
).toBe(0);
3073 // error trying to generate addresses from xpub with hardened derivation
3074 it('Shows error for hardened addresses with xpub root key', function(done
) {
3075 driver
.findElement(By
.css('#bip32-tab a'))
3077 driver
.executeScript(function() {
3078 $(".hardened-addresses").prop("checked", true);
3080 // set xpub for account 0 of bip44 for 'abandon abandon ability'
3081 driver
.findElement(By
.css("#root-key"))
3082 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
3083 driver
.sleep(feedbackDelay
).then(function() {
3084 // Check feedback is correct
3085 driver
.findElement(By
.css('.feedback'))
3087 .then(function(feedback
) {
3088 var msg
= "Hardened derivation path is invalid with xpub key";
3089 expect(feedback
).toBe(msg
);
3095 // Litecoin uses ltub by default, and can optionally be set to xprv
3097 // https://github.com/iancoleman/bip39/issues/96
3098 // Issue with extended keys on Litecoin
3099 it('Uses ltub by default for litecoin, but can be set to xprv', function(done
) {
3100 driver
.findElement(By
.css('.phrase'))
3101 .sendKeys("abandon abandon ability");
3102 selectNetwork("LTC - Litecoin");
3103 driver
.sleep(generateDelay
).then(function() {
3104 // check the extended key is generated correctly
3105 driver
.findElement(By
.css('.root-key'))
3106 .getAttribute("value")
3107 .then(function(rootKey
) {
3108 expect(rootKey
).toBe("Ltpv71G8qDifUiNesiPqf6h5V6eQ8ic77oxQiYtawiACjBEx3sTXNR2HGDGnHETYxESjqkMLFBkKhWVq67ey1B2MKQXannUqNy1RZVHbmrEjnEU");
3109 // set litecoin to use ltub
3110 driver
.executeScript(function() {
3111 $(".litecoin-use-ltub").prop("checked", false);
3112 $(".litecoin-use-ltub").trigger("change");
3114 driver
.sleep(generateDelay
).then(function() {
3115 driver
.findElement(By
.css('.root-key'))
3116 .getAttribute("value")
3117 .then(function(rootKey
) {
3118 expect(rootKey
).toBe("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
3127 // https://github.com/iancoleman/bip39/issues/99#issuecomment-327094159
3128 // "warn me emphatically when they have detected invalid input" to the entropy field
3129 // A warning is shown when entropy is filtered and discarded
3130 it('Warns when entropy is filtered and discarded', function(done
) {
3131 driver
.findElement(By
.css('.use-entropy'))
3133 // set entropy to have no filtered content
3134 driver
.findElement(By
.css('.entropy'))
3135 .sendKeys("00000000 00000000 00000000 00000000");
3136 driver
.sleep(generateDelay
).then(function() {
3137 // check the filter warning does not show
3138 driver
.findElement(By
.css('.entropy-container .filter-warning'))
3139 .getAttribute("class")
3140 .then(function(classes
) {
3141 expect(classes
).toContain("hidden");
3142 // set entropy to have some filtered content
3143 driver
.findElement(By
.css('.entropy'))
3144 .sendKeys("10000000 zxcvbn 00000000 00000000 00000000");
3145 driver
.sleep(entropyFeedbackDelay
).then(function() {
3146 // check the filter warning shows
3147 driver
.findElement(By
.css('.entropy-container .filter-warning'))
3148 .getAttribute("class")
3149 .then(function(classes
) {
3150 expect(classes
).not
.toContain("hidden");
3158 // Bitcoin Cash address can be set to use cashaddr format
3159 it('Can use cashaddr format for bitcoin cash addresses', function(done
) {
3160 driver
.executeScript(function() {
3161 $(".use-bch-cashaddr-addresses").prop("checked", true);
3163 driver
.findElement(By
.css('.phrase'))
3164 .sendKeys("abandon abandon ability");
3165 selectNetwork("BCH - Bitcoin Cash");
3166 driver
.sleep(generateDelay
).then(function() {
3167 getFirstAddress(function(address
) {
3168 expect(address
).toBe("bitcoincash:qzlquk7w4hkudxypl4fgv8x279r754dkvur7jpcsps");
3174 // Bitcoin Cash address can be set to use bitpay format
3175 it('Can use bitpay format for bitcoin cash addresses', function(done
) {
3176 driver
.executeScript(function() {
3177 $(".use-bch-bitpay-addresses").prop("checked", true);
3179 driver
.findElement(By
.css('.phrase'))
3180 .sendKeys("abandon abandon ability");
3181 selectNetwork("BCH - Bitcoin Cash");
3182 driver
.sleep(generateDelay
).then(function() {
3183 getFirstAddress(function(address
) {
3184 expect(address
).toBe("CZnpA9HPmvhuhLLPWJP8rNDpLUYXy1LXFk");
3190 // Bitcoin Cash address can be set to use legacy format
3191 it('Can use legacy format for bitcoin cash addresses', function(done
) {
3192 driver
.executeScript(function() {
3193 $(".use-bch-legacy-addresses").prop("checked", true);
3195 driver
.findElement(By
.css('.phrase'))
3196 .sendKeys("abandon abandon ability");
3197 selectNetwork("BCH - Bitcoin Cash");
3198 driver
.sleep(generateDelay
).then(function() {
3199 getFirstAddress(function(address
) {
3200 expect(address
).toBe("1JKvb6wKtsjNoCRxpZ4DGrbniML7z5U16A");
3206 // End of tests ported from old suit, so no more comments above each test now
3208 it('Can generate more addresses from a custom index', function(done
) {
3209 var expectedIndexes
= [
3210 0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,
3211 40,41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59
3213 driver
.findElement(By
.css('.phrase'))
3214 .sendKeys("abandon abandon ability");
3215 driver
.sleep(generateDelay
).then(function() {
3216 // Set start of next lot of rows to be from index 40
3217 // which means indexes 20-39 will not be in the table.
3218 driver
.findElement(By
.css('.more-rows-start-index'))
3220 driver
.findElement(By
.css('.more'))
3222 driver
.sleep(generateDelay
).then(function() {
3223 // Check actual indexes in the table match the expected pattern
3224 driver
.findElements(By
.css(".index"))
3225 .then(function(els
) {
3226 expect(els
.length
).toBe(expectedIndexes
.length
);
3227 var testRowAtIndex = function(i
) {
3228 if (i
>= expectedIndexes
.length
) {
3233 .then(function(actualPath
) {
3234 var noHardened
= actualPath
.replace(/'/g, "");
3235 var pathBits = noHardened.split("/")
3236 var lastBit = pathBits[pathBits.length-1];
3237 var actualIndex = parseInt(lastBit);
3238 var expectedIndex = expectedIndexes[i];
3239 expect(actualIndex).toBe(expectedIndex);
3240 testRowAtIndex(i+1);
3250 it('Can generate BIP141 addresses
with P2WPKH
-in-P2SH semanitcs
', function(done) {
3251 // Sourced from BIP49 official test specs
3252 driver.findElement(By.css('#bip141
-tab a
'))
3254 driver.findElement(By.css('.bip141
-path
'))
3256 driver.findElement(By.css('.bip141
-path
'))
3257 .sendKeys("m/49'/1'/0'/0");
3258 selectNetwork("BTC
- Bitcoin Testnet
");
3259 driver.findElement(By.css(".phrase
"))
3260 .sendKeys("abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about
");
3261 driver.sleep(generateDelay).then(function() {
3262 getFirstAddress(function(address) {
3263 expect(address).toBe("2Mww8dCYPUpKHofjgcXcBCEGmniw9CoaiD2
");
3269 it('Can generate BIP141 addresses with P2WPKH semanitcs', function(done) {
3270 // This result tested against bitcoinjs-lib test spec for segwit address
3271 // using the first private key of this mnemonic and default path m/0
3272 // https://github.com/bitcoinjs/bitcoinjs-lib/blob/9c8503cab0c6c30a95127042703bc18e8d28c76d/test/integration/addresses.js#L50
3273 // so whilst not directly comparable, substituting the private key produces
3274 // identical results between this tool and the bitcoinjs-lib test.
3275 // Private key generated is:
3276 // L3L8Nu9whawPBNLGtFqDhKut9DKKfG3CQoysupT7BimqVCZsLFNP
3277 driver.findElement(By.css('#bip141-tab a'))
3280 driver.executeScript(function() {
3281 $(".bip141
-semantics option
[selected
]").removeAttr("selected
");
3282 $(".bip141
-semantics option
").filter(function(i,e) {
3283 return $(e).html() == "P2WPKH
";
3284 }).prop("selected
", true);
3285 $(".bip141
-semantics
").trigger("change
");
3287 driver.findElement(By.css(".phrase
"))
3288 .sendKeys("abandon abandon ability
");
3289 driver.sleep(generateDelay).then(function() {
3290 getFirstAddress(function(address) {
3291 expect(address).toBe("bc1qfwu6a5a3evygrk8zvdxxvz4547lmpyx5vsfxe9
");
3297 it('Shows the entropy used by the PRNG when clicking generate', function(done) {
3298 driver.findElement(By.css('.generate')).click();
3299 driver.sleep(generateDelay).then(function() {
3300 driver.findElement(By.css('.entropy'))
3301 .getAttribute("value
")
3302 .then(function(entropy) {
3303 expect(entropy).not.toBe("");
3309 it('Shows the index of each word in the mnemonic', function(done) {
3310 driver.findElement(By.css('.phrase'))
3311 .sendKeys("abandon abandon ability
");
3312 driver.sleep(generateDelay).then(function() {
3313 driver.findElement(By.css('.use-entropy'))
3315 driver.findElement(By.css('.word-indexes'))
3317 .then(function(indexes) {
3318 expect(indexes).toBe("0, 0, 1");
3324 it('Shows the derivation path for bip84 tab', function(done) {
3325 driver.findElement(By.css('#bip84-tab a'))
3327 driver.findElement(By.css('.phrase'))
3328 .sendKeys('abandon abandon ability');
3329 driver.sleep(generateDelay).then(function() {
3330 driver.findElement(By.css('#bip84 .path'))
3331 .getAttribute("value
")
3332 .then(function(path) {
3333 expect(path).toBe("m
/84'/0'/0'/0");
3339 it('Shows the extended private key for bip84 tab', function(done) {
3340 driver.findElement(By.css('#bip84-tab a'))
3342 driver.findElement(By.css('.phrase'))
3343 .sendKeys('abandon abandon ability');
3344 driver.sleep(generateDelay).then(function() {
3345 driver.findElement(By.css('.extended-priv-key'))
3346 .getAttribute("value
")
3347 .then(function(path) {
3348 expect(path).toBe("zprvAev3RKrZ3QVKiUFCfdeMRen1BPDJgdNt1XpxiDy8acSs4kkAGTCvq7HeRYRNNpo8EtEjCFQBWavJwtCUR29y4TUCH4X5RXMcyq48uN8y9BP
");
3354 it('Shows the extended public key for bip84 tab', function(done) {
3355 driver.findElement(By.css('#bip84-tab a'))
3357 driver.findElement(By.css('.phrase'))
3358 .sendKeys('abandon abandon ability');
3359 driver.sleep(generateDelay).then(function() {
3360 driver.findElement(By.css('.extended-pub-key'))
3361 .getAttribute("value
")
3362 .then(function(path) {
3363 expect(path).toBe("zpub6suPpqPSsn3cvxKfmfBMnnijjR3o666jNkkZWcNk8wyqwZ5JozXBNuc8Gs7DB3uLwTDvGVTspVEAUQcEjKF3pZHgywVbubdTqbXTUg7usyx
");
3369 it('Changes the address list if bip84 account is changed', function(done) {
3370 driver.findElement(By.css('#bip84-tab a'))
3372 driver.findElement(By.css('#bip84 .account'))
3374 driver.findElement(By.css('.phrase'))
3375 .sendKeys('abandon abandon ability');
3376 driver.sleep(generateDelay).then(function() {
3377 getFirstAddress(function(address) {
3378 expect(address).toBe("bc1qp7vv669t2fy965jdzvqwrraana89ctd5ewc662
");
3384 it('Changes the address list if bip84 change is changed', function(done) {
3385 driver.findElement(By.css('#bip84-tab a'))
3387 driver.findElement(By.css('#bip84 .change'))
3389 driver.findElement(By.css('.phrase'))
3390 .sendKeys('abandon abandon ability');
3391 driver.sleep(generateDelay).then(function() {
3392 getFirstAddress(function(address) {
3393 expect(address).toBe("bc1qr39vj6rh06ff05m53uxq8uazehwhccswylhrs2
");
3399 it('Passes the official BIP84 test spec for rootpriv', function(done) {
3400 driver.findElement(By.css('#bip84-tab a'))
3402 driver.findElement(By.css('.phrase'))
3403 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3404 driver.sleep(generateDelay).then(function() {
3405 driver.findElement(By.css(".root
-key
"))
3406 .getAttribute("value
")
3407 .then(function(rootKey) {
3408 expect(rootKey).toBe("zprvAWgYBBk7JR8Gjrh4UJQ2uJdG1r3WNRRfURiABBE3RvMXYSrRJL62XuezvGdPvG6GFBZduosCc1YP5wixPox7zhZLfiUm8aunE96BBa4Kei5
");
3414 it('Passes the official BIP84 test spec for account 0 xprv', function(done) {
3415 driver.findElement(By.css('#bip84-tab a'))
3417 driver.findElement(By.css('.phrase'))
3418 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3419 driver.sleep(generateDelay).then(function() {
3420 driver.findElement(By.css("#bip84
.account
-xprv
"))
3421 .getAttribute("value
")
3422 .then(function(rootKey) {
3423 expect(rootKey).toBe("zprvAdG4iTXWBoARxkkzNpNh8r6Qag3irQB8PzEMkAFeTRXxHpbF9z4QgEvBRmfvqWvGp42t42nvgGpNgYSJA9iefm1yYNZKEm7z6qUWCroSQnE
");
3429 it('Passes the official BIP84 test spec for account 0 xpub', function(done) {
3430 driver.findElement(By.css('#bip84-tab a'))
3432 driver.findElement(By.css('.phrase'))
3433 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3434 driver.sleep(generateDelay).then(function() {
3435 driver.findElement(By.css("#bip84
.account
-xpub
"))
3436 .getAttribute("value
")
3437 .then(function(rootKey) {
3438 expect(rootKey).toBe("zpub6rFR7y4Q2AijBEqTUquhVz398htDFrtymD9xYYfG1m4wAcvPhXNfE3EfH1r1ADqtfSdVCToUG868RvUUkgDKf31mGDtKsAYz2oz2AGutZYs
");
3444 it('Passes the official BIP84 test spec for account 0 first address', function(done) {
3445 driver.findElement(By.css('#bip84-tab a'))
3447 driver.findElement(By.css('.phrase'))
3448 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3449 driver.sleep(generateDelay).then(function() {
3450 getFirstAddress(function(address) {
3451 expect(address).toBe("bc1qcr8te4kr609gcawutmrza0j4xv80jy8z306fyu
");
3457 it('Passes the official BIP84 test spec for account 0 first change address', function(done) {
3458 driver.findElement(By.css('#bip84-tab a'))
3460 driver.findElement(By.css('.phrase'))
3461 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3462 driver.findElement(By.css('#bip84 .change'))
3464 driver.sleep(generateDelay).then(function() {
3465 getFirstAddress(function(address) {
3466 expect(address).toBe("bc1q8c6fshw2dlwun7ekn9qwf37cu2rn755upcp6el
");
3472 it('Can display the table as csv', function(done) {
3473 var headings = "path
,address
,public key
,private key
";
3474 var row1 = "m
/44'/0'/0'/0/0,1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug
,033f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3
,L26cVSpWFkJ6aQkPkKmTzLqTdLJ923e6CzrVh9cmx21QHsoUmrEE
";
3475 var row20 = "m
/44'/0'/0'/0/19,1KhBy28XLAciXnnRvm71PvQJaETyrxGV55
,02b4b3e396434d8cdd20c03ac4aaa07387784d5d867b75987f516f5705ee68cb3a
,L4GrDrjReMsCAu5DkLXn79jSb95qR7Zfx7eshybCQZ1qL32MXJab
";
3476 driver.findElement(By.css('.phrase'))
3477 .sendKeys('abandon abandon ability');
3478 driver.sleep(generateDelay).then(function() {
3479 driver.findElement(By.css('.csv'))
3480 .getAttribute("value
")
3481 .then(function(csv) {
3482 expect(csv).toContain(headings);
3483 expect(csv).toContain(row1);
3484 expect(csv).toContain(row20);
3490 it('LeftPads ethereum keys that are less than 32 bytes', function(done) {
3491 // see https://github.com/iancoleman/bip39/issues/155
3492 selectNetwork("ETH
- Ethereum
");
3493 driver.findElement(By.css('#bip32-tab a'))
3495 driver.findElement(By.css('#bip32-path'))
3497 driver.findElement(By.css('#bip32-path'))
3498 .sendKeys("m
/44'/60'/0'");
3499 driver.findElement(By.css('.phrase'))
3500 .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');
3501 driver.sleep(generateDelay).then(function() {
3502 getFirstAddress(function(address) {
3503 expect(address).toBe("0x8943E785B4a5714FC87a3aFAad1eB1FeB602B118");
3509 it('Can encrypt private keys using BIP38', function(done) {
3510 // see https://github.com/iancoleman/bip39/issues/140
3511 driver.executeScript(function() {
3512 $(".use-bip38
").prop("checked
", true);
3514 driver.findElement(By.css('.bip38-password'))
3515 .sendKeys('bip38password');
3516 driver.findElement(By.css('.rows-to-add'))
3518 driver.findElement(By.css('.rows-to-add'))
3520 driver.findElement(By.css('.phrase'))
3521 .sendKeys('abandon abandon ability');
3522 driver.sleep(bip38delay).then(function() {
3524 getFirstRowValue(function(address) {
3525 expect(address).toBe("1NCvSdumA3ngMM9c4aqU56AM6rqXddfuXB
");
3527 getFirstRowValue(function(pubkey) {
3528 expect(pubkey).toBe("043f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3884a74447ea901729b1e73a999b7520e7cb55b4120e6432c64153ccab8a848e1
");
3530 getFirstRowValue(function(privkey) {
3531 expect(privkey).toBe("6PRNRiFnj1RoR3sXhymdCvoZCgnUHQpfupNdKkFbWJkwWQEKesWt1EDMDM
");
3537 }, bip38delay + 5000);
3539 it('Shows the checksum for the entropy', function(done) {
3540 driver.findElement(By.css('.use-entropy'))
3542 driver.findElement(By.css('.entropy'))
3543 .sendKeys("00000000000000000000000000000000");
3544 driver.sleep(generateDelay).then(function() {
3545 driver.findElement(By.css('.checksum'))
3547 .then(function(text) {
3548 expect(text).toBe("1");
3554 it('Shows the checksum for the entropy with the correct groupings', function(done) {
3555 driver.findElement(By.css('.use-entropy'))
3557 // create a checksum of 20 bits, which spans multiple words
3558 driver.findElement(By.css('.entropy'))
3559 .sendKeys("F000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
");
3560 driver.sleep(generateDelay).then(function() {
3561 driver.findElement(By.css('.checksum'))
3563 .then(function(text) {
3564 // first group is 9 bits, second group is 11
3565 expect(text).toBe("011010111 01110000110");
3571 it('Uses vprv for bitcoin testnet p2wpkh', function(done) {
3572 selectNetwork("BTC
- Bitcoin Testnet
");
3573 driver.findElement(By.css('#bip84-tab a'))
3575 driver.findElement(By.css('.phrase'))
3576 .sendKeys('abandon abandon ability');
3577 driver.sleep(generateDelay).then(function() {
3578 driver.findElement(By.css('.root-key'))
3579 .getAttribute("value
")
3580 .then(function(path) {
3581 expect(path).toBe("vprv9DMUxX4ShgxML9N2YV5CvWEebWrM9aJ5ULpbRRyzyWu6vs4BzTvbfFFrH41N5hVi7MYSfiugd765L3JmAfDM5po36Y8ouCKRDeYQwByCmS7
");
3587 it('Shows a warning if generating weak mnemonics', function(done) {
3588 driver.executeScript(function() {
3589 $(".strength option
[selected
]").removeAttr("selected
");
3590 $(".strength option
[value
=6]").prop("selected
", true);
3591 $(".strength
").trigger("change
");
3593 driver.findElement(By.css(".generate
-container
.warning
"))
3594 .getAttribute("class")
3595 .then(function(classes) {
3596 expect(classes).not.toContain("hidden
");
3601 it('Does not show a warning if generating strong mnemonics', function(done) {
3602 driver.executeScript(function() {
3603 $(".strength option
[selected
]").removeAttr("selected
");
3604 $(".strength option
[value
=12]").prop("selected
", true);
3606 driver.findElement(By.css(".generate
-container
.warning
"))
3607 .getAttribute("class")
3608 .then(function(classes) {
3609 expect(classes).toContain("hidden
");
3614 it('Shows a warning if overriding weak entropy with longer mnemonics', function(done) {
3615 driver.findElement(By.css('.use-entropy'))
3617 driver.findElement(By.css('.entropy'))
3618 .sendKeys("0123456789abcdef
"); // 6 words
3619 driver.executeScript(function() {
3620 $(".mnemonic
-length
").val("12").trigger("change
");
3622 driver.findElement(By.css(".weak
-entropy
-override
-warning
"))
3623 .getAttribute("class")
3624 .then(function(classes) {
3625 expect(classes).not.toContain("hidden
");
3630 it('Does not show a warning if entropy is stronger than mnemonic length', function(done) {
3631 driver.findElement(By.css('.use-entropy'))
3633 driver.findElement(By.css('.entropy'))
3634 .sendKeys("0123456789abcdef0123456789abcdef0123456789abcdef
"); // 18 words
3635 driver.executeScript(function() {
3636 $(".mnemonic
-length
").val("12").trigger("change
");
3638 driver.findElement(By.css(".weak
-entropy
-override
-warning
"))
3639 .getAttribute("class")
3640 .then(function(classes) {
3641 expect(classes).toContain("hidden
");
3646 it('Shows a warning for litecoin BIP84 (which does not have p2wpkh params)', function(done) {
3647 driver.findElement(By.css('.phrase'))
3648 .sendKeys('abandon abandon ability');
3649 selectNetwork("LTC
- Litecoin
");
3650 driver.findElement(By.css('#bip84-tab a'))
3652 // bip84 unavailable is shown
3653 driver.sleep(feedbackDelay).then(function() {
3654 driver.findElement(By.css('#bip84 .unavailable'))
3655 .getAttribute("class")
3656 .then(function(classes) {
3657 expect(classes).not.toContain("hidden
");
3663 it('Shows litecoin BIP49 addresses', function(done) {
3664 driver.findElement(By.css('.phrase'))
3665 .sendKeys('abandon abandon ability');
3666 selectNetwork("LTC
- Litecoin
");
3667 driver.findElement(By.css('#bip49-tab a'))
3669 // bip49 addresses are shown
3670 driver.sleep(generateDelay).then(function() {
3671 driver.findElement(By.css('#bip49 .available'))
3672 .getAttribute("class")
3673 .then(function(classes) {
3674 expect(classes).not.toContain("hidden
");
3675 // check first address
3676 getFirstAddress(function(address) {
3677 expect(address).toBe("MFwLPhsXoBuSLL8cLmW9uK6tChkzduV8qN
");