add manifestdata struct, write manifest file at the end

This commit is contained in:
Robert Habermeier 2016-06-10 17:19:55 +02:00
parent b047eb29a1
commit 3b1cca5622
2 changed files with 55 additions and 11 deletions

View File

@ -17,6 +17,7 @@
//! Blockchain database client. //! Blockchain database client.
use std::marker::PhantomData; use std::marker::PhantomData;
use std::fs::{create_dir, File};
use std::path::PathBuf; use std::path::PathBuf;
use util::*; use util::*;
use util::panics::*; use util::panics::*;
@ -755,33 +756,39 @@ impl<V> BlockChainClient for Client<V> where V: Verifier {
} }
fn take_snapshot(&self, root_dir: &Path) { fn take_snapshot(&self, root_dir: &Path) {
use pv64::BlockChunker; use pv64::{BlockChunker, ManifestData};
let best_header_bytes = self.best_block_header(); let best_header_bytes = self.best_block_header();
let best_header = HeaderView::new(&best_header_bytes); let best_header = HeaderView::new(&best_header_bytes);
let state_root = best_header.state_root();
trace!(target: "pv64_snapshot", "Taking snapshot starting at block {}", best_header.number()); trace!(target: "pv64_snapshot", "Taking snapshot starting at block {}", best_header.number());
let mut manifest_hashes = Vec::new();
// lock the state db while we create the state chunks. // lock the state db while we create the state chunks.
{ let state_hashes = {
let _state_db = self.state_db.lock().unwrap(); let _state_db = self.state_db.lock().unwrap();
let _state_root = best_header.state_root();
// todo [rob] actually create the state chunks. // todo [rob] actually create the state chunks.
}
Vec::new()
};
let best_hash = best_header.hash(); let best_hash = best_header.hash();
let genesis_hash = self.chain.genesis_hash(); let genesis_hash = self.chain.genesis_hash();
let mut path = root_dir.to_owned(); let mut path = root_dir.to_owned();
path.push("snapshot/"); path.push("snapshot/");
let _ = ::std::fs::create_dir(&path); let _ = create_dir(&path);
let block_chunk_hashes = BlockChunker::new(self, best_hash, genesis_hash).chunk_all(&path); let block_chunk_hashes = BlockChunker::new(self, best_hash, genesis_hash).chunk_all(&path);
for hash in block_chunk_hashes {
manifest_hashes.push(hash); let manifest_data = ManifestData {
} state_hashes: state_hashes,
block_hashes: block_chunk_hashes,
state_root: state_root,
};
let mut manifest_file = File::create("MANIFEST").unwrap();
manifest_file.write_all(&manifest_data.to_rlp()).unwrap();
} }
} }

View File

@ -30,7 +30,7 @@ use views::BlockView;
use util::{Bytes, Hashable}; use util::{Bytes, Hashable};
use util::hash::H256; use util::hash::H256;
use util::rlp::{Stream, RlpStream}; use util::rlp::{DecoderError, Stream, RlpStream, UntrustedRlp, View};
/// Used to build block chunks. /// Used to build block chunks.
pub struct BlockChunker<'a> { pub struct BlockChunker<'a> {
@ -123,4 +123,41 @@ impl<'a> BlockChunker<'a> {
chunk_hashes chunk_hashes
} }
}
/// Manifest data.
pub struct ManifestData {
/// List of state chunk hashes.
pub state_hashes: Vec<H256>,
/// List of block chunk hashes.
pub block_hashes: Vec<H256>,
/// The final, expected state root.
pub state_root: H256,
}
impl ManifestData {
/// Encode the manifest data to.
pub fn to_rlp(self) -> Bytes {
let mut stream = RlpStream::new_list(3);
stream.append(&self.state_hashes);
stream.append(&self.block_hashes);
stream.append(&self.state_root);
stream.out()
}
/// Try to restore manifest data from raw bytes interpreted as RLP.
pub fn from_rlp(raw: &[u8]) -> Result<Self, DecoderError> {
let decoder = UntrustedRlp::new(raw);
let state_hashes: Vec<H256> = try!(try!(decoder.at(0)).as_val());
let block_hashes: Vec<H256> = try!(try!(decoder.at(1)).as_val());
let state_root: H256 = try!(try!(decoder.at(2)).as_val());
Ok(ManifestData {
state_hashes: state_hashes,
block_hashes: block_hashes,
state_root: state_root,
})
}
} }