2 // cd /path/to/repo/tests
3 // jasmine spec/tests.js
9 // see https://developer.mozilla.org/en-US/docs/Mozilla/Firefox/Headless_mode#Automated_testing_with_headless_mode
11 // USER SPECIFIED OPTIONS
12 var browser
= process
.env
.BROWSER
; //"firefox"; // or "chrome"
14 console
.log("Browser can be set via environment variable, eg");
15 console
.log("BROWSER=firefox jasmine spec/tests.js");
16 console
.log("Options for BROWSER are firefox chrome");
17 console
.log("Using default browser: chrome");
21 console
.log("Using browser: " + browser
);
26 var webdriver
= require('selenium-webdriver');
27 var By
= webdriver
.By
;
28 var Key
= webdriver
.Key
;
29 var until
= webdriver
.until
;
33 var generateDelay
= 1500;
34 var feedbackDelay
= 500;
35 var entropyFeedbackDelay
= 500;
36 var bip38delay
= 15000;
38 // url uses file:// scheme
39 var path
= require('path')
40 var parentDir
= path
.resolve(process
.cwd(), '..', 'src', 'index.html');
41 var url
= "file://" + parentDir
;
42 if (browser
== "firefox") {
43 // TODO loading local html in firefox is broken
44 console
.log("Loading local html in firefox is broken, see https://stackoverflow.com/q/46367054");
45 console
.log("You must run a server in this case, ie do this:");
46 console
.log("$ cd /path/to/bip39/src");
47 console
.log("$ python -m http.server");
48 url
= "http://localhost:8000";
51 // Variables dependent on specific browser selection
53 if (browser
== "firefox") {
54 var firefox
= require('selenium-webdriver/firefox');
55 var binary
= new firefox
.Binary(firefox
.Channel
.NIGHTLY
);
56 binary
.addArguments("-headless");
57 newDriver = function() {
58 return new webdriver
.Builder()
59 .forBrowser('firefox')
60 .setFirefoxOptions(new firefox
.Options().setBinary(binary
))
64 else if (browser
== "chrome") {
65 var chrome
= require('selenium-webdriver/chrome');
66 newDriver = function() {
67 return new webdriver
.Builder()
69 .setChromeOptions(new chrome
.Options().addArguments("headless"))
76 function testNetwork(done
, params
) {
77 var phrase
= params
.phrase
|| 'abandon abandon ability';
78 driver
.findElement(By
.css('.phrase'))
80 selectNetwork(params
.selectText
);
81 driver
.sleep(generateDelay
).then(function() {
82 getFirstAddress(function(address
) {
83 expect(address
).toBe(params
.firstAddress
);
89 function getFirstRowValue(handler
, selector
) {
90 driver
.findElements(By
.css(selector
))
97 function getFirstAddress(handler
) {
98 getFirstRowValue(handler
, ".address");
101 function getFirstPath(handler
) {
102 getFirstRowValue(handler
, ".index");
105 function testColumnValuesAreInvisible(done
, columnClassName
) {
106 var selector
= "." + columnClassName
+ " span";
107 driver
.findElements(By
.css(selector
))
108 .then(function(els
) {
109 els
[0].getAttribute("class")
110 .then(function(classes
) {
111 expect(classes
).toContain("invisible");
117 function testRowsAreInCorrectOrder(done
) {
118 driver
.findElements(By
.css('.index'))
119 .then(function(els
) {
120 var testRowAtIndex = function(i
) {
121 if (i
>= els
.length
) {
126 .then(function(actualPath
) {
127 var noHardened
= actualPath
.replace(/'/g, "");
128 var pathBits = noHardened.split("/")
129 var lastBit = pathBits[pathBits.length-1];
130 var actualIndex = parseInt(lastBit);
131 expect(actualIndex).toBe(i);
140 function selectNetwork(name) {
141 driver.executeScript(function() {
142 var selectText = arguments[0];
143 $(".network option[selected]").removeAttr("selected");
144 $(".network option").filter(function(i,e) {
145 return $(e).html() == selectText;
146 }).prop("selected", true);
147 $(".network").trigger("change");
151 function testEntropyType(done, entropyText, entropyTypeUnsafe) {
152 // entropy type is compiled into regexp so needs escaping
153 // see https://stackoverflow.com/a/2593661
154 var entropyType = (entropyTypeUnsafe+'').replace(/[.?*+^$[\]\\(){}|-]/g, "\\$&");
155 driver.findElement(By.css('.use-entropy
'))
157 driver.findElement(By.css('.entropy
'))
158 .sendKeys(entropyText);
159 driver.sleep(generateDelay).then(function() {
160 driver.findElement(By.css('.entropy
-container
'))
162 .then(function(text) {
163 var re = new RegExp("Entropy Type\\s+" + entropyType);
164 expect(text).toMatch(re);
170 function testEntropyBits(done, entropyText, entropyBits) {
171 driver.findElement(By.css('.use-entropy
'))
173 driver.findElement(By.css('.entropy
'))
174 .sendKeys(entropyText);
175 driver.sleep(generateDelay).then(function() {
176 driver.findElement(By.css('.entropy
-container
'))
178 .then(function(text) {
179 var re = new RegExp("Total Bits\\s+" + entropyBits);
180 expect(text).toMatch(re);
186 function testEntropyFeedback(done, entropyDetail) {
187 // entropy type is compiled into regexp so needs escaping
188 // see https://stackoverflow.com/a/2593661
189 if ("type" in entropyDetail) {
190 entropyDetail.type = (entropyDetail.type+'').replace(/[.?*+^$[\]\\(){}|-]/g, "\\$&");
192 driver.findElement(By.css('.use-entropy
'))
194 driver.findElement(By.css('.entropy
'))
195 .sendKeys(entropyDetail.entropy);
196 driver.sleep(entropyFeedbackDelay).then(function() {
197 driver.findElement(By.css('.entropy
-container
'))
199 .then(function(text) {
200 driver.findElement(By.css('.phrase
'))
201 .getAttribute("value")
202 .then(function(phrase) {
203 if ("filtered" in entropyDetail) {
204 var key = "Filtered Entropy";
205 var value = entropyDetail.filtered;
206 var reText = key + "\\s+" + value;
207 var re = new RegExp(reText);
208 expect(text).toMatch(re);
210 if ("type" in entropyDetail) {
211 var key = "Entropy Type";
212 var value = entropyDetail.type;
213 var reText = key + "\\s+" + value;
214 var re = new RegExp(reText);
215 expect(text).toMatch(re);
217 if ("events" in entropyDetail) {
218 var key = "Event Count";
219 var value = entropyDetail.events;
220 var reText = key + "\\s+" + value;
221 var re = new RegExp(reText);
222 expect(text).toMatch(re);
224 if ("bits" in entropyDetail) {
225 var key = "Total Bits";
226 var value = entropyDetail.bits;
227 var reText = key + "\\s+" + value;
228 var re = new RegExp(reText);
229 expect(text).toMatch(re);
231 if ("bitsPerEvent" in entropyDetail) {
232 var key = "Bits Per Event";
233 var value = entropyDetail.bitsPerEvent;
234 var reText = key + "\\s+" + value;
235 var re = new RegExp(reText);
236 expect(text).toMatch(re);
238 if ("words" in entropyDetail) {
239 var actualWords = phrase.split(/\s+/)
240 .filter(function(w) { return w.length > 0 })
242 expect(actualWords).toBe(entropyDetail.words);
244 if ("strength" in entropyDetail) {
245 var key = "Time To Crack";
246 var value = entropyDetail.strength;
247 var reText = key + "\\s+" + value;
248 var re = new RegExp(reText);
249 expect(text).toMatch(re);
257 function testClientSelect(done, params) {
258 // set mnemonic and select bip32 tab
259 driver.findElement(By.css('#bip32
-tab a
'))
261 driver.findElement(By.css('.phrase
'))
262 .sendKeys("abandon abandon ability");
263 driver.sleep(generateDelay).then(function() {
265 // set bip32 client to bitcoin core
266 driver.executeScript(function() {
267 $("#bip32-client").val(arguments[0]).trigger("change");
268 }, params.selectValue);
269 driver.sleep(generateDelay).then(function() {
270 // check the derivation path is correct
271 driver.findElement(By.css("#bip32-path"))
272 .getAttribute("value")
273 .then(function(path) {
274 expect(path).toBe(params.bip32path);
275 // check hardened addresses is selected
276 driver.findElement(By.css(".hardened-addresses"))
277 .getAttribute("checked")
278 .then(function(isChecked) {
279 expect(isChecked).toBe(params.useHardenedAddresses);
280 // check input is readonly
281 driver.findElement(By.css("#bip32-path"))
282 .getAttribute("readonly")
283 .then(function(isReadonly) {
284 expect(isReadonly).toBe("true");
295 describe('BIP39 Tool Tests
', function() {
297 beforeEach(function(done) {
298 driver = newDriver();
299 driver.get(url).then(done);
302 // Close the website after each test is run (so that it is opened fresh each time)
303 afterEach(function(done) {
304 driver.quit().then(done);
309 // Page initially loads with blank phrase
310 it('Should load the page
', function(done) {
311 driver.findElement(By.css('.phrase
'))
312 .getAttribute('value
').then(function(value) {
313 expect(value).toBe('');
319 it('Should have text on the page
', function(done) {
320 driver.findElement(By.css('body
'))
322 .then(function(text) {
323 var textToFind = "You can enter an existing BIP39 mnemonic";
324 expect(text).toContain(textToFind);
329 // Entering mnemonic generates addresses
330 it('Should have a list
of addresses
', function(done) {
331 driver.findElement(By.css('.phrase
'))
332 .sendKeys('abandon abandon ability
');
333 driver.sleep(generateDelay).then(function() {
334 driver.findElements(By.css('.address
'))
335 .then(function(els) {
336 expect(els.length).toBe(20);
342 // Generate button generates random mnemonic
343 it('Should be able to generate a random mnemonic
', function(done) {
344 // initial phrase is blank
345 driver.findElement(By.css('.phrase
'))
346 .getAttribute("value")
347 .then(function(phrase) {
348 expect(phrase.length).toBe(0);
350 driver.findElement(By.css('.generate
')).click();
351 driver.sleep(generateDelay).then(function() {
352 // new phrase is not blank
353 driver.findElement(By.css('.phrase
'))
354 .getAttribute("value")
355 .then(function(phrase) {
356 expect(phrase.length).toBeGreaterThan(0);
363 // Mnemonic length can be customized
364 it('Should allow custom length mnemonics
', function(done) {
366 driver.executeScript(function() {
367 $(".strength option[selected]").removeAttr("selected");
368 $(".strength option[value=6]").prop("selected", true);
370 driver.findElement(By.css('.generate
')).click();
371 driver.sleep(generateDelay).then(function() {
372 driver.findElement(By.css('.phrase
'))
373 .getAttribute("value")
374 .then(function(phrase) {
375 var words = phrase.split(" ");
376 expect(words.length).toBe(6);
382 // Passphrase can be set
383 it('Allows a passphrase to be
set', function(done) {
384 driver.findElement(By.css('.phrase
'))
385 .sendKeys('abandon abandon ability
');
386 driver.findElement(By.css('.passphrase
'))
387 .sendKeys('secure_passphrase
');
388 driver.sleep(generateDelay).then(function() {
389 getFirstAddress(function(address) {
390 expect(address).toBe("15pJzUWPGzR7avffV9nY5by4PSgSKG9rba");
396 // Network can be set to networks other than bitcoin
397 it('Allows selection
of bitcoin testnet
', function(done) {
399 selectText: "BTC - Bitcoin Testnet",
400 firstAddress: "mucaU5iiDaJDb69BHLeDv8JFfGiyg2nJKi",
402 testNetwork(done, params);
404 it('Allows selection
of litecoin
', function(done) {
406 selectText: "LTC - Litecoin",
407 firstAddress: "LQ4XU8RX2ULPmPq9FcUHdVmPVchP9nwXdn",
409 testNetwork(done, params);
411 it('Allows selection
of ripple
', function(done) {
413 selectText: "XRP - Ripple",
414 firstAddress: "rLTFnqbmCVPGx6VfaygdtuKWJgcN4v1zRS",
415 phrase: "ill clump only blind unit burden thing track silver cloth review awake useful craft whale all satisfy else trophy sunset walk vanish hope valve",
417 testNetwork(done, params);
419 it('Allows selection
of dogecoin
', function(done) {
421 selectText: "DOGE - Dogecoin",
422 firstAddress: "DPQH2AtuzkVSG6ovjKk4jbUmZ6iXLpgbJA",
424 testNetwork(done, params);
426 it('Allows selection
of denarius
', function(done) {
428 selectText: "DNR - Denarius",
429 firstAddress: "DFdFMVUMzU9xX88EywXvAGwjiwpxyh9vKb",
431 testNetwork(done, params);
433 it('Allows selection
of shadowcash
', function(done) {
435 selectText: "SDC - ShadowCash",
436 firstAddress: "SiSZtfYAXEFvMm3XM8hmtkGDyViRwErtCG",
438 testNetwork(done, params);
440 it('Allows selection
of shadowcash testnet
', function(done) {
442 selectText: "SDC - ShadowCash Testnet",
443 firstAddress: "tM2EDpVKaTiEg2NZg3yKg8eqjLr55BErHe",
445 testNetwork(done, params);
447 it('Allows selection
of viacoin
', function(done) {
449 selectText: "VIA - Viacoin",
450 firstAddress: "Vq9Eq4N5SQnjqZvxtxzo7hZPW5XnyJsmXT",
452 testNetwork(done, params);
454 it('Allows selection
of viacoin testnet
', function(done) {
456 selectText: "VIA - Viacoin Testnet",
457 firstAddress: "tM2EDpVKaTiEg2NZg3yKg8eqjLr55BErHe",
459 testNetwork(done, params);
461 it('Allows selection
of jumbucks
', function(done) {
463 selectText: "JBS - Jumbucks",
464 firstAddress: "JLEXccwDXADK4RxBPkRez7mqsHVoJBEUew",
466 testNetwork(done, params);
468 it('Allows selection
of clam
', function(done) {
470 selectText: "CLAM - Clams",
471 firstAddress: "xCp4sakjVx4pUAZ6cBCtuin8Ddb6U1sk9y",
473 testNetwork(done, params);
475 it('Allows selection
of crown
', function(done) {
477 selectText: "CRW - Crown",
478 firstAddress: "18pWSwSUAQdiwMHUfFZB1fM2xue9X1FqE5",
480 testNetwork(done, params);
482 it('Allows selection
of dash
', function(done) {
484 selectText: "DASH - Dash",
485 firstAddress: "XdbhtMuGsPSkE6bPdNTHoFSszQKmK4S5LT",
487 testNetwork(done, params);
489 it('Allows selection
of dash testnet
', function(done) {
491 selectText: "DASH - Dash Testnet",
492 firstAddress: "yaR52EN4oojdJfBgzWJTymC4uuCLPT29Gw",
494 testNetwork(done, params);
496 it('Allows selection
of game
', function(done) {
498 selectText: "GAME - GameCredits",
499 firstAddress: "GSMY9bAp36cMR4zyT4uGVS7GFjpdXbao5Q",
501 testNetwork(done, params);
503 it('Allows selection
of komodo
', function(done) {
505 selectText: "KMD - Komodo",
506 firstAddress: "RMPPzJwAjPVZZAwJvXivHJGGjdCx6WBD2t",
508 testNetwork(done, params);
510 it('Allows selection
of namecoin
', function(done) {
512 selectText: "NMC - Namecoin",
513 firstAddress: "Mw2vK2Bvex1yYtYF6sfbEg2YGoUc98YUD2",
515 testNetwork(done, params);
517 it('Allows selection
of onixcoin
', function(done) {
519 selectText: "ONX - Onixcoin",
520 firstAddress: "XGwMqddeKjT3ddgX73QokjVbCL3aK6Yxfk",
522 testNetwork(done, params);
524 it('Allows selection
of peercoin
', function(done) {
526 selectText: "PPC - Peercoin",
527 firstAddress: "PVAiioTaK2eDHSEo3tppT9AVdBYqxRTBAm",
529 testNetwork(done, params);
531 it('Allows selection
of ethereum
', function(done) {
533 selectText: "ETH - Ethereum",
534 firstAddress: "0xe5815d5902Ad612d49283DEdEc02100Bd44C2772",
536 testNetwork(done, params);
537 // TODO test private key and public key
539 it('Allows selection
of slimcoin
', function(done) {
541 selectText: "SLM - Slimcoin",
542 firstAddress: "SNzPi1CafHFm3WWjRo43aMgiaEEj3ogjww",
544 testNetwork(done, params);
546 it('Allows selection
of slimcoin testnet
', function(done) {
548 selectText: "SLM - Slimcoin Testnet",
549 firstAddress: "n3nMgWufTek5QQAr6uwMhg5xbzj8xqc4Dq",
551 testNetwork(done, params);
553 it('Allows selection
of bitcoin cash
', function(done) {
555 selectText: "BCH - Bitcoin Cash",
556 firstAddress: "bitcoincash:qzlquk7w4hkudxypl4fgv8x279r754dkvur7jpcsps",
558 testNetwork(done, params);
560 it('Allows selection
of myriadcoin
', function(done) {
562 selectText: "XMY - Myriadcoin",
563 firstAddress: "MJEswvRR46wh9BoiVj9DzKYMBkCramhoBV",
565 testNetwork(done, params);
567 it('Allows selection
of pivx
', function(done) {
569 selectText: "PIVX - PIVX",
570 firstAddress: "DBxgT7faCuno7jmtKuu6KWCiwqsVPqh1tS",
572 testNetwork(done, params);
574 it('Allows selection
of pivx testnet
', function(done) {
576 selectText: "PIVX - PIVX Testnet",
577 firstAddress: "yB5U384n6dGkVE3by5y9VdvHHPwPg68fQj",
579 testNetwork(done, params);
581 it('Allows selection
of maza
', function(done) {
583 selectText: "MAZA - Maza",
584 firstAddress: "MGW4Bmi2NEm4PxSjgeFwhP9vg18JHoRnfw",
586 testNetwork(done, params);
588 it('Allows selection
of fujicoin
', function(done) {
590 selectText: "FJC - Fujicoin",
591 firstAddress: "FgiaLpG7C99DyR4WnPxXedRVHXSfKzUDhF",
593 testNetwork(done, params);
595 it('Allows selection
of nubits
', function(done) {
597 selectText: "USNBT - NuBits",
598 firstAddress: "BLxkabXuZSJSdesLD7KxZdqovd4YwyBTU6",
600 testNetwork(done, params);
602 it('Allows selection
of bitcoin gold
', function(done) {
604 selectText: "BTG - Bitcoin Gold",
605 firstAddress: "GdDqug4WUsn5syNbSTHatNn4XnuwZtzedx",
607 testNetwork(done, params);
609 it('Allows selection
of monacoin
', function(done) {
611 selectText: "MONA - Monacoin",
612 firstAddress: "MKMiMr7MyjDKjJbCBzgF6u4ByqTS4NkRB1",
614 testNetwork(done, params);
616 it('Allows selection
of AXE
', function(done) {
618 selectText: "AXE - Axe",
619 firstAddress: "PScwtLUyPiGrqtKXrHF37DGETLXLZdw4up",
621 testNetwork(done, params);
623 it('Allows selection
of BlackCoin
', function(done) {
625 selectText: "BLK - BlackCoin",
626 firstAddress: "B5MznAKwj7uQ42vDz3w4onhBXPcqhTwJ9z",
628 testNetwork(done, params);
630 it('Allows selection
of Neblio
', function(done) {
632 selectText: "NEBL - Neblio",
633 firstAddress: "NefkeEEvhusbHMmTRrxx7H9wFnUXd8qQsE",
635 testNetwork(done, params);
637 it('Allows selection
of Beetlecoin
', function(done) {
639 selectText: "BEET - Beetlecoin",
640 firstAddress: "BVmtbEsGrjpknprmpHFq26z4kYHJUFHE71",
642 testNetwork(done, params);
644 it('Allows selection
of Adcoin
', function(done) {
646 selectText: "ACC - Adcoin",
647 firstAddress: "AcEDM6V5sF4kFHC76MJjjfProtS5Sw2qcd",
649 testNetwork(done, params);
651 it('Allows selection
of Asiacoin
', function(done) {
653 selectText: "AC - Asiacoin",
654 firstAddress: "ALupuEEz7kJjQTAvmtcBMBVuEjPa7GqZzE",
656 testNetwork(done, params);
658 it('Allows selection
of Auroracoin
', function(done) {
660 selectText: "AUR - Auroracoin",
661 firstAddress: "ANuraS6F4Jpi413FEnavjYkKYJJRHkgYCm",
663 testNetwork(done, params);
665 it('Allows selection
of Bata
', function(done) {
667 selectText: "BTA - Bata",
668 firstAddress: "BGxBdNeYPtF3GCuTtZBPQdFxCkdBYSF3fj",
670 testNetwork(done, params);
672 it('Allows selection
of Belacoin
', function(done) {
674 selectText: "BELA - Belacoin",
675 firstAddress: "BEeetqpNffdzeknSpNmQp5KAFh2KK1Qx7S",
677 testNetwork(done, params);
679 it('Allows selection
of Bitcoin Atom
', function(done) {
681 selectText: "BCA - Bitcoin Atom",
682 firstAddress: "AMy6qMbJeC4zsGRL6iWszmeCdQH65fgfih",
684 testNetwork(done, params);
686 it('Allows selection
of Bitcoinplus
', function(done) {
688 selectText: "XBC - Bitcoinplus",
689 firstAddress: "B7FSynZoDbEwTCSgsXq9nJ5ue8owYLVL8r",
691 testNetwork(done, params);
693 it('Allows selection
of Bitcoin Private
', function(done) {
695 selectText: "BTCP - Bitcoin Private",
696 firstAddress: "b1M3PbiXXyN6Hdivdw5rJv5VKpLjPzhm4jM",
698 testNetwork(done, params);
700 it('Allows selection
of Bitcoinz
', function(done) {
702 selectText: "BTCZ - Bitcoinz",
703 firstAddress: "t1X2YQoxs8cYRo2oaBYgVEwW5QNjCC59NYc",
705 testNetwork(done, params);
707 it('Allows selection
of Bitcore
', function(done) {
709 selectText: "BTX - Bitcore",
710 firstAddress: "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: "RtgRvXMBng1y51ftteveFqwNfyRG18HpxQ",
1062 testNetwork(done, params);
1064 it('Allows selection
of RevolutionVR
', function(done) {
1066 selectText: "RVR - RevolutionVR",
1067 firstAddress: "VXeeoP2jkzZnMFxtc66ZBZK1NHN5QJnnjL",
1069 testNetwork(done, params);
1071 it('Allows selection
of Rubycoin
', function(done) {
1073 selectText: "RBY - Rubycoin",
1074 firstAddress: "RV76JDtjTs11JdMDRToYn6CHecMRPLnKS6",
1076 testNetwork(done, params);
1078 it('Allows selection
of Smileycoin
', function(done) {
1080 selectText: "SMLY - Smileycoin",
1081 firstAddress: "BEZVnEBCAyFByrgKpwAgYgtvP4rKAd9Sj2",
1083 testNetwork(done, params);
1085 it('Allows selection
of Solarcoin
', function(done) {
1087 selectText: "SLR - Solarcoin",
1088 firstAddress: "8LZ13HbnjtaMJWSvvVFNTLf71zFfDrhwLu",
1090 testNetwork(done, params);
1092 it('Allows selection
of Stratis
', function(done) {
1094 selectText: "STRAT - Stratis",
1095 firstAddress: "ScfJnq3QDhKgDMEds6sqUE1ot6ShfhmXXq",
1097 testNetwork(done, params);
1099 it('Allows selection
of Stratis Test
', function(done) {
1101 selectText: "TSTRAT - Stratis Test",
1102 firstAddress: "TRLWm3dye4FRrDWouwYUSUZP96xb76mBE3",
1104 testNetwork(done, params);
1106 it('Allows selection
of Syscoin
', function(done) {
1108 selectText: "SYS - Syscoin",
1109 firstAddress: "SZwJi42Pst3VAMomyK5DG4157WM5ofRmSj",
1111 testNetwork(done, params);
1113 it('Allows selection
of Toa
', function(done) {
1115 selectText: "TOA - Toa",
1116 firstAddress: "TSe1QAnUwQzUfbBusDzRJ9URttrRGKoNKF",
1118 testNetwork(done, params);
1120 it('Allows selection
of Ultimatesecurecash
', function(done) {
1122 selectText: "USC - Ultimatesecurecash",
1123 firstAddress: "UPyLAZU2Che5fiy7Ed8xVJFmXAUhitA4ug",
1125 testNetwork(done, params);
1127 it('Allows selection
of Unobtanium
', function(done) {
1129 selectText: "UNO - Unobtanium",
1130 firstAddress: "uUBMPVMXrR6qhqornJqKTWgr8L69vihSL9",
1132 testNetwork(done, params);
1134 it('Allows selection
of Vcash
', function(done) {
1136 selectText: "XVC - Vcash",
1137 firstAddress: "VuL53MSY6KjvAjKSeRkh3NDnKykacDVeps",
1139 testNetwork(done, params);
1141 it('Allows selection
of Verge
', function(done) {
1143 selectText: "XVG - Verge",
1144 firstAddress: "DCrVuGkMjLJpTGgwAgv9AcMdeb1nkWbjZA",
1146 testNetwork(done, params);
1148 it('Allows selection
of Vertcoin
', function(done) {
1150 selectText: "VTC - Vertcoin",
1151 firstAddress: "Vf6koGuiWdXQfx8tNqxoNeEDxh4xh5cxsG",
1153 testNetwork(done, params);
1155 it('Allows selection
of Vivo
', function(done) {
1157 selectText: "VIVO - Vivo",
1158 firstAddress: "VFmBwuXXGhJe7MarQG2GfzHMFebRHgfSpB",
1160 testNetwork(done, params);
1162 it('Allows selection
of Vpncoin
', function(done) {
1164 selectText: "VASH - Vpncoin",
1165 firstAddress: "VoEmH1qXC4TsSgBAStR21QYetwnFqbqCx9",
1167 testNetwork(done, params);
1169 it('Allows selection
of Whitecoin
', function(done) {
1171 selectText: "XWC - Whitecoin",
1172 firstAddress: "WcSwCAUqrSgeSYbsaS3SSWWhsx8KRYTFDR",
1174 testNetwork(done, params);
1176 it('Allows selection
of Wincoin
', function(done) {
1178 selectText: "WC - Wincoin",
1179 firstAddress: "WaDVCESMGgyKgNESdn3u43NnwmGSkZED3Z",
1181 testNetwork(done, params);
1183 it('Allows selection
of Zcoin
', function(done) {
1185 selectText: "XZC - Zcoin",
1186 firstAddress: "a6VcMdP4XgAA9Tr7xNszmPG5FZpfRf17Cq",
1188 testNetwork(done, params);
1190 it('Allows selection
of Zcash
', function(done) {
1192 selectText: "ZEC - Zcash",
1193 firstAddress: "t1Sz8AneMcVuzUg3tPJ8et5AS5LFJ7K2EF9",
1195 testNetwork(done, params);
1197 it('Allows selection
of Zclassic
', function(done) {
1199 selectText: "ZCL - Zclassic",
1200 firstAddress: "t1TBMxTvVJRybUbMLGWq8H4A8F4VUL7czEc",
1202 testNetwork(done, params);
1204 it('Allows selection
of Zencash
', function(done) {
1206 selectText: "ZEN - Zencash",
1207 firstAddress: "znWh9XASyW2dZq5tck84wFjiwuqVysi7q3p",
1209 testNetwork(done, params);
1211 it('Allows selection
of Energi
', function(done) {
1213 selectText: "NRG - Energi",
1214 firstAddress: "EejRy4t4nidzhGGzkJUgFP3z4HYBjhTsRt",
1216 testNetwork(done, params);
1220 // BIP39 seed is set from phrase
1221 it('Sets the bip39 seed
from the prhase
', function(done) {
1222 driver.findElement(By.css('.phrase
'))
1223 .sendKeys('abandon abandon ability
');
1224 driver.sleep(generateDelay).then(function() {
1225 driver.findElement(By.css('.seed
'))
1226 .getAttribute("value")
1227 .then(function(seed) {
1228 expect(seed).toBe("20da140d3dd1df8713cefcc4d54ce0e445b4151027a1ab567b832f6da5fcc5afc1c3a3f199ab78b8e0ab4652efd7f414ac2c9a3b81bceb879a70f377aa0a58f3");
1234 // BIP32 root key is set from phrase
1235 it('Sets the bip39 root key
from the prhase
', function(done) {
1236 driver.findElement(By.css('.phrase
'))
1237 .sendKeys('abandon abandon ability
');
1238 driver.sleep(generateDelay).then(function() {
1239 driver.findElement(By.css('.root
-key
'))
1240 .getAttribute("value")
1241 .then(function(seed) {
1242 expect(seed).toBe("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
1248 // Tabs show correct addresses when changed
1249 it('Shows the correct address when tab is changed
', function(done) {
1250 driver.findElement(By.css('.phrase
'))
1251 .sendKeys('abandon abandon ability
');
1252 driver.sleep(generateDelay).then(function() {
1253 driver.findElement(By.css('#bip32
-tab a
'))
1255 driver.sleep(generateDelay).then(function() {
1256 getFirstAddress(function(address) {
1257 expect(address).toBe("17uQ7s2izWPwBmEVFikTmZUjbBKWYdJchz");
1264 // BIP44 derivation path is shown
1265 it('Shows the derivation path
for bip44 tab
', function(done) {
1266 driver.findElement(By.css('.phrase
'))
1267 .sendKeys('abandon abandon ability
');
1268 driver.sleep(generateDelay).then(function() {
1269 driver.findElement(By.css('#bip44
.path
'))
1270 .getAttribute("value")
1271 .then(function(path) {
1272 expect(path).toBe("m/44'/0'/0'/0");
1278 // BIP44 extended private key is shown
1279 it('Shows the extended private key for bip44 tab', function(done) {
1280 driver.findElement(By.css('.phrase'))
1281 .sendKeys('abandon abandon ability');
1282 driver.sleep(generateDelay).then(function() {
1283 driver.findElement(By.css('.extended-priv-key'))
1284 .getAttribute("value
")
1285 .then(function(path) {
1286 expect(path).toBe("xprvA2DxxvPZcyRvYgZMGS53nadR32mVDeCyqQYyFhrCVbJNjPoxMeVf7QT5g7mQASbTf9Kp4cryvcXnu2qurjWKcrdsr91jXymdCDNxKgLFKJG
");
1292 // BIP44 extended public key is shown
1293 it('Shows the extended public key for bip44 tab', function(done) {
1294 driver.findElement(By.css('.phrase'))
1295 .sendKeys('abandon abandon ability');
1296 driver.sleep(generateDelay).then(function() {
1297 driver.findElement(By.css('.extended-pub-key'))
1298 .getAttribute("value
")
1299 .then(function(path) {
1300 expect(path).toBe("xpub6FDKNRvTTLzDmAdpNTc49ia9b4byd6vqCdUa46Fp3vqMcC96uBoufCmZXQLiN5AK3iSCJMhf9gT2sxkpyaPepRuA7W3MujV5tGmF5VfbueM
");
1306 // BIP44 account field changes address list
1307 it('Changes the address list if bip44 account is changed', function(done) {
1308 driver.findElement(By.css('#bip44 .account'))
1310 driver.findElement(By.css('.phrase'))
1311 .sendKeys('abandon abandon ability');
1312 driver.sleep(generateDelay).then(function() {
1313 getFirstAddress(function(address) {
1314 expect(address).toBe("1Nq2Wmu726XHCuGhctEtGmhxo3wzk5wZ1H
");
1320 // BIP44 change field changes address list
1321 it('Changes the address list if bip44 change is changed', function(done) {
1322 driver.findElement(By.css('#bip44 .change'))
1324 driver.findElement(By.css('.phrase'))
1325 .sendKeys('abandon abandon ability');
1326 driver.sleep(generateDelay).then(function() {
1327 getFirstAddress(function(address) {
1328 expect(address).toBe("1KAGfWgqfVbSSXY56fNQ7YnhyKuoskHtYo
");
1334 // BIP32 derivation path can be set
1335 it('Can use a custom bip32 derivation path', function(done) {
1336 driver.findElement(By.css('#bip32-tab a'))
1338 driver.findElement(By.css('#bip32 .path'))
1340 driver.findElement(By.css('#bip32 .path'))
1342 driver.findElement(By.css('.phrase'))
1343 .sendKeys('abandon abandon ability');
1344 driver.sleep(generateDelay).then(function() {
1345 getFirstAddress(function(address) {
1346 expect(address).toBe("16pYQQdLD1hH4hwTGLXBaZ9Teboi1AGL8L
");
1352 // BIP32 can use hardened derivation paths
1353 it('Can use a hardened derivation paths', function(done) {
1354 driver.findElement(By.css('#bip32-tab a'))
1356 driver.findElement(By.css('#bip32 .path'))
1358 driver.findElement(By.css('#bip32 .path'))
1360 driver.findElement(By.css('.phrase
'))
1361 .sendKeys('abandon abandon ability
');
1362 driver.sleep(generateDelay).then(function() {
1363 getFirstAddress(function(address) {
1364 expect(address).toBe("14aXZeprXAE3UUKQc4ihvwBvww2LuEoHo4");
1370 // BIP32 extended private key is shown
1371 it('Shows the BIP32 extended
private key
', function(done) {
1372 driver.findElement(By.css('#bip32
-tab a
'))
1374 driver.findElement(By.css('.phrase
'))
1375 .sendKeys('abandon abandon ability
');
1376 driver.sleep(generateDelay).then(function() {
1377 driver.findElement(By.css('.extended
-priv
-key
'))
1378 .getAttribute("value")
1379 .then(function(privKey) {
1380 expect(privKey).toBe("xprv9va99uTVE5aLiutUVLTyfxfe8v8aaXjSQ1XxZbK6SezYVuikA9MnjQVTA8rQHpNA5LKvyQBpLiHbBQiiccKiBDs7eRmBogsvq3THFeLHYbe");
1386 // BIP32 extended public key is shown
1387 it('Shows the BIP32 extended
public key
', function(done) {
1388 driver.findElement(By.css('#bip32
-tab a
'))
1390 driver.findElement(By.css('.phrase
'))
1391 .sendKeys('abandon abandon ability
');
1392 driver.sleep(generateDelay).then(function() {
1393 driver.findElement(By.css('.extended
-pub
-key
'))
1394 .getAttribute("value")
1395 .then(function(pubKey) {
1396 expect(pubKey).toBe("xpub69ZVZQzP4T8dwPxwbMzz36cNgwy4yzTHmETZMyihzzXXNi3thgg3HCow1RtY252wdw5rS8369xKnraN5Q93y3FkFfJp2XEHWUrkyXsjS93P");
1402 // Derivation path is shown in table
1403 it('Shows the derivation path
in the table
', function(done) {
1404 driver.findElement(By.css('.phrase
'))
1405 .sendKeys('abandon abandon ability
');
1406 driver.sleep(generateDelay).then(function() {
1407 getFirstPath(function(path) {
1408 expect(path).toBe("m/44'/0'/0'/0/0");
1414 // Derivation path for address can be hardened
1415 it('Can derive hardened addresses', function(done) {
1416 driver.findElement(By.css('#bip32-tab a'))
1418 driver.executeScript(function() {
1419 $(".hardened
-addresses
").prop("checked
", true);
1421 driver.findElement(By.css('.phrase'))
1422 .sendKeys('abandon abandon ability');
1423 driver.sleep(generateDelay).then(function() {
1424 getFirstAddress(function(address) {
1425 expect(address).toBe("18exLzUv7kfpiXRzmCjFDoC9qwNLFyvwyd
");
1431 // Derivation path visibility can be toggled
1432 it('Can toggle visibility of the derivation path column', function(done) {
1433 driver.findElement(By.css('.phrase'))
1434 .sendKeys('abandon abandon ability');
1435 driver.sleep(generateDelay).then(function() {
1436 driver.findElement(By.css('.index-toggle'))
1438 testColumnValuesAreInvisible(done, "index
");
1443 it('Shows the address in the table', function(done) {
1444 driver.findElement(By.css('.phrase'))
1445 .sendKeys('abandon abandon ability');
1446 driver.sleep(generateDelay).then(function() {
1447 getFirstAddress(function(address) {
1448 expect(address).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug
");
1454 // Addresses are shown in order of derivation path
1455 it('Shows the address in order of derivation path', function(done) {
1456 driver.findElement(By.css('.phrase'))
1457 .sendKeys('abandon abandon ability');
1458 driver.sleep(generateDelay).then(function() {
1459 testRowsAreInCorrectOrder(done);
1463 // Address visibility can be toggled
1464 it('Can toggle visibility of the address column', function(done) {
1465 driver.findElement(By.css('.phrase'))
1466 .sendKeys('abandon abandon ability');
1467 driver.sleep(generateDelay).then(function() {
1468 driver.findElement(By.css('.address-toggle'))
1470 testColumnValuesAreInvisible(done, "address
");
1474 // Public key is shown in table
1475 it('Shows the public key in the table', function(done) {
1476 driver.findElement(By.css('.phrase'))
1477 .sendKeys('abandon abandon ability');
1478 driver.sleep(generateDelay).then(function() {
1479 driver.findElements(By.css('.pubkey'))
1480 .then(function(els) {
1482 .then(function(pubkey) {
1483 expect(pubkey).toBe("033f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3
");
1490 // Public key visibility can be toggled
1491 it('Can toggle visibility of the public key column', function(done) {
1492 driver.findElement(By.css('.phrase'))
1493 .sendKeys('abandon abandon ability');
1494 driver.sleep(generateDelay).then(function() {
1495 driver.findElement(By.css('.public-key-toggle'))
1497 testColumnValuesAreInvisible(done, "pubkey
");
1501 // Private key is shown in table
1502 it('Shows the private key in the table', function(done) {
1503 driver.findElement(By.css('.phrase'))
1504 .sendKeys('abandon abandon ability');
1505 driver.sleep(generateDelay).then(function() {
1506 driver.findElements(By.css('.privkey'))
1507 .then(function(els) {
1509 .then(function(pubkey) {
1510 expect(pubkey).toBe("L26cVSpWFkJ6aQkPkKmTzLqTdLJ923e6CzrVh9cmx21QHsoUmrEE
");
1517 // Private key visibility can be toggled
1518 it('Can toggle visibility of the private key column', function(done) {
1519 driver.findElement(By.css('.phrase'))
1520 .sendKeys('abandon abandon ability');
1521 driver.sleep(generateDelay).then(function() {
1522 driver.findElement(By.css('.private-key-toggle'))
1524 testColumnValuesAreInvisible(done, "privkey
");
1528 // More addresses can be generated
1529 it('Can generate more rows in the table', function(done) {
1530 driver.findElement(By.css('.phrase'))
1531 .sendKeys('abandon abandon ability');
1532 driver.sleep(generateDelay).then(function() {
1533 driver.findElement(By.css('.more'))
1535 driver.sleep(generateDelay).then(function() {
1536 driver.findElements(By.css('.address'))
1537 .then(function(els) {
1538 expect(els.length).toBe(40);
1545 // A custom number of additional addresses can be generated
1546 it('Can generate more rows in the table', function(done) {
1547 driver.findElement(By.css('.phrase'))
1548 .sendKeys('abandon abandon ability');
1549 driver.sleep(generateDelay).then(function() {
1550 driver.findElement(By.css('.rows-to-add'))
1552 driver.findElement(By.css('.rows-to-add'))
1554 driver.findElement(By.css('.more'))
1556 driver.sleep(generateDelay).then(function() {
1557 driver.findElements(By.css('.address'))
1558 .then(function(els) {
1559 expect(els.length).toBe(21);
1566 // Additional addresses are shown in order of derivation path
1567 it('Shows additional addresses in order of derivation path', function(done) {
1568 driver.findElement(By.css('.phrase'))
1569 .sendKeys('abandon abandon ability');
1570 driver.sleep(generateDelay).then(function() {
1571 driver.findElement(By.css('.more'))
1573 driver.sleep(generateDelay).then(function() {
1574 testRowsAreInCorrectOrder(done);
1579 // BIP32 root key can be set by the user
1580 it('Allows the user to set the BIP32 root key', function(done) {
1581 driver.findElement(By.css('.root-key'))
1582 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1583 driver.sleep(generateDelay).then(function() {
1584 getFirstAddress(function(address) {
1585 expect(address).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug
");
1591 // Setting BIP32 root key clears the existing phrase, passphrase and seed
1592 // TODO this doesn't work in selenium with chrome
1593 it('Confirms the existing phrase should be cleared', function(done) {
1594 if (browser == "chrome
") {
1595 pending("Selenium
+ Chrome headless bug
for alert
, see
https://stackoverflow.com/q/45242264");
1597 driver
.findElement(By
.css('.phrase'))
1598 .sendKeys('A non-blank but invalid value');
1599 driver
.findElement(By
.css('.root-key'))
1600 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1601 driver
.switchTo().alert().accept();
1602 driver
.findElement(By
.css('.phrase'))
1603 .getAttribute("value").then(function(value
) {
1604 expect(value
).toBe("");
1609 // Clearing of phrase, passphrase and seed can be cancelled by user
1610 // TODO this doesn't work in selenium with chrome
1611 it('Allows the clearing of the phrase to be cancelled', function(done
) {
1612 if (browser
== "chrome") {
1613 pending("Selenium + Chrome headless bug for alert, see https://stackoverflow.com/q/45242264");
1615 driver
.findElement(By
.css('.phrase'))
1616 .sendKeys('abandon abandon ability');
1617 driver
.sleep(generateDelay
).then(function() {
1618 driver
.findElement(By
.css('.root-key'))
1620 driver
.findElement(By
.css('.root-key'))
1622 driver
.switchTo().alert().dismiss();
1623 driver
.findElement(By
.css('.phrase'))
1624 .getAttribute("value").then(function(value
) {
1625 expect(value
).toBe("abandon abandon ability");
1631 // Custom BIP32 root key is used when changing the derivation path
1632 it('Can set derivation path for root key instead of phrase', function(done
) {
1633 driver
.findElement(By
.css('#bip44 .account'))
1635 driver
.findElement(By
.css('.root-key'))
1636 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1637 driver
.sleep(generateDelay
).then(function() {
1638 getFirstAddress(function(address
) {
1639 expect(address
).toBe("1Nq2Wmu726XHCuGhctEtGmhxo3wzk5wZ1H");
1645 // Incorrect mnemonic shows error
1646 it('Shows an error for incorrect mnemonic', function(done
) {
1647 driver
.findElement(By
.css('.phrase'))
1648 .sendKeys('abandon abandon abandon');
1649 driver
.sleep(feedbackDelay
).then(function() {
1650 driver
.findElement(By
.css('.feedback'))
1652 .then(function(feedback
) {
1653 expect(feedback
).toBe("Invalid mnemonic");
1659 // Incorrect word shows suggested replacement
1660 it('Shows word suggestion for incorrect word', function(done
) {
1661 driver
.findElement(By
.css('.phrase'))
1662 .sendKeys('abandon abandon abiliti');
1663 driver
.sleep(feedbackDelay
).then(function() {
1664 driver
.findElement(By
.css('.feedback'))
1666 .then(function(feedback
) {
1667 var msg
= "abiliti not in wordlist, did you mean ability?";
1668 expect(feedback
).toBe(msg
);
1674 // Github pull request 48
1675 // First four letters of word shows that word, not closest
1676 // since first four letters gives unique word in BIP39 wordlist
1677 // eg ille should show illegal, not idle
1678 it('Shows word suggestion based on first four chars', function(done
) {
1679 driver
.findElement(By
.css('.phrase'))
1681 driver
.sleep(feedbackDelay
).then(function() {
1682 driver
.findElement(By
.css('.feedback'))
1684 .then(function(feedback
) {
1685 var msg
= "ille not in wordlist, did you mean illegal?";
1686 expect(feedback
).toBe(msg
);
1692 // Incorrect BIP32 root key shows error
1693 it('Shows error for incorrect root key', function(done
) {
1694 driver
.findElement(By
.css('.root-key'))
1695 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpj');
1696 driver
.sleep(feedbackDelay
).then(function() {
1697 driver
.findElement(By
.css('.feedback'))
1699 .then(function(feedback
) {
1700 var msg
= "Invalid root key";
1701 expect(feedback
).toBe(msg
);
1707 // Derivation path not starting with m shows error
1708 it('Shows error for derivation path not starting with m', function(done
) {
1709 driver
.findElement(By
.css('#bip32-tab a'))
1711 driver
.findElement(By
.css('#bip32 .path'))
1713 driver
.findElement(By
.css('#bip32 .path'))
1715 driver
.findElement(By
.css('.phrase'))
1716 .sendKeys('abandon abandon ability');
1717 driver
.sleep(feedbackDelay
).then(function() {
1718 driver
.findElement(By
.css('.feedback'))
1720 .then(function(feedback
) {
1721 var msg
= "First character must be 'm'";
1722 expect(feedback
).toBe(msg
);
1728 // Derivation path containing invalid characters shows useful error
1729 it('Shows error for derivation path not starting with m', function(done
) {
1730 driver
.findElement(By
.css('#bip32-tab a'))
1732 driver
.findElement(By
.css('#bip32 .path'))
1734 driver
.findElement(By
.css('#bip32 .path'))
1735 .sendKeys('m/1/0wrong1/1');
1736 driver
.findElement(By
.css('.phrase'))
1737 .sendKeys('abandon abandon ability');
1738 driver
.sleep(feedbackDelay
).then(function() {
1739 driver
.findElement(By
.css('.feedback'))
1741 .then(function(feedback
) {
1742 var msg
= "Invalid characters 0wrong1 found at depth 2";
1743 expect(feedback
).toBe(msg
);
1749 // Github Issue 11: Default word length is 15
1750 // https://github.com/iancoleman/bip39/issues/11
1751 it('Sets the default word length to 15', function(done
) {
1752 driver
.findElement(By
.css('.strength'))
1753 .getAttribute("value")
1754 .then(function(strength
) {
1755 expect(strength
).toBe("15");
1760 // Github Issue 12: Generate more rows with private keys hidden
1761 // https://github.com/iancoleman/bip39/issues/12
1762 it('Sets the correct hidden column state on new rows', function(done
) {
1763 driver
.findElement(By
.css('.phrase'))
1764 .sendKeys("abandon abandon ability");
1765 driver
.sleep(generateDelay
).then(function() {
1766 driver
.findElement(By
.css('.private-key-toggle'))
1768 driver
.findElement(By
.css('.more'))
1770 driver
.sleep(generateDelay
).then(function() {
1771 driver
.findElements(By
.css('.privkey'))
1772 .then(function(els
) {
1773 expect(els
.length
).toBe(40);
1775 testColumnValuesAreInvisible(done
, "privkey");
1780 // Github Issue 19: Mnemonic is not sensitive to whitespace
1781 // https://github.com/iancoleman/bip39/issues/19
1782 it('Ignores excess whitespace in the mnemonic', function(done
) {
1783 var doublespace
= " ";
1784 var mnemonic
= "urge cat" + doublespace
+ "bid";
1785 driver
.findElement(By
.css('.phrase'))
1786 .sendKeys(mnemonic
);
1787 driver
.sleep(generateDelay
).then(function() {
1788 driver
.findElement(By
.css('.root-key'))
1789 .getAttribute("value")
1790 .then(function(seed
) {
1791 expect(seed
).toBe("xprv9s21ZrQH143K3isaZsWbKVoTtbvd34Y1ZGRugGdMeBGbM3AgBVzTH159mj1cbbtYSJtQr65w6L5xy5L9SFC7c9VJZWHxgAzpj4mun5LhrbC");
1797 // Github Issue 23: Part 1: Use correct derivation path when changing tabs
1798 // https://github.com/iancoleman/bip39/issues/23
1799 it('Uses the correct derivation path when changing tabs', function(done
) {
1800 // 1) and 2) set the phrase
1801 driver
.findElement(By
.css('.phrase'))
1802 .sendKeys("abandon abandon ability");
1803 driver
.sleep(generateDelay
).then(function() {
1804 // 3) select bip32 tab
1805 driver
.findElement(By
.css('#bip32-tab a'))
1807 driver
.sleep(generateDelay
).then(function() {
1808 // 4) switch from bitcoin to litecoin
1809 selectNetwork("LTC - Litecoin");
1810 driver
.sleep(generateDelay
).then(function() {
1811 // 5) Check address is displayed correctly
1812 getFirstAddress(function(address
) {
1813 expect(address
).toBe("LS8MP5LZ5AdzSZveRrjm3aYVoPgnfFh5T5");
1814 // 5) Check derivation path is displayed correctly
1815 getFirstPath(function(path
) {
1816 expect(path
).toBe("m/0/0");
1825 // Github Issue 23 Part 2: Coin selection in derivation path
1826 // https://github.com/iancoleman/bip39/issues/23#issuecomment-238011920
1827 it('Uses the correct derivation path when changing coins', function(done
) {
1829 driver
.findElement(By
.css('.phrase'))
1830 .sendKeys("abandon abandon ability");
1831 driver
.sleep(generateDelay
).then(function() {
1832 // switch from bitcoin to clam
1833 selectNetwork("CLAM - Clams");
1834 driver
.sleep(generateDelay
).then(function() {
1835 // check derivation path is displayed correctly
1836 getFirstPath(function(path
) {
1837 expect(path
).toBe("m/44'/23'/0'/0/0");
1844 // Github Issue 26: When using a Root key derrived altcoins are incorrect
1845 // https://github.com/iancoleman/bip39/issues/26
1846 it('Uses the correct derivation for altcoins with root keys', function(done
) {
1847 // 1) 2) and 3) set the root key
1848 driver
.findElement(By
.css('.root-key'))
1849 .sendKeys("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
1850 driver
.sleep(generateDelay
).then(function() {
1851 // 4) switch from bitcoin to viacoin
1852 selectNetwork("VIA - Viacoin");
1853 driver
.sleep(generateDelay
).then(function() {
1854 // 5) ensure the derived address is correct
1855 getFirstAddress(function(address
) {
1856 expect(address
).toBe("Vq9Eq4N5SQnjqZvxtxzo7hZPW5XnyJsmXT");
1863 // Selecting a language with no existing phrase should generate a phrase in
1865 it('Generate a random phrase when language is selected and no current phrase', function(done
) {
1866 driver
.findElement(By
.css("a[href='#japanese']"))
1868 driver
.sleep(generateDelay
).then(function() {
1869 driver
.findElement(By
.css(".phrase"))
1870 .getAttribute("value").then(function(phrase
) {
1871 expect(phrase
.search(/[a-z]/)).toBe(-1);
1872 expect(phrase
.length
).toBeGreaterThan(0);
1878 // Selecting a language with existing phrase should update the phrase to use
1880 it('Updates existing phrases when the language is changed', function(done
) {
1881 driver
.findElement(By
.css(".phrase"))
1882 .sendKeys("abandon abandon ability");
1883 driver
.sleep(generateDelay
).then(function() {
1884 driver
.findElement(By
.css("a[href='#italian']"))
1886 driver
.sleep(generateDelay
).then(function() {
1887 driver
.findElement(By
.css(".phrase"))
1888 .getAttribute("value").then(function(phrase
) {
1889 // Check only the language changes, not the phrase
1890 expect(phrase
).toBe("abaco abaco abbaglio");
1891 getFirstAddress(function(address
) {
1892 // Check the address is correct
1893 expect(address
).toBe("1Dz5TgDhdki9spa6xbPFbBqv5sjMrx3xgV");
1901 // Suggested replacement for erroneous word in non-English language
1902 it('Shows word suggestion for incorrect word in non-English language', function(done
) {
1903 driver
.findElement(By
.css('.phrase'))
1904 .sendKeys('abaco abaco zbbaglio');
1905 driver
.sleep(feedbackDelay
).then(function() {
1906 driver
.findElement(By
.css('.feedback'))
1908 .then(function(feedback
) {
1909 var msg
= "zbbaglio not in wordlist, did you mean abbaglio?";
1910 expect(feedback
).toBe(msg
);
1916 // Japanese word does not break across lines.
1918 // https://github.com/bitcoin/bips/blob/master/bip-0039/bip-0039-wordlists.md#japanese
1919 it('Does not break Japanese words across lines', function(done
) {
1920 driver
.findElement(By
.css('.phrase'))
1921 .getCssValue("word-break")
1922 .then(function(value
) {
1923 expect(value
).toBe("keep-all");
1928 // Language can be specified at page load using hash value in url
1929 it('Can set the language from the url hash', function(done
) {
1930 driver
.get(url
+ "#japanese").then(function() {
1931 driver
.findElement(By
.css('.generate')).click();
1932 driver
.sleep(generateDelay
).then(function() {
1933 driver
.findElement(By
.css(".phrase"))
1934 .getAttribute("value").then(function(phrase
) {
1935 expect(phrase
.search(/[a-z]/)).toBe(-1);
1936 expect(phrase
.length
).toBeGreaterThan(0);
1943 // Entropy can be entered by the user
1944 it('Allows entropy to be entered', function(done
) {
1945 driver
.findElement(By
.css('.use-entropy'))
1947 driver
.findElement(By
.css('.entropy'))
1948 .sendKeys('00000000 00000000 00000000 00000000');
1949 driver
.sleep(generateDelay
).then(function() {
1950 driver
.findElement(By
.css(".phrase"))
1951 .getAttribute("value").then(function(phrase
) {
1952 expect(phrase
).toBe("abandon abandon ability");
1953 getFirstAddress(function(address
) {
1954 expect(address
).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug");
1961 // A warning about entropy is shown to the user, with additional information
1962 it('Shows a warning about using entropy', function(done
) {
1963 driver
.findElement(By
.css('.use-entropy'))
1965 driver
.findElement(By
.css('.entropy-container'))
1967 .then(function(containerText
) {
1968 var warning
= "mnemonic may be insecure";
1969 expect(containerText
).toContain(warning
);
1970 driver
.findElement(By
.css('#entropy-notes'))
1971 .findElement(By
.xpath("parent::*"))
1973 .then(function(notesText
) {
1974 var detail
= "flipping a fair coin, rolling a fair dice, noise measurements etc";
1975 expect(notesText
).toContain(detail
);
1981 // The types of entropy available are described to the user
1982 it('Shows the types of entropy available', function(done
) {
1983 driver
.findElement(By
.css('.entropy'))
1984 .getAttribute("placeholder")
1985 .then(function(placeholderText
) {
1994 for (var i
=0; i
<options
.length
; i
++) {
1995 var option
= options
[i
];
1996 expect(placeholderText
).toContain(option
);
2002 // The actual entropy used is shown to the user
2003 it('Shows the actual entropy used', function(done
) {
2004 driver
.findElement(By
.css('.use-entropy'))
2006 driver
.findElement(By
.css('.entropy'))
2007 .sendKeys('Not A Very Good Entropy Source At All');
2008 driver
.sleep(generateDelay
).then(function() {
2009 driver
.findElement(By
.css('.entropy-container'))
2011 .then(function(text
) {
2012 expect(text
).toMatch(/Filtered Entropy
\s
+AedEceAA
/);
2018 // Binary entropy can be entered
2019 it('Allows binary entropy to be entered', function(done
) {
2020 testEntropyType(done
, "01", "binary");
2023 // Base 6 entropy can be entered
2024 it('Allows base 6 entropy to be entered', function(done
) {
2025 testEntropyType(done
, "012345", "base 6");
2028 // Base 6 dice entropy can be entered
2029 it('Allows base 6 dice entropy to be entered', function(done
) {
2030 testEntropyType(done
, "123456", "base 6 (dice)");
2033 // Base 10 entropy can be entered
2034 it('Allows base 10 entropy to be entered', function(done
) {
2035 testEntropyType(done
, "789", "base 10");
2038 // Hexadecimal entropy can be entered
2039 it('Allows hexadecimal entropy to be entered', function(done
) {
2040 testEntropyType(done
, "abcdef", "hexadecimal");
2043 // Dice entropy value is shown as the converted base 6 value
2044 // ie 123456 is converted to 123450
2045 it('Shows dice entropy as base 6', function(done
) {
2046 driver
.findElement(By
.css('.use-entropy'))
2048 driver
.findElement(By
.css('.entropy'))
2049 .sendKeys("123456");
2050 driver
.sleep(generateDelay
).then(function() {
2051 driver
.findElement(By
.css('.entropy-container'))
2053 .then(function(text
) {
2054 expect(text
).toMatch(/Filtered Entropy
\s
+123450/);
2060 // The number of bits of entropy accumulated is shown
2061 it("Shows the number of bits of entropy for 20 bits of binary", function(done
) {
2062 testEntropyBits(done
, "0000 0000 0000 0000 0000", "20");
2064 it("Shows the number of bits of entropy for 1 bit of binary", function(done
) {
2065 testEntropyBits(done
, "0", "1");
2067 it("Shows the number of bits of entropy for 4 bits of binary", function(done
) {
2068 testEntropyBits(done
, "0000", "4");
2070 it("Shows the number of bits of entropy for 1 character of base 6 (dice)", function(done
) {
2071 // 6 in card is 0 in base 6, 0 in base 6 is 2.6 bits (rounded down to 2 bits)
2072 testEntropyBits(done
, "6", "2");
2074 it("Shows the number of bits of entropy for 1 character of base 10 with 3 bits", function(done
) {
2075 // 7 in base 10 is 111 in base 2, no leading zeros
2076 testEntropyBits(done
, "7", "3");
2078 it("Shows the number of bits of entropy for 1 character of base 10 with 4 bis", function(done
) {
2079 testEntropyBits(done
, "8", "4");
2081 it("Shows the number of bits of entropy for 1 character of hex", function(done
) {
2082 testEntropyBits(done
, "F", "4");
2084 it("Shows the number of bits of entropy for 2 characters of base 10", function(done
) {
2085 testEntropyBits(done
, "29", "6");
2087 it("Shows the number of bits of entropy for 2 characters of hex", function(done
) {
2088 testEntropyBits(done
, "0A", "8");
2090 it("Shows the number of bits of entropy for 2 characters of hex with 3 leading zeros", function(done
) {
2091 // hex is always multiple of 4 bits of entropy
2092 testEntropyBits(done
, "1A", "8");
2094 it("Shows the number of bits of entropy for 2 characters of hex with 2 leading zeros", function(done
) {
2095 testEntropyBits(done
, "2A", "8");
2097 it("Shows the number of bits of entropy for 2 characters of hex with 1 leading zero", function(done
) {
2098 testEntropyBits(done
, "4A", "8");
2100 it("Shows the number of bits of entropy for 2 characters of hex with no leading zeros", function(done
) {
2101 testEntropyBits(done
, "8A", "8");
2103 it("Shows the number of bits of entropy for 2 characters of hex starting with F", function(done
) {
2104 testEntropyBits(done
, "FA", "8");
2106 it("Shows the number of bits of entropy for 4 characters of hex with leading zeros", function(done
) {
2107 testEntropyBits(done
, "000A", "16");
2109 it("Shows the number of bits of entropy for 4 characters of base 6", function(done
) {
2110 testEntropyBits(done
, "5555", "11");
2112 it("Shows the number of bits of entropy for 4 characters of base 6 dice", function(done
) {
2113 // uses dice, so entropy is actually 0000 in base 6, which is 4 lots of
2114 // 2.58 bits, which is 10.32 bits (rounded down to 10 bits)
2115 testEntropyBits(done
, "6666", "10");
2117 it("Shows the number of bits of entropy for 4 charactes of base 10", function(done
) {
2118 // Uses base 10, which is 4 lots of 3.32 bits, which is 13.3 bits (rounded
2120 testEntropyBits(done
, "2227", "13");
2122 it("Shows the number of bits of entropy for 4 characters of hex with 2 leading zeros", function(done
) {
2123 testEntropyBits(done
, "222F", "16");
2125 it("Shows the number of bits of entropy for 4 characters of hex starting with F", function(done
) {
2126 testEntropyBits(done
, "FFFF", "16");
2128 it("Shows the number of bits of entropy for 10 characters of base 10", function(done
) {
2129 // 10 events at 3.32 bits per event
2130 testEntropyBits(done
, "0000101017", "33");
2132 it("Shows the number of bits of entropy for a full deck of cards", function(done
) {
2133 // cards are not replaced, so a full deck is not 52^52 entropy which is 296
2134 // bits, it's 52!, which is 225 bits
2135 testEntropyBits(done
, "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks", "225");
2138 it("Shows details about the entered entropy", function(done
) {
2139 testEntropyFeedback(done
,
2143 type: "hexadecimal",
2147 strength: "less than a second",
2151 it("Shows details about the entered entropy", function(done
) {
2152 testEntropyFeedback(done
,
2154 entropy: "AAAAAAAA",
2155 filtered: "AAAAAAAA",
2156 type: "hexadecimal",
2160 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2164 it("Shows details about the entered entropy", function(done
) {
2165 testEntropyFeedback(done
,
2167 entropy: "AAAAAAAA B",
2168 filtered: "AAAAAAAAB",
2169 type: "hexadecimal",
2173 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2177 it("Shows details about the entered entropy", function(done
) {
2178 testEntropyFeedback(done
,
2180 entropy: "AAAAAAAA BBBBBBBB",
2181 filtered: "AAAAAAAABBBBBBBB",
2182 type: "hexadecimal",
2186 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2190 it("Shows details about the entered entropy", function(done
) {
2191 testEntropyFeedback(done
,
2193 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC",
2194 filtered: "AAAAAAAABBBBBBBBCCCCCCCC",
2195 type: "hexadecimal",
2199 strength: "less than a second",
2203 it("Shows details about the entered entropy", function(done
) {
2204 testEntropyFeedback(done
,
2206 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDD",
2207 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDD",
2208 type: "hexadecimal",
2212 strength: "2 minutes",
2216 it("Shows details about the entered entropy", function(done
) {
2217 testEntropyFeedback(done
,
2219 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA",
2220 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDA",
2221 type: "hexadecimal",
2229 it("Shows details about the entered entropy", function(done
) {
2230 testEntropyFeedback(done
,
2232 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA EEEEEEEE",
2233 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDAEEEEEEEE",
2234 type: "hexadecimal",
2238 strength: "3 years",
2242 it("Shows details about the entered entropy", function(done
) {
2243 testEntropyFeedback(done
,
2245 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA EEEEEEEE FFFFFFFF",
2246 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDAEEEEEEEEFFFFFFFF",
2247 type: "hexadecimal",
2251 strength: "centuries",
2255 it("Shows details about the entered entropy", function(done
) {
2256 testEntropyFeedback(done
,
2263 strength: "less than a second",
2267 it("Shows details about the entered entropy", function(done
) {
2268 testEntropyFeedback(done
,
2270 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2271 type: "card (full deck)",
2275 strength: "centuries",
2279 it("Shows details about the entered entropy", function(done
) {
2280 testEntropyFeedback(done
,
2282 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks3d",
2283 type: "card (full deck, 1 duplicate: 3d)",
2287 strength: "centuries",
2291 it("Shows details about the entered entropy", function(done
) {
2292 testEntropyFeedback(done
,
2294 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqs3d4d",
2295 type: "card (2 duplicates: 3d 4d, 1 missing: KS)",
2299 strength: "centuries",
2303 it("Shows details about the entered entropy", function(done
) {
2304 testEntropyFeedback(done
,
2306 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqs3d4d5d6d",
2307 type: "card (4 duplicates: 3d 4d 5d..., 1 missing: KS)",
2311 strength: "centuries",
2315 it("Shows details about the entered entropy", function(done
) {
2316 testEntropyFeedback(done
,
2317 // Next test was throwing uncaught error in zxcvbn
2318 // Also tests 451 bits, ie Math.log2(52!)*2 = 225.58 * 2
2320 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsksac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2321 type: "card (full deck, 52 duplicates: ac 2c 3c...)",
2325 strength: "centuries",
2329 it("Shows details about the entered entropy", function(done
) {
2330 testEntropyFeedback(done
,
2331 // Case insensitivity to duplicate cards
2334 type: "card (1 duplicate: AS)",
2338 strength: "less than a second",
2342 it("Shows details about the entered entropy", function(done
) {
2343 testEntropyFeedback(done
,
2346 type: "card (1 duplicate: as)",
2350 strength: "less than a second",
2354 it("Shows details about the entered entropy", function(done
) {
2355 testEntropyFeedback(done
,
2356 // Missing cards are detected
2358 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2359 type: "card (1 missing: 9C)",
2363 strength: "centuries",
2367 it("Shows details about the entered entropy", function(done
) {
2368 testEntropyFeedback(done
,
2370 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2371 type: "card (2 missing: 9C 5D)",
2375 strength: "centuries",
2379 it("Shows details about the entered entropy", function(done
) {
2380 testEntropyFeedback(done
,
2382 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d7d8d9dtdjd kdah2h3h 5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2383 type: "card (4 missing: 9C 5D QD...)",
2387 strength: "centuries",
2391 it("Shows details about the entered entropy", function(done
) {
2392 testEntropyFeedback(done
,
2393 // More than six missing cards does not show message
2395 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d 8d9d jd kdah2h3h 5h6h7h8h9hthjhqhkh 2s3s4s5s6s7s8s9stsjsqsks",
2400 strength: "centuries",
2404 it("Shows details about the entered entropy", function(done
) {
2405 testEntropyFeedback(done
,
2406 // Multiple decks of cards increases bits per event
2411 bitsPerEvent: "4.34",
2415 it("Shows details about the entered entropy", function(done
) {
2416 testEntropyFeedback(done
,
2421 bitsPerEvent: "4.80",
2425 it("Shows details about the entered entropy", function(done
) {
2426 testEntropyFeedback(done
,
2431 bitsPerEvent: "5.01",
2435 it("Shows details about the entered entropy", function(done
) {
2436 testEntropyFeedback(done
,
2438 entropy: "3d3d3d3d",
2441 bitsPerEvent: "5.14",
2445 it("Shows details about the entered entropy", function(done
) {
2446 testEntropyFeedback(done
,
2448 entropy: "3d3d3d3d3d",
2451 bitsPerEvent: "5.22",
2455 it("Shows details about the entered entropy", function(done
) {
2456 testEntropyFeedback(done
,
2458 entropy: "3d3d3d3d3d3d",
2461 bitsPerEvent: "5.28",
2465 it("Shows details about the entered entropy", function(done
) {
2466 testEntropyFeedback(done
,
2468 entropy: "3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d",
2471 bitsPerEvent: "5.59",
2472 strength: 'less than a second - Repeats like "abcabcabc" are only slightly harder to guess than "abc"',
2477 // Entropy is truncated from the left
2478 it('Truncates entropy from the left', function(done
) {
2479 // Truncate from left means 0000 is removed from the start
2480 // which gives mnemonic 'avocado zoo zone'
2481 // not 1111 removed from the end
2482 // which gives the mnemonic 'abstract zoo zoo'
2483 var entropy
= "00000000 00000000 00000000 00000000";
2484 entropy
+= "11111111 11111111 11111111 1111"; // Missing last byte
2485 driver
.findElement(By
.css('.use-entropy'))
2487 driver
.findElement(By
.css('.entropy'))
2489 driver
.sleep(generateDelay
).then(function() {
2490 driver
.findElement(By
.css(".phrase"))
2491 .getAttribute("value").then(function(phrase
) {
2492 expect(phrase
).toBe("avocado zoo zone");
2498 // Very large entropy results in very long mnemonics
2499 it('Converts very long entropy to very long mnemonics', function(done
) {
2501 for (var i
=0; i
<33; i
++) {
2502 entropy
+= "AAAAAAAA"; // 3 words * 33 iterations = 99 words
2504 driver
.findElement(By
.css('.use-entropy'))
2506 driver
.findElement(By
.css('.entropy'))
2508 driver
.sleep(generateDelay
).then(function() {
2509 driver
.findElement(By
.css(".phrase"))
2510 .getAttribute("value").then(function(phrase
) {
2511 var wordCount
= phrase
.split(/\s+/g).length
;
2512 expect(wordCount
).toBe(99);
2518 // Is compatible with bip32jp entropy
2519 // https://bip32jp.github.io/english/index.html
2521 // Is incompatible with:
2523 it('Is compatible with bip32jp.github.io', function(done
) {
2524 var entropy
= "543210543210543210543210543210543210543210543210543210543210543210543210543210543210543210543210543";
2525 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";
2526 driver
.findElement(By
.css('.use-entropy'))
2528 driver
.findElement(By
.css('.entropy'))
2530 driver
.sleep(generateDelay
).then(function() {
2531 driver
.findElement(By
.css(".phrase"))
2532 .getAttribute("value").then(function(phrase
) {
2533 expect(phrase
).toBe(expectedPhrase
);
2539 // Blank entropy does not generate mnemonic or addresses
2540 it('Does not generate mnemonic for blank entropy', function(done
) {
2541 driver
.findElement(By
.css('.use-entropy'))
2543 driver
.findElement(By
.css('.entropy'))
2545 // check there is no mnemonic
2546 driver
.sleep(generateDelay
).then(function() {
2547 driver
.findElement(By
.css(".phrase"))
2548 .getAttribute("value").then(function(phrase
) {
2549 expect(phrase
).toBe("");
2550 // check there is no mnemonic
2551 driver
.findElements(By
.css(".address"))
2552 .then(function(addresses
) {
2553 expect(addresses
.length
).toBe(0);
2554 // Check the feedback says 'blank entropy'
2555 driver
.findElement(By
.css(".feedback"))
2557 .then(function(feedbackText
) {
2558 expect(feedbackText
).toBe("Blank entropy");
2566 // Mnemonic length can be selected even for weak entropy
2567 it('Allows selection of mnemonic length even for weak entropy', function(done
) {
2568 driver
.findElement(By
.css('.use-entropy'))
2570 driver
.executeScript(function() {
2571 $(".mnemonic-length").val("18").trigger("change");
2573 driver
.findElement(By
.css('.entropy'))
2574 .sendKeys("012345");
2575 driver
.sleep(generateDelay
).then(function() {
2576 driver
.findElement(By
.css(".phrase"))
2577 .getAttribute("value").then(function(phrase
) {
2578 var wordCount
= phrase
.split(/\s+/g).length
;
2579 expect(wordCount
).toBe(18);
2586 // https://github.com/iancoleman/bip39/issues/33
2587 // Final cards should contribute entropy
2588 it('Uses as much entropy as possible for the mnemonic', function(done
) {
2589 driver
.findElement(By
.css('.use-entropy'))
2591 driver
.findElement(By
.css('.entropy'))
2592 .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");
2593 driver
.sleep(generateDelay
).then(function() {
2595 driver
.findElement(By
.css(".phrase"))
2596 .getAttribute("value").then(function(originalPhrase
) {
2597 // Set the last 12 cards to be AS
2598 driver
.findElement(By
.css('.entropy'))
2600 driver
.findElement(By
.css('.entropy'))
2601 .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");
2602 driver
.sleep(generateDelay
).then(function() {
2604 driver
.findElement(By
.css(".phrase"))
2605 .getAttribute("value").then(function(newPhrase
) {
2606 expect(originalPhrase
).not
.toEqual(newPhrase
);
2615 // https://github.com/iancoleman/bip39/issues/35
2617 // TODO this doesn't work in selenium with firefox
2618 // see https://stackoverflow.com/q/40360223
2619 it('Shows a qr code on hover for the phrase', function(done
) {
2620 if (browser
== "firefox") {
2621 pending("Selenium + Firefox bug for mouseMove, see https://stackoverflow.com/q/40360223");
2623 // generate a random mnemonic
2624 var generateEl
= driver
.findElement(By
.css('.generate'));
2626 // toggle qr to show (hidden by default)
2627 var phraseEl
= driver
.findElement(By
.css(".phrase"));
2629 var rootKeyEl
= driver
.findElement(By
.css(".root-key"));
2630 driver
.sleep(generateDelay
).then(function() {
2631 // hover over the root key
2632 driver
.actions().mouseMove(rootKeyEl
).perform().then(function() {
2633 // check the qr code shows
2634 driver
.executeScript(function() {
2635 return $(".qr-container").find("canvas").length
> 0;
2637 .then(function(qrShowing
) {
2638 expect(qrShowing
).toBe(true);
2639 // hover away from the phrase
2640 driver
.actions().mouseMove(generateEl
).perform().then(function() {;
2641 // check the qr code hides
2642 driver
.executeScript(function() {
2643 return $(".qr-container").find("canvas").length
== 0;
2645 .then(function(qrHidden
) {
2646 expect(qrHidden
).toBe(true);
2655 // BIP44 account extendend private key is shown
2656 // github issue 37 - compatibility with electrum
2657 it('Shows the bip44 account extended private key', function(done
) {
2658 driver
.findElement(By
.css(".phrase"))
2659 .sendKeys("abandon abandon ability");
2660 driver
.sleep(generateDelay
).then(function() {
2661 driver
.findElement(By
.css("#bip44 .account-xprv"))
2662 .getAttribute("value")
2663 .then(function(xprv
) {
2664 expect(xprv
).toBe("xprv9yzrnt4zWVJUr1k2VxSPy9ettKz5PpeDMgaVG7UKedhqnw1tDkxP2UyYNhuNSumk2sLE5ctwKZs9vwjsq3e1vo9egCK6CzP87H2cVYXpfwQ");
2670 // BIP44 account extendend public key is shown
2671 // github issue 37 - compatibility with electrum
2672 it('Shows the bip44 account extended public key', function(done
) {
2673 driver
.findElement(By
.css(".phrase"))
2674 .sendKeys("abandon abandon ability");
2675 driver
.sleep(generateDelay
).then(function() {
2676 driver
.findElement(By
.css("#bip44 .account-xpub"))
2677 .getAttribute("value")
2678 .then(function(xprv
) {
2679 expect(xprv
).toBe("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2686 // BIP32 root key can be set as an xpub
2687 it('Generates addresses from xpub as bip32 root key', function(done
) {
2688 driver
.findElement(By
.css('#bip32-tab a'))
2690 // set xpub for account 0 of bip44 for 'abandon abandon ability'
2691 driver
.findElement(By
.css("#root-key"))
2692 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2693 driver
.sleep(generateDelay
).then(function() {
2694 // check the addresses are generated
2695 getFirstAddress(function(address
) {
2696 expect(address
).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug");
2697 // check the xprv key is not set
2698 driver
.findElement(By
.css(".extended-priv-key"))
2699 .getAttribute("value")
2700 .then(function(xprv
) {
2701 expect(xprv
).toBe("NA");
2702 // check the private key is not set
2703 driver
.findElements(By
.css(".privkey"))
2704 .then(function(els
) {
2707 .then(function(privkey
) {
2708 expect(xprv
).toBe("NA");
2718 // xpub for bip32 root key will not work with hardened derivation paths
2719 it('Shows error for hardened derivation paths with xpub root key', function(done
) {
2720 // set xpub for account 0 of bip44 for 'abandon abandon ability'
2721 driver
.findElement(By
.css("#root-key"))
2722 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2723 driver
.sleep(feedbackDelay
).then(function() {
2724 // Check feedback is correct
2725 driver
.findElement(By
.css('.feedback'))
2727 .then(function(feedback
) {
2728 var msg
= "Hardened derivation path is invalid with xpub key";
2729 expect(feedback
).toBe(msg
);
2730 // Check no addresses are shown
2731 driver
.findElements(By
.css('.addresses tr'))
2732 .then(function(rows
) {
2733 expect(rows
.length
).toBe(0);
2741 // no root key shows feedback
2742 it('Shows feedback for no root key', function(done
) {
2743 // set xpub for account 0 of bip44 for 'abandon abandon ability'
2744 driver
.findElement(By
.css('#bip32-tab a'))
2746 driver
.sleep(feedbackDelay
).then(function() {
2747 // Check feedback is correct
2748 driver
.findElement(By
.css('.feedback'))
2750 .then(function(feedback
) {
2751 expect(feedback
).toBe("Invalid root key");
2758 // display error switching tabs while addresses are generating
2759 it('Can change details while old addresses are still being generated', function(done
) {
2760 // Set to generate 199 more addresses.
2761 // This will take a long time allowing a new set of addresses to be
2762 // generated midway through this lot.
2763 // The newly generated addresses should not include any from the old set.
2764 // Any more than 199 will show an alert which needs to be accepted.
2765 driver
.findElement(By
.css('.rows-to-add'))
2767 driver
.findElement(By
.css('.rows-to-add'))
2770 driver
.findElement(By
.css('.phrase'))
2771 .sendKeys("abandon abandon ability");
2772 driver
.sleep(generateDelay
).then(function() {
2773 // change tabs which should cancel the previous generating
2774 driver
.findElement(By
.css('.rows-to-add'))
2776 driver
.findElement(By
.css('.rows-to-add'))
2778 driver
.findElement(By
.css('#bip32-tab a'))
2780 driver
.sleep(generateDelay
).then(function() {
2781 driver
.findElements(By
.css('.index'))
2782 .then(function(els
) {
2783 // check the derivation paths have the right quantity
2784 expect(els
.length
).toBe(20);
2785 // check the derivation paths are in order
2786 testRowsAreInCorrectOrder(done
);
2790 }, generateDelay
+ 5000);
2793 // padding for binary should give length with multiple of 256
2794 // hashed entropy 1111 is length 252, so requires 4 leading zeros
2795 // prior to issue 49 it would only generate 2 leading zeros, ie missing 2
2796 it('Pads hashed entropy with leading zeros', function(done
) {
2797 driver
.findElement(By
.css('.use-entropy'))
2799 driver
.executeScript(function() {
2800 $(".mnemonic-length").val("15").trigger("change");
2802 driver
.findElement(By
.css('.entropy'))
2804 driver
.sleep(generateDelay
).then(function() {
2805 driver
.findElement(By
.css('.phrase'))
2806 .getAttribute("value")
2807 .then(function(phrase
) {
2808 expect(phrase
).toBe("avocado valid quantum cross link predict excuse edit street able flame large galaxy ginger nuclear");
2814 // Github pull request 55
2815 // https://github.com/iancoleman/bip39/pull/55
2817 it('Can set the derivation path on bip32 tab for bitcoincore', function(done
) {
2818 testClientSelect(done
, {
2820 bip32path: "m/0'/0'",
2821 useHardenedAddresses: "true",
2824 it('Can set the derivation path on bip32 tab for multibit', function(done
) {
2825 testClientSelect(done
, {
2827 bip32path: "m/0'/0",
2828 useHardenedAddresses: null,
2833 // https://github.com/iancoleman/bip39/issues/58
2834 // bip32 derivation is correct, does not drop leading zeros
2836 // https://medium.com/@alexberegszaszi/why-do-my-bip32-wallets-disagree-6f3254cc5846
2837 it('Retains leading zeros for bip32 derivation', function(done
) {
2838 driver
.findElement(By
.css(".phrase"))
2839 .sendKeys("fruit wave dwarf banana earth journey tattoo true farm silk olive fence");
2840 driver
.findElement(By
.css(".passphrase"))
2841 .sendKeys("banana");
2842 driver
.sleep(generateDelay
).then(function() {
2843 getFirstAddress(function(address
) {
2844 // Note that bitcore generates an incorrect address
2845 // 13EuKhffWkBE2KUwcbkbELZb1MpzbimJ3Y
2846 // see the medium.com link above for more details
2847 expect(address
).toBe("17rxURoF96VhmkcEGCj5LNQkmN9HVhWb7F");
2854 // Japanese mnemonics generate incorrect bip32 seed
2855 // BIP39 seed is set from phrase
2856 it('Generates correct seed for Japanese mnemonics', function(done
) {
2857 driver
.findElement(By
.css(".phrase"))
2858 .sendKeys("あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あおぞら");
2859 driver
.findElement(By
.css(".passphrase"))
2860 .sendKeys("メートルガバヴァぱばぐゞちぢ十人十色");
2861 driver
.sleep(generateDelay
).then(function() {
2862 driver
.findElement(By
.css(".seed"))
2863 .getAttribute("value")
2864 .then(function(seed
) {
2865 expect(seed
).toBe("a262d6fb6122ecf45be09c50492b31f92e9beb7d9a845987a02cefda57a15f9c467a17872029a9e92299b5cbdf306e3a0ee620245cbd508959b6cb7ca637bd55");
2871 // BIP49 official test vectors
2872 // https://github.com/bitcoin/bips/blob/master/bip-0049.mediawiki#test-vectors
2873 it('Generates BIP49 addresses matching the official test vectors', function(done
) {
2874 driver
.findElement(By
.css('#bip49-tab a'))
2876 selectNetwork("BTC - Bitcoin Testnet");
2877 driver
.findElement(By
.css(".phrase"))
2878 .sendKeys("abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about");
2879 driver
.sleep(generateDelay
).then(function() {
2880 getFirstAddress(function(address
) {
2881 expect(address
).toBe("2Mww8dCYPUpKHofjgcXcBCEGmniw9CoaiD2");
2887 // BIP49 derivation path is shown
2888 it('Shows the bip49 derivation path', function(done
) {
2889 driver
.findElement(By
.css('#bip49-tab a'))
2891 driver
.findElement(By
.css(".phrase"))
2892 .sendKeys("abandon abandon ability");
2893 driver
.sleep(generateDelay
).then(function() {
2894 driver
.findElement(By
.css('#bip49 .path'))
2895 .getAttribute("value")
2896 .then(function(path
) {
2897 expect(path
).toBe("m/49'/0'/0'/0");
2903 // BIP49 extended private key is shown
2904 it('Shows the bip49 extended private key', function(done
) {
2905 driver
.findElement(By
.css('#bip49-tab a'))
2907 driver
.findElement(By
.css(".phrase"))
2908 .sendKeys("abandon abandon ability");
2909 driver
.sleep(generateDelay
).then(function() {
2910 driver
.findElement(By
.css('.extended-priv-key'))
2911 .getAttribute("value")
2912 .then(function(xprv
) {
2913 expect(xprv
).toBe("yprvALYB4DYRG6CzzVgzQZwwqjAA2wjBGC3iEd7KYYScpoDdmf75qMRWZWxoFcRXBJjgEXdFqJ9vDRGRLJQsrL22Su5jMbNFeM9vetaGVqy9Qy2");
2919 // BIP49 extended public key is shown
2920 it('Shows the bip49 extended public key', function(done
) {
2921 driver
.findElement(By
.css('#bip49-tab a'))
2923 driver
.findElement(By
.css(".phrase"))
2924 .sendKeys("abandon abandon ability");
2925 driver
.sleep(generateDelay
).then(function() {
2926 driver
.findElement(By
.css('.extended-pub-key'))
2927 .getAttribute("value")
2928 .then(function(xprv
) {
2929 expect(xprv
).toBe("ypub6ZXXTj5K6TmJCymTWbUxCs6tayZffemZbr2vLvrEP8kceTSENtjm7KHH6thvAKxVar9fGe8rgsPEX369zURLZ68b4f7Vexz7RuXsjQ69YDt");
2935 // BIP49 account field changes address list
2936 it('Can set the bip49 account field', function(done
) {
2937 driver
.findElement(By
.css('#bip49-tab a'))
2939 driver
.findElement(By
.css('#bip49 .account'))
2941 driver
.findElement(By
.css('#bip49 .account'))
2943 driver
.findElement(By
.css(".phrase"))
2944 .sendKeys("abandon abandon ability");
2945 driver
.sleep(generateDelay
).then(function() {
2946 getFirstAddress(function(address
) {
2947 expect(address
).toBe("381wg1GGN4rP88rNC9v7QWsiww63yLVPsn");
2953 // BIP49 change field changes address list
2954 it('Can set the bip49 change field', function(done
) {
2955 driver
.findElement(By
.css('#bip49-tab a'))
2957 driver
.findElement(By
.css('#bip49 .change'))
2959 driver
.findElement(By
.css('#bip49 .change'))
2961 driver
.findElement(By
.css(".phrase"))
2962 .sendKeys("abandon abandon ability");
2963 driver
.sleep(generateDelay
).then(function() {
2964 getFirstAddress(function(address
) {
2965 expect(address
).toBe("3PEM7MiKed5konBoN66PQhK8r3hjGhy9dT");
2971 // BIP49 account extendend private key is shown
2972 it('Shows the bip49 account extended private key', function(done
) {
2973 driver
.findElement(By
.css('#bip49-tab a'))
2975 driver
.findElement(By
.css(".phrase"))
2976 .sendKeys("abandon abandon ability");
2977 driver
.sleep(generateDelay
).then(function() {
2978 driver
.findElement(By
.css('#bip49 .account-xprv'))
2979 .getAttribute("value")
2980 .then(function(xprv
) {
2981 expect(xprv
).toBe("yprvAHtB1M5Wp675aLzFy9TJYK2mSsLkg6mcBRh5DZTR7L4EnYSmYPaL63KFA4ycg1PngW5LfkmejxzosCs17TKZMpRFKc3z5SJar6QAKaFcaZL");
2987 // BIP49 account extendend public key is shown
2988 it('Shows the bip49 account extended public key', function(done
) {
2989 driver
.findElement(By
.css('#bip49-tab a'))
2991 driver
.findElement(By
.css(".phrase"))
2992 .sendKeys("abandon abandon ability");
2993 driver
.sleep(generateDelay
).then(function() {
2994 driver
.findElement(By
.css('#bip49 .account-xpub'))
2995 .getAttribute("value")
2996 .then(function(xprv
) {
2997 expect(xprv
).toBe("ypub6WsXQrcQeTfNnq4j5AzJuSyVzuBF5ZVTYecg1ws2ffbDfLmv5vtadqdj1NgR6C6gufMpMfJpHxvb6JEQKvETVNWCRanNedfJtnTchZiJtsL");
3003 // Test selecting coin where bip49 is unavailable (eg CLAM)
3004 it('Shows an error on bip49 tab for coins without bip49', function(done
) {
3005 driver
.findElement(By
.css('#bip49-tab a'))
3007 driver
.findElement(By
.css(".phrase"))
3008 .sendKeys("abandon abandon ability");
3009 driver
.sleep(generateDelay
).then(function() {
3010 selectNetwork("CLAM - Clams");
3011 // bip49 available is hidden
3012 driver
.findElement(By
.css('#bip49 .available'))
3013 .getAttribute("class")
3014 .then(function(classes
) {
3015 expect(classes
).toContain("hidden");
3016 // bip49 unavailable is shown
3017 driver
.findElement(By
.css('#bip49 .unavailable'))
3018 .getAttribute("class")
3019 .then(function(classes
) {
3020 expect(classes
).not
.toContain("hidden");
3021 // check there are no addresses shown
3022 driver
.findElements(By
.css('.addresses tr'))
3023 .then(function(rows
) {
3024 expect(rows
.length
).toBe(0);
3025 // check the derived private key is blank
3026 driver
.findElement(By
.css('.extended-priv-key'))
3027 .getAttribute("value")
3028 .then(function(xprv
) {
3029 expect(xprv
).toBe('');
3030 // check the derived public key is blank
3031 driver
.findElement(By
.css('.extended-pub-key'))
3032 .getAttribute("value")
3033 .then(function(xpub
) {
3034 expect(xpub
).toBe('');
3045 // Cleared mnemonic and root key still allows addresses to be generated
3046 // https://github.com/iancoleman/bip39/issues/43
3047 it('Clears old root keys from memory when mnemonic is cleared', function(done
) {
3049 driver
.findElement(By
.css(".phrase"))
3050 .sendKeys("abandon abandon ability");
3051 driver
.sleep(generateDelay
).then(function() {
3052 // clear the mnemonic and root key
3053 // using selenium .clear() doesn't seem to trigger the 'input' event
3054 // so clear it using keys instead
3055 driver
.findElement(By
.css('.phrase'))
3056 .sendKeys(Key
.CONTROL
,"a");
3057 driver
.findElement(By
.css('.phrase'))
3058 .sendKeys(Key
.DELETE
);
3059 driver
.findElement(By
.css('.root-key'))
3060 .sendKeys(Key
.CONTROL
,"a");
3061 driver
.findElement(By
.css('.root-key'))
3062 .sendKeys(Key
.DELETE
);
3063 driver
.sleep(generateDelay
).then(function() {
3064 // try to generate more addresses
3065 driver
.findElement(By
.css('.more'))
3067 driver
.sleep(generateDelay
).then(function() {
3068 driver
.findElements(By
.css(".addresses tr"))
3069 .then(function(els
) {
3070 // check there are no addresses shown
3071 expect(els
.length
).toBe(0);
3080 // error trying to generate addresses from xpub with hardened derivation
3081 it('Shows error for hardened addresses with xpub root key', function(done
) {
3082 driver
.findElement(By
.css('#bip32-tab a'))
3084 driver
.executeScript(function() {
3085 $(".hardened-addresses").prop("checked", true);
3087 // set xpub for account 0 of bip44 for 'abandon abandon ability'
3088 driver
.findElement(By
.css("#root-key"))
3089 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
3090 driver
.sleep(feedbackDelay
).then(function() {
3091 // Check feedback is correct
3092 driver
.findElement(By
.css('.feedback'))
3094 .then(function(feedback
) {
3095 var msg
= "Hardened derivation path is invalid with xpub key";
3096 expect(feedback
).toBe(msg
);
3102 // Litecoin uses ltub by default, and can optionally be set to xprv
3104 // https://github.com/iancoleman/bip39/issues/96
3105 // Issue with extended keys on Litecoin
3106 it('Uses ltub by default for litecoin, but can be set to xprv', function(done
) {
3107 driver
.findElement(By
.css('.phrase'))
3108 .sendKeys("abandon abandon ability");
3109 selectNetwork("LTC - Litecoin");
3110 driver
.sleep(generateDelay
).then(function() {
3111 // check the extended key is generated correctly
3112 driver
.findElement(By
.css('.root-key'))
3113 .getAttribute("value")
3114 .then(function(rootKey
) {
3115 expect(rootKey
).toBe("Ltpv71G8qDifUiNesiPqf6h5V6eQ8ic77oxQiYtawiACjBEx3sTXNR2HGDGnHETYxESjqkMLFBkKhWVq67ey1B2MKQXannUqNy1RZVHbmrEjnEU");
3116 // set litecoin to use ltub
3117 driver
.executeScript(function() {
3118 $(".litecoin-use-ltub").prop("checked", false);
3119 $(".litecoin-use-ltub").trigger("change");
3121 driver
.sleep(generateDelay
).then(function() {
3122 driver
.findElement(By
.css('.root-key'))
3123 .getAttribute("value")
3124 .then(function(rootKey
) {
3125 expect(rootKey
).toBe("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
3134 // https://github.com/iancoleman/bip39/issues/99#issuecomment-327094159
3135 // "warn me emphatically when they have detected invalid input" to the entropy field
3136 // A warning is shown when entropy is filtered and discarded
3137 it('Warns when entropy is filtered and discarded', function(done
) {
3138 driver
.findElement(By
.css('.use-entropy'))
3140 // set entropy to have no filtered content
3141 driver
.findElement(By
.css('.entropy'))
3142 .sendKeys("00000000 00000000 00000000 00000000");
3143 driver
.sleep(generateDelay
).then(function() {
3144 // check the filter warning does not show
3145 driver
.findElement(By
.css('.entropy-container .filter-warning'))
3146 .getAttribute("class")
3147 .then(function(classes
) {
3148 expect(classes
).toContain("hidden");
3149 // set entropy to have some filtered content
3150 driver
.findElement(By
.css('.entropy'))
3151 .sendKeys("10000000 zxcvbn 00000000 00000000 00000000");
3152 driver
.sleep(entropyFeedbackDelay
).then(function() {
3153 // check the filter warning shows
3154 driver
.findElement(By
.css('.entropy-container .filter-warning'))
3155 .getAttribute("class")
3156 .then(function(classes
) {
3157 expect(classes
).not
.toContain("hidden");
3165 // Bitcoin Cash address can be set to use cashaddr format
3166 it('Can use cashaddr format for bitcoin cash addresses', function(done
) {
3167 driver
.executeScript(function() {
3168 $(".use-bch-cashaddr-addresses").prop("checked", true);
3170 driver
.findElement(By
.css('.phrase'))
3171 .sendKeys("abandon abandon ability");
3172 selectNetwork("BCH - Bitcoin Cash");
3173 driver
.sleep(generateDelay
).then(function() {
3174 getFirstAddress(function(address
) {
3175 expect(address
).toBe("bitcoincash:qzlquk7w4hkudxypl4fgv8x279r754dkvur7jpcsps");
3181 // Bitcoin Cash address can be set to use bitpay format
3182 it('Can use bitpay format for bitcoin cash addresses', function(done
) {
3183 driver
.executeScript(function() {
3184 $(".use-bch-bitpay-addresses").prop("checked", true);
3186 driver
.findElement(By
.css('.phrase'))
3187 .sendKeys("abandon abandon ability");
3188 selectNetwork("BCH - Bitcoin Cash");
3189 driver
.sleep(generateDelay
).then(function() {
3190 getFirstAddress(function(address
) {
3191 expect(address
).toBe("CZnpA9HPmvhuhLLPWJP8rNDpLUYXy1LXFk");
3197 // Bitcoin Cash address can be set to use legacy format
3198 it('Can use legacy format for bitcoin cash addresses', function(done
) {
3199 driver
.executeScript(function() {
3200 $(".use-bch-legacy-addresses").prop("checked", true);
3202 driver
.findElement(By
.css('.phrase'))
3203 .sendKeys("abandon abandon ability");
3204 selectNetwork("BCH - Bitcoin Cash");
3205 driver
.sleep(generateDelay
).then(function() {
3206 getFirstAddress(function(address
) {
3207 expect(address
).toBe("1JKvb6wKtsjNoCRxpZ4DGrbniML7z5U16A");
3213 // End of tests ported from old suit, so no more comments above each test now
3215 it('Can generate more addresses from a custom index', function(done
) {
3216 var expectedIndexes
= [
3217 0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,
3218 40,41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59
3220 driver
.findElement(By
.css('.phrase'))
3221 .sendKeys("abandon abandon ability");
3222 driver
.sleep(generateDelay
).then(function() {
3223 // Set start of next lot of rows to be from index 40
3224 // which means indexes 20-39 will not be in the table.
3225 driver
.findElement(By
.css('.more-rows-start-index'))
3227 driver
.findElement(By
.css('.more'))
3229 driver
.sleep(generateDelay
).then(function() {
3230 // Check actual indexes in the table match the expected pattern
3231 driver
.findElements(By
.css(".index"))
3232 .then(function(els
) {
3233 expect(els
.length
).toBe(expectedIndexes
.length
);
3234 var testRowAtIndex = function(i
) {
3235 if (i
>= expectedIndexes
.length
) {
3240 .then(function(actualPath
) {
3241 var noHardened
= actualPath
.replace(/'/g, "");
3242 var pathBits = noHardened.split("/")
3243 var lastBit = pathBits[pathBits.length-1];
3244 var actualIndex = parseInt(lastBit);
3245 var expectedIndex = expectedIndexes[i];
3246 expect(actualIndex).toBe(expectedIndex);
3247 testRowAtIndex(i+1);
3257 it('Can generate BIP141 addresses
with P2WPKH
-in-P2SH semanitcs
', function(done) {
3258 // Sourced from BIP49 official test specs
3259 driver.findElement(By.css('#bip141
-tab a
'))
3261 driver.findElement(By.css('.bip141
-path
'))
3263 driver.findElement(By.css('.bip141
-path
'))
3264 .sendKeys("m/49'/1'/0'/0");
3265 selectNetwork("BTC
- Bitcoin Testnet
");
3266 driver.findElement(By.css(".phrase
"))
3267 .sendKeys("abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about
");
3268 driver.sleep(generateDelay).then(function() {
3269 getFirstAddress(function(address) {
3270 expect(address).toBe("2Mww8dCYPUpKHofjgcXcBCEGmniw9CoaiD2
");
3276 it('Can generate BIP141 addresses with P2WPKH semanitcs', function(done) {
3277 // This result tested against bitcoinjs-lib test spec for segwit address
3278 // using the first private key of this mnemonic and default path m/0
3279 // https://github.com/bitcoinjs/bitcoinjs-lib/blob/9c8503cab0c6c30a95127042703bc18e8d28c76d/test/integration/addresses.js#L50
3280 // so whilst not directly comparable, substituting the private key produces
3281 // identical results between this tool and the bitcoinjs-lib test.
3282 // Private key generated is:
3283 // L3L8Nu9whawPBNLGtFqDhKut9DKKfG3CQoysupT7BimqVCZsLFNP
3284 driver.findElement(By.css('#bip141-tab a'))
3287 driver.executeScript(function() {
3288 $(".bip141
-semantics option
[selected
]").removeAttr("selected
");
3289 $(".bip141
-semantics option
").filter(function(i,e) {
3290 return $(e).html() == "P2WPKH
";
3291 }).prop("selected
", true);
3292 $(".bip141
-semantics
").trigger("change
");
3294 driver.findElement(By.css(".phrase
"))
3295 .sendKeys("abandon abandon ability
");
3296 driver.sleep(generateDelay).then(function() {
3297 getFirstAddress(function(address) {
3298 expect(address).toBe("bc1qfwu6a5a3evygrk8zvdxxvz4547lmpyx5vsfxe9
");
3304 it('Shows the entropy used by the PRNG when clicking generate', function(done) {
3305 driver.findElement(By.css('.generate')).click();
3306 driver.sleep(generateDelay).then(function() {
3307 driver.findElement(By.css('.entropy'))
3308 .getAttribute("value
")
3309 .then(function(entropy) {
3310 expect(entropy).not.toBe("");
3316 it('Shows the index of each word in the mnemonic', function(done) {
3317 driver.findElement(By.css('.phrase'))
3318 .sendKeys("abandon abandon ability
");
3319 driver.sleep(generateDelay).then(function() {
3320 driver.findElement(By.css('.use-entropy'))
3322 driver.findElement(By.css('.word-indexes'))
3324 .then(function(indexes) {
3325 expect(indexes).toBe("0, 0, 1");
3331 it('Shows the derivation path for bip84 tab', function(done) {
3332 driver.findElement(By.css('#bip84-tab a'))
3334 driver.findElement(By.css('.phrase'))
3335 .sendKeys('abandon abandon ability');
3336 driver.sleep(generateDelay).then(function() {
3337 driver.findElement(By.css('#bip84 .path'))
3338 .getAttribute("value
")
3339 .then(function(path) {
3340 expect(path).toBe("m
/84'/0'/0'/0");
3346 it('Shows the extended private key for bip84 tab', function(done) {
3347 driver.findElement(By.css('#bip84-tab a'))
3349 driver.findElement(By.css('.phrase'))
3350 .sendKeys('abandon abandon ability');
3351 driver.sleep(generateDelay).then(function() {
3352 driver.findElement(By.css('.extended-priv-key'))
3353 .getAttribute("value
")
3354 .then(function(path) {
3355 expect(path).toBe("zprvAev3RKrZ3QVKiUFCfdeMRen1BPDJgdNt1XpxiDy8acSs4kkAGTCvq7HeRYRNNpo8EtEjCFQBWavJwtCUR29y4TUCH4X5RXMcyq48uN8y9BP
");
3361 it('Shows the extended public key for bip84 tab', function(done) {
3362 driver.findElement(By.css('#bip84-tab a'))
3364 driver.findElement(By.css('.phrase'))
3365 .sendKeys('abandon abandon ability');
3366 driver.sleep(generateDelay).then(function() {
3367 driver.findElement(By.css('.extended-pub-key'))
3368 .getAttribute("value
")
3369 .then(function(path) {
3370 expect(path).toBe("zpub6suPpqPSsn3cvxKfmfBMnnijjR3o666jNkkZWcNk8wyqwZ5JozXBNuc8Gs7DB3uLwTDvGVTspVEAUQcEjKF3pZHgywVbubdTqbXTUg7usyx
");
3376 it('Changes the address list if bip84 account is changed', function(done) {
3377 driver.findElement(By.css('#bip84-tab a'))
3379 driver.findElement(By.css('#bip84 .account'))
3381 driver.findElement(By.css('.phrase'))
3382 .sendKeys('abandon abandon ability');
3383 driver.sleep(generateDelay).then(function() {
3384 getFirstAddress(function(address) {
3385 expect(address).toBe("bc1qp7vv669t2fy965jdzvqwrraana89ctd5ewc662
");
3391 it('Changes the address list if bip84 change is changed', function(done) {
3392 driver.findElement(By.css('#bip84-tab a'))
3394 driver.findElement(By.css('#bip84 .change'))
3396 driver.findElement(By.css('.phrase'))
3397 .sendKeys('abandon abandon ability');
3398 driver.sleep(generateDelay).then(function() {
3399 getFirstAddress(function(address) {
3400 expect(address).toBe("bc1qr39vj6rh06ff05m53uxq8uazehwhccswylhrs2
");
3406 it('Passes the official BIP84 test spec for rootpriv', function(done) {
3407 driver.findElement(By.css('#bip84-tab a'))
3409 driver.findElement(By.css('.phrase'))
3410 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3411 driver.sleep(generateDelay).then(function() {
3412 driver.findElement(By.css(".root
-key
"))
3413 .getAttribute("value
")
3414 .then(function(rootKey) {
3415 expect(rootKey).toBe("zprvAWgYBBk7JR8Gjrh4UJQ2uJdG1r3WNRRfURiABBE3RvMXYSrRJL62XuezvGdPvG6GFBZduosCc1YP5wixPox7zhZLfiUm8aunE96BBa4Kei5
");
3421 it('Passes the official BIP84 test spec for account 0 xprv', function(done) {
3422 driver.findElement(By.css('#bip84-tab a'))
3424 driver.findElement(By.css('.phrase'))
3425 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3426 driver.sleep(generateDelay).then(function() {
3427 driver.findElement(By.css("#bip84
.account
-xprv
"))
3428 .getAttribute("value
")
3429 .then(function(rootKey) {
3430 expect(rootKey).toBe("zprvAdG4iTXWBoARxkkzNpNh8r6Qag3irQB8PzEMkAFeTRXxHpbF9z4QgEvBRmfvqWvGp42t42nvgGpNgYSJA9iefm1yYNZKEm7z6qUWCroSQnE
");
3436 it('Passes the official BIP84 test spec for account 0 xpub', function(done) {
3437 driver.findElement(By.css('#bip84-tab a'))
3439 driver.findElement(By.css('.phrase'))
3440 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3441 driver.sleep(generateDelay).then(function() {
3442 driver.findElement(By.css("#bip84
.account
-xpub
"))
3443 .getAttribute("value
")
3444 .then(function(rootKey) {
3445 expect(rootKey).toBe("zpub6rFR7y4Q2AijBEqTUquhVz398htDFrtymD9xYYfG1m4wAcvPhXNfE3EfH1r1ADqtfSdVCToUG868RvUUkgDKf31mGDtKsAYz2oz2AGutZYs
");
3451 it('Passes the official BIP84 test spec for account 0 first address', function(done) {
3452 driver.findElement(By.css('#bip84-tab a'))
3454 driver.findElement(By.css('.phrase'))
3455 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3456 driver.sleep(generateDelay).then(function() {
3457 getFirstAddress(function(address) {
3458 expect(address).toBe("bc1qcr8te4kr609gcawutmrza0j4xv80jy8z306fyu
");
3464 it('Passes the official BIP84 test spec for account 0 first change address', function(done) {
3465 driver.findElement(By.css('#bip84-tab a'))
3467 driver.findElement(By.css('.phrase'))
3468 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3469 driver.findElement(By.css('#bip84 .change'))
3471 driver.sleep(generateDelay).then(function() {
3472 getFirstAddress(function(address) {
3473 expect(address).toBe("bc1q8c6fshw2dlwun7ekn9qwf37cu2rn755upcp6el
");
3479 it('Can display the table as csv', function(done) {
3480 var headings = "path
,address
,public key
,private key
";
3481 var row1 = "m
/44'/0'/0'/0/0,1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug
,033f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3
,L26cVSpWFkJ6aQkPkKmTzLqTdLJ923e6CzrVh9cmx21QHsoUmrEE
";
3482 var row20 = "m
/44'/0'/0'/0/19,1KhBy28XLAciXnnRvm71PvQJaETyrxGV55
,02b4b3e396434d8cdd20c03ac4aaa07387784d5d867b75987f516f5705ee68cb3a
,L4GrDrjReMsCAu5DkLXn79jSb95qR7Zfx7eshybCQZ1qL32MXJab
";
3483 driver.findElement(By.css('.phrase'))
3484 .sendKeys('abandon abandon ability');
3485 driver.sleep(generateDelay).then(function() {
3486 driver.findElement(By.css('.csv'))
3487 .getAttribute("value
")
3488 .then(function(csv) {
3489 expect(csv).toContain(headings);
3490 expect(csv).toContain(row1);
3491 expect(csv).toContain(row20);
3497 it('LeftPads ethereum keys that are less than 32 bytes', function(done) {
3498 // see https://github.com/iancoleman/bip39/issues/155
3499 selectNetwork("ETH
- Ethereum
");
3500 driver.findElement(By.css('#bip32-tab a'))
3502 driver.findElement(By.css('#bip32-path'))
3504 driver.findElement(By.css('#bip32-path'))
3505 .sendKeys("m
/44'/60'/0'");
3506 driver.findElement(By.css('.phrase'))
3507 .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');
3508 driver.sleep(generateDelay).then(function() {
3509 getFirstAddress(function(address) {
3510 expect(address).toBe("0x8943E785B4a5714FC87a3aFAad1eB1FeB602B118");
3516 it('Can encrypt private keys using BIP38', function(done) {
3517 // see https://github.com/iancoleman/bip39/issues/140
3518 driver.executeScript(function() {
3519 $(".use-bip38
").prop("checked
", true);
3521 driver.findElement(By.css('.bip38-password'))
3522 .sendKeys('bip38password');
3523 driver.findElement(By.css('.rows-to-add'))
3525 driver.findElement(By.css('.rows-to-add'))
3527 driver.findElement(By.css('.phrase'))
3528 .sendKeys('abandon abandon ability');
3529 driver.sleep(bip38delay).then(function() {
3531 getFirstRowValue(function(address) {
3532 expect(address).toBe("1NCvSdumA3ngMM9c4aqU56AM6rqXddfuXB
");
3534 getFirstRowValue(function(pubkey) {
3535 expect(pubkey).toBe("043f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3884a74447ea901729b1e73a999b7520e7cb55b4120e6432c64153ccab8a848e1
");
3537 getFirstRowValue(function(privkey) {
3538 expect(privkey).toBe("6PRNRiFnj1RoR3sXhymdCvoZCgnUHQpfupNdKkFbWJkwWQEKesWt1EDMDM
");
3544 }, bip38delay + 5000);
3546 it('Shows the checksum for the entropy', function(done) {
3547 driver.findElement(By.css('.use-entropy'))
3549 driver.findElement(By.css('.entropy'))
3550 .sendKeys("00000000000000000000000000000000");
3551 driver.sleep(generateDelay).then(function() {
3552 driver.findElement(By.css('.checksum'))
3554 .then(function(text) {
3555 expect(text).toBe("1");
3561 it('Shows the checksum for the entropy with the correct groupings', function(done) {
3562 driver.findElement(By.css('.use-entropy'))
3564 // create a checksum of 20 bits, which spans multiple words
3565 driver.findElement(By.css('.entropy'))
3566 .sendKeys("F000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
");
3567 driver.sleep(generateDelay).then(function() {
3568 driver.findElement(By.css('.checksum'))
3570 .then(function(text) {
3571 // first group is 9 bits, second group is 11
3572 expect(text).toBe("011010111 01110000110");
3578 it('Uses vprv for bitcoin testnet p2wpkh', function(done) {
3579 selectNetwork("BTC
- Bitcoin Testnet
");
3580 driver.findElement(By.css('#bip84-tab a'))
3582 driver.findElement(By.css('.phrase'))
3583 .sendKeys('abandon abandon ability');
3584 driver.sleep(generateDelay).then(function() {
3585 driver.findElement(By.css('.root-key'))
3586 .getAttribute("value
")
3587 .then(function(path) {
3588 expect(path).toBe("vprv9DMUxX4ShgxML9N2YV5CvWEebWrM9aJ5ULpbRRyzyWu6vs4BzTvbfFFrH41N5hVi7MYSfiugd765L3JmAfDM5po36Y8ouCKRDeYQwByCmS7
");
3594 it('Shows a warning if generating weak mnemonics', function(done) {
3595 driver.executeScript(function() {
3596 $(".strength option
[selected
]").removeAttr("selected
");
3597 $(".strength option
[value
=6]").prop("selected
", true);
3598 $(".strength
").trigger("change
");
3600 driver.findElement(By.css(".generate
-container
.warning
"))
3601 .getAttribute("class")
3602 .then(function(classes) {
3603 expect(classes).not.toContain("hidden
");
3608 it('Does not show a warning if generating strong mnemonics', function(done) {
3609 driver.executeScript(function() {
3610 $(".strength option
[selected
]").removeAttr("selected
");
3611 $(".strength option
[value
=12]").prop("selected
", true);
3613 driver.findElement(By.css(".generate
-container
.warning
"))
3614 .getAttribute("class")
3615 .then(function(classes) {
3616 expect(classes).toContain("hidden
");
3621 it('Shows a warning if overriding weak entropy with longer mnemonics', function(done) {
3622 driver.findElement(By.css('.use-entropy'))
3624 driver.findElement(By.css('.entropy'))
3625 .sendKeys("0123456789abcdef
"); // 6 words
3626 driver.executeScript(function() {
3627 $(".mnemonic
-length
").val("12").trigger("change
");
3629 driver.findElement(By.css(".weak
-entropy
-override
-warning
"))
3630 .getAttribute("class")
3631 .then(function(classes) {
3632 expect(classes).not.toContain("hidden
");
3637 it('Does not show a warning if entropy is stronger than mnemonic length', function(done) {
3638 driver.findElement(By.css('.use-entropy'))
3640 driver.findElement(By.css('.entropy'))
3641 .sendKeys("0123456789abcdef0123456789abcdef0123456789abcdef
"); // 18 words
3642 driver.executeScript(function() {
3643 $(".mnemonic
-length
").val("12").trigger("change
");
3645 driver.findElement(By.css(".weak
-entropy
-override
-warning
"))
3646 .getAttribute("class")
3647 .then(function(classes) {
3648 expect(classes).toContain("hidden
");
3653 it('Shows a warning for litecoin BIP84 (which does not have p2wpkh params)', function(done) {
3654 driver.findElement(By.css('.phrase'))
3655 .sendKeys('abandon abandon ability');
3656 selectNetwork("LTC
- Litecoin
");
3657 driver.findElement(By.css('#bip84-tab a'))
3659 // bip84 unavailable is shown
3660 driver.sleep(feedbackDelay).then(function() {
3661 driver.findElement(By.css('#bip84 .unavailable'))
3662 .getAttribute("class")
3663 .then(function(classes) {
3664 expect(classes).not.toContain("hidden
");
3670 it('Shows litecoin BIP49 addresses', function(done) {
3671 driver.findElement(By.css('.phrase'))
3672 .sendKeys('abandon abandon ability');
3673 selectNetwork("LTC
- Litecoin
");
3674 driver.findElement(By.css('#bip49-tab a'))
3676 // bip49 addresses are shown
3677 driver.sleep(generateDelay).then(function() {
3678 driver.findElement(By.css('#bip49 .available'))
3679 .getAttribute("class")
3680 .then(function(classes) {
3681 expect(classes).not.toContain("hidden
");
3682 // check first address
3683 getFirstAddress(function(address) {
3684 expect(address).toBe("MFwLPhsXoBuSLL8cLmW9uK6tChkzduV8qN
");
3691 it('Can use root keys to generate segwit table rows', function(done) {
3692 // segwit uses ypub / zpub instead of xpub but the root key should still
3693 // be valid regardless of the encoding used to import that key.
3694 // Maybe this breaks the reason for the different extended key prefixes, but
3695 // since the parsed root key is used behind the scenes anyhow this should be
3697 driver.findElement(By.css('#root-key'))
3698 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
3699 driver.findElement(By.css('#bip49-tab a'))
3701 // bip49 addresses are shown
3702 driver.sleep(generateDelay).then(function() {
3703 getFirstAddress(function(address) {
3704 expect(address).toBe("3QG2Y9AA4xZ846gKHZqNf7mvVKbLqMKxr2
");