lbry-sdk/torba/baseaccount.py

403 lines
15 KiB
Python
Raw Normal View History

import typing
2018-09-25 22:46:22 -04:00
from typing import List, Dict, Tuple, Type, Optional, Any
2018-06-11 09:33:32 -04:00
from twisted.internet import defer
from torba.mnemonic import Mnemonic
from torba.bip32 import PrivateKey, PubKey, from_extended_key_string
2018-09-21 14:49:16 -04:00
from torba.hash import aes_encrypt, aes_decrypt
2018-08-30 11:50:11 -04:00
from torba.constants import COIN
2018-06-11 09:33:32 -04:00
if typing.TYPE_CHECKING:
from torba import baseledger
2018-08-07 21:47:30 -04:00
from torba import wallet as basewallet
2018-09-25 22:46:22 -04:00
from torba import basetransaction
2018-06-11 09:33:32 -04:00
2018-07-29 13:13:40 -04:00
class AddressManager:
name: str
2018-06-11 09:33:32 -04:00
2018-07-14 17:47:18 -04:00
__slots__ = 'account', 'public_key', 'chain_number'
def __init__(self, account, public_key, chain_number):
2018-06-11 09:33:32 -04:00
self.account = account
2018-07-14 17:47:18 -04:00
self.public_key = public_key
2018-06-11 09:33:32 -04:00
self.chain_number = chain_number
2018-07-29 13:13:40 -04:00
@classmethod
def from_dict(cls, account: 'BaseAccount', d: dict) \
-> Tuple['AddressManager', 'AddressManager']:
raise NotImplementedError
@classmethod
2018-07-29 14:34:56 -04:00
def to_dict(cls, receiving: 'AddressManager', change: 'AddressManager') -> Dict:
d: Dict[str, Any] = {'name': cls.name}
receiving_dict = receiving.to_dict_instance()
if receiving_dict:
d['receiving'] = receiving_dict
change_dict = change.to_dict_instance()
if change_dict:
d['change'] = change_dict
return d
def to_dict_instance(self) -> Optional[dict]:
raise NotImplementedError
2018-07-29 13:13:40 -04:00
2018-07-14 17:47:18 -04:00
@property
def db(self):
return self.account.ledger.db
2018-06-11 09:33:32 -04:00
def _query_addresses(self, **constraints):
2018-07-14 17:47:18 -04:00
return self.db.get_addresses(
account=self.account,
chain=self.chain_number,
**constraints
2018-06-11 09:33:32 -04:00
)
2018-07-29 13:13:40 -04:00
def get_private_key(self, index: int) -> PrivateKey:
raise NotImplementedError
def get_max_gap(self) -> defer.Deferred:
2018-07-25 23:29:41 -04:00
raise NotImplementedError
def ensure_address_gap(self) -> defer.Deferred:
2018-07-14 17:47:18 -04:00
raise NotImplementedError
def get_address_records(self, only_usable: bool = False, **constraints) -> defer.Deferred:
2018-07-14 17:47:18 -04:00
raise NotImplementedError
@defer.inlineCallbacks
def get_addresses(self, only_usable: bool = False, **constraints) -> defer.Deferred:
records = yield self.get_address_records(only_usable=only_usable, **constraints)
2018-08-30 11:50:11 -04:00
return [r['address'] for r in records]
2018-07-14 17:47:18 -04:00
@defer.inlineCallbacks
def get_or_create_usable_address(self) -> defer.Deferred:
addresses = yield self.get_addresses(only_usable=True, limit=1)
2018-07-14 17:47:18 -04:00
if addresses:
2018-08-30 11:50:11 -04:00
return addresses[0]
2018-07-14 17:47:18 -04:00
addresses = yield self.ensure_address_gap()
2018-08-30 11:50:11 -04:00
return addresses[0]
2018-07-14 17:47:18 -04:00
2018-07-29 13:13:40 -04:00
class HierarchicalDeterministic(AddressManager):
2018-07-14 17:47:18 -04:00
""" Implements simple version of Bitcoin Hierarchical Deterministic key management. """
2018-07-29 13:13:40 -04:00
name = "deterministic-chain"
2018-07-14 17:47:18 -04:00
__slots__ = 'gap', 'maximum_uses_per_address'
2018-07-29 13:13:40 -04:00
def __init__(self, account: 'BaseAccount', chain: int, gap: int, maximum_uses_per_address: int) -> None:
super().__init__(account, account.public_key.child(chain), chain)
2018-07-14 17:47:18 -04:00
self.gap = gap
self.maximum_uses_per_address = maximum_uses_per_address
2018-07-29 13:13:40 -04:00
@classmethod
def from_dict(cls, account: 'BaseAccount', d: dict) -> Tuple[AddressManager, AddressManager]:
return (
cls(account, 0, **d.get('receiving', {'gap': 20, 'maximum_uses_per_address': 2})),
cls(account, 1, **d.get('change', {'gap': 6, 'maximum_uses_per_address': 2}))
)
def to_dict_instance(self):
return {'gap': self.gap, 'maximum_uses_per_address': self.maximum_uses_per_address}
def get_private_key(self, index: int) -> PrivateKey:
return self.account.private_key.child(self.chain_number).child(index)
2018-06-12 10:02:04 -04:00
@defer.inlineCallbacks
def generate_keys(self, start: int, end: int) -> defer.Deferred:
2018-06-11 09:33:32 -04:00
new_keys = []
2018-06-12 10:02:04 -04:00
for index in range(start, end+1):
2018-07-14 17:47:18 -04:00
new_keys.append((index, self.public_key.child(index)))
2018-06-11 09:33:32 -04:00
yield self.db.add_keys(
self.account, self.chain_number, new_keys
)
2018-08-30 11:50:11 -04:00
return [key[1].address for key in new_keys]
2018-06-12 10:02:04 -04:00
2018-07-25 23:29:41 -04:00
@defer.inlineCallbacks
def get_max_gap(self) -> defer.Deferred:
2018-07-25 23:29:41 -04:00
addresses = yield self._query_addresses(order_by="position ASC")
max_gap = 0
current_gap = 0
for address in addresses:
if address['used_times'] == 0:
current_gap += 1
else:
max_gap = max(max_gap, current_gap)
current_gap = 0
2018-08-30 11:50:11 -04:00
return max_gap
2018-07-25 23:29:41 -04:00
2018-06-12 10:02:04 -04:00
@defer.inlineCallbacks
def ensure_address_gap(self) -> defer.Deferred:
addresses = yield self._query_addresses(limit=self.gap, order_by="position DESC")
2018-06-11 09:33:32 -04:00
2018-06-12 10:02:04 -04:00
existing_gap = 0
for address in addresses:
if address['used_times'] == 0:
existing_gap += 1
else:
break
if existing_gap == self.gap:
2018-08-30 11:50:11 -04:00
return []
2018-06-12 10:02:04 -04:00
start = addresses[0]['position']+1 if addresses else 0
end = start + (self.gap - existing_gap)
new_keys = yield self.generate_keys(start, end-1)
2018-08-30 11:50:11 -04:00
return new_keys
2018-06-11 09:33:32 -04:00
def get_address_records(self, only_usable: bool = False, **constraints):
if only_usable:
constraints['used_times__lte'] = self.maximum_uses_per_address
return self._query_addresses(order_by="used_times ASC, position ASC", **constraints)
2018-07-14 17:47:18 -04:00
2018-07-29 13:13:40 -04:00
class SingleKey(AddressManager):
""" Single Key address manager always returns the same address for all operations. """
name = "single-address"
2018-07-14 17:47:18 -04:00
__slots__ = ()
2018-07-29 13:13:40 -04:00
@classmethod
def from_dict(cls, account: 'BaseAccount', d: dict)\
-> Tuple[AddressManager, AddressManager]:
same_address_manager = cls(account, account.public_key, 0)
return same_address_manager, same_address_manager
2018-07-29 14:34:56 -04:00
def to_dict_instance(self):
return None
2018-07-29 13:13:40 -04:00
def get_private_key(self, index: int) -> PrivateKey:
return self.account.private_key
def get_max_gap(self) -> defer.Deferred:
2018-07-25 23:29:41 -04:00
return defer.succeed(0)
2018-06-11 09:33:32 -04:00
@defer.inlineCallbacks
def ensure_address_gap(self) -> defer.Deferred:
2018-07-14 17:47:18 -04:00
exists = yield self.get_address_records()
if not exists:
yield self.db.add_keys(
self.account, self.chain_number, [(0, self.public_key)]
)
2018-08-30 11:50:11 -04:00
return [self.public_key.address]
return []
2018-07-14 17:47:18 -04:00
def get_address_records(self, only_usable: bool = False, **constraints) -> defer.Deferred:
return self._query_addresses(**constraints)
2018-06-11 09:33:32 -04:00
class BaseAccount:
2018-06-11 09:33:32 -04:00
mnemonic_class = Mnemonic
private_key_class = PrivateKey
public_key_class = PubKey
2018-07-29 14:34:56 -04:00
address_generators: Dict[str, Type[AddressManager]] = {
2018-07-29 13:13:40 -04:00
SingleKey.name: SingleKey,
HierarchicalDeterministic.name: HierarchicalDeterministic,
}
2018-06-11 09:33:32 -04:00
2018-08-07 21:47:30 -04:00
def __init__(self, ledger: 'baseledger.BaseLedger', wallet: 'basewallet.Wallet', name: str,
seed: str, private_key_string: str, encrypted: bool,
private_key: Optional[PrivateKey], public_key: PubKey,
address_generator: dict) -> None:
2018-06-12 10:02:04 -04:00
self.ledger = ledger
self.wallet = wallet
2018-08-30 11:50:11 -04:00
self.id = public_key.address
2018-07-14 17:47:18 -04:00
self.name = name
2018-06-12 10:02:04 -04:00
self.seed = seed
self.private_key_string = private_key_string
self.password: Optional[str] = None
2018-09-21 14:49:16 -04:00
self.encryption_init_vector = None
2018-06-12 10:02:04 -04:00
self.encrypted = encrypted
2018-09-21 14:49:16 -04:00
self.serialize_encrypted = encrypted
self.private_key = private_key
2018-06-12 10:02:04 -04:00
self.public_key = public_key
2018-07-29 13:13:40 -04:00
generator_name = address_generator.get('name', HierarchicalDeterministic.name)
2018-07-29 14:34:56 -04:00
self.address_generator = self.address_generators[generator_name]
2018-07-29 13:13:40 -04:00
self.receiving, self.change = self.address_generator.from_dict(self, address_generator)
self.address_managers = {self.receiving, self.change}
2018-06-13 20:57:57 -04:00
ledger.add_account(self)
wallet.add_account(self)
2018-06-11 09:33:32 -04:00
@classmethod
2018-08-07 21:47:30 -04:00
def generate(cls, ledger: 'baseledger.BaseLedger', wallet: 'basewallet.Wallet',
2018-08-07 21:36:44 -04:00
name: str = None, address_generator: dict = None):
return cls.from_dict(ledger, wallet, {
'name': name,
'seed': cls.mnemonic_class().make_seed(),
'address_generator': address_generator or {}
})
2018-06-11 09:33:32 -04:00
@classmethod
def get_private_key_from_seed(cls, ledger: 'baseledger.BaseLedger', seed: str, password: str):
2018-06-11 09:33:32 -04:00
return cls.private_key_class.from_seed(
ledger, cls.mnemonic_class.mnemonic_to_seed(seed, password)
)
@classmethod
2018-08-07 21:47:30 -04:00
def from_dict(cls, ledger: 'baseledger.BaseLedger', wallet: 'basewallet.Wallet', d: dict):
seed = d.get('seed', '')
private_key_string = d.get('private_key', '')
private_key = None
public_key = None
encrypted = d.get('encrypted', False)
if not encrypted:
if seed:
private_key = cls.get_private_key_from_seed(ledger, seed, '')
public_key = private_key.public_key
elif private_key:
private_key = from_extended_key_string(ledger, private_key_string)
public_key = private_key.public_key
if public_key is None:
2018-06-11 09:33:32 -04:00
public_key = from_extended_key_string(ledger, d['public_key'])
name = d.get('name')
if not name:
name = 'Account #{}'.format(public_key.address)
2018-07-29 13:13:40 -04:00
return cls(
2018-06-11 09:33:32 -04:00
ledger=ledger,
wallet=wallet,
name=name,
seed=seed,
private_key_string=private_key_string,
encrypted=encrypted,
2018-06-11 09:33:32 -04:00
private_key=private_key,
public_key=public_key,
address_generator=d.get('address_generator', {})
2018-06-11 09:33:32 -04:00
)
def to_dict(self):
private_key_string, seed = self.private_key_string, self.seed
2018-07-14 17:47:18 -04:00
if not self.encrypted and self.private_key:
private_key_string = self.private_key.extended_key_string()
2018-09-21 14:49:16 -04:00
if not self.encrypted and self.serialize_encrypted:
private_key_string = aes_encrypt(self.password, private_key_string, self.encryption_init_vector)
seed = aes_encrypt(self.password, self.seed, self.encryption_init_vector)
2018-07-29 13:13:40 -04:00
return {
2018-06-11 09:33:32 -04:00
'ledger': self.ledger.get_id(),
2018-07-14 17:47:18 -04:00
'name': self.name,
'seed': seed,
2018-06-11 09:33:32 -04:00
'encrypted': self.encrypted,
'private_key': private_key_string,
'public_key': self.public_key.extended_key_string(),
2018-07-29 13:13:40 -04:00
'address_generator': self.address_generator.to_dict(self.receiving, self.change)
2018-06-11 09:33:32 -04:00
}
2018-08-30 11:50:11 -04:00
@defer.inlineCallbacks
def get_details(self, show_seed=False, **kwargs):
satoshis = yield self.get_balance(**kwargs)
details = {
'id': self.id,
'name': self.name,
'coins': round(satoshis/COIN, 2),
'satoshis': satoshis,
'encrypted': self.encrypted,
'public_key': self.public_key.extended_key_string(),
'address_generator': self.address_generator.to_dict(self.receiving, self.change)
}
if show_seed:
details['seed'] = self.seed
return details
2018-09-21 14:49:16 -04:00
def decrypt(self, password: str) -> None:
2018-06-11 09:33:32 -04:00
assert self.encrypted, "Key is not encrypted."
2018-09-21 14:49:16 -04:00
self.seed = aes_decrypt(password, self.seed)
2018-09-21 13:10:16 -04:00
self.private_key = from_extended_key_string(
self.ledger, aes_decrypt(password, self.private_key_string)
2018-09-21 13:10:16 -04:00
)
self.password = password
2018-06-11 09:33:32 -04:00
self.encrypted = False
2018-09-21 14:49:16 -04:00
def encrypt(self, password: str) -> None:
2018-06-11 09:33:32 -04:00
assert not self.encrypted, "Key is already encrypted."
2018-09-21 14:49:16 -04:00
assert isinstance(self.private_key, PrivateKey)
self.seed = aes_encrypt(password, self.seed, self.encryption_init_vector)
self.private_key_string = aes_encrypt(
password, self.private_key.extended_key_string(), self.encryption_init_vector
)
self.private_key = None
self.password = None
2018-06-11 09:33:32 -04:00
self.encrypted = True
@defer.inlineCallbacks
2018-06-12 10:02:04 -04:00
def ensure_address_gap(self):
2018-06-11 09:33:32 -04:00
addresses = []
2018-07-29 13:13:40 -04:00
for address_manager in self.address_managers:
new_addresses = yield address_manager.ensure_address_gap()
2018-06-11 09:33:32 -04:00
addresses.extend(new_addresses)
2018-08-30 11:50:11 -04:00
return addresses
2018-06-11 09:33:32 -04:00
2018-07-14 17:47:18 -04:00
@defer.inlineCallbacks
def get_addresses(self, **constraints) -> defer.Deferred:
rows = yield self.ledger.db.select_addresses('address', **constraints)
return [r[0] for r in rows]
2018-06-12 10:02:04 -04:00
def get_address_records(self, **constraints) -> defer.Deferred:
return self.ledger.db.get_addresses(account=self, **constraints)
2018-06-12 10:02:04 -04:00
def get_private_key(self, chain: int, index: int) -> PrivateKey:
2018-06-11 09:33:32 -04:00
assert not self.encrypted, "Cannot get private key on encrypted wallet account."
2018-07-29 13:13:40 -04:00
address_manager = {0: self.receiving, 1: self.change}[chain]
return address_manager.get_private_key(index)
2018-06-11 09:33:32 -04:00
def get_balance(self, confirmations: int = 0, **constraints):
2018-07-16 23:58:29 -04:00
if confirmations > 0:
2018-07-14 17:47:18 -04:00
height = self.ledger.headers.height - (confirmations-1)
constraints.update({'height__lte': height, 'height__gt': 0})
return self.ledger.db.get_balance(account=self, **constraints)
2018-07-25 23:29:41 -04:00
@defer.inlineCallbacks
def get_max_gap(self):
change_gap = yield self.change.get_max_gap()
receiving_gap = yield self.receiving.get_max_gap()
2018-08-30 11:50:11 -04:00
return {
2018-07-25 23:29:41 -04:00
'max_change_gap': change_gap,
'max_receiving_gap': receiving_gap,
2018-08-30 11:50:11 -04:00
}
2018-07-25 23:29:41 -04:00
def get_utxos(self, **constraints):
2018-10-03 07:08:02 -04:00
return self.ledger.db.get_utxos(account=self, **constraints)
def get_transactions(self, **constraints) -> List['basetransaction.BaseTransaction']:
return self.ledger.db.get_transactions(account=self, **constraints)
2018-09-21 09:47:31 -04:00
@defer.inlineCallbacks
def fund(self, to_account, amount=None, everything=False,
outputs=1, broadcast=False, **constraints):
assert self.ledger == to_account.ledger, 'Can only transfer between accounts of the same ledger.'
tx_class = self.ledger.transaction_class
if everything:
utxos = yield self.get_utxos(**constraints)
yield self.ledger.reserve_outputs(utxos)
tx = yield tx_class.create(
inputs=[tx_class.input_class.spend(txo) for txo in utxos],
outputs=[],
funding_accounts=[self],
change_account=to_account
)
elif amount > 0:
to_address = yield to_account.change.get_or_create_usable_address()
to_hash160 = to_account.ledger.address_to_hash160(to_address)
tx = yield tx_class.create(
inputs=[],
outputs=[
tx_class.output_class.pay_pubkey_hash(amount//outputs, to_hash160)
for _ in range(outputs)
],
funding_accounts=[self],
change_account=self
)
else:
raise ValueError('An amount is required.')
if broadcast:
yield self.ledger.broadcast(tx)
else:
yield self.ledger.release_outputs(
[txi.txo_ref.txo for txi in tx.inputs]
)
2018-08-30 11:50:11 -04:00
return tx