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