Skip to content
This repository has been archived by the owner on Nov 15, 2023. It is now read-only.

Limit number of blocks per level (2nd attempt) #1559

Merged
merged 68 commits into from
Dec 20, 2022
Merged
Changes from 1 commit
Commits
Show all changes
68 commits
Select commit Hold shift + click to select a range
b1db89b
Temporary hack to fetch leaves from parachain block import
davxy Jul 28, 2022
c211faa
Merge branch 'master' into davxy/remove-leaves
davxy Aug 1, 2022
e9bd438
Working temporary PoC implementation
davxy Aug 1, 2022
8aaf2e8
Merge branch 'master' into davxy/remove-leaves
davxy Aug 10, 2022
e0a7da1
TEMPORARY COMMIT
davxy Aug 10, 2022
48367c6
Test excercising leaves pruning feature
davxy Aug 11, 2022
0613b3b
Removed global backend hack and introduced configurable leaves level …
davxy Aug 12, 2022
0ddaedc
Cleanup
davxy Aug 12, 2022
e811887
Nitpicks
davxy Aug 12, 2022
be75d7b
Removed debug logs
davxy Aug 12, 2022
bd8852b
Adjust leaves per level default value to be equal to the substrate ba…
davxy Aug 12, 2022
5b299dd
Better docs
davxy Aug 12, 2022
6e3cef9
Don't rely on the backend implementation order for the leaves
davxy Aug 19, 2022
7437611
Merge branch 'master' into davxy/remove-leaves-portable
davxy Aug 19, 2022
7453fe0
Leaves monitoring code encapsulation
davxy Aug 19, 2022
45fc0d1
Fix panic for unsigned subtraction underflow
davxy Aug 19, 2022
12427d5
Cargo fmt
davxy Aug 19, 2022
b2c197f
Improval leaves cache insertion/deletion strategy
davxy Aug 21, 2022
0c81d6b
Apply suggestions from code review
davxy Aug 22, 2022
aabc69a
Fix to default level limit
davxy Aug 22, 2022
55e598e
Remove redundant type
davxy Aug 23, 2022
d939757
Allow removal of non-leaves blocks and restore from backend
davxy Aug 30, 2022
ca4b383
Encapsulate limit monitor in its own module
davxy Aug 30, 2022
9c13d60
Proper cleanup for descendants
davxy Aug 31, 2022
765b59a
Apply suggestions from code review
davxy Sep 1, 2022
ffeeb22
Optimization
davxy Sep 1, 2022
7e6ed47
Fix
davxy Sep 1, 2022
5dbfe14
Fix typo
davxy Sep 1, 2022
2d9242c
Nit. Rename some fresher to freshest
davxy Sep 1, 2022
eff06c5
Dedicated ParachainBlockImport constructor with explicit LevelLimit
davxy Sep 8, 2022
7a8d663
Apply suggestions from code review
davxy Sep 14, 2022
4f15278
Fmt
davxy Sep 14, 2022
cc60027
Fix typo
davxy Sep 14, 2022
48e0cc9
Apply suggestions from code review
davxy Sep 14, 2022
7b51bb0
Use block number instead of u64 for import counter
davxy Sep 14, 2022
a9d0b1e
Separate target find and removal logic
davxy Sep 14, 2022
27e8f04
Code adjustments
davxy Sep 14, 2022
b675d48
Refactory to reduce cognitive complexity
davxy Sep 15, 2022
5b5f1d0
Merge branch 'master' into davxy/remove-leaves-portable
davxy Sep 15, 2022
1029cff
Apply suggestions from code review
davxy Sep 28, 2022
1bc4e61
Fix type
davxy Sep 28, 2022
d8f1c84
Restore monitor within the constructor
davxy Sep 28, 2022
9766f86
Rename Monitor 'update' to 'block_imported'
davxy Sep 28, 2022
19695b4
Keep track of invalidated leaves
davxy Oct 1, 2022
77e2549
Check import before add block to the monitor
davxy Oct 1, 2022
8ecb772
Update test
davxy Oct 1, 2022
3d86b2e
Adjust log levels
davxy Oct 1, 2022
59fcb7a
[ci] Apply cargo-fmt
Oct 1, 2022
283d33b
Extra debugging logs
davxy Oct 5, 2022
fd939e9
Added target to debug logs
davxy Oct 5, 2022
bbde030
Report errors as such
davxy Oct 7, 2022
7dd4ebe
Added extra log to the monitor restore method
davxy Oct 13, 2022
c89b2c2
Merge branch 'master' into davxy/remove-leaves-portable
davxy Oct 13, 2022
f08c6e4
Fix after master merge
davxy Oct 13, 2022
3cd5d31
Bugfix: level monitor cache should use block 'post-hash'
davxy Oct 13, 2022
f0c18c1
TEMPORARY
davxy Oct 18, 2022
d945832
Merge branch 'master' into davxy/remove-leaves-portable
davxy Oct 21, 2022
01c37b7
Minor tweaks
davxy Oct 21, 2022
70c7c7e
Merge branch 'master' into davxy/remove-leaves-portable
davxy Nov 10, 2022
61304ec
Small code refactory
davxy Nov 11, 2022
e92d651
Invalidate leaf only on removal
davxy Nov 11, 2022
bc1643d
Removed temporary logs
davxy Nov 16, 2022
408580e
Explicit recovery for announced candidate blocks (#1891)
davxy Dec 9, 2022
f396157
Merge branch 'master' into davxy/remove-leaves-portable
davxy Dec 9, 2022
92bb6ac
Fix after master merge
davxy Dec 9, 2022
819ade6
Merge branch 'master' into davxy/remove-leaves-portable
davxy Dec 20, 2022
8886f67
Moved block recovery related types in pov-recovery crate
davxy Dec 20, 2022
0eff586
Fix after refactory
davxy Dec 20, 2022
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
Prev Previous commit
Next Next commit
Separate target find and removal logic
  • Loading branch information
davxy committed Sep 14, 2022
commit a9d0b1e081b19456ba343fc373a34b0eb5f72a08
149 changes: 94 additions & 55 deletions client/consensus/common/src/level_monitor.rs
Original file line number Diff line number Diff line change
Expand Up @@ -76,6 +76,14 @@ impl<Block: BlockT, BE> LevelMonitor<Block, BE> {
}
}

