Fix pruning of last PMMR leaf, additional tests

Due to the construction of PMMRs the last element, when its a leaf,
can never be pruned as it has no parent yet and it will be needed
to calculate that hash. To work around this, we now insert coinbase
outputs first to add at least one output of padding.

Also changed the `set_sumtree_root` function on chain a bit to allow
setting the roots on a fork. Mostly useful for tests.

Added new test case to handle both the issue above and spending
transactions within a fork.
This commit is contained in:
Ignotus Peverell 2018-01-08 01:23:23 +00:00
parent 7f52b6c361
commit 6a9a584c43
No known key found for this signature in database
GPG key ID: 99CD25F39F8F8211
6 changed files with 224 additions and 99 deletions

View file

@ -358,11 +358,15 @@ impl Chain {
/// Sets the sumtree roots on a brand new block by applying the block on the /// Sets the sumtree roots on a brand new block by applying the block on the
/// current sumtree state. /// current sumtree state.
pub fn set_sumtree_roots(&self, b: &mut Block) -> Result<(), Error> { pub fn set_sumtree_roots(&self, b: &mut Block, is_fork: bool) -> Result<(), Error> {
let mut sumtrees = self.sumtrees.write().unwrap(); let mut sumtrees = self.sumtrees.write().unwrap();
let store = self.store.clone();
let roots = sumtree::extending(&mut sumtrees, |extension| { let roots = sumtree::extending(&mut sumtrees, |extension| {
// apply the block on the sumtrees and check the resulting root // apply the block on the sumtrees and check the resulting root
if is_fork {
pipe::rewind_and_apply_fork(b, store, extension)?;
}
extension.apply_block(b)?; extension.apply_block(b)?;
extension.force_rollback(); extension.force_rollback();
Ok(extension.roots()) Ok(extension.roots())
@ -374,7 +378,7 @@ impl Chain {
Ok(()) Ok(())
} }
/// returs sumtree roots /// Returns current sumtree roots
pub fn get_sumtree_roots( pub fn get_sumtree_roots(
&self, &self,
) -> ( ) -> (

View file

@ -393,11 +393,11 @@ fn update_header_head(bh: &BlockHeader, ctx: &mut BlockContext) -> Result<Option
} }
} }
// Utility function to handle forks. From the forked block, jump backward /// Utility function to handle forks. From the forked block, jump backward
// to find to fork root. Rewind the sumtrees to the root and apply all the /// to find to fork root. Rewind the sumtrees to the root and apply all the
// forked blocks prior to the one being processed to set the sumtrees in /// forked blocks prior to the one being processed to set the sumtrees in
// the expected state. /// the expected state.
fn rewind_and_apply_fork( pub fn rewind_and_apply_fork(
b: &Block, b: &Block,
store: Arc<ChainStore>, store: Arc<ChainStore>,
ext: &mut sumtree::Extension, ext: &mut sumtree::Extension,

View file

@ -22,7 +22,7 @@ use std::sync::Arc;
use util::secp::pedersen::{RangeProof, Commitment}; use util::secp::pedersen::{RangeProof, Commitment};
use core::core::{Block, SumCommit, TxKernel}; use core::core::{Block, SumCommit, Input, Output, TxKernel, COINBASE_OUTPUT};
use core::core::pmmr::{HashSum, NoSum, Summable, PMMR}; use core::core::pmmr::{HashSum, NoSum, Summable, PMMR};
use core::core::hash::Hashed; use core::core::hash::Hashed;
use grin_store; use grin_store;
@ -244,16 +244,70 @@ impl<'a> Extension<'a> {
/// prune MMR data. /// prune MMR data.
pub fn apply_block(&mut self, b: &Block) -> Result<(), Error> { pub fn apply_block(&mut self, b: &Block) -> Result<(), Error> {
// doing inputs first guarantees an input can't spend an output in the // first applying coinbase outputs. due to the construction of PMMRs the
// last element, when its a leaf, can never be pruned as it has no parent
// yet and it will be needed to calculate that hash. to work around this,
// we insert coinbase outputs first to add at least one output of padding
for out in &b.outputs {
if out.features.contains(COINBASE_OUTPUT) {
self.apply_output(out)?;
}
}
// then doing inputsm guarantees an input can't spend an output in the
// same block, enforcing block cut-through // same block, enforcing block cut-through
for input in &b.inputs { for input in &b.inputs {
self.apply_input(input, b.header.height)?;
}
// now all regular, non coinbase outputs
for out in &b.outputs {
if !out.features.contains(COINBASE_OUTPUT) {
self.apply_output(out)?;
}
}
// finally, applying all kernels
for kernel in &b.kernels {
self.apply_kernel(kernel)?;
}
Ok(())
}
fn save_pos_index(&self) -> Result<(), Error> {
debug!(
LOGGER,
"sumtree: save_pos_index: outputs: {}, {:?}",
self.new_output_commits.len(),
self.new_output_commits.values().collect::<Vec<_>>(),
);
for (commit, pos) in &self.new_output_commits {
self.commit_index.save_output_pos(commit, *pos)?;
}
debug!(
LOGGER,
"sumtree: save_pos_index: kernels: {}, {:?}",
self.new_kernel_excesses.len(),
self.new_kernel_excesses.values().collect::<Vec<_>>(),
);
for (excess, pos) in &self.new_kernel_excesses {
self.commit_index.save_kernel_pos(excess, *pos)?;
}
Ok(())
}
fn apply_input(&mut self, input: &Input, height: u64) -> Result<(), Error> {
let commit = input.commitment(); let commit = input.commitment();
let pos_res = self.get_output_pos(&commit); let pos_res = self.get_output_pos(&commit);
if let Ok(pos) = pos_res { if let Ok(pos) = pos_res {
match self.output_pmmr.prune(pos, b.header.height as u32) { match self.output_pmmr.prune(pos, height as u32) {
Ok(true) => { Ok(true) => {
self.rproof_pmmr self.rproof_pmmr
.prune(pos, b.header.height as u32) .prune(pos, height as u32)
.map_err(|s| Error::SumTreeErr(s))?; .map_err(|s| Error::SumTreeErr(s))?;
} }
Ok(false) => return Err(Error::AlreadySpent(commit)), Ok(false) => return Err(Error::AlreadySpent(commit)),
@ -262,9 +316,10 @@ impl<'a> Extension<'a> {
} else { } else {
return Err(Error::AlreadySpent(commit)); return Err(Error::AlreadySpent(commit));
} }
Ok(())
} }
for out in &b.outputs { fn apply_output(&mut self, out: &Output) -> Result<(), Error> {
let commit = out.commitment(); let commit = out.commitment();
let switch_commit_hash = out.switch_commit_hash(); let switch_commit_hash = out.switch_commit_hash();
let sum_commit = SumCommit { let sum_commit = SumCommit {
@ -300,9 +355,10 @@ impl<'a> Extension<'a> {
self.rproof_pmmr self.rproof_pmmr
.push(NoSum(out.proof)) .push(NoSum(out.proof))
.map_err(&Error::SumTreeErr)?; .map_err(&Error::SumTreeErr)?;
Ok(())
} }
for kernel in &b.kernels { fn apply_kernel(&mut self, kernel: &TxKernel) -> Result<(), Error> {
if let Ok(pos) = self.get_kernel_pos(&kernel.excess) { if let Ok(pos) = self.get_kernel_pos(&kernel.excess) {
// same as outputs // same as outputs
if let Some(k) = self.kernel_pmmr.get(pos) { if let Some(k) = self.kernel_pmmr.get(pos) {
@ -317,33 +373,6 @@ impl<'a> Extension<'a> {
.push(NoSum(kernel.clone())) .push(NoSum(kernel.clone()))
.map_err(&Error::SumTreeErr)?; .map_err(&Error::SumTreeErr)?;
self.new_kernel_excesses.insert(kernel.excess, pos); self.new_kernel_excesses.insert(kernel.excess, pos);
}
Ok(())
}
fn save_pos_index(&self) -> Result<(), Error> {
debug!(
LOGGER,
"sumtree: save_pos_index: outputs: {}, {:?}",
self.new_output_commits.len(),
self.new_output_commits.values().collect::<Vec<_>>(),
);
for (commit, pos) in &self.new_output_commits {
self.commit_index.save_output_pos(commit, *pos)?;
}
debug!(
LOGGER,
"sumtree: save_pos_index: kernels: {}, {:?}",
self.new_kernel_excesses.len(),
self.new_kernel_excesses.values().collect::<Vec<_>>(),
);
for (excess, pos) in &self.new_kernel_excesses {
self.commit_index.save_kernel_pos(excess, *pos)?;
}
Ok(()) Ok(())
} }

View file

@ -17,6 +17,7 @@ extern crate grin_chain as chain;
extern crate grin_core as core; extern crate grin_core as core;
extern crate grin_keychain as keychain; extern crate grin_keychain as keychain;
extern crate grin_pow as pow; extern crate grin_pow as pow;
extern crate grin_util as util;
extern crate rand; extern crate rand;
extern crate time; extern crate time;
@ -25,7 +26,7 @@ use std::sync::Arc;
use chain::Chain; use chain::Chain;
use chain::types::*; use chain::types::*;
use core::core::{Block, BlockHeader}; use core::core::{Block, BlockHeader, Transaction, build};
use core::core::hash::Hashed; use core::core::hash::Hashed;
use core::core::target::Difficulty; use core::core::target::Difficulty;
use core::consensus; use core::consensus;
@ -79,7 +80,7 @@ fn mine_empty_chain() {
let difficulty = consensus::next_difficulty(chain.difficulty_iter()).unwrap(); let difficulty = consensus::next_difficulty(chain.difficulty_iter()).unwrap();
b.header.difficulty = difficulty.clone(); b.header.difficulty = difficulty.clone();
chain.set_sumtree_roots(&mut b).unwrap(); chain.set_sumtree_roots(&mut b, false).unwrap();
pow::pow_size( pow::pow_size(
&mut cuckoo_miner, &mut cuckoo_miner,
@ -123,10 +124,11 @@ fn mine_empty_chain() {
#[test] #[test]
fn mine_forks() { fn mine_forks() {
let chain = setup(".grin2"); let chain = setup(".grin2");
let kc = Keychain::from_random_seed().unwrap();
// add a first block to not fork genesis // add a first block to not fork genesis
let prev = chain.head_header().unwrap(); let prev = chain.head_header().unwrap();
let b = prepare_block(&prev, &chain, 2); let b = prepare_block(&kc, &prev, &chain, 2);
chain.process_block(b, chain::SKIP_POW).unwrap(); chain.process_block(b, chain::SKIP_POW).unwrap();
// mine and add a few blocks // mine and add a few blocks
@ -134,10 +136,10 @@ fn mine_forks() {
for n in 1..4 { for n in 1..4 {
// first block for one branch // first block for one branch
let prev = chain.head_header().unwrap(); let prev = chain.head_header().unwrap();
let b1 = prepare_block(&prev, &chain, 3 * n); let b1 = prepare_block(&kc, &prev, &chain, 3 * n);
// 2nd block with higher difficulty for other branch // 2nd block with higher difficulty for other branch
let b2 = prepare_block(&prev, &chain, 3 * n + 1); let b2 = prepare_block(&kc, &prev, &chain, 3 * n + 1);
// process the first block to extend the chain // process the first block to extend the chain
let bhash = b1.hash(); let bhash = b1.hash();
@ -163,24 +165,25 @@ fn mine_forks() {
#[test] #[test]
fn mine_losing_fork() { fn mine_losing_fork() {
let kc = Keychain::from_random_seed().unwrap();
let chain = setup(".grin3"); let chain = setup(".grin3");
// add a first block we'll be forking from // add a first block we'll be forking from
let prev = chain.head_header().unwrap(); let prev = chain.head_header().unwrap();
let b1 = prepare_block(&prev, &chain, 2); let b1 = prepare_block(&kc, &prev, &chain, 2);
let b1head = b1.header.clone(); let b1head = b1.header.clone();
chain.process_block(b1, chain::SKIP_POW).unwrap(); chain.process_block(b1, chain::SKIP_POW).unwrap();
// prepare the 2 successor, sibling blocks, one with lower diff // prepare the 2 successor, sibling blocks, one with lower diff
let b2 = prepare_block(&b1head, &chain, 4); let b2 = prepare_block(&kc, &b1head, &chain, 4);
let b2head = b2.header.clone(); let b2head = b2.header.clone();
let bfork = prepare_block(&b1head, &chain, 3); let bfork = prepare_block(&kc, &b1head, &chain, 3);
// add higher difficulty first, prepare its successor, then fork // add higher difficulty first, prepare its successor, then fork
// with lower diff // with lower diff
chain.process_block(b2, chain::SKIP_POW).unwrap(); chain.process_block(b2, chain::SKIP_POW).unwrap();
assert_eq!(chain.head_header().unwrap().hash(), b2head.hash()); assert_eq!(chain.head_header().unwrap().hash(), b2head.hash());
let b3 = prepare_block(&b2head, &chain, 5); let b3 = prepare_block(&kc, &b2head, &chain, 5);
chain.process_block(bfork, chain::SKIP_POW).unwrap(); chain.process_block(bfork, chain::SKIP_POW).unwrap();
// adding the successor // adding the successor
@ -191,6 +194,7 @@ fn mine_losing_fork() {
#[test] #[test]
fn longer_fork() { fn longer_fork() {
let kc = Keychain::from_random_seed().unwrap();
// to make it easier to compute the sumtree roots in the test, we // to make it easier to compute the sumtree roots in the test, we
// prepare 2 chains, the 2nd will be have the forked blocks we can // prepare 2 chains, the 2nd will be have the forked blocks we can
// then send back on the 1st // then send back on the 1st
@ -201,7 +205,7 @@ fn longer_fork() {
// for the forked chain // for the forked chain
let mut prev = chain.head_header().unwrap(); let mut prev = chain.head_header().unwrap();
for n in 0..10 { for n in 0..10 {
let b = prepare_block(&prev, &chain, n + 2); let b = prepare_block(&kc, &prev, &chain, 2*n + 2);
let bh = b.header.clone(); let bh = b.header.clone();
if n < 5 { if n < 5 {
@ -222,7 +226,7 @@ fn longer_fork() {
let mut prev_fork = head_fork.clone(); let mut prev_fork = head_fork.clone();
for n in 0..7 { for n in 0..7 {
let b_fork = prepare_block(&prev_fork, &chain_fork, n + 7); let b_fork = prepare_block(&kc, &prev_fork, &chain_fork, 2*n + 11);
let bh_fork = b_fork.header.clone(); let bh_fork = b_fork.header.clone();
let b = b_fork.clone(); let b = b_fork.clone();
@ -233,17 +237,105 @@ fn longer_fork() {
} }
} }
fn prepare_block(prev: &BlockHeader, chain: &Chain, diff: u64) -> Block { #[test]
let mut b = prepare_block_nosum(prev, diff); fn spend_in_fork() {
chain.set_sumtree_roots(&mut b).unwrap(); util::init_test_logger();
let chain = setup(".grin6");
let prev = chain.head_header().unwrap();
let kc = Keychain::from_random_seed().unwrap();
// mine 4 blocks, the 4th will be the root of the fork
let mut fork_head = prev;
for n in 2..6 {
let b = prepare_block(&kc, &fork_head, &chain, n);
fork_head = b.header.clone();
chain.process_block(b, chain::SKIP_POW).unwrap();
}
let (tx1, _) = build::transaction(
vec![
build::input(consensus::REWARD, kc.derive_key_id(2).unwrap()),
build::output(consensus::REWARD - 20000, kc.derive_key_id(30).unwrap()),
build::with_fee(20000),
],
&kc,
).unwrap();
let next = prepare_block_tx(&kc, &fork_head, &chain, 7, vec![&tx1]);
let prev_main = next.header.clone();
chain.process_block(next, chain::SKIP_POW).unwrap();
let (tx2, _) = build::transaction(
vec![
build::input(consensus::REWARD - 20000, kc.derive_key_id(30).unwrap()),
build::output(consensus::REWARD - 40000, kc.derive_key_id(31).unwrap()),
build::with_fee(20000),
],
&kc,
).unwrap();
let next = prepare_block_tx(&kc, &prev_main, &chain, 9, vec![&tx2]);
let prev_main = next.header.clone();
chain.process_block(next, chain::SKIP_POW).unwrap();
// mine 2 forked blocks from the first
let fork = prepare_fork_block_tx(&kc, &fork_head, &chain, 6, vec![&tx1]);
let prev_fork = fork.header.clone();
chain.process_block(fork, chain::SKIP_POW).unwrap();
let fork_next = prepare_fork_block_tx(&kc, &prev_fork, &chain, 8, vec![&tx2]);
let prev_fork = fork_next.header.clone();
chain.process_block(fork_next, chain::SKIP_POW).unwrap();
// check state
let head = chain.head_header().unwrap();
assert_eq!(head.height, 6);
assert_eq!(head.hash(), prev_main.hash());
assert!(chain.is_unspent(&tx2.outputs[0].commitment()).unwrap());
let res = chain.is_unspent(&tx1.outputs[0].commitment());
assert!(!res.unwrap());
// make the fork win
let fork_next = prepare_fork_block(&kc, &prev_fork, &chain, 10);
let prev_fork = fork_next.header.clone();
chain.process_block(fork_next, chain::SKIP_POW).unwrap();
// check state
let head = chain.head_header().unwrap();
assert_eq!(head.height, 7);
assert_eq!(head.hash(), prev_fork.hash());
assert!(chain.is_unspent(&tx2.outputs[0].commitment()).unwrap());
assert!(!chain.is_unspent(&tx1.outputs[0].commitment()).unwrap());
}
fn prepare_block(kc: &Keychain, prev: &BlockHeader, chain: &Chain, diff: u64) -> Block {
let mut b = prepare_block_nosum(kc, prev, diff, vec![]);
chain.set_sumtree_roots(&mut b, false).unwrap();
b b
} }
fn prepare_block_nosum(prev: &BlockHeader, diff: u64) -> Block { fn prepare_block_tx(kc: &Keychain, prev: &BlockHeader, chain: &Chain, diff: u64, txs: Vec<&Transaction>) -> Block {
let keychain = Keychain::from_random_seed().unwrap(); let mut b = prepare_block_nosum(kc, prev, diff, txs);
let key_id = keychain.derive_key_id(1).unwrap(); chain.set_sumtree_roots(&mut b, false).unwrap();
b
}
let mut b = core::core::Block::new(prev, vec![], &keychain, &key_id).unwrap(); fn prepare_fork_block(kc: &Keychain, prev: &BlockHeader, chain: &Chain, diff: u64) -> Block {
let mut b = prepare_block_nosum(kc, prev, diff, vec![]);
chain.set_sumtree_roots(&mut b, true).unwrap();
b
}
fn prepare_fork_block_tx(kc: &Keychain, prev: &BlockHeader, chain: &Chain, diff: u64, txs: Vec<&Transaction>) -> Block {
let mut b = prepare_block_nosum(kc, prev, diff, txs);
chain.set_sumtree_roots(&mut b, true).unwrap();
b
}
fn prepare_block_nosum(kc: &Keychain, prev: &BlockHeader, diff: u64, txs: Vec<&Transaction>) -> Block {
let key_id = kc.derive_key_id(diff as u32).unwrap();
let mut b = core::core::Block::new(prev, txs, kc, &key_id).unwrap();
b.header.timestamp = prev.timestamp + time::Duration::seconds(60); b.header.timestamp = prev.timestamp + time::Duration::seconds(60);
b.header.total_difficulty = Difficulty::from_num(diff); b.header.total_difficulty = Difficulty::from_num(diff);
b b

View file

@ -79,7 +79,7 @@ fn test_coinbase_maturity() {
let difficulty = consensus::next_difficulty(chain.difficulty_iter()).unwrap(); let difficulty = consensus::next_difficulty(chain.difficulty_iter()).unwrap();
block.header.difficulty = difficulty.clone(); block.header.difficulty = difficulty.clone();
chain.set_sumtree_roots(&mut block).unwrap(); chain.set_sumtree_roots(&mut block, false).unwrap();
pow::pow_size( pow::pow_size(
&mut cuckoo_miner, &mut cuckoo_miner,
@ -115,7 +115,7 @@ fn test_coinbase_maturity() {
let difficulty = consensus::next_difficulty(chain.difficulty_iter()).unwrap(); let difficulty = consensus::next_difficulty(chain.difficulty_iter()).unwrap();
block.header.difficulty = difficulty.clone(); block.header.difficulty = difficulty.clone();
chain.set_sumtree_roots(&mut block).unwrap(); chain.set_sumtree_roots(&mut block, false).unwrap();
pow::pow_size( pow::pow_size(
&mut cuckoo_miner, &mut cuckoo_miner,
@ -143,7 +143,7 @@ fn test_coinbase_maturity() {
let difficulty = consensus::next_difficulty(chain.difficulty_iter()).unwrap(); let difficulty = consensus::next_difficulty(chain.difficulty_iter()).unwrap();
block.header.difficulty = difficulty.clone(); block.header.difficulty = difficulty.clone();
chain.set_sumtree_roots(&mut block).unwrap(); chain.set_sumtree_roots(&mut block, false).unwrap();
pow::pow_size( pow::pow_size(
&mut cuckoo_miner, &mut cuckoo_miner,
@ -164,7 +164,7 @@ fn test_coinbase_maturity() {
let difficulty = consensus::next_difficulty(chain.difficulty_iter()).unwrap(); let difficulty = consensus::next_difficulty(chain.difficulty_iter()).unwrap();
block.header.difficulty = difficulty.clone(); block.header.difficulty = difficulty.clone();
chain.set_sumtree_roots(&mut block).unwrap(); chain.set_sumtree_roots(&mut block, false).unwrap();
pow::pow_size( pow::pow_size(
&mut cuckoo_miner, &mut cuckoo_miner,

View file

@ -591,7 +591,7 @@ impl Miner {
b.header.timestamp = time::at_utc(time::Timespec::new(now_sec, 0)); b.header.timestamp = time::at_utc(time::Timespec::new(now_sec, 0));
trace!(LOGGER, "Block: {:?}", b); trace!(LOGGER, "Block: {:?}", b);
let roots_result = self.chain.set_sumtree_roots(&mut b); let roots_result = self.chain.set_sumtree_roots(&mut b, false);
match roots_result { match roots_result {
Ok(_) => Ok((b, block_fees)), Ok(_) => Ok((b, block_fees)),