161 lines
5.7 KiB
Python
161 lines
5.7 KiB
Python
from datetime import datetime, timedelta
|
|
|
|
from PyQt5.QtCore import pyqtProperty, pyqtSignal, pyqtSlot, QObject, Q_ENUMS
|
|
from PyQt5.QtCore import Qt, QAbstractListModel, QModelIndex
|
|
|
|
from electrum.logging import get_logger
|
|
from electrum.util import Satoshis
|
|
from electrum.lnutil import LOCAL, REMOTE
|
|
from electrum.lnchannel import ChannelState
|
|
|
|
from .qetypes import QEAmount
|
|
from .util import QtEventListener, qt_event_listener
|
|
|
|
class QEChannelListModel(QAbstractListModel, QtEventListener):
|
|
_logger = get_logger(__name__)
|
|
|
|
# define listmodel rolemap
|
|
_ROLE_NAMES=('cid','state','state_code','initiator','capacity','can_send',
|
|
'can_receive','l_csv_delay','r_csv_delay','send_frozen','receive_frozen',
|
|
'type','node_id','node_alias','short_cid','funding_tx','is_trampoline')
|
|
_ROLE_KEYS = range(Qt.UserRole, Qt.UserRole + len(_ROLE_NAMES))
|
|
_ROLE_MAP = dict(zip(_ROLE_KEYS, [bytearray(x.encode()) for x in _ROLE_NAMES]))
|
|
_ROLE_RMAP = dict(zip(_ROLE_NAMES, _ROLE_KEYS))
|
|
|
|
_network_signal = pyqtSignal(str, object)
|
|
|
|
def __init__(self, wallet, parent=None):
|
|
super().__init__(parent)
|
|
self.wallet = wallet
|
|
self.init_model()
|
|
|
|
# To avoid leaking references to "self" that prevent the
|
|
# window from being GC-ed when closed, callbacks should be
|
|
# methods of this class only, and specifically not be
|
|
# partials, lambdas or methods of subobjects. Hence...
|
|
self.register_callbacks()
|
|
self.destroyed.connect(lambda: self.on_destroy())
|
|
|
|
@qt_event_listener
|
|
def on_event_channel(self, wallet, channel):
|
|
if wallet == self.wallet:
|
|
self.on_channel_updated(channel)
|
|
|
|
# elif event == 'channels_updated':
|
|
@qt_event_listener
|
|
def on_event_channels_updated(self, wallet):
|
|
if wallet == self.wallet:
|
|
self.init_model() # TODO: remove/add less crude than full re-init
|
|
|
|
def on_destroy(self):
|
|
self.unregister_callbacks()
|
|
|
|
def rowCount(self, index):
|
|
return len(self.channels)
|
|
|
|
def roleNames(self):
|
|
return self._ROLE_MAP
|
|
|
|
def data(self, index, role):
|
|
tx = self.channels[index.row()]
|
|
role_index = role - Qt.UserRole
|
|
value = tx[self._ROLE_NAMES[role_index]]
|
|
if isinstance(value, (bool, list, int, str, QEAmount)) or value is None:
|
|
return value
|
|
if isinstance(value, Satoshis):
|
|
return value.value
|
|
return str(value)
|
|
|
|
def clear(self):
|
|
self.beginResetModel()
|
|
self.channels = []
|
|
self.endResetModel()
|
|
|
|
def channel_to_model(self, lnc):
|
|
lnworker = self.wallet.lnworker
|
|
item = {}
|
|
item['cid'] = lnc.channel_id.hex()
|
|
item['node_alias'] = lnworker.get_node_alias(lnc.node_id) or lnc.node_id.hex()
|
|
item['short_cid'] = lnc.short_id_for_GUI()
|
|
item['state'] = lnc.get_state_for_GUI()
|
|
item['state_code'] = int(lnc.get_state())
|
|
item['capacity'] = QEAmount(amount_sat=lnc.get_capacity())
|
|
item['can_send'] = QEAmount(amount_msat=lnc.available_to_spend(LOCAL))
|
|
item['can_receive'] = QEAmount(amount_msat=lnc.available_to_spend(REMOTE))
|
|
item['is_trampoline'] = lnworker.is_trampoline_peer(lnc.node_id)
|
|
return item
|
|
|
|
numOpenChannelsChanged = pyqtSignal()
|
|
@pyqtProperty(int, notify=numOpenChannelsChanged)
|
|
def numOpenChannels(self):
|
|
return sum([1 if x['state_code'] == ChannelState.OPEN else 0 for x in self.channels])
|
|
|
|
@pyqtSlot()
|
|
def init_model(self):
|
|
self._logger.debug('init_model')
|
|
if not self.wallet.lnworker:
|
|
self._logger.warning('lnworker should be defined')
|
|
return
|
|
|
|
channels = []
|
|
|
|
lnchannels = self.wallet.lnworker.get_channel_objects()
|
|
for channel in lnchannels.values():
|
|
if channel.is_backup():
|
|
# not implemented
|
|
continue
|
|
item = self.channel_to_model(channel)
|
|
channels.append(item)
|
|
|
|
# sort, for now simply by state
|
|
def chan_sort_score(c):
|
|
return c['state_code']
|
|
channels.sort(key=chan_sort_score)
|
|
|
|
self.clear()
|
|
self.beginInsertRows(QModelIndex(), 0, len(channels) - 1)
|
|
self.channels = channels
|
|
self.endInsertRows()
|
|
|
|
def on_channel_updated(self, channel):
|
|
i = 0
|
|
for c in self.channels:
|
|
if c['cid'] == channel.channel_id.hex():
|
|
self.do_update(i,channel)
|
|
break
|
|
i = i + 1
|
|
|
|
def do_update(self, modelindex, channel):
|
|
modelitem = self.channels[modelindex]
|
|
modelitem.update(self.channel_to_model(channel))
|
|
|
|
mi = self.createIndex(modelindex, 0)
|
|
self.dataChanged.emit(mi, mi, self._ROLE_KEYS)
|
|
self.numOpenChannelsChanged.emit()
|
|
|
|
@pyqtSlot(str)
|
|
def new_channel(self, cid):
|
|
self._logger.debug('new channel with cid %s' % cid)
|
|
lnchannels = self.wallet.lnworker.channels
|
|
for channel in lnchannels.values():
|
|
self._logger.debug(repr(channel))
|
|
if cid == channel.channel_id.hex():
|
|
item = self.channel_to_model(channel)
|
|
self._logger.debug(item)
|
|
self.beginInsertRows(QModelIndex(), 0, 0)
|
|
self.channels.insert(0,item)
|
|
self.endInsertRows()
|
|
|
|
@pyqtSlot(str)
|
|
def remove_channel(self, cid):
|
|
self._logger.debug('remove channel with cid %s' % cid)
|
|
i = 0
|
|
for channel in self.channels:
|
|
if cid == channel['cid']:
|
|
self._logger.debug(cid)
|
|
self.beginRemoveRows(QModelIndex(), i, i)
|
|
self.channels.remove(channel)
|
|
self.endRemoveRows()
|
|
return
|
|
i = i + 1
|