From 3f8641e7c40cb659cd5e275952cf6e5def2aabfa Mon Sep 17 00:00:00 2001 From: jackjack-jj Date: Sun, 14 Apr 2013 22:57:34 +0200 Subject: [PATCH] v2.0: ui, tx creation and few minor changes --- pywallet.py | 1152 +++++++++++++++++++++++++++++++++++++++++++++++++-- 1 file changed, 1119 insertions(+), 33 deletions(-) diff --git a/pywallet.py b/pywallet.py index 3908d1b..1749273 100755 --- a/pywallet.py +++ b/pywallet.py @@ -1,5 +1,6 @@ #!/usr/bin/env python -pywversion="1.9.4" +#-*- coding: utf-8 -*- +pywversion="2.0" never_update=False # @@ -21,8 +22,8 @@ except: missing_dep.append('bsddb') import os, sys, time -pyw_filename = sys.argv[0].split('/')[len(sys.argv[0].split('/'))-1] -pyw_path = os.getcwd() +pyw_filename = os.path.basename(__file__) +pyw_path = os.path.dirname(os.path.realpath(__file__)) try: for i in os.listdir('/usr/lib/python2.5/site-packages'): @@ -1431,8 +1432,11 @@ def md5_2(a): return hashlib.md5(a).digest() def md5_file(nf): + try: fichier = file(nf, 'r').read() return md5_2(fichier) + except: + return 'zz' def md5_onlinefile(add): page = urllib.urlopen(add).read() @@ -2038,8 +2042,9 @@ def rewrite_wallet(db_env, walletfile, destFileName, pre_put_callback=None): # wallet.dat reader / writer +addr_to_keys={} def read_wallet(json_db, db_env, walletfile, print_wallet, print_wallet_transactions, transaction_filter, include_balance, vers=-1, FillPool=False): - global passphrase + global passphrase, addr_to_keys crypted=False private_keys = [] @@ -2085,6 +2090,7 @@ def read_wallet(json_db, db_env, walletfile, print_wallet, print_wallet_transact sec = SecretToASecret(PrivKeyToSecret(d['private_key']), compressed) hexsec = ASecretToSecret(sec).encode('hex') private_keys.append(sec) + addr_to_keys[addr]=[hexsec, d['public_key'].encode('hex')] json_db['keys'].append({'addr' : addr, 'sec' : sec, 'hexsec' : hexsec, 'secret' : hexsec, 'pubkey':d['public_key'].encode('hex'), 'compressed':compressed, 'private':d['private_key'].encode('hex')}) elif type == "wkey": @@ -2134,7 +2140,7 @@ def read_wallet(json_db, db_env, walletfile, print_wallet, print_wallet_transact else: json_db[type] = 'unsupported' - print d + print "Wallet data not recognized: "+str(d) list_of_reserve_not_in_pool=[] parse_wallet(db, item_callback) @@ -2201,8 +2207,10 @@ def read_wallet(json_db, db_env, walletfile, print_wallet, print_wallet_transact sec = SecretToASecret(secret, compressed) k['sec'] = sec + k['hexsec'] = sec k['secret'] = secret.encode('hex') k['compressed'] = compressed + addr_to_keys[k['addr']]=[sec, k['pubkey']] # del(k['ckey']) # del(k['secret']) # del(k['pubkey']) @@ -2326,7 +2334,499 @@ def keyinfo(sec, keyishex): return True +def css_wui(): + return """html, body { + height: 100%; + width: 100%; + padding: 0; + margin: 0; +} + +body { + margin: 0px; + padding: 0px; + background: url(%3D%3D) repeat; + font-family: 'Open Sans', sans-serif; + font-size: 10pt; + color: #B0B0B0; +} + + +h1, h2, h3 { + margin: 0; + padding: 0; +} + +h2 +{ + font-weight: 400; + font-family: 'Archivo Narrow', sans-serif; + font-size: 2.50em; +} + +p, ol, ul { + margin-top: 0px; +} + +p { + line-height: 180%; +} + +strong { +} + +a { + color: #1492C4; +} + +a:hover { + text-decoration: none; +} + +a img { + border: none; +} + +img.border { + border: 10px solid rgba(255,255,255,.10); +} + +img.alignleft { + float: left; + margin-right: 30px; +} + +img.alignright { + float: right; +} + +img.aligncenter { + margin: 0px auto; +} + +hr { + display: none; +} + +#retour-pyw{ + overflow: auto; +} + +#uptodate{ + position:absolute; + top:0px; + right:0px; + background: rgba(0,0,0,0.70); + padding:10px; +} + +#full-screen-background-image { + z-index: -999; + min-height: 100%; + min-width: 1024px; + width: 100%; + height: auto; + position: fixed; + top: 0; + left: 0; +} + +#wrapper { + position: relative; + width: 100%; + min-height: 400px; + #margin: 30px auto; + margin-top:10px; #decalage p/r haut +} + +#wrapper { + overflow: hidden; +} + +.container { + width: 1000px; + margin: 0px auto; +} + +.clearfix { + clear: both; +} + +/** HEADER */ + +#header-wrapper-title +{ + overflow: hidden; + height: 80px; + margin-bottom: 10px; + background: rgba(0,0,0,0); +} + +#header-wrapper +{ + overflow: hidden; + height: 50px; + margin-bottom: 20px; + background: rgba(0,0,0,0.70); +} + +#header { + overflow: hidden; +} + +/** LOGO */ + +#logo { + float: left; + #width: 300px; + height: 50px; + +} + +#logo h1, #logo p { + margin: 0px; + line-height: normal; +} + +#logo h1 a { + padding-left: 00px; + text-decoration: none; + font-size: 2.50em; + font-weight: 400; + font-family: 'Archivo Narrow', sans-serif; + color: #FFFFFF; +} + +/** MENU */ + +#menu { + float: left; + height: 50px; +} + +#menu ul { + margin: 0px; + padding: 0px; + list-style: none; + line-height: normal; +} + +#menu li { + float: left; + margin-right: 10px; + padding: 0px 5px 0px 5px; +} + +#menu a { + display: block; + height: 50px; + padding: 0px 10px; + line-height: 50px; + text-decoration: none; + text-transform: uppercase; + color: #FFFFFF; +} + +#menu a:hover { + text-decoration: none; + background: rgba(0,0,0,0.70); +} + +#menu .active +{ + background: rgba(0,0,0,0.70); +} + +/** PAGE */ + +#page { + overflow: hidden; + margin-bottom: 20px; +} + +/** CONTENT */ + +#content { + float: left; + width: 950px; + padding: 40px; + background: rgba(0,0,0,0.70); +} + +#content h2 a +{ + display: block; + padding: 0px 0px 20px 0px; + text-decoration: none; + color: #FFFFFF; +} + +#content #box1 +{ + margin-bottom: 0px; +} + +/** SIDEBAR */ + +#sidebar { + float: right; + width: 350px; + padding: 20px; + background: rgba(0,0,0,0.70); +} + +#sidebar h2 +{ + padding: 0px 0px 00px 0px; + color: #FFFFFF; +} + +/* Footer */ + +#footer { + overflow: hidden; + margin: 00px auto 0px auto; + padding: 10px 0px; + background: rgba(0,0,0,0.70); +} + +#footer p { + text-align: center; + font-size: 12px; +} + +#footer a { +} + +/** LIST STYLE 1 */ + +ul.style1 { + margin: 0px; + padding: 10px 0px 0px 0px; + list-style: none; +} + +ul.style1 li { + clear: both; + margin-bottom: 25px; + padding: 30px 0px 40px 0px; + border-top: 1px solid #000000; + box-shadow: inset 0 1px 0 rgba(255,255,255,.10); +} + +ul.style1 h3 { + padding-bottom: 5px; + font-size: 14px; + color: #FFFFFF; +} + +ul.style1 p { + line-height: 150%; +} + +ul.style1 .button-style { + float: left; + margin-top: 0px; +} + +ul.style1 .first { + padding-top: 0px; + border-top: none; + box-shadow: none; +} + +/** LIST STYLE 3 */ + +ul.style3 { + margin: 0px; + padding: 0px; + list-style: none; +} + +ul.style3 li { + padding: 10px 0px 10px 0px; + border-top: 1px solid #000000; + box-shadow: inset 0 1px 0 rgba(255,255,255,.10); +} + +ul.style3 a { + text-decoration: none; + color: #949494; +} + +ul.style3 a:hover { + text-decoration: underline; +} + +ul.style3 .first { + padding-top: 0px; + border-top: none; + box-shadow: none; +} + +ul.style3 .date { + width: 87px; + background-color: #1F768D; + margin-top: 20px; + height: 24px; + line-height: 24px; + text-align: center; + font-size: 12px; + color: #FFFFFF; +} + +ul.style3 .first .date +{ + margin-top: 0px; +} + +.button-style +{ + display: inline-block; + background-color: #1F768D; + margin-top: 0px; + padding: 5px 30px; + height: 24px; + line-height: 24px; + text-decoration: none; + text-align: center; + color: #FFFFFF; +} + +.button-style-red +{ + color: #ffffff; + display: inline-block; + background-color: #a12323; + margin-top: 20px; + padding: 5px 30px; + height: 24px; + line-height: 24px; + text-decoration: none; + text-align: center; +} + +.entry +{ + margin-bottom: 30px; +} +""" + +def onclick_on_tab(page): + list=['DumpPage','ImportPage','DeletePage','InfoPage','AboutPage','PassphrasePage','TxPage'] + r='' + for p in list: + if p!=page: + r+="document.getElementById('"+p+"').style.display='none';" + r+="document.getElementById('"+p+"Button').className='';" + r+="document.getElementById('"+page+"').style.display='block';" + r+="document.getElementById('"+page+"Button').className='active';" + return r + +def html_wui(listcontent,uptodate_text): + global pywversion + return """ + + + +Pywallet Web Interface - Pywallet """+pywversion+""" + + + + + +
+ +
+
 
