mirror of
https://github.com/0glabs/0g-storage-node.git
synced 2025-01-24 05:55:17 +00:00
193e154361
* 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
112 lines
3.4 KiB
Rust
112 lines
3.4 KiB
Rust
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))
|
|
}
|