2024-05-31 05:11:06 +00:00
|
|
|
use serde::{Deserialize, Serialize};
|
|
|
|
use ssz_derive::{Decode, Encode};
|
2024-07-19 03:55:18 +00:00
|
|
|
use std::{cell::RefCell, path::PathBuf, rc::Rc};
|
2024-01-03 10:24:52 +00:00
|
|
|
|
2024-05-31 05:11:06 +00:00
|
|
|
pub const SHARD_CONFIG_KEY: &str = "shard_config";
|
|
|
|
|
2024-01-03 10:24:52 +00:00
|
|
|
#[derive(Clone)]
|
|
|
|
pub struct Config {
|
|
|
|
pub db_dir: PathBuf,
|
|
|
|
}
|
2024-05-31 05:11:06 +00:00
|
|
|
|
2024-06-07 08:58:15 +00:00
|
|
|
#[derive(Clone, Copy, Debug, Decode, Encode, Serialize, Deserialize, Eq, PartialEq)]
|
2024-05-31 05:11:06 +00:00
|
|
|
#[serde(rename_all = "camelCase")]
|
|
|
|
pub struct ShardConfig {
|
|
|
|
pub shard_id: usize,
|
|
|
|
pub num_shard: usize,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Default for ShardConfig {
|
|
|
|
fn default() -> Self {
|
|
|
|
Self {
|
|
|
|
shard_id: 0,
|
|
|
|
num_shard: 1,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ShardConfig {
|
|
|
|
pub fn new(shard_position: &Option<String>) -> Result<Self, String> {
|
|
|
|
let (id, num) = if let Some(position) = shard_position {
|
|
|
|
Self::parse_position(position)?
|
|
|
|
} else {
|
|
|
|
(0, 1)
|
|
|
|
};
|
|
|
|
|
|
|
|
if id >= num {
|
|
|
|
return Err(format!(
|
|
|
|
"Incorrect shard_id: expected [0, {}), actual {}",
|
|
|
|
num, id
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
|
|
|
if !num.is_power_of_two() {
|
|
|
|
return Err(format!(
|
|
|
|
"Incorrect shard group bytes: {}, should be power of two",
|
|
|
|
num
|
|
|
|
));
|
|
|
|
}
|
|
|
|
Ok(ShardConfig {
|
|
|
|
shard_id: id,
|
|
|
|
num_shard: num,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn miner_shard_mask(&self) -> u64 {
|
|
|
|
!(self.num_shard - 1) as u64
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn miner_shard_id(&self) -> u64 {
|
|
|
|
self.shard_id as u64
|
|
|
|
}
|
|
|
|
|
2024-07-19 03:55:18 +00:00
|
|
|
pub fn is_valid(&self) -> bool {
|
|
|
|
self.num_shard > 0 && self.num_shard.is_power_of_two() && self.shard_id < self.num_shard
|
|
|
|
}
|
|
|
|
|
2024-05-31 05:11:06 +00:00
|
|
|
pub fn in_range(&self, segment_index: u64) -> bool {
|
|
|
|
segment_index as usize % self.num_shard == self.shard_id
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn parse_position(input: &str) -> Result<(usize, usize), String> {
|
|
|
|
let parts: Vec<&str> = input.trim().split('/').map(|s| s.trim()).collect();
|
|
|
|
|
|
|
|
if parts.len() != 2 {
|
|
|
|
return Err("Incorrect format, expected like: '0 / 8'".into());
|
|
|
|
}
|
|
|
|
|
|
|
|
let numerator = parts[0]
|
|
|
|
.parse::<usize>()
|
|
|
|
.map_err(|e| format!("Cannot parse shard position {:?}", e))?;
|
|
|
|
let denominator = parts[1]
|
|
|
|
.parse::<usize>()
|
|
|
|
.map_err(|e| format!("Cannot parse shard position {:?}", e))?;
|
|
|
|
|
|
|
|
Ok((numerator, denominator))
|
|
|
|
}
|
2024-06-07 18:50:36 +00:00
|
|
|
|
|
|
|
pub fn next_segment_index(&self, current: usize, start_index: usize) -> usize {
|
|
|
|
// `shift` should be 0 if `current` was returned by the same config.
|
|
|
|
let shift = (start_index + current + self.num_shard - self.shard_id) % self.num_shard;
|
|
|
|
current + self.num_shard - shift
|
|
|
|
}
|
2024-05-31 05:11:06 +00:00
|
|
|
}
|
2024-07-19 03:55:18 +00:00
|
|
|
|
|
|
|
struct ShardSegmentTreeNode {
|
|
|
|
pub num_shard: usize,
|
|
|
|
pub covered: bool,
|
|
|
|
pub childs: [Option<Rc<RefCell<ShardSegmentTreeNode>>>; 2],
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ShardSegmentTreeNode {
|
|
|
|
pub fn new(num_shard: usize) -> Self {
|
|
|
|
ShardSegmentTreeNode {
|
|
|
|
num_shard,
|
|
|
|
covered: false,
|
|
|
|
childs: [None, None],
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn push_down(&mut self) {
|
|
|
|
if self.childs[0].is_none() {
|
|
|
|
for i in 0..2 {
|
|
|
|
self.childs[i] = Some(Rc::new(RefCell::new(ShardSegmentTreeNode::new(
|
|
|
|
self.num_shard << 1,
|
|
|
|
))));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn update(&mut self) {
|
|
|
|
let mut covered = true;
|
|
|
|
for i in 0..2 {
|
|
|
|
if let Some(child) = &self.childs[i] {
|
|
|
|
covered = covered && child.borrow().covered;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
self.covered = covered;
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn insert(&mut self, num_shard: usize, shard_id: usize) {
|
|
|
|
if self.covered {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if num_shard == self.num_shard {
|
|
|
|
self.covered = true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
self.push_down();
|
|
|
|
if let Some(child) = &self.childs[shard_id % 2] {
|
|
|
|
child.borrow_mut().insert(num_shard, shard_id >> 1);
|
|
|
|
}
|
|
|
|
self.update();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn all_shards_available(shard_configs: Vec<ShardConfig>) -> bool {
|
|
|
|
let mut root = ShardSegmentTreeNode::new(1);
|
|
|
|
for shard_config in shard_configs.iter() {
|
|
|
|
if !shard_config.is_valid() {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
root.insert(shard_config.num_shard, shard_config.shard_id);
|
|
|
|
if root.covered {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
false
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use crate::config::all_shards_available;
|
|
|
|
|
|
|
|
use super::ShardConfig;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_all_shards_available() {
|
|
|
|
assert!(all_shards_available(vec![
|
|
|
|
ShardConfig {
|
|
|
|
shard_id: 3,
|
|
|
|
num_shard: 8
|
|
|
|
},
|
|
|
|
ShardConfig {
|
|
|
|
shard_id: 7,
|
|
|
|
num_shard: 8
|
|
|
|
},
|
|
|
|
ShardConfig {
|
|
|
|
shard_id: 0,
|
|
|
|
num_shard: 4
|
|
|
|
},
|
|
|
|
ShardConfig {
|
|
|
|
shard_id: 1,
|
|
|
|
num_shard: 4
|
|
|
|
},
|
|
|
|
ShardConfig {
|
|
|
|
shard_id: 0,
|
|
|
|
num_shard: 2
|
|
|
|
},
|
|
|
|
ShardConfig {
|
|
|
|
shard_id: 0,
|
|
|
|
num_shard: 1 << 25
|
|
|
|
},
|
|
|
|
]));
|
|
|
|
assert!(!all_shards_available(vec![
|
|
|
|
ShardConfig {
|
|
|
|
shard_id: 0,
|
|
|
|
num_shard: 4
|
|
|
|
},
|
|
|
|
ShardConfig {
|
|
|
|
shard_id: 1,
|
|
|
|
num_shard: 4
|
|
|
|
},
|
|
|
|
ShardConfig {
|
|
|
|
shard_id: 3,
|
|
|
|
num_shard: 8
|
|
|
|
},
|
|
|
|
ShardConfig {
|
|
|
|
shard_id: 0,
|
|
|
|
num_shard: 2
|
|
|
|
},
|
|
|
|
]));
|
|
|
|
}
|
|
|
|
}
|