mirror of
https://github.com/0glabs/0g-storage-node.git
synced 2025-01-23 13:36:08 +00:00
Change miner id logic & request miner id automatically (#60)
* Change miner id logic & request miner id automatically * Not enable all features in the test workflow. * Auto configurable mining period * Adjust test params for ci
This commit is contained in:
parent
5bcd3602b0
commit
193e154361
2
.github/workflows/cc.yml
vendored
2
.github/workflows/cc.yml
vendored
@ -37,7 +37,7 @@ jobs:
|
||||
uses: ./.github/actions/setup-rust
|
||||
|
||||
- name: Run unittest
|
||||
run: cargo test --all-features --no-fail-fast
|
||||
run: cargo test --no-fail-fast
|
||||
env:
|
||||
CARGO_INCREMENTAL: '0'
|
||||
RUSTC_BOOTSTRAP: '1'
|
||||
|
1
.gitignore
vendored
1
.gitignore
vendored
@ -5,3 +5,4 @@
|
||||
tests/**/__pycache__
|
||||
tests/tmp/**
|
||||
.vscode/*.json
|
||||
/0g-storage-contracts-dev
|
@ -1 +1 @@
|
||||
Subproject commit d466311abb6f629a6489450f2e684b2c6e7b1089
|
||||
Subproject commit 6a9f52e8c10ff9b5cd7a5844c543c0951b97d395
|
@ -11,4 +11,4 @@ ethers = "^2"
|
||||
serde_json = "1.0.82"
|
||||
|
||||
[features]
|
||||
compile-contracts = []
|
||||
dev = []
|
@ -1,3 +1,7 @@
|
||||
fn main() {
|
||||
println!("cargo:rerun-if-changed=../../0g-storage-contracts/artifacts/");
|
||||
if cfg!(not(feature = "dev")) {
|
||||
println!("cargo:rerun-if-changed=../../0g-storage-contracts/artifacts/");
|
||||
} else {
|
||||
println!("cargo:rerun-if-changed=../../0g-storage-contracts-dev/artifacts/");
|
||||
}
|
||||
}
|
||||
|
@ -2,12 +2,26 @@ use ethers::prelude::abigen;
|
||||
|
||||
// run `cargo doc -p contract-interface --open` to read struct definition
|
||||
|
||||
#[cfg(not(feature = "dev"))]
|
||||
abigen!(
|
||||
ZgsFlow,
|
||||
"../../0g-storage-contracts/artifacts/contracts/dataFlow/Flow.sol/Flow.json"
|
||||
);
|
||||
|
||||
#[cfg(not(feature = "dev"))]
|
||||
abigen!(
|
||||
PoraMine,
|
||||
"../../0g-storage-contracts/artifacts/contracts/test/PoraMineTest.sol/PoraMineTest.json"
|
||||
"../../0g-storage-contracts/artifacts/contracts/miner/Mine.sol/PoraMine.json"
|
||||
);
|
||||
|
||||
#[cfg(feature = "dev")]
|
||||
abigen!(
|
||||
ZgsFlow,
|
||||
"../../0g-storage-contracts-dev/artifacts/contracts/dataFlow/Flow.sol/Flow.json"
|
||||
);
|
||||
|
||||
#[cfg(feature = "dev")]
|
||||
abigen!(
|
||||
PoraMine,
|
||||
"../../0g-storage-contracts-dev/artifacts/contracts/miner/Mine.sol/PoraMine.json"
|
||||
);
|
||||
|
@ -8,7 +8,7 @@ use ethers::signers::LocalWallet;
|
||||
use ethers::signers::Signer;
|
||||
|
||||
pub struct MinerConfig {
|
||||
pub(crate) miner_id: H256,
|
||||
pub(crate) miner_id: Option<H256>,
|
||||
pub(crate) miner_key: H256,
|
||||
pub(crate) rpc_endpoint_url: String,
|
||||
pub(crate) mine_address: Address,
|
||||
@ -32,19 +32,16 @@ impl MinerConfig {
|
||||
cpu_percentage: u64,
|
||||
iter_batch: usize,
|
||||
) -> Option<MinerConfig> {
|
||||
match (miner_id, miner_key) {
|
||||
(Some(miner_id), Some(miner_key)) => Some(MinerConfig {
|
||||
miner_id,
|
||||
miner_key,
|
||||
rpc_endpoint_url,
|
||||
mine_address,
|
||||
flow_address,
|
||||
submission_gas,
|
||||
cpu_percentage,
|
||||
iter_batch,
|
||||
}),
|
||||
_ => None,
|
||||
}
|
||||
miner_key.map(|miner_key| MinerConfig {
|
||||
miner_id,
|
||||
miner_key,
|
||||
rpc_endpoint_url,
|
||||
mine_address,
|
||||
flow_address,
|
||||
submission_gas,
|
||||
cpu_percentage,
|
||||
iter_batch,
|
||||
})
|
||||
}
|
||||
|
||||
pub(crate) async fn make_provider(&self) -> Result<MineServiceMiddleware, String> {
|
||||
|
@ -7,6 +7,7 @@ extern crate lazy_static;
|
||||
mod config;
|
||||
mod loader;
|
||||
mod mine;
|
||||
mod miner_id;
|
||||
pub mod pora;
|
||||
mod sealer;
|
||||
mod service;
|
||||
@ -16,4 +17,5 @@ mod watcher;
|
||||
pub use config::MinerConfig;
|
||||
pub use loader::PoraLoader;
|
||||
pub use mine::CustomMineRange;
|
||||
pub use miner_id::load_miner_id;
|
||||
pub use service::{MineService, MinerMessage};
|
||||
|
@ -82,6 +82,7 @@ impl PoraService {
|
||||
mine_context_receiver: mpsc::UnboundedReceiver<MineContextMessage>,
|
||||
loader: Arc<dyn PoraLoader>,
|
||||
config: &MinerConfig,
|
||||
miner_id: H256,
|
||||
) -> mpsc::UnboundedReceiver<AnswerWithoutProof> {
|
||||
let (mine_answer_sender, mine_answer_receiver) =
|
||||
mpsc::unbounded_channel::<AnswerWithoutProof>();
|
||||
@ -95,7 +96,7 @@ impl PoraService {
|
||||
msg_recv,
|
||||
puzzle: None,
|
||||
mine_range,
|
||||
miner_id: config.miner_id,
|
||||
miner_id,
|
||||
loader,
|
||||
cpu_percentage: config.cpu_percentage,
|
||||
iter_batch: config.iter_batch,
|
||||
|
111
node/miner/src/miner_id.rs
Normal file
111
node/miner/src/miner_id.rs
Normal file
@ -0,0 +1,111 @@
|
||||
use crate::config::MineServiceMiddleware;
|
||||
use crate::config::MinerConfig;
|
||||
use contract_interface::{NewMinerIdFilter, PoraMine};
|
||||
use ethereum_types::Address;
|
||||
use ethers::contract::ContractCall;
|
||||
use ethers::contract::EthEvent;
|
||||
use std::sync::Arc;
|
||||
use storage::log_store::{config::ConfigurableExt, Store};
|
||||
use storage::H256;
|
||||
use tokio::sync::RwLock;
|
||||
|
||||
const MINER_ID: &str = "mine.miner_id";
|
||||
|
||||
pub fn load_miner_id(store: &dyn Store) -> storage::error::Result<Option<H256>> {
|
||||
store.get_config_decoded(&MINER_ID)
|
||||
}
|
||||
|
||||
fn set_miner_id(store: &dyn Store, miner_id: &H256) -> storage::error::Result<()> {
|
||||
store.set_config_encoded(&MINER_ID, miner_id)
|
||||
}
|
||||
|
||||
pub(crate) async fn check_and_request_miner_id(
|
||||
config: &MinerConfig,
|
||||
store: &RwLock<dyn Store>,
|
||||
provider: &Arc<MineServiceMiddleware>,
|
||||
) -> Result<H256, String> {
|
||||
let db_miner_id = load_miner_id(&*store.read().await)
|
||||
.map_err(|e| format!("miner_id on db corrupt: {:?}", e))?;
|
||||
|
||||
let mine_contract = PoraMine::new(config.mine_address, provider.clone());
|
||||
|
||||
match (db_miner_id, config.miner_id) {
|
||||
(Some(d_id), Some(c_id)) => {
|
||||
if d_id != c_id {
|
||||
Err(format!(
|
||||
"database miner id {} != configuration miner id {}",
|
||||
d_id, c_id
|
||||
))
|
||||
} else {
|
||||
Ok(d_id)
|
||||
}
|
||||
}
|
||||
(None, Some(c_id)) => {
|
||||
check_miner_id(&mine_contract, c_id).await?;
|
||||
set_miner_id(&*store.write().await, &c_id)
|
||||
.map_err(|e| format!("set miner id on db corrupt: {:?}", e))?;
|
||||
Ok(c_id)
|
||||
}
|
||||
(Some(d_id), None) => {
|
||||
check_miner_id(&mine_contract, d_id).await?;
|
||||
Ok(d_id)
|
||||
}
|
||||
(None, None) => {
|
||||
let beneficiary = provider.address();
|
||||
let id = request_miner_id(&mine_contract, beneficiary).await?;
|
||||
set_miner_id(&*store.write().await, &id)
|
||||
.map_err(|e| format!("set miner id on db corrupt: {:?}", e))?;
|
||||
Ok(id)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
async fn check_miner_id(
|
||||
mine_contract: &PoraMine<MineServiceMiddleware>,
|
||||
miner_id: H256,
|
||||
) -> Result<Address, String> {
|
||||
debug!("Checking miner id on chain...");
|
||||
|
||||
let beneficiary = mine_contract
|
||||
.beneficiaries(miner_id.0)
|
||||
.call()
|
||||
.await
|
||||
.map_err(|e| format!("Fail to query miner id information: {:?}", e))?;
|
||||
|
||||
if beneficiary == Address::zero() {
|
||||
Err("candidate miner id is not registered".into())
|
||||
} else {
|
||||
Ok(beneficiary)
|
||||
}
|
||||
}
|
||||
|
||||
async fn request_miner_id(
|
||||
mine_contract: &PoraMine<MineServiceMiddleware>,
|
||||
beneficiary: Address,
|
||||
) -> Result<H256, String> {
|
||||
debug!("Requesting miner id on chain...");
|
||||
|
||||
let submission_call: ContractCall<_, _> =
|
||||
mine_contract.request_miner_id(beneficiary, 0).legacy();
|
||||
|
||||
let pending_tx = submission_call
|
||||
.send()
|
||||
.await
|
||||
.map_err(|e| format!("Fail to request miner id: {:?}", e))?;
|
||||
|
||||
let receipt = pending_tx
|
||||
.retries(3)
|
||||
.await
|
||||
.map_err(|e| format!("Fail to execute mine answer transaction: {:?}", e))?
|
||||
.ok_or("Request miner id transaction dropped after 3 retires")?;
|
||||
|
||||
let first_log = receipt
|
||||
.logs
|
||||
.first()
|
||||
.ok_or("Fail to find minerId in receipt")?;
|
||||
|
||||
let new_id_event = NewMinerIdFilter::decode_log(&first_log.clone().into())
|
||||
.map_err(|e| format!("Fail to decode NewMinerId event: {:?}", e))?;
|
||||
|
||||
Ok(H256(new_id_event.miner_id))
|
||||
}
|
@ -34,6 +34,7 @@ impl Sealer {
|
||||
provider: Arc<MineServiceMiddleware>,
|
||||
store: Arc<RwLock<dyn Store>>,
|
||||
config: &MinerConfig,
|
||||
miner_id: H256,
|
||||
) {
|
||||
let flow_contract = ZgsFlow::new(config.flow_address, provider);
|
||||
let sealer = Sealer {
|
||||
@ -41,7 +42,7 @@ impl Sealer {
|
||||
store,
|
||||
context_cache: Default::default(),
|
||||
last_context_flow_length: 0,
|
||||
miner_id: config.miner_id,
|
||||
miner_id,
|
||||
};
|
||||
|
||||
executor.spawn(async move { Box::pin(sealer.start()).await }, "data_sealer");
|
||||
|
@ -1,3 +1,4 @@
|
||||
use crate::miner_id::check_and_request_miner_id;
|
||||
use crate::sealer::Sealer;
|
||||
use crate::submitter::Submitter;
|
||||
use crate::{config::MinerConfig, mine::PoraService, watcher::MineContextWatcher};
|
||||
@ -30,6 +31,9 @@ impl MineService {
|
||||
|
||||
let (msg_send, msg_recv) = broadcast::channel(1024);
|
||||
|
||||
let miner_id = check_and_request_miner_id(&config, &store, &provider).await?;
|
||||
debug!("miner id setting complete.");
|
||||
|
||||
let mine_context_receiver = MineContextWatcher::spawn(
|
||||
executor.clone(),
|
||||
msg_recv.resubscribe(),
|
||||
@ -43,6 +47,7 @@ impl MineService {
|
||||
mine_context_receiver,
|
||||
Arc::new(store.clone()),
|
||||
&config,
|
||||
miner_id,
|
||||
);
|
||||
|
||||
Submitter::spawn(
|
||||
@ -53,7 +58,7 @@ impl MineService {
|
||||
&config,
|
||||
);
|
||||
|
||||
Sealer::spawn(executor, provider, store, &config);
|
||||
Sealer::spawn(executor, provider, store, &config, miner_id);
|
||||
|
||||
debug!("Starting miner service");
|
||||
|
||||
|
@ -97,7 +97,7 @@ impl Submitter {
|
||||
mine_length: mine_answer.mining_length.into(),
|
||||
recall_position: mine_answer.recall_position.into(),
|
||||
seal_offset: mine_answer.seal_offset.into(),
|
||||
sealed_context_digest: sealed_context_digest.digest, // TODO(kevin): wait for implementation of data sealing.
|
||||
sealed_context_digest: sealed_context_digest.digest,
|
||||
sealed_data: unsafe { std::mem::transmute(mine_answer.sealed_data) },
|
||||
merkle_proof: flow_proof_to_pora_merkle_proof(flow_proof),
|
||||
};
|
||||
|
@ -16,6 +16,8 @@ class LongTimeMineTest(TestFramework):
|
||||
"mine_iter_batch_size": 50,
|
||||
}
|
||||
self.mine_period = 15
|
||||
self.launch_wait_seconds = 15
|
||||
|
||||
|
||||
def submit_data(self, item, size):
|
||||
submissions_before = self.contract.num_submissions()
|
||||
|
@ -3,6 +3,7 @@ from test_framework.test_framework import TestFramework
|
||||
from config.node_config import MINER_ID, GENESIS_PRIV_KEY
|
||||
from utility.submission import create_submission, submit_data
|
||||
from utility.utils import wait_until
|
||||
from test_framework.blockchain_node import BlockChainNodeType
|
||||
|
||||
|
||||
class MineTest(TestFramework):
|
||||
@ -10,10 +11,10 @@ class MineTest(TestFramework):
|
||||
self.num_blockchain_nodes = 1
|
||||
self.num_nodes = 1
|
||||
self.zgs_node_configs[0] = {
|
||||
"miner_id": MINER_ID,
|
||||
"miner_key": GENESIS_PRIV_KEY,
|
||||
}
|
||||
self.mine_period = 15
|
||||
self.mine_period = int(45 / self.block_time)
|
||||
self.launch_wait_seconds = 15
|
||||
|
||||
def submit_data(self, item, size):
|
||||
submissions_before = self.contract.num_submissions()
|
||||
|
@ -3,6 +3,8 @@ from test_framework.test_framework import TestFramework
|
||||
from config.node_config import MINER_ID, GENESIS_PRIV_KEY
|
||||
from utility.submission import create_submission, submit_data
|
||||
from utility.utils import wait_until, assert_equal, assert_greater_than
|
||||
from test_framework.blockchain_node import BlockChainNodeType
|
||||
|
||||
|
||||
import math
|
||||
|
||||
@ -13,11 +15,12 @@ class MineTest(TestFramework):
|
||||
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 = 20
|
||||
self.mine_period = int(60 / self.block_time)
|
||||
self.launch_wait_seconds = 15
|
||||
|
||||
|
||||
def submit_data(self, item, size, no_submit = False):
|
||||
submissions_before = self.contract.num_submissions()
|
||||
|
@ -28,6 +28,13 @@ class BlockChainNodeType(Enum):
|
||||
BSC = 1
|
||||
Evmos = 2
|
||||
|
||||
def block_time(self):
|
||||
if self == BlockChainNodeType.Conflux:
|
||||
return 0.5
|
||||
elif self == BlockChainNodeType.BSC:
|
||||
return 0.25
|
||||
else:
|
||||
return 3.0
|
||||
|
||||
@unique
|
||||
class NodeType(Enum):
|
||||
@ -299,8 +306,8 @@ class BlockchainNode(TestNode):
|
||||
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)
|
||||
# tx_hash = mine_contract.functions.setMiner(decode_hex(MINER_ID)).transact(TX_PARAMS)
|
||||
# self.wait_for_transaction_receipt(w3, tx_hash)
|
||||
|
||||
dummy_reward_contract = w3.eth.contract(
|
||||
address = book.functions.reward().call(),
|
||||
@ -328,8 +335,8 @@ class BlockchainNode(TestNode):
|
||||
|
||||
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)
|
||||
# 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
|
||||
|
||||
|
@ -47,8 +47,10 @@ class TestFramework:
|
||||
self.blockchain_node_configs = {}
|
||||
self.zgs_node_configs = {}
|
||||
self.blockchain_node_type = blockchain_node_type
|
||||
self.block_time = blockchain_node_type.block_time()
|
||||
self.enable_market = False
|
||||
self.mine_period = 100
|
||||
self.launch_wait_seconds = 1
|
||||
|
||||
# Set default binary path
|
||||
binary_ext = ".exe" if is_windows_platform() else ""
|
||||
@ -203,7 +205,9 @@ class TestFramework:
|
||||
time.sleep(1)
|
||||
node.start()
|
||||
|
||||
time.sleep(1)
|
||||
self.log.info("Wait the zgs_node launch for %d seconds", self.launch_wait_seconds)
|
||||
time.sleep(self.launch_wait_seconds)
|
||||
|
||||
for node in self.nodes:
|
||||
node.wait_for_rpc_connection()
|
||||
|
||||
@ -268,6 +272,10 @@ class TestFramework:
|
||||
"--tmpdir", dest="tmpdir", help="Root directory for datadirs"
|
||||
)
|
||||
|
||||
parser.add_argument(
|
||||
"--devdir", dest="devdir", help="A softlink point to the last run"
|
||||
)
|
||||
|
||||
parser.add_argument(
|
||||
"--randomseed", dest="random_seed", type=int, help="Set a random seed"
|
||||
)
|
||||
@ -433,6 +441,19 @@ class TestFramework:
|
||||
self.__start_logging()
|
||||
self.log.info("Root dir: %s", self.root_dir)
|
||||
|
||||
if self.options.devdir:
|
||||
dst = self.options.devdir
|
||||
|
||||
if os.path.islink(dst):
|
||||
os.remove(dst)
|
||||
elif os.path.isdir(dst):
|
||||
shutil.rmtree(dst)
|
||||
elif os.path.exists(dst):
|
||||
os.remove(dst)
|
||||
|
||||
os.symlink(self.options.tmpdir, dst)
|
||||
self.log.info("Symlink: %s", Path(dst).absolute())
|
||||
|
||||
if self.blockchain_node_type == BlockChainNodeType.Conflux:
|
||||
self.blockchain_binary = os.path.abspath(self.options.conflux)
|
||||
elif self.blockchain_node_type == BlockChainNodeType.BSC:
|
||||
|
Loading…
Reference in New Issue
Block a user