2019-03-11 12:04:06 -04:00
|
|
|
import os
|
2018-05-25 02:03:25 -04:00
|
|
|
import json
|
2019-03-11 12:04:06 -04:00
|
|
|
import zlib
|
2020-05-06 10:53:31 -04:00
|
|
|
import asyncio
|
2019-10-16 01:18:39 -04:00
|
|
|
import logging
|
2020-05-06 10:53:31 -04:00
|
|
|
from typing import List, Sequence, Tuple, Optional, Iterable
|
2019-03-11 12:12:26 -04:00
|
|
|
from hashlib import sha256
|
2019-03-11 12:04:06 -04:00
|
|
|
from operator import attrgetter
|
2020-05-01 09:33:10 -04:00
|
|
|
from decimal import Decimal
|
|
|
|
|
2020-05-06 10:53:31 -04:00
|
|
|
|
2020-05-01 23:25:07 -04:00
|
|
|
from lbry.db import Database, SPENDABLE_TYPE_CODES
|
2020-05-01 09:33:10 -04:00
|
|
|
from lbry.blockchain.ledger import Ledger
|
|
|
|
from lbry.constants import COIN, NULL_HASH32
|
|
|
|
from lbry.blockchain.transaction import Transaction, Input, Output
|
|
|
|
from lbry.blockchain.dewies import dewies_to_lbc
|
2019-12-31 14:52:57 -05:00
|
|
|
from lbry.crypto.crypt import better_aes_encrypt, better_aes_decrypt
|
2020-05-01 09:33:10 -04:00
|
|
|
from lbry.crypto.bip32 import PubKey, PrivateKey
|
|
|
|
from lbry.schema.claim import Claim
|
|
|
|
from lbry.schema.purchase import Purchase
|
|
|
|
from lbry.error import InsufficientFundsError, KeyFeeAboveMaxAllowedError
|
|
|
|
|
2020-05-06 10:53:31 -04:00
|
|
|
from .account import Account, SingleKey, HierarchicalDeterministic
|
2020-05-01 09:33:10 -04:00
|
|
|
from .coinselection import CoinSelector, OutputEffectiveAmountEstimator
|
2020-05-06 10:53:31 -04:00
|
|
|
from .storage import WalletStorage
|
|
|
|
from .preferences import TimestampedPreferences
|
2018-05-25 02:03:25 -04:00
|
|
|
|
|
|
|
|
2019-10-16 01:18:39 -04:00
|
|
|
log = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
ENCRYPT_ON_DISK = 'encrypt-on-disk'
|
|
|
|
|
|
|
|
|
2018-05-25 02:03:25 -04:00
|
|
|
class Wallet:
|
|
|
|
""" The primary role of Wallet is to encapsulate a collection
|
|
|
|
of accounts (seed/private keys) and the spending rules / settings
|
|
|
|
for the coins attached to those accounts. Wallets are represented
|
|
|
|
by physical files on the filesystem.
|
|
|
|
"""
|
|
|
|
|
2020-05-06 10:53:31 -04:00
|
|
|
def __init__(self, ledger: Ledger, db: Database, name: str, storage: WalletStorage, preferences: dict):
|
2020-05-01 09:33:10 -04:00
|
|
|
self.ledger = ledger
|
|
|
|
self.db = db
|
2018-05-25 02:03:25 -04:00
|
|
|
self.name = name
|
2020-05-06 10:53:31 -04:00
|
|
|
self.storage = storage
|
2019-10-12 19:33:16 -04:00
|
|
|
self.preferences = TimestampedPreferences(preferences or {})
|
2020-05-06 10:53:31 -04:00
|
|
|
self.encryption_password: Optional[str] = None
|
2020-01-27 13:17:59 -05:00
|
|
|
self.id = self.get_id()
|
2018-05-25 02:03:25 -04:00
|
|
|
|
2020-05-06 10:53:31 -04:00
|
|
|
self.utxo_lock = asyncio.Lock()
|
2020-05-01 09:33:10 -04:00
|
|
|
|
2020-05-06 10:53:31 -04:00
|
|
|
self.accounts = AccountListManager(self)
|
|
|
|
self.claims = ClaimListManager(self)
|
|
|
|
self.streams = StreamListManager(self)
|
|
|
|
self.channels = ChannelListManager(self)
|
|
|
|
self.collections = CollectionListManager(self)
|
|
|
|
self.purchases = PurchaseListManager(self)
|
|
|
|
self.supports = SupportListManager(self)
|
2020-05-01 09:33:10 -04:00
|
|
|
|
2020-05-06 10:53:31 -04:00
|
|
|
def get_id(self):
|
|
|
|
return os.path.basename(self.storage.path) if self.storage.path else self.name
|
2020-05-01 09:33:10 -04:00
|
|
|
|
2020-05-06 10:53:31 -04:00
|
|
|
@classmethod
|
|
|
|
async def create(cls, ledger: Ledger, db: Database, path: str, name: str, create_account=False, single_key=False):
|
|
|
|
wallet = cls(ledger, db, name, WalletStorage(path), {})
|
|
|
|
if create_account:
|
|
|
|
wallet.accounts.generate(address_generator={
|
|
|
|
'name': SingleKey.name if single_key else HierarchicalDeterministic.name
|
|
|
|
})
|
|
|
|
await wallet.save()
|
|
|
|
return wallet
|
2020-05-01 09:33:10 -04:00
|
|
|
|
2020-05-06 10:53:31 -04:00
|
|
|
@classmethod
|
|
|
|
async def from_path(cls, ledger: Ledger, db: Database, path: str):
|
|
|
|
return await cls.from_storage(ledger, db, WalletStorage(path))
|
2020-05-01 09:33:10 -04:00
|
|
|
|
2018-05-25 02:03:25 -04:00
|
|
|
@classmethod
|
2020-05-06 10:53:31 -04:00
|
|
|
async def from_storage(cls, ledger: Ledger, db: Database, storage: WalletStorage) -> 'Wallet':
|
|
|
|
json_dict = await storage.read()
|
2020-05-01 09:33:10 -04:00
|
|
|
if 'ledger' in json_dict and json_dict['ledger'] != ledger.get_id():
|
|
|
|
raise ValueError(
|
|
|
|
f"Using ledger {ledger.get_id()} but wallet is {json_dict['ledger']}."
|
|
|
|
)
|
2018-08-07 21:31:29 -04:00
|
|
|
wallet = cls(
|
2020-05-01 09:33:10 -04:00
|
|
|
ledger, db,
|
2018-05-25 02:03:25 -04:00
|
|
|
name=json_dict.get('name', 'Wallet'),
|
2020-05-06 10:53:31 -04:00
|
|
|
storage=storage,
|
2019-10-12 19:33:16 -04:00
|
|
|
preferences=json_dict.get('preferences', {}),
|
2018-05-25 02:03:25 -04:00
|
|
|
)
|
2020-05-06 10:53:31 -04:00
|
|
|
for account_dict in json_dict.get('accounts', []):
|
|
|
|
wallet.accounts.add_from_dict(account_dict)
|
2018-08-07 21:31:29 -04:00
|
|
|
return wallet
|
2018-05-25 02:03:25 -04:00
|
|
|
|
2019-10-16 01:18:39 -04:00
|
|
|
def to_dict(self, encrypt_password: str = None):
|
2018-05-25 02:03:25 -04:00
|
|
|
return {
|
2020-05-06 10:53:31 -04:00
|
|
|
'version': WalletStorage.VERSION,
|
2020-05-01 09:33:10 -04:00
|
|
|
'ledger': self.ledger.get_id(),
|
2020-05-06 10:53:31 -04:00
|
|
|
'name': self.name,
|
2019-10-12 19:33:16 -04:00
|
|
|
'preferences': self.preferences.data,
|
2019-10-16 01:18:39 -04:00
|
|
|
'accounts': [a.to_dict(encrypt_password) for a in self.accounts]
|
2018-05-25 02:03:25 -04:00
|
|
|
}
|
|
|
|
|
2020-05-06 10:53:31 -04:00
|
|
|
async def save(self):
|
2019-10-18 13:38:51 -04:00
|
|
|
if self.preferences.get(ENCRYPT_ON_DISK, False):
|
2019-10-22 09:41:04 -04:00
|
|
|
if self.encryption_password is not None:
|
2020-05-06 10:53:31 -04:00
|
|
|
return await self.storage.write(self.to_dict(encrypt_password=self.encryption_password))
|
2019-10-18 13:38:51 -04:00
|
|
|
elif not self.is_locked:
|
2019-10-16 01:18:39 -04:00
|
|
|
log.warning(
|
|
|
|
"Disk encryption requested but no password available for encryption. "
|
|
|
|
"Saving wallet in an unencrypted state."
|
|
|
|
)
|
2020-05-06 10:53:31 -04:00
|
|
|
return await self.storage.write(self.to_dict())
|
2018-05-25 02:03:25 -04:00
|
|
|
|
2019-03-11 12:04:06 -04:00
|
|
|
@property
|
2019-03-11 12:30:32 -04:00
|
|
|
def hash(self) -> bytes:
|
2019-03-11 12:04:06 -04:00
|
|
|
h = sha256()
|
2019-10-16 12:27:18 -04:00
|
|
|
if self.preferences.get(ENCRYPT_ON_DISK, False):
|
2019-10-22 09:41:04 -04:00
|
|
|
assert self.encryption_password is not None, \
|
2019-10-16 12:27:18 -04:00
|
|
|
"Encryption is enabled but no password is available, cannot generate hash."
|
|
|
|
h.update(self.encryption_password.encode())
|
2019-10-12 19:33:16 -04:00
|
|
|
h.update(self.preferences.hash)
|
2019-03-11 12:04:06 -04:00
|
|
|
for account in sorted(self.accounts, key=attrgetter('id')):
|
|
|
|
h.update(account.hash)
|
|
|
|
return h.digest()
|
|
|
|
|
|
|
|
def pack(self, password):
|
2019-10-16 01:18:39 -04:00
|
|
|
assert not self.is_locked, "Cannot pack a wallet with locked/encrypted accounts."
|
2019-03-11 12:04:06 -04:00
|
|
|
new_data = json.dumps(self.to_dict())
|
|
|
|
new_data_compressed = zlib.compress(new_data.encode())
|
|
|
|
return better_aes_encrypt(password, new_data_compressed)
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def unpack(cls, password, encrypted):
|
|
|
|
decrypted = better_aes_decrypt(password, encrypted)
|
|
|
|
decompressed = zlib.decompress(decrypted)
|
|
|
|
return json.loads(decompressed)
|
|
|
|
|
2020-05-01 09:33:10 -04:00
|
|
|
def merge(self, password: str, data: str) -> List[Account]:
|
2019-10-16 01:18:39 -04:00
|
|
|
assert not self.is_locked, "Cannot sync apply on a locked wallet."
|
2019-10-12 19:33:16 -04:00
|
|
|
added_accounts = []
|
|
|
|
decrypted_data = self.unpack(password, data)
|
|
|
|
self.preferences.merge(decrypted_data.get('preferences', {}))
|
|
|
|
for account_dict in decrypted_data['accounts']:
|
2020-05-01 09:33:10 -04:00
|
|
|
_, _, pubkey = Account.keys_from_dict(self.ledger, account_dict)
|
2019-10-12 19:33:16 -04:00
|
|
|
account_id = pubkey.address
|
|
|
|
local_match = None
|
|
|
|
for local_account in self.accounts:
|
|
|
|
if account_id == local_account.id:
|
|
|
|
local_match = local_account
|
|
|
|
break
|
|
|
|
if local_match is not None:
|
|
|
|
local_match.merge(account_dict)
|
|
|
|
else:
|
2020-05-01 09:33:10 -04:00
|
|
|
added_accounts.append(
|
2020-05-06 10:53:31 -04:00
|
|
|
self.accounts.add_from_dict(account_dict)
|
2020-05-01 09:33:10 -04:00
|
|
|
)
|
2019-10-12 19:33:16 -04:00
|
|
|
return added_accounts
|
|
|
|
|
2019-10-13 23:43:06 -04:00
|
|
|
@property
|
|
|
|
def is_locked(self) -> bool:
|
|
|
|
for account in self.accounts:
|
|
|
|
if account.encrypted:
|
|
|
|
return True
|
|
|
|
return False
|
|
|
|
|
|
|
|
def unlock(self, password):
|
|
|
|
for account in self.accounts:
|
|
|
|
if account.encrypted:
|
2019-10-16 01:18:39 -04:00
|
|
|
if not account.decrypt(password):
|
|
|
|
return False
|
2019-10-22 09:41:04 -04:00
|
|
|
self.encryption_password = password
|
2019-10-14 12:53:02 -04:00
|
|
|
return True
|
2019-10-13 23:43:06 -04:00
|
|
|
|
|
|
|
def lock(self):
|
2019-10-16 01:18:39 -04:00
|
|
|
assert self.encryption_password is not None, "Cannot lock an unencrypted wallet, encrypt first."
|
2019-10-13 23:43:06 -04:00
|
|
|
for account in self.accounts:
|
|
|
|
if not account.encrypted:
|
2019-10-16 01:18:39 -04:00
|
|
|
account.encrypt(self.encryption_password)
|
2019-10-14 12:53:02 -04:00
|
|
|
return True
|
2019-10-13 23:43:06 -04:00
|
|
|
|
|
|
|
@property
|
|
|
|
def is_encrypted(self) -> bool:
|
2019-10-16 01:18:39 -04:00
|
|
|
return self.is_locked or self.preferences.get(ENCRYPT_ON_DISK, False)
|
2019-10-13 23:43:06 -04:00
|
|
|
|
|
|
|
def decrypt(self):
|
2019-10-16 01:18:39 -04:00
|
|
|
assert not self.is_locked, "Cannot decrypt a locked wallet, unlock first."
|
|
|
|
self.preferences[ENCRYPT_ON_DISK] = False
|
2019-10-13 23:43:06 -04:00
|
|
|
self.save()
|
2019-10-14 12:53:02 -04:00
|
|
|
return True
|
2019-10-13 23:43:06 -04:00
|
|
|
|
|
|
|
def encrypt(self, password):
|
2019-10-16 01:18:39 -04:00
|
|
|
assert not self.is_locked, "Cannot re-encrypt a locked wallet, unlock first."
|
2019-10-18 12:43:28 -04:00
|
|
|
assert password, "Cannot encrypt with blank password."
|
2019-10-16 01:18:39 -04:00
|
|
|
self.encryption_password = password
|
|
|
|
self.preferences[ENCRYPT_ON_DISK] = True
|
2019-10-13 23:43:06 -04:00
|
|
|
self.save()
|
2019-10-14 12:53:02 -04:00
|
|
|
return True
|
2019-10-13 23:43:06 -04:00
|
|
|
|
2020-05-06 10:53:31 -04:00
|
|
|
@property
|
|
|
|
def has_accounts(self):
|
|
|
|
return len(self.accounts) > 0
|
|
|
|
|
|
|
|
async def _get_account_and_address_info_for_address(self, address):
|
|
|
|
match = await self.db.get_address(accounts=self.accounts, address=address)
|
|
|
|
if match:
|
|
|
|
for account in self.accounts:
|
|
|
|
if match['account'] == account.public_key.address:
|
|
|
|
return account, match
|
|
|
|
|
|
|
|
async def get_private_key_for_address(self, address) -> Optional[PrivateKey]:
|
|
|
|
match = await self._get_account_and_address_info_for_address(address)
|
|
|
|
if match:
|
|
|
|
account, address_info = match
|
|
|
|
return account.get_private_key(address_info['chain'], address_info['pubkey'].n)
|
|
|
|
return None
|
|
|
|
|
|
|
|
async def get_public_key_for_address(self, address) -> Optional[PubKey]:
|
|
|
|
match = await self._get_account_and_address_info_for_address(address)
|
|
|
|
if match:
|
|
|
|
_, address_info = match
|
|
|
|
return address_info['pubkey']
|
|
|
|
return None
|
|
|
|
|
|
|
|
async def get_account_for_address(self, address):
|
|
|
|
match = await self._get_account_and_address_info_for_address(address)
|
|
|
|
if match:
|
|
|
|
return match[0]
|
|
|
|
|
|
|
|
async def save_max_gap(self):
|
|
|
|
gap_changed = False
|
|
|
|
for account in self.accounts:
|
|
|
|
if await account.save_max_gap():
|
|
|
|
gap_changed = True
|
|
|
|
if gap_changed:
|
|
|
|
self.save()
|
|
|
|
|
2020-05-01 09:33:10 -04:00
|
|
|
async def get_effective_amount_estimators(self, funding_accounts: Iterable[Account]):
|
|
|
|
estimators = []
|
2020-05-01 23:25:07 -04:00
|
|
|
utxos = await self.db.get_utxos(
|
|
|
|
accounts=funding_accounts,
|
|
|
|
txo_type__in=SPENDABLE_TYPE_CODES
|
|
|
|
)
|
|
|
|
for utxo in utxos[0]:
|
2020-05-01 09:33:10 -04:00
|
|
|
estimators.append(OutputEffectiveAmountEstimator(self.ledger, utxo))
|
|
|
|
return estimators
|
|
|
|
|
|
|
|
async def get_spendable_utxos(self, amount: int, funding_accounts: Iterable[Account]):
|
2020-05-06 10:53:31 -04:00
|
|
|
async with self.utxo_lock:
|
|
|
|
txos = await self.get_effective_amount_estimators(funding_accounts)
|
|
|
|
fee = Output.pay_pubkey_hash(COIN, NULL_HASH32).get_fee(self.ledger)
|
|
|
|
selector = CoinSelector(amount, fee)
|
|
|
|
spendables = selector.select(txos, self.ledger.coin_selection_strategy)
|
|
|
|
if spendables:
|
|
|
|
await self.db.reserve_outputs(s.txo for s in spendables)
|
|
|
|
return spendables
|
|
|
|
|
|
|
|
async def list_transactions(self, **constraints):
|
|
|
|
return txs_to_dict(await self.db.get_transactions(
|
|
|
|
include_is_my_output=True, include_is_spent=True,
|
|
|
|
**constraints
|
|
|
|
), self.ledger)
|
2020-05-01 09:33:10 -04:00
|
|
|
|
|
|
|
async def create_transaction(self, inputs: Iterable[Input], outputs: Iterable[Output],
|
|
|
|
funding_accounts: Iterable[Account], change_account: Account,
|
|
|
|
sign: bool = True):
|
|
|
|
""" Find optimal set of inputs when only outputs are provided; add change
|
|
|
|
outputs if only inputs are provided or if inputs are greater than outputs. """
|
|
|
|
|
|
|
|
tx = Transaction() \
|
|
|
|
.add_inputs(inputs) \
|
|
|
|
.add_outputs(outputs)
|
|
|
|
|
|
|
|
# value of the outputs plus associated fees
|
|
|
|
cost = (
|
|
|
|
tx.get_base_fee(self.ledger) +
|
|
|
|
tx.get_total_output_sum(self.ledger)
|
|
|
|
)
|
|
|
|
# value of the inputs less the cost to spend those inputs
|
|
|
|
payment = tx.get_effective_input_sum(self.ledger)
|
|
|
|
|
|
|
|
try:
|
|
|
|
|
|
|
|
for _ in range(5):
|
|
|
|
|
|
|
|
if payment < cost:
|
|
|
|
deficit = cost - payment
|
|
|
|
spendables = await self.get_spendable_utxos(deficit, funding_accounts)
|
|
|
|
if not spendables:
|
|
|
|
raise InsufficientFundsError()
|
|
|
|
payment += sum(s.effective_amount for s in spendables)
|
|
|
|
tx.add_inputs(s.txi for s in spendables)
|
|
|
|
|
|
|
|
cost_of_change = (
|
|
|
|
tx.get_base_fee(self.ledger) +
|
|
|
|
Output.pay_pubkey_hash(COIN, NULL_HASH32).get_fee(self.ledger)
|
|
|
|
)
|
|
|
|
if payment > cost:
|
|
|
|
change = payment - cost
|
|
|
|
if change > cost_of_change:
|
|
|
|
change_address = await change_account.change.get_or_create_usable_address()
|
|
|
|
change_hash160 = change_account.ledger.address_to_hash160(change_address)
|
|
|
|
change_amount = change - cost_of_change
|
|
|
|
change_output = Output.pay_pubkey_hash(change_amount, change_hash160)
|
|
|
|
change_output.is_internal_transfer = True
|
|
|
|
tx.add_outputs([Output.pay_pubkey_hash(change_amount, change_hash160)])
|
|
|
|
|
|
|
|
if tx._outputs:
|
|
|
|
break
|
|
|
|
# this condition and the outer range(5) loop cover an edge case
|
|
|
|
# whereby a single input is just enough to cover the fee and
|
|
|
|
# has some change left over, but the change left over is less
|
|
|
|
# than the cost_of_change: thus the input is completely
|
|
|
|
# consumed and no output is added, which is an invalid tx.
|
|
|
|
# to be able to spend this input we must increase the cost
|
|
|
|
# of the TX and run through the balance algorithm a second time
|
|
|
|
# adding an extra input and change output, making tx valid.
|
|
|
|
# we do this 5 times in case the other UTXOs added are also
|
|
|
|
# less than the fee, after 5 attempts we give up and go home
|
|
|
|
cost += cost_of_change + 1
|
|
|
|
|
|
|
|
if sign:
|
|
|
|
await self.sign(tx)
|
|
|
|
|
|
|
|
except Exception as e:
|
|
|
|
log.exception('Failed to create transaction:')
|
|
|
|
await self.db.release_tx(tx)
|
|
|
|
raise e
|
|
|
|
|
|
|
|
return tx
|
|
|
|
|
|
|
|
async def sign(self, tx):
|
|
|
|
for i, txi in enumerate(tx._inputs):
|
|
|
|
assert txi.script is not None
|
|
|
|
assert txi.txo_ref.txo is not None
|
|
|
|
txo_script = txi.txo_ref.txo.script
|
|
|
|
if txo_script.is_pay_pubkey_hash:
|
|
|
|
address = self.ledger.hash160_to_address(txo_script.values['pubkey_hash'])
|
|
|
|
private_key = await self.get_private_key_for_address(address)
|
|
|
|
assert private_key is not None, 'Cannot find private key for signing output.'
|
|
|
|
serialized = tx._serialize_for_signature(i)
|
|
|
|
txi.script.values['signature'] = \
|
|
|
|
private_key.sign(serialized) + bytes((tx.signature_hash_type(1),))
|
|
|
|
txi.script.values['pubkey'] = private_key.public_key.pubkey_bytes
|
|
|
|
txi.script.generate()
|
|
|
|
else:
|
|
|
|
raise NotImplementedError("Don't know how to spend this output.")
|
|
|
|
tx._reset()
|
|
|
|
|
2020-05-06 10:53:31 -04:00
|
|
|
async def pay(self, amount: int, address: bytes, funding_accounts: List[Account], change_account: Account):
|
|
|
|
output = Output.pay_pubkey_hash(amount, self.ledger.address_to_hash160(address))
|
|
|
|
return await self.create_transaction([], [output], funding_accounts, change_account)
|
|
|
|
|
|
|
|
async def _report_state(self):
|
|
|
|
try:
|
|
|
|
for account in self.accounts:
|
|
|
|
balance = dewies_to_lbc(await account.get_balance(include_claims=True))
|
|
|
|
_, channel_count = await account.get_channels(limit=1)
|
|
|
|
claim_count = await account.get_claim_count()
|
|
|
|
if isinstance(account.receiving, SingleKey):
|
|
|
|
log.info("Loaded single key account %s with %s LBC. "
|
|
|
|
"%d channels, %d certificates and %d claims",
|
|
|
|
account.id, balance, channel_count, len(account.channel_keys), claim_count)
|
|
|
|
else:
|
|
|
|
total_receiving = len(await account.receiving.get_addresses())
|
|
|
|
total_change = len(await account.change.get_addresses())
|
|
|
|
log.info("Loaded account %s with %s LBC, %d receiving addresses (gap: %d), "
|
|
|
|
"%d change addresses (gap: %d), %d channels, %d certificates and %d claims. ",
|
|
|
|
account.id, balance, total_receiving, account.receiving.gap, total_change,
|
|
|
|
account.change.gap, channel_count, len(account.channel_keys), claim_count)
|
|
|
|
except Exception as err:
|
|
|
|
if isinstance(err, asyncio.CancelledError): # TODO: remove when updated to 3.8
|
|
|
|
raise
|
|
|
|
log.exception(
|
|
|
|
'Failed to display wallet state, please file issue '
|
|
|
|
'for this bug along with the traceback you see below:')
|
|
|
|
|
|
|
|
|
|
|
|
class AccountListManager:
|
|
|
|
__slots__ = 'wallet', '_accounts'
|
|
|
|
|
|
|
|
def __init__(self, wallet: Wallet):
|
|
|
|
self.wallet = wallet
|
|
|
|
self._accounts: List[Account] = []
|
|
|
|
|
|
|
|
def __len__(self):
|
|
|
|
return self._accounts.__len__()
|
|
|
|
|
|
|
|
def __iter__(self):
|
|
|
|
return self._accounts.__iter__()
|
|
|
|
|
|
|
|
def __getitem__(self, account_id: str) -> Account:
|
|
|
|
for account in self:
|
|
|
|
if account.id == account_id:
|
|
|
|
return account
|
|
|
|
raise ValueError(f"Couldn't find account: {account_id}.")
|
|
|
|
|
|
|
|
@property
|
|
|
|
def default(self) -> Optional[Account]:
|
|
|
|
for account in self:
|
|
|
|
return account
|
2020-05-01 09:33:10 -04:00
|
|
|
|
2020-05-06 10:53:31 -04:00
|
|
|
def generate(self, name: str = None, address_generator: dict = None) -> Account:
|
|
|
|
account = Account.generate(self.wallet.ledger, self.wallet.db, name, address_generator)
|
|
|
|
self._accounts.append(account)
|
|
|
|
return account
|
|
|
|
|
|
|
|
def add_from_dict(self, account_dict: dict) -> Account:
|
|
|
|
account = Account.from_dict(self.wallet.ledger, self.wallet.db, account_dict)
|
|
|
|
self._accounts.append(account)
|
|
|
|
return account
|
|
|
|
|
|
|
|
async def remove(self, account_id: str) -> Account:
|
|
|
|
account = self[account_id]
|
|
|
|
self._accounts.remove(account)
|
|
|
|
await self.wallet.save()
|
|
|
|
return account
|
|
|
|
|
|
|
|
def get_or_none(self, account_id: str) -> Optional[Account]:
|
|
|
|
if account_id is not None:
|
|
|
|
return self[account_id]
|
|
|
|
|
|
|
|
def get_or_default(self, account_id: str) -> Optional[Account]:
|
|
|
|
if account_id is None:
|
|
|
|
return self.default
|
|
|
|
return self[account_id]
|
|
|
|
|
|
|
|
def get_or_all(self, account_ids: List[str]) -> List[Account]:
|
|
|
|
return [self[account_id] for account_id in account_ids] if account_ids else self._accounts
|
|
|
|
|
|
|
|
async def get_account_details(self, **kwargs):
|
|
|
|
accounts = []
|
|
|
|
for i, account in enumerate(self._accounts):
|
|
|
|
details = await account.get_details(**kwargs)
|
|
|
|
details['is_default'] = i == 0
|
|
|
|
accounts.append(details)
|
|
|
|
return accounts
|
|
|
|
|
|
|
|
|
|
|
|
class BaseListManager:
|
|
|
|
__slots__ = 'wallet', 'db'
|
|
|
|
|
|
|
|
def __init__(self, wallet: Wallet):
|
|
|
|
self.wallet = wallet
|
|
|
|
self.db = wallet.db
|
|
|
|
|
|
|
|
async def create(self, **kwargs) -> Transaction:
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
async def delete(self, **constraints):
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
async def list(self, **constraints) -> Tuple[List[Output], Optional[int]]:
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
async def get(self, **constraints) -> Output:
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
async def get_or_none(self, **constraints) -> Optional[Output]:
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
|
|
|
|
class ClaimListManager(BaseListManager):
|
|
|
|
name = 'claim'
|
|
|
|
__slots__ = ()
|
|
|
|
|
|
|
|
async def create(
|
2020-05-01 09:33:10 -04:00
|
|
|
self, name: str, claim: Claim, amount: int, holding_address: str,
|
2020-05-06 10:53:31 -04:00
|
|
|
funding_accounts: List[Account], change_account: Account, signing_channel: Output = None):
|
2020-05-01 09:33:10 -04:00
|
|
|
claim_output = Output.pay_claim_name_pubkey_hash(
|
2020-05-06 10:53:31 -04:00
|
|
|
amount, name, claim, self.wallet.ledger.address_to_hash160(holding_address)
|
2020-05-01 09:33:10 -04:00
|
|
|
)
|
|
|
|
if signing_channel is not None:
|
|
|
|
claim_output.sign(signing_channel, b'placeholder txid:nout')
|
2020-05-06 10:53:31 -04:00
|
|
|
return await self.wallet.create_transaction(
|
2020-05-01 09:33:10 -04:00
|
|
|
[], [claim_output], funding_accounts, change_account, sign=False
|
|
|
|
)
|
|
|
|
|
2020-05-06 10:53:31 -04:00
|
|
|
async def update(
|
|
|
|
self, previous_claim: Output, claim: Claim, amount: int, holding_address: str,
|
|
|
|
funding_accounts: List[Account], change_account: Account, signing_channel: Output = None):
|
2020-05-01 09:33:10 -04:00
|
|
|
updated_claim = Output.pay_update_claim_pubkey_hash(
|
|
|
|
amount, previous_claim.claim_name, previous_claim.claim_id,
|
2020-05-06 10:53:31 -04:00
|
|
|
claim, self.wallet.ledger.address_to_hash160(holding_address)
|
2020-05-01 09:33:10 -04:00
|
|
|
)
|
|
|
|
if signing_channel is not None:
|
|
|
|
updated_claim.sign(signing_channel, b'placeholder txid:nout')
|
|
|
|
else:
|
|
|
|
updated_claim.clear_signature()
|
2020-05-06 10:53:31 -04:00
|
|
|
return await self.wallet.create_transaction(
|
2020-05-01 09:33:10 -04:00
|
|
|
[Input.spend(previous_claim)], [updated_claim], funding_accounts, change_account, sign=False
|
|
|
|
)
|
|
|
|
|
2020-05-06 10:53:31 -04:00
|
|
|
async def delete(self, claim_id=None, txid=None, nout=None):
|
|
|
|
claim = await self.get(claim_id=claim_id, txid=txid, nout=nout)
|
|
|
|
return await self.wallet.create_transaction(
|
|
|
|
[Input.spend(claim)], [], self.wallet._accounts, self.wallet._accounts[0]
|
|
|
|
)
|
|
|
|
|
|
|
|
async def list(self, **constraints) -> Tuple[List[Output], Optional[int]]:
|
|
|
|
return await self.db.get_claims(wallet=self.wallet, **constraints)
|
|
|
|
|
|
|
|
async def get(self, claim_id=None, claim_name=None, txid=None, nout=None) -> Output:
|
|
|
|
if txid is not None and nout is not None:
|
|
|
|
key, value, constraints = 'txid:nout', f'{txid}:{nout}', {'tx_hash': '', 'position': nout}
|
|
|
|
elif claim_id is not None:
|
|
|
|
key, value, constraints = 'id', claim_id, {'claim_id': claim_id}
|
|
|
|
elif claim_name is not None:
|
|
|
|
key, value, constraints = 'name', claim_name, {'claim_name': claim_name}
|
|
|
|
else:
|
|
|
|
raise ValueError(f"Couldn't find {self.name} because an {self.name}_id or name was not provided.")
|
|
|
|
claims, _ = await self.list(**constraints)
|
|
|
|
if len(claims) == 1:
|
|
|
|
return claims[0]
|
|
|
|
elif len(claims) > 1:
|
|
|
|
raise ValueError(
|
|
|
|
f"Multiple {self.name}s found with {key} '{value}', "
|
|
|
|
f"pass a {self.name}_id to narrow it down."
|
|
|
|
)
|
|
|
|
raise ValueError(f"Couldn't find {self.name} with {key} '{value}'.")
|
|
|
|
|
|
|
|
async def get_or_none(self, claim_id=None, claim_name=None, txid=None, nout=None) -> Optional[Output]:
|
|
|
|
if any((claim_id, claim_name, all((txid, nout)))):
|
|
|
|
return await self.get(claim_id, claim_name, txid, nout)
|
|
|
|
|
|
|
|
|
|
|
|
class StreamListManager(ClaimListManager):
|
|
|
|
__slots__ = ()
|
|
|
|
|
|
|
|
async def create(self, *args, **kwargs):
|
|
|
|
return await super().create(*args, **kwargs)
|
|
|
|
|
|
|
|
async def list(self, **constraints) -> Tuple[List[Output], Optional[int]]:
|
|
|
|
return await self.db.get_streams(wallet=self.wallet, **constraints)
|
|
|
|
|
|
|
|
|
|
|
|
class CollectionListManager(ClaimListManager):
|
|
|
|
__slots__ = ()
|
|
|
|
|
|
|
|
async def create(self, *args, **kwargs):
|
|
|
|
return await super().create(*args, **kwargs)
|
|
|
|
|
|
|
|
async def list(self, **constraints) -> Tuple[List[Output], Optional[int]]:
|
|
|
|
return await self.db.get_collections(wallet=self.wallet, **constraints)
|
|
|
|
|
|
|
|
|
|
|
|
class ChannelListManager(ClaimListManager):
|
|
|
|
name = 'channel'
|
|
|
|
__slots__ = ()
|
|
|
|
|
|
|
|
async def create(self, name: str, amount: int, account: Account, funding_accounts: List[Account],
|
|
|
|
claim_address: str, preview=False, **kwargs):
|
|
|
|
claim = Claim()
|
|
|
|
claim.channel.update(**kwargs)
|
|
|
|
tx = await super().create(
|
|
|
|
name, claim, amount, claim_address, funding_accounts, funding_accounts[0]
|
|
|
|
)
|
|
|
|
txo = tx.outputs[0]
|
|
|
|
txo.generate_channel_private_key()
|
|
|
|
await self.wallet.sign(tx)
|
|
|
|
if not preview:
|
|
|
|
account.add_channel_private_key(txo.private_key)
|
|
|
|
await self.wallet.save()
|
|
|
|
return tx
|
|
|
|
|
|
|
|
async def list(self, **constraints) -> Tuple[List[Output], Optional[int]]:
|
|
|
|
return await self.db.get_channels(wallet=self.wallet, **constraints)
|
|
|
|
|
|
|
|
async def get_for_signing(self, **kwargs) -> Output:
|
|
|
|
channel = await self.get(**kwargs)
|
|
|
|
if not channel.has_private_key:
|
|
|
|
raise Exception(
|
|
|
|
f"Couldn't find private key for channel '{channel.claim_name}', can't use channel for signing. "
|
|
|
|
)
|
|
|
|
return channel
|
|
|
|
|
|
|
|
async def get_for_signing_or_none(self, **kwargs) -> Optional[Output]:
|
|
|
|
if any(kwargs.values()):
|
|
|
|
return await self.get_for_signing(**kwargs)
|
|
|
|
|
|
|
|
|
|
|
|
class SupportListManager(BaseListManager):
|
|
|
|
__slots__ = ()
|
|
|
|
|
|
|
|
async def create(self, name: str, claim_id: str, amount: int, holding_address: str,
|
|
|
|
funding_accounts: List[Account], change_account: Account) -> Transaction:
|
2020-05-01 09:33:10 -04:00
|
|
|
support_output = Output.pay_support_pubkey_hash(
|
2020-05-06 10:53:31 -04:00
|
|
|
amount, name, claim_id, self.wallet.ledger.address_to_hash160(holding_address)
|
|
|
|
)
|
|
|
|
return await self.wallet.create_transaction(
|
|
|
|
[], [support_output], funding_accounts, change_account
|
|
|
|
)
|
|
|
|
|
|
|
|
async def list(self, **constraints) -> Tuple[List[Output], Optional[int]]:
|
|
|
|
return await self.db.get_supports(**constraints)
|
|
|
|
|
|
|
|
async def get(self, **constraints) -> Output:
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
async def get_or_none(self, **constraints) -> Optional[Output]:
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
|
|
|
|
class PurchaseListManager(BaseListManager):
|
|
|
|
__slots__ = ()
|
|
|
|
|
|
|
|
async def create(self, name: str, claim_id: str, amount: int, holding_address: str,
|
|
|
|
funding_accounts: List[Account], change_account: Account) -> Transaction:
|
|
|
|
support_output = Output.pay_support_pubkey_hash(
|
|
|
|
amount, name, claim_id, self.wallet.ledger.address_to_hash160(holding_address)
|
|
|
|
)
|
|
|
|
return await self.wallet.create_transaction(
|
|
|
|
[], [support_output], funding_accounts, change_account
|
2020-05-01 09:33:10 -04:00
|
|
|
)
|
|
|
|
|
|
|
|
def purchase(self, claim_id: str, amount: int, merchant_address: bytes,
|
|
|
|
funding_accounts: List['Account'], change_account: 'Account'):
|
|
|
|
payment = Output.pay_pubkey_hash(amount, self.ledger.address_to_hash160(merchant_address))
|
|
|
|
data = Output.add_purchase_data(Purchase(claim_id))
|
|
|
|
return self.create_transaction(
|
|
|
|
[], [payment, data], funding_accounts, change_account
|
|
|
|
)
|
|
|
|
|
|
|
|
async def create_purchase_transaction(
|
|
|
|
self, accounts: List[Account], txo: Output, exchange: 'ExchangeRateManager',
|
|
|
|
override_max_key_fee=False):
|
|
|
|
fee = txo.claim.stream.fee
|
|
|
|
fee_amount = exchange.to_dewies(fee.currency, fee.amount)
|
|
|
|
if not override_max_key_fee and self.ledger.conf.max_key_fee:
|
|
|
|
max_fee = self.ledger.conf.max_key_fee
|
|
|
|
max_fee_amount = exchange.to_dewies(max_fee['currency'], Decimal(max_fee['amount']))
|
|
|
|
if max_fee_amount and fee_amount > max_fee_amount:
|
|
|
|
error_fee = f"{dewies_to_lbc(fee_amount)} LBC"
|
|
|
|
if fee.currency != 'LBC':
|
|
|
|
error_fee += f" ({fee.amount} {fee.currency})"
|
|
|
|
error_max_fee = f"{dewies_to_lbc(max_fee_amount)} LBC"
|
|
|
|
if max_fee['currency'] != 'LBC':
|
|
|
|
error_max_fee += f" ({max_fee['amount']} {max_fee['currency']})"
|
|
|
|
raise KeyFeeAboveMaxAllowedError(
|
|
|
|
f"Purchase price of {error_fee} exceeds maximum "
|
|
|
|
f"configured price of {error_max_fee}."
|
|
|
|
)
|
|
|
|
fee_address = fee.address or txo.get_address(self.ledger)
|
|
|
|
return await self.purchase(
|
|
|
|
txo.claim_id, fee_amount, fee_address, accounts, accounts[0]
|
|
|
|
)
|
|
|
|
|
2020-05-06 10:53:31 -04:00
|
|
|
async def list(self, **constraints) -> Tuple[List[Output], Optional[int]]:
|
|
|
|
return await self.db.get_purchases(**constraints)
|
|
|
|
|
|
|
|
async def get(self, **constraints) -> Output:
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
async def get_or_none(self, **constraints) -> Optional[Output]:
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
|
|
|
|
def txs_to_dict(txs, ledger):
|
|
|
|
history = []
|
|
|
|
for tx in txs: # pylint: disable=too-many-nested-blocks
|
|
|
|
ts = headers.estimated_timestamp(tx.height)
|
|
|
|
item = {
|
|
|
|
'txid': tx.id,
|
|
|
|
'timestamp': ts,
|
|
|
|
'date': datetime.fromtimestamp(ts).isoformat(' ')[:-3] if tx.height > 0 else None,
|
|
|
|
'confirmations': (headers.height + 1) - tx.height if tx.height > 0 else 0,
|
|
|
|
'claim_info': [],
|
|
|
|
'update_info': [],
|
|
|
|
'support_info': [],
|
|
|
|
'abandon_info': [],
|
|
|
|
'purchase_info': []
|
2018-07-11 22:37:15 -04:00
|
|
|
}
|
2020-05-06 10:53:31 -04:00
|
|
|
is_my_inputs = all([txi.is_my_input for txi in tx.inputs])
|
|
|
|
if is_my_inputs:
|
|
|
|
# fees only matter if we are the ones paying them
|
|
|
|
item['value'] = dewies_to_lbc(tx.net_account_balance + tx.fee)
|
|
|
|
item['fee'] = dewies_to_lbc(-tx.fee)
|
2018-05-25 02:03:25 -04:00
|
|
|
else:
|
2020-05-06 10:53:31 -04:00
|
|
|
# someone else paid the fees
|
|
|
|
item['value'] = dewies_to_lbc(tx.net_account_balance)
|
|
|
|
item['fee'] = '0.0'
|
|
|
|
for txo in tx.my_claim_outputs:
|
|
|
|
item['claim_info'].append({
|
|
|
|
'address': txo.get_address(self.ledger),
|
|
|
|
'balance_delta': dewies_to_lbc(-txo.amount),
|
|
|
|
'amount': dewies_to_lbc(txo.amount),
|
|
|
|
'claim_id': txo.claim_id,
|
|
|
|
'claim_name': txo.claim_name,
|
|
|
|
'nout': txo.position,
|
|
|
|
'is_spent': txo.is_spent,
|
|
|
|
})
|
|
|
|
for txo in tx.my_update_outputs:
|
|
|
|
if is_my_inputs: # updating my own claim
|
|
|
|
previous = None
|
|
|
|
for txi in tx.inputs:
|
|
|
|
if txi.txo_ref.txo is not None:
|
|
|
|
other_txo = txi.txo_ref.txo
|
|
|
|
if (other_txo.is_claim or other_txo.script.is_support_claim) \
|
|
|
|
and other_txo.claim_id == txo.claim_id:
|
|
|
|
previous = other_txo
|
|
|
|
break
|
|
|
|
if previous is not None:
|
|
|
|
item['update_info'].append({
|
|
|
|
'address': txo.get_address(self),
|
|
|
|
'balance_delta': dewies_to_lbc(previous.amount - txo.amount),
|
|
|
|
'amount': dewies_to_lbc(txo.amount),
|
|
|
|
'claim_id': txo.claim_id,
|
|
|
|
'claim_name': txo.claim_name,
|
|
|
|
'nout': txo.position,
|
|
|
|
'is_spent': txo.is_spent,
|
|
|
|
})
|
|
|
|
else: # someone sent us their claim
|
|
|
|
item['update_info'].append({
|
|
|
|
'address': txo.get_address(self),
|
|
|
|
'balance_delta': dewies_to_lbc(0),
|
|
|
|
'amount': dewies_to_lbc(txo.amount),
|
|
|
|
'claim_id': txo.claim_id,
|
|
|
|
'claim_name': txo.claim_name,
|
|
|
|
'nout': txo.position,
|
|
|
|
'is_spent': txo.is_spent,
|
|
|
|
})
|
|
|
|
for txo in tx.my_support_outputs:
|
|
|
|
item['support_info'].append({
|
|
|
|
'address': txo.get_address(self.ledger),
|
|
|
|
'balance_delta': dewies_to_lbc(txo.amount if not is_my_inputs else -txo.amount),
|
|
|
|
'amount': dewies_to_lbc(txo.amount),
|
|
|
|
'claim_id': txo.claim_id,
|
|
|
|
'claim_name': txo.claim_name,
|
|
|
|
'is_tip': not is_my_inputs,
|
|
|
|
'nout': txo.position,
|
|
|
|
'is_spent': txo.is_spent,
|
|
|
|
})
|
|
|
|
if is_my_inputs:
|
|
|
|
for txo in tx.other_support_outputs:
|
|
|
|
item['support_info'].append({
|
|
|
|
'address': txo.get_address(self.ledger),
|
|
|
|
'balance_delta': dewies_to_lbc(-txo.amount),
|
|
|
|
'amount': dewies_to_lbc(txo.amount),
|
|
|
|
'claim_id': txo.claim_id,
|
|
|
|
'claim_name': txo.claim_name,
|
|
|
|
'is_tip': is_my_inputs,
|
|
|
|
'nout': txo.position,
|
|
|
|
'is_spent': txo.is_spent,
|
|
|
|
})
|
|
|
|
for txo in tx.my_abandon_outputs:
|
|
|
|
item['abandon_info'].append({
|
|
|
|
'address': txo.get_address(self.ledger),
|
|
|
|
'balance_delta': dewies_to_lbc(txo.amount),
|
|
|
|
'amount': dewies_to_lbc(txo.amount),
|
|
|
|
'claim_id': txo.claim_id,
|
|
|
|
'claim_name': txo.claim_name,
|
|
|
|
'nout': txo.position
|
|
|
|
})
|
|
|
|
for txo in tx.any_purchase_outputs:
|
|
|
|
item['purchase_info'].append({
|
|
|
|
'address': txo.get_address(self.ledger),
|
|
|
|
'balance_delta': dewies_to_lbc(txo.amount if not is_my_inputs else -txo.amount),
|
|
|
|
'amount': dewies_to_lbc(txo.amount),
|
|
|
|
'claim_id': txo.purchased_claim_id,
|
|
|
|
'nout': txo.position,
|
|
|
|
'is_spent': txo.is_spent,
|
|
|
|
})
|
|
|
|
history.append(item)
|
|
|
|
return history
|