compatibility with 0.6 protocol

This commit is contained in:
ThomasV 2012-11-14 15:33:44 +01:00
parent 259c5c1b06
commit 8ec6a60daf
5 changed files with 63 additions and 39 deletions

View File

@ -845,6 +845,8 @@ class ElectrumWindow(QMainWindow):
label = self.wallet.labels.get(address,'') label = self.wallet.labels.get(address,'')
n = 0 n = 0
h = self.wallet.history.get(address,[]) h = self.wallet.history.get(address,[])
if h == ['*']: h = []
for tx_hash, tx_height in h: for tx_hash, tx_height in h:
tx = self.wallet.transactions.get(tx_hash) tx = self.wallet.transactions.get(tx_hash)
if tx: n += 1 if tx: n += 1

View File

@ -84,6 +84,12 @@ class Interface(threading.Thread):
if error: if error:
print_error("received error:", c) print_error("received error:", c)
if msg_id is not None:
with self.lock:
method, params, channel = self.unanswered_requests.pop(msg_id)
response_queue = self.responses[channel]
response_queue.put({'method':method, 'params':params, 'error':error, 'id':msg_id})
return return
if msg_id is not None: if msg_id is not None:

View File

@ -32,7 +32,7 @@ class WalletVerifier(threading.Thread):
self.daemon = True self.daemon = True
self.config = config self.config = config
self.interface = interface self.interface = interface
self.transactions = [] # monitored transactions self.transactions = {} # monitored transactions
self.interface.register_channel('verifier') self.interface.register_channel('verifier')
self.verified_tx = config.get('verified_tx',{}) # height of verified tx self.verified_tx = config.get('verified_tx',{}) # height of verified tx
@ -49,16 +49,16 @@ class WalletVerifier(threading.Thread):
def get_confirmations(self, tx): def get_confirmations(self, tx):
""" return the number of confirmations of a monitored transaction. """ """ return the number of confirmations of a monitored transaction. """
with self.lock: with self.lock:
if tx in self.transactions: if tx in self.transactions.keys():
return (self.local_height - self.verified_tx[tx] + 1) if tx in self.verified_tx else 0 return (self.local_height - self.verified_tx[tx] + 1) if tx in self.verified_tx else 0
else: else:
return 0 return 0
def add(self, tx_hash): def add(self, tx_hash, tx_height):
""" add a transaction to the list of monitored transactions. """ """ add a transaction to the list of monitored transactions. """
with self.lock: with self.lock:
if tx_hash not in self.transactions: if tx_hash not in self.transactions.keys():
self.transactions.append(tx_hash) self.transactions[tx_hash] = tx_height
def run(self): def run(self):
requested_merkle = [] requested_merkle = []
@ -87,11 +87,11 @@ class WalletVerifier(threading.Thread):
# request missing tx # request missing tx
if all_chunks: if all_chunks:
for tx_hash in self.transactions: for tx_hash, tx_height in self.transactions.items():
if tx_hash not in self.verified_tx: if tx_hash not in self.verified_tx:
if self.merkle_roots.get(tx_hash) is None and tx_hash not in requested_merkle: if self.merkle_roots.get(tx_hash) is None and tx_hash not in requested_merkle:
print_error('requesting merkle', tx_hash) print_error('requesting merkle', tx_hash)
self.interface.send([ ('blockchain.transaction.get_merkle',[tx_hash]) ], 'verifier') self.interface.send([ ('blockchain.transaction.get_merkle',[tx_hash, tx_height]) ], 'verifier')
requested_merkle.append(tx_hash) requested_merkle.append(tx_hash)
# process pending headers # process pending headers

View File

@ -1,4 +1,4 @@
ELECTRUM_VERSION = "1.3" # version of the client package ELECTRUM_VERSION = "1.3" # version of the client package
PROTOCOL_VERSION = '0.5' # protocol version requested PROTOCOL_VERSION = '0.6' # protocol version requested
SEED_VERSION = 4 # bump this everytime the seed generation is modified SEED_VERSION = 4 # bump this everytime the seed generation is modified
TRANSLATION_ID = 32150 # version of the wiki page TRANSLATION_ID = 32150 # version of the wiki page

View File

