156 lines
4.9 KiB
Python
156 lines
4.9 KiB
Python
import pickle
|
|
from os import path
|
|
from pathlib import Path
|
|
from typing import List
|
|
|
|
import aiosqlite
|
|
import pytest
|
|
|
|
from chia.consensus.blockchain import Blockchain
|
|
from chia.consensus.constants import ConsensusConstants
|
|
from chia.full_node.block_store import BlockStore
|
|
from chia.full_node.coin_store import CoinStore
|
|
from chia.types.full_block import FullBlock
|
|
from chia.util.db_wrapper import DBWrapper
|
|
from chia.util.path import mkdir
|
|
from tests.setup_nodes import bt, test_constants
|
|
|
|
|
|
async def create_blockchain(constants: ConsensusConstants):
|
|
db_path = Path("blockchain_test.db")
|
|
if db_path.exists():
|
|
db_path.unlink()
|
|
connection = await aiosqlite.connect(db_path)
|
|
wrapper = DBWrapper(connection)
|
|
coin_store = await CoinStore.create(wrapper)
|
|
store = await BlockStore.create(wrapper)
|
|
bc1 = await Blockchain.create(coin_store, store, constants)
|
|
assert bc1.get_peak() is None
|
|
return bc1, connection, db_path
|
|
|
|
|
|
@pytest.fixture(scope="function")
|
|
async def empty_blockchain():
|
|
"""
|
|
Provides a list of 10 valid blocks, as well as a blockchain with 9 blocks added to it.
|
|
"""
|
|
bc1, connection, db_path = await create_blockchain(test_constants)
|
|
yield bc1
|
|
|
|
await connection.close()
|
|
bc1.shut_down()
|
|
db_path.unlink()
|
|
|
|
|
|
block_format_version = "rc4"
|
|
|
|
|
|
@pytest.fixture(scope="session")
|
|
async def default_400_blocks():
|
|
return persistent_blocks(400, f"test_blocks_400_{block_format_version}.db", seed=b"alternate2")
|
|
|
|
|
|
@pytest.fixture(scope="session")
|
|
async def default_1000_blocks():
|
|
return persistent_blocks(1000, f"test_blocks_1000_{block_format_version}.db")
|
|
|
|
|
|
@pytest.fixture(scope="session")
|
|
async def pre_genesis_empty_slots_1000_blocks():
|
|
return persistent_blocks(
|
|
1000, f"pre_genesis_empty_slots_1000_blocks{block_format_version}.db", seed=b"alternate2", empty_sub_slots=1
|
|
)
|
|
|
|
|
|
@pytest.fixture(scope="session")
|
|
async def default_10000_blocks():
|
|
return persistent_blocks(10000, f"test_blocks_10000_{block_format_version}.db")
|
|
|
|
|
|
@pytest.fixture(scope="session")
|
|
async def default_20000_blocks():
|
|
return persistent_blocks(20000, f"test_blocks_20000_{block_format_version}.db")
|
|
|
|
|
|
@pytest.fixture(scope="session")
|
|
async def default_10000_blocks_compact():
|
|
return persistent_blocks(
|
|
10000,
|
|
f"test_blocks_10000_compact_{block_format_version}.db",
|
|
normalized_to_identity_cc_eos=True,
|
|
normalized_to_identity_icc_eos=True,
|
|
normalized_to_identity_cc_ip=True,
|
|
normalized_to_identity_cc_sp=True,
|
|
)
|
|
|
|
|
|
def persistent_blocks(
|
|
num_of_blocks: int,
|
|
db_name: str,
|
|
seed: bytes = b"",
|
|
empty_sub_slots=0,
|
|
normalized_to_identity_cc_eos: bool = False,
|
|
normalized_to_identity_icc_eos: bool = False,
|
|
normalized_to_identity_cc_sp: bool = False,
|
|
normalized_to_identity_cc_ip: bool = False,
|
|
):
|
|
# try loading from disc, if not create new blocks.db file
|
|
# TODO hash fixtures.py and blocktool.py, add to path, delete if the files changed
|
|
block_path_dir = Path("~/.chia/blocks").expanduser()
|
|
file_path = Path(f"~/.chia/blocks/{db_name}").expanduser()
|
|
if not path.exists(block_path_dir):
|
|
mkdir(block_path_dir.parent)
|
|
mkdir(block_path_dir)
|
|
|
|
if file_path.exists():
|
|
try:
|
|
bytes_list = file_path.read_bytes()
|
|
block_bytes_list: List[bytes] = pickle.loads(bytes_list)
|
|
blocks: List[FullBlock] = []
|
|
for block_bytes in block_bytes_list:
|
|
blocks.append(FullBlock.from_bytes(block_bytes))
|
|
if len(blocks) == num_of_blocks:
|
|
print(f"\n loaded {file_path} with {len(blocks)} blocks")
|
|
return blocks
|
|
except EOFError:
|
|
print("\n error reading db file")
|
|
|
|
return new_test_db(
|
|
file_path,
|
|
num_of_blocks,
|
|
seed,
|
|
empty_sub_slots,
|
|
normalized_to_identity_cc_eos,
|
|
normalized_to_identity_icc_eos,
|
|
normalized_to_identity_cc_sp,
|
|
normalized_to_identity_cc_ip,
|
|
)
|
|
|
|
|
|
def new_test_db(
|
|
path: Path,
|
|
num_of_blocks: int,
|
|
seed: bytes,
|
|
empty_sub_slots: int,
|
|
normalized_to_identity_cc_eos: bool = False, # CC_EOS,
|
|
normalized_to_identity_icc_eos: bool = False, # ICC_EOS
|
|
normalized_to_identity_cc_sp: bool = False, # CC_SP,
|
|
normalized_to_identity_cc_ip: bool = False, # CC_IP
|
|
):
|
|
print(f"create {path} with {num_of_blocks} blocks with ")
|
|
blocks: List[FullBlock] = bt.get_consecutive_blocks(
|
|
num_of_blocks,
|
|
seed=seed,
|
|
skip_slots=empty_sub_slots,
|
|
normalized_to_identity_cc_eos=normalized_to_identity_cc_eos,
|
|
normalized_to_identity_icc_eos=normalized_to_identity_icc_eos,
|
|
normalized_to_identity_cc_sp=normalized_to_identity_cc_sp,
|
|
normalized_to_identity_cc_ip=normalized_to_identity_cc_ip,
|
|
)
|
|
block_bytes_list: List[bytes] = []
|
|
for block in blocks:
|
|
block_bytes_list.append(bytes(block))
|
|
bytes_fn = pickle.dumps(block_bytes_list)
|
|
path.write_bytes(bytes_fn)
|
|
return blocks
|