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