lbry-sdk/tests/integration/blockchain/test_blockchain_reorganization.py

234 lines
10 KiB
Python
Raw Permalink Normal View History

2018-11-04 06:55:50 +01:00
import logging
2020-04-01 19:52:14 +02:00
import asyncio
2020-04-01 21:26:35 +02:00
from binascii import hexlify
2020-03-31 16:30:36 +02:00
from lbry.testcase import CommandTestCase
2020-03-31 16:30:36 +02:00
class BlockchainReorganizationTests(CommandTestCase):
2018-11-04 06:55:50 +01:00
VERBOSITY = logging.WARN
async def assertBlockHash(self, height):
bp = self.conductor.spv_node.writer
2022-05-09 17:34:28 +02:00
reader = self.conductor.spv_node.server
2020-11-18 20:38:15 +01:00
def get_txids():
return [
2022-05-09 17:34:28 +02:00
reader.db.fs_tx_hash(tx_num)[0][::-1].hex()
2020-11-18 20:38:15 +01:00
for tx_num in range(bp.db.tx_counts[height - 1], bp.db.tx_counts[height])
]
block_hash = await self.blockchain.get_block_hash(height)
self.assertEqual(block_hash, (await self.ledger.headers.hash(height)).decode())
2022-05-09 17:34:28 +02:00
self.assertEqual(block_hash, (await reader.db.fs_block_hashes(height, 1))[0][::-1].hex())
2020-11-18 20:38:15 +01:00
2021-07-22 23:33:54 +02:00
txids = await asyncio.get_event_loop().run_in_executor(None, get_txids)
2022-05-09 17:34:28 +02:00
txs = await reader.db.get_transactions_and_merkles(txids)
2020-11-18 20:38:15 +01:00
block_txs = (await bp.daemon.deserialised_block(block_hash))['tx']
self.assertSetEqual(set(block_txs), set(txs.keys()), msg='leveldb/lbrycrd is missing transactions')
self.assertListEqual(block_txs, list(txs.keys()), msg='leveldb/lbrycrd transactions are of order')
2018-09-26 03:06:22 +02:00
async def test_reorg(self):
bp = self.conductor.spv_node.writer
2020-04-24 03:17:44 +02:00
bp.reorg_count_metric.set(0)
2019-06-18 23:50:41 +02:00
# invalidate current block, move forward 2
2020-11-18 20:38:15 +01:00
height = 206
self.assertEqual(self.ledger.headers.height, height)
await self.assertBlockHash(height)
block_hash = (await self.ledger.headers.hash(206)).decode()
await self.blockchain.invalidate_block(block_hash)
2019-06-18 23:50:41 +02:00
await self.blockchain.generate(2)
await asyncio.wait_for(self.on_header(207), 3.0)
2020-03-31 16:30:36 +02:00
self.assertEqual(self.ledger.headers.height, 207)
await self.assertBlockHash(206)
await self.assertBlockHash(207)
2020-04-24 03:17:44 +02:00
self.assertEqual(1, bp.reorg_count_metric._samples()[0][2])
2018-09-26 03:06:22 +02:00
2019-06-18 23:50:41 +02:00
# invalidate current block, move forward 3
2020-03-31 16:30:36 +02:00
await self.blockchain.invalidate_block((await self.ledger.headers.hash(206)).decode())
2019-06-18 23:50:41 +02:00
await self.blockchain.generate(3)
await asyncio.wait_for(self.on_header(208), 3.0)
2020-03-31 16:30:36 +02:00
self.assertEqual(self.ledger.headers.height, 208)
await self.assertBlockHash(206)
await self.assertBlockHash(207)
await self.assertBlockHash(208)
2020-04-24 03:17:44 +02:00
self.assertEqual(2, bp.reorg_count_metric._samples()[0][2])
2020-11-18 20:38:15 +01:00
await self.blockchain.generate(3)
await asyncio.wait_for(self.on_header(211), 3.0)
2020-11-18 20:38:15 +01:00
await self.assertBlockHash(209)
await self.assertBlockHash(210)
await self.assertBlockHash(211)
2021-03-10 02:15:50 +01:00
still_valid = await self.daemon.jsonrpc_stream_create(
'still-valid', '1.0', file_path=self.create_upload_file(data=b'hi!')
)
await self.ledger.wait(still_valid)
await self.blockchain.generate(1)
await asyncio.wait_for(self.on_header(212), 1.0)
2021-03-10 02:15:50 +01:00
claim_id = still_valid.outputs[0].claim_id
c1 = (await self.resolve(f'still-valid#{claim_id}'))['claim_id']
c2 = (await self.resolve(f'still-valid#{claim_id[:2]}'))['claim_id']
c3 = (await self.resolve(f'still-valid'))['claim_id']
self.assertTrue(c1 == c2 == c3)
abandon_tx = await self.daemon.jsonrpc_stream_abandon(claim_id=claim_id)
await self.blockchain.generate(1)
await asyncio.wait_for(self.on_header(213), 1.0)
2021-03-10 02:15:50 +01:00
c1 = await self.resolve(f'still-valid#{still_valid.outputs[0].claim_id}')
c2 = await self.daemon.jsonrpc_resolve([f'still-valid#{claim_id[:2]}'])
c3 = await self.daemon.jsonrpc_resolve([f'still-valid'])
2020-04-01 19:52:14 +02:00
2020-04-01 21:26:35 +02:00
async def test_reorg_change_claim_height(self):
2020-04-01 19:52:14 +02:00
# sanity check
2021-03-10 02:15:50 +01:00
result = await self.resolve('hovercraft') # TODO: do these for claim_search and resolve both
self.assertIn('error', result)
2020-04-01 19:52:14 +02:00
2020-04-05 22:58:36 +02:00
still_valid = await self.daemon.jsonrpc_stream_create(
'still-valid', '1.0', file_path=self.create_upload_file(data=b'hi!')
)
await self.ledger.wait(still_valid)
await self.generate(1)
2020-04-01 21:11:45 +02:00
# create a claim and verify it's returned by claim_search
2021-05-05 22:19:23 +02:00
self.assertEqual(self.ledger.headers.height, 207)
await self.assertBlockHash(207)
broadcast_tx = await self.daemon.jsonrpc_stream_create(
'hovercraft', '1.0', file_path=self.create_upload_file(data=b'hi!')
)
await self.ledger.wait(broadcast_tx)
await self.generate(1)
await self.ledger.wait(broadcast_tx, self.blockchain.block_expected)
self.assertEqual(self.ledger.headers.height, 208)
await self.assertBlockHash(208)
claim = await self.resolve('hovercraft')
self.assertEqual(claim['txid'], broadcast_tx.id)
self.assertEqual(claim['height'], 208)
# check that our tx is in block 208 as returned by lbrycrdd
invalidated_block_hash = (await self.ledger.headers.hash(208)).decode()
block_207 = await self.blockchain.get_block(invalidated_block_hash)
self.assertIn(claim['txid'], block_207['tx'])
self.assertEqual(208, claim['height'])
# reorg the last block dropping our claim tx
await self.blockchain.invalidate_block(invalidated_block_hash)
await self.conductor.clear_mempool()
2021-05-05 22:19:23 +02:00
await self.blockchain.generate(2)
await asyncio.wait_for(self.on_header(209), 3.0)
2021-05-05 22:19:23 +02:00
await self.assertBlockHash(207)
await self.assertBlockHash(208)
await self.assertBlockHash(209)
# verify the claim was dropped from block 208 as returned by lbrycrdd
reorg_block_hash = await self.blockchain.get_block_hash(208)
self.assertNotEqual(invalidated_block_hash, reorg_block_hash)
block_207 = await self.blockchain.get_block(reorg_block_hash)
self.assertNotIn(claim['txid'], block_207['tx'])
client_reorg_block_hash = (await self.ledger.headers.hash(208)).decode()
self.assertEqual(client_reorg_block_hash, reorg_block_hash)
# verify the dropped claim is no longer returned by claim search
self.assertDictEqual(
{'error': {'name': 'NOT_FOUND', 'text': 'Could not find claim at "hovercraft".'}},
await self.resolve('hovercraft')
)
# verify the claim published a block earlier wasn't also reverted
self.assertEqual(207, (await self.resolve('still-valid'))['height'])
# broadcast the claim in a different block
new_txid = await self.blockchain.sendrawtransaction(hexlify(broadcast_tx.raw).decode())
self.assertEqual(broadcast_tx.id, new_txid)
await self.blockchain.generate(1)
2021-05-05 22:19:23 +02:00
await asyncio.wait_for(self.on_header(210), 1.0)
# verify the claim is in the new block and that it is returned by claim_search
republished = await self.resolve('hovercraft')
self.assertEqual(210, republished['height'])
self.assertEqual(claim['claim_id'], republished['claim_id'])
# this should still be unchanged
self.assertEqual(207, (await self.resolve('still-valid'))['height'])
async def test_reorg_drop_claim(self):
# sanity check
result = await self.resolve('hovercraft') # TODO: do these for claim_search and resolve both
self.assertIn('error', result)
still_valid = await self.daemon.jsonrpc_stream_create(
'still-valid', '1.0', file_path=self.create_upload_file(data=b'hi!')
)
await self.ledger.wait(still_valid)
await self.generate(1)
# create a claim and verify it's returned by claim_search
2020-04-05 22:58:36 +02:00
self.assertEqual(self.ledger.headers.height, 207)
2020-11-18 20:38:15 +01:00
await self.assertBlockHash(207)
2020-04-01 21:11:45 +02:00
broadcast_tx = await self.daemon.jsonrpc_stream_create(
'hovercraft', '1.0', file_path=self.create_upload_file(data=b'hi!')
)
await self.ledger.wait(broadcast_tx)
await self.generate(1)
await self.ledger.wait(broadcast_tx, self.blockchain.block_expected)
2020-04-05 22:58:36 +02:00
self.assertEqual(self.ledger.headers.height, 208)
2020-11-18 20:38:15 +01:00
await self.assertBlockHash(208)
2021-03-10 02:15:50 +01:00
claim = await self.resolve('hovercraft')
self.assertEqual(claim['txid'], broadcast_tx.id)
self.assertEqual(claim['height'], 208)
2020-04-01 19:52:14 +02:00
2020-04-05 22:58:36 +02:00
# check that our tx is in block 208 as returned by lbrycrdd
invalidated_block_hash = (await self.ledger.headers.hash(208)).decode()
2020-04-01 19:52:14 +02:00
block_207 = await self.blockchain.get_block(invalidated_block_hash)
2021-03-10 02:15:50 +01:00
self.assertIn(claim['txid'], block_207['tx'])
self.assertEqual(208, claim['height'])
2020-04-01 19:52:14 +02:00
# reorg the last block dropping our claim tx
await self.blockchain.invalidate_block(invalidated_block_hash)
await self.conductor.clear_mempool()
2020-04-01 19:52:14 +02:00
await self.blockchain.generate(2)
2020-11-20 16:57:28 +01:00
# wait for the client to catch up and verify the reorg
await asyncio.wait_for(self.on_header(209), 3.0)
await self.assertBlockHash(207)
await self.assertBlockHash(208)
await self.assertBlockHash(209)
2020-04-05 22:58:36 +02:00
# verify the claim was dropped from block 208 as returned by lbrycrdd
reorg_block_hash = await self.blockchain.get_block_hash(208)
2020-04-01 19:52:14 +02:00
self.assertNotEqual(invalidated_block_hash, reorg_block_hash)
block_207 = await self.blockchain.get_block(reorg_block_hash)
2021-03-10 02:15:50 +01:00
self.assertNotIn(claim['txid'], block_207['tx'])
2020-04-01 19:52:14 +02:00
2020-04-05 22:58:36 +02:00
client_reorg_block_hash = (await self.ledger.headers.hash(208)).decode()
2020-04-01 19:52:14 +02:00
self.assertEqual(client_reorg_block_hash, reorg_block_hash)
2021-03-10 02:15:50 +01:00
# verify the dropped claim is no longer returned by claim search
self.assertDictEqual(
{'error': {'name': 'NOT_FOUND', 'text': 'Could not find claim at "hovercraft".'}},
await self.resolve('hovercraft')
)
# verify the claim published a block earlier wasn't also reverted
self.assertEqual(207, (await self.resolve('still-valid'))['height'])
2020-04-01 21:26:35 +02:00
# broadcast the claim in a different block
new_txid = await self.blockchain.sendrawtransaction(hexlify(broadcast_tx.raw).decode())
self.assertEqual(broadcast_tx.id, new_txid)
await self.blockchain.generate(1)
2020-04-05 22:58:36 +02:00
await asyncio.wait_for(self.on_header(210), 1.0)
2020-04-01 21:26:35 +02:00
# verify the claim is in the new block and that it is returned by claim_search
2021-03-10 02:15:50 +01:00
republished = await self.resolve('hovercraft')
self.assertEqual(210, republished['height'])
self.assertEqual(claim['claim_id'], republished['claim_id'])
2020-04-05 22:58:36 +02:00
# this should still be unchanged
2021-03-10 02:15:50 +01:00
self.assertEqual(207, (await self.resolve('still-valid'))['height'])