]> git.immae.eu Git - perso/Immae/Projets/Cryptomonnaies/BIP39.git/blob - tests/spec/tests.js
Add test for Neblio coin
[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
638 // BIP39 seed is set from phrase
639 it('Sets the bip39 seed from the prhase', function(done) {
640 driver.findElement(By.css('.phrase'))
641 .sendKeys('abandon abandon ability');
642 driver.sleep(generateDelay).then(function() {
643 driver.findElement(By.css('.seed'))
644 .getAttribute("value")
645 .then(function(seed) {
646 expect(seed).toBe("20da140d3dd1df8713cefcc4d54ce0e445b4151027a1ab567b832f6da5fcc5afc1c3a3f199ab78b8e0ab4652efd7f414ac2c9a3b81bceb879a70f377aa0a58f3");
647 done();
648 })
649 });
650 });
651
652 // BIP32 root key is set from phrase
653 it('Sets the bip39 root key from the prhase', function(done) {
654 driver.findElement(By.css('.phrase'))
655 .sendKeys('abandon abandon ability');
656 driver.sleep(generateDelay).then(function() {
657 driver.findElement(By.css('.root-key'))
658 .getAttribute("value")
659 .then(function(seed) {
660 expect(seed).toBe("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
661 done();
662 })
663 });
664 });
665
666 // Tabs show correct addresses when changed
667 it('Shows the correct address when tab is changed', function(done) {
668 driver.findElement(By.css('.phrase'))
669 .sendKeys('abandon abandon ability');
670 driver.sleep(generateDelay).then(function() {
671 driver.findElement(By.css('#bip32-tab a'))
672 .click();
673 driver.sleep(generateDelay).then(function() {
674 getFirstAddress(function(address) {
675 expect(address).toBe("17uQ7s2izWPwBmEVFikTmZUjbBKWYdJchz");
676 done();
677 });
678 });
679 });
680 });
681
682 // BIP44 derivation path is shown
683 it('Shows the derivation path for bip44 tab', function(done) {
684 driver.findElement(By.css('.phrase'))
685 .sendKeys('abandon abandon ability');
686 driver.sleep(generateDelay).then(function() {
687 driver.findElement(By.css('#bip44 .path'))
688 .getAttribute("value")
689 .then(function(path) {
690 expect(path).toBe("m/44'/0'/0'/0");
691 done();
692 })
693 });
694 });
695
696 // BIP44 extended private key is shown
697 it('Shows the extended private key for bip44 tab', function(done) {
698 driver.findElement(By.css('.phrase'))
699 .sendKeys('abandon abandon ability');
700 driver.sleep(generateDelay).then(function() {
701 driver.findElement(By.css('.extended-priv-key'))
702 .getAttribute("value")
703 .then(function(path) {
704 expect(path).toBe("xprvA2DxxvPZcyRvYgZMGS53nadR32mVDeCyqQYyFhrCVbJNjPoxMeVf7QT5g7mQASbTf9Kp4cryvcXnu2qurjWKcrdsr91jXymdCDNxKgLFKJG");
705 done();
706 })
707 });
708 });
709
710 // BIP44 extended public key is shown
711 it('Shows the extended public key for bip44 tab', function(done) {
712 driver.findElement(By.css('.phrase'))
713 .sendKeys('abandon abandon ability');
714 driver.sleep(generateDelay).then(function() {
715 driver.findElement(By.css('.extended-pub-key'))
716 .getAttribute("value")
717 .then(function(path) {
718 expect(path).toBe("xpub6FDKNRvTTLzDmAdpNTc49ia9b4byd6vqCdUa46Fp3vqMcC96uBoufCmZXQLiN5AK3iSCJMhf9gT2sxkpyaPepRuA7W3MujV5tGmF5VfbueM");
719 done();
720 })
721 });
722 });
723
724 // BIP44 account field changes address list
725 it('Changes the address list if bip44 account is changed', function(done) {
726 driver.findElement(By.css('#bip44 .account'))
727 .sendKeys('1');
728 driver.findElement(By.css('.phrase'))
729 .sendKeys('abandon abandon ability');
730 driver.sleep(generateDelay).then(function() {
731 getFirstAddress(function(address) {
732 expect(address).toBe("1Nq2Wmu726XHCuGhctEtGmhxo3wzk5wZ1H");
733 done();
734 });
735 });
736 });
737
738 // BIP44 change field changes address list
739 it('Changes the address list if bip44 change is changed', function(done) {
740 driver.findElement(By.css('#bip44 .change'))
741 .sendKeys('1');
742 driver.findElement(By.css('.phrase'))
743 .sendKeys('abandon abandon ability');
744 driver.sleep(generateDelay).then(function() {
745 getFirstAddress(function(address) {
746 expect(address).toBe("1KAGfWgqfVbSSXY56fNQ7YnhyKuoskHtYo");
747 done();
748 });
749 });
750 });
751
752 // BIP32 derivation path can be set
753 it('Can use a custom bip32 derivation path', function(done) {
754 driver.findElement(By.css('#bip32-tab a'))
755 .click();
756 driver.findElement(By.css('#bip32 .path'))
757 .clear();
758 driver.findElement(By.css('#bip32 .path'))
759 .sendKeys('m/1');
760 driver.findElement(By.css('.phrase'))
761 .sendKeys('abandon abandon ability');
762 driver.sleep(generateDelay).then(function() {
763 getFirstAddress(function(address) {
764 expect(address).toBe("16pYQQdLD1hH4hwTGLXBaZ9Teboi1AGL8L");
765 done();
766 });
767 });
768 });
769
770 // BIP32 can use hardened derivation paths
771 it('Can use a hardened derivation paths', function(done) {
772 driver.findElement(By.css('#bip32-tab a'))
773 .click();
774 driver.findElement(By.css('#bip32 .path'))
775 .clear();
776 driver.findElement(By.css('#bip32 .path'))
777 .sendKeys("m/0'");
778 driver.findElement(By.css('.phrase'))
779 .sendKeys('abandon abandon ability');
780 driver.sleep(generateDelay).then(function() {
781 getFirstAddress(function(address) {
782 expect(address).toBe("14aXZeprXAE3UUKQc4ihvwBvww2LuEoHo4");
783 done();
784 });
785 });
786 });
787
788 // BIP32 extended private key is shown
789 it('Shows the BIP32 extended private key', function(done) {
790 driver.findElement(By.css('#bip32-tab a'))
791 .click();
792 driver.findElement(By.css('.phrase'))
793 .sendKeys('abandon abandon ability');
794 driver.sleep(generateDelay).then(function() {
795 driver.findElement(By.css('.extended-priv-key'))
796 .getAttribute("value")
797 .then(function(privKey) {
798 expect(privKey).toBe("xprv9va99uTVE5aLiutUVLTyfxfe8v8aaXjSQ1XxZbK6SezYVuikA9MnjQVTA8rQHpNA5LKvyQBpLiHbBQiiccKiBDs7eRmBogsvq3THFeLHYbe");
799 done();
800 });
801 });
802 });
803
804 // BIP32 extended public key is shown
805 it('Shows the BIP32 extended public key', function(done) {
806 driver.findElement(By.css('#bip32-tab a'))
807 .click();
808 driver.findElement(By.css('.phrase'))
809 .sendKeys('abandon abandon ability');
810 driver.sleep(generateDelay).then(function() {
811 driver.findElement(By.css('.extended-pub-key'))
812 .getAttribute("value")
813 .then(function(pubKey) {
814 expect(pubKey).toBe("xpub69ZVZQzP4T8dwPxwbMzz36cNgwy4yzTHmETZMyihzzXXNi3thgg3HCow1RtY252wdw5rS8369xKnraN5Q93y3FkFfJp2XEHWUrkyXsjS93P");
815 done();
816 });
817 });
818 });
819
820 // Derivation path is shown in table
821 it('Shows the derivation path in the table', function(done) {
822 driver.findElement(By.css('.phrase'))
823 .sendKeys('abandon abandon ability');
824 driver.sleep(generateDelay).then(function() {
825 getFirstPath(function(path) {
826 expect(path).toBe("m/44'/0'/0'/0/0");
827 done();
828 });
829 });
830 });
831
832 // Derivation path for address can be hardened
833 it('Can derive hardened addresses', function(done) {
834 driver.findElement(By.css('#bip32-tab a'))
835 .click();
836 driver.executeScript(function() {
837 $(".hardened-addresses").prop("checked", true);
838 });
839 driver.findElement(By.css('.phrase'))
840 .sendKeys('abandon abandon ability');
841 driver.sleep(generateDelay).then(function() {
842 getFirstAddress(function(address) {
843 expect(address).toBe("18exLzUv7kfpiXRzmCjFDoC9qwNLFyvwyd");
844 done();
845 });
846 });
847 });
848
849 // Derivation path visibility can be toggled
850 it('Can toggle visibility of the derivation path column', function(done) {
851 driver.findElement(By.css('.phrase'))
852 .sendKeys('abandon abandon ability');
853 driver.sleep(generateDelay).then(function() {
854 driver.findElement(By.css('.index-toggle'))
855 .click();
856 testColumnValuesAreInvisible(done, "index");
857 });
858 });
859
860 // Address is shown
861 it('Shows the address in the table', function(done) {
862 driver.findElement(By.css('.phrase'))
863 .sendKeys('abandon abandon ability');
864 driver.sleep(generateDelay).then(function() {
865 getFirstAddress(function(address) {
866 expect(address).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug");
867 done();
868 });
869 });
870 });
871
872 // Addresses are shown in order of derivation path
873 it('Shows the address in order of derivation path', function(done) {
874 driver.findElement(By.css('.phrase'))
875 .sendKeys('abandon abandon ability');
876 driver.sleep(generateDelay).then(function() {
877 testRowsAreInCorrectOrder(done);
878 });
879 });
880
881 // Address visibility can be toggled
882 it('Can toggle visibility of the address column', function(done) {
883 driver.findElement(By.css('.phrase'))
884 .sendKeys('abandon abandon ability');
885 driver.sleep(generateDelay).then(function() {
886 driver.findElement(By.css('.address-toggle'))
887 .click();
888 testColumnValuesAreInvisible(done, "address");
889 });
890 });
891
892 // Public key is shown in table
893 it('Shows the public key in the table', function(done) {
894 driver.findElement(By.css('.phrase'))
895 .sendKeys('abandon abandon ability');
896 driver.sleep(generateDelay).then(function() {
897 driver.findElements(By.css('.pubkey'))
898 .then(function(els) {
899 els[0].getText()
900 .then(function(pubkey) {
901 expect(pubkey).toBe("033f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3");
902 done();
903 });
904 });
905 });
906 });
907
908 // Public key visibility can be toggled
909 it('Can toggle visibility of the public key column', function(done) {
910 driver.findElement(By.css('.phrase'))
911 .sendKeys('abandon abandon ability');
912 driver.sleep(generateDelay).then(function() {
913 driver.findElement(By.css('.public-key-toggle'))
914 .click();
915 testColumnValuesAreInvisible(done, "pubkey");
916 });
917 });
918
919 // Private key is shown in table
920 it('Shows the private key in the table', function(done) {
921 driver.findElement(By.css('.phrase'))
922 .sendKeys('abandon abandon ability');
923 driver.sleep(generateDelay).then(function() {
924 driver.findElements(By.css('.privkey'))
925 .then(function(els) {
926 els[0].getText()
927 .then(function(pubkey) {
928 expect(pubkey).toBe("L26cVSpWFkJ6aQkPkKmTzLqTdLJ923e6CzrVh9cmx21QHsoUmrEE");
929 done();
930 });
931 });
932 });
933 });
934
935 // Private key visibility can be toggled
936 it('Can toggle visibility of the private key column', function(done) {
937 driver.findElement(By.css('.phrase'))
938 .sendKeys('abandon abandon ability');
939 driver.sleep(generateDelay).then(function() {
940 driver.findElement(By.css('.private-key-toggle'))
941 .click();
942 testColumnValuesAreInvisible(done, "privkey");
943 });
944 });
945
946 // More addresses can be generated
947 it('Can generate more rows in the table', function(done) {
948 driver.findElement(By.css('.phrase'))
949 .sendKeys('abandon abandon ability');
950 driver.sleep(generateDelay).then(function() {
951 driver.findElement(By.css('.more'))
952 .click();
953 driver.sleep(generateDelay).then(function() {
954 driver.findElements(By.css('.address'))
955 .then(function(els) {
956 expect(els.length).toBe(40);
957 done();
958 });
959 });
960 });
961 });
962
963 // A custom number of additional addresses can be generated
964 it('Can generate more rows in the table', function(done) {
965 driver.findElement(By.css('.phrase'))
966 .sendKeys('abandon abandon ability');
967 driver.sleep(generateDelay).then(function() {
968 driver.findElement(By.css('.rows-to-add'))
969 .clear();
970 driver.findElement(By.css('.rows-to-add'))
971 .sendKeys('1');
972 driver.findElement(By.css('.more'))
973 .click();
974 driver.sleep(generateDelay).then(function() {
975 driver.findElements(By.css('.address'))
976 .then(function(els) {
977 expect(els.length).toBe(21);
978 done();
979 });
980 });
981 });
982 });
983
984 // Additional addresses are shown in order of derivation path
985 it('Shows additional addresses in order of derivation path', function(done) {
986 driver.findElement(By.css('.phrase'))
987 .sendKeys('abandon abandon ability');
988 driver.sleep(generateDelay).then(function() {
989 driver.findElement(By.css('.more'))
990 .click();
991 driver.sleep(generateDelay).then(function() {
992 testRowsAreInCorrectOrder(done);
993 });
994 });
995 });
996
997 // BIP32 root key can be set by the user
998 it('Allows the user to set the BIP32 root key', function(done) {
999 driver.findElement(By.css('.root-key'))
1000 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1001 driver.sleep(generateDelay).then(function() {
1002 getFirstAddress(function(address) {
1003 expect(address).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug");
1004 done();
1005 });
1006 });
1007 });
1008
1009 // Setting BIP32 root key clears the existing phrase, passphrase and seed
1010 // TODO this doesn't work in selenium with chrome
1011 it('Confirms the existing phrase should be cleared', function(done) {
1012 if (browser == "chrome") {
1013 pending("Selenium + Chrome headless bug for alert, see https://stackoverflow.com/q/45242264");
1014 }
1015 driver.findElement(By.css('.phrase'))
1016 .sendKeys('A non-blank but invalid value');
1017 driver.findElement(By.css('.root-key'))
1018 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1019 driver.switchTo().alert().accept();
1020 driver.findElement(By.css('.phrase'))
1021 .getAttribute("value").then(function(value) {
1022 expect(value).toBe("");
1023 done();
1024 });
1025 });
1026
1027 // Clearing of phrase, passphrase and seed can be cancelled by user
1028 // TODO this doesn't work in selenium with chrome
1029 it('Allows the clearing of the phrase to be cancelled', function(done) {
1030 if (browser == "chrome") {
1031 pending("Selenium + Chrome headless bug for alert, see https://stackoverflow.com/q/45242264");
1032 }
1033 driver.findElement(By.css('.phrase'))
1034 .sendKeys('abandon abandon ability');
1035 driver.sleep(generateDelay).then(function() {
1036 driver.findElement(By.css('.root-key'))
1037 .clear();
1038 driver.findElement(By.css('.root-key'))
1039 .sendKeys('x');
1040 driver.switchTo().alert().dismiss();
1041 driver.findElement(By.css('.phrase'))
1042 .getAttribute("value").then(function(value) {
1043 expect(value).toBe("abandon abandon ability");
1044 done();
1045 });
1046 });
1047 });
1048
1049 // Custom BIP32 root key is used when changing the derivation path
1050 it('Can set derivation path for root key instead of phrase', function(done) {
1051 driver.findElement(By.css('#bip44 .account'))
1052 .sendKeys('1');
1053 driver.findElement(By.css('.root-key'))
1054 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi');
1055 driver.sleep(generateDelay).then(function() {
1056 getFirstAddress(function(address) {
1057 expect(address).toBe("1Nq2Wmu726XHCuGhctEtGmhxo3wzk5wZ1H");
1058 done();
1059 });
1060 });
1061 });
1062
1063 // Incorrect mnemonic shows error
1064 it('Shows an error for incorrect mnemonic', function(done) {
1065 driver.findElement(By.css('.phrase'))
1066 .sendKeys('abandon abandon abandon');
1067 driver.sleep(feedbackDelay).then(function() {
1068 driver.findElement(By.css('.feedback'))
1069 .getText()
1070 .then(function(feedback) {
1071 expect(feedback).toBe("Invalid mnemonic");
1072 done();
1073 });
1074 });
1075 });
1076
1077 // Incorrect word shows suggested replacement
1078 it('Shows word suggestion for incorrect word', function(done) {
1079 driver.findElement(By.css('.phrase'))
1080 .sendKeys('abandon abandon abiliti');
1081 driver.sleep(feedbackDelay).then(function() {
1082 driver.findElement(By.css('.feedback'))
1083 .getText()
1084 .then(function(feedback) {
1085 var msg = "abiliti not in wordlist, did you mean ability?";
1086 expect(feedback).toBe(msg);
1087 done();
1088 });
1089 });
1090 });
1091
1092 // Github pull request 48
1093 // First four letters of word shows that word, not closest
1094 // since first four letters gives unique word in BIP39 wordlist
1095 // eg ille should show illegal, not idle
1096 it('Shows word suggestion based on first four chars', function(done) {
1097 driver.findElement(By.css('.phrase'))
1098 .sendKeys('ille');
1099 driver.sleep(feedbackDelay).then(function() {
1100 driver.findElement(By.css('.feedback'))
1101 .getText()
1102 .then(function(feedback) {
1103 var msg = "ille not in wordlist, did you mean illegal?";
1104 expect(feedback).toBe(msg);
1105 done();
1106 });
1107 });
1108 });
1109
1110 // Incorrect BIP32 root key shows error
1111 it('Shows error for incorrect root key', function(done) {
1112 driver.findElement(By.css('.root-key'))
1113 .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpj');
1114 driver.sleep(feedbackDelay).then(function() {
1115 driver.findElement(By.css('.feedback'))
1116 .getText()
1117 .then(function(feedback) {
1118 var msg = "Invalid root key";
1119 expect(feedback).toBe(msg);
1120 done();
1121 });
1122 });
1123 });
1124
1125 // Derivation path not starting with m shows error
1126 it('Shows error for derivation path not starting with m', function(done) {
1127 driver.findElement(By.css('#bip32-tab a'))
1128 .click();
1129 driver.findElement(By.css('#bip32 .path'))
1130 .clear();
1131 driver.findElement(By.css('#bip32 .path'))
1132 .sendKeys('n/0');
1133 driver.findElement(By.css('.phrase'))
1134 .sendKeys('abandon abandon ability');
1135 driver.sleep(feedbackDelay).then(function() {
1136 driver.findElement(By.css('.feedback'))
1137 .getText()
1138 .then(function(feedback) {
1139 var msg = "First character must be 'm'";
1140 expect(feedback).toBe(msg);
1141 done();
1142 });
1143 });
1144 });
1145
1146 // Derivation path containing invalid characters shows useful error
1147 it('Shows error for derivation path not starting with m', function(done) {
1148 driver.findElement(By.css('#bip32-tab a'))
1149 .click();
1150 driver.findElement(By.css('#bip32 .path'))
1151 .clear();
1152 driver.findElement(By.css('#bip32 .path'))
1153 .sendKeys('m/1/0wrong1/1');
1154 driver.findElement(By.css('.phrase'))
1155 .sendKeys('abandon abandon ability');
1156 driver.sleep(feedbackDelay).then(function() {
1157 driver.findElement(By.css('.feedback'))
1158 .getText()
1159 .then(function(feedback) {
1160 var msg = "Invalid characters 0wrong1 found at depth 2";
1161 expect(feedback).toBe(msg);
1162 done();
1163 });
1164 });
1165 });
1166
1167 // Github Issue 11: Default word length is 15
1168 // https://github.com/iancoleman/bip39/issues/11
1169 it('Sets the default word length to 15', function(done) {
1170 driver.findElement(By.css('.strength'))
1171 .getAttribute("value")
1172 .then(function(strength) {
1173 expect(strength).toBe("15");
1174 done();
1175 });
1176 });
1177
1178 // Github Issue 12: Generate more rows with private keys hidden
1179 // https://github.com/iancoleman/bip39/issues/12
1180 it('Sets the correct hidden column state on new rows', function(done) {
1181 driver.findElement(By.css('.phrase'))
1182 .sendKeys("abandon abandon ability");
1183 driver.sleep(generateDelay).then(function() {
1184 driver.findElement(By.css('.private-key-toggle'))
1185 .click();
1186 driver.findElement(By.css('.more'))
1187 .click();
1188 driver.sleep(generateDelay).then(function() {
1189 driver.findElements(By.css('.privkey'))
1190 .then(function(els) {
1191 expect(els.length).toBe(40);
1192 });
1193 testColumnValuesAreInvisible(done, "privkey");
1194 });
1195 });
1196 });
1197
1198 // Github Issue 19: Mnemonic is not sensitive to whitespace
1199 // https://github.com/iancoleman/bip39/issues/19
1200 it('Ignores excess whitespace in the mnemonic', function(done) {
1201 var doublespace = " ";
1202 var mnemonic = "urge cat" + doublespace + "bid";
1203 driver.findElement(By.css('.phrase'))
1204 .sendKeys(mnemonic);
1205 driver.sleep(generateDelay).then(function() {
1206 driver.findElement(By.css('.root-key'))
1207 .getAttribute("value")
1208 .then(function(seed) {
1209 expect(seed).toBe("xprv9s21ZrQH143K3isaZsWbKVoTtbvd34Y1ZGRugGdMeBGbM3AgBVzTH159mj1cbbtYSJtQr65w6L5xy5L9SFC7c9VJZWHxgAzpj4mun5LhrbC");
1210 done();
1211 });
1212 });
1213 });
1214
1215 // Github Issue 23: Part 1: Use correct derivation path when changing tabs
1216 // https://github.com/iancoleman/bip39/issues/23
1217 it('Uses the correct derivation path when changing tabs', function(done) {
1218 // 1) and 2) set the phrase
1219 driver.findElement(By.css('.phrase'))
1220 .sendKeys("abandon abandon ability");
1221 driver.sleep(generateDelay).then(function() {
1222 // 3) select bip32 tab
1223 driver.findElement(By.css('#bip32-tab a'))
1224 .click();
1225 driver.sleep(generateDelay).then(function() {
1226 // 4) switch from bitcoin to litecoin
1227 selectNetwork("LTC - Litecoin");
1228 driver.sleep(generateDelay).then(function() {
1229 // 5) Check address is displayed correctly
1230 getFirstAddress(function(address) {
1231 expect(address).toBe("LS8MP5LZ5AdzSZveRrjm3aYVoPgnfFh5T5");
1232 // 5) Check derivation path is displayed correctly
1233 getFirstPath(function(path) {
1234 expect(path).toBe("m/0/0");
1235 done();
1236 });
1237 });
1238 });
1239 });
1240 });
1241 });
1242
1243 // Github Issue 23 Part 2: Coin selection in derivation path
1244 // https://github.com/iancoleman/bip39/issues/23#issuecomment-238011920
1245 it('Uses the correct derivation path when changing coins', function(done) {
1246 // set the phrase
1247 driver.findElement(By.css('.phrase'))
1248 .sendKeys("abandon abandon ability");
1249 driver.sleep(generateDelay).then(function() {
1250 // switch from bitcoin to clam
1251 selectNetwork("CLAM - Clams");
1252 driver.sleep(generateDelay).then(function() {
1253 // check derivation path is displayed correctly
1254 getFirstPath(function(path) {
1255 expect(path).toBe("m/44'/23'/0'/0/0");
1256 done();
1257 });
1258 });
1259 });
1260 });
1261
1262 // Github Issue 26: When using a Root key derrived altcoins are incorrect
1263 // https://github.com/iancoleman/bip39/issues/26
1264 it('Uses the correct derivation for altcoins with root keys', function(done) {
1265 // 1) 2) and 3) set the root key
1266 driver.findElement(By.css('.root-key'))
1267 .sendKeys("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
1268 driver.sleep(generateDelay).then(function() {
1269 // 4) switch from bitcoin to viacoin
1270 selectNetwork("VIA - Viacoin");
1271 driver.sleep(generateDelay).then(function() {
1272 // 5) ensure the derived address is correct
1273 getFirstAddress(function(address) {
1274 expect(address).toBe("Vq9Eq4N5SQnjqZvxtxzo7hZPW5XnyJsmXT");
1275 done();
1276 });
1277 });
1278 });
1279 });
1280
1281 // Selecting a language with no existing phrase should generate a phrase in
1282 // that language.
1283 it('Generate a random phrase when language is selected and no current phrase', function(done) {
1284 driver.findElement(By.css("a[href='#japanese']"))
1285 .click();
1286 driver.sleep(generateDelay).then(function() {
1287 driver.findElement(By.css(".phrase"))
1288 .getAttribute("value").then(function(phrase) {
1289 expect(phrase.search(/[a-z]/)).toBe(-1);
1290 expect(phrase.length).toBeGreaterThan(0);
1291 done();
1292 });
1293 });
1294 });
1295
1296 // Selecting a language with existing phrase should update the phrase to use
1297 // that language.
1298 it('Updates existing phrases when the language is changed', function(done) {
1299 driver.findElement(By.css(".phrase"))
1300 .sendKeys("abandon abandon ability");
1301 driver.sleep(generateDelay).then(function() {
1302 driver.findElement(By.css("a[href='#italian']"))
1303 .click();
1304 driver.sleep(generateDelay).then(function() {
1305 driver.findElement(By.css(".phrase"))
1306 .getAttribute("value").then(function(phrase) {
1307 // Check only the language changes, not the phrase
1308 expect(phrase).toBe("abaco abaco abbaglio");
1309 getFirstAddress(function(address) {
1310 // Check the address is correct
1311 expect(address).toBe("1Dz5TgDhdki9spa6xbPFbBqv5sjMrx3xgV");
1312 done();
1313 });
1314 });
1315 });
1316 });
1317 });
1318
1319 // Suggested replacement for erroneous word in non-English language
1320 it('Shows word suggestion for incorrect word in non-English language', function(done) {
1321 driver.findElement(By.css('.phrase'))
1322 .sendKeys('abaco abaco zbbaglio');
1323 driver.sleep(feedbackDelay).then(function() {
1324 driver.findElement(By.css('.feedback'))
1325 .getText()
1326 .then(function(feedback) {
1327 var msg = "zbbaglio not in wordlist, did you mean abbaglio?";
1328 expect(feedback).toBe(msg);
1329 done();
1330 });
1331 });
1332 });
1333
1334 // Japanese word does not break across lines.
1335 // Point 2 from
1336 // https://github.com/bitcoin/bips/blob/master/bip-0039/bip-0039-wordlists.md#japanese
1337 it('Does not break Japanese words across lines', function(done) {
1338 driver.findElement(By.css('.phrase'))
1339 .getCssValue("word-break")
1340 .then(function(value) {
1341 expect(value).toBe("keep-all");
1342 done();
1343 });
1344 });
1345
1346 // Language can be specified at page load using hash value in url
1347 it('Can set the language from the url hash', function(done) {
1348 driver.get(url + "#japanese").then(function() {
1349 driver.findElement(By.css('.generate')).click();
1350 driver.sleep(generateDelay).then(function() {
1351 driver.findElement(By.css(".phrase"))
1352 .getAttribute("value").then(function(phrase) {
1353 expect(phrase.search(/[a-z]/)).toBe(-1);
1354 expect(phrase.length).toBeGreaterThan(0);
1355 done();
1356 });
1357 });
1358 });
1359 });
1360
1361 // Entropy can be entered by the user
1362 it('Allows entropy to be entered', function(done) {
1363 driver.findElement(By.css('.use-entropy'))
1364 .click();
1365 driver.findElement(By.css('.entropy'))
1366 .sendKeys('00000000 00000000 00000000 00000000');
1367 driver.sleep(generateDelay).then(function() {
1368 driver.findElement(By.css(".phrase"))
1369 .getAttribute("value").then(function(phrase) {
1370 expect(phrase).toBe("abandon abandon ability");
1371 getFirstAddress(function(address) {
1372 expect(address).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug");
1373 done();
1374 })
1375 });
1376 });
1377 });
1378
1379 // A warning about entropy is shown to the user, with additional information
1380 it('Shows a warning about using entropy', function(done) {
1381 driver.findElement(By.css('.use-entropy'))
1382 .click();
1383 driver.findElement(By.css('.entropy-container'))
1384 .getText()
1385 .then(function(containerText) {
1386 var warning = "mnemonic may be insecure";
1387 expect(containerText).toContain(warning);
1388 driver.findElement(By.css('#entropy-notes'))
1389 .findElement(By.xpath("parent::*"))
1390 .getText()
1391 .then(function(notesText) {
1392 var detail = "flipping a fair coin, rolling a fair dice, noise measurements etc";
1393 expect(notesText).toContain(detail);
1394 done();
1395 });
1396 });
1397 });
1398
1399 // The types of entropy available are described to the user
1400 it('Shows the types of entropy available', function(done) {
1401 driver.findElement(By.css('.entropy'))
1402 .getAttribute("placeholder")
1403 .then(function(placeholderText) {
1404 var options = [
1405 "binary",
1406 "base 6",
1407 "dice",
1408 "base 10",
1409 "hexadecimal",
1410 "cards",
1411 ];
1412 for (var i=0; i<options.length; i++) {
1413 var option = options[i];
1414 expect(placeholderText).toContain(option);
1415 }
1416 done();
1417 });
1418 });
1419
1420 // The actual entropy used is shown to the user
1421 it('Shows the actual entropy used', function(done) {
1422 driver.findElement(By.css('.use-entropy'))
1423 .click();
1424 driver.findElement(By.css('.entropy'))
1425 .sendKeys('Not A Very Good Entropy Source At All');
1426 driver.sleep(generateDelay).then(function() {
1427 driver.findElement(By.css('.entropy-container'))
1428 .getText()
1429 .then(function(text) {
1430 expect(text).toMatch(/Filtered Entropy\s+AedEceAA/);
1431 done();
1432 });
1433 });
1434 });
1435
1436 // Binary entropy can be entered
1437 it('Allows binary entropy to be entered', function(done) {
1438 testEntropyType(done, "01", "binary");
1439 });
1440
1441 // Base 6 entropy can be entered
1442 it('Allows base 6 entropy to be entered', function(done) {
1443 testEntropyType(done, "012345", "base 6");
1444 });
1445
1446 // Base 6 dice entropy can be entered
1447 it('Allows base 6 dice entropy to be entered', function(done) {
1448 testEntropyType(done, "123456", "base 6 (dice)");
1449 });
1450
1451 // Base 10 entropy can be entered
1452 it('Allows base 10 entropy to be entered', function(done) {
1453 testEntropyType(done, "789", "base 10");
1454 });
1455
1456 // Hexadecimal entropy can be entered
1457 it('Allows hexadecimal entropy to be entered', function(done) {
1458 testEntropyType(done, "abcdef", "hexadecimal");
1459 });
1460
1461 // Dice entropy value is shown as the converted base 6 value
1462 // ie 123456 is converted to 123450
1463 it('Shows dice entropy as base 6', function(done) {
1464 driver.findElement(By.css('.use-entropy'))
1465 .click();
1466 driver.findElement(By.css('.entropy'))
1467 .sendKeys("123456");
1468 driver.sleep(generateDelay).then(function() {
1469 driver.findElement(By.css('.entropy-container'))
1470 .getText()
1471 .then(function(text) {
1472 expect(text).toMatch(/Filtered Entropy\s+123450/);
1473 done();
1474 });
1475 });
1476 });
1477
1478 // The number of bits of entropy accumulated is shown
1479 it("Shows the number of bits of entropy for 20 bits of binary", function(done) {
1480 testEntropyBits(done, "0000 0000 0000 0000 0000", "20");
1481 });
1482 it("Shows the number of bits of entropy for 1 bit of binary", function(done) {
1483 testEntropyBits(done, "0", "1");
1484 });
1485 it("Shows the number of bits of entropy for 4 bits of binary", function(done) {
1486 testEntropyBits(done, "0000", "4");
1487 });
1488 it("Shows the number of bits of entropy for 1 character of base 6 (dice)", function(done) {
1489 // 6 in card is 0 in base 6, 0 in base 6 is 2.6 bits (rounded down to 2 bits)
1490 testEntropyBits(done, "6", "2");
1491 });
1492 it("Shows the number of bits of entropy for 1 character of base 10 with 3 bits", function(done) {
1493 // 7 in base 10 is 111 in base 2, no leading zeros
1494 testEntropyBits(done, "7", "3");
1495 });
1496 it("Shows the number of bits of entropy for 1 character of base 10 with 4 bis", function(done) {
1497 testEntropyBits(done, "8", "4");
1498 });
1499 it("Shows the number of bits of entropy for 1 character of hex", function(done) {
1500 testEntropyBits(done, "F", "4");
1501 });
1502 it("Shows the number of bits of entropy for 2 characters of base 10", function(done) {
1503 testEntropyBits(done, "29", "6");
1504 });
1505 it("Shows the number of bits of entropy for 2 characters of hex", function(done) {
1506 testEntropyBits(done, "0A", "8");
1507 });
1508 it("Shows the number of bits of entropy for 2 characters of hex with 3 leading zeros", function(done) {
1509 // hex is always multiple of 4 bits of entropy
1510 testEntropyBits(done, "1A", "8");
1511 });
1512 it("Shows the number of bits of entropy for 2 characters of hex with 2 leading zeros", function(done) {
1513 testEntropyBits(done, "2A", "8");
1514 });
1515 it("Shows the number of bits of entropy for 2 characters of hex with 1 leading zero", function(done) {
1516 testEntropyBits(done, "4A", "8");
1517 });
1518 it("Shows the number of bits of entropy for 2 characters of hex with no leading zeros", function(done) {
1519 testEntropyBits(done, "8A", "8");
1520 });
1521 it("Shows the number of bits of entropy for 2 characters of hex starting with F", function(done) {
1522 testEntropyBits(done, "FA", "8");
1523 });
1524 it("Shows the number of bits of entropy for 4 characters of hex with leading zeros", function(done) {
1525 testEntropyBits(done, "000A", "16");
1526 });
1527 it("Shows the number of bits of entropy for 4 characters of base 6", function(done) {
1528 testEntropyBits(done, "5555", "11");
1529 });
1530 it("Shows the number of bits of entropy for 4 characters of base 6 dice", function(done) {
1531 // uses dice, so entropy is actually 0000 in base 6, which is 4 lots of
1532 // 2.58 bits, which is 10.32 bits (rounded down to 10 bits)
1533 testEntropyBits(done, "6666", "10");
1534 });
1535 it("Shows the number of bits of entropy for 4 charactes of base 10", function(done) {
1536 // Uses base 10, which is 4 lots of 3.32 bits, which is 13.3 bits (rounded
1537 // down to 13)
1538 testEntropyBits(done, "2227", "13");
1539 });
1540 it("Shows the number of bits of entropy for 4 characters of hex with 2 leading zeros", function(done) {
1541 testEntropyBits(done, "222F", "16");
1542 });
1543 it("Shows the number of bits of entropy for 4 characters of hex starting with F", function(done) {
1544 testEntropyBits(done, "FFFF", "16");
1545 });
1546 it("Shows the number of bits of entropy for 10 characters of base 10", function(done) {
1547 // 10 events at 3.32 bits per event
1548 testEntropyBits(done, "0000101017", "33");
1549 });
1550 it("Shows the number of bits of entropy for a full deck of cards", function(done) {
1551 // cards are not replaced, so a full deck is not 52^52 entropy which is 296
1552 // bits, it's 52!, which is 225 bits
1553 testEntropyBits(done, "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks", "225");
1554 });
1555
1556 it("Shows details about the entered entropy", function(done) {
1557 testEntropyFeedback(done,
1558 {
1559 entropy: "A",
1560 filtered: "A",
1561 type: "hexadecimal",
1562 events: "1",
1563 bits: "4",
1564 words: 0,
1565 strength: "less than a second",
1566 }
1567 );
1568 });
1569 it("Shows details about the entered entropy", function(done) {
1570 testEntropyFeedback(done,
1571 {
1572 entropy: "AAAAAAAA",
1573 filtered: "AAAAAAAA",
1574 type: "hexadecimal",
1575 events: "8",
1576 bits: "32",
1577 words: 3,
1578 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
1579 }
1580 );
1581 });
1582 it("Shows details about the entered entropy", function(done) {
1583 testEntropyFeedback(done,
1584 {
1585 entropy: "AAAAAAAA B",
1586 filtered: "AAAAAAAAB",
1587 type: "hexadecimal",
1588 events: "9",
1589 bits: "36",
1590 words: 3,
1591 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
1592 }
1593 );
1594 });
1595 it("Shows details about the entered entropy", function(done) {
1596 testEntropyFeedback(done,
1597 {
1598 entropy: "AAAAAAAA BBBBBBBB",
1599 filtered: "AAAAAAAABBBBBBBB",
1600 type: "hexadecimal",
1601 events: "16",
1602 bits: "64",
1603 words: 6,
1604 strength: "less than a second - Repeats like \"aaa\" are easy to guess",
1605 }
1606 );
1607 });
1608 it("Shows details about the entered entropy", function(done) {
1609 testEntropyFeedback(done,
1610 {
1611 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC",
1612 filtered: "AAAAAAAABBBBBBBBCCCCCCCC",
1613 type: "hexadecimal",
1614 events: "24",
1615 bits: "96",
1616 words: 9,
1617 strength: "less than a second",
1618 }
1619 );
1620 });
1621 it("Shows details about the entered entropy", function(done) {
1622 testEntropyFeedback(done,
1623 {
1624 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDD",
1625 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDD",
1626 type: "hexadecimal",
1627 events: "32",
1628 bits: "128",
1629 words: 12,
1630 strength: "2 minutes",
1631 }
1632 );
1633 });
1634 it("Shows details about the entered entropy", function(done) {
1635 testEntropyFeedback(done,
1636 {
1637 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA",
1638 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDA",
1639 type: "hexadecimal",
1640 events: "32",
1641 bits: "128",
1642 words: 12,
1643 strength: "2 days",
1644 }
1645 );
1646 });
1647 it("Shows details about the entered entropy", function(done) {
1648 testEntropyFeedback(done,
1649 {
1650 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA EEEEEEEE",
1651 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDAEEEEEEEE",
1652 type: "hexadecimal",
1653 events: "40",
1654 bits: "160",
1655 words: 15,
1656 strength: "3 years",
1657 }
1658 );
1659 });
1660 it("Shows details about the entered entropy", function(done) {
1661 testEntropyFeedback(done,
1662 {
1663 entropy: "AAAAAAAA BBBBBBBB CCCCCCCC DDDDDDDA EEEEEEEE FFFFFFFF",
1664 filtered: "AAAAAAAABBBBBBBBCCCCCCCCDDDDDDDAEEEEEEEEFFFFFFFF",
1665 type: "hexadecimal",
1666 events: "48",
1667 bits: "192",
1668 words: 18,
1669 strength: "centuries",
1670 }
1671 );
1672 });
1673 it("Shows details about the entered entropy", function(done) {
1674 testEntropyFeedback(done,
1675 {
1676 entropy: "7d",
1677 type: "card",
1678 events: "1",
1679 bits: "4",
1680 words: 0,
1681 strength: "less than a second",
1682 }
1683 );
1684 });
1685 it("Shows details about the entered entropy", function(done) {
1686 testEntropyFeedback(done,
1687 {
1688 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
1689 type: "card (full deck)",
1690 events: "52",
1691 bits: "225",
1692 words: 21,
1693 strength: "centuries",
1694 }
1695 );
1696 });
1697 it("Shows details about the entered entropy", function(done) {
1698 testEntropyFeedback(done,
1699 {
1700 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks3d",
1701 type: "card (full deck, 1 duplicate: 3d)",
1702 events: "53",
1703 bits: "254",
1704 words: 21,
1705 strength: "centuries",
1706 }
1707 );
1708 });
1709 it("Shows details about the entered entropy", function(done) {
1710 testEntropyFeedback(done,
1711 {
1712 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqs3d4d",
1713 type: "card (2 duplicates: 3d 4d, 1 missing: KS)",
1714 events: "53",
1715 bits: "254",
1716 words: 21,
1717 strength: "centuries",
1718 }
1719 );
1720 });
1721 it("Shows details about the entered entropy", function(done) {
1722 testEntropyFeedback(done,
1723 {
1724 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqs3d4d5d6d",
1725 type: "card (4 duplicates: 3d 4d 5d..., 1 missing: KS)",
1726 events: "55",
1727 bits: "264",
1728 words: 24,
1729 strength: "centuries",
1730 }
1731 );
1732 });
1733 it("Shows details about the entered entropy", function(done) {
1734 testEntropyFeedback(done,
1735 // Next test was throwing uncaught error in zxcvbn
1736 // Also tests 451 bits, ie Math.log2(52!)*2 = 225.58 * 2
1737 {
1738 entropy: "ac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsksac2c3c4c5c6c7c8c9ctcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
1739 type: "card (full deck, 52 duplicates: ac 2c 3c...)",
1740 events: "104",
1741 bits: "499",
1742 words: 45,
1743 strength: "centuries",
1744 }
1745 );
1746 });
1747 it("Shows details about the entered entropy", function(done) {
1748 testEntropyFeedback(done,
1749 // Case insensitivity to duplicate cards
1750 {
1751 entropy: "asAS",
1752 type: "card (1 duplicate: AS)",
1753 events: "2",
1754 bits: "9",
1755 words: 0,
1756 strength: "less than a second",
1757 }
1758 );
1759 });
1760 it("Shows details about the entered entropy", function(done) {
1761 testEntropyFeedback(done,
1762 {
1763 entropy: "ASas",
1764 type: "card (1 duplicate: as)",
1765 events: "2",
1766 bits: "9",
1767 words: 0,
1768 strength: "less than a second",
1769 }
1770 );
1771 });
1772 it("Shows details about the entered entropy", function(done) {
1773 testEntropyFeedback(done,
1774 // Missing cards are detected
1775 {
1776 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d5d6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
1777 type: "card (1 missing: 9C)",
1778 events: "51",
1779 bits: "221",
1780 words: 18,
1781 strength: "centuries",
1782 }
1783 );
1784 });
1785 it("Shows details about the entered entropy", function(done) {
1786 testEntropyFeedback(done,
1787 {
1788 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d7d8d9dtdjdqdkdah2h3h4h5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
1789 type: "card (2 missing: 9C 5D)",
1790 events: "50",
1791 bits: "216",
1792 words: 18,
1793 strength: "centuries",
1794 }
1795 );
1796 });
1797 it("Shows details about the entered entropy", function(done) {
1798 testEntropyFeedback(done,
1799 {
1800 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d7d8d9dtdjd kdah2h3h 5h6h7h8h9hthjhqhkhas2s3s4s5s6s7s8s9stsjsqsks",
1801 type: "card (4 missing: 9C 5D QD...)",
1802 events: "48",
1803 bits: "208",
1804 words: 18,
1805 strength: "centuries",
1806 }
1807 );
1808 });
1809 it("Shows details about the entered entropy", function(done) {
1810 testEntropyFeedback(done,
1811 // More than six missing cards does not show message
1812 {
1813 entropy: "ac2c3c4c5c6c7c8c tcjcqckcad2d3d4d 6d 8d9d jd kdah2h3h 5h6h7h8h9hthjhqhkh 2s3s4s5s6s7s8s9stsjsqsks",
1814 type: "card",
1815 events: "45",
1816 bits: "195",
1817 words: 18,
1818 strength: "centuries",
1819 }
1820 );
1821 });
1822 it("Shows details about the entered entropy", function(done) {
1823 testEntropyFeedback(done,
1824 // Multiple decks of cards increases bits per event
1825 {
1826 entropy: "3d",
1827 events: "1",
1828 bits: "4",
1829 bitsPerEvent: "4.34",
1830 }
1831 );
1832 });
1833 it("Shows details about the entered entropy", function(done) {
1834 testEntropyFeedback(done,
1835 {
1836 entropy: "3d3d",
1837 events: "2",
1838 bits: "9",
1839 bitsPerEvent: "4.80",
1840 }
1841 );
1842 });
1843 it("Shows details about the entered entropy", function(done) {
1844 testEntropyFeedback(done,
1845 {
1846 entropy: "3d3d3d",
1847 events: "3",
1848 bits: "15",
1849 bitsPerEvent: "5.01",
1850 }
1851 );
1852 });
1853 it("Shows details about the entered entropy", function(done) {
1854 testEntropyFeedback(done,
1855 {
1856 entropy: "3d3d3d3d",
1857 events: "4",
1858 bits: "20",
1859 bitsPerEvent: "5.14",
1860 }
1861 );
1862 });
1863 it("Shows details about the entered entropy", function(done) {
1864 testEntropyFeedback(done,
1865 {
1866 entropy: "3d3d3d3d3d",
1867 events: "5",
1868 bits: "26",
1869 bitsPerEvent: "5.22",
1870 }
1871 );
1872 });
1873 it("Shows details about the entered entropy", function(done) {
1874 testEntropyFeedback(done,
1875 {
1876 entropy: "3d3d3d3d3d3d",
1877 events: "6",
1878 bits: "31",
1879 bitsPerEvent: "5.28",
1880 }
1881 );
1882 });
1883 it("Shows details about the entered entropy", function(done) {
1884 testEntropyFeedback(done,
1885 {
1886 entropy: "3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d",
1887 events: "33",
1888 bits: "184",
1889 bitsPerEvent: "5.59",
1890 strength: 'less than a second - Repeats like "abcabcabc" are only slightly harder to guess than "abc"',
1891 }
1892 );
1893 });
1894
1895 // Entropy is truncated from the left
1896 it('Truncates entropy from the left', function(done) {
1897 // Truncate from left means 0000 is removed from the start
1898 // which gives mnemonic 'avocado zoo zone'
1899 // not 1111 removed from the end
1900 // which gives the mnemonic 'abstract zoo zoo'
1901 var entropy = "00000000 00000000 00000000 00000000";
1902 entropy += "11111111 11111111 11111111 1111"; // Missing last byte
1903 driver.findElement(By.css('.use-entropy'))
1904 .click();
1905 driver.findElement(By.css('.entropy'))
1906 .sendKeys(entropy);
1907 driver.sleep(generateDelay).then(function() {
1908 driver.findElement(By.css(".phrase"))
1909 .getAttribute("value").then(function(phrase) {
1910 expect(phrase).toBe("avocado zoo zone");
1911 done();
1912 });
1913 });
1914 });
1915
1916 // Very large entropy results in very long mnemonics
1917 it('Converts very long entropy to very long mnemonics', function(done) {
1918 var entropy = "";
1919 for (var i=0; i<33; i++) {
1920 entropy += "AAAAAAAA"; // 3 words * 33 iterations = 99 words
1921 }
1922 driver.findElement(By.css('.use-entropy'))
1923 .click();
1924 driver.findElement(By.css('.entropy'))
1925 .sendKeys(entropy);
1926 driver.sleep(generateDelay).then(function() {
1927 driver.findElement(By.css(".phrase"))
1928 .getAttribute("value").then(function(phrase) {
1929 var wordCount = phrase.split(/\s+/g).length;
1930 expect(wordCount).toBe(99);
1931 done();
1932 });
1933 });
1934 });
1935
1936 // Is compatible with bip32jp entropy
1937 // https://bip32jp.github.io/english/index.html
1938 // NOTES:
1939 // Is incompatible with:
1940 // base 20
1941 it('Is compatible with bip32jp.github.io', function(done) {
1942 var entropy = "543210543210543210543210543210543210543210543210543210543210543210543210543210543210543210543210543";
1943 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";
1944 driver.findElement(By.css('.use-entropy'))
1945 .click();
1946 driver.findElement(By.css('.entropy'))
1947 .sendKeys(entropy);
1948 driver.sleep(generateDelay).then(function() {
1949 driver.findElement(By.css(".phrase"))
1950 .getAttribute("value").then(function(phrase) {
1951 expect(phrase).toBe(expectedPhrase);
1952 done();
1953 });
1954 });
1955 });
1956
1957 // Blank entropy does not generate mnemonic or addresses
1958 it('Does not generate mnemonic for blank entropy', function(done) {
1959 driver.findElement(By.css('.use-entropy'))
1960 .click();
1961 driver.findElement(By.css('.entropy'))
1962 .clear();
1963 // check there is no mnemonic
1964 driver.sleep(generateDelay).then(function() {
1965 driver.findElement(By.css(".phrase"))
1966 .getAttribute("value").then(function(phrase) {
1967 expect(phrase).toBe("");
1968 // check there is no mnemonic
1969 driver.findElements(By.css(".address"))
1970 .then(function(addresses) {
1971 expect(addresses.length).toBe(0);
1972 // Check the feedback says 'blank entropy'
1973 driver.findElement(By.css(".feedback"))
1974 .getText()
1975 .then(function(feedbackText) {
1976 expect(feedbackText).toBe("Blank entropy");
1977 done();
1978 });
1979 })
1980 });
1981 });
1982 });
1983
1984 // Mnemonic length can be selected even for weak entropy
1985 it('Allows selection of mnemonic length even for weak entropy', function(done) {
1986 driver.findElement(By.css('.use-entropy'))
1987 .click();
1988 driver.executeScript(function() {
1989 $(".mnemonic-length").val("18").trigger("change");
1990 });
1991 driver.findElement(By.css('.entropy'))
1992 .sendKeys("012345");
1993 driver.sleep(generateDelay).then(function() {
1994 driver.findElement(By.css(".phrase"))
1995 .getAttribute("value").then(function(phrase) {
1996 var wordCount = phrase.split(/\s+/g).length;
1997 expect(wordCount).toBe(18);
1998 done();
1999 });
2000 });
2001 });
2002
2003 // Github issue 33
2004 // https://github.com/iancoleman/bip39/issues/33
2005 // Final cards should contribute entropy
2006 it('Uses as much entropy as possible for the mnemonic', function(done) {
2007 driver.findElement(By.css('.use-entropy'))
2008 .click();
2009 driver.findElement(By.css('.entropy'))
2010 .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");
2011 driver.sleep(generateDelay).then(function() {
2012 // Get mnemonic
2013 driver.findElement(By.css(".phrase"))
2014 .getAttribute("value").then(function(originalPhrase) {
2015 // Set the last 12 cards to be AS
2016 driver.findElement(By.css('.entropy'))
2017 .clear();
2018 driver.findElement(By.css('.entropy'))
2019 .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");
2020 driver.sleep(generateDelay).then(function() {
2021 // Get new mnemonic
2022 driver.findElement(By.css(".phrase"))
2023 .getAttribute("value").then(function(newPhrase) {
2024 expect(originalPhrase).not.toEqual(newPhrase);
2025 done();
2026 });
2027 });
2028 });
2029 });
2030 });
2031
2032 // Github issue 35
2033 // https://github.com/iancoleman/bip39/issues/35
2034 // QR Code support
2035 // TODO this doesn't work in selenium with firefox
2036 // see https://stackoverflow.com/q/40360223
2037 it('Shows a qr code on hover for the phrase', function(done) {
2038 if (browser == "firefox") {
2039 pending("Selenium + Firefox bug for mouseMove, see https://stackoverflow.com/q/40360223");
2040 }
2041 // generate a random mnemonic
2042 var generateEl = driver.findElement(By.css('.generate'));
2043 generateEl.click();
2044 // toggle qr to show (hidden by default)
2045 var phraseEl = driver.findElement(By.css(".phrase"));
2046 phraseEl.click();
2047 var rootKeyEl = driver.findElement(By.css(".root-key"));
2048 driver.sleep(generateDelay).then(function() {
2049 // hover over the root key
2050 driver.actions().mouseMove(rootKeyEl).perform().then(function() {
2051 // check the qr code shows
2052 driver.executeScript(function() {
2053 return $(".qr-container").find("canvas").length > 0;
2054 })
2055 .then(function(qrShowing) {
2056 expect(qrShowing).toBe(true);
2057 // hover away from the phrase
2058 driver.actions().mouseMove(generateEl).perform().then(function() {;
2059 // check the qr code hides
2060 driver.executeScript(function() {
2061 return $(".qr-container").find("canvas").length == 0;
2062 })
2063 .then(function(qrHidden) {
2064 expect(qrHidden).toBe(true);
2065 done();
2066 });
2067 });
2068 });
2069 });
2070 });
2071 });
2072
2073 // BIP44 account extendend private key is shown
2074 // github issue 37 - compatibility with electrum
2075 it('Shows the bip44 account extended private key', function(done) {
2076 driver.findElement(By.css(".phrase"))
2077 .sendKeys("abandon abandon ability");
2078 driver.sleep(generateDelay).then(function() {
2079 driver.findElement(By.css("#bip44 .account-xprv"))
2080 .getAttribute("value")
2081 .then(function(xprv) {
2082 expect(xprv).toBe("xprv9yzrnt4zWVJUr1k2VxSPy9ettKz5PpeDMgaVG7UKedhqnw1tDkxP2UyYNhuNSumk2sLE5ctwKZs9vwjsq3e1vo9egCK6CzP87H2cVYXpfwQ");
2083 done();
2084 });
2085 });
2086 });
2087
2088 // BIP44 account extendend public key is shown
2089 // github issue 37 - compatibility with electrum
2090 it('Shows the bip44 account extended public key', function(done) {
2091 driver.findElement(By.css(".phrase"))
2092 .sendKeys("abandon abandon ability");
2093 driver.sleep(generateDelay).then(function() {
2094 driver.findElement(By.css("#bip44 .account-xpub"))
2095 .getAttribute("value")
2096 .then(function(xprv) {
2097 expect(xprv).toBe("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2098 done();
2099 });
2100 });
2101 });
2102
2103 // github issue 40
2104 // BIP32 root key can be set as an xpub
2105 it('Generates addresses from xpub as bip32 root key', function(done) {
2106 driver.findElement(By.css('#bip32-tab a'))
2107 .click();
2108 // set xpub for account 0 of bip44 for 'abandon abandon ability'
2109 driver.findElement(By.css("#root-key"))
2110 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2111 driver.sleep(generateDelay).then(function() {
2112 // check the addresses are generated
2113 getFirstAddress(function(address) {
2114 expect(address).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug");
2115 // check the xprv key is not set
2116 driver.findElement(By.css(".extended-priv-key"))
2117 .getAttribute("value")
2118 .then(function(xprv) {
2119 expect(xprv).toBe("NA");
2120 // check the private key is not set
2121 driver.findElements(By.css(".privkey"))
2122 .then(function(els) {
2123 els[0]
2124 .getText()
2125 .then(function(privkey) {
2126 expect(xprv).toBe("NA");
2127 done();
2128 });
2129 });
2130 });
2131 });
2132 });
2133 });
2134
2135 // github issue 40
2136 // xpub for bip32 root key will not work with hardened derivation paths
2137 it('Shows error for hardened derivation paths with xpub root key', function(done) {
2138 // set xpub for account 0 of bip44 for 'abandon abandon ability'
2139 driver.findElement(By.css("#root-key"))
2140 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2141 driver.sleep(feedbackDelay).then(function() {
2142 // Check feedback is correct
2143 driver.findElement(By.css('.feedback'))
2144 .getText()
2145 .then(function(feedback) {
2146 var msg = "Hardened derivation path is invalid with xpub key";
2147 expect(feedback).toBe(msg);
2148 // Check no addresses are shown
2149 driver.findElements(By.css('.addresses tr'))
2150 .then(function(rows) {
2151 expect(rows.length).toBe(0);
2152 done();
2153 });
2154 });
2155 });
2156 });
2157
2158 // github issue 39
2159 // no root key shows feedback
2160 it('Shows feedback for no root key', function(done) {
2161 // set xpub for account 0 of bip44 for 'abandon abandon ability'
2162 driver.findElement(By.css('#bip32-tab a'))
2163 .click();
2164 driver.sleep(feedbackDelay).then(function() {
2165 // Check feedback is correct
2166 driver.findElement(By.css('.feedback'))
2167 .getText()
2168 .then(function(feedback) {
2169 expect(feedback).toBe("Invalid root key");
2170 done();
2171 });
2172 });
2173 });
2174
2175 // Github issue 44
2176 // display error switching tabs while addresses are generating
2177 it('Can change details while old addresses are still being generated', function(done) {
2178 // Set to generate 199 more addresses.
2179 // This will take a long time allowing a new set of addresses to be
2180 // generated midway through this lot.
2181 // The newly generated addresses should not include any from the old set.
2182 // Any more than 199 will show an alert which needs to be accepted.
2183 driver.findElement(By.css('.rows-to-add'))
2184 .clear();
2185 driver.findElement(By.css('.rows-to-add'))
2186 .sendKeys('199');
2187 // set the prhase
2188 driver.findElement(By.css('.phrase'))
2189 .sendKeys("abandon abandon ability");
2190 driver.sleep(generateDelay).then(function() {
2191 // change tabs which should cancel the previous generating
2192 driver.findElement(By.css('.rows-to-add'))
2193 .clear();
2194 driver.findElement(By.css('.rows-to-add'))
2195 .sendKeys('20');
2196 driver.findElement(By.css('#bip32-tab a'))
2197 .click()
2198 driver.sleep(generateDelay).then(function() {
2199 driver.findElements(By.css('.index'))
2200 .then(function(els) {
2201 // check the derivation paths have the right quantity
2202 expect(els.length).toBe(20);
2203 // check the derivation paths are in order
2204 testRowsAreInCorrectOrder(done);
2205 });
2206 });
2207 });
2208 }, generateDelay + 5000);
2209
2210 // Github issue 49
2211 // padding for binary should give length with multiple of 256
2212 // hashed entropy 1111 is length 252, so requires 4 leading zeros
2213 // prior to issue 49 it would only generate 2 leading zeros, ie missing 2
2214 it('Pads hashed entropy with leading zeros', function(done) {
2215 driver.findElement(By.css('.use-entropy'))
2216 .click();
2217 driver.executeScript(function() {
2218 $(".mnemonic-length").val("15").trigger("change");
2219 });
2220 driver.findElement(By.css('.entropy'))
2221 .sendKeys("1111");
2222 driver.sleep(generateDelay).then(function() {
2223 driver.findElement(By.css('.phrase'))
2224 .getAttribute("value")
2225 .then(function(phrase) {
2226 expect(phrase).toBe("avocado valid quantum cross link predict excuse edit street able flame large galaxy ginger nuclear");
2227 done();
2228 });
2229 });
2230 });
2231
2232 // Github pull request 55
2233 // https://github.com/iancoleman/bip39/pull/55
2234 // Client select
2235 it('Can set the derivation path on bip32 tab for bitcoincore', function(done) {
2236 testClientSelect(done, {
2237 selectValue: "0",
2238 bip32path: "m/0'/0'",
2239 useHardenedAddresses: "true",
2240 });
2241 });
2242 it('Can set the derivation path on bip32 tab for multibit', function(done) {
2243 testClientSelect(done, {
2244 selectValue: "2",
2245 bip32path: "m/0'/0",
2246 useHardenedAddresses: null,
2247 });
2248 });
2249
2250 // github issue 58
2251 // https://github.com/iancoleman/bip39/issues/58
2252 // bip32 derivation is correct, does not drop leading zeros
2253 // see also
2254 // https://medium.com/@alexberegszaszi/why-do-my-bip32-wallets-disagree-6f3254cc5846
2255 it('Retains leading zeros for bip32 derivation', function(done) {
2256 driver.findElement(By.css(".phrase"))
2257 .sendKeys("fruit wave dwarf banana earth journey tattoo true farm silk olive fence");
2258 driver.findElement(By.css(".passphrase"))
2259 .sendKeys("banana");
2260 driver.sleep(generateDelay).then(function() {
2261 getFirstAddress(function(address) {
2262 // Note that bitcore generates an incorrect address
2263 // 13EuKhffWkBE2KUwcbkbELZb1MpzbimJ3Y
2264 // see the medium.com link above for more details
2265 expect(address).toBe("17rxURoF96VhmkcEGCj5LNQkmN9HVhWb7F");
2266 done();
2267 });
2268 });
2269 });
2270
2271 // github issue 60
2272 // Japanese mnemonics generate incorrect bip32 seed
2273 // BIP39 seed is set from phrase
2274 it('Generates correct seed for Japanese mnemonics', function(done) {
2275 driver.findElement(By.css(".phrase"))
2276 .sendKeys("あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あおぞら");
2277 driver.findElement(By.css(".passphrase"))
2278 .sendKeys("メートルガバヴァぱばぐゞちぢ十人十色");
2279 driver.sleep(generateDelay).then(function() {
2280 driver.findElement(By.css(".seed"))
2281 .getAttribute("value")
2282 .then(function(seed) {
2283 expect(seed).toBe("a262d6fb6122ecf45be09c50492b31f92e9beb7d9a845987a02cefda57a15f9c467a17872029a9e92299b5cbdf306e3a0ee620245cbd508959b6cb7ca637bd55");
2284 done();
2285 });
2286 });
2287 });
2288
2289 // BIP49 official test vectors
2290 // https://github.com/bitcoin/bips/blob/master/bip-0049.mediawiki#test-vectors
2291 it('Generates BIP49 addresses matching the official test vectors', function(done) {
2292 driver.findElement(By.css('#bip49-tab a'))
2293 .click();
2294 selectNetwork("BTC - Bitcoin Testnet");
2295 driver.findElement(By.css(".phrase"))
2296 .sendKeys("abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about");
2297 driver.sleep(generateDelay).then(function() {
2298 getFirstAddress(function(address) {
2299 expect(address).toBe("2Mww8dCYPUpKHofjgcXcBCEGmniw9CoaiD2");
2300 done();
2301 });
2302 });
2303 });
2304
2305 // BIP49 derivation path is shown
2306 it('Shows the bip49 derivation path', function(done) {
2307 driver.findElement(By.css('#bip49-tab a'))
2308 .click();
2309 driver.findElement(By.css(".phrase"))
2310 .sendKeys("abandon abandon ability");
2311 driver.sleep(generateDelay).then(function() {
2312 driver.findElement(By.css('#bip49 .path'))
2313 .getAttribute("value")
2314 .then(function(path) {
2315 expect(path).toBe("m/49'/0'/0'/0");
2316 done();
2317 });
2318 });
2319 });
2320
2321 // BIP49 extended private key is shown
2322 it('Shows the bip49 extended private key', function(done) {
2323 driver.findElement(By.css('#bip49-tab a'))
2324 .click();
2325 driver.findElement(By.css(".phrase"))
2326 .sendKeys("abandon abandon ability");
2327 driver.sleep(generateDelay).then(function() {
2328 driver.findElement(By.css('.extended-priv-key'))
2329 .getAttribute("value")
2330 .then(function(xprv) {
2331 expect(xprv).toBe("yprvALYB4DYRG6CzzVgzQZwwqjAA2wjBGC3iEd7KYYScpoDdmf75qMRWZWxoFcRXBJjgEXdFqJ9vDRGRLJQsrL22Su5jMbNFeM9vetaGVqy9Qy2");
2332 done();
2333 });
2334 });
2335 });
2336
2337 // BIP49 extended public key is shown
2338 it('Shows the bip49 extended public key', function(done) {
2339 driver.findElement(By.css('#bip49-tab a'))
2340 .click();
2341 driver.findElement(By.css(".phrase"))
2342 .sendKeys("abandon abandon ability");
2343 driver.sleep(generateDelay).then(function() {
2344 driver.findElement(By.css('.extended-pub-key'))
2345 .getAttribute("value")
2346 .then(function(xprv) {
2347 expect(xprv).toBe("ypub6ZXXTj5K6TmJCymTWbUxCs6tayZffemZbr2vLvrEP8kceTSENtjm7KHH6thvAKxVar9fGe8rgsPEX369zURLZ68b4f7Vexz7RuXsjQ69YDt");
2348 done();
2349 });
2350 });
2351 });
2352
2353 // BIP49 account field changes address list
2354 it('Can set the bip49 account field', function(done) {
2355 driver.findElement(By.css('#bip49-tab a'))
2356 .click();
2357 driver.findElement(By.css('#bip49 .account'))
2358 .clear();
2359 driver.findElement(By.css('#bip49 .account'))
2360 .sendKeys("1");
2361 driver.findElement(By.css(".phrase"))
2362 .sendKeys("abandon abandon ability");
2363 driver.sleep(generateDelay).then(function() {
2364 getFirstAddress(function(address) {
2365 expect(address).toBe("381wg1GGN4rP88rNC9v7QWsiww63yLVPsn");
2366 done();
2367 });
2368 });
2369 });
2370
2371 // BIP49 change field changes address list
2372 it('Can set the bip49 change field', function(done) {
2373 driver.findElement(By.css('#bip49-tab a'))
2374 .click();
2375 driver.findElement(By.css('#bip49 .change'))
2376 .clear();
2377 driver.findElement(By.css('#bip49 .change'))
2378 .sendKeys("1");
2379 driver.findElement(By.css(".phrase"))
2380 .sendKeys("abandon abandon ability");
2381 driver.sleep(generateDelay).then(function() {
2382 getFirstAddress(function(address) {
2383 expect(address).toBe("3PEM7MiKed5konBoN66PQhK8r3hjGhy9dT");
2384 done();
2385 });
2386 });
2387 });
2388
2389 // BIP49 account extendend private key is shown
2390 it('Shows the bip49 account extended private key', function(done) {
2391 driver.findElement(By.css('#bip49-tab a'))
2392 .click();
2393 driver.findElement(By.css(".phrase"))
2394 .sendKeys("abandon abandon ability");
2395 driver.sleep(generateDelay).then(function() {
2396 driver.findElement(By.css('#bip49 .account-xprv'))
2397 .getAttribute("value")
2398 .then(function(xprv) {
2399 expect(xprv).toBe("yprvAHtB1M5Wp675aLzFy9TJYK2mSsLkg6mcBRh5DZTR7L4EnYSmYPaL63KFA4ycg1PngW5LfkmejxzosCs17TKZMpRFKc3z5SJar6QAKaFcaZL");
2400 done();
2401 });
2402 });
2403 });
2404
2405 // BIP49 account extendend public key is shown
2406 it('Shows the bip49 account extended public key', function(done) {
2407 driver.findElement(By.css('#bip49-tab a'))
2408 .click();
2409 driver.findElement(By.css(".phrase"))
2410 .sendKeys("abandon abandon ability");
2411 driver.sleep(generateDelay).then(function() {
2412 driver.findElement(By.css('#bip49 .account-xpub'))
2413 .getAttribute("value")
2414 .then(function(xprv) {
2415 expect(xprv).toBe("ypub6WsXQrcQeTfNnq4j5AzJuSyVzuBF5ZVTYecg1ws2ffbDfLmv5vtadqdj1NgR6C6gufMpMfJpHxvb6JEQKvETVNWCRanNedfJtnTchZiJtsL");
2416 done();
2417 });
2418 });
2419 });
2420
2421 // Test selecting coin where bip49 is unavailable (eg CLAM)
2422 it('Shows an error on bip49 tab for coins without bip49', function(done) {
2423 driver.findElement(By.css('#bip49-tab a'))
2424 .click();
2425 driver.findElement(By.css(".phrase"))
2426 .sendKeys("abandon abandon ability");
2427 driver.sleep(generateDelay).then(function() {
2428 selectNetwork("CLAM - Clams");
2429 // bip49 available is hidden
2430 driver.findElement(By.css('#bip49 .available'))
2431 .getAttribute("class")
2432 .then(function(classes) {
2433 expect(classes).toContain("hidden");
2434 // bip49 unavailable is shown
2435 driver.findElement(By.css('#bip49 .unavailable'))
2436 .getAttribute("class")
2437 .then(function(classes) {
2438 expect(classes).not.toContain("hidden");
2439 // check there are no addresses shown
2440 driver.findElements(By.css('.addresses tr'))
2441 .then(function(rows) {
2442 expect(rows.length).toBe(0);
2443 // check the derived private key is blank
2444 driver.findElement(By.css('.extended-priv-key'))
2445 .getAttribute("value")
2446 .then(function(xprv) {
2447 expect(xprv).toBe('');
2448 // check the derived public key is blank
2449 driver.findElement(By.css('.extended-pub-key'))
2450 .getAttribute("value")
2451 .then(function(xpub) {
2452 expect(xpub).toBe('');
2453 done();
2454 });
2455 });
2456 })
2457 });
2458 });
2459 });
2460 });
2461
2462 // github issue 43
2463 // Cleared mnemonic and root key still allows addresses to be generated
2464 // https://github.com/iancoleman/bip39/issues/43
2465 it('Clears old root keys from memory when mnemonic is cleared', function(done) {
2466 // set the phrase
2467 driver.findElement(By.css(".phrase"))
2468 .sendKeys("abandon abandon ability");
2469 driver.sleep(generateDelay).then(function() {
2470 // clear the mnemonic and root key
2471 // using selenium .clear() doesn't seem to trigger the 'input' event
2472 // so clear it using keys instead
2473 driver.findElement(By.css('.phrase'))
2474 .sendKeys(Key.CONTROL,"a");
2475 driver.findElement(By.css('.phrase'))
2476 .sendKeys(Key.DELETE);
2477 driver.findElement(By.css('.root-key'))
2478 .sendKeys(Key.CONTROL,"a");
2479 driver.findElement(By.css('.root-key'))
2480 .sendKeys(Key.DELETE);
2481 driver.sleep(generateDelay).then(function() {
2482 // try to generate more addresses
2483 driver.findElement(By.css('.more'))
2484 .click();
2485 driver.sleep(generateDelay).then(function() {
2486 driver.findElements(By.css(".addresses tr"))
2487 .then(function(els) {
2488 // check there are no addresses shown
2489 expect(els.length).toBe(0);
2490 done();
2491 });
2492 });
2493 });
2494 });
2495 });
2496
2497 // Github issue 95
2498 // error trying to generate addresses from xpub with hardened derivation
2499 it('Shows error for hardened addresses with xpub root key', function(done) {
2500 driver.findElement(By.css('#bip32-tab a'))
2501 .click()
2502 driver.executeScript(function() {
2503 $(".hardened-addresses").prop("checked", true);
2504 });
2505 // set xpub for account 0 of bip44 for 'abandon abandon ability'
2506 driver.findElement(By.css("#root-key"))
2507 .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf");
2508 driver.sleep(feedbackDelay).then(function() {
2509 // Check feedback is correct
2510 driver.findElement(By.css('.feedback'))
2511 .getText()
2512 .then(function(feedback) {
2513 var msg = "Hardened derivation path is invalid with xpub key";
2514 expect(feedback).toBe(msg);
2515 done();
2516 });
2517 });
2518 });
2519
2520 // Litecoin uses ltub by default, and can optionally be set to xprv
2521 // github issue 96
2522 // https://github.com/iancoleman/bip39/issues/96
2523 // Issue with extended keys on Litecoin
2524 it('Uses ltub by default for litecoin, but can be set to xprv', function(done) {
2525 driver.findElement(By.css('.phrase'))
2526 .sendKeys("abandon abandon ability");
2527 selectNetwork("LTC - Litecoin");
2528 driver.sleep(generateDelay).then(function() {
2529 // check the extended key is generated correctly
2530 driver.findElement(By.css('.root-key'))
2531 .getAttribute("value")
2532 .then(function(rootKey) {
2533 expect(rootKey).toBe("Ltpv71G8qDifUiNesiPqf6h5V6eQ8ic77oxQiYtawiACjBEx3sTXNR2HGDGnHETYxESjqkMLFBkKhWVq67ey1B2MKQXannUqNy1RZVHbmrEjnEU");
2534 // set litecoin to use ltub
2535 driver.executeScript(function() {
2536 $(".litecoin-use-ltub").prop("checked", false);
2537 $(".litecoin-use-ltub").trigger("change");
2538 });
2539 driver.sleep(generateDelay).then(function() {
2540 driver.findElement(By.css('.root-key'))
2541 .getAttribute("value")
2542 .then(function(rootKey) {
2543 expect(rootKey).toBe("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi");
2544 done();
2545 });
2546 })
2547 });
2548 });
2549 });
2550
2551 // github issue 99
2552 // https://github.com/iancoleman/bip39/issues/99#issuecomment-327094159
2553 // "warn me emphatically when they have detected invalid input" to the entropy field
2554 // A warning is shown when entropy is filtered and discarded
2555 it('Warns when entropy is filtered and discarded', function(done) {
2556 driver.findElement(By.css('.use-entropy'))
2557 .click();
2558 // set entropy to have no filtered content
2559 driver.findElement(By.css('.entropy'))
2560 .sendKeys("00000000 00000000 00000000 00000000");
2561 driver.sleep(generateDelay).then(function() {
2562 // check the filter warning does not show
2563 driver.findElement(By.css('.entropy-container .filter-warning'))
2564 .getAttribute("class")
2565 .then(function(classes) {
2566 expect(classes).toContain("hidden");
2567 // set entropy to have some filtered content
2568 driver.findElement(By.css('.entropy'))
2569 .sendKeys("10000000 zxcvbn 00000000 00000000 00000000");
2570 driver.sleep(entropyFeedbackDelay).then(function() {
2571 // check the filter warning shows
2572 driver.findElement(By.css('.entropy-container .filter-warning'))
2573 .getAttribute("class")
2574 .then(function(classes) {
2575 expect(classes).not.toContain("hidden");
2576 done();
2577 });
2578 });
2579 });
2580 });
2581 });
2582
2583 // Bitcoin Cash address can be set to use bitpay format
2584 it('Can use bitpay format for bitcoin cash addresses', function(done) {
2585 driver.executeScript(function() {
2586 $(".use-bitpay-addresses").prop("checked", true);
2587 });
2588 driver.findElement(By.css('.phrase'))
2589 .sendKeys("abandon abandon ability");
2590 selectNetwork("BCH - Bitcoin Cash");
2591 driver.sleep(generateDelay).then(function() {
2592 getFirstAddress(function(address) {
2593 expect(address).toBe("CZnpA9HPmvhuhLLPWJP8rNDpLUYXy1LXFk");
2594 done();
2595 });
2596 });
2597 });
2598
2599 // End of tests ported from old suit, so no more comments above each test now
2600
2601 it('Can generate more addresses from a custom index', function(done) {
2602 var expectedIndexes = [
2603 0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,
2604 40,41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59
2605 ];
2606 driver.findElement(By.css('.phrase'))
2607 .sendKeys("abandon abandon ability");
2608 driver.sleep(generateDelay).then(function() {
2609 // Set start of next lot of rows to be from index 40
2610 // which means indexes 20-39 will not be in the table.
2611 driver.findElement(By.css('.more-rows-start-index'))
2612 .sendKeys("40");
2613 driver.findElement(By.css('.more'))
2614 .click();
2615 driver.sleep(generateDelay).then(function() {
2616 // Check actual indexes in the table match the expected pattern
2617 driver.findElements(By.css(".index"))
2618 .then(function(els) {
2619 expect(els.length).toBe(expectedIndexes.length);
2620 var testRowAtIndex = function(i) {
2621 if (i >= expectedIndexes.length) {
2622 done();
2623 }
2624 else {
2625 els[i].getText()
2626 .then(function(actualPath) {
2627 var noHardened = actualPath.replace(/'/g, "");
2628 var pathBits = noHardened.split("/")
2629 var lastBit = pathBits[pathBits.length-1];
2630 var actualIndex = parseInt(lastBit);
2631 var expectedIndex = expectedIndexes[i];
2632 expect(actualIndex).toBe(expectedIndex);
2633 testRowAtIndex(i+1);
2634 });
2635 }
2636 }
2637 testRowAtIndex(0);
2638 });
2639 });
2640 });
2641 });
2642
2643 it('Can generate BIP141 addresses with P2WPKH-in-P2SH semanitcs', function(done) {
2644 // Sourced from BIP49 official test specs
2645 driver.findElement(By.css('#bip141-tab a'))
2646 .click();
2647 driver.findElement(By.css('.bip141-path'))
2648 .clear();
2649 driver.findElement(By.css('.bip141-path'))
2650 .sendKeys("m/49'/1'/0'/0");
2651 selectNetwork("BTC - Bitcoin Testnet");
2652 driver.findElement(By.css(".phrase"))
2653 .sendKeys("abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about");
2654 driver.sleep(generateDelay).then(function() {
2655 getFirstAddress(function(address) {
2656 expect(address).toBe("2Mww8dCYPUpKHofjgcXcBCEGmniw9CoaiD2");
2657 done();
2658 });
2659 });
2660 });
2661
2662 it('Can generate BIP141 addresses with P2WPKH semanitcs', function(done) {
2663 // This result tested against bitcoinjs-lib test spec for segwit address
2664 // using the first private key of this mnemonic and default path m/0
2665 // https://github.com/bitcoinjs/bitcoinjs-lib/blob/9c8503cab0c6c30a95127042703bc18e8d28c76d/test/integration/addresses.js#L50
2666 // so whilst not directly comparable, substituting the private key produces
2667 // identical results between this tool and the bitcoinjs-lib test.
2668 // Private key generated is:
2669 // L3L8Nu9whawPBNLGtFqDhKut9DKKfG3CQoysupT7BimqVCZsLFNP
2670 driver.findElement(By.css('#bip141-tab a'))
2671 .click();
2672 // Choose P2WPKH
2673 driver.executeScript(function() {
2674 $(".bip141-semantics option[selected]").removeAttr("selected");
2675 $(".bip141-semantics option").filter(function(i,e) {
2676 return $(e).html() == "P2WPKH";
2677 }).prop("selected", true);
2678 $(".bip141-semantics").trigger("change");
2679 });
2680 driver.findElement(By.css(".phrase"))
2681 .sendKeys("abandon abandon ability");
2682 driver.sleep(generateDelay).then(function() {
2683 getFirstAddress(function(address) {
2684 expect(address).toBe("bc1qfwu6a5a3evygrk8zvdxxvz4547lmpyx5vsfxe9");
2685 done();
2686 });
2687 });
2688 });
2689
2690 it('Shows the entropy used by the PRNG when clicking generate', function(done) {
2691 driver.findElement(By.css('.generate')).click();
2692 driver.sleep(generateDelay).then(function() {
2693 driver.findElement(By.css('.entropy'))
2694 .getAttribute("value")
2695 .then(function(entropy) {
2696 expect(entropy).not.toBe("");
2697 done();
2698 });
2699 });
2700 });
2701
2702 it('Shows the index of each word in the mnemonic', function(done) {
2703 driver.findElement(By.css('.phrase'))
2704 .sendKeys("abandon abandon ability");
2705 driver.sleep(generateDelay).then(function() {
2706 driver.findElement(By.css('.use-entropy'))
2707 .click();
2708 driver.findElement(By.css('.word-indexes'))
2709 .getText()
2710 .then(function(indexes) {
2711 expect(indexes).toBe("0, 0, 1");
2712 done();
2713 });
2714 });
2715 });
2716
2717 it('Shows the derivation path for bip84 tab', function(done) {
2718 driver.findElement(By.css('#bip84-tab a'))
2719 .click()
2720 driver.findElement(By.css('.phrase'))
2721 .sendKeys('abandon abandon ability');
2722 driver.sleep(generateDelay).then(function() {
2723 driver.findElement(By.css('#bip84 .path'))
2724 .getAttribute("value")
2725 .then(function(path) {
2726 expect(path).toBe("m/84'/0'/0'/0");
2727 done();
2728 })
2729 });
2730 });
2731
2732 it('Shows the extended private key for bip84 tab', function(done) {
2733 driver.findElement(By.css('#bip84-tab a'))
2734 .click()
2735 driver.findElement(By.css('.phrase'))
2736 .sendKeys('abandon abandon ability');
2737 driver.sleep(generateDelay).then(function() {
2738 driver.findElement(By.css('.extended-priv-key'))
2739 .getAttribute("value")
2740 .then(function(path) {
2741 expect(path).toBe("zprvAev3RKrZ3QVKiUFCfdeMRen1BPDJgdNt1XpxiDy8acSs4kkAGTCvq7HeRYRNNpo8EtEjCFQBWavJwtCUR29y4TUCH4X5RXMcyq48uN8y9BP");
2742 done();
2743 })
2744 });
2745 });
2746
2747 it('Shows the extended public key for bip84 tab', function(done) {
2748 driver.findElement(By.css('#bip84-tab a'))
2749 .click()
2750 driver.findElement(By.css('.phrase'))
2751 .sendKeys('abandon abandon ability');
2752 driver.sleep(generateDelay).then(function() {
2753 driver.findElement(By.css('.extended-pub-key'))
2754 .getAttribute("value")
2755 .then(function(path) {
2756 expect(path).toBe("zpub6suPpqPSsn3cvxKfmfBMnnijjR3o666jNkkZWcNk8wyqwZ5JozXBNuc8Gs7DB3uLwTDvGVTspVEAUQcEjKF3pZHgywVbubdTqbXTUg7usyx");
2757 done();
2758 })
2759 });
2760 });
2761
2762 it('Changes the address list if bip84 account is changed', function(done) {
2763 driver.findElement(By.css('#bip84-tab a'))
2764 .click()
2765 driver.findElement(By.css('#bip84 .account'))
2766 .sendKeys('1');
2767 driver.findElement(By.css('.phrase'))
2768 .sendKeys('abandon abandon ability');
2769 driver.sleep(generateDelay).then(function() {
2770 getFirstAddress(function(address) {
2771 expect(address).toBe("bc1qp7vv669t2fy965jdzvqwrraana89ctd5ewc662");
2772 done();
2773 });
2774 });
2775 });
2776
2777 it('Changes the address list if bip84 change is changed', function(done) {
2778 driver.findElement(By.css('#bip84-tab a'))
2779 .click()
2780 driver.findElement(By.css('#bip84 .change'))
2781 .sendKeys('1');
2782 driver.findElement(By.css('.phrase'))
2783 .sendKeys('abandon abandon ability');
2784 driver.sleep(generateDelay).then(function() {
2785 getFirstAddress(function(address) {
2786 expect(address).toBe("bc1qr39vj6rh06ff05m53uxq8uazehwhccswylhrs2");
2787 done();
2788 });
2789 });
2790 });
2791
2792 it('Passes the official BIP84 test spec for rootpriv', function(done) {
2793 driver.findElement(By.css('#bip84-tab a'))
2794 .click()
2795 driver.findElement(By.css('.phrase'))
2796 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
2797 driver.sleep(generateDelay).then(function() {
2798 driver.findElement(By.css(".root-key"))
2799 .getAttribute("value")
2800 .then(function(rootKey) {
2801 expect(rootKey).toBe("zprvAWgYBBk7JR8Gjrh4UJQ2uJdG1r3WNRRfURiABBE3RvMXYSrRJL62XuezvGdPvG6GFBZduosCc1YP5wixPox7zhZLfiUm8aunE96BBa4Kei5");
2802 done();
2803 })
2804 });
2805 });
2806
2807 it('Passes the official BIP84 test spec for account 0 xprv', function(done) {
2808 driver.findElement(By.css('#bip84-tab a'))
2809 .click()
2810 driver.findElement(By.css('.phrase'))
2811 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
2812 driver.sleep(generateDelay).then(function() {
2813 driver.findElement(By.css("#bip84 .account-xprv"))
2814 .getAttribute("value")
2815 .then(function(rootKey) {
2816 expect(rootKey).toBe("zprvAdG4iTXWBoARxkkzNpNh8r6Qag3irQB8PzEMkAFeTRXxHpbF9z4QgEvBRmfvqWvGp42t42nvgGpNgYSJA9iefm1yYNZKEm7z6qUWCroSQnE");
2817 done();
2818 })
2819 });
2820 });
2821
2822 it('Passes the official BIP84 test spec for account 0 xpub', function(done) {
2823 driver.findElement(By.css('#bip84-tab a'))
2824 .click()
2825 driver.findElement(By.css('.phrase'))
2826 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
2827 driver.sleep(generateDelay).then(function() {
2828 driver.findElement(By.css("#bip84 .account-xpub"))
2829 .getAttribute("value")
2830 .then(function(rootKey) {
2831 expect(rootKey).toBe("zpub6rFR7y4Q2AijBEqTUquhVz398htDFrtymD9xYYfG1m4wAcvPhXNfE3EfH1r1ADqtfSdVCToUG868RvUUkgDKf31mGDtKsAYz2oz2AGutZYs");
2832 done();
2833 })
2834 });
2835 });
2836
2837 it('Passes the official BIP84 test spec for account 0 first address', function(done) {
2838 driver.findElement(By.css('#bip84-tab a'))
2839 .click()
2840 driver.findElement(By.css('.phrase'))
2841 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
2842 driver.sleep(generateDelay).then(function() {
2843 getFirstAddress(function(address) {
2844 expect(address).toBe("bc1qcr8te4kr609gcawutmrza0j4xv80jy8z306fyu");
2845 done();
2846 });
2847 });
2848 });
2849
2850 it('Passes the official BIP84 test spec for account 0 first change address', function(done) {
2851 driver.findElement(By.css('#bip84-tab a'))
2852 .click()
2853 driver.findElement(By.css('.phrase'))
2854 .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
2855 driver.findElement(By.css('#bip84 .change'))
2856 .sendKeys('1');
2857 driver.sleep(generateDelay).then(function() {
2858 getFirstAddress(function(address) {
2859 expect(address).toBe("bc1q8c6fshw2dlwun7ekn9qwf37cu2rn755upcp6el");
2860 done();
2861 });
2862 });
2863 });
2864
2865 it('Can display the table as csv', function(done) {
2866 var headings = "path,address,public key,private key";
2867 var row1 = "m/44'/0'/0'/0/0,1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug,033f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3,L26cVSpWFkJ6aQkPkKmTzLqTdLJ923e6CzrVh9cmx21QHsoUmrEE";
2868 var row20 = "m/44'/0'/0'/0/19,1KhBy28XLAciXnnRvm71PvQJaETyrxGV55,02b4b3e396434d8cdd20c03ac4aaa07387784d5d867b75987f516f5705ee68cb3a,L4GrDrjReMsCAu5DkLXn79jSb95qR7Zfx7eshybCQZ1qL32MXJab";
2869 driver.findElement(By.css('.phrase'))
2870 .sendKeys('abandon abandon ability');
2871 driver.sleep(generateDelay).then(function() {
2872 driver.findElement(By.css('.csv'))
2873 .getAttribute("value")
2874 .then(function(csv) {
2875 expect(csv).toContain(headings);
2876 expect(csv).toContain(row1);
2877 expect(csv).toContain(row20);
2878 done();
2879 });
2880 });
2881 });
2882
2883 it('LeftPads ethereum keys that are less than 32 bytes', function(done) {
2884 // see https://github.com/iancoleman/bip39/issues/155
2885 selectNetwork("ETH - Ethereum");
2886 driver.findElement(By.css('#bip32-tab a'))
2887 .click()
2888 driver.findElement(By.css('#bip32-path'))
2889 .clear();
2890 driver.findElement(By.css('#bip32-path'))
2891 .sendKeys("m/44'/60'/0'");
2892 driver.findElement(By.css('.phrase'))
2893 .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');
2894 driver.sleep(generateDelay).then(function() {
2895 getFirstAddress(function(address) {
2896 expect(address).toBe("0x8943E785B4a5714FC87a3aFAad1eB1FeB602B118");
2897 done();
2898 });
2899 });
2900 });
2901
2902 it('Can encrypt private keys using BIP38', function(done) {
2903 // see https://github.com/iancoleman/bip39/issues/140
2904 driver.executeScript(function() {
2905 $(".use-bip38").prop("checked", true);
2906 });
2907 driver.findElement(By.css('.bip38-password'))
2908 .sendKeys('bip38password');
2909 driver.findElement(By.css('.rows-to-add'))
2910 .clear();
2911 driver.findElement(By.css('.rows-to-add'))
2912 .sendKeys('1');
2913 driver.findElement(By.css('.phrase'))
2914 .sendKeys('abandon abandon ability');
2915 driver.sleep(bip38delay).then(function() {
2916 // address
2917 getFirstRowValue(function(address) {
2918 expect(address).toBe("1NCvSdumA3ngMM9c4aqU56AM6rqXddfuXB");
2919 // pubkey
2920 getFirstRowValue(function(pubkey) {
2921 expect(pubkey).toBe("043f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3884a74447ea901729b1e73a999b7520e7cb55b4120e6432c64153ccab8a848e1");
2922 // privkey
2923 getFirstRowValue(function(privkey) {
2924 expect(privkey).toBe("6PRNRiFnj1RoR3sXhymdCvoZCgnUHQpfupNdKkFbWJkwWQEKesWt1EDMDM");
2925 done();
2926 }, ".privkey");
2927 }, ".pubkey");
2928 }, ".address");
2929 });
2930 }, bip38delay + 5000);
2931
2932 it('Shows the checksum for the entropy', function(done) {
2933 driver.findElement(By.css('.use-entropy'))
2934 .click();
2935 driver.findElement(By.css('.entropy'))
2936 .sendKeys("00000000000000000000000000000000");
2937 driver.sleep(generateDelay).then(function() {
2938 driver.findElement(By.css('.checksum'))
2939 .getText()
2940 .then(function(text) {
2941 expect(text).toBe("1");
2942 done();
2943 });
2944 });
2945 });
2946
2947 it('Shows the checksum for the entropy with the correct groupings', function(done) {
2948 driver.findElement(By.css('.use-entropy'))
2949 .click();
2950 // create a checksum of 20 bits, which spans multiple words
2951 driver.findElement(By.css('.entropy'))
2952 .sendKeys("F000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000");
2953 driver.sleep(generateDelay).then(function() {
2954 driver.findElement(By.css('.checksum'))
2955 .getText()
2956 .then(function(text) {
2957 // first group is 9 bits, second group is 11
2958 expect(text).toBe("011010111 01110000110");
2959 done();
2960 });
2961 });
2962 });
2963
2964 });