mirror of
				https://github.com/0glabs/0g-storage-node.git
				synced 2025-11-03 16:17:27 +00:00 
			
		
		
		
	Test mining with the simple market (#31)
* Test mining with the simple market * Support reward donation
This commit is contained in:
		
							parent
							
								
									306c43c9dc
								
							
						
					
					
						commit
						16dfc56437
					
				@ -1 +1 @@
 | 
			
		||||
Subproject commit 1f3f759236b48dd8682deba91011292cb5966f4c
 | 
			
		||||
Subproject commit d466311abb6f629a6489450f2e684b2c6e7b1089
 | 
			
		||||
@ -43,6 +43,10 @@ impl<'a> Miner<'a> {
 | 
			
		||||
    pub async fn iteration(&self, nonce: H256) -> Option<AnswerWithoutProof> {
 | 
			
		||||
        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)
 | 
			
		||||
            .div_mod(U256::from((self.mining_length as usize) / SECTORS_PER_LOAD));
 | 
			
		||||
        let recall_offset = recall_offset.as_u64();
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										90
									
								
								tests/mine_with_market_test.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										90
									
								
								tests/mine_with_market_test.py
									
									
									
									
									
										Executable 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()
 | 
			
		||||
@ -248,7 +248,7 @@ class BlockchainNode(TestNode):
 | 
			
		||||
    def wait_for_transaction_receipt(self, w3, tx_hash, timeout=120, parent_hash=None):
 | 
			
		||||
        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))
 | 
			
		||||
 | 
			
		||||
        account1 = w3.eth.account.from_key(GENESIS_PRIV_KEY)
 | 
			
		||||
@ -285,7 +285,7 @@ class BlockchainNode(TestNode):
 | 
			
		||||
            contract_address = '0x' + contract_address_bytes.hex()           
 | 
			
		||||
            return Web3.to_checksum_address(contract_address)
 | 
			
		||||
        
 | 
			
		||||
        
 | 
			
		||||
        def deploy_no_market():
 | 
			
		||||
            flowAddress = predict_contract_address(1)
 | 
			
		||||
            mineAddress = predict_contract_address(2)
 | 
			
		||||
 | 
			
		||||
@ -295,22 +295,51 @@ class BlockchainNode(TestNode):
 | 
			
		||||
            book, _ = deploy_contract("AddressBook", [flowAddress, ZERO, ZERO, mineAddress]);
 | 
			
		||||
            self.log.debug("AddressBook deployed")
 | 
			
		||||
 | 
			
		||||
        flow_contract, flow_contract_hash = deploy_contract(
 | 
			
		||||
            "Flow", [book.address, 100, 0]
 | 
			
		||||
        )
 | 
			
		||||
            flow_contract, flow_contract_hash = deploy_contract("Flow", [book.address, mine_period, 0])
 | 
			
		||||
            self.log.debug("Flow deployed")
 | 
			
		||||
 | 
			
		||||
        mine_contract, _ = deploy_contract(
 | 
			
		||||
            "PoraMineTest",
 | 
			
		||||
            [book.address, 3],
 | 
			
		||||
        )
 | 
			
		||||
            mine_contract, _ = deploy_contract("PoraMineTest", [book.address, 3])
 | 
			
		||||
            self.log.debug("Mine deployed")
 | 
			
		||||
            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
 | 
			
		||||
            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):
 | 
			
		||||
        w3 = Web3(HTTPProvider(self.rpc_url))
 | 
			
		||||
 | 
			
		||||
@ -1,6 +1,7 @@
 | 
			
		||||
from gettext import npgettext
 | 
			
		||||
from config.node_config import TX_PARAMS
 | 
			
		||||
from utility.utils import assert_equal
 | 
			
		||||
from copy import copy
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class ContractProxy:
 | 
			
		||||
@ -28,18 +29,39 @@ class ContractProxy:
 | 
			
		||||
        contract = self._get_contract(node_idx)
 | 
			
		||||
        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):
 | 
			
		||||
        return self.contract_address
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class FlowContractProxy(ContractProxy):
 | 
			
		||||
    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)
 | 
			
		||||
 | 
			
		||||
        combined_tx_prarams = TX_PARAMS
 | 
			
		||||
 | 
			
		||||
        if tx_prarams is not None:
 | 
			
		||||
            combined_tx_prarams.update(tx_prarams)
 | 
			
		||||
            
 | 
			
		||||
 | 
			
		||||
        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(
 | 
			
		||||
            contract.w3, tx_hash, parent_hash=parent_hash
 | 
			
		||||
        )
 | 
			
		||||
@ -57,6 +79,9 @@ class FlowContractProxy(ContractProxy):
 | 
			
		||||
    def epoch(self, node_idx=0):
 | 
			
		||||
        return self._call("epoch", node_idx)
 | 
			
		||||
 | 
			
		||||
    def get_mine_context(self, node_idx=0):
 | 
			
		||||
        return self._call("makeContextWithResult", node_idx)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class MineContractProxy(ContractProxy):
 | 
			
		||||
    def last_mined_epoch(self, node_idx=0):
 | 
			
		||||
