2020-01-20 14:40:58 +03:00
|
|
|
// Copyright 2020 The Grin Developers
|
2019-11-26 23:21:49 +03:00
|
|
|
//
|
|
|
|
// 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 self::chain::txhashset::BitmapAccumulator;
|
|
|
|
use self::core::core::hash::Hash;
|
|
|
|
use self::core::ser::PMMRIndexHashable;
|
|
|
|
use bit_vec::BitVec;
|
|
|
|
use grin_chain as chain;
|
|
|
|
use grin_core as core;
|
|
|
|
use grin_util as util;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_bitmap_accumulator() {
|
|
|
|
util::init_test_logger();
|
|
|
|
|
|
|
|
let mut accumulator = BitmapAccumulator::new();
|
|
|
|
assert_eq!(accumulator.root(), Hash::default());
|
|
|
|
|
|
|
|
// 1000... (rebuild from 0, setting [0] true)
|
|
|
|
accumulator.apply(vec![0], vec![0], 1).unwrap();
|
|
|
|
let expected_hash = {
|
|
|
|
let mut bit_vec = BitVec::from_elem(1024, false);
|
|
|
|
bit_vec.set(0, true);
|
|
|
|
bit_vec.to_bytes().hash_with_index(0)
|
|
|
|
};
|
|
|
|
assert_eq!(accumulator.root(), expected_hash);
|
|
|
|
|
|
|
|
// 1100... (rebuild from 0, setting [0, 1] true)
|
|
|
|
accumulator.apply(vec![0], vec![0, 1], 2).unwrap();
|
|
|
|
let expected_hash = {
|
|
|
|
let mut bit_vec = BitVec::from_elem(1024, false);
|
|
|
|
bit_vec.set(0, true);
|
|
|
|
bit_vec.set(1, true);
|
|
|
|
bit_vec.to_bytes().hash_with_index(0)
|
|
|
|
};
|
|
|
|
assert_eq!(accumulator.root(), expected_hash);
|
|
|
|
|
|
|
|
// 0100... (rebuild from 0, setting [1] true, which will reset [0] false)
|
|
|
|
accumulator.apply(vec![0], vec![1], 2).unwrap();
|
|
|
|
let expected_hash = {
|
|
|
|
let mut bit_vec = BitVec::from_elem(1024, false);
|
|
|
|
bit_vec.set(1, true);
|
|
|
|
let expected_bytes = bit_vec.to_bytes();
|
|
|
|
expected_bytes.hash_with_index(0)
|
|
|
|
};
|
|
|
|
assert_eq!(accumulator.root(), expected_hash);
|
|
|
|
|
|
|
|
// 0100... (rebuild from 1, setting [1] true)
|
|
|
|
accumulator.apply(vec![1], vec![1], 2).unwrap();
|
|
|
|
let expected_hash = {
|
|
|
|
let mut bit_vec = BitVec::from_elem(1024, false);
|
|
|
|
bit_vec.set(1, true);
|
|
|
|
let expected_bytes = bit_vec.to_bytes();
|
|
|
|
expected_bytes.hash_with_index(0)
|
|
|
|
};
|
|
|
|
assert_eq!(accumulator.root(), expected_hash);
|
|
|
|
|
|
|
|
// 0100...0001 (rebuild from 0, setting [1, 1023] true)
|
|
|
|
accumulator.apply(vec![0], vec![1, 1023], 1024).unwrap();
|
|
|
|
let expected_hash = {
|
|
|
|
let mut bit_vec = BitVec::from_elem(1024, false);
|
|
|
|
bit_vec.set(1, true);
|
|
|
|
bit_vec.set(1023, true);
|
|
|
|
let expected_bytes = bit_vec.to_bytes();
|
|
|
|
expected_bytes.hash_with_index(0)
|
|
|
|
};
|
|
|
|
assert_eq!(accumulator.root(), expected_hash);
|
|
|
|
|
|
|
|
// Now set bits such that we extend the bitmap accumulator across multiple 1024 bit chunks.
|
|
|
|
// We need a second bit_vec here to reflect the additional chunk.
|
|
|
|
// 0100...0001, 1000...0000 (rebuild from 0, setting [1, 1023, 1024] true)
|
|
|
|
accumulator
|
|
|
|
.apply(vec![0], vec![1, 1023, 1024], 1025)
|
|
|
|
.unwrap();
|
|
|
|
let expected_hash = {
|
|
|
|
let mut bit_vec = BitVec::from_elem(1024, false);
|
|
|
|
bit_vec.set(1, true);
|
|
|
|
bit_vec.set(1023, true);
|
|
|
|
let mut bit_vec2 = BitVec::from_elem(1024, false);
|
|
|
|
bit_vec2.set(0, true);
|
|
|
|
let expected_bytes_0 = bit_vec.to_bytes();
|
|
|
|
let expected_bytes_1 = bit_vec2.to_bytes();
|
|
|
|
let expected_hash_0 = expected_bytes_0.hash_with_index(0);
|
|
|
|
let expected_hash_1 = expected_bytes_1.hash_with_index(1);
|
|
|
|
(expected_hash_0, expected_hash_1).hash_with_index(2)
|
|
|
|
};
|
|
|
|
assert_eq!(accumulator.root(), expected_hash);
|
|
|
|
|
|
|
|
// Just rebuild the second bitmap chunk.
|
|
|
|
// 0100...0001, 0100...0000 (rebuild from 1025, setting [1025] true)
|
|
|
|
accumulator.apply(vec![1025], vec![1025], 1026).unwrap();
|
|
|
|
let expected_hash = {
|
|
|
|
let mut bit_vec = BitVec::from_elem(1024, false);
|
|
|
|
bit_vec.set(1, true);
|
|
|
|
bit_vec.set(1023, true);
|
|
|
|
let mut bit_vec2 = BitVec::from_elem(1024, false);
|
|
|
|
bit_vec2.set(1, true);
|
|
|
|
let expected_bytes_0 = bit_vec.to_bytes();
|
|
|
|
let expected_bytes_1 = bit_vec2.to_bytes();
|
|
|
|
let expected_hash_0 = expected_bytes_0.hash_with_index(0);
|
|
|
|
let expected_hash_1 = expected_bytes_1.hash_with_index(1);
|
|
|
|
(expected_hash_0, expected_hash_1).hash_with_index(2)
|
|
|
|
};
|
|
|
|
assert_eq!(accumulator.root(), expected_hash);
|
|
|
|
|
|
|
|
// Rebuild the first bitmap chunk and all chunks after it.
|
|
|
|
// 0100...0000, 0100...0000 (rebuild from 1, setting [1, 1025] true)
|
|
|
|
accumulator.apply(vec![1], vec![1, 1025], 1026).unwrap();
|
|
|
|
let expected_hash = {
|
|
|
|
let mut bit_vec = BitVec::from_elem(1024, false);
|
|
|
|
bit_vec.set(1, true);
|
|
|
|
let mut bit_vec2 = BitVec::from_elem(1024, false);
|
|
|
|
bit_vec2.set(1, true);
|
|
|
|
let expected_bytes_0 = bit_vec.to_bytes();
|
|
|
|
let expected_bytes_1 = bit_vec2.to_bytes();
|
|
|
|
let expected_hash_0 = expected_bytes_0.hash_with_index(0);
|
|
|
|
let expected_hash_1 = expected_bytes_1.hash_with_index(1);
|
|
|
|
(expected_hash_0, expected_hash_1).hash_with_index(2)
|
|
|
|
};
|
|
|
|
assert_eq!(accumulator.root(), expected_hash);
|
|
|
|
|
|
|
|
// Make sure we handle the case where the first chunk is all 0s
|
|
|
|
// 0000...0000, 0100...0000 (rebuild from 1, setting [1025] true)
|
|
|
|
accumulator.apply(vec![1], vec![1025], 1026).unwrap();
|
|
|
|
let expected_hash = {
|
|
|
|
let bit_vec = BitVec::from_elem(1024, false);
|
|
|
|
let mut bit_vec2 = BitVec::from_elem(1024, false);
|
|
|
|
bit_vec2.set(1, true);
|
|
|
|
let expected_bytes_0 = bit_vec.to_bytes();
|
|
|
|
let expected_bytes_1 = bit_vec2.to_bytes();
|
|
|
|
let expected_hash_0 = expected_bytes_0.hash_with_index(0);
|
|
|
|
let expected_hash_1 = expected_bytes_1.hash_with_index(1);
|
|
|
|
(expected_hash_0, expected_hash_1).hash_with_index(2)
|
|
|
|
};
|
|
|
|
assert_eq!(accumulator.root(), expected_hash);
|
|
|
|
|
|
|
|
// Check that removing the last bit in a chunk removes the now empty chunk
|
|
|
|
// if it is the rightmost chunk.
|
|
|
|
// 0000...0001 (rebuild from 1023, setting [1023] true)
|
|
|
|
accumulator.apply(vec![1023], vec![1023], 1024).unwrap();
|
|
|
|
let expected_hash = {
|
|
|
|
let mut bit_vec = BitVec::from_elem(1024, false);
|
|
|
|
bit_vec.set(1023, true);
|
|
|
|
let expected_bytes = bit_vec.to_bytes();
|
|
|
|
expected_bytes.hash_with_index(0)
|
|
|
|
};
|
|
|
|
assert_eq!(accumulator.root(), expected_hash);
|
|
|
|
|
|
|
|
// Make sure we pad appropriately with 0s if we set a distant bit to 1.
|
|
|
|
// Start with an empty accumulator.
|
|
|
|
// 0000...0000, 0000...0000, 0000...0000, 0000...0001 (rebuild from 4095, setting [4095] true)
|
|
|
|
let mut accumulator = BitmapAccumulator::new();
|
|
|
|
accumulator.apply(vec![4095], vec![4095], 4096).unwrap();
|
|
|
|
let expected_hash = {
|
|
|
|
let bit_vec0 = BitVec::from_elem(1024, false);
|
|
|
|
let bit_vec1 = BitVec::from_elem(1024, false);
|
|
|
|
let bit_vec2 = BitVec::from_elem(1024, false);
|
|
|
|
let mut bit_vec3 = BitVec::from_elem(1024, false);
|
|
|
|
bit_vec3.set(1023, true);
|
|
|
|
|
|
|
|
let expected_bytes_0 = bit_vec0.to_bytes();
|
|
|
|
let expected_bytes_1 = bit_vec1.to_bytes();
|
|
|
|
let expected_bytes_2 = bit_vec2.to_bytes();
|
|
|
|
let expected_bytes_3 = bit_vec3.to_bytes();
|
|
|
|
|
|
|
|
let expected_hash_0 = expected_bytes_0.hash_with_index(0);
|
|
|
|
let expected_hash_1 = expected_bytes_1.hash_with_index(1);
|
|
|
|
let expected_hash_2 = (expected_hash_0, expected_hash_1).hash_with_index(2);
|
|
|
|
|
|
|
|
let expected_hash_3 = expected_bytes_2.hash_with_index(3);
|
|
|
|
let expected_hash_4 = expected_bytes_3.hash_with_index(4);
|
|
|
|
let expected_hash_5 = (expected_hash_3, expected_hash_4).hash_with_index(5);
|
|
|
|
|
|
|
|
(expected_hash_2, expected_hash_5).hash_with_index(6)
|
|
|
|
};
|
|
|
|
assert_eq!(accumulator.root(), expected_hash);
|
|
|
|
}
|