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