X-Git-Url: https://git.immae.eu/?a=blobdiff_plain;f=tests%2Fspec%2Ftests.js;h=b81294b64a532bf0a59d0db958b17a30d26cf539;hb=9369eaa6d56500061c952927691e31cd503ae11c;hp=72edd281d19efe7e494b74748124a9d93b853595;hpb=0460b53f3aff1745fd2727c094d12f90cfbe1d10;p=perso%2FImmae%2FProjets%2FCryptomonnaies%2FBIP39.git diff --git a/tests/spec/tests.js b/tests/spec/tests.js index 72edd28..b81294b 100644 --- a/tests/spec/tests.js +++ b/tests/spec/tests.js @@ -30,9 +30,10 @@ var until = webdriver.until; var newDriver = null; var driver = null; // Delays in ms -var generateDelay = 1000; +var generateDelay = 1500; var feedbackDelay = 500; var entropyFeedbackDelay = 500; +var bip38delay = 15000; // url uses file:// scheme var path = require('path') @@ -422,6 +423,13 @@ it('Allows selection of dogecoin', function(done) { }; testNetwork(done, params); }); +it('Allows selection of denarius', function(done) { + var params = { + selectText: "DNR - Denarius", + firstAddress: "DFdFMVUMzU9xX88EywXvAGwjiwpxyh9vKb", + }; + testNetwork(done, params); +}); it('Allows selection of shadowcash', function(done) { var params = { selectText: "SDC - ShadowCash", @@ -492,6 +500,13 @@ it('Allows selection of game', function(done) { }; testNetwork(done, params); }); +it('Allows selection of komodo', function(done) { + var params = { + selectText: "KMD - Komodo", + firstAddress: "RMPPzJwAjPVZZAwJvXivHJGGjdCx6WBD2t", + }; + testNetwork(done, params); +}); it('Allows selection of namecoin', function(done) { var params = { selectText: "NMC - Namecoin", @@ -499,6 +514,13 @@ it('Allows selection of namecoin', function(done) { }; testNetwork(done, params); }); +it('Allows selection of onixcoin', function(done) { + var params = { + selectText: "ONX - Onixcoin", + firstAddress: "XGwMqddeKjT3ddgX73QokjVbCL3aK6Yxfk", + }; + testNetwork(done, params); +}); it('Allows selection of peercoin', function(done) { var params = { selectText: "PPC - Peercoin", @@ -531,7 +553,7 @@ it('Allows selection of slimcoin testnet', function(done) { it('Allows selection of bitcoin cash', function(done) { var params = { selectText: "BCH - Bitcoin Cash", - firstAddress: "1JKvb6wKtsjNoCRxpZ4DGrbniML7z5U16A", + firstAddress: "bitcoincash:qzlquk7w4hkudxypl4fgv8x279r754dkvur7jpcsps", }; testNetwork(done, params); }); @@ -577,222 +599,832 @@ it('Allows selection of nubits', function(done) { }; testNetwork(done, params); }); - -// BIP39 seed is set from phrase -it('Sets the bip39 seed from the prhase', function(done) { - driver.findElement(By.css('.phrase')) - .sendKeys('abandon abandon ability'); - driver.sleep(generateDelay).then(function() { - driver.findElement(By.css('.seed')) - .getAttribute("value") - .then(function(seed) { - expect(seed).toBe("20da140d3dd1df8713cefcc4d54ce0e445b4151027a1ab567b832f6da5fcc5afc1c3a3f199ab78b8e0ab4652efd7f414ac2c9a3b81bceb879a70f377aa0a58f3"); - done(); - }) - }); +it('Allows selection of bitcoin gold', function(done) { + var params = { + selectText: "BTG - Bitcoin Gold", + firstAddress: "GdDqug4WUsn5syNbSTHatNn4XnuwZtzedx", + }; + testNetwork(done, params); }); - -// BIP32 root key is set from phrase -it('Sets the bip39 root key from the prhase', function(done) { - driver.findElement(By.css('.phrase')) - .sendKeys('abandon abandon ability'); - driver.sleep(generateDelay).then(function() { - driver.findElement(By.css('.root-key')) - .getAttribute("value") - .then(function(seed) { - expect(seed).toBe("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi"); - done(); - }) - }); +it('Allows selection of monacoin', function(done) { + var params = { + selectText: "MONA - Monacoin", + firstAddress: "MKMiMr7MyjDKjJbCBzgF6u4ByqTS4NkRB1", + }; + testNetwork(done, params); }); - -// Tabs show correct addresses when changed -it('Shows the correct address when tab is changed', function(done) { - driver.findElement(By.css('.phrase')) - .sendKeys('abandon abandon ability'); - driver.sleep(generateDelay).then(function() { - driver.findElement(By.css('#bip32-tab a')) - .click(); - driver.sleep(generateDelay).then(function() { - getFirstAddress(function(address) { - expect(address).toBe("17uQ7s2izWPwBmEVFikTmZUjbBKWYdJchz"); - done(); - }); - }); - }); +it('Allows selection of AXE', function(done) { + var params = { + selectText: "AXE - Axe", + firstAddress: "PScwtLUyPiGrqtKXrHF37DGETLXLZdw4up", + }; + testNetwork(done, params); }); - -// BIP44 derivation path is shown -it('Shows the derivation path for bip44 tab', function(done) { - driver.findElement(By.css('.phrase')) - .sendKeys('abandon abandon ability'); - driver.sleep(generateDelay).then(function() { - driver.findElement(By.css('#bip44 .path')) - .getAttribute("value") - .then(function(path) { - expect(path).toBe("m/44'/0'/0'/0"); - done(); - }) - }); +it('Allows selection of BlackCoin', function(done) { + var params = { + selectText: "BLK - BlackCoin", + firstAddress: "B5MznAKwj7uQ42vDz3w4onhBXPcqhTwJ9z", + }; + testNetwork(done, params); }); - -// BIP44 extended private key is shown -it('Shows the extended private key for bip44 tab', function(done) { - driver.findElement(By.css('.phrase')) - .sendKeys('abandon abandon ability'); - driver.sleep(generateDelay).then(function() { - driver.findElement(By.css('.extended-priv-key')) - .getAttribute("value") - .then(function(path) { - expect(path).toBe("xprvA2DxxvPZcyRvYgZMGS53nadR32mVDeCyqQYyFhrCVbJNjPoxMeVf7QT5g7mQASbTf9Kp4cryvcXnu2qurjWKcrdsr91jXymdCDNxKgLFKJG"); - done(); - }) - }); +it('Allows selection of Neblio', function(done) { + var params = { + selectText: "NEBL - Neblio", + firstAddress: "NefkeEEvhusbHMmTRrxx7H9wFnUXd8qQsE", + }; + testNetwork(done, params); }); - -// BIP44 extended public key is shown -it('Shows the extended public key for bip44 tab', function(done) { - driver.findElement(By.css('.phrase')) - .sendKeys('abandon abandon ability'); - driver.sleep(generateDelay).then(function() { - driver.findElement(By.css('.extended-pub-key')) - .getAttribute("value") - .then(function(path) { - expect(path).toBe("xpub6FDKNRvTTLzDmAdpNTc49ia9b4byd6vqCdUa46Fp3vqMcC96uBoufCmZXQLiN5AK3iSCJMhf9gT2sxkpyaPepRuA7W3MujV5tGmF5VfbueM"); - done(); - }) - }); +it('Allows selection of Beetlecoin', function(done) { + var params = { + selectText: "BEET - Beetlecoin", + firstAddress: "BVmtbEsGrjpknprmpHFq26z4kYHJUFHE71", + }; + testNetwork(done, params); }); - -// BIP44 account field changes address list -it('Changes the address list if bip44 account is changed', function(done) { - driver.findElement(By.css('#bip44 .account')) - .sendKeys('1'); - driver.findElement(By.css('.phrase')) - .sendKeys('abandon abandon ability'); - driver.sleep(generateDelay).then(function() { - getFirstAddress(function(address) { - expect(address).toBe("1Nq2Wmu726XHCuGhctEtGmhxo3wzk5wZ1H"); - done(); - }); - }); +it('Allows selection of Adcoin', function(done) { + var params = { + selectText: "ACC - Adcoin", + firstAddress: "AcEDM6V5sF4kFHC76MJjjfProtS5Sw2qcd", + }; + testNetwork(done, params); }); - -// BIP44 change field changes address list -it('Changes the address list if bip44 change is changed', function(done) { - driver.findElement(By.css('#bip44 .change')) - .sendKeys('1'); - driver.findElement(By.css('.phrase')) - .sendKeys('abandon abandon ability'); - driver.sleep(generateDelay).then(function() { - getFirstAddress(function(address) { - expect(address).toBe("1KAGfWgqfVbSSXY56fNQ7YnhyKuoskHtYo"); - done(); - }); - }); +it('Allows selection of Asiacoin', function(done) { + var params = { + selectText: "AC - Asiacoin", + firstAddress: "ALupuEEz7kJjQTAvmtcBMBVuEjPa7GqZzE", + }; + testNetwork(done, params); }); - -// BIP32 derivation path can be set -it('Can use a custom bip32 derivation path', function(done) { - driver.findElement(By.css('#bip32-tab a')) - .click(); - driver.findElement(By.css('#bip32 .path')) - .clear(); - driver.findElement(By.css('#bip32 .path')) - .sendKeys('m/1'); - driver.findElement(By.css('.phrase')) - .sendKeys('abandon abandon ability'); - driver.sleep(generateDelay).then(function() { - getFirstAddress(function(address) { - expect(address).toBe("16pYQQdLD1hH4hwTGLXBaZ9Teboi1AGL8L"); - done(); - }); - }); +it('Allows selection of Auroracoin', function(done) { + var params = { + selectText: "AUR - Auroracoin", + firstAddress: "ANuraS6F4Jpi413FEnavjYkKYJJRHkgYCm", + }; + testNetwork(done, params); }); - -// BIP32 can use hardened derivation paths -it('Can use a hardened derivation paths', function(done) { - driver.findElement(By.css('#bip32-tab a')) - .click(); - driver.findElement(By.css('#bip32 .path')) - .clear(); - driver.findElement(By.css('#bip32 .path')) - .sendKeys("m/0'"); - driver.findElement(By.css('.phrase')) - .sendKeys('abandon abandon ability'); - driver.sleep(generateDelay).then(function() { - getFirstAddress(function(address) { - expect(address).toBe("14aXZeprXAE3UUKQc4ihvwBvww2LuEoHo4"); - done(); - }); - }); +it('Allows selection of Bata', function(done) { + var params = { + selectText: "BTA - Bata", + firstAddress: "BGxBdNeYPtF3GCuTtZBPQdFxCkdBYSF3fj", + }; + testNetwork(done, params); }); - -// BIP32 extended private key is shown -it('Shows the BIP32 extended private key', function(done) { - driver.findElement(By.css('#bip32-tab a')) - .click(); - driver.findElement(By.css('.phrase')) - .sendKeys('abandon abandon ability'); - driver.sleep(generateDelay).then(function() { - driver.findElement(By.css('.extended-priv-key')) - .getAttribute("value") - .then(function(privKey) { - expect(privKey).toBe("xprv9va99uTVE5aLiutUVLTyfxfe8v8aaXjSQ1XxZbK6SezYVuikA9MnjQVTA8rQHpNA5LKvyQBpLiHbBQiiccKiBDs7eRmBogsvq3THFeLHYbe"); - done(); - }); - }); +it('Allows selection of Belacoin', function(done) { + var params = { + selectText: "BELA - Belacoin", + firstAddress: "BEeetqpNffdzeknSpNmQp5KAFh2KK1Qx7S", + }; + testNetwork(done, params); }); - -// BIP32 extended public key is shown -it('Shows the BIP32 extended public key', function(done) { - driver.findElement(By.css('#bip32-tab a')) - .click(); - driver.findElement(By.css('.phrase')) - .sendKeys('abandon abandon ability'); - driver.sleep(generateDelay).then(function() { - driver.findElement(By.css('.extended-pub-key')) - .getAttribute("value") - .then(function(pubKey) { - expect(pubKey).toBe("xpub69ZVZQzP4T8dwPxwbMzz36cNgwy4yzTHmETZMyihzzXXNi3thgg3HCow1RtY252wdw5rS8369xKnraN5Q93y3FkFfJp2XEHWUrkyXsjS93P"); - done(); - }); - }); +it('Allows selection of Bitcoin Atom', function(done) { + var params = { + selectText: "BCA - Bitcoin Atom", + firstAddress: "AMy6qMbJeC4zsGRL6iWszmeCdQH65fgfih", + }; + testNetwork(done, params); }); - -// Derivation path is shown in table -it('Shows the derivation path in the table', function(done) { - driver.findElement(By.css('.phrase')) - .sendKeys('abandon abandon ability'); - driver.sleep(generateDelay).then(function() { - getFirstPath(function(path) { - expect(path).toBe("m/44'/0'/0'/0/0"); - done(); - }); - }); +it('Allows selection of Bitcoinplus', function(done) { + var params = { + selectText: "XBC - Bitcoinplus", + firstAddress: "B7FSynZoDbEwTCSgsXq9nJ5ue8owYLVL8r", + }; + testNetwork(done, params); }); - -// Derivation path for address can be hardened -it('Can derive hardened addresses', function(done) { - driver.findElement(By.css('#bip32-tab a')) - .click(); - driver.executeScript(function() { - $(".hardened-addresses").prop("checked", true); - }); - driver.findElement(By.css('.phrase')) - .sendKeys('abandon abandon ability'); - driver.sleep(generateDelay).then(function() { - getFirstAddress(function(address) { - expect(address).toBe("18exLzUv7kfpiXRzmCjFDoC9qwNLFyvwyd"); - done(); - }); - }); +it('Allows selection of Bitcoin Private', function(done) { + var params = { + selectText: "BTCP - Bitcoin Private", + firstAddress: "b1M3PbiXXyN6Hdivdw5rJv5VKpLjPzhm4jM", + }; + testNetwork(done, params); }); - -// Derivation path visibility can be toggled -it('Can toggle visibility of the derivation path column', function(done) { - driver.findElement(By.css('.phrase')) - .sendKeys('abandon abandon ability'); +it('Allows selection of Bitcoinz', function(done) { + var params = { + selectText: "BTCZ - Bitcoinz", + firstAddress: "t1X2YQoxs8cYRo2oaBYgVEwW5QNjCC59NYc", + }; + testNetwork(done, params); +}); +it('Allows selection of Bitcore', function(done) { + var params = { + selectText: "BTX - Bitcore", + firstAddress: "1Dg18EtqhReS11e9h8khkLjWGLHVjPM2AB", + }; + testNetwork(done, params); +}); +it('Allows selection of Bitsend', function(done) { + var params = { + selectText: "BSD - Bitsend", + firstAddress: "iBPk7LYjDun3EPk7CRR8UUmnPoceVc1bp2", + }; + testNetwork(done, params); +}); +it('Allows selection of Britcoin', function(done) { + var params = { + selectText: "BRIT - Britcoin", + firstAddress: "B6Aue4J2XLs1f1dtD4H1SHYFfh4XrmEbrw", + }; + testNetwork(done, params); +}); +it('Allows selection of Canadaecoin', function(done) { + var params = { + selectText: "CDN - Canadaecoin", + firstAddress: "CanAyCfd5Rj2CQVfaoAmvDUZunPM5W1AEQ", + }; + testNetwork(done, params); +}); +it('Allows selection of Cannacoin', function(done) { + var params = { + selectText: "CCN - Cannacoin", + firstAddress: "CYjW8xWB43g6krLJTmmrPk1PonoQX7h9Qd", + }; + testNetwork(done, params); +}); +it('Allows selection of Clubcoin', function(done) { + var params = { + selectText: "CLUB - Clubcoin", + firstAddress: "CHMDEXN4sihpSVX4GyAa2hZ62shnby7uyN", + }; + testNetwork(done, params); +}); +it('Allows selection of Compcoin', function(done) { + var params = { + selectText: "CMP - Compcoin", + firstAddress: "CLshtw3zhxkseBJS46UF12v3AFy9Dx7JVv", + }; + testNetwork(done, params); +}); +it('Allows selection of Crave', function(done) { + var params = { + selectText: "CRAVE - Crave", + firstAddress: "VCYJeti6uKMNBFKCL7eP96UwuFWYHM7c85", + }; + testNetwork(done, params); +}); +it('Allows selection of Defcoin', function(done) { + var params = { + selectText: "DFC - Defcoin", + firstAddress: "D8swcgyaaFUrXZU3ATwbgy16buCpWqbG1M", + }; + testNetwork(done, params); +}); +it('Allows selection of Diamond', function(done) { + var params = { + selectText: "DMD - Diamond", + firstAddress: "dJnrVbLL9UPjdaVRz2C8VpqHZknqAqjLek", + }; + testNetwork(done, params); +}); +it('Allows selection of Digibyte', function(done) { + var params = { + selectText: "DGB - Digibyte", + firstAddress: "D85Rp9jwLtMdmP6wGjTiqHBdVQLST3YCEq", + }; + testNetwork(done, params); +}); +it('Allows selection of Digitalcoin', function(done) { + var params = { + selectText: "DGC - Digitalcoin", + firstAddress: "DKw4UGKEAZWweDNEbBFNQx4EM8x1mpUdia", + }; + testNetwork(done, params); +}); +it('Allows selection of Ecoin', function(done) { + var params = { + selectText: "ECN - Ecoin", + firstAddress: "e6WFPLG5gcXyF7cESFteH1hE2XSmowW5yB", + }; + testNetwork(done, params); +}); +it('Allows selection of Edrcoin', function(done) { + var params = { + selectText: "EDRC - Edrcoin", + firstAddress: "eh1nUJsvgKPFv6ebMBfcwJ299GMCpjeZUG", + }; + testNetwork(done, params); +}); +it('Allows selection of Egulden', function(done) { + var params = { + selectText: "EFL - Egulden", + firstAddress: "Lg66yt55R7edRM58cDhKzXik2kFme3viX7", + }; + testNetwork(done, params); +}); +it('Allows selection of Einsteinium', function(done) { + var params = { + selectText: "EMC2 - Einsteinium", + firstAddress: "EVAABm9hXKHk2MpVMbwNakRubFnNha5m8m", + }; + testNetwork(done, params); +}); +it('Allows selection of Europecoin', function(done) { + var params = { + selectText: "ERC - Europecoin", + firstAddress: "ESA2YwPYntAoaPrE8Fm5qkKRtkcwLcwD6R", + }; + testNetwork(done, params); +}); +it('Allows selection of Exclusivecoin', function(done) { + var params = { + selectText: "EXCL - Exclusivecoin", + firstAddress: "EbUa6m8UZW6nTxsYZD2FsDjkadKbp5M6JT", + }; + testNetwork(done, params); +}); +it('Allows selection of Feathercoin', function(done) { + var params = { + selectText: "FTC - Feathercoin", + firstAddress: "6gDdjAMoSgQaW8UhqK3oboHs6ftGAroKkM", + }; + testNetwork(done, params); +}); +it('Allows selection of Firstcoin', function(done) { + var params = { + selectText: "FRST - Firstcoin", + firstAddress: "FJN9GzfMm7Q8R4DJwK1H9F6A1GTghvFiMJ", + }; + testNetwork(done, params); +}); +it('Allows selection of Flashcoin', function(done) { + var params = { + selectText: "FLASH - Flashcoin", + firstAddress: "UWfpf5LfMmLxZYooEb2EyvWhZ8NG7EZDRt", + }; + testNetwork(done, params); +}); +it('Allows selection of GCRCoin', function(done) { + var params = { + selectText: "GCR - GCRCoin", + firstAddress: "GJjF5cLwyXLacpuvXAVksxGxKvHDjx58d6", + }; + testNetwork(done, params); +}); +it('Allows selection of Gobyte', function(done) { + var params = { + selectText: "GBX - Gobyte", + firstAddress: "GS813Ys2brkmvSUw1rUqGPm2HqQVDHJRyA", + }; + testNetwork(done, params); +}); +it('Allows selection of Gridcoin', function(done) { + var params = { + selectText: "GRC - Gridcoin", + firstAddress: "SGrWbBPvobgqKRF8td1Kdc9vbRY7MJ78Y9", + }; + testNetwork(done, params); +}); +it('Allows selection of Gulden', function(done) { + var params = { + selectText: "NLG - Gulden", + firstAddress: "GcDP7cNEc33MPPdTFNJ8pZc6VMZJ2CbKxY", + }; + testNetwork(done, params); +}); +it('Allows selection of Helleniccoin', function(done) { + var params = { + selectText: "HNC - Helleniccoin", + firstAddress: "LbHEKe5H72zp9G1fuWNiiNePTUfJb88915", + }; + testNetwork(done, params); +}); +it('Allows selection of Hempcoin', function(done) { + var params = { + selectText: "THC - Hempcoin", + firstAddress: "H8sdWbZyJV4gyXyHtLXDaNnAuUDhK5mfTV", + }; + testNetwork(done, params); +}); +it('Allows selection of Insane', function(done) { + var params = { + selectText: "INSN - Insane", + firstAddress: "iMPqEJMiXWuxC9U2NVinCCMr4t72h58EWx", + }; + testNetwork(done, params); +}); +it('Allows selection of Iop', function(done) { + var params = { + selectText: "IOP - Iop", + firstAddress: "pGKQmcaPf95Ur5o6oHK4qdiZ52p1yaTvq1", + }; + testNetwork(done, params); +}); +it('Allows selection of Ixcoin', function(done) { + var params = { + selectText: "IXC - Ixcoin", + firstAddress: "xgE9bTZ6YypT3E6ByzkTt31Hq68E9BqywH", + }; + testNetwork(done, params); +}); +it('Allows selection of Kobocoin', function(done) { + var params = { + selectText: "KOBO - Kobocoin", + firstAddress: "FTVoNJETXDAM8x7MnmdE8RwWndSr9PQWhy", + }; + testNetwork(done, params); +}); +it('Allows selection of Landcoin', function(done) { + var params = { + selectText: "LDCN - Landcoin", + firstAddress: "LLvLwNjG1aJcn1RS4W4GJUbv8fNaRATG7c", + }; + testNetwork(done, params); +}); +it('Allows selection of Library Credits', function(done) { + var params = { + selectText: "LBC - Library Credits", + firstAddress: "bQJEQrHDJyHdqycB32uysh1SWn8Ln8LMdg", + }; + testNetwork(done, params); +}); +it('Allows selection of Linx', function(done) { + var params = { + selectText: "LINX - Linx", + firstAddress: "XGWQ3cb3LGUB3VnHmj6xYSMgnokNbf6dyk", + }; + testNetwork(done, params); +}); +it('Allows selection of Litecoincash', function(done) { + var params = { + selectText: "LCC - Litecoincash", + firstAddress: "Ce5n7fjUuQPLutJ4W5nCCfQLKdKLE1mv9A", + }; + testNetwork(done, params); +}); +it('Allows selection of Lynx', function(done) { + var params = { + selectText: "LYNX - Lynx", + firstAddress: "KUeY3ZdZkg96p4W98pj1JjygCFU1XqWdw3", + }; + testNetwork(done, params); +}); +it('Allows selection of Minexcoin', function(done) { + var params = { + selectText: "MNX - Minexcoin", + firstAddress: "XC1VnyJVfiMDwWgFtAHDp41cgY3AHk3dJT", + }; + testNetwork(done, params); +}); +it('Allows selection of Navcoin', function(done) { + var params = { + selectText: "NAV - Navcoin", + firstAddress: "NTQVTPK3NWSQLKoffkiQw99T8PifkF1Y2U", + }; + testNetwork(done, params); +}); +it('Allows selection of Neoscoin', function(done) { + var params = { + selectText: "NEOS - Neoscoin", + firstAddress: "NgATz6QbQNXvayHQ4CpZayugb9HeaPDdby", + }; + testNetwork(done, params); +}); +it('Allows selection of Neurocoin', function(done) { + var params = { + selectText: "NRO - Neurocoin", + firstAddress: "NVdYErQ3mFpDuF5DquW9WMiT7sLc8ufFTn", + }; + testNetwork(done, params); +}); +it('Allows selection of Newyorkc', function(done) { + var params = { + selectText: "NYC - Newyorkc", + firstAddress: "RSVMfyH1fKfy3puADJEhut2vfkRyon6imm", + }; + testNetwork(done, params); +}); +it('Allows selection of Novacoin', function(done) { + var params = { + selectText: "NVC - Novacoin", + firstAddress: "4JRvUmxcKCJmaMXZyvRoSS1cmG2XvnZfHN", + }; + testNetwork(done, params); +}); +it('Allows selection of Nushares', function(done) { + var params = { + selectText: "NSR - Nushares", + firstAddress: "SecjXzU3c7EecdT7EbC4vvmbdtBBokWh6J", + }; + testNetwork(done, params); +}); +it('Allows selection of Okcash', function(done) { + var params = { + selectText: "OK - Okcash", + firstAddress: "PV4Qp1TUYuGv4TqVtLZtqvrsWWRycfx1Yi", + }; + testNetwork(done, params); +}); +it('Allows selection of Omnicore', function(done) { + var params = { + selectText: "OMNI - Omnicore", + firstAddress: "1Q1t3gonjCT3rW38TsTsCvgSc3hh7zBGbi", + }; + testNetwork(done, params); +}); +it('Allows selection of Pesobit', function(done) { + var params = { + selectText: "PSB - Pesobit", + firstAddress: "PDePsF7ALyXP7JaywokdYiRTDtKa14MAr1", + }; + testNetwork(done, params); +}); +it('Allows selection of Pinkcoin', function(done) { + var params = { + selectText: "PINK - Pinkcoin", + firstAddress: "2TgjYQffjbzUHJghNaVbdsjHbRwruC3yzC", + }; + testNetwork(done, params); +}); +it('Allows selection of POSWcoin', function(done) { + var params = { + selectText: "POSW - POSWcoin", + firstAddress: "PNxewmZoPnGBvoEbH6hgQZCK1igDiBCdgC", + }; + testNetwork(done, params); +}); +it('Allows selection of Potcoin', function(done) { + var params = { + selectText: "POT - Potcoin", + firstAddress: "PEo7Vg2ctXgpP4vuLPeY9aGJtZotyrmiHc", + }; + testNetwork(done, params); +}); +it('Allows selection of Putincoin', function(done) { + var params = { + selectText: "PUT - Putincoin", + firstAddress: "PViWnfr2uFtovd6e7joM49C94CsGSnqJis", + }; + testNetwork(done, params); +}); +it('Allows selection of Reddcoin', function(done) { + var params = { + selectText: "RDD - Reddcoin", + firstAddress: "RtgRvXMBng1y51ftteveFqwNfyRG18HpxQ", + }; + testNetwork(done, params); +}); +it('Allows selection of RevolutionVR', function(done) { + var params = { + selectText: "RVR - RevolutionVR", + firstAddress: "VXeeoP2jkzZnMFxtc66ZBZK1NHN5QJnnjL", + }; + testNetwork(done, params); +}); +it('Allows selection of Rubycoin', function(done) { + var params = { + selectText: "RBY - Rubycoin", + firstAddress: "RV76JDtjTs11JdMDRToYn6CHecMRPLnKS6", + }; + testNetwork(done, params); +}); +it('Allows selection of Smileycoin', function(done) { + var params = { + selectText: "SMLY - Smileycoin", + firstAddress: "BEZVnEBCAyFByrgKpwAgYgtvP4rKAd9Sj2", + }; + testNetwork(done, params); +}); +it('Allows selection of Solarcoin', function(done) { + var params = { + selectText: "SLR - Solarcoin", + firstAddress: "8LZ13HbnjtaMJWSvvVFNTLf71zFfDrhwLu", + }; + testNetwork(done, params); +}); +it('Allows selection of Stratis', function(done) { + var params = { + selectText: "STRAT - Stratis", + firstAddress: "ScfJnq3QDhKgDMEds6sqUE1ot6ShfhmXXq", + }; + testNetwork(done, params); +}); +it('Allows selection of Syscoin', function(done) { + var params = { + selectText: "SYS - Syscoin", + firstAddress: "SZwJi42Pst3VAMomyK5DG4157WM5ofRmSj", + }; + testNetwork(done, params); +}); +it('Allows selection of Toa', function(done) { + var params = { + selectText: "TOA - Toa", + firstAddress: "TSe1QAnUwQzUfbBusDzRJ9URttrRGKoNKF", + }; + testNetwork(done, params); +}); +it('Allows selection of Ultimatesecurecash', function(done) { + var params = { + selectText: "USC - Ultimatesecurecash", + firstAddress: "UPyLAZU2Che5fiy7Ed8xVJFmXAUhitA4ug", + }; + testNetwork(done, params); +}); +it('Allows selection of Unobtanium', function(done) { + var params = { + selectText: "UNO - Unobtanium", + firstAddress: "uUBMPVMXrR6qhqornJqKTWgr8L69vihSL9", + }; + testNetwork(done, params); +}); +it('Allows selection of Vcash', function(done) { + var params = { + selectText: "XVC - Vcash", + firstAddress: "VuL53MSY6KjvAjKSeRkh3NDnKykacDVeps", + }; + testNetwork(done, params); +}); +it('Allows selection of Verge', function(done) { + var params = { + selectText: "XVG - Verge", + firstAddress: "DCrVuGkMjLJpTGgwAgv9AcMdeb1nkWbjZA", + }; + testNetwork(done, params); +}); +it('Allows selection of Vertcoin', function(done) { + var params = { + selectText: "VTC - Vertcoin", + firstAddress: "Vf6koGuiWdXQfx8tNqxoNeEDxh4xh5cxsG", + }; + testNetwork(done, params); +}); +it('Allows selection of Vivo', function(done) { + var params = { + selectText: "VIVO - Vivo", + firstAddress: "VFmBwuXXGhJe7MarQG2GfzHMFebRHgfSpB", + }; + testNetwork(done, params); +}); +it('Allows selection of Vpncoin', function(done) { + var params = { + selectText: "VASH - Vpncoin", + firstAddress: "VoEmH1qXC4TsSgBAStR21QYetwnFqbqCx9", + }; + testNetwork(done, params); +}); +it('Allows selection of Whitecoin', function(done) { + var params = { + selectText: "XWC - Whitecoin", + firstAddress: "WcSwCAUqrSgeSYbsaS3SSWWhsx8KRYTFDR", + }; + testNetwork(done, params); +}); +it('Allows selection of Wincoin', function(done) { + var params = { + selectText: "WC - Wincoin", + firstAddress: "WaDVCESMGgyKgNESdn3u43NnwmGSkZED3Z", + }; + testNetwork(done, params); +}); +it('Allows selection of Zcoin', function(done) { + var params = { + selectText: "XZC - Zcoin", + firstAddress: "a6VcMdP4XgAA9Tr7xNszmPG5FZpfRf17Cq", + }; + testNetwork(done, params); +}); +it('Allows selection of Zcash', function(done) { + var params = { + selectText: "ZEC - Zcash", + firstAddress: "t1Sz8AneMcVuzUg3tPJ8et5AS5LFJ7K2EF9", + }; + testNetwork(done, params); +}); +it('Allows selection of Zclassic', function(done) { + var params = { + selectText: "ZCL - Zclassic", + firstAddress: "t1TBMxTvVJRybUbMLGWq8H4A8F4VUL7czEc", + }; + testNetwork(done, params); +}); +it('Allows selection of Zencash', function(done) { + var params = { + selectText: "ZEN - Zencash", + firstAddress: "znWh9XASyW2dZq5tck84wFjiwuqVysi7q3p", + }; + testNetwork(done, params); +}); +it('Allows selection of Energi', function(done) { + var params = { + selectText: "NRG - Energi", + firstAddress: "EejRy4t4nidzhGGzkJUgFP3z4HYBjhTsRt", + }; + testNetwork(done, params); +}); + + +// BIP39 seed is set from phrase +it('Sets the bip39 seed from the prhase', function(done) { + driver.findElement(By.css('.phrase')) + .sendKeys('abandon abandon ability'); + driver.sleep(generateDelay).then(function() { + driver.findElement(By.css('.seed')) + .getAttribute("value") + .then(function(seed) { + expect(seed).toBe("20da140d3dd1df8713cefcc4d54ce0e445b4151027a1ab567b832f6da5fcc5afc1c3a3f199ab78b8e0ab4652efd7f414ac2c9a3b81bceb879a70f377aa0a58f3"); + done(); + }) + }); +}); + +// BIP32 root key is set from phrase +it('Sets the bip39 root key from the prhase', function(done) { + driver.findElement(By.css('.phrase')) + .sendKeys('abandon abandon ability'); + driver.sleep(generateDelay).then(function() { + driver.findElement(By.css('.root-key')) + .getAttribute("value") + .then(function(seed) { + expect(seed).toBe("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi"); + done(); + }) + }); +}); + +// Tabs show correct addresses when changed +it('Shows the correct address when tab is changed', function(done) { + driver.findElement(By.css('.phrase')) + .sendKeys('abandon abandon ability'); + driver.sleep(generateDelay).then(function() { + driver.findElement(By.css('#bip32-tab a')) + .click(); + driver.sleep(generateDelay).then(function() { + getFirstAddress(function(address) { + expect(address).toBe("17uQ7s2izWPwBmEVFikTmZUjbBKWYdJchz"); + done(); + }); + }); + }); +}); + +// BIP44 derivation path is shown +it('Shows the derivation path for bip44 tab', function(done) { + driver.findElement(By.css('.phrase')) + .sendKeys('abandon abandon ability'); + driver.sleep(generateDelay).then(function() { + driver.findElement(By.css('#bip44 .path')) + .getAttribute("value") + .then(function(path) { + expect(path).toBe("m/44'/0'/0'/0"); + done(); + }) + }); +}); + +// BIP44 extended private key is shown +it('Shows the extended private key for bip44 tab', function(done) { + driver.findElement(By.css('.phrase')) + .sendKeys('abandon abandon ability'); + driver.sleep(generateDelay).then(function() { + driver.findElement(By.css('.extended-priv-key')) + .getAttribute("value") + .then(function(path) { + expect(path).toBe("xprvA2DxxvPZcyRvYgZMGS53nadR32mVDeCyqQYyFhrCVbJNjPoxMeVf7QT5g7mQASbTf9Kp4cryvcXnu2qurjWKcrdsr91jXymdCDNxKgLFKJG"); + done(); + }) + }); +}); + +// BIP44 extended public key is shown +it('Shows the extended public key for bip44 tab', function(done) { + driver.findElement(By.css('.phrase')) + .sendKeys('abandon abandon ability'); + driver.sleep(generateDelay).then(function() { + driver.findElement(By.css('.extended-pub-key')) + .getAttribute("value") + .then(function(path) { + expect(path).toBe("xpub6FDKNRvTTLzDmAdpNTc49ia9b4byd6vqCdUa46Fp3vqMcC96uBoufCmZXQLiN5AK3iSCJMhf9gT2sxkpyaPepRuA7W3MujV5tGmF5VfbueM"); + done(); + }) + }); +}); + +// BIP44 account field changes address list +it('Changes the address list if bip44 account is changed', function(done) { + driver.findElement(By.css('#bip44 .account')) + .sendKeys('1'); + driver.findElement(By.css('.phrase')) + .sendKeys('abandon abandon ability'); + driver.sleep(generateDelay).then(function() { + getFirstAddress(function(address) { + expect(address).toBe("1Nq2Wmu726XHCuGhctEtGmhxo3wzk5wZ1H"); + done(); + }); + }); +}); + +// BIP44 change field changes address list +it('Changes the address list if bip44 change is changed', function(done) { + driver.findElement(By.css('#bip44 .change')) + .sendKeys('1'); + driver.findElement(By.css('.phrase')) + .sendKeys('abandon abandon ability'); + driver.sleep(generateDelay).then(function() { + getFirstAddress(function(address) { + expect(address).toBe("1KAGfWgqfVbSSXY56fNQ7YnhyKuoskHtYo"); + done(); + }); + }); +}); + +// BIP32 derivation path can be set +it('Can use a custom bip32 derivation path', function(done) { + driver.findElement(By.css('#bip32-tab a')) + .click(); + driver.findElement(By.css('#bip32 .path')) + .clear(); + driver.findElement(By.css('#bip32 .path')) + .sendKeys('m/1'); + driver.findElement(By.css('.phrase')) + .sendKeys('abandon abandon ability'); + driver.sleep(generateDelay).then(function() { + getFirstAddress(function(address) { + expect(address).toBe("16pYQQdLD1hH4hwTGLXBaZ9Teboi1AGL8L"); + done(); + }); + }); +}); + +// BIP32 can use hardened derivation paths +it('Can use a hardened derivation paths', function(done) { + driver.findElement(By.css('#bip32-tab a')) + .click(); + driver.findElement(By.css('#bip32 .path')) + .clear(); + driver.findElement(By.css('#bip32 .path')) + .sendKeys("m/0'"); + driver.findElement(By.css('.phrase')) + .sendKeys('abandon abandon ability'); + driver.sleep(generateDelay).then(function() { + getFirstAddress(function(address) { + expect(address).toBe("14aXZeprXAE3UUKQc4ihvwBvww2LuEoHo4"); + done(); + }); + }); +}); + +// BIP32 extended private key is shown +it('Shows the BIP32 extended private key', function(done) { + driver.findElement(By.css('#bip32-tab a')) + .click(); + driver.findElement(By.css('.phrase')) + .sendKeys('abandon abandon ability'); + driver.sleep(generateDelay).then(function() { + driver.findElement(By.css('.extended-priv-key')) + .getAttribute("value") + .then(function(privKey) { + expect(privKey).toBe("xprv9va99uTVE5aLiutUVLTyfxfe8v8aaXjSQ1XxZbK6SezYVuikA9MnjQVTA8rQHpNA5LKvyQBpLiHbBQiiccKiBDs7eRmBogsvq3THFeLHYbe"); + done(); + }); + }); +}); + +// BIP32 extended public key is shown +it('Shows the BIP32 extended public key', function(done) { + driver.findElement(By.css('#bip32-tab a')) + .click(); + driver.findElement(By.css('.phrase')) + .sendKeys('abandon abandon ability'); + driver.sleep(generateDelay).then(function() { + driver.findElement(By.css('.extended-pub-key')) + .getAttribute("value") + .then(function(pubKey) { + expect(pubKey).toBe("xpub69ZVZQzP4T8dwPxwbMzz36cNgwy4yzTHmETZMyihzzXXNi3thgg3HCow1RtY252wdw5rS8369xKnraN5Q93y3FkFfJp2XEHWUrkyXsjS93P"); + done(); + }); + }); +}); + +// Derivation path is shown in table +it('Shows the derivation path in the table', function(done) { + driver.findElement(By.css('.phrase')) + .sendKeys('abandon abandon ability'); + driver.sleep(generateDelay).then(function() { + getFirstPath(function(path) { + expect(path).toBe("m/44'/0'/0'/0/0"); + done(); + }); + }); +}); + +// Derivation path for address can be hardened +it('Can derive hardened addresses', function(done) { + driver.findElement(By.css('#bip32-tab a')) + .click(); + driver.executeScript(function() { + $(".hardened-addresses").prop("checked", true); + }); + driver.findElement(By.css('.phrase')) + .sendKeys('abandon abandon ability'); + driver.sleep(generateDelay).then(function() { + getFirstAddress(function(address) { + expect(address).toBe("18exLzUv7kfpiXRzmCjFDoC9qwNLFyvwyd"); + done(); + }); + }); +}); + +// Derivation path visibility can be toggled +it('Can toggle visibility of the derivation path column', function(done) { + driver.findElement(By.css('.phrase')) + .sendKeys('abandon abandon ability'); driver.sleep(generateDelay).then(function() { driver.findElement(By.css('.index-toggle')) .click(); @@ -905,13 +1537,13 @@ it('Can generate more rows in the table', function(done) { // A custom number of additional addresses can be generated it('Can generate more rows in the table', function(done) { - driver.findElement(By.css('.rows-to-add')) - .clear(); - driver.findElement(By.css('.rows-to-add')) - .sendKeys('1'); driver.findElement(By.css('.phrase')) .sendKeys('abandon abandon ability'); driver.sleep(generateDelay).then(function() { + driver.findElement(By.css('.rows-to-add')) + .clear(); + driver.findElement(By.css('.rows-to-add')) + .sendKeys('1'); driver.findElement(By.css('.more')) .click(); driver.sleep(generateDelay).then(function() { @@ -1856,228 +2488,595 @@ it('Truncates entropy from the left', function(done) { }); }); -// Very large entropy results in very long mnemonics -it('Converts very long entropy to very long mnemonics', function(done) { - var entropy = ""; - for (var i=0; i<33; i++) { - entropy += "AAAAAAAA"; // 3 words * 33 iterations = 99 words - } +// Very large entropy results in very long mnemonics +it('Converts very long entropy to very long mnemonics', function(done) { + var entropy = ""; + for (var i=0; i<33; i++) { + entropy += "AAAAAAAA"; // 3 words * 33 iterations = 99 words + } + driver.findElement(By.css('.use-entropy')) + .click(); + driver.findElement(By.css('.entropy')) + .sendKeys(entropy); + driver.sleep(generateDelay).then(function() { + driver.findElement(By.css(".phrase")) + .getAttribute("value").then(function(phrase) { + var wordCount = phrase.split(/\s+/g).length; + expect(wordCount).toBe(99); + done(); + }); + }); +}); + +// Is compatible with bip32jp entropy +// https://bip32jp.github.io/english/index.html +// NOTES: +// Is incompatible with: +// base 20 +it('Is compatible with bip32jp.github.io', function(done) { + var entropy = "543210543210543210543210543210543210543210543210543210543210543210543210543210543210543210543210543"; + 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"; + driver.findElement(By.css('.use-entropy')) + .click(); + driver.findElement(By.css('.entropy')) + .sendKeys(entropy); + driver.sleep(generateDelay).then(function() { + driver.findElement(By.css(".phrase")) + .getAttribute("value").then(function(phrase) { + expect(phrase).toBe(expectedPhrase); + done(); + }); + }); +}); + +// Blank entropy does not generate mnemonic or addresses +it('Does not generate mnemonic for blank entropy', function(done) { + driver.findElement(By.css('.use-entropy')) + .click(); + driver.findElement(By.css('.entropy')) + .clear(); + // check there is no mnemonic + driver.sleep(generateDelay).then(function() { + driver.findElement(By.css(".phrase")) + .getAttribute("value").then(function(phrase) { + expect(phrase).toBe(""); + // check there is no mnemonic + driver.findElements(By.css(".address")) + .then(function(addresses) { + expect(addresses.length).toBe(0); + // Check the feedback says 'blank entropy' + driver.findElement(By.css(".feedback")) + .getText() + .then(function(feedbackText) { + expect(feedbackText).toBe("Blank entropy"); + done(); + }); + }) + }); + }); +}); + +// Mnemonic length can be selected even for weak entropy +it('Allows selection of mnemonic length even for weak entropy', function(done) { + driver.findElement(By.css('.use-entropy')) + .click(); + driver.executeScript(function() { + $(".mnemonic-length").val("18").trigger("change"); + }); + driver.findElement(By.css('.entropy')) + .sendKeys("012345"); + driver.sleep(generateDelay).then(function() { + driver.findElement(By.css(".phrase")) + .getAttribute("value").then(function(phrase) { + var wordCount = phrase.split(/\s+/g).length; + expect(wordCount).toBe(18); + done(); + }); + }); +}); + +// Github issue 33 +// https://github.com/iancoleman/bip39/issues/33 +// Final cards should contribute entropy +it('Uses as much entropy as possible for the mnemonic', function(done) { + driver.findElement(By.css('.use-entropy')) + .click(); + driver.findElement(By.css('.entropy')) + .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"); + driver.sleep(generateDelay).then(function() { + // Get mnemonic + driver.findElement(By.css(".phrase")) + .getAttribute("value").then(function(originalPhrase) { + // Set the last 12 cards to be AS + driver.findElement(By.css('.entropy')) + .clear(); + driver.findElement(By.css('.entropy')) + .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"); + driver.sleep(generateDelay).then(function() { + // Get new mnemonic + driver.findElement(By.css(".phrase")) + .getAttribute("value").then(function(newPhrase) { + expect(originalPhrase).not.toEqual(newPhrase); + done(); + }); + }); + }); + }); +}); + +// Github issue 35 +// https://github.com/iancoleman/bip39/issues/35 +// QR Code support +// TODO this doesn't work in selenium with firefox +// see https://stackoverflow.com/q/40360223 +it('Shows a qr code on hover for the phrase', function(done) { + if (browser == "firefox") { + pending("Selenium + Firefox bug for mouseMove, see https://stackoverflow.com/q/40360223"); + } + // generate a random mnemonic + var generateEl = driver.findElement(By.css('.generate')); + generateEl.click(); + // toggle qr to show (hidden by default) + var phraseEl = driver.findElement(By.css(".phrase")); + phraseEl.click(); + var rootKeyEl = driver.findElement(By.css(".root-key")); + driver.sleep(generateDelay).then(function() { + // hover over the root key + driver.actions().mouseMove(rootKeyEl).perform().then(function() { + // check the qr code shows + driver.executeScript(function() { + return $(".qr-container").find("canvas").length > 0; + }) + .then(function(qrShowing) { + expect(qrShowing).toBe(true); + // hover away from the phrase + driver.actions().mouseMove(generateEl).perform().then(function() {; + // check the qr code hides + driver.executeScript(function() { + return $(".qr-container").find("canvas").length == 0; + }) + .then(function(qrHidden) { + expect(qrHidden).toBe(true); + done(); + }); + }); + }); + }); + }); +}); + +// BIP44 account extendend private key is shown +// github issue 37 - compatibility with electrum +it('Shows the bip44 account extended private key', function(done) { + driver.findElement(By.css(".phrase")) + .sendKeys("abandon abandon ability"); + driver.sleep(generateDelay).then(function() { + driver.findElement(By.css("#bip44 .account-xprv")) + .getAttribute("value") + .then(function(xprv) { + expect(xprv).toBe("xprv9yzrnt4zWVJUr1k2VxSPy9ettKz5PpeDMgaVG7UKedhqnw1tDkxP2UyYNhuNSumk2sLE5ctwKZs9vwjsq3e1vo9egCK6CzP87H2cVYXpfwQ"); + done(); + }); + }); +}); + +// BIP44 account extendend public key is shown +// github issue 37 - compatibility with electrum +it('Shows the bip44 account extended public key', function(done) { + driver.findElement(By.css(".phrase")) + .sendKeys("abandon abandon ability"); + driver.sleep(generateDelay).then(function() { + driver.findElement(By.css("#bip44 .account-xpub")) + .getAttribute("value") + .then(function(xprv) { + expect(xprv).toBe("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf"); + done(); + }); + }); +}); + +// github issue 40 +// BIP32 root key can be set as an xpub +it('Generates addresses from xpub as bip32 root key', function(done) { + driver.findElement(By.css('#bip32-tab a')) + .click(); + // set xpub for account 0 of bip44 for 'abandon abandon ability' + driver.findElement(By.css("#root-key")) + .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf"); + driver.sleep(generateDelay).then(function() { + // check the addresses are generated + getFirstAddress(function(address) { + expect(address).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug"); + // check the xprv key is not set + driver.findElement(By.css(".extended-priv-key")) + .getAttribute("value") + .then(function(xprv) { + expect(xprv).toBe("NA"); + // check the private key is not set + driver.findElements(By.css(".privkey")) + .then(function(els) { + els[0] + .getText() + .then(function(privkey) { + expect(xprv).toBe("NA"); + done(); + }); + }); + }); + }); + }); +}); + +// github issue 40 +// xpub for bip32 root key will not work with hardened derivation paths +it('Shows error for hardened derivation paths with xpub root key', function(done) { + // set xpub for account 0 of bip44 for 'abandon abandon ability' + driver.findElement(By.css("#root-key")) + .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf"); + driver.sleep(feedbackDelay).then(function() { + // Check feedback is correct + driver.findElement(By.css('.feedback')) + .getText() + .then(function(feedback) { + var msg = "Hardened derivation path is invalid with xpub key"; + expect(feedback).toBe(msg); + // Check no addresses are shown + driver.findElements(By.css('.addresses tr')) + .then(function(rows) { + expect(rows.length).toBe(0); + done(); + }); + }); + }); +}); + +// github issue 39 +// no root key shows feedback +it('Shows feedback for no root key', function(done) { + // set xpub for account 0 of bip44 for 'abandon abandon ability' + driver.findElement(By.css('#bip32-tab a')) + .click(); + driver.sleep(feedbackDelay).then(function() { + // Check feedback is correct + driver.findElement(By.css('.feedback')) + .getText() + .then(function(feedback) { + expect(feedback).toBe("Invalid root key"); + done(); + }); + }); +}); + +// Github issue 44 +// display error switching tabs while addresses are generating +it('Can change details while old addresses are still being generated', function(done) { + // Set to generate 199 more addresses. + // This will take a long time allowing a new set of addresses to be + // generated midway through this lot. + // The newly generated addresses should not include any from the old set. + // Any more than 199 will show an alert which needs to be accepted. + driver.findElement(By.css('.rows-to-add')) + .clear(); + driver.findElement(By.css('.rows-to-add')) + .sendKeys('199'); + // set the prhase + driver.findElement(By.css('.phrase')) + .sendKeys("abandon abandon ability"); + driver.sleep(generateDelay).then(function() { + // change tabs which should cancel the previous generating + driver.findElement(By.css('.rows-to-add')) + .clear(); + driver.findElement(By.css('.rows-to-add')) + .sendKeys('20'); + driver.findElement(By.css('#bip32-tab a')) + .click() + driver.sleep(generateDelay).then(function() { + driver.findElements(By.css('.index')) + .then(function(els) { + // check the derivation paths have the right quantity + expect(els.length).toBe(20); + // check the derivation paths are in order + testRowsAreInCorrectOrder(done); + }); + }); + }); +}, generateDelay + 5000); + +// Github issue 49 +// padding for binary should give length with multiple of 256 +// hashed entropy 1111 is length 252, so requires 4 leading zeros +// prior to issue 49 it would only generate 2 leading zeros, ie missing 2 +it('Pads hashed entropy with leading zeros', function(done) { driver.findElement(By.css('.use-entropy')) .click(); + driver.executeScript(function() { + $(".mnemonic-length").val("15").trigger("change"); + }); driver.findElement(By.css('.entropy')) - .sendKeys(entropy); + .sendKeys("1111"); driver.sleep(generateDelay).then(function() { - driver.findElement(By.css(".phrase")) - .getAttribute("value").then(function(phrase) { - var wordCount = phrase.split(/\s+/g).length; - expect(wordCount).toBe(99); + driver.findElement(By.css('.phrase')) + .getAttribute("value") + .then(function(phrase) { + expect(phrase).toBe("avocado valid quantum cross link predict excuse edit street able flame large galaxy ginger nuclear"); done(); }); }); }); -// Is compatible with bip32jp entropy -// https://bip32jp.github.io/english/index.html -// NOTES: -// Is incompatible with: -// base 20 -it('Is compatible with bip32jp.github.io', function(done) { - var entropy = "543210543210543210543210543210543210543210543210543210543210543210543210543210543210543210543210543"; - 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"; - driver.findElement(By.css('.use-entropy')) - .click(); - driver.findElement(By.css('.entropy')) - .sendKeys(entropy); +// Github pull request 55 +// https://github.com/iancoleman/bip39/pull/55 +// Client select +it('Can set the derivation path on bip32 tab for bitcoincore', function(done) { + testClientSelect(done, { + selectValue: "0", + bip32path: "m/0'/0'", + useHardenedAddresses: "true", + }); +}); +it('Can set the derivation path on bip32 tab for multibit', function(done) { + testClientSelect(done, { + selectValue: "2", + bip32path: "m/0'/0", + useHardenedAddresses: null, + }); +}); + +// github issue 58 +// https://github.com/iancoleman/bip39/issues/58 +// bip32 derivation is correct, does not drop leading zeros +// see also +// https://medium.com/@alexberegszaszi/why-do-my-bip32-wallets-disagree-6f3254cc5846 +it('Retains leading zeros for bip32 derivation', function(done) { + driver.findElement(By.css(".phrase")) + .sendKeys("fruit wave dwarf banana earth journey tattoo true farm silk olive fence"); + driver.findElement(By.css(".passphrase")) + .sendKeys("banana"); driver.sleep(generateDelay).then(function() { - driver.findElement(By.css(".phrase")) - .getAttribute("value").then(function(phrase) { - expect(phrase).toBe(expectedPhrase); + getFirstAddress(function(address) { + // Note that bitcore generates an incorrect address + // 13EuKhffWkBE2KUwcbkbELZb1MpzbimJ3Y + // see the medium.com link above for more details + expect(address).toBe("17rxURoF96VhmkcEGCj5LNQkmN9HVhWb7F"); + done(); + }); + }); +}); + +// github issue 60 +// Japanese mnemonics generate incorrect bip32 seed +// BIP39 seed is set from phrase +it('Generates correct seed for Japanese mnemonics', function(done) { + driver.findElement(By.css(".phrase")) + .sendKeys("あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あおぞら"); + driver.findElement(By.css(".passphrase")) + .sendKeys("メートルガバヴァぱばぐゞちぢ十人十色"); + driver.sleep(generateDelay).then(function() { + driver.findElement(By.css(".seed")) + .getAttribute("value") + .then(function(seed) { + expect(seed).toBe("a262d6fb6122ecf45be09c50492b31f92e9beb7d9a845987a02cefda57a15f9c467a17872029a9e92299b5cbdf306e3a0ee620245cbd508959b6cb7ca637bd55"); done(); }); }); }); -// Blank entropy does not generate mnemonic or addresses -it('Does not generate mnemonic for blank entropy', function(done) { - driver.findElement(By.css('.use-entropy')) +// BIP49 official test vectors +// https://github.com/bitcoin/bips/blob/master/bip-0049.mediawiki#test-vectors +it('Generates BIP49 addresses matching the official test vectors', function(done) { + driver.findElement(By.css('#bip49-tab a')) .click(); - driver.findElement(By.css('.entropy')) - .clear(); - // check there is no mnemonic + selectNetwork("BTC - Bitcoin Testnet"); + driver.findElement(By.css(".phrase")) + .sendKeys("abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about"); driver.sleep(generateDelay).then(function() { - driver.findElement(By.css(".phrase")) - .getAttribute("value").then(function(phrase) { - expect(phrase).toBe(""); - // check there is no mnemonic - driver.findElements(By.css(".address")) - .then(function(addresses) { - expect(addresses.length).toBe(0); - // Check the feedback says 'blank entropy' - driver.findElement(By.css(".feedback")) - .getText() - .then(function(feedbackText) { - expect(feedbackText).toBe("Blank entropy"); - done(); - }); - }) - }); + getFirstAddress(function(address) { + expect(address).toBe("2Mww8dCYPUpKHofjgcXcBCEGmniw9CoaiD2"); + done(); + }); }); }); -// Mnemonic length can be selected even for weak entropy -it('Allows selection of mnemonic length even for weak entropy', function(done) { - driver.findElement(By.css('.use-entropy')) +// BIP49 derivation path is shown +it('Shows the bip49 derivation path', function(done) { + driver.findElement(By.css('#bip49-tab a')) .click(); - driver.executeScript(function() { - $(".mnemonic-length").val("18").trigger("change"); - }); - driver.findElement(By.css('.entropy')) - .sendKeys("012345"); + driver.findElement(By.css(".phrase")) + .sendKeys("abandon abandon ability"); driver.sleep(generateDelay).then(function() { - driver.findElement(By.css(".phrase")) - .getAttribute("value").then(function(phrase) { - var wordCount = phrase.split(/\s+/g).length; - expect(wordCount).toBe(18); + driver.findElement(By.css('#bip49 .path')) + .getAttribute("value") + .then(function(path) { + expect(path).toBe("m/49'/0'/0'/0"); done(); }); }); }); -// Github issue 33 -// https://github.com/iancoleman/bip39/issues/33 -// Final cards should contribute entropy -it('Uses as much entropy as possible for the mnemonic', function(done) { - driver.findElement(By.css('.use-entropy')) +// BIP49 extended private key is shown +it('Shows the bip49 extended private key', function(done) { + driver.findElement(By.css('#bip49-tab a')) .click(); - driver.findElement(By.css('.entropy')) - .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"); + driver.findElement(By.css(".phrase")) + .sendKeys("abandon abandon ability"); driver.sleep(generateDelay).then(function() { - // Get mnemonic - driver.findElement(By.css(".phrase")) - .getAttribute("value").then(function(originalPhrase) { - // Set the last 12 cards to be AS - driver.findElement(By.css('.entropy')) - .clear(); - driver.findElement(By.css('.entropy')) - .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"); - driver.sleep(generateDelay).then(function() { - // Get new mnemonic - driver.findElement(By.css(".phrase")) - .getAttribute("value").then(function(newPhrase) { - expect(originalPhrase).not.toEqual(newPhrase); - done(); - }); - }); + driver.findElement(By.css('.extended-priv-key')) + .getAttribute("value") + .then(function(xprv) { + expect(xprv).toBe("yprvALYB4DYRG6CzzVgzQZwwqjAA2wjBGC3iEd7KYYScpoDdmf75qMRWZWxoFcRXBJjgEXdFqJ9vDRGRLJQsrL22Su5jMbNFeM9vetaGVqy9Qy2"); + done(); }); }); }); -// Github issue 35 -// https://github.com/iancoleman/bip39/issues/35 -// QR Code support -// TODO this doesn't work in selenium with firefox -// see https://stackoverflow.com/q/40360223 -it('Shows a qr code on hover for the phrase', function(done) { - if (browser == "firefox") { - pending("Selenium + Firefox bug for mouseMove, see https://stackoverflow.com/q/40360223"); - } - // generate a random mnemonic - var generateEl = driver.findElement(By.css('.generate')); - generateEl.click(); - // toggle qr to show (hidden by default) - var phraseEl = driver.findElement(By.css(".phrase")); - phraseEl.click(); - var rootKeyEl = driver.findElement(By.css(".root-key")); +// BIP49 extended public key is shown +it('Shows the bip49 extended public key', function(done) { + driver.findElement(By.css('#bip49-tab a')) + .click(); + driver.findElement(By.css(".phrase")) + .sendKeys("abandon abandon ability"); driver.sleep(generateDelay).then(function() { - // hover over the root key - driver.actions().mouseMove(rootKeyEl).perform().then(function() { - // check the qr code shows - driver.executeScript(function() { - return $(".qr-container").find("canvas").length > 0; - }) - .then(function(qrShowing) { - expect(qrShowing).toBe(true); - // hover away from the phrase - driver.actions().mouseMove(generateEl).perform().then(function() {; - // check the qr code hides - driver.executeScript(function() { - return $(".qr-container").find("canvas").length == 0; - }) - .then(function(qrHidden) { - expect(qrHidden).toBe(true); - done(); - }); - }); + driver.findElement(By.css('.extended-pub-key')) + .getAttribute("value") + .then(function(xprv) { + expect(xprv).toBe("ypub6ZXXTj5K6TmJCymTWbUxCs6tayZffemZbr2vLvrEP8kceTSENtjm7KHH6thvAKxVar9fGe8rgsPEX369zURLZ68b4f7Vexz7RuXsjQ69YDt"); + done(); }); + }); +}); + +// BIP49 account field changes address list +it('Can set the bip49 account field', function(done) { + driver.findElement(By.css('#bip49-tab a')) + .click(); + driver.findElement(By.css('#bip49 .account')) + .clear(); + driver.findElement(By.css('#bip49 .account')) + .sendKeys("1"); + driver.findElement(By.css(".phrase")) + .sendKeys("abandon abandon ability"); + driver.sleep(generateDelay).then(function() { + getFirstAddress(function(address) { + expect(address).toBe("381wg1GGN4rP88rNC9v7QWsiww63yLVPsn"); + done(); }); }); }); -// BIP44 account extendend private key is shown -// github issue 37 - compatibility with electrum -it('Shows the bip44 account extended private key', function(done) { +// BIP49 change field changes address list +it('Can set the bip49 change field', function(done) { + driver.findElement(By.css('#bip49-tab a')) + .click(); + driver.findElement(By.css('#bip49 .change')) + .clear(); + driver.findElement(By.css('#bip49 .change')) + .sendKeys("1"); driver.findElement(By.css(".phrase")) .sendKeys("abandon abandon ability"); driver.sleep(generateDelay).then(function() { - driver.findElement(By.css("#bip44 .account-xprv")) + getFirstAddress(function(address) { + expect(address).toBe("3PEM7MiKed5konBoN66PQhK8r3hjGhy9dT"); + done(); + }); + }); +}); + +// BIP49 account extendend private key is shown +it('Shows the bip49 account extended private key', function(done) { + driver.findElement(By.css('#bip49-tab a')) + .click(); + driver.findElement(By.css(".phrase")) + .sendKeys("abandon abandon ability"); + driver.sleep(generateDelay).then(function() { + driver.findElement(By.css('#bip49 .account-xprv')) .getAttribute("value") .then(function(xprv) { - expect(xprv).toBe("xprv9yzrnt4zWVJUr1k2VxSPy9ettKz5PpeDMgaVG7UKedhqnw1tDkxP2UyYNhuNSumk2sLE5ctwKZs9vwjsq3e1vo9egCK6CzP87H2cVYXpfwQ"); + expect(xprv).toBe("yprvAHtB1M5Wp675aLzFy9TJYK2mSsLkg6mcBRh5DZTR7L4EnYSmYPaL63KFA4ycg1PngW5LfkmejxzosCs17TKZMpRFKc3z5SJar6QAKaFcaZL"); done(); - }); + }); }); }); -// BIP44 account extendend public key is shown -// github issue 37 - compatibility with electrum -it('Shows the bip44 account extended public key', function(done) { +// BIP49 account extendend public key is shown +it('Shows the bip49 account extended public key', function(done) { + driver.findElement(By.css('#bip49-tab a')) + .click(); driver.findElement(By.css(".phrase")) .sendKeys("abandon abandon ability"); driver.sleep(generateDelay).then(function() { - driver.findElement(By.css("#bip44 .account-xpub")) + driver.findElement(By.css('#bip49 .account-xpub')) .getAttribute("value") .then(function(xprv) { - expect(xprv).toBe("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf"); + expect(xprv).toBe("ypub6WsXQrcQeTfNnq4j5AzJuSyVzuBF5ZVTYecg1ws2ffbDfLmv5vtadqdj1NgR6C6gufMpMfJpHxvb6JEQKvETVNWCRanNedfJtnTchZiJtsL"); done(); - }); + }); }); }); -// github issue 40 -// BIP32 root key can be set as an xpub -it('Generates addresses from xpub as bip32 root key', function(done) { - driver.findElement(By.css('#bip32-tab a')) +// Test selecting coin where bip49 is unavailable (eg CLAM) +it('Shows an error on bip49 tab for coins without bip49', function(done) { + driver.findElement(By.css('#bip49-tab a')) .click(); - // set xpub for account 0 of bip44 for 'abandon abandon ability' - driver.findElement(By.css("#root-key")) - .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf"); + driver.findElement(By.css(".phrase")) + .sendKeys("abandon abandon ability"); driver.sleep(generateDelay).then(function() { - // check the addresses are generated - getFirstAddress(function(address) { - expect(address).toBe("1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug"); - // check the xprv key is not set - driver.findElement(By.css(".extended-priv-key")) - .getAttribute("value") - .then(function(xprv) { - expect(xprv).toBe("NA"); - // check the private key is not set - driver.findElements(By.css(".privkey")) - .then(function(els) { - els[0] - .getText() - .then(function(privkey) { - expect(xprv).toBe("NA"); - done(); - }); - }); + selectNetwork("CLAM - Clams"); + // bip49 available is hidden + driver.findElement(By.css('#bip49 .available')) + .getAttribute("class") + .then(function(classes) { + expect(classes).toContain("hidden"); + // bip49 unavailable is shown + driver.findElement(By.css('#bip49 .unavailable')) + .getAttribute("class") + .then(function(classes) { + expect(classes).not.toContain("hidden"); + // check there are no addresses shown + driver.findElements(By.css('.addresses tr')) + .then(function(rows) { + expect(rows.length).toBe(0); + // check the derived private key is blank + driver.findElement(By.css('.extended-priv-key')) + .getAttribute("value") + .then(function(xprv) { + expect(xprv).toBe(''); + // check the derived public key is blank + driver.findElement(By.css('.extended-pub-key')) + .getAttribute("value") + .then(function(xpub) { + expect(xpub).toBe(''); + done(); + }); + }); + }) + }); + }); + }); +}); + +// github issue 43 +// Cleared mnemonic and root key still allows addresses to be generated +// https://github.com/iancoleman/bip39/issues/43 +it('Clears old root keys from memory when mnemonic is cleared', function(done) { + // set the phrase + driver.findElement(By.css(".phrase")) + .sendKeys("abandon abandon ability"); + driver.sleep(generateDelay).then(function() { + // clear the mnemonic and root key + // using selenium .clear() doesn't seem to trigger the 'input' event + // so clear it using keys instead + driver.findElement(By.css('.phrase')) + .sendKeys(Key.CONTROL,"a"); + driver.findElement(By.css('.phrase')) + .sendKeys(Key.DELETE); + driver.findElement(By.css('.root-key')) + .sendKeys(Key.CONTROL,"a"); + driver.findElement(By.css('.root-key')) + .sendKeys(Key.DELETE); + driver.sleep(generateDelay).then(function() { + // try to generate more addresses + driver.findElement(By.css('.more')) + .click(); + driver.sleep(generateDelay).then(function() { + driver.findElements(By.css(".addresses tr")) + .then(function(els) { + // check there are no addresses shown + expect(els.length).toBe(0); + done(); + }); }); - }); + }); }); }); -// github issue 40 -// xpub for bip32 root key will not work with hardened derivation paths -it('Shows error for hardened derivation paths with xpub root key', function(done) { +// Github issue 95 +// error trying to generate addresses from xpub with hardened derivation +it('Shows error for hardened addresses with xpub root key', function(done) { + driver.findElement(By.css('#bip32-tab a')) + .click() + driver.executeScript(function() { + $(".hardened-addresses").prop("checked", true); + }); // set xpub for account 0 of bip44 for 'abandon abandon ability' driver.findElement(By.css("#root-key")) .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf"); @@ -2088,476 +3087,614 @@ it('Shows error for hardened derivation paths with xpub root key', function(done .then(function(feedback) { var msg = "Hardened derivation path is invalid with xpub key"; expect(feedback).toBe(msg); - // Check no addresses are shown - driver.findElements(By.css('.addresses tr')) - .then(function(rows) { - expect(rows.length).toBe(0); - done(); - }); + done(); + }); + }); +}); + +// Litecoin uses ltub by default, and can optionally be set to xprv +// github issue 96 +// https://github.com/iancoleman/bip39/issues/96 +// Issue with extended keys on Litecoin +it('Uses ltub by default for litecoin, but can be set to xprv', function(done) { + driver.findElement(By.css('.phrase')) + .sendKeys("abandon abandon ability"); + selectNetwork("LTC - Litecoin"); + driver.sleep(generateDelay).then(function() { + // check the extended key is generated correctly + driver.findElement(By.css('.root-key')) + .getAttribute("value") + .then(function(rootKey) { + expect(rootKey).toBe("Ltpv71G8qDifUiNesiPqf6h5V6eQ8ic77oxQiYtawiACjBEx3sTXNR2HGDGnHETYxESjqkMLFBkKhWVq67ey1B2MKQXannUqNy1RZVHbmrEjnEU"); + // set litecoin to use ltub + driver.executeScript(function() { + $(".litecoin-use-ltub").prop("checked", false); + $(".litecoin-use-ltub").trigger("change"); + }); + driver.sleep(generateDelay).then(function() { + driver.findElement(By.css('.root-key')) + .getAttribute("value") + .then(function(rootKey) { + expect(rootKey).toBe("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi"); + done(); + }); + }) + }); + }); +}); + +// github issue 99 +// https://github.com/iancoleman/bip39/issues/99#issuecomment-327094159 +// "warn me emphatically when they have detected invalid input" to the entropy field +// A warning is shown when entropy is filtered and discarded +it('Warns when entropy is filtered and discarded', function(done) { + driver.findElement(By.css('.use-entropy')) + .click(); + // set entropy to have no filtered content + driver.findElement(By.css('.entropy')) + .sendKeys("00000000 00000000 00000000 00000000"); + driver.sleep(generateDelay).then(function() { + // check the filter warning does not show + driver.findElement(By.css('.entropy-container .filter-warning')) + .getAttribute("class") + .then(function(classes) { + expect(classes).toContain("hidden"); + // set entropy to have some filtered content + driver.findElement(By.css('.entropy')) + .sendKeys("10000000 zxcvbn 00000000 00000000 00000000"); + driver.sleep(entropyFeedbackDelay).then(function() { + // check the filter warning shows + driver.findElement(By.css('.entropy-container .filter-warning')) + .getAttribute("class") + .then(function(classes) { + expect(classes).not.toContain("hidden"); + done(); + }); + }); }); }); }); -// github issue 39 -// no root key shows feedback -it('Shows feedback for no root key', function(done) { - // set xpub for account 0 of bip44 for 'abandon abandon ability' - driver.findElement(By.css('#bip32-tab a')) - .click(); - driver.sleep(feedbackDelay).then(function() { - // Check feedback is correct - driver.findElement(By.css('.feedback')) - .getText() - .then(function(feedback) { - expect(feedback).toBe("Invalid root key"); - done(); - }); +// Bitcoin Cash address can be set to use cashaddr format +it('Can use cashaddr format for bitcoin cash addresses', function(done) { + driver.executeScript(function() { + $(".use-bch-cashaddr-addresses").prop("checked", true); + }); + driver.findElement(By.css('.phrase')) + .sendKeys("abandon abandon ability"); + selectNetwork("BCH - Bitcoin Cash"); + driver.sleep(generateDelay).then(function() { + getFirstAddress(function(address) { + expect(address).toBe("bitcoincash:qzlquk7w4hkudxypl4fgv8x279r754dkvur7jpcsps"); + done(); + }); + }); +}); + +// Bitcoin Cash address can be set to use bitpay format +it('Can use bitpay format for bitcoin cash addresses', function(done) { + driver.executeScript(function() { + $(".use-bch-bitpay-addresses").prop("checked", true); + }); + driver.findElement(By.css('.phrase')) + .sendKeys("abandon abandon ability"); + selectNetwork("BCH - Bitcoin Cash"); + driver.sleep(generateDelay).then(function() { + getFirstAddress(function(address) { + expect(address).toBe("CZnpA9HPmvhuhLLPWJP8rNDpLUYXy1LXFk"); + done(); + }); + }); +}); + +// Bitcoin Cash address can be set to use legacy format +it('Can use legacy format for bitcoin cash addresses', function(done) { + driver.executeScript(function() { + $(".use-bch-legacy-addresses").prop("checked", true); + }); + driver.findElement(By.css('.phrase')) + .sendKeys("abandon abandon ability"); + selectNetwork("BCH - Bitcoin Cash"); + driver.sleep(generateDelay).then(function() { + getFirstAddress(function(address) { + expect(address).toBe("1JKvb6wKtsjNoCRxpZ4DGrbniML7z5U16A"); + done(); + }); }); }); -// Github issue 44 -// display error switching tabs while addresses are generating -it('Can change details while old addresses are still being generated', function(done) { - // Set to generate 199 more addresses. - // This will take a long time allowing a new set of addresses to be - // generated midway through this lot. - // The newly generated addresses should not include any from the old set. - // Any more than 199 will show an alert which needs to be accepted. - driver.findElement(By.css('.rows-to-add')) - .clear(); - driver.findElement(By.css('.rows-to-add')) - .sendKeys('199'); - // set the prhase +// End of tests ported from old suit, so no more comments above each test now + +it('Can generate more addresses from a custom index', function(done) { + var expectedIndexes = [ + 0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19, + 40,41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59 + ]; driver.findElement(By.css('.phrase')) .sendKeys("abandon abandon ability"); driver.sleep(generateDelay).then(function() { - // generate more addresses + // Set start of next lot of rows to be from index 40 + // which means indexes 20-39 will not be in the table. + driver.findElement(By.css('.more-rows-start-index')) + .sendKeys("40"); driver.findElement(By.css('.more')) .click(); - // change tabs which should cancel the previous generating - driver.findElement(By.css('#bip32-tab a')) - .click() driver.sleep(generateDelay).then(function() { - driver.findElements(By.css('.index')) + // Check actual indexes in the table match the expected pattern + driver.findElements(By.css(".index")) .then(function(els) { - // check the derivation paths have the right quantity - expect(els.length).toBe(20); - // check the derivation paths are in order - testRowsAreInCorrectOrder(done); + expect(els.length).toBe(expectedIndexes.length); + var testRowAtIndex = function(i) { + if (i >= expectedIndexes.length) { + done(); + } + else { + els[i].getText() + .then(function(actualPath) { + var noHardened = actualPath.replace(/'/g, ""); + var pathBits = noHardened.split("/") + var lastBit = pathBits[pathBits.length-1]; + var actualIndex = parseInt(lastBit); + var expectedIndex = expectedIndexes[i]; + expect(actualIndex).toBe(expectedIndex); + testRowAtIndex(i+1); + }); + } + } + testRowAtIndex(0); }); }); }); }); -// Github issue 49 -// padding for binary should give length with multiple of 256 -// hashed entropy 1111 is length 252, so requires 4 leading zeros -// prior to issue 49 it would only generate 2 leading zeros, ie missing 2 -it('Pads hashed entropy with leading zeros', function(done) { - driver.findElement(By.css('.use-entropy')) +it('Can generate BIP141 addresses with P2WPKH-in-P2SH semanitcs', function(done) { + // Sourced from BIP49 official test specs + driver.findElement(By.css('#bip141-tab a')) .click(); - driver.executeScript(function() { - $(".mnemonic-length").val("15").trigger("change"); - }); - driver.findElement(By.css('.entropy')) - .sendKeys("1111"); + driver.findElement(By.css('.bip141-path')) + .clear(); + driver.findElement(By.css('.bip141-path')) + .sendKeys("m/49'/1'/0'/0"); + selectNetwork("BTC - Bitcoin Testnet"); + driver.findElement(By.css(".phrase")) + .sendKeys("abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about"); driver.sleep(generateDelay).then(function() { - driver.findElement(By.css('.phrase')) - .getAttribute("value") - .then(function(phrase) { - expect(phrase).toBe("avocado valid quantum cross link predict excuse edit street able flame large galaxy ginger nuclear"); - done(); - }); + getFirstAddress(function(address) { + expect(address).toBe("2Mww8dCYPUpKHofjgcXcBCEGmniw9CoaiD2"); + done(); + }); }); }); -// Github pull request 55 -// https://github.com/iancoleman/bip39/pull/55 -// Client select -it('Can set the derivation path on bip32 tab for bitcoincore', function(done) { - testClientSelect(done, { - selectValue: "0", - bip32path: "m/0'/0'", - useHardenedAddresses: "true", - }); -}); -it('Can set the derivation path on bip32 tab for multibit', function(done) { - testClientSelect(done, { - selectValue: "2", - bip32path: "m/0'/0", - useHardenedAddresses: null, +it('Can generate BIP141 addresses with P2WPKH semanitcs', function(done) { + // This result tested against bitcoinjs-lib test spec for segwit address + // using the first private key of this mnemonic and default path m/0 + // https://github.com/bitcoinjs/bitcoinjs-lib/blob/9c8503cab0c6c30a95127042703bc18e8d28c76d/test/integration/addresses.js#L50 + // so whilst not directly comparable, substituting the private key produces + // identical results between this tool and the bitcoinjs-lib test. + // Private key generated is: + // L3L8Nu9whawPBNLGtFqDhKut9DKKfG3CQoysupT7BimqVCZsLFNP + driver.findElement(By.css('#bip141-tab a')) + .click(); + // Choose P2WPKH + driver.executeScript(function() { + $(".bip141-semantics option[selected]").removeAttr("selected"); + $(".bip141-semantics option").filter(function(i,e) { + return $(e).html() == "P2WPKH"; + }).prop("selected", true); + $(".bip141-semantics").trigger("change"); }); -}); - -// github issue 58 -// https://github.com/iancoleman/bip39/issues/58 -// bip32 derivation is correct, does not drop leading zeros -// see also -// https://medium.com/@alexberegszaszi/why-do-my-bip32-wallets-disagree-6f3254cc5846 -it('Retains leading zeros for bip32 derivation', function(done) { driver.findElement(By.css(".phrase")) - .sendKeys("fruit wave dwarf banana earth journey tattoo true farm silk olive fence"); - driver.findElement(By.css(".passphrase")) - .sendKeys("banana"); + .sendKeys("abandon abandon ability"); driver.sleep(generateDelay).then(function() { getFirstAddress(function(address) { - // Note that bitcore generates an incorrect address - // 13EuKhffWkBE2KUwcbkbELZb1MpzbimJ3Y - // see the medium.com link above for more details - expect(address).toBe("17rxURoF96VhmkcEGCj5LNQkmN9HVhWb7F"); + expect(address).toBe("bc1qfwu6a5a3evygrk8zvdxxvz4547lmpyx5vsfxe9"); done(); }); }); }); -// github issue 60 -// Japanese mnemonics generate incorrect bip32 seed -// BIP39 seed is set from phrase -it('Generates correct seed for Japanese mnemonics', function(done) { - driver.findElement(By.css(".phrase")) - .sendKeys("あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あいこくしん あおぞら"); - driver.findElement(By.css(".passphrase")) - .sendKeys("メートルガバヴァぱばぐゞちぢ十人十色"); +it('Shows the entropy used by the PRNG when clicking generate', function(done) { + driver.findElement(By.css('.generate')).click(); driver.sleep(generateDelay).then(function() { - driver.findElement(By.css(".seed")) + driver.findElement(By.css('.entropy')) .getAttribute("value") - .then(function(seed) { - expect(seed).toBe("a262d6fb6122ecf45be09c50492b31f92e9beb7d9a845987a02cefda57a15f9c467a17872029a9e92299b5cbdf306e3a0ee620245cbd508959b6cb7ca637bd55"); + .then(function(entropy) { + expect(entropy).not.toBe(""); done(); }); }); }); -// BIP49 official test vectors -// https://github.com/bitcoin/bips/blob/master/bip-0049.mediawiki#test-vectors -it('Generates BIP49 addresses matching the official test vectors', function(done) { - driver.findElement(By.css('#bip49-tab a')) - .click(); - selectNetwork("BTC - Bitcoin Testnet"); - driver.findElement(By.css(".phrase")) - .sendKeys("abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about"); +it('Shows the index of each word in the mnemonic', function(done) { + driver.findElement(By.css('.phrase')) + .sendKeys("abandon abandon ability"); driver.sleep(generateDelay).then(function() { - getFirstAddress(function(address) { - expect(address).toBe("2Mww8dCYPUpKHofjgcXcBCEGmniw9CoaiD2"); - done(); - }); + driver.findElement(By.css('.use-entropy')) + .click(); + driver.findElement(By.css('.word-indexes')) + .getText() + .then(function(indexes) { + expect(indexes).toBe("0, 0, 1"); + done(); + }); }); }); -// BIP49 derivation path is shown -it('Shows the bip49 derivation path', function(done) { - driver.findElement(By.css('#bip49-tab a')) - .click(); - driver.findElement(By.css(".phrase")) - .sendKeys("abandon abandon ability"); +it('Shows the derivation path for bip84 tab', function(done) { + driver.findElement(By.css('#bip84-tab a')) + .click() + driver.findElement(By.css('.phrase')) + .sendKeys('abandon abandon ability'); driver.sleep(generateDelay).then(function() { - driver.findElement(By.css('#bip49 .path')) + driver.findElement(By.css('#bip84 .path')) .getAttribute("value") .then(function(path) { - expect(path).toBe("m/49'/0'/0'/0"); + expect(path).toBe("m/84'/0'/0'/0"); done(); - }); + }) }); }); -// BIP49 extended private key is shown -it('Shows the bip49 extended private key', function(done) { - driver.findElement(By.css('#bip49-tab a')) - .click(); - driver.findElement(By.css(".phrase")) - .sendKeys("abandon abandon ability"); +it('Shows the extended private key for bip84 tab', function(done) { + driver.findElement(By.css('#bip84-tab a')) + .click() + driver.findElement(By.css('.phrase')) + .sendKeys('abandon abandon ability'); driver.sleep(generateDelay).then(function() { driver.findElement(By.css('.extended-priv-key')) .getAttribute("value") - .then(function(xprv) { - expect(xprv).toBe("yprvALYB4DYRG6CzzVgzQZwwqjAA2wjBGC3iEd7KYYScpoDdmf75qMRWZWxoFcRXBJjgEXdFqJ9vDRGRLJQsrL22Su5jMbNFeM9vetaGVqy9Qy2"); + .then(function(path) { + expect(path).toBe("zprvAev3RKrZ3QVKiUFCfdeMRen1BPDJgdNt1XpxiDy8acSs4kkAGTCvq7HeRYRNNpo8EtEjCFQBWavJwtCUR29y4TUCH4X5RXMcyq48uN8y9BP"); done(); - }); + }) }); }); -// BIP49 extended public key is shown -it('Shows the bip49 extended public key', function(done) { - driver.findElement(By.css('#bip49-tab a')) - .click(); - driver.findElement(By.css(".phrase")) - .sendKeys("abandon abandon ability"); +it('Shows the extended public key for bip84 tab', function(done) { + driver.findElement(By.css('#bip84-tab a')) + .click() + driver.findElement(By.css('.phrase')) + .sendKeys('abandon abandon ability'); driver.sleep(generateDelay).then(function() { driver.findElement(By.css('.extended-pub-key')) .getAttribute("value") - .then(function(xprv) { - expect(xprv).toBe("ypub6ZXXTj5K6TmJCymTWbUxCs6tayZffemZbr2vLvrEP8kceTSENtjm7KHH6thvAKxVar9fGe8rgsPEX369zURLZ68b4f7Vexz7RuXsjQ69YDt"); + .then(function(path) { + expect(path).toBe("zpub6suPpqPSsn3cvxKfmfBMnnijjR3o666jNkkZWcNk8wyqwZ5JozXBNuc8Gs7DB3uLwTDvGVTspVEAUQcEjKF3pZHgywVbubdTqbXTUg7usyx"); done(); - }); + }) }); }); -// BIP49 account field changes address list -it('Can set the bip49 account field', function(done) { - driver.findElement(By.css('#bip49-tab a')) - .click(); - driver.findElement(By.css('#bip49 .account')) - .clear(); - driver.findElement(By.css('#bip49 .account')) - .sendKeys("1"); - driver.findElement(By.css(".phrase")) - .sendKeys("abandon abandon ability"); +it('Changes the address list if bip84 account is changed', function(done) { + driver.findElement(By.css('#bip84-tab a')) + .click() + driver.findElement(By.css('#bip84 .account')) + .sendKeys('1'); + driver.findElement(By.css('.phrase')) + .sendKeys('abandon abandon ability'); + driver.sleep(generateDelay).then(function() { + getFirstAddress(function(address) { + expect(address).toBe("bc1qp7vv669t2fy965jdzvqwrraana89ctd5ewc662"); + done(); + }); + }); +}); + +it('Changes the address list if bip84 change is changed', function(done) { + driver.findElement(By.css('#bip84-tab a')) + .click() + driver.findElement(By.css('#bip84 .change')) + .sendKeys('1'); + driver.findElement(By.css('.phrase')) + .sendKeys('abandon abandon ability'); + driver.sleep(generateDelay).then(function() { + getFirstAddress(function(address) { + expect(address).toBe("bc1qr39vj6rh06ff05m53uxq8uazehwhccswylhrs2"); + done(); + }); + }); +}); + +it('Passes the official BIP84 test spec for rootpriv', function(done) { + driver.findElement(By.css('#bip84-tab a')) + .click() + driver.findElement(By.css('.phrase')) + .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about'); + driver.sleep(generateDelay).then(function() { + driver.findElement(By.css(".root-key")) + .getAttribute("value") + .then(function(rootKey) { + expect(rootKey).toBe("zprvAWgYBBk7JR8Gjrh4UJQ2uJdG1r3WNRRfURiABBE3RvMXYSrRJL62XuezvGdPvG6GFBZduosCc1YP5wixPox7zhZLfiUm8aunE96BBa4Kei5"); + done(); + }) + }); +}); + +it('Passes the official BIP84 test spec for account 0 xprv', function(done) { + driver.findElement(By.css('#bip84-tab a')) + .click() + driver.findElement(By.css('.phrase')) + .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about'); + driver.sleep(generateDelay).then(function() { + driver.findElement(By.css("#bip84 .account-xprv")) + .getAttribute("value") + .then(function(rootKey) { + expect(rootKey).toBe("zprvAdG4iTXWBoARxkkzNpNh8r6Qag3irQB8PzEMkAFeTRXxHpbF9z4QgEvBRmfvqWvGp42t42nvgGpNgYSJA9iefm1yYNZKEm7z6qUWCroSQnE"); + done(); + }) + }); +}); + +it('Passes the official BIP84 test spec for account 0 xpub', function(done) { + driver.findElement(By.css('#bip84-tab a')) + .click() + driver.findElement(By.css('.phrase')) + .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about'); + driver.sleep(generateDelay).then(function() { + driver.findElement(By.css("#bip84 .account-xpub")) + .getAttribute("value") + .then(function(rootKey) { + expect(rootKey).toBe("zpub6rFR7y4Q2AijBEqTUquhVz398htDFrtymD9xYYfG1m4wAcvPhXNfE3EfH1r1ADqtfSdVCToUG868RvUUkgDKf31mGDtKsAYz2oz2AGutZYs"); + done(); + }) + }); +}); + +it('Passes the official BIP84 test spec for account 0 first address', function(done) { + driver.findElement(By.css('#bip84-tab a')) + .click() + driver.findElement(By.css('.phrase')) + .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about'); + driver.sleep(generateDelay).then(function() { + getFirstAddress(function(address) { + expect(address).toBe("bc1qcr8te4kr609gcawutmrza0j4xv80jy8z306fyu"); + done(); + }); + }); +}); + +it('Passes the official BIP84 test spec for account 0 first change address', function(done) { + driver.findElement(By.css('#bip84-tab a')) + .click() + driver.findElement(By.css('.phrase')) + .sendKeys('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about'); + driver.findElement(By.css('#bip84 .change')) + .sendKeys('1'); driver.sleep(generateDelay).then(function() { getFirstAddress(function(address) { - expect(address).toBe("381wg1GGN4rP88rNC9v7QWsiww63yLVPsn"); + expect(address).toBe("bc1q8c6fshw2dlwun7ekn9qwf37cu2rn755upcp6el"); done(); }); }); }); -// BIP49 change field changes address list -it('Can set the bip49 change field', function(done) { - driver.findElement(By.css('#bip49-tab a')) - .click(); - driver.findElement(By.css('#bip49 .change')) +it('Can display the table as csv', function(done) { + var headings = "path,address,public key,private key"; + var row1 = "m/44'/0'/0'/0/0,1Di3Vp7tBWtyQaDABLAjfWtF6V7hYKJtug,033f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3,L26cVSpWFkJ6aQkPkKmTzLqTdLJ923e6CzrVh9cmx21QHsoUmrEE"; + var row20 = "m/44'/0'/0'/0/19,1KhBy28XLAciXnnRvm71PvQJaETyrxGV55,02b4b3e396434d8cdd20c03ac4aaa07387784d5d867b75987f516f5705ee68cb3a,L4GrDrjReMsCAu5DkLXn79jSb95qR7Zfx7eshybCQZ1qL32MXJab"; + driver.findElement(By.css('.phrase')) + .sendKeys('abandon abandon ability'); + driver.sleep(generateDelay).then(function() { + driver.findElement(By.css('.csv')) + .getAttribute("value") + .then(function(csv) { + expect(csv).toContain(headings); + expect(csv).toContain(row1); + expect(csv).toContain(row20); + done(); + }); + }); +}); + +it('LeftPads ethereum keys that are less than 32 bytes', function(done) { + // see https://github.com/iancoleman/bip39/issues/155 + selectNetwork("ETH - Ethereum"); + driver.findElement(By.css('#bip32-tab a')) + .click() + driver.findElement(By.css('#bip32-path')) .clear(); - driver.findElement(By.css('#bip49 .change')) - .sendKeys("1"); - driver.findElement(By.css(".phrase")) - .sendKeys("abandon abandon ability"); + driver.findElement(By.css('#bip32-path')) + .sendKeys("m/44'/60'/0'"); + driver.findElement(By.css('.phrase')) + .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'); driver.sleep(generateDelay).then(function() { getFirstAddress(function(address) { - expect(address).toBe("3PEM7MiKed5konBoN66PQhK8r3hjGhy9dT"); + expect(address).toBe("0x8943E785B4a5714FC87a3aFAad1eB1FeB602B118"); done(); }); }); }); -// BIP49 account extendend private key is shown -it('Shows the bip49 account extended private key', function(done) { - driver.findElement(By.css('#bip49-tab a')) - .click(); - driver.findElement(By.css(".phrase")) - .sendKeys("abandon abandon ability"); - driver.sleep(generateDelay).then(function() { - driver.findElement(By.css('#bip49 .account-xprv')) - .getAttribute("value") - .then(function(xprv) { - expect(xprv).toBe("yprvAHtB1M5Wp675aLzFy9TJYK2mSsLkg6mcBRh5DZTR7L4EnYSmYPaL63KFA4ycg1PngW5LfkmejxzosCs17TKZMpRFKc3z5SJar6QAKaFcaZL"); - done(); - }); +it('Can encrypt private keys using BIP38', function(done) { + // see https://github.com/iancoleman/bip39/issues/140 + driver.executeScript(function() { + $(".use-bip38").prop("checked", true); }); -}); + driver.findElement(By.css('.bip38-password')) + .sendKeys('bip38password'); + driver.findElement(By.css('.rows-to-add')) + .clear(); + driver.findElement(By.css('.rows-to-add')) + .sendKeys('1'); + driver.findElement(By.css('.phrase')) + .sendKeys('abandon abandon ability'); + driver.sleep(bip38delay).then(function() { + // address + getFirstRowValue(function(address) { + expect(address).toBe("1NCvSdumA3ngMM9c4aqU56AM6rqXddfuXB"); + // pubkey + getFirstRowValue(function(pubkey) { + expect(pubkey).toBe("043f5aed5f6cfbafaf223188095b5980814897295f723815fea5d3f4b648d0d0b3884a74447ea901729b1e73a999b7520e7cb55b4120e6432c64153ccab8a848e1"); + // privkey + getFirstRowValue(function(privkey) { + expect(privkey).toBe("6PRNRiFnj1RoR3sXhymdCvoZCgnUHQpfupNdKkFbWJkwWQEKesWt1EDMDM"); + done(); + }, ".privkey"); + }, ".pubkey"); + }, ".address"); + }); +}, bip38delay + 5000); -// BIP49 account extendend public key is shown -it('Shows the bip49 account extended public key', function(done) { - driver.findElement(By.css('#bip49-tab a')) +it('Shows the checksum for the entropy', function(done) { + driver.findElement(By.css('.use-entropy')) .click(); - driver.findElement(By.css(".phrase")) - .sendKeys("abandon abandon ability"); + driver.findElement(By.css('.entropy')) + .sendKeys("00000000000000000000000000000000"); driver.sleep(generateDelay).then(function() { - driver.findElement(By.css('#bip49 .account-xpub')) - .getAttribute("value") - .then(function(xprv) { - expect(xprv).toBe("ypub6WsXQrcQeTfNnq4j5AzJuSyVzuBF5ZVTYecg1ws2ffbDfLmv5vtadqdj1NgR6C6gufMpMfJpHxvb6JEQKvETVNWCRanNedfJtnTchZiJtsL"); + driver.findElement(By.css('.checksum')) + .getText() + .then(function(text) { + expect(text).toBe("1"); done(); }); }); }); -// Test selecting coin where bip49 is unavailable (eg CLAM) -it('Shows an error on bip49 tab for coins without bip49', function(done) { - driver.findElement(By.css('#bip49-tab a')) +it('Shows the checksum for the entropy with the correct groupings', function(done) { + driver.findElement(By.css('.use-entropy')) .click(); - driver.findElement(By.css(".phrase")) - .sendKeys("abandon abandon ability"); + // create a checksum of 20 bits, which spans multiple words + driver.findElement(By.css('.entropy')) + .sendKeys("F000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000"); driver.sleep(generateDelay).then(function() { - selectNetwork("CLAM - Clams"); - // bip49 available is hidden - driver.findElement(By.css('#bip49 .available')) - .getAttribute("class") - .then(function(classes) { - expect(classes).toContain("hidden"); - // bip49 unavailable is shown - driver.findElement(By.css('#bip49 .unavailable')) - .getAttribute("class") - .then(function(classes) { - expect(classes).not.toContain("hidden"); - // check there are no addresses shown - driver.findElements(By.css('.addresses tr')) - .then(function(rows) { - expect(rows.length).toBe(0); - // check the derived private key is blank - driver.findElement(By.css('.extended-priv-key')) - .getAttribute("value") - .then(function(xprv) { - expect(xprv).toBe(''); - // check the derived public key is blank - driver.findElement(By.css('.extended-pub-key')) - .getAttribute("value") - .then(function(xpub) { - expect(xpub).toBe(''); - done(); - }); - }); - }) - }); + driver.findElement(By.css('.checksum')) + .getText() + .then(function(text) { + // first group is 9 bits, second group is 11 + expect(text).toBe("011010111 01110000110"); + done(); }); }); }); -// github issue 43 -// Cleared mnemonic and root key still allows addresses to be generated -// https://github.com/iancoleman/bip39/issues/43 -it('Clears old root keys from memory when mnemonic is cleared', function(done) { - // set the phrase - driver.findElement(By.css(".phrase")) - .sendKeys("abandon abandon ability"); +it('Uses vprv for bitcoin testnet p2wpkh', function(done) { + selectNetwork("BTC - Bitcoin Testnet"); + driver.findElement(By.css('#bip84-tab a')) + .click() + driver.findElement(By.css('.phrase')) + .sendKeys('abandon abandon ability'); driver.sleep(generateDelay).then(function() { - // clear the mnemonic and root key - // using selenium .clear() doesn't seem to trigger the 'input' event - // so clear it using keys instead - driver.findElement(By.css('.phrase')) - .sendKeys(Key.CONTROL,"a"); - driver.findElement(By.css('.phrase')) - .sendKeys(Key.DELETE); - driver.findElement(By.css('.root-key')) - .sendKeys(Key.CONTROL,"a"); driver.findElement(By.css('.root-key')) - .sendKeys(Key.DELETE); - driver.sleep(generateDelay).then(function() { - // try to generate more addresses - driver.findElement(By.css('.more')) - .click(); - driver.sleep(generateDelay).then(function() { - driver.findElements(By.css(".addresses tr")) - .then(function(els) { - // check there are no addresses shown - expect(els.length).toBe(0); - done(); - }); - }); - }); + .getAttribute("value") + .then(function(path) { + expect(path).toBe("vprv9DMUxX4ShgxML9N2YV5CvWEebWrM9aJ5ULpbRRyzyWu6vs4BzTvbfFFrH41N5hVi7MYSfiugd765L3JmAfDM5po36Y8ouCKRDeYQwByCmS7"); + done(); + }) }); }); -// Github issue 95 -// error trying to generate addresses from xpub with hardened derivation -it('Shows error for hardened addresses with xpub root key', function(done) { - driver.findElement(By.css('#bip32-tab a')) - .click() +it('Shows a warning if generating weak mnemonics', function(done) { driver.executeScript(function() { - $(".hardened-addresses").prop("checked", true); - }); - // set xpub for account 0 of bip44 for 'abandon abandon ability' - driver.findElement(By.css("#root-key")) - .sendKeys("xpub6CzDCPbtLrrn4VpVbyyQLHbdSMpZoHN4iuW64VswCyEpfjM2mJGdaHJ2DyuZwtst96E16VvcERb8BBeJdHSCVmAq9RhtRQg6eAZFrTKCNqf"); - driver.sleep(feedbackDelay).then(function() { - // Check feedback is correct - driver.findElement(By.css('.feedback')) - .getText() - .then(function(feedback) { - var msg = "Hardened derivation path is invalid with xpub key"; - expect(feedback).toBe(msg); - done(); - }); + $(".strength option[selected]").removeAttr("selected"); + $(".strength option[value=6]").prop("selected", true); + $(".strength").trigger("change"); }); + driver.findElement(By.css(".generate-container .warning")) + .getAttribute("class") + .then(function(classes) { + expect(classes).not.toContain("hidden"); + done(); + }); }); -// Litecoin uses xprv by default, and can optionally be set to ltpv -// github issue 96 -// https://github.com/iancoleman/bip39/issues/96 -// Issue with extended keys on Litecoin -it('Uses xprv by default for litecoin, but can be set to ltpv', function(done) { - driver.findElement(By.css('.phrase')) - .sendKeys("abandon abandon ability"); - selectNetwork("LTC - Litecoin"); - driver.sleep(generateDelay).then(function() { - // check the extended key is generated correctly - driver.findElement(By.css('.root-key')) - .getAttribute("value") - .then(function(rootKey) { - expect(rootKey).toBe("xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi"); - // set litecoin to use ltub - driver.executeScript(function() { - $(".litecoin-use-ltub").prop("checked", true); - $(".litecoin-use-ltub").trigger("change"); - }); - driver.sleep(generateDelay).then(function() { - driver.findElement(By.css('.root-key')) - .getAttribute("value") - .then(function(rootKey) { - expect(rootKey).toBe("Ltpv71G8qDifUiNesiPqf6h5V6eQ8ic77oxQiYtawiACjBEx3sTXNR2HGDGnHETYxESjqkMLFBkKhWVq67ey1B2MKQXannUqNy1RZVHbmrEjnEU"); - done(); - }); - }) - }); +it('Does not show a warning if generating strong mnemonics', function(done) { + driver.executeScript(function() { + $(".strength option[selected]").removeAttr("selected"); + $(".strength option[value=12]").prop("selected", true); }); + driver.findElement(By.css(".generate-container .warning")) + .getAttribute("class") + .then(function(classes) { + expect(classes).toContain("hidden"); + done(); + }); }); -// BIP32 tab can use P2WPKH Nested In P2SH -// github issue 91 part 2 -// https://github.com/iancoleman/bip39/issues/91 -// generate new addresses from xpub? -it('Uses xprv by default for litecoin, but can be set to ltpv', function(done) { - // use p2wpkh addresses +it('Shows a warning if overriding weak entropy with longer mnemonics', function(done) { + driver.findElement(By.css('.use-entropy')) + .click(); + driver.findElement(By.css('.entropy')) + .sendKeys("0123456789abcdef"); // 6 words driver.executeScript(function() { - $(".p2wpkh-nested-in-p2sh").prop("checked", true); + $(".mnemonic-length").val("12").trigger("change"); }); - // use bip32 tab - driver.findElement(By.css('#bip32-tab a')) - .click() - // use testnet - selectNetwork("BTC - Bitcoin Testnet"); - // Set root xpub to BIP49 official test vector account 0 - driver.findElement(By.css('.root-key')) - .sendKeys("tpubDD7tXK8KeQ3YY83yWq755fHY2JW8Ha8Q765tknUM5rSvjPcGWfUppDFMpQ1ScziKfW3ZNtZvAD7M3u7bSs7HofjTD3KP3YxPK7X6hwV8Rk2"); - driver.sleep(generateDelay).then(function() { - getFirstAddress(function(address) { - expect(address).toBe("2Mww8dCYPUpKHofjgcXcBCEGmniw9CoaiD2"); + driver.findElement(By.css(".weak-entropy-override-warning")) + .getAttribute("class") + .then(function(classes) { + expect(classes).not.toContain("hidden"); done(); }); - }); }); -// github issue 99 -// https://github.com/iancoleman/bip39/issues/99#issuecomment-327094159 -// "warn me emphatically when they have detected invalid input" to the entropy field -// A warning is shown when entropy is filtered and discarded -it('Warns when entropy is filtered and discarded', function(done) { +it('Does not show a warning if entropy is stronger than mnemonic length', function(done) { driver.findElement(By.css('.use-entropy')) .click(); - // set entropy to have no filtered content driver.findElement(By.css('.entropy')) - .sendKeys("00000000 00000000 00000000 00000000"); + .sendKeys("0123456789abcdef0123456789abcdef0123456789abcdef"); // 18 words + driver.executeScript(function() { + $(".mnemonic-length").val("12").trigger("change"); + }); + driver.findElement(By.css(".weak-entropy-override-warning")) + .getAttribute("class") + .then(function(classes) { + expect(classes).toContain("hidden"); + done(); + }); +}); + +it('Shows a warning for litecoin BIP84 (which does not have p2wpkh params)', function(done) { + driver.findElement(By.css('.phrase')) + .sendKeys('abandon abandon ability'); + selectNetwork("LTC - Litecoin"); + driver.findElement(By.css('#bip84-tab a')) + .click() + // bip84 unavailable is shown + driver.sleep(feedbackDelay).then(function() { + driver.findElement(By.css('#bip84 .unavailable')) + .getAttribute("class") + .then(function(classes) { + expect(classes).not.toContain("hidden"); + done(); + }); + }); +}); + +it('Shows litecoin BIP49 addresses', function(done) { + driver.findElement(By.css('.phrase')) + .sendKeys('abandon abandon ability'); + selectNetwork("LTC - Litecoin"); + driver.findElement(By.css('#bip49-tab a')) + .click() + // bip49 addresses are shown driver.sleep(generateDelay).then(function() { - // check the filter warning does not show - driver.findElement(By.css('.entropy-container .filter-warning')) + driver.findElement(By.css('#bip49 .available')) .getAttribute("class") .then(function(classes) { - expect(classes).toContain("hidden"); - // set entropy to have some filtered content - driver.findElement(By.css('.entropy')) - .sendKeys("10000000 zxcvbn 00000000 00000000 00000000"); - driver.sleep(entropyFeedbackDelay).then(function() { - // check the filter warning shows - driver.findElement(By.css('.entropy-container .filter-warning')) - .getAttribute("class") - .then(function(classes) { - expect(classes).not.toContain("hidden"); - done(); - }); + expect(classes).not.toContain("hidden"); + // check first address + getFirstAddress(function(address) { + expect(address).toBe("MFwLPhsXoBuSLL8cLmW9uK6tChkzduV8qN"); + done(); }); }); }); }); -// Bitcoin Cash address can be set to use bitpay format -it('Can use bitpay format for bitcoin cash addresses', function(done) { - driver.executeScript(function() { - $(".use-bitpay-addresses").prop("checked", true); - }); - driver.findElement(By.css('.phrase')) - .sendKeys("abandon abandon ability"); - selectNetwork("BCH - Bitcoin Cash"); +it('Can use root keys to generate segwit table rows', function(done) { + // segwit uses ypub / zpub instead of xpub but the root key should still + // be valid regardless of the encoding used to import that key. + // Maybe this breaks the reason for the different extended key prefixes, but + // since the parsed root key is used behind the scenes anyhow this should be + // allowed. + driver.findElement(By.css('#root-key')) + .sendKeys('xprv9s21ZrQH143K2jkGDCeTLgRewT9F2pH5JZs2zDmmjXes34geVnFiuNa8KTvY5WoYvdn4Ag6oYRoB6cXtc43NgJAEqDXf51xPm6fhiMCKwpi'); + driver.findElement(By.css('#bip49-tab a')) + .click() + // bip49 addresses are shown driver.sleep(generateDelay).then(function() { getFirstAddress(function(address) { - expect(address).toBe("CZnpA9HPmvhuhLLPWJP8rNDpLUYXy1LXFk"); + expect(address).toBe("3QG2Y9AA4xZ846gKHZqNf7mvVKbLqMKxr2"); done(); }); });