2018-07-29 02:52:54 +02:00
|
|
|
import typing
|
2018-07-29 20:34:56 +02:00
|
|
|
from typing import Dict, Tuple, Type, Optional, Any
|
2018-06-11 15:33:32 +02:00
|
|
|
from twisted.internet import defer
|
|
|
|
|
|
|
|
from torba.mnemonic import Mnemonic
|
|
|
|
from torba.bip32 import PrivateKey, PubKey, from_extended_key_string
|
|
|
|
from torba.hash import double_sha256, aes_encrypt, aes_decrypt
|
|
|
|
|
2018-07-29 02:52:54 +02:00
|
|
|
if typing.TYPE_CHECKING:
|
|
|
|
from torba import baseledger
|
2018-06-11 15:33:32 +02:00
|
|
|
|
2018-07-29 02:52:54 +02:00
|
|
|
|
2018-07-29 19:13:40 +02:00
|
|
|
class AddressManager:
|
|
|
|
|
|
|
|
name: str
|
2018-06-11 15:33:32 +02:00
|
|
|
|
2018-07-14 23:47:18 +02:00
|
|
|
__slots__ = 'account', 'public_key', 'chain_number'
|
|
|
|
|
|
|
|
def __init__(self, account, public_key, chain_number):
|
2018-06-11 15:33:32 +02:00
|
|
|
self.account = account
|
2018-07-14 23:47:18 +02:00
|
|
|
self.public_key = public_key
|
2018-06-11 15:33:32 +02:00
|
|
|
self.chain_number = chain_number
|
|
|
|
|
2018-07-29 19:13:40 +02:00
|
|
|
@classmethod
|
|
|
|
def from_dict(cls, account: 'BaseAccount', d: dict) \
|
|
|
|
-> Tuple['AddressManager', 'AddressManager']:
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
@classmethod
|
2018-07-29 20:34:56 +02: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 19:13:40 +02:00
|
|
|
|
2018-07-14 23:47:18 +02:00
|
|
|
@property
|
|
|
|
def db(self):
|
|
|
|
return self.account.ledger.db
|
2018-06-11 15:33:32 +02:00
|
|
|
|
2018-07-29 02:52:54 +02:00
|
|
|
def _query_addresses(self, limit: int = None, max_used_times: int = None, order_by=None):
|
2018-07-14 23:47:18 +02:00
|
|
|
return self.db.get_addresses(
|
|
|
|
self.account, self.chain_number, limit, max_used_times, order_by
|
2018-06-11 15:33:32 +02:00
|
|
|
)
|
|
|
|
|
2018-07-29 19:13:40 +02:00
|
|
|
def get_private_key(self, index: int) -> PrivateKey:
|
|
|
|
raise NotImplementedError
|
|
|
|
|
2018-07-29 02:52:54 +02:00
|
|
|
def get_max_gap(self) -> defer.Deferred:
|
2018-07-26 05:29:41 +02:00
|
|
|
raise NotImplementedError
|
|
|
|
|
2018-07-29 02:52:54 +02:00
|
|
|
def ensure_address_gap(self) -> defer.Deferred:
|
2018-07-14 23:47:18 +02:00
|
|
|
raise NotImplementedError
|
|
|
|
|
2018-07-29 02:52:54 +02:00
|
|
|
def get_address_records(self, limit: int = None, only_usable: bool = False) -> defer.Deferred:
|
2018-07-14 23:47:18 +02:00
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
@defer.inlineCallbacks
|
2018-07-29 02:52:54 +02:00
|
|
|
def get_addresses(self, limit: int = None, only_usable: bool = False) -> defer.Deferred:
|
2018-07-14 23:47:18 +02:00
|
|
|
records = yield self.get_address_records(limit=limit, only_usable=only_usable)
|
|
|
|
defer.returnValue([r['address'] for r in records])
|
|
|
|
|
|
|
|
@defer.inlineCallbacks
|
2018-07-29 02:52:54 +02:00
|
|
|
def get_or_create_usable_address(self) -> defer.Deferred:
|
2018-07-14 23:47:18 +02:00
|
|
|
addresses = yield self.get_addresses(limit=1, only_usable=True)
|
|
|
|
if addresses:
|
|
|
|
defer.returnValue(addresses[0])
|
|
|
|
addresses = yield self.ensure_address_gap()
|
|
|
|
defer.returnValue(addresses[0])
|
|
|
|
|
|
|
|
|
2018-07-29 19:13:40 +02:00
|
|
|
class HierarchicalDeterministic(AddressManager):
|
2018-07-14 23:47:18 +02:00
|
|
|
""" Implements simple version of Bitcoin Hierarchical Deterministic key management. """
|
|
|
|
|
2018-07-29 19:13:40 +02:00
|
|
|
name = "deterministic-chain"
|
|
|
|
|
2018-07-14 23:47:18 +02:00
|
|
|
__slots__ = 'gap', 'maximum_uses_per_address'
|
|
|
|
|
2018-07-29 19:13:40 +02: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 23:47:18 +02:00
|
|
|
self.gap = gap
|
|
|
|
self.maximum_uses_per_address = maximum_uses_per_address
|
|
|
|
|
2018-07-29 19:13:40 +02: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 16:02:04 +02:00
|
|
|
@defer.inlineCallbacks
|
2018-07-29 02:52:54 +02:00
|
|
|
def generate_keys(self, start: int, end: int) -> defer.Deferred:
|
2018-06-11 15:33:32 +02:00
|
|
|
new_keys = []
|
2018-06-12 16:02:04 +02:00
|
|
|
for index in range(start, end+1):
|
2018-07-14 23:47:18 +02:00
|
|
|
new_keys.append((index, self.public_key.child(index)))
|
2018-06-11 15:33:32 +02:00
|
|
|
yield self.db.add_keys(
|
|
|
|
self.account, self.chain_number, new_keys
|
|
|
|
)
|
2018-06-12 16:02:04 +02:00
|
|
|
defer.returnValue([key[1].address for key in new_keys])
|
|
|
|
|
2018-07-26 05:29:41 +02:00
|
|
|
@defer.inlineCallbacks
|
2018-07-29 02:52:54 +02:00
|
|
|
def get_max_gap(self) -> defer.Deferred:
|
2018-07-26 05:29:41 +02: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
|
|
|
|
defer.returnValue(max_gap)
|
|
|
|
|
2018-06-12 16:02:04 +02:00
|
|
|
@defer.inlineCallbacks
|
2018-07-29 02:52:54 +02:00
|
|
|
def ensure_address_gap(self) -> defer.Deferred:
|
2018-07-14 23:47:18 +02:00
|
|
|
addresses = yield self._query_addresses(self.gap, None, "position DESC")
|
2018-06-11 15:33:32 +02:00
|
|
|
|
2018-06-12 16:02:04 +02:00
|
|
|
existing_gap = 0
|
|
|
|
for address in addresses:
|
|
|
|
if address['used_times'] == 0:
|
|
|
|
existing_gap += 1
|
|
|
|
else:
|
|
|
|
break
|
|
|
|
|
|
|
|
if existing_gap == self.gap:
|
|
|
|
defer.returnValue([])
|
|
|
|
|
|
|
|
start = addresses[0]['position']+1 if addresses else 0
|
|
|
|
end = start + (self.gap - existing_gap)
|
|
|
|
new_keys = yield self.generate_keys(start, end-1)
|
|
|
|
defer.returnValue(new_keys)
|
2018-06-11 15:33:32 +02:00
|
|
|
|
2018-07-29 02:52:54 +02:00
|
|
|
def get_address_records(self, limit: int = None, only_usable: bool = False):
|
2018-07-14 23:47:18 +02:00
|
|
|
return self._query_addresses(
|
|
|
|
limit, self.maximum_uses_per_address if only_usable else None,
|
|
|
|
"used_times ASC, position ASC"
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2018-07-29 19:13:40 +02:00
|
|
|
class SingleKey(AddressManager):
|
|
|
|
""" Single Key address manager always returns the same address for all operations. """
|
|
|
|
|
|
|
|
name = "single-address"
|
2018-07-14 23:47:18 +02:00
|
|
|
|
|
|
|
__slots__ = ()
|
|
|
|
|
2018-07-29 19:13:40 +02: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 20:34:56 +02:00
|
|
|
def to_dict_instance(self):
|
|
|
|
return None
|
|
|
|
|
2018-07-29 19:13:40 +02:00
|
|
|
def get_private_key(self, index: int) -> PrivateKey:
|
|
|
|
return self.account.private_key
|
|
|
|
|
2018-07-29 02:52:54 +02:00
|
|
|
def get_max_gap(self) -> defer.Deferred:
|
2018-07-26 05:29:41 +02:00
|
|
|
return defer.succeed(0)
|
|
|
|
|
2018-06-11 15:33:32 +02:00
|
|
|
@defer.inlineCallbacks
|
2018-07-29 02:52:54 +02:00
|
|
|
def ensure_address_gap(self) -> defer.Deferred:
|
2018-07-14 23:47:18 +02:00
|
|
|
exists = yield self.get_address_records()
|
|
|
|
if not exists:
|
|
|
|
yield self.db.add_keys(
|
|
|
|
self.account, self.chain_number, [(0, self.public_key)]
|
|
|
|
)
|
|
|
|
defer.returnValue([self.public_key.address])
|
|
|
|
defer.returnValue([])
|
|
|
|
|
2018-07-29 02:52:54 +02:00
|
|
|
def get_address_records(self, limit: int = None, only_usable: bool = False) -> defer.Deferred:
|
2018-07-14 23:47:18 +02:00
|
|
|
return self._query_addresses()
|
2018-06-11 15:33:32 +02:00
|
|
|
|
|
|
|
|
2018-07-29 02:52:54 +02:00
|
|
|
class BaseAccount:
|
2018-06-11 15:33:32 +02:00
|
|
|
|
|
|
|
mnemonic_class = Mnemonic
|
|
|
|
private_key_class = PrivateKey
|
|
|
|
public_key_class = PubKey
|
2018-07-29 20:34:56 +02:00
|
|
|
address_generators: Dict[str, Type[AddressManager]] = {
|
2018-07-29 19:13:40 +02:00
|
|
|
SingleKey.name: SingleKey,
|
|
|
|
HierarchicalDeterministic.name: HierarchicalDeterministic,
|
|
|
|
}
|
2018-06-11 15:33:32 +02:00
|
|
|
|
2018-07-29 19:13:40 +02:00
|
|
|
def __init__(self, ledger: 'baseledger.BaseLedger', name: str, seed: str, encrypted: bool,
|
|
|
|
private_key: PrivateKey, public_key: PubKey, address_generator: dict
|
2018-07-29 02:52:54 +02:00
|
|
|
) -> None:
|
2018-06-12 16:02:04 +02:00
|
|
|
self.ledger = ledger
|
2018-07-14 23:47:18 +02:00
|
|
|
self.name = name
|
2018-06-12 16:02:04 +02:00
|
|
|
self.seed = seed
|
|
|
|
self.encrypted = encrypted
|
|
|
|
self.private_key = private_key
|
|
|
|
self.public_key = public_key
|
2018-07-29 19:13:40 +02:00
|
|
|
generator_name = address_generator.get('name', HierarchicalDeterministic.name)
|
2018-07-29 20:34:56 +02:00
|
|
|
self.address_generator = self.address_generators[generator_name]
|
2018-07-29 19:13:40 +02:00
|
|
|
self.receiving, self.change = self.address_generator.from_dict(self, address_generator)
|
|
|
|
self.address_managers = {self.receiving, self.change}
|
2018-06-14 02:57:57 +02:00
|
|
|
ledger.add_account(self)
|
2018-06-11 15:33:32 +02:00
|
|
|
|
|
|
|
@classmethod
|
2018-07-29 19:13:40 +02:00
|
|
|
def generate(cls, ledger: 'baseledger.BaseLedger', password: str, address_generator: dict = None):
|
2018-06-11 15:33:32 +02:00
|
|
|
seed = cls.mnemonic_class().make_seed()
|
2018-07-29 19:13:40 +02:00
|
|
|
return cls.from_seed(ledger, seed, password, address_generator or {})
|
2018-06-11 15:33:32 +02:00
|
|
|
|
|
|
|
@classmethod
|
2018-07-29 19:13:40 +02:00
|
|
|
def from_seed(cls, ledger: 'baseledger.BaseLedger', seed: str, password: str, address_generator: dict):
|
2018-06-11 15:33:32 +02:00
|
|
|
private_key = cls.get_private_key_from_seed(ledger, seed, password)
|
|
|
|
return cls(
|
2018-07-14 23:47:18 +02:00
|
|
|
ledger=ledger, name='Account #{}'.format(private_key.public_key.address),
|
2018-07-29 19:13:40 +02:00
|
|
|
seed=seed, encrypted=False,
|
2018-06-11 15:33:32 +02:00
|
|
|
private_key=private_key,
|
2018-07-14 23:47:18 +02:00
|
|
|
public_key=private_key.public_key,
|
2018-07-29 19:13:40 +02:00
|
|
|
address_generator=address_generator
|
2018-06-11 15:33:32 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
@classmethod
|
2018-07-29 02:52:54 +02:00
|
|
|
def get_private_key_from_seed(cls, ledger: 'baseledger.BaseLedger', seed: str, password: str):
|
2018-06-11 15:33:32 +02:00
|
|
|
return cls.private_key_class.from_seed(
|
|
|
|
ledger, cls.mnemonic_class.mnemonic_to_seed(seed, password)
|
|
|
|
)
|
|
|
|
|
|
|
|
@classmethod
|
2018-07-29 02:52:54 +02:00
|
|
|
def from_dict(cls, ledger: 'baseledger.BaseLedger', d: dict):
|
2018-07-14 23:47:18 +02:00
|
|
|
if not d['encrypted'] and d['private_key']:
|
2018-06-11 15:33:32 +02:00
|
|
|
private_key = from_extended_key_string(ledger, d['private_key'])
|
|
|
|
public_key = private_key.public_key
|
|
|
|
else:
|
|
|
|
private_key = d['private_key']
|
|
|
|
public_key = from_extended_key_string(ledger, d['public_key'])
|
2018-07-29 19:13:40 +02:00
|
|
|
return cls(
|
2018-06-11 15:33:32 +02:00
|
|
|
ledger=ledger,
|
2018-07-14 23:47:18 +02:00
|
|
|
name=d['name'],
|
2018-06-11 15:33:32 +02:00
|
|
|
seed=d['seed'],
|
|
|
|
encrypted=d['encrypted'],
|
|
|
|
private_key=private_key,
|
|
|
|
public_key=public_key,
|
2018-07-29 19:13:40 +02:00
|
|
|
address_generator=d['address_generator']
|
2018-06-11 15:33:32 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
def to_dict(self):
|
2018-07-14 23:47:18 +02:00
|
|
|
private_key = self.private_key
|
|
|
|
if not self.encrypted and self.private_key:
|
2018-07-15 03:34:07 +02:00
|
|
|
private_key = self.private_key.extended_key_string()
|
2018-07-29 19:13:40 +02:00
|
|
|
return {
|
2018-06-11 15:33:32 +02:00
|
|
|
'ledger': self.ledger.get_id(),
|
2018-07-14 23:47:18 +02:00
|
|
|
'name': self.name,
|
2018-06-11 15:33:32 +02:00
|
|
|
'seed': self.seed,
|
|
|
|
'encrypted': self.encrypted,
|
2018-07-14 23:47:18 +02:00
|
|
|
'private_key': private_key,
|
2018-07-15 03:34:07 +02:00
|
|
|
'public_key': self.public_key.extended_key_string(),
|
2018-07-29 19:13:40 +02:00
|
|
|
'address_generator': self.address_generator.to_dict(self.receiving, self.change)
|
2018-06-11 15:33:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
def decrypt(self, password):
|
|
|
|
assert self.encrypted, "Key is not encrypted."
|
|
|
|
secret = double_sha256(password)
|
|
|
|
self.seed = aes_decrypt(secret, self.seed)
|
|
|
|
self.private_key = from_extended_key_string(self.ledger, aes_decrypt(secret, self.private_key))
|
|
|
|
self.encrypted = False
|
|
|
|
|
|
|
|
def encrypt(self, password):
|
|
|
|
assert not self.encrypted, "Key is already encrypted."
|
|
|
|
secret = double_sha256(password)
|
|
|
|
self.seed = aes_encrypt(secret, self.seed)
|
|
|
|
self.private_key = aes_encrypt(secret, self.private_key.extended_key_string())
|
|
|
|
self.encrypted = True
|
|
|
|
|
|
|
|
@defer.inlineCallbacks
|
2018-06-12 16:02:04 +02:00
|
|
|
def ensure_address_gap(self):
|
2018-06-11 15:33:32 +02:00
|
|
|
addresses = []
|
2018-07-29 19:13:40 +02:00
|
|
|
for address_manager in self.address_managers:
|
|
|
|
new_addresses = yield address_manager.ensure_address_gap()
|
2018-06-11 15:33:32 +02:00
|
|
|
addresses.extend(new_addresses)
|
|
|
|
defer.returnValue(addresses)
|
|
|
|
|
2018-07-14 23:47:18 +02:00
|
|
|
@defer.inlineCallbacks
|
2018-07-29 02:52:54 +02:00
|
|
|
def get_addresses(self, limit: int = None, max_used_times: int = None) -> defer.Deferred:
|
2018-07-14 23:47:18 +02:00
|
|
|
records = yield self.get_address_records(limit, max_used_times)
|
|
|
|
defer.returnValue([r['address'] for r in records])
|
2018-06-12 16:02:04 +02:00
|
|
|
|
2018-07-29 02:52:54 +02:00
|
|
|
def get_address_records(self, limit: int = None, max_used_times: int = None) -> defer.Deferred:
|
2018-07-14 23:47:18 +02:00
|
|
|
return self.ledger.db.get_addresses(self, None, limit, max_used_times)
|
2018-06-12 16:02:04 +02:00
|
|
|
|
2018-07-29 02:52:54 +02:00
|
|
|
def get_private_key(self, chain: int, index: int) -> PrivateKey:
|
2018-06-11 15:33:32 +02:00
|
|
|
assert not self.encrypted, "Cannot get private key on encrypted wallet account."
|
2018-07-29 19:13:40 +02:00
|
|
|
address_manager = {0: self.receiving, 1: self.change}[chain]
|
|
|
|
return address_manager.get_private_key(index)
|
2018-06-11 15:33:32 +02:00
|
|
|
|
2018-07-29 02:52:54 +02:00
|
|
|
def get_balance(self, confirmations: int = 6, **constraints):
|
2018-07-17 05:58:29 +02:00
|
|
|
if confirmations > 0:
|
2018-07-14 23:47:18 +02:00
|
|
|
height = self.ledger.headers.height - (confirmations-1)
|
2018-07-23 04:52:21 +02:00
|
|
|
constraints.update({'height__lte': height, 'height__gt': 0})
|
2018-07-17 05:58:29 +02:00
|
|
|
return self.ledger.db.get_balance_for_account(self, **constraints)
|
2018-07-10 02:22:04 +02:00
|
|
|
|
2018-07-26 05:29:41 +02:00
|
|
|
@defer.inlineCallbacks
|
|
|
|
def get_max_gap(self):
|
|
|
|
change_gap = yield self.change.get_max_gap()
|
|
|
|
receiving_gap = yield self.receiving.get_max_gap()
|
|
|
|
defer.returnValue({
|
|
|
|
'max_change_gap': change_gap,
|
|
|
|
'max_receiving_gap': receiving_gap,
|
|
|
|
})
|
|
|
|
|
2018-07-10 02:22:04 +02:00
|
|
|
def get_unspent_outputs(self, **constraints):
|
|
|
|
return self.ledger.db.get_utxos_for_account(self, **constraints)
|