Browse Source

open_channel_coroutine: do not timeout on sign_transaction. fixes #7027

patch-4
ThomasV 4 years ago
parent
commit
1323bd4f9c
  1. 7
      electrum/lnpeer.py
  2. 25
      electrum/lnworker.py

7
electrum/lnpeer.py

@ -563,7 +563,6 @@ class Peer(Logger):
@temporarily_reserve_funding_tx_change_address @temporarily_reserve_funding_tx_change_address
async def channel_establishment_flow( async def channel_establishment_flow(
self, *, self, *,
password: Optional[str],
funding_tx: 'PartialTransaction', funding_tx: 'PartialTransaction',
funding_sat: int, funding_sat: int,
push_msat: int, push_msat: int,
@ -578,6 +577,7 @@ class Peer(Logger):
Channel configurations are initialized in this method. Channel configurations are initialized in this method.
""" """
# will raise if init fails
await asyncio.wait_for(self.initialized, LN_P2P_NETWORK_TIMEOUT) await asyncio.wait_for(self.initialized, LN_P2P_NETWORK_TIMEOUT)
feerate = self.lnworker.current_feerate_per_kw() feerate = self.lnworker.current_feerate_per_kw()
@ -675,9 +675,8 @@ class Peer(Logger):
funding_tx.outputs().remove(dummy_output) funding_tx.outputs().remove(dummy_output)
funding_tx.add_outputs([funding_output]) funding_tx.add_outputs([funding_output])
funding_tx.set_rbf(False) funding_tx.set_rbf(False)
self.lnworker.wallet.sign_transaction(funding_tx, password) if not funding_tx.is_segwit():
if not funding_tx.is_complete() and not funding_tx.is_segwit(): raise Exception('Funding transaction is not segwit')
raise Exception('Funding transaction is not complete')
funding_txid = funding_tx.txid() funding_txid = funding_tx.txid()
assert funding_txid assert funding_txid
funding_index = funding_tx.outputs().index(funding_output) funding_index = funding_tx.outputs().index(funding_output)

25
electrum/lnworker.py

@ -861,20 +861,21 @@ class LNWallet(LNWorker):
await self.network.try_broadcasting(force_close_tx, 'force-close') await self.network.try_broadcasting(force_close_tx, 'force-close')
@log_exceptions @log_exceptions
async def _open_channel_coroutine(self, *, connect_str: str, funding_tx: PartialTransaction, async def _open_channel_coroutine(
funding_sat: int, push_sat: int, self, *, connect_str: str,
password: Optional[str]) -> Tuple[Channel, PartialTransaction]: funding_tx: PartialTransaction,
funding_sat: int, push_sat: int,
password: Optional[str]) -> Tuple[Channel, PartialTransaction]:
peer = await self.add_peer(connect_str) peer = await self.add_peer(connect_str)
# will raise if init fails coro = peer.channel_establishment_flow(
await asyncio.wait_for(peer.initialized, LN_P2P_NETWORK_TIMEOUT)
chan, funding_tx = await peer.channel_establishment_flow(
password=password,
funding_tx=funding_tx, funding_tx=funding_tx,
funding_sat=funding_sat, funding_sat=funding_sat,
push_msat=push_sat * 1000, push_msat=push_sat * 1000,
temp_channel_id=os.urandom(32)) temp_channel_id=os.urandom(32))
chan, funding_tx = await asyncio.wait_for(coro, LN_P2P_NETWORK_TIMEOUT)
util.trigger_callback('channels_updated', self.wallet) util.trigger_callback('channels_updated', self.wallet)
self.wallet.add_transaction(funding_tx) # save tx as local into the wallet self.wallet.add_transaction(funding_tx) # save tx as local into the wallet
self.wallet.sign_transaction(funding_tx, password)
self.wallet.set_label(funding_tx.txid(), _('Open channel')) self.wallet.set_label(funding_tx.txid(), _('Open channel'))
if funding_tx.is_complete(): if funding_tx.is_complete():
await self.network.try_broadcasting(funding_tx, 'open_channel') await self.network.try_broadcasting(funding_tx, 'open_channel')
@ -911,15 +912,15 @@ class LNWallet(LNWorker):
return tx return tx
def open_channel(self, *, connect_str: str, funding_tx: PartialTransaction, def open_channel(self, *, connect_str: str, funding_tx: PartialTransaction,
funding_sat: int, push_amt_sat: int, password: str = None, funding_sat: int, push_amt_sat: int, password: str = None) -> Tuple[Channel, PartialTransaction]:
timeout: Optional[int] = 20) -> Tuple[Channel, PartialTransaction]:
if funding_sat > LN_MAX_FUNDING_SAT: if funding_sat > LN_MAX_FUNDING_SAT:
raise Exception(_("Requested channel capacity is over protocol allowed maximum.")) raise Exception(_("Requested channel capacity is over protocol allowed maximum."))
coro = self._open_channel_coroutine(connect_str=connect_str, funding_tx=funding_tx, funding_sat=funding_sat, coro = self._open_channel_coroutine(
push_sat=push_amt_sat, password=password) connect_str=connect_str, funding_tx=funding_tx, funding_sat=funding_sat,
push_sat=push_amt_sat, password=password)
fut = asyncio.run_coroutine_threadsafe(coro, self.network.asyncio_loop) fut = asyncio.run_coroutine_threadsafe(coro, self.network.asyncio_loop)
try: try:
chan, funding_tx = fut.result(timeout=timeout) chan, funding_tx = fut.result()
except concurrent.futures.TimeoutError: except concurrent.futures.TimeoutError:
raise Exception(_("open_channel timed out")) raise Exception(_("open_channel timed out"))
# at this point the channel opening was successful # at this point the channel opening was successful

Loading…
Cancel
Save