Browse Source

asynchronous processing: use a queue, handle responses in wallet class

283
ThomasV 13 years ago
parent
commit
f60f6c28d3
  1. 6
      client/electrum
  2. 20
      client/gui_qt.py
  3. 104
      client/interface.py
  4. 83
      client/wallet.py

6
client/electrum

@ -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)

20
client/gui_qt.py

@ -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

104
client/interface.py

@ -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,74 +76,16 @@ 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)
self.responses.put({'method':method, 'params':params, 'result':result}) if method == 'address.subscribe':
#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] addr = params[-1]
if addr in self.addresses_waiting_for_status: if addr in self.addresses_waiting_for_status:
self.addresses_waiting_for_status.remove(addr) self.addresses_waiting_for_status.remove(addr)
apply(self.address_callback,(addr, result))
elif method == 'address.get_history': elif method == 'address.get_history':
addr = params[0] addr = params[0]
if addr in self.addresses_waiting_for_history: if addr in self.addresses_waiting_for_history:
self.addresses_waiting_for_history.remove(addr) self.addresses_waiting_for_history.remove(addr)
apply(self.history_callback, (addr, result)) self.responses.put({'method':method, 'params':params, 'result':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):
@ -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:

83
client/wallet.py

@ -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…
Cancel
Save