tSPV example script (first commit) - electrum - Electrum Bitcoin wallet HTML git clone https://git.parazyd.org/electrum DIR Log DIR Files DIR Refs DIR Submodules --- DIR commit 633a65878f024dc686542d1a18171fc58d86a3be DIR parent 11a8e04dd07f39988a278f907d2d9dd8a9e12949 HTML Author: thomasv <thomasv@gitorious> Date: Thu, 18 Oct 2012 17:07:27 +0200 SPV example script (first commit) Diffstat: A scripts/validate_tx | 93 +++++++++++++++++++++++++++++++ 1 file changed, 93 insertions(+), 0 deletions(-) --- DIR diff --git a/scripts/validate_tx b/scripts/validate_tx t@@ -0,0 +1,93 @@ +#!/usr/bin/env python + +import sys, hashlib +from electrum import Interface + +"""validate a transaction (SPV)""" + + +Hash = lambda x: hashlib.sha256(hashlib.sha256(x).digest()).digest() + +def rev_hex(s): + return s.decode('hex')[::-1].encode('hex') + +def int_to_hex(i, length=1): + s = hex(i)[2:].rstrip('L') + s = "0"*(2*length - len(s)) + s + return rev_hex(s) + + +i = Interface({'server':'ecdsa.org:50002:s'}) +i.start() + + +def get_header(i, block_height): + i.send([('blockchain.block.get_header',[block_height])]) + while True: + r = i.responses.get(True, 100000000000) + method = r.get('method') + if method == 'blockchain.block.get_header': + break + return r.get('result') + + + +def get_merkle(i, tx_hash): + + i.send([('blockchain.transaction.get_merkle',[tx_hash])]) + while True: + r = i.responses.get(True, 100000000000) + method = r.get('method') + if method == 'blockchain.transaction.get_merkle': + break + + return r.get('result') + + +def merkle_root(merkle): + + merkle = map (lambda tx_hash: tx_hash.decode('hex')[::-1], merkle) + while len(merkle) != 1: + if len(merkle)%2: merkle.append( merkle[-1] ) + n = [] + while merkle: + n.append( Hash( merkle[0] + merkle[1] ) ) + merkle = merkle[2:] + + merkle = n + + return merkle[0][::-1].encode('hex') + + +def hash_header(res): + header = int_to_hex(res.get('version'),4) \ + + rev_hex(res.get('prev_block_hash')) \ + + rev_hex(res.get('merkle_root')) \ + + int_to_hex(int(res.get('timestamp')),4) \ + + int_to_hex(int(res.get('bits')),4) \ + + int_to_hex(int(res.get('nonce')),4) + return rev_hex(Hash(header.decode('hex')).encode('hex')) + + +def verify_tx(tx_hash): + res = get_merkle(i, tx_hash) + assert res.get('merkle_root') == merkle_root(res['merkle']) + block_height = res.get('block_height') + _hash = None + for height in range(block_height-10,block_height+10): + header = get_header(i, height) + if _hash: assert _hash == header.get('prev_block_hash') + _hash = hash_header(header) + #print _hash + if height==block_height: + assert header.get('merkle_root') == res.get('merkle_root') + + + +try: + tx = sys.argv[1] +except: + tx = '587430e52af2cec98b3fd543083469ffa7a5f5dd2bd569898a7897a64e2eb031' + +verify_tx(tx) +