0g-storage-node/node/sync/src/test_util.rs
2024-01-04 18:05:32 +08:00

151 lines
4.4 KiB
Rust

use rand::random;
use shared_types::{compute_padded_chunk_size, ChunkArray, Transaction, CHUNK_SIZE};
use std::{cmp, sync::Arc};
use storage::{
log_store::{
log_manager::{sub_merkle_tree, tx_subtree_root_list_padded, LogConfig, PORA_CHUNK_SIZE},
LogStoreChunkWrite, LogStoreWrite,
},
LogManager,
};
use tokio::sync::RwLock;
/// Creates stores for local node and peers with initialized transaction of specified chunk count.
/// The first store is for local node, and data not stored. The second store is for peers, and all
/// transactions are finalized for file sync.
#[allow(clippy::type_complexity)]
pub fn create_2_store(
chunk_count: Vec<usize>,
) -> (
Arc<RwLock<LogManager>>,
Arc<RwLock<LogManager>>,
Vec<Transaction>,
Vec<Vec<u8>>,
) {
let config = LogConfig::default();
let mut store = LogManager::memorydb(config.clone()).unwrap();
let mut peer_store = LogManager::memorydb(config).unwrap();
let mut offset = 1;
let mut txs = vec![];
let mut data = vec![];
for (tx_seq, chunks) in chunk_count.iter().enumerate() {
let ret = generate_data(*chunks, &mut store, &mut peer_store, tx_seq as u64, offset);
txs.push(ret.0);
data.push(ret.1);
offset = ret.2;
}
(
Arc::new(RwLock::new(store)),
Arc::new(RwLock::new(peer_store)),
txs,
data,
)
}
fn generate_data(
chunk_count: usize,
store: &mut LogManager,
peer_store: &mut LogManager,
seq: u64,
offset: u64,
) -> (Transaction, Vec<u8>, u64) {
let data_size = CHUNK_SIZE * chunk_count;
let mut data = vec![0u8; data_size];
for i in 0..chunk_count {
data[i * CHUNK_SIZE] = random();
}
let merkel_nodes = tx_subtree_root_list_padded(&data);
let first_tree_size = 1 << (merkel_nodes[0].0 - 1);
let start_offset = if offset % first_tree_size == 0 {
offset
} else {
(offset / first_tree_size + 1) * first_tree_size
};
let merkle = sub_merkle_tree(&data).unwrap();
let tx = Transaction {
stream_ids: vec![],
size: data_size as u64,
data_merkle_root: merkle.root().into(),
seq,
data: vec![],
start_entry_index: start_offset,
merkle_nodes: merkel_nodes,
};
store.put_tx(tx.clone()).unwrap();
peer_store.put_tx(tx.clone()).unwrap();
for start_index in (0..chunk_count).step_by(PORA_CHUNK_SIZE) {
let end = cmp::min((start_index + PORA_CHUNK_SIZE) * CHUNK_SIZE, data.len());
let chunk_array = ChunkArray {
data: data[start_index * CHUNK_SIZE..end].to_vec(),
start_index: start_index as u64,
};
peer_store.put_chunks(tx.seq, chunk_array.clone()).unwrap();
}
peer_store.finalize_tx(tx.seq).unwrap();
let (padded_chunk_count, _) = compute_padded_chunk_size(data_size);
(tx, data, start_offset + padded_chunk_count as u64)
}
#[cfg(test)]
pub mod tests {
use file_location_cache::{test_util::AnnounceFileBuilder, FileLocationCache};
use libp2p::PeerId;
use shared_types::TxID;
use std::sync::Arc;
use storage::{
log_store::{log_manager::LogConfig, Store as LogStore},
LogManager,
};
use storage_async::Store;
use task_executor::test_utils::TestRuntime;
use tokio::sync::RwLock;
pub struct TestStoreRuntime {
pub runtime: TestRuntime,
pub store: Store,
}
impl Default for TestStoreRuntime {
fn default() -> Self {
let store = Arc::new(RwLock::new(Self::new_store()));
Self::new(store)
}
}
impl TestStoreRuntime {
pub fn new_store() -> impl LogStore {
LogManager::memorydb(LogConfig::default()).unwrap()
}
pub fn new(store: Arc<RwLock<dyn LogStore>>) -> TestStoreRuntime {
let runtime = TestRuntime::default();
let executor = runtime.task_executor.clone();
Self {
runtime,
store: Store::new(store, executor),
}
}
}
pub fn create_file_location_cache(peer_id: PeerId, txs: Vec<TxID>) -> Arc<FileLocationCache> {
let cache = FileLocationCache::default();
for tx_id in txs {
let announcement = AnnounceFileBuilder::default()
.with_tx_id(tx_id)
.with_peer_id(peer_id)
.build();
cache.insert(announcement);
}
Arc::new(cache)
}
}