Skip to content

Commit

Permalink
Adding cache hints for outpoint txs
Browse files Browse the repository at this point in the history
  • Loading branch information
Gerson2102 committed Sep 6, 2024
1 parent 6223152 commit 5e50e80
Showing 1 changed file with 94 additions and 85 deletions.
179 changes: 94 additions & 85 deletions scripts/data/generate_data.py
Original file line number Diff line number Diff line change
Expand Up @@ -13,24 +13,24 @@

def request_rpc(method: str, params: list):
"""Makes a JSON-RPC call to a Bitcoin API endpoint.
Uses environment variables BITCOIN_RPC and USERPWD
Uses environment variables BITCOIN_RPC and USERPWD
or the default public endpoint if those variables are not set.
:return: parsed JSON result as Python object
"""
url = BITCOIN_RPC or DEFAULT_URL
auth = USERPWD.split(":") if USERPWD else None
headers = {'content-type': 'application/json'}
headers = {"content-type": "application/json"}
payload = {
"jsonrpc": "2.0",
"method": method,
"params": params,
"id": 0,
}
data = requests.post(url, auth=auth, headers=headers, json=payload).json()
if data['result'] is None:
if data["result"] is None:
raise ConnectionError("RPC response is null")
return data['result']
return data["result"]


def fetch_chain_state(block_height: int):
Expand All @@ -45,20 +45,22 @@ def fetch_chain_state(block_height: int):

# In order to init prev_timestamps we need to query 10 previous headers
prev_header = head
prev_timestamps = [head['time']]
prev_timestamps = [head["time"]]
for _ in range(10):
if prev_header['height'] == 0:
if prev_header["height"] == 0:
prev_timestamps.insert(0, 0)
else:
prev_header = request_rpc("getblockheader", [prev_header['previousblockhash']])
prev_timestamps.insert(0, prev_header['time'])
head['prev_timestamps'] = prev_timestamps
prev_header = request_rpc(
"getblockheader", [prev_header["previousblockhash"]]
)
prev_timestamps.insert(0, prev_header["time"])
head["prev_timestamps"] = prev_timestamps

# In order to init epoch start we need to query block header at epoch start
if block_height < 2016:
head['epoch_start_time'] = 1231006505
head["epoch_start_time"] = 1231006505
else:
head['epoch_start_time'] = get_epoch_start_time(block_height)
head["epoch_start_time"] = get_epoch_start_time(block_height)

return head

Expand All @@ -67,41 +69,39 @@ def next_chain_state(head: dict, blocks: list):
"""Computes resulting chain state given the initial chain state
and all blocks that were applied to it.
"""
block_height = head['height'] + len(blocks)
block_height = head["height"] + len(blocks)
next_head = blocks[-1]

# We need to recalculate the prev_timestamps field given the previous chain state
# and all the blocks we applied to it
prev_timestamps = head['prev_timestamps'] + list(map(lambda x: x['time'], blocks))
next_head['prev_timestamps'] = prev_timestamps[-11:]
prev_timestamps = head["prev_timestamps"] + list(map(lambda x: x["time"], blocks))
next_head["prev_timestamps"] = prev_timestamps[-11:]

# Update epoch start time if neccesary
if head['height'] // 2016 != block_height // 2016:
next_head['epoch_start_time'] = get_epoch_start_time(block_height)
if head["height"] // 2016 != block_height // 2016:
next_head["epoch_start_time"] = get_epoch_start_time(block_height)
else:
next_head['epoch_start_time'] = head['epoch_start_time']
next_head["epoch_start_time"] = head["epoch_start_time"]
return next_head


