You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
277 lines
13 KiB
277 lines
13 KiB
from typing import TYPE_CHECKING, Sequence |
|
|
|
import PyQt5.QtGui as QtGui |
|
import PyQt5.QtWidgets as QtWidgets |
|
import PyQt5.QtCore as QtCore |
|
from PyQt5.QtWidgets import QLabel, QLineEdit, QHBoxLayout, QGridLayout |
|
|
|
from electrum.util import EventListener, ShortID |
|
from electrum.i18n import _ |
|
from electrum.util import format_time |
|
from electrum.lnutil import format_short_channel_id, LOCAL, REMOTE, UpdateAddHtlc, Direction |
|
from electrum.lnchannel import htlcsum, Channel, AbstractChannel, HTLCWithStatus |
|
from electrum.lnaddr import LnAddr, lndecode |
|
from electrum.bitcoin import COIN |
|
from electrum.wallet import Abstract_Wallet |
|
|
|
from .util import Buttons, CloseButton, ShowQRLineEdit, MessageBoxMixin, WWLabel |
|
from .util import QtEventListener, qt_event_listener, VLine |
|
|
|
if TYPE_CHECKING: |
|
from .main_window import ElectrumWindow |
|
|
|
class HTLCItem(QtGui.QStandardItem): |
|
def __init__(self, *args, **kwargs): |
|
super().__init__(*args, **kwargs) |
|
self.setEditable(False) |
|
|
|
class SelectableLabel(QtWidgets.QLabel): |
|
def __init__(self, text=''): |
|
super().__init__(text) |
|
self.setTextInteractionFlags(QtCore.Qt.TextSelectableByMouse) |
|
|
|
class LinkedLabel(QtWidgets.QLabel): |
|
def __init__(self, text, on_clicked): |
|
super().__init__(text) |
|
self.linkActivated.connect(on_clicked) |
|
|
|
|
|
class ChannelDetailsDialog(QtWidgets.QDialog, MessageBoxMixin, QtEventListener): |
|
|
|
def __init__(self, window: 'ElectrumWindow', chan: AbstractChannel): |
|
super().__init__(window) |
|
# initialize instance fields |
|
self.window = window |
|
self.wallet = window.wallet |
|
self.chan = chan |
|
self.format_msat = lambda msat: window.format_amount_and_units(msat / 1000) |
|
self.format_sat = lambda sat: window.format_amount_and_units(sat) |
|
# register callbacks for updating |
|
self.register_callbacks() |
|
title = _('Lightning Channel') if not self.chan.is_backup() else _('Channel Backup') |
|
self.setWindowTitle(title) |
|
self.setMinimumSize(800, 400) |
|
# activity labels. not used for backups. |
|
self.local_balance_label = SelectableLabel() |
|
self.remote_balance_label = SelectableLabel() |
|
self.can_send_label = SelectableLabel() |
|
self.can_receive_label = SelectableLabel() |
|
# add widgets |
|
vbox = QtWidgets.QVBoxLayout(self) |
|
if self.chan.is_backup(): |
|
vbox.addWidget(QLabel('\n'.join([ |
|
_("This is a channel backup."), |
|
_("It shows a channel that was opened with another instance of this wallet"), |
|
_("A backup does not contain information about your local balance in the channel."), |
|
_("You can use it to request a force close.") |
|
]))) |
|
|
|
form = self.get_common_form(chan) |
|
vbox.addLayout(form) |
|
if not self.chan.is_closed() and not self.chan.is_backup(): |
|
hbox_stats = self.get_hbox_stats(chan) |
|
form.addRow(QLabel(_('Channel stats')+ ':'), hbox_stats) |
|
|
|
if not self.chan.is_backup(): |
|
# add htlc tree view to vbox (wouldn't scale correctly in QFormLayout) |
|
vbox.addWidget(QLabel(_('Payments (HTLCs):'))) |
|
w = self.create_htlc_list(chan) |
|
vbox.addWidget(w) |
|
|
|
vbox.addLayout(Buttons(CloseButton(self))) |
|
# initialize sent/received fields |
|
self.update() |
|
|
|
def make_htlc_item(self, i: UpdateAddHtlc, direction: Direction) -> HTLCItem: |
|
it = HTLCItem(_('Sent HTLC with ID {}' if Direction.SENT == direction else 'Received HTLC with ID {}').format(i.htlc_id)) |
|
it.appendRow([HTLCItem(_('Amount')),HTLCItem(self.format_msat(i.amount_msat))]) |
|
it.appendRow([HTLCItem(_('CLTV expiry')), HTLCItem(str(i.cltv_abs))]) |
|
it.appendRow([HTLCItem(_('Payment hash')),HTLCItem(i.payment_hash.hex())]) |
|
return it |
|
|
|
def make_model(self, htlcs: Sequence[HTLCWithStatus]) -> QtGui.QStandardItemModel: |
|
model = QtGui.QStandardItemModel(0, 2) |
|
model.setHorizontalHeaderLabels(['HTLC', 'Property value']) |
|
parentItem = model.invisibleRootItem() |
|
folder_types = { |
|
'settled': _('Fulfilled HTLCs'), |
|
'inflight': _('HTLCs in current commitment transaction'), |
|
'failed': _('Failed HTLCs'), |
|
} |
|
self.folders = {} |
|
self.keyname_rows = {} |
|
|
|
for keyname, i in folder_types.items(): |
|
myFont=QtGui.QFont() |
|
myFont.setBold(True) |
|
folder = HTLCItem(i) |
|
folder.setFont(myFont) |
|
parentItem.appendRow(folder) |
|
self.folders[keyname] = folder |
|
mapping = {} |
|
num = 0 |
|
for htlc_with_status in htlcs: |
|
if htlc_with_status.status != keyname: |
|
continue |
|
htlc = htlc_with_status.htlc |
|
it = self.make_htlc_item(htlc, htlc_with_status.direction) |
|
self.folders[keyname].appendRow(it) |
|
mapping[htlc.payment_hash] = num |
|
num += 1 |
|
self.keyname_rows[keyname] = mapping |
|
return model |
|
|
|
def move(self, fro: str, to: str, payment_hash: bytes): |
|
assert fro != to |
|
row_idx = self.keyname_rows[fro].pop(payment_hash) |
|
row = self.folders[fro].takeRow(row_idx) |
|
self.folders[to].appendRow(row) |
|
dest_mapping = self.keyname_rows[to] |
|
dest_mapping[payment_hash] = len(dest_mapping) |
|
|
|
@qt_event_listener |
|
def on_event_channel(self, wallet, chan): |
|
if chan == self.chan: |
|
self.update() |
|
|
|
@qt_event_listener |
|
def on_event_htlc_added(self, chan, htlc, direction): |
|
if chan != self.chan: |
|
return |
|
mapping = self.keyname_rows['inflight'] |
|
mapping[htlc.payment_hash] = len(mapping) |
|
self.folders['inflight'].appendRow(self.make_htlc_item(htlc, direction)) |
|
|
|
@qt_event_listener |
|
def on_event_htlc_fulfilled(self, payment_hash, chan, htlc_id): |
|
if chan.channel_id != self.chan.channel_id: |
|
return |
|
self.move('inflight', 'settled', payment_hash) |
|
self.update() |
|
|
|
@qt_event_listener |
|
def on_event_htlc_failed(self, payment_hash, chan, htlc_id): |
|
if chan.channel_id != self.chan.channel_id: |
|
return |
|
self.move('inflight', 'failed', payment_hash) |
|
self.update() |
|
|
|
def update(self): |
|
if self.chan.is_closed() or self.chan.is_backup(): |
|
return |
|
assert isinstance(self.chan, Channel), type(self.chan) |
|
self.can_send_label.setText(self.format_msat(self.chan.available_to_spend(LOCAL))) |
|
self.can_receive_label.setText(self.format_msat(self.chan.available_to_spend(REMOTE))) |
|
self.sent_label.setText(self.format_msat(self.chan.total_msat(Direction.SENT))) |
|
self.received_label.setText(self.format_msat(self.chan.total_msat(Direction.RECEIVED))) |
|
self.local_balance_label.setText(self.format_msat(self.chan.balance(LOCAL))) |
|
self.remote_balance_label.setText(self.format_msat(self.chan.balance(REMOTE))) |
|
self.current_feerate.setText(self.window.format_fee_rate(4 * self.chan.get_latest_feerate(LOCAL))) |
|
|
|
@QtCore.pyqtSlot(str) |
|
def show_tx(self, link_text: str): |
|
tx = self.wallet.adb.get_transaction(link_text) |
|
if not tx: |
|
self.show_error(_("Transaction not found.")) |
|
return |
|
self.window.show_transaction(tx) |
|
|
|
def get_common_form(self, chan): |
|
form = QtWidgets.QFormLayout(None) |
|
remote_id_e = ShowQRLineEdit(chan.node_id.hex(), self.window.config, title=_("Remote Node ID")) |
|
form.addRow(QLabel(_('Remote Node') + ':'), remote_id_e) |
|
channel_id_e = ShowQRLineEdit(chan.channel_id.hex(), self.window.config, title=_("Channel ID")) |
|
form.addRow(QLabel(_('Channel ID') + ':'), channel_id_e) |
|
form.addRow(QLabel(_('Short Channel ID') + ':'), SelectableLabel(str(chan.short_channel_id))) |
|
if local_scid_alias := chan.get_local_scid_alias(): |
|
form.addRow(QLabel('Local SCID Alias:'), SelectableLabel(str(ShortID(local_scid_alias)))) |
|
if remote_scid_alias := chan.get_remote_scid_alias(): |
|
form.addRow(QLabel('Remote SCID Alias:'), SelectableLabel(str(ShortID(remote_scid_alias)))) |
|
form.addRow(QLabel(_('State') + ':'), SelectableLabel(chan.get_state_for_GUI())) |
|
self.capacity = self.format_sat(chan.get_capacity()) |
|
form.addRow(QLabel(_('Capacity') + ':'), SelectableLabel(self.capacity)) |
|
if not chan.is_backup(): |
|
form.addRow(QLabel(_('Channel type:')), SelectableLabel(chan.storage['channel_type'].name_minimal)) |
|
initiator = 'Local' if chan.constraints.is_initiator else 'Remote' |
|
form.addRow(QLabel(_('Initiator:')), SelectableLabel(initiator)) |
|
else: |
|
form.addRow(QLabel("Backup Type"), QLabel("imported" if self.chan.is_imported else "on-chain")) |
|
funding_txid = chan.funding_outpoint.txid |
|
funding_label_text = f'<a href={funding_txid}>{funding_txid}</a>:{chan.funding_outpoint.output_index}' |
|
form.addRow(QLabel(_('Funding Outpoint') + ':'), LinkedLabel(funding_label_text, self.show_tx)) |
|
if chan.is_closed(): |
|
item = chan.get_closing_height() |
|
if item: |
|
closing_txid, closing_height, timestamp = item |
|
closing_label_text = f'<a href={closing_txid}>{closing_txid}</a>' |
|
form.addRow(QLabel(_('Closing Transaction') + ':'), LinkedLabel(closing_label_text, self.show_tx)) |
|
return form |
|
|
|
def get_hbox_stats(self, chan: Channel): |
|
hbox_stats = QHBoxLayout() |
|
form_layout_left = QtWidgets.QFormLayout(None) |
|
form_layout_right = QtWidgets.QFormLayout(None) |
|
form_layout_left.addRow(_('Local balance') + ':', self.local_balance_label) |
|
form_layout_right.addRow(_('Remote balance') + ':', self.remote_balance_label) |
|
form_layout_left.addRow(_('Can send') + ':', self.can_send_label) |
|
form_layout_right.addRow(_('Can receive') + ':', self.can_receive_label) |
|
local_reserve_label = SelectableLabel("{}".format( |
|
self.format_sat(chan.config[LOCAL].reserve_sat), |
|
)) |
|
remote_reserve_label = SelectableLabel("{}".format( |
|
self.format_sat(chan.config[REMOTE].reserve_sat), |
|
)) |
|
form_layout_left.addRow(_('Local reserve') + ':', local_reserve_label) |
|
form_layout_right.addRow(_('Remote reserve' + ':'), remote_reserve_label) |
|
#self.htlc_minimum_msat = SelectableLabel(str(chan.config[REMOTE].htlc_minimum_msat)) |
|
#form_layout_left.addRow(_('Minimum HTLC value accepted by peer (mSAT):'), self.htlc_minimum_msat) |
|
#self.max_htlcs = SelectableLabel(str(chan.config[REMOTE].max_accepted_htlcs)) |
|
#form_layout_left.addRow(_('Maximum number of concurrent HTLCs accepted by peer:'), self.max_htlcs) |
|
#self.max_htlc_value = SelectableLabel(self.format_sat(chan.config[REMOTE].max_htlc_value_in_flight_msat / 1000)) |
|
#form_layout_left.addRow(_('Maximum value of in-flight HTLCs accepted by peer:'), self.max_htlc_value) |
|
local_dust_limit_label = SelectableLabel("{}".format( |
|
self.format_sat(chan.config[LOCAL].dust_limit_sat), |
|
)) |
|
remote_dust_limit_label = SelectableLabel("{}".format( |
|
self.format_sat(chan.config[REMOTE].dust_limit_sat), |
|
)) |
|
form_layout_left.addRow(_('Local dust limit') + ':', local_dust_limit_label) |
|
form_layout_right.addRow(_('Remote dust limit') + ':', remote_dust_limit_label) |
|
self.received_label = SelectableLabel() |
|
self.sent_label = SelectableLabel() |
|
form_layout_left.addRow(_('Total sent') + ':', self.sent_label) |
|
form_layout_right.addRow(_('Total received') + ':', self.received_label) |
|
# to-self-delay |
|
csv_local_header = SelectableLabel(_("Remote force-close CSV delay") + ":") |
|
csv_local_header.setToolTip(_("Force-close CSV delay imposed on them")) |
|
csv_remote_header = SelectableLabel(_("Local force-close CSV delay") + ":") |
|
csv_remote_header.setToolTip(_("Force-close CSV delay imposed on us")) |
|
csv_local_label = SelectableLabel(_("{} blocks").format(chan.config[LOCAL].to_self_delay)) |
|
csv_remote_label = SelectableLabel(_("{} blocks").format(chan.config[REMOTE].to_self_delay)) |
|
form_layout_left.addRow(csv_local_header, csv_local_label) |
|
form_layout_right.addRow(csv_remote_header, csv_remote_label) |
|
# onchain feerate |
|
self.current_feerate = SelectableLabel() |
|
form_layout_left.addRow(_('Current feerate') + ':', self.current_feerate) |
|
# channel stats left column |
|
hbox_stats.addLayout(form_layout_left, 50) |
|
# vertical line separator |
|
hbox_stats.addWidget(VLine()) |
|
# channel stats right column |
|
hbox_stats.addLayout(form_layout_right, 50) |
|
return hbox_stats |
|
|
|
def create_htlc_list(self, chan): |
|
w = QtWidgets.QTreeView(self) |
|
htlc_dict = chan.get_payments() |
|
htlc_list = [] |
|
for rhash, plist in htlc_dict.items(): |
|
for htlc_with_status in plist: |
|
htlc_list.append(htlc_with_status) |
|
w.setModel(self.make_model(htlc_list)) |
|
w.header().setSectionResizeMode(0, QtWidgets.QHeaderView.ResizeToContents) |
|
return w |
|
|
|
def closeEvent(self, event): |
|
self.unregister_callbacks() |
|
event.accept()
|
|
|