¿Cómo convertir bytes de scriptPubKey a una dirección de Bitcoin?

Actualmente estoy intentando analizar la cadena de bloques usando pyblockchain . Mi problema es que no puedo codificar scriptPubKeycorrectamente, aunque no sé qué podría hacer mal.

A continuación puedes ver cómo atravieso la cadena de bloques:

from blockchain.reader import BlockchainFileReader

import hashlib
import base58

block_reader = BlockchainFileReader('/media/Data/btc/blocks/blk00325.dat')

count = 0

for block in block_reader:    
    count +=1        
    for t in block.transactions:
        for outp in t.outputs:
            addr = base58.b58encode(outp.script_pub_key)
            if addr.startswith('1'):
                print(addr)        
    if count >= 5:
        break

Si yo, en mi cuaderno de Jupyter, echo un vistazo a outp, encontraré por

outp.script_pub_key
>> b'v\xa9\x14\x1e\xbev\x83\xceJd\xad\xc9\x17\xe9\xb1\x93\x7f\x12&Q\xcb\xab\xa1\x88\xac'

este:

base58.b58encode(outp.script_pub_key)
>> 'pkJBVCg6k54E7ZiP7cvxbCvtN9aY9zEcgK'

y esta no es una dirección de bitcoin válida .

Aparentemente, se supone que las direcciones de bitcoin están codificadas en Base58Check ; sin embargo, eso tampoco funciona:

base58.b58encode_check(outp.script_pub_key)
>> '6PSJQapdQn8VeG9SBuZdH8q2ysyP4ND6dmspzLZb'

Entonces, ¿qué estoy haciendo mal aquí?

Respuestas (1)

No está considerando los códigos de operación en la cadena pub_key. Por ejemplo, una script_pub_key podría ser OP_DUP OP_HASH160 [pub_key] ... y debe extraer pub_key.
Este código de muestra muestra las direcciones, puede compararlas con un explorador de bloques para obtener la misma identificación de transacción. Tenga en cuenta que el byte después de OP_HASH160 da la longitud de pub_key en bytes después, por lo que también debe omitirse para generar la dirección.


import sys
import base58
import hashlib
import binascii
from blockchain.reader import BlockchainFileReader
block_reader=BlockchainFileReader('/var/data/bitcoin-data/blocks/blk00325.dat')
satochi_convert=1e8
def sha256(x):
    h=hashlib.new('sha256')
    h.update(x)
    return h.digest()

def hashStr(buffer): return binascii.hexlify(buffer)

for block in block_reader: #block has .header, .transactions for iter,tx in enumerate(block.transactions): print(' Transaction (txn_hash):{} {}'.format(type(tx.txn_hash),tx.txn_hash)) for x in tx.outputs: script_pub_key_str=hashStr(x.script_pub_key) if script_pub_key_str[0:4]==b'76a9': #This is a pubkeyhash bytes=x.script_pub_key[2] #number of bytes in the pub_key assert bytes==20 public_key = x.script_pub_key[3:3+bytes] #20 bytes z=b'\00'+public_key #checksum=sha256(sha256(z))[:4] #address1=base58.b58encode(z + checksum) address2=base58.b58encode_check(z) # adds checksum for you print(' output value {:<20} address {}'.format(float(x.value)/satochi_convert,address2)) elif script_pub_key_str[0:2]==b'a9': #this is a scripthash (pay-to-script address) bytes=x.script_pub_key[1] #number of bytes in the pub_key assert bytes==20 public_key = x.script_pub_key[2:2+bytes] #20 bytes z=b'\05'+public_key #used for mainnet address2=base58.b58encode_check(z) print(' output value {:<20} address {}'.format(float(x.value)/satochi_convert,address2)) else: print(' output value {:<20} other {}'.format(float(x.value)/satochi_convert,script_pub_key_str)) print()

La salida de muestra para la primera transacción en su archivo de bloque es:

 Transaction (txn_hash): cc728403552d5e1fddf06e6a7e8552353f315be6c1a43a8e64e4d11b081d4ca3
  output value 25.17686501          address 1N6LrEDiHuFwSyJYj2GedZM2FGk7kkLjn
  output value 0.22864963           address 1Hr9uwzwXWpjQDNUWdZ1i9qnoSpnniJe4U
....

Códigos OP aquí: https://en.bitcoin.it/wiki/Script , información de scriptpubkey: https://en.bitcoin.it/wiki/Transaction NOTA: este es solo un ejemplo de cómo calcular direcciones, no lo es muy robusto ya que no verifica otros códigos OP después, otros posibles scripts de pago, etc.