+ +
+
+
+ """+listcontent+""" +
+ +
+ +
+
"""+uptodate_text+"""
+ + +""" + def WI_FormInit(title, action, divname): + return "
  • %s

    "%title return '

      '+title+'

    ' def WI_InputText(label, name, id, value, size=30): @@ -2336,8 +2836,8 @@ def WI_InputPassword(label, name, id, value, size=30): return '%s
    '%(label, name, id, value, size) def WI_Submit(value, local_block, local_button, function): - return ''%(value, local_block, local_button, function) - + return """
    %s"""%(function,value) + def WI_CloseButton(local_block, local_button): return ''%(local_block, local_button, local_button) @@ -2345,7 +2845,7 @@ def WI_ReturnDiv(local_block): return ''%(local_block) def WI_FormEnd(): - return '

    ' + return "
  • " def WI_RadioButton(name, value, id, checked, label): return '   %s
    '%(name, value, id, checked, label) @@ -2353,6 +2853,9 @@ def WI_RadioButton(name, value, id, checked, label): def WI_Checkbox(name, value, id, other, label): return '%s'%(name, value, id, other, label) +def WI_Endiv(t,name,title, desc,hidden=False): + return '

    '+title+'

    '+X_if_else('

    '+desc+'

    ',desc!='','')+'
      '+t+'
    ' + def WI_AjaxFunction(name, command_when_ready, query_string, command_until_ready): return '\n\ function ajax%s(){\n\ @@ -2445,13 +2948,21 @@ def import_csv_keys(filename, wdir, wname, nbremax=9999999): return True +def dep_text_aboutpage(val): + if val: + return "Not found" + else: + return "Found" + +CTX_adds='' + if 'twisted' not in missing_dep: class WIRoot(resource.Resource): def render_GET(self, request): try: request.args['update'][0] - update_pyw() + return update_pyw() except: True @@ -2470,7 +2981,7 @@ if 'twisted' not in missing_dep: X_if_else( 'Pywallet is up-to-date', uptodate, - 'Pywallet is NOT up-to-date
    Click to update, then restart Pywallet'), + 'Pywallet is not up-to-date
    Click to update'), checking_finished, 'Checking version...' ) @@ -2612,10 +3123,33 @@ if 'twisted' not in missing_dep: WI_InputText('Key:', 'key', 'bf-key', '', 35) + \ '
    ' + \ WI_FormEnd() + + global CTX_adds, addr_to_keys + CTX_adds2=CTX_adds.split('|')+addr_to_keys.keys() + + + CreateTxForm = WI_FormInit('Create transaction', 'CTX', 'divformctx') + \ + X_if_else("Additional addresses used: " + ', '.join(CTX_adds.split('|'))+"

    ",len(CTX_adds)>0,"No additional addresses used

    ") + \ + listtx_txt(CTX_adds) + \ + WI_FormEnd() + + CreateTxForm2 = WI_FormInit('Check addresses funds', 'ListTransactions', 'divformctx') + \ + WI_InputText('Addresses: ', 'adds', 'ctx-adds', '', 35) + \ + WI_Submit('Check ', '', '', 'ajaxCTx') + \ + WI_FormEnd() Misc = '' Javascript = '' # WI_AjaxFunction('Import', 'document.getElementById("ImportDiv").innerHTML = ajaxRequest.responseText;', '"/Import?dir="+document.getElementById("impf-dir").value+"&name="+document.getElementById("impf-name").value+"&key="+document.getElementById("impf-key").value+"&label="+document.getElementById("impf-label").value+"&vers="+document.getElementById("impf-vers").value+"&format="+(document.getElementById("impf-hex").checked?"hex":"reg")+(document.getElementById("impf-reserve").checked?"&reserve=1":"")', 'document.getElementById("ImportDiv").innerHTML = "Loading...";') + \ page = 'Pywallet Web Interface' + header + Javascript + CPPForm + DWForm + MWForm + DKForm + IKForm + DTxForm + InfoForm + ImportForm + ImportTxForm + DeleteForm + BalanceForm + Misc + '' + + AboutPage="\ +Pywallet is a tool to manage wallet files, developped by jackjack. Support thread is on bitcointalk.
    \ +
    \ +To support pywallet's development or if you think it's worth something, you can send anything you want to 1AQDfx22pKGgXnUZFL1e4UKos3QqvRzNh5.\ +\ +



    Dependencies:
    \ + \ +    ecdsa: "+dep_text_aboutpage('ecdsa' in missing_dep)+"\ + \ +

    Pywallet path:   "+pyw_path+"/"+pyw_filename+"\ + " + + return html_wui(Javascript + \ + WI_Endiv(DWForm+DKForm+DTxForm, 'DumpPage', 'Dump', '') + \ + WI_Endiv(ImportForm+IKForm+MWForm+ImportTxForm,'ImportPage', 'Import', "Don't forget to close Bitcoin when you modify your wallet", True) + \ + WI_Endiv(DeleteForm,'DeletePage', 'Delete', "Don't forget to close Bitcoin when you modify your wallet", True) + \ + WI_Endiv(CPPForm,'PassphrasePage', 'Change passphrase', '', True) + \ + WI_Endiv(InfoForm+BalanceForm,'InfoPage', 'Info', '', True) + \ + WI_Endiv(CreateTxForm2+CreateTxForm,'TxPage', 'Manage transactions', 'You can here create your own transactions.
    By default, the unspent transactions from addresses previously dumped are shown, but you can add other addresses to check.
    You can\'t create a transaction if you didn\'t dump the private keys of each input beforehand.', True) + \ + WI_Endiv(AboutPage,'AboutPage','About','', True) + ,check_version_text + ) + + return page def getChild(self, name, request): @@ -2875,8 +3436,9 @@ def ct(l_prevh, l_prevn, l_prevsig, l_prevpubkey, l_value_out, l_pubkey_out, is_ txin_ret += inverse_str("%08x"%l_prevn[i]) if scriptSig: - txin_ret2 += "%02x"%(len(l_prevsig[i])/2) + txin_ret2 += "%02x"%(1+len(l_prevsig[i])/2) txin_ret2 += l_prevsig[i] + txin_ret2 += "01" txin_ret2 += "%02x"%(len(l_prevpubkey[i])/2) txin_ret2 += l_prevpubkey[i] @@ -2886,6 +3448,8 @@ def ct(l_prevh, l_prevn, l_prevsig, l_prevpubkey, l_value_out, l_pubkey_out, is_ elif index == i: txin_ret += "%02x"%(len(oldScriptPubkey)/2) txin_ret += oldScriptPubkey + else: + txin_ret += "00" ret += txin_ret ret += "ffffffff" @@ -2910,7 +3474,46 @@ def ct(l_prevh, l_prevn, l_prevsig, l_prevpubkey, l_value_out, l_pubkey_out, is_ if not scriptSig: ret += "01000000" return ret + +def create_transaction(secret_key, hashes_txin, indexes_txin, pubkey_txin, prevScriptPubKey, amounts_txout, scriptPubkey): + li1 = len(secret_key) + li2 = len(hashes_txin) + li3 = len(indexes_txin) + li4 = len(pubkey_txin) + li5 = len(prevScriptPubKey) + + if li1 != li2 or li2 != li3 or li3 != li4 or li4 != li5: + print("Error in the number of tx inputs") + exit(0) + lo1 = len(amounts_txout) + lo2 = len(scriptPubkey) + + if lo1 != lo2: + print("Error in the number of tx outputs") + exit(0) + + sig_txin = [] + i=0 + for cpt in hashes_txin: + sig_txin.append(sign_message(secret_key[i].decode('hex'), ct(hashes_txin, indexes_txin, sig_txin, pubkey_txin, amounts_txout, scriptPubkey, i, prevScriptPubKey[i]), True)+"01") + i+=1 + + tx = ct(hashes_txin, indexes_txin, sig_txin, pubkey_txin, amounts_txout, scriptPubkey) + hashtx = Hash(tx.decode('hex')).encode('hex') + + for i in range(len(sig_txin)): + try: + verify_message_signature(pubkey_txin[i], sig_txin[i][:-2], ct(hashes_txin, indexes_txin, sig_txin, pubkey_txin, amounts_txout, scriptPubkey, i, prevScriptPubKey[i]), True) + print("sig %2d: verif ok"%i) + except: + print("sig %2d: verif error"%i) + exit(0) + +# tx += end_of_wallettx([], int(time.time())) +# return [inverse_str(hashtx), "027478" + hashtx, tx] + return [inverse_str(hashtx), "", tx] + def inverse_str(string): ret = "" for i in range(len(string)/2): @@ -2918,6 +3521,302 @@ def inverse_str(string): ret += string[len(string)-2-2*i+1]; return ret +def read_table(table, beg, end): + rows = table.split(beg) + for i in range(len(rows)): + rows[i] = rows[i].split(end)[0] + return rows + +def read_blockexplorer_table(table): + cell = [] + rows = read_table(table, '', '') + for i in range(len(rows)): + cell.append(read_table(rows[i], '', '')) + del cell[i][0] + del cell[0] + del cell[0] + return cell + +txin_amounts = {} + +def bc_address_to_available_tx(address, testnet=False): + TN="" + if testnet: + TN="testnet" + + blockexplorer_url = "http://blockexplorer.com/"+TN+"/address/" + ret = "" + txin = [] + txin_no = {} + global txin_amounts + txout = [] + balance = 0 + txin_is_used = {} + + page = urllib.urlopen("%s/%s" % (blockexplorer_url, address)) + try: + table = page.read().split('')[1] + table = table.split("
    ")[0] + except: + return {address:[]} + + cell = read_blockexplorer_table(table) + + for i in range(len(cell)): + txhash = read_table(cell[i][0], '/tx/', '#')[1] + post_hash = read_table(cell[i][0], '#', '">')[1] + io = post_hash[0] + no_tx = post_hash[1:] + if io in 'i': + txout.append([txhash, post_hash]) + else: + txin.append(txhash+no_tx) + txin_no[txhash+no_tx] = post_hash[1:] + txin_is_used[txhash+no_tx] = 0 + + #hashblock = read_table(cell[i][1], '/block/', '">')[1] + #blocknumber = read_table(cell[i][1], 'Block ', '')[1] + + txin_amounts[txhash+no_tx] = round(float(cell[i][2]), 8) + +# if cell[i][3][:4] in 'Sent' and io in 'o': +# print(cell[i][3][:4]) +# print(io) +# return 'error' +# if cell[i][3][:4] in 'Rece' and io in 'i': +# print(cell[i][3][:4]) +# print(io) +# return 'error' + + balance = round(float(cell[i][5]), 8) + + + for tx in txout: + pagetx = urllib.urlopen("http://blockexplorer.com/"+TN+"/tx/"+tx[0]) + table_in = pagetx.read().split('Outputs')[0].split('')[1].split("
    ")[0] + + cell = read_blockexplorer_table(table_in) + for i in range(len(cell)): + txhash = read_table(cell[i][0], '/tx/', '#')[1] + no_tx = read_table(cell[i][0], '#', '">')[1][1:] + + if txhash+no_tx in txin: + txin_is_used[txhash+no_tx] = 1 + + ret = [] + for tx in txin: + if not txin_is_used[tx]: + ret.append([tx,txin_amounts[tx],txin_no[tx]]) + + return {address : ret} + +def write_avtx(list_avtx, testnet=False): + TN="" + if testnet: + TN="testnet" + gret = "" + for add in list_avtx: + notnull = False + try: + hexsec = " -> " + bc_address_to_sec[add] + except: + hexsec = "" + ret = '' + a = list_avtx[add] + for array in a: + notnull = True + no_tx = array[0][64:] + array[0] = array[0][:64] + link = "http://blockexplorer.com/"+TN+"/rawtx/"+array[0] + pagetx = urllib.urlopen(link) + ScriptPubkey = str(json.loads(pagetx.read())['out'][int(array[2])]['scriptPubKey']) +# ret += '' + array[0] + "#" + no_tx + ": " + str(array[1]) + " & " + ScriptPubkey + "
    " + ret += '" + ret+="" +# ret += "
    " + "
    " + if notnull is False: + ret = "" + gret += ret + gret=gret[:-len("")] + + return gret+"
    ' + add + "" + hexsec + '
    ' + array[0] + "" + str(array[1]) + "" + ScriptPubkey + "
    " + +ct_txin = [] +ct_txout = [] + + +empty_txin={'hash':'', 'index':'', 'sig':'##', 'pubkey':'', 'oldscript':'', 'addr':''} +empty_txout={'amount':'', 'script':''} + +class tx(): + ins=[] + outs=[] + tosign=False + + def hashtypeone(index,script): + global empty_txin + for i in range(len(ins)): + self.ins[i]=empty_txin + self.ins[index]['pubkey']="" + self.ins[index]['oldscript']=s + self.tosign=True + + def copy(): + r=tx() + r.ins=self.ins[:] + r.outs=self.outs[:] + return r + + def sign(n=-1): + if n==-1: + for i in range(len(ins)): + self.sign(i) + return "done" + + global json_db + txcopy=self.copy() + txcopy.hashtypeone(i, self.ins[n]['oldscript']) + + sec='' + for k in json_db['keys']: + if k['addr']==self.ins[n]['addr'] and 'hexsec' in k: + sec=k['hexsec'] + if sec=='': + print "priv key not found (addr:"+self.ins[n]['addr']+")" + return "" + + self.ins[n]['sig']=sign_message(sec.decode('hex'), txcopy.get_tx(), True) + + def ser(): + r={} + r['ins']=self.ins + r['outs']=self.outs + r['tosign']=self.tosign + return json.dumps(r) + + def unser(r): + s=json.loads(r) + self.ins=s['ins'] + self.outs=s['outs'] + self.tosign=s['tosign'] + + def get_tx(): + r='' + ret += inverse_str("%08x"%1) + ret += "%02x"%len(self.ins) + + for i in range(len(self.ins)): + txin=self.ins[i] + ret += inverse_str(txin['hash']) + ret += inverse_str("%08x"%txin['index']) + + if txin['pubkey']!="": + tmp += "%02x"%(1+len(txin['sig'])/2) + tmp += txin['sig'] + tmp += "01" + tmp += "%02x"%(len(txin['pubkey'])/2) + tmp += txin['pubkey'] + + ret += "%02x"%(len(tmp)/2) + ret += tmp + + elif txin['oldscript']!="": + ret += "%02x"%(len(txin['oldscript'])/2) + ret += txin['oldscript'] + + else: + ret += "00" + + ret += "ffffffff" + + ret += "%02x"%len(self.outs) + + for i in range(len(self.outs)): + txout=self.outs[i] + ret += inverse_str("%016x"%(txout['amount'])) + + if txout['script'][:2]=='s:': #script + script=txout['script'][:2] + ret += "%02x"%(len(script)/2) + ret += script + else: #address + ret += "%02x"%(len(txout['script'])/2+5) + ret += "%02x"%OP_DUP + ret += "%02x"%OP_HASH160 + ret += "%02x"%(len(txout['script'])/2) + ret += txout['script'] + ret += "%02x"%OP_EQUALVERIFY + ret += "%02x"%OP_CHECKSIG + + ret += "00000000" + if not self.tosign: + ret += "01000000" + return ret + + +def listtx_txt(adds): + untx_site="http://blockchain.info/unspent?active=" + ret='' + table="""
    """ + utx=untx_site+adds + try: + utxs=json.loads(urllib.urlopen(utx).read())["unspent_outputs"] + except: + return "No inputs" + + table+="\ + \ + \ + \ + \ + " + for tx in utxs: + txhash=str(tx["tx_hash"]).decode('hex')[::-1].encode('hex') + txn=int(tx["tx_output_n"]) + txscript=str(tx["script"]) + txvalue=int(tx["value"]) + table+="" + table+="" +# table+="" + table+="" + table+="" + if txscript[:6]+txscript[-4:]=="76a91488ac": + table+="" + table+="" + else: + table+="" + + table+="" + table+="\n" + table+="
    UseTx hashScriptAmount
    \ + \ + \ + \ + \ +\ + "+a+""+txhash+"Address "+hash_160_to_bc_address(txscript[6:-4].decode('hex'))+""+txscript+""+str(txvalue/1e8)+"
    " + ret+=table + + ret+="0 BTC (inputs) - 0 BTC (outputs) = 0 BTC (fee)

    " + + txouts="" + nbretxouts=30 + unserouts=["parseFloat(document.getElementById(\"txout_am_"+str(i)+"\").value)" for i in range(nbretxouts)] + serouts="+".join(unserouts) + for i in range(nbretxouts): + txouts+="\ +Amount:     Script: \ +\ +\ +\ +
    "+X_if_else("",i>=2,"")+"
    " + + ret+=txouts + ret+="
    " +# ret+=WI_Submit('Create', '', '', 'ajaxCTX2') + + return ret + if 'twisted' not in missing_dep: class WIInfo(resource.Resource): @@ -3050,6 +3949,171 @@ if 'twisted' not in missing_dep: def render_POST(self, request): return self.render_GET(request) + class WICTListTx(resource.Resource): + def render_GET(self, request): + global CTX_adds + try: + adds=request.args['addresses'][0] + CTX_adds=adds + except: + return "You must provide at least one address to see the transaction you can spend. Divided by |" + + ret="" + ret=listtx_txt(adds) + return "Refresh to display available incoming transactions" + + def render_POST(self, request): + return self.render_GET(request) + + class WICT(resource.Resource): + def render_GET(self, request): + #CT?sec=s&hashesin=h&indexes=1&pubkeys=p&prevspk=r&amounts=2453628&spk=spk#tbend + global txin_amounts, json_db + display = "" + + + + try: + testnet=request.args['testnet'][0] + TN="testnet" + except: + TN="" + + try: + + list_sec, list_hin, list_indexes, list_pubs, list_scriptin, list_outam, list_scriptout, list_amin = [[] for i in range(8)] + + txin_to_use=[] + txouts_nos=[] + txouts_not_empty=[] + for i in request.args: + if i[:4]=='txin' and i[-10:-7]=='use': + txin_to_use.append(i.split('_')[1]) + if i[:4]=='txou' and i.split('_')[2] not in txouts_nos: + p=i.split('_')[1] + no=i.split('_')[2] + txouts_nos.append(no) + + for no in txouts_nos: + if request.args['txout_am_'+no][0]!='' and request.args['txout_am_'+no][0]!='0': + list_outam.append(request.args['txout_am_'+no][0]) + list_scriptout.append(request.args['txout_script_'+no][0]) + + + global addr_to_keys + for h in txin_to_use: + if request.args['txin_'+h+'_add'][0] not in addr_to_keys.keys(): + return "
    No private key for "+request.args['txin_'+h+'_add'][0]+", please dump a wallet containing this address

    Return to Pywallet" + + list_hin.append(h) + list_indexes.append(request.args['txin_'+h+'_n'][0]) + list_scriptin.append(request.args['txin_'+h+'_script'][0]) + list_sec.append(addr_to_keys[request.args['txin_'+h+'_add'][0]][0]) + list_pubs.append(addr_to_keys[request.args['txin_'+h+'_add'][0]][1]) + list_amin.append(request.args['txin_'+h+'_amin'][0]) + + sec=",".join(list_sec) + hashesin=",".join(list_hin) + indexes=",".join(list_indexes) + pubkeys=",".join(list_pubs) + prevspk=",".join(list_scriptin) + amins=",".join(list_amin) + amounts=",".join(list_outam) + spk=",".join(list_scriptout) + + + except: + display += "error" + return display + + secret_key = sec.split(',') + hashes_txin = hashesin.split(',') + indexes_txin = indexes.split(',') + for i in range(len(indexes_txin)): + indexes_txin[i] = int(indexes_txin[i]) + pubkey_txin = pubkeys.split(',') + am_txin = amins.split(',') + prevScriptPubKey = prevspk.split(',') + + amounts_txout = amounts.split(',') + for i in range(len(amounts_txout)): + amounts_txout[i] = int(1e8*float(amounts_txout[i])) + spk_txout = spk.split(',') + + tx = create_transaction(secret_key, hashes_txin, indexes_txin, pubkey_txin, prevScriptPubKey, amounts_txout, spk_txout) + + display += "Inputs: (go to CTTest before)
    " + sum_in = 0 + for i in range(len(hashes_txin)): + try: +# ain = txin_amounts[hashes_txin[i] + ("%d"%indexes_txin[i])] + ain=int(am_txin[i])/1e8 + aaa = ", %.8f BTC"%ain + sum_in += ain + except: + aaa = "" + display += ('%d: %s #%d%s
    ')%(i, hashes_txin[i], indexes_txin[i], hashes_txin[i], indexes_txin[i], aaa) + + display += "

    Outputs:
    " + sum_out = 0 + for i in range(len(spk_txout)): + sum_out += amounts_txout[i]/BTC + display += '%.8f BTC to %s
    '%(amounts_txout[i]/BTC, hash_160_to_bc_address(spk_txout[i].decode('hex'))) + + display += "
    " + display += "
    " + display += "In: %.8f BTC"%sum_in + display += "
    " + display += "Out: %.8f BTC"%sum_out + display += "
    " + display += "Fee: %.8f BTC"%(sum_in-sum_out) + display += "
    " + display += "
    " + display += "
    " + display += ("
    Transaction hash: "+tx[0])
    +			display += "
    " +# display += ("tx_k: "+tx[1]) +# display += "
    " + display += ("Raw transaction: "+tx[2]) + + display += "

    " + + + return display + + def render_POST(self, request): + return self.render_GET(request) + + class WICTTest(resource.Resource): + + def render_GET(self, request): + try: + request.args['testnet'][0] + testnet=True + except: + testnet=False + list_avtx = {} + i = 0 + + try: + for add in request.args['addresses'][0].split(','): + print "Address %d: %s"%(i, add) + list_avtx[add] = bc_address_to_available_tx(add, testnet)[add] + i += 1 + + print(list_avtx) + + display = "" + display += write_avtx(list_avtx, testnet) + except: + display="You must provide at least one address to see the transaction you can spend.
    Like this" + + return display + + def render_POST(self, request): + return self.render_GET(request) + + class WIImport(resource.Resource): def render_GET(self, request): @@ -3149,11 +4213,26 @@ if 'twisted' not in missing_dep: def update_pyw(): if md5_last_pywallet[0] and md5_last_pywallet[1] not in md5_pywallet: - filout = open(pyw_filename, 'w') - filout.write(urllib.urlopen('https://raw.github.com/jackjack-jj/pywallet/master/pywallet.py').read()) - filout.close() - reactor.stop() + dl=urllib.urlopen('https://raw.github.com/jackjack-jj/pywallet/master/pywallet.py').read() + if len(dl)>40 and md5_2(dl)==md5_last_pywallet[1]: + filout = open(pyw_path+"/"+pyw_filename, 'w') + filout.write(dl) + filout.close() + thread.start_new_thread(restart_pywallet, ()) + return "Updated, please wait..." + else: + return "Problem when downloading new version ("+md5_2(dl)+"/"+md5_last_pywallet[1]+")" +def restart_pywallet(): + time.sleep(1) + thread.start_new_thread(start_pywallet, ()) + time.sleep(0.3) + reactor.stop() + +def start_pywallet(): + a=Popen("python "+pyw_path+"/"+pyw_filename+" --web --port "+str(webport)+" --wait 2", shell=True, bufsize=-1, stdout=PIPE).stdout + a.close() + import thread @@ -3163,15 +4242,11 @@ def retrieve_last_pywallet_md5(): global md5_last_pywallet md5_last_pywallet = [True, md5_onlinefile('https://raw.github.com/jackjack-jj/pywallet/master/pywallet.py')] -def zzz(a): - global pywversion - pywversion=a - from optparse import OptionParser if __name__ == '__main__': - md5_pywallet = md5_file(pyw_filename) + md5_pywallet = md5_file(pyw_path+"/"+pyw_filename) thread.start_new_thread(retrieve_last_pywallet_md5, ()) @@ -3243,6 +4318,8 @@ if __name__ == '__main__': parser.add_option("--dont_check_walletversion", dest="dcv", action="store_true", help="don't check if wallet version > %d before running (WARNING: this may break your wallet, be sure you know what you do)"%max_version) + parser.add_option("--wait", dest="nseconds", + help="wait NSECONDS seconds before launch") # parser.add_option("--forcerun", dest="forcerun", @@ -3260,6 +4337,9 @@ if __name__ == '__main__': # if options.forcerun is None: # exit(0) + if options.nseconds: + time.sleep(int(options.nseconds)) + if options.passphrase: passphrase = options.passphrase @@ -3315,7 +4395,13 @@ if __name__ == '__main__': 'Delete': WIDelete(), 'Balance': WIBalance(), 'ChangePP': WIChangePP(), - 'Others': WIOthers() + 'Others': WIOthers(), + 'LoadBalances': WICTTest(), + 'CTTest': WICTTest(), + 'ListTransactions': WICTListTx(), + 'CreateTransaction': WICT(), + 'CT': WICT() + } if options.dcv is not None: