Browse Source

turn lightning_payments_completed into dict. Show status of lightning payments in GUI. Make 'listchannels' available offline

dependabot/pip/contrib/deterministic-build/ecdsa-0.13.3
ThomasV 6 years ago
parent
commit
b0d6000771
  1. 35
      electrum/commands.py
  2. 2
      electrum/gui/kivy/uix/screens.py
  3. 12
      electrum/gui/qt/request_list.py
  4. 11
      electrum/gui/qt/util.py
  5. 52
      electrum/lnworker.py
  6. 7
      electrum/paymentrequest.py
  7. 7
      electrum/util.py

35
electrum/commands.py

@ -786,7 +786,7 @@ class Commands:
def nodeid(self): def nodeid(self):
return bh2u(self.wallet.lnworker.node_keypair.pubkey) return bh2u(self.wallet.lnworker.node_keypair.pubkey)
@command('wn') @command('w')
def listchannels(self): def listchannels(self):
return list(self.wallet.lnworker.list_channels()) return list(self.wallet.lnworker.list_channels())
@ -806,7 +806,38 @@ class Commands:
@command('w') @command('w')
def listinvoices(self): def listinvoices(self):
return "\n".join(self.wallet.lnworker.list_invoices()) report = self.wallet.lnworker._list_invoices()
return '\n'.join(self._format_ln_invoices(report))
def _format_ln_invoices(self, report):
from .lnutil import SENT
if report['settled']:
yield 'Settled invoices:'
yield '-----------------'
for date, direction, htlc, preimage in sorted(report['settled']):
# astimezone converts to local time
# replace removes the tz info since we don't need to display it
yield 'Paid at: ' + date.astimezone().replace(tzinfo=None).isoformat(sep=' ', timespec='minutes')
yield 'We paid' if direction == SENT else 'They paid'
yield str(htlc)
yield 'Preimage: ' + (bh2u(preimage) if preimage else 'Not available') # if delete_invoice was called
yield ''
if report['unsettled']:
yield 'Your unsettled invoices:'
yield '------------------------'
for addr, preimage, pay_req in report['unsettled']:
yield pay_req
yield str(addr)
yield 'Preimage: ' + bh2u(preimage)
yield ''
if report['inflight']:
yield 'Outgoing payments in progress:'
yield '------------------------------'
for addr, htlc, direction in report['inflight']:
yield str(addr)
yield str(htlc)
yield ''
@command('wn') @command('wn')
def closechannel(self, channel_point, force=False): def closechannel(self, channel_point, force=False):

2
electrum/gui/kivy/uix/screens.py

@ -26,7 +26,7 @@ from electrum.util import profiler, parse_URI, format_time, InvalidPassword, Not
from electrum import bitcoin, constants from electrum import bitcoin, constants
from electrum.transaction import TxOutput, Transaction, tx_from_str from electrum.transaction import TxOutput, Transaction, tx_from_str
from electrum.util import send_exception_to_crash_reporter, parse_URI, InvalidBitcoinURI from electrum.util import send_exception_to_crash_reporter, parse_URI, InvalidBitcoinURI
from electrum.paymentrequest import PR_UNPAID, PR_PAID, PR_UNKNOWN, PR_EXPIRED from electrum.util import PR_UNPAID, PR_PAID, PR_UNKNOWN, PR_EXPIRED
from electrum.plugin import run_hook from electrum.plugin import run_hook
from electrum.wallet import InternalAddressCorruption from electrum.wallet import InternalAddressCorruption
from electrum import simple_config from electrum import simple_config

12
electrum/gui/qt/request_list.py

@ -31,8 +31,8 @@ from PyQt5.QtCore import Qt, QItemSelectionModel
from electrum.i18n import _ from electrum.i18n import _
from electrum.util import format_time, age from electrum.util import format_time, age
from electrum.util import PR_UNPAID, PR_EXPIRED, PR_PAID, PR_UNKNOWN, PR_INFLIGHT
from electrum.plugin import run_hook from electrum.plugin import run_hook
from electrum.paymentrequest import PR_UNKNOWN
from electrum.wallet import InternalAddressCorruption from electrum.wallet import InternalAddressCorruption
from electrum.bitcoin import COIN from electrum.bitcoin import COIN
from electrum.lnaddr import lndecode from electrum.lnaddr import lndecode
@ -144,7 +144,9 @@ class RequestList(MyTreeView):
items[0].setData(address, ROLE_RHASH_OR_ADDR) items[0].setData(address, ROLE_RHASH_OR_ADDR)
self.filter() self.filter()
# lightning # lightning
for payreq_key, (preimage_hex, invoice) in self.wallet.lnworker.invoices.items(): lnworker = self.wallet.lnworker
for key, (preimage_hex, invoice) in lnworker.invoices.items():
status = lnworker.get_invoice_status(key)
lnaddr = lndecode(invoice, expected_hrp=constants.net.SEGWIT_HRP) lnaddr = lndecode(invoice, expected_hrp=constants.net.SEGWIT_HRP)
amount_sat = lnaddr.amount*COIN if lnaddr.amount else None amount_sat = lnaddr.amount*COIN if lnaddr.amount else None
amount_str = self.parent.format_amount(amount_sat) if amount_sat else '' amount_str = self.parent.format_amount(amount_sat) if amount_sat else ''
@ -154,11 +156,13 @@ class RequestList(MyTreeView):
description = v description = v
break break
date = format_time(lnaddr.date) date = format_time(lnaddr.date)
labels = [date, 'lightning', description, amount_str, ''] labels = [date, 'lightning', description, amount_str, pr_tooltips.get(status,'')]
items = [QStandardItem(e) for e in labels] items = [QStandardItem(e) for e in labels]
items[1].setIcon(self.icon_cache.get(":icons/lightning.png")) items[1].setIcon(self.icon_cache.get(":icons/lightning.png"))
items[0].setData(REQUEST_TYPE_LN, ROLE_REQUEST_TYPE) items[0].setData(REQUEST_TYPE_LN, ROLE_REQUEST_TYPE)
items[0].setData(payreq_key, ROLE_RHASH_OR_ADDR) items[0].setData(key, ROLE_RHASH_OR_ADDR)
if status is not PR_UNKNOWN:
items[4].setIcon(self.icon_cache.get(pr_icons.get(status)))
self.model().insertRow(self.model().rowCount(), items) self.model().insertRow(self.model().rowCount(), items)
# sort requests by date # sort requests by date
self.model().sort(0) self.model().sort(0)

11
electrum/gui/qt/util.py

