English |
@@ -4318,7 +5206,9 @@
+
+
@@ -4417,6 +5307,11 @@
(/^c[123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz]{51}$/.test(key)) :
(/^[LK][123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz]{51}$/.test(key));
},
+ // 58 base58 characters starting with 6P
+ isBIP38Format: function (key) {
+ key = key.toString();
+ return (/^6P[123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz]{56}$/.test(key));
+ },
// 64 characters [0-9A-F]
isHexFormat: function (key) {
key = key.toString();
@@ -4469,6 +5364,120 @@
anyPrivateKeyToECKey: function (privKey) {
var bytes = ninja.privateKey.anyPrivateKeyToByteArray(privKey);
return (bytes == null)? null : new Bitcoin.ECKey(bytes);
+ },
+ BIP38EncryptedKeyToByteArrayAsync: function (base58Encrypted, passphrase, callback) {
+ var hex;
+ try {
+ hex = Bitcoin.Base58.decode(base58Encrypted);
+ } catch (e) {
+ callback(new Error(ninja.translator.get("detailalertnotvalidprivatekey")));
+ return;
+ }
+
+ if (hex.length != 43) {
+ callback(new Error(ninja.translator.get("detailalertnotvalidprivatekey")));
+ return;
+ } else if (hex[0] != 0x01) {
+ callback(new Error(ninja.translator.get("detailalertnotvalidprivatekey")));
+ return;
+ }
+
+ var expChecksum = hex.slice(-4);
+ hex = hex.slice(0, -4);
+
+ var checksum = Bitcoin.Util.dsha256(hex);
+ if (checksum[0] != expChecksum[0] || checksum[1] != expChecksum[1] || checksum[2] != expChecksum[2] || checksum[3] != expChecksum[3]) {
+ callback(new Error(ninja.translator.get("detailalertnotvalidprivatekey")));
+ return;
+ }
+
+ var isCompPoint = false;
+ var isECMult = false;
+ var hasLotSeq = false;
+ if (hex[1] == 0x42) {
+ if (hex[2] == 0xe0) {
+ isCompPoint = true;
+ } else if (hex[2] != 0xc0) {
+ callback(new Error(ninja.translator.get("detailalertnotvalidprivatekey")));
+ return;
+ }
+ } else if (hex[1] == 0x43) {
+ isECMult = true;
+ isCompPoint = (hex[2] & 0x20) != 0;
+ hasLotSeq = (hex[2] & 0x04) != 0;
+ if ((hex[2] & 0x24) != hex[2]) {
+ callback(new Error(ninja.translator.get("detailalertnotvalidprivatekey")));
+ return;
+ }
+ } else {
+ callback(new Error(ninja.translator.get("detailalertnotvalidprivatekey")));
+ return;
+ }
+
+ var decrypted;
+ var AES_opts = {mode: new Crypto.mode.ECB(Crypto.pad.NoPadding), asBytes: true};
+
+ var verifyHashAndReturn = function() {
+ var tmpkey = new Bitcoin.ECKey(decrypted);
+ var base58AddrText = isCompPoint ? tmpkey.getBitcoinAddressCompressed() : tmpkey.getBitcoinAddress();
+ checksum = Bitcoin.Util.dsha256(base58AddrText);
+
+ if (checksum[0] != hex[3] || checksum[1] != hex[4] || checksum[2] != hex[5] || checksum[3] != hex[6]) {
+ callback(new Error(ninja.translator.get("bip38alertincorrectpassphrase")));
+ return;
+ }
+
+ callback(tmpkey.getBitcoinPrivateKeyByteArray());
+ };
+
+ if (!isECMult) {
+ var addresshash = hex.slice(3, 7);
+ Crypto_scrypt(passphrase, addresshash, 16384, 8, 8, 64, function(derivedBytes) {
+ var k = derivedBytes.slice(32, 32+32);
+ decrypted = Crypto.AES.decrypt(hex.slice(7, 7+32), k, AES_opts);
+ for (var x = 0; x < 32; x++) decrypted[x] ^= derivedBytes[x];
+ verifyHashAndReturn();
+ });
+ } else {
+ var ownerentropy = hex.slice(7, 7+8);
+ var ownersalt = !hasLotSeq ? ownerentropy : ownerentropy.slice(0, 4);
+ Crypto_scrypt(passphrase, ownersalt, 16384, 8, 8, 32, function(prefactorA) {
+ var passfactor;
+ if (!hasLotSeq) {
+ passfactor = prefactorA;
+ } else {
+ var prefactorB = prefactorA.concat(ownerentropy);
+ passfactor = Bitcoin.Util.dsha256(prefactorB);
+ }
+ var kp = new Bitcoin.ECKey(passfactor);
+ var passpoint = kp.getPubCompressed();
+
+ var encryptedpart2 = hex.slice(23, 23+16);
+
+ var addresshashplusownerentropy = hex.slice(3, 3+12);
+ Crypto_scrypt(passpoint, addresshashplusownerentropy, 1024, 1, 1, 64, function(derived) {
+ var k = derived.slice(32);
+
+ var unencryptedpart2 = Crypto.AES.decrypt(encryptedpart2, k, AES_opts);
+ for (var i = 0; i < 16; i++) { unencryptedpart2[i] ^= derived[i+16]; }
+
+ var encryptedpart1 = hex.slice(15, 15+8).concat(unencryptedpart2.slice(0, 0+8));
+ var unencryptedpart1 = Crypto.AES.decrypt(encryptedpart1, k, AES_opts);
+ for (var i = 0; i < 16; i++) { unencryptedpart1[i] ^= derived[i]; }
+
+ var seedb = unencryptedpart1.slice(0, 0+16).concat(unencryptedpart2.slice(8, 8+8));
+
+ var factorb = Bitcoin.Util.dsha256(seedb);
+
+ var ps = EllipticCurve.getSECCurveByName("secp256k1");
+ var privateKey = BigInteger.fromByteArrayUnsigned(passfactor).multiply(BigInteger.fromByteArrayUnsigned(factorb)).remainder(ps.getN());
+
+ decrypted = privateKey.toByteArrayUnsigned();
+ verifyHashAndReturn();
+ });
+ });
+ }
+
}
};
@@ -4714,7 +5723,8 @@
"vanityalertinvalidinputpublickeysmatch": "Invalid input. The Public Key of both entries match. You must input two different keys.",
"vanityalertinvalidinputcannotmultiple": "Invalid input. Cannot multiple two public keys. Select 'Add' to add two public keys to get a bitcoin address.",
"vanityprivatekeyonlyavailable": "Only available when combining two private keys",
- "vanityalertinvalidinputprivatekeysmatch": "Invalid input. The Private Key of both entries match. You must input two different keys."
+ "vanityalertinvalidinputprivatekeysmatch": "Invalid input. The Private Key of both entries match. You must input two different keys.",
+ "bip38alertincorrectpassphrase": "Incorrect passphrase for this encrypted private key."
},
"es": {
@@ -4733,6 +5743,7 @@
"vanityalertinvalidinputcannotmultiple": "Invalid input. Cannot multiple two public keys. Select 'Add' to add two public keys to get a bitcoin address.", //TODO: please translate
"vanityprivatekeyonlyavailable": "Only available when combining two private keys", //TODO: please translate
"vanityalertinvalidinputprivatekeysmatch": "Invalid input. The Private Key of both entries match. You must input two different keys.", //TODO: please translate
+ "bip38alertincorrectpassphrase": "Incorrect passphrase for this encrypted private key.", //TODO: please translate
// header and menu html
"tagline": "Generador de carteras Bitcoin de código abierto en lado de cliente con Javascript",
@@ -4831,7 +5842,9 @@
"detailcompwifprefix": "'K' o 'L'",
"detaillabelprivhex": "Clave privada en formato hexadecimal (64 caracteres [0-9A-F]):",
"detaillabelprivb64": "Clave privada en base64 (44 caracteres):",
- "detaillabelprivmini": "Clave privada en formato mini (22, 26 o 30 caracteres, empieza por 'S'):"
+ "detaillabelprivmini": "Clave privada en formato mini (22, 26 o 30 caracteres, empieza por 'S'):",
+ "detaillabelpassphrase": "BIP38 Passphrase", // TODO: please translate,
+ "detaildecrypt": "Decrypt BIP38", // TODO: please translate
}
}
};
@@ -5326,7 +6339,7 @@
document.getElementById("detailarea").style.display = "none";
},
- viewDetails: function () {
+ viewDetails: function (bip38) {
var key = document.getElementById("detailprivkey").value.toString().replace(/^\s+|\s+$/g, ""); // trim white space
document.getElementById("detailprivkey").value = key;
var keyFormat = ninja.privateKey;
@@ -5339,8 +6352,13 @@
// hide Private Key Mini Format
document.getElementById("detailmini").style.display = "none";
}
- var btcKey = ninja.privateKey.anyPrivateKeyToECKey(key);
- if (btcKey == null) {
+ var btcKey = null;
+
+ if (!bip38) {
+ btcKey = ninja.privateKey.anyPrivateKeyToECKey(key);
+ }
+
+ if (btcKey == null && !bip38) {
// enforce a minimum passphrase length
if (key.length >= ninja.wallets.brainwallet.minPassphraseLength) {
// Deterministic Wallet confirm box to ask if user wants to SHA256 the input to get a private key
@@ -5358,7 +6376,8 @@
ninja.wallets.detailwallet.clear();
}
}
- if (btcKey != undefined) {
+
+ var populateKeyDetails = function(btcKey) {
document.getElementById("detailpubkey").innerHTML = btcKey.getPubKeyHex();
document.getElementById("detailpubkeycomp").innerHTML = btcKey.getPubKeyHexCompressed();
document.getElementById("detailaddress").innerHTML = btcKey.getBitcoinAddress();
@@ -5373,6 +6392,28 @@
"detailqrcodeprivate": btcKey.getBitcoinWalletImportFormat(),
"detailqrcodeprivatecomp": btcKey.getBitcoinWalletImportFormatCompressed()
});
+ };
+ if (btcKey != undefined) {
+ populateKeyDetails(btcKey);
+ }
+
+ if (bip38) {
+ var passphrase = document.getElementById("detailprivkeypassphrase").value.toString().replace(/^\s+|\s+$/g, ""); // trim white space
+ if (passphrase == "") {
+ alert("Passphrase required for BIP38 key");
+ return;
+ }
+ ninja.privateKey.BIP38EncryptedKeyToByteArrayAsync(key, passphrase, function (btcKeyOrError) {
+ document.getElementById("busyblock").className = "";
+ if (btcKeyOrError.message) {
+ alert(btcKeyOrError.message);
+ ninja.wallets.detailwallet.clear();
+ } else {
+ populateKeyDetails(new Bitcoin.ECKey(btcKeyOrError));
+ }
+ });
+ document.getElementById("busyblock").className = "busy";
+ return;
}
},
@@ -5533,6 +6574,38 @@
if (btcKey.getBitcoinWalletImportFormat() != "5JrBLQseeZdYw4jWEAHmNxGMr5fxh9NJU3fUwnv4khfKcg2rJVh") {
alert("fail testMini30CharsToECKey");
}
+ },
+ testBip38: function() {
+ var tests = [["6PRVWUbkzzsbcVac2qwfssoUJAN1Xhrg6bNk8J7Nzm5H7kxEbn2Nh2ZoGg","TestingOneTwoThree","5KN7MzqK5wt2TP1fQCYyHBtDrXdJuXbUzm4A9rKAteGu3Qi5CVR"],
+ ["6PRNFFkZc2NZ6dJqFfhRoFNMR9Lnyj7dYGrzdgXXVMXcxoKTePPX1dWByq","Satoshi","5HtasZ6ofTHP6HCwTqTkLDuLQisYPah7aUnSKfC7h4hMUVw2gi5"],
+ ["6PYNKZ1EAgYgmQfmNVamxyXVWHzK5s6DGhwP4J5o44cvXdoY7sRzhtpUeo","TestingOneTwoThree","L44B5gGEpqEDRS9vVPz7QT35jcBG2r3CZwSwQ4fCewXAhAhqGVpP"],
+ ["6PYLtMnXvfG3oJde97zRyLYFZCYizPU5T3LwgdYJz1fRhh16bU7u6PPmY7","Satoshi","KwYgW8gcxj1JWJXhPSu4Fqwzfhp5Yfi42mdYmMa4XqK7NJxXUSK7"],
+ ["6PfQu77ygVyJLZjfvMLyhLMQbYnu5uguoJJ4kMCLqWwPEdfpwANVS76gTX","TestingOneTwoThree","5K4caxezwjGCGfnoPTZ8tMcJBLB7Jvyjv4xxeacadhq8nLisLR2"],
+ ["6PfLGnQs6VZnrNpmVKfjotbnQuaJK4KZoPFrAjx1JMJUa1Ft8gnf5WxfKd","Satoshi","5KJ51SgxWaAYR13zd9ReMhJpwrcX47xTJh2D3fGPG9CM8vkv5sH"],
+ ["6PgNBNNzDkKdhkT6uJntUXwwzQV8Rr2tZcbkDcuC9DZRsS6AtHts4Ypo1j","MOLON LABE","5JLdxTtcTHcfYcmJsNVy1v2PMDx432JPoYcBTVVRHpPaxUrdtf8"],
+ ["6PgGWtx25kUg8QWvwuJAgorN6k9FbE25rv5dMRwu5SKMnfpfVe5mar2ngH",Crypto.charenc.UTF8.bytesToString([206,156,206,159,206,155,206,169,206,157,32,206,155,206,145,206,146,206,149])/*UTF-8 characters, encoded in source so they don't get corrupted*/,"5KMKKuUmAkiNbA3DazMQiLfDq47qs8MAEThm4yL8R2PhV1ov33D"]];
+ var testsDone = 0;
+ var runTest = function(test, i) {
+ ninja.privateKey.BIP38EncryptedKeyToByteArrayAsync(test[0], test[1], function(privBytes) {
+ if (privBytes.constructor == Error) {
+ alert('fail testBip38 #'+i+', error: '+privBytes.message);
+ } else {
+ var btcKey = new Bitcoin.ECKey(privBytes);
+ var wif = !test[2].substr(0,1).match(/[LK]/) ? btcKey.getBitcoinWalletImportFormat() : btcKey.getBitcoinWalletImportFormatCompressed();
+ if (wif != test[2]) {
+ alert("fail testBip38 #"+i);
+ } else {
+ console.log('pass testBip38 #'+i);
+ }
+ }
+ if (++testsDone == tests.length) {
+ alert('BIP38 tests finished');
+ }
+ });
+ }
+ for (var i = 0; i < tests.length; i++) {
+ runTest(tests[i], i);
+ }
}
};
@@ -5543,7 +6616,7 @@
for (var name in ninja.unitTests) {
ninja.unitTests[name]();
}
- alert("running of unit tests complete");
+ alert("running of synchronous unit tests complete");
}
// change language