view src/puppy/block.py @ 8:4d259e84160d

fix OverFlow bug
author Dennis Concepcion Martin <dennisconcepcionmartin@gmail.com>
date Sun, 24 Oct 2021 17:38:23 +0200
parents
children e218f70e19e9
line wrap: on
line source

from puppy.block_structure import *
from puppy.helpers import get_var_int
from puppy.helpers import compute_hash

"""
Deserialize methods
"""


def deserialize_block(f):
    """
    Deserialize block
    :param f: buffer, required
    :return: dict
    """

    block = Block()
    block.magic_number = f.read(4).hex()
    block.block_size = f.read(4)[::-1].hex()
    block_header, block.block_hash = deserialize_header(f)
    block.number_of_transactions = get_var_int(f)

    transactions = []
    for transaction_number in range(int(block.number_of_transactions, 16)):
        transactions.append(deserialize_transaction_data(f, block.block_size))

    block_dict = block.__dict__
    block_dict['header'] = block_header
    block_dict['transactions'] = transactions

    return block_dict


def deserialize_header(f):
    """
    Deserialize block header
    More info: https://learnmeabitcoin.com/technical/block-header
    :param f: buffer, required
    :return: (dict, string)
    """

    # Compute block hash
    before = f.tell()
    header = f.read(80)
    block_hash = compute_hash(header)
    f.seek(before)

    header = Header()
    header.version = f.read(4)[::-1].hex()
    header.previous_block_hash = f.read(32)[::-1].hex()
    header.merkle_root = f.read(32)[::-1].hex()
    header.time = f.read(4)[::-1].hex()
    header.bits = f.read(4)[::-1].hex()
    header.nonce = f.read(4)[::-1].hex()

    return header.__dict__, block_hash


def deserialize_transaction_data(f, block_size):
    """
    Deserialize transaction data
    More info: https://learnmeabitcoin.com/technical/transaction-data
    :param f: buffer, required
    :param block_size: string, required
    :return: dict
    """

    start_transaction_data = f.tell()

    transaction = Transaction()
    transaction.version = f.read(4)[::-1].hex()
    transaction.number_of_inputs = get_var_int(f)

    transaction_inputs = []
    for input_number in range(int(transaction.number_of_inputs, 16)):
        transaction_input = TransactionInput()
        transaction_input.id = f.read(32)[::-1].hex()
        transaction_input.vout = f.read(4)[::-1].hex()
        transaction_input.script_sig_size = get_var_int(f)
        transaction_input.script_sig = f.read(int(transaction_input.script_sig_size, 16)).hex()
        transaction_input.sequence = f.read(4)[::-1].hex()
        transaction_inputs.append(transaction_input.__dict__)

    transaction.number_of_outputs = get_var_int(f)

    transaction_outputs = []
    for output_number in range(int(transaction.number_of_outputs, 16)):
        transaction_output = TransactionOutput()
        transaction_output.value = f.read(8)[::-1].hex()
        transaction_output.script_pub_key_size = get_var_int(f)
        transaction_output.script_pub_key = f.read(int(transaction_output.script_pub_key_size, 16)).hex()
        transaction_outputs.append(transaction_output.__dict__)

    transaction.lock_time = f.read(4)[::-1].hex()

    # Compute transaction id
    end_transaction_data = f.tell()
    transaction_data_size = end_transaction_data - start_transaction_data
    f.seek(start_transaction_data)
    transaction_data = f.read(transaction_data_size)
    f.seek(end_transaction_data)
    transaction.id = compute_hash(transaction_data)

    transaction_dict = transaction.__dict__
    transaction_dict['inputs'] = transaction_inputs
    transaction_dict['outputs'] = transaction_outputs

    return transaction_dict