use ckb_chain_spec::consensus::ProposalWindow;
use ckb_types::{core::BlockNumber, packed::ProposalShortId};
use std::collections::{BTreeMap, HashSet};
use std::ops::Bound;
#[cfg(test)]
mod tests;
#[derive(Default, Clone, Debug)]
pub struct ProposalView {
pub(crate) gap: HashSet<ProposalShortId>,
pub(crate) set: HashSet<ProposalShortId>,
}
impl ProposalView {
pub fn new(gap: HashSet<ProposalShortId>, set: HashSet<ProposalShortId>) -> ProposalView {
ProposalView { gap, set }
}
pub fn gap(&self) -> &HashSet<ProposalShortId> {
&self.gap
}
pub fn set(&self) -> &HashSet<ProposalShortId> {
&self.set
}
pub fn contains_proposed(&self, id: &ProposalShortId) -> bool {
self.set.contains(id)
}
pub fn contains_gap(&self, id: &ProposalShortId) -> bool {
self.gap.contains(id)
}
}
#[derive(Debug, PartialEq, Clone, Eq)]
pub struct ProposalTable {
pub(crate) table: BTreeMap<BlockNumber, HashSet<ProposalShortId>>,
pub(crate) proposal_window: ProposalWindow,
}
impl ProposalTable {
pub fn new(proposal_window: ProposalWindow) -> Self {
ProposalTable {
proposal_window,
table: BTreeMap::default(),
}
}
pub fn insert(&mut self, number: BlockNumber, ids: HashSet<ProposalShortId>) -> bool {
self.table.insert(number, ids).is_none()
}
pub fn remove(&mut self, number: BlockNumber) -> Option<HashSet<ProposalShortId>> {
self.table.remove(&number)
}
pub fn all(&self) -> &BTreeMap<BlockNumber, HashSet<ProposalShortId>> {
&self.table
}
pub fn finalize(
&mut self,
origin: &ProposalView,
number: BlockNumber,
) -> (HashSet<ProposalShortId>, ProposalView) {
let candidate_number = number + 1;
let proposal_start = candidate_number.saturating_sub(self.proposal_window.farthest());
let proposal_end = candidate_number.saturating_sub(self.proposal_window.closest());
if proposal_start > 1 {
self.table = self.table.split_off(&proposal_start);
}
ckb_logger::trace!("[proposal_finalize] table {:?}", self.table);
let (new_ids, gap) = if candidate_number <= self.proposal_window.closest() {
(
HashSet::new(),
self.table
.range((Bound::Unbounded, Bound::Included(&number)))
.flat_map(|pair| pair.1)
.cloned()
.collect(),
)
} else {
(
self.table
.range((
Bound::Included(&proposal_start),
Bound::Included(&proposal_end),
))
.flat_map(|pair| pair.1)
.cloned()
.collect(),
self.table
.range((Bound::Excluded(&proposal_end), Bound::Included(&number)))
.flat_map(|pair| pair.1)
.cloned()
.collect(),
)
};
let removed_ids: HashSet<ProposalShortId> =
origin.set().difference(&new_ids).cloned().collect();
ckb_logger::trace!(
"[proposal_finalize] number {} proposal_start {}----proposal_end {}",
number,
proposal_start,
proposal_end
);
ckb_logger::trace!(
"[proposal_finalize] number {} new_ids {:?}----removed_ids {:?}",
number,
new_ids,
removed_ids
);
(removed_ids, ProposalView::new(gap, new_ids))
}
}