53414ae105
* use 0-based positions in methods pmmr_leaf_to_insertion_index and bintree_postorder_height; add round_up_to_leaf_pos method
* use 0-based positions in method insertion_to_pmmr_index
* use 0-based positions in method is_leaf
* use 0-based positions in method family()
* use 0-based positions in method is_left_sibling
* use 0-based positions in method family_branch
* use 0-based positions in methods bintree_{left,right}most
* use 0-based positions in method bintree_pos_iter
* use 0-based positions in method bintree_range
* use 0-based positions in method bintree_leaf_pos_iter
* rename last_pos in MMR related structs to size
* use 0-based positions in method prune
* use 0-based positions in method push and apply_output return value
* use 0-based position argument of method merkle_proof
* use 0-based outputs in method pmmr::peaks
* fix peaks() code comments
* refix peaks() code comments
* use 0-based positions in method get_peak_from_file
* use 0-based positions in methods get_data_from_file
* use 0-based positions in methods get_from_file
* use 0-based positions in methods get_data
* use 0-based positions in methods get_hash
* use 0-based positions in method peak_path
* use 0-based positions in method bag_the_rhs
* use 0-based positions in method Backend::remove
* use 0-based positions in method leaf_pos_iter
* use 0-based positions in method self.LeafSet::includes
* use 0-based positions in methods self.LeafSet::{add,remove}
* use 0-based positions in methods is_pruned,is_pruned_root,is_compacted
* use 0-based positions in methods PruneList::append
* use 0-based positions in methods append_pruned_subtree
* use 0-based positions in method calculate_next_leaf_shift
* use 0-based positions in method append_single
* use 0-based positions in method calculate_next_shift
* use 0-based positions in method segment_pos_range
* use 0-based positions in method reconstruct_root
* use 0-based positions in method validate_with
* use 0-based positions in method validate
* rename size (formerly last_pos) to mmr_size
* use 0-based positions in Segment's hash_pos and leaf_pos
* minimize use of saturating_sub(1) and rename some pos/idx to size
* use 0-based positions in methods get_output_pos
* use 0-based positions in method get_unspent_output_at
* use 0-based positions in method get_header_hash
* use 0-based positions in methods MerkleProof::verify{,_consume}
* use 0-based positions in method cleanup_subtree
* don't allow 0 in prunelist bitmap
* use 0-based positions in methods get_{,leaf_}shift
* rename some 1-based pos to pos1; identify TODO
* Address yeastplume's PR review comments
109 lines
3.0 KiB
Rust
109 lines
3.0 KiB
Rust
// Copyright 2021 The Grin Developers
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
use env_logger;
|
|
|
|
use grin_store as store;
|
|
|
|
use chrono::prelude::Utc;
|
|
use std::fs;
|
|
use std::time::{Duration, Instant};
|
|
|
|
use croaring::Bitmap;
|
|
|
|
use crate::store::leaf_set::LeafSet;
|
|
|
|
pub fn as_millis(d: Duration) -> u128 {
|
|
d.as_secs() as u128 * 1_000 as u128 + (d.subsec_nanos() / (1_000 * 1_000)) as u128
|
|
}
|
|
|
|
#[test]
|
|
fn test_leaf_set_performance() {
|
|
let (mut leaf_set, data_dir) = setup("leaf_set_perf");
|
|
|
|
println!("Timing some common operations:");
|
|
|
|
// Add a million pos to the set, syncing data to disk in 1,000 pos chunks
|
|
// Simulating 1,000 blocks with 1,000 outputs each.
|
|
let now = Instant::now();
|
|
for x in 0..1_000 {
|
|
for y in 0..1_000 {
|
|
let pos = (x * 1_000) + y;
|
|
leaf_set.add(pos);
|
|
}
|
|
leaf_set.flush().unwrap();
|
|
}
|
|
assert_eq!(leaf_set.len(), 1_000_000);
|
|
println!(
|
|
"Adding 1,000 chunks of 1,000 pos to leaf_set took {}ms",
|
|
as_millis(now.elapsed())
|
|
);
|
|
|
|
// Simulate looking up existence of a large number of pos in the leaf_set.
|
|
let now = Instant::now();
|
|
for x in 0..1_000_000 {
|
|
assert!(leaf_set.includes(x));
|
|
}
|
|
println!(
|
|
"Checking 1,000,000 inclusions in leaf_set took {}ms",
|
|
as_millis(now.elapsed())
|
|
);
|
|
|
|
// Remove a large number of pos in chunks to simulate blocks containing tx
|
|
// spending outputs. Simulate 1,000 blocks each spending 1,000 outputs.
|
|
let now = Instant::now();
|
|
for x in 0..1_000 {
|
|
for y in 0..1_000 {
|
|
let pos = (x * 1_000) + y;
|
|
leaf_set.remove(pos);
|
|
}
|
|
leaf_set.flush().unwrap();
|
|
}
|
|
assert_eq!(leaf_set.len(), 0);
|
|
println!(
|
|
"Removing 1,000 chunks of 1,000 pos from leaf_set took {}ms",
|
|
as_millis(now.elapsed())
|
|
);
|
|
|
|
// Rewind pos in chunks of 1,000 to simulate rewinding over the same blocks.
|
|
let now = Instant::now();
|
|
for x in 0..1_000 {
|
|
let from_pos = x * 1_000 + 1;
|
|
let to_pos = from_pos + 1_000;
|
|
let bitmap: Bitmap = (from_pos..to_pos).collect();
|
|
leaf_set.rewind(1_000_000, &bitmap);
|
|
}
|
|
assert_eq!(leaf_set.len(), 1_000_000);
|
|
println!(
|
|
"Rewinding 1,000 chunks of 1,000 pos from leaf_set took {}ms",
|
|
as_millis(now.elapsed())
|
|
);
|
|
|
|
// panic!("stop here to display results");
|
|
|
|
teardown(data_dir);
|
|
}
|
|
|
|
fn setup(test_name: &str) -> (LeafSet, String) {
|
|
let _ = env_logger::init();
|
|
let data_dir = format!("./target/{}-{}", test_name, Utc::now().timestamp());
|
|
fs::create_dir_all(data_dir.clone()).unwrap();
|
|
let leaf_set = LeafSet::open(&format!("{}/{}", data_dir, "utxo.bin")).unwrap();
|
|
(leaf_set, data_dir)
|
|
}
|
|
|
|
fn teardown(data_dir: String) {
|
|
fs::remove_dir_all(data_dir).unwrap();
|
|
}
|