2018-03-02 23:47:27 +03:00
|
|
|
// Copyright 2018 The Grin Developers
|
2017-09-28 02:46:32 +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.
|
|
|
|
|
2018-03-05 22:33:44 +03:00
|
|
|
//! Utility structs to handle the 3 hashtrees (output, range proof, kernel) more
|
2017-09-28 02:46:32 +03:00
|
|
|
//! conveniently and transactionally.
|
|
|
|
|
|
|
|
use std::collections::HashMap;
|
2018-05-28 21:22:22 +03:00
|
|
|
use std::fs;
|
2018-02-10 01:32:16 +03:00
|
|
|
use std::fs::File;
|
|
|
|
use std::path::{Path, PathBuf};
|
2017-09-28 02:46:32 +03:00
|
|
|
use std::sync::Arc;
|
2018-03-21 03:34:19 +03:00
|
|
|
use std::time::Instant;
|
2017-09-28 02:46:32 +03:00
|
|
|
|
2018-06-18 18:18:38 +03:00
|
|
|
use croaring::Bitmap;
|
|
|
|
|
2018-03-04 03:19:54 +03:00
|
|
|
use util::secp::pedersen::{Commitment, RangeProof};
|
2018-02-10 01:32:16 +03:00
|
|
|
|
2018-06-02 21:00:44 +03:00
|
|
|
use core::core::committed::Committed;
|
2018-05-28 21:22:22 +03:00
|
|
|
use core::core::hash::{Hash, Hashed};
|
2018-06-20 22:18:52 +03:00
|
|
|
use core::core::merkle_proof::MerkleProof;
|
|
|
|
use core::core::pmmr;
|
|
|
|
use core::core::pmmr::PMMR;
|
2018-06-02 21:00:44 +03:00
|
|
|
use core::core::{Block, BlockHeader, Input, Output, OutputFeatures, OutputIdentifier, Transaction,
|
|
|
|
TxKernel};
|
2018-03-06 20:58:33 +03:00
|
|
|
use core::global;
|
2018-03-21 15:28:05 +03:00
|
|
|
use core::ser::{PMMRIndexHashable, PMMRable};
|
2018-02-22 16:45:13 +03:00
|
|
|
|
2017-09-28 02:46:32 +03:00
|
|
|
use grin_store;
|
2018-04-24 22:53:01 +03:00
|
|
|
use grin_store::pmmr::PMMRBackend;
|
2018-03-06 20:58:33 +03:00
|
|
|
use grin_store::types::prune_noop;
|
2018-05-07 16:21:41 +03:00
|
|
|
use types::{BlockMarker, BlockSums, ChainStore, Error, TxHashSetRoots};
|
|
|
|
use util::{secp_static, zip, LOGGER};
|
2017-09-28 02:46:32 +03:00
|
|
|
|
2018-03-05 22:33:44 +03:00
|
|
|
const TXHASHSET_SUBDIR: &'static str = "txhashset";
|
|
|
|
const OUTPUT_SUBDIR: &'static str = "output";
|
2017-09-28 02:46:32 +03:00
|
|
|
const RANGE_PROOF_SUBDIR: &'static str = "rangeproof";
|
|
|
|
const KERNEL_SUBDIR: &'static str = "kernel";
|
2018-03-05 22:33:44 +03:00
|
|
|
const TXHASHSET_ZIP: &'static str = "txhashset_snapshot.zip";
|
2017-09-28 02:46:32 +03:00
|
|
|
|
2017-09-29 21:44:25 +03:00
|
|
|
struct PMMRHandle<T>
|
2017-10-17 00:23:10 +03:00
|
|
|
where
|
2018-02-22 16:45:13 +03:00
|
|
|
T: PMMRable,
|
2017-09-29 21:44:25 +03:00
|
|
|
{
|
2017-09-28 02:46:32 +03:00
|
|
|
backend: PMMRBackend<T>,
|
|
|
|
last_pos: u64,
|
|
|
|
}
|
|
|
|
|
2017-09-29 21:44:25 +03:00
|
|
|
impl<T> PMMRHandle<T>
|
2017-10-17 00:23:10 +03:00
|
|
|
where
|
2018-03-13 21:22:34 +03:00
|
|
|
T: PMMRable + ::std::fmt::Debug,
|
2017-09-29 21:44:25 +03:00
|
|
|
{
|
2018-06-18 18:18:38 +03:00
|
|
|
fn new(
|
|
|
|
root_dir: String,
|
|
|
|
file_name: &str,
|
|
|
|
header: Option<&BlockHeader>,
|
|
|
|
) -> Result<PMMRHandle<T>, Error> {
|
2018-03-05 22:33:44 +03:00
|
|
|
let path = Path::new(&root_dir).join(TXHASHSET_SUBDIR).join(file_name);
|
2017-09-28 02:46:32 +03:00
|
|
|
fs::create_dir_all(path.clone())?;
|
2018-06-18 18:18:38 +03:00
|
|
|
let be = PMMRBackend::new(path.to_str().unwrap().to_string(), header)?;
|
2017-09-28 02:46:32 +03:00
|
|
|
let sz = be.unpruned_size()?;
|
|
|
|
Ok(PMMRHandle {
|
|
|
|
backend: be,
|
|
|
|
last_pos: sz,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// An easy to manipulate structure holding the 3 sum trees necessary to
|
2018-03-05 22:33:44 +03:00
|
|
|
/// validate blocks and capturing the Output set, the range proofs and the
|
2017-09-28 02:46:32 +03:00
|
|
|
/// kernels. Also handles the index of Commitments to positions in the
|
2018-02-22 16:45:13 +03:00
|
|
|
/// output and range proof pmmr trees.
|
2017-09-28 02:46:32 +03:00
|
|
|
///
|
|
|
|
/// Note that the index is never authoritative, only the trees are
|
|
|
|
/// guaranteed to indicate whether an output is spent or not. The index
|
|
|
|
/// may have commitments that have already been spent, even with
|
|
|
|
/// pruning enabled.
|
2018-02-22 16:45:13 +03:00
|
|
|
|
2018-03-05 22:33:44 +03:00
|
|
|
pub struct TxHashSet {
|
2018-03-22 03:10:11 +03:00
|
|
|
output_pmmr_h: PMMRHandle<OutputIdentifier>,
|
2018-02-22 16:45:13 +03:00
|
|
|
rproof_pmmr_h: PMMRHandle<RangeProof>,
|
|
|
|
kernel_pmmr_h: PMMRHandle<TxKernel>,
|
2017-09-28 02:46:32 +03:00
|
|
|
|
|
|
|
// chain store used as index of commitments to MMR positions
|
|
|
|
commit_index: Arc<ChainStore>,
|
|
|
|
}
|
|
|
|
|
2018-03-05 22:33:44 +03:00
|
|
|
impl TxHashSet {
|
|
|
|
/// Open an existing or new set of backends for the TxHashSet
|
2018-06-18 18:18:38 +03:00
|
|
|
pub fn open(
|
|
|
|
root_dir: String,
|
|
|
|
commit_index: Arc<ChainStore>,
|
|
|
|
header: Option<&BlockHeader>,
|
|
|
|
) -> Result<TxHashSet, Error> {
|
2018-03-06 20:58:33 +03:00
|
|
|
let output_file_path: PathBuf = [&root_dir, TXHASHSET_SUBDIR, OUTPUT_SUBDIR]
|
|
|
|
.iter()
|
|
|
|
.collect();
|
2018-03-05 22:33:44 +03:00
|
|
|
fs::create_dir_all(output_file_path.clone())?;
|
2018-02-22 16:45:13 +03:00
|
|
|
|
2018-03-05 22:33:44 +03:00
|
|
|
let rproof_file_path: PathBuf = [&root_dir, TXHASHSET_SUBDIR, RANGE_PROOF_SUBDIR]
|
2018-03-04 03:19:54 +03:00
|
|
|
.iter()
|
|
|
|
.collect();
|
2018-02-22 16:45:13 +03:00
|
|
|
fs::create_dir_all(rproof_file_path.clone())?;
|
|
|
|
|
2018-03-06 20:58:33 +03:00
|
|
|
let kernel_file_path: PathBuf = [&root_dir, TXHASHSET_SUBDIR, KERNEL_SUBDIR]
|
|
|
|
.iter()
|
|
|
|
.collect();
|
2018-02-10 01:32:16 +03:00
|
|
|
fs::create_dir_all(kernel_file_path.clone())?;
|
|
|
|
|
2018-03-05 22:33:44 +03:00
|
|
|
Ok(TxHashSet {
|
2018-06-18 18:18:38 +03:00
|
|
|
output_pmmr_h: PMMRHandle::new(root_dir.clone(), OUTPUT_SUBDIR, header)?,
|
|
|
|
rproof_pmmr_h: PMMRHandle::new(root_dir.clone(), RANGE_PROOF_SUBDIR, header)?,
|
|
|
|
kernel_pmmr_h: PMMRHandle::new(root_dir.clone(), KERNEL_SUBDIR, None)?,
|
2018-04-24 22:53:01 +03:00
|
|
|
commit_index,
|
2017-09-28 02:46:32 +03:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2018-03-02 23:47:27 +03:00
|
|
|
/// Check if an output is unspent.
|
2018-01-17 06:03:40 +03:00
|
|
|
/// We look in the index to find the output MMR pos.
|
|
|
|
/// Then we check the entry in the output MMR and confirm the hash matches.
|
2018-03-02 23:47:27 +03:00
|
|
|
pub fn is_unspent(&mut self, output_id: &OutputIdentifier) -> Result<Hash, Error> {
|
2018-02-22 16:45:13 +03:00
|
|
|
match self.commit_index.get_output_pos(&output_id.commit) {
|
2017-11-22 23:14:42 +03:00
|
|
|
Ok(pos) => {
|
2018-03-22 03:10:11 +03:00
|
|
|
let output_pmmr: PMMR<OutputIdentifier, _> =
|
2018-03-05 22:33:44 +03:00
|
|
|
PMMR::at(&mut self.output_pmmr_h.backend, self.output_pmmr_h.last_pos);
|
2018-03-24 02:33:59 +03:00
|
|
|
if let Some(hash) = output_pmmr.get_hash(pos) {
|
2018-03-30 09:02:40 +03:00
|
|
|
if hash == output_id.hash_with_index(pos - 1) {
|
2018-03-02 23:47:27 +03:00
|
|
|
Ok(hash)
|
2018-01-17 06:03:40 +03:00
|
|
|
} else {
|
2018-03-05 22:33:44 +03:00
|
|
|
Err(Error::TxHashSetErr(format!("txhashset hash mismatch")))
|
2018-01-17 06:03:40 +03:00
|
|
|
}
|
2017-11-22 23:14:42 +03:00
|
|
|
} else {
|
2018-01-17 06:03:40 +03:00
|
|
|
Err(Error::OutputNotFound)
|
2017-11-22 23:14:42 +03:00
|
|
|
}
|
|
|
|
}
|
2018-01-17 06:03:40 +03:00
|
|
|
Err(grin_store::Error::NotFoundErr) => Err(Error::OutputNotFound),
|
2018-03-05 22:33:44 +03:00
|
|
|
Err(e) => Err(Error::StoreErr(e, format!("txhashset unspent check"))),
|
2017-09-28 02:46:32 +03:00
|
|
|
}
|
|
|
|
}
|
2017-10-24 21:11:58 +03:00
|
|
|
|
2017-10-28 00:57:04 +03:00
|
|
|
/// returns the last N nodes inserted into the tree (i.e. the 'bottom'
|
|
|
|
/// nodes at level 0
|
2018-05-28 21:22:22 +03:00
|
|
|
/// TODO: These need to return the actual data from the flat-files instead
|
|
|
|
/// of hashes now
|
2018-03-24 02:33:59 +03:00
|
|
|
pub fn last_n_output(&mut self, distance: u64) -> Vec<(Hash, OutputIdentifier)> {
|
2018-03-22 03:10:11 +03:00
|
|
|
let output_pmmr: PMMR<OutputIdentifier, _> =
|
2018-03-05 22:33:44 +03:00
|
|
|
PMMR::at(&mut self.output_pmmr_h.backend, self.output_pmmr_h.last_pos);
|
|
|
|
output_pmmr.get_last_n_insertions(distance)
|
2017-10-28 00:57:04 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/// as above, for range proofs
|
2018-03-24 02:33:59 +03:00
|
|
|
pub fn last_n_rangeproof(&mut self, distance: u64) -> Vec<(Hash, RangeProof)> {
|
2018-03-04 03:19:54 +03:00
|
|
|
let rproof_pmmr: PMMR<RangeProof, _> =
|
|
|
|
PMMR::at(&mut self.rproof_pmmr_h.backend, self.rproof_pmmr_h.last_pos);
|
2017-10-28 00:57:04 +03:00
|
|
|
rproof_pmmr.get_last_n_insertions(distance)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// as above, for kernels
|
2018-03-24 02:33:59 +03:00
|
|
|
pub fn last_n_kernel(&mut self, distance: u64) -> Vec<(Hash, TxKernel)> {
|
2018-03-04 03:19:54 +03:00
|
|
|
let kernel_pmmr: PMMR<TxKernel, _> =
|
|
|
|
PMMR::at(&mut self.kernel_pmmr_h.backend, self.kernel_pmmr_h.last_pos);
|
2017-10-28 00:57:04 +03:00
|
|
|
kernel_pmmr.get_last_n_insertions(distance)
|
|
|
|
}
|
|
|
|
|
2018-05-28 21:22:22 +03:00
|
|
|
/// returns outputs from the given insertion (leaf) index up to the
|
|
|
|
/// specified limit. Also returns the last index actually populated
|
2018-04-11 12:02:07 +03:00
|
|
|
pub fn outputs_by_insertion_index(
|
|
|
|
&mut self,
|
|
|
|
start_index: u64,
|
|
|
|
max_count: u64,
|
|
|
|
) -> (u64, Vec<OutputIdentifier>) {
|
|
|
|
let output_pmmr: PMMR<OutputIdentifier, _> =
|
|
|
|
PMMR::at(&mut self.output_pmmr_h.backend, self.output_pmmr_h.last_pos);
|
|
|
|
output_pmmr.elements_from_insertion_index(start_index, max_count)
|
|
|
|
}
|
|
|
|
|
2018-06-13 19:03:34 +03:00
|
|
|
/// highest output insertion index available
|
2018-04-11 12:02:07 +03:00
|
|
|
pub fn highest_output_insertion_index(&mut self) -> u64 {
|
|
|
|
pmmr::n_leaves(self.output_pmmr_h.last_pos)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// As above, for rangeproofs
|
|
|
|
pub fn rangeproofs_by_insertion_index(
|
|
|
|
&mut self,
|
|
|
|
start_index: u64,
|
|
|
|
max_count: u64,
|
|
|
|
) -> (u64, Vec<RangeProof>) {
|
|
|
|
let rproof_pmmr: PMMR<RangeProof, _> =
|
|
|
|
PMMR::at(&mut self.rproof_pmmr_h.backend, self.rproof_pmmr_h.last_pos);
|
|
|
|
rproof_pmmr.elements_from_insertion_index(start_index, max_count)
|
|
|
|
}
|
|
|
|
|
2018-02-10 01:32:16 +03:00
|
|
|
/// Output and kernel MMR indexes at the end of the provided block
|
2018-04-24 22:53:01 +03:00
|
|
|
pub fn indexes_at(&self, bh: &Hash) -> Result<BlockMarker, Error> {
|
2018-03-09 00:36:51 +03:00
|
|
|
self.commit_index.get_block_marker(bh).map_err(&From::from)
|
2018-02-10 01:32:16 +03:00
|
|
|
}
|
|
|
|
|
2017-10-24 21:11:58 +03:00
|
|
|
/// Get sum tree roots
|
2018-02-22 16:45:13 +03:00
|
|
|
/// TODO: Return data instead of hashes
|
2018-03-04 03:19:54 +03:00
|
|
|
pub fn roots(&mut self) -> (Hash, Hash, Hash) {
|
2018-03-22 03:10:11 +03:00
|
|
|
let output_pmmr: PMMR<OutputIdentifier, _> =
|
2018-03-05 22:33:44 +03:00
|
|
|
PMMR::at(&mut self.output_pmmr_h.backend, self.output_pmmr_h.last_pos);
|
2018-03-04 03:19:54 +03:00
|
|
|
let rproof_pmmr: PMMR<RangeProof, _> =
|
|
|
|
PMMR::at(&mut self.rproof_pmmr_h.backend, self.rproof_pmmr_h.last_pos);
|
|
|
|
let kernel_pmmr: PMMR<TxKernel, _> =
|
|
|
|
PMMR::at(&mut self.kernel_pmmr_h.backend, self.kernel_pmmr_h.last_pos);
|
2017-10-28 00:57:04 +03:00
|
|
|
(output_pmmr.root(), rproof_pmmr.root(), kernel_pmmr.root())
|
2017-10-24 21:11:58 +03:00
|
|
|
}
|
2018-03-06 20:58:33 +03:00
|
|
|
|
2018-04-26 16:01:01 +03:00
|
|
|
/// build a new merkle proof for the given position
|
|
|
|
pub fn merkle_proof(&mut self, commit: Commitment) -> Result<MerkleProof, String> {
|
|
|
|
let pos = self.commit_index.get_output_pos(&commit).unwrap();
|
|
|
|
let output_pmmr: PMMR<OutputIdentifier, _> =
|
|
|
|
PMMR::at(&mut self.output_pmmr_h.backend, self.output_pmmr_h.last_pos);
|
|
|
|
output_pmmr.merkle_proof(pos)
|
|
|
|
}
|
|
|
|
|
2018-03-06 20:58:33 +03:00
|
|
|
/// Compact the MMR data files and flush the rm logs
|
|
|
|
pub fn compact(&mut self) -> Result<(), Error> {
|
|
|
|
let commit_index = self.commit_index.clone();
|
2018-06-18 18:18:38 +03:00
|
|
|
let head_header = commit_index.head_header()?;
|
|
|
|
let current_height = head_header.height;
|
2018-03-13 21:22:34 +03:00
|
|
|
|
|
|
|
// horizon for compacting is based on current_height
|
2018-06-18 18:18:38 +03:00
|
|
|
let horizon = current_height.saturating_sub(global::cut_through_horizon().into());
|
|
|
|
let horizon_header = self.commit_index.get_header_by_height(horizon)?;
|
|
|
|
let horizon_marker = self.commit_index.get_block_marker(&horizon_header.hash())?;
|
|
|
|
|
|
|
|
let rewind_add_pos =
|
|
|
|
output_pos_to_rewind(self.commit_index.clone(), &horizon_header, &head_header)?;
|
|
|
|
let rewind_rm_pos =
|
|
|
|
input_pos_to_rewind(self.commit_index.clone(), &horizon_header, &head_header)?;
|
2018-03-13 21:22:34 +03:00
|
|
|
|
2018-03-06 20:58:33 +03:00
|
|
|
let clean_output_index = |commit: &[u8]| {
|
2018-03-13 21:22:34 +03:00
|
|
|
// do we care if this fails?
|
2018-03-06 20:58:33 +03:00
|
|
|
let _ = commit_index.delete_output_pos(commit);
|
|
|
|
};
|
2018-03-13 21:22:34 +03:00
|
|
|
|
2018-06-18 18:18:38 +03:00
|
|
|
self.output_pmmr_h.backend.check_compact(
|
|
|
|
horizon_marker.output_pos,
|
|
|
|
&rewind_add_pos,
|
|
|
|
&rewind_rm_pos,
|
|
|
|
clean_output_index,
|
|
|
|
)?;
|
2018-03-13 21:22:34 +03:00
|
|
|
|
2018-06-18 18:18:38 +03:00
|
|
|
self.rproof_pmmr_h.backend.check_compact(
|
|
|
|
horizon_marker.output_pos,
|
|
|
|
&rewind_add_pos,
|
|
|
|
&rewind_rm_pos,
|
|
|
|
&prune_noop,
|
|
|
|
)?;
|
2018-03-13 21:22:34 +03:00
|
|
|
|
2018-03-06 20:58:33 +03:00
|
|
|
Ok(())
|
|
|
|
}
|
2017-09-28 02:46:32 +03:00
|
|
|
}
|
|
|
|
|
2018-05-28 21:22:22 +03:00
|
|
|
/// Starts a new unit of work to extend (or rewind) the chain with additional
|
|
|
|
/// blocks. Accepts a closure that will operate within that unit of work.
|
2018-04-09 19:37:46 +03:00
|
|
|
/// The closure has access to an Extension object that allows the addition
|
|
|
|
/// of blocks to the txhashset and the checking of the current tree roots.
|
|
|
|
///
|
|
|
|
/// The unit of work is always discarded (always rollback) as this is read-only.
|
|
|
|
pub fn extending_readonly<'a, F, T>(trees: &'a mut TxHashSet, inner: F) -> Result<T, Error>
|
|
|
|
where
|
|
|
|
F: FnOnce(&mut Extension) -> Result<T, Error>,
|
|
|
|
{
|
|
|
|
let sizes: (u64, u64, u64);
|
|
|
|
let res: Result<T, Error>;
|
|
|
|
{
|
|
|
|
let commit_index = trees.commit_index.clone();
|
|
|
|
|
|
|
|
trace!(LOGGER, "Starting new txhashset (readonly) extension.");
|
|
|
|
let mut extension = Extension::new(trees, commit_index);
|
2018-05-30 23:57:13 +03:00
|
|
|
extension.force_rollback();
|
2018-04-09 19:37:46 +03:00
|
|
|
res = inner(&mut extension);
|
|
|
|
|
|
|
|
sizes = extension.sizes();
|
|
|
|
}
|
|
|
|
|
2018-05-30 23:57:13 +03:00
|
|
|
trace!(
|
2018-04-09 19:37:46 +03:00
|
|
|
LOGGER,
|
2018-05-30 23:57:13 +03:00
|
|
|
"Rollbacking txhashset (readonly) extension. sizes {:?}",
|
|
|
|
sizes
|
2018-04-09 19:37:46 +03:00
|
|
|
);
|
|
|
|
|
|
|
|
trees.output_pmmr_h.backend.discard();
|
|
|
|
trees.rproof_pmmr_h.backend.discard();
|
|
|
|
trees.kernel_pmmr_h.backend.discard();
|
|
|
|
|
|
|
|
trace!(LOGGER, "TxHashSet (readonly) extension done.");
|
|
|
|
|
|
|
|
res
|
|
|
|
}
|
|
|
|
|
2017-09-28 02:46:32 +03:00
|
|
|
/// Starts a new unit of work to extend the chain with additional blocks,
|
|
|
|
/// accepting a closure that will work within that unit of work. The closure
|
|
|
|
/// has access to an Extension object that allows the addition of blocks to
|
2018-03-05 22:33:44 +03:00
|
|
|
/// the txhashset and the checking of the current tree roots.
|
2017-09-28 02:46:32 +03:00
|
|
|
///
|
|
|
|
/// If the closure returns an error, modifications are canceled and the unit
|
|
|
|
/// of work is abandoned. Otherwise, the unit of work is permanently applied.
|
2018-03-05 22:33:44 +03:00
|
|
|
pub fn extending<'a, F, T>(trees: &'a mut TxHashSet, inner: F) -> Result<T, Error>
|
2017-10-17 00:23:10 +03:00
|
|
|
where
|
|
|
|
F: FnOnce(&mut Extension) -> Result<T, Error>,
|
2017-09-29 21:44:25 +03:00
|
|
|
{
|
2017-09-28 02:46:32 +03:00
|
|
|
let sizes: (u64, u64, u64);
|
|
|
|
let res: Result<T, Error>;
|
|
|
|
let rollback: bool;
|
2018-04-24 22:53:01 +03:00
|
|
|
|
2017-09-28 02:46:32 +03:00
|
|
|
{
|
|
|
|
let commit_index = trees.commit_index.clone();
|
2018-02-10 01:32:16 +03:00
|
|
|
|
2018-03-25 19:41:12 +03:00
|
|
|
trace!(LOGGER, "Starting new txhashset extension.");
|
2017-09-28 02:46:32 +03:00
|
|
|
let mut extension = Extension::new(trees, commit_index);
|
|
|
|
res = inner(&mut extension);
|
2018-02-10 01:32:16 +03:00
|
|
|
|
2017-09-28 02:46:32 +03:00
|
|
|
rollback = extension.rollback;
|
|
|
|
if res.is_ok() && !rollback {
|
2018-03-09 00:36:51 +03:00
|
|
|
extension.save_indexes()?;
|
2017-09-28 02:46:32 +03:00
|
|
|
}
|
|
|
|
sizes = extension.sizes();
|
|
|
|
}
|
2018-04-24 22:53:01 +03:00
|
|
|
|
2017-09-28 02:46:32 +03:00
|
|
|
match res {
|
|
|
|
Err(e) => {
|
2018-03-05 22:33:44 +03:00
|
|
|
debug!(LOGGER, "Error returned, discarding txhashset extension.");
|
|
|
|
trees.output_pmmr_h.backend.discard();
|
2017-09-28 02:46:32 +03:00
|
|
|
trees.rproof_pmmr_h.backend.discard();
|
|
|
|
trees.kernel_pmmr_h.backend.discard();
|
|
|
|
Err(e)
|
|
|
|
}
|
|
|
|
Ok(r) => {
|
|
|
|
if rollback {
|
2018-05-30 23:57:13 +03:00
|
|
|
trace!(LOGGER, "Rollbacking txhashset extension. sizes {:?}", sizes);
|
2018-03-05 22:33:44 +03:00
|
|
|
trees.output_pmmr_h.backend.discard();
|
2017-09-28 02:46:32 +03:00
|
|
|
trees.rproof_pmmr_h.backend.discard();
|
|
|
|
trees.kernel_pmmr_h.backend.discard();
|
|
|
|
} else {
|
2018-05-30 23:57:13 +03:00
|
|
|
trace!(LOGGER, "Committing txhashset extension. sizes {:?}", sizes);
|
2018-03-05 22:33:44 +03:00
|
|
|
trees.output_pmmr_h.backend.sync()?;
|
2017-09-28 02:46:32 +03:00
|
|
|
trees.rproof_pmmr_h.backend.sync()?;
|
|
|
|
trees.kernel_pmmr_h.backend.sync()?;
|
2018-03-05 22:33:44 +03:00
|
|
|
trees.output_pmmr_h.last_pos = sizes.0;
|
2017-09-28 02:46:32 +03:00
|
|
|
trees.rproof_pmmr_h.last_pos = sizes.1;
|
|
|
|
trees.kernel_pmmr_h.last_pos = sizes.2;
|
|
|
|
}
|
|
|
|
|
2018-03-25 19:41:12 +03:00
|
|
|
trace!(LOGGER, "TxHashSet extension done.");
|
2017-09-28 02:46:32 +03:00
|
|
|
Ok(r)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Allows the application of new blocks on top of the sum trees in a
|
|
|
|
/// reversible manner within a unit of work provided by the `extending`
|
|
|
|
/// function.
|
|
|
|
pub struct Extension<'a> {
|
2018-03-22 03:10:11 +03:00
|
|
|
output_pmmr: PMMR<'a, OutputIdentifier, PMMRBackend<OutputIdentifier>>,
|
2018-02-22 16:45:13 +03:00
|
|
|
rproof_pmmr: PMMR<'a, RangeProof, PMMRBackend<RangeProof>>,
|
|
|
|
kernel_pmmr: PMMR<'a, TxKernel, PMMRBackend<TxKernel>>,
|
2017-09-28 02:46:32 +03:00
|
|
|
|
|
|
|
commit_index: Arc<ChainStore>,
|
|
|
|
new_output_commits: HashMap<Commitment, u64>,
|
2018-04-24 22:53:01 +03:00
|
|
|
new_block_markers: HashMap<Hash, BlockMarker>,
|
2017-09-29 21:44:25 +03:00
|
|
|
rollback: bool,
|
2017-09-28 02:46:32 +03:00
|
|
|
}
|
|
|
|
|
2018-05-08 17:23:33 +03:00
|
|
|
impl<'a> Committed for Extension<'a> {
|
|
|
|
fn inputs_committed(&self) -> Vec<Commitment> {
|
|
|
|
vec![]
|
|
|
|
}
|
|
|
|
|
|
|
|
fn outputs_committed(&self) -> Vec<Commitment> {
|
|
|
|
let mut commitments = vec![];
|
|
|
|
for n in 1..self.output_pmmr.unpruned_size() + 1 {
|
|
|
|
if pmmr::is_leaf(n) {
|
|
|
|
if let Some(out) = self.output_pmmr.get_data(n) {
|
|
|
|
commitments.push(out.commit);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
commitments
|
|
|
|
}
|
|
|
|
|
|
|
|
fn kernels_committed(&self) -> Vec<Commitment> {
|
|
|
|
let mut commitments = vec![];
|
|
|
|
for n in 1..self.kernel_pmmr.unpruned_size() + 1 {
|
|
|
|
if pmmr::is_leaf(n) {
|
|
|
|
if let Some(kernel) = self.kernel_pmmr.get_data(n) {
|
|
|
|
commitments.push(kernel.excess);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
commitments
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-28 02:46:32 +03:00
|
|
|
impl<'a> Extension<'a> {
|
|
|
|
// constructor
|
2018-03-05 22:33:44 +03:00
|
|
|
fn new(trees: &'a mut TxHashSet, commit_index: Arc<ChainStore>) -> Extension<'a> {
|
2017-09-28 02:46:32 +03:00
|
|
|
Extension {
|
2018-03-06 20:58:33 +03:00
|
|
|
output_pmmr: PMMR::at(
|
|
|
|
&mut trees.output_pmmr_h.backend,
|
|
|
|
trees.output_pmmr_h.last_pos,
|
|
|
|
),
|
2017-09-29 21:44:25 +03:00
|
|
|
rproof_pmmr: PMMR::at(
|
|
|
|
&mut trees.rproof_pmmr_h.backend,
|
|
|
|
trees.rproof_pmmr_h.last_pos,
|
|
|
|
),
|
|
|
|
kernel_pmmr: PMMR::at(
|
|
|
|
&mut trees.kernel_pmmr_h.backend,
|
|
|
|
trees.kernel_pmmr_h.last_pos,
|
|
|
|
),
|
2017-09-28 02:46:32 +03:00
|
|
|
commit_index: commit_index,
|
|
|
|
new_output_commits: HashMap::new(),
|
2018-03-09 00:36:51 +03:00
|
|
|
new_block_markers: HashMap::new(),
|
2017-09-28 02:46:32 +03:00
|
|
|
rollback: false,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-18 18:18:38 +03:00
|
|
|
// Rewind the MMR backend to undo applying a raw tx to the txhashset extension.
|
|
|
|
// This is used during txpool validation to undo an invalid tx.
|
|
|
|
fn rewind_raw_tx(
|
|
|
|
&mut self,
|
|
|
|
output_pos: u64,
|
|
|
|
kernel_pos: u64,
|
|
|
|
rewind_rm_pos: &Bitmap,
|
|
|
|
) -> Result<(), Error> {
|
|
|
|
let latest_output_pos = self.output_pmmr.unpruned_size();
|
|
|
|
let rewind_add_pos: Bitmap = ((output_pos + 1)..(latest_output_pos + 1))
|
|
|
|
.map(|x| x as u32)
|
|
|
|
.collect();
|
|
|
|
self.rewind_to_pos(output_pos, kernel_pos, &rewind_add_pos, rewind_rm_pos)?;
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2018-05-30 23:57:13 +03:00
|
|
|
/// Apply a "raw" transaction to the txhashset.
|
|
|
|
/// We will never commit a txhashset extension that includes raw txs.
|
|
|
|
/// But we can use this when validating txs in the tx pool.
|
|
|
|
/// If we can add a tx to the tx pool and then successfully add the
|
|
|
|
/// aggregated tx from the tx pool to the current chain state (via a
|
|
|
|
/// txhashset extension) then we know the tx pool is valid (including the
|
|
|
|
/// new tx).
|
2018-06-18 18:18:38 +03:00
|
|
|
pub fn apply_raw_tx(&mut self, tx: &Transaction) -> Result<(), Error> {
|
2018-05-30 23:57:13 +03:00
|
|
|
// This should *never* be called on a writeable extension...
|
|
|
|
if !self.rollback {
|
|
|
|
panic!("attempted to apply a raw tx to a writeable txhashset extension");
|
|
|
|
}
|
|
|
|
|
|
|
|
// Checkpoint the MMR positions before we apply the new tx,
|
|
|
|
// anything goes wrong we will rewind to these positions.
|
|
|
|
let output_pos = self.output_pmmr.unpruned_size();
|
|
|
|
let kernel_pos = self.kernel_pmmr.unpruned_size();
|
|
|
|
|
2018-06-18 18:18:38 +03:00
|
|
|
// Build bitmap of output pos spent (as inputs) by this tx for rewind.
|
|
|
|
let rewind_rm_pos = tx.inputs
|
|
|
|
.iter()
|
|
|
|
.filter_map(|x| self.get_output_pos(&x.commitment()).ok())
|
|
|
|
.map(|x| x as u32)
|
|
|
|
.collect();
|
2018-06-02 14:29:18 +03:00
|
|
|
|
2018-05-30 23:57:13 +03:00
|
|
|
for ref output in &tx.outputs {
|
|
|
|
if let Err(e) = self.apply_output(output) {
|
2018-06-18 18:18:38 +03:00
|
|
|
self.rewind_raw_tx(output_pos, kernel_pos, &rewind_rm_pos)?;
|
2018-05-30 23:57:13 +03:00
|
|
|
return Err(e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for ref input in &tx.inputs {
|
2018-06-18 18:18:38 +03:00
|
|
|
if let Err(e) = self.apply_input(input) {
|
|
|
|
self.rewind_raw_tx(output_pos, kernel_pos, &rewind_rm_pos)?;
|
2018-05-30 23:57:13 +03:00
|
|
|
return Err(e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for ref kernel in &tx.kernels {
|
|
|
|
if let Err(e) = self.apply_kernel(kernel) {
|
2018-06-18 18:18:38 +03:00
|
|
|
self.rewind_raw_tx(output_pos, kernel_pos, &rewind_rm_pos)?;
|
2018-05-30 23:57:13 +03:00
|
|
|
return Err(e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Validate a vector of "raw" transactions against the current chain state.
|
|
|
|
/// We support rewind on a "dirty" txhashset - so we can apply each tx in
|
|
|
|
/// turn, rewinding if any particular tx is not valid and continuing
|
|
|
|
/// through the vec of txs provided. This allows us to efficiently apply
|
|
|
|
/// all the txs, filtering out those that are not valid and returning the
|
|
|
|
/// final vec of txs that were successfully validated against the txhashset.
|
|
|
|
///
|
|
|
|
/// Note: We also pass in a "pre_tx". This tx is applied to and validated
|
|
|
|
/// before we start applying the vec of txs. We use this when validating
|
|
|
|
/// txs in the stempool as we need to account for txs in the txpool as
|
|
|
|
/// well (new_tx + stempool + txpool + txhashset). So we aggregate the
|
|
|
|
/// contents of the txpool into a single aggregated tx and pass it in here
|
|
|
|
/// as the "pre_tx" so we apply it to the txhashset before we start
|
|
|
|
/// validating the stempool txs.
|
|
|
|
/// This is optional and we pass in None when validating the txpool txs
|
|
|
|
/// themselves.
|
|
|
|
///
|
|
|
|
pub fn validate_raw_txs(
|
|
|
|
&mut self,
|
|
|
|
txs: Vec<Transaction>,
|
|
|
|
pre_tx: Option<Transaction>,
|
|
|
|
) -> Result<Vec<Transaction>, Error> {
|
|
|
|
let mut valid_txs = vec![];
|
2018-06-18 18:18:38 +03:00
|
|
|
|
|
|
|
// First apply the "pre_tx" to account for any state that need adding to
|
|
|
|
// the chain state before we can validate our vec of txs.
|
|
|
|
// This is the aggregate tx from the txpool if we are validating the stempool.
|
2018-05-30 23:57:13 +03:00
|
|
|
if let Some(tx) = pre_tx {
|
2018-06-18 18:18:38 +03:00
|
|
|
self.apply_raw_tx(&tx)?;
|
2018-05-30 23:57:13 +03:00
|
|
|
}
|
2018-06-18 18:18:38 +03:00
|
|
|
|
|
|
|
// Now validate each tx, rewinding any tx (and only that tx)
|
|
|
|
// if it fails to validate successfully.
|
2018-05-30 23:57:13 +03:00
|
|
|
for tx in txs {
|
2018-06-18 18:18:38 +03:00
|
|
|
if self.apply_raw_tx(&tx).is_ok() {
|
2018-05-30 23:57:13 +03:00
|
|
|
valid_txs.push(tx);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Ok(valid_txs)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Verify we are not attempting to spend any coinbase outputs
|
|
|
|
/// that have not sufficiently matured.
|
|
|
|
pub fn verify_coinbase_maturity(
|
|
|
|
&mut self,
|
|
|
|
inputs: &Vec<Input>,
|
|
|
|
height: u64,
|
|
|
|
) -> Result<(), Error> {
|
2018-06-20 22:18:52 +03:00
|
|
|
for input in inputs {
|
|
|
|
if input.features.contains(OutputFeatures::COINBASE_OUTPUT) {
|
|
|
|
self.verify_maturity_via_merkle_proof(input, height)?;
|
2018-05-30 23:57:13 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2018-06-20 22:18:52 +03:00
|
|
|
fn verify_maturity_via_merkle_proof(&self, input: &Input, height: u64) -> Result<(), Error> {
|
|
|
|
let header = self.commit_index.get_block_header(&input.block_hash())?;
|
|
|
|
|
|
|
|
// Check that the height indicates it has matured sufficiently
|
|
|
|
// we will check the Merkle proof below to ensure we are being
|
|
|
|
// honest about the height
|
|
|
|
if header.height + global::coinbase_maturity() >= height {
|
|
|
|
return Err(Error::ImmatureCoinbase);
|
|
|
|
}
|
|
|
|
|
|
|
|
// We need the MMR pos to verify the Merkle proof
|
|
|
|
let pos = self.get_output_pos(&input.commitment())?;
|
|
|
|
|
|
|
|
let out_id = OutputIdentifier::from_input(input);
|
|
|
|
let res = input
|
|
|
|
.merkle_proof()
|
|
|
|
.verify(header.output_root, &out_id, pos)
|
|
|
|
.map_err(|_| Error::MerkleProof)?;
|
|
|
|
|
|
|
|
Ok(res)
|
|
|
|
}
|
|
|
|
|
2017-09-28 02:46:32 +03:00
|
|
|
/// Apply a new set of blocks on top the existing sum trees. Blocks are
|
|
|
|
/// applied in order of the provided Vec. If pruning is enabled, inputs also
|
|
|
|
/// prune MMR data.
|
|
|
|
pub fn apply_block(&mut self, b: &Block) -> Result<(), Error> {
|
2018-06-20 22:47:32 +03:00
|
|
|
// A block is not valid if it has not been fully cut-through.
|
|
|
|
// So we can safely apply outputs first (we will not spend these in the same
|
|
|
|
// block).
|
2018-01-08 04:23:23 +03:00
|
|
|
for out in &b.outputs {
|
2018-06-20 22:47:32 +03:00
|
|
|
self.apply_output(out)?;
|
2018-01-08 04:23:23 +03:00
|
|
|
}
|
2018-01-17 06:03:40 +03:00
|
|
|
|
2017-09-28 02:46:32 +03:00
|
|
|
for input in &b.inputs {
|
2018-06-18 18:18:38 +03:00
|
|
|
self.apply_input(input)?;
|
2017-09-28 02:46:32 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
for kernel in &b.kernels {
|
2018-01-08 04:23:23 +03:00
|
|
|
self.apply_kernel(kernel)?;
|
2017-09-28 02:46:32 +03:00
|
|
|
}
|
2018-03-02 23:47:27 +03:00
|
|
|
|
2018-03-09 00:36:51 +03:00
|
|
|
// finally, recording the PMMR positions after this block for future rewind
|
2018-04-24 22:53:01 +03:00
|
|
|
let marker = BlockMarker {
|
|
|
|
output_pos: self.output_pmmr.unpruned_size(),
|
|
|
|
kernel_pos: self.kernel_pmmr.unpruned_size(),
|
|
|
|
};
|
|
|
|
self.new_block_markers.insert(b.hash(), marker);
|
2017-09-28 02:46:32 +03:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2018-05-30 23:57:13 +03:00
|
|
|
// Store all new output pos in the index.
|
|
|
|
// Also store any new block_markers.
|
2018-03-09 00:36:51 +03:00
|
|
|
fn save_indexes(&self) -> Result<(), Error> {
|
2017-09-28 02:46:32 +03:00
|
|
|
for (commit, pos) in &self.new_output_commits {
|
|
|
|
self.commit_index.save_output_pos(commit, *pos)?;
|
|
|
|
}
|
2018-04-24 22:53:01 +03:00
|
|
|
for (bh, marker) in &self.new_block_markers {
|
|
|
|
self.commit_index.save_block_marker(bh, marker)?;
|
2017-09-28 02:46:32 +03:00
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2018-06-18 18:18:38 +03:00
|
|
|
fn apply_input(&mut self, input: &Input) -> Result<(), Error> {
|
2018-01-08 04:23:23 +03:00
|
|
|
let commit = input.commitment();
|
|
|
|
let pos_res = self.get_output_pos(&commit);
|
|
|
|
if let Ok(pos) = pos_res {
|
2018-03-30 09:02:40 +03:00
|
|
|
let output_id_hash = OutputIdentifier::from_input(input).hash_with_index(pos - 1);
|
2018-03-24 02:33:59 +03:00
|
|
|
if let Some(read_hash) = self.output_pmmr.get_hash(pos) {
|
2018-02-22 16:45:13 +03:00
|
|
|
// check hash from pmmr matches hash from input (or corresponding output)
|
2018-01-17 06:03:40 +03:00
|
|
|
// if not then the input is not being honest about
|
|
|
|
// what it is attempting to spend...
|
2018-03-24 02:33:59 +03:00
|
|
|
let read_elem = self.output_pmmr.get_data(pos);
|
2018-05-30 23:57:13 +03:00
|
|
|
let read_elem_hash = read_elem
|
|
|
|
.expect("no output at pos")
|
|
|
|
.hash_with_index(pos - 1);
|
|
|
|
if output_id_hash != read_hash || output_id_hash != read_elem_hash {
|
2018-03-05 22:33:44 +03:00
|
|
|
return Err(Error::TxHashSetErr(format!("output pmmr hash mismatch")));
|
2018-01-17 06:03:40 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-05 22:33:44 +03:00
|
|
|
// Now prune the output_pmmr, rproof_pmmr and their storage.
|
2018-03-04 03:19:54 +03:00
|
|
|
// Input is not valid if we cannot prune successfully (to spend an unspent
|
|
|
|
// output).
|
2018-06-18 18:18:38 +03:00
|
|
|
match self.output_pmmr.prune(pos) {
|
2018-01-08 04:23:23 +03:00
|
|
|
Ok(true) => {
|
|
|
|
self.rproof_pmmr
|
2018-06-18 18:18:38 +03:00
|
|
|
.prune(pos)
|
2018-03-05 22:33:44 +03:00
|
|
|
.map_err(|s| Error::TxHashSetErr(s))?;
|
2018-01-08 04:23:23 +03:00
|
|
|
}
|
|
|
|
Ok(false) => return Err(Error::AlreadySpent(commit)),
|
2018-03-05 22:33:44 +03:00
|
|
|
Err(s) => return Err(Error::TxHashSetErr(s)),
|
2018-01-08 04:23:23 +03:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return Err(Error::AlreadySpent(commit));
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn apply_output(&mut self, out: &Output) -> Result<(), Error> {
|
|
|
|
let commit = out.commitment();
|
|
|
|
if let Ok(pos) = self.get_output_pos(&commit) {
|
|
|
|
// we need to check whether the commitment is in the current MMR view
|
|
|
|
// as well as the index doesn't support rewind and is non-authoritative
|
|
|
|
// (non-historical node will have a much smaller one)
|
|
|
|
// note that this doesn't show the commitment *never* existed, just
|
|
|
|
// that this is not an existing unspent commitment right now
|
2018-03-24 02:33:59 +03:00
|
|
|
if let Some(hash) = self.output_pmmr.get_hash(pos) {
|
2018-01-08 04:23:23 +03:00
|
|
|
// processing a new fork so we may get a position on the old
|
|
|
|
// fork that exists but matches a different node
|
|
|
|
// filtering that case out
|
2018-05-30 17:15:37 +03:00
|
|
|
//
|
|
|
|
// TODO - the following call to hash() is *INCORRECT*
|
|
|
|
// TODO - we should be calling hash_with_index(pos - 1)
|
|
|
|
// TODO - but we cannot safely make this change in testnet2
|
|
|
|
// TODO - with this incorrect behavior we never get a match, so duplicates are
|
|
|
|
// allowed
|
|
|
|
//
|
|
|
|
if hash == OutputIdentifier::from_output(out).hash() {
|
2018-01-08 04:23:23 +03:00
|
|
|
return Err(Error::DuplicateCommitment(commit));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-02-22 16:45:13 +03:00
|
|
|
// push new outputs in their MMR and save them in the index
|
2018-03-05 22:33:44 +03:00
|
|
|
let pos = self.output_pmmr
|
2018-03-22 03:10:11 +03:00
|
|
|
.push(OutputIdentifier::from_output(out))
|
2018-03-05 22:33:44 +03:00
|
|
|
.map_err(&Error::TxHashSetErr)?;
|
2018-01-08 04:23:23 +03:00
|
|
|
self.new_output_commits.insert(out.commitment(), pos);
|
|
|
|
|
2018-02-22 16:45:13 +03:00
|
|
|
// push range proofs in their MMR and file
|
2018-01-08 04:23:23 +03:00
|
|
|
self.rproof_pmmr
|
2018-02-22 16:45:13 +03:00
|
|
|
.push(out.proof)
|
2018-03-05 22:33:44 +03:00
|
|
|
.map_err(&Error::TxHashSetErr)?;
|
2018-01-08 04:23:23 +03:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn apply_kernel(&mut self, kernel: &TxKernel) -> Result<(), Error> {
|
2018-02-10 01:32:16 +03:00
|
|
|
// push kernels in their MMR and file
|
2018-03-09 00:36:51 +03:00
|
|
|
self.kernel_pmmr
|
2018-02-22 16:45:13 +03:00
|
|
|
.push(kernel.clone())
|
2018-03-05 22:33:44 +03:00
|
|
|
.map_err(&Error::TxHashSetErr)?;
|
2018-02-10 01:32:16 +03:00
|
|
|
|
2018-01-08 04:23:23 +03:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2018-03-02 23:47:27 +03:00
|
|
|
/// Build a Merkle proof for the given output and the block by
|
|
|
|
/// rewinding the MMR to the last pos of the block.
|
|
|
|
/// Note: this relies on the MMR being stable even after pruning/compaction.
|
|
|
|
/// We need the hash of each sibling pos from the pos up to the peak
|
|
|
|
/// including the sibling leaf node which may have been removed.
|
2018-03-22 19:57:37 +03:00
|
|
|
pub fn merkle_proof(
|
2018-03-02 23:47:27 +03:00
|
|
|
&mut self,
|
|
|
|
output: &OutputIdentifier,
|
2018-03-09 00:36:51 +03:00
|
|
|
block_header: &BlockHeader,
|
2018-03-02 23:47:27 +03:00
|
|
|
) -> Result<MerkleProof, Error> {
|
2018-03-04 03:19:54 +03:00
|
|
|
debug!(
|
|
|
|
LOGGER,
|
2018-03-22 19:57:37 +03:00
|
|
|
"txhashset: merkle_proof: output: {:?}, block: {:?}",
|
|
|
|
output.commit,
|
2018-03-09 00:36:51 +03:00
|
|
|
block_header.hash()
|
2018-03-04 03:19:54 +03:00
|
|
|
);
|
2018-03-09 00:36:51 +03:00
|
|
|
|
2018-04-09 19:37:46 +03:00
|
|
|
// rewind to the specified block for a consistent view
|
2018-06-18 18:18:38 +03:00
|
|
|
let head_header = self.commit_index.head_header()?;
|
|
|
|
self.rewind(block_header, &head_header)?;
|
2018-03-22 19:57:37 +03:00
|
|
|
|
2018-03-02 23:47:27 +03:00
|
|
|
// then calculate the Merkle Proof based on the known pos
|
|
|
|
let pos = self.get_output_pos(&output.commit)?;
|
2018-03-05 22:33:44 +03:00
|
|
|
let merkle_proof = self.output_pmmr
|
2018-03-02 23:47:27 +03:00
|
|
|
.merkle_proof(pos)
|
2018-03-05 22:33:44 +03:00
|
|
|
.map_err(&Error::TxHashSetErr)?;
|
2018-03-02 23:47:27 +03:00
|
|
|
|
|
|
|
Ok(merkle_proof)
|
|
|
|
}
|
|
|
|
|
2018-06-18 18:18:38 +03:00
|
|
|
/// Saves a snapshot of the output and rangeproof MMRs to disk.
|
|
|
|
/// Specifically - saves a snapshot of the utxo file, tagged with
|
|
|
|
/// the block hash as filename suffix.
|
|
|
|
/// Needed for fast-sync (utxo file needs to be rewound before sending
|
|
|
|
/// across).
|
|
|
|
pub fn snapshot(&mut self, header: &BlockHeader) -> Result<(), Error> {
|
|
|
|
self.output_pmmr
|
|
|
|
.snapshot(header)
|
|
|
|
.map_err(|e| Error::Other(e))?;
|
|
|
|
self.rproof_pmmr
|
|
|
|
.snapshot(header)
|
|
|
|
.map_err(|e| Error::Other(e))?;
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Rewinds the MMRs to the provided block, rewinding to the last output pos
|
|
|
|
/// and last kernel pos of that block.
|
|
|
|
pub fn rewind(
|
|
|
|
&mut self,
|
|
|
|
block_header: &BlockHeader,
|
|
|
|
head_header: &BlockHeader,
|
|
|
|
) -> Result<(), Error> {
|
2018-03-09 00:36:51 +03:00
|
|
|
let hash = block_header.hash();
|
2018-06-18 18:18:38 +03:00
|
|
|
trace!(
|
|
|
|
LOGGER,
|
|
|
|
"Rewind to header {} @ {}",
|
|
|
|
block_header.height,
|
|
|
|
hash
|
|
|
|
);
|
2017-11-15 23:37:40 +03:00
|
|
|
|
2018-06-18 18:18:38 +03:00
|
|
|
// Rewind our MMRs to the appropriate positions
|
|
|
|
// based on the block_marker.
|
2018-04-24 22:53:01 +03:00
|
|
|
let marker = self.commit_index.get_block_marker(&hash)?;
|
2018-06-18 18:18:38 +03:00
|
|
|
|
|
|
|
// We need to build bitmaps of added and removed output positions
|
|
|
|
// so we can correctly rewind all operations applied to the output MMR
|
|
|
|
// after the position we are rewinding to (these operations will be
|
|
|
|
// undone during rewind).
|
|
|
|
// Rewound output pos will be removed from the MMR.
|
|
|
|
// Rewound input (spent) pos will be added back to the MMR.
|
|
|
|
let rewind_add_pos =
|
|
|
|
output_pos_to_rewind(self.commit_index.clone(), block_header, head_header)?;
|
|
|
|
let rewind_rm_pos =
|
|
|
|
input_pos_to_rewind(self.commit_index.clone(), block_header, head_header)?;
|
|
|
|
|
|
|
|
self.rewind_to_pos(
|
|
|
|
marker.output_pos,
|
|
|
|
marker.kernel_pos,
|
|
|
|
&rewind_add_pos,
|
|
|
|
&rewind_rm_pos,
|
|
|
|
)?;
|
2018-05-30 23:57:13 +03:00
|
|
|
|
2018-02-10 01:32:16 +03:00
|
|
|
Ok(())
|
|
|
|
}
|
2017-11-15 23:37:40 +03:00
|
|
|
|
2018-02-10 01:32:16 +03:00
|
|
|
/// Rewinds the MMRs to the provided positions, given the output and
|
|
|
|
/// kernel we want to rewind to.
|
2018-05-30 23:57:13 +03:00
|
|
|
fn rewind_to_pos(
|
|
|
|
&mut self,
|
|
|
|
output_pos: u64,
|
|
|
|
kernel_pos: u64,
|
2018-06-18 18:18:38 +03:00
|
|
|
rewind_add_pos: &Bitmap,
|
|
|
|
rewind_rm_pos: &Bitmap,
|
2018-05-30 23:57:13 +03:00
|
|
|
) -> Result<(), Error> {
|
|
|
|
trace!(
|
|
|
|
LOGGER,
|
2018-06-18 18:18:38 +03:00
|
|
|
"Rewind txhashset to output {}, kernel {}",
|
2018-05-30 23:57:13 +03:00
|
|
|
output_pos,
|
|
|
|
kernel_pos,
|
|
|
|
);
|
2017-11-15 23:37:40 +03:00
|
|
|
|
2018-06-18 18:18:38 +03:00
|
|
|
// Remember to "rewind" our new_output_commits
|
|
|
|
// in case we are rewinding state that has not yet
|
|
|
|
// been sync'd to disk.
|
|
|
|
self.new_output_commits.retain(|_, &mut v| v <= output_pos);
|
|
|
|
|
2018-03-05 22:33:44 +03:00
|
|
|
self.output_pmmr
|
2018-06-18 18:18:38 +03:00
|
|
|
.rewind(output_pos, rewind_add_pos, rewind_rm_pos)
|
2018-03-05 22:33:44 +03:00
|
|
|
.map_err(&Error::TxHashSetErr)?;
|
2017-09-29 21:44:25 +03:00
|
|
|
self.rproof_pmmr
|
2018-06-18 18:18:38 +03:00
|
|
|
.rewind(output_pos, rewind_add_pos, rewind_rm_pos)
|
2018-03-05 22:33:44 +03:00
|
|
|
.map_err(&Error::TxHashSetErr)?;
|
2017-09-29 21:44:25 +03:00
|
|
|
self.kernel_pmmr
|
2018-06-18 18:18:38 +03:00
|
|
|
.rewind(kernel_pos, rewind_add_pos, rewind_rm_pos)
|
2018-03-05 22:33:44 +03:00
|
|
|
.map_err(&Error::TxHashSetErr)?;
|
2017-11-15 23:37:40 +03:00
|
|
|
|
2017-09-28 02:46:32 +03:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2017-12-25 03:27:13 +03:00
|
|
|
fn get_output_pos(&self, commit: &Commitment) -> Result<u64, grin_store::Error> {
|
|
|
|
if let Some(pos) = self.new_output_commits.get(commit) {
|
|
|
|
Ok(*pos)
|
|
|
|
} else {
|
|
|
|
self.commit_index.get_output_pos(commit)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-05 22:33:44 +03:00
|
|
|
/// Current root hashes and sums (if applicable) for the Output, range proof
|
2017-09-28 02:46:32 +03:00
|
|
|
/// and kernel sum trees.
|
2018-03-05 22:33:44 +03:00
|
|
|
pub fn roots(&self) -> TxHashSetRoots {
|
|
|
|
TxHashSetRoots {
|
|
|
|
output_root: self.output_pmmr.root(),
|
2018-02-22 16:45:13 +03:00
|
|
|
rproof_root: self.rproof_pmmr.root(),
|
|
|
|
kernel_root: self.kernel_pmmr.root(),
|
|
|
|
}
|
2017-09-28 02:46:32 +03:00
|
|
|
}
|
|
|
|
|
2018-04-24 22:53:01 +03:00
|
|
|
/// Validate the various MMR roots against the block header.
|
|
|
|
pub fn validate_roots(&self, header: &BlockHeader) -> Result<(), Error> {
|
|
|
|
// If we are validating the genesis block then
|
|
|
|
// we have no outputs or kernels.
|
|
|
|
// So we are done here.
|
|
|
|
if header.height == 0 {
|
|
|
|
return Ok(());
|
|
|
|
}
|
|
|
|
|
|
|
|
let roots = self.roots();
|
|
|
|
if roots.output_root != header.output_root || roots.rproof_root != header.range_proof_root
|
|
|
|
|| roots.kernel_root != header.kernel_root
|
|
|
|
{
|
|
|
|
return Err(Error::InvalidRoot);
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2018-05-07 16:21:41 +03:00
|
|
|
fn validate_mmrs(&self) -> Result<(), Error> {
|
|
|
|
let now = Instant::now();
|
|
|
|
|
2018-02-10 01:32:16 +03:00
|
|
|
// validate all hashes and sums within the trees
|
2018-03-05 22:33:44 +03:00
|
|
|
if let Err(e) = self.output_pmmr.validate() {
|
|
|
|
return Err(Error::InvalidTxHashSet(e));
|
2018-02-10 01:32:16 +03:00
|
|
|
}
|
|
|
|
if let Err(e) = self.rproof_pmmr.validate() {
|
2018-03-05 22:33:44 +03:00
|
|
|
return Err(Error::InvalidTxHashSet(e));
|
2018-02-10 01:32:16 +03:00
|
|
|
}
|
|
|
|
if let Err(e) = self.kernel_pmmr.validate() {
|
2018-03-05 22:33:44 +03:00
|
|
|
return Err(Error::InvalidTxHashSet(e));
|
2018-02-10 01:32:16 +03:00
|
|
|
}
|
|
|
|
|
2018-05-07 16:21:41 +03:00
|
|
|
debug!(
|
|
|
|
LOGGER,
|
|
|
|
"txhashset: validated the output|rproof|kernel mmrs, took {}s",
|
|
|
|
now.elapsed().as_secs(),
|
|
|
|
);
|
2018-04-24 22:53:01 +03:00
|
|
|
|
2018-05-07 16:21:41 +03:00
|
|
|
Ok(())
|
|
|
|
}
|
2018-02-10 01:32:16 +03:00
|
|
|
|
2018-05-07 16:21:41 +03:00
|
|
|
/// Validate the txhashset state against the provided block header.
|
|
|
|
pub fn validate(
|
|
|
|
&mut self,
|
|
|
|
header: &BlockHeader,
|
|
|
|
skip_rproofs: bool,
|
|
|
|
) -> Result<((Commitment, Commitment)), Error> {
|
|
|
|
self.validate_mmrs()?;
|
|
|
|
self.validate_roots(header)?;
|
|
|
|
|
|
|
|
if header.height == 0 {
|
|
|
|
let zero_commit = secp_static::commit_to_zero_value();
|
|
|
|
return Ok((zero_commit.clone(), zero_commit.clone()));
|
|
|
|
}
|
|
|
|
|
2018-06-02 21:00:44 +03:00
|
|
|
// The real magicking happens here.
|
|
|
|
// Sum of kernel excesses should equal sum of
|
|
|
|
// unspent outputs minus total supply.
|
|
|
|
let (output_sum, kernel_sum) = self.verify_kernel_sums(
|
|
|
|
header.total_overage(),
|
|
|
|
header.total_kernel_offset(),
|
|
|
|
None,
|
|
|
|
None,
|
|
|
|
)?;
|
2018-05-07 16:21:41 +03:00
|
|
|
|
2018-06-02 21:00:44 +03:00
|
|
|
// This is an expensive verification step.
|
2018-05-07 16:21:41 +03:00
|
|
|
self.verify_kernel_signatures()?;
|
|
|
|
|
2018-06-02 21:00:44 +03:00
|
|
|
// Verify the rangeproof for each output in the sum above.
|
|
|
|
// This is an expensive verification step (skip for faster verification).
|
2018-03-21 15:28:05 +03:00
|
|
|
if !skip_rproofs {
|
|
|
|
self.verify_rangeproofs()?;
|
|
|
|
}
|
2018-03-21 03:34:19 +03:00
|
|
|
|
2018-05-07 16:21:41 +03:00
|
|
|
Ok((output_sum, kernel_sum))
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Save blocks sums (the output_sum and kernel_sum) for the given block
|
|
|
|
/// header.
|
|
|
|
pub fn save_latest_block_sums(
|
|
|
|
&self,
|
|
|
|
header: &BlockHeader,
|
|
|
|
output_sum: Commitment,
|
|
|
|
kernel_sum: Commitment,
|
|
|
|
) -> Result<(), Error> {
|
|
|
|
self.commit_index.save_block_sums(
|
|
|
|
&header.hash(),
|
|
|
|
&BlockSums {
|
|
|
|
output_sum,
|
|
|
|
kernel_sum,
|
|
|
|
},
|
|
|
|
)?;
|
2018-02-10 01:32:16 +03:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2018-05-28 21:22:22 +03:00
|
|
|
/// Rebuild the index of MMR positions to the corresponding Output and
|
|
|
|
/// kernel by iterating over the whole MMR data. This is a costly operation
|
2018-02-10 01:32:16 +03:00
|
|
|
/// performed only when we receive a full new chain state.
|
|
|
|
pub fn rebuild_index(&self) -> Result<(), Error> {
|
2018-03-05 22:33:44 +03:00
|
|
|
for n in 1..self.output_pmmr.unpruned_size() + 1 {
|
2018-02-10 01:32:16 +03:00
|
|
|
// non-pruned leaves only
|
|
|
|
if pmmr::bintree_postorder_height(n) == 0 {
|
2018-03-24 02:33:59 +03:00
|
|
|
if let Some(out) = self.output_pmmr.get_data(n) {
|
|
|
|
self.commit_index.save_output_pos(&out.commit, n)?;
|
2018-02-10 01:32:16 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2017-09-28 02:46:32 +03:00
|
|
|
/// Force the rollback of this extension, no matter the result
|
|
|
|
pub fn force_rollback(&mut self) {
|
|
|
|
self.rollback = true;
|
|
|
|
}
|
|
|
|
|
2018-03-13 21:22:34 +03:00
|
|
|
/// Dumps the output MMR.
|
|
|
|
/// We use this after compacting for visual confirmation that it worked.
|
|
|
|
pub fn dump_output_pmmr(&self) {
|
|
|
|
debug!(LOGGER, "-- outputs --");
|
|
|
|
self.output_pmmr.dump_from_file(false);
|
2018-03-15 19:53:40 +03:00
|
|
|
debug!(LOGGER, "--");
|
|
|
|
self.output_pmmr.dump_stats();
|
2018-03-13 21:22:34 +03:00
|
|
|
debug!(LOGGER, "-- end of outputs --");
|
|
|
|
}
|
|
|
|
|
2017-10-28 00:57:04 +03:00
|
|
|
/// Dumps the state of the 3 sum trees to stdout for debugging. Short
|
2018-03-05 22:33:44 +03:00
|
|
|
/// version only prints the Output tree.
|
2017-10-22 10:11:45 +03:00
|
|
|
pub fn dump(&self, short: bool) {
|
|
|
|
debug!(LOGGER, "-- outputs --");
|
2018-03-05 22:33:44 +03:00
|
|
|
self.output_pmmr.dump(short);
|
2017-10-28 00:57:04 +03:00
|
|
|
if !short {
|
|
|
|
debug!(LOGGER, "-- range proofs --");
|
|
|
|
self.rproof_pmmr.dump(short);
|
|
|
|
debug!(LOGGER, "-- kernels --");
|
|
|
|
self.kernel_pmmr.dump(short);
|
|
|
|
}
|
2017-10-12 22:23:58 +03:00
|
|
|
}
|
|
|
|
|
2017-09-28 02:46:32 +03:00
|
|
|
// Sizes of the sum trees, used by `extending` on rollback.
|
|
|
|
fn sizes(&self) -> (u64, u64, u64) {
|
2017-10-17 00:23:10 +03:00
|
|
|
(
|
2018-03-05 22:33:44 +03:00
|
|
|
self.output_pmmr.unpruned_size(),
|
2017-10-17 00:23:10 +03:00
|
|
|
self.rproof_pmmr.unpruned_size(),
|
|
|
|
self.kernel_pmmr.unpruned_size(),
|
|
|
|
)
|
2017-09-28 02:46:32 +03:00
|
|
|
}
|
2018-02-10 01:32:16 +03:00
|
|
|
|
2018-05-07 16:21:41 +03:00
|
|
|
fn verify_kernel_signatures(&self) -> Result<(), Error> {
|
2018-03-21 03:34:19 +03:00
|
|
|
let now = Instant::now();
|
|
|
|
|
2018-05-07 16:21:41 +03:00
|
|
|
let mut kern_count = 0;
|
2018-03-21 15:28:05 +03:00
|
|
|
for n in 1..self.kernel_pmmr.unpruned_size() + 1 {
|
|
|
|
if pmmr::is_leaf(n) {
|
2018-03-24 02:33:59 +03:00
|
|
|
if let Some(kernel) = self.kernel_pmmr.get_data(n) {
|
2018-02-10 01:32:16 +03:00
|
|
|
kernel.verify()?;
|
2018-05-07 16:21:41 +03:00
|
|
|
kern_count += 1;
|
2018-02-10 01:32:16 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-03-13 21:22:34 +03:00
|
|
|
|
|
|
|
debug!(
|
|
|
|
LOGGER,
|
2018-05-07 16:21:41 +03:00
|
|
|
"txhashset: verified {} kernel signatures, pmmr size {}, took {}s",
|
2018-03-21 03:34:19 +03:00
|
|
|
kern_count,
|
|
|
|
self.kernel_pmmr.unpruned_size(),
|
|
|
|
now.elapsed().as_secs(),
|
2018-03-13 21:22:34 +03:00
|
|
|
);
|
2018-05-07 16:21:41 +03:00
|
|
|
|
|
|
|
Ok(())
|
2018-02-10 01:32:16 +03:00
|
|
|
}
|
|
|
|
|
2018-03-21 03:34:19 +03:00
|
|
|
fn verify_rangeproofs(&self) -> Result<(), Error> {
|
|
|
|
let now = Instant::now();
|
|
|
|
|
|
|
|
let mut proof_count = 0;
|
2018-03-05 22:33:44 +03:00
|
|
|
for n in 1..self.output_pmmr.unpruned_size() + 1 {
|
2018-03-13 21:22:34 +03:00
|
|
|
if pmmr::is_leaf(n) {
|
2018-03-24 02:33:59 +03:00
|
|
|
if let Some(out) = self.output_pmmr.get_data(n) {
|
|
|
|
if let Some(rp) = self.rproof_pmmr.get_data(n) {
|
2018-06-01 22:41:26 +03:00
|
|
|
out.into_output(rp).verify_proof()?;
|
2018-03-21 15:28:05 +03:00
|
|
|
} else {
|
|
|
|
// TODO - rangeproof not found
|
|
|
|
return Err(Error::OutputNotFound);
|
2018-02-25 02:39:26 +03:00
|
|
|
}
|
2018-03-21 03:34:19 +03:00
|
|
|
proof_count += 1;
|
2018-04-18 22:12:39 +03:00
|
|
|
|
|
|
|
if proof_count % 500 == 0 {
|
|
|
|
debug!(
|
|
|
|
LOGGER,
|
|
|
|
"txhashset: verify_rangeproofs: verified {} rangeproofs", proof_count,
|
|
|
|
);
|
|
|
|
}
|
2018-03-21 03:34:19 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
debug!(
|
|
|
|
LOGGER,
|
2018-05-07 16:21:41 +03:00
|
|
|
"txhashset: verified {} rangeproofs, pmmr size {}, took {}s",
|
2018-03-21 03:34:19 +03:00
|
|
|
proof_count,
|
|
|
|
self.rproof_pmmr.unpruned_size(),
|
|
|
|
now.elapsed().as_secs(),
|
|
|
|
);
|
|
|
|
Ok(())
|
|
|
|
}
|
2018-02-10 01:32:16 +03:00
|
|
|
}
|
|
|
|
|
2018-03-05 22:33:44 +03:00
|
|
|
/// Packages the txhashset data files into a zip and returns a Read to the
|
2018-02-10 01:32:16 +03:00
|
|
|
/// resulting file
|
|
|
|
pub fn zip_read(root_dir: String) -> Result<File, Error> {
|
2018-03-05 22:33:44 +03:00
|
|
|
let txhashset_path = Path::new(&root_dir).join(TXHASHSET_SUBDIR);
|
|
|
|
let zip_path = Path::new(&root_dir).join(TXHASHSET_ZIP);
|
2018-02-10 01:32:16 +03:00
|
|
|
|
|
|
|
// create the zip archive
|
|
|
|
{
|
2018-03-05 22:33:44 +03:00
|
|
|
zip::compress(&txhashset_path, &File::create(zip_path.clone())?)
|
2018-02-10 01:32:16 +03:00
|
|
|
.map_err(|ze| Error::Other(ze.to_string()))?;
|
|
|
|
}
|
|
|
|
|
|
|
|
// open it again to read it back
|
|
|
|
let zip_file = File::open(zip_path)?;
|
|
|
|
Ok(zip_file)
|
|
|
|
}
|
|
|
|
|
2018-03-05 22:33:44 +03:00
|
|
|
/// Extract the txhashset data from a zip file and writes the content into the
|
|
|
|
/// txhashset storage dir
|
|
|
|
pub fn zip_write(root_dir: String, txhashset_data: File) -> Result<(), Error> {
|
|
|
|
let txhashset_path = Path::new(&root_dir).join(TXHASHSET_SUBDIR);
|
2018-02-10 01:32:16 +03:00
|
|
|
|
2018-03-05 22:33:44 +03:00
|
|
|
fs::create_dir_all(txhashset_path.clone())?;
|
|
|
|
zip::decompress(txhashset_data, &txhashset_path).map_err(|ze| Error::Other(ze.to_string()))
|
2017-09-28 02:46:32 +03:00
|
|
|
}
|
2018-06-18 18:18:38 +03:00
|
|
|
|
|
|
|
/// Given a block header to rewind to and the block header at the
|
|
|
|
/// head of the current chain state, we need to calculate the positions
|
|
|
|
/// of all outputs we need to "undo" during a rewind.
|
|
|
|
/// The MMR is append-only so we can simply look for all positions added after
|
|
|
|
/// the rewind pos.
|
|
|
|
fn output_pos_to_rewind(
|
|
|
|
commit_index: Arc<ChainStore>,
|
|
|
|
block_header: &BlockHeader,
|
|
|
|
head_header: &BlockHeader,
|
|
|
|
) -> Result<Bitmap, Error> {
|
|
|
|
let marker_to = commit_index.get_block_marker(&head_header.hash())?;
|
|
|
|
let marker_from = commit_index.get_block_marker(&block_header.hash())?;
|
|
|
|
Ok(((marker_from.output_pos + 1)..=marker_to.output_pos)
|
|
|
|
.map(|x| x as u32)
|
|
|
|
.collect())
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Given a block header to rewind to and the block header at the
|
|
|
|
/// head of the current chain state, we need to calculate the positions
|
|
|
|
/// of all inputs (spent outputs) we need to "undo" during a rewind.
|
|
|
|
/// We do this by leveraging the "block_input_bitmap" cache and OR'ing
|
|
|
|
/// the set of bitmaps together for the set of blocks being rewound.
|
|
|
|
fn input_pos_to_rewind(
|
|
|
|
commit_index: Arc<ChainStore>,
|
|
|
|
block_header: &BlockHeader,
|
|
|
|
head_header: &BlockHeader,
|
|
|
|
) -> Result<Bitmap, Error> {
|
|
|
|
let mut bitmap = Bitmap::create();
|
|
|
|
let mut current = head_header.hash();
|
|
|
|
loop {
|
|
|
|
if current == block_header.hash() {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// We cache recent block headers and block_input_bitmaps
|
|
|
|
// internally in our db layer (commit_index).
|
|
|
|
// I/O should be minimized or eliminated here for most
|
|
|
|
// rewind scenarios.
|
|
|
|
let current_header = commit_index.get_block_header(¤t)?;
|
|
|
|
let input_bitmap = commit_index.get_block_input_bitmap(¤t)?;
|
|
|
|
|
|
|
|
bitmap.or_inplace(&input_bitmap);
|
|
|
|
current = current_header.previous;
|
|
|
|
}
|
|
|
|
Ok(bitmap)
|
|
|
|
}
|