Generate txhashset archives on 720 block intervals. (#2813)

* generate txhashset archives on 250 block intervals.

* moved txhashset_archive_interval to global and added a simple test.

* cleaning up the tests and adding license.

* increasing cleanup duration to 24 hours to prevent premature deletion of the current txhashset archive

* bug fixes and changing request_state to request height using archive_interval.

* removing stopstate from chain_test_helper to fix compile issue
This commit is contained in:
Cadmus Peverell 2019-06-06 02:39:07 +00:00 committed by Gary Yu
parent dfb4d5afae
commit 5ebe2aa397
13 changed files with 220 additions and 13 deletions

View file

@ -685,6 +685,27 @@ impl Chain {
)) ))
} }
/// To support the ability to download the txhashset from multiple peers in parallel,
/// the peers must all agree on the exact binary representation of the txhashset.
/// This means compacting and rewinding to the exact same header.
/// Since compaction is a heavy operation, peers can agree to compact every 12 hours,
/// and no longer support requesting arbitrary txhashsets.
/// Here we return the header of the txhashset we are currently offering to peers.
pub fn txhashset_archive_header(&self) -> Result<BlockHeader, Error> {
let sync_threshold = global::state_sync_threshold() as u64;
let body_head = self.head()?;
let archive_interval = global::txhashset_archive_interval();
let mut txhashset_height = body_head.height.saturating_sub(sync_threshold);
txhashset_height = txhashset_height.saturating_sub(txhashset_height % archive_interval);
debug!(
"txhashset_archive_header: body_head - {}, {}, txhashset height - {}",
body_head.last_block_h, body_head.height, txhashset_height,
);
self.get_header_by_height(txhashset_height)
}
// Special handling to make sure the whole kernel set matches each of its // Special handling to make sure the whole kernel set matches each of its
// roots in each block header, without truncation. We go back header by // roots in each block header, without truncation. We go back header by
// header, rewind and check each root. This fixes a potential weakness in // header, rewind and check each root. This fixes a potential weakness in

View file

