Files
purple-electrumwallet/tests/test_txbatcher.py
T

292 lines
14 KiB
Python
Raw Normal View History

2025-02-14 14:12:12 +01:00
import logging
from unittest import mock
import asyncio
import dataclasses
2025-06-05 14:37:05 +02:00
from aiorpcx import timeout_after
2025-02-14 14:12:12 +01:00
import electrum.fee_policy
from electrum import keystore, wallet, lnutil
2025-02-14 14:12:12 +01:00
from electrum import SimpleConfig
from electrum import util
from electrum.address_synchronizer import TX_HEIGHT_UNCONFIRMED
2025-02-14 14:12:12 +01:00
from electrum.transaction import Transaction, PartialTxInput, PartialTxOutput, TxOutpoint
from electrum.logging import console_stderr_handler, Logger
from electrum.submarine_swaps import SwapManager, SwapData
from electrum.lnsweep import SweepInfo, sweep_ctx_anchor
2025-02-14 14:12:12 +01:00
from electrum.fee_policy import FeeTimeEstimates
from . import ElectrumTestCase
2025-03-22 09:46:18 +01:00
from .test_wallet_vertical import WalletIntegrityHelper, read_test_vector
WALLET_DATA = read_test_vector('cause_carbon_wallet.json')
2025-02-14 14:12:12 +01:00
class MockNetwork(Logger):
def __init__(self, config):
Logger.__init__(self)
2025-02-14 14:12:12 +01:00
self.config = config
self.fee_estimates = FeeTimeEstimates()
self.asyncio_loop = util.get_asyncio_loop()
self.interface = None
self.relay_fee = 1000
self.wallets = []
self._tx_queue = asyncio.Queue()
2025-02-14 14:12:12 +01:00
def get_local_height(self):
return 42
def blockchain(self):
class BlockchainMock:
def is_tip_stale(self):
return True
return BlockchainMock()
async def try_broadcasting(self, tx, name):
for w in self.wallets:
w.adb.receive_tx_callback(tx, tx_height=TX_HEIGHT_UNCONFIRMED)
self._tx_queue.put_nowait(tx)
2025-02-14 14:12:12 +01:00
return tx.txid()
async def next_tx(self):
tx = await util.wait_for2(self._tx_queue.get(), timeout=10)
return tx
2025-02-14 14:12:12 +01:00
2025-03-28 15:49:25 +01:00
def is_connected(self):
return True
2025-02-14 14:12:12 +01:00
SWAP_FUNDING_TX = "01000000000101500e9d67647481864edfb020b5c45e1c40d90f06b0130f9faed1a5149c6d26450000000000ffffffff0226080300000000002200205059c44bf57534303ab8f090f06b7bde58f5d2522440247a1ff6b41bdca9348df312c20100000000160014021d4f3b17921d790e1c022367a5bb078ce4deb402483045022100d41331089a2031396a1db8e4dec6dda9cacefe1288644b92f8e08a23325aa19b02204159230691601f7d726e4e6e0b7124d3377620f400d699a01095f0b0a09ee26a012102d60315c72c0cefd41c6d07883c20b88be3fc37aac7912f0052722a95de0de71600000000"
2025-05-13 09:09:22 +02:00
SWAP_CLAIM_TX = "02000000000101f9db8580febd5c0f85b6f1576c83f7739109e3a2d772743e3217e9537fea7e89000000000001000000017005030000000000160014b113a47f3718da3fd161339a6681c150fef2cfe30347304402204c6d40103589b1a8177a37a824f0c66a3a7b22bc570b14c9e07965b56f6ace8f02203a35cffe0ab10de00f3e15ecf5aafdd2c7f6c62da11edd9054a1bce7a9e1455c0120f1939b5723155713855d7ebea6e174f77d41d669269e7f138856c3de190e7a366a8201208763a914d7a62ef0270960fe23f0f351b28caadab62c21838821030bfd61153816df786036ea293edce851d3a4b9f4a1c66bdc1a17f00ffef3d6b167750334ef24b1752102fc8128f17f9e666ea281c702171ab16c1dd2a4337b71f08970f5aa10c608a93268ac00000000"
2025-02-14 14:12:12 +01:00
2025-06-05 14:37:05 +02:00
SWAPDATA = SwapData(
is_reverse=True,
locktime=2420532,
onchain_amount=198694,
lightning_amount=200000,
redeem_script=bytes.fromhex('8201208763a914d7a62ef0270960fe23f0f351b28caadab62c21838821030bfd61153816df786036ea293edce851d3a4b9f4a1c66bdc1a17f00ffef3d6b167750334ef24b1752102fc8128f17f9e666ea281c702171ab16c1dd2a4337b71f08970f5aa10c608a93268ac'),
preimage=bytes.fromhex('f1939b5723155713855d7ebea6e174f77d41d669269e7f138856c3de190e7a36'),
prepay_hash=None,
privkey=bytes.fromhex('58fd0018a9a2737d1d6b81d380df96bf0c858473a9592015508a270a7c9b1d8d'),
lockup_address='tb1q2pvugjl4w56rqw4c7zg0q6mmmev0t5jjy3qzg7sl766phh9fxjxsrtl77t',
claim_to_output=None,
2025-06-05 14:37:05 +02:00
funding_txid='897eea7f53e917323e7472d7a2e3099173f7836c57f1b6850f5cbdfe8085dbf9',
spending_txid=None,
is_redeemed=False,
)
txin = PartialTxInput(
prevout=TxOutpoint(txid=bytes.fromhex(SWAPDATA.funding_txid), out_idx=0),
)
txin._trusted_value_sats = SWAPDATA.onchain_amount
txin, locktime = SwapManager.create_claim_txin(txin=txin, swap=SWAPDATA)
SWAP_SWEEP_INFO = SweepInfo(
txin=txin,
cltv_abs=locktime,
txout=None,
name='swap claim',
can_be_batched=True,
2025-11-03 13:36:37 +01:00
dust_override=False,
2025-06-05 14:37:05 +02:00
)
anchor_chan_ctx = Transaction(
"02000000000101d24af3b7adefff5a068f736d64842c18da7087b41ba43ab5b999c545c5f1606501000000008d0d5d8"
"0024a010000000000002200207b95cb2555b3f8fc246d26ac38023ec8edf423d70b41dfe17efc89baa6e0cc72740003"
"000000000022002075f8af76a5b5c4b25e4aee3a4f96a190a168bde5d9761de8d45d3e49cd6f1d82040047304402200"
"6524eb2f467bf1eacd2116ea79a80c182eb95e18d0fa24fc5c600581ec4aa5f02206f50bdfc3577ca3bc9892eade7d8"
"1a9b06b9a8803296fad689af9fee9375191d0147304402202112d08ffa79010b1d698ca9fb0790119a42f7b70e183a2"
"a05dbf79c32806d3b022001cd3e3aec8c1142c8689e9e679c684ea82dcde8c86cf8e55466c743b3647b1f0147522103"
"0551e6017a0e9dbffd468c2a08ecf8446b532f0a6d5db291eb77026f2ef3deb421034f986fd43561d52a96b19fbdd0c"
"296f0442034d3f3f63fc394320a95750d42a852ae08572d20"
)
chan_multisig_key = lnutil.Keypair(
privkey="4c8b2c19d6528f54a4c900d87b3d8dbf746314925d126e0abf8e2ed965a9302f",
pubkey="034f986fd43561d52a96b19fbdd0c296f0442034d3f3f63fc394320a95750d42a8",
)
anchor_txin = sweep_ctx_anchor(ctx=anchor_chan_ctx, multisig_key=chan_multisig_key)
ANCHOR_SWEEP_INFO = SweepInfo(
name='local_anchor',
cltv_abs=None,
txin=anchor_txin,
txout=None,
can_be_batched=True,
dust_override=True,
)
2025-06-05 14:37:05 +02:00
2025-02-14 14:12:12 +01:00
class TestTxBatcher(ElectrumTestCase):
TESTNET = True
@classmethod
def setUpClass(cls):
super().setUpClass()
console_stderr_handler.setLevel(logging.DEBUG)
def setUp(self):
super().setUp()
self.config = SimpleConfig({'electrum_path': self.electrum_path})
self.config.FEE_POLICY = 'feerate:5000'
2025-02-14 14:12:12 +01:00
async def asyncSetUp(self):
await super().asyncSetUp()
self.network = MockNetwork(self.config)
def create_standard_wallet_from_seed(self, seed_words, *, config=None, gap_limit=2):
if config is None:
config = self.config
ks = keystore.from_seed(seed_words, passphrase='', for_multisig=False)
return WalletIntegrityHelper.create_standard_wallet(ks, gap_limit=gap_limit, config=config)
def _create_wallet(self):
2025-03-22 09:46:18 +01:00
wallet = self.create_standard_wallet_from_seed(WALLET_DATA["seed"])
wallet.start_network(self.network)
wallet.txbatcher.SLEEP_INTERVAL = 0.01
self.network.wallets.append(wallet)
return wallet
2025-02-14 14:12:12 +01:00
@mock.patch.object(wallet.Abstract_Wallet, 'save_db')
async def test_batch_payments(self, mock_save_db):
# output 1: tx1(o1) ---------------
# \
# output 2: tx1'(o1,o2) ----> tx2(tx1|o2)
#
# tx1 is broadcast, and replaced by tx1'
# tx1 gets mined
# txbatcher creates a new transaction tx2, child of tx1
#
OUTGOING_ADDRESS = 'tb1q7rl9cxr85962ztnsze089zs8ycv52hk43f3m9n'
wallet = self._create_wallet()
2025-02-14 14:12:12 +01:00
# fund wallet
2025-03-22 09:46:18 +01:00
funding_tx = Transaction(WALLET_DATA["funding_tx"])
2025-02-14 14:12:12 +01:00
await self.network.try_broadcasting(funding_tx, 'funding')
await self.network.next_tx()
2025-02-14 14:12:12 +01:00
assert wallet.adb.get_transaction(funding_tx.txid()) is not None
self.logger.info(f'wallet balance {wallet.get_balance()}')
# payment 1 -> tx1(output1)
output1 = PartialTxOutput.from_address_and_value(OUTGOING_ADDRESS, 10_000)
wallet.txbatcher.add_payment_output('default', output1)
2025-02-14 14:12:12 +01:00
tx1 = await self.network.next_tx()
assert output1 in tx1.outputs()
# payment 2 -> tx2(output1, output2)
output2 = PartialTxOutput.from_address_and_value(OUTGOING_ADDRESS, 20_000)
wallet.txbatcher.add_payment_output('default', output2)
2025-02-14 14:12:12 +01:00
tx1_prime = await self.network.next_tx()
assert wallet.adb.get_transaction(tx1_prime.txid()) is not None
assert len(tx1_prime.outputs()) == 3
assert output1 in tx1_prime.outputs()
assert output2 in tx1_prime.outputs()
# tx1 gets confirmed, tx2 gets removed
wallet.adb.receive_tx_callback(tx1, tx_height=1)
2025-02-14 14:12:12 +01:00
tx_mined_status = wallet.adb.get_tx_height(tx1.txid())
wallet.adb.add_verified_tx(tx1.txid(), dataclasses.replace(tx_mined_status, conf=1))
2025-02-14 14:12:12 +01:00
assert wallet.adb.get_transaction(tx1.txid()) is not None
assert wallet.adb.get_transaction(tx1_prime.txid()) is None
# txbatcher creates tx2
tx2 = await self.network.next_tx()
assert output1 in tx1.outputs()
assert output2 in tx2.outputs()
# check that tx2 is child of tx1
assert len(tx2.inputs()) == 1
assert tx2.inputs()[0].prevout.txid.hex() == tx1.txid()
@mock.patch.object(wallet.Abstract_Wallet, 'save_db')
async def test_rbf_batching__cannot_batch_as_would_need_to_use_ismine_outputs_of_basetx(self, mock_save_db):
"""Wallet history contains unconf tx1 that spends all its coins to two ismine outputs,
one 'recv' address (20k sats) and one 'change' (80k sats).
The user tries to create tx2, that pays an invoice for 90k sats.
The tx batcher fails to batch, and should create a child transaction
"""
wallet = self._create_wallet()
2025-02-14 14:12:12 +01:00
# fund wallet
2025-03-22 09:46:18 +01:00
funding_tx = Transaction(WALLET_DATA['funding_tx'])
2025-02-14 14:12:12 +01:00
await self.network.try_broadcasting(funding_tx, 'funding')
await self.network.next_tx()
2025-02-14 14:12:12 +01:00
assert wallet.adb.get_transaction(funding_tx.txid()) is not None
self.logger.info(f'wallet balance1 {wallet.get_balance()}')
# to_self_payment tx1
output1 = PartialTxOutput.from_address_and_value("tb1qyfnv3y866ufedugxxxfksyratv4pz3h78g9dad", 20_000)
wallet.txbatcher.add_payment_output('default', output1)
tx1 = await self.network.next_tx()
assert len(tx1.outputs()) == 2
assert output1 in tx1.outputs()
2025-02-14 14:12:12 +01:00
# outgoing payment tx2
output2 = PartialTxOutput.from_address_and_value("tb1qkfn0fude7z789uys2u7sf80kd4805zpvs3na0h", 90_000)
wallet.txbatcher.add_payment_output('default', output2)
# before tx1 gets confirmed, txbatch.create_transaction will raise notenoughfunds
await asyncio.sleep(wallet.txbatcher.SLEEP_INTERVAL)
# tx1 gets confirmed
wallet.adb.receive_tx_callback(tx1, tx_height=1)
tx_mined_status = wallet.adb.get_tx_height(tx1.txid())
wallet.adb.add_verified_tx(tx1.txid(), dataclasses.replace(tx_mined_status, conf=1))
2025-02-14 14:12:12 +01:00
tx2 = await self.network.next_tx()
assert len(tx2.outputs()) == 2
assert output2 in tx2.outputs()
@mock.patch.object(wallet.Abstract_Wallet, 'save_db')
async def test_sweep_from_submarine_swap(self, mock_save_db):
self.maxDiff = None
# create wallet
wallet = self._create_wallet()
2025-06-05 14:37:05 +02:00
wallet.adb.db.transactions[SWAPDATA.funding_txid] = tx = Transaction(SWAP_FUNDING_TX)
wallet.adb.receive_tx_callback(tx, tx_height=1)
tx_mined_status = wallet.adb.get_tx_height(tx.txid())
wallet.adb.add_verified_tx(tx.txid(), dataclasses.replace(tx_mined_status, conf=1))
2025-06-05 14:37:05 +02:00
wallet.txbatcher.add_sweep_input('default', SWAP_SWEEP_INFO)
2025-02-14 14:12:12 +01:00
tx = await self.network.next_tx()
txid = tx.txid()
self.assertEqual(SWAP_CLAIM_TX, str(tx))
# add a new payment, reusing the same input
# this tests that txin.make_witness() can be called more than once
output1 = PartialTxOutput.from_address_and_value("tb1qyfnv3y866ufedugxxxfksyratv4pz3h78g9dad", 20_000)
wallet.txbatcher.add_payment_output('default', output1)
2025-02-14 14:12:12 +01:00
new_tx = await self.network.next_tx()
# check that we batched with previous tx
assert new_tx.inputs()[0].prevout == tx.inputs()[0].prevout == txin.prevout
assert output1 in new_tx.outputs()
2025-06-05 14:37:05 +02:00
async def test_to_sweep_after_anchor_sweep_conditions(self):
# create wallet
wallet = self._create_wallet()
wallet.txbatcher.add_sweep_input('lnwatcher', ANCHOR_SWEEP_INFO)
anchor_batch = wallet.txbatcher.tx_batches['lnwatcher']
# does not return sweep info if prev_tx is not in db
to_sweep_no_tx = anchor_batch._to_sweep_after(tx=None)
self.assertFalse(to_sweep_no_tx)
# returns sweep input if anchor_chan_ctx conf < 1
wallet.adb.db.transactions[anchor_chan_ctx.txid()] = anchor_chan_ctx
to_sweep_no_conf = anchor_batch._to_sweep_after(tx=None)
self.assertEqual(to_sweep_no_conf[anchor_txin.prevout], ANCHOR_SWEEP_INFO)
self.assertEqual(len(to_sweep_no_conf), 1)
# does not return sweep input if ctx fee is already higher than target fee
with mock.patch.object(wallet.adb, 'get_tx_fee', return_value=2000), \
mock.patch.object(electrum.fee_policy.FeePolicy, 'estimate_fee', return_value=1000):
to_sweep_high_fee = anchor_batch._to_sweep_after(tx=None)
self.assertFalse(to_sweep_high_fee)
# after the ctx is confirmed the anchor claim shouldn't be broadcast anymore
wallet.adb.receive_tx_callback(anchor_chan_ctx, tx_height=1)
tx_mined_status = wallet.adb.get_tx_height(anchor_chan_ctx.txid())
wallet.adb.add_verified_tx(anchor_chan_ctx.txid(), dataclasses.replace(tx_mined_status, conf=1))
self.assertIn(anchor_txin.prevout, anchor_batch.batch_inputs)
to_sweep_ctx_conf = anchor_batch._to_sweep_after(tx=None)
self.assertFalse(to_sweep_ctx_conf)
self.assertFalse(anchor_batch.batch_inputs)
self.assertEqual(wallet.txbatcher.tx_batches['lnwatcher'], anchor_batch)
2025-06-05 14:37:05 +02:00
async def _wait_for_base_tx(self, txbatch, should_be_none=False):
async with timeout_after(10):
while True:
base_tx = txbatch._base_tx
if (base_tx is not None) ^ should_be_none:
return base_tx
await asyncio.sleep(0.1)