2020-01-20 14:40:58 +03:00
|
|
|
// Copyright 2020 The Grin Developers
|
2018-06-20 22:18:52 +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.
|
|
|
|
|
2019-11-26 23:21:49 +03:00
|
|
|
mod common;
|
2018-06-20 22:18:52 +03:00
|
|
|
|
2018-12-08 02:59:40 +03:00
|
|
|
use self::core::core::merkle_proof::MerkleProof;
|
2020-09-29 16:57:33 +03:00
|
|
|
use self::core::core::pmmr::{ReadablePMMR, VecBackend, PMMR};
|
2019-06-27 19:19:41 +03:00
|
|
|
use self::core::ser::{self, PMMRIndexHashable};
|
2019-11-26 23:21:49 +03:00
|
|
|
use crate::common::TestElem;
|
2018-12-08 02:59:40 +03:00
|
|
|
use grin_core as core;
|
2018-06-20 22:18:52 +03:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn empty_merkle_proof() {
|
|
|
|
let proof = MerkleProof::empty();
|
|
|
|
assert_eq!(proof.path, vec![]);
|
|
|
|
assert_eq!(proof.mmr_size, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn merkle_proof_ser_deser() {
|
|
|
|
let mut ba = VecBackend::new();
|
|
|
|
let mut pmmr = PMMR::new(&mut ba);
|
|
|
|
for x in 0..15 {
|
2018-12-02 22:59:24 +03:00
|
|
|
pmmr.push(&TestElem([0, 0, 0, x])).unwrap();
|
2018-06-20 22:18:52 +03:00
|
|
|
}
|
|
|
|
let proof = pmmr.merkle_proof(9).unwrap();
|
|
|
|
|
|
|
|
let mut vec = Vec::new();
|
2019-07-06 17:51:03 +03:00
|
|
|
ser::serialize_default(&mut vec, &proof).expect("serialization failed");
|
2019-06-27 19:19:41 +03:00
|
|
|
let proof_2: MerkleProof = ser::deserialize_default(&mut &vec[..]).unwrap();
|
2018-06-20 22:18:52 +03:00
|
|
|
|
|
|
|
assert_eq!(proof, proof_2);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn pmmr_merkle_proof_prune_and_rewind() {
|
|
|
|
let mut ba = VecBackend::new();
|
|
|
|
let mut pmmr = PMMR::new(&mut ba);
|
2018-12-02 22:59:24 +03:00
|
|
|
pmmr.push(&TestElem([0, 0, 0, 1])).unwrap();
|
|
|
|
pmmr.push(&TestElem([0, 0, 0, 2])).unwrap();
|
2018-06-20 22:18:52 +03:00
|
|
|
let proof = pmmr.merkle_proof(2).unwrap();
|
|
|
|
|
|
|
|
// now prune an element and check we can still generate
|
|
|
|
// the correct Merkle proof for the other element (after sibling pruned)
|
|
|
|
pmmr.prune(1).unwrap();
|
|
|
|
let proof_2 = pmmr.merkle_proof(2).unwrap();
|
|
|
|
assert_eq!(proof, proof_2);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn pmmr_merkle_proof() {
|
|
|
|
let elems = [
|
|
|
|
TestElem([0, 0, 0, 1]),
|
|
|
|
TestElem([0, 0, 0, 2]),
|
|
|
|
TestElem([0, 0, 0, 3]),
|
|
|
|
TestElem([0, 0, 0, 4]),
|
|
|
|
TestElem([0, 0, 0, 5]),
|
|
|
|
TestElem([0, 0, 0, 6]),
|
|
|
|
TestElem([0, 0, 0, 7]),
|
|
|
|
TestElem([0, 0, 0, 8]),
|
|
|
|
TestElem([1, 0, 0, 0]),
|
|
|
|
];
|
|
|
|
|
|
|
|
let mut ba = VecBackend::new();
|
|
|
|
let mut pmmr = PMMR::new(&mut ba);
|
|
|
|
|
2018-12-02 22:59:24 +03:00
|
|
|
pmmr.push(&elems[0]).unwrap();
|
2018-06-20 22:18:52 +03:00
|
|
|
let pos_0 = elems[0].hash_with_index(0);
|
|
|
|
assert_eq!(pmmr.get_hash(1).unwrap(), pos_0);
|
|
|
|
|
|
|
|
let proof = pmmr.merkle_proof(1).unwrap();
|
|
|
|
assert_eq!(proof.path, vec![]);
|
2019-08-09 17:10:54 +03:00
|
|
|
assert!(proof.verify(pmmr.root().unwrap(), &elems[0], 1).is_ok());
|
2018-06-20 22:18:52 +03:00
|
|
|
|
2018-12-02 22:59:24 +03:00
|
|
|
pmmr.push(&elems[1]).unwrap();
|
2018-06-20 22:18:52 +03:00
|
|
|
let pos_1 = elems[1].hash_with_index(1);
|
|
|
|
assert_eq!(pmmr.get_hash(2).unwrap(), pos_1);
|
|
|
|
let pos_2 = (pos_0, pos_1).hash_with_index(2);
|
|
|
|
assert_eq!(pmmr.get_hash(3).unwrap(), pos_2);
|
|
|
|
|
2019-08-09 17:10:54 +03:00
|
|
|
assert_eq!(pmmr.root().unwrap(), pos_2);
|
2020-09-29 16:57:33 +03:00
|
|
|
assert_eq!(pmmr.peaks(), vec![pos_2]);
|
2018-06-20 22:18:52 +03:00
|
|
|
|
|
|
|
// single peak, path with single sibling
|
|
|
|
let proof = pmmr.merkle_proof(1).unwrap();
|
|
|
|
assert_eq!(proof.path, vec![pos_1]);
|
2019-08-09 17:10:54 +03:00
|
|
|
assert!(proof.verify(pmmr.root().unwrap(), &elems[0], 1).is_ok());
|
2018-06-20 22:18:52 +03:00
|
|
|
|
|
|
|
let proof = pmmr.merkle_proof(2).unwrap();
|
|
|
|
assert_eq!(proof.path, vec![pos_0]);
|
2019-08-09 17:10:54 +03:00
|
|
|
assert!(proof.verify(pmmr.root().unwrap(), &elems[1], 2).is_ok());
|
2018-06-20 22:18:52 +03:00
|
|
|
|
|
|
|
// three leaves, two peaks (one also the right-most leaf)
|
2018-12-02 22:59:24 +03:00
|
|
|
pmmr.push(&elems[2]).unwrap();
|
2018-06-20 22:18:52 +03:00
|
|
|
let pos_3 = elems[2].hash_with_index(3);
|
|
|
|
assert_eq!(pmmr.get_hash(4).unwrap(), pos_3);
|
|
|
|
|
2019-08-09 17:10:54 +03:00
|
|
|
assert_eq!(pmmr.root().unwrap(), (pos_2, pos_3).hash_with_index(4));
|
2020-09-29 16:57:33 +03:00
|
|
|
assert_eq!(pmmr.peaks(), vec![pos_2, pos_3]);
|
2018-06-20 22:18:52 +03:00
|
|
|
|
|
|
|
let proof = pmmr.merkle_proof(1).unwrap();
|
|
|
|
assert_eq!(proof.path, vec![pos_1, pos_3]);
|
2019-08-09 17:10:54 +03:00
|
|
|
assert!(proof.verify(pmmr.root().unwrap(), &elems[0], 1).is_ok());
|
2018-06-20 22:18:52 +03:00
|
|
|
|
|
|
|
let proof = pmmr.merkle_proof(2).unwrap();
|
|
|
|
assert_eq!(proof.path, vec![pos_0, pos_3]);
|
2019-08-09 17:10:54 +03:00
|
|
|
assert!(proof.verify(pmmr.root().unwrap(), &elems[1], 2).is_ok());
|
2018-06-20 22:18:52 +03:00
|
|
|
|
|
|
|
let proof = pmmr.merkle_proof(4).unwrap();
|
|
|
|
assert_eq!(proof.path, vec![pos_2]);
|
2019-08-09 17:10:54 +03:00
|
|
|
assert!(proof.verify(pmmr.root().unwrap(), &elems[2], 4).is_ok());
|
2018-06-20 22:18:52 +03:00
|
|
|
|
|
|
|
// 7 leaves, 3 peaks, 11 pos in total
|
2018-12-02 22:59:24 +03:00
|
|
|
pmmr.push(&elems[3]).unwrap();
|
2018-06-20 22:18:52 +03:00
|
|
|
let pos_4 = elems[3].hash_with_index(4);
|
|
|
|
assert_eq!(pmmr.get_hash(5).unwrap(), pos_4);
|
|
|
|
let pos_5 = (pos_3, pos_4).hash_with_index(5);
|
|
|
|
assert_eq!(pmmr.get_hash(6).unwrap(), pos_5);
|
|
|
|
let pos_6 = (pos_2, pos_5).hash_with_index(6);
|
|
|
|
assert_eq!(pmmr.get_hash(7).unwrap(), pos_6);
|
|
|
|
|
2018-12-02 22:59:24 +03:00
|
|
|
pmmr.push(&elems[4]).unwrap();
|
2018-06-20 22:18:52 +03:00
|
|
|
let pos_7 = elems[4].hash_with_index(7);
|
|
|
|
assert_eq!(pmmr.get_hash(8).unwrap(), pos_7);
|
|
|
|
|
2018-12-02 22:59:24 +03:00
|
|
|
pmmr.push(&elems[5]).unwrap();
|
2018-06-20 22:18:52 +03:00
|
|
|
let pos_8 = elems[5].hash_with_index(8);
|
|
|
|
assert_eq!(pmmr.get_hash(9).unwrap(), pos_8);
|
|
|
|
|
|
|
|
let pos_9 = (pos_7, pos_8).hash_with_index(9);
|
|
|
|
assert_eq!(pmmr.get_hash(10).unwrap(), pos_9);
|
|
|
|
|
2018-12-02 22:59:24 +03:00
|
|
|
pmmr.push(&elems[6]).unwrap();
|
2018-06-20 22:18:52 +03:00
|
|
|
let pos_10 = elems[6].hash_with_index(10);
|
|
|
|
assert_eq!(pmmr.get_hash(11).unwrap(), pos_10);
|
|
|
|
|
|
|
|
assert_eq!(pmmr.unpruned_size(), 11);
|
|
|
|
|
|
|
|
let proof = pmmr.merkle_proof(1).unwrap();
|
|
|
|
assert_eq!(
|
|
|
|
proof.path,
|
|
|
|
vec![pos_1, pos_5, (pos_9, pos_10).hash_with_index(11)]
|
|
|
|
);
|
2019-08-09 17:10:54 +03:00
|
|
|
assert!(proof.verify(pmmr.root().unwrap(), &elems[0], 1).is_ok());
|
2018-06-20 22:18:52 +03:00
|
|
|
|
|
|
|
let proof = pmmr.merkle_proof(2).unwrap();
|
|
|
|
assert_eq!(
|
|
|
|
proof.path,
|
|
|
|
vec![pos_0, pos_5, (pos_9, pos_10).hash_with_index(11)]
|
|
|
|
);
|
2019-08-09 17:10:54 +03:00
|
|
|
assert!(proof.verify(pmmr.root().unwrap(), &elems[1], 2).is_ok());
|
2018-06-20 22:18:52 +03:00
|
|
|
|
|
|
|
let proof = pmmr.merkle_proof(4).unwrap();
|
|
|
|
assert_eq!(
|
|
|
|
proof.path,
|
|
|
|
vec![pos_4, pos_2, (pos_9, pos_10).hash_with_index(11)]
|
|
|
|
);
|
2019-08-09 17:10:54 +03:00
|
|
|
assert!(proof.verify(pmmr.root().unwrap(), &elems[2], 4).is_ok());
|
2018-06-20 22:18:52 +03:00
|
|
|
|
|
|
|
let proof = pmmr.merkle_proof(5).unwrap();
|
|
|
|
assert_eq!(
|
|
|
|
proof.path,
|
|
|
|
vec![pos_3, pos_2, (pos_9, pos_10).hash_with_index(11)]
|
|
|
|
);
|
2019-08-09 17:10:54 +03:00
|
|
|
assert!(proof.verify(pmmr.root().unwrap(), &elems[3], 5).is_ok());
|
2018-06-20 22:18:52 +03:00
|
|
|
|
|
|
|
let proof = pmmr.merkle_proof(8).unwrap();
|
|
|
|
assert_eq!(proof.path, vec![pos_8, pos_10, pos_6]);
|
2019-08-09 17:10:54 +03:00
|
|
|
assert!(proof.verify(pmmr.root().unwrap(), &elems[4], 8).is_ok());
|
2018-06-20 22:18:52 +03:00
|
|
|
|
|
|
|
let proof = pmmr.merkle_proof(9).unwrap();
|
|
|
|
assert_eq!(proof.path, vec![pos_7, pos_10, pos_6]);
|
2019-08-09 17:10:54 +03:00
|
|
|
assert!(proof.verify(pmmr.root().unwrap(), &elems[5], 9).is_ok());
|
2018-06-20 22:18:52 +03:00
|
|
|
|
|
|
|
let proof = pmmr.merkle_proof(11).unwrap();
|
|
|
|
assert_eq!(proof.path, vec![pos_9, pos_6]);
|
2019-08-09 17:10:54 +03:00
|
|
|
assert!(proof.verify(pmmr.root().unwrap(), &elems[6], 11).is_ok());
|
2018-06-20 22:18:52 +03:00
|
|
|
}
|