// Internal support structure containing information about the target scheduled for removal.
struct TargetInfo<Block: BlockT> {
// Index of freshest leaf in the leaves array.
freshest_leaf_idx: usize,
// Route from target to its freshest leaf.
davxy marked this conversation as resolved.
Show resolved Hide resolved
freshest_route: TreeRoute<Block>,
}

impl<Block, BE> LevelMonitor<Block, BE>
where
Block: BlockT,
Expand Down Expand Up @@ -124,6 +132,7 @@ where
///
/// If the given level is found to have a number of blocks greater than or equal the limit
/// then the limit is enforced by chosing one (or more) blocks to remove.
///
/// The removal strategy is driven by the block freshness.
///
/// A block freshness is determined by the most recent leaf freshness descending from the block
Expand All @@ -148,28 +157,97 @@ where
);

let mut leaves = self.backend.blockchain().leaves().unwrap_or_default();
// Sort leaves by freshness (least recent first).
// Sort leaves by freshness (least fresh first).
leaves.sort_unstable_by(|a, b| self.freshness.get(a).cmp(&self.freshness.get(b)));

// This may not be the most efficient way to remove **multiple** entries, but is the clear
// one. Should be considered that in normal conditions the number of blocks to remove is 0
// or 1, it is not worth to complicate the code too much.
// One condition that may trigger multiple removals is if we restart the node using an
// existing db and a smaller limit wrt the one previously used.
// This may not be the most efficient way to remove **multiple** entries, but is the easy
// one :-). Should be considered that in "normal" conditions the number of blocks to remove
// is 0 or 1, it is not worth to complicate the code too much. One condition that may
// trigger multiple removals (2+) is if we restart the node using an existing db and a
// smaller limit wrt the one previously used.
let remove_count = level_len - self.level_limit + 1;
(0..remove_count).for_each(|_| self.remove_block(number, &leaves));
(0..remove_count).all(|_| {
davxy marked this conversation as resolved.
Show resolved Hide resolved
self.find_target(number, &leaves).map_or(false, |target| {
self.remove_target(target, &leaves);
true
})
});
}

