]> git.immae.eu Git - perso/Immae/Projets/Cryptomonnaies/BIP39.git/blob - tests/spec/tests.js
Merge pull request #207 from Coinomi/kobocoin
[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 Kobocoin', function(done) {
904 var params = {
905 selectText: "KOBO - Kobocoin",
906 firstAddress: "FTVoNJETXDAM8x7MnmdE8RwWndSr9PQWhy",
907 };
908 testNetwork(done, params);
909 });
910 it('Allows selection of Landcoin', function(done) {
911 var params = {
912 selectText: "LDCN - Landcoin",
913 firstAddress: "LLvLwNjG1aJcn1RS4W4GJUbv8fNaRATG7c",
914 };
915 testNetwork(done, params);
916 });
917 it('Allows selection of Library Credits', function(done) {
918 var params = {
919 selectText: "LBC - Library Credits",
920 firstAddress: "bQJEQrHDJyHdqycB32uysh1SWn8Ln8LMdg",
921 };
922 testNetwork(done, params);
923 });
924 it('Allows selection of Linx', function(done) {
925 var params = {
926 selectText: "LINX - Linx",
927 firstAddress: "XGWQ3cb3LGUB3VnHmj6xYSMgnokNbf6dyk",
928 };
929 testNetwork(done, params);
930 });
931 it('Allows selection of Litecoincash', function(done) {
932 var params = {
933 selectText: "LCC - Litecoincash",
934 firstAddress: "Ce5n7fjUuQPLutJ4W5nCCfQLKdKLE1mv9A",
935 };
936 testNetwork(done, params);
937 });
938 it('Allows selection of Lynx', function(done) {
939 var params = {
940 selectText: "LYNX - Lynx",
941 firstAddress: "KUeY3ZdZkg96p4W98pj1JjygCFU1XqWdw3",
942 };
943 testNetwork(done, params);
944 });
945 it('Allows selection of Minexcoin', function(done) {
946 var params = {
947 selectText: "MNX - Minexcoin",
948 firstAddress: "XC1VnyJVfiMDwWgFtAHDp41cgY3AHk3dJT",
949 };
950 testNetwork(done, params);
951 });
952 it('Allows selection of Navcoin', function(done) {
953 var params = {
954 selectText: "NAV - Navcoin",
955 firstAddress: "NTQVTPK3NWSQLKoffkiQw99T8PifkF1Y2U",
956 };
957 testNetwork(done, params);
958 });
959 it('Allows selection of Neoscoin', function(done) {
960 var params = {
961 selectText: "NEOS - Neoscoin",
962 firstAddress: "NgATz6QbQNXvayHQ4CpZayugb9HeaPDdby",
963 };
964 testNetwork(done, params);
965 });
966 it('Allows selection of Neurocoin', function(done) {
967 var params = {
968 selectText: "NRO - Neurocoin",
969 firstAddress: "NVdYErQ3mFpDuF5DquW9WMiT7sLc8ufFTn",
970 };
971 testNetwork(done, params);
972 });
973 it('Allows selection of Newyorkc', function(done) {
974 var params = {
975 selectText: "NYC - Newyorkc",
976 firstAddress: "RSVMfyH1fKfy3puADJEhut2vfkRyon6imm",
977 };
978 testNetwork(done, params);
979 });
980 it('Allows selection of Novacoin', function(done) {
981 var params = {
982 selectText: "NVC - Novacoin",
983 firstAddress: "4JRvUmxcKCJmaMXZyvRoSS1cmG2XvnZfHN",
984 };
985 testNetwork(done, params);
986 });
987 it('Allows selection of Nushares', function(done) {
988 var params = {
989 selectText: "NSR - Nushares",
990 firstAddress: "SecjXzU3c7EecdT7EbC4vvmbdtBBokWh6J",
991 };
992 testNetwork(done, params);
993 });
994 it('Allows selection of Okcash', function(done) {
995 var params = {
996 selectText: "OK - Okcash",
997 firstAddress: "PV4Qp1TUYuGv4TqVtLZtqvrsWWRycfx1Yi",
998 };
999 testNetwork(done, params);
1000 });
1001 it('Allows selection of Omnicore', function(done) {
1002 var params = {
1003 selectText: "OMNI - Omnicore",
1004 firstAddress: "1Q1t3gonjCT3rW38TsTsCvgSc3hh7zBGbi",
1005 };
1006 testNetwork(done, params);
1007 });
1008 it('Allows selection of Pesobit', function(done) {
1009 var params = {
1010 selectText: "PSB - Pesobit",
1011 firstAddress: "PDePsF7ALyXP7JaywokdYiRTDtKa14MAr1",
1012 };
1013 testNetwork(done, params);
1014 });
1015 it('Allows selection of Pinkcoin', function(done) {
1016 var params = {
1017 selectText: "PINK - Pinkcoin",
1018 firstAddress: "2TgjYQffjbzUHJghNaVbdsjHbRwruC3yzC",
1019 };
1020 testNetwork(done, params);
1021 });
1022 it('Allows selection of POSWcoin', function(done) {
1023 var params = {
1024 selectText: "POSW - POSWcoin",
1025 firstAddress: "PNxewmZoPnGBvoEbH6hgQZCK1igDiBCdgC",
1026 };
1027 testNetwork(done, params);
1028 });
1029 it('Allows selection of Potcoin', function(done) {
1030 var params = {
1031 selectText: "POT - Potcoin",
1032 firstAddress: "PEo7Vg2ctXgpP4vuLPeY9aGJtZotyrmiHc",
1033 };
1034 testNetwork(done, params);
1035 });
1036 it('Allows selection of Putincoin', function(done) {
1037 var params = {
1038 selectText: "PUT - Putincoin",
1039 firstAddress: "PViWnfr2uFtovd6e7joM49C94CsGSnqJis",
1040 };
1041 testNetwork(done, params);
1042 });
1043 it('Allows selection of Reddcoin', function(done) {
1044 var params = {
1045 selectText: "RDD - Reddcoin",
1046 firstAddress: "1M4druAcUfkXBaAcQ4cCgCLPHChiaib6kL",
1047 };
1048 testNetwork(done, params);
1049 });
1050 it('Allows selection of RevolutionVR', function(done) {
1051 var params = {
1052 selectText: "RVR - RevolutionVR",
1053 firstAddress: "VXeeoP2jkzZnMFxtc66ZBZK1NHN5QJnnjL",
1054 };
1055 testNetwork(done, params);
1056 });
1057 it('Allows selection of Rubycoin', function(done) {
1058 var params = {
1059 selectText: "RBY - Rubycoin",
1060 firstAddress: "RV76JDtjTs11JdMDRToYn6CHecMRPLnKS6",
1061 };
1062 testNetwork(done, params);
1063 });
1064 it('Allows selection of Smileycoin', function(done) {
1065 var params = {
1066 selectText: "SMLY - Smileycoin",
1067 firstAddress: "BEZVnEBCAyFByrgKpwAgYgtvP4rKAd9Sj2",
1068 };
1069 testNetwork(done, params);
1070 });
1071 it('Allows selection of Solarcoin', function(done) {
1072 var params = {
1073 selectText: "SLR - Solarcoin",
1074 firstAddress: "8LZ13HbnjtaMJWSvvVFNTLf71zFfDrhwLu",
1075 };
1076 testNetwork(done, params);
1077 });
1078 it('Allows selection of Stratis', function(done) {
1079 var params = {
1080 selectText: "STRAT - Stratis",
1081 firstAddress: "ScfJnq3QDhKgDMEds6sqUE1ot6ShfhmXXq",
1082 };
1083 testNetwork(done, params);
1084 });
1085 it('Allows selection of Syscoin', function(done) {
1086 var params = {
1087 selectText: "SYS - Syscoin",
1088 firstAddress: "SZwJi42Pst3VAMomyK5DG4157WM5ofRmSj",
1089 };
1090 testNetwork(done, params);
1091 });
1092 it('Allows selection of Toa', function(done) {
1093 var params = {
1094 selectText: "TOA - Toa",
1095 firstAddress: "TSe1QAnUwQzUfbBusDzRJ9URttrRGKoNKF",
1096 };
1097 testNetwork(done, params);
1098 });
1099 it('Allows selection of Ultimatesecurecash', function(done) {
1100 var params = {
1101 selectText: "USC - Ultimatesecurecash",
1102 firstAddress: "UPyLAZU2Che5fiy7Ed8xVJFmXAUhitA4ug",
1103 };
1104 testNetwork(done, params);
1105 });
1106 it('Allows selection of Unobtanium', function(done) {
1107 var params = {
1108 selectText: "UNO - Unobtanium",
1109 firstAddress: "uUBMPVMXrR6qhqornJqKTWgr8L69vihSL9",
1110 };
1111 testNetwork(done, params);
1112 });
1113 it('Allows selection of Vcash', function(done) {
1114 var params = {
1115 selectText: "XVC - Vcash",
1116 firstAddress: "VuL53MSY6KjvAjKSeRkh3NDnKykacDVeps",
1117 };
1118 testNetwork(done, params);
1119 });
1120 it('Allows selection of Verge', function(done) {
1121 var params = {
1122 selectText: "XVG - Verge",
1123 firstAddress: "DCrVuGkMjLJpTGgwAgv9AcMdeb1nkWbjZA",
1124 };
1125 testNetwork(done, params);
1126 });
1127 it('Allows selection of Vertcoin', function(done) {
1128 var params = {
1129 selectText: "VTC - Vertcoin",
1130 firstAddress: "Vf6koGuiWdXQfx8tNqxoNeEDxh4xh5cxsG",
1131 };
1132 testNetwork(done, params);
1133 });
1134 it('Allows selection of Vivo', function(done) {
1135 var params = {
1136 selectText: "VIVO - Vivo",
1137 firstAddress: "VFmBwuXXGhJe7MarQG2GfzHMFebRHgfSpB",
1138 };
1139 testNetwork(done, params);
1140 });
1141 it('Allows selection of Vpncoin', function(done) {
1142 var params = {
1143 selectText: "VASH - Vpncoin",
1144 firstAddress: "VoEmH1qXC4TsSgBAStR21QYetwnFqbqCx9",
1145 };
1146 testNetwork(done, params);
1147 });
1148 it('Allows selection of Whitecoin', function(done) {
1149 var params = {
1150 selectText: "XWC - Whitecoin",
1151 firstAddress: "WcSwCAUqrSgeSYbsaS3SSWWhsx8KRYTFDR",
1152 };
1153 testNetwork(done, params);
1154 });
1155 it('Allows selection of Wincoin', function(done) {
1156 var params = {
1157 selectText: "WC - Wincoin",
1158 firstAddress: "WaDVCESMGgyKgNESdn3u43NnwmGSkZED3Z",
1159 };
1160 testNetwork(done, params);
1161 });
1162 it('Allows selection of Zcoin', function(done) {
1163 var params = {
1164 selectText: "XZC - Zcoin",
1165 firstAddress: "a6VcMdP4XgAA9Tr7xNszmPG5FZpfRf17Cq",
1166 };
1167 testNetwork(done, params);
1168 });
1169
1170
1171 // BIP39 seed is set from phrase
1172 it('Sets the bip39 seed from the prhase', function(done) {
1173 driver.findElement(By.css('.phrase'))
1174 .sendKeys('abandon abandon ability');
1175 driver.sleep(generateDelay).then(function() {
1176 driver.findElement(By.css('.seed'))
1177 .getAttribute("value")
1178 .then(function(seed) {
1179 expect(seed).toBe("20da140d3dd1df8713cefcc4d54ce0e445b4151027a1ab567b832f6da5fcc5afc1c3a3f199ab78b8e0ab4652efd7f414ac2c9a3b81bceb879a70f377aa0a58f3");
1180 done();
1181 })
1182 });
1183 });
1184
1185 // BIP32 root key is set from phrase
1186 it('Sets the bip39 root key from the prhase', function(done) {
1187 driver.findElement(By.css('.phrase'))
1188 .sendKeys('abandon abandon ability');
1189 driver.sleep(generateDelay).then(function() {
1190 driver.findElement(By.css('.root-key'))
1191 .getAttribute("value")
1192 .then(function(seed) {
1193 expect(seed).toBe("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
1194 done();
1195 })
1196 });
1197 });
1198
1199 // Tabs show correct addresses when changed
1200 it('Shows the correct address when tab is changed', function(done) {
1201 driver.findElement(By.css('.phrase'))
1202 .sendKeys('abandon abandon ability');
1203 driver.sleep(generateDelay).then(function() {
1204 driver.findElement(By.css('#bip32-tab a'))
1205 .click();
1206 driver.sleep(generateDelay).then(function() {
1207 getFirstAddress(function(address) {
1208 expect(address).toBe("17uQ7s2izWPwBmEVFikTmZUjbBKWYdJchz");
1209 done();
1210 });
1211 });
1212 });
1213 });
1214
1215 // BIP44 derivation path is shown
1216 it('Shows the derivation path for bip44 tab', function(done) {
1217 driver.findElement(By.css('.phrase'))
1218 .sendKeys('abandon abandon ability');
1219 driver.sleep(generateDelay).then(function() {
1220 driver.findElement(By.css('#bip44 .path'))
1221 .getAttribute("value")
1222 .then(function(path) {
1223 expect(path).toBe("m/44'/0'/0'/0");
1224 done();
1225 })
1226 });
1227 });
1228
1229 // BIP44 extended private key is shown
1230 it('Shows the extended private key for bip44 tab', function(done) {
1231 driver.findElement(By.css('.phrase'))
1232 .sendKeys('abandon abandon ability');
1233 driver.sleep(generateDelay).then(function() {
1234 driver.findElement(By.css('.extended-priv-key'))
1235 .getAttribute("value")
1236 .then(function(path) {
1237 expect(path).toBe("xprvA2DxxvPZcyRvYgZMGS53nadR32mVDeCyqQYyFhrCVbJNjPoxMeVf7QT5g7mQASbTf9Kp4cryvcXnu2qurjWKcrdsr91jXymdCDNxKgLFKJG");
1238 done();
1239 })
1240 });
1241 });
1242
1243 // BIP44 extended public key is shown
1244 it('Shows the extended public key for bip44 tab', function(done) {
1245 driver.findElement(By.css('.phrase'))
1246 .sendKeys('abandon abandon ability');
1247 driver.sleep(generateDelay).then(function() {
1248 driver.findElement(By.css('.extended-pub-key'))
1249 .getAttribute("value")
1250 .then(function(path) {
1251 expect(path).toBe("xpub6FDKNRvTTLzDmAdpNTc49ia9b4byd6vqCdUa46Fp3vqMcC96uBoufCmZXQLiN5AK3iSCJMhf9gT2sxkpyaPepRuA7W3MujV5tGmF5VfbueM");
1252 done();
1253 })
1254 });
1255 });
1256
1257 // BIP44 account field changes address list
1258 it('Changes the address list if bip44 account is changed', function(done) {
1259 driver.findElement(By.css('#bip44 .account'))
1260 .sendKeys('1');
1261 driver.findElement(By.css('.phrase'))
1262 .sendKeys('abandon abandon ability');
1263 driver.sleep(generateDelay).then(function() {
1264 getFirstAddress(function(address) {
1265 expect(address).toBe("1Nq2Wmu726XHCuGhctEtGmhxo3wzk5wZ1H");
1266 done();
1267 });
1268 });
1269 });
1270
1271 // BIP44 change field changes address list
1272 it('Changes the address list if bip44 change is changed', function(done) {
1273 driver.findElement(By.css('#bip44 .change'))
1274 .sendKeys('1');
1275 driver.findElement(By.css('.phrase'))
1276 .sendKeys('abandon abandon ability');
1277 driver.sleep(generateDelay).then(function() {
1278 getFirstAddress(function(address) {
1279 expect(address).toBe("1KAGfWgqfVbSSXY56fNQ7YnhyKuoskHtYo");
1280 done();
1281 });
1282 });
1283 });
1284
1285 // BIP32 derivation path can be set
1286 it('Can use a custom bip32 derivation path', function(done) {
1287 driver.findElement(By.css('#bip32-tab a'))
1288 .click();
1289 driver.findElement(By.css('#bip32 .path'))
1290 .clear();
1291 driver.findElement(By.css('#bip32 .path'))
1292 .sendKeys('m/1');
1293 driver.findElement(By.css('.phrase'))
1294 .sendKeys('abandon abandon ability');
1295 driver.sleep(generateDelay).then(function() {
1296 getFirstAddress(function(address) {
1297 expect(address).toBe("16pYQQdLD1hH4hwTGLXBaZ9Teboi1AGL8L");
1298 done();
1299 });
1300 });
1301 });
1302
1303 // BIP32 can use hardened derivation paths
1304 it('Can use a hardened derivation paths', function(done) {
1305 driver.findElement(By.css('#bip32-tab a'))
1306 .click();
1307 driver.findElement(By.css('#bip32 .path'))
1308 .clear();
1309 driver.findElement(By.css('#bip32 .path'))
1310 .sendKeys("m/0'");
1311 driver.findElement(By.css('.phrase'))
1312 .sendKeys('abandon abandon ability');
1313 driver.sleep(generateDelay).then(function() {
1314 getFirstAddress(function(address) {
1315 expect(address).toBe("14aXZeprXAE3UUKQc4ihvwBvww2LuEoHo4");
1316 done();
1317 });
1318 });
1319 });
1320
1321 // BIP32 extended private key is shown
1322 it('Shows the BIP32 extended private key', function(done) {
1323 driver.findElement(By.css('#bip32-tab a'))
1324 .click();
1325 driver.findElement(By.css('.phrase'))
1326 .sendKeys('abandon abandon ability');
1327 driver.sleep(generateDelay).then(function() {
1328 driver.findElement(By.css('.extended-priv-key'))
1329 .getAttribute("value")
1330 .then(function(privKey) {
1331 expect(privKey).toBe("xprv9va99uTVE5aLiutUVLTyfxfe8v8aaXjSQ1XxZbK6SezYVuikA9MnjQVTA8rQHpNA5LKvyQBpLiHbBQiiccKiBDs7eRmBogsvq3THFeLHYbe");
1332 done();
1333 });
1334 });
1335 });
1336
1337 // BIP32 extended public key is shown
1338 it('Shows the BIP32 extended public key', function(done) {
1339 driver.findElement(By.css('#bip32-tab a'))
1340 .click();
1341 driver.findElement(By.css('.phrase'))
1342 .sendKeys('abandon abandon ability');
1343 driver.sleep(generateDelay).then(function() {
1344 driver.findElement(By.css('.extended-pub-key'))
1345 .getAttribute("value")
1346 .then(function(pubKey) {
1347 expect(pubKey).toBe("xpub69ZVZQzP4T8dwPxwbMzz36cNgwy4yzTHmETZMyihzzXXNi3thgg3HCow1RtY252wdw5rS8369xKnraN5Q93y3FkFfJp2XEHWUrkyXsjS93P");
1348 done();
1349 });
1350 });
1351 });
1352
1353 // Derivation path is shown in table
1354 it('Shows the derivation path in the table', function(done) {
1355 driver.findElement(By.css('.phrase'))
1356 .sendKeys('abandon abandon ability');
1357 driver.sleep(generateDelay).then(function() {
1358 getFirstPath(function(path) {
1359 expect(path).toBe("m/44'/0'/0'/0/0");
1360 done();
1361 });
1362 });
1363 });
1364
1365 // Derivation path for address can be hardened
1366 it('Can derive hardened addresses', function(done) {
1367 driver.findElement(By.css('#bip32-tab a'))
1368 .click();
1369 driver.executeScript(function() {
1370 $(".hardened-addresses").prop("checked", true);
1371 });
1372 driver.findElement(By.css('.phrase'))
1373 .sendKeys('abandon abandon ability');
1374 driver.sleep(generateDelay).then(function() {
1375 getFirstAddress(function(address) {
1376 expect(address).toBe("18exLzUv7kfpiXRzmCjFDoC9qwNLFyvwyd");
1377 done();
1378 });
1379 });
1380 });
1381
1382 // Derivation path visibility can be toggled
1383 it('Can toggle visibility of the derivation path column', function(done) {
1384 driver.findElement(By.css('.phrase'))
1385 .sendKeys('abandon abandon ability');
1386 driver.sleep(generateDelay).then(function() {
1387 driver.findElement(By.css('.index-toggle'))
1388 .click();
1389 testColumnValuesAreInvisible(done, "index");
1390 });
1391 });
1392
1393 // Address is shown
1394 it('Shows the address in the table', function(done) {
1395 driver.findElement(By.css('.phrase'))
1396 .sendKeys('abandon abandon ability');
1397 driver.sleep(generateDelay).then(function() {
1398 getFirstAddress(function(address) {
1399 expect(address).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug");
1400 done();
1401 });
1402 });
1403 });
1404
1405 // Addresses are shown in order of derivation path
1406 it('Shows the address in order of derivation path', function(done) {
1407 driver.findElement(By.css('.phrase'))
1408 .sendKeys('abandon abandon ability');
1409 driver.sleep(generateDelay).then(function() {
1410 testRowsAreInCorrectOrder(done);
1411 });
1412 });
1413
1414 // Address visibility can be toggled
1415 it('Can toggle visibility of the address column', function(done) {
1416 driver.findElement(By.css('.phrase'))
1417 .sendKeys('abandon abandon ability');
1418 driver.sleep(generateDelay).then(function() {
1419 driver.findElement(By.css('.address-toggle'))
1420 .click();
1421 testColumnValuesAreInvisible(done, "address");
1422 });
1423 });
1424
1425 // Public key is shown in table
1426 it('Shows the public key in the table', function(done) {
1427 driver.findElement(By.css('.phrase'))
1428 .sendKeys('abandon abandon ability');
1429 driver.sleep(generateDelay).then(function() {
1430 driver.findElements(By.css('.pubkey'))
1431 .then(function(els) {
1432 els[0].getText()
1433 .then(function(pubkey) {
1434 expect(pubkey).toBe("033f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3");
1435 done();
1436 });
1437 });
1438 });
1439 });
1440
1441 // Public key visibility can be toggled
1442 it('Can toggle visibility of the public key column', function(done) {
1443 driver.findElement(By.css('.phrase'))
1444 .sendKeys('abandon abandon ability');
1445 driver.sleep(generateDelay).then(function() {
1446 driver.findElement(By.css('.public-key-toggle'))
1447 .click();
1448 testColumnValuesAreInvisible(done, "pubkey");
1449 });
1450 });
1451
1452 // Private key is shown in table
1453 it('Shows the private key in the table', function(done) {
1454 driver.findElement(By.css('.phrase'))
1455 .sendKeys('abandon abandon ability');
1456 driver.sleep(generateDelay).then(function() {
1457 driver.findElements(By.css('.privkey'))
1458 .then(function(els) {
1459 els[0].getText()
1460 .then(function(pubkey) {
1461 expect(pubkey).toBe("L26cVSpWFkJ6aQkPkKmTzLqTdLJ923e6CzrVh9cmx21QHsoUmrEE");
1462 done();
1463 });
1464 });
1465 });
1466 });
1467
1468 // Private key visibility can be toggled
1469 it('Can toggle visibility of the private key column', function(done) {
1470 driver.findElement(By.css('.phrase'))
1471 .sendKeys('abandon abandon ability');
1472 driver.sleep(generateDelay).then(function() {
1473 driver.findElement(By.css('.private-key-toggle'))
1474 .click();
1475 testColumnValuesAreInvisible(done, "privkey");
1476 });
1477 });
1478
1479 // More addresses can be generated
1480 it('Can generate more rows in the table', function(done) {
1481 driver.findElement(By.css('.phrase'))
1482 .sendKeys('abandon abandon ability');
1483 driver.sleep(generateDelay).then(function() {
1484 driver.findElement(By.css('.more'))
1485 .click();
1486 driver.sleep(generateDelay).then(function() {
1487 driver.findElements(By.css('.address'))
1488 .then(function(els) {
1489 expect(els.length).toBe(40);
1490 done();
1491 });
1492 });
1493 });
1494 });
1495
1496 // A custom number of additional addresses can be generated
1497 it('Can generate more rows in the table', function(done) {
1498 driver.findElement(By.css('.phrase'))
1499 .sendKeys('abandon abandon ability');
1500 driver.sleep(generateDelay).then(function() {
1501 driver.findElement(By.css('.rows-to-add'))
1502 .clear();
1503 driver.findElement(By.css('.rows-to-add'))
1504 .sendKeys('1');
1505 driver.findElement(By.css('.more'))
1506 .click();
1507 driver.sleep(generateDelay).then(function() {
1508 driver.findElements(By.css('.address'))
1509 .then(function(els) {
1510 expect(els.length).toBe(21);
1511 done();
1512 });
1513 });
1514 });
1515 });
1516
1517 // Additional addresses are shown in order of derivation path
1518 it('Shows additional addresses in order of derivation path', function(done) {
1519 driver.findElement(By.css('.phrase'))
1520 .sendKeys('abandon abandon ability');
1521 driver.sleep(generateDelay).then(function() {
1522 driver.findElement(By.css('.more'))
1523 .click();
1524 driver.sleep(generateDelay).then(function() {
1525 testRowsAreInCorrectOrder(done);
1526 });
1527 });
1528 });
1529
1530 // BIP32 root key can be set by the user
1531 it('Allows the user to set the BIP32 root key', function(done) {
1532 driver.findElement(By.css('.root-key'))
1533 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1534 driver.sleep(generateDelay).then(function() {
1535 getFirstAddress(function(address) {
1536 expect(address).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug");
1537 done();
1538 });
1539 });
1540 });
1541
1542 // Setting BIP32 root key clears the existing phrase, passphrase and seed
1543 // TODO this doesn't work in selenium with chrome
1544 it('Confirms the existing phrase should be cleared', function(done) {
1545 if (browser == "chrome") {
1546 pending("Selenium + Chrome headless bug for alert, see https://stackoverflow.com/q/45242264");
1547 }
1548 driver.findElement(By.css('.phrase'))
1549 .sendKeys('A non-blank but invalid value');
1550 driver.findElement(By.css('.root-key'))
1551 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1552 driver.switchTo().alert().accept();
1553 driver.findElement(By.css('.phrase'))
1554 .getAttribute("value").then(function(value) {
1555 expect(value).toBe("");
1556 done();
1557 });
1558 });
1559
1560 // Clearing of phrase, passphrase and seed can be cancelled by user
1561 // TODO this doesn't work in selenium with chrome
1562 it('Allows the clearing of the phrase to be cancelled', function(done) {
1563 if (browser == "chrome") {
1564 pending("Selenium + Chrome headless bug for alert, see https://stackoverflow.com/q/45242264");
1565 }
1566 driver.findElement(By.css('.phrase'))
1567 .sendKeys('abandon abandon ability');
1568 driver.sleep(generateDelay).then(function() {
1569 driver.findElement(By.css('.root-key'))
1570 .clear();
1571 driver.findElement(By.css('.root-key'))
1572 .sendKeys('x');
1573 driver.switchTo().alert().dismiss();
1574 driver.findElement(By.css('.phrase'))
1575 .getAttribute("value").then(function(value) {
1576 expect(value).toBe("abandon abandon ability");
1577 done();
1578 });
1579 });
1580 });
1581
1582 // Custom BIP32 root key is used when changing the derivation path
1583 it('Can set derivation path for root key instead of phrase', function(done) {
1584 driver.findElement(By.css('#bip44 .account'))
1585 .sendKeys('1');
1586 driver.findElement(By.css('.root-key'))
1587 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1588 driver.sleep(generateDelay).then(function() {
1589 getFirstAddress(function(address) {
1590 expect(address).toBe("1Nq2Wmu726XHCuGhctEtGmhxo3wzk5wZ1H");
1591 done();
1592 });
1593 });
1594 });
1595
1596 // Incorrect mnemonic shows error
1597 it('Shows an error for incorrect mnemonic', function(done) {
1598 driver.findElement(By.css('.phrase'))
1599 .sendKeys('abandon abandon abandon');
1600 driver.sleep(feedbackDelay).then(function() {
1601 driver.findElement(By.css('.feedback'))
1602 .getText()
1603 .then(function(feedback) {
1604 expect(feedback).toBe("Invalid mnemonic");
1605 done();
1606 });
1607 });
1608 });
1609
1610 // Incorrect word shows suggested replacement
1611 it('Shows word suggestion for incorrect word', function(done) {
1612 driver.findElement(By.css('.phrase'))
1613 .sendKeys('abandon abandon abiliti');
1614 driver.sleep(feedbackDelay).then(function() {
1615 driver.findElement(By.css('.feedback'))
1616 .getText()
1617 .then(function(feedback) {
1618 var msg = "abiliti not in wordlist, did you mean ability?";
1619 expect(feedback).toBe(msg);
1620 done();
1621 });
1622 });
1623 });
1624
1625 // Github pull request 48
1626 // First four letters of word shows that word, not closest
1627 // since first four letters gives unique word in BIP39 wordlist
1628 // eg ille should show illegal, not idle
1629 it('Shows word suggestion based on first four chars', function(done) {
1630 driver.findElement(By.css('.phrase'))
1631 .sendKeys('ille');
1632 driver.sleep(feedbackDelay).then(function() {
1633 driver.findElement(By.css('.feedback'))
1634 .getText()
1635 .then(function(feedback) {
1636 var msg = "ille not in wordlist, did you mean illegal?";
1637 expect(feedback).toBe(msg);
1638 done();
1639 });
1640 });
1641 });
1642
1643 // Incorrect BIP32 root key shows error
1644 it('Shows error for incorrect root key', function(done) {
1645 driver.findElement(By.css('.root-key'))
1646 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpj');
1647 driver.sleep(feedbackDelay).then(function() {
1648 driver.findElement(By.css('.feedback'))
1649 .getText()
1650 .then(function(feedback) {
1651 var msg = "Invalid root key";
1652 expect(feedback).toBe(msg);
1653 done();
1654 });
1655 });
1656 });
1657
1658 // Derivation path not starting with m shows error
1659 it('Shows error for derivation path not starting with m', function(done) {
1660 driver.findElement(By.css('#bip32-tab a'))
1661 .click();
1662 driver.findElement(By.css('#bip32 .path'))
1663 .clear();
1664 driver.findElement(By.css('#bip32 .path'))
1665 .sendKeys('n/0');
1666 driver.findElement(By.css('.phrase'))
1667 .sendKeys('abandon abandon ability');
1668 driver.sleep(feedbackDelay).then(function() {
1669 driver.findElement(By.css('.feedback'))
1670 .getText()
1671 .then(function(feedback) {
1672 var msg = "First character must be 'm'";
1673 expect(feedback).toBe(msg);
1674 done();
1675 });
1676 });
1677 });
1678
1679 // Derivation path containing invalid characters shows useful error
1680 it('Shows error for derivation path not starting with m', function(done) {
1681 driver.findElement(By.css('#bip32-tab a'))
1682 .click();
1683 driver.findElement(By.css('#bip32 .path'))
1684 .clear();
1685 driver.findElement(By.css('#bip32 .path'))
1686 .sendKeys('m/1/0wrong1/1');
1687 driver.findElement(By.css('.phrase'))
1688 .sendKeys('abandon abandon ability');
1689 driver.sleep(feedbackDelay).then(function() {
1690 driver.findElement(By.css('.feedback'))
1691 .getText()
1692 .then(function(feedback) {
1693 var msg = "Invalid characters 0wrong1 found at depth 2";
1694 expect(feedback).toBe(msg);
1695 done();
1696 });
1697 });
1698 });
1699
1700 // Github Issue 11: Default word length is 15
1701 // https://github.com/iancoleman/bip39/issues/11
1702 it('Sets the default word length to 15', function(done) {
1703 driver.findElement(By.css('.strength'))
1704 .getAttribute("value")
1705 .then(function(strength) {
1706 expect(strength).toBe("15");
1707 done();
1708 });
1709 });
1710
1711 // Github Issue 12: Generate more rows with private keys hidden
1712 // https://github.com/iancoleman/bip39/issues/12
1713 it('Sets the correct hidden column state on new rows', function(done) {
1714 driver.findElement(By.css('.phrase'))
1715 .sendKeys("abandon abandon ability");
1716 driver.sleep(generateDelay).then(function() {
1717 driver.findElement(By.css('.private-key-toggle'))
1718 .click();
1719 driver.findElement(By.css('.more'))
1720 .click();
1721 driver.sleep(generateDelay).then(function() {
1722 driver.findElements(By.css('.privkey'))
1723 .then(function(els) {
1724 expect(els.length).toBe(40);
1725 });
1726 testColumnValuesAreInvisible(done, "privkey");
1727 });
1728 });
1729 });
1730
1731 // Github Issue 19: Mnemonic is not sensitive to whitespace
1732 // https://github.com/iancoleman/bip39/issues/19
1733 it('Ignores excess whitespace in the mnemonic', function(done) {
1734 var doublespace = " ";
1735 var mnemonic = "urge cat" + doublespace + "bid";
1736 driver.findElement(By.css('.phrase'))
1737 .sendKeys(mnemonic);
1738 driver.sleep(generateDelay).then(function() {
1739 driver.findElement(By.css('.root-key'))
1740 .getAttribute("value")
1741 .then(function(seed) {
1742 expect(seed).toBe("xprv9s21ZrQH143K3isaZsWbKVoTtbvd34Y1ZGRugGdMeBGbM3AgBVzTH159mj1cbbtYSJtQr65w6L5xy5L9SFC7c9VJZWHxgAzpj4mun5LhrbC");
1743 done();
1744 });
1745 });
1746 });
1747
1748 // Github Issue 23: Part 1: Use correct derivation path when changing tabs
1749 // https://github.com/iancoleman/bip39/issues/23
1750 it('Uses the correct derivation path when changing tabs', function(done) {
1751 // 1) and 2) set the phrase
1752 driver.findElement(By.css('.phrase'))
1753 .sendKeys("abandon abandon ability");
1754 driver.sleep(generateDelay).then(function() {
1755 // 3) select bip32 tab
1756 driver.findElement(By.css('#bip32-tab a'))
1757 .click();
1758 driver.sleep(generateDelay).then(function() {
1759 // 4) switch from bitcoin to litecoin
1760 selectNetwork("LTC - Litecoin");
1761 driver.sleep(generateDelay).then(function() {
1762 // 5) Check address is displayed correctly
1763 getFirstAddress(function(address) {
1764 expect(address).toBe("LS8MP5LZ5AdzSZveRrjm3aYVoPgnfFh5T5");
1765 // 5) Check derivation path is displayed correctly
1766 getFirstPath(function(path) {
1767 expect(path).toBe("m/0/0");
1768 done();
1769 });
1770 });
1771 });
1772 });
1773 });
1774 });
1775
1776 // Github Issue 23 Part 2: Coin selection in derivation path
1777 // https://github.com/iancoleman/bip39/issues/23#issuecomment-238011920
1778 it('Uses the correct derivation path when changing coins', function(done) {
1779 // set the phrase
1780 driver.findElement(By.css('.phrase'))
1781 .sendKeys("abandon abandon ability");
1782 driver.sleep(generateDelay).then(function() {
1783 // switch from bitcoin to clam
1784 selectNetwork("CLAM - Clams");
1785 driver.sleep(generateDelay).then(function() {
1786 // check derivation path is displayed correctly
1787 getFirstPath(function(path) {
1788 expect(path).toBe("m/44'/23'/0'/0/0");
1789 done();
1790 });
1791 });
1792 });
1793 });
1794
1795 // Github Issue 26: When using a Root key derrived altcoins are incorrect
1796 // https://github.com/iancoleman/bip39/issues/26
1797 it('Uses the correct derivation for altcoins with root keys', function(done) {
1798 // 1) 2) and 3) set the root key
1799 driver.findElement(By.css('.root-key'))
1800 .sendKeys("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
1801 driver.sleep(generateDelay).then(function() {
1802 // 4) switch from bitcoin to viacoin
1803 selectNetwork("VIA - Viacoin");
1804 driver.sleep(generateDelay).then(function() {
1805 // 5) ensure the derived address is correct
1806 getFirstAddress(function(address) {
1807 expect(address).toBe("Vq9Eq4N5SQnjqZvxtxzo7hZPW5XnyJsmXT");
1808 done();
1809 });
1810 });
1811 });
1812 });
1813
1814 // Selecting a language with no existing phrase should generate a phrase in
1815 // that language.
1816 it('Generate a random phrase when language is selected and no current phrase', function(done) {
1817 driver.findElement(By.css("a[href='#japanese']"))
1818 .click();
1819 driver.sleep(generateDelay).then(function() {
1820 driver.findElement(By.css(".phrase"))
1821 .getAttribute("value").then(function(phrase) {
1822 expect(phrase.search(/[a-z]/)).toBe(-1);
1823 expect(phrase.length).toBeGreaterThan(0);
1824 done();
1825 });
1826 });
1827 });
1828
1829 // Selecting a language with existing phrase should update the phrase to use
1830 // that language.
1831 it('Updates existing phrases when the language is changed', function(done) {
1832 driver.findElement(By.css(".phrase"))
1833 .sendKeys("abandon abandon ability");
1834 driver.sleep(generateDelay).then(function() {
1835 driver.findElement(By.css("a[href='#italian']"))
1836 .click();
1837 driver.sleep(generateDelay).then(function() {
1838 driver.findElement(By.css(".phrase"))
1839 .getAttribute("value").then(function(phrase) {
1840 // Check only the language changes, not the phrase
1841 expect(phrase).toBe("abaco abaco abbaglio");
1842 getFirstAddress(function(address) {
1843 // Check the address is correct
1844 expect(address).toBe("1Dz5TgDhdki9spa6xbPFbBqv5sjMrx3xgV");
1845 done();
1846 });
1847 });
1848 });
1849 });
1850 });
1851
1852 // Suggested replacement for erroneous word in non-English language
1853 it('Shows word suggestion for incorrect word in non-English language', function(done) {
1854 driver.findElement(By.css('.phrase'))
1855 .sendKeys('abaco abaco zbbaglio');
1856 driver.sleep(feedbackDelay).then(function() {
1857 driver.findElement(By.css('.feedback'))
1858 .getText()
1859 .then(function(feedback) {
1860 var msg = "zbbaglio not in wordlist, did you mean abbaglio?";
1861 expect(feedback).toBe(msg);
1862 done();
1863 });
1864 });
1865 });
1866
1867 // Japanese word does not break across lines.
1868 // Point 2 from
1869 // https://github.com/bitcoin/bips/blob/master/bip-0039/bip-0039-wordlists.md#japanese
1870 it('Does not break Japanese words across lines', function(done) {
1871 driver.findElement(By.css('.phrase'))
1872 .getCssValue("word-break")
1873 .then(function(value) {
1874 expect(value).toBe("keep-all");
1875 done();
1876 });
1877 });
1878
1879 // Language can be specified at page load using hash value in url
1880 it('Can set the language from the url hash', function(done) {
1881 driver.get(url + "#japanese").then(function() {
1882 driver.findElement(By.css('.generate')).click();
1883 driver.sleep(generateDelay).then(function() {
1884 driver.findElement(By.css(".phrase"))
1885 .getAttribute("value").then(function(phrase) {
1886 expect(phrase.search(/[a-z]/)).toBe(-1);
1887 expect(phrase.length).toBeGreaterThan(0);
1888 done();
1889 });
1890 });
1891 });
1892 });
1893
1894 // Entropy can be entered by the user
1895 it('Allows entropy to be entered', function(done) {
1896 driver.findElement(By.css('.use-entropy'))
1897 .click();
1898 driver.findElement(By.css('.entropy'))
1899 .sendKeys('00000000 00000000 00000000 00000000');
1900 driver.sleep(generateDelay).then(function() {
1901 driver.findElement(By.css(".phrase"))
1902 .getAttribute("value").then(function(phrase) {
1903 expect(phrase).toBe("abandon abandon ability");
1904 getFirstAddress(function(address) {
1905 expect(address).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug");
1906 done();
1907 })
1908 });
1909 });
1910 });
1911
1912 // A warning about entropy is shown to the user, with additional information
1913 it('Shows a warning about using entropy', function(done) {
1914 driver.findElement(By.css('.use-entropy'))
1915 .click();
1916 driver.findElement(By.css('.entropy-container'))
1917 .getText()
1918 .then(function(containerText) {
1919 var warning = "mnemonic may be insecure";
1920 expect(containerText).toContain(warning);
1921 driver.findElement(By.css('#entropy-notes'))
1922 .findElement(By.xpath("parent::*"))
1923 .getText()
1924 .then(function(notesText) {
1925 var detail = "flipping a fair coin, rolling a fair dice, noise measurements etc";
1926 expect(notesText).toContain(detail);
1927 done();
1928 });
1929 });
1930 });
1931
1932 // The types of entropy available are described to the user
1933 it('Shows the types of entropy available', function(done) {
1934 driver.findElement(By.css('.entropy'))
1935 .getAttribute("placeholder")
1936 .then(function(placeholderText) {
1937 var options = [
1938 "binary",
1939 "base 6",
1940 "dice",
1941 "base 10",
1942 "hexadecimal",
1943 "cards",
1944 ];
1945 for (var i=0; i<options.length; i++) {
1946 var option = options[i];
1947 expect(placeholderText).toContain(option);
1948 }
1949 done();
1950 });
1951 });
1952
1953 // The actual entropy used is shown to the user
1954 it('Shows the actual entropy used', function(done) {
1955 driver.findElement(By.css('.use-entropy'))
1956 .click();
1957 driver.findElement(By.css('.entropy'))
1958 .sendKeys('Not A Very Good Entropy Source At All');
1959 driver.sleep(generateDelay).then(function() {
1960 driver.findElement(By.css('.entropy-container'))
1961 .getText()
1962 .then(function(text) {
1963 expect(text).toMatch(/Filtered Entropy\s+AedEceAA/);
1964 done();
1965 });
1966 });
1967 });
1968
1969 // Binary entropy can be entered
1970 it('Allows binary entropy to be entered', function(done) {
1971 testEntropyType(done, "01", "binary");
1972 });
1973
1974 // Base 6 entropy can be entered
1975 it('Allows base 6 entropy to be entered', function(done) {
1976 testEntropyType(done, "012345", "base 6");
1977 });
1978
1979 // Base 6 dice entropy can be entered
1980 it('Allows base 6 dice entropy to be entered', function(done) {
1981 testEntropyType(done, "123456", "base 6 (dice)");
1982 });
1983
1984 // Base 10 entropy can be entered
1985 it('Allows base 10 entropy to be entered', function(done) {
1986 testEntropyType(done, "789", "base 10");
1987 });
1988
1989 // Hexadecimal entropy can be entered
1990 it('Allows hexadecimal entropy to be entered', function(done) {
1991 testEntropyType(done, "abcdef", "hexadecimal");
1992 });
1993
1994 // Dice entropy value is shown as the converted base 6 value
1995 // ie 123456 is converted to 123450
1996 it('Shows dice entropy as base 6', function(done) {
1997 driver.findElement(By.css('.use-entropy'))
1998 .click();
1999 driver.findElement(By.css('.entropy'))
2000 .sendKeys("123456");
2001 driver.sleep(generateDelay).then(function() {
2002 driver.findElement(By.css('.entropy-container'))
2003 .getText()
2004 .then(function(text) {
2005 expect(text).toMatch(/Filtered Entropy\s+123450/);
2006 done();
2007 });
2008 });
2009 });
2010
2011 // The number of bits of entropy accumulated is shown
2012 it("Shows the number of bits of entropy for 20 bits of binary", function(done) {
2013 testEntropyBits(done, "0000 0000 0000 0000 0000", "20");
2014 });
2015 it("Shows the number of bits of entropy for 1 bit of binary", function(done) {
2016 testEntropyBits(done, "0", "1");
2017 });
2018 it("Shows the number of bits of entropy for 4 bits of binary", function(done) {
2019 testEntropyBits(done, "0000", "4");
2020 });
2021 it("Shows the number of bits of entropy for 1 character of base 6 (dice)", function(done) {
2022 // 6 in card is 0 in base 6, 0 in base 6 is 2.6 bits (rounded down to 2 bits)
2023 testEntropyBits(done, "6", "2");
2024 });
2025 it("Shows the number of bits of entropy for 1 character of base 10 with 3 bits", function(done) {
2026 // 7 in base 10 is 111 in base 2, no leading zeros
2027 testEntropyBits(done, "7", "3");
2028 });
2029 it("Shows the number of bits of entropy for 1 character of base 10 with 4 bis", function(done) {
2030 testEntropyBits(done, "8", "4");
2031 });
2032 it("Shows the number of bits of entropy for 1 character of hex", function(done) {
2033 testEntropyBits(done, "F", "4");
2034 });
2035 it("Shows the number of bits of entropy for 2 characters of base 10", function(done) {
2036 testEntropyBits(done, "29", "6");
2037 });
2038 it("Shows the number of bits of entropy for 2 characters of hex", function(done) {
2039 testEntropyBits(done, "0A", "8");
2040 });
2041 it("Shows the number of bits of entropy for 2 characters of hex with 3 leading zeros", function(done) {
2042 // hex is always multiple of 4 bits of entropy
2043 testEntropyBits(done, "1A", "8");
2044 });
2045 it("Shows the number of bits of entropy for 2 characters of hex with 2 leading zeros", function(done) {
2046 testEntropyBits(done, "2A", "8");
2047 });
2048 it("Shows the number of bits of entropy for 2 characters of hex with 1 leading zero", function(done) {
2049 testEntropyBits(done, "4A", "8");
2050 });
2051 it("Shows the number of bits of entropy for 2 characters of hex with no leading zeros", function(done) {
2052 testEntropyBits(done, "8A", "8");
2053 });
2054 it("Shows the number of bits of entropy for 2 characters of hex starting with F", function(done) {
2055 testEntropyBits(done, "FA", "8");
2056 });
2057 it("Shows the number of bits of entropy for 4 characters of hex with leading zeros", function(done) {
2058 testEntropyBits(done, "000A", "16");
2059 });
2060 it("Shows the number of bits of entropy for 4 characters of base 6", function(done) {
2061 testEntropyBits(done, "5555", "11");
2062 });
2063 it("Shows the number of bits of entropy for 4 characters of base 6 dice", function(done) {
2064 // uses dice, so entropy is actually 0000 in base 6, which is 4 lots of
2065 // 2.58 bits, which is 10.32 bits (rounded down to 10 bits)
2066 testEntropyBits(done, "6666", "10");
2067 });
2068 it("Shows the number of bits of entropy for 4 charactes of base 10", function(done) {
2069 // Uses base 10, which is 4 lots of 3.32 bits, which is 13.3 bits (rounded
2070 // down to 13)
2071 testEntropyBits(done, "2227", "13");
2072 });
2073 it("Shows the number of bits of entropy for 4 characters of hex with 2 leading zeros", function(done) {
2074 testEntropyBits(done, "222F", "16");
2075 });
2076 it("Shows the number of bits of entropy for 4 characters of hex starting with F", function(done) {
2077 testEntropyBits(done, "FFFF", "16");
2078 });
2079 it("Shows the number of bits of entropy for 10 characters of base 10", function(done) {
2080 // 10 events at 3.32 bits per event
2081 testEntropyBits(done, "0000101017", "33");
2082 });
2083 it("Shows the number of bits of entropy for a full deck of cards", function(done) {
2084 // cards are not replaced, so a full deck is not 52^52 entropy which is 296
2085 // bits, it's 52!, which is 225 bits
2086 testEntropyBits(done, "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks", "225");
2087 });
2088
2089 it("Shows details about the entered entropy", function(done) {
2090 testEntropyFeedback(done,
2091 {
2092 entropy: "A",
2093 filtered: "A",
2094 type: "hexadecimal",
2095 events: "1",
2096 bits: "4",
2097 words: 0,
2098 strength: "less than a second",
2099 }
2100 );
2101 });
2102 it("Shows details about the entered entropy", function(done) {
2103 testEntropyFeedback(done,
2104 {
2105 entropy: "AAAAAAAA",
2106 filtered: "AAAAAAAA",
2107 type: "hexadecimal",
2108 events: "8",
2109 bits: "32",
2110 words: 3,
2111 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2112 }
2113 );
2114 });
2115 it("Shows details about the entered entropy", function(done) {
2116 testEntropyFeedback(done,
2117 {
2118 entropy: "AAAAAAAA B",
2119 filtered: "AAAAAAAAB",
2120 type: "hexadecimal",
2121 events: "9",
2122 bits: "36",
2123 words: 3,
2124 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2125 }
2126 );
2127 });
2128 it("Shows details about the entered entropy", function(done) {
2129 testEntropyFeedback(done,
2130 {
2131 entropy: "AAAAAAAA BBBBBBBB",
2132 filtered: "AAAAAAAABBBBBBBB",
2133 type: "hexadecimal",
2134 events: "16",
2135 bits: "64",
2136 words: 6,
2137 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
2138 }
2139 );
2140 });
2141 it("Shows details about the entered entropy", function(done) {
2142 testEntropyFeedback(done,
2143 {
2144 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC",
2145 filtered: "AAAAAAAABBBBBBBBCCCCCCCC",
2146 type: "hexadecimal",
2147 events: "24",
2148 bits: "96",
2149 words: 9,
2150 strength: "less than a second",
2151 }
2152 );
2153 });
2154 it("Shows details about the entered entropy", function(done) {
2155 testEntropyFeedback(done,
2156 {
2157 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDD",
2158 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDD",
2159 type: "hexadecimal",
2160 events: "32",
2161 bits: "128",
2162 words: 12,
2163 strength: "2 minutes",
2164 }
2165 );
2166 });
2167 it("Shows details about the entered entropy", function(done) {
2168 testEntropyFeedback(done,
2169 {
2170 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA",
2171 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDA",
2172 type: "hexadecimal",
2173 events: "32",
2174 bits: "128",
2175 words: 12,
2176 strength: "2 days",
2177 }
2178 );
2179 });
2180 it("Shows details about the entered entropy", function(done) {
2181 testEntropyFeedback(done,
2182 {
2183 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA EEEEEEEE",
2184 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDAEEEEEEEE",
2185 type: "hexadecimal",
2186 events: "40",
2187 bits: "160",
2188 words: 15,
2189 strength: "3 years",
2190 }
2191 );
2192 });
2193 it("Shows details about the entered entropy", function(done) {
2194 testEntropyFeedback(done,
2195 {
2196 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA EEEEEEEE FFFFFFFF",
2197 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDAEEEEEEEEFFFFFFFF",
2198 type: "hexadecimal",
2199 events: "48",
2200 bits: "192",
2201 words: 18,
2202 strength: "centuries",
2203 }
2204 );
2205 });
2206 it("Shows details about the entered entropy", function(done) {
2207 testEntropyFeedback(done,
2208 {
2209 entropy: "7d",
2210 type: "card",
2211 events: "1",
2212 bits: "4",
2213 words: 0,
2214 strength: "less than a second",
2215 }
2216 );
2217 });
2218 it("Shows details about the entered entropy", function(done) {
2219 testEntropyFeedback(done,
2220 {
2221 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2222 type: "card (full deck)",
2223 events: "52",
2224 bits: "225",
2225 words: 21,
2226 strength: "centuries",
2227 }
2228 );
2229 });
2230 it("Shows details about the entered entropy", function(done) {
2231 testEntropyFeedback(done,
2232 {
2233 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks3d",
2234 type: "card (full deck, 1 duplicate: 3d)",
2235 events: "53",
2236 bits: "254",
2237 words: 21,
2238 strength: "centuries",
2239 }
2240 );
2241 });
2242 it("Shows details about the entered entropy", function(done) {
2243 testEntropyFeedback(done,
2244 {
2245 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqs3d4d",
2246 type: "card (2 duplicates: 3d 4d, 1 missing: KS)",
2247 events: "53",
2248 bits: "254",
2249 words: 21,
2250 strength: "centuries",
2251 }
2252 );
2253 });
2254 it("Shows details about the entered entropy", function(done) {
2255 testEntropyFeedback(done,
2256 {
2257 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqs3d4d5d6d",
2258 type: "card (4 duplicates: 3d 4d 5d..., 1 missing: KS)",
2259 events: "55",
2260 bits: "264",
2261 words: 24,
2262 strength: "centuries",
2263 }
2264 );
2265 });
2266 it("Shows details about the entered entropy", function(done) {
2267 testEntropyFeedback(done,
2268 // Next test was throwing uncaught error in zxcvbn
2269 // Also tests 451 bits, ie Math.log2(52!)*2 = 225.58 * 2
2270 {
2271 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsksac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2272 type: "card (full deck, 52 duplicates: ac 2c 3c...)",
2273 events: "104",
2274 bits: "499",
2275 words: 45,
2276 strength: "centuries",
2277 }
2278 );
2279 });
2280 it("Shows details about the entered entropy", function(done) {
2281 testEntropyFeedback(done,
2282 // Case insensitivity to duplicate cards
2283 {
2284 entropy: "asAS",
2285 type: "card (1 duplicate: AS)",
2286 events: "2",
2287 bits: "9",
2288 words: 0,
2289 strength: "less than a second",
2290 }
2291 );
2292 });
2293 it("Shows details about the entered entropy", function(done) {
2294 testEntropyFeedback(done,
2295 {
2296 entropy: "ASas",
2297 type: "card (1 duplicate: as)",
2298 events: "2",
2299 bits: "9",
2300 words: 0,
2301 strength: "less than a second",
2302 }
2303 );
2304 });
2305 it("Shows details about the entered entropy", function(done) {
2306 testEntropyFeedback(done,
2307 // Missing cards are detected
2308 {
2309 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2310 type: "card (1 missing: 9C)",
2311 events: "51",
2312 bits: "221",
2313 words: 18,
2314 strength: "centuries",
2315 }
2316 );
2317 });
2318 it("Shows details about the entered entropy", function(done) {
2319 testEntropyFeedback(done,
2320 {
2321 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2322 type: "card (2 missing: 9C 5D)",
2323 events: "50",
2324 bits: "216",
2325 words: 18,
2326 strength: "centuries",
2327 }
2328 );
2329 });
2330 it("Shows details about the entered entropy", function(done) {
2331 testEntropyFeedback(done,
2332 {
2333 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d7d8d9dtdjd kdah2h3h 5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
2334 type: "card (4 missing: 9C 5D QD...)",
2335 events: "48",
2336 bits: "208",
2337 words: 18,
2338 strength: "centuries",
2339 }
2340 );
2341 });
2342 it("Shows details about the entered entropy", function(done) {
2343 testEntropyFeedback(done,
2344 // More than six missing cards does not show message
2345 {
2346 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d 8d9d jd kdah2h3h 5h6h7h8h9hthjhqhkh 2s3s4s5s6s7s8s9stsjsqsks",
2347 type: "card",
2348 events: "45",
2349 bits: "195",
2350 words: 18,
2351 strength: "centuries",
2352 }
2353 );
2354 });
2355 it("Shows details about the entered entropy", function(done) {
2356 testEntropyFeedback(done,
2357 // Multiple decks of cards increases bits per event
2358 {
2359 entropy: "3d",
2360 events: "1",
2361 bits: "4",
2362 bitsPerEvent: "4.34",
2363 }
2364 );
2365 });
2366 it("Shows details about the entered entropy", function(done) {
2367 testEntropyFeedback(done,
2368 {
2369 entropy: "3d3d",
2370 events: "2",
2371 bits: "9",
2372 bitsPerEvent: "4.80",
2373 }
2374 );
2375 });
2376 it("Shows details about the entered entropy", function(done) {
2377 testEntropyFeedback(done,
2378 {
2379 entropy: "3d3d3d",
2380 events: "3",
2381 bits: "15",
2382 bitsPerEvent: "5.01",
2383 }
2384 );
2385 });
2386 it("Shows details about the entered entropy", function(done) {
2387 testEntropyFeedback(done,
2388 {
2389 entropy: "3d3d3d3d",
2390 events: "4",
2391 bits: "20",
2392 bitsPerEvent: "5.14",
2393 }
2394 );
2395 });
2396 it("Shows details about the entered entropy", function(done) {
2397 testEntropyFeedback(done,
2398 {
2399 entropy: "3d3d3d3d3d",
2400 events: "5",
2401 bits: "26",
2402 bitsPerEvent: "5.22",
2403 }
2404 );
2405 });
2406 it("Shows details about the entered entropy", function(done) {
2407 testEntropyFeedback(done,
2408 {
2409 entropy: "3d3d3d3d3d3d",
2410 events: "6",
2411 bits: "31",
2412 bitsPerEvent: "5.28",
2413 }
2414 );
2415 });
2416 it("Shows details about the entered entropy", function(done) {
2417 testEntropyFeedback(done,
2418 {
2419 entropy: "3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d",
2420 events: "33",
2421 bits: "184",
2422 bitsPerEvent: "5.59",
2423 strength: 'less than a second - Repeats like "abcabcabc" are only slightly harder to guess than "abc"',
2424 }
2425 );
2426 });
2427
2428 // Entropy is truncated from the left
2429 it('Truncates entropy from the left', function(done) {
2430 // Truncate from left means 0000 is removed from the start
2431 // which gives mnemonic 'avocado zoo zone'
2432 // not 1111 removed from the end
2433 // which gives the mnemonic 'abstract zoo zoo'
2434 var entropy = "00000000 00000000 00000000 00000000";
2435 entropy += "11111111 11111111 11111111 1111"; // Missing last byte
2436 driver.findElement(By.css('.use-entropy'))
2437 .click();
2438 driver.findElement(By.css('.entropy'))
2439 .sendKeys(entropy);
2440 driver.sleep(generateDelay).then(function() {
2441 driver.findElement(By.css(".phrase"))
2442 .getAttribute("value").then(function(phrase) {
2443 expect(phrase).toBe("avocado zoo zone");
2444 done();
2445 });
2446 });
2447 });
2448
2449 // Very large entropy results in very long mnemonics
2450 it('Converts very long entropy to very long mnemonics', function(done) {
2451 var entropy = "";
2452 for (var i=0; i<33; i++) {
2453 entropy += "AAAAAAAA"; // 3 words * 33 iterations = 99 words
2454 }
2455 driver.findElement(By.css('.use-entropy'))
2456 .click();
2457 driver.findElement(By.css('.entropy'))
2458 .sendKeys(entropy);
2459 driver.sleep(generateDelay).then(function() {
2460 driver.findElement(By.css(".phrase"))
2461 .getAttribute("value").then(function(phrase) {
2462 var wordCount = phrase.split(/\s+/g).length;
2463 expect(wordCount).toBe(99);
2464 done();
2465 });
2466 });
2467 });
2468
2469 // Is compatible with bip32jp entropy
2470 // https://bip32jp.github.io/english/index.html
2471 // NOTES:
2472 // Is incompatible with:
2473 // base 20
2474 it('Is compatible with bip32jp.github.io', function(done) {
2475 var entropy = "543210543210543210543210543210543210543210543210543210543210543210543210543210543210543210543210543";
2476 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";
2477 driver.findElement(By.css('.use-entropy'))
2478 .click();
2479 driver.findElement(By.css('.entropy'))
2480 .sendKeys(entropy);
2481 driver.sleep(generateDelay).then(function() {
2482 driver.findElement(By.css(".phrase"))
2483 .getAttribute("value").then(function(phrase) {
2484 expect(phrase).toBe(expectedPhrase);
2485 done();
2486 });
2487 });
2488 });
2489
2490 // Blank entropy does not generate mnemonic or addresses
2491 it('Does not generate mnemonic for blank entropy', function(done) {
2492 driver.findElement(By.css('.use-entropy'))
2493 .click();
2494 driver.findElement(By.css('.entropy'))
2495 .clear();
2496 // check there is no mnemonic
2497 driver.sleep(generateDelay).then(function() {
2498 driver.findElement(By.css(".phrase"))
2499 .getAttribute("value").then(function(phrase) {
2500 expect(phrase).toBe("");
2501 // check there is no mnemonic
2502 driver.findElements(By.css(".address"))
2503 .then(function(addresses) {
2504 expect(addresses.length).toBe(0);
2505 // Check the feedback says 'blank entropy'
2506 driver.findElement(By.css(".feedback"))
2507 .getText()
2508 .then(function(feedbackText) {
2509 expect(feedbackText).toBe("Blank entropy");
2510 done();
2511 });
2512 })
2513 });
2514 });
2515 });
2516
2517 // Mnemonic length can be selected even for weak entropy
2518 it('Allows selection of mnemonic length even for weak entropy', function(done) {
2519 driver.findElement(By.css('.use-entropy'))
2520 .click();
2521 driver.executeScript(function() {
2522 $(".mnemonic-length").val("18").trigger("change");
2523 });
2524 driver.findElement(By.css('.entropy'))
2525 .sendKeys("012345");
2526 driver.sleep(generateDelay).then(function() {
2527 driver.findElement(By.css(".phrase"))
2528 .getAttribute("value").then(function(phrase) {
2529 var wordCount = phrase.split(/\s+/g).length;
2530 expect(wordCount).toBe(18);
2531 done();
2532 });
2533 });
2534 });
2535
2536 // Github issue 33
2537 // https://github.com/iancoleman/bip39/issues/33
2538 // Final cards should contribute entropy
2539 it('Uses as much entropy as possible for the mnemonic', function(done) {
2540 driver.findElement(By.css('.use-entropy'))
2541 .click();
2542 driver.findElement(By.css('.entropy'))
2543 .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");
2544 driver.sleep(generateDelay).then(function() {
2545 // Get mnemonic
2546 driver.findElement(By.css(".phrase"))
2547 .getAttribute("value").then(function(originalPhrase) {
2548 // Set the last 12 cards to be AS
2549 driver.findElement(By.css('.entropy'))
2550 .clear();
2551 driver.findElement(By.css('.entropy'))
2552 .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");
2553 driver.sleep(generateDelay).then(function() {
2554 // Get new mnemonic
2555 driver.findElement(By.css(".phrase"))
2556 .getAttribute("value").then(function(newPhrase) {
2557 expect(originalPhrase).not.toEqual(newPhrase);
2558 done();
2559 });
2560 });
2561 });
2562 });
2563 });
2564
2565 // Github issue 35
2566 // https://github.com/iancoleman/bip39/issues/35
2567 // QR Code support
2568 // TODO this doesn't work in selenium with firefox
2569 // see https://stackoverflow.com/q/40360223
2570 it('Shows a qr code on hover for the phrase', function(done) {
2571 if (browser == "firefox") {
2572 pending("Selenium + Firefox bug for mouseMove, see https://stackoverflow.com/q/40360223");
2573 }
2574 // generate a random mnemonic
2575 var generateEl = driver.findElement(By.css('.generate'));
2576 generateEl.click();
2577 // toggle qr to show (hidden by default)
2578 var phraseEl = driver.findElement(By.css(".phrase"));
2579 phraseEl.click();
2580 var rootKeyEl = driver.findElement(By.css(".root-key"));
2581 driver.sleep(generateDelay).then(function() {
2582 // hover over the root key
2583 driver.actions().mouseMove(rootKeyEl).perform().then(function() {
2584 // check the qr code shows
2585 driver.executeScript(function() {
2586 return $(".qr-container").find("canvas").length > 0;
2587 })
2588 .then(function(qrShowing) {
2589 expect(qrShowing).toBe(true);
2590 // hover away from the phrase
2591 driver.actions().mouseMove(generateEl).perform().then(function() {;
2592 // check the qr code hides
2593 driver.executeScript(function() {
2594 return $(".qr-container").find("canvas").length == 0;
2595 })
2596 .then(function(qrHidden) {
2597 expect(qrHidden).toBe(true);
2598 done();
2599 });
2600 });
2601 });
2602 });
2603 });
2604 });
2605
2606 // BIP44 account extendend private key is shown
2607 // github issue 37 - compatibility with electrum
2608 it('Shows the bip44 account extended private key', function(done) {
2609 driver.findElement(By.css(".phrase"))
2610 .sendKeys("abandon abandon ability");
2611 driver.sleep(generateDelay).then(function() {
2612 driver.findElement(By.css("#bip44 .account-xprv"))
2613 .getAttribute("value")
2614 .then(function(xprv) {
2615 expect(xprv).toBe("xprv9yzrnt4zWVJUr1k2VxSPy9ettKz5PpeDMgaVG7UKedhqnw1tDkxP2UyYNhuNSumk2sLE5ctwKZs9vwjsq3e1vo9egCK6CzP87H2cVYXpfwQ");
2616 done();
2617 });
2618 });
2619 });
2620
2621 // BIP44 account extendend public key is shown
2622 // github issue 37 - compatibility with electrum
2623 it('Shows the bip44 account extended public key', function(done) {
2624 driver.findElement(By.css(".phrase"))
2625 .sendKeys("abandon abandon ability");
2626 driver.sleep(generateDelay).then(function() {
2627 driver.findElement(By.css("#bip44 .account-xpub"))
2628 .getAttribute("value")
2629 .then(function(xprv) {
2630 expect(xprv).toBe("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2631 done();
2632 });
2633 });
2634 });
2635
2636 // github issue 40
2637 // BIP32 root key can be set as an xpub
2638 it('Generates addresses from xpub as bip32 root key', function(done) {
2639 driver.findElement(By.css('#bip32-tab a'))
2640 .click();
2641 // set xpub for account 0 of bip44 for 'abandon abandon ability'
2642 driver.findElement(By.css("#root-key"))
2643 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2644 driver.sleep(generateDelay).then(function() {
2645 // check the addresses are generated
2646 getFirstAddress(function(address) {
2647 expect(address).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug");
2648 // check the xprv key is not set
2649 driver.findElement(By.css(".extended-priv-key"))
2650 .getAttribute("value")
2651 .then(function(xprv) {
2652 expect(xprv).toBe("NA");
2653 // check the private key is not set
2654 driver.findElements(By.css(".privkey"))
2655 .then(function(els) {
2656 els[0]
2657 .getText()
2658 .then(function(privkey) {
2659 expect(xprv).toBe("NA");
2660 done();
2661 });
2662 });
2663 });
2664 });
2665 });
2666 });
2667
2668 // github issue 40
2669 // xpub for bip32 root key will not work with hardened derivation paths
2670 it('Shows error for hardened derivation paths with xpub root key', function(done) {
2671 // set xpub for account 0 of bip44 for 'abandon abandon ability'
2672 driver.findElement(By.css("#root-key"))
2673 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2674 driver.sleep(feedbackDelay).then(function() {
2675 // Check feedback is correct
2676 driver.findElement(By.css('.feedback'))
2677 .getText()
2678 .then(function(feedback) {
2679 var msg = "Hardened derivation path is invalid with xpub key";
2680 expect(feedback).toBe(msg);
2681 // Check no addresses are shown
2682 driver.findElements(By.css('.addresses tr'))
2683 .then(function(rows) {
2684 expect(rows.length).toBe(0);
2685 done();
2686 });
2687 });
2688 });
2689 });
2690
2691 // github issue 39
2692 // no root key shows feedback
2693 it('Shows feedback for no root key', function(done) {
2694 // set xpub for account 0 of bip44 for 'abandon abandon ability'
2695 driver.findElement(By.css('#bip32-tab a'))
2696 .click();
2697 driver.sleep(feedbackDelay).then(function() {
2698 // Check feedback is correct
2699 driver.findElement(By.css('.feedback'))
2700 .getText()
2701 .then(function(feedback) {
2702 expect(feedback).toBe("Invalid root key");
2703 done();
2704 });
2705 });
2706 });
2707
2708 // Github issue 44
2709 // display error switching tabs while addresses are generating
2710 it('Can change details while old addresses are still being generated', function(done) {
2711 // Set to generate 199 more addresses.
2712 // This will take a long time allowing a new set of addresses to be
2713 // generated midway through this lot.
2714 // The newly generated addresses should not include any from the old set.
2715 // Any more than 199 will show an alert which needs to be accepted.
2716 driver.findElement(By.css('.rows-to-add'))
2717 .clear();
2718 driver.findElement(By.css('.rows-to-add'))
2719 .sendKeys('199');
2720 // set the prhase
2721 driver.findElement(By.css('.phrase'))
2722 .sendKeys("abandon abandon ability");
2723 driver.sleep(generateDelay).then(function() {
2724 // change tabs which should cancel the previous generating
2725 driver.findElement(By.css('.rows-to-add'))
2726 .clear();
2727 driver.findElement(By.css('.rows-to-add'))
2728 .sendKeys('20');
2729 driver.findElement(By.css('#bip32-tab a'))
2730 .click()
2731 driver.sleep(generateDelay).then(function() {
2732 driver.findElements(By.css('.index'))
2733 .then(function(els) {
2734 // check the derivation paths have the right quantity
2735 expect(els.length).toBe(20);
2736 // check the derivation paths are in order
2737 testRowsAreInCorrectOrder(done);
2738 });
2739 });
2740 });
2741 }, generateDelay + 5000);
2742
2743 // Github issue 49
2744 // padding for binary should give length with multiple of 256
2745 // hashed entropy 1111 is length 252, so requires 4 leading zeros
2746 // prior to issue 49 it would only generate 2 leading zeros, ie missing 2
2747 it('Pads hashed entropy with leading zeros', function(done) {
2748 driver.findElement(By.css('.use-entropy'))
2749 .click();
2750 driver.executeScript(function() {
2751 $(".mnemonic-length").val("15").trigger("change");
2752 });
2753 driver.findElement(By.css('.entropy'))
2754 .sendKeys("1111");
2755 driver.sleep(generateDelay).then(function() {
2756 driver.findElement(By.css('.phrase'))
2757 .getAttribute("value")
2758 .then(function(phrase) {
2759 expect(phrase).toBe("avocado valid quantum cross link predict excuse edit street able flame large galaxy ginger nuclear");
2760 done();
2761 });
2762 });
2763 });
2764
2765 // Github pull request 55
2766 // https://github.com/iancoleman/bip39/pull/55
2767 // Client select
2768 it('Can set the derivation path on bip32 tab for bitcoincore', function(done) {
2769 testClientSelect(done, {
2770 selectValue: "0",
2771 bip32path: "m/0'/0'",
2772 useHardenedAddresses: "true",
2773 });
2774 });
2775 it('Can set the derivation path on bip32 tab for multibit', function(done) {
2776 testClientSelect(done, {
2777 selectValue: "2",
2778 bip32path: "m/0'/0",
2779 useHardenedAddresses: null,
2780 });
2781 });
2782
2783 // github issue 58
2784 // https://github.com/iancoleman/bip39/issues/58
2785 // bip32 derivation is correct, does not drop leading zeros
2786 // see also
2787 // https://medium.com/@alexberegszaszi/why-do-my-bip32-wallets-disagree-6f3254cc5846
2788 it('Retains leading zeros for bip32 derivation', function(done) {
2789 driver.findElement(By.css(".phrase"))
2790 .sendKeys("fruit wave dwarf banana earth journey tattoo true farm silk olive fence");
2791 driver.findElement(By.css(".passphrase"))
2792 .sendKeys("banana");
2793 driver.sleep(generateDelay).then(function() {
2794 getFirstAddress(function(address) {
2795 // Note that bitcore generates an incorrect address
2796 // 13EuKhffWkBE2KUwcbkbELZb1MpzbimJ3Y
2797 // see the medium.com link above for more details
2798 expect(address).toBe("17rxURoF96VhmkcEGCj5LNQkmN9HVhWb7F");
2799 done();
2800 });
2801 });
2802 });
2803
2804 // github issue 60
2805 // Japanese mnemonics generate incorrect bip32 seed
2806 // BIP39 seed is set from phrase
2807 it('Generates correct seed for Japanese mnemonics', function(done) {
2808 driver.findElement(By.css(".phrase"))
2809 .sendKeys("あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あおぞら");
2810 driver.findElement(By.css(".passphrase"))
2811 .sendKeys("メートルガバヴァぱばぐゞちぢ十人十色");
2812 driver.sleep(generateDelay).then(function() {
2813 driver.findElement(By.css(".seed"))
2814 .getAttribute("value")
2815 .then(function(seed) {
2816 expect(seed).toBe("a262d6fb6122ecf45be09c50492b31f92e9beb7d9a845987a02cefda57a15f9c467a17872029a9e92299b5cbdf306e3a0ee620245cbd508959b6cb7ca637bd55");
2817 done();
2818 });
2819 });
2820 });
2821
2822 // BIP49 official test vectors
2823 // https://github.com/bitcoin/bips/blob/master/bip-0049.mediawiki#test-vectors
2824 it('Generates BIP49 addresses matching the official test vectors', function(done) {
2825 driver.findElement(By.css('#bip49-tab a'))
2826 .click();
2827 selectNetwork("BTC - Bitcoin Testnet");
2828 driver.findElement(By.css(".phrase"))
2829 .sendKeys("abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about");
2830 driver.sleep(generateDelay).then(function() {
2831 getFirstAddress(function(address) {
2832 expect(address).toBe("2Mww8dCYPUpKHofjgcXcBCEGmniw9CoaiD2");
2833 done();
2834 });
2835 });
2836 });
2837
2838 // BIP49 derivation path is shown
2839 it('Shows the bip49 derivation path', function(done) {
2840 driver.findElement(By.css('#bip49-tab a'))
2841 .click();
2842 driver.findElement(By.css(".phrase"))
2843 .sendKeys("abandon abandon ability");
2844 driver.sleep(generateDelay).then(function() {
2845 driver.findElement(By.css('#bip49 .path'))
2846 .getAttribute("value")
2847 .then(function(path) {
2848 expect(path).toBe("m/49'/0'/0'/0");
2849 done();
2850 });
2851 });
2852 });
2853
2854 // BIP49 extended private key is shown
2855 it('Shows the bip49 extended private key', function(done) {
2856 driver.findElement(By.css('#bip49-tab a'))
2857 .click();
2858 driver.findElement(By.css(".phrase"))
2859 .sendKeys("abandon abandon ability");
2860 driver.sleep(generateDelay).then(function() {
2861 driver.findElement(By.css('.extended-priv-key'))
2862 .getAttribute("value")
2863 .then(function(xprv) {
2864 expect(xprv).toBe("yprvALYB4DYRG6CzzVgzQZwwqjAA2wjBGC3iEd7KYYScpoDdmf75qMRWZWxoFcRXBJjgEXdFqJ9vDRGRLJQsrL22Su5jMbNFeM9vetaGVqy9Qy2");
2865 done();
2866 });
2867 });
2868 });
2869
2870 // BIP49 extended public key is shown
2871 it('Shows the bip49 extended public key', function(done) {
2872 driver.findElement(By.css('#bip49-tab a'))
2873 .click();
2874 driver.findElement(By.css(".phrase"))
2875 .sendKeys("abandon abandon ability");
2876 driver.sleep(generateDelay).then(function() {
2877 driver.findElement(By.css('.extended-pub-key'))
2878 .getAttribute("value")
2879 .then(function(xprv) {
2880 expect(xprv).toBe("ypub6ZXXTj5K6TmJCymTWbUxCs6tayZffemZbr2vLvrEP8kceTSENtjm7KHH6thvAKxVar9fGe8rgsPEX369zURLZ68b4f7Vexz7RuXsjQ69YDt");
2881 done();
2882 });
2883 });
2884 });
2885
2886 // BIP49 account field changes address list
2887 it('Can set the bip49 account field', function(done) {
2888 driver.findElement(By.css('#bip49-tab a'))
2889 .click();
2890 driver.findElement(By.css('#bip49 .account'))
2891 .clear();
2892 driver.findElement(By.css('#bip49 .account'))
2893 .sendKeys("1");
2894 driver.findElement(By.css(".phrase"))
2895 .sendKeys("abandon abandon ability");
2896 driver.sleep(generateDelay).then(function() {
2897 getFirstAddress(function(address) {
2898 expect(address).toBe("381wg1GGN4rP88rNC9v7QWsiww63yLVPsn");
2899 done();
2900 });
2901 });
2902 });
2903
2904 // BIP49 change field changes address list
2905 it('Can set the bip49 change field', function(done) {
2906 driver.findElement(By.css('#bip49-tab a'))
2907 .click();
2908 driver.findElement(By.css('#bip49 .change'))
2909 .clear();
2910 driver.findElement(By.css('#bip49 .change'))
2911 .sendKeys("1");
2912 driver.findElement(By.css(".phrase"))
2913 .sendKeys("abandon abandon ability");
2914 driver.sleep(generateDelay).then(function() {
2915 getFirstAddress(function(address) {
2916 expect(address).toBe("3PEM7MiKed5konBoN66PQhK8r3hjGhy9dT");
2917 done();
2918 });
2919 });
2920 });
2921
2922 // BIP49 account extendend private key is shown
2923 it('Shows the bip49 account extended private key', function(done) {
2924 driver.findElement(By.css('#bip49-tab a'))
2925 .click();
2926 driver.findElement(By.css(".phrase"))
2927 .sendKeys("abandon abandon ability");
2928 driver.sleep(generateDelay).then(function() {
2929 driver.findElement(By.css('#bip49 .account-xprv'))
2930 .getAttribute("value")
2931 .then(function(xprv) {
2932 expect(xprv).toBe("yprvAHtB1M5Wp675aLzFy9TJYK2mSsLkg6mcBRh5DZTR7L4EnYSmYPaL63KFA4ycg1PngW5LfkmejxzosCs17TKZMpRFKc3z5SJar6QAKaFcaZL");
2933 done();
2934 });
2935 });
2936 });
2937
2938 // BIP49 account extendend public key is shown
2939 it('Shows the bip49 account extended public key', function(done) {
2940 driver.findElement(By.css('#bip49-tab a'))
2941 .click();
2942 driver.findElement(By.css(".phrase"))
2943 .sendKeys("abandon abandon ability");
2944 driver.sleep(generateDelay).then(function() {
2945 driver.findElement(By.css('#bip49 .account-xpub'))
2946 .getAttribute("value")
2947 .then(function(xprv) {
2948 expect(xprv).toBe("ypub6WsXQrcQeTfNnq4j5AzJuSyVzuBF5ZVTYecg1ws2ffbDfLmv5vtadqdj1NgR6C6gufMpMfJpHxvb6JEQKvETVNWCRanNedfJtnTchZiJtsL");
2949 done();
2950 });
2951 });
2952 });
2953
2954 // Test selecting coin where bip49 is unavailable (eg CLAM)
2955 it('Shows an error on bip49 tab for coins without bip49', function(done) {
2956 driver.findElement(By.css('#bip49-tab a'))
2957 .click();
2958 driver.findElement(By.css(".phrase"))
2959 .sendKeys("abandon abandon ability");
2960 driver.sleep(generateDelay).then(function() {
2961 selectNetwork("CLAM - Clams");
2962 // bip49 available is hidden
2963 driver.findElement(By.css('#bip49 .available'))
2964 .getAttribute("class")
2965 .then(function(classes) {
2966 expect(classes).toContain("hidden");
2967 // bip49 unavailable is shown
2968 driver.findElement(By.css('#bip49 .unavailable'))
2969 .getAttribute("class")
2970 .then(function(classes) {
2971 expect(classes).not.toContain("hidden");
2972 // check there are no addresses shown
2973 driver.findElements(By.css('.addresses tr'))
2974 .then(function(rows) {
2975 expect(rows.length).toBe(0);
2976 // check the derived private key is blank
2977 driver.findElement(By.css('.extended-priv-key'))
2978 .getAttribute("value")
2979 .then(function(xprv) {
2980 expect(xprv).toBe('');
2981 // check the derived public key is blank
2982 driver.findElement(By.css('.extended-pub-key'))
2983 .getAttribute("value")
2984 .then(function(xpub) {
2985 expect(xpub).toBe('');
2986 done();
2987 });
2988 });
2989 })
2990 });
2991 });
2992 });
2993 });
2994
2995 // github issue 43
2996 // Cleared mnemonic and root key still allows addresses to be generated
2997 // https://github.com/iancoleman/bip39/issues/43
2998 it('Clears old root keys from memory when mnemonic is cleared', function(done) {
2999 // set the phrase
3000 driver.findElement(By.css(".phrase"))
3001 .sendKeys("abandon abandon ability");
3002 driver.sleep(generateDelay).then(function() {
3003 // clear the mnemonic and root key
3004 // using selenium .clear() doesn't seem to trigger the 'input' event
3005 // so clear it using keys instead
3006 driver.findElement(By.css('.phrase'))
3007 .sendKeys(Key.CONTROL,"a");
3008 driver.findElement(By.css('.phrase'))
3009 .sendKeys(Key.DELETE);
3010 driver.findElement(By.css('.root-key'))
3011 .sendKeys(Key.CONTROL,"a");
3012 driver.findElement(By.css('.root-key'))
3013 .sendKeys(Key.DELETE);
3014 driver.sleep(generateDelay).then(function() {
3015 // try to generate more addresses
3016 driver.findElement(By.css('.more'))
3017 .click();
3018 driver.sleep(generateDelay).then(function() {
3019 driver.findElements(By.css(".addresses tr"))
3020 .then(function(els) {
3021 // check there are no addresses shown
3022 expect(els.length).toBe(0);
3023 done();
3024 });
3025 });
3026 });
3027 });
3028 });
3029
3030 // Github issue 95
3031 // error trying to generate addresses from xpub with hardened derivation
3032 it('Shows error for hardened addresses with xpub root key', function(done) {
3033 driver.findElement(By.css('#bip32-tab a'))
3034 .click()
3035 driver.executeScript(function() {
3036 $(".hardened-addresses").prop("checked", true);
3037 });
3038 // set xpub for account 0 of bip44 for 'abandon abandon ability'
3039 driver.findElement(By.css("#root-key"))
3040 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
3041 driver.sleep(feedbackDelay).then(function() {
3042 // Check feedback is correct
3043 driver.findElement(By.css('.feedback'))
3044 .getText()
3045 .then(function(feedback) {
3046 var msg = "Hardened derivation path is invalid with xpub key";
3047 expect(feedback).toBe(msg);
3048 done();
3049 });
3050 });
3051 });
3052
3053 // Litecoin uses ltub by default, and can optionally be set to xprv
3054 // github issue 96
3055 // https://github.com/iancoleman/bip39/issues/96
3056 // Issue with extended keys on Litecoin
3057 it('Uses ltub by default for litecoin, but can be set to xprv', function(done) {
3058 driver.findElement(By.css('.phrase'))
3059 .sendKeys("abandon abandon ability");
3060 selectNetwork("LTC - Litecoin");
3061 driver.sleep(generateDelay).then(function() {
3062 // check the extended key is generated correctly
3063 driver.findElement(By.css('.root-key'))
3064 .getAttribute("value")
3065 .then(function(rootKey) {
3066 expect(rootKey).toBe("Ltpv71G8qDifUiNesiPqf6h5V6eQ8ic77oxQiYtawiACjBEx3sTXNR2HGDGnHETYxESjqkMLFBkKhWVq67ey1B2MKQXannUqNy1RZVHbmrEjnEU");
3067 // set litecoin to use ltub
3068 driver.executeScript(function() {
3069 $(".litecoin-use-ltub").prop("checked", false);
3070 $(".litecoin-use-ltub").trigger("change");
3071 });
3072 driver.sleep(generateDelay).then(function() {
3073 driver.findElement(By.css('.root-key'))
3074 .getAttribute("value")
3075 .then(function(rootKey) {
3076 expect(rootKey).toBe("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
3077 done();
3078 });
3079 })
3080 });
3081 });
3082 });
3083
3084 // github issue 99
3085 // https://github.com/iancoleman/bip39/issues/99#issuecomment-327094159
3086 // "warn me emphatically when they have detected invalid input" to the entropy field
3087 // A warning is shown when entropy is filtered and discarded
3088 it('Warns when entropy is filtered and discarded', function(done) {
3089 driver.findElement(By.css('.use-entropy'))
3090 .click();
3091 // set entropy to have no filtered content
3092 driver.findElement(By.css('.entropy'))
3093 .sendKeys("00000000 00000000 00000000 00000000");
3094 driver.sleep(generateDelay).then(function() {
3095 // check the filter warning does not show
3096 driver.findElement(By.css('.entropy-container .filter-warning'))
3097 .getAttribute("class")
3098 .then(function(classes) {
3099 expect(classes).toContain("hidden");
3100 // set entropy to have some filtered content
3101 driver.findElement(By.css('.entropy'))
3102 .sendKeys("10000000 zxcvbn 00000000 00000000 00000000");
3103 driver.sleep(entropyFeedbackDelay).then(function() {
3104 // check the filter warning shows
3105 driver.findElement(By.css('.entropy-container .filter-warning'))
3106 .getAttribute("class")
3107 .then(function(classes) {
3108 expect(classes).not.toContain("hidden");
3109 done();
3110 });
3111 });
3112 });
3113 });
3114 });
3115
3116 // Bitcoin Cash address can be set to use bitpay format
3117 it('Can use bitpay format for bitcoin cash addresses', function(done) {
3118 driver.executeScript(function() {
3119 $(".use-bitpay-addresses").prop("checked", true);
3120 });
3121 driver.findElement(By.css('.phrase'))
3122 .sendKeys("abandon abandon ability");
3123 selectNetwork("BCH - Bitcoin Cash");
3124 driver.sleep(generateDelay).then(function() {
3125 getFirstAddress(function(address) {
3126 expect(address).toBe("CZnpA9HPmvhuhLLPWJP8rNDpLUYXy1LXFk");
3127 done();
3128 });
3129 });
3130 });
3131
3132 // End of tests ported from old suit, so no more comments above each test now
3133
3134 it('Can generate more addresses from a custom index', function(done) {
3135 var expectedIndexes = [
3136 0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,
3137 40,41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59
3138 ];
3139 driver.findElement(By.css('.phrase'))
3140 .sendKeys("abandon abandon ability");
3141 driver.sleep(generateDelay).then(function() {
3142 // Set start of next lot of rows to be from index 40
3143 // which means indexes 20-39 will not be in the table.
3144 driver.findElement(By.css('.more-rows-start-index'))
3145 .sendKeys("40");
3146 driver.findElement(By.css('.more'))
3147 .click();
3148 driver.sleep(generateDelay).then(function() {
3149 // Check actual indexes in the table match the expected pattern
3150 driver.findElements(By.css(".index"))
3151 .then(function(els) {
3152 expect(els.length).toBe(expectedIndexes.length);
3153 var testRowAtIndex = function(i) {
3154 if (i >= expectedIndexes.length) {
3155 done();
3156 }
3157 else {
3158 els[i].getText()
3159 .then(function(actualPath) {
3160 var noHardened = actualPath.replace(/'/g, "");
3161 var pathBits = noHardened.split("/")
3162 var lastBit = pathBits[pathBits.length-1];
3163 var actualIndex = parseInt(lastBit);
3164 var expectedIndex = expectedIndexes[i];
3165 expect(actualIndex).toBe(expectedIndex);
3166 testRowAtIndex(i+1);
3167 });
3168 }
3169 }
3170 testRowAtIndex(0);
3171 });
3172 });
3173 });
3174 });
3175
3176 it('Can generate BIP141 addresses with P2WPKH-in-P2SH semanitcs', function(done) {
3177 // Sourced from BIP49 official test specs
3178 driver.findElement(By.css('#bip141-tab a'))
3179 .click();
3180 driver.findElement(By.css('.bip141-path'))
3181 .clear();
3182 driver.findElement(By.css('.bip141-path'))
3183 .sendKeys("m/49'/1'/0'/0");
3184 selectNetwork("BTC - Bitcoin Testnet");
3185 driver.findElement(By.css(".phrase"))
3186 .sendKeys("abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about");
3187 driver.sleep(generateDelay).then(function() {
3188 getFirstAddress(function(address) {
3189 expect(address).toBe("2Mww8dCYPUpKHofjgcXcBCEGmniw9CoaiD2");
3190 done();
3191 });
3192 });
3193 });
3194
3195 it('Can generate BIP141 addresses with P2WPKH semanitcs', function(done) {
3196 // This result tested against bitcoinjs-lib test spec for segwit address
3197 // using the first private key of this mnemonic and default path m/0
3198 // https://github.com/bitcoinjs/bitcoinjs-lib/blob/9c8503cab0c6c30a95127042703bc18e8d28c76d/test/integration/addresses.js#L50
3199 // so whilst not directly comparable, substituting the private key produces
3200 // identical results between this tool and the bitcoinjs-lib test.
3201 // Private key generated is:
3202 // L3L8Nu9whawPBNLGtFqDhKut9DKKfG3CQoysupT7BimqVCZsLFNP
3203 driver.findElement(By.css('#bip141-tab a'))
3204 .click();
3205 // Choose P2WPKH
3206 driver.executeScript(function() {
3207 $(".bip141-semantics option[selected]").removeAttr("selected");
3208 $(".bip141-semantics option").filter(function(i,e) {
3209 return $(e).html() == "P2WPKH";
3210 }).prop("selected", true);
3211 $(".bip141-semantics").trigger("change");
3212 });
3213 driver.findElement(By.css(".phrase"))
3214 .sendKeys("abandon abandon ability");
3215 driver.sleep(generateDelay).then(function() {
3216 getFirstAddress(function(address) {
3217 expect(address).toBe("bc1qfwu6a5a3evygrk8zvdxxvz4547lmpyx5vsfxe9");
3218 done();
3219 });
3220 });
3221 });
3222
3223 it('Shows the entropy used by the PRNG when clicking generate', function(done) {
3224 driver.findElement(By.css('.generate')).click();
3225 driver.sleep(generateDelay).then(function() {
3226 driver.findElement(By.css('.entropy'))
3227 .getAttribute("value")
3228 .then(function(entropy) {
3229 expect(entropy).not.toBe("");
3230 done();
3231 });
3232 });
3233 });
3234
3235 it('Shows the index of each word in the mnemonic', function(done) {
3236 driver.findElement(By.css('.phrase'))
3237 .sendKeys("abandon abandon ability");
3238 driver.sleep(generateDelay).then(function() {
3239 driver.findElement(By.css('.use-entropy'))
3240 .click();
3241 driver.findElement(By.css('.word-indexes'))
3242 .getText()
3243 .then(function(indexes) {
3244 expect(indexes).toBe("0, 0, 1");
3245 done();
3246 });
3247 });
3248 });
3249
3250 it('Shows the derivation path for bip84 tab', function(done) {
3251 driver.findElement(By.css('#bip84-tab a'))
3252 .click()
3253 driver.findElement(By.css('.phrase'))
3254 .sendKeys('abandon abandon ability');
3255 driver.sleep(generateDelay).then(function() {
3256 driver.findElement(By.css('#bip84 .path'))
3257 .getAttribute("value")
3258 .then(function(path) {
3259 expect(path).toBe("m/84'/0'/0'/0");
3260 done();
3261 })
3262 });
3263 });
3264
3265 it('Shows the extended private key for bip84 tab', function(done) {
3266 driver.findElement(By.css('#bip84-tab a'))
3267 .click()
3268 driver.findElement(By.css('.phrase'))
3269 .sendKeys('abandon abandon ability');
3270 driver.sleep(generateDelay).then(function() {
3271 driver.findElement(By.css('.extended-priv-key'))
3272 .getAttribute("value")
3273 .then(function(path) {
3274 expect(path).toBe("zprvAev3RKrZ3QVKiUFCfdeMRen1BPDJgdNt1XpxiDy8acSs4kkAGTCvq7HeRYRNNpo8EtEjCFQBWavJwtCUR29y4TUCH4X5RXMcyq48uN8y9BP");
3275 done();
3276 })
3277 });
3278 });
3279
3280 it('Shows the extended public key for bip84 tab', function(done) {
3281 driver.findElement(By.css('#bip84-tab a'))
3282 .click()
3283 driver.findElement(By.css('.phrase'))
3284 .sendKeys('abandon abandon ability');
3285 driver.sleep(generateDelay).then(function() {
3286 driver.findElement(By.css('.extended-pub-key'))
3287 .getAttribute("value")
3288 .then(function(path) {
3289 expect(path).toBe("zpub6suPpqPSsn3cvxKfmfBMnnijjR3o666jNkkZWcNk8wyqwZ5JozXBNuc8Gs7DB3uLwTDvGVTspVEAUQcEjKF3pZHgywVbubdTqbXTUg7usyx");
3290 done();
3291 })
3292 });
3293 });
3294
3295 it('Changes the address list if bip84 account is changed', function(done) {
3296 driver.findElement(By.css('#bip84-tab a'))
3297 .click()
3298 driver.findElement(By.css('#bip84 .account'))
3299 .sendKeys('1');
3300 driver.findElement(By.css('.phrase'))
3301 .sendKeys('abandon abandon ability');
3302 driver.sleep(generateDelay).then(function() {
3303 getFirstAddress(function(address) {
3304 expect(address).toBe("bc1qp7vv669t2fy965jdzvqwrraana89ctd5ewc662");
3305 done();
3306 });
3307 });
3308 });
3309
3310 it('Changes the address list if bip84 change is changed', function(done) {
3311 driver.findElement(By.css('#bip84-tab a'))
3312 .click()
3313 driver.findElement(By.css('#bip84 .change'))
3314 .sendKeys('1');
3315 driver.findElement(By.css('.phrase'))
3316 .sendKeys('abandon abandon ability');
3317 driver.sleep(generateDelay).then(function() {
3318 getFirstAddress(function(address) {
3319 expect(address).toBe("bc1qr39vj6rh06ff05m53uxq8uazehwhccswylhrs2");
3320 done();
3321 });
3322 });
3323 });
3324
3325 it('Passes the official BIP84 test spec for rootpriv', function(done) {
3326 driver.findElement(By.css('#bip84-tab a'))
3327 .click()
3328 driver.findElement(By.css('.phrase'))
3329 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3330 driver.sleep(generateDelay).then(function() {
3331 driver.findElement(By.css(".root-key"))
3332 .getAttribute("value")
3333 .then(function(rootKey) {
3334 expect(rootKey).toBe("zprvAWgYBBk7JR8Gjrh4UJQ2uJdG1r3WNRRfURiABBE3RvMXYSrRJL62XuezvGdPvG6GFBZduosCc1YP5wixPox7zhZLfiUm8aunE96BBa4Kei5");
3335 done();
3336 })
3337 });
3338 });
3339
3340 it('Passes the official BIP84 test spec for account 0 xprv', function(done) {
3341 driver.findElement(By.css('#bip84-tab a'))
3342 .click()
3343 driver.findElement(By.css('.phrase'))
3344 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3345 driver.sleep(generateDelay).then(function() {
3346 driver.findElement(By.css("#bip84 .account-xprv"))
3347 .getAttribute("value")
3348 .then(function(rootKey) {
3349 expect(rootKey).toBe("zprvAdG4iTXWBoARxkkzNpNh8r6Qag3irQB8PzEMkAFeTRXxHpbF9z4QgEvBRmfvqWvGp42t42nvgGpNgYSJA9iefm1yYNZKEm7z6qUWCroSQnE");
3350 done();
3351 })
3352 });
3353 });
3354
3355 it('Passes the official BIP84 test spec for account 0 xpub', function(done) {
3356 driver.findElement(By.css('#bip84-tab a'))
3357 .click()
3358 driver.findElement(By.css('.phrase'))
3359 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3360 driver.sleep(generateDelay).then(function() {
3361 driver.findElement(By.css("#bip84 .account-xpub"))
3362 .getAttribute("value")
3363 .then(function(rootKey) {
3364 expect(rootKey).toBe("zpub6rFR7y4Q2AijBEqTUquhVz398htDFrtymD9xYYfG1m4wAcvPhXNfE3EfH1r1ADqtfSdVCToUG868RvUUkgDKf31mGDtKsAYz2oz2AGutZYs");
3365 done();
3366 })
3367 });
3368 });
3369
3370 it('Passes the official BIP84 test spec for account 0 first address', function(done) {
3371 driver.findElement(By.css('#bip84-tab a'))
3372 .click()
3373 driver.findElement(By.css('.phrase'))
3374 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3375 driver.sleep(generateDelay).then(function() {
3376 getFirstAddress(function(address) {
3377 expect(address).toBe("bc1qcr8te4kr609gcawutmrza0j4xv80jy8z306fyu");
3378 done();
3379 });
3380 });
3381 });
3382
3383 it('Passes the official BIP84 test spec for account 0 first change address', function(done) {
3384 driver.findElement(By.css('#bip84-tab a'))
3385 .click()
3386 driver.findElement(By.css('.phrase'))
3387 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
3388 driver.findElement(By.css('#bip84 .change'))
3389 .sendKeys('1');
3390 driver.sleep(generateDelay).then(function() {
3391 getFirstAddress(function(address) {
3392 expect(address).toBe("bc1q8c6fshw2dlwun7ekn9qwf37cu2rn755upcp6el");
3393 done();
3394 });
3395 });
3396 });
3397
3398 it('Can display the table as csv', function(done) {
3399 var headings = "path,address,public key,private key";
3400 var row1 = "m/44'/0'/0'/0/0,1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug,033f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3,L26cVSpWFkJ6aQkPkKmTzLqTdLJ923e6CzrVh9cmx21QHsoUmrEE";
3401 var row20 = "m/44'/0'/0'/0/19,1KhBy28XLAciXnnRvm71PvQJaETyrxGV55,02b4b3e396434d8cdd20c03ac4aaa07387784d5d867b75987f516f5705ee68cb3a,L4GrDrjReMsCAu5DkLXn79jSb95qR7Zfx7eshybCQZ1qL32MXJab";
3402 driver.findElement(By.css('.phrase'))
3403 .sendKeys('abandon abandon ability');
3404 driver.sleep(generateDelay).then(function() {
3405 driver.findElement(By.css('.csv'))
3406 .getAttribute("value")
3407 .then(function(csv) {
3408 expect(csv).toContain(headings);
3409 expect(csv).toContain(row1);
3410 expect(csv).toContain(row20);
3411 done();
3412 });
3413 });
3414 });
3415
3416 it('LeftPads ethereum keys that are less than 32 bytes', function(done) {
3417 // see https://github.com/iancoleman/bip39/issues/155
3418 selectNetwork("ETH - Ethereum");
3419 driver.findElement(By.css('#bip32-tab a'))
3420 .click()
3421 driver.findElement(By.css('#bip32-path'))
3422 .clear();
3423 driver.findElement(By.css('#bip32-path'))
3424 .sendKeys("m/44'/60'/0'");
3425 driver.findElement(By.css('.phrase'))
3426 .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');
3427 driver.sleep(generateDelay).then(function() {
3428 getFirstAddress(function(address) {
3429 expect(address).toBe("0x8943E785B4a5714FC87a3aFAad1eB1FeB602B118");
3430 done();
3431 });
3432 });
3433 });
3434
3435 it('Can encrypt private keys using BIP38', function(done) {
3436 // see https://github.com/iancoleman/bip39/issues/140
3437 driver.executeScript(function() {
3438 $(".use-bip38").prop("checked", true);
3439 });
3440 driver.findElement(By.css('.bip38-password'))
3441 .sendKeys('bip38password');
3442 driver.findElement(By.css('.rows-to-add'))
3443 .clear();
3444 driver.findElement(By.css('.rows-to-add'))
3445 .sendKeys('1');
3446 driver.findElement(By.css('.phrase'))
3447 .sendKeys('abandon abandon ability');
3448 driver.sleep(bip38delay).then(function() {
3449 // address
3450 getFirstRowValue(function(address) {
3451 expect(address).toBe("1NCvSdumA3ngMM9c4aqU56AM6rqXddfuXB");
3452 // pubkey
3453 getFirstRowValue(function(pubkey) {
3454 expect(pubkey).toBe("043f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3884a74447ea901729b1e73a999b7520e7cb55b4120e6432c64153ccab8a848e1");
3455 // privkey
3456 getFirstRowValue(function(privkey) {
3457 expect(privkey).toBe("6PRNRiFnj1RoR3sXhymdCvoZCgnUHQpfupNdKkFbWJkwWQEKesWt1EDMDM");
3458 done();
3459 }, ".privkey");
3460 }, ".pubkey");
3461 }, ".address");
3462 });
3463 }, bip38delay + 5000);
3464
3465 it('Shows the checksum for the entropy', function(done) {
3466 driver.findElement(By.css('.use-entropy'))
3467 .click();
3468 driver.findElement(By.css('.entropy'))
3469 .sendKeys("00000000000000000000000000000000");
3470 driver.sleep(generateDelay).then(function() {
3471 driver.findElement(By.css('.checksum'))
3472 .getText()
3473 .then(function(text) {
3474 expect(text).toBe("1");
3475 done();
3476 });
3477 });
3478 });
3479
3480 it('Shows the checksum for the entropy with the correct groupings', function(done) {
3481 driver.findElement(By.css('.use-entropy'))
3482 .click();
3483 // create a checksum of 20 bits, which spans multiple words
3484 driver.findElement(By.css('.entropy'))
3485 .sendKeys("F000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000");
3486 driver.sleep(generateDelay).then(function() {
3487 driver.findElement(By.css('.checksum'))
3488 .getText()
3489 .then(function(text) {
3490 // first group is 9 bits, second group is 11
3491 expect(text).toBe("011010111 01110000110");
3492 done();
3493 });
3494 });
3495 });
3496
3497 it('Uses vprv for bitcoin testnet p2wpkh', function(done) {
3498 selectNetwork("BTC - Bitcoin Testnet");
3499 driver.findElement(By.css('#bip84-tab a'))
3500 .click()
3501 driver.findElement(By.css('.phrase'))
3502 .sendKeys('abandon abandon ability');
3503 driver.sleep(generateDelay).then(function() {
3504 driver.findElement(By.css('.root-key'))
3505 .getAttribute("value")
3506 .then(function(path) {
3507 expect(path).toBe("vprv9DMUxX4ShgxML9N2YV5CvWEebWrM9aJ5ULpbRRyzyWu6vs4BzTvbfFFrH41N5hVi7MYSfiugd765L3JmAfDM5po36Y8ouCKRDeYQwByCmS7");
3508 done();
3509 })
3510 });
3511 });
3512
3513 it('Shows a warning if generating weak mnemonics', function(done) {
3514 driver.executeScript(function() {
3515 $(".strength option[selected]").removeAttr("selected");
3516 $(".strength option[value=6]").prop("selected", true);
3517 $(".strength").trigger("change");
3518 });
3519 driver.findElement(By.css(".generate-container .warning"))
3520 .getAttribute("class")
3521 .then(function(classes) {
3522 expect(classes).not.toContain("hidden");
3523 done();
3524 });
3525 });
3526
3527 it('Does not show a warning if generating strong mnemonics', function(done) {
3528 driver.executeScript(function() {
3529 $(".strength option[selected]").removeAttr("selected");
3530 $(".strength option[value=12]").prop("selected", true);
3531 });
3532 driver.findElement(By.css(".generate-container .warning"))
3533 .getAttribute("class")
3534 .then(function(classes) {
3535 expect(classes).toContain("hidden");
3536 done();
3537 });
3538 });
3539
3540 it('Shows a warning if overriding weak entropy with longer mnemonics', function(done) {
3541 driver.findElement(By.css('.use-entropy'))
3542 .click();
3543 driver.findElement(By.css('.entropy'))
3544 .sendKeys("0123456789abcdef"); // 6 words
3545 driver.executeScript(function() {
3546 $(".mnemonic-length").val("12").trigger("change");
3547 });
3548 driver.findElement(By.css(".weak-entropy-override-warning"))
3549 .getAttribute("class")
3550 .then(function(classes) {
3551 expect(classes).not.toContain("hidden");
3552 done();
3553 });
3554 });
3555
3556 it('Does not show a warning if entropy is stronger than mnemonic length', function(done) {
3557 driver.findElement(By.css('.use-entropy'))
3558 .click();
3559 driver.findElement(By.css('.entropy'))
3560 .sendKeys("0123456789abcdef0123456789abcdef0123456789abcdef"); // 18 words
3561 driver.executeScript(function() {
3562 $(".mnemonic-length").val("12").trigger("change");
3563 });
3564 driver.findElement(By.css(".weak-entropy-override-warning"))
3565 .getAttribute("class")
3566 .then(function(classes) {
3567 expect(classes).toContain("hidden");
3568 done();
3569 });
3570 });
3571
3572 });