Decouple turns from segments in PoRep (#5004)
* Decouple Segments from Turns in Storage * Get replicator local cluster tests running in a reasonable amount of time * Fix unused imports * Document new RPC APIs * Check for exit while polling
This commit is contained in:
@ -30,6 +30,8 @@ Methods
|
|||||||
* [getSignatureStatus](#getsignaturestatus)
|
* [getSignatureStatus](#getsignaturestatus)
|
||||||
* [getSlotLeader](#getslotleader)
|
* [getSlotLeader](#getslotleader)
|
||||||
* [getSlotsPerSegment](#getslotspersegment)
|
* [getSlotsPerSegment](#getslotspersegment)
|
||||||
|
* [getStorageTurn](#getstorageturn)
|
||||||
|
* [getStorageTurnRate](#getstorageturnrate)
|
||||||
* [getNumBlocksSinceSignatureConfirmation](#getnumblockssincesignatureconfirmation)
|
* [getNumBlocksSinceSignatureConfirmation](#getnumblockssincesignatureconfirmation)
|
||||||
* [getTransactionCount](#gettransactioncount)
|
* [getTransactionCount](#gettransactioncount)
|
||||||
* [getTotalSupply](#gettotalsupply)
|
* [getTotalSupply](#gettotalsupply)
|
||||||
@ -278,13 +280,53 @@ None
|
|||||||
```bash
|
```bash
|
||||||
// Request
|
// Request
|
||||||
curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc":"2.0","id":1, "method":"getSlotsPerSegment"}' http://localhost:8899
|
curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc":"2.0","id":1, "method":"getSlotsPerSegment"}' http://localhost:8899
|
||||||
|
|
||||||
// Result
|
// Result
|
||||||
{"jsonrpc":"2.0","result":"1024","id":1}
|
{"jsonrpc":"2.0","result":"1024","id":1}
|
||||||
```
|
```
|
||||||
|
|
||||||
----
|
----
|
||||||
|
|
||||||
|
### getStorageTurn
|
||||||
|
Returns the current storage turn's blockhash and slot
|
||||||
|
|
||||||
|
##### Parameters:
|
||||||
|
None
|
||||||
|
|
||||||
|
##### Results:
|
||||||
|
An array consisting of
|
||||||
|
* `string` - a Hash as base-58 encoded string indicating the blockhash of the turn slot
|
||||||
|
* `u64` - the current storage turn slot
|
||||||
|
|
||||||
|
##### Example:
|
||||||
|
```bash
|
||||||
|
// Request
|
||||||
|
curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc":"2.0","id":1, "method":"getStorageTurn"}' http://localhost:8899
|
||||||
|
// Result
|
||||||
|
{"jsonrpc":"2.0","result":["GH7ome3EiwEr7tu9JuTh2dpYWBJK3z69Xm1ZE3MEE6JC", "2048"],"id":1}
|
||||||
|
```
|
||||||
|
|
||||||
|
----
|
||||||
|
|
||||||
|
### getStorageTurnRate
|
||||||
|
Returns the current storage turn rate in terms of slots per turn
|
||||||
|
|
||||||
|
##### Parameters:
|
||||||
|
None
|
||||||
|
|
||||||
|
##### Results:
|
||||||
|
* `u64` - Number of slots in storage turn
|
||||||
|
|
||||||
|
##### Example:
|
||||||
|
```bash
|
||||||
|
// Request
|
||||||
|
curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc":"2.0","id":1, "method":"getStorageTurnRate"}' http://localhost:8899
|
||||||
|
// Result
|
||||||
|
{"jsonrpc":"2.0","result":"1024","id":1}
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
----
|
||||||
|
|
||||||
### getNumBlocksSinceSignatureConfirmation
|
### getNumBlocksSinceSignatureConfirmation
|
||||||
Returns the current number of blocks since signature has been confirmed.
|
Returns the current number of blocks since signature has been confirmed.
|
||||||
|
|
||||||
|
@ -15,8 +15,8 @@ pub enum RpcRequest {
|
|||||||
GetSlot,
|
GetSlot,
|
||||||
GetSlotLeader,
|
GetSlotLeader,
|
||||||
GetEpochVoteAccounts,
|
GetEpochVoteAccounts,
|
||||||
GetStorageBlockhash,
|
GetStorageTurn,
|
||||||
GetStorageSlot,
|
GetStorageTurnRate,
|
||||||
GetSlotsPerSegment,
|
GetSlotsPerSegment,
|
||||||
GetStoragePubkeysForSlot,
|
GetStoragePubkeysForSlot,
|
||||||
GetTransactionCount,
|
GetTransactionCount,
|
||||||
@ -44,8 +44,8 @@ impl RpcRequest {
|
|||||||
RpcRequest::GetSlot => "getSlot",
|
RpcRequest::GetSlot => "getSlot",
|
||||||
RpcRequest::GetSlotLeader => "getSlotLeader",
|
RpcRequest::GetSlotLeader => "getSlotLeader",
|
||||||
RpcRequest::GetEpochVoteAccounts => "getEpochVoteAccounts",
|
RpcRequest::GetEpochVoteAccounts => "getEpochVoteAccounts",
|
||||||
RpcRequest::GetStorageBlockhash => "getStorageBlockhash",
|
RpcRequest::GetStorageTurn => "getStorageTurn",
|
||||||
RpcRequest::GetStorageSlot => "getStorageSlot",
|
RpcRequest::GetStorageTurnRate => "getStorageTurnRate",
|
||||||
RpcRequest::GetSlotsPerSegment => "getSlotsPerSegment",
|
RpcRequest::GetSlotsPerSegment => "getSlotsPerSegment",
|
||||||
RpcRequest::GetStoragePubkeysForSlot => "getStoragePubkeysForSlot",
|
RpcRequest::GetStoragePubkeysForSlot => "getStoragePubkeysForSlot",
|
||||||
RpcRequest::GetTransactionCount => "getTransactionCount",
|
RpcRequest::GetTransactionCount => "getTransactionCount",
|
||||||
|
@ -16,8 +16,8 @@ use solana_sdk::poh_config::PohConfig;
|
|||||||
use solana_sdk::pubkey::Pubkey;
|
use solana_sdk::pubkey::Pubkey;
|
||||||
use solana_sdk::signature::{Keypair, KeypairUtil};
|
use solana_sdk::signature::{Keypair, KeypairUtil};
|
||||||
use solana_sdk::system_transaction;
|
use solana_sdk::system_transaction;
|
||||||
use solana_sdk::timing::DEFAULT_SLOTS_PER_EPOCH;
|
|
||||||
use solana_sdk::timing::DEFAULT_TICKS_PER_SLOT;
|
use solana_sdk::timing::DEFAULT_TICKS_PER_SLOT;
|
||||||
|
use solana_sdk::timing::{DEFAULT_SLOTS_PER_EPOCH, DEFAULT_SLOTS_PER_SEGMENT};
|
||||||
use solana_sdk::transaction::Transaction;
|
use solana_sdk::transaction::Transaction;
|
||||||
use solana_stake_api::stake_instruction;
|
use solana_stake_api::stake_instruction;
|
||||||
use solana_storage_api::storage_contract;
|
use solana_storage_api::storage_contract;
|
||||||
@ -80,6 +80,7 @@ pub struct ClusterConfig {
|
|||||||
pub cluster_lamports: u64,
|
pub cluster_lamports: u64,
|
||||||
pub ticks_per_slot: u64,
|
pub ticks_per_slot: u64,
|
||||||
pub slots_per_epoch: u64,
|
pub slots_per_epoch: u64,
|
||||||
|
pub slots_per_segment: u64,
|
||||||
pub stakers_slot_offset: u64,
|
pub stakers_slot_offset: u64,
|
||||||
pub native_instruction_processors: Vec<(String, Pubkey)>,
|
pub native_instruction_processors: Vec<(String, Pubkey)>,
|
||||||
pub poh_config: PohConfig,
|
pub poh_config: PohConfig,
|
||||||
@ -95,6 +96,7 @@ impl Default for ClusterConfig {
|
|||||||
cluster_lamports: 0,
|
cluster_lamports: 0,
|
||||||
ticks_per_slot: DEFAULT_TICKS_PER_SLOT,
|
ticks_per_slot: DEFAULT_TICKS_PER_SLOT,
|
||||||
slots_per_epoch: DEFAULT_SLOTS_PER_EPOCH,
|
slots_per_epoch: DEFAULT_SLOTS_PER_EPOCH,
|
||||||
|
slots_per_segment: DEFAULT_SLOTS_PER_SEGMENT,
|
||||||
stakers_slot_offset: DEFAULT_SLOTS_PER_EPOCH,
|
stakers_slot_offset: DEFAULT_SLOTS_PER_EPOCH,
|
||||||
native_instruction_processors: vec![],
|
native_instruction_processors: vec![],
|
||||||
poh_config: PohConfig::default(),
|
poh_config: PohConfig::default(),
|
||||||
@ -147,6 +149,7 @@ impl LocalCluster {
|
|||||||
);
|
);
|
||||||
genesis_block.ticks_per_slot = config.ticks_per_slot;
|
genesis_block.ticks_per_slot = config.ticks_per_slot;
|
||||||
genesis_block.slots_per_epoch = config.slots_per_epoch;
|
genesis_block.slots_per_epoch = config.slots_per_epoch;
|
||||||
|
genesis_block.slots_per_segment = config.slots_per_segment;
|
||||||
genesis_block.stakers_slot_offset = config.stakers_slot_offset;
|
genesis_block.stakers_slot_offset = config.stakers_slot_offset;
|
||||||
genesis_block.poh_config = config.poh_config.clone();
|
genesis_block.poh_config = config.poh_config.clone();
|
||||||
genesis_block
|
genesis_block
|
||||||
@ -574,7 +577,7 @@ impl Drop for LocalCluster {
|
|||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod test {
|
mod test {
|
||||||
use super::*;
|
use super::*;
|
||||||
use crate::storage_stage::STORAGE_ROTATE_TEST_COUNT;
|
use crate::storage_stage::SLOTS_PER_TURN_TEST;
|
||||||
use solana_runtime::epoch_schedule::MINIMUM_SLOTS_PER_EPOCH;
|
use solana_runtime::epoch_schedule::MINIMUM_SLOTS_PER_EPOCH;
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
@ -591,7 +594,7 @@ mod test {
|
|||||||
solana_logger::setup();
|
solana_logger::setup();
|
||||||
let mut validator_config = ValidatorConfig::default();
|
let mut validator_config = ValidatorConfig::default();
|
||||||
validator_config.rpc_config.enable_fullnode_exit = true;
|
validator_config.rpc_config.enable_fullnode_exit = true;
|
||||||
validator_config.storage_rotate_count = STORAGE_ROTATE_TEST_COUNT;
|
validator_config.storage_slots_per_turn = SLOTS_PER_TURN_TEST;
|
||||||
const NUM_NODES: usize = 1;
|
const NUM_NODES: usize = 1;
|
||||||
let num_replicators = 1;
|
let num_replicators = 1;
|
||||||
let config = ClusterConfig {
|
let config = ClusterConfig {
|
||||||
|
@ -104,7 +104,7 @@ pub(crate) fn sample_file(in_path: &Path, sample_offsets: &[u64]) -> io::Result<
|
|||||||
Ok(hasher.result())
|
Ok(hasher.result())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn get_slot_from_blockhash(
|
fn get_slot_from_signature(
|
||||||
signature: &ed25519_dalek::Signature,
|
signature: &ed25519_dalek::Signature,
|
||||||
storage_turn: u64,
|
storage_turn: u64,
|
||||||
slots_per_segment: u64,
|
slots_per_segment: u64,
|
||||||
@ -265,7 +265,7 @@ impl Replicator {
|
|||||||
};
|
};
|
||||||
spawn(move || {
|
spawn(move || {
|
||||||
// setup replicator
|
// setup replicator
|
||||||
let window_service = Self::setup(
|
let window_service = match Self::setup(
|
||||||
&mut meta,
|
&mut meta,
|
||||||
cluster_info.clone(),
|
cluster_info.clone(),
|
||||||
&blocktree,
|
&blocktree,
|
||||||
@ -275,8 +275,21 @@ impl Replicator {
|
|||||||
repair_socket,
|
repair_socket,
|
||||||
blob_fetch_receiver,
|
blob_fetch_receiver,
|
||||||
slot_sender,
|
slot_sender,
|
||||||
)
|
) {
|
||||||
.ok();
|
Ok(window_service) => window_service,
|
||||||
|
Err(e) => {
|
||||||
|
//shutdown services before exiting
|
||||||
|
error!("setup failed {:?}; replicator thread exiting...", e);
|
||||||
|
exit.store(true, Ordering::Relaxed);
|
||||||
|
request_processor
|
||||||
|
.into_iter()
|
||||||
|
.for_each(|t| t.join().unwrap());
|
||||||
|
fetch_stage.join().unwrap();
|
||||||
|
gossip_service.join().unwrap();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
info!("setup complete");
|
info!("setup complete");
|
||||||
// run replicator
|
// run replicator
|
||||||
Self::run(
|
Self::run(
|
||||||
@ -293,9 +306,7 @@ impl Replicator {
|
|||||||
.for_each(|t| t.join().unwrap());
|
.for_each(|t| t.join().unwrap());
|
||||||
fetch_stage.join().unwrap();
|
fetch_stage.join().unwrap();
|
||||||
gossip_service.join().unwrap();
|
gossip_service.join().unwrap();
|
||||||
if let Some(window) = window_service {
|
window_service.join().unwrap()
|
||||||
window.join().unwrap()
|
|
||||||
}
|
|
||||||
})
|
})
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -342,6 +353,7 @@ impl Replicator {
|
|||||||
&cluster_info,
|
&cluster_info,
|
||||||
meta.slots_per_segment,
|
meta.slots_per_segment,
|
||||||
&meta.blockhash,
|
&meta.blockhash,
|
||||||
|
exit,
|
||||||
) {
|
) {
|
||||||
Ok(blockhash_and_slot) => blockhash_and_slot,
|
Ok(blockhash_and_slot) => blockhash_and_slot,
|
||||||
Err(e) => {
|
Err(e) => {
|
||||||
@ -410,27 +422,27 @@ impl Replicator {
|
|||||||
return Err(e);
|
return Err(e);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
let (storage_blockhash, storage_slot) = match Self::poll_for_blockhash_and_slot(
|
let (segment_blockhash, segment_slot) = match Self::poll_for_segment(
|
||||||
&cluster_info,
|
&cluster_info,
|
||||||
slots_per_segment,
|
slots_per_segment,
|
||||||
&Hash::default(),
|
&Hash::default(),
|
||||||
|
exit,
|
||||||
) {
|
) {
|
||||||
Ok(blockhash_and_slot) => blockhash_and_slot,
|
Ok(blockhash_and_slot) => blockhash_and_slot,
|
||||||
Err(e) => {
|
Err(e) => {
|
||||||
error!("unable to get turn status, exiting...");
|
|
||||||
//shutdown services before exiting
|
//shutdown services before exiting
|
||||||
exit.store(true, Ordering::Relaxed);
|
exit.store(true, Ordering::Relaxed);
|
||||||
return Err(e);
|
return Err(e);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
let signature = storage_keypair.sign(storage_blockhash.as_ref());
|
let signature = storage_keypair.sign(segment_blockhash.as_ref());
|
||||||
let slot = get_slot_from_blockhash(&signature, storage_slot, slots_per_segment);
|
let slot = get_slot_from_signature(&signature, segment_slot, slots_per_segment);
|
||||||
info!("replicating slot: {}", slot);
|
info!("replicating slot: {}", slot);
|
||||||
slot_sender.send(slot)?;
|
slot_sender.send(slot)?;
|
||||||
meta.slot = slot;
|
meta.slot = slot;
|
||||||
meta.slots_per_segment = slots_per_segment;
|
meta.slots_per_segment = slots_per_segment;
|
||||||
meta.signature = Signature::new(&signature.to_bytes());
|
meta.signature = Signature::new(&signature.to_bytes());
|
||||||
meta.blockhash = storage_blockhash;
|
meta.blockhash = segment_blockhash;
|
||||||
|
|
||||||
let mut repair_slot_range = RepairSlotRange::default();
|
let mut repair_slot_range = RepairSlotRange::default();
|
||||||
repair_slot_range.end = slot + slots_per_segment;
|
repair_slot_range.end = slot + slots_per_segment;
|
||||||
@ -682,13 +694,35 @@ impl Replicator {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Waits until the first segment is ready, and returns the current segment
|
||||||
|
fn poll_for_segment(
|
||||||
|
cluster_info: &Arc<RwLock<ClusterInfo>>,
|
||||||
|
slots_per_segment: u64,
|
||||||
|
previous_blockhash: &Hash,
|
||||||
|
exit: &Arc<AtomicBool>,
|
||||||
|
) -> result::Result<(Hash, u64), Error> {
|
||||||
|
loop {
|
||||||
|
let (blockhash, turn_slot) = Self::poll_for_blockhash_and_slot(
|
||||||
|
cluster_info,
|
||||||
|
slots_per_segment,
|
||||||
|
previous_blockhash,
|
||||||
|
exit,
|
||||||
|
)?;
|
||||||
|
if get_segment_from_slot(turn_slot, slots_per_segment) != 0 {
|
||||||
|
return Ok((blockhash, turn_slot));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// Poll for a different blockhash and associated max_slot than `previous_blockhash`
|
/// Poll for a different blockhash and associated max_slot than `previous_blockhash`
|
||||||
fn poll_for_blockhash_and_slot(
|
fn poll_for_blockhash_and_slot(
|
||||||
cluster_info: &Arc<RwLock<ClusterInfo>>,
|
cluster_info: &Arc<RwLock<ClusterInfo>>,
|
||||||
slots_per_segment: u64,
|
slots_per_segment: u64,
|
||||||
previous_blockhash: &Hash,
|
previous_blockhash: &Hash,
|
||||||
|
exit: &Arc<AtomicBool>,
|
||||||
) -> result::Result<(Hash, u64), Error> {
|
) -> result::Result<(Hash, u64), Error> {
|
||||||
for _ in 0..10 {
|
info!("waiting for the next turn...");
|
||||||
|
loop {
|
||||||
let rpc_peers = {
|
let rpc_peers = {
|
||||||
let cluster_info = cluster_info.read().unwrap();
|
let cluster_info = cluster_info.read().unwrap();
|
||||||
cluster_info.rpc_peers()
|
cluster_info.rpc_peers()
|
||||||
@ -700,19 +734,19 @@ impl Replicator {
|
|||||||
RpcClient::new_socket(rpc_peers[node_index].rpc)
|
RpcClient::new_socket(rpc_peers[node_index].rpc)
|
||||||
};
|
};
|
||||||
let response = rpc_client
|
let response = rpc_client
|
||||||
.retry_make_rpc_request(&RpcRequest::GetStorageBlockhash, None, 0)
|
.retry_make_rpc_request(&RpcRequest::GetStorageTurn, None, 0)
|
||||||
.map_err(|err| {
|
.map_err(|err| {
|
||||||
warn!("Error while making rpc request {:?}", err);
|
warn!("Error while making rpc request {:?}", err);
|
||||||
Error::IO(io::Error::new(ErrorKind::Other, "rpc error"))
|
Error::IO(io::Error::new(ErrorKind::Other, "rpc error"))
|
||||||
})?;
|
})?;
|
||||||
let storage_blockhash =
|
let (storage_blockhash, turn_slot) =
|
||||||
serde_json::from_value::<(String)>(response).map_err(|err| {
|
serde_json::from_value::<((String, u64))>(response).map_err(|err| {
|
||||||
io::Error::new(
|
io::Error::new(
|
||||||
io::ErrorKind::Other,
|
io::ErrorKind::Other,
|
||||||
format!("Couldn't parse response: {:?}", err),
|
format!("Couldn't parse response: {:?}", err),
|
||||||
)
|
)
|
||||||
})?;
|
})?;
|
||||||
let storage_blockhash = storage_blockhash.parse().map_err(|err| {
|
let turn_blockhash = storage_blockhash.parse().map_err(|err| {
|
||||||
io::Error::new(
|
io::Error::new(
|
||||||
io::ErrorKind::Other,
|
io::ErrorKind::Other,
|
||||||
format!(
|
format!(
|
||||||
@ -721,28 +755,21 @@ impl Replicator {
|
|||||||
),
|
),
|
||||||
)
|
)
|
||||||
})?;
|
})?;
|
||||||
if storage_blockhash != *previous_blockhash {
|
if turn_blockhash != *previous_blockhash {
|
||||||
let storage_slot = rpc_client
|
info!("turn slot: {}", turn_slot);
|
||||||
.retry_make_rpc_request(&RpcRequest::GetStorageSlot, None, 0)
|
if get_segment_from_slot(turn_slot, slots_per_segment) != 0 {
|
||||||
.map_err(|err| {
|
return Ok((turn_blockhash, turn_slot));
|
||||||
warn!("Error while making rpc request {:?}", err);
|
|
||||||
Error::IO(io::Error::new(ErrorKind::Other, "rpc error"))
|
|
||||||
})?
|
|
||||||
.as_u64()
|
|
||||||
.unwrap();
|
|
||||||
info!("storage slot: {}", storage_slot);
|
|
||||||
if get_segment_from_slot(storage_slot, slots_per_segment) != 0 {
|
|
||||||
return Ok((storage_blockhash, storage_slot));
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
info!("waiting for segment...");
|
if exit.load(Ordering::Relaxed) {
|
||||||
|
return Err(Error::IO(io::Error::new(
|
||||||
|
ErrorKind::Other,
|
||||||
|
"exit signalled...",
|
||||||
|
)));
|
||||||
|
}
|
||||||
sleep(Duration::from_secs(5));
|
sleep(Duration::from_secs(5));
|
||||||
}
|
}
|
||||||
Err(io::Error::new(
|
|
||||||
ErrorKind::Other,
|
|
||||||
"Couldn't get blockhash or slot",
|
|
||||||
))?
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Ask a replicator to populate a given blocktree with its segment.
|
/// Ask a replicator to populate a given blocktree with its segment.
|
||||||
@ -867,7 +894,7 @@ impl Replicator {
|
|||||||
}
|
}
|
||||||
sleep(Duration::from_millis(500));
|
sleep(Duration::from_millis(500));
|
||||||
}
|
}
|
||||||
panic!("Couldn't get slot height!");
|
panic!("Couldn't get segment slot from replicator!");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -139,12 +139,15 @@ impl JsonRpcRequestProcessor {
|
|||||||
.collect::<Vec<_>>())
|
.collect::<Vec<_>>())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn get_storage_blockhash(&self) -> Result<String> {
|
fn get_storage_turn_rate(&self) -> Result<u64> {
|
||||||
Ok(self.storage_state.get_storage_blockhash().to_string())
|
Ok(self.storage_state.get_storage_turn_rate())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn get_storage_slot(&self) -> Result<u64> {
|
fn get_storage_turn(&self) -> Result<(String, u64)> {
|
||||||
Ok(self.storage_state.get_slot())
|
Ok((
|
||||||
|
self.storage_state.get_storage_blockhash().to_string(),
|
||||||
|
self.storage_state.get_slot(),
|
||||||
|
))
|
||||||
}
|
}
|
||||||
|
|
||||||
fn get_slots_per_segment(&self) -> Result<u64> {
|
fn get_slots_per_segment(&self) -> Result<u64> {
|
||||||
@ -267,11 +270,11 @@ pub trait RpcSol {
|
|||||||
#[rpc(meta, name = "getEpochVoteAccounts")]
|
#[rpc(meta, name = "getEpochVoteAccounts")]
|
||||||
fn get_epoch_vote_accounts(&self, _: Self::Metadata) -> Result<Vec<RpcVoteAccountInfo>>;
|
fn get_epoch_vote_accounts(&self, _: Self::Metadata) -> Result<Vec<RpcVoteAccountInfo>>;
|
||||||
|
|
||||||
#[rpc(meta, name = "getStorageBlockhash")]
|
#[rpc(meta, name = "getStorageTurnRate")]
|
||||||
fn get_storage_blockhash(&self, _: Self::Metadata) -> Result<String>;
|
fn get_storage_turn_rate(&self, _: Self::Metadata) -> Result<u64>;
|
||||||
|
|
||||||
#[rpc(meta, name = "getStorageSlot")]
|
#[rpc(meta, name = "getStorageTurn")]
|
||||||
fn get_storage_slot(&self, _: Self::Metadata) -> Result<u64>;
|
fn get_storage_turn(&self, _: Self::Metadata) -> Result<(String, u64)>;
|
||||||
|
|
||||||
#[rpc(meta, name = "getSlotsPerSegment")]
|
#[rpc(meta, name = "getSlotsPerSegment")]
|
||||||
fn get_slots_per_segment(&self, _: Self::Metadata) -> Result<u64>;
|
fn get_slots_per_segment(&self, _: Self::Metadata) -> Result<u64>;
|
||||||
@ -526,15 +529,15 @@ impl RpcSol for RpcSolImpl {
|
|||||||
.get_epoch_vote_accounts()
|
.get_epoch_vote_accounts()
|
||||||
}
|
}
|
||||||
|
|
||||||
fn get_storage_blockhash(&self, meta: Self::Metadata) -> Result<String> {
|
fn get_storage_turn_rate(&self, meta: Self::Metadata) -> Result<u64> {
|
||||||
meta.request_processor
|
meta.request_processor
|
||||||
.read()
|
.read()
|
||||||
.unwrap()
|
.unwrap()
|
||||||
.get_storage_blockhash()
|
.get_storage_turn_rate()
|
||||||
}
|
}
|
||||||
|
|
||||||
fn get_storage_slot(&self, meta: Self::Metadata) -> Result<u64> {
|
fn get_storage_turn(&self, meta: Self::Metadata) -> Result<(String, u64)> {
|
||||||
meta.request_processor.read().unwrap().get_storage_slot()
|
meta.request_processor.read().unwrap().get_storage_turn()
|
||||||
}
|
}
|
||||||
|
|
||||||
fn get_slots_per_segment(&self, meta: Self::Metadata) -> Result<u64> {
|
fn get_slots_per_segment(&self, meta: Self::Metadata) -> Result<u64> {
|
||||||
|
@ -49,6 +49,7 @@ pub struct StorageStateInner {
|
|||||||
storage_blockhash: Hash,
|
storage_blockhash: Hash,
|
||||||
slot: u64,
|
slot: u64,
|
||||||
slots_per_segment: u64,
|
slots_per_segment: u64,
|
||||||
|
slots_per_turn: u64,
|
||||||
}
|
}
|
||||||
|
|
||||||
// Used to track root slots in storage stage
|
// Used to track root slots in storage stage
|
||||||
@ -69,7 +70,7 @@ pub struct StorageStage {
|
|||||||
t_storage_create_accounts: JoinHandle<()>,
|
t_storage_create_accounts: JoinHandle<()>,
|
||||||
}
|
}
|
||||||
|
|
||||||
pub const STORAGE_ROTATE_TEST_COUNT: u64 = 2;
|
pub const SLOTS_PER_TURN_TEST: u64 = 2;
|
||||||
// TODO: some way to dynamically size NUM_IDENTITIES
|
// TODO: some way to dynamically size NUM_IDENTITIES
|
||||||
const NUM_IDENTITIES: usize = 1024;
|
const NUM_IDENTITIES: usize = 1024;
|
||||||
pub const NUM_STORAGE_SAMPLES: usize = 4;
|
pub const NUM_STORAGE_SAMPLES: usize = 4;
|
||||||
@ -88,7 +89,7 @@ fn get_identity_index_from_signature(key: &Signature) -> usize {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl StorageState {
|
impl StorageState {
|
||||||
pub fn new(hash: &Hash, slots_per_segment: u64) -> Self {
|
pub fn new(hash: &Hash, slots_per_turn: u64, slots_per_segment: u64) -> Self {
|
||||||
let storage_keys = vec![0u8; KEY_SIZE * NUM_IDENTITIES];
|
let storage_keys = vec![0u8; KEY_SIZE * NUM_IDENTITIES];
|
||||||
let storage_results = vec![Hash::default(); NUM_IDENTITIES];
|
let storage_results = vec![Hash::default(); NUM_IDENTITIES];
|
||||||
let replicator_map = vec![];
|
let replicator_map = vec![];
|
||||||
@ -97,6 +98,7 @@ impl StorageState {
|
|||||||
storage_keys,
|
storage_keys,
|
||||||
storage_results,
|
storage_results,
|
||||||
replicator_map,
|
replicator_map,
|
||||||
|
slots_per_turn,
|
||||||
slot: 0,
|
slot: 0,
|
||||||
slots_per_segment,
|
slots_per_segment,
|
||||||
storage_blockhash: *hash,
|
storage_blockhash: *hash,
|
||||||
@ -121,6 +123,10 @@ impl StorageState {
|
|||||||
self.state.read().unwrap().storage_blockhash
|
self.state.read().unwrap().storage_blockhash
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn get_storage_turn_rate(&self) -> u64 {
|
||||||
|
self.state.read().unwrap().slots_per_turn
|
||||||
|
}
|
||||||
|
|
||||||
pub fn get_slot(&self) -> u64 {
|
pub fn get_slot(&self) -> u64 {
|
||||||
self.state.read().unwrap().slot
|
self.state.read().unwrap().slot
|
||||||
}
|
}
|
||||||
@ -171,12 +177,12 @@ impl StorageStage {
|
|||||||
storage_keypair: &Arc<Keypair>,
|
storage_keypair: &Arc<Keypair>,
|
||||||
exit: &Arc<AtomicBool>,
|
exit: &Arc<AtomicBool>,
|
||||||
bank_forks: &Arc<RwLock<BankForks>>,
|
bank_forks: &Arc<RwLock<BankForks>>,
|
||||||
storage_rotate_count: u64,
|
|
||||||
cluster_info: &Arc<RwLock<ClusterInfo>>,
|
cluster_info: &Arc<RwLock<ClusterInfo>>,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
let (instruction_sender, instruction_receiver) = channel();
|
let (instruction_sender, instruction_receiver) = channel();
|
||||||
|
|
||||||
let t_storage_mining_verifier = {
|
let t_storage_mining_verifier = {
|
||||||
|
let slots_per_turn = storage_state.state.read().unwrap().slots_per_turn;
|
||||||
let storage_state_inner = storage_state.state.clone();
|
let storage_state_inner = storage_state.state.clone();
|
||||||
let exit = exit.clone();
|
let exit = exit.clone();
|
||||||
let storage_keypair = storage_keypair.clone();
|
let storage_keypair = storage_keypair.clone();
|
||||||
@ -194,7 +200,7 @@ impl StorageStage {
|
|||||||
&some_blocktree,
|
&some_blocktree,
|
||||||
&mut storage_slots,
|
&mut storage_slots,
|
||||||
&mut current_key,
|
&mut current_key,
|
||||||
storage_rotate_count,
|
slots_per_turn,
|
||||||
&instruction_sender,
|
&instruction_sender,
|
||||||
) {
|
) {
|
||||||
match e {
|
match e {
|
||||||
@ -488,7 +494,7 @@ impl StorageStage {
|
|||||||
blocktree: &Arc<Blocktree>,
|
blocktree: &Arc<Blocktree>,
|
||||||
storage_slots: &mut StorageSlots,
|
storage_slots: &mut StorageSlots,
|
||||||
current_key_idx: &mut usize,
|
current_key_idx: &mut usize,
|
||||||
storage_rotate_count: u64,
|
slots_per_turn: u64,
|
||||||
instruction_sender: &InstructionSender,
|
instruction_sender: &InstructionSender,
|
||||||
) -> Result<()> {
|
) -> Result<()> {
|
||||||
let timeout = Duration::new(1, 0);
|
let timeout = Duration::new(1, 0);
|
||||||
@ -503,8 +509,7 @@ impl StorageStage {
|
|||||||
if bank.slot() > storage_slots.last_root {
|
if bank.slot() > storage_slots.last_root {
|
||||||
storage_slots.slot_count += 1;
|
storage_slots.slot_count += 1;
|
||||||
storage_slots.last_root = bank.slot();
|
storage_slots.last_root = bank.slot();
|
||||||
|
if storage_slots.slot_count % slots_per_turn == 0 {
|
||||||
if storage_slots.slot_count % storage_rotate_count == 0 {
|
|
||||||
// load all the replicator accounts in the bank. collect all their proofs at the current slot
|
// load all the replicator accounts in the bank. collect all their proofs at the current slot
|
||||||
let replicator_accounts = replicator_accounts(bank.as_ref());
|
let replicator_accounts = replicator_accounts(bank.as_ref());
|
||||||
// find proofs, and use them to update
|
// find proofs, and use them to update
|
||||||
@ -656,7 +661,11 @@ mod tests {
|
|||||||
let bank = Arc::new(Bank::new(&genesis_block));
|
let bank = Arc::new(Bank::new(&genesis_block));
|
||||||
let bank_forks = Arc::new(RwLock::new(BankForks::new_from_banks(&[bank.clone()], 0)));
|
let bank_forks = Arc::new(RwLock::new(BankForks::new_from_banks(&[bank.clone()], 0)));
|
||||||
let (_slot_sender, slot_receiver) = channel();
|
let (_slot_sender, slot_receiver) = channel();
|
||||||
let storage_state = StorageState::new(&bank.last_blockhash(), bank.slots_per_segment());
|
let storage_state = StorageState::new(
|
||||||
|
&bank.last_blockhash(),
|
||||||
|
SLOTS_PER_TURN_TEST,
|
||||||
|
bank.slots_per_segment(),
|
||||||
|
);
|
||||||
let storage_stage = StorageStage::new(
|
let storage_stage = StorageStage::new(
|
||||||
&storage_state,
|
&storage_state,
|
||||||
slot_receiver,
|
slot_receiver,
|
||||||
@ -665,7 +674,6 @@ mod tests {
|
|||||||
&storage_keypair,
|
&storage_keypair,
|
||||||
&exit.clone(),
|
&exit.clone(),
|
||||||
&bank_forks,
|
&bank_forks,
|
||||||
STORAGE_ROTATE_TEST_COUNT,
|
|
||||||
&cluster_info,
|
&cluster_info,
|
||||||
);
|
);
|
||||||
exit.store(true, Ordering::Relaxed);
|
exit.store(true, Ordering::Relaxed);
|
||||||
@ -695,7 +703,11 @@ mod tests {
|
|||||||
|
|
||||||
let cluster_info = test_cluster_info(&keypair.pubkey());
|
let cluster_info = test_cluster_info(&keypair.pubkey());
|
||||||
let (bank_sender, bank_receiver) = channel();
|
let (bank_sender, bank_receiver) = channel();
|
||||||
let storage_state = StorageState::new(&bank.last_blockhash(), bank.slots_per_segment());
|
let storage_state = StorageState::new(
|
||||||
|
&bank.last_blockhash(),
|
||||||
|
SLOTS_PER_TURN_TEST,
|
||||||
|
bank.slots_per_segment(),
|
||||||
|
);
|
||||||
let storage_stage = StorageStage::new(
|
let storage_stage = StorageStage::new(
|
||||||
&storage_state,
|
&storage_state,
|
||||||
bank_receiver,
|
bank_receiver,
|
||||||
@ -704,7 +716,6 @@ mod tests {
|
|||||||
&storage_keypair,
|
&storage_keypair,
|
||||||
&exit.clone(),
|
&exit.clone(),
|
||||||
&bank_forks,
|
&bank_forks,
|
||||||
STORAGE_ROTATE_TEST_COUNT,
|
|
||||||
&cluster_info,
|
&cluster_info,
|
||||||
);
|
);
|
||||||
bank_sender.send(vec![bank.clone()]).unwrap();
|
bank_sender.send(vec![bank.clone()]).unwrap();
|
||||||
@ -784,7 +795,11 @@ mod tests {
|
|||||||
let cluster_info = test_cluster_info(&keypair.pubkey());
|
let cluster_info = test_cluster_info(&keypair.pubkey());
|
||||||
|
|
||||||
let (bank_sender, bank_receiver) = channel();
|
let (bank_sender, bank_receiver) = channel();
|
||||||
let storage_state = StorageState::new(&bank.last_blockhash(), bank.slots_per_segment());
|
let storage_state = StorageState::new(
|
||||||
|
&bank.last_blockhash(),
|
||||||
|
SLOTS_PER_TURN_TEST,
|
||||||
|
bank.slots_per_segment(),
|
||||||
|
);
|
||||||
let storage_stage = StorageStage::new(
|
let storage_stage = StorageStage::new(
|
||||||
&storage_state,
|
&storage_state,
|
||||||
bank_receiver,
|
bank_receiver,
|
||||||
@ -793,7 +808,6 @@ mod tests {
|
|||||||
&storage_keypair,
|
&storage_keypair,
|
||||||
&exit.clone(),
|
&exit.clone(),
|
||||||
&bank_forks,
|
&bank_forks,
|
||||||
STORAGE_ROTATE_TEST_COUNT,
|
|
||||||
&cluster_info,
|
&cluster_info,
|
||||||
);
|
);
|
||||||
bank_sender.send(vec![bank.clone()]).unwrap();
|
bank_sender.send(vec![bank.clone()]).unwrap();
|
||||||
|
@ -62,7 +62,6 @@ impl Tvu {
|
|||||||
cluster_info: &Arc<RwLock<ClusterInfo>>,
|
cluster_info: &Arc<RwLock<ClusterInfo>>,
|
||||||
sockets: Sockets,
|
sockets: Sockets,
|
||||||
blocktree: Arc<Blocktree>,
|
blocktree: Arc<Blocktree>,
|
||||||
storage_rotate_count: u64,
|
|
||||||
storage_state: &StorageState,
|
storage_state: &StorageState,
|
||||||
blockstream: Option<&String>,
|
blockstream: Option<&String>,
|
||||||
ledger_signal_receiver: Receiver<bool>,
|
ledger_signal_receiver: Receiver<bool>,
|
||||||
@ -145,7 +144,6 @@ impl Tvu {
|
|||||||
storage_keypair,
|
storage_keypair,
|
||||||
&exit,
|
&exit,
|
||||||
&bank_forks,
|
&bank_forks,
|
||||||
storage_rotate_count,
|
|
||||||
&cluster_info,
|
&cluster_info,
|
||||||
);
|
);
|
||||||
|
|
||||||
@ -181,7 +179,6 @@ pub mod tests {
|
|||||||
use crate::blocktree::get_tmp_ledger_path;
|
use crate::blocktree::get_tmp_ledger_path;
|
||||||
use crate::cluster_info::{ClusterInfo, Node};
|
use crate::cluster_info::{ClusterInfo, Node};
|
||||||
use crate::genesis_utils::{create_genesis_block, GenesisBlockInfo};
|
use crate::genesis_utils::{create_genesis_block, GenesisBlockInfo};
|
||||||
use crate::storage_stage::STORAGE_ROTATE_TEST_COUNT;
|
|
||||||
use solana_runtime::bank::Bank;
|
use solana_runtime::bank::Bank;
|
||||||
use std::sync::atomic::Ordering;
|
use std::sync::atomic::Ordering;
|
||||||
|
|
||||||
@ -227,7 +224,6 @@ pub mod tests {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
blocktree,
|
blocktree,
|
||||||
STORAGE_ROTATE_TEST_COUNT,
|
|
||||||
&StorageState::default(),
|
&StorageState::default(),
|
||||||
None,
|
None,
|
||||||
l_receiver,
|
l_receiver,
|
||||||
|
@ -23,7 +23,7 @@ use solana_sdk::genesis_block::GenesisBlock;
|
|||||||
use solana_sdk::poh_config::PohConfig;
|
use solana_sdk::poh_config::PohConfig;
|
||||||
use solana_sdk::pubkey::Pubkey;
|
use solana_sdk::pubkey::Pubkey;
|
||||||
use solana_sdk::signature::{Keypair, KeypairUtil};
|
use solana_sdk::signature::{Keypair, KeypairUtil};
|
||||||
use solana_sdk::timing::{timestamp, DEFAULT_SLOTS_PER_SEGMENT};
|
use solana_sdk::timing::{timestamp, DEFAULT_SLOTS_PER_TURN};
|
||||||
use std::net::{IpAddr, Ipv4Addr, SocketAddr};
|
use std::net::{IpAddr, Ipv4Addr, SocketAddr};
|
||||||
use std::sync::atomic::{AtomicBool, Ordering};
|
use std::sync::atomic::{AtomicBool, Ordering};
|
||||||
use std::sync::mpsc::Receiver;
|
use std::sync::mpsc::Receiver;
|
||||||
@ -35,7 +35,7 @@ pub struct ValidatorConfig {
|
|||||||
pub sigverify_disabled: bool,
|
pub sigverify_disabled: bool,
|
||||||
pub voting_disabled: bool,
|
pub voting_disabled: bool,
|
||||||
pub blockstream: Option<String>,
|
pub blockstream: Option<String>,
|
||||||
pub storage_rotate_count: u64,
|
pub storage_slots_per_turn: u64,
|
||||||
pub account_paths: Option<String>,
|
pub account_paths: Option<String>,
|
||||||
pub rpc_config: JsonRpcConfig,
|
pub rpc_config: JsonRpcConfig,
|
||||||
pub snapshot_path: Option<String>,
|
pub snapshot_path: Option<String>,
|
||||||
@ -44,15 +44,11 @@ pub struct ValidatorConfig {
|
|||||||
|
|
||||||
impl Default for ValidatorConfig {
|
impl Default for ValidatorConfig {
|
||||||
fn default() -> Self {
|
fn default() -> Self {
|
||||||
// TODO: remove this, temporary parameter to configure
|
|
||||||
// storage amount differently for test configurations
|
|
||||||
// so tests don't take forever to run.
|
|
||||||
const NUM_HASHES_FOR_STORAGE_ROTATE: u64 = DEFAULT_SLOTS_PER_SEGMENT;
|
|
||||||
Self {
|
Self {
|
||||||
sigverify_disabled: false,
|
sigverify_disabled: false,
|
||||||
voting_disabled: false,
|
voting_disabled: false,
|
||||||
blockstream: None,
|
blockstream: None,
|
||||||
storage_rotate_count: NUM_HASHES_FOR_STORAGE_ROTATE,
|
storage_slots_per_turn: DEFAULT_SLOTS_PER_TURN,
|
||||||
account_paths: None,
|
account_paths: None,
|
||||||
rpc_config: JsonRpcConfig::default(),
|
rpc_config: JsonRpcConfig::default(),
|
||||||
snapshot_path: None,
|
snapshot_path: None,
|
||||||
@ -157,7 +153,11 @@ impl Validator {
|
|||||||
keypair.clone(),
|
keypair.clone(),
|
||||||
)));
|
)));
|
||||||
|
|
||||||
let storage_state = StorageState::new(&bank.last_blockhash(), bank.slots_per_segment());
|
let storage_state = StorageState::new(
|
||||||
|
&bank.last_blockhash(),
|
||||||
|
config.storage_slots_per_turn,
|
||||||
|
bank.slots_per_segment(),
|
||||||
|
);
|
||||||
|
|
||||||
let rpc_service = if node.info.rpc.port() == 0 {
|
let rpc_service = if node.info.rpc.port() == 0 {
|
||||||
None
|
None
|
||||||
@ -240,7 +240,6 @@ impl Validator {
|
|||||||
&cluster_info,
|
&cluster_info,
|
||||||
sockets,
|
sockets,
|
||||||
blocktree.clone(),
|
blocktree.clone(),
|
||||||
config.storage_rotate_count,
|
|
||||||
&storage_state,
|
&storage_state,
|
||||||
config.blockstream.as_ref(),
|
config.blockstream.as_ref(),
|
||||||
ledger_signal_receiver,
|
ledger_signal_receiver,
|
||||||
|
@ -11,12 +11,11 @@ use solana::contact_info::ContactInfo;
|
|||||||
use solana::gossip_service::discover_cluster;
|
use solana::gossip_service::discover_cluster;
|
||||||
use solana::local_cluster::{ClusterConfig, LocalCluster};
|
use solana::local_cluster::{ClusterConfig, LocalCluster};
|
||||||
use solana::replicator::Replicator;
|
use solana::replicator::Replicator;
|
||||||
use solana::storage_stage::STORAGE_ROTATE_TEST_COUNT;
|
use solana::storage_stage::SLOTS_PER_TURN_TEST;
|
||||||
use solana::validator::ValidatorConfig;
|
use solana::validator::ValidatorConfig;
|
||||||
use solana_client::thin_client::create_client;
|
use solana_client::thin_client::create_client;
|
||||||
use solana_sdk::genesis_block::create_genesis_block;
|
use solana_sdk::genesis_block::create_genesis_block;
|
||||||
use solana_sdk::signature::{Keypair, KeypairUtil};
|
use solana_sdk::signature::{Keypair, KeypairUtil};
|
||||||
use solana_sdk::timing::DEFAULT_SLOTS_PER_SEGMENT;
|
|
||||||
use std::fs::remove_dir_all;
|
use std::fs::remove_dir_all;
|
||||||
use std::sync::{Arc, RwLock};
|
use std::sync::{Arc, RwLock};
|
||||||
|
|
||||||
@ -27,12 +26,15 @@ fn run_replicator_startup_basic(num_nodes: usize, num_replicators: usize) {
|
|||||||
info!("starting replicator test");
|
info!("starting replicator test");
|
||||||
|
|
||||||
let mut validator_config = ValidatorConfig::default();
|
let mut validator_config = ValidatorConfig::default();
|
||||||
validator_config.storage_rotate_count = STORAGE_ROTATE_TEST_COUNT;
|
let slots_per_segment = 8;
|
||||||
|
validator_config.storage_slots_per_turn = SLOTS_PER_TURN_TEST;
|
||||||
let config = ClusterConfig {
|
let config = ClusterConfig {
|
||||||
validator_configs: vec![validator_config; num_nodes],
|
validator_configs: vec![validator_config; num_nodes],
|
||||||
num_replicators,
|
num_replicators,
|
||||||
node_stakes: vec![100; num_nodes],
|
node_stakes: vec![100; num_nodes],
|
||||||
cluster_lamports: 10_000,
|
cluster_lamports: 10_000,
|
||||||
|
// keep a low slot/segment count to speed up the test
|
||||||
|
slots_per_segment,
|
||||||
..ClusterConfig::default()
|
..ClusterConfig::default()
|
||||||
};
|
};
|
||||||
let cluster = LocalCluster::new(&config);
|
let cluster = LocalCluster::new(&config);
|
||||||
@ -62,16 +64,13 @@ fn run_replicator_startup_basic(num_nodes: usize, num_replicators: usize) {
|
|||||||
)));
|
)));
|
||||||
let path = get_tmp_ledger_path("test");
|
let path = get_tmp_ledger_path("test");
|
||||||
let blocktree = Arc::new(Blocktree::open(&path).unwrap());
|
let blocktree = Arc::new(Blocktree::open(&path).unwrap());
|
||||||
assert_eq!(
|
Replicator::download_from_replicator(
|
||||||
Replicator::download_from_replicator(
|
&cluster_info,
|
||||||
&cluster_info,
|
&replicator_info,
|
||||||
&replicator_info,
|
&blocktree,
|
||||||
&blocktree,
|
slots_per_segment,
|
||||||
DEFAULT_SLOTS_PER_SEGMENT,
|
)
|
||||||
)
|
.unwrap();
|
||||||
.unwrap(),
|
|
||||||
0
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
@ -131,7 +130,7 @@ fn test_replicator_startup_ledger_hang() {
|
|||||||
solana_logger::setup();
|
solana_logger::setup();
|
||||||
info!("starting replicator test");
|
info!("starting replicator test");
|
||||||
let mut validator_config = ValidatorConfig::default();
|
let mut validator_config = ValidatorConfig::default();
|
||||||
validator_config.storage_rotate_count = STORAGE_ROTATE_TEST_COUNT;
|
validator_config.storage_slots_per_turn = SLOTS_PER_TURN_TEST;
|
||||||
let cluster = LocalCluster::new_with_equal_stakes(2, 10_000, 100);;
|
let cluster = LocalCluster::new_with_equal_stakes(2, 10_000, 100);;
|
||||||
|
|
||||||
info!("starting replicator node");
|
info!("starting replicator node");
|
||||||
@ -160,7 +159,7 @@ fn test_account_setup() {
|
|||||||
let num_nodes = 1;
|
let num_nodes = 1;
|
||||||
let num_replicators = 1;
|
let num_replicators = 1;
|
||||||
let mut validator_config = ValidatorConfig::default();
|
let mut validator_config = ValidatorConfig::default();
|
||||||
validator_config.storage_rotate_count = STORAGE_ROTATE_TEST_COUNT;
|
validator_config.storage_slots_per_turn = SLOTS_PER_TURN_TEST;
|
||||||
let config = ClusterConfig {
|
let config = ClusterConfig {
|
||||||
validator_configs: vec![ValidatorConfig::default(); num_nodes],
|
validator_configs: vec![ValidatorConfig::default(); num_nodes],
|
||||||
num_replicators,
|
num_replicators,
|
||||||
|
@ -13,7 +13,6 @@ use solana::packet::index_blobs;
|
|||||||
use solana::rpc_subscriptions::RpcSubscriptions;
|
use solana::rpc_subscriptions::RpcSubscriptions;
|
||||||
use solana::service::Service;
|
use solana::service::Service;
|
||||||
use solana::storage_stage::StorageState;
|
use solana::storage_stage::StorageState;
|
||||||
use solana::storage_stage::STORAGE_ROTATE_TEST_COUNT;
|
|
||||||
use solana::streamer;
|
use solana::streamer;
|
||||||
use solana::tvu::{Sockets, Tvu};
|
use solana::tvu::{Sockets, Tvu};
|
||||||
use solana::validator;
|
use solana::validator;
|
||||||
@ -133,7 +132,6 @@ fn test_replay() {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
blocktree,
|
blocktree,
|
||||||
STORAGE_ROTATE_TEST_COUNT,
|
|
||||||
&StorageState::default(),
|
&StorageState::default(),
|
||||||
None,
|
None,
|
||||||
ledger_signal_receiver,
|
ledger_signal_receiver,
|
||||||
|
@ -14,7 +14,10 @@ pub const DEFAULT_TICKS_PER_SLOT: u64 = 4;
|
|||||||
pub const DEFAULT_SLOTS_PER_EPOCH: u64 = 8192;
|
pub const DEFAULT_SLOTS_PER_EPOCH: u64 = 8192;
|
||||||
|
|
||||||
// Storage segment configuration
|
// Storage segment configuration
|
||||||
pub const DEFAULT_SLOTS_PER_SEGMENT: u64 = 16;
|
pub const DEFAULT_SLOTS_PER_SEGMENT: u64 = 1024;
|
||||||
|
|
||||||
|
// 4 times longer than the max_lockout to allow enough time for PoRep (128 slots)
|
||||||
|
pub const DEFAULT_SLOTS_PER_TURN: u64 = 32 * 4;
|
||||||
|
|
||||||
pub const NUM_CONSECUTIVE_LEADER_SLOTS: u64 = 4;
|
pub const NUM_CONSECUTIVE_LEADER_SLOTS: u64 = 4;
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user