// Remove the target block and all its descendants.
//
// Leaves should have already been ordered by "freshness" (older first).
fn remove_target(&mut self, target: TargetInfo<Block>, leaves: &[Block::Hash]) {
let mut remove_leaf = |number, hash| {
if self.backend.remove_leaf_block(&hash).is_err() {
return false
}
log::debug!(target: "parachain", "Removing block {}", hash);
self.levels.get_mut(&number).map(|level| level.remove(&hash));
self.freshness.remove(&hash);
true
};

// Takes care of route removal. Starts from the leaf and stops as soon as an error is
// encountered. In this case an error is interpreted as the block being not a leaf
// and it will be removed while removing another route from the same block but to a
// different leaf.
let mut remove_route = |route: TreeRoute<Block>| {
route.enacted().iter().rev().all(|elem| remove_leaf(elem.number, elem.hash));
};

let target_number = target.freshest_route.common_block().number;
let target_hash = target.freshest_route.common_block().hash;

remove_route(target.freshest_route);

// Don't bother trying with leaves we already found to not be our descendants.
let to_skip = leaves.len() - target.freshest_leaf_idx;
leaves.iter().rev().skip(to_skip).for_each(|leaf_hash| {
match sp_blockchain::tree_route(self.backend.blockchain(), target_hash, *leaf_hash) {
Ok(route) if route.retracted().is_empty() => remove_route(route),
Err(err) => {
log::warn!(
target: "parachain",
"Unable getting route from {:?} to {:?}: {}",
target_hash, leaf_hash, err,
);
},
_ => (),
};
});

remove_leaf(target_number, target_hash);
}

/// This will search for the less fresh block and removes it along with all its descendants.
/// Leaves should have already been ordered by "freshness" (older first).
fn remove_block(&mut self, number: NumberFor<Block>, leaves: &[Block::Hash]) {
// Helper function to find the best candidate to be removed.
//
// Given a set of blocks with height equal to `number` (potential candidates)
// 1. For each candidate fetch all the leaves that are descending from it.
// 2. Set the candidate freshness equal to the fresher of its descending leaves.
// 3. The target is set as the candidate that is less fresh.
//
// Input `leaves` are assumed to be already ordered by "freshness" (fresher first).
//
// Returns the index of the target fresher leaf within `leaves` and the route from target to
// such leaf.
fn find_target(
&self,
number: NumberFor<Block>,
leaves: &[Block::Hash],
) -> Option<TargetInfo<Block>> {
let mut candidate_freshest_leaf_idx = usize::MAX;
let mut candidate_freshest_route = None;

let blockchain = self.backend.blockchain();
let best_hash = blockchain.info().best_hash;

for blk_hash in self.levels.entry(number).or_default().iter() {
let level = match self.levels.get(&number) {
Some(level) => level,
None => return None,
};

for blk_hash in level.iter() {
// Start checking from the most fresh leaf (the last). Because of the leaves ordering,
// here leaf index is synonym of its freshness, that is the greater the index the
// fresher it is.
Expand Down Expand Up @@ -199,54 +277,15 @@ where
};
}
if candidate_freshest_leaf_idx == 0 {
// We can't find a candidate with an older leaf.
// We can't find a candidate with an older leaf that the current one.
break
}
}

let candidate_freshest_route = match candidate_freshest_route {
Some(route) => route,
None => return,
};

// We have a candidate, proceed removing the block and all its descendants.

let candidate_hash = candidate_freshest_route.common_block().hash;

// Takes care of route removal. Starts from the leaf and stops as soon as an error is
// encountered. In this case an error is interpreted as the block being not a leaf
// and it will be removed while removing another route from the same block but to a
// different leaf.
let mut remove_route = |route: TreeRoute<Block>| {
std::iter::once(route.common_block()).chain(route.enacted()).rev().all(|elem| {
if self.backend.remove_leaf_block(&elem.hash).is_err() {
return false
}
log::debug!(target: "parachain", "Removing block {}", elem.hash);
self.levels.get_mut(&elem.number).map(|level| level.remove(&elem.hash));
self.freshness.remove(&elem.hash);
true
});
};

remove_route(candidate_freshest_route);

let to_skip = leaves.len() - candidate_freshest_leaf_idx;
leaves.iter().rev().skip(to_skip).for_each(|leaf_hash| {
match sp_blockchain::tree_route(blockchain, candidate_hash, *leaf_hash) {
Ok(route) if route.retracted().is_empty() => {
remove_route(route);
},
Err(err) => {
log::warn!(
target: "parachain",
"Unable getting route from {} to {}: {}",
candidate_hash, leaf_hash, err,
);
},
_ => (),
};
});
candidate_freshest_route.map(|route| TargetInfo {
freshest_leaf_idx: candidate_freshest_leaf_idx,
freshest_route: route,
})
}

/// Add a new imported block information to the monitor.
Expand Down