Skip to content
This repository has been archived by the owner on Jan 13, 2025. It is now read-only.

Wen restart aggregate last voted fork slots #33892

Merged
Show file tree
Hide file tree
Changes from 16 commits
Commits
Show all changes
61 commits
Select commit Hold shift + click to select a range
47f8891
Push and aggregate RestartLastVotedForkSlots.
wen-coding Oct 25, 2023
a8d0c08
Fix API and lint errors.
wen-coding Oct 26, 2023
f7b8232
Reduce clutter.
wen-coding Oct 26, 2023
f5f71b4
Put my own LastVotedForkSlots into the aggregate.
wen-coding Oct 26, 2023
630cc70
Merge branch 'solana-labs:master' into wen_restart_aggregate_last_vot…
wen-coding Oct 27, 2023
ce32c03
Write LastVotedForkSlots aggregate progress into local file.
wen-coding Oct 27, 2023
b90185d
Fix typo and name constants.
wen-coding Oct 27, 2023
3c819f0
Fix flaky test.
wen-coding Oct 27, 2023
e21efe3
Clarify the comments.
wen-coding Oct 27, 2023
b24b8db
- Use constant for wait_for_supermajority
wen-coding Nov 8, 2023
a2204f3
Fix delay_after_first_shred and remove loop in wen_restart.
wen-coding Nov 9, 2023
0c1ef0f
Read wen_restart slots inside the loop instead.
wen-coding Nov 9, 2023
b9324c8
Merge branch 'master' into wen_restart_aggregate_last_voted_fork_slots
wen-coding Nov 11, 2023
122314d
Discard turbine shreds while in wen_restart in windows insert rather …
wen-coding Nov 12, 2023
e1252a4
Merge branch 'master' into wen_restart_aggregate_last_voted_fork_slots
wen-coding Nov 14, 2023
31ca285
Merge branch 'master' into wen_restart_aggregate_last_voted_fork_slots
wen-coding Nov 15, 2023
c3ab972
Merge branch 'master' into wen_restart_aggregate_last_voted_fork_slots
wen-coding Nov 16, 2023
8fc2327
Use the new Gossip API.
wen-coding Nov 16, 2023
229f447
Rename slots_to_repair_for_wen_restart and a few others.
wen-coding Nov 16, 2023
bc1b4b5
Rename a few more and list all states.
wen-coding Nov 17, 2023
8743b5c
Pipe exit down to aggregate loop so we can exit early.
wen-coding Nov 17, 2023
4ebbde8
Merge branch 'master' into wen_restart_aggregate_last_voted_fork_slots
wen-coding Nov 17, 2023
0d82a7c
Fix import of RestartLastVotedForkSlots.
wen-coding Nov 17, 2023
5e0a5b1
Merge branch 'master' into wen_restart_aggregate_last_voted_fork_slots
wen-coding Nov 17, 2023
ec21ec1
Merge branch 'master' into wen_restart_aggregate_last_voted_fork_slots
wen-coding Dec 12, 2023
c172c26
Use the new method to generate test bank.
wen-coding Dec 12, 2023
08de626
Make linter happy.
wen-coding Jan 4, 2024
ea4d800
Merge branch 'master' into wen_restart_aggregate_last_voted_fork_slots
wen-coding Jan 4, 2024
4f91be7
Use new bank constructor for tests.
wen-coding Jan 4, 2024
de89a4e
Merge branch 'master' into wen_restart_aggregate_last_voted_fork_slots
wen-coding Jan 20, 2024
1e98324
Fix a bad merge.
wen-coding Jan 20, 2024
777523f
Merge branch 'master' into wen_restart_aggregate_last_voted_fork_slots
wen-coding Jan 21, 2024
1e478e8
- add new const for wen_restart
wen-coding Jan 23, 2024
b0980e4
Merge branch 'master' into wen_restart_aggregate_last_voted_fork_slots
wen-coding Jan 23, 2024
f4acd69
Add initialize and put the main logic into a loop.
wen-coding Jan 23, 2024
bb471c1
Merge branch 'wen_restart_aggregate_last_voted_fork_slots' of https:/…
wen-coding Jan 23, 2024
c45a29b
Change aggregate interface and other fixes.
wen-coding Jan 24, 2024
167b790
Add failure tests and tests for state transition.
wen-coding Jan 24, 2024
e0a070f
Add more tests and add ability to recover from written records in
wen-coding Jan 27, 2024
8be5cd0
Merge branch 'master' into wen_restart_aggregate_last_voted_fork_slots
wen-coding Jan 27, 2024
ddd144e
Merge branch 'master' into wen_restart_aggregate_last_voted_fork_slots
wen-coding Jan 29, 2024
1cfc510
Various name changes.
wen-coding Jan 31, 2024
5b10c6e
We don't really care what type of error is returned.
wen-coding Jan 31, 2024
0620aaf
Wait on expected progress message in proto file instead of sleep.
wen-coding Jan 31, 2024
1ceda56
Merge branch 'master' into wen_restart_aggregate_last_voted_fork_slots
wen-coding Jan 31, 2024
93abe45
Code reorganization and cleanup.
wen-coding Feb 5, 2024
cb1788e
Make linter happy.
wen-coding Feb 5, 2024
72a732e
Add WenRestartError.
wen-coding Feb 5, 2024
4c920cb
Split WenRestartErrors into separate erros per state.
wen-coding Feb 7, 2024
bf71c9b
Revert "Split WenRestartErrors into separate erros per state."
wen-coding Feb 9, 2024
056aef7
Merge branch 'master' into wen_restart_aggregate_last_voted_fork_slots
wen-coding Feb 9, 2024
645452f
Use individual functions when testing for failures.
wen-coding Feb 10, 2024
f46e62a
Move initialization errors into initialize().
wen-coding Feb 28, 2024
e3d0194
Use anyhow instead of thiserror to generate backtrace for error.
wen-coding Feb 28, 2024
ffbb20c
Add missing Cargo.lock.
wen-coding Feb 29, 2024
3b50964
Merge branch 'master' into wen_restart_aggregate_last_voted_fork_slots
wen-coding Feb 29, 2024
59fd5ff
Add error log when last_vote is missing in the tower storage.
wen-coding Feb 29, 2024
e7c320c
Merge branch 'master' into wen_restart_aggregate_last_voted_fork_slots
wen-coding Feb 29, 2024
9ccf5ca
Merge branch 'master' into wen_restart_aggregate_last_voted_fork_slots
wen-coding Mar 1, 2024
021dbe9
Change error log info.
wen-coding Mar 1, 2024
40c0fb6
Change test to match exact error.
wen-coding Mar 1, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

