basicswap_miserver/basicswap/interface/xmr.py

573 lines
23 KiB
Python
Raw Normal View History

2020-10-31 20:08:30 +00:00
#!/usr/bin/env python
# -*- coding: utf-8 -*-
# Copyright (c) 2020-2024 tecnovert
2020-10-31 20:08:30 +00:00
# Distributed under the MIT software license, see the accompanying
# file LICENSE or http://www.opensource.org/licenses/mit-license.php.
2022-06-11 21:13:12 +00:00
import json
2020-10-31 20:08:30 +00:00
import logging
2020-11-14 22:13:11 +00:00
import basicswap.contrib.ed25519_fast as edf
import basicswap.ed25519_fast_util as edu
2020-11-21 13:16:27 +00:00
import basicswap.util_xmr as xmr_util
2021-01-24 15:53:41 +00:00
from coincurve.ed25519 import (
2022-06-14 22:25:17 +00:00
ed25519_add,
2021-01-24 15:53:41 +00:00
ed25519_get_pubkey,
ed25519_scalar_add,
2022-06-14 22:25:17 +00:00
)
2020-11-14 22:13:11 +00:00
from coincurve.keys import PrivateKey
from coincurve.dleag import (
2022-06-14 22:25:17 +00:00
dleag_prove,
2020-11-14 22:13:11 +00:00
dleag_verify,
2022-06-14 22:25:17 +00:00
dleag_proof_len,
verify_ed25519_point,
)
2020-11-14 22:13:11 +00:00
from basicswap.interface import (
Curves)
from basicswap.util import (
i2b, b2i, b2h,
2020-11-21 13:16:27 +00:00
dumpj,
2022-06-14 22:25:17 +00:00
ensure,
make_int,
2021-10-23 14:00:32 +00:00
TemporaryError)
from basicswap.util.network import (
is_private_ip_address)
from basicswap.rpc_xmr import (
2020-11-14 22:13:11 +00:00
make_xmr_rpc_func,
2022-11-28 17:54:41 +00:00
make_xmr_rpc2_func)
from basicswap.chainparams import XMR_COIN, CoinInterface, Coins
2020-10-31 20:08:30 +00:00
class XMRInterface(CoinInterface):
@staticmethod
def curve_type():
return Curves.ed25519
@staticmethod
def coin_type():
return Coins.XMR
2020-11-28 23:04:26 +00:00
@staticmethod
def COIN():
return XMR_COIN
2020-10-31 20:08:30 +00:00
@staticmethod
2021-02-13 22:54:01 +00:00
def exp() -> int:
2020-10-31 20:08:30 +00:00
return 12
@staticmethod
2021-02-13 22:54:01 +00:00
def nbk() -> int:
2020-10-31 20:08:30 +00:00
return 32
@staticmethod
2021-02-13 22:54:01 +00:00
def nbK() -> int: # No. of bytes requires to encode a public key
2020-10-31 20:08:30 +00:00
return 32
@staticmethod
def depth_spendable() -> int:
return 10
2023-07-18 23:19:04 +00:00
@staticmethod
def xmr_swap_a_lock_spend_tx_vsize() -> int:
raise ValueError('Not possible')
@staticmethod
def xmr_swap_b_lock_spend_tx_vsize() -> int:
# TODO: Estimate with ringsize
return 1507
def __init__(self, coin_settings, network, swap_client=None):
super().__init__(network)
2020-10-31 20:08:30 +00:00
2020-11-21 13:16:27 +00:00
self.blocks_confirmed = coin_settings['blocks_confirmed']
self._restore_height = coin_settings.get('restore_height', 0)
2020-12-22 20:48:07 +00:00
self.setFeePriority(coin_settings.get('fee_priority', 0))
self._sc = swap_client
self._log = self._sc.log if self._sc and self._sc.log else logging
self._wallet_password = None
self._have_checked_seed = False
2020-12-22 11:21:25 +00:00
daemon_login = None
if coin_settings.get('rpcuser', '') != '':
daemon_login = (coin_settings.get('rpcuser', ''), coin_settings.get('rpcpassword', ''))
rpchost = coin_settings.get('rpchost', '127.0.0.1')
proxy_host = None
proxy_port = None
# Connect to the daemon over a proxy if not running locally
if swap_client:
chain_client_settings = swap_client.getChainClientSettings(self.coin_type())
manage_daemon: bool = chain_client_settings['manage_daemon']
if swap_client.use_tor_proxy:
if manage_daemon is False:
log_str: str = ''
have_cc_tor_opt = 'use_tor' in chain_client_settings
if have_cc_tor_opt and chain_client_settings['use_tor'] is False:
log_str = ' bypassing proxy (use_tor false for XMR)'
elif have_cc_tor_opt is False and is_private_ip_address(rpchost):
log_str = ' bypassing proxy (private ip address)'
else:
proxy_host = swap_client.tor_proxy_host
proxy_port = swap_client.tor_proxy_port
log_str = f' through proxy at {proxy_host}'
self._log.info(f'Connecting to remote {self.coin_name()} daemon at {rpchost}{log_str}.')
else:
self._log.info(f'Not connecting to local {self.coin_name()} daemon through proxy.')
elif manage_daemon is False:
self._log.info(f'Connecting to remote {self.coin_name()} daemon at {rpchost}.')
2024-02-05 15:17:44 +00:00
self._rpctimeout = coin_settings.get('rpctimeout', 60)
self._walletrpctimeout = coin_settings.get('walletrpctimeout', 120)
self._walletrpctimeoutlong = coin_settings.get('walletrpctimeoutlong', 600)
2024-02-06 21:36:07 +00:00
self.rpc = make_xmr_rpc_func(coin_settings['rpcport'], daemon_login, host=rpchost, proxy_host=proxy_host, proxy_port=proxy_port, default_timeout=self._rpctimeout, tag='Node(j) ')
self.rpc2 = make_xmr_rpc2_func(coin_settings['rpcport'], daemon_login, host=rpchost, proxy_host=proxy_host, proxy_port=proxy_port, default_timeout=self._rpctimeout, tag='Node ') # non-json endpoint
self.rpc_wallet = make_xmr_rpc_func(coin_settings['walletrpcport'], coin_settings['walletrpcauth'], host=coin_settings.get('walletrpchost', '127.0.0.1'), default_timeout=self._walletrpctimeout, tag='Wallet ')
2023-12-29 13:36:00 +00:00
def checkWallets(self) -> int:
return 1
2020-12-22 11:21:25 +00:00
def setFeePriority(self, new_priority):
2021-10-21 22:47:04 +00:00
ensure(new_priority >= 0 and new_priority < 4, 'Invalid fee_priority value')
2020-12-22 11:21:25 +00:00
self._fee_priority = new_priority
2020-11-21 13:16:27 +00:00
def setWalletFilename(self, wallet_filename):
self._wallet_filename = wallet_filename
2020-10-31 20:08:30 +00:00
def createWallet(self, params):
if self._wallet_password is not None:
params['password'] = self._wallet_password
2023-12-29 13:36:00 +00:00
rv = self.rpc_wallet('generate_from_keys', params)
self._log.info('generate_from_keys %s', dumpj(rv))
def openWallet(self, filename):
params = {'filename': filename}
if self._wallet_password is not None:
params['password'] = self._wallet_password
2022-12-02 23:07:41 +00:00
try:
# Can't reopen the same wallet in windows, !is_keys_file_locked()
2023-12-29 13:36:00 +00:00
self.rpc_wallet('close_wallet')
2022-12-02 23:07:41 +00:00
except Exception:
pass
2023-12-29 13:36:00 +00:00
self.rpc_wallet('open_wallet', params)
def initialiseWallet(self, key_view, key_spend, restore_height=None):
2021-09-04 23:18:34 +00:00
with self._mx_wallet:
try:
self.openWallet(self._wallet_filename)
2021-09-04 23:18:34 +00:00
# TODO: Check address
return # Wallet exists
except Exception as e:
pass
Kbv = self.getPubkey(key_view)
Kbs = self.getPubkey(key_spend)
address_b58 = xmr_util.encode_address(Kbv, Kbs)
params = {
'filename': self._wallet_filename,
'address': address_b58,
'viewkey': b2h(key_view[::-1]),
'spendkey': b2h(key_spend[::-1]),
'restore_height': self._restore_height,
}
self.createWallet(params)
self.openWallet(self._wallet_filename)
2023-02-17 23:47:44 +00:00
def ensureWalletExists(self) -> None:
2021-09-04 23:18:34 +00:00
with self._mx_wallet:
self.openWallet(self._wallet_filename)
2023-12-29 13:36:00 +00:00
def testDaemonRPC(self, with_wallet=True) -> None:
self.rpc_wallet('get_languages')
2020-11-07 11:08:07 +00:00
def getDaemonVersion(self):
2023-12-29 13:36:00 +00:00
return self.rpc_wallet('get_version')['version']
2020-11-07 11:08:07 +00:00
def getBlockchainInfo(self):
2024-02-05 15:17:44 +00:00
get_height = self.rpc2('get_height', timeout=self._rpctimeout)
rv = {
'blocks': get_height['height'],
'verificationprogress': 0.0,
}
try:
# get_block_count.block_count is how many blocks are in the longest chain known to the node.
# get_block_count returns "Internal error" if bootstrap-daemon is active
if get_height['untrusted'] is True:
rv['bootstrapping'] = True
2024-02-05 15:17:44 +00:00
get_info = self.rpc2('get_info', timeout=self._rpctimeout)
if 'height_without_bootstrap' in get_info:
rv['blocks'] = get_info['height_without_bootstrap']
rv['known_block_count'] = get_info['height']
if rv['known_block_count'] > rv['blocks']:
rv['verificationprogress'] = rv['blocks'] / rv['known_block_count']
else:
2024-02-05 15:17:44 +00:00
rv['known_block_count'] = self.rpc('get_block_count', timeout=self._rpctimeout)['count']
rv['verificationprogress'] = rv['blocks'] / rv['known_block_count']
except Exception as e:
self._log.warning('XMR get_block_count failed with: %s', str(e))
rv['verificationprogress'] = 0.0
2021-11-01 14:47:32 +00:00
2020-11-07 11:08:07 +00:00
return rv
2020-11-14 22:13:11 +00:00
def getChainHeight(self):
2024-02-05 15:17:44 +00:00
return self.rpc2('get_height', timeout=self._rpctimeout)['height']
2020-11-14 22:13:11 +00:00
2020-11-07 11:08:07 +00:00
def getWalletInfo(self):
2021-09-04 23:18:34 +00:00
with self._mx_wallet:
try:
self.openWallet(self._wallet_filename)
except Exception as e:
if 'Failed to open wallet' in str(e):
rv = {'encrypted': True, 'locked': True, 'balance': 0, 'unconfirmed_balance': 0}
return rv
raise e
2021-09-04 23:18:34 +00:00
rv = {}
2023-12-29 13:36:00 +00:00
self.rpc_wallet('refresh')
balance_info = self.rpc_wallet('get_balance')
rv['balance'] = self.format_amount(balance_info['unlocked_balance'])
rv['unconfirmed_balance'] = self.format_amount(balance_info['balance'] - balance_info['unlocked_balance'])
rv['encrypted'] = False if self._wallet_password is None else True
rv['locked'] = False
2021-09-04 23:18:34 +00:00
return rv
2020-11-07 11:08:07 +00:00
def walletRestoreHeight(self):
return self._restore_height
2023-02-17 23:47:44 +00:00
def getMainWalletAddress(self) -> str:
2021-09-04 23:18:34 +00:00
with self._mx_wallet:
self.openWallet(self._wallet_filename)
2023-12-29 13:36:00 +00:00
return self.rpc_wallet('get_address')['address']
2020-11-21 13:16:27 +00:00
2023-02-17 23:47:44 +00:00
def getNewAddress(self, placeholder) -> str:
2021-09-04 23:18:34 +00:00
with self._mx_wallet:
self.openWallet(self._wallet_filename)
2023-12-29 13:36:00 +00:00
new_address = self.rpc_wallet('create_address', {'account_index': 0})['address']
self.rpc_wallet('store')
2023-10-11 15:46:52 +00:00
return new_address
2020-11-07 11:08:07 +00:00
2023-07-14 07:31:05 +00:00
def get_fee_rate(self, conf_target: int = 2):
2024-02-01 09:28:21 +00:00
self._log.warning('TODO - estimate XMR fee rate?')
return 0.0, 'unused'
2020-11-27 22:20:35 +00:00
2023-02-17 23:47:44 +00:00
def getNewSecretKey(self) -> bytes:
2023-07-14 07:31:05 +00:00
# Note: Returned bytes are in big endian order
return i2b(edu.get_secret())
2020-10-31 20:08:30 +00:00
2023-02-17 23:47:44 +00:00
def pubkey(self, key: bytes) -> bytes:
2020-10-31 20:08:30 +00:00
return edf.scalarmult_B(key)
2023-02-17 23:47:44 +00:00
def encodeKey(self, vk: bytes) -> str:
return vk[::-1].hex()
2023-02-17 23:47:44 +00:00
def decodeKey(self, k_hex: str) -> bytes:
return bytes.fromhex(k_hex)[::-1]
2023-02-17 23:47:44 +00:00
def encodePubkey(self, pk: bytes) -> str:
2020-10-31 20:08:30 +00:00
return edu.encodepoint(pk)
def decodePubkey(self, pke):
return edf.decodepoint(pke)
2020-11-14 22:13:11 +00:00
def getPubkey(self, privkey):
return ed25519_get_pubkey(privkey)
2023-02-17 23:47:44 +00:00
def getAddressFromKeys(self, key_view: bytes, key_spend: bytes) -> str:
pk_view = self.getPubkey(key_view)
pk_spend = self.getPubkey(key_spend)
return xmr_util.encode_address(pk_view, pk_spend)
2023-02-17 23:47:44 +00:00
def verifyKey(self, k: int) -> bool:
2020-11-14 22:13:11 +00:00
i = b2i(k)
return (i < edf.l and i > 8)
2020-11-14 22:13:11 +00:00
def verifyPubkey(self, pubkey_bytes):
# Calls ed25519_decode_check_point() in secp256k1
# Checks for small order
2020-11-14 22:13:11 +00:00
return verify_ed25519_point(pubkey_bytes)
2023-02-17 23:47:44 +00:00
def proveDLEAG(self, key: bytes) -> bytes:
2020-11-14 22:13:11 +00:00
privkey = PrivateKey(key)
return dleag_prove(privkey)
2023-02-17 23:47:44 +00:00
def verifyDLEAG(self, dleag_bytes: bytes) -> bool:
2020-11-14 22:13:11 +00:00
return dleag_verify(dleag_bytes)
2023-02-17 23:47:44 +00:00
def lengthDLEAG(self) -> int:
2020-11-14 22:13:11 +00:00
return dleag_proof_len()
2023-02-17 23:47:44 +00:00
def sumKeys(self, ka: bytes, kb: bytes) -> bytes:
2021-01-24 15:53:41 +00:00
return ed25519_scalar_add(ka, kb)
2020-10-31 20:08:30 +00:00
2023-02-17 23:47:44 +00:00
def sumPubkeys(self, Ka: bytes, Kb: bytes) -> bytes:
2021-01-24 15:53:41 +00:00
return ed25519_add(Ka, Kb)
2020-10-31 20:08:30 +00:00
2023-02-17 23:47:44 +00:00
def encodeSharedAddress(self, Kbv: bytes, Kbs: bytes) -> str:
2021-09-04 23:18:34 +00:00
return xmr_util.encode_address(Kbv, Kbs)
2023-07-18 23:19:04 +00:00
def publishBLockTx(self, kbv: bytes, Kbs: bytes, output_amount: int, feerate: int, delay_for: int = 10, unlock_time: int = 0) -> bytes:
2021-09-04 23:18:34 +00:00
with self._mx_wallet:
self.openWallet(self._wallet_filename)
2023-12-29 13:36:00 +00:00
self.rpc_wallet('refresh')
2020-10-31 20:08:30 +00:00
2022-12-20 20:19:01 +00:00
Kbv = self.getPubkey(kbv)
2021-09-04 23:18:34 +00:00
shared_addr = xmr_util.encode_address(Kbv, Kbs)
2020-10-31 20:08:30 +00:00
params = {'destinations': [{'amount': output_amount, 'address': shared_addr}], 'unlock_time': unlock_time}
2021-09-04 23:18:34 +00:00
if self._fee_priority > 0:
params['priority'] = self._fee_priority
2023-12-29 13:36:00 +00:00
rv = self.rpc_wallet('transfer', params)
2021-09-04 23:18:34 +00:00
self._log.info('publishBLockTx %s to address_b58 %s', rv['tx_hash'], shared_addr)
tx_hash = bytes.fromhex(rv['tx_hash'])
2020-11-30 14:29:40 +00:00
2022-06-11 21:13:12 +00:00
if self._sc.debug:
i = 0
while not self._sc.delay_event.is_set():
2022-12-11 18:31:43 +00:00
gt_params = {'out': True, 'pending': True, 'failed': True, 'pool': True, }
2023-12-29 13:36:00 +00:00
rv = self.rpc_wallet('get_transfers', gt_params)
2022-06-11 21:13:12 +00:00
self._log.debug('get_transfers {}'.format(dumpj(rv)))
if 'pending' not in rv:
break
if i >= delay_for:
break
self._sc.delay_event.wait(1.0)
2020-10-31 20:08:30 +00:00
2021-09-04 23:18:34 +00:00
return tx_hash
2020-10-31 20:08:30 +00:00
2021-10-23 14:00:32 +00:00
def findTxB(self, kbv, Kbs, cb_swap_value, cb_block_confirmed, restore_height, bid_sender):
2021-09-04 23:18:34 +00:00
with self._mx_wallet:
Kbv = self.getPubkey(kbv)
address_b58 = xmr_util.encode_address(Kbv, Kbs)
2020-10-31 20:08:30 +00:00
2021-09-04 23:18:34 +00:00
kbv_le = kbv[::-1]
params = {
'restore_height': restore_height,
'filename': address_b58,
'address': address_b58,
'viewkey': b2h(kbv_le),
}
2020-10-31 20:08:30 +00:00
2021-09-04 23:18:34 +00:00
try:
self.openWallet(address_b58)
2021-09-04 23:18:34 +00:00
except Exception as e:
self.createWallet(params)
self.openWallet(address_b58)
2020-10-31 20:08:30 +00:00
2024-02-05 15:17:44 +00:00
self.rpc_wallet('refresh', timeout=self._walletrpctimeoutlong)
2020-10-31 20:08:30 +00:00
2021-09-04 23:18:34 +00:00
'''
# Debug
2020-10-31 20:08:30 +00:00
try:
2023-12-29 13:36:00 +00:00
current_height = self.rpc_wallet('get_height')['height']
2021-09-04 23:18:34 +00:00
self._log.info('findTxB XMR current_height %d\nAddress: %s', current_height, address_b58)
2020-10-31 20:08:30 +00:00
except Exception as e:
2023-12-29 13:36:00 +00:00
self._log.info('rpc failed %s', str(e))
2020-10-31 20:08:30 +00:00
current_height = None # If the transfer is available it will be deep enough
2021-09-04 23:18:34 +00:00
# and (current_height is None or current_height - transfer['block_height'] > cb_block_confirmed):
'''
2020-10-31 20:08:30 +00:00
params = {'transfer_type': 'available'}
2023-12-29 13:36:00 +00:00
transfers = self.rpc_wallet('incoming_transfers', params)
rv = None
if 'transfers' in transfers:
for transfer in transfers['transfers']:
2022-12-11 18:31:43 +00:00
# unlocked <- wallet->is_transfer_unlocked() checks unlock_time and CRYPTONOTE_DEFAULT_TX_SPENDABLE_AGE
if not transfer['unlocked']:
2023-12-29 13:36:00 +00:00
full_tx = self.rpc_wallet('get_transfer_by_txid', {'txid': transfer['tx_hash']})
2022-12-11 18:31:43 +00:00
unlock_time = full_tx['transfer']['unlock_time']
if unlock_time != 0:
self._log.warning('Coin b lock txn is locked: {}, unlock_time {}'.format(transfer['tx_hash'], unlock_time))
rv = -1
continue
2021-09-04 23:18:34 +00:00
if transfer['amount'] == cb_swap_value:
return {'txid': transfer['tx_hash'], 'amount': transfer['amount'], 'height': 0 if 'block_height' not in transfer else transfer['block_height']}
else:
self._log.warning('Incorrect amount detected for coin b lock txn: {}'.format(transfer['tx_hash']))
rv = -1
return rv
2020-10-31 20:08:30 +00:00
2020-11-21 13:16:27 +00:00
def findTxnByHash(self, txid):
2021-09-04 23:18:34 +00:00
with self._mx_wallet:
self.openWallet(self._wallet_filename)
2024-02-05 15:17:44 +00:00
self.rpc_wallet('refresh', timeout=self._walletrpctimeoutlong)
2020-11-21 13:16:27 +00:00
2021-09-04 23:18:34 +00:00
try:
2024-02-05 15:17:44 +00:00
current_height = self.rpc2('get_height', timeout=self._rpctimeout)['height']
2021-09-04 23:18:34 +00:00
self._log.info('findTxnByHash XMR current_height %d\nhash: %s', current_height, txid)
except Exception as e:
2023-12-29 13:36:00 +00:00
self._log.info('rpc failed %s', str(e))
2021-09-04 23:18:34 +00:00
current_height = None # If the transfer is available it will be deep enough
params = {'transfer_type': 'available'}
2023-12-29 13:36:00 +00:00
rv = self.rpc_wallet('incoming_transfers', params)
2021-09-04 23:18:34 +00:00
if 'transfers' in rv:
for transfer in rv['transfers']:
if transfer['tx_hash'] == txid \
and (current_height is None or current_height - transfer['block_height'] > self.blocks_confirmed):
return {'txid': transfer['tx_hash'], 'amount': transfer['amount'], 'height': transfer['block_height']}
2020-10-31 20:08:30 +00:00
2021-09-04 23:18:34 +00:00
return None
2020-10-31 20:08:30 +00:00
2023-07-18 23:19:04 +00:00
def spendBLockTx(self, chain_b_lock_txid: bytes, address_to: str, kbv: bytes, kbs: bytes, cb_swap_value: int, b_fee_rate: int, restore_height: int, spend_actual_balance: bool = False) -> bytes:
'''
Notes:
"Error: No unlocked balance in the specified subaddress(es)" can mean not enough funds after tx fee.
'''
2021-09-04 23:18:34 +00:00
with self._mx_wallet:
Kbv = self.getPubkey(kbv)
Kbs = self.getPubkey(kbs)
address_b58 = xmr_util.encode_address(Kbv, Kbs)
2020-10-31 20:08:30 +00:00
2021-09-04 23:18:34 +00:00
wallet_filename = address_b58 + '_spend'
2020-10-31 20:08:30 +00:00
2021-09-04 23:18:34 +00:00
params = {
'filename': wallet_filename,
'address': address_b58,
'viewkey': b2h(kbv[::-1]),
'spendkey': b2h(kbs[::-1]),
'restore_height': restore_height,
}
try:
self.openWallet(wallet_filename)
2021-09-04 23:18:34 +00:00
except Exception as e:
self.createWallet(params)
self.openWallet(wallet_filename)
2021-09-04 23:18:34 +00:00
2023-12-29 13:36:00 +00:00
self.rpc_wallet('refresh')
rv = self.rpc_wallet('get_balance')
2021-09-04 23:18:34 +00:00
if rv['balance'] < cb_swap_value:
self._log.warning('Balance is too low, checking for existing spend.')
2023-12-29 13:36:00 +00:00
txns = self.rpc_wallet('get_transfers', {'out': True})
if 'out' in txns:
txns = txns['out']
if len(txns) > 0:
txid = txns[0]['txid']
self._log.warning(f'spendBLockTx detected spending tx: {txid}.')
if txns[0]['address'] == address_b58:
return bytes.fromhex(txid)
2021-09-04 23:18:34 +00:00
self._log.error('wallet {} balance {}, expected {}'.format(wallet_filename, rv['balance'], cb_swap_value))
if not spend_actual_balance:
raise TemporaryError('Invalid balance')
if spend_actual_balance and rv['balance'] != cb_swap_value:
self._log.warning('Spending actual balance {}, not swap value {}.'.format(rv['balance'], cb_swap_value))
cb_swap_value = rv['balance']
2021-09-04 23:18:34 +00:00
if rv['unlocked_balance'] < cb_swap_value:
self._log.error('wallet {} balance {}, expected {}, blocks_to_unlock {}'.format(wallet_filename, rv['unlocked_balance'], cb_swap_value, rv['blocks_to_unlock']))
2021-10-23 14:00:32 +00:00
raise TemporaryError('Invalid unlocked_balance')
2021-09-04 23:18:34 +00:00
params = {'address': address_to}
if self._fee_priority > 0:
params['priority'] = self._fee_priority
2023-12-29 13:36:00 +00:00
rv = self.rpc_wallet('sweep_all', params)
2022-06-11 21:13:12 +00:00
self._log.debug('sweep_all {}'.format(json.dumps(rv)))
2021-09-04 23:18:34 +00:00
return bytes.fromhex(rv['tx_hash_list'][0])
def withdrawCoin(self, value: int, addr_to: str, sweepall: bool) -> str:
2021-09-04 23:18:34 +00:00
with self._mx_wallet:
self.openWallet(self._wallet_filename)
2023-12-29 13:36:00 +00:00
self.rpc_wallet('refresh')
2021-09-04 23:18:34 +00:00
if sweepall:
2023-12-29 13:36:00 +00:00
balance = self.rpc_wallet('get_balance')
if balance['balance'] != balance['unlocked_balance']:
raise ValueError('Balance must be fully confirmed to use sweep all.')
self._log.info('XMR withdraw sweep_all.')
self._log.debug('XMR balance: {}'.format(balance['balance']))
params = {'address': addr_to}
if self._fee_priority > 0:
params['priority'] = self._fee_priority
rv = self.rpc_wallet('sweep_all', params)
return rv['tx_hash_list'][0]
value_sats: int = make_int(value, self.exp())
2021-09-04 23:18:34 +00:00
params = {'destinations': [{'amount': value_sats, 'address': addr_to}]}
if self._fee_priority > 0:
params['priority'] = self._fee_priority
2023-12-29 13:36:00 +00:00
rv = self.rpc_wallet('transfer', params)
2021-09-04 23:18:34 +00:00
return rv['tx_hash']
2022-12-11 18:31:43 +00:00
def showLockTransfers(self, kbv, Kbs, restore_height):
with self._mx_wallet:
try:
2022-12-11 18:31:43 +00:00
Kbv = self.getPubkey(kbv)
address_b58 = xmr_util.encode_address(Kbv, Kbs)
wallet_file = address_b58 + '_spend'
try:
self.openWallet(wallet_file)
except Exception:
wallet_file = address_b58
2022-12-11 18:31:43 +00:00
try:
self.openWallet(wallet_file)
except Exception:
self._log.info(f'showLockTransfers trying to create wallet for address {address_b58}.')
kbv_le = kbv[::-1]
params = {
'restore_height': restore_height,
'filename': address_b58,
'address': address_b58,
'viewkey': b2h(kbv_le),
}
self.createWallet(params)
self.openWallet(address_b58)
2023-12-29 13:36:00 +00:00
self.rpc_wallet('refresh')
2023-12-29 13:36:00 +00:00
rv = self.rpc_wallet('get_transfers', {'in': True, 'out': True, 'pending': True, 'failed': True})
rv['filename'] = wallet_file
return rv
except Exception as e:
return {'error': str(e)}
2023-07-05 21:35:25 +00:00
def getSpendableBalance(self) -> int:
with self._mx_wallet:
self.openWallet(self._wallet_filename)
2023-12-29 13:36:00 +00:00
self.rpc_wallet('refresh')
balance_info = self.rpc_wallet('get_balance')
return balance_info['unlocked_balance']
def changeWalletPassword(self, old_password, new_password):
self._log.info('changeWalletPassword - {}'.format(self.ticker()))
orig_password = self._wallet_password
if old_password != '':
self._wallet_password = old_password
try:
self.openWallet(self._wallet_filename)
2023-12-29 13:36:00 +00:00
self.rpc_wallet('change_wallet_password', {'old_password': old_password, 'new_password': new_password})
except Exception as e:
self._wallet_password = orig_password
raise e
2023-02-17 23:47:44 +00:00
def unlockWallet(self, password: str) -> None:
self._log.info('unlockWallet - {}'.format(self.ticker()))
self._wallet_password = password
if not self._have_checked_seed:
self._sc.checkWalletSeed(self.coin_type())
2023-02-17 23:47:44 +00:00
def lockWallet(self) -> None:
self._log.info('lockWallet - {}'.format(self.ticker()))
self._wallet_password = None
def isAddressMine(self, address):
# TODO
return True
2023-07-05 21:35:25 +00:00
2023-07-18 23:19:04 +00:00
def ensureFunds(self, amount: int) -> None:
2023-07-05 21:35:25 +00:00
if self.getSpendableBalance() < amount:
raise ValueError('Balance too low')
2023-07-18 23:19:04 +00:00
def getTransaction(self, txid: bytes):
2023-12-29 13:36:00 +00:00
return self.rpc2('get_transactions', {'txs_hashes': [txid.hex(), ]})