You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
litecoin/test/functional/mweb_p2p.py

144 lines
6.0 KiB

#!/usr/bin/env python3
# Copyright (c) 2020 The Bitcoin Core developers
# Distributed under the MIT software license, see the accompanying
# file COPYING or http://www.opensource.org/licenses/mit-license.php.
"""
Test LIP-0006
1. Test getdata 'mwebheader' *before* MWEB activation
2. Test getdata 'mwebheader' *after* MWEB activation
3. Test getdata 'mwebleafset' *before* MWEB activation
4. Test getdata 'mwebleafset' *after* MWEB activation
- Request from earlier block (not tip) to make sure rewind works
"""
from test_framework.messages import (
CBlockHeader,
CInv,
Hash,
hash256,
msg_getdata,
MSG_MWEB_HEADER,
MSG_MWEB_LEAFSET,
)
from test_framework.p2p import P2PInterface, p2p_lock
from test_framework.script import MAX_SCRIPT_ELEMENT_SIZE
from test_framework.test_framework import BitcoinTestFramework
from test_framework.ltc_util import FIRST_MWEB_HEIGHT, get_hogex_tx, get_mweb_header
from test_framework.util import assert_equal
# Can be used to mimic a light client requesting MWEB data from a full node
class MockLightClient(P2PInterface):
def __init__(self):
super().__init__()
self.merkle_blocks_with_mweb = {}
self.block_headers = {}
self.leafsets = {}
def request_mweb_header(self, block_hash):
want = msg_getdata([CInv(MSG_MWEB_HEADER, int(block_hash, 16))])
self.send_message(want)
def on_mwebheader(self, message):
self.merkle_blocks_with_mweb[message.header_hash()] = message.merkleblockwithmweb
def request_mweb_leafset(self, block_hash):
want = msg_getdata([CInv(MSG_MWEB_LEAFSET, int(block_hash, 16))])
self.send_message(want)
def on_mwebleafset(self, message):
self.leafsets[message.block_hash] = message.leafset
def on_block(self, message):
message.block.calc_sha256()
self.block_headers[Hash(message.block.sha256)] = CBlockHeader(message.block)
class MWEBP2PTest(BitcoinTestFramework):
def set_test_params(self):
self.setup_clean_chain = True
self.extra_args = [['-whitelist=noban@127.0.0.1']] # immediate tx relay
self.num_nodes = 1
def assert_mweb_header(self, node, light_client, post_mweb_block_hash):
assert post_mweb_block_hash in light_client.merkle_blocks_with_mweb
merkle_block_with_mweb = light_client.merkle_blocks_with_mweb[post_mweb_block_hash]
# Check block header is correct
assert Hash.from_hex(post_mweb_block_hash) in light_client.block_headers
block_header = light_client.block_headers[Hash.from_hex(post_mweb_block_hash)]
assert_equal(block_header, merkle_block_with_mweb.merkle.header)
# Check MWEB header is correct
mweb_header = get_mweb_header(node, post_mweb_block_hash)
assert_equal(mweb_header, merkle_block_with_mweb.mweb_header)
# Check HogEx transaction is correct
hogex_tx = get_hogex_tx(node, post_mweb_block_hash)
assert_equal(hogex_tx, merkle_block_with_mweb.hogex)
# Check Merkle tree
merkle_tree = merkle_block_with_mweb.merkle.txn
assert_equal(3, merkle_tree.nTransactions)
assert_equal(2, len(merkle_tree.vHash))
left_hash = Hash(merkle_tree.vHash[0])
right_hash = Hash(merkle_tree.vHash[1])
assert_equal(Hash.from_hex(hogex_tx.hash), right_hash)
right_branch_bytes = hash256(right_hash.serialize() + right_hash.serialize())
merkle_root_bytes = hash256(left_hash.serialize() + right_branch_bytes)
assert_equal(Hash.from_byte_arr(merkle_root_bytes), Hash(block_header.hashMerkleRoot))
def run_test(self):
node = self.nodes[0]
light_client = node.add_p2p_connection(MockLightClient())
self.log.info("Generate pre-MWEB blocks")
pre_mweb_block_hash = node.generate(FIRST_MWEB_HEIGHT - 1)[-1]
self.log.info("Request 'mwebheader' and 'mwebleafset' for pre-MWEB block '{}'".format(pre_mweb_block_hash))
light_client.request_mweb_header(pre_mweb_block_hash)
light_client.request_mweb_leafset(pre_mweb_block_hash)
self.log.info("Activate MWEB")
node.sendtoaddress(node.getnewaddress(address_type='mweb'), 1)
post_mweb_block_hash = node.generate(1)[0]
self.log.info("Waiting for mweb_header")
light_client.wait_for_block(int(post_mweb_block_hash, 16), 5)
self.log.info("Pegin some additional coins")
node.sendtoaddress(node.getnewaddress(address_type='mweb'), 10)
post_mweb_block_hash2 = node.generate(1)[0]
light_client.wait_for_block(int(post_mweb_block_hash2, 16), 5)
self.log.info("Request 'mwebheader' and 'mwebleafset' for block '{}'".format(post_mweb_block_hash))
light_client.request_mweb_header(post_mweb_block_hash)
light_client.request_mweb_leafset(post_mweb_block_hash)
self.log.info("Waiting for 'mwebheader' and 'mwebleafset'")
light_client.wait_for_mwebheader(post_mweb_block_hash, 5)
light_client.wait_for_mwebleafset(post_mweb_block_hash, 5)
self.log.info("Assert results")
# Before MWEB activation, no merkle block should be returned
assert pre_mweb_block_hash not in light_client.merkle_blocks_with_mweb
# After MWEB activation, the requested merkle block should be returned
self.assert_mweb_header(node, light_client, post_mweb_block_hash)
# Before MWEB activation, no leafset should be returned
assert pre_mweb_block_hash not in light_client.leafsets
# After MWEB activation, the leafset should be returned
# Only 2 outputs should be in the UTXO set (the pegin and its change)
# That's '11' and then padded to the right with 0's, then serialized in big endian.
# So we expect the serialized leafset to be 0b11000000 or 0xc0
assert Hash.from_hex(post_mweb_block_hash) in light_client.leafsets
leafset = light_client.leafsets[Hash.from_hex(post_mweb_block_hash)]
assert_equal([0xc0], leafset)
if __name__ == '__main__':
MWEBP2PTest().main()