@ -381,7 +381,8 @@ class Wallet:
addr = item.get('address') addr = item.get('address')
if addr in addresses: if addr in addresses:
key = item['prevout_hash'] + ':%d'%item['prevout_n'] key = item['prevout_hash'] + ':%d'%item['prevout_n']
value = self.prevout_values[ key ] value = self.prevout_values.get( key )
if value is None: continue
v -= value v -= value
for item in d.get('outputs'): for item in d.get('outputs'):
addr = item.get('address') addr = item.get('address')
@ -409,6 +410,7 @@ class Wallet:
def get_addr_balance(self, addr): def get_addr_balance(self, addr):
assert self.is_mine(addr) assert self.is_mine(addr)
h = self.history.get(addr,[]) h = self.history.get(addr,[])
if h == ['*']: return 0,0
c = u = 0 c = u = 0
for tx_hash, tx_height in h: for tx_hash, tx_height in h:
v = self.get_tx_value(tx_hash, [addr]) v = self.get_tx_value(tx_hash, [addr])
@ -531,6 +533,7 @@ class Wallet:
def get_status(self, h): def get_status(self, h):
if not h: return None if not h: return None
if h == ['*']: return '*'
status = '' status = ''
for tx_hash, height in h: for tx_hash, height in h:
status += tx_hash + ':%d:' % height status += tx_hash + ':%d:' % height
@ -553,16 +556,18 @@ class Wallet:
def receive_history_callback(self, addr, hist): def receive_history_callback(self, addr, hist):
if not self.check_new_history(addr, hist): if hist != ['*']:
raise BaseException("error: received history for %s is not consistent with known transactions"%addr) if not self.check_new_history(addr, hist):
raise BaseException("error: received history for %s is not consistent with known transactions"%addr)
with self.lock: with self.lock:
self.history[addr] = hist self.history[addr] = hist
self.save() self.save()
if hist != ['*']:
for tx_hash, tx_height in hist: for tx_hash, tx_height in hist:
if tx_height>0: if tx_height>0:
self.verifier.add(tx_hash) self.verifier.add(tx_hash, tx_height)
def get_tx_history(self): def get_tx_history(self):
@ -886,6 +891,7 @@ class Wallet:
# set the timestamp for transactions that need it # set the timestamp for transactions that need it
for hist in self.history.values(): for hist in self.history.values():
if hist == ['*']: continue
for tx_hash, tx_height in hist: for tx_hash, tx_height in hist:
tx = self.transactions.get(tx_hash) tx = self.transactions.get(tx_hash)
if tx and not tx.get('timestamp'): if tx and not tx.get('timestamp'):
@ -894,7 +900,7 @@ class Wallet:
self.set_tx_timestamp(tx_hash, timestamp) self.set_tx_timestamp(tx_hash, timestamp)
if tx_height>0: if tx_height>0:
self.verifier.add(tx_hash) self.verifier.add(tx_hash, tx_height)
@ -936,6 +942,7 @@ class Wallet:
# 1 check that tx is referenced in addr_history. # 1 check that tx is referenced in addr_history.
addresses = [] addresses = []
for addr, hist in self.history.items(): for addr, hist in self.history.items():
if hist == ['*']:continue
for txh, height in hist: for txh, height in hist:
if txh == tx_hash: if txh == tx_hash:
addresses.append(addr) addresses.append(addr)
@ -998,6 +1005,7 @@ class WalletSynchronizer(threading.Thread):
# request any missing transactions # request any missing transactions
for history in self.wallet.history.values(): for history in self.wallet.history.values():
if history == ['*']: continue
for tx_hash, tx_height in history: for tx_hash, tx_height in history:
if self.wallet.transactions.get(tx_hash) is None and (tx_hash, tx_height) not in missing_tx: if self.wallet.transactions.get(tx_hash) is None and (tx_hash, tx_height) not in missing_tx:
missing_tx.append( (tx_hash, tx_height) ) missing_tx.append( (tx_hash, tx_height) )
@ -1035,7 +1043,11 @@ class WalletSynchronizer(threading.Thread):
# 3. handle response # 3. handle response
method = r['method'] method = r['method']
params = r['params'] params = r['params']
result = r['result'] result = r.get('result')
error = r.get('error')
if error:
print "error", r
continue
if method == 'blockchain.address.subscribe': if method == 'blockchain.address.subscribe':
addr = params[0] addr = params[0]
@ -1045,34 +1057,38 @@ class WalletSynchronizer(threading.Thread):
elif method == 'blockchain.address.get_history': elif method == 'blockchain.address.get_history':
addr = params[0] addr = params[0]
hist = [] if result == ['*']:
assert requested_histories.pop(addr) == '*'
self.wallet.receive_history_callback(addr, result)
else:
hist = []
# check that txids are unique
txids = []
for item in result:
tx_hash = item['tx_hash']
if tx_hash not in txids:
txids.append(tx_hash)
hist.append( (tx_hash, item['height']) )
# check that txids are unique if len(hist) != len(result):
txids = [] raise BaseException("error: server sent history with non-unique txid")
for item in result:
tx_hash = item['tx_hash']
if tx_hash not in txids:
txids.append(tx_hash)
hist.append( (tx_hash, item['height']) )
if len(hist) != len(result): # check that the status corresponds to what was announced
raise BaseException("error: server sent history with non-unique txid") rs = requested_histories.pop(addr)
if self.wallet.get_status(hist) != rs:
# check that the status corresponds to what was announced raise BaseException("error: status mismatch: %s"%addr)
if self.wallet.get_status(hist) != requested_histories.pop(addr):
raise BaseException("error: status mismatch: %s"%addr)
# store received history # store received history
self.wallet.receive_history_callback(addr, hist) self.wallet.receive_history_callback(addr, hist)
# request transactions that we don't have # request transactions that we don't have
for tx_hash, tx_height in hist: for tx_hash, tx_height in hist:
if self.wallet.transactions.get(tx_hash) is None: if self.wallet.transactions.get(tx_hash) is None:
if (tx_hash, tx_height) not in requested_tx and (tx_hash, tx_height) not in missing_tx: if (tx_hash, tx_height) not in requested_tx and (tx_hash, tx_height) not in missing_tx:
missing_tx.append( (tx_hash, tx_height) ) missing_tx.append( (tx_hash, tx_height) )
else: else:
timestamp = self.wallet.verifier.get_timestamp(tx_height) timestamp = self.wallet.verifier.get_timestamp(tx_height)
self.wallet.set_tx_timestamp(tx_hash, timestamp) self.wallet.set_tx_timestamp(tx_hash, timestamp)
elif method == 'blockchain.transaction.get': elif method == 'blockchain.transaction.get':
tx_hash = params[0] tx_hash = params[0]