2018-05-25 23:26:07 -04:00
|
|
|
import os
|
2020-01-02 22:18:49 -05:00
|
|
|
import asyncio
|
2020-05-06 10:53:31 -04:00
|
|
|
import logging
|
|
|
|
from typing import Optional, Dict
|
2018-06-14 15:18:36 -04:00
|
|
|
|
2020-05-01 09:33:10 -04:00
|
|
|
from lbry.db import Database
|
|
|
|
from lbry.blockchain.ledger import Ledger
|
2020-01-02 22:50:27 -05:00
|
|
|
|
2020-05-06 10:53:31 -04:00
|
|
|
from .wallet import Wallet
|
2018-09-25 09:41:31 -04:00
|
|
|
|
2018-08-06 00:28:11 -04:00
|
|
|
log = logging.getLogger(__name__)
|
|
|
|
|
2018-03-25 22:59:57 -04:00
|
|
|
|
2020-01-02 22:18:49 -05:00
|
|
|
class WalletManager:
|
|
|
|
|
2020-05-06 10:53:31 -04:00
|
|
|
def __init__(self, ledger: Ledger, db: Database):
|
2020-05-01 09:33:10 -04:00
|
|
|
self.ledger = ledger
|
|
|
|
self.db = db
|
2020-05-06 10:53:31 -04:00
|
|
|
self.wallets: Dict[str, Wallet] = {}
|
|
|
|
|
|
|
|
def __getitem__(self, wallet_id: str) -> Wallet:
|
|
|
|
try:
|
|
|
|
return self.wallets[wallet_id]
|
|
|
|
except KeyError:
|
|
|
|
raise ValueError(f"Couldn't find wallet: {wallet_id}.")
|
2019-12-14 20:33:25 -05:00
|
|
|
|
2020-01-02 22:18:49 -05:00
|
|
|
@property
|
2020-05-06 10:53:31 -04:00
|
|
|
def default(self) -> Optional[Wallet]:
|
|
|
|
for wallet in self.wallets.values():
|
2020-01-02 22:18:49 -05:00
|
|
|
return wallet
|
2018-05-25 23:26:07 -04:00
|
|
|
|
2020-05-18 08:26:36 -04:00
|
|
|
def get_or_default(self, wallet_id: Optional[str]) -> Wallet:
|
2020-05-06 10:53:31 -04:00
|
|
|
if wallet_id:
|
|
|
|
return self[wallet_id]
|
2020-05-18 08:26:36 -04:00
|
|
|
wallet = self.default
|
|
|
|
if not wallet:
|
|
|
|
raise ValueError("No wallets available.")
|
|
|
|
return wallet
|
|
|
|
|
|
|
|
def get_or_default_for_spending(self, wallet_id: Optional[str]) -> Wallet:
|
|
|
|
wallet = self.get_or_default(wallet_id)
|
|
|
|
if wallet.is_locked:
|
|
|
|
raise ValueError("Cannot spend funds with locked wallet, unlock first.")
|
|
|
|
return wallet
|
2020-01-02 22:18:49 -05:00
|
|
|
|
|
|
|
@property
|
2020-05-06 10:53:31 -04:00
|
|
|
def path(self):
|
|
|
|
return os.path.join(self.ledger.conf.wallet_dir, 'wallets')
|
|
|
|
|
|
|
|
def sync_ensure_path_exists(self):
|
|
|
|
if not os.path.exists(self.path):
|
|
|
|
os.mkdir(self.path)
|
|
|
|
|
|
|
|
async def ensure_path_exists(self):
|
|
|
|
await asyncio.get_running_loop().run_in_executor(
|
|
|
|
None, self.sync_ensure_path_exists
|
|
|
|
)
|
|
|
|
|
|
|
|
async def load(self):
|
|
|
|
wallets_directory = self.path
|
|
|
|
for wallet_id in self.ledger.conf.wallets:
|
|
|
|
if wallet_id in self.wallets:
|
2020-06-05 00:35:22 -04:00
|
|
|
log.warning("Ignoring duplicate wallet_id in config: %s", wallet_id)
|
2020-05-06 10:53:31 -04:00
|
|
|
continue
|
|
|
|
wallet_path = os.path.join(wallets_directory, wallet_id)
|
|
|
|
if not os.path.exists(wallet_path):
|
|
|
|
if not wallet_id == "default_wallet": # we'll probably generate this wallet, don't show error
|
2020-06-05 00:35:22 -04:00
|
|
|
log.error("Could not load wallet, file does not exist: %s", wallet_path)
|
2020-05-06 10:53:31 -04:00
|
|
|
continue
|
|
|
|
wallet = await Wallet.from_path(self.ledger, self.db, wallet_path)
|
|
|
|
self.add(wallet)
|
|
|
|
default_wallet = self.default
|
|
|
|
if default_wallet is None:
|
|
|
|
if self.ledger.conf.create_default_wallet:
|
|
|
|
assert self.ledger.conf.wallets[0] == "default_wallet", (
|
|
|
|
"Requesting to generate the default wallet but the 'wallets' "
|
|
|
|
"config setting does not include 'default_wallet' as the first wallet."
|
|
|
|
)
|
|
|
|
await self.create(
|
|
|
|
self.ledger.conf.wallets[0], 'Wallet',
|
|
|
|
create_account=self.ledger.conf.create_default_account
|
|
|
|
)
|
|
|
|
elif not default_wallet.has_accounts and self.ledger.conf.create_default_account:
|
2020-05-18 08:26:36 -04:00
|
|
|
await default_wallet.accounts.generate()
|
2020-05-06 10:53:31 -04:00
|
|
|
|
|
|
|
def add(self, wallet: Wallet) -> Wallet:
|
|
|
|
self.wallets[wallet.id] = wallet
|
|
|
|
return wallet
|
2019-10-29 01:26:25 -04:00
|
|
|
|
2020-05-06 10:53:31 -04:00
|
|
|
async def add_from_path(self, wallet_path) -> Wallet:
|
|
|
|
wallet_id = os.path.basename(wallet_path)
|
|
|
|
if wallet_id in self.wallets:
|
|
|
|
existing = self.wallets.get(wallet_id)
|
|
|
|
if existing.storage.path == wallet_path:
|
|
|
|
raise Exception(f"Wallet '{wallet_id}' is already loaded.")
|
|
|
|
raise Exception(
|
|
|
|
f"Wallet '{wallet_id}' is already loaded from '{existing.storage.path}'"
|
|
|
|
f" and cannot be loaded from '{wallet_path}'. Consider changing the wallet"
|
|
|
|
f" filename to be unique in order to avoid conflicts."
|
|
|
|
)
|
|
|
|
wallet = await Wallet.from_path(self.ledger, self.db, wallet_path)
|
|
|
|
return self.add(wallet)
|
|
|
|
|
2020-05-18 08:26:36 -04:00
|
|
|
async def create(
|
|
|
|
self, wallet_id: str, name: str,
|
|
|
|
create_account=False, language='en', single_key=False) -> Wallet:
|
2020-05-06 10:53:31 -04:00
|
|
|
if wallet_id in self.wallets:
|
|
|
|
raise Exception(f"Wallet with id '{wallet_id}' is already loaded and cannot be created.")
|
|
|
|
wallet_path = os.path.join(self.path, wallet_id)
|
|
|
|
if os.path.exists(wallet_path):
|
|
|
|
raise Exception(f"Wallet at path '{wallet_path}' already exists, use 'wallet_add' to load wallet.")
|
2020-05-18 08:26:36 -04:00
|
|
|
wallet = await Wallet.create(
|
|
|
|
self.ledger, self.db, wallet_path, name,
|
|
|
|
create_account, language, single_key
|
|
|
|
)
|
2020-05-06 10:53:31 -04:00
|
|
|
return self.add(wallet)
|