1 change: 1 addition & 0 deletions core/src/repair/ancestor_hashes_service.rs
Original file line number Diff line number Diff line change
Expand Up @@ -1369,6 +1369,7 @@ mod test {
ancestor_duplicate_slots_sender,
repair_validators: None,
repair_whitelist,
slots_to_repair_for_wen_restart: None,
};

let (ancestor_hashes_replay_update_sender, ancestor_hashes_replay_update_receiver) =
Expand Down
1 change: 1 addition & 0 deletions core/src/repair/repair_generic_traversal.rs
Original file line number Diff line number Diff line change
Expand Up @@ -191,6 +191,7 @@ pub fn get_closest_completion(
path_slot,
slot_meta,
limit - repairs.len(),
true,
);
repairs.extend(new_repairs);
total_processed_slots += 1;
Expand Down
131 changes: 118 additions & 13 deletions core/src/repair/repair_service.rs
Original file line number Diff line number Diff line change
Expand Up @@ -213,6 +213,8 @@ pub struct RepairInfo {
pub repair_validators: Option<HashSet<Pubkey>>,
// Validators which should be given priority when serving
pub repair_whitelist: Arc<RwLock<HashSet<Pubkey>>>,
// A given list of slots to repair when in wen_restart
pub slots_to_repair_for_wen_restart: Option<Arc<RwLock<Vec<Slot>>>>,
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

suggestion: s/slots_to_repair_for_wen_restart/wen_restart_repair_slots/ for brevity

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Done.

}

