2018-03-05 22:33:44 +03:00
|
|
|
// Copyright 2018 The Grin Developers
|
2016-10-22 21:35:48 +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.
|
|
|
|
|
2016-10-21 03:06:12 +03:00
|
|
|
//! Implements storage primitives required by the chain
|
|
|
|
|
2018-04-23 21:55:25 +03:00
|
|
|
use std::sync::{Arc, RwLock};
|
|
|
|
|
2018-06-18 18:18:38 +03:00
|
|
|
use croaring::Bitmap;
|
2018-06-22 11:44:38 +03:00
|
|
|
use lmdb;
|
2018-04-23 21:55:25 +03:00
|
|
|
use lru_cache::LruCache;
|
2017-06-19 18:59:56 +03:00
|
|
|
|
2017-11-01 02:20:55 +03:00
|
|
|
use util::secp::pedersen::Commitment;
|
2017-04-20 21:26:53 +03:00
|
|
|
|
2017-06-19 18:59:56 +03:00
|
|
|
use core::consensus::TargetError;
|
2018-06-04 19:05:49 +03:00
|
|
|
use core::core::hash::{Hash, Hashed};
|
2018-09-20 11:19:32 +03:00
|
|
|
use core::core::{Block, BlockHeader, BlockSums};
|
2018-09-19 01:12:57 +03:00
|
|
|
use core::pow::Difficulty;
|
2018-06-22 11:08:06 +03:00
|
|
|
use grin_store as store;
|
2018-07-01 01:36:38 +03:00
|
|
|
use grin_store::{option_to_not_found, to_key, u64_to_key, Error};
|
2018-06-23 02:36:10 +03:00
|
|
|
use types::Tip;
|
2016-10-21 03:06:12 +03:00
|
|
|
|
2016-11-30 05:45:39 +03:00
|
|
|
const STORE_SUBPATH: &'static str = "chain";
|
2016-10-21 03:06:12 +03:00
|
|
|
|
2016-11-17 04:03:23 +03:00
|
|
|
const BLOCK_HEADER_PREFIX: u8 = 'h' as u8;
|
|
|
|
const BLOCK_PREFIX: u8 = 'b' as u8;
|
2016-10-21 03:06:12 +03:00
|
|
|
const HEAD_PREFIX: u8 = 'H' as u8;
|
2017-02-08 00:50:01 +03:00
|
|
|
const HEADER_HEAD_PREFIX: u8 = 'I' as u8;
|
2017-12-04 22:16:57 +03:00
|
|
|
const SYNC_HEAD_PREFIX: u8 = 's' as u8;
|
2017-02-08 00:50:01 +03:00
|
|
|
const HEADER_HEIGHT_PREFIX: u8 = '8' as u8;
|
2017-09-28 02:46:32 +03:00
|
|
|
const COMMIT_POS_PREFIX: u8 = 'c' as u8;
|
2018-06-18 18:18:38 +03:00
|
|
|
const BLOCK_INPUT_BITMAP_PREFIX: u8 = 'B' as u8;
|
2018-09-20 11:19:32 +03:00
|
|
|
const BLOCK_SUMS_PREFIX: u8 = 'M' as u8;
|
2016-10-21 03:06:12 +03:00
|
|
|
|
2018-06-22 11:08:06 +03:00
|
|
|
/// All chain-related database operations
|
|
|
|
pub struct ChainStore {
|
|
|
|
db: store::Store,
|
2018-04-23 21:55:25 +03:00
|
|
|
header_cache: Arc<RwLock<LruCache<Hash, BlockHeader>>>,
|
2018-06-22 13:44:50 +03:00
|
|
|
block_input_bitmap_cache: Arc<RwLock<LruCache<Hash, Vec<u8>>>>,
|
2016-10-21 03:06:12 +03:00
|
|
|
}
|
|
|
|
|
2018-06-22 11:08:06 +03:00
|
|
|
impl ChainStore {
|
2017-08-10 03:54:10 +03:00
|
|
|
/// Create new chain store
|
2018-06-22 11:08:06 +03:00
|
|
|
pub fn new(db_env: Arc<lmdb::Environment>) -> Result<ChainStore, Error> {
|
|
|
|
let db = store::Store::open(db_env, STORE_SUBPATH);
|
|
|
|
Ok(ChainStore {
|
2018-04-23 21:55:25 +03:00
|
|
|
db,
|
2018-06-18 18:18:38 +03:00
|
|
|
header_cache: Arc::new(RwLock::new(LruCache::new(1_000))),
|
2018-06-22 13:44:50 +03:00
|
|
|
block_input_bitmap_cache: Arc::new(RwLock::new(LruCache::new(1_000))),
|
2018-04-23 21:55:25 +03:00
|
|
|
})
|
2016-10-21 03:06:12 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-22 11:08:06 +03:00
|
|
|
#[allow(missing_docs)]
|
|
|
|
impl ChainStore {
|
|
|
|
pub fn head(&self) -> Result<Tip, Error> {
|
2018-07-01 01:36:38 +03:00
|
|
|
option_to_not_found(self.db.get_ser(&vec![HEAD_PREFIX]), "HEAD")
|
2016-10-21 03:06:12 +03:00
|
|
|
}
|
|
|
|
|
2018-10-05 10:29:33 +03:00
|
|
|
/// Header of the block at the head of the block chain (not the same thing as header_head).
|
2018-06-22 11:08:06 +03:00
|
|
|
pub fn head_header(&self) -> Result<BlockHeader, Error> {
|
2018-06-04 19:05:49 +03:00
|
|
|
self.get_block_header(&self.head()?.last_block_h)
|
2016-11-30 05:45:39 +03:00
|
|
|
}
|
2016-11-27 23:31:15 +03:00
|
|
|
|
2018-10-05 10:29:33 +03:00
|
|
|
/// Head of the header chain (not the same thing as head_header).
|
|
|
|
pub fn header_head(&self) -> Result<Tip, Error> {
|
2018-07-01 01:36:38 +03:00
|
|
|
option_to_not_found(self.db.get_ser(&vec![HEADER_HEAD_PREFIX]), "HEADER_HEAD")
|
2017-02-08 00:50:01 +03:00
|
|
|
}
|
|
|
|
|
2018-06-22 11:08:06 +03:00
|
|
|
pub fn get_sync_head(&self) -> Result<Tip, Error> {
|
2018-07-01 01:36:38 +03:00
|
|
|
option_to_not_found(self.db.get_ser(&vec![SYNC_HEAD_PREFIX]), "SYNC_HEAD")
|
2017-12-04 22:16:57 +03:00
|
|
|
}
|
|
|
|
|
2018-06-22 11:08:06 +03:00
|
|
|
pub fn get_block(&self, h: &Hash) -> Result<Block, Error> {
|
2018-07-01 01:36:38 +03:00
|
|
|
option_to_not_found(
|
|
|
|
self.db.get_ser(&to_key(BLOCK_PREFIX, &mut h.to_vec())),
|
2018-07-05 15:18:09 +03:00
|
|
|
&format!("BLOCK: {}", h),
|
2018-07-01 01:36:38 +03:00
|
|
|
)
|
2017-02-08 00:50:01 +03:00
|
|
|
}
|
|
|
|
|
2018-06-22 11:08:06 +03:00
|
|
|
pub fn block_exists(&self, h: &Hash) -> Result<bool, Error> {
|
2018-02-10 01:32:16 +03:00
|
|
|
self.db.exists(&to_key(BLOCK_PREFIX, &mut h.to_vec()))
|
|
|
|
}
|
|
|
|
|
2018-09-24 11:24:10 +03:00
|
|
|
pub fn get_block_sums(&self, bh: &Hash) -> Result<BlockSums, Error> {
|
|
|
|
option_to_not_found(
|
|
|
|
self.db
|
|
|
|
.get_ser(&to_key(BLOCK_SUMS_PREFIX, &mut bh.to_vec())),
|
|
|
|
&format!("Block sums for block: {}", bh),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2018-06-22 11:08:06 +03:00
|
|
|
pub fn get_block_header(&self, h: &Hash) -> Result<BlockHeader, Error> {
|
2018-04-23 21:55:25 +03:00
|
|
|
{
|
|
|
|
let mut header_cache = self.header_cache.write().unwrap();
|
|
|
|
|
|
|
|
// cache hit - return the value from the cache
|
|
|
|
if let Some(header) = header_cache.get_mut(h) {
|
|
|
|
return Ok(header.clone());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-22 11:08:06 +03:00
|
|
|
let header: Result<BlockHeader, Error> = option_to_not_found(
|
|
|
|
self.db
|
|
|
|
.get_ser(&to_key(BLOCK_HEADER_PREFIX, &mut h.to_vec())),
|
2018-07-01 01:36:38 +03:00
|
|
|
&format!("BLOCK HEADER: {}", h),
|
2018-06-22 11:08:06 +03:00
|
|
|
);
|
2018-02-17 20:56:22 +03:00
|
|
|
|
2018-06-22 11:08:06 +03:00
|
|
|
// cache miss - so adding to the cache for next time
|
|
|
|
if let Ok(header) = header {
|
|
|
|
{
|
|
|
|
let mut header_cache = self.header_cache.write().unwrap();
|
|
|
|
header_cache.insert(*h, header.clone());
|
|
|
|
}
|
|
|
|
Ok(header)
|
|
|
|
} else {
|
|
|
|
header
|
|
|
|
}
|
2018-02-15 02:25:12 +03:00
|
|
|
}
|
2017-02-08 00:50:01 +03:00
|
|
|
|
2018-09-15 02:51:36 +03:00
|
|
|
pub fn get_hash_by_height(&self, height: u64) -> Result<Hash, Error> {
|
|
|
|
option_to_not_found(
|
|
|
|
self.db.get_ser(&u64_to_key(HEADER_HEIGHT_PREFIX, height)),
|
|
|
|
&format!("Hash at height: {}", height),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2018-10-09 15:37:06 +03:00
|
|
|
// We are on the current chain if -
|
|
|
|
// * the header by height index matches the header, and
|
|
|
|
// * we are not ahead of the current head
|
|
|
|
pub fn is_on_current_chain(&self, header: &BlockHeader) -> Result<(), Error> {
|
|
|
|
let head = self.head()?;
|
|
|
|
|
|
|
|
// check we are not out ahead of the current head
|
|
|
|
if header.height > head.height {
|
|
|
|
return Err(Error::NotFoundErr(String::from(
|
|
|
|
"header.height > head.height",
|
|
|
|
)));
|
|
|
|
}
|
|
|
|
|
|
|
|
let header_at_height = self.get_header_by_height(header.height)?;
|
|
|
|
if header.hash() == header_at_height.hash() {
|
|
|
|
Ok(())
|
|
|
|
} else {
|
|
|
|
Err(Error::NotFoundErr(String::from(
|
|
|
|
"header.hash == header_at_height.hash",
|
|
|
|
)))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-22 11:08:06 +03:00
|
|
|
pub fn get_header_by_height(&self, height: u64) -> Result<BlockHeader, Error> {
|
2018-07-01 01:36:38 +03:00
|
|
|
option_to_not_found(
|
|
|
|
self.db.get_ser(&u64_to_key(HEADER_HEIGHT_PREFIX, height)),
|
|
|
|
&format!("Header at height: {}", height),
|
|
|
|
).and_then(|hash| self.get_block_header(&hash))
|
2018-06-22 11:08:06 +03:00
|
|
|
}
|
2018-06-18 18:18:38 +03:00
|
|
|
|
2018-06-22 11:08:06 +03:00
|
|
|
pub fn get_output_pos(&self, commit: &Commitment) -> Result<u64, Error> {
|
|
|
|
option_to_not_found(
|
|
|
|
self.db
|
|
|
|
.get_ser(&to_key(COMMIT_POS_PREFIX, &mut commit.as_ref().to_vec())),
|
2018-07-01 01:36:38 +03:00
|
|
|
&format!("Output position for: {:?}", commit),
|
2018-06-22 11:08:06 +03:00
|
|
|
)
|
|
|
|
}
|
2018-06-18 18:18:38 +03:00
|
|
|
|
2018-06-22 11:08:06 +03:00
|
|
|
/// Builds a new batch to be used with this store.
|
|
|
|
pub fn batch(&self) -> Result<Batch, Error> {
|
|
|
|
Ok(Batch {
|
|
|
|
store: self,
|
|
|
|
db: self.db.batch()?,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// An atomic batch in which all changes can be committed all at once or
|
|
|
|
/// discarded on error.
|
|
|
|
pub struct Batch<'a> {
|
|
|
|
store: &'a ChainStore,
|
|
|
|
db: store::Batch<'a>,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[allow(missing_docs)]
|
|
|
|
impl<'a> Batch<'a> {
|
2018-09-25 13:01:19 +03:00
|
|
|
pub fn head(&self) -> Result<Tip, Error> {
|
|
|
|
option_to_not_found(self.db.get_ser(&vec![HEAD_PREFIX]), "HEAD")
|
|
|
|
}
|
|
|
|
|
2018-10-05 10:29:33 +03:00
|
|
|
/// Header of the block at the head of the block chain (not the same thing as header_head).
|
2018-09-25 13:01:19 +03:00
|
|
|
pub fn head_header(&self) -> Result<BlockHeader, Error> {
|
|
|
|
self.get_block_header(&self.head()?.last_block_h)
|
|
|
|
}
|
|
|
|
|
2018-10-05 10:29:33 +03:00
|
|
|
/// Head of the header chain (not the same thing as head_header).
|
|
|
|
pub fn header_head(&self) -> Result<Tip, Error> {
|
2018-09-27 11:35:25 +03:00
|
|
|
option_to_not_found(self.db.get_ser(&vec![HEADER_HEAD_PREFIX]), "HEADER_HEAD")
|
|
|
|
}
|
|
|
|
|
2018-09-27 13:44:50 +03:00
|
|
|
pub fn get_sync_head(&self) -> Result<Tip, Error> {
|
|
|
|
option_to_not_found(self.db.get_ser(&vec![SYNC_HEAD_PREFIX]), "SYNC_HEAD")
|
|
|
|
}
|
|
|
|
|
2018-06-22 11:08:06 +03:00
|
|
|
pub fn save_head(&self, t: &Tip) -> Result<(), Error> {
|
|
|
|
self.db.put_ser(&vec![HEAD_PREFIX], t)?;
|
|
|
|
self.db.put_ser(&vec![HEADER_HEAD_PREFIX], t)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn save_body_head(&self, t: &Tip) -> Result<(), Error> {
|
|
|
|
self.db.put_ser(&vec![HEAD_PREFIX], t)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn save_header_head(&self, t: &Tip) -> Result<(), Error> {
|
|
|
|
self.db.put_ser(&vec![HEADER_HEAD_PREFIX], t)
|
|
|
|
}
|
|
|
|
|
2018-09-25 13:01:19 +03:00
|
|
|
pub fn get_hash_by_height(&self, height: u64) -> Result<Hash, Error> {
|
|
|
|
option_to_not_found(
|
|
|
|
self.db.get_ser(&u64_to_key(HEADER_HEIGHT_PREFIX, height)),
|
|
|
|
&format!("Hash at height: {}", height),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2018-06-22 11:08:06 +03:00
|
|
|
pub fn save_sync_head(&self, t: &Tip) -> Result<(), Error> {
|
|
|
|
self.db.put_ser(&vec![SYNC_HEAD_PREFIX], t)
|
|
|
|
}
|
|
|
|
|
2018-10-01 18:03:06 +03:00
|
|
|
pub fn reset_sync_head(&self) -> Result<(), Error> {
|
2018-10-05 10:29:33 +03:00
|
|
|
let head = self.header_head()?;
|
2018-10-01 18:03:06 +03:00
|
|
|
self.save_sync_head(&head)
|
2018-06-22 11:08:06 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Reset both header_head and sync_head to the current head of the body chain
|
|
|
|
pub fn reset_head(&self) -> Result<(), Error> {
|
2018-10-01 18:03:06 +03:00
|
|
|
let tip = self.head()?;
|
2018-06-22 11:08:06 +03:00
|
|
|
self.save_header_head(&tip)?;
|
|
|
|
self.save_sync_head(&tip)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// get block
|
2018-06-22 16:04:51 +03:00
|
|
|
pub fn get_block(&self, h: &Hash) -> Result<Block, Error> {
|
2018-07-01 01:36:38 +03:00
|
|
|
option_to_not_found(
|
|
|
|
self.db.get_ser(&to_key(BLOCK_PREFIX, &mut h.to_vec())),
|
|
|
|
&format!("Block with hash: {}", h),
|
|
|
|
)
|
2018-06-22 11:08:06 +03:00
|
|
|
}
|
|
|
|
|
2018-09-27 11:35:25 +03:00
|
|
|
pub fn block_exists(&self, h: &Hash) -> Result<bool, Error> {
|
|
|
|
self.db.exists(&to_key(BLOCK_PREFIX, &mut h.to_vec()))
|
|
|
|
}
|
|
|
|
|
2018-06-22 11:08:06 +03:00
|
|
|
/// Save the block and its header
|
|
|
|
pub fn save_block(&self, b: &Block) -> Result<(), Error> {
|
|
|
|
self.db
|
|
|
|
.put_ser(&to_key(BLOCK_PREFIX, &mut b.hash().to_vec())[..], b)?;
|
|
|
|
self.db.put_ser(
|
|
|
|
&to_key(BLOCK_HEADER_PREFIX, &mut b.hash().to_vec())[..],
|
|
|
|
&b.header,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Delete a full block. Does not delete any record associated with a block
|
|
|
|
/// header.
|
|
|
|
pub fn delete_block(&self, bh: &Hash) -> Result<(), Error> {
|
2018-10-02 18:17:15 +03:00
|
|
|
self.db
|
|
|
|
.delete(&to_key(BLOCK_PREFIX, &mut bh.to_vec())[..])?;
|
|
|
|
|
|
|
|
// Best effort at deleting associated data for this block.
|
|
|
|
// Not an error if these fail.
|
|
|
|
{
|
|
|
|
let _ = self.delete_block_sums(bh);
|
|
|
|
let _ = self.delete_block_input_bitmap(bh);
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
2018-06-22 11:08:06 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn save_block_header(&self, bh: &BlockHeader) -> Result<(), Error> {
|
|
|
|
let hash = bh.hash();
|
|
|
|
self.db
|
|
|
|
.put_ser(&to_key(BLOCK_HEADER_PREFIX, &mut hash.to_vec())[..], bh)?;
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn save_header_height(&self, bh: &BlockHeader) -> Result<(), Error> {
|
2018-02-10 01:32:16 +03:00
|
|
|
self.db
|
2018-03-09 21:59:15 +03:00
|
|
|
.put_ser(&u64_to_key(HEADER_HEIGHT_PREFIX, bh.height), &bh.hash())
|
2018-02-10 01:32:16 +03:00
|
|
|
}
|
|
|
|
|
2018-06-22 11:08:06 +03:00
|
|
|
pub fn delete_header_by_height(&self, height: u64) -> Result<(), Error> {
|
2017-12-16 03:27:37 +03:00
|
|
|
self.db.delete(&u64_to_key(HEADER_HEIGHT_PREFIX, height))
|
|
|
|
}
|
|
|
|
|
2018-06-22 11:08:06 +03:00
|
|
|
pub fn save_output_pos(&self, commit: &Commitment, pos: u64) -> Result<(), Error> {
|
2017-09-29 21:44:25 +03:00
|
|
|
self.db.put_ser(
|
|
|
|
&to_key(COMMIT_POS_PREFIX, &mut commit.as_ref().to_vec())[..],
|
|
|
|
&pos,
|
|
|
|
)
|
2017-09-28 02:46:32 +03:00
|
|
|
}
|
|
|
|
|
2018-06-22 11:08:06 +03:00
|
|
|
pub fn get_output_pos(&self, commit: &Commitment) -> Result<u64, Error> {
|
2017-11-01 02:32:33 +03:00
|
|
|
option_to_not_found(
|
|
|
|
self.db
|
|
|
|
.get_ser(&to_key(COMMIT_POS_PREFIX, &mut commit.as_ref().to_vec())),
|
2018-07-01 01:36:38 +03:00
|
|
|
&format!("Output position for commit: {:?}", commit),
|
2017-11-01 02:32:33 +03:00
|
|
|
)
|
2017-09-28 02:46:32 +03:00
|
|
|
}
|
|
|
|
|
2018-06-22 11:08:06 +03:00
|
|
|
pub fn delete_output_pos(&self, commit: &[u8]) -> Result<(), Error> {
|
2018-03-06 20:58:33 +03:00
|
|
|
self.db
|
|
|
|
.delete(&to_key(COMMIT_POS_PREFIX, &mut commit.to_vec()))
|
|
|
|
}
|
|
|
|
|
2018-09-20 11:19:32 +03:00
|
|
|
pub fn get_block_header(&self, h: &Hash) -> Result<BlockHeader, Error> {
|
2017-11-01 02:32:33 +03:00
|
|
|
option_to_not_found(
|
|
|
|
self.db
|
2018-06-22 11:08:06 +03:00
|
|
|
.get_ser(&to_key(BLOCK_HEADER_PREFIX, &mut h.to_vec())),
|
2018-07-01 01:36:38 +03:00
|
|
|
&format!("Block header for block: {}", h),
|
2017-11-01 02:32:33 +03:00
|
|
|
)
|
2017-04-20 21:26:53 +03:00
|
|
|
}
|
|
|
|
|
2018-08-20 22:34:12 +03:00
|
|
|
fn save_block_input_bitmap(&self, bh: &Hash, bm: &Bitmap) -> Result<(), Error> {
|
2018-06-22 13:45:25 +03:00
|
|
|
self.db.put(
|
|
|
|
&to_key(BLOCK_INPUT_BITMAP_PREFIX, &mut bh.to_vec())[..],
|
|
|
|
bm.serialize(),
|
|
|
|
)
|
2018-06-18 18:18:38 +03:00
|
|
|
}
|
|
|
|
|
2018-10-02 18:17:15 +03:00
|
|
|
fn delete_block_input_bitmap(&self, bh: &Hash) -> Result<(), Error> {
|
2018-06-18 18:18:38 +03:00
|
|
|
self.db
|
|
|
|
.delete(&to_key(BLOCK_INPUT_BITMAP_PREFIX, &mut bh.to_vec()))
|
2018-03-09 00:36:51 +03:00
|
|
|
}
|
|
|
|
|
2018-09-20 11:19:32 +03:00
|
|
|
pub fn save_block_sums(&self, bh: &Hash, sums: &BlockSums) -> Result<(), Error> {
|
|
|
|
self.db
|
|
|
|
.put_ser(&to_key(BLOCK_SUMS_PREFIX, &mut bh.to_vec())[..], &sums)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn get_block_sums(&self, bh: &Hash) -> Result<BlockSums, Error> {
|
|
|
|
option_to_not_found(
|
|
|
|
self.db
|
|
|
|
.get_ser(&to_key(BLOCK_SUMS_PREFIX, &mut bh.to_vec())),
|
|
|
|
&format!("Block sums for block: {}", bh),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2018-10-02 18:17:15 +03:00
|
|
|
fn delete_block_sums(&self, bh: &Hash) -> Result<(), Error> {
|
2018-09-20 11:19:32 +03:00
|
|
|
self.db.delete(&to_key(BLOCK_SUMS_PREFIX, &mut bh.to_vec()))
|
|
|
|
}
|
|
|
|
|
2018-09-27 11:35:25 +03:00
|
|
|
// We are on the current chain if -
|
|
|
|
// * the header by height index matches the header, and
|
|
|
|
// * we are not ahead of the current head
|
|
|
|
pub fn is_on_current_chain(&self, header: &BlockHeader) -> Result<(), Error> {
|
|
|
|
let head = self.head()?;
|
|
|
|
|
|
|
|
// check we are not out ahead of the current head
|
|
|
|
if header.height > head.height {
|
|
|
|
return Err(Error::NotFoundErr(String::from(
|
|
|
|
"header.height > head.height",
|
|
|
|
)));
|
|
|
|
}
|
|
|
|
|
|
|
|
let header_at_height = self.get_header_by_height(header.height)?;
|
|
|
|
if header.hash() == header_at_height.hash() {
|
|
|
|
Ok(())
|
|
|
|
} else {
|
|
|
|
Err(Error::NotFoundErr(String::from(
|
|
|
|
"header.hash == header_at_height.hash",
|
|
|
|
)))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-25 13:01:19 +03:00
|
|
|
pub fn get_header_by_height(&self, height: u64) -> Result<BlockHeader, Error> {
|
|
|
|
option_to_not_found(
|
|
|
|
self.db.get_ser(&u64_to_key(HEADER_HEIGHT_PREFIX, height)),
|
|
|
|
&format!("Header at height: {}", height),
|
|
|
|
).and_then(|hash| self.get_block_header(&hash))
|
|
|
|
}
|
|
|
|
|
2017-09-29 21:44:25 +03:00
|
|
|
/// Maintain consistency of the "header_by_height" index by traversing back
|
2017-12-16 03:27:37 +03:00
|
|
|
/// through the current chain and updating "header_by_height" until we reach
|
|
|
|
/// a block_header that is consistent with its height (everything prior to
|
|
|
|
/// this will be consistent).
|
|
|
|
/// We need to handle the case where we have no index entry for a given
|
|
|
|
/// height to account for the case where we just switched to a new fork and
|
|
|
|
/// the height jumped beyond current chain height.
|
2018-06-22 11:08:06 +03:00
|
|
|
pub fn setup_height(&self, header: &BlockHeader, old_tip: &Tip) -> Result<(), Error> {
|
2017-12-16 03:27:37 +03:00
|
|
|
// remove headers ahead if we backtracked
|
|
|
|
for n in header.height..old_tip.height {
|
2018-04-21 04:11:41 +03:00
|
|
|
self.delete_header_by_height(n + 1)?;
|
2017-12-16 03:27:37 +03:00
|
|
|
}
|
2018-04-03 02:10:30 +03:00
|
|
|
self.build_by_height_index(header, false)
|
|
|
|
}
|
2017-12-16 03:27:37 +03:00
|
|
|
|
2018-06-22 11:08:06 +03:00
|
|
|
pub fn build_by_height_index(&self, header: &BlockHeader, force: bool) -> Result<(), Error> {
|
2018-03-09 21:59:15 +03:00
|
|
|
self.save_header_height(&header)?;
|
2017-12-05 21:32:57 +03:00
|
|
|
|
2017-12-16 03:27:37 +03:00
|
|
|
if header.height > 0 {
|
2018-06-22 11:08:06 +03:00
|
|
|
let mut prev_header = self.store.get_block_header(&header.previous)?;
|
2017-12-16 03:27:37 +03:00
|
|
|
while prev_header.height > 0 {
|
2018-04-03 02:10:30 +03:00
|
|
|
if !force {
|
2018-09-27 11:35:25 +03:00
|
|
|
if let Ok(_) = self.is_on_current_chain(&prev_header) {
|
2018-04-03 02:10:30 +03:00
|
|
|
break;
|
|
|
|
}
|
2017-12-16 03:27:37 +03:00
|
|
|
}
|
2018-03-09 21:59:15 +03:00
|
|
|
self.save_header_height(&prev_header)?;
|
2017-02-08 00:50:01 +03:00
|
|
|
|
2018-06-22 11:08:06 +03:00
|
|
|
prev_header = self.store.get_block_header(&prev_header.previous)?;
|
2017-12-16 03:27:37 +03:00
|
|
|
}
|
2017-02-08 00:50:01 +03:00
|
|
|
}
|
2017-12-16 03:27:37 +03:00
|
|
|
Ok(())
|
2017-02-08 00:50:01 +03:00
|
|
|
}
|
2018-06-22 11:08:06 +03:00
|
|
|
|
2018-08-20 22:34:12 +03:00
|
|
|
fn build_block_input_bitmap(&self, block: &Block) -> Result<Bitmap, Error> {
|
|
|
|
let bitmap = block
|
|
|
|
.inputs()
|
|
|
|
.iter()
|
|
|
|
.filter_map(|x| self.get_output_pos(&x.commitment()).ok())
|
|
|
|
.map(|x| x as u32)
|
|
|
|
.collect();
|
|
|
|
Ok(bitmap)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn build_and_cache_block_input_bitmap(&self, block: &Block) -> Result<Bitmap, Error> {
|
|
|
|
// Build the bitmap.
|
|
|
|
let bitmap = self.build_block_input_bitmap(block)?;
|
|
|
|
|
|
|
|
// Save the bitmap to the db (via the batch).
|
|
|
|
self.save_block_input_bitmap(&block.hash(), &bitmap)?;
|
|
|
|
|
|
|
|
// Finally cache it locally for use later.
|
|
|
|
let mut cache = self.store.block_input_bitmap_cache.write().unwrap();
|
|
|
|
cache.insert(block.hash(), bitmap.serialize());
|
|
|
|
|
|
|
|
Ok(bitmap)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn get_block_input_bitmap(&self, bh: &Hash) -> Result<Bitmap, Error> {
|
|
|
|
{
|
|
|
|
let mut cache = self.store.block_input_bitmap_cache.write().unwrap();
|
|
|
|
|
|
|
|
// cache hit - return the value from the cache
|
|
|
|
if let Some(bytes) = cache.get_mut(bh) {
|
|
|
|
return Ok(Bitmap::deserialize(&bytes));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// cache miss - get it from db (build it, store it and cache it as necessary)
|
|
|
|
self.get_block_input_bitmap_db(bh)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the block input bitmap from the db or build the bitmap from
|
|
|
|
// the full block from the db (if the block is found).
|
|
|
|
// (bool, Bitmap) : (false if bitmap was built and not found in db)
|
|
|
|
fn get_block_input_bitmap_db(&self, bh: &Hash) -> Result<Bitmap, Error> {
|
|
|
|
if let Ok(Some(bytes)) = self
|
|
|
|
.db
|
|
|
|
.get(&to_key(BLOCK_INPUT_BITMAP_PREFIX, &mut bh.to_vec()))
|
|
|
|
{
|
|
|
|
Ok(Bitmap::deserialize(&bytes))
|
|
|
|
} else {
|
|
|
|
match self.get_block(bh) {
|
|
|
|
Ok(block) => {
|
|
|
|
let bitmap = self.build_and_cache_block_input_bitmap(&block)?;
|
|
|
|
Ok(bitmap)
|
|
|
|
}
|
|
|
|
Err(e) => Err(e),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-22 11:08:06 +03:00
|
|
|
/// Commits this batch. If it's a child batch, it will be merged with the
|
|
|
|
/// parent, otherwise the batch is written to db.
|
|
|
|
pub fn commit(self) -> Result<(), Error> {
|
|
|
|
self.db.commit()
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Creates a child of this batch. It will be merged with its parent on
|
|
|
|
/// commit, abandoned otherwise.
|
|
|
|
pub fn child(&mut self) -> Result<Batch, Error> {
|
|
|
|
Ok(Batch {
|
|
|
|
store: self.store,
|
|
|
|
db: self.db.child()?,
|
|
|
|
})
|
|
|
|
}
|
2017-02-08 00:50:01 +03:00
|
|
|
}
|
2017-06-19 18:59:56 +03:00
|
|
|
|
|
|
|
/// An iterator on blocks, from latest to earliest, specialized to return
|
|
|
|
/// information pertaining to block difficulty calculation (timestamp and
|
|
|
|
/// previous difficulties). Mostly used by the consensus next difficulty
|
|
|
|
/// calculation.
|
2018-09-27 11:35:25 +03:00
|
|
|
pub struct DifficultyIter<'a> {
|
2018-03-15 22:16:34 +03:00
|
|
|
start: Hash,
|
2018-09-27 11:35:25 +03:00
|
|
|
batch: Batch<'a>,
|
2018-03-15 22:16:34 +03:00
|
|
|
|
|
|
|
// maintain state for both the "next" header in this iteration
|
|
|
|
// and its previous header in the chain ("next next" in the iteration)
|
|
|
|
// so we effectively read-ahead as we iterate through the chain back
|
|
|
|
// toward the genesis block (while maintaining current state)
|
|
|
|
header: Option<BlockHeader>,
|
|
|
|
prev_header: Option<BlockHeader>,
|
2017-06-19 18:59:56 +03:00
|
|
|
}
|
|
|
|
|
2018-09-27 11:35:25 +03:00
|
|
|
impl<'a> DifficultyIter<'a> {
|
2017-07-04 02:46:25 +03:00
|
|
|
/// Build a new iterator using the provided chain store and starting from
|
|
|
|
/// the provided block hash.
|
2018-09-27 11:35:25 +03:00
|
|
|
pub fn from(start: Hash, batch: Batch) -> DifficultyIter {
|
2017-06-19 18:59:56 +03:00
|
|
|
DifficultyIter {
|
2018-09-27 11:35:25 +03:00
|
|
|
start,
|
|
|
|
batch,
|
2018-03-15 22:16:34 +03:00
|
|
|
header: None,
|
|
|
|
prev_header: None,
|
2017-06-19 18:59:56 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-27 11:35:25 +03:00
|
|
|
impl<'a> Iterator for DifficultyIter<'a> {
|
2017-08-03 19:57:55 +03:00
|
|
|
type Item = Result<(u64, Difficulty), TargetError>;
|
2017-06-19 18:59:56 +03:00
|
|
|
|
|
|
|
fn next(&mut self) -> Option<Self::Item> {
|
2018-03-15 22:16:34 +03:00
|
|
|
// Get both header and previous_header if this is the initial iteration.
|
|
|
|
// Otherwise move prev_header to header and get the next prev_header.
|
|
|
|
self.header = if self.header.is_none() {
|
2018-09-27 11:35:25 +03:00
|
|
|
self.batch.get_block_header(&self.start).ok()
|
2018-03-15 22:16:34 +03:00
|
|
|
} else {
|
|
|
|
self.prev_header.clone()
|
|
|
|
};
|
|
|
|
|
|
|
|
// If we have a header we can do this iteration.
|
|
|
|
// Otherwise we are done.
|
|
|
|
if let Some(header) = self.header.clone() {
|
2018-09-27 11:35:25 +03:00
|
|
|
self.prev_header = self.batch.get_block_header(&header.previous).ok();
|
2018-03-15 22:16:34 +03:00
|
|
|
|
2018-08-10 16:56:35 +03:00
|
|
|
let prev_difficulty = self
|
|
|
|
.prev_header
|
2018-03-15 22:16:34 +03:00
|
|
|
.clone()
|
2018-09-11 01:36:57 +03:00
|
|
|
.map_or(Difficulty::zero(), |x| x.total_difficulty());
|
|
|
|
let difficulty = header.total_difficulty() - prev_difficulty;
|
2018-03-15 22:16:34 +03:00
|
|
|
|
2018-07-30 11:33:28 +03:00
|
|
|
Some(Ok((header.timestamp.timestamp() as u64, difficulty)))
|
2018-03-15 22:16:34 +03:00
|
|
|
} else {
|
|
|
|
return None;
|
2017-06-19 18:59:56 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|