mirror of
https://github.com/mimblewimble/grin.git
synced 2025-01-21 03:21:08 +03:00
GET /v1/chain/validate (#832)
chain.validate() now takes skip_rproof=true|false
This commit is contained in:
parent
f0a3479ea3
commit
c595a4b35c
5 changed files with 73 additions and 73 deletions
|
@ -405,6 +405,20 @@ impl Handler for ChainHandler {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Chain validation handler.
|
||||||
|
/// GET /v1/chain/validate
|
||||||
|
pub struct ChainValidationHandler {
|
||||||
|
pub chain: Weak<chain::Chain>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Handler for ChainValidationHandler {
|
||||||
|
fn handle(&self, _req: &mut Request) -> IronResult<Response> {
|
||||||
|
// TODO - read skip_rproofs from query params
|
||||||
|
w(&self.chain).validate(true).unwrap();
|
||||||
|
Ok(Response::with((status::Ok, "{}")))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// Chain compaction handler. Trigger a compaction of the chain state to regain
|
/// Chain compaction handler. Trigger a compaction of the chain state to regain
|
||||||
/// storage space.
|
/// storage space.
|
||||||
/// GET /v1/chain/compact
|
/// GET /v1/chain/compact
|
||||||
|
@ -636,6 +650,9 @@ pub fn start_rest_apis<T>(
|
||||||
let chain_compact_handler = ChainCompactHandler {
|
let chain_compact_handler = ChainCompactHandler {
|
||||||
chain: chain.clone(),
|
chain: chain.clone(),
|
||||||
};
|
};
|
||||||
|
let chain_validation_handler = ChainValidationHandler {
|
||||||
|
chain: chain.clone(),
|
||||||
|
};
|
||||||
let status_handler = StatusHandler {
|
let status_handler = StatusHandler {
|
||||||
chain: chain.clone(),
|
chain: chain.clone(),
|
||||||
peers: peers.clone(),
|
peers: peers.clone(),
|
||||||
|
@ -667,6 +684,7 @@ pub fn start_rest_apis<T>(
|
||||||
"get blocks".to_string(),
|
"get blocks".to_string(),
|
||||||
"get chain".to_string(),
|
"get chain".to_string(),
|
||||||
"get chain/compact".to_string(),
|
"get chain/compact".to_string(),
|
||||||
|
"get chain/validate".to_string(),
|
||||||
"get chain/outputs".to_string(),
|
"get chain/outputs".to_string(),
|
||||||
"get status".to_string(),
|
"get status".to_string(),
|
||||||
"get txhashset/roots".to_string(),
|
"get txhashset/roots".to_string(),
|
||||||
|
@ -688,6 +706,7 @@ pub fn start_rest_apis<T>(
|
||||||
blocks: get "/blocks/*" => block_handler,
|
blocks: get "/blocks/*" => block_handler,
|
||||||
chain_tip: get "/chain" => chain_tip_handler,
|
chain_tip: get "/chain" => chain_tip_handler,
|
||||||
chain_compact: get "/chain/compact" => chain_compact_handler,
|
chain_compact: get "/chain/compact" => chain_compact_handler,
|
||||||
|
chain_validate: get "/chain/validate" => chain_validation_handler,
|
||||||
chain_outputs: get "/chain/outputs/*" => output_handler,
|
chain_outputs: get "/chain/outputs/*" => output_handler,
|
||||||
status: get "/status" => status_handler,
|
status: get "/status" => status_handler,
|
||||||
txhashset_roots: get "/txhashset/*" => txhashset_handler,
|
txhashset_roots: get "/txhashset/*" => txhashset_handler,
|
||||||
|
|
|
@ -412,7 +412,7 @@ impl Chain {
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Validate the current chain state.
|
/// Validate the current chain state.
|
||||||
pub fn validate(&self) -> Result<(), Error> {
|
pub fn validate(&self, skip_rproofs: bool) -> Result<(), Error> {
|
||||||
let header = self.store.head_header()?;
|
let header = self.store.head_header()?;
|
||||||
let mut txhashset = self.txhashset.write().unwrap();
|
let mut txhashset = self.txhashset.write().unwrap();
|
||||||
|
|
||||||
|
@ -423,7 +423,7 @@ impl Chain {
|
||||||
// Force rollback first as this is a "read-only" extension.
|
// Force rollback first as this is a "read-only" extension.
|
||||||
txhashset::extending(&mut txhashset, |extension| {
|
txhashset::extending(&mut txhashset, |extension| {
|
||||||
extension.force_rollback();
|
extension.force_rollback();
|
||||||
extension.validate(&header)
|
extension.validate(&header, skip_rproofs)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -532,7 +532,7 @@ impl Chain {
|
||||||
let mut txhashset =
|
let mut txhashset =
|
||||||
txhashset::TxHashSet::open(self.db_root.clone(), self.store.clone(), None)?;
|
txhashset::TxHashSet::open(self.db_root.clone(), self.store.clone(), None)?;
|
||||||
txhashset::extending(&mut txhashset, |extension| {
|
txhashset::extending(&mut txhashset, |extension| {
|
||||||
extension.validate(&header)?;
|
extension.validate(&header, false)?;
|
||||||
// TODO validate kernels and their sums with Outputs
|
// TODO validate kernels and their sums with Outputs
|
||||||
extension.rebuild_index()?;
|
extension.rebuild_index()?;
|
||||||
Ok(())
|
Ok(())
|
||||||
|
@ -572,7 +572,7 @@ impl Chain {
|
||||||
// First check we can successfully validate the full chain state.
|
// First check we can successfully validate the full chain state.
|
||||||
// If we cannot then do not attempt to compact.
|
// If we cannot then do not attempt to compact.
|
||||||
// This should not be required long term - but doing this for debug purposes.
|
// This should not be required long term - but doing this for debug purposes.
|
||||||
self.validate()?;
|
self.validate(true)?;
|
||||||
|
|
||||||
// Now compact the txhashset via the extension.
|
// Now compact the txhashset via the extension.
|
||||||
{
|
{
|
||||||
|
@ -588,7 +588,7 @@ impl Chain {
|
||||||
|
|
||||||
// Now check we can still successfully validate the chain state after
|
// Now check we can still successfully validate the chain state after
|
||||||
// compacting.
|
// compacting.
|
||||||
self.validate()?;
|
self.validate(true)?;
|
||||||
|
|
||||||
// we need to be careful here in testing as 20 blocks is not that long
|
// we need to be careful here in testing as 20 blocks is not that long
|
||||||
// in wall clock time
|
// in wall clock time
|
||||||
|
|
|
@ -31,7 +31,7 @@ use core::core::{Block, BlockHeader, Input, Output, OutputFeatures, OutputIdenti
|
||||||
use core::core::pmmr::{self, MerkleProof, PMMR};
|
use core::core::pmmr::{self, MerkleProof, PMMR};
|
||||||
use core::global;
|
use core::global;
|
||||||
use core::core::hash::{Hash, Hashed};
|
use core::core::hash::{Hash, Hashed};
|
||||||
use core::ser::{self, PMMRIndexHashable, PMMRable};
|
use core::ser::{PMMRIndexHashable, PMMRable};
|
||||||
|
|
||||||
use grin_store;
|
use grin_store;
|
||||||
use grin_store::pmmr::{PMMRBackend, PMMRFileMetadata};
|
use grin_store::pmmr::{PMMRBackend, PMMRFileMetadata};
|
||||||
|
@ -554,7 +554,7 @@ impl<'a> Extension<'a> {
|
||||||
/// Validate the txhashset state against the provided block header.
|
/// Validate the txhashset state against the provided block header.
|
||||||
/// Rewinds to that pos for the header first so we see a consistent
|
/// Rewinds to that pos for the header first so we see a consistent
|
||||||
/// view of the world.
|
/// view of the world.
|
||||||
pub fn validate(&mut self, header: &BlockHeader) -> Result<(), Error> {
|
pub fn validate(&mut self, header: &BlockHeader, skip_rproofs: bool) -> Result<(), Error> {
|
||||||
// first rewind to the provided header
|
// first rewind to the provided header
|
||||||
&self.rewind(header)?;
|
&self.rewind(header)?;
|
||||||
|
|
||||||
|
@ -599,9 +599,11 @@ impl<'a> Extension<'a> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// now verify the rangeproof for each output included in the sum above
|
// now verify the rangeproof for each output in the sum above
|
||||||
// this is an expensive operation
|
// this is an expensive operation (only verified if requested)
|
||||||
self.verify_rangeproofs()?;
|
if !skip_rproofs {
|
||||||
|
self.verify_rangeproofs()?;
|
||||||
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
@ -679,40 +681,24 @@ impl<'a> Extension<'a> {
|
||||||
fn sum_kernels(&self, kernel_offset: Option<Commitment>) -> Result<Commitment, Error> {
|
fn sum_kernels(&self, kernel_offset: Option<Commitment>) -> Result<Commitment, Error> {
|
||||||
let now = Instant::now();
|
let now = Instant::now();
|
||||||
|
|
||||||
// make sure we have the right count of kernels using the MMR, the storage
|
let mut commitments = vec![];
|
||||||
// file may have a few more
|
if let Some(offset) = kernel_offset {
|
||||||
let mmr_sz = self.kernel_pmmr.unpruned_size();
|
commitments.push(offset);
|
||||||
let count = pmmr::n_leaves(mmr_sz);
|
}
|
||||||
|
|
||||||
let mut kernel_file = File::open(self.kernel_pmmr.data_file_path())?;
|
for n in 1..self.kernel_pmmr.unpruned_size() + 1 {
|
||||||
let first: TxKernel = ser::deserialize(&mut kernel_file)?;
|
if pmmr::is_leaf(n) {
|
||||||
first.verify()?;
|
if let Some((_, Some(kernel))) = self.kernel_pmmr.get(n, true) {
|
||||||
let mut sum_kernel = first.excess;
|
kernel.verify()?;
|
||||||
|
commitments.push(kernel.excess.clone());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
let secp = static_secp_instance();
|
let secp = static_secp_instance();
|
||||||
let mut kern_count = 1;
|
let secp = secp.lock().unwrap();
|
||||||
loop {
|
let kern_count = commitments.len();
|
||||||
match ser::deserialize::<TxKernel>(&mut kernel_file) {
|
let sum_kernel = secp.commit_sum(commitments, vec![])?;
|
||||||
Ok(kernel) => {
|
|
||||||
kernel.verify()?;
|
|
||||||
let secp = secp.lock().unwrap();
|
|
||||||
sum_kernel = secp.commit_sum(vec![sum_kernel, kernel.excess], vec![])?;
|
|
||||||
kern_count += 1;
|
|
||||||
if kern_count == count {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Err(_) => break,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// now apply the kernel offset of we have one
|
|
||||||
{
|
|
||||||
let secp = secp.lock().unwrap();
|
|
||||||
if let Some(kernel_offset) = kernel_offset {
|
|
||||||
sum_kernel = secp.commit_sum(vec![sum_kernel, kernel_offset], vec![])?;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
debug!(
|
debug!(
|
||||||
LOGGER,
|
LOGGER,
|
||||||
|
@ -730,13 +716,12 @@ impl<'a> Extension<'a> {
|
||||||
let mut proof_count = 0;
|
let mut proof_count = 0;
|
||||||
for n in 1..self.output_pmmr.unpruned_size() + 1 {
|
for n in 1..self.output_pmmr.unpruned_size() + 1 {
|
||||||
if pmmr::is_leaf(n) {
|
if pmmr::is_leaf(n) {
|
||||||
if let Some((_, output)) = self.output_pmmr.get(n, true) {
|
if let Some((_, Some(out))) = self.output_pmmr.get(n, true) {
|
||||||
let out = output.expect("not a leaf node");
|
if let Some((_, Some(rp))) = self.rproof_pmmr.get(n, true) {
|
||||||
match self.rproof_pmmr.get(n, true) {
|
out.to_output(rp).verify_proof()?;
|
||||||
Some((_, Some(rp))) => out.to_output(rp).verify_proof()?,
|
} else {
|
||||||
_res => {
|
// TODO - rangeproof not found
|
||||||
return Err(Error::OutputNotFound);
|
return Err(Error::OutputNotFound);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
proof_count += 1;
|
proof_count += 1;
|
||||||
}
|
}
|
||||||
|
@ -756,33 +741,29 @@ impl<'a> Extension<'a> {
|
||||||
fn sum_outputs(&self) -> Result<Commitment, Error> {
|
fn sum_outputs(&self) -> Result<Commitment, Error> {
|
||||||
let now = Instant::now();
|
let now = Instant::now();
|
||||||
|
|
||||||
let mut sum_output = None;
|
let mut commitments = vec![];
|
||||||
let mut output_count = 0;
|
|
||||||
let secp = static_secp_instance();
|
|
||||||
for n in 1..self.output_pmmr.unpruned_size() + 1 {
|
for n in 1..self.output_pmmr.unpruned_size() + 1 {
|
||||||
if pmmr::is_leaf(n) {
|
if pmmr::is_leaf(n) {
|
||||||
if let Some((_, output)) = self.output_pmmr.get(n, true) {
|
if let Some((_, Some(out))) = self.output_pmmr.get(n, true) {
|
||||||
let out = output.expect("not a leaf node");
|
commitments.push(out.commit.clone());
|
||||||
let commit = out.commit.clone();
|
|
||||||
if let None = sum_output {
|
|
||||||
sum_output = Some(commit);
|
|
||||||
} else {
|
|
||||||
let secp = secp.lock().unwrap();
|
|
||||||
sum_output =
|
|
||||||
Some(secp.commit_sum(vec![sum_output.unwrap(), commit], vec![])?);
|
|
||||||
}
|
|
||||||
output_count += 1;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let secp = static_secp_instance();
|
||||||
|
let secp = secp.lock().unwrap();
|
||||||
|
let commit_count = commitments.len();
|
||||||
|
let sum_output = secp.commit_sum(commitments, vec![])?;
|
||||||
|
|
||||||
debug!(
|
debug!(
|
||||||
LOGGER,
|
LOGGER,
|
||||||
"Summed {} Outputs, pmmr size {}, took {}s",
|
"Summed {} Outputs, pmmr size {}, took {}s",
|
||||||
output_count,
|
commit_count,
|
||||||
self.output_pmmr.unpruned_size(),
|
self.output_pmmr.unpruned_size(),
|
||||||
now.elapsed().as_secs(),
|
now.elapsed().as_secs(),
|
||||||
);
|
);
|
||||||
Ok(sum_output.unwrap())
|
|
||||||
|
Ok(sum_output)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -118,13 +118,13 @@ fn data_files() {
|
||||||
.expect("previous block pmmr file data doesn't exist");
|
.expect("previous block pmmr file data doesn't exist");
|
||||||
|
|
||||||
println!("Cur_pmmr_md: {:?}", cur_pmmr_md);
|
println!("Cur_pmmr_md: {:?}", cur_pmmr_md);
|
||||||
chain.validate().unwrap();
|
chain.validate(false).unwrap();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// Now reload the chain, should have valid indices
|
// Now reload the chain, should have valid indices
|
||||||
{
|
{
|
||||||
let chain = reload_chain(chain_dir);
|
let chain = reload_chain(chain_dir);
|
||||||
chain.validate().unwrap();
|
chain.validate(false).unwrap();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -112,7 +112,7 @@ fn mine_empty_chain() {
|
||||||
let header_by_height = chain.get_header_by_height(n).unwrap();
|
let header_by_height = chain.get_header_by_height(n).unwrap();
|
||||||
assert_eq!(header_by_height.hash(), bhash);
|
assert_eq!(header_by_height.hash(), bhash);
|
||||||
|
|
||||||
chain.validate().unwrap();
|
chain.validate(false).unwrap();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -295,7 +295,7 @@ fn spend_in_fork_and_compact() {
|
||||||
chain
|
chain
|
||||||
.process_block(next.clone(), chain::Options::SKIP_POW)
|
.process_block(next.clone(), chain::Options::SKIP_POW)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
chain.validate().unwrap();
|
chain.validate(false).unwrap();
|
||||||
|
|
||||||
println!("tx 1 processed, should have 6 outputs or 396 bytes in file, first skipped");
|
println!("tx 1 processed, should have 6 outputs or 396 bytes in file, first skipped");
|
||||||
|
|
||||||
|
@ -311,7 +311,7 @@ fn spend_in_fork_and_compact() {
|
||||||
let next = prepare_block_tx(&kc, &prev_main, &chain, 9, vec![&tx2]);
|
let next = prepare_block_tx(&kc, &prev_main, &chain, 9, vec![&tx2]);
|
||||||
let prev_main = next.header.clone();
|
let prev_main = next.header.clone();
|
||||||
chain.process_block(next, chain::Options::SKIP_POW).unwrap();
|
chain.process_block(next, chain::Options::SKIP_POW).unwrap();
|
||||||
chain.validate().unwrap();
|
chain.validate(false).unwrap();
|
||||||
|
|
||||||
println!("tx 2 processed");
|
println!("tx 2 processed");
|
||||||
/* panic!("Stop"); */
|
/* panic!("Stop"); */
|
||||||
|
@ -326,7 +326,7 @@ fn spend_in_fork_and_compact() {
|
||||||
chain
|
chain
|
||||||
.process_block(fork_next, chain::Options::SKIP_POW)
|
.process_block(fork_next, chain::Options::SKIP_POW)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
chain.validate().unwrap();
|
chain.validate(false).unwrap();
|
||||||
|
|
||||||
// check state
|
// check state
|
||||||
let head = chain.head_header().unwrap();
|
let head = chain.head_header().unwrap();
|
||||||
|
@ -349,7 +349,7 @@ fn spend_in_fork_and_compact() {
|
||||||
chain
|
chain
|
||||||
.process_block(fork_next, chain::Options::SKIP_POW)
|
.process_block(fork_next, chain::Options::SKIP_POW)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
chain.validate().unwrap();
|
chain.validate(false).unwrap();
|
||||||
|
|
||||||
// check state
|
// check state
|
||||||
let head = chain.head_header().unwrap();
|
let head = chain.head_header().unwrap();
|
||||||
|
@ -374,9 +374,9 @@ fn spend_in_fork_and_compact() {
|
||||||
chain.process_block(next, chain::Options::SKIP_POW).unwrap();
|
chain.process_block(next, chain::Options::SKIP_POW).unwrap();
|
||||||
}
|
}
|
||||||
|
|
||||||
chain.validate().unwrap();
|
chain.validate(false).unwrap();
|
||||||
chain.compact().unwrap();
|
chain.compact().unwrap();
|
||||||
chain.validate().unwrap();
|
chain.validate(false).unwrap();
|
||||||
}
|
}
|
||||||
|
|
||||||
fn prepare_block(kc: &Keychain, prev: &BlockHeader, chain: &Chain, diff: u64) -> Block {
|
fn prepare_block(kc: &Keychain, prev: &BlockHeader, chain: &Chain, diff: u64) -> Block {
|
||||||
|
|
Loading…
Reference in a new issue