@ -64,3 +89,9 @@ class MineContractProxy(ContractProxy):
 | 
			
		||||
 | 
			
		||||
    def set_quality(self, quality, node_idx=0):
 | 
			
		||||
        return self._send("setQuality", node_idx, _targetQuality=quality)
 | 
			
		||||
    
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class IRewardContractProxy(ContractProxy):
 | 
			
		||||
    def reward_distributes(self, node_idx=0):
 | 
			
		||||
        return self._logs("DistributeReward", node_idx)
 | 
			
		||||
@ -13,7 +13,7 @@ import traceback
 | 
			
		||||
 | 
			
		||||
from eth_utils import encode_hex
 | 
			
		||||
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.blockchain_node import BlockChainNodeType
 | 
			
		||||
from test_framework.conflux_node import ConfluxNode, connect_sample_nodes, sync_blocks
 | 
			
		||||
@ -34,6 +34,10 @@ TEST_EXIT_FAILED = 1
 | 
			
		||||
 | 
			
		||||
class TestFramework:
 | 
			
		||||
    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_nodes = None
 | 
			
		||||
        self.blockchain_nodes = []
 | 
			
		||||
@ -42,6 +46,8 @@ class TestFramework:
 | 
			
		||||
        self.blockchain_node_configs = {}
 | 
			
		||||
        self.zgs_node_configs = {}
 | 
			
		||||
        self.blockchain_node_type = blockchain_node_type
 | 
			
		||||
        self.enable_market = False
 | 
			
		||||
        self.mine_period = 100
 | 
			
		||||
 | 
			
		||||
        binary_ext = ".exe" if is_windows_platform() else ""
 | 
			
		||||
        tests_dir = os.path.dirname(__file_path__)
 | 
			
		||||
@ -134,9 +140,11 @@ class TestFramework:
 | 
			
		||||
                connect_sample_nodes(self.blockchain_nodes, self.log)
 | 
			
		||||
                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.mine_contract = MineContractProxy(mine_contract, self.blockchain_nodes)
 | 
			
		||||
        self.reward_contract = IRewardContractProxy(reward_contract, self.blockchain_nodes)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
        for node in self.blockchain_nodes[1:]:
 | 
			
		||||
            node.wait_for_transaction(tx_hash)
 | 
			
		||||
 | 
			
		||||
@ -1,6 +1,7 @@
 | 
			
		||||
import sha3
 | 
			
		||||
 | 
			
		||||
from math import log2
 | 
			
		||||
from utility.spec import ENTRY_SIZE
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def decompose(num):
 | 
			
		||||
@ -154,6 +155,28 @@ class MerkleTree:
 | 
			
		||||
        leaf = Leaf.from_data(data, self.hasher)
 | 
			
		||||
        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):
 | 
			
		||||
        if self:
 | 
			
		||||
            subroot = self.get_last_subroot()
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										2
									
								
								tests/utility/spec.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								tests/utility/spec.py
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,2 @@
 | 
			
		||||
ENTRY_SIZE = 256
 | 
			
		||||
PORA_CHUNK_SIZE = 1024
 | 
			
		||||
@ -3,11 +3,7 @@ import base64
 | 
			
		||||
from eth_utils import encode_hex, decode_hex
 | 
			
		||||
from math import log2
 | 
			
		||||
from utility.merkle_tree import add_0x_prefix, Leaf, MerkleTree
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
ENTRY_SIZE = 256
 | 
			
		||||
PORA_CHUNK_SIZE = 1024
 | 
			
		||||
 | 
			
		||||
from utility.spec import ENTRY_SIZE, PORA_CHUNK_SIZE
 | 
			
		||||
 | 
			
		||||
def log2_pow2(n):
 | 
			
		||||
    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):
 | 
			
		||||
    tree = MerkleTree()
 | 
			
		||||
    i = offset
 | 
			
		||||
    n = len(data)
 | 
			
		||||
    while i < offset + size:
 | 
			
		||||
    for i in range(offset, offset + size, batch):
 | 
			
		||||
        start = i
 | 
			
		||||
        end = min(offset + size, i + batch)
 | 
			
		||||
 | 
			
		||||
@ -111,21 +106,34 @@ def create_segment_node(data, offset, batch, size):
 | 
			
		||||
        else:
 | 
			
		||||
            tree.add_leaf(Leaf(segment_root(data[start:end])))
 | 
			
		||||
 | 
			
		||||
        i += batch
 | 
			
		||||
 | 
			
		||||
    return tree.get_root_hash()
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
segment_root_cached_chunks = None
 | 
			
		||||
segment_root_cached_output = None
 | 
			
		||||
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)
 | 
			
		||||
    if data_len == 0:
 | 
			
		||||
        return b"\x00" * 32
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    tree = MerkleTree()
 | 
			
		||||
    for i in range(0, data_len, 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):
 | 
			
		||||
 | 
			
		||||
		Loading…
	
		Reference in New Issue
	
	Block a user