]> git.immae.eu Git - perso/Immae/Projets/Cryptomonnaies/BIP39.git/blob - tests/spec/tests.js
Warn when generating low entropy mnemonics
[perso/Immae/Projets/Cryptomonnaies/BIP39.git] / tests / spec / tests.js
1 // Usage:
2 // cd /path/to/repo/tests
3 // jasmine spec/tests.js
4 //
5 // Dependencies:
6 // nodejs
7 // selenium
8 // jasmine
9 // see https://developer.mozilla.org/en-US/docs/Mozilla/Firefox/Headless_mode#Automated_testing_with_headless_mode
10
11 // USER SPECIFIED OPTIONS
12 var browser = process.env.BROWSER; //"firefox"; // or "chrome"
13 if (!browser) {
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");
18 browser = "chrome";
19 }
20 else {
21 console.log("Using browser: " + browser);
22 }
23
24 // Globals
25
26 var webdriver = require('selenium-webdriver');
27 var By = webdriver.By;
28 var Key = webdriver.Key;
29 var until = webdriver.until;
30 var newDriver = null;
31 var driver = null;
32 // Delays in ms
33 var generateDelay = 1500;
34 var feedbackDelay = 500;
35 var entropyFeedbackDelay = 500;
36 var bip38delay = 15000;
37
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";
49 }
50
51 // Variables dependent on specific browser selection
52
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))
61 .build();
62 }
63 }
64 else if (browser == "chrome") {
65 var chrome = require('selenium-webdriver/chrome');
66 newDriver = function() {
67 return new webdriver.Builder()
68 .forBrowser('chrome')
69 .setChromeOptions(new chrome.Options().addArguments("headless"))
70 .build();
71 }
72 }
73
74 // Helper functions
75
76 function testNetwork(done, params) {
77 var phrase = params.phrase || 'abandon abandon ability';
78 driver.findElement(By.css('.phrase'))
79 .sendKeys(phrase);
80 selectNetwork(params.selectText);
81 driver.sleep(generateDelay).then(function() {
82 getFirstAddress(function(address) {
83 expect(address).toBe(params.firstAddress);
84 done();
85 });
86 });
87 }
88
89 function getFirstRowValue(handler, selector) {
90 driver.findElements(By.css(selector))
91 .then(function(els) {
92 els[0].getText()
93 .then(handler);
94 })
95 }
96
97 function getFirstAddress(handler) {
98 getFirstRowValue(handler, ".address");
99 }
100
101 function getFirstPath(handler) {
102 getFirstRowValue(handler, ".index");
103 }
104
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");
112 done();
113 });
114 })
115 }
116
117 function testRowsAreInCorrectOrder(done) {
118 driver.findElements(By.css('.index'))
119 .then(function(els) {
120 var testRowAtIndex = function(i) {
121 if (i >= els.length) {
122 done();
123 }
124 else {
125 els[i].getText()
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);
132 testRowAtIndex(i+1);
133 });
134 }
135 }
136 testRowAtIndex(0);
137 });
138 }
139
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");
148 }, name);
149 }
150
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'))
156 .click();
157 driver.findElement(By.css('.entropy'))
158 .sendKeys(entropyText);
159 driver.sleep(generateDelay).then(function() {
160 driver.findElement(By.css('.entropy-container'))
161 .getText()
162 .then(function(text) {
163 var re = new RegExp("Entropy Type\\s+" + entropyType);
164 expect(text).toMatch(re);
165 done();
166 });
167 });
168 }
169
170 function testEntropyBits(done, entropyText, entropyBits) {
171 driver.findElement(By.css('.use-entropy'))
172 .click();
173 driver.findElement(By.css('.entropy'))
174 .sendKeys(entropyText);
175 driver.sleep(generateDelay).then(function() {
176 driver.findElement(By.css('.entropy-container'))
177 .getText()
178 .then(function(text) {
179 var re = new RegExp("Total Bits\\s+" + entropyBits);
180 expect(text).toMatch(re);
181 done();
182 });
183 });
184 }
185
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, "\\$&");
191 }
192 driver.findElement(By.css('.use-entropy'))
193 .click();
194 driver.findElement(By.css('.entropy'))
195 .sendKeys(entropyDetail.entropy);
196 driver.sleep(entropyFeedbackDelay).then(function() {
197 driver.findElement(By.css('.entropy-container'))
198 .getText()
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);
209 }
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);
216 }
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);
223 }
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);
230 }
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);
237 }
238 if ("words" in entropyDetail) {
239 var actualWords = phrase.split(/\s+/)
240 .filter(function(w) { return w.length > 0 })
241 .length;
242 expect(actualWords).toBe(entropyDetail.words);
243 }
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);
250 }
251 done();
252 });
253 });
254 });
255 }
256
257 function testClientSelect(done, params) {
258 // set mnemonic and select bip32 tab
259 driver.findElement(By.css('#bip32-tab a'))
260 .click()
261 driver.findElement(By.css('.phrase'))
262 .sendKeys("abandon abandon ability");
263 driver.sleep(generateDelay).then(function() {
264 // BITCOIN CORE
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");
285 done();
286 });
287 });
288 });
289 });
290 });
291 }
292
293 // Tests
294
295 describe('BIP39 Tool Tests', function() {
296
297 beforeEach(function(done) {
298 driver = newDriver();
299 driver.get(url).then(done);
300 });
301
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);
305 });
306
307 // BEGIN TESTS
308
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('');
314 done();
315 });
316 });
317
318 // Page has text
319 it('Should have text on the page', function(done) {
320 driver.findElement(By.css('body'))
321 .getText()
322 .then(function(text) {
323 var textToFind = "You can enter an existing BIP39 mnemonic";
324 expect(text).toContain(textToFind);
325 done();
326 });
327 });
328
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);
337 done();
338 })
339 });
340 });
341
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);
349 // press generate
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);
357 done();
358 });
359 });
360 });
361 });
362
363 // Mnemonic length can be customized
364 it('Should allow custom length mnemonics', function(done) {
365 // set strength to 6
366 driver.executeScript(function() {
367 $(".strength option[selected]").removeAttr("selected");
368 $(".strength option[value=6]").prop("selected", true);
369 });
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);
377 done();
378 });
379 });
380 });
381
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");
391 done();
392 })
393 });
394 });
395
396 // Network can be set to networks other than bitcoin
397 it('Allows selection of bitcoin testnet', function(done) {
398 var params = {
399 selectText: "BTC - Bitcoin Testnet",
400 firstAddress: "mucaU5iiDaJDb69BHLeDv8JFfGiyg2nJKi",
401 };
402 testNetwork(done, params);
403 });
404 it('Allows selection of litecoin', function(done) {
405 var params = {
406 selectText: "LTC - Litecoin",
407 firstAddress: "LQ4XU8RX2ULPmPq9FcUHdVmPVchP9nwXdn",
408 };
409 testNetwork(done, params);
410 });
411 it('Allows selection of ripple', function(done) {
412 var params = {
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",
416 };
417 testNetwork(done, params);
418 });
419 it('Allows selection of dogecoin', function(done) {
420 var params = {
421 selectText: "DOGE - Dogecoin",
422 firstAddress: "DPQH2AtuzkVSG6ovjKk4jbUmZ6iXLpgbJA",
423 };
424 testNetwork(done, params);
425 });
426 it('Allows selection of denarius', function(done) {
427 var params = {
428 selectText: "DNR - Denarius",
429 firstAddress: "DFdFMVUMzU9xX88EywXvAGwjiwpxyh9vKb",
430 };
431 testNetwork(done, params);
432 });
433 it('Allows selection of shadowcash', function(done) {
434 var params = {
435 selectText: "SDC - ShadowCash",
436 firstAddress: "SiSZtfYAXEFvMm3XM8hmtkGDyViRwErtCG",
437 };
438 testNetwork(done, params);
439 });
440 it('Allows selection of shadowcash testnet', function(done) {
441 var params = {
442 selectText: "SDC - ShadowCash Testnet",
443 firstAddress: "tM2EDpVKaTiEg2NZg3yKg8eqjLr55BErHe",
444 };
445 testNetwork(done, params);
446 });
447 it('Allows selection of viacoin', function(done) {
448 var params = {
449 selectText: "VIA - Viacoin",
450 firstAddress: "Vq9Eq4N5SQnjqZvxtxzo7hZPW5XnyJsmXT",
451 };
452 testNetwork(done, params);
453 });
454 it('Allows selection of viacoin testnet', function(done) {
455 var params = {
456 selectText: "VIA - Viacoin Testnet",
457 firstAddress: "tM2EDpVKaTiEg2NZg3yKg8eqjLr55BErHe",
458 };
459 testNetwork(done, params);
460 });
461 it('Allows selection of jumbucks', function(done) {
462 var params = {
463 selectText: "JBS - Jumbucks",
464 firstAddress: "JLEXccwDXADK4RxBPkRez7mqsHVoJBEUew",
465 };
466 testNetwork(done, params);
467 });
468 it('Allows selection of clam', function(done) {
469 var params = {
470 selectText: "CLAM - Clams",
471 firstAddress: "xCp4sakjVx4pUAZ6cBCtuin8Ddb6U1sk9y",
472 };
473 testNetwork(done, params);
474 });
475 it('Allows selection of crown', function(done) {
476 var params = {
477 selectText: "CRW - Crown",
478 firstAddress: "18pWSwSUAQdiwMHUfFZB1fM2xue9X1FqE5",
479 };
480 testNetwork(done, params);
481 });
482 it('Allows selection of dash', function(done) {
483 var params = {
484 selectText: "DASH - Dash",
485 firstAddress: "XdbhtMuGsPSkE6bPdNTHoFSszQKmK4S5LT",
486 };
487 testNetwork(done, params);
488 });
489 it('Allows selection of dash testnet', function(done) {
490 var params = {
491 selectText: "DASH - Dash Testnet",
492 firstAddress: "yaR52EN4oojdJfBgzWJTymC4uuCLPT29Gw",
493 };
494 testNetwork(done, params);
495 });
496 it('Allows selection of game', function(done) {
497 var params = {
498 selectText: "GAME - GameCredits",
499 firstAddress: "GSMY9bAp36cMR4zyT4uGVS7GFjpdXbao5Q",
500 };
501 testNetwork(done, params);
502 });
503 it('Allows selection of komodo', function(done) {
504 var params = {
505 selectText: "KMD - Komodo",
506 firstAddress: "RMPPzJwAjPVZZAwJvXivHJGGjdCx6WBD2t",
507 };
508 testNetwork(done, params);
509 });
510 it('Allows selection of namecoin', function(done) {
511 var params = {
512 selectText: "NMC - Namecoin",
513 firstAddress: "Mw2vK2Bvex1yYtYF6sfbEg2YGoUc98YUD2",
514 };
515 testNetwork(done, params);
516 });
517 it('Allows selection of onixcoin', function(done) {
518 var params = {
519 selectText: "ONX - Onixcoin",
520 firstAddress: "XGwMqddeKjT3ddgX73QokjVbCL3aK6Yxfk",
521 };
522 testNetwork(done, params);
523 });
524 it('Allows selection of peercoin', function(done) {
525 var params = {
526 selectText: "PPC - Peercoin",
527 firstAddress: "PVAiioTaK2eDHSEo3tppT9AVdBYqxRTBAm",
528 };
529 testNetwork(done, params);
530 });
531 it('Allows selection of ethereum', function(done) {
532 var params = {
533 selectText: "ETH - Ethereum",
534 firstAddress: "0xe5815d5902Ad612d49283DEdEc02100Bd44C2772",
535 };
536 testNetwork(done, params);
537 // TODO test private key and public key
538 });
539 it('Allows selection of slimcoin', function(done) {
540 var params = {
541 selectText: "SLM - Slimcoin",
542 firstAddress: "SNzPi1CafHFm3WWjRo43aMgiaEEj3ogjww",
543 };
544 testNetwork(done, params);
545 });
546 it('Allows selection of slimcoin testnet', function(done) {
547 var params = {
548 selectText: "SLM - Slimcoin Testnet",
549 firstAddress: "n3nMgWufTek5QQAr6uwMhg5xbzj8xqc4Dq",
550 };
551 testNetwork(done, params);
552 });
553 it('Allows selection of bitcoin cash', function(done) {
554 var params = {
555 selectText: "BCH - Bitcoin Cash",
556 firstAddress: "1JKvb6wKtsjNoCRxpZ4DGrbniML7z5U16A",
557 };
558 testNetwork(done, params);
559 });
560 it('Allows selection of myriadcoin', function(done) {
561 var params = {
562 selectText: "XMY - Myriadcoin",
563 firstAddress: "MJEswvRR46wh9BoiVj9DzKYMBkCramhoBV",
564 };
565 testNetwork(done, params);
566 });
567 it('Allows selection of pivx', function(done) {
568 var params = {
569 selectText: "PIVX - PIVX",
570 firstAddress: "DBxgT7faCuno7jmtKuu6KWCiwqsVPqh1tS",
571 };
572 testNetwork(done, params);
573 });
574 it('Allows selection of pivx testnet', function(done) {
575 var params = {
576 selectText: "PIVX - PIVX Testnet",
577 firstAddress: "yB5U384n6dGkVE3by5y9VdvHHPwPg68fQj",
578 };
579 testNetwork(done, params);
580 });
581 it('Allows selection of maza', function(done) {
582 var params = {
583 selectText: "MAZA - Maza",
584 firstAddress: "MGW4Bmi2NEm4PxSjgeFwhP9vg18JHoRnfw",
585 };
586 testNetwork(done, params);
587 });
588 it('Allows selection of fujicoin', function(done) {
589 var params = {
590 selectText: "FJC - Fujicoin",
591 firstAddress: "FgiaLpG7C99DyR4WnPxXedRVHXSfKzUDhF",
592 };
593 testNetwork(done, params);
594 });
595 it('Allows selection of nubits', function(done) {
596 var params = {
597 selectText: "USNBT - NuBits",
598 firstAddress: "BLxkabXuZSJSdesLD7KxZdqovd4YwyBTU6",
599 };
600 testNetwork(done, params);
601 });
602 it('Allows selection of bitcoin gold', function(done) {
603 var params = {
604 selectText: "BTG - Bitcoin Gold",
605 firstAddress: "GdDqug4WUsn5syNbSTHatNn4XnuwZtzedx",
606 };
607 testNetwork(done, params);
608 });
609 it('Allows selection of monacoin', function(done) {
610 var params = {
611 selectText: "MONA - Monacoin",
612 firstAddress: "MKMiMr7MyjDKjJbCBzgF6u4ByqTS4NkRB1",
613 };
614 testNetwork(done, params);
615 });
616 it('Allows selection of AXE', function(done) {
617 var params = {
618 selectText: "AXE - Axe",
619 firstAddress: "XQ4HLxUVS3egk5ff1o9e2vJFJKSSsUH3B7",
620 };
621 testNetwork(done, params);
622 });
623 it('Allows selection of BlackCoin', function(done) {
624 var params = {
625 selectText: "BLK - BlackCoin",
626 firstAddress: "B5MznAKwj7uQ42vDz3w4onhBXPcqhTwJ9z",
627 };
628 testNetwork(done, params);
629 });
630 it('Allows selection of Neblio', function(done) {
631 var params = {
632 selectText: "NEBL - Neblio",
633 firstAddress: "NefkeEEvhusbHMmTRrxx7H9wFnUXd8qQsE",
634 };
635 testNetwork(done, params);
636 });
637 it('Allows selection of Beetlecoin', function(done) {
638 var params = {
639 selectText: "BEET - Beetlecoin",
640 firstAddress: "BVmtbEsGrjpknprmpHFq26z4kYHJUFHE71",
641 };
642 testNetwork(done, params);
643 });
644 it('Allows selection of Adcoin', function(done) {
645 var params = {
646 selectText: "ACC - Adcoin",
647 firstAddress: "AcEDM6V5sF4kFHC76MJjjfProtS5Sw2qcd",
648 };
649 testNetwork(done, params);
650 });
651 it('Allows selection of Asiacoin', function(done) {
652 var params = {
653 selectText: "AC - Asiacoin",
654 firstAddress: "ALupuEEz7kJjQTAvmtcBMBVuEjPa7GqZzE",
655 };
656 testNetwork(done, params);
657 });
658 it('Allows selection of Auroracoin', function(done) {
659 var params = {
660 selectText: "AUR - Auroracoin",
661 firstAddress: "ANuraS6F4Jpi413FEnavjYkKYJJRHkgYCm",
662 };
663 testNetwork(done, params);
664 });
665 it('Allows selection of Bata', function(done) {
666 var params = {
667 selectText: "BTA - Bata",
668 firstAddress: "BGxBdNeYPtF3GCuTtZBPQdFxCkdBYSF3fj",
669 };
670 testNetwork(done, params);
671 });
672 it('Allows selection of Belacoin', function(done) {
673 var params = {
674 selectText: "BELA - Belacoin",
675 firstAddress: "BEeetqpNffdzeknSpNmQp5KAFh2KK1Qx7S",
676 };
677 testNetwork(done, params);
678 });
679 it('Allows selection of Bitcoin Atom', function(done) {
680 var params = {
681 selectText: "BCA - Bitcoin Atom",
682 firstAddress: "AMy6qMbJeC4zsGRL6iWszmeCdQH65fgfih",
683 };
684 testNetwork(done, params);
685 });
686 it('Allows selection of Bitcoinplus', function(done) {
687 var params = {
688 selectText: "XBC - Bitcoinplus",
689 firstAddress: "B7FSynZoDbEwTCSgsXq9nJ5ue8owYLVL8r",
690 };
691 testNetwork(done, params);
692 });
693 it('Allows selection of Bitcore', function(done) {
694 var params = {
695 selectText: "BTX - Bitcore",
696 firstAddress: "1Dg18EtqhReS11e9h8khkLjWGLHVjPM2AB",
697 };
698 testNetwork(done, params);
699 });
700 it('Allows selection of Bitsend', function(done) {
701 var params = {
702 selectText: "BSD - Bitsend",
703 firstAddress: "iBPk7LYjDun3EPk7CRR8UUmnPoceVc1bp2",
704 };
705 testNetwork(done, params);
706 });
707 it('Allows selection of Britcoin', function(done) {
708 var params = {
709 selectText: "BRIT - Britcoin",
710 firstAddress: "B6Aue4J2XLs1f1dtD4H1SHYFfh4XrmEbrw",
711 };
712 testNetwork(done, params);
713 });
714 it('Allows selection of Canadaecoin', function(done) {
715 var params = {
716 selectText: "CDN - Canadaecoin",
717 firstAddress: "CanAyCfd5Rj2CQVfaoAmvDUZunPM5W1AEQ",
718 };
719 testNetwork(done, params);
720 });
721 it('Allows selection of Cannacoin', function(done) {
722 var params = {
723 selectText: "CCN - Cannacoin",
724 firstAddress: "CYjW8xWB43g6krLJTmmrPk1PonoQX7h9Qd",
725 };
726 testNetwork(done, params);
727 });
728 it('Allows selection of Clubcoin', function(done) {
729 var params = {
730 selectText: "CLUB - Clubcoin",
731 firstAddress: "CHMDEXN4sihpSVX4GyAa2hZ62shnby7uyN",
732 };
733 testNetwork(done, params);
734 });
735 it('Allows selection of Compcoin', function(done) {
736 var params = {
737 selectText: "CMP - Compcoin",
738 firstAddress: "CLshtw3zhxkseBJS46UF12v3AFy9Dx7JVv",
739 };
740 testNetwork(done, params);
741 });
742 it('Allows selection of Crave', function(done) {
743 var params = {
744 selectText: "CRAVE - Crave",
745 firstAddress: "VCYJeti6uKMNBFKCL7eP96UwuFWYHM7c85",
746 };
747 testNetwork(done, params);
748 });
749 it('Allows selection of Defcoin', function(done) {
750 var params = {
751 selectText: "DFC - Defcoin",
752 firstAddress: "D8swcgyaaFUrXZU3ATwbgy16buCpWqbG1M",
753 };
754 testNetwork(done, params);
755 });
756 it('Allows selection of Diamond', function(done) {
757 var params = {
758 selectText: "DMD - Diamond",
759 firstAddress: "dJnrVbLL9UPjdaVRz2C8VpqHZknqAqjLek",
760 };
761 testNetwork(done, params);
762 });
763 it('Allows selection of Digibyte', function(done) {
764 var params = {
765 selectText: "DGB - Digibyte",
766 firstAddress: "D85Rp9jwLtMdmP6wGjTiqHBdVQLST3YCEq",
767 };
768 testNetwork(done, params);
769 });
770 it('Allows selection of Digitalcoin', function(done) {
771 var params = {
772 selectText: "DGC - Digitalcoin",
773 firstAddress: "DKw4UGKEAZWweDNEbBFNQx4EM8x1mpUdia",
774 };
775 testNetwork(done, params);
776 });
777 it('Allows selection of Ecoin', function(done) {
778 var params = {
779 selectText: "ECN - Ecoin",
780 firstAddress: "e6WFPLG5gcXyF7cESFteH1hE2XSmowW5yB",
781 };
782 testNetwork(done, params);
783 });
784 it('Allows selection of Edrcoin', function(done) {
785 var params = {
786 selectText: "EDRC - Edrcoin",
787 firstAddress: "eh1nUJsvgKPFv6ebMBfcwJ299GMCpjeZUG",
788 };
789 testNetwork(done, params);
790 });
791 it('Allows selection of Egulden', function(done) {
792 var params = {
793 selectText: "EFL - Egulden",
794 firstAddress: "Lg66yt55R7edRM58cDhKzXik2kFme3viX7",
795 };
796 testNetwork(done, params);
797 });
798 it('Allows selection of Einsteinium', function(done) {
799 var params = {
800 selectText: "EMC2 - Einsteinium",
801 firstAddress: "EVAABm9hXKHk2MpVMbwNakRubFnNha5m8m",
802 };
803 testNetwork(done, params);
804 });
805 it('Allows selection of Europecoin', function(done) {
806 var params = {
807 selectText: "ERC - Europecoin",
808 firstAddress: "ESA2YwPYntAoaPrE8Fm5qkKRtkcwLcwD6R",
809 };
810 testNetwork(done, params);
811 });
812 it('Allows selection of Exclusivecoin', function(done) {
813 var params = {
814 selectText: "EXCL - Exclusivecoin",
815 firstAddress: "EbUa6m8UZW6nTxsYZD2FsDjkadKbp5M6JT",
816 };
817 testNetwork(done, params);
818 });
819 it('Allows selection of Feathercoin', function(done) {
820 var params = {
821 selectText: "FTC - Feathercoin",
822 firstAddress: "6gDdjAMoSgQaW8UhqK3oboHs6ftGAroKkM",
823 };
824 testNetwork(done, params);
825 });
826 it('Allows selection of Firstcoin', function(done) {
827 var params = {
828 selectText: "FRST - Firstcoin",
829 firstAddress: "FJN9GzfMm7Q8R4DJwK1H9F6A1GTghvFiMJ",
830 };
831 testNetwork(done, params);
832 });
833 it('Allows selection of Flashcoin', function(done) {
834 var params = {
835 selectText: "FLASH - Flashcoin",
836 firstAddress: "UWfpf5LfMmLxZYooEb2EyvWhZ8NG7EZDRt",
837 };
838 testNetwork(done, params);
839 });
840 it('Allows selection of GCRCoin', function(done) {
841 var params = {
842 selectText: "GCR - GCRCoin",
843 firstAddress: "GJjF5cLwyXLacpuvXAVksxGxKvHDjx58d6",
844 };
845 testNetwork(done, params);
846 });
847 it('Allows selection of Gobyte', function(done) {
848 var params = {
849 selectText: "GBX - Gobyte",
850 firstAddress: "GS813Ys2brkmvSUw1rUqGPm2HqQVDHJRyA",
851 };
852 testNetwork(done, params);
853 });
854 it('Allows selection of Gridcoin', function(done) {
855 var params = {
856 selectText: "GRC - Gridcoin",
857 firstAddress: "SGrWbBPvobgqKRF8td1Kdc9vbRY7MJ78Y9",
858 };
859 testNetwork(done, params);
860 });
861 it('Allows selection of Gulden', function(done) {
862 var params = {
863 selectText: "NLG - Gulden",
864 firstAddress: "GcDP7cNEc33MPPdTFNJ8pZc6VMZJ2CbKxY",
865 };
866 testNetwork(done, params);
867 });
868 it('Allows selection of Helleniccoin', function(done) {
869 var params = {
870 selectText: "HNC - Helleniccoin",
871 firstAddress: "LbHEKe5H72zp9G1fuWNiiNePTUfJb88915",
872 };
873 testNetwork(done, params);
874 });
875 it('Allows selection of Hempcoin', function(done) {
876 var params = {
877 selectText: "THC - Hempcoin",
878 firstAddress: "H8sdWbZyJV4gyXyHtLXDaNnAuUDhK5mfTV",
879 };
880 testNetwork(done, params);
881 });
882 it('Allows selection of Insane', function(done) {
883 var params = {
884 selectText: "INSN - Insane",
885 firstAddress: "iMPqEJMiXWuxC9U2NVinCCMr4t72h58EWx",
886 };
887 testNetwork(done, params);
888 });
889 it('Allows selection of Iop', function(done) {
890 var params = {
891 selectText: "IOP - Iop",
892 firstAddress: "pGKQmcaPf95Ur5o6oHK4qdiZ52p1yaTvq1",
893 };
894 testNetwork(done, params);
895 });
896 it('Allows selection of Ixcoin', function(done) {
897 var params = {
898 selectText: "IXC - Ixcoin",
899 firstAddress: "xgE9bTZ6YypT3E6ByzkTt31Hq68E9BqywH",
900 };
901 testNetwork(done, params);
902 });
903 it('Allows selection of Landcoin', function(done) {
904 var params = {
905 selectText: "LDCN - Landcoin",
906 firstAddress: "LLvLwNjG1aJcn1RS4W4GJUbv8fNaRATG7c",
907 };
908 testNetwork(done, params);
909 });
910 it('Allows selection of Library Credits', function(done) {
911 var params = {
912 selectText: "LBC - Library Credits",
913 firstAddress: "bQJEQrHDJyHdqycB32uysh1SWn8Ln8LMdg",
914 };
915 testNetwork(done, params);
916 });
917 it('Allows selection of Linx', function(done) {
918 var params = {
919 selectText: "LINX - Linx",
920 firstAddress: "XGWQ3cb3LGUB3VnHmj6xYSMgnokNbf6dyk",
921 };
922 testNetwork(done, params);
923 });
924 it('Allows selection of Litecoincash', function(done) {
925 var params = {
926 selectText: "LCC - Litecoincash",
927 firstAddress: "Ce5n7fjUuQPLutJ4W5nCCfQLKdKLE1mv9A",
928 };
929 testNetwork(done, params);
930 });
931 it('Allows selection of Lynx', function(done) {
932 var params = {
933 selectText: "LYNX - Lynx",
934 firstAddress: "KUeY3ZdZkg96p4W98pj1JjygCFU1XqWdw3",
935 };
936 testNetwork(done, params);
937 });
938 it('Allows selection of Minexcoin', function(done) {
939 var params = {
940 selectText: "MNX - Minexcoin",
941 firstAddress: "XC1VnyJVfiMDwWgFtAHDp41cgY3AHk3dJT",
942 };
943 testNetwork(done, params);
944 });
945 it('Allows selection of Navcoin', function(done) {
946 var params = {
947 selectText: "NAV - Navcoin",
948 firstAddress: "NTQVTPK3NWSQLKoffkiQw99T8PifkF1Y2U",
949 };
950 testNetwork(done, params);
951 });
952 it('Allows selection of Neoscoin', function(done) {
953 var params = {
954 selectText: "NEOS - Neoscoin",
955 firstAddress: "NgATz6QbQNXvayHQ4CpZayugb9HeaPDdby",
956 };
957 testNetwork(done, params);
958 });
959 it('Allows selection of Neurocoin', function(done) {
960 var params = {
961 selectText: "NRO - Neurocoin",
962 firstAddress: "NVdYErQ3mFpDuF5DquW9WMiT7sLc8ufFTn",
963 };
964 testNetwork(done, params);
965 });
966 it('Allows selection of Newyorkc', function(done) {
967 var params = {
968 selectText: "NYC - Newyorkc",
969 firstAddress: "RSVMfyH1fKfy3puADJEhut2vfkRyon6imm",
970 };
971 testNetwork(done, params);
972 });
973 it('Allows selection of Novacoin', function(done) {
974 var params = {
975 selectText: "NVC - Novacoin",
976 firstAddress: "4JRvUmxcKCJmaMXZyvRoSS1cmG2XvnZfHN",
977 };
978 testNetwork(done, params);
979 });
980 it('Allows selection of Nushares', function(done) {
981 var params = {
982 selectText: "NSR - Nushares",
983 firstAddress: "SecjXzU3c7EecdT7EbC4vvmbdtBBokWh6J",
984 };
985 testNetwork(done, params);
986 });
987 it('Allows selection of Okcash', function(done) {
988 var params = {
989 selectText: "OK - Okcash",
990 firstAddress: "PV4Qp1TUYuGv4TqVtLZtqvrsWWRycfx1Yi",
991 };
992 testNetwork(done, params);
993 });
994 it('Allows selection of Omnicore', function(done) {
995 var params = {
996 selectText: "OMNI - Omnicore",
997 firstAddress: "1Q1t3gonjCT3rW38TsTsCvgSc3hh7zBGbi",
998 };
999 testNetwork(done, params);
1000 });
1001 it('Allows selection of Pesobit', function(done) {
1002 var params = {
1003 selectText: "PSB - Pesobit",
1004 firstAddress: "PDePsF7ALyXP7JaywokdYiRTDtKa14MAr1",
1005 };
1006 testNetwork(done, params);
1007 });
1008 it('Allows selection of Pinkcoin', function(done) {
1009 var params = {
1010 selectText: "PINK - Pinkcoin",
1011 firstAddress: "2TgjYQffjbzUHJghNaVbdsjHbRwruC3yzC",
1012 };
1013 testNetwork(done, params);
1014 });
1015 it('Allows selection of POSWcoin', function(done) {
1016 var params = {
1017 selectText: "POSW - POSWcoin",
1018 firstAddress: "PNxewmZoPnGBvoEbH6hgQZCK1igDiBCdgC",
1019 };
1020 testNetwork(done, params);
1021 });
1022 it('Allows selection of Potcoin', function(done) {
1023 var params = {
1024 selectText: "POT - Potcoin",
1025 firstAddress: "PEo7Vg2ctXgpP4vuLPeY9aGJtZotyrmiHc",
1026 };
1027 testNetwork(done, params);
1028 });
1029 it('Allows selection of Putincoin', function(done) {
1030 var params = {
1031 selectText: "PUT - Putincoin",
1032 firstAddress: "PViWnfr2uFtovd6e7joM49C94CsGSnqJis",
1033 };
1034 testNetwork(done, params);
1035 });
1036 it('Allows selection of Reddcoin', function(done) {
1037 var params = {
1038 selectText: "RDD - Reddcoin",
1039 firstAddress: "1M4druAcUfkXBaAcQ4cCgCLPHChiaib6kL",
1040 };
1041 testNetwork(done, params);
1042 });
1043 it('Allows selection of RevolutionVR', function(done) {
1044 var params = {
1045 selectText: "RVR - RevolutionVR",
1046 firstAddress: "VXeeoP2jkzZnMFxtc66ZBZK1NHN5QJnnjL",
1047 };
1048 testNetwork(done, params);
1049 });
1050 it('Allows selection of Rubycoin', function(done) {
1051 var params = {
1052 selectText: "RBY - Rubycoin",
1053 firstAddress: "RV76JDtjTs11JdMDRToYn6CHecMRPLnKS6",
1054 };
1055 testNetwork(done, params);
1056 });
1057 it('Allows selection of Smileycoin', function(done) {
1058 var params = {
1059 selectText: "SMLY - Smileycoin",
1060 firstAddress: "BEZVnEBCAyFByrgKpwAgYgtvP4rKAd9Sj2",
1061 };
1062 testNetwork(done, params);
1063 });
1064 it('Allows selection of Solarcoin', function(done) {
1065 var params = {
1066 selectText: "SLR - Solarcoin",
1067 firstAddress: "8LZ13HbnjtaMJWSvvVFNTLf71zFfDrhwLu",
1068 };
1069 testNetwork(done, params);
1070 });
1071 it('Allows selection of Stratis', function(done) {
1072 var params = {
1073 selectText: "STRAT - Stratis",
1074 firstAddress: "ScfJnq3QDhKgDMEds6sqUE1ot6ShfhmXXq",
1075 };
1076 testNetwork(done, params);
1077 });
1078 it('Allows selection of Syscoin', function(done) {
1079 var params = {
1080 selectText: "SYS - Syscoin",
1081 firstAddress: "SZwJi42Pst3VAMomyK5DG4157WM5ofRmSj",
1082 };
1083 testNetwork(done, params);
1084 });
1085 it('Allows selection of Toa', function(done) {
1086 var params = {
1087 selectText: "TOA - Toa",
1088 firstAddress: "TSe1QAnUwQzUfbBusDzRJ9URttrRGKoNKF",
1089 };
1090 testNetwork(done, params);
1091 });
1092 it('Allows selection of Ultimatesecurecash', function(done) {
1093 var params = {
1094 selectText: "USC - Ultimatesecurecash",
1095 firstAddress: "UPyLAZU2Che5fiy7Ed8xVJFmXAUhitA4ug",
1096 };
1097 testNetwork(done, params);
1098 });
1099 it('Allows selection of Unobtanium', function(done) {
1100 var params = {
1101 selectText: "UNO - Unobtanium",
1102 firstAddress: "uUBMPVMXrR6qhqornJqKTWgr8L69vihSL9",
1103 };
1104 testNetwork(done, params);
1105 });
1106 it('Allows selection of Vcash', function(done) {
1107 var params = {
1108 selectText: "XVC - Vcash",
1109 firstAddress: "VuL53MSY6KjvAjKSeRkh3NDnKykacDVeps",
1110 };
1111 testNetwork(done, params);
1112 });
1113 it('Allows selection of Verge', function(done) {
1114 var params = {
1115 selectText: "XVG - Verge",
1116 firstAddress: "DCrVuGkMjLJpTGgwAgv9AcMdeb1nkWbjZA",
1117 };
1118 testNetwork(done, params);
1119 });
1120 it('Allows selection of Vertcoin', function(done) {
1121 var params = {
1122 selectText: "VTC - Vertcoin",
1123 firstAddress: "Vf6koGuiWdXQfx8tNqxoNeEDxh4xh5cxsG",
1124 };
1125 testNetwork(done, params);
1126 });
1127 it('Allows selection of Vivo', function(done) {
1128 var params = {
1129 selectText: "VIVO - Vivo",
1130 firstAddress: "VFmBwuXXGhJe7MarQG2GfzHMFebRHgfSpB",
1131 };
1132 testNetwork(done, params);
1133 });
1134 it('Allows selection of Vpncoin', function(done) {
1135 var params = {
1136 selectText: "VASH - Vpncoin",
1137 firstAddress: "VoEmH1qXC4TsSgBAStR21QYetwnFqbqCx9",
1138 };
1139 testNetwork(done, params);
1140 });
1141 it('Allows selection of Whitecoin', function(done) {
1142 var params = {
1143 selectText: "XWC - Whitecoin",
1144 firstAddress: "WcSwCAUqrSgeSYbsaS3SSWWhsx8KRYTFDR",
1145 };
1146 testNetwork(done, params);
1147 });
1148 it('Allows selection of Wincoin', function(done) {
1149 var params = {
1150 selectText: "WC - Wincoin",
1151 firstAddress: "WaDVCESMGgyKgNESdn3u43NnwmGSkZED3Z",
1152 };
1153 testNetwork(done, params);
1154 });
1155 it('Allows selection of Zcoin', function(done) {
1156 var params = {
1157 selectText: "XZC - Zcoin",
1158 firstAddress: "a6VcMdP4XgAA9Tr7xNszmPG5FZpfRf17Cq",
1159 };
1160 testNetwork(done, params);
1161 });
1162
1163
1164 // BIP39 seed is set from phrase
1165 it('Sets the bip39 seed from the prhase', function(done) {
1166 driver.findElement(By.css('.phrase'))
1167 .sendKeys('abandon abandon ability');
1168 driver.sleep(generateDelay).then(function() {
1169 driver.findElement(By.css('.seed'))
1170 .getAttribute("value")
1171 .then(function(seed) {
1172 expect(seed).toBe("20da140d3dd1df8713cefcc4d54ce0e445b4151027a1ab567b832f6da5fcc5afc1c3a3f199ab78b8e0ab4652efd7f414ac2c9a3b81bceb879a70f377aa0a58f3");
1173 done();
1174 })
1175 });
1176 });
1177
1178 // BIP32 root key is set from phrase
1179 it('Sets the bip39 root key from the prhase', function(done) {
1180 driver.findElement(By.css('.phrase'))
1181 .sendKeys('abandon abandon ability');
1182 driver.sleep(generateDelay).then(function() {
1183 driver.findElement(By.css('.root-key'))
1184 .getAttribute("value")
1185 .then(function(seed) {
1186 expect(seed).toBe("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
1187 done();
1188 })
1189 });
1190 });
1191
1192 // Tabs show correct addresses when changed
1193 it('Shows the correct address when tab is changed', function(done) {
1194 driver.findElement(By.css('.phrase'))
1195 .sendKeys('abandon abandon ability');
1196 driver.sleep(generateDelay).then(function() {
1197 driver.findElement(By.css('#bip32-tab a'))
1198 .click();
1199 driver.sleep(generateDelay).then(function() {
1200 getFirstAddress(function(address) {
1201 expect(address).toBe("17uQ7s2izWPwBmEVFikTmZUjbBKWYdJchz");
1202 done();
1203 });
1204 });
1205 });
1206 });
1207
1208 // BIP44 derivation path is shown
1209 it('Shows the derivation path for bip44 tab', function(done) {
1210 driver.findElement(By.css('.phrase'))
1211 .sendKeys('abandon abandon ability');
1212 driver.sleep(generateDelay).then(function() {
1213 driver.findElement(By.css('#bip44 .path'))
1214 .getAttribute("value")
1215 .then(function(path) {
1216 expect(path).toBe("m/44'/0'/0'/0");
1217 done();
1218 })
1219 });
1220 });
1221
1222 // BIP44 extended private key is shown
1223 it('Shows the extended private key for bip44 tab', function(done) {
1224 driver.findElement(By.css('.phrase'))
1225 .sendKeys('abandon abandon ability');
1226 driver.sleep(generateDelay).then(function() {
1227 driver.findElement(By.css('.extended-priv-key'))
1228 .getAttribute("value")
1229 .then(function(path) {
1230 expect(path).toBe("xprvA2DxxvPZcyRvYgZMGS53nadR32mVDeCyqQYyFhrCVbJNjPoxMeVf7QT5g7mQASbTf9Kp4cryvcXnu2qurjWKcrdsr91jXymdCDNxKgLFKJG");
1231 done();
1232 })
1233 });
1234 });
1235
1236 // BIP44 extended public key is shown
1237 it('Shows the extended public key for bip44 tab', function(done) {
1238 driver.findElement(By.css('.phrase'))
1239 .sendKeys('abandon abandon ability');
1240 driver.sleep(generateDelay).then(function() {
1241 driver.findElement(By.css('.extended-pub-key'))
1242 .getAttribute("value")
1243 .then(function(path) {
1244 expect(path).toBe("xpub6FDKNRvTTLzDmAdpNTc49ia9b4byd6vqCdUa46Fp3vqMcC96uBoufCmZXQLiN5AK3iSCJMhf9gT2sxkpyaPepRuA7W3MujV5tGmF5VfbueM");
1245 done();
1246 })
1247 });
1248 });
1249
1250 // BIP44 account field changes address list
1251 it('Changes the address list if bip44 account is changed', function(done) {
1252 driver.findElement(By.css('#bip44 .account'))
1253 .sendKeys('1');
1254 driver.findElement(By.css('.phrase'))
1255 .sendKeys('abandon abandon ability');
1256 driver.sleep(generateDelay).then(function() {
1257 getFirstAddress(function(address) {
1258 expect(address).toBe("1Nq2Wmu726XHCuGhctEtGmhxo3wzk5wZ1H");
1259 done();
1260 });
1261 });
1262 });
1263
1264 // BIP44 change field changes address list
1265 it('Changes the address list if bip44 change is changed', function(done) {
1266 driver.findElement(By.css('#bip44 .change'))
1267 .sendKeys('1');
1268 driver.findElement(By.css('.phrase'))
1269 .sendKeys('abandon abandon ability');
1270 driver.sleep(generateDelay).then(function() {
1271 getFirstAddress(function(address) {
1272 expect(address).toBe("1KAGfWgqfVbSSXY56fNQ7YnhyKuoskHtYo");
1273 done();
1274 });
1275 });
1276 });
1277
1278 // BIP32 derivation path can be set
1279 it('Can use a custom bip32 derivation path', function(done) {
1280 driver.findElement(By.css('#bip32-tab a'))
1281 .click();
1282 driver.findElement(By.css('#bip32 .path'))
1283 .clear();
1284 driver.findElement(By.css('#bip32 .path'))
1285 .sendKeys('m/1');
1286 driver.findElement(By.css('.phrase'))
1287 .sendKeys('abandon abandon ability');
1288 driver.sleep(generateDelay).then(function() {
1289 getFirstAddress(function(address) {
1290 expect(address).toBe("16pYQQdLD1hH4hwTGLXBaZ9Teboi1AGL8L");
1291 done();
1292 });
1293 });
1294 });
1295
1296 // BIP32 can use hardened derivation paths
1297 it('Can use a hardened derivation paths', function(done) {
1298 driver.findElement(By.css('#bip32-tab a'))
1299 .click();
1300 driver.findElement(By.css('#bip32 .path'))
1301 .clear();
1302 driver.findElement(By.css('#bip32 .path'))
1303 .sendKeys("m/0'");
1304 driver.findElement(By.css('.phrase'))
1305 .sendKeys('abandon abandon ability');
1306 driver.sleep(generateDelay).then(function() {
1307 getFirstAddress(function(address) {
1308 expect(address).toBe("14aXZeprXAE3UUKQc4ihvwBvww2LuEoHo4");
1309 done();
1310 });
1311 });
1312 });
1313
1314 // BIP32 extended private key is shown
1315 it('Shows the BIP32 extended private key', function(done) {
1316 driver.findElement(By.css('#bip32-tab a'))
1317 .click();
1318 driver.findElement(By.css('.phrase'))
1319 .sendKeys('abandon abandon ability');
1320 driver.sleep(generateDelay).then(function() {
1321 driver.findElement(By.css('.extended-priv-key'))
1322 .getAttribute("value")
1323 .then(function(privKey) {
1324 expect(privKey).toBe("xprv9va99uTVE5aLiutUVLTyfxfe8v8aaXjSQ1XxZbK6SezYVuikA9MnjQVTA8rQHpNA5LKvyQBpLiHbBQiiccKiBDs7eRmBogsvq3THFeLHYbe");
1325 done();
1326 });
1327 });
1328 });
1329
1330 // BIP32 extended public key is shown
1331 it('Shows the BIP32 extended public key', function(done) {
1332 driver.findElement(By.css('#bip32-tab a'))
1333 .click();
1334 driver.findElement(By.css('.phrase'))
1335 .sendKeys('abandon abandon ability');
1336 driver.sleep(generateDelay).then(function() {
1337 driver.findElement(By.css('.extended-pub-key'))
1338 .getAttribute("value")
1339 .then(function(pubKey) {
1340 expect(pubKey).toBe("xpub69ZVZQzP4T8dwPxwbMzz36cNgwy4yzTHmETZMyihzzXXNi3thgg3HCow1RtY252wdw5rS8369xKnraN5Q93y3FkFfJp2XEHWUrkyXsjS93P");
1341 done();
1342 });
1343 });
1344 });
1345
1346 // Derivation path is shown in table
1347 it('Shows the derivation path in the table', function(done) {
1348 driver.findElement(By.css('.phrase'))
1349 .sendKeys('abandon abandon ability');
1350 driver.sleep(generateDelay).then(function() {
1351 getFirstPath(function(path) {
1352 expect(path).toBe("m/44'/0'/0'/0/0");
1353 done();
1354 });
1355 });
1356 });
1357
1358 // Derivation path for address can be hardened
1359 it('Can derive hardened addresses', function(done) {
1360 driver.findElement(By.css('#bip32-tab a'))
1361 .click();
1362 driver.executeScript(function() {
1363 $(".hardened-addresses").prop("checked", true);
1364 });
1365 driver.findElement(By.css('.phrase'))
1366 .sendKeys('abandon abandon ability');
1367 driver.sleep(generateDelay).then(function() {
1368 getFirstAddress(function(address) {
1369 expect(address).toBe("18exLzUv7kfpiXRzmCjFDoC9qwNLFyvwyd");
1370 done();
1371 });
1372 });
1373 });
1374
1375 // Derivation path visibility can be toggled
1376 it('Can toggle visibility of the derivation path column', function(done) {
1377 driver.findElement(By.css('.phrase'))
1378 .sendKeys('abandon abandon ability');
1379 driver.sleep(generateDelay).then(function() {
1380 driver.findElement(By.css('.index-toggle'))
1381 .click();
1382 testColumnValuesAreInvisible(done, "index");
1383 });
1384 });
1385
1386 // Address is shown
1387 it('Shows the address in the table', function(done) {
1388 driver.findElement(By.css('.phrase'))
1389 .sendKeys('abandon abandon ability');
1390 driver.sleep(generateDelay).then(function() {
1391 getFirstAddress(function(address) {
1392 expect(address).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug");
1393 done();
1394 });
1395 });
1396 });
1397
1398 // Addresses are shown in order of derivation path
1399 it('Shows the address in order of derivation path', function(done) {
1400 driver.findElement(By.css('.phrase'))
1401 .sendKeys('abandon abandon ability');
1402 driver.sleep(generateDelay).then(function() {
1403 testRowsAreInCorrectOrder(done);
1404 });
1405 });
1406
1407 // Address visibility can be toggled
1408 it('Can toggle visibility of the address column', function(done) {
1409 driver.findElement(By.css('.phrase'))
1410 .sendKeys('abandon abandon ability');
1411 driver.sleep(generateDelay).then(function() {
1412 driver.findElement(By.css('.address-toggle'))
1413 .click();
1414 testColumnValuesAreInvisible(done, "address");
1415 });
1416 });
1417
1418 // Public key is shown in table
1419 it('Shows the public key in the table', function(done) {
1420 driver.findElement(By.css('.phrase'))
1421 .sendKeys('abandon abandon ability');
1422 driver.sleep(generateDelay).then(function() {
1423 driver.findElements(By.css('.pubkey'))
1424 .then(function(els) {
1425 els[0].getText()
1426 .then(function(pubkey) {
1427 expect(pubkey).toBe("033f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3");
1428 done();
1429 });
1430 });
1431 });
1432 });
1433
1434 // Public key visibility can be toggled
1435 it('Can toggle visibility of the public key column', function(done) {
1436 driver.findElement(By.css('.phrase'))
1437 .sendKeys('abandon abandon ability');
1438 driver.sleep(generateDelay).then(function() {
1439 driver.findElement(By.css('.public-key-toggle'))
1440 .click();
1441 testColumnValuesAreInvisible(done, "pubkey");
1442 });
1443 });
1444
1445 // Private key is shown in table
1446 it('Shows the private key in the table', function(done) {
1447 driver.findElement(By.css('.phrase'))
1448 .sendKeys('abandon abandon ability');
1449 driver.sleep(generateDelay).then(function() {
1450 driver.findElements(By.css('.privkey'))
1451 .then(function(els) {
1452 els[0].getText()
1453 .then(function(pubkey) {
1454 expect(pubkey).toBe("L26cVSpWFkJ6aQkPkKmTzLqTdLJ923e6CzrVh9cmx21QHsoUmrEE");
1455 done();
1456 });
1457 });
1458 });
1459 });
1460
1461 // Private key visibility can be toggled
1462 it('Can toggle visibility of the private key column', function(done) {
1463 driver.findElement(By.css('.phrase'))
1464 .sendKeys('abandon abandon ability');
1465 driver.sleep(generateDelay).then(function() {
1466 driver.findElement(By.css('.private-key-toggle'))
1467 .click();
1468 testColumnValuesAreInvisible(done, "privkey");
1469 });
1470 });
1471
1472 // More addresses can be generated
1473 it('Can generate more rows in the table', function(done) {
1474 driver.findElement(By.css('.phrase'))
1475 .sendKeys('abandon abandon ability');
1476 driver.sleep(generateDelay).then(function() {
1477 driver.findElement(By.css('.more'))
1478 .click();
1479 driver.sleep(generateDelay).then(function() {
1480 driver.findElements(By.css('.address'))
1481 .then(function(els) {
1482 expect(els.length).toBe(40);
1483 done();
1484 });
1485 });
1486 });
1487 });
1488
1489 // A custom number of additional addresses can be generated
1490 it('Can generate more rows in the table', function(done) {
1491 driver.findElement(By.css('.phrase'))
1492 .sendKeys('abandon abandon ability');
1493 driver.sleep(generateDelay).then(function() {
1494 driver.findElement(By.css('.rows-to-add'))
1495 .clear();
1496 driver.findElement(By.css('.rows-to-add'))
1497 .sendKeys('1');
1498 driver.findElement(By.css('.more'))
1499 .click();
1500 driver.sleep(generateDelay).then(function() {
1501 driver.findElements(By.css('.address'))
1502 .then(function(els) {
1503 expect(els.length).toBe(21);
1504 done();
1505 });
1506 });
1507 });
1508 });
1509
1510 // Additional addresses are shown in order of derivation path
1511 it('Shows additional addresses in order of derivation path', function(done) {
1512 driver.findElement(By.css('.phrase'))
1513 .sendKeys('abandon abandon ability');
1514 driver.sleep(generateDelay).then(function() {
1515 driver.findElement(By.css('.more'))
1516 .click();
1517 driver.sleep(generateDelay).then(function() {
1518 testRowsAreInCorrectOrder(done);
1519 });
1520 });
1521 });
1522
1523 // BIP32 root key can be set by the user
1524 it('Allows the user to set the BIP32 root key', function(done) {
1525 driver.findElement(By.css('.root-key'))
1526 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1527 driver.sleep(generateDelay).then(function() {
1528 getFirstAddress(function(address) {
1529 expect(address).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug");
1530 done();
1531 });
1532 });
1533 });
1534
1535 // Setting BIP32 root key clears the existing phrase, passphrase and seed
1536 // TODO this doesn't work in selenium with chrome
1537 it('Confirms the existing phrase should be cleared', function(done) {
1538 if (browser == "chrome") {
1539 pending("Selenium + Chrome headless bug for alert, see https://stackoverflow.com/q/45242264");
1540 }
1541 driver.findElement(By.css('.phrase'))
1542 .sendKeys('A non-blank but invalid value');
1543 driver.findElement(By.css('.root-key'))
1544 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1545 driver.switchTo().alert().accept();
1546 driver.findElement(By.css('.phrase'))
1547 .getAttribute("value").then(function(value) {
1548 expect(value).toBe("");
1549 done();
1550 });
1551 });
1552
1553 // Clearing of phrase, passphrase and seed can be cancelled by user
1554 // TODO this doesn't work in selenium with chrome
1555 it('Allows the clearing of the phrase to be cancelled', function(done) {
1556 if (browser == "chrome") {
1557 pending("Selenium + Chrome headless bug for alert, see https://stackoverflow.com/q/45242264");
1558 }
1559 driver.findElement(By.css('.phrase'))
1560 .sendKeys('abandon abandon ability');
1561 driver.sleep(generateDelay).then(function() {
1562 driver.findElement(By.css('.root-key'))
1563 .clear();
1564 driver.findElement(By.css('.root-key'))
1565 .sendKeys('x');
1566 driver.switchTo().alert().dismiss();
1567 driver.findElement(By.css('.phrase'))
1568 .getAttribute("value").then(function(value) {
1569 expect(value).toBe("abandon abandon ability");
1570 done();
1571 });
1572 });
1573 });
1574
1575 // Custom BIP32 root key is used when changing the derivation path
1576 it('Can set derivation path for root key instead of phrase', function(done) {
1577 driver.findElement(By.css('#bip44 .account'))
1578 .sendKeys('1');
1579 driver.findElement(By.css('.root-key'))
1580 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1581 driver.sleep(generateDelay).then(function() {
1582 getFirstAddress(function(address) {
1583 expect(address).toBe("1Nq2Wmu726XHCuGhctEtGmhxo3wzk5wZ1H");
1584 done();
1585 });
1586 });
1587 });
1588
1589 // Incorrect mnemonic shows error
1590 it('Shows an error for incorrect mnemonic', function(done) {
1591 driver.findElement(By.css('.phrase'))
1592 .sendKeys('abandon abandon abandon');
1593 driver.sleep(feedbackDelay).then(function() {
1594 driver.findElement(By.css('.feedback'))
1595 .getText()
1596 .then(function(feedback) {
1597 expect(feedback).toBe("Invalid mnemonic");
1598 done();
1599 });
1600 });
1601 });
1602
1603 // Incorrect word shows suggested replacement
1604 it('Shows word suggestion for incorrect word', function(done) {
1605 driver.findElement(By.css('.phrase'))
1606 .sendKeys('abandon abandon abiliti');
1607 driver.sleep(feedbackDelay).then(function() {
1608 driver.findElement(By.css('.feedback'))
1609 .getText()
1610 .then(function(feedback) {
1611 var msg = "abiliti not in wordlist, did you mean ability?";
1612 expect(feedback).toBe(msg);
1613 done();
1614 });
1615 });
1616 });
1617
1618 // Github pull request 48
1619 // First four letters of word shows that word, not closest
1620 // since first four letters gives unique word in BIP39 wordlist
1621 // eg ille should show illegal, not idle
1622 it('Shows word suggestion based on first four chars', function(done) {
1623 driver.findElement(By.css('.phrase'))
1624 .sendKeys('ille');
1625 driver.sleep(feedbackDelay).then(function() {
1626 driver.findElement(By.css('.feedback'))
1627 .getText()
1628 .then(function(feedback) {
1629 var msg = "ille not in wordlist, did you mean illegal?";
1630 expect(feedback).toBe(msg);
1631 done();
1632 });
1633 });
1634 });
1635
1636 // Incorrect BIP32 root key shows error
1637 it('Shows error for incorrect root key', function(done) {
1638 driver.findElement(By.css('.root-key'))
1639 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpj');
1640 driver.sleep(feedbackDelay).then(function() {
1641 driver.findElement(By.css('.feedback'))
1642 .getText()
1643 .then(function(feedback) {
1644 var msg = "Invalid root key";
1645 expect(feedback).toBe(msg);
1646 done();
1647 });
1648 });
1649 });
1650
1651 // Derivation path not starting with m shows error
1652 it('Shows error for derivation path not starting with m', function(done) {
1653 driver.findElement(By.css('#bip32-tab a'))
1654 .click();
1655 driver.findElement(By.css('#bip32 .path'))
1656 .clear();
1657 driver.findElement(By.css('#bip32 .path'))
1658 .sendKeys('n/0');
1659 driver.findElement(By.css('.phrase'))
1660 .sendKeys('abandon abandon ability');
1661 driver.sleep(feedbackDelay).then(function() {
1662 driver.findElement(By.css('.feedback'))
1663 .getText()
1664 .then(function(feedback) {
1665 var msg = "First character must be 'm'";
1666 expect(feedback).toBe(msg);
1667 done();
1668 });
1669 });
1670 });
1671
1672 // Derivation path containing invalid characters shows useful error
1673 it('Shows error for derivation path not starting with m', function(done) {
1674 driver.findElement(By.css('#bip32-tab a'))
1675 .click();
1676 driver.findElement(By.css('#bip32 .path'))
1677 .clear();
1678 driver.findElement(By.css('#bip32 .path'))
1679 .sendKeys('m/1/0wrong1/1');
1680 driver.findElement(By.css('.phrase'))
1681 .sendKeys('abandon abandon ability');
1682 driver.sleep(feedbackDelay).then(function() {
1683 driver.findElement(By.css('.feedback'))
1684 .getText()
1685 .then(function(feedback) {
1686 var msg = "Invalid characters 0wrong1 found at depth 2";
1687 expect(feedback).toBe(msg);
1688 done();
1689 });
1690 });
1691 });
1692
1693 // Github Issue 11: Default word length is 15
1694 // https://github.com/iancoleman/bip39/issues/11
1695 it('Sets the default word length to 15', function(done) {
1696 driver.findElement(By.css('.strength'))
1697 .getAttribute("value")
1698 .then(function(strength) {
1699 expect(strength).toBe("15");
1700 done();
1701 });
1702 });
1703
1704 // Github Issue 12: Generate more rows with private keys hidden
1705 // https://github.com/iancoleman/bip39/issues/12
1706 it('Sets the correct hidden column state on new rows', function(done) {
1707 driver.findElement(By.css('.phrase'))
1708 .sendKeys("abandon abandon ability");
1709 driver.sleep(generateDelay).then(function() {
1710 driver.findElement(By.css('.private-key-toggle'))
1711 .click();
1712 driver.findElement(By.css('.more'))
1713 .click();
1714 driver.sleep(generateDelay).then(function() {
1715 driver.findElements(By.css('.privkey'))
1716 .then(function(els) {
1717 expect(els.length).toBe(40);
1718 });
1719 testColumnValuesAreInvisible(done, "privkey");
1720 });
1721 });
1722 });
1723
1724 // Github Issue 19: Mnemonic is not sensitive to whitespace
1725 // https://github.com/iancoleman/bip39/issues/19
1726 it('Ignores excess whitespace in the mnemonic', function(done) {
1727 var doublespace = " ";
1728 var mnemonic = "urge cat" + doublespace + "bid";
1729 driver.findElement(By.css('.phrase'))
1730 .sendKeys(mnemonic);
1731 driver.sleep(generateDelay).then(function() {
1732 driver.findElement(By.css('.root-key'))
1733 .getAttribute("value")
1734 .then(function(seed) {
1735 expect(seed).toBe("xprv9s21ZrQH143K3isaZsWbKVoTtbvd34Y1ZGRugGdMeBGbM3AgBVzTH159mj1cbbtYSJtQr65w6L5xy5L9SFC7c9VJZWHxgAzpj4mun5LhrbC");
1736 done();
1737 });
1738 });
1739 });
1740
1741 // Github Issue 23: Part 1: Use correct derivation path when changing tabs
1742 // https://github.com/iancoleman/bip39/issues/23
1743 it('Uses the correct derivation path when changing tabs', function(done) {
1744 // 1) and 2) set the phrase
1745 driver.findElement(By.css('.phrase'))
1746 .sendKeys("abandon abandon ability");
1747 driver.sleep(generateDelay).then(function() {
1748 // 3) select bip32 tab
1749 driver.findElement(By.css('#bip32-tab a'))
1750 .click();
1751 driver.sleep(generateDelay).then(function() {
1752 // 4) switch from bitcoin to litecoin
1753 selectNetwork("LTC - Litecoin");
1754 driver.sleep(generateDelay).then(function() {
1755 // 5) Check address is displayed correctly
1756 getFirstAddress(function(address) {
1757 expect(address).toBe("LS8MP5LZ5AdzSZveRrjm3aYVoPgnfFh5T5");
1758 // 5) Check derivation path is displayed correctly
1759 getFirstPath(function(path) {
1760 expect(path).toBe("m/0/0");
1761 done();
1762 });
1763 });
1764 });
1765 });
1766 });
1767 });
1768
1769 // Github Issue 23 Part 2: Coin selection in derivation path
1770 // https://github.com/iancoleman/bip39/issues/23#issuecomment-238011920
1771 it('Uses the correct derivation path when changing coins', function(done) {
1772 // set the phrase
1773 driver.findElement(By.css('.phrase'))
1774 .sendKeys("abandon abandon ability");
1775 driver.sleep(generateDelay).then(function() {
1776 // switch from bitcoin to clam
1777 selectNetwork("CLAM - Clams");
1778 driver.sleep(generateDelay).then(function() {
1779 // check derivation path is displayed correctly
1780 getFirstPath(function(path) {
1781 expect(path).toBe("m/44'/23'/0'/0/0");
1782 done();
1783 });
1784 });
1785 });
1786 });
1787
1788 // Github Issue 26: When using a Root key derrived altcoins are incorrect
1789 // https://github.com/iancoleman/bip39/issues/26
1790 it('Uses the correct derivation for altcoins with root keys', function(done) {
1791 // 1) 2) and 3) set the root key
1792 driver.findElement(By.css('.root-key'))
1793 .sendKeys("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
1794 driver.sleep(generateDelay).then(function() {
1795 // 4) switch from bitcoin to viacoin
1796 selectNetwork("VIA - Viacoin");
1797 driver.sleep(generateDelay).then(function() {
1798 // 5) ensure the derived address is correct
1799 getFirstAddress(function(address) {
1800 expect(address).toBe("Vq9Eq4N5SQnjqZvxtxzo7hZPW5XnyJsmXT");
1801 done();
1802 });
1803 });
1804 });
1805 });
1806
1807 // Selecting a language with no existing phrase should generate a phrase in
1808 // that language.
1809 it('Generate a random phrase when language is selected and no current phrase', function(done) {
1810 driver.findElement(By.css("a[href='#japanese']"))
1811 .click();
1812 driver.sleep(generateDelay).then(function() {
1813 driver.findElement(By.css(".phrase"))
1814 .getAttribute("value").then(function(phrase) {
1815 expect(phrase.search(/[a-z]/)).toBe(-1);
1816 expect(phrase.length).toBeGreaterThan(0);
1817 done();
1818 });
1819 });
1820 });
1821
1822 // Selecting a language with existing phrase should update the phrase to use
1823 // that language.
1824 it('Updates existing phrases when the language is changed', function(done) {
1825 driver.findElement(By.css(".phrase"))
1826 .sendKeys("abandon abandon ability");
1827 driver.sleep(generateDelay).then(function() {
1828 driver.findElement(By.css("a[href='#italian']"))
1829 .click();
1830 driver.sleep(generateDelay).then(function() {
1831 driver.findElement(By.css(".phrase"))
1832 .getAttribute("value").then(function(phrase) {
1833 // Check only the language changes, not the phrase
1834 expect(phrase).toBe("abaco abaco abbaglio");
1835 getFirstAddress(function(address) {
1836 // Check the address is correct
1837 expect(address).toBe("1Dz5TgDhdki9spa6xbPFbBqv5sjMrx3xgV");
1838 done();
1839 });
1840 });
1841 });
1842 });
1843 });
1844
1845 // Suggested replacement for erroneous word in non-English language
1846 it('Shows word suggestion for incorrect word in non-English language', function(done) {
1847 driver.findElement(By.css('.phrase'))
1848 .sendKeys('abaco abaco zbbaglio');
1849 driver.sleep(feedbackDelay).then(function() {
1850 driver.findElement(By.css('.feedback'))
1851 .getText()
1852 .then(function(feedback) {
1853 var msg = "zbbaglio not in wordlist, did you mean abbaglio?";
1854 expect(feedback).toBe(msg);
1855 done();
1856 });
1857 });
1858 });
1859
1860 // Japanese word does not break across lines.
1861 // Point 2 from
1862 // https://github.com/bitcoin/bips/blob/master/bip-0039/bip-0039-wordlists.md#japanese
1863 it('Does not break Japanese words across lines', function(done) {
1864 driver.findElement(By.css('.phrase'))
1865 .getCssValue("word-break")
1866 .then(function(value) {
1867 expect(value).toBe("keep-all");
1868 done();
1869 });
1870 });
1871
1872 // Language can be specified at page load using hash value in url
1873 it('Can set the language from the url hash', function(done) {
1874 driver.get(url + "#japanese").then(function() {
1875 driver.findElement(By.css('.generate')).click();
1876 driver.sleep(generateDelay).then(function() {
1877 driver.findElement(By.css(".phrase"))
1878 .getAttribute("value").then(function(phrase) {
1879 expect(phrase.search(/[a-z]/)).toBe(-1);
1880 expect(phrase.length).toBeGreaterThan(0);
1881 done();
1882 });
1883 });
1884 });
1885 });
1886
1887 // Entropy can be entered by the user
1888 it('Allows entropy to be entered', function(done) {
1889 driver.findElement(By.css('.use-entropy'))
1890 .click();
1891 driver.findElement(By.css('.entropy'))
1892 .sendKeys('00000000 00000000 00000000 00000000');
1893 driver.sleep(generateDelay).then(function() {
1894 driver.findElement(By.css(".phrase"))
1895 .getAttribute("value").then(function(phrase) {
1896 expect(phrase).toBe("abandon abandon ability");
1897 getFirstAddress(function(address) {
1898 expect(address).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug");
1899 done();
1900 })
1901 });
1902 });
1903 });
1904
1905 // A warning about entropy is shown to the user, with additional information
1906 it('Shows a warning about using entropy', function(done) {
1907 driver.findElement(By.css('.use-entropy'))
1908 .click();
1909 driver.findElement(By.css('.entropy-container'))
1910 .getText()
1911 .then(function(containerText) {
1912 var warning = "mnemonic may be insecure";
1913 expect(containerText).toContain(warning);
1914 driver.findElement(By.css('#entropy-notes'))
1915 .findElement(By.xpath("parent::*"))
1916 .getText()
1917 .then(function(notesText) {
1918 var detail = "flipping a fair coin, rolling a fair dice, noise measurements etc";
1919 expect(notesText).toContain(detail);
1920 done();
1921 });
1922 });
1923 });
1924
1925 // The types of entropy available are described to the user
1926 it('Shows the types of entropy available', function(done) {
1927 driver.findElement(By.css('.entropy'))
1928 .getAttribute("placeholder")
1929 .then(function(placeholderText) {
1930 var options = [
1931 "binary",
1932 "base 6",
1933 "dice",
1934 "base 10",
1935 "hexadecimal",
1936 "cards",
1937 ];
1938 for (var i=0; i<options.length; i++) {
1939 var option = options[i];
1940 expect(placeholderText).toContain(option);
1941 }
1942 done();
1943 });
1944 });
1945
1946 // The actual entropy used is shown to the user
1947 it('Shows the actual entropy used', function(done) {
1948 driver.findElement(By.css('.use-entropy'))
1949 .click();
1950 driver.findElement(By.css('.entropy'))
1951 .sendKeys('Not A Very Good Entropy Source At All');
1952 driver.sleep(generateDelay).then(function() {
1953 driver.findElement(By.css('.entropy-container'))
1954 .getText()
1955 .then(function(text) {
1956 expect(text).toMatch(/Filtered Entropy\s+AedEceAA/);
1957 done();
1958 });
1959 });
1960 });
1961
1962 // Binary entropy can be entered
1963 it('Allows binary entropy to be entered', function(done) {
1964 testEntropyType(done, "01", "binary");
1965 });
1966
1967 // Base 6 entropy can be entered
1968 it('Allows base 6 entropy to be entered', function(done) {
1969 testEntropyType(done, "012345", "base 6");
1970 });
1971
1972 // Base 6 dice entropy can be entered
1973 it('Allows base 6 dice entropy to be entered', function(done) {
1974 testEntropyType(done, "123456", "base 6 (dice)");
1975 });
1976
1977 // Base 10 entropy can be entered
1978 it('Allows base 10 entropy to be entered', function(done) {
1979 testEntropyType(done, "789", "base 10");
1980 });
1981
1982 // Hexadecimal entropy can be entered
1983 it('Allows hexadecimal entropy to be entered', function(done) {
1984 testEntropyType(done, "abcdef", "hexadecimal");
1985 });
1986
1987 // Dice entropy value is shown as the converted base 6 value
1988 // ie 123456 is converted to 123450
1989 it('Shows dice entropy as base 6', function(done) {
1990 driver.findElement(By.css('.use-entropy'))
1991 .click();
1992 driver.findElement(By.css('.entropy'))
1993 .sendKeys("123456");
1994 driver.sleep(generateDelay).then(function() {
1995 driver.findElement(By.css('.entropy-container'))
1996 .getText()
1997 .then(function(text) {
1998 expect(text).toMatch(/Filtered Entropy\s+123450/);
1999 done();
2000 });
2001 });
2002 });
2003
2004 // The number of bits of entropy accumulated is shown
2005 it("Shows the number of bits of entropy for 20 bits of binary", function(done) {
2006 testEntropyBits(done, "0000 0000 0000 0000 0000", "20");
2007 });
2008 it("Shows the number of bits of entropy for 1 bit of binary", function(done) {
2009 testEntropyBits(done, "0", "1");
2010 });
2011 it("Shows the number of bits of entropy for 4 bits of binary", function(done) {
2012 testEntropyBits(done, "0000", "4");
2013 });
2014 it("Shows the number of bits of entropy for 1 character of base 6 (dice)", function(done) {
2015 // 6 in card is 0 in base 6, 0 in base 6 is 2.6 bits (rounded down to 2 bits)
2016 testEntropyBits(done, "6", "2");
2017 });
2018 it("Shows the number of bits of entropy for 1 character of base 10 with 3 bits", function(done) {
2019 // 7 in base 10 is 111 in base 2, no leading zeros
2020 testEntropyBits(done, "7", "3");
2021 });
2022 it("Shows the number of bits of entropy for 1 character of base 10 with 4 bis", function(done) {
2023 testEntropyBits(done, "8", "4");
2024 });
2025 it("Shows the number of bits of entropy for 1 character of hex", function(done) {
2026 testEntropyBits(done, "F", "4");
2027 });
2028 it("Shows the number of bits of entropy for 2 characters of base 10", function(done) {
2029 testEntropyBits(done, "29", "6");
2030 });
2031 it("Shows the number of bits of entropy for 2 characters of hex", function(done) {
2032 testEntropyBits(done, "0A", "8");
2033 });
2034 it("Shows the number of bits of entropy for 2 characters of hex with 3 leading zeros", function(done) {
2035 // hex is always multiple of 4 bits of entropy
2036 testEntropyBits(done, "1A", "8");
2037 });
2038 it("Shows the number of bits of entropy for 2 characters of hex with 2 leading zeros", function(done) {
2039 testEntropyBits(done, "2A", "8");
2040 });
2041 it("Shows the number of bits of entropy for 2 characters of hex with 1 leading zero", function(done) {
2042 testEntropyBits(done, "4A", "8");
2043 });
2044 it("Shows the number of bits of entropy for 2 characters of hex with no leading zeros", function(done) {
2045 testEntropyBits(done, "8A", "8");
2046 });
2047 it("Shows the number of bits of entropy for 2 characters of hex starting with F", function(done) {
2048 testEntropyBits(done, "FA", "8");
2049 });
2050 it("Shows the number of bits of entropy for 4 characters of hex with leading zeros", function(done) {
2051 testEntropyBits(done, "000A", "16");
2052 });
2053 it("Shows the number of bits of entropy for 4 characters of base 6", function(done) {
2054 testEntropyBits(done, "5555", "11");
2055 });
2056 it("Shows the number of bits of entropy for 4 characters of base 6 dice", function(done) {
2057 // uses dice, so entropy is actually 0000 in base 6, which is 4 lots of
2058 // 2.58 bits, which is 10.32 bits (rounded down to 10 bits)
2059 testEntropyBits(done, "6666", "10");
2060 });
2061 it("Shows the number of bits of entropy for 4 charactes of base 10", function(done) {
2062 // Uses base 10, which is 4 lots of 3.32 bits, which is 13.3 bits (rounded
2063 // down to 13)
2064 testEntropyBits(done, "2227", "13");
2065 });
2066 it("Shows the number of bits of entropy for 4 characters of hex with 2 leading zeros", function(done) {
2067 testEntropyBits(done, "222F", "16");
2068 });
2069 it("Shows the number of bits of entropy for 4 characters of hex starting with F", function(done) {
2070 testEntropyBits(done, "FFFF", "16");
2071 });
2072 it("Shows the number of bits of entropy for 10 characters of base 10", function(done) {
2073 // 10 events at 3.32 bits per event
2074 testEntropyBits(done, "0000101017", "33");
2075 });
2076 it("Shows the number of bits of entropy for a full deck of cards", function(done) {
2077 // cards are not replaced, so a full deck is not 52^52 entropy which is 296
2078 // bits, it's 52!, which is 225 bits
2079 testEntropyBits(done, "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks", "225");
2080 });
2081
2082 it("Shows details about the entered entropy", function(done) {
2083 testEntropyFeedback(done,
2084 {
2085 entropy: "A",
2086 filtered: "A",
2087 type: "hexadecimal",
2088 events: "1",
2089 bits: "4",
2090 words: 0,
2091 strength: "less than a second",
2092 }
2093 );
2094 });
2095 it("Shows details about the entered entropy", function(done) {
2096 testEntropyFeedback(done,
2097 {
2098 entropy: "AAAAAAAA",
2099 filtered: "AAAAAAAA",
2100 type: "hexadecimal",
2101 events: "8",
2102 bits: "32",
2103 words: 3,
2104 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2105 }
2106 );
2107 });
2108 it("Shows details about the entered entropy", function(done) {
2109 testEntropyFeedback(done,
2110 {
2111 entropy: "AAAAAAAA B",
2112 filtered: "AAAAAAAAB",
2113 type: "hexadecimal",
2114 events: "9",
2115 bits: "36",
2116 words: 3,
2117 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2118 }
2119 );
2120 });
2121 it("Shows details about the entered entropy", function(done) {
2122 testEntropyFeedback(done,
2123 {
2124 entropy: "AAAAAAAA BBBBBBBB",
2125 filtered: "AAAAAAAABBBBBBBB",
2126 type: "hexadecimal",
2127 events: "16",
2128 bits: "64",
2129 words: 6,
2130 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2131 }
2132 );
2133 });
2134 it("Shows details about the entered entropy", function(done) {
2135 testEntropyFeedback(done,
2136 {
2137 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC",
2138 filtered: "AAAAAAAABBBBBBBBCCCCCCCC",
2139 type: "hexadecimal",
2140 events: "24",
2141 bits: "96",
2142 words: 9,
2143 strength: "less than a second",
2144 }
2145 );
2146 });
2147 it("Shows details about the entered entropy", function(done) {
2148 testEntropyFeedback(done,
2149 {
2150 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDD",
2151 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDD",
2152 type: "hexadecimal",
2153 events: "32",
2154 bits: "128",
2155 words: 12,
2156 strength: "2 minutes",
2157 }
2158 );
2159 });
2160 it("Shows details about the entered entropy", function(done) {
2161 testEntropyFeedback(done,
2162 {
2163 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA",
2164 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDA",
2165 type: "hexadecimal",
2166 events: "32",
2167 bits: "128",
2168 words: 12,
2169 strength: "2 days",
2170 }
2171 );
2172 });
2173 it("Shows details about the entered entropy", function(done) {
2174 testEntropyFeedback(done,
2175 {
2176 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA EEEEEEEE",
2177 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDAEEEEEEEE",
2178 type: "hexadecimal",
2179 events: "40",
2180 bits: "160",
2181 words: 15,
2182 strength: "3 years",
2183 }
2184 );
2185 });
2186 it("Shows details about the entered entropy", function(done) {
2187 testEntropyFeedback(done,
2188 {
2189 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA EEEEEEEE FFFFFFFF",
2190 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDAEEEEEEEEFFFFFFFF",
2191 type: "hexadecimal",
2192 events: "48",
2193 bits: "192",
2194 words: 18,
2195 strength: "centuries",
2196 }
2197 );
2198 });
2199 it("Shows details about the entered entropy", function(done) {
2200 testEntropyFeedback(done,
2201 {
2202 entropy: "7d",
2203 type: "card",
2204 events: "1",
2205 bits: "4",
2206 words: 0,
2207 strength: "less than a second",
2208 }
2209 );
2210 });
2211 it("Shows details about the entered entropy", function(done) {
2212 testEntropyFeedback(done,
2213 {
2214 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2215 type: "card (full deck)",
2216 events: "52",
2217 bits: "225",
2218 words: 21,
2219 strength: "centuries",
2220 }
2221 );
2222 });
2223 it("Shows details about the entered entropy", function(done) {
2224 testEntropyFeedback(done,
2225 {
2226 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks3d",
2227 type: "card (full deck, 1 duplicate: 3d)",
2228 events: "53",
2229 bits: "254",
2230 words: 21,
2231 strength: "centuries",
2232 }
2233 );
2234 });
2235 it("Shows details about the entered entropy", function(done) {
2236 testEntropyFeedback(done,
2237 {
2238 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqs3d4d",
2239 type: "card (2 duplicates: 3d 4d, 1 missing: KS)",
2240 events: "53",
2241 bits: "254",
2242 words: 21,
2243 strength: "centuries",
2244 }
2245 );
2246 });
2247 it("Shows details about the entered entropy", function(done) {
2248 testEntropyFeedback(done,
2249 {
2250 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqs3d4d5d6d",
2251 type: "card (4 duplicates: 3d 4d 5d..., 1 missing: KS)",
2252 events: "55",
2253 bits: "264",
2254 words: 24,
2255 strength: "centuries",
2256 }
2257 );
2258 });
2259 it("Shows details about the entered entropy", function(done) {
2260 testEntropyFeedback(done,
2261 // Next test was throwing uncaught error in zxcvbn
2262 // Also tests 451 bits, ie Math.log2(52!)*2 = 225.58 * 2
2263 {
2264 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsksac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2265 type: "card (full deck, 52 duplicates: ac 2c 3c...)",
2266 events: "104",
2267 bits: "499",
2268 words: 45,
2269 strength: "centuries",
2270 }
2271 );
2272 });
2273 it("Shows details about the entered entropy", function(done) {
2274 testEntropyFeedback(done,
2275 // Case insensitivity to duplicate cards
2276 {
2277 entropy: "asAS",
2278 type: "card (1 duplicate: AS)",
2279 events: "2",
2280 bits: "9",
2281 words: 0,
2282 strength: "less than a second",
2283 }
2284 );
2285 });
2286 it("Shows details about the entered entropy", function(done) {
2287 testEntropyFeedback(done,
2288 {
2289 entropy: "ASas",
2290 type: "card (1 duplicate: as)",
2291 events: "2",
2292 bits: "9",
2293 words: 0,
2294 strength: "less than a second",
2295 }
2296 );
2297 });
2298 it("Shows details about the entered entropy", function(done) {
2299 testEntropyFeedback(done,
2300 // Missing cards are detected
2301 {
2302 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2303 type: "card (1 missing: 9C)",
2304 events: "51",
2305 bits: "221",
2306 words: 18,
2307 strength: "centuries",
2308 }
2309 );
2310 });
2311 it("Shows details about the entered entropy", function(done) {
2312 testEntropyFeedback(done,
2313 {
2314 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2315 type: "card (2 missing: 9C 5D)",
2316 events: "50",
2317 bits: "216",
2318 words: 18,
2319 strength: "centuries",
2320 }
2321 );
2322 });
2323 it("Shows details about the entered entropy", function(done) {
2324 testEntropyFeedback(done,
2325 {
2326 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d7d8d9dtdjd kdah2h3h 5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2327 type: "card (4 missing: 9C 5D QD...)",
2328 events: "48",
2329 bits: "208",
2330 words: 18,
2331 strength: "centuries",
2332 }
2333 );
2334 });
2335 it("Shows details about the entered entropy", function(done) {
2336 testEntropyFeedback(done,
2337 // More than six missing cards does not show message
2338 {
2339 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d 8d9d jd kdah2h3h 5h6h7h8h9hthjhqhkh 2s3s4s5s6s7s8s9stsjsqsks",
2340 type: "card",
2341 events: "45",
2342 bits: "195",
2343 words: 18,
2344 strength: "centuries",
2345 }
2346 );
2347 });
2348 it("Shows details about the entered entropy", function(done) {
2349 testEntropyFeedback(done,
2350 // Multiple decks of cards increases bits per event
2351 {
2352 entropy: "3d",
2353 events: "1",
2354 bits: "4",
2355 bitsPerEvent: "4.34",
2356 }
2357 );
2358 });
2359 it("Shows details about the entered entropy", function(done) {
2360 testEntropyFeedback(done,
2361 {
2362 entropy: "3d3d",
2363 events: "2",
2364 bits: "9",
2365 bitsPerEvent: "4.80",
2366 }
2367 );
2368 });
2369 it("Shows details about the entered entropy", function(done) {
2370 testEntropyFeedback(done,
2371 {
2372 entropy: "3d3d3d",
2373 events: "3",
2374 bits: "15",
2375 bitsPerEvent: "5.01",
2376 }
2377 );
2378 });
2379 it("Shows details about the entered entropy", function(done) {
2380 testEntropyFeedback(done,
2381 {
2382 entropy: "3d3d3d3d",
2383 events: "4",
2384 bits: "20",
2385 bitsPerEvent: "5.14",
2386 }
2387 );
2388 });
2389 it("Shows details about the entered entropy", function(done) {
2390 testEntropyFeedback(done,
2391 {
2392 entropy: "3d3d3d3d3d",
2393 events: "5",
2394 bits: "26",
2395 bitsPerEvent: "5.22",
2396 }
2397 );
2398 });
2399 it("Shows details about the entered entropy", function(done) {
2400 testEntropyFeedback(done,
2401 {
2402 entropy: "3d3d3d3d3d3d",
2403 events: "6",
2404 bits: "31",
2405 bitsPerEvent: "5.28",
2406 }
2407 );
2408 });
2409 it("Shows details about the entered entropy", function(done) {
2410 testEntropyFeedback(done,
2411 {
2412 entropy: "3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d",
2413 events: "33",
2414 bits: "184",
2415 bitsPerEvent: "5.59",
2416 strength: 'less than a second - Repeats like "abcabcabc" are only slightly harder to guess than "abc"',
2417 }
2418 );
2419 });
2420
2421 // Entropy is truncated from the left
2422 it('Truncates entropy from the left', function(done) {
2423 // Truncate from left means 0000 is removed from the start
2424 // which gives mnemonic 'avocado zoo zone'
2425 // not 1111 removed from the end
2426 // which gives the mnemonic 'abstract zoo zoo'
2427 var entropy = "00000000 00000000 00000000 00000000";
2428 entropy += "11111111 11111111 11111111 1111"; // Missing last byte
2429 driver.findElement(By.css('.use-entropy'))
2430 .click();
2431 driver.findElement(By.css('.entropy'))
2432 .sendKeys(entropy);
2433 driver.sleep(generateDelay).then(function() {
2434 driver.findElement(By.css(".phrase"))
2435 .getAttribute("value").then(function(phrase) {
2436 expect(phrase).toBe("avocado zoo zone");
2437 done();
2438 });
2439 });
2440 });
2441
2442 // Very large entropy results in very long mnemonics
2443 it('Converts very long entropy to very long mnemonics', function(done) {
2444 var entropy = "";
2445 for (var i=0; i<33; i++) {
2446 entropy += "AAAAAAAA"; // 3 words * 33 iterations = 99 words
2447 }
2448 driver.findElement(By.css('.use-entropy'))
2449 .click();
2450 driver.findElement(By.css('.entropy'))
2451 .sendKeys(entropy);
2452 driver.sleep(generateDelay).then(function() {
2453 driver.findElement(By.css(".phrase"))
2454 .getAttribute("value").then(function(phrase) {
2455 var wordCount = phrase.split(/\s+/g).length;
2456 expect(wordCount).toBe(99);
2457 done();
2458 });
2459 });
2460 });
2461
2462 // Is compatible with bip32jp entropy
2463 // https://bip32jp.github.io/english/index.html
2464 // NOTES:
2465 // Is incompatible with:
2466 // base 20
2467 it('Is compatible with bip32jp.github.io', function(done) {
2468 var entropy = "543210543210543210543210543210543210543210543210543210543210543210543210543210543210543210543210543";
2469 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";
2470 driver.findElement(By.css('.use-entropy'))
2471 .click();
2472 driver.findElement(By.css('.entropy'))
2473 .sendKeys(entropy);
2474 driver.sleep(generateDelay).then(function() {
2475 driver.findElement(By.css(".phrase"))
2476 .getAttribute("value").then(function(phrase) {
2477 expect(phrase).toBe(expectedPhrase);
2478 done();
2479 });
2480 });
2481 });
2482
2483 // Blank entropy does not generate mnemonic or addresses
2484 it('Does not generate mnemonic for blank entropy', function(done) {
2485 driver.findElement(By.css('.use-entropy'))
2486 .click();
2487 driver.findElement(By.css('.entropy'))
2488 .clear();
2489 // check there is no mnemonic
2490 driver.sleep(generateDelay).then(function() {
2491 driver.findElement(By.css(".phrase"))
2492 .getAttribute("value").then(function(phrase) {
2493 expect(phrase).toBe("");
2494 // check there is no mnemonic
2495 driver.findElements(By.css(".address"))
2496 .then(function(addresses) {
2497 expect(addresses.length).toBe(0);
2498 // Check the feedback says 'blank entropy'
2499 driver.findElement(By.css(".feedback"))
2500 .getText()
2501 .then(function(feedbackText) {
2502 expect(feedbackText).toBe("Blank entropy");
2503 done();
2504 });
2505 })
2506 });
2507 });
2508 });
2509
2510 // Mnemonic length can be selected even for weak entropy
2511 it('Allows selection of mnemonic length even for weak entropy', function(done) {
2512 driver.findElement(By.css('.use-entropy'))
2513 .click();
2514 driver.executeScript(function() {
2515 $(".mnemonic-length").val("18").trigger("change");
2516 });
2517 driver.findElement(By.css('.entropy'))
2518 .sendKeys("012345");
2519 driver.sleep(generateDelay).then(function() {
2520 driver.findElement(By.css(".phrase"))
2521 .getAttribute("value").then(function(phrase) {
2522 var wordCount = phrase.split(/\s+/g).length;
2523 expect(wordCount).toBe(18);
2524 done();
2525 });
2526 });
2527 });
2528
2529 // Github issue 33
2530 // https://github.com/iancoleman/bip39/issues/33
2531 // Final cards should contribute entropy
2532 it('Uses as much entropy as possible for the mnemonic', function(done) {
2533 driver.findElement(By.css('.use-entropy'))
2534 .click();
2535 driver.findElement(By.css('.entropy'))
2536 .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");
2537 driver.sleep(generateDelay).then(function() {
2538 // Get mnemonic
2539 driver.findElement(By.css(".phrase"))
2540 .getAttribute("value").then(function(originalPhrase) {
2541 // Set the last 12 cards to be AS
2542 driver.findElement(By.css('.entropy'))
2543 .clear();
2544 driver.findElement(By.css('.entropy'))
2545 .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");
2546 driver.sleep(generateDelay).then(function() {
2547 // Get new mnemonic
2548 driver.findElement(By.css(".phrase"))
2549 .getAttribute("value").then(function(newPhrase) {
2550 expect(originalPhrase).not.toEqual(newPhrase);
2551 done();
2552 });
2553 });
2554 });
2555 });
2556 });
2557
2558 // Github issue 35
2559 // https://github.com/iancoleman/bip39/issues/35
2560 // QR Code support
2561 // TODO this doesn't work in selenium with firefox
2562 // see https://stackoverflow.com/q/40360223
2563 it('Shows a qr code on hover for the phrase', function(done) {
2564 if (browser == "firefox") {
2565 pending("Selenium + Firefox bug for mouseMove, see https://stackoverflow.com/q/40360223");
2566 }
2567 // generate a random mnemonic
2568 var generateEl = driver.findElement(By.css('.generate'));
2569 generateEl.click();
2570 // toggle qr to show (hidden by default)
2571 var phraseEl = driver.findElement(By.css(".phrase"));
2572 phraseEl.click();
2573 var rootKeyEl = driver.findElement(By.css(".root-key"));
2574 driver.sleep(generateDelay).then(function() {
2575 // hover over the root key
2576 driver.actions().mouseMove(rootKeyEl).perform().then(function() {
2577 // check the qr code shows
2578 driver.executeScript(function() {
2579 return $(".qr-container").find("canvas").length > 0;
2580 })
2581 .then(function(qrShowing) {
2582 expect(qrShowing).toBe(true);
2583 // hover away from the phrase
2584 driver.actions().mouseMove(generateEl).perform().then(function() {;
2585 // check the qr code hides
2586 driver.executeScript(function() {
2587 return $(".qr-container").find("canvas").length == 0;
2588 })
2589 .then(function(qrHidden) {
2590 expect(qrHidden).toBe(true);
2591 done();
2592 });
2593 });
2594 });
2595 });
2596 });
2597 });
2598
2599 // BIP44 account extendend private key is shown
2600 // github issue 37 - compatibility with electrum
2601 it('Shows the bip44 account extended private key', function(done) {
2602 driver.findElement(By.css(".phrase"))
2603 .sendKeys("abandon abandon ability");
2604 driver.sleep(generateDelay).then(function() {
2605 driver.findElement(By.css("#bip44 .account-xprv"))
2606 .getAttribute("value")
2607 .then(function(xprv) {
2608 expect(xprv).toBe("xprv9yzrnt4zWVJUr1k2VxSPy9ettKz5PpeDMgaVG7UKedhqnw1tDkxP2UyYNhuNSumk2sLE5ctwKZs9vwjsq3e1vo9egCK6CzP87H2cVYXpfwQ");
2609 done();
2610 });
2611 });
2612 });
2613
2614 // BIP44 account extendend public key is shown
2615 // github issue 37 - compatibility with electrum
2616 it('Shows the bip44 account extended public key', function(done) {
2617 driver.findElement(By.css(".phrase"))
2618 .sendKeys("abandon abandon ability");
2619 driver.sleep(generateDelay).then(function() {
2620 driver.findElement(By.css("#bip44 .account-xpub"))
2621 .getAttribute("value")
2622 .then(function(xprv) {
2623 expect(xprv).toBe("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2624 done();
2625 });
2626 });
2627 });
2628
2629 // github issue 40
2630 // BIP32 root key can be set as an xpub
2631 it('Generates addresses from xpub as bip32 root key', function(done) {
2632 driver.findElement(By.css('#bip32-tab a'))
2633 .click();
2634 // set xpub for account 0 of bip44 for 'abandon abandon ability'
2635 driver.findElement(By.css("#root-key"))
2636 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2637 driver.sleep(generateDelay).then(function() {
2638 // check the addresses are generated
2639 getFirstAddress(function(address) {
2640 expect(address).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug");
2641 // check the xprv key is not set
2642 driver.findElement(By.css(".extended-priv-key"))
2643 .getAttribute("value")
2644 .then(function(xprv) {
2645 expect(xprv).toBe("NA");
2646 // check the private key is not set
2647 driver.findElements(By.css(".privkey"))
2648 .then(function(els) {
2649 els[0]
2650 .getText()
2651 .then(function(privkey) {
2652 expect(xprv).toBe("NA");
2653 done();
2654 });
2655 });
2656 });
2657 });
2658 });
2659 });
2660
2661 // github issue 40
2662 // xpub for bip32 root key will not work with hardened derivation paths
2663 it('Shows error for hardened derivation paths with xpub root key', function(done) {
2664 // set xpub for account 0 of bip44 for 'abandon abandon ability'
2665 driver.findElement(By.css("#root-key"))
2666 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2667 driver.sleep(feedbackDelay).then(function() {
2668 // Check feedback is correct
2669 driver.findElement(By.css('.feedback'))
2670 .getText()
2671 .then(function(feedback) {
2672 var msg = "Hardened derivation path is invalid with xpub key";
2673 expect(feedback).toBe(msg);
2674 // Check no addresses are shown
2675 driver.findElements(By.css('.addresses tr'))
2676 .then(function(rows) {
2677 expect(rows.length).toBe(0);
2678 done();
2679 });
2680 });
2681 });
2682 });
2683
2684 // github issue 39
2685 // no root key shows feedback
2686 it('Shows feedback for no root key', function(done) {
2687 // set xpub for account 0 of bip44 for 'abandon abandon ability'
2688 driver.findElement(By.css('#bip32-tab a'))
2689 .click();
2690 driver.sleep(feedbackDelay).then(function() {
2691 // Check feedback is correct
2692 driver.findElement(By.css('.feedback'))
2693 .getText()
2694 .then(function(feedback) {
2695 expect(feedback).toBe("Invalid root key");
2696 done();
2697 });
2698 });
2699 });
2700
2701 // Github issue 44
2702 // display error switching tabs while addresses are generating
2703 it('Can change details while old addresses are still being generated', function(done) {
2704 // Set to generate 199 more addresses.
2705 // This will take a long time allowing a new set of addresses to be
2706 // generated midway through this lot.
2707 // The newly generated addresses should not include any from the old set.
2708 // Any more than 199 will show an alert which needs to be accepted.
2709 driver.findElement(By.css('.rows-to-add'))
2710 .clear();
2711 driver.findElement(By.css('.rows-to-add'))
2712 .sendKeys('199');
2713 // set the prhase
2714 driver.findElement(By.css('.phrase'))
2715 .sendKeys("abandon abandon ability");
2716 driver.sleep(generateDelay).then(function() {
2717 // change tabs which should cancel the previous generating
2718 driver.findElement(By.css('.rows-to-add'))
2719 .clear();
2720 driver.findElement(By.css('.rows-to-add'))
2721 .sendKeys('20');
2722 driver.findElement(By.css('#bip32-tab a'))
2723 .click()
2724 driver.sleep(generateDelay).then(function() {
2725 driver.findElements(By.css('.index'))
2726 .then(function(els) {
2727 // check the derivation paths have the right quantity
2728 expect(els.length).toBe(20);
2729 // check the derivation paths are in order
2730 testRowsAreInCorrectOrder(done);
2731 });
2732 });
2733 });
2734 }, generateDelay + 5000);
2735
2736 // Github issue 49
2737 // padding for binary should give length with multiple of 256
2738 // hashed entropy 1111 is length 252, so requires 4 leading zeros
2739 // prior to issue 49 it would only generate 2 leading zeros, ie missing 2
2740 it('Pads hashed entropy with leading zeros', function(done) {
2741 driver.findElement(By.css('.use-entropy'))
2742 .click();
2743 driver.executeScript(function() {
2744 $(".mnemonic-length").val("15").trigger("change");
2745 });
2746 driver.findElement(By.css('.entropy'))
2747 .sendKeys("1111");
2748 driver.sleep(generateDelay).then(function() {
2749 driver.findElement(By.css('.phrase'))
2750 .getAttribute("value")
2751 .then(function(phrase) {
2752 expect(phrase).toBe("avocado valid quantum cross link predict excuse edit street able flame large galaxy ginger nuclear");
2753 done();
2754 });
2755 });
2756 });
2757
2758 // Github pull request 55
2759 // https://github.com/iancoleman/bip39/pull/55
2760 // Client select
2761 it('Can set the derivation path on bip32 tab for bitcoincore', function(done) {
2762 testClientSelect(done, {
2763 selectValue: "0",
2764 bip32path: "m/0'/0'",
2765 useHardenedAddresses: "true",
2766 });
2767 });
2768 it('Can set the derivation path on bip32 tab for multibit', function(done) {
2769 testClientSelect(done, {
2770 selectValue: "2",
2771 bip32path: "m/0'/0",
2772 useHardenedAddresses: null,
2773 });
2774 });
2775
2776 // github issue 58
2777 // https://github.com/iancoleman/bip39/issues/58
2778 // bip32 derivation is correct, does not drop leading zeros
2779 // see also
2780 // https://medium.com/@alexberegszaszi/why-do-my-bip32-wallets-disagree-6f3254cc5846
2781 it('Retains leading zeros for bip32 derivation', function(done) {
2782 driver.findElement(By.css(".phrase"))
2783 .sendKeys("fruit wave dwarf banana earth journey tattoo true farm silk olive fence");
2784 driver.findElement(By.css(".passphrase"))
2785 .sendKeys("banana");
2786 driver.sleep(generateDelay).then(function() {
2787 getFirstAddress(function(address) {
2788 // Note that bitcore generates an incorrect address
2789 // 13EuKhffWkBE2KUwcbkbELZb1MpzbimJ3Y
2790 // see the medium.com link above for more details
2791 expect(address).toBe("17rxURoF96VhmkcEGCj5LNQkmN9HVhWb7F");
2792 done();
2793 });
2794 });
2795 });
2796
2797 // github issue 60
2798 // Japanese mnemonics generate incorrect bip32 seed
2799 // BIP39 seed is set from phrase
2800 it('Generates correct seed for Japanese mnemonics', function(done) {
2801 driver.findElement(By.css(".phrase"))
2802 .sendKeys("あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あおぞら");
2803 driver.findElement(By.css(".passphrase"))
2804 .sendKeys("メートルガバヴァぱばぐゞちぢ十人十色");
2805 driver.sleep(generateDelay).then(function() {
2806 driver.findElement(By.css(".seed"))
2807 .getAttribute("value")
2808 .then(function(seed) {
2809 expect(seed).toBe("a262d6fb6122ecf45be09c50492b31f92e9beb7d9a845987a02cefda57a15f9c467a17872029a9e92299b5cbdf306e3a0ee620245cbd508959b6cb7ca637bd55");
2810 done();
2811 });
2812 });
2813 });
2814
2815 // BIP49 official test vectors
2816 // https://github.com/bitcoin/bips/blob/master/bip-0049.mediawiki#test-vectors
2817 it('Generates BIP49 addresses matching the official test vectors', function(done) {
2818 driver.findElement(By.css('#bip49-tab a'))
2819 .click();
2820 selectNetwork("BTC - Bitcoin Testnet");
2821 driver.findElement(By.css(".phrase"))
2822 .sendKeys("abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about");
2823 driver.sleep(generateDelay).then(function() {
2824 getFirstAddress(function(address) {
2825 expect(address).toBe("2Mww8dCYPUpKHofjgcXcBCEGmniw9CoaiD2");
2826 done();
2827 });
2828 });
2829 });
2830
2831 // BIP49 derivation path is shown
2832 it('Shows the bip49 derivation path', function(done) {
2833 driver.findElement(By.css('#bip49-tab a'))
2834 .click();
2835 driver.findElement(By.css(".phrase"))
2836 .sendKeys("abandon abandon ability");
2837 driver.sleep(generateDelay).then(function() {
2838 driver.findElement(By.css('#bip49 .path'))
2839 .getAttribute("value")
2840 .then(function(path) {
2841 expect(path).toBe("m/49'/0'/0'/0");
2842 done();
2843 });
2844 });
2845 });
2846
2847 // BIP49 extended private key is shown
2848 it('Shows the bip49 extended private key', function(done) {
2849 driver.findElement(By.css('#bip49-tab a'))
2850 .click();
2851 driver.findElement(By.css(".phrase"))
2852 .sendKeys("abandon abandon ability");
2853 driver.sleep(generateDelay).then(function() {
2854 driver.findElement(By.css('.extended-priv-key'))
2855 .getAttribute("value")
2856 .then(function(xprv) {
2857 expect(xprv).toBe("yprvALYB4DYRG6CzzVgzQZwwqjAA2wjBGC3iEd7KYYScpoDdmf75qMRWZWxoFcRXBJjgEXdFqJ9vDRGRLJQsrL22Su5jMbNFeM9vetaGVqy9Qy2");
2858 done();
2859 });
2860 });
2861 });
2862
2863 // BIP49 extended public key is shown
2864 it('Shows the bip49 extended public key', function(done) {
2865 driver.findElement(By.css('#bip49-tab a'))
2866 .click();
2867 driver.findElement(By.css(".phrase"))
2868 .sendKeys("abandon abandon ability");
2869 driver.sleep(generateDelay).then(function() {
2870 driver.findElement(By.css('.extended-pub-key'))
2871 .getAttribute("value")
2872 .then(function(xprv) {
2873 expect(xprv).toBe("ypub6ZXXTj5K6TmJCymTWbUxCs6tayZffemZbr2vLvrEP8kceTSENtjm7KHH6thvAKxVar9fGe8rgsPEX369zURLZ68b4f7Vexz7RuXsjQ69YDt");
2874 done();
2875 });
2876 });
2877 });
2878
2879 // BIP49 account field changes address list
2880 it('Can set the bip49 account field', function(done) {
2881 driver.findElement(By.css('#bip49-tab a'))
2882 .click();
2883 driver.findElement(By.css('#bip49 .account'))
2884 .clear();
2885 driver.findElement(By.css('#bip49 .account'))
2886 .sendKeys("1");
2887 driver.findElement(By.css(".phrase"))
2888 .sendKeys("abandon abandon ability");
2889 driver.sleep(generateDelay).then(function() {
2890 getFirstAddress(function(address) {
2891 expect(address).toBe("381wg1GGN4rP88rNC9v7QWsiww63yLVPsn");
2892 done();
2893 });
2894 });
2895 });
2896
2897 // BIP49 change field changes address list
2898 it('Can set the bip49 change field', function(done) {
2899 driver.findElement(By.css('#bip49-tab a'))
2900 .click();
2901 driver.findElement(By.css('#bip49 .change'))
2902 .clear();
2903 driver.findElement(By.css('#bip49 .change'))
2904 .sendKeys("1");
2905 driver.findElement(By.css(".phrase"))
2906 .sendKeys("abandon abandon ability");
2907 driver.sleep(generateDelay).then(function() {
2908 getFirstAddress(function(address) {
2909 expect(address).toBe("3PEM7MiKed5konBoN66PQhK8r3hjGhy9dT");
2910 done();
2911 });
2912 });
2913 });
2914
2915 // BIP49 account extendend private key is shown
2916 it('Shows the bip49 account extended private key', function(done) {
2917 driver.findElement(By.css('#bip49-tab a'))
2918 .click();
2919 driver.findElement(By.css(".phrase"))
2920 .sendKeys("abandon abandon ability");
2921 driver.sleep(generateDelay).then(function() {
2922 driver.findElement(By.css('#bip49 .account-xprv'))
2923 .getAttribute("value")
2924 .then(function(xprv) {
2925 expect(xprv).toBe("yprvAHtB1M5Wp675aLzFy9TJYK2mSsLkg6mcBRh5DZTR7L4EnYSmYPaL63KFA4ycg1PngW5LfkmejxzosCs17TKZMpRFKc3z5SJar6QAKaFcaZL");
2926 done();
2927 });
2928 });
2929 });
2930
2931 // BIP49 account extendend public key is shown
2932 it('Shows the bip49 account extended public key', function(done) {
2933 driver.findElement(By.css('#bip49-tab a'))
2934 .click();
2935 driver.findElement(By.css(".phrase"))
2936 .sendKeys("abandon abandon ability");
2937 driver.sleep(generateDelay).then(function() {
2938 driver.findElement(By.css('#bip49 .account-xpub'))
2939 .getAttribute("value")
2940 .then(function(xprv) {
2941 expect(xprv).toBe("ypub6WsXQrcQeTfNnq4j5AzJuSyVzuBF5ZVTYecg1ws2ffbDfLmv5vtadqdj1NgR6C6gufMpMfJpHxvb6JEQKvETVNWCRanNedfJtnTchZiJtsL");
2942 done();
2943 });
2944 });
2945 });
2946
2947 // Test selecting coin where bip49 is unavailable (eg CLAM)
2948 it('Shows an error on bip49 tab for coins without bip49', function(done) {
2949 driver.findElement(By.css('#bip49-tab a'))
2950 .click();
2951 driver.findElement(By.css(".phrase"))
2952 .sendKeys("abandon abandon ability");
2953 driver.sleep(generateDelay).then(function() {
2954 selectNetwork("CLAM - Clams");
2955 // bip49 available is hidden
2956 driver.findElement(By.css('#bip49 .available'))
2957 .getAttribute("class")
2958 .then(function(classes) {
2959 expect(classes).toContain("hidden");
2960 // bip49 unavailable is shown
2961 driver.findElement(By.css('#bip49 .unavailable'))
2962 .getAttribute("class")
2963 .then(function(classes) {
2964 expect(classes).not.toContain("hidden");
2965 // check there are no addresses shown
2966 driver.findElements(By.css('.addresses tr'))
2967 .then(function(rows) {
2968 expect(rows.length).toBe(0);
2969 // check the derived private key is blank
2970 driver.findElement(By.css('.extended-priv-key'))
2971 .getAttribute("value")
2972 .then(function(xprv) {
2973 expect(xprv).toBe('');
2974 // check the derived public key is blank
2975 driver.findElement(By.css('.extended-pub-key'))
2976 .getAttribute("value")
2977 .then(function(xpub) {
2978 expect(xpub).toBe('');
2979 done();
2980 });
2981 });
2982 })
2983 });
2984 });
2985 });
2986 });
2987
2988 // github issue 43
2989 // Cleared mnemonic and root key still allows addresses to be generated
2990 // https://github.com/iancoleman/bip39/issues/43
2991 it('Clears old root keys from memory when mnemonic is cleared', function(done) {
2992 // set the phrase
2993 driver.findElement(By.css(".phrase"))
2994 .sendKeys("abandon abandon ability");
2995 driver.sleep(generateDelay).then(function() {
2996 // clear the mnemonic and root key
2997 // using selenium .clear() doesn't seem to trigger the 'input' event
2998 // so clear it using keys instead
2999 driver.findElement(By.css('.phrase'))
3000 .sendKeys(Key.CONTROL,"a");
3001 driver.findElement(By.css('.phrase'))
3002 .sendKeys(Key.DELETE);
3003 driver.findElement(By.css('.root-key'))
3004 .sendKeys(Key.CONTROL,"a");
3005 driver.findElement(By.css('.root-key'))
3006 .sendKeys(Key.DELETE);
3007 driver.sleep(generateDelay).then(function() {
3008 // try to generate more addresses
3009 driver.findElement(By.css('.more'))
3010 .click();
3011 driver.sleep(generateDelay).then(function() {
3012 driver.findElements(By.css(".addresses tr"))
3013 .then(function(els) {
3014 // check there are no addresses shown
3015 expect(els.length).toBe(0);
3016 done();
3017 });
3018 });
3019 });
3020 });
3021 });
3022
3023 // Github issue 95
3024 // error trying to generate addresses from xpub with hardened derivation
3025 it('Shows error for hardened addresses with xpub root key', function(done) {
3026 driver.findElement(By.css('#bip32-tab a'))
3027 .click()
3028 driver.executeScript(function() {
3029 $(".hardened-addresses").prop("checked", true);
3030 });
3031 // set xpub for account 0 of bip44 for 'abandon abandon ability'
3032 driver.findElement(By.css("#root-key"))
3033 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
3034 driver.sleep(feedbackDelay).then(function() {
3035 // Check feedback is correct
3036 driver.findElement(By.css('.feedback'))
3037 .getText()
3038 .then(function(feedback) {
3039 var msg = "Hardened derivation path is invalid with xpub key";
3040 expect(feedback).toBe(msg);
3041 done();
3042 });
3043 });
3044 });
3045
3046 // Litecoin uses ltub by default, and can optionally be set to xprv
3047 // github issue 96
3048 // https://github.com/iancoleman/bip39/issues/96
3049 // Issue with extended keys on Litecoin
3050 it('Uses ltub by default for litecoin, but can be set to xprv', function(done) {
3051 driver.findElement(By.css('.phrase'))
3052 .sendKeys("abandon abandon ability");
3053 selectNetwork("LTC - Litecoin");
3054 driver.sleep(generateDelay).then(function() {
3055 // check the extended key is generated correctly
3056 driver.findElement(By.css('.root-key'))
3057 .getAttribute("value")
3058 .then(function(rootKey) {
3059 expect(rootKey).toBe("Ltpv71G8qDifUiNesiPqf6h5V6eQ8ic77oxQiYtawiACjBEx3sTXNR2HGDGnHETYxESjqkMLFBkKhWVq67ey1B2MKQXannUqNy1RZVHbmrEjnEU");
3060 // set litecoin to use ltub
3061 driver.executeScript(function() {
3062 $(".litecoin-use-ltub").prop("checked", false);
3063 $(".litecoin-use-ltub").trigger("change");
3064 });
3065 driver.sleep(generateDelay).then(function() {
3066 driver.findElement(By.css('.root-key'))
3067 .getAttribute("value")
3068 .then(function(rootKey) {
3069 expect(rootKey).toBe("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
3070 done();
3071 });
3072 })
3073 });
3074 });
3075 });
3076
3077 // github issue 99
3078 // https://github.com/iancoleman/bip39/issues/99#issuecomment-327094159
3079 // "warn me emphatically when they have detected invalid input" to the entropy field
3080 // A warning is shown when entropy is filtered and discarded
3081 it('Warns when entropy is filtered and discarded', function(done) {
3082 driver.findElement(By.css('.use-entropy'))
3083 .click();
3084 // set entropy to have no filtered content
3085 driver.findElement(By.css('.entropy'))
3086 .sendKeys("00000000 00000000 00000000 00000000");
3087 driver.sleep(generateDelay).then(function() {
3088 // check the filter warning does not show
3089 driver.findElement(By.css('.entropy-container .filter-warning'))
3090 .getAttribute("class")
3091 .then(function(classes) {
3092 expect(classes).toContain("hidden");
3093 // set entropy to have some filtered content
3094 driver.findElement(By.css('.entropy'))
3095 .sendKeys("10000000 zxcvbn 00000000 00000000 00000000");
3096 driver.sleep(entropyFeedbackDelay).then(function() {
3097 // check the filter warning shows
3098 driver.findElement(By.css('.entropy-container .filter-warning'))
3099 .getAttribute("class")
3100 .then(function(classes) {
3101 expect(classes).not.toContain("hidden");
3102 done();
3103 });
3104 });
3105 });
3106 });
3107 });
3108
3109 // Bitcoin Cash address can be set to use bitpay format
3110 it('Can use bitpay format for bitcoin cash addresses', function(done) {
3111 driver.executeScript(function() {
3112 $(".use-bitpay-addresses").prop("checked", true);
3113 });
3114 driver.findElement(By.css('.phrase'))
3115 .sendKeys("abandon abandon ability");
3116 selectNetwork("BCH - Bitcoin Cash");
3117 driver.sleep(generateDelay).then(function() {
3118 getFirstAddress(function(address) {
3119 expect(address).toBe("CZnpA9HPmvhuhLLPWJP8rNDpLUYXy1LXFk");
3120 done();
3121 });
3122 });
3123 });
3124
3125 // End of tests ported from old suit, so no more comments above each test now
3126
3127 it('Can generate more addresses from a custom index', function(done) {
3128 var expectedIndexes = [
3129 0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,
3130 40,41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59
3131 ];
3132 driver.findElement(By.css('.phrase'))
3133 .sendKeys("abandon abandon ability");
3134 driver.sleep(generateDelay).then(function() {
3135 // Set start of next lot of rows to be from index 40
3136 // which means indexes 20-39 will not be in the table.
3137 driver.findElement(By.css('.more-rows-start-index'))
3138 .sendKeys("40");
3139 driver.findElement(By.css('.more'))
3140 .click();
3141 driver.sleep(generateDelay).then(function() {
3142 // Check actual indexes in the table match the expected pattern
3143 driver.findElements(By.css(".index"))
3144 .then(function(els) {
3145 expect(els.length).toBe(expectedIndexes.length);
3146 var testRowAtIndex = function(i) {
3147 if (i >= expectedIndexes.length) {
3148 done();
3149 }
3150 else {
3151 els[i].getText()
3152 .then(function(actualPath) {
3153 var noHardened = actualPath.replace(/'/g, "");
3154 var pathBits = noHardened.split("/")
3155 var lastBit = pathBits[pathBits.length-1];
3156 var actualIndex = parseInt(lastBit);
3157 var expectedIndex = expectedIndexes[i];
3158 expect(actualIndex).toBe(expectedIndex);
3159 testRowAtIndex(i+1);
3160 });
3161 }
3162 }
3163 testRowAtIndex(0);
3164 });
3165 });
3166 });
3167 });
3168
3169 it('Can generate BIP141 addresses with P2WPKH-in-P2SH semanitcs', function(done) {
3170 // Sourced from BIP49 official test specs
3171 driver.findElement(By.css('#bip141-tab a'))
3172 .click();
3173 driver.findElement(By.css('.bip141-path'))
3174 .clear();
3175 driver.findElement(By.css('.bip141-path'))
3176 .sendKeys("m/49'/1'/0'/0");
3177 selectNetwork("BTC - Bitcoin Testnet");
3178 driver.findElement(By.css(".phrase"))
3179 .sendKeys("abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about");
3180 driver.sleep(generateDelay).then(function() {
3181 getFirstAddress(function(address) {
3182 expect(address).toBe("2Mww8dCYPUpKHofjgcXcBCEGmniw9CoaiD2");
3183 done();
3184 });
3185 });
3186 });
3187
3188 it('Can generate BIP141 addresses with P2WPKH semanitcs', function(done) {
3189 // This result tested against bitcoinjs-lib test spec for segwit address
3190 // using the first private key of this mnemonic and default path m/0
3191 // https://github.com/bitcoinjs/bitcoinjs-lib/blob/9c8503cab0c6c30a95127042703bc18e8d28c76d/test/integration/addresses.js#L50
3192 // so whilst not directly comparable, substituting the private key produces
3193 // identical results between this tool and the bitcoinjs-lib test.
3194 // Private key generated is:
3195 // L3L8Nu9whawPBNLGtFqDhKut9DKKfG3CQoysupT7BimqVCZsLFNP
3196 driver.findElement(By.css('#bip141-tab a'))
3197 .click();
3198 // Choose P2WPKH
3199 driver.executeScript(function() {
3200 $(".bip141-semantics option[selected]").removeAttr("selected");
3201 $(".bip141-semantics option").filter(function(i,e) {
3202 return $(e).html() == "P2WPKH";
3203 }).prop("selected", true);
3204 $(".bip141-semantics").trigger("change");
3205 });
3206 driver.findElement(By.css(".phrase"))
3207 .sendKeys("abandon abandon ability");
3208 driver.sleep(generateDelay).then(function() {
3209 getFirstAddress(function(address) {
3210 expect(address).toBe("bc1qfwu6a5a3evygrk8zvdxxvz4547lmpyx5vsfxe9");
3211 done();
3212 });
3213 });
3214 });
3215
3216 it('Shows the entropy used by the PRNG when clicking generate', function(done) {
3217 driver.findElement(By.css('.generate')).click();
3218 driver.sleep(generateDelay).then(function() {
3219 driver.findElement(By.css('.entropy'))
3220 .getAttribute("value")
3221 .then(function(entropy) {
3222 expect(entropy).not.toBe("");
3223 done();
3224 });
3225 });
3226 });
3227
3228 it('Shows the index of each word in the mnemonic', function(done) {
3229 driver.findElement(By.css('.phrase'))
3230 .sendKeys("abandon abandon ability");
3231 driver.sleep(generateDelay).then(function() {
3232 driver.findElement(By.css('.use-entropy'))
3233 .click();
3234 driver.findElement(By.css('.word-indexes'))
3235 .getText()
3236 .then(function(indexes) {
3237 expect(indexes).toBe("0, 0, 1");
3238 done();
3239 });
3240 });
3241 });
3242
3243 it('Shows the derivation path for bip84 tab', function(done) {
3244 driver.findElement(By.css('#bip84-tab a'))
3245 .click()
3246 driver.findElement(By.css('.phrase'))
3247 .sendKeys('abandon abandon ability');
3248 driver.sleep(generateDelay).then(function() {
3249 driver.findElement(By.css('#bip84 .path'))
3250 .getAttribute("value")
3251 .then(function(path) {
3252 expect(path).toBe("m/84'/0'/0'/0");
3253 done();
3254 })
3255 });
3256 });
3257
3258 it('Shows the extended private key for bip84 tab', function(done) {
3259 driver.findElement(By.css('#bip84-tab a'))
3260 .click()
3261 driver.findElement(By.css('.phrase'))
3262 .sendKeys('abandon abandon ability');
3263 driver.sleep(generateDelay).then(function() {
3264 driver.findElement(By.css('.extended-priv-key'))
3265 .getAttribute("value")
3266 .then(function(path) {
3267 expect(path).toBe("zprvAev3RKrZ3QVKiUFCfdeMRen1BPDJgdNt1XpxiDy8acSs4kkAGTCvq7HeRYRNNpo8EtEjCFQBWavJwtCUR29y4TUCH4X5RXMcyq48uN8y9BP");
3268 done();
3269 })
3270 });
3271 });
3272
3273 it('Shows the extended public key for bip84 tab', function(done) {
3274 driver.findElement(By.css('#bip84-tab a'))
3275 .click()
3276 driver.findElement(By.css('.phrase'))
3277 .sendKeys('abandon abandon ability');
3278 driver.sleep(generateDelay).then(function() {
3279 driver.findElement(By.css('.extended-pub-key'))
3280 .getAttribute("value")
3281 .then(function(path) {
3282 expect(path).toBe("zpub6suPpqPSsn3cvxKfmfBMnnijjR3o666jNkkZWcNk8wyqwZ5JozXBNuc8Gs7DB3uLwTDvGVTspVEAUQcEjKF3pZHgywVbubdTqbXTUg7usyx");
3283 done();
3284 })
3285 });
3286 });
3287
3288 it('Changes the address list if bip84 account is changed', function(done) {
3289 driver.findElement(By.css('#bip84-tab a'))
3290 .click()
3291 driver.findElement(By.css('#bip84 .account'))
3292 .sendKeys('1');
3293 driver.findElement(By.css('.phrase'))
3294 .sendKeys('abandon abandon ability');
3295 driver.sleep(generateDelay).then(function() {
3296 getFirstAddress(function(address) {
3297 expect(address).toBe("bc1qp7vv669t2fy965jdzvqwrraana89ctd5ewc662");
3298 done();
3299 });
3300 });
3301 });
3302
3303 it('Changes the address list if bip84 change is changed', function(done) {
3304 driver.findElement(By.css('#bip84-tab a'))
3305 .click()
3306 driver.findElement(By.css('#bip84 .change'))
3307 .sendKeys('1');
3308 driver.findElement(By.css('.phrase'))
3309 .sendKeys('abandon abandon ability');
3310 driver.sleep(generateDelay).then(function() {
3311 getFirstAddress(function(address) {
3312 expect(address).toBe("bc1qr39vj6rh06ff05m53uxq8uazehwhccswylhrs2");
3313 done();
3314 });
3315 });
3316 });
3317
3318 it('Passes the official BIP84 test spec for rootpriv', function(done) {
3319 driver.findElement(By.css('#bip84-tab a'))
3320 .click()
3321 driver.findElement(By.css('.phrase'))
3322 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3323 driver.sleep(generateDelay).then(function() {
3324 driver.findElement(By.css(".root-key"))
3325 .getAttribute("value")
3326 .then(function(rootKey) {
3327 expect(rootKey).toBe("zprvAWgYBBk7JR8Gjrh4UJQ2uJdG1r3WNRRfURiABBE3RvMXYSrRJL62XuezvGdPvG6GFBZduosCc1YP5wixPox7zhZLfiUm8aunE96BBa4Kei5");
3328 done();
3329 })
3330 });
3331 });
3332
3333 it('Passes the official BIP84 test spec for account 0 xprv', function(done) {
3334 driver.findElement(By.css('#bip84-tab a'))
3335 .click()
3336 driver.findElement(By.css('.phrase'))
3337 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3338 driver.sleep(generateDelay).then(function() {
3339 driver.findElement(By.css("#bip84 .account-xprv"))
3340 .getAttribute("value")
3341 .then(function(rootKey) {
3342 expect(rootKey).toBe("zprvAdG4iTXWBoARxkkzNpNh8r6Qag3irQB8PzEMkAFeTRXxHpbF9z4QgEvBRmfvqWvGp42t42nvgGpNgYSJA9iefm1yYNZKEm7z6qUWCroSQnE");
3343 done();
3344 })
3345 });
3346 });
3347
3348 it('Passes the official BIP84 test spec for account 0 xpub', function(done) {
3349 driver.findElement(By.css('#bip84-tab a'))
3350 .click()
3351 driver.findElement(By.css('.phrase'))
3352 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3353 driver.sleep(generateDelay).then(function() {
3354 driver.findElement(By.css("#bip84 .account-xpub"))
3355 .getAttribute("value")
3356 .then(function(rootKey) {
3357 expect(rootKey).toBe("zpub6rFR7y4Q2AijBEqTUquhVz398htDFrtymD9xYYfG1m4wAcvPhXNfE3EfH1r1ADqtfSdVCToUG868RvUUkgDKf31mGDtKsAYz2oz2AGutZYs");
3358 done();
3359 })
3360 });
3361 });
3362
3363 it('Passes the official BIP84 test spec for account 0 first address', function(done) {
3364 driver.findElement(By.css('#bip84-tab a'))
3365 .click()
3366 driver.findElement(By.css('.phrase'))
3367 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3368 driver.sleep(generateDelay).then(function() {
3369 getFirstAddress(function(address) {
3370 expect(address).toBe("bc1qcr8te4kr609gcawutmrza0j4xv80jy8z306fyu");
3371 done();
3372 });
3373 });
3374 });
3375
3376 it('Passes the official BIP84 test spec for account 0 first change address', function(done) {
3377 driver.findElement(By.css('#bip84-tab a'))
3378 .click()
3379 driver.findElement(By.css('.phrase'))
3380 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3381 driver.findElement(By.css('#bip84 .change'))
3382 .sendKeys('1');
3383 driver.sleep(generateDelay).then(function() {
3384 getFirstAddress(function(address) {
3385 expect(address).toBe("bc1q8c6fshw2dlwun7ekn9qwf37cu2rn755upcp6el");
3386 done();
3387 });
3388 });
3389 });
3390
3391 it('Can display the table as csv', function(done) {
3392 var headings = "path,address,public key,private key";
3393 var row1 = "m/44'/0'/0'/0/0,1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug,033f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3,L26cVSpWFkJ6aQkPkKmTzLqTdLJ923e6CzrVh9cmx21QHsoUmrEE";
3394 var row20 = "m/44'/0'/0'/0/19,1KhBy28XLAciXnnRvm71PvQJaETyrxGV55,02b4b3e396434d8cdd20c03ac4aaa07387784d5d867b75987f516f5705ee68cb3a,L4GrDrjReMsCAu5DkLXn79jSb95qR7Zfx7eshybCQZ1qL32MXJab";
3395 driver.findElement(By.css('.phrase'))
3396 .sendKeys('abandon abandon ability');
3397 driver.sleep(generateDelay).then(function() {
3398 driver.findElement(By.css('.csv'))
3399 .getAttribute("value")
3400 .then(function(csv) {
3401 expect(csv).toContain(headings);
3402 expect(csv).toContain(row1);
3403 expect(csv).toContain(row20);
3404 done();
3405 });
3406 });
3407 });
3408
3409 it('LeftPads ethereum keys that are less than 32 bytes', function(done) {
3410 // see https://github.com/iancoleman/bip39/issues/155
3411 selectNetwork("ETH - Ethereum");
3412 driver.findElement(By.css('#bip32-tab a'))
3413 .click()
3414 driver.findElement(By.css('#bip32-path'))
3415 .clear();
3416 driver.findElement(By.css('#bip32-path'))
3417 .sendKeys("m/44'/60'/0'");
3418 driver.findElement(By.css('.phrase'))
3419 .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');
3420 driver.sleep(generateDelay).then(function() {
3421 getFirstAddress(function(address) {
3422 expect(address).toBe("0x8943E785B4a5714FC87a3aFAad1eB1FeB602B118");
3423 done();
3424 });
3425 });
3426 });
3427
3428 it('Can encrypt private keys using BIP38', function(done) {
3429 // see https://github.com/iancoleman/bip39/issues/140
3430 driver.executeScript(function() {
3431 $(".use-bip38").prop("checked", true);
3432 });
3433 driver.findElement(By.css('.bip38-password'))
3434 .sendKeys('bip38password');
3435 driver.findElement(By.css('.rows-to-add'))
3436 .clear();
3437 driver.findElement(By.css('.rows-to-add'))
3438 .sendKeys('1');
3439 driver.findElement(By.css('.phrase'))
3440 .sendKeys('abandon abandon ability');
3441 driver.sleep(bip38delay).then(function() {
3442 // address
3443 getFirstRowValue(function(address) {
3444 expect(address).toBe("1NCvSdumA3ngMM9c4aqU56AM6rqXddfuXB");
3445 // pubkey
3446 getFirstRowValue(function(pubkey) {
3447 expect(pubkey).toBe("043f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3884a74447ea901729b1e73a999b7520e7cb55b4120e6432c64153ccab8a848e1");
3448 // privkey
3449 getFirstRowValue(function(privkey) {
3450 expect(privkey).toBe("6PRNRiFnj1RoR3sXhymdCvoZCgnUHQpfupNdKkFbWJkwWQEKesWt1EDMDM");
3451 done();
3452 }, ".privkey");
3453 }, ".pubkey");
3454 }, ".address");
3455 });
3456 }, bip38delay + 5000);
3457
3458 it('Shows the checksum for the entropy', function(done) {
3459 driver.findElement(By.css('.use-entropy'))
3460 .click();
3461 driver.findElement(By.css('.entropy'))
3462 .sendKeys("00000000000000000000000000000000");
3463 driver.sleep(generateDelay).then(function() {
3464 driver.findElement(By.css('.checksum'))
3465 .getText()
3466 .then(function(text) {
3467 expect(text).toBe("1");
3468 done();
3469 });
3470 });
3471 });
3472
3473 it('Shows the checksum for the entropy with the correct groupings', function(done) {
3474 driver.findElement(By.css('.use-entropy'))
3475 .click();
3476 // create a checksum of 20 bits, which spans multiple words
3477 driver.findElement(By.css('.entropy'))
3478 .sendKeys("F000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000");
3479 driver.sleep(generateDelay).then(function() {
3480 driver.findElement(By.css('.checksum'))
3481 .getText()
3482 .then(function(text) {
3483 // first group is 9 bits, second group is 11
3484 expect(text).toBe("011010111 01110000110");
3485 done();
3486 });
3487 });
3488 });
3489
3490 it('Uses vprv for bitcoin testnet p2wpkh', function(done) {
3491 selectNetwork("BTC - Bitcoin Testnet");
3492 driver.findElement(By.css('#bip84-tab a'))
3493 .click()
3494 driver.findElement(By.css('.phrase'))
3495 .sendKeys('abandon abandon ability');
3496 driver.sleep(generateDelay).then(function() {
3497 driver.findElement(By.css('.root-key'))
3498 .getAttribute("value")
3499 .then(function(path) {
3500 expect(path).toBe("vprv9DMUxX4ShgxML9N2YV5CvWEebWrM9aJ5ULpbRRyzyWu6vs4BzTvbfFFrH41N5hVi7MYSfiugd765L3JmAfDM5po36Y8ouCKRDeYQwByCmS7");
3501 done();
3502 })
3503 });
3504 });
3505
3506 it('Shows a warning if generating weak mnemonics', function(done) {
3507 driver.executeScript(function() {
3508 $(".strength option[selected]").removeAttr("selected");
3509 $(".strength option[value=6]").prop("selected", true);
3510 $(".strength").trigger("change");
3511 });
3512 driver.findElement(By.css(".generate-container .warning"))
3513 .getAttribute("class")
3514 .then(function(classes) {
3515 expect(classes).not.toContain("hidden");
3516 done();
3517 });
3518 });
3519
3520 it('Does not show a warning if generating strong mnemonics', function(done) {
3521 driver.executeScript(function() {
3522 $(".strength option[selected]").removeAttr("selected");
3523 $(".strength option[value=12]").prop("selected", true);
3524 });
3525 driver.findElement(By.css(".generate-container .warning"))
3526 .getAttribute("class")
3527 .then(function(classes) {
3528 expect(classes).toContain("hidden");
3529 done();
3530 });
3531 });
3532
3533 });