mirror of
https://github.com/BlueWallet/BlueWallet.git
synced 2025-01-19 05:45:15 +01:00
197 lines
11 KiB
JavaScript
197 lines
11 KiB
JavaScript
import assert from 'assert';
|
||
import * as bitcoin from 'bitcoinjs-lib';
|
||
|
||
import { HDLegacyP2PKHWallet } from '../../class';
|
||
|
||
describe('Legacy HD (BIP44)', () => {
|
||
it('works', async () => {
|
||
if (!process.env.HD_MNEMONIC) {
|
||
console.error('process.env.HD_MNEMONIC not set, skipped');
|
||
return;
|
||
}
|
||
const hd = new HDLegacyP2PKHWallet();
|
||
hd.setSecret(process.env.HD_MNEMONIC);
|
||
assert.ok(hd.validateMnemonic());
|
||
|
||
assert.strictEqual(
|
||
hd.getXpub(),
|
||
'xpub6ByZUAv558PPheJgcPYHpxPLwz8M7TtueYMAik84NADeQcvbzS8W3WxxJ3C9NzfYkMoChiMAumWbeEvMWhTVpH75NqGv5c9wF3wKDbfQShb',
|
||
);
|
||
|
||
assert.strictEqual(hd._getExternalAddressByIndex(0), '186FBQmCV5W1xY7ywaWtTZPAQNciVN8Por');
|
||
assert.strictEqual(hd._getInternalAddressByIndex(0), '1J9zoJz5LsAJ361SQHYnLTWg46Tc2AXUCj');
|
||
|
||
assert.strictEqual(hd._getInternalWIFByIndex(0), 'L4ojevRtK81A8Kof3qyLS2M7HvsVDbUDENNhJqU4vf79w9yGnQLb');
|
||
assert.strictEqual(hd._getExternalWIFByIndex(0), 'Kz6kLhdyDfSbKuVH25XVqBRztjmFe8X22Xe1hnFzEv79gJNMkTAH');
|
||
|
||
assert.ok(hd.getAllExternalAddresses().includes('186FBQmCV5W1xY7ywaWtTZPAQNciVN8Por'));
|
||
assert.ok(!hd.getAllExternalAddresses().includes('1J9zoJz5LsAJ361SQHYnLTWg46Tc2AXUCj')); // not internal
|
||
|
||
assert.strictEqual(
|
||
hd._getPubkeyByAddress(hd._getExternalAddressByIndex(0)).toString('hex'),
|
||
'0316e84a2556f30a199541633f5dda6787710ccab26771b7084f4c9e1104f47667',
|
||
);
|
||
assert.strictEqual(
|
||
hd._getPubkeyByAddress(hd._getInternalAddressByIndex(0)).toString('hex'),
|
||
'02ad7b2216f3a2b38d56db8a7ee5c540fd12c4bbb7013106eff78cc2ace65aa002',
|
||
);
|
||
|
||
assert.strictEqual(hd._getDerivationPathByAddress(hd._getExternalAddressByIndex(0)), "m/44'/0'/0'/0/0");
|
||
assert.strictEqual(hd._getDerivationPathByAddress(hd._getInternalAddressByIndex(0)), "m/44'/0'/0'/1/0");
|
||
});
|
||
|
||
it('can create TX', async () => {
|
||
if (!process.env.HD_MNEMONIC) {
|
||
console.error('process.env.HD_MNEMONIC not set, skipped');
|
||
return;
|
||
}
|
||
const hd = new HDLegacyP2PKHWallet();
|
||
hd.setSecret(process.env.HD_MNEMONIC);
|
||
assert.ok(hd.validateMnemonic());
|
||
|
||
const utxo = [
|
||
{
|
||
height: 554830,
|
||
value: 10000,
|
||
address: '186FBQmCV5W1xY7ywaWtTZPAQNciVN8Por',
|
||
txId: '4f65c8cb159585c00d4deba9c5b36a2bcdfb1399a561114dcf6f2d0c1174bc5f',
|
||
vout: 0,
|
||
txid: '4f65c8cb159585c00d4deba9c5b36a2bcdfb1399a561114dcf6f2d0c1174bc5f',
|
||
amount: 10000,
|
||
wif: 'Kz6kLhdyDfSbKuVH25XVqBRztjmFe8X22Xe1hnFzEv79gJNMkTAH',
|
||
confirmations: 1,
|
||
txhex:
|
||
'01000000000101e8d98effbb4fba4f0a89bcf217eb5a7e2f8efcae44f32ecacbc5d8cc3ce683c301000000171600148ba6d02e74c0a6e000e8b174eb2ed44e5ea211a6ffffffff0510270000000000001976a9144dc6cbf64df9ab106cee812c7501960b93e9217788ac204e0000000000001976a914bc2db6b74c8db9b188711dcedd511e6a305603f588ac30750000000000001976a9144dc6cbf64df9ab106cee812c7501960b93e9217788ac409c0000000000001976a914bc2db6b74c8db9b188711dcedd511e6a305603f588ac204716000000000017a914e286d58e53f9247a4710e51232cce0686f16873c8702483045022100af3800cd8171f154785cf13f46c092f61c1668f97db432bb4e7ed7bc812a8c6d022051bddca1eaf1ad8b5f3bd0ccde7447e56fd3c8709e5906f02ec6326e9a5b2ff30121039a421d5eb7c9de6590ae2a471cb556b60de8c6b056beb907dbdc1f5e6092f58800000000',
|
||
},
|
||
{
|
||
height: 554830,
|
||
value: 20000,
|
||
address: '1J9zoJz5LsAJ361SQHYnLTWg46Tc2AXUCj',
|
||
txId: '4f65c8cb159585c00d4deba9c5b36a2bcdfb1399a561114dcf6f2d0c1174bc5f',
|
||
vout: 1,
|
||
txid: '4f65c8cb159585c00d4deba9c5b36a2bcdfb1399a561114dcf6f2d0c1174bc5f',
|
||
amount: 20000,
|
||
wif: 'L4ojevRtK81A8Kof3qyLS2M7HvsVDbUDENNhJqU4vf79w9yGnQLb',
|
||
confirmations: 1,
|
||
txhex:
|
||
'01000000000101e8d98effbb4fba4f0a89bcf217eb5a7e2f8efcae44f32ecacbc5d8cc3ce683c301000000171600148ba6d02e74c0a6e000e8b174eb2ed44e5ea211a6ffffffff0510270000000000001976a9144dc6cbf64df9ab106cee812c7501960b93e9217788ac204e0000000000001976a914bc2db6b74c8db9b188711dcedd511e6a305603f588ac30750000000000001976a9144dc6cbf64df9ab106cee812c7501960b93e9217788ac409c0000000000001976a914bc2db6b74c8db9b188711dcedd511e6a305603f588ac204716000000000017a914e286d58e53f9247a4710e51232cce0686f16873c8702483045022100af3800cd8171f154785cf13f46c092f61c1668f97db432bb4e7ed7bc812a8c6d022051bddca1eaf1ad8b5f3bd0ccde7447e56fd3c8709e5906f02ec6326e9a5b2ff30121039a421d5eb7c9de6590ae2a471cb556b60de8c6b056beb907dbdc1f5e6092f58800000000',
|
||
},
|
||
{
|
||
height: 554830,
|
||
value: 30000,
|
||
address: '186FBQmCV5W1xY7ywaWtTZPAQNciVN8Por',
|
||
txId: '4f65c8cb159585c00d4deba9c5b36a2bcdfb1399a561114dcf6f2d0c1174bc5f',
|
||
vout: 2,
|
||
txid: '4f65c8cb159585c00d4deba9c5b36a2bcdfb1399a561114dcf6f2d0c1174bc5f',
|
||
amount: 30000,
|
||
wif: 'Kz6kLhdyDfSbKuVH25XVqBRztjmFe8X22Xe1hnFzEv79gJNMkTAH',
|
||
confirmations: 1,
|
||
txhex:
|
||
'01000000000101e8d98effbb4fba4f0a89bcf217eb5a7e2f8efcae44f32ecacbc5d8cc3ce683c301000000171600148ba6d02e74c0a6e000e8b174eb2ed44e5ea211a6ffffffff0510270000000000001976a9144dc6cbf64df9ab106cee812c7501960b93e9217788ac204e0000000000001976a914bc2db6b74c8db9b188711dcedd511e6a305603f588ac30750000000000001976a9144dc6cbf64df9ab106cee812c7501960b93e9217788ac409c0000000000001976a914bc2db6b74c8db9b188711dcedd511e6a305603f588ac204716000000000017a914e286d58e53f9247a4710e51232cce0686f16873c8702483045022100af3800cd8171f154785cf13f46c092f61c1668f97db432bb4e7ed7bc812a8c6d022051bddca1eaf1ad8b5f3bd0ccde7447e56fd3c8709e5906f02ec6326e9a5b2ff30121039a421d5eb7c9de6590ae2a471cb556b60de8c6b056beb907dbdc1f5e6092f58800000000',
|
||
},
|
||
{
|
||
height: 554830,
|
||
value: 40000,
|
||
address: '1J9zoJz5LsAJ361SQHYnLTWg46Tc2AXUCj',
|
||
txId: '4f65c8cb159585c00d4deba9c5b36a2bcdfb1399a561114dcf6f2d0c1174bc5f',
|
||
vout: 3,
|
||
txid: '4f65c8cb159585c00d4deba9c5b36a2bcdfb1399a561114dcf6f2d0c1174bc5f',
|
||
amount: 40000,
|
||
wif: 'L4ojevRtK81A8Kof3qyLS2M7HvsVDbUDENNhJqU4vf79w9yGnQLb',
|
||
confirmations: 1,
|
||
txhex:
|
||
'01000000000101e8d98effbb4fba4f0a89bcf217eb5a7e2f8efcae44f32ecacbc5d8cc3ce683c301000000171600148ba6d02e74c0a6e000e8b174eb2ed44e5ea211a6ffffffff0510270000000000001976a9144dc6cbf64df9ab106cee812c7501960b93e9217788ac204e0000000000001976a914bc2db6b74c8db9b188711dcedd511e6a305603f588ac30750000000000001976a9144dc6cbf64df9ab106cee812c7501960b93e9217788ac409c0000000000001976a914bc2db6b74c8db9b188711dcedd511e6a305603f588ac204716000000000017a914e286d58e53f9247a4710e51232cce0686f16873c8702483045022100af3800cd8171f154785cf13f46c092f61c1668f97db432bb4e7ed7bc812a8c6d022051bddca1eaf1ad8b5f3bd0ccde7447e56fd3c8709e5906f02ec6326e9a5b2ff30121039a421d5eb7c9de6590ae2a471cb556b60de8c6b056beb907dbdc1f5e6092f58800000000',
|
||
},
|
||
];
|
||
|
||
let txNew = hd.createTransaction(
|
||
utxo,
|
||
[{ address: '3GcKN7q7gZuZ8eHygAhHrvPa5zZbG5Q1rK', value: 80000 }],
|
||
1,
|
||
hd._getInternalAddressByIndex(hd.next_free_change_address_index),
|
||
);
|
||
let tx = bitcoin.Transaction.fromHex(txNew.tx.toHex());
|
||
assert.strictEqual(tx.ins.length, 3);
|
||
assert.strictEqual(tx.outs.length, 2);
|
||
assert.strictEqual(tx.outs[0].value, 80000); // payee
|
||
assert.strictEqual(tx.outs[1].value, 9478); // change
|
||
let toAddress = bitcoin.address.fromOutputScript(tx.outs[0].script);
|
||
const changeAddress = bitcoin.address.fromOutputScript(tx.outs[1].script);
|
||
assert.strictEqual('3GcKN7q7gZuZ8eHygAhHrvPa5zZbG5Q1rK', toAddress);
|
||
assert.strictEqual(hd._getInternalAddressByIndex(hd.next_free_change_address_index), changeAddress);
|
||
|
||
// testing sendMax
|
||
txNew = hd.createTransaction(
|
||
utxo,
|
||
[{ address: '3GcKN7q7gZuZ8eHygAhHrvPa5zZbG5Q1rK' }],
|
||
1,
|
||
hd._getInternalAddressByIndex(hd.next_free_change_address_index),
|
||
);
|
||
tx = bitcoin.Transaction.fromHex(txNew.tx.toHex());
|
||
assert.strictEqual(tx.ins.length, 4);
|
||
assert.strictEqual(tx.outs.length, 1);
|
||
toAddress = bitcoin.address.fromOutputScript(tx.outs[0].script);
|
||
assert.strictEqual('3GcKN7q7gZuZ8eHygAhHrvPa5zZbG5Q1rK', toAddress);
|
||
});
|
||
|
||
it('can sign and verify messages', async () => {
|
||
const mnemonic = 'abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about';
|
||
const hd = new HDLegacyP2PKHWallet();
|
||
hd.setSecret(mnemonic);
|
||
let signature;
|
||
|
||
// external address
|
||
signature = hd.signMessage('vires is numeris', hd._getExternalAddressByIndex(0));
|
||
assert.strictEqual(signature, 'H5J8DbqvuBy8lqRW7+LTVrrtrsaqLSwRDyj+5XtCrZpdCgPlxKM4EKRD6qvdKeyEh1fiSfIVB/edPAum3gKcJZo=');
|
||
assert.strictEqual(hd.verifyMessage('vires is numeris', hd._getExternalAddressByIndex(0), signature), true);
|
||
|
||
// internal address
|
||
signature = hd.signMessage('vires is numeris', hd._getInternalAddressByIndex(0));
|
||
assert.strictEqual(signature, 'H98hmvtyPFUbR6E5Tcsqmc+eSjlYhP2vy41Y6IyHS9DVKEI5n8VEMpIEDtvlMARVce96nOqbRHXo9nD05WXH/Eo=');
|
||
assert.strictEqual(hd.verifyMessage('vires is numeris', hd._getInternalAddressByIndex(0), signature), true);
|
||
});
|
||
|
||
it('can show fingerprint', async () => {
|
||
const mnemonic = 'abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about';
|
||
const hd = new HDLegacyP2PKHWallet();
|
||
hd.setSecret(mnemonic);
|
||
assert.strictEqual(hd.getMasterFingerprintHex(), '73C5DA0A');
|
||
});
|
||
|
||
// from electrum tests https://github.com/spesmilo/electrum/blob/9c1a51547a301e765b9b0f9935c6d940bb9d658e/electrum/tests/test_wallet_vertical.py#L292
|
||
it('can use mnemonic with passphrase', () => {
|
||
const mnemonic = 'treat dwarf wealth gasp brass outside high rent blood crowd make initial';
|
||
const UNICODE_HORROR = '₿ 😀 😈 う けたま わる w͢͢͝h͡o͢͡ ̸͢k̵͟n̴͘ǫw̸̛s͘ ̀́w͘͢ḩ̵a҉̡͢t ̧̕h́o̵r͏̵rors̡ ̶͡͠lį̶e͟͟ ̶͝in͢ ͏t̕h̷̡͟e ͟͟d̛a͜r̕͡k̢̨ ͡h̴e͏a̷̢̡rt́͏ ̴̷͠ò̵̶f̸ u̧͘ní̛͜c͢͏o̷͏d̸͢e̡͝?͞';
|
||
const hd = new HDLegacyP2PKHWallet();
|
||
hd.setSecret(mnemonic);
|
||
hd.setPassphrase(UNICODE_HORROR);
|
||
|
||
assert.strictEqual(
|
||
hd.getXpub(),
|
||
'xpub6D85QDBajeLe2JXJrZeGyQCaw47PWKi3J9DPuHakjTkVBWCxVQQkmMVMSSfnw39tj9FntbozpRtb1AJ8ubjeVSBhyK4M5mzdvsXZzKPwodT',
|
||
);
|
||
|
||
assert.strictEqual(hd._getExternalAddressByIndex(0), '1F88g2naBMhDB7pYFttPWGQgryba3hPevM');
|
||
assert.strictEqual(hd._getInternalAddressByIndex(0), '1H4QD1rg2zQJ4UjuAVJr5eW1fEM8WMqyxh');
|
||
assert.strictEqual(hd._getExternalWIFByIndex(0), 'L3HLzdVcwo4711gFiZG4fiLzLVNJpR6nejfo6J85wuYn9YF2G5zk');
|
||
});
|
||
|
||
it('can create with custom derivation path', async () => {
|
||
const hd = new HDLegacyP2PKHWallet();
|
||
hd.setSecret('abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about');
|
||
hd.setDerivationPath("m/44'/0'/1'");
|
||
|
||
assert.strictEqual(
|
||
hd.getXpub(),
|
||
'xpub6BosfCnifzxcJJ1wYuntGJfF2zPJkDeG9ELNHcKNjezuea4tumswN9sH1psMdSVqCMoJC21Bv8usSeqSP4Sp1tLzW7aY59fGn9GCYzx5UTo',
|
||
);
|
||
|
||
assert.strictEqual(hd._getExternalAddressByIndex(0), '15qucUWKf95Fo58FdCBhUTSAtsm22HHE2Q');
|
||
assert.strictEqual(hd._getInternalAddressByIndex(0), '1DgjtFUiXvqxGic9A9fiDPrHNyKC4cGtTH');
|
||
assert.strictEqual(hd._getExternalWIFByIndex(0), 'KzReLDRfwGJ7bBH6WjLQ36e2WxjHob3d61EKnZQT86nutd5tpkvC');
|
||
|
||
assert.strictEqual(hd._getDerivationPathByAddress(hd._getExternalAddressByIndex(0)), "m/44'/0'/1'/0/0");
|
||
assert.strictEqual(hd._getDerivationPathByAddress(hd._getInternalAddressByIndex(0)), "m/44'/0'/1'/1/0");
|
||
});
|
||
});
|