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.

210 lines
7.0 KiB

# -*- coding: utf-8 -*-
import binascii, base64
from PyQt5 import QtCore, QtWidgets
from collections import OrderedDict
import logging
from electrum.lightning import lightningCall
import traceback
mapping = {0: "channel_point"}
revMapp = {"channel_point": 0}
datatable = OrderedDict([])
class MyTableRow(QtWidgets.QTreeWidgetItem):
def __init__(self, di):
strs = [str(di[mapping[key]]) for key in range(len(mapping))]
super(MyTableRow, self).__init__(strs)
assert isinstance(di, dict)
self.di = di
def __getitem__(self, idx):
return self.di[idx]
def __setitem__(self, idx, val):
self.di[idx] = val
try:
self.setData(revMapp[idx], QtCore.Qt.DisplayRole, '{0}'.format(val))
except KeyError:
logging.warning("Lightning Channel field %s unknown", idx)
def __str__(self):
return str(self.di)
def addChannelRow(new):
made = MyTableRow(new)
datatable[new["channel_point"]] = made
datatable.move_to_end(new["channel_point"], last=False)
return made
def clickHandler(nodeIdInput, local_amt_inp, push_amt_inp, lightningRpc):
nodeId = nodeIdInput.text()
print("creating channel with connstr {}".format(nodeId))
lightningCall(lightningRpc, "openchannel")(str(nodeId), local_amt_inp.text(), push_amt_inp.text())
class LightningChannelsList(QtWidgets.QWidget):
update_rows = QtCore.pyqtSignal(str, dict)
def create_menu(self, position):
menu = QtWidgets.QMenu()
cur = self._tv.currentItem()
channel_point = cur["channel_point"]
def close():
params = [str(channel_point)] + (["--force"] if not cur["active"] else []) # TODO test if force is being used correctly
lightningCall(self.lightningRpc, "closechannel")(*params)
menu.addAction("Close channel", close)
menu.exec_(self._tv.viewport().mapToGlobal(position))
def lightningWorkerHandler(self, sourceClassName, obj):
new = {}
for k, v in obj.items():
try:
v = binascii.hexlify(base64.b64decode(v)).decode("ascii")
except:
pass
new[k] = v
try:
obj = datatable[new["channel_point"]]
except KeyError:
print("lightning channel_point {} unknown!".format(new["channel_point"]))
else:
for k, v in new.items():
try:
if obj[k] != v: obj[k] = v
except KeyError:
obj[k] = v
def lightningRpcHandler(self, methodName, obj):
if isinstance(obj, Exception):
try:
raise obj
except:
traceback.print_exc()
else:
self.update_rows.emit(methodName, obj)
def do_update_rows(self, methodName, obj):
if methodName != "listchannels":
print("channel list ignoring reply {} to {}".format(obj, methodName))
return
self._tv.clear()
for i in obj["channels"]:
self._tv.insertTopLevelItem(0, addChannelRow(i))
def __init__(self, parent, lightningWorker, lightningRpc):
QtWidgets.QWidget.__init__(self, parent)
self.update_rows.connect(self.do_update_rows)
def tick():
lightningCall(lightningRpc, "listchannels")()
timer = QtCore.QTimer(self)
timer.timeout.connect(tick)
timer.start(5000)
lightningWorker.subscribe(self.lightningWorkerHandler)
lightningRpc.subscribe(self.lightningRpcHandler)
self.lightningRpc = lightningRpc
self._tv=QtWidgets.QTreeWidget(self)
self._tv.setHeaderLabels([mapping[i] for i in range(len(mapping))])
self._tv.setColumnCount(len(mapping))
self._tv.setContextMenuPolicy(QtCore.Qt.CustomContextMenu)
self._tv.customContextMenuRequested.connect(self.create_menu)
nodeid_inp = QtWidgets.QLineEdit(self)
local_amt_inp = QtWidgets.QLineEdit(self)
push_amt_inp = QtWidgets.QLineEdit(self)
button = QtWidgets.QPushButton('Open channel', self)
button.clicked.connect(lambda: clickHandler(nodeid_inp, local_amt_inp, push_amt_inp, lightningRpc))
l=QtWidgets.QVBoxLayout(self)
h=QtWidgets.QGridLayout(self)
nodeid_label = QtWidgets.QLabel(self)
nodeid_label.setText("Node ID")
local_amt_label = QtWidgets.QLabel(self)
local_amt_label.setText("Local amount (sat)")
push_amt_label = QtWidgets.QLabel(self)
push_amt_label.setText("Push amount (sat)")
h.addWidget(nodeid_label, 0, 0)
h.addWidget(local_amt_label, 0, 1)
h.addWidget(push_amt_label, 0, 2)
h.addWidget(nodeid_inp, 1, 0)
h.addWidget(local_amt_inp, 1, 1)
h.addWidget(push_amt_inp, 1, 2)
h.addWidget(button, 1, 3)
h.setColumnStretch(0, 3)
h.setColumnStretch(1, 1)
h.setColumnStretch(2, 1)
h.setColumnStretch(3, 1)
l.addLayout(h)
l.addWidget(self._tv)
self.resize(2500,1000)
class MockLightningWorker:
def subscribe(self, handler):
pass
if __name__=="__main__":
import queue, threading, asyncio
from sys import argv, exit
import signal , traceback, os
loop = asyncio.new_event_loop()
async def loopstop():
loop.stop()
def signal_handler(signal, frame):
asyncio.run_coroutine_threadsafe(loopstop(), loop)
signal.signal(signal.SIGINT, signal_handler)
a=QtWidgets.QApplication(argv)
gotReplyHandlerLock = threading.Lock()
gotReplyHandlerLock.acquire()
replyHandler = None
class MockLightningRPC:
def __init__(self, q):
self.queue = q
def subscribe(self, handler):
global replyHandler
replyHandler = handler
gotReplyHandlerLock.release()
q = queue.Queue()
w=LightningChannelsList(None, MockLightningWorker(), MockLightningRPC(q))
w.show()
w.raise_()
async def the_job():
try:
acquired_once = False
while loop.is_running():
try:
cmd = q.get_nowait()
except queue.Empty:
await asyncio.sleep(1)
continue
if not acquired_once:
gotReplyHandlerLock.acquire()
acquired_once = True
if cmd[0] == "listchannels":
#replyHandler("listchannels", Exception("Test exception"))
replyHandler("listchannels", {"channels": [{"channel_point": binascii.hexlify(os.urandom(32)).decode("ascii"), "active": True}]})
elif cmd[0] == "openchannel":
replyHandler("openchannel", {})
else:
print("mock rpc server ignoring", cmd[0])
except:
traceback.print_exc()
def asyncioThread():
loop.create_task(the_job())
loop.run_forever()
threading.Thread(target=asyncioThread).start()
exit(a.exec_())