mirror of
https://github.com/LBRYFoundation/LBRY-Vault.git
synced 2025-08-23 17:47:31 +00:00
asynchronous processing: use a queue, handle responses in wallet class
This commit is contained in:
parent
39895f41cc
commit
f60f6c28d3
4 changed files with 116 additions and 111 deletions
|
@ -22,6 +22,7 @@ from optparse import OptionParser
|
||||||
from wallet import Wallet, SecretToASecret
|
from wallet import Wallet, SecretToASecret
|
||||||
from decimal import Decimal
|
from decimal import Decimal
|
||||||
|
|
||||||
|
import thread
|
||||||
|
|
||||||
from wallet import format_satoshis
|
from wallet import format_satoshis
|
||||||
from interface import loop_interfaces_thread, new_interface
|
from interface import loop_interfaces_thread, new_interface
|
||||||
|
@ -62,8 +63,6 @@ if __name__ == '__main__':
|
||||||
firstarg = args[1] if len(args) > 1 else ''
|
firstarg = args[1] if len(args) > 1 else ''
|
||||||
|
|
||||||
if cmd == 'gui':
|
if cmd == 'gui':
|
||||||
import thread
|
|
||||||
|
|
||||||
if options.gui=='gtk':
|
if options.gui=='gtk':
|
||||||
import gui
|
import gui
|
||||||
elif options.gui=='qt':
|
elif options.gui=='qt':
|
||||||
|
@ -169,7 +168,8 @@ if __name__ == '__main__':
|
||||||
addresses = wallet.all_addresses()
|
addresses = wallet.all_addresses()
|
||||||
version = wallet.electrum_version
|
version = wallet.electrum_version
|
||||||
interface.start_session(addresses, version)
|
interface.start_session(addresses, version)
|
||||||
interface.update_wallet()
|
thread.start_new_thread(wallet.run, ())
|
||||||
|
wallet.update()
|
||||||
wallet.save()
|
wallet.save()
|
||||||
|
|
||||||
# check if --from_addr not in wallet (for mktx/payto)
|
# check if --from_addr not in wallet (for mktx/payto)
|
||||||
|
|
|
@ -187,10 +187,10 @@ class ElectrumWindow(QMainWindow):
|
||||||
|
|
||||||
def update_wallet(self):
|
def update_wallet(self):
|
||||||
if self.wallet.interface.is_connected:
|
if self.wallet.interface.is_connected:
|
||||||
if self.wallet.interface.blocks == 0:
|
if self.wallet.blocks == 0:
|
||||||
text = "Server not ready"
|
text = "Server not ready"
|
||||||
icon = QIcon(":icons/status_disconnected.png")
|
icon = QIcon(":icons/status_disconnected.png")
|
||||||
elif not self.wallet.interface.is_up_to_date:
|
elif not self.wallet.up_to_date:
|
||||||
text = "Synchronizing..."
|
text = "Synchronizing..."
|
||||||
icon = QIcon(":icons/status_waiting.png")
|
icon = QIcon(":icons/status_waiting.png")
|
||||||
else:
|
else:
|
||||||
|
@ -208,9 +208,9 @@ class ElectrumWindow(QMainWindow):
|
||||||
self.statusBar().showMessage(text)
|
self.statusBar().showMessage(text)
|
||||||
self.status_button.setIcon( icon )
|
self.status_button.setIcon( icon )
|
||||||
|
|
||||||
if self.wallet.interface.was_updated and self.wallet.interface.is_up_to_date:
|
if self.wallet.was_updated and self.wallet.up_to_date:
|
||||||
self.wallet.interface.was_updated = False
|
self.wallet.was_updated = False
|
||||||
self.textbox.setText( self.wallet.interface.message )
|
self.textbox.setText( self.wallet.banner )
|
||||||
self.update_history_tab()
|
self.update_history_tab()
|
||||||
self.update_receive_tab()
|
self.update_receive_tab()
|
||||||
self.update_contacts_tab()
|
self.update_contacts_tab()
|
||||||
|
@ -236,7 +236,7 @@ class ElectrumWindow(QMainWindow):
|
||||||
tx = self.wallet.tx_history.get(tx_hash)
|
tx = self.wallet.tx_history.get(tx_hash)
|
||||||
|
|
||||||
if tx['height']:
|
if tx['height']:
|
||||||
conf = self.wallet.interface.blocks - tx['height'] + 1
|
conf = self.wallet.blocks - tx['height'] + 1
|
||||||
time_str = datetime.datetime.fromtimestamp( tx['nTime']).isoformat(' ')[:-3]
|
time_str = datetime.datetime.fromtimestamp( tx['nTime']).isoformat(' ')[:-3]
|
||||||
else:
|
else:
|
||||||
conf = 0
|
conf = 0
|
||||||
|
@ -309,7 +309,7 @@ class ElectrumWindow(QMainWindow):
|
||||||
for tx in self.wallet.get_tx_history():
|
for tx in self.wallet.get_tx_history():
|
||||||
tx_hash = tx['tx_hash']
|
tx_hash = tx['tx_hash']
|
||||||
if tx['height']:
|
if tx['height']:
|
||||||
conf = self.wallet.interface.blocks - tx['height'] + 1
|
conf = self.wallet.blocks - tx['height'] + 1
|
||||||
time_str = datetime.datetime.fromtimestamp( tx['nTime']).isoformat(' ')[:-3]
|
time_str = datetime.datetime.fromtimestamp( tx['nTime']).isoformat(' ')[:-3]
|
||||||
icon = QIcon(":icons/confirmed.png")
|
icon = QIcon(":icons/confirmed.png")
|
||||||
else:
|
else:
|
||||||
|
@ -832,7 +832,7 @@ class ElectrumWindow(QMainWindow):
|
||||||
interface = wallet.interface
|
interface = wallet.interface
|
||||||
if parent:
|
if parent:
|
||||||
if interface.is_connected:
|
if interface.is_connected:
|
||||||
status = "Connected to %s:%d\n%d blocks\nresponse time: %f"%(interface.host, interface.port, interface.blocks, interface.rtime)
|
status = "Connected to %s:%d\n%d blocks\nresponse time: %f"%(interface.host, interface.port, wallet.blocks, interface.rtime)
|
||||||
else:
|
else:
|
||||||
status = "Not connected"
|
status = "Not connected"
|
||||||
host = wallet.host
|
host = wallet.host
|
||||||
|
@ -926,7 +926,7 @@ class ElectrumGui():
|
||||||
if not is_recovery:
|
if not is_recovery:
|
||||||
wallet.new_seed(None)
|
wallet.new_seed(None)
|
||||||
# generate first key
|
# generate first key
|
||||||
wallet.synchronize()
|
#wallet.synchronize()
|
||||||
# run a dialog indicating the seed, ask the user to remember it
|
# run a dialog indicating the seed, ask the user to remember it
|
||||||
ElectrumWindow.show_seed_dialog(wallet)
|
ElectrumWindow.show_seed_dialog(wallet)
|
||||||
#ask for password
|
#ask for password
|
||||||
|
@ -935,7 +935,7 @@ class ElectrumGui():
|
||||||
# ask for seed and gap.
|
# ask for seed and gap.
|
||||||
if not ElectrumWindow.seed_dialog( wallet ): return False
|
if not ElectrumWindow.seed_dialog( wallet ): return False
|
||||||
wallet.init_mpk( wallet.seed ) # not encrypted at this point
|
wallet.init_mpk( wallet.seed ) # not encrypted at this point
|
||||||
wallet.synchronize()
|
#wallet.synchronize()
|
||||||
|
|
||||||
if wallet.is_found():
|
if wallet.is_found():
|
||||||
# history and addressbook
|
# history and addressbook
|
||||||
|
|
|
@ -25,19 +25,16 @@ DEFAULT_SERVERS = ['electrum.bitcoins.sk','ecdsa.org','electrum.novit.ro'] # li
|
||||||
|
|
||||||
|
|
||||||
class Interface:
|
class Interface:
|
||||||
def __init__(self, host, port, address_callback=None, history_callback=None, newblock_callback=None):
|
def __init__(self, host, port, address_callback=None, history_callback=None, newblock_callback=None, sync_cb=None):
|
||||||
self.host = host
|
self.host = host
|
||||||
self.port = port
|
self.port = port
|
||||||
|
self.sync_callback = sync_cb
|
||||||
self.address_callback = address_callback
|
self.address_callback = address_callback
|
||||||
self.history_callback = history_callback
|
self.history_callback = history_callback
|
||||||
self.newblock_callback = newblock_callback
|
self.newblock_callback = newblock_callback
|
||||||
|
|
||||||
self.servers = [] # actual list from IRC
|
self.servers = [] # actual list from IRC
|
||||||
self.rtime = 0
|
self.rtime = 0
|
||||||
self.blocks = 0
|
|
||||||
self.message = ''
|
|
||||||
self.was_updated = True # fixme: use a semaphore
|
|
||||||
self.is_up_to_date = False
|
|
||||||
|
|
||||||
self.is_connected = False
|
self.is_connected = False
|
||||||
|
|
||||||
|
@ -45,15 +42,17 @@ class Interface:
|
||||||
self.addresses_waiting_for_status = []
|
self.addresses_waiting_for_status = []
|
||||||
self.addresses_waiting_for_history = []
|
self.addresses_waiting_for_history = []
|
||||||
self.tx_event = threading.Event()
|
self.tx_event = threading.Event()
|
||||||
self.up_to_date_event = threading.Event()
|
|
||||||
self.up_to_date_event.clear()
|
|
||||||
|
|
||||||
#json
|
#json
|
||||||
self.message_id = 0
|
self.message_id = 0
|
||||||
self.messages = {}
|
self.messages = {}
|
||||||
|
|
||||||
self.responses = Queue.Queue()
|
self.responses = Queue.Queue()
|
||||||
|
|
||||||
|
|
||||||
|
def is_up_to_date(self):
|
||||||
|
return not ( self.addresses_waiting_for_status or self.addresses_waiting_for_history )
|
||||||
|
|
||||||
|
|
||||||
def send_tx(self, data):
|
def send_tx(self, data):
|
||||||
self.tx_event.clear()
|
self.tx_event.clear()
|
||||||
self.send([('transaction.broadcast', [data])])
|
self.send([('transaction.broadcast', [data])])
|
||||||
|
@ -61,9 +60,6 @@ class Interface:
|
||||||
return self.tx_result
|
return self.tx_result
|
||||||
|
|
||||||
|
|
||||||
def start_session(self, addresses, version):
|
|
||||||
pass
|
|
||||||
|
|
||||||
|
|
||||||
def queue_json_response(self, c):
|
def queue_json_response(self, c):
|
||||||
#print repr(c)
|
#print repr(c)
|
||||||
|
@ -80,76 +76,18 @@ class Interface:
|
||||||
print "received error:", c, method, params
|
print "received error:", c, method, params
|
||||||
else:
|
else:
|
||||||
#self.handle_response(method, params, result)
|
#self.handle_response(method, params, result)
|
||||||
|
if method == 'address.subscribe':
|
||||||
|
addr = params[-1]
|
||||||
|
if addr in self.addresses_waiting_for_status:
|
||||||
|
self.addresses_waiting_for_status.remove(addr)
|
||||||
|
elif method == 'address.get_history':
|
||||||
|
addr = params[0]
|
||||||
|
if addr in self.addresses_waiting_for_history:
|
||||||
|
self.addresses_waiting_for_history.remove(addr)
|
||||||
self.responses.put({'method':method, 'params':params, 'result':result})
|
self.responses.put({'method':method, 'params':params, 'result':result})
|
||||||
#self.is_up_to_date = True
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
def handle_response(self, r):
|
|
||||||
if r is None:
|
|
||||||
print "empty item"
|
|
||||||
return
|
|
||||||
|
|
||||||
method = r['method']
|
|
||||||
params = r['params']
|
|
||||||
result = r['result']
|
|
||||||
|
|
||||||
if method == 'server.banner':
|
|
||||||
self.message = result
|
|
||||||
self.was_updated = True
|
|
||||||
|
|
||||||
elif method == 'session.poll':
|
|
||||||
# native poll
|
|
||||||
blocks, changed_addresses = result
|
|
||||||
if blocks == -1: raise BaseException("session not found")
|
|
||||||
self.blocks = int(blocks)
|
|
||||||
if changed_addresses:
|
|
||||||
self.is_up_to_date = False
|
|
||||||
self.was_updated = True
|
|
||||||
for addr, status in changed_addresses.items():
|
|
||||||
apply(self.address_callback, (addr, status))
|
|
||||||
else:
|
|
||||||
self.is_up_to_date = True
|
|
||||||
|
|
||||||
elif method == 'server.peers':
|
|
||||||
#print "Received server list: ", result
|
|
||||||
self.servers = map( lambda x:x[1], result )
|
|
||||||
|
|
||||||
elif method == 'address.subscribe':
|
|
||||||
addr = params[-1]
|
|
||||||
if addr in self.addresses_waiting_for_status:
|
|
||||||
self.addresses_waiting_for_status.remove(addr)
|
|
||||||
apply(self.address_callback,(addr, result))
|
|
||||||
|
|
||||||
elif method == 'address.get_history':
|
|
||||||
addr = params[0]
|
|
||||||
if addr in self.addresses_waiting_for_history:
|
|
||||||
self.addresses_waiting_for_history.remove(addr)
|
|
||||||
apply(self.history_callback, (addr, result))
|
|
||||||
self.was_updated = True
|
|
||||||
|
|
||||||
elif method == 'transaction.broadcast':
|
|
||||||
self.tx_result = result
|
|
||||||
self.tx_event.set()
|
|
||||||
|
|
||||||
elif method == 'numblocks.subscribe':
|
|
||||||
self.blocks = result
|
|
||||||
if self.newblock_callback: apply(self.newblock_callback,(result,))
|
|
||||||
|
|
||||||
elif method == 'client.version':
|
|
||||||
pass
|
|
||||||
|
|
||||||
else:
|
|
||||||
print "unknown message:", method, params, result
|
|
||||||
|
|
||||||
if self.addresses_waiting_for_status or self.addresses_waiting_for_history:
|
|
||||||
self.is_up_to_date = False
|
|
||||||
else:
|
|
||||||
self.is_up_to_date = True
|
|
||||||
self.up_to_date_event.set()
|
|
||||||
|
|
||||||
|
|
||||||
def subscribe(self, addresses):
|
def subscribe(self, addresses):
|
||||||
messages = []
|
messages = []
|
||||||
for addr in addresses:
|
for addr in addresses:
|
||||||
|
@ -196,11 +134,6 @@ class PollingInterface(Interface):
|
||||||
def poll(self):
|
def poll(self):
|
||||||
self.send([('session.poll', [])])
|
self.send([('session.poll', [])])
|
||||||
|
|
||||||
def update_wallet(self):
|
|
||||||
while True:
|
|
||||||
self.poll()
|
|
||||||
if self.is_up_to_date: break
|
|
||||||
|
|
||||||
#if is_new or wallet.remote_url:
|
#if is_new or wallet.remote_url:
|
||||||
# self.was_updated = True
|
# self.was_updated = True
|
||||||
# is_new = wallet.synchronize()
|
# is_new = wallet.synchronize()
|
||||||
|
@ -213,7 +146,7 @@ class PollingInterface(Interface):
|
||||||
def poll_thread(self, poll_interval):
|
def poll_thread(self, poll_interval):
|
||||||
while self.is_connected:
|
while self.is_connected:
|
||||||
try:
|
try:
|
||||||
self.update_wallet()
|
self.poll()
|
||||||
time.sleep(poll_interval)
|
time.sleep(poll_interval)
|
||||||
except socket.gaierror:
|
except socket.gaierror:
|
||||||
break
|
break
|
||||||
|
@ -281,7 +214,6 @@ class NativeInterface(PollingInterface):
|
||||||
|
|
||||||
if cmd == 'new_session':
|
if cmd == 'new_session':
|
||||||
self.session_id, self.message = ast.literal_eval( out )
|
self.session_id, self.message = ast.literal_eval( out )
|
||||||
self.was_updated = True
|
|
||||||
else:
|
else:
|
||||||
self.responses.put({'method':method, 'params':params, 'result':out})
|
self.responses.put({'method':method, 'params':params, 'result':out})
|
||||||
|
|
||||||
|
@ -379,9 +311,6 @@ class AsynchronousInterface(Interface):
|
||||||
self.is_connected = False
|
self.is_connected = False
|
||||||
self.responses.put(None)
|
self.responses.put(None)
|
||||||
|
|
||||||
def update_wallet(self):
|
|
||||||
self.up_to_date_event.wait()
|
|
||||||
|
|
||||||
def send(self, messages):
|
def send(self, messages):
|
||||||
out = ''
|
out = ''
|
||||||
for m in messages:
|
for m in messages:
|
||||||
|
@ -416,8 +345,6 @@ def new_interface(wallet):
|
||||||
else:
|
else:
|
||||||
host = random.choice( DEFAULT_SERVERS ) # random choice when the wallet is created
|
host = random.choice( DEFAULT_SERVERS ) # random choice when the wallet is created
|
||||||
port = wallet.port
|
port = wallet.port
|
||||||
address_cb = wallet.receive_status_callback
|
|
||||||
history_cb = wallet.receive_history_callback
|
|
||||||
|
|
||||||
if port == 50000:
|
if port == 50000:
|
||||||
InterfaceClass = NativeInterface
|
InterfaceClass = NativeInterface
|
||||||
|
@ -429,21 +356,20 @@ def new_interface(wallet):
|
||||||
print "unknown port number: %d. using native protocol."%port
|
print "unknown port number: %d. using native protocol."%port
|
||||||
InterfaceClass = NativeInterface
|
InterfaceClass = NativeInterface
|
||||||
|
|
||||||
interface = InterfaceClass(host, port, address_cb, history_cb)
|
interface = InterfaceClass(host, port)
|
||||||
|
|
||||||
return interface
|
return interface
|
||||||
|
|
||||||
|
|
||||||
def loop_interfaces_thread(wallet):
|
def loop_interfaces_thread(wallet):
|
||||||
|
|
||||||
while True:
|
while True:
|
||||||
|
interface = wallet.interface
|
||||||
try:
|
try:
|
||||||
addresses = wallet.all_addresses()
|
addresses = wallet.all_addresses()
|
||||||
version = wallet.electrum_version
|
version = wallet.electrum_version
|
||||||
wallet.interface.start_session(addresses, version)
|
interface.start_session(addresses, version)
|
||||||
|
wallet.run()
|
||||||
while wallet.interface.is_connected:
|
|
||||||
response = wallet.interface.responses.get()
|
|
||||||
wallet.interface.handle_response(response)
|
|
||||||
|
|
||||||
print "Disconnected"
|
print "Disconnected"
|
||||||
except socket.error:
|
except socket.error:
|
||||||
|
|
|
@ -17,7 +17,7 @@
|
||||||
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
|
||||||
import sys, base64, os, re, hashlib, copy, operator, ast
|
import sys, base64, os, re, hashlib, copy, operator, ast, threading
|
||||||
|
|
||||||
try:
|
try:
|
||||||
import ecdsa
|
import ecdsa
|
||||||
|
@ -265,6 +265,12 @@ class Wallet:
|
||||||
self.imported_keys = {}
|
self.imported_keys = {}
|
||||||
self.remote_url = None
|
self.remote_url = None
|
||||||
|
|
||||||
|
self.was_updated = False
|
||||||
|
self.blocks = 0
|
||||||
|
self.banner = ''
|
||||||
|
self.up_to_date_event = threading.Event()
|
||||||
|
self.up_to_date_event.clear()
|
||||||
|
|
||||||
|
|
||||||
def set_server(self, host, port):
|
def set_server(self, host, port):
|
||||||
if host!= self.host or port!=self.port:
|
if host!= self.host or port!=self.port:
|
||||||
|
@ -461,6 +467,8 @@ class Wallet:
|
||||||
|
|
||||||
|
|
||||||
def synchronize(self):
|
def synchronize(self):
|
||||||
|
if not self.master_public_key:
|
||||||
|
return False
|
||||||
|
|
||||||
is_new = False
|
is_new = False
|
||||||
while True:
|
while True:
|
||||||
|
@ -495,6 +503,7 @@ class Wallet:
|
||||||
|
|
||||||
return is_new
|
return is_new
|
||||||
|
|
||||||
|
|
||||||
def get_remote_number(self):
|
def get_remote_number(self):
|
||||||
import jsonrpclib
|
import jsonrpclib
|
||||||
server = jsonrpclib.Server(self.remote_url)
|
server = jsonrpclib.Server(self.remote_url)
|
||||||
|
@ -708,7 +717,6 @@ class Wallet:
|
||||||
def receive_history_callback(self, addr, data):
|
def receive_history_callback(self, addr, data):
|
||||||
#print "updating history for", addr
|
#print "updating history for", addr
|
||||||
self.history[addr] = data
|
self.history[addr] = data
|
||||||
self.synchronize()
|
|
||||||
self.update_tx_history()
|
self.update_tx_history()
|
||||||
self.save()
|
self.save()
|
||||||
|
|
||||||
|
@ -920,3 +928,74 @@ class Wallet:
|
||||||
address = address + ' <' + payto_address + '>'
|
address = address + ' <' + payto_address + '>'
|
||||||
|
|
||||||
return address, amount, label, message, signature, identity, url
|
return address, amount, label, message, signature, identity, url
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
def handle_response(self, r):
|
||||||
|
if r is None:
|
||||||
|
print "empty item"
|
||||||
|
return
|
||||||
|
|
||||||
|
method = r['method']
|
||||||
|
params = r['params']
|
||||||
|
result = r['result']
|
||||||
|
|
||||||
|
if method == 'server.banner':
|
||||||
|
self.banner = result
|
||||||
|
self.was_updated = True
|
||||||
|
|
||||||
|
elif method == 'session.poll':
|
||||||
|
# native poll
|
||||||
|
blocks, changed_addresses = result
|
||||||
|
if blocks == -1: raise BaseException("session not found")
|
||||||
|
self.blocks = int(blocks)
|
||||||
|
if changed_addresses:
|
||||||
|
self.is_up_to_date = False
|
||||||
|
self.was_updated = True
|
||||||
|
for addr, status in changed_addresses.items():
|
||||||
|
self.receive_status_callback(addr, status)
|
||||||
|
else:
|
||||||
|
self.is_up_to_date = True
|
||||||
|
|
||||||
|
elif method == 'server.peers':
|
||||||
|
#print "Received server list: ", result
|
||||||
|
self.servers = map( lambda x:x[1], result )
|
||||||
|
|
||||||
|
elif method == 'address.subscribe':
|
||||||
|
addr = params[-1]
|
||||||
|
self.receive_status_callback(addr, result)
|
||||||
|
|
||||||
|
elif method == 'address.get_history':
|
||||||
|
addr = params[0]
|
||||||
|
self.receive_history_callback(addr, result)
|
||||||
|
self.was_updated = True
|
||||||
|
|
||||||
|
elif method == 'transaction.broadcast':
|
||||||
|
self.tx_result = result
|
||||||
|
self.tx_event.set()
|
||||||
|
|
||||||
|
elif method == 'numblocks.subscribe':
|
||||||
|
self.blocks = result
|
||||||
|
#self.newblock_callback,(result,))
|
||||||
|
|
||||||
|
elif method == 'client.version':
|
||||||
|
pass
|
||||||
|
|
||||||
|
else:
|
||||||
|
print "unknown message:", method, params, result
|
||||||
|
|
||||||
|
def update(self):
|
||||||
|
self.up_to_date_event.wait()
|
||||||
|
|
||||||
|
def run(self):
|
||||||
|
while self.interface.is_connected:
|
||||||
|
new = self.synchronize()
|
||||||
|
if self.interface.is_up_to_date() and not new:
|
||||||
|
self.up_to_date = True
|
||||||
|
self.up_to_date_event.set()
|
||||||
|
else:
|
||||||
|
self.up_to_date = False
|
||||||
|
|
||||||
|
response = self.interface.responses.get()
|
||||||
|
self.handle_response(response)
|
||||||
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue