160 lines
7.7 KiB
Python
160 lines
7.7 KiB
Python
import unittest
|
|
from unittest import mock
|
|
|
|
import lib.bitcoin as bitcoin
|
|
import lib.keystore as keystore
|
|
import lib.storage as storage
|
|
import lib.wallet as wallet
|
|
from lib import constants
|
|
|
|
from . import TestCaseForTestnet
|
|
|
|
|
|
class WalletIntegrityHelper:
|
|
|
|
gap_limit = 1 # make tests run faster
|
|
|
|
@classmethod
|
|
def check_seeded_keystore_sanity(cls, test_obj, ks):
|
|
test_obj.assertTrue(ks.is_deterministic())
|
|
test_obj.assertFalse(ks.is_watching_only())
|
|
test_obj.assertFalse(ks.can_import())
|
|
test_obj.assertTrue(ks.has_seed())
|
|
|
|
@classmethod
|
|
def check_xpub_keystore_sanity(cls, test_obj, ks):
|
|
test_obj.assertTrue(ks.is_deterministic())
|
|
test_obj.assertTrue(ks.is_watching_only())
|
|
test_obj.assertFalse(ks.can_import())
|
|
test_obj.assertFalse(ks.has_seed())
|
|
|
|
@classmethod
|
|
def create_standard_wallet(cls, ks):
|
|
store = storage.WalletStorage('if_this_exists_mocking_failed_648151893')
|
|
store.put('keystore', ks.dump())
|
|
store.put('gap_limit', cls.gap_limit)
|
|
w = wallet.Standard_Wallet(store)
|
|
w.synchronize()
|
|
return w
|
|
|
|
@classmethod
|
|
def create_multisig_wallet(cls, ks1, ks2, ks3=None):
|
|
"""Creates a 2-of-2 or 2-of-3 multisig wallet."""
|
|
store = storage.WalletStorage('if_this_exists_mocking_failed_648151893')
|
|
store.put('x%d/' % 1, ks1.dump())
|
|
store.put('x%d/' % 2, ks2.dump())
|
|
if ks3 is None:
|
|
multisig_type = "%dof%d" % (2, 2)
|
|
else:
|
|
multisig_type = "%dof%d" % (2, 3)
|
|
store.put('x%d/' % 3, ks3.dump())
|
|
store.put('wallet_type', multisig_type)
|
|
store.put('gap_limit', cls.gap_limit)
|
|
w = wallet.Multisig_Wallet(store)
|
|
w.synchronize()
|
|
return w
|
|
|
|
|
|
# TODO passphrase/seed_extension
|
|
class TestWalletKeystoreAddressIntegrityForMainnet(unittest.TestCase):
|
|
|
|
@mock.patch.object(storage.WalletStorage, '_write')
|
|
def test_electrum_seed_standard(self, mock_write):
|
|
seed_words = 'cycle rocket west magnet parrot shuffle foot correct salt library feed song'
|
|
self.assertEqual(bitcoin.seed_type(seed_words), 'standard')
|
|
|
|
ks = keystore.from_seed(seed_words, '', False)
|
|
|
|
WalletIntegrityHelper.check_seeded_keystore_sanity(self, ks)
|
|
self.assertTrue(isinstance(ks, keystore.BIP32_KeyStore))
|
|
|
|
self.assertEqual(ks.xprv, 'xprv9s21ZrQH143K32jECVM729vWgGq4mUDJCk1ozqAStTphzQtCTuoFmFafNoG1g55iCnBTXUzz3zWnDb5CVLGiFvmaZjuazHDL8a81cPQ8KL6')
|
|
self.assertEqual(ks.xpub, 'xpub661MyMwAqRbcFWohJWt7PHsFEJfZAvw9ZxwQoDa4SoMgsDDM1T7WK3u9E4edkC4ugRnZ8E4xDZRpk8Rnts3Nbt97dPwT52CwBdDWroaZf8U')
|
|
|
|
w = WalletIntegrityHelper.create_standard_wallet(ks)
|
|
self.assertEqual(w.txin_type, 'p2pkh')
|
|
|
|
self.assertEqual(w.get_receiving_addresses()[0], 't1fFMuEC9XFGsEUEPzpEE8jhxpcEMs369xJ')
|
|
self.assertEqual(w.get_change_addresses()[0], 't1cKFzsmq8d97RtePBbqS1WebLofuXaXkzF')
|
|
|
|
@mock.patch.object(storage.WalletStorage, '_write')
|
|
def test_electrum_seed_old(self, mock_write):
|
|
seed_words = 'powerful random nobody notice nothing important anyway look away hidden message over'
|
|
self.assertEqual(bitcoin.seed_type(seed_words), 'old')
|
|
|
|
ks = keystore.from_seed(seed_words, '', False)
|
|
|
|
WalletIntegrityHelper.check_seeded_keystore_sanity(self, ks)
|
|
self.assertTrue(isinstance(ks, keystore.Old_KeyStore))
|
|
|
|
self.assertEqual(ks.mpk, 'e9d4b7866dd1e91c862aebf62a49548c7dbf7bcc6e4b7b8c9da820c7737968df9c09d5a3e271dc814a29981f81b3faaf2737b551ef5dcc6189cf0f8252c442b3')
|
|
|
|
w = WalletIntegrityHelper.create_standard_wallet(ks)
|
|
self.assertEqual(w.txin_type, 'p2pkh')
|
|
|
|
self.assertEqual(w.get_receiving_addresses()[0], 't1YAqEWYrfi9CbW5LgmayAvjDE5T5MgaYiD')
|
|
self.assertEqual(w.get_change_addresses()[0], 't1cJ799hEFa5AHmB3ReeDo3Rr2X4quderf4')
|
|
|
|
@mock.patch.object(storage.WalletStorage, '_write')
|
|
def test_bip39_seed_bip44_standard(self, mock_write):
|
|
seed_words = 'treat dwarf wealth gasp brass outside high rent blood crowd make initial'
|
|
self.assertEqual(keystore.bip39_is_checksum_valid(seed_words), (True, True))
|
|
|
|
ks = keystore.from_bip39_seed(seed_words, '', "m/44'/0'/0'")
|
|
|
|
self.assertTrue(isinstance(ks, keystore.BIP32_KeyStore))
|
|
|
|
self.assertEqual(ks.xprv, 'xprv9zGLcNEb3cHUKizLVBz6RYeE9bEZAVPjH2pD1DEzCnPcsemWc3d3xTao8sfhfUmDLMq6e3RcEMEvJG1Et8dvfL8DV4h7mwm9J6AJsW9WXQD')
|
|
self.assertEqual(ks.xpub, 'xpub6DFh1smUsyqmYD4obDX6ngaxhd53Zx7aeFjoobebm7vbkT6f9awJWFuGzBT9FQJEWFBL7UyhMXtYzRcwDuVbcxtv9Ce2W9eMm4KXLdvdbjv')
|
|
|
|
w = WalletIntegrityHelper.create_standard_wallet(ks)
|
|
self.assertEqual(w.txin_type, 'p2pkh')
|
|
|
|
self.assertEqual(w.get_receiving_addresses()[0], 't1PbiEBABXU1E4GEnE31cUPULDoYYWVMpjs')
|
|
self.assertEqual(w.get_change_addresses()[0], 't1Z8gbq4eeVbg89ACGddq1T6yfEP9bQx9Ki')
|
|
|
|
@mock.patch.object(storage.WalletStorage, '_write')
|
|
def test_electrum_multisig_seed_standard(self, mock_write):
|
|
seed_words = 'blast uniform dragon fiscal ensure vast young utility dinosaur abandon rookie sure'
|
|
self.assertEqual(bitcoin.seed_type(seed_words), 'standard')
|
|
|
|
ks1 = keystore.from_seed(seed_words, '', True)
|
|
WalletIntegrityHelper.check_seeded_keystore_sanity(self, ks1)
|
|
self.assertTrue(isinstance(ks1, keystore.BIP32_KeyStore))
|
|
self.assertEqual(ks1.xprv, 'xprv9s21ZrQH143K3t9vo23J3hajRbzvkRLJ6Y1zFrUFAfU3t8oooMPfb7f87cn5KntgqZs5nipZkCiBFo5ZtaSD2eDo7j7CMuFV8Zu6GYLTpY6')
|
|
self.assertEqual(ks1.xpub, 'xpub661MyMwAqRbcGNEPu3aJQqXTydqR9t49Tkwb4Esrj112kw8xLthv8uybxvaki4Ygt9xiwZUQGeFTG7T2TUzR3eA4Zp3aq5RXsABHFBUrq4c')
|
|
|
|
# electrum seed: ghost into match ivory badge robot record tackle radar elbow traffic loud
|
|
ks2 = keystore.from_xpub('xpub661MyMwAqRbcGfCPEkkyo5WmcrhTq8mi3xuBS7VEZ3LYvsgY1cCFDbenT33bdD12axvrmXhuX3xkAbKci3yZY9ZEk8vhLic7KNhLjqdh5ec')
|
|
WalletIntegrityHelper.check_xpub_keystore_sanity(self, ks2)
|
|
self.assertTrue(isinstance(ks2, keystore.BIP32_KeyStore))
|
|
|
|
w = WalletIntegrityHelper.create_multisig_wallet(ks1, ks2)
|
|
self.assertEqual(w.txin_type, 'p2sh')
|
|
|
|
self.assertEqual(w.get_receiving_addresses()[0], 't3KcK3kAJerAahSJhN6Pt729u7ficQqK4XX')
|
|
self.assertEqual(w.get_change_addresses()[0], 't3PQ7wZhzpoywPLnD1nfcd5sPYLtw2qh5ak')
|
|
|
|
@mock.patch.object(storage.WalletStorage, '_write')
|
|
def test_bip39_multisig_seed_bip45_standard(self, mock_write):
|
|
seed_words = 'treat dwarf wealth gasp brass outside high rent blood crowd make initial'
|
|
self.assertEqual(keystore.bip39_is_checksum_valid(seed_words), (True, True))
|
|
|
|
ks1 = keystore.from_bip39_seed(seed_words, '', "m/45'/0")
|
|
self.assertTrue(isinstance(ks1, keystore.BIP32_KeyStore))
|
|
self.assertEqual(ks1.xprv, 'xprv9vyEFyXf7pYVv4eDU3hhuCEAHPHNGuxX73nwtYdpbLcqwJCPwFKknAK8pHWuHHBirCzAPDZ7UJHrYdhLfn1NkGp9rk3rVz2aEqrT93qKRD9')
|
|
self.assertEqual(ks1.xpub, 'xpub69xafV4YxC6o8Yiga5EiGLAtqR7rgNgNUGiYgw3S9g9pp6XYUne1KxdcfYtxwmA3eBrzMFuYcNQKfqsXCygCo4GxQFHfywxpUbKNfYvGJka')
|
|
|
|
# bip39 seed: tray machine cook badge night page project uncover ritual toward person enact
|
|
# der: m/45'/0
|
|
ks2 = keystore.from_xpub('xpub6Bco9vrgo8rNUSi8Bjomn8xLA41DwPXeuPcgJamNRhTTyGVHsp8fZXaGzp9ypHoei16J6X3pumMAP1u3Dy4jTSWjm4GZowL7Dcn9u4uZC9W')
|
|
WalletIntegrityHelper.check_xpub_keystore_sanity(self, ks2)
|
|
self.assertTrue(isinstance(ks2, keystore.BIP32_KeyStore))
|
|
|
|
w = WalletIntegrityHelper.create_multisig_wallet(ks1, ks2)
|
|
self.assertEqual(w.txin_type, 'p2sh')
|
|
|
|
self.assertEqual(w.get_receiving_addresses()[0], 't3ZvKyVcMRf5rofUFgN8gk23jtCkRh12Lja')
|
|
self.assertEqual(w.get_change_addresses()[0], 't3JaafdGtfhXJzCrupYct6tJBdD2XZUtm8H')
|
|
|