@ -23,9 +23,8 @@ from PyQt5.QtWidgets import (QPushButton, QLabel, QMessageBox, QHBoxLayout,
QHeaderView, QApplication, QToolTip, QTreeWidget, QStyledItemDelegate) QHeaderView, QApplication, QToolTip, QTreeWidget, QStyledItemDelegate)
from electrum.i18n import _, languages from electrum.i18n import _, languages
from electrum.util import (FileImportFailed, FileExportFailed, from electrum.util import FileImportFailed, FileExportFailed, make_aiohttp_session, PrintError, resource_path
resource_path) from electrum.util import PR_UNPAID, PR_PAID, PR_EXPIRED, PR_INFLIGHT
from electrum.paymentrequest import PR_UNPAID, PR_PAID, PR_EXPIRED
if TYPE_CHECKING: if TYPE_CHECKING:
from .main_window import ElectrumWindow from .main_window import ElectrumWindow
@ -44,13 +43,15 @@ dialogs = []
pr_icons = { pr_icons = {
PR_UNPAID:"unpaid.png", PR_UNPAID:"unpaid.png",
PR_PAID:"confirmed.png", PR_PAID:"confirmed.png",
PR_EXPIRED:"expired.png" PR_EXPIRED:"expired.png",
PR_INFLIGHT:"lightning.png",
} }
pr_tooltips = { pr_tooltips = {
PR_UNPAID:_('Pending'), PR_UNPAID:_('Pending'),
PR_PAID:_('Paid'), PR_PAID:_('Paid'),
PR_EXPIRED:_('Expired') PR_EXPIRED:_('Expired'),
PR_INFLIGHT:_('Inflight')
} }
expiration_values = [ expiration_values = [

52
electrum/lnworker.py

@ -67,13 +67,14 @@ class LNWorker(PrintError):
def __init__(self, wallet: 'Abstract_Wallet'): def __init__(self, wallet: 'Abstract_Wallet'):
self.wallet = wallet self.wallet = wallet
# invoices we are currently trying to pay (might be pending HTLCs on a commitment transaction) # invoices we are currently trying to pay (might be pending HTLCs on a commitment transaction)
self.invoices = self.wallet.storage.get('lightning_invoices', {}) # type: Dict[str, Tuple[str,str]] # RHASH -> (preimage, invoice)
self.paying = self.wallet.storage.get('lightning_payments_inflight', {}) # type: Dict[bytes, Tuple[str, Optional[int], str]] self.paying = self.wallet.storage.get('lightning_payments_inflight', {}) # type: Dict[bytes, Tuple[str, Optional[int], str]]
self.completed = self.wallet.storage.get('lightning_payments_completed', {})
self.sweep_address = wallet.get_receiving_address() self.sweep_address = wallet.get_receiving_address()
self.lock = threading.RLock() self.lock = threading.RLock()
self.ln_keystore = self._read_ln_keystore() self.ln_keystore = self._read_ln_keystore()
self.node_keypair = generate_keypair(self.ln_keystore, LnKeyFamily.NODE_KEY, 0) self.node_keypair = generate_keypair(self.ln_keystore, LnKeyFamily.NODE_KEY, 0)
self.peers = {} # type: Dict[bytes, Peer] # pubkey -> Peer self.peers = {} # type: Dict[bytes, Peer] # pubkey -> Peer
self.invoices = wallet.storage.get('lightning_invoices', {}) # type: Dict[str, Tuple[str,str]] # RHASH -> (preimage, invoice)
self.channels = {} # type: Dict[bytes, Channel] self.channels = {} # type: Dict[bytes, Channel]
for x in wallet.storage.get("channels", []): for x in wallet.storage.get("channels", []):
c = Channel(x, sweep_address=self.sweep_address, payment_completed=self.payment_completed) c = Channel(x, sweep_address=self.sweep_address, payment_completed=self.payment_completed)
@ -123,56 +124,37 @@ class LNWorker(PrintError):
def payment_completed(self, chan, direction, htlc, preimage): def payment_completed(self, chan, direction, htlc, preimage):
assert type(direction) is Direction assert type(direction) is Direction
key = bh2u(htlc.payment_hash)
chan_id = chan.channel_id chan_id = chan.channel_id
if direction == SENT: if direction == SENT:
assert htlc.payment_hash not in self.invoices assert htlc.payment_hash not in self.invoices
self.paying.pop(bh2u(htlc.payment_hash)) self.paying.pop(key)
self.wallet.storage.put('lightning_payments_inflight', self.paying) self.wallet.storage.put('lightning_payments_inflight', self.paying)
l = self.wallet.storage.get('lightning_payments_completed', [])
if not preimage: if not preimage:
preimage, _addr = self.get_invoice(htlc.payment_hash) preimage, _addr = self.get_invoice(htlc.payment_hash)
tupl = (time.time(), direction, json.loads(encoder.encode(htlc)), bh2u(preimage), bh2u(chan_id)) tupl = (time.time(), direction, json.loads(encoder.encode(htlc)), bh2u(preimage), bh2u(chan_id))
l.append(tupl) self.completed[key] = tupl
self.wallet.storage.put('lightning_payments_completed', l) self.wallet.storage.put('lightning_payments_completed', self.completed)
self.wallet.storage.write() self.wallet.storage.write()
self.network.trigger_callback('ln_payment_completed', tupl[0], direction, htlc, preimage, chan_id) self.network.trigger_callback('ln_payment_completed', tupl[0], direction, htlc, preimage, chan_id)
def list_invoices(self): def get_invoice_status(self, key):
report = self._list_invoices() from electrum.util import PR_UNPAID, PR_EXPIRED, PR_PAID, PR_UNKNOWN, PR_INFLIGHT
if report['settled']: if key in self.completed:
yield 'Settled invoices:' return PR_PAID
yield '-----------------' elif key in self.paying:
for date, direction, htlc, preimage in sorted(report['settled']): return PR_INFLIGHT
# astimezone converts to local time elif key in self.invoices:
# replace removes the tz info since we don't need to display it return PR_UNPAID
yield 'Paid at: ' + date.astimezone().replace(tzinfo=None).isoformat(sep=' ', timespec='minutes') else:
yield 'We paid' if direction == SENT else 'They paid' return PR_UNKNOWN
yield str(htlc)
yield 'Preimage: ' + (bh2u(preimage) if preimage else 'Not available') # if delete_invoice was called
yield ''
if report['unsettled']:
yield 'Your unsettled invoices:'
yield '------------------------'
for addr, preimage, pay_req in report['unsettled']:
yield pay_req
yield str(addr)
yield 'Preimage: ' + bh2u(preimage)
yield ''
if report['inflight']:
yield 'Outgoing payments in progress:'
yield '------------------------------'
for addr, htlc, direction in report['inflight']:
yield str(addr)
yield str(htlc)
yield ''
def _list_invoices(self, chan_id=None): def _list_invoices(self, chan_id=None):
invoices = dict(self.invoices) invoices = dict(self.invoices)
completed = self.wallet.storage.get('lightning_payments_completed', [])
settled = [] settled = []
unsettled = [] unsettled = []
inflight = [] inflight = []
for date, direction, htlc, hex_preimage, hex_chan_id in completed: for date, direction, htlc, hex_preimage, hex_chan_id in self.completed.values():
direction = Direction(direction) direction = Direction(direction)
if chan_id is not None: if chan_id is not None:
if bfh(hex_chan_id) != chan_id: if bfh(hex_chan_id) != chan_id:

7
electrum/paymentrequest.py

@ -41,6 +41,7 @@ except ImportError:
from . import bitcoin, ecc, util, transaction, x509, rsakey from . import bitcoin, ecc, util, transaction, x509, rsakey
from .util import bh2u, bfh, export_meta, import_meta, make_aiohttp_session from .util import bh2u, bfh, export_meta, import_meta, make_aiohttp_session
from .util import PR_UNPAID, PR_EXPIRED, PR_PAID, PR_UNKNOWN, PR_INFLIGHT
from .crypto import sha256 from .crypto import sha256
from .bitcoin import TYPE_ADDRESS from .bitcoin import TYPE_ADDRESS
from .transaction import TxOutput from .transaction import TxOutput
@ -65,12 +66,6 @@ def load_ca_list():
# status of payment requests
PR_UNPAID = 0
PR_EXPIRED = 1
PR_UNKNOWN = 2 # sent but not propagated
PR_PAID = 3 # send and propagated
async def get_payment_request(url: str) -> 'PaymentRequest': async def get_payment_request(url: str) -> 'PaymentRequest':
u = urllib.parse.urlparse(url) u = urllib.parse.urlparse(url)

7
electrum/util.py

@ -73,6 +73,13 @@ base_units_list = ['BTC', 'mBTC', 'bits', 'sat'] # list(dict) does not guarante
DECIMAL_POINT_DEFAULT = 5 # mBTC DECIMAL_POINT_DEFAULT = 5 # mBTC
# status of payment requests
PR_UNPAID = 0
PR_EXPIRED = 1
PR_UNKNOWN = 2 # sent but not propagated
PR_PAID = 3 # send and propagated
PR_INFLIGHT = 4 # lightning
class UnknownBaseUnit(Exception): pass class UnknownBaseUnit(Exception): pass

Loading…
Cancel
Save