@ -1432,10 +1432,10 @@ pub fn zip_read(root_dir: String, header: &BlockHeader) -> Result<File, Error> {
} else { } else {
// clean up old zips. // clean up old zips.
// Theoretically, we only need clean-up those zip files older than STATE_SYNC_THRESHOLD. // Theoretically, we only need clean-up those zip files older than STATE_SYNC_THRESHOLD.
// But practically, these zip files are not small ones, we just keep the zips in last one hour // But practically, these zip files are not small ones, we just keep the zips in last 24 hours
let data_dir = Path::new(&root_dir); let data_dir = Path::new(&root_dir);
let pattern = format!("{}_", TXHASHSET_ZIP); let pattern = format!("{}_", TXHASHSET_ZIP);
if let Ok(n) = clean_files_by_prefix(data_dir.clone(), &pattern, 60 * 60) { if let Ok(n) = clean_files_by_prefix(data_dir.clone(), &pattern, 24 * 60 * 60) {
debug!( debug!(
"{} zip files have been clean up in folder: {:?}", "{} zip files have been clean up in folder: {:?}",
n, data_dir n, data_dir

View file

@ -0,0 +1,118 @@
// Copyright 2018 The Grin Developers
//
// 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.
use self::chain::types::NoopAdapter;
use self::chain::types::Options;
use self::chain::Chain;
use self::core::core::verifier_cache::LruVerifierCache;
use self::core::core::Block;
use self::core::genesis;
use self::core::global::ChainTypes;
use self::core::libtx::{self, reward};
use self::core::pow::Difficulty;
use self::core::{consensus, global, pow};
use self::keychain::{ExtKeychainPath, Keychain};
use self::util::RwLock;
use chrono::Duration;
use grin_chain as chain;
use grin_core as core;
use grin_keychain as keychain;
use grin_util as util;
use std::fs;
use std::sync::Arc;
pub fn clean_output_dir(dir_name: &str) {
let _ = fs::remove_dir_all(dir_name);
}
pub fn setup(dir_name: &str, genesis: Block) -> Chain {
util::init_test_logger();
clean_output_dir(dir_name);
let verifier_cache = Arc::new(RwLock::new(LruVerifierCache::new()));
Chain::init(
dir_name.to_string(),
Arc::new(NoopAdapter {}),
genesis,
pow::verify_size,
verifier_cache,
false,
)
.unwrap()
}
/// Mine a chain of specified length to assist with automated tests.
/// Must call clean_output_dir at the end of your test.
pub fn mine_chain(dir_name: &str, chain_length: u64) -> Chain {
global::set_mining_mode(ChainTypes::AutomatedTesting);
// add coinbase data from the dev genesis block
let mut genesis = genesis::genesis_dev();
let keychain = keychain::ExtKeychain::from_random_seed(false).unwrap();
let key_id = keychain::ExtKeychain::derive_key_id(0, 1, 0, 0, 0);
let reward = reward::output(&keychain, &key_id, 0, false).unwrap();
genesis = genesis.with_reward(reward.0, reward.1);
let mut chain = setup(dir_name, pow::mine_genesis_block().unwrap());
chain.set_txhashset_roots(&mut genesis).unwrap();
genesis.header.output_mmr_size = 1;
genesis.header.kernel_mmr_size = 1;
// get a valid PoW
pow::pow_size(
&mut genesis.header,
Difficulty::unit(),
global::proofsize(),
global::min_edge_bits(),
)
.unwrap();
mine_some_on_top(&mut chain, chain_length, &keychain);
chain
}
fn mine_some_on_top<K>(chain: &mut Chain, chain_length: u64, keychain: &K)
where
K: Keychain,
{
for n in 1..chain_length {
let prev = chain.head_header().unwrap();
let next_header_info = consensus::next_difficulty(1, chain.difficulty_iter().unwrap());
let pk = ExtKeychainPath::new(1, n as u32, 0, 0, 0).to_identifier();
let reward = libtx::reward::output(keychain, &pk, 0, false).unwrap();
let mut b =
core::core::Block::new(&prev, vec![], next_header_info.clone().difficulty, reward)
.unwrap();
b.header.timestamp = prev.timestamp + Duration::seconds(160);
b.header.pow.secondary_scaling = next_header_info.secondary_scaling;
chain.set_txhashset_roots(&mut b).unwrap();
let edge_bits = if n == 2 {
global::min_edge_bits() + 1
} else {
global::min_edge_bits()
};
b.header.pow.proof.edge_bits = edge_bits;
pow::pow_size(
&mut b.header,
next_header_info.difficulty,
global::proofsize(),
edge_bits,
)
.unwrap();
b.header.pow.proof.edge_bits = edge_bits;
chain.process_block(b, Options::MINE).unwrap();
}
}

View file

@ -0,0 +1,25 @@
// Copyright 2018 The Grin Developers
//
// 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.
mod chain_test_helper;
use self::chain_test_helper::{clean_output_dir, mine_chain};
#[test]
fn test() {
let chain = mine_chain(".txhashset_archive_test", 35);
let header = chain.txhashset_archive_header().unwrap();
assert_eq!(10, header.height);
clean_output_dir(".txhashset_archive_test");
}

View file

@ -51,7 +51,10 @@ pub const AUTOMATED_TESTING_COINBASE_MATURITY: u64 = 3;
pub const USER_TESTING_COINBASE_MATURITY: u64 = 3; pub const USER_TESTING_COINBASE_MATURITY: u64 = 3;
/// Testing cut through horizon in blocks /// Testing cut through horizon in blocks
pub const TESTING_CUT_THROUGH_HORIZON: u32 = 70; pub const AUTOMATED_TESTING_CUT_THROUGH_HORIZON: u32 = 20;
/// Testing cut through horizon in blocks
pub const USER_TESTING_CUT_THROUGH_HORIZON: u32 = 70;
/// Testing state sync threshold in blocks /// Testing state sync threshold in blocks
pub const TESTING_STATE_SYNC_THRESHOLD: u32 = 20; pub const TESTING_STATE_SYNC_THRESHOLD: u32 = 20;
@ -82,6 +85,12 @@ pub const PEER_EXPIRATION_REMOVE_TIME: i64 = PEER_EXPIRATION_DAYS * 24 * 3600;
/// For a node configured as "archival_mode = true" only the txhashset will be compacted. /// For a node configured as "archival_mode = true" only the txhashset will be compacted.
pub const COMPACTION_CHECK: u64 = DAY_HEIGHT; pub const COMPACTION_CHECK: u64 = DAY_HEIGHT;
/// Automated testing number of blocks to reuse a txhashset zip for.
pub const AUTOMATED_TESTING_TXHASHSET_ARCHIVE_INTERVAL: u64 = 10;
/// Number of blocks to reuse a txhashset zip for.
pub const TXHASHSET_ARCHIVE_INTERVAL: u64 = 12 * 60;
/// Types of chain a server can run with, dictates the genesis block and /// Types of chain a server can run with, dictates the genesis block and
/// and mining parameters used. /// and mining parameters used.
#[derive(Debug, Clone, Serialize, Deserialize, PartialEq)] #[derive(Debug, Clone, Serialize, Deserialize, PartialEq)]
@ -245,8 +254,8 @@ pub fn max_block_weight() -> usize {
pub fn cut_through_horizon() -> u32 { pub fn cut_through_horizon() -> u32 {
let param_ref = CHAIN_TYPE.read(); let param_ref = CHAIN_TYPE.read();
match *param_ref { match *param_ref {
ChainTypes::AutomatedTesting => TESTING_CUT_THROUGH_HORIZON, ChainTypes::AutomatedTesting => AUTOMATED_TESTING_CUT_THROUGH_HORIZON,
ChainTypes::UserTesting => TESTING_CUT_THROUGH_HORIZON, ChainTypes::UserTesting => USER_TESTING_CUT_THROUGH_HORIZON,
_ => CUT_THROUGH_HORIZON, _ => CUT_THROUGH_HORIZON,
} }
} }
@ -261,6 +270,15 @@ pub fn state_sync_threshold() -> u32 {
} }
} }
/// Number of blocks to reuse a txhashset zip for.
pub fn txhashset_archive_interval() -> u64 {
let param_ref = CHAIN_TYPE.read();
match *param_ref {
ChainTypes::AutomatedTesting => AUTOMATED_TESTING_TXHASHSET_ARCHIVE_INTERVAL,
_ => TXHASHSET_ARCHIVE_INTERVAL,
}
}
/// Are we in automated testing mode? /// Are we in automated testing mode?
pub fn is_automated_testing_mode() -> bool { pub fn is_automated_testing_mode() -> bool {
let param_ref = CHAIN_TYPE.read(); let param_ref = CHAIN_TYPE.read();

View file

@ -71,7 +71,6 @@ pub fn verify_size(bh: &BlockHeader) -> Result<(), Error> {
pub fn mine_genesis_block() -> Result<Block, Error> { pub fn mine_genesis_block() -> Result<Block, Error> {
let mut gen = genesis::genesis_dev(); let mut gen = genesis::genesis_dev();
if global::is_user_testing_mode() || global::is_automated_testing_mode() { if global::is_user_testing_mode() || global::is_automated_testing_mode() {
gen = genesis::genesis_dev();
gen.header.timestamp = Utc::now(); gen.header.timestamp = Utc::now();
} }

View file

@ -562,6 +562,10 @@ impl ChainAdapter for TrackingAdapter {
self.adapter.txhashset_read(h) self.adapter.txhashset_read(h)
} }
fn txhashset_archive_header(&self) -> Result<core::BlockHeader, chain::Error> {
self.adapter.txhashset_archive_header()
}
fn txhashset_receive_ready(&self) -> bool { fn txhashset_receive_ready(&self) -> bool {
self.adapter.txhashset_receive_ready() self.adapter.txhashset_receive_ready()
} }

View file

@ -674,6 +674,10 @@ impl ChainAdapter for Peers {
self.adapter.txhashset_read(h) self.adapter.txhashset_read(h)
} }
fn txhashset_archive_header(&self) -> Result<core::BlockHeader, chain::Error> {
self.adapter.txhashset_archive_header()
}
fn txhashset_receive_ready(&self) -> bool { fn txhashset_receive_ready(&self) -> bool {
self.adapter.txhashset_receive_ready() self.adapter.txhashset_receive_ready()
} }

View file

@ -12,8 +12,10 @@
// See the License for the specific language governing permissions and // See the License for the specific language governing permissions and
// limitations under the License. // limitations under the License.
use crate::conn::{Message, MessageHandler, Response, Tracker}; use crate::conn::{Message, MessageHandler, Response, Tracker};
use crate::core::core::{self, hash::Hash, CompactBlock}; use crate::core::core::{self, hash::Hash, hash::Hashed, CompactBlock};
use crate::msg::{ use crate::msg::{
BanReason, GetPeerAddrs, Headers, KernelDataResponse, Locator, PeerAddrs, Ping, Pong, BanReason, GetPeerAddrs, Headers, KernelDataResponse, Locator, PeerAddrs, Ping, Pong,
TxHashSetArchive, TxHashSetRequest, Type, TxHashSetArchive, TxHashSetRequest, Type,
@ -297,15 +299,17 @@ impl MessageHandler for Protocol {
sm_req.hash, sm_req.height sm_req.hash, sm_req.height
); );
let txhashset = self.adapter.txhashset_read(sm_req.hash); let txhashset_header = self.adapter.txhashset_archive_header()?;
let txhashset_header_hash = txhashset_header.hash();
let txhashset = self.adapter.txhashset_read(txhashset_header_hash);
if let Some(txhashset) = txhashset { if let Some(txhashset) = txhashset {
let file_sz = txhashset.reader.metadata()?.len(); let file_sz = txhashset.reader.metadata()?.len();
let mut resp = Response::new( let mut resp = Response::new(
Type::TxHashSetArchive, Type::TxHashSetArchive,
&TxHashSetArchive { &TxHashSetArchive {
height: sm_req.height as u64, height: txhashset_header.height as u64,
hash: sm_req.hash, hash: txhashset_header_hash,
bytes: file_sz, bytes: file_sz,
}, },
writer, writer,

View file

@ -302,6 +302,10 @@ impl ChainAdapter for DummyAdapter {
unimplemented!() unimplemented!()
} }
fn txhashset_archive_header(&self) -> Result<core::BlockHeader, chain::Error> {
unimplemented!()
}
fn txhashset_receive_ready(&self) -> bool { fn txhashset_receive_ready(&self) -> bool {
false false
} }

View file

@ -536,6 +536,9 @@ pub trait ChainAdapter: Sync + Send {
/// at the provided block hash. /// at the provided block hash.
fn txhashset_read(&self, h: Hash) -> Option<TxHashSetRead>; fn txhashset_read(&self, h: Hash) -> Option<TxHashSetRead>;
/// Header of the txhashset archive currently being served to peers.
fn txhashset_archive_header(&self) -> Result<core::BlockHeader, chain::Error>;
/// Whether the node is ready to accept a new txhashset. If this isn't the /// Whether the node is ready to accept a new txhashset. If this isn't the
/// case, the archive is provided without being requested and likely an /// case, the archive is provided without being requested and likely an
/// attack attempt. This should be checked *before* downloading the whole /// attack attempt. This should be checked *before* downloading the whole

View file

@ -372,6 +372,10 @@ impl p2p::ChainAdapter for NetToChainAdapter {
} }
} }
fn txhashset_archive_header(&self) -> Result<core::BlockHeader, chain::Error> {
self.chain().txhashset_archive_header()
}
fn txhashset_receive_ready(&self) -> bool { fn txhashset_receive_ready(&self) -> bool {
match self.sync_state.status() { match self.sync_state.status() {
SyncStatus::TxHashsetDownload { .. } => true, SyncStatus::TxHashsetDownload { .. } => true,

View file

@ -160,6 +160,9 @@ impl StateSync {
fn request_state(&self, header_head: &chain::Tip) -> Result<Arc<Peer>, p2p::Error> { fn request_state(&self, header_head: &chain::Tip) -> Result<Arc<Peer>, p2p::Error> {
let threshold = global::state_sync_threshold() as u64; let threshold = global::state_sync_threshold() as u64;
let archive_interval = global::txhashset_archive_interval();
let mut txhashset_height = header_head.height.saturating_sub(threshold);
txhashset_height = txhashset_height.saturating_sub(txhashset_height % archive_interval);
if let Some(peer) = self.peers.most_work_peer() { if let Some(peer) = self.peers.most_work_peer() {
// ask for txhashset at state_sync_threshold // ask for txhashset at state_sync_threshold
@ -168,18 +171,18 @@ impl StateSync {
.get_block_header(&header_head.prev_block_h) .get_block_header(&header_head.prev_block_h)
.map_err(|e| { .map_err(|e| {
error!( error!(
"chain error dirung getting a block header {}: {:?}", "chain error during getting a block header {}: {:?}",
&header_head.prev_block_h, e &header_head.prev_block_h, e
); );
p2p::Error::Internal p2p::Error::Internal
})?; })?;
for _ in 0..threshold { while txhashset_head.height > txhashset_height {
txhashset_head = self txhashset_head = self
.chain .chain
.get_previous_header(&txhashset_head) .get_previous_header(&txhashset_head)
.map_err(|e| { .map_err(|e| {
error!( error!(
"chain error dirung getting a previous block header {}: {:?}", "chain error during getting a previous block header {}: {:?}",
txhashset_head.hash(), txhashset_head.hash(),
e e
); );