def get_epoch_start_time(block_height: int) -> int:
"""Computes the corresponding epoch start time given the current block height.
"""
"""Computes the corresponding epoch start time given the current block height."""
epoch_start_block_height = (block_height // 2016) * 2016
epoch_start_block_hash = request_rpc("getblockhash", [epoch_start_block_height])
epoch_start_header = request_rpc("getblockheader", [epoch_start_block_hash])
return epoch_start_header['time']
return epoch_start_header["time"]


def format_chain_state(head: dict):
"""Formats chain state according to the respective Cairo type.
"""
"""Formats chain state according to the respective Cairo type."""
return {
"block_height": head['height'],
"total_work": str(int.from_bytes(bytes.fromhex(head['chainwork']), 'big')),
"best_block_hash": head['hash'],
"current_target": str(bits_to_target(head['bits'])),
"epoch_start_time": head['epoch_start_time'],
"prev_timestamps": head['prev_timestamps'],
"block_height": head["height"],
"total_work": str(int.from_bytes(bytes.fromhex(head["chainwork"]), "big")),
"best_block_hash": head["hash"],
"current_target": str(bits_to_target(head["bits"])),
"epoch_start_time": head["epoch_start_time"],
"prev_timestamps": head["prev_timestamps"],
}


Expand All @@ -111,79 +111,74 @@ def bits_to_target(bits: str) -> int:
:param bits: bits as a hex string (without 0x prefix)
:return: target as integer
"""
exponent = int.from_bytes(bytes.fromhex(bits[:2]), 'big')
mantissa = int.from_bytes(bytes.fromhex(bits[2:]), 'big')
exponent = int.from_bytes(bytes.fromhex(bits[:2]), "big")
mantissa = int.from_bytes(bytes.fromhex(bits[2:]), "big")
if exponent == 0:
return mantissa
elif exponent <=3:
elif exponent <= 3:
return mantissa >> (8 * (3 - exponent))
else:
return mantissa << (8 * (exponent - 3))


def fetch_block(block_hash: str):
"""Downloads block with transactions (and referred UTXOs) from RPC given the block hash.
"""
"""Downloads block with transactions (and referred UTXOs) from RPC given the block hash."""
block = request_rpc("getblock", [block_hash, 2])
block['data'] = [resolve_transaction(tx) for tx in block['tx']]
block["data"] = [resolve_transaction(tx) for tx in block["tx"]]
return block


def resolve_transaction(transaction: dict):
"""
"""
""" """
return {
"version": transaction['version'],
"version": transaction["version"],
# Skip the first 4 bytes (version) and take the next 4 bytes (marker + flag)
"is_segwit": transaction["hex"][8:12] == "0001",
"lock_time": transaction['locktime'],
"inputs": [resolve_input(input) for input in transaction['vin']],
"outputs": [format_output(output) for output in transaction['vout']],
"lock_time": transaction["locktime"],
"inputs": [resolve_input(input) for input in transaction["vin"]],
"outputs": [format_output(output) for output in transaction["vout"]],
}


def resolve_input(input: dict):
"""
"""
if input.get('coinbase'):
""" """
if input.get("coinbase"):
return format_coinbase_input(input)
else:
return {
"script": f'0x{input["scriptSig"]["hex"]}',
"sequence": input['sequence'],
"witness": [f'0x{item}' for item in input.get('txinwitness', [])],
"sequence": input["sequence"],
"witness": [f"0x{item}" for item in input.get("txinwitness", [])],
"previous_output": resolve_outpoint(input),
}


def resolve_outpoint(input: dict):
"""
"""
tx = request_rpc("getrawtransaction", [input['txid'], True])
block = request_rpc("getblockheader", [tx['blockhash']])
""" """
tx = request_rpc("getrawtransaction", [input["txid"], True])
block = request_rpc("getblockheader", [tx["blockhash"]])
return {
"txid": input['txid'],
"vout": input['vout'],
"data": format_output(tx['vout'][input['vout']]),
"block_height": block['height'],
"block_time": block['time'],
"is_coinbase": tx['vin'][0].get('coinbase') is not None
"txid": input["txid"],
"vout": input["vout"],
"data": format_output(tx["vout"][input["vout"]]),
"block_height": block["height"],
"block_time": block["time"],
"is_coinbase": tx["vin"][0].get("coinbase") is not None,
}


def format_coinbase_input(input: dict):
"""
"""
""" """
return {
"script": f'0x{input["coinbase"]}',
"sequence": input["sequence"],
"witness": [],
"previous_output": {
"txid": "0" * 64,
"vout": 0xffffffff,
"vout": 0xFFFFFFFF,
"data": {
"value": 0,
"pk_script": '0x',
"pk_script": "0x",
"cached": False,
},
"block_height": 0,
Expand All @@ -194,8 +189,7 @@ def format_coinbase_input(input: dict):


def format_output(output: dict):
"""
"""
""" """
return {
"value": int(output["value"] * 100000000),
"pk_script": f'0x{output["scriptPubKey"]["hex"]}',
Expand All @@ -205,20 +199,18 @@ def format_output(output: dict):


def format_block_with_transactions(block: dict):
"""Formats block with transactions according to the respective Cairo type.
"""
"""Formats block with transactions according to the respective Cairo type."""
return {
"header": format_header(block),
"data": {
"variant_id": 1,
"transactions": block['data'], # already formatted
}
"transactions": block["data"], # already formatted
},
}


def fetch_block_header(block_hash: str):
"""Downloads block header (without trasnasction) from RPC given the block hash.
"""
"""Downloads block header (without trasnasction) from RPC given the block hash."""
return request_rpc("getblockheader", [block_hash])


Expand All @@ -231,10 +223,7 @@ def format_block(header: dict):
"""
return {
"header": format_header(header),
"data": {
"variant_id": 0,
"merkle_root": header['merkleroot']
}
"data": {"variant_id": 0, "merkle_root": header["merkleroot"]},
}


Expand All @@ -244,53 +233,73 @@ def format_header(header: dict):
:param header: block header obtained from RPC
"""
return {
"version": header['version'],
"time": header['time'],
"bits": int.from_bytes(bytes.fromhex(header['bits']), 'big'),
"nonce": header['nonce']
"version": header["version"],
"time": header["time"],
"bits": int.from_bytes(bytes.fromhex(header["bits"]), "big"),
"nonce": header["nonce"],
}


def generate_data(mode: str, initial_height: int, num_blocks: int, include_expected: bool):
def generate_data(
mode: str, initial_height: int, num_blocks: int, include_expected: bool
):
"""Generates arguments for Raito program in a human readable form and the expected result.
:param mode: Validation mode:
:param mode: Validation mode:
"light" — generate block headers with Merkle root only
"full" — generate full blocks with transactions (and referenced UTXOs)
:param initial_height: The block height of the initial chain state (0 means the state after genesis)
:param num_blocks: The number of blocks to apply on top of it (has to be at least 1)
:return: tuple (arguments, expected output)
"""
chain_state = fetch_chain_state(initial_height)
next_block_hash = chain_state['nextblockhash']
next_block_hash = chain_state["nextblockhash"]
blocks = []

# UTXO set to track unspent outputs
utxo_set = {}

for _ in range(num_blocks):
if mode == 'light':
if mode == "light":
block = fetch_block_header(next_block_hash)
elif mode == 'full':
elif mode == "full":
block = fetch_block(next_block_hash)
for tx in block["data"]:
for idx, output in enumerate(tx["outputs"]):
outpoint = (tx["txid"], idx)
utxo_set[outpoint] = output

for tx_input in tx["inputs"]:
outpoint = (
tx_input["previous_output"]["txid"],
tx_input["previous_output"]["vout"],
)
if outpoint in utxo_set:
utxo_set[outpoint]["cached"] = True

else:
raise NotImplementedError(mode)
next_block_hash = block['nextblockhash']
next_block_hash = block["nextblockhash"]
blocks.append(block)

block_formatter = format_block if mode == 'light' else format_block_with_transactions
block_formatter = (
format_block if mode == "light" else format_block_with_transactions
)
result = {
"chain_state": format_chain_state(chain_state),
"blocks": list(map(block_formatter, blocks)),
}

if include_expected:
result['expected'] = format_chain_state(next_chain_state(chain_state, blocks))
result["expected"] = format_chain_state(next_chain_state(chain_state, blocks))

return result


# Usage: generate_data.py MODE INITIAL_HEIGHT NUM_BLOCKS INCLUDE_EXPECTED OUTPUT_FILE
# Example: generate_data.py 'light' 0 10 false light_0_10.json
if __name__ == '__main__':
if (len(sys.argv) != 6):
if __name__ == "__main__":
if len(sys.argv) != 6:
raise TypeError("Expected five arguments")

data = generate_data(
Expand Down

0 comments on commit 5e50e80

Please sign in to comment.