Test mining with the simple market (#31)

* Test mining with the simple market

* Support reward donation
This commit is contained in:
Chenxing Li 2024-03-19 14:09:17 +08:00 committed by GitHub
parent 306c43c9dc
commit 16dfc56437
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
9 changed files with 229 additions and 34 deletions

@ -1 +1 @@
Subproject commit 1f3f759236b48dd8682deba91011292cb5966f4c Subproject commit d466311abb6f629a6489450f2e684b2c6e7b1089

View File

@ -43,6 +43,10 @@ impl<'a> Miner<'a> {
pub async fn iteration(&self, nonce: H256) -> Option<AnswerWithoutProof> { pub async fn iteration(&self, nonce: H256) -> Option<AnswerWithoutProof> {
let (scratch_pad, recall_seed) = self.make_scratch_pad(&nonce); let (scratch_pad, recall_seed) = self.make_scratch_pad(&nonce);
if self.mining_length == 0 {
return None;
}
let (_, recall_offset) = U256::from_big_endian(&recall_seed) let (_, recall_offset) = U256::from_big_endian(&recall_seed)
.div_mod(U256::from((self.mining_length as usize) / SECTORS_PER_LOAD)); .div_mod(U256::from((self.mining_length as usize) / SECTORS_PER_LOAD));
let recall_offset = recall_offset.as_u64(); let recall_offset = recall_offset.as_u64();

90
tests/mine_with_market_test.py Executable file
View File

@ -0,0 +1,90 @@
#!/usr/bin/env python3
from test_framework.blockchain_node import BlockChainNodeType
from test_framework.test_framework import TestFramework
from config.node_config import MINER_ID, GENESIS_PRIV_KEY
from test_framework.blockchain_node import BlockChainNodeType
from utility.submission import create_submission, submit_data
from utility.utils import wait_until, assert_equal, assert_greater_than
import math
PRICE_PER_SECTOR = math.ceil(10 * (10 ** 18) / (2 ** 30) * 256 / 12)
class MineTest(TestFramework):
def setup_params(self):
self.num_blockchain_nodes = 1
self.num_nodes = 1
self.zgs_node_configs[0] = {
"miner_id": MINER_ID,
"miner_key": GENESIS_PRIV_KEY,
}
self.enable_market = True
self.mine_period = 150
def submit_data(self, item, size, no_submit = False):
submissions_before = self.contract.num_submissions()
client = self.nodes[0]
chunk_data = item * 256 * size
submissions, data_root = create_submission(chunk_data)
value = int(size * PRICE_PER_SECTOR * 1.1)
self.contract.submit(submissions, tx_prarams = {"value": value})
wait_until(lambda: self.contract.num_submissions() == submissions_before + 1)
if not no_submit:
wait_until(lambda: client.zgs_get_file_info(data_root) is not None)
segment = submit_data(client, chunk_data)
wait_until(lambda: client.zgs_get_file_info(data_root)["finalized"])
def run_test(self):
blockchain = self.blockchain_nodes[0]
self.log.info("flow address: %s", self.contract.address())
self.log.info("mine address: %s", self.mine_contract.address())
quality = int(2**256 / 4096)
self.mine_contract.set_quality(quality)
SECTORS_PER_PRICING = int(8 * ( 2 ** 30 ) / 256)
self.log.info("Submit the actual data chunk (256 MB)")
self.submit_data(b"\x11", int(SECTORS_PER_PRICING / 32))
self.log.info("Submit the data hash only (8 GB)")
self.submit_data(b"\x11", int(SECTORS_PER_PRICING), no_submit=True)
self.log.info("Sumission Done, Current block number %d", int(blockchain.eth_blockNumber(), 16))
self.log.info("Wait for mine context release")
wait_until(lambda: self.contract.get_mine_context()[0] > 0, timeout=180)
self.log.info("Current flow length: %d", self.contract.get_mine_context()[3])
self.log.info("Wait for mine answer")
wait_until(lambda: self.mine_contract.last_mined_epoch() == 1)
rewards = self.reward_contract.reward_distributes()
assert_equal(len(self.reward_contract.reward_distributes()), 1)
firstReward = rewards[0].args.amount
self.log.info("Received reward %d Gwei", firstReward / (10**9))
self.reward_contract.transfer(10000 * 10 ** 18)
self.log.info("Donation Done")
self.log.info("Submit the data hash only (8 GB)")
self.submit_data(b"\x11", int(SECTORS_PER_PRICING), no_submit=True)
self.log.info("Sumission Done, Current block number %d", int(blockchain.eth_blockNumber(), 16))
self.log.info("Wait for mine context release")
wait_until(lambda: self.contract.get_mine_context()[0] > 1, timeout=180)
self.log.info("Wait for mine answer")
wait_until(lambda: self.mine_contract.last_mined_epoch() == 2)
rewards = self.reward_contract.reward_distributes()
assert_equal(len(self.reward_contract.reward_distributes()), 2)
secondReward = rewards[1].args.amount
self.log.info("Received reward %d Gwei", secondReward / (10**9))
assert_greater_than(secondReward, 100 * firstReward)
if __name__ == "__main__":
MineTest(blockchain_node_type=BlockChainNodeType.BSC).main()

View File

@ -248,7 +248,7 @@ class BlockchainNode(TestNode):
def wait_for_transaction_receipt(self, w3, tx_hash, timeout=120, parent_hash=None): def wait_for_transaction_receipt(self, w3, tx_hash, timeout=120, parent_hash=None):
return w3.eth.wait_for_transaction_receipt(tx_hash, timeout) return w3.eth.wait_for_transaction_receipt(tx_hash, timeout)
def setup_contract(self): def setup_contract(self, enable_market, mine_period):
w3 = Web3(HTTPProvider(self.rpc_url)) w3 = Web3(HTTPProvider(self.rpc_url))
account1 = w3.eth.account.from_key(GENESIS_PRIV_KEY) account1 = w3.eth.account.from_key(GENESIS_PRIV_KEY)
@ -285,7 +285,7 @@ class BlockchainNode(TestNode):
contract_address = '0x' + contract_address_bytes.hex() contract_address = '0x' + contract_address_bytes.hex()
return Web3.to_checksum_address(contract_address) return Web3.to_checksum_address(contract_address)
def deploy_no_market():
flowAddress = predict_contract_address(1) flowAddress = predict_contract_address(1)
mineAddress = predict_contract_address(2) mineAddress = predict_contract_address(2)
@ -295,22 +295,51 @@ class BlockchainNode(TestNode):
book, _ = deploy_contract("AddressBook", [flowAddress, ZERO, ZERO, mineAddress]); book, _ = deploy_contract("AddressBook", [flowAddress, ZERO, ZERO, mineAddress]);
self.log.debug("AddressBook deployed") self.log.debug("AddressBook deployed")
flow_contract, flow_contract_hash = deploy_contract( flow_contract, flow_contract_hash = deploy_contract("Flow", [book.address, mine_period, 0])
"Flow", [book.address, 100, 0]
)
self.log.debug("Flow deployed") self.log.debug("Flow deployed")
mine_contract, _ = deploy_contract( mine_contract, _ = deploy_contract("PoraMineTest", [book.address, 3])
"PoraMineTest",
[book.address, 3],
)
self.log.debug("Mine deployed") self.log.debug("Mine deployed")
self.log.info("All contracts deployed") self.log.info("All contracts deployed")
tx_hash = mine_contract.functions.setMiner(decode_hex(MINER_ID)).transact(TX_PARAMS) tx_hash = mine_contract.functions.setMiner(decode_hex(MINER_ID)).transact(TX_PARAMS)
self.wait_for_transaction_receipt(w3, tx_hash) self.wait_for_transaction_receipt(w3, tx_hash)
return flow_contract, flow_contract_hash, mine_contract dummy_reward_contract = w3.eth.contract(
address = book.functions.reward().call(),
abi=load_contract_metadata(base_path=self.contract_path, name="IReward")["abi"],
)
return flow_contract, flow_contract_hash, mine_contract, dummy_reward_contract
def deploy_with_market():
mineAddress = predict_contract_address(1)
marketAddress = predict_contract_address(2)
rewardAddress = predict_contract_address(3)
flowAddress = predict_contract_address(4)
LIFETIME_MONTH = 1
self.log.debug("Start deploy contracts")
book, _ = deploy_contract("AddressBook", [flowAddress, marketAddress, rewardAddress, mineAddress]);
self.log.debug("AddressBook deployed")
mine_contract, _ = deploy_contract("PoraMineTest", [book.address, 3])
deploy_contract("FixedPrice", [book.address, LIFETIME_MONTH])
reward_contract, _ =deploy_contract("OnePoolReward", [book.address, LIFETIME_MONTH])
flow_contract, flow_contract_hash = deploy_contract("FixedPriceFlow", [book.address, mine_period, 0])
self.log.info("All contracts deployed")
tx_hash = mine_contract.functions.setMiner(decode_hex(MINER_ID)).transact(TX_PARAMS)
self.wait_for_transaction_receipt(w3, tx_hash)
return flow_contract, flow_contract_hash, mine_contract, reward_contract
if enable_market:
return deploy_with_market()
else:
return deploy_no_market()
def get_contract(self, contract_address): def get_contract(self, contract_address):
w3 = Web3(HTTPProvider(self.rpc_url)) w3 = Web3(HTTPProvider(self.rpc_url))

View File

@ -1,6 +1,7 @@
from gettext import npgettext from gettext import npgettext
from config.node_config import TX_PARAMS from config.node_config import TX_PARAMS
from utility.utils import assert_equal from utility.utils import assert_equal
from copy import copy
class ContractProxy: class ContractProxy:
@ -28,18 +29,39 @@ class ContractProxy:
contract = self._get_contract(node_idx) contract = self._get_contract(node_idx)
return getattr(contract.functions, fn_name)(**args).transact(TX_PARAMS) return getattr(contract.functions, fn_name)(**args).transact(TX_PARAMS)
def _logs(self, event_name, node_idx, **args):
assert node_idx < len(self.blockchain_nodes)
contract = self._get_contract(node_idx)
return getattr(contract.events, event_name).create_filter(fromBlock =0, toBlock="latest").get_all_entries()
def transfer(self, value, node_idx = 0):
tx_params = TX_PARAMS
tx_params["value"] = value
contract = self._get_contract(node_idx)
contract.receive.transact(tx_params)
def address(self): def address(self):
return self.contract_address return self.contract_address
class FlowContractProxy(ContractProxy): class FlowContractProxy(ContractProxy):
def submit( def submit(
self, submission_nodes, node_idx=0, tx_prarams=TX_PARAMS, parent_hash=None self, submission_nodes, node_idx=0, tx_prarams=None, parent_hash=None,
): ):
assert node_idx < len(self.blockchain_nodes) assert node_idx < len(self.blockchain_nodes)
combined_tx_prarams = TX_PARAMS
if tx_prarams is not None:
combined_tx_prarams.update(tx_prarams)
contract = self._get_contract(node_idx) contract = self._get_contract(node_idx)
tx_hash = contract.functions.submit(submission_nodes).transact(tx_prarams) # print(contract.functions.submit(submission_nodes).estimate_gas(combined_tx_prarams))
tx_hash = contract.functions.submit(submission_nodes).transact(combined_tx_prarams)
receipt = self.blockchain_nodes[node_idx].wait_for_transaction_receipt( receipt = self.blockchain_nodes[node_idx].wait_for_transaction_receipt(
contract.w3, tx_hash, parent_hash=parent_hash contract.w3, tx_hash, parent_hash=parent_hash
) )
@ -57,6 +79,9 @@ class FlowContractProxy(ContractProxy):
def epoch(self, node_idx=0): def epoch(self, node_idx=0):
return self._call("epoch", node_idx) return self._call("epoch", node_idx)
def get_mine_context(self, node_idx=0):
return self._call("makeContextWithResult", node_idx)
class MineContractProxy(ContractProxy): class MineContractProxy(ContractProxy):
def last_mined_epoch(self, node_idx=0): def last_mined_epoch(self, node_idx=0):
@ -64,3 +89,9 @@ class MineContractProxy(ContractProxy):
def set_quality(self, quality, node_idx=0): def set_quality(self, quality, node_idx=0):
return self._send("setQuality", node_idx, _targetQuality=quality) return self._send("setQuality", node_idx, _targetQuality=quality)
class IRewardContractProxy(ContractProxy):
def reward_distributes(self, node_idx=0):
return self._logs("DistributeReward", node_idx)

View File

@ -13,7 +13,7 @@ import traceback
from eth_utils import encode_hex from eth_utils import encode_hex
from test_framework.bsc_node import BSCNode from test_framework.bsc_node import BSCNode
from test_framework.contract_proxy import FlowContractProxy, MineContractProxy from test_framework.contract_proxy import FlowContractProxy, MineContractProxy, IRewardContractProxy
from test_framework.zgs_node import ZgsNode from test_framework.zgs_node import ZgsNode
from test_framework.blockchain_node import BlockChainNodeType from test_framework.blockchain_node import BlockChainNodeType
from test_framework.conflux_node import ConfluxNode, connect_sample_nodes, sync_blocks from test_framework.conflux_node import ConfluxNode, connect_sample_nodes, sync_blocks
@ -34,6 +34,10 @@ TEST_EXIT_FAILED = 1
class TestFramework: class TestFramework:
def __init__(self, blockchain_node_type=BlockChainNodeType.Conflux): def __init__(self, blockchain_node_type=BlockChainNodeType.Conflux):
if "http_proxy" in os.environ:
# Print a warning message in yellow color
print("\n\033[93m ⚠️ Warning: You've set the environment variable 'http_proxy', which might lead to testing issues.\033[0m\n")
self.num_blockchain_nodes = None self.num_blockchain_nodes = None
self.num_nodes = None self.num_nodes = None
self.blockchain_nodes = [] self.blockchain_nodes = []
@ -42,6 +46,8 @@ class TestFramework:
self.blockchain_node_configs = {} self.blockchain_node_configs = {}
self.zgs_node_configs = {} self.zgs_node_configs = {}
self.blockchain_node_type = blockchain_node_type self.blockchain_node_type = blockchain_node_type
self.enable_market = False
self.mine_period = 100
binary_ext = ".exe" if is_windows_platform() else "" binary_ext = ".exe" if is_windows_platform() else ""
tests_dir = os.path.dirname(__file_path__) tests_dir = os.path.dirname(__file_path__)
@ -134,9 +140,11 @@ class TestFramework:
connect_sample_nodes(self.blockchain_nodes, self.log) connect_sample_nodes(self.blockchain_nodes, self.log)
sync_blocks(self.blockchain_nodes) sync_blocks(self.blockchain_nodes)
contract, tx_hash, mine_contract = self.blockchain_nodes[0].setup_contract() contract, tx_hash, mine_contract, reward_contract = self.blockchain_nodes[0].setup_contract(self.enable_market, self.mine_period)
self.contract = FlowContractProxy(contract, self.blockchain_nodes) self.contract = FlowContractProxy(contract, self.blockchain_nodes)
self.mine_contract = MineContractProxy(mine_contract, self.blockchain_nodes) self.mine_contract = MineContractProxy(mine_contract, self.blockchain_nodes)
self.reward_contract = IRewardContractProxy(reward_contract, self.blockchain_nodes)
for node in self.blockchain_nodes[1:]: for node in self.blockchain_nodes[1:]:
node.wait_for_transaction(tx_hash) node.wait_for_transaction(tx_hash)

View File

@ -1,6 +1,7 @@
import sha3 import sha3
from math import log2 from math import log2
from utility.spec import ENTRY_SIZE
def decompose(num): def decompose(num):
@ -154,6 +155,28 @@ class MerkleTree:
leaf = Leaf.from_data(data, self.hasher) leaf = Leaf.from_data(data, self.hasher)
self.add_leaf(leaf) self.add_leaf(leaf)
@classmethod
def from_data_list(cls, data, encoding="utf-8"):
tree = cls(encoding)
n = len(data)
if n < ENTRY_SIZE or (n & (n - 1)) != 0:
raise Exception("Input length is not power of 2")
leaves = [Leaf.from_data(data[i:i + ENTRY_SIZE], tree.hasher) for i in range(0, n, ENTRY_SIZE)]
tree.__leaves = leaves
nodes = leaves
while len(nodes) > 1:
next_nodes = []
for i in range(0, len(nodes), 2):
next_nodes.append(Node.from_children(nodes[i], nodes[i+1], tree.hasher))
nodes = next_nodes
tree.__root = nodes[0]
return tree
def add_leaf(self, leaf): def add_leaf(self, leaf):
if self: if self:
subroot = self.get_last_subroot() subroot = self.get_last_subroot()

2
tests/utility/spec.py Normal file
View File

@ -0,0 +1,2 @@
ENTRY_SIZE = 256
PORA_CHUNK_SIZE = 1024

View File

@ -3,11 +3,7 @@ import base64
from eth_utils import encode_hex, decode_hex from eth_utils import encode_hex, decode_hex
from math import log2 from math import log2
from utility.merkle_tree import add_0x_prefix, Leaf, MerkleTree from utility.merkle_tree import add_0x_prefix, Leaf, MerkleTree
from utility.spec import ENTRY_SIZE, PORA_CHUNK_SIZE
ENTRY_SIZE = 256
PORA_CHUNK_SIZE = 1024
def log2_pow2(n): def log2_pow2(n):
return int(log2(((n ^ (n - 1)) >> 1) + 1)) return int(log2(((n ^ (n - 1)) >> 1) + 1))
@ -98,9 +94,8 @@ def create_node(data, offset, chunks):
def create_segment_node(data, offset, batch, size): def create_segment_node(data, offset, batch, size):
tree = MerkleTree() tree = MerkleTree()
i = offset
n = len(data) n = len(data)
while i < offset + size: for i in range(offset, offset + size, batch):
start = i start = i
end = min(offset + size, i + batch) end = min(offset + size, i + batch)
@ -111,21 +106,34 @@ def create_segment_node(data, offset, batch, size):
else: else:
tree.add_leaf(Leaf(segment_root(data[start:end]))) tree.add_leaf(Leaf(segment_root(data[start:end])))
i += batch
return tree.get_root_hash() return tree.get_root_hash()
segment_root_cached_chunks = None
segment_root_cached_output = None
def segment_root(chunks): def segment_root(chunks):
global segment_root_cached_chunks, segment_root_cached_output
if segment_root_cached_chunks == chunks:
return segment_root_cached_output
data_len = len(chunks) data_len = len(chunks)
if data_len == 0: if data_len == 0:
return b"\x00" * 32 return b"\x00" * 32
tree = MerkleTree() tree = MerkleTree()
for i in range(0, data_len, ENTRY_SIZE): for i in range(0, data_len, ENTRY_SIZE):
tree.encrypt(chunks[i : i + ENTRY_SIZE]) tree.encrypt(chunks[i : i + ENTRY_SIZE])
return tree.get_root_hash() digest = tree.get_root_hash()
segment_root_cached_chunks = chunks
segment_root_cached_output = digest
return digest
def generate_merkle_tree(data): def generate_merkle_tree(data):