pub struct RepairSlotRange {
Expand Down Expand Up @@ -386,17 +388,24 @@ impl RepairService {
);
add_votes_elapsed.stop();

let repairs = repair_weight.get_best_weighted_repairs(
blockstore,
root_bank.epoch_stakes_map(),
root_bank.epoch_schedule(),
MAX_ORPHANS,
MAX_REPAIR_LENGTH,
MAX_UNKNOWN_LAST_INDEX_REPAIRS,
MAX_CLOSEST_COMPLETION_REPAIRS,
&mut repair_timing,
&mut best_repairs_stats,
);
let repairs = match repair_info.slots_to_repair_for_wen_restart.clone() {
Some(slots_to_repair) => Self::generate_repairs_for_wen_restart(
blockstore,
MAX_REPAIR_LENGTH,
AshwinSekar marked this conversation as resolved.
Show resolved Hide resolved
&slots_to_repair.read().unwrap(),
),
None => repair_weight.get_best_weighted_repairs(
blockstore,
root_bank.epoch_stakes_map(),
root_bank.epoch_schedule(),
MAX_ORPHANS,
MAX_REPAIR_LENGTH,
MAX_UNKNOWN_LAST_INDEX_REPAIRS,
MAX_CLOSEST_COMPLETION_REPAIRS,
&mut repair_timing,
&mut best_repairs_stats,
),
};

let mut popular_pruned_forks = repair_weight.get_popular_pruned_forks(
root_bank.epoch_stakes_map(),
Expand Down Expand Up @@ -613,7 +622,14 @@ impl RepairService {
slot: Slot,
slot_meta: &SlotMeta,
max_repairs: usize,
add_delay_after_first_shred: bool,
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

nit: maybe change name to throttle_requests_by_shred_tick?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Done.

) -> Vec<ShredRepairType> {
// When in wen_restart, turbine not running, no need to wait after first shred.
carllin marked this conversation as resolved.
Show resolved Hide resolved
let defer_repair_threshold_ticks = if add_delay_after_first_shred {
DEFER_REPAIR_THRESHOLD_TICKS
} else {
0
};
if max_repairs == 0 || slot_meta.is_full() {
vec![]
} else if slot_meta.consumed == slot_meta.received {
Expand All @@ -630,7 +646,7 @@ impl RepairService {
* timestamp().saturating_sub(slot_meta.first_shred_timestamp)
/ 1_000;
if ticks_since_first_insert
< reference_tick.saturating_add(DEFER_REPAIR_THRESHOLD_TICKS)
< reference_tick.saturating_add(defer_repair_threshold_ticks)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Should we bypass the reference_tick comparison completely if add_delay_after_first_shred == false?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Done.

{
return vec![];
}
Expand All @@ -641,7 +657,7 @@ impl RepairService {
.find_missing_data_indexes(
slot,
slot_meta.first_shred_timestamp,
DEFER_REPAIR_THRESHOLD_TICKS,
defer_repair_threshold_ticks,
slot_meta.consumed,
slot_meta.received,
max_repairs,
Expand All @@ -668,6 +684,7 @@ impl RepairService {
slot,
&slot_meta,
max_repairs - repairs.len(),
true,
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

nit: could you add the param name in a comment like /*add_delay_after_first_shred*/ true for readability?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Done.

);
repairs.extend(new_repairs);
let next_slots = slot_meta.next_slots;
Expand All @@ -678,6 +695,32 @@ impl RepairService {
}
}

pub(crate) fn generate_repairs_for_wen_restart(
blockstore: &Blockstore,
max_repairs: usize,
slots: &Vec<Slot>,
) -> Vec<ShredRepairType> {
let mut result: Vec<ShredRepairType> = Vec::new();
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

s/result/repairs/?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Done.

for slot in slots {
if let Some(slot_meta) = blockstore.meta(*slot).unwrap() {
let new_repairs = Self::generate_repairs_for_slot(
AshwinSekar marked this conversation as resolved.
Show resolved Hide resolved
blockstore,
*slot,
&slot_meta,
max_repairs - result.len(),
false,
);
result.extend(new_repairs);
} else {
result.push(ShredRepairType::HighestShred(*slot, 0));
}
if result.len() >= max_repairs {
break;
}
}
result
}

/// Generate repairs for all slots `x` in the repair_range.start <= x <= repair_range.end
#[cfg(test)]
pub fn generate_repairs_in_range(
Expand Down Expand Up @@ -705,6 +748,7 @@ impl RepairService {
slot,
&meta,
max_repairs - repairs.len(),
true,
);
repairs.extend(new_repairs);
}
Expand All @@ -727,6 +771,7 @@ impl RepairService {
slot,
&slot_meta,
MAX_REPAIR_PER_DUPLICATE,
true,
))
}
} else {
Expand Down Expand Up @@ -1349,4 +1394,64 @@ mod test {
);
assert_ne!(duplicate_status.repair_pubkey_and_addr, dummy_addr);
}

#[test]
fn test_generate_repairs_for_wen_restart() {
solana_logger::setup();
let ledger_path = get_tmp_ledger_path_auto_delete!();
let blockstore = Blockstore::open(ledger_path.path()).unwrap();
let max_repairs = 3;

let slots: Vec<u64> = vec![2, 3, 5, 7];
let num_entries_per_slot = max_ticks_per_n_shreds(1, None) + 1;

let shreds = make_chaining_slot_entries(&slots, num_entries_per_slot);
for (mut slot_shreds, _) in shreds.into_iter() {
slot_shreds.remove(1);
blockstore.insert_shreds(slot_shreds, None, false).unwrap();
}
sleep_shred_deferment_period();
carllin marked this conversation as resolved.
Show resolved Hide resolved

let mut slots_to_repair: Vec<Slot> = vec![];

// When slots_to_repair is empty, ignore all and return empty result.
let result = RepairService::generate_repairs_for_wen_restart(
&blockstore,
max_repairs,
&slots_to_repair,
);
assert!(result.is_empty());

// When asked to repair dead_slot and some unknown slot, return correct results.
carllin marked this conversation as resolved.
Show resolved Hide resolved
slots_to_repair = vec![2, 81];
let result = RepairService::generate_repairs_for_wen_restart(
&blockstore,
max_repairs,
&slots_to_repair,
);
assert_eq!(
result,
vec![
ShredRepairType::HighestShred(2, 1),
ShredRepairType::HighestShred(81, 0),
],
);

// Test that it will not generate more than max_repairs.e().unwrap();
slots_to_repair = vec![3, 82, 5, 83, 7, 84];
let result = RepairService::generate_repairs_for_wen_restart(
&blockstore,
max_repairs,
&slots_to_repair,
);
assert_eq!(result.len(), max_repairs);
assert_eq!(
result,
vec![
ShredRepairType::HighestShred(3, 1),
ShredRepairType::HighestShred(82, 0),
ShredRepairType::HighestShred(5, 1),
],
carllin marked this conversation as resolved.
Show resolved Hide resolved
);
}
}
1 change: 1 addition & 0 deletions core/src/repair/repair_weighted_traversal.rs
Original file line number Diff line number Diff line change
Expand Up @@ -103,6 +103,7 @@ pub fn get_best_repair_shreds(
slot,
slot_meta,
max_repairs - repairs.len(),
true,
);
repairs.extend(new_repairs);
visited_set.insert(slot);
Expand Down
3 changes: 3 additions & 0 deletions core/src/tvu.rs
Original file line number Diff line number Diff line change
Expand Up @@ -138,6 +138,7 @@ impl Tvu {
turbine_quic_endpoint_sender: AsyncSender<(SocketAddr, Bytes)>,
turbine_quic_endpoint_receiver: Receiver<(Pubkey, SocketAddr, Bytes)>,
repair_quic_endpoint_sender: AsyncSender<LocalRequest>,
slots_to_repair_for_wen_restart: Option<Arc<RwLock<Vec<Slot>>>>,
) -> Result<Self, String> {
let TvuSockets {
repair: repair_socket,
Expand Down Expand Up @@ -205,6 +206,7 @@ impl Tvu {
repair_whitelist: tvu_config.repair_whitelist,
cluster_info: cluster_info.clone(),
cluster_slots: cluster_slots.clone(),
slots_to_repair_for_wen_restart: slots_to_repair_for_wen_restart.clone(),
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

do you need to clone() here?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actually I don't now, removed.

};
WindowService::new(
blockstore.clone(),
Expand Down Expand Up @@ -491,6 +493,7 @@ pub mod tests {
turbine_quic_endpoint_sender,
turbine_quic_endpoint_receiver,
repair_quic_endpoint_sender,
None,
)
.expect("assume success");
exit.store(true, Ordering::Relaxed);
Expand Down
9 changes: 9 additions & 0 deletions core/src/validator.rs
Original file line number Diff line number Diff line change
Expand Up @@ -1217,6 +1217,11 @@ impl Validator {
.unwrap();

let in_wen_restart = config.wen_restart_proto_path.is_some() && !waited_for_supermajority;
let slots_to_repair_for_wen_restart = if in_wen_restart {
Some(Arc::new(RwLock::new(Vec::new())))
} else {
None
};
let tower = match process_blockstore.process_to_create_tower() {
Ok(tower) => {
info!("Tower state: {:?}", tower);
Expand Down Expand Up @@ -1284,6 +1289,7 @@ impl Validator {
turbine_quic_endpoint_sender.clone(),
turbine_quic_endpoint_receiver,
repair_quic_endpoint_sender,
slots_to_repair_for_wen_restart.clone(),
)?;

if in_wen_restart {
Expand All @@ -1293,6 +1299,9 @@ impl Validator {
last_vote,
blockstore.clone(),
cluster_info.clone(),
bank_forks.clone(),
slots_to_repair_for_wen_restart.clone(),
WAIT_FOR_SUPERMAJORITY_THRESHOLD_PERCENT,
carllin marked this conversation as resolved.
Show resolved Hide resolved
) {
Ok(()) => {
return Err("wen_restart phase one completedy".to_string());
Expand Down
35 changes: 31 additions & 4 deletions core/src/window_service.rs
Original file line number Diff line number Diff line change
Expand Up @@ -204,15 +204,19 @@ fn prune_shreds_invalid_repair(
shreds: &mut Vec<Shred>,
repair_infos: &mut Vec<Option<RepairMeta>>,
outstanding_requests: &RwLock<OutstandingShredRepairs>,
in_wen_restart: bool,
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

maybe something like: s/in_wen_restart/accept_only_repairs/

I think the function name should probably change to something more generic like prune_shreds_by_repair_status

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Done.

) {
assert_eq!(shreds.len(), repair_infos.len());
let mut i = 0;
let mut removed = HashSet::new();
{
let mut outstanding_requests = outstanding_requests.write().unwrap();
shreds.retain(|shred| {
// In wen_restart, we discard all shreds from Turbine and keep only those from repair to
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If the param name is changed to accept_only_repairs this comment could be moved to the caller.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Done.

// avoid new shreds make validator OOM before wen_restart is over.
let should_keep = (
verify_repair(&mut outstanding_requests, shred, &repair_infos[i]),
(!in_wen_restart || repair_infos[i].is_some())
&& verify_repair(&mut outstanding_requests, shred, &repair_infos[i]),
i += 1,
)
.0;
Expand Down Expand Up @@ -240,6 +244,7 @@ fn run_insert<F>(
retransmit_sender: &Sender<Vec<ShredPayload>>,
outstanding_requests: &RwLock<OutstandingShredRepairs>,
reed_solomon_cache: &ReedSolomonCache,
in_wen_restart: bool,
) -> Result<()>
where
F: Fn(PossibleDuplicateShred),
Expand Down Expand Up @@ -285,7 +290,12 @@ where

let mut prune_shreds_elapsed = Measure::start("prune_shreds_elapsed");
let num_shreds = shreds.len();
prune_shreds_invalid_repair(&mut shreds, &mut repair_infos, outstanding_requests);
prune_shreds_invalid_repair(
&mut shreds,
&mut repair_infos,
outstanding_requests,
in_wen_restart,
);
ws_metrics.num_shreds_pruned_invalid_repair = num_shreds - shreds.len();
let repairs: Vec<_> = repair_infos
.iter()
Expand Down Expand Up @@ -343,6 +353,8 @@ impl WindowService {

let cluster_info = repair_info.cluster_info.clone();

let in_wen_restart = repair_info.slots_to_repair_for_wen_restart.is_some();

let repair_service = RepairService::new(
blockstore.clone(),
exit.clone(),
Expand Down Expand Up @@ -377,6 +389,7 @@ impl WindowService {
completed_data_sets_sender,
retransmit_sender,
outstanding_requests,
in_wen_restart,
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If in_wen_restart is true, will this behavior be enabled indefinitely?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It will be enabled for as long as the validator is not restarted. When wen_restart is done, we will set wen_restart state to DONE and restart the validator, then in_wen_restart will be false and this behavior will be turned off.

);

WindowService {
Expand Down Expand Up @@ -424,6 +437,7 @@ impl WindowService {
completed_data_sets_sender: CompletedDataSetsSender,
retransmit_sender: Sender<Vec<ShredPayload>>,
outstanding_requests: Arc<RwLock<OutstandingShredRepairs>>,
in_wen_restart: bool,
) -> JoinHandle<()> {
let handle_error = || {
inc_new_counter_error!("solana-window-insert-error", 1, 1);
Expand Down Expand Up @@ -456,6 +470,7 @@ impl WindowService {
&retransmit_sender,
&outstanding_requests,
&reed_solomon_cache,
in_wen_restart,
) {
ws_metrics.record_error(&e);
if Self::should_exit_on_error(e, &handle_error) {
Expand Down Expand Up @@ -683,7 +698,7 @@ mod test {
4, // position
0, // version
);
let mut shreds = vec![shred.clone(), shred.clone(), shred];
let mut shreds = vec![shred.clone(), shred.clone(), shred.clone()];
let repair_meta = RepairMeta { nonce: 0 };
let outstanding_requests = Arc::new(RwLock::new(OutstandingShredRepairs::default()));
let repair_type = ShredRepairType::Orphan(9);
Expand All @@ -693,9 +708,21 @@ mod test {
.add_request(repair_type, timestamp());
let repair_meta1 = RepairMeta { nonce };
let mut repair_infos = vec![None, Some(repair_meta), Some(repair_meta1)];
prune_shreds_invalid_repair(&mut shreds, &mut repair_infos, &outstanding_requests);
prune_shreds_invalid_repair(&mut shreds, &mut repair_infos, &outstanding_requests, false);
assert_eq!(shreds.len(), 2);
assert_eq!(repair_infos.len(), 2);
assert!(repair_infos[0].is_none());
assert_eq!(repair_infos[1].as_ref().unwrap().nonce, nonce);

shreds = vec![shred.clone(), shred.clone(), shred];
let repair_meta2 = RepairMeta { nonce: 0 };
let repair_meta3 = RepairMeta { nonce };
repair_infos = vec![None, Some(repair_meta2), Some(repair_meta3)];
// In wen_restart, we discard all Turbine shreds and only keep valid repair shreds.
prune_shreds_invalid_repair(&mut shreds, &mut repair_infos, &outstanding_requests, true);
assert_eq!(shreds.len(), 1);
assert_eq!(repair_infos.len(), 1);
assert!(repair_infos[0].is_some());
assert_eq!(repair_infos[0].as_ref().unwrap().nonce, nonce);
}
}
Loading