2018-05-30 23:57:13 +03:00
|
|
|
// 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.
|
|
|
|
|
|
|
|
extern crate grin_chain as chain;
|
|
|
|
extern crate grin_core as core;
|
|
|
|
extern crate grin_keychain as keychain;
|
2018-06-22 11:08:06 +03:00
|
|
|
extern crate grin_store as store;
|
2018-05-30 23:57:13 +03:00
|
|
|
extern crate grin_wallet as wallet;
|
2018-08-03 05:16:16 +03:00
|
|
|
extern crate grin_util as util;
|
2018-05-30 23:57:13 +03:00
|
|
|
|
2018-08-03 05:16:16 +03:00
|
|
|
use std::collections::HashSet;
|
|
|
|
use std::iter::FromIterator;
|
|
|
|
use std::fs::{self, File, OpenOptions};
|
|
|
|
use std::path::{Path, PathBuf};
|
2018-05-30 23:57:13 +03:00
|
|
|
use std::sync::Arc;
|
|
|
|
|
2018-06-22 11:08:06 +03:00
|
|
|
use chain::store::ChainStore;
|
2018-06-22 11:33:42 +03:00
|
|
|
use chain::txhashset;
|
2018-05-30 23:57:13 +03:00
|
|
|
use chain::types::Tip;
|
|
|
|
use core::core::target::Difficulty;
|
|
|
|
use core::core::{Block, BlockHeader};
|
2018-06-08 08:21:54 +03:00
|
|
|
use keychain::{ExtKeychain, Keychain};
|
2018-08-03 05:16:16 +03:00
|
|
|
use util::file;
|
2018-05-30 23:57:13 +03:00
|
|
|
use wallet::libtx::{build, reward};
|
|
|
|
|
|
|
|
fn clean_output_dir(dir_name: &str) {
|
|
|
|
let _ = fs::remove_dir_all(dir_name);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_some_raw_txs() {
|
|
|
|
let db_root = format!(".grin_txhashset_raw_txs");
|
|
|
|
clean_output_dir(&db_root);
|
|
|
|
|
2018-06-22 11:08:06 +03:00
|
|
|
let db_env = Arc::new(store::new_env(db_root.clone()));
|
2018-06-22 11:33:42 +03:00
|
|
|
|
|
|
|
let chain_store = ChainStore::new(db_env).unwrap();
|
|
|
|
let store = Arc::new(chain_store);
|
|
|
|
// open the txhashset, creating a new one if necessary
|
|
|
|
let mut txhashset = txhashset::TxHashSet::open(db_root.clone(), store.clone(), None).unwrap();
|
2018-05-30 23:57:13 +03:00
|
|
|
|
2018-06-08 08:21:54 +03:00
|
|
|
let keychain = ExtKeychain::from_random_seed().unwrap();
|
2018-05-30 23:57:13 +03:00
|
|
|
let key_id1 = keychain.derive_key_id(1).unwrap();
|
|
|
|
let key_id2 = keychain.derive_key_id(2).unwrap();
|
|
|
|
let key_id3 = keychain.derive_key_id(3).unwrap();
|
|
|
|
let key_id4 = keychain.derive_key_id(4).unwrap();
|
|
|
|
let key_id5 = keychain.derive_key_id(5).unwrap();
|
|
|
|
let key_id6 = keychain.derive_key_id(6).unwrap();
|
|
|
|
|
|
|
|
// Create a simple block with a single coinbase output
|
|
|
|
// so we have something to spend.
|
|
|
|
let prev_header = BlockHeader::default();
|
|
|
|
let reward_out = reward::output(&keychain, &key_id1, 0, prev_header.height).unwrap();
|
|
|
|
let block = Block::new(&prev_header, vec![], Difficulty::one(), reward_out).unwrap();
|
|
|
|
|
|
|
|
// Now apply this initial block to the (currently empty) MMRs.
|
|
|
|
// Note: this results in an output MMR with a single leaf node.
|
|
|
|
// We need to be careful with pruning while processing the txs below
|
|
|
|
// as we cannot prune a tree with a single node in it (no sibling or parent).
|
2018-06-22 11:08:06 +03:00
|
|
|
let mut batch = store.batch().unwrap();
|
|
|
|
txhashset::extending(&mut txhashset, &mut batch, |extension| {
|
|
|
|
extension.apply_block(&block)
|
|
|
|
}).unwrap();
|
2018-05-30 23:57:13 +03:00
|
|
|
|
|
|
|
// Make sure we setup the head in the store based on block we just accepted.
|
|
|
|
let head = Tip::from_block(&block.header);
|
2018-06-22 11:08:06 +03:00
|
|
|
batch.save_head(&head).unwrap();
|
|
|
|
batch.commit().unwrap();
|
2018-05-30 23:57:13 +03:00
|
|
|
|
|
|
|
let coinbase_reward = 60_000_000_000;
|
|
|
|
|
|
|
|
// tx1 spends the original coinbase output from the block
|
|
|
|
let tx1 = build::transaction_with_offset(
|
|
|
|
vec![
|
|
|
|
build::coinbase_input(
|
|
|
|
coinbase_reward,
|
|
|
|
key_id1.clone(),
|
|
|
|
),
|
|
|
|
build::output(100, key_id2.clone()),
|
|
|
|
build::output(150, key_id3.clone()),
|
|
|
|
],
|
|
|
|
&keychain,
|
|
|
|
).unwrap();
|
|
|
|
|
|
|
|
// tx2 attempts to "double spend" the coinbase output from the block (conflicts
|
|
|
|
// with tx1)
|
|
|
|
let tx2 = build::transaction_with_offset(
|
|
|
|
vec![
|
|
|
|
build::coinbase_input(
|
|
|
|
coinbase_reward,
|
|
|
|
key_id1.clone(),
|
|
|
|
),
|
|
|
|
build::output(100, key_id4.clone()),
|
|
|
|
],
|
|
|
|
&keychain,
|
|
|
|
).unwrap();
|
|
|
|
|
|
|
|
// tx3 spends one output from tx1
|
|
|
|
let tx3 = build::transaction_with_offset(
|
|
|
|
vec![
|
|
|
|
build::input(100, key_id2.clone()),
|
|
|
|
build::output(90, key_id5.clone()),
|
|
|
|
],
|
|
|
|
&keychain,
|
|
|
|
).unwrap();
|
|
|
|
|
|
|
|
// tx4 spends the other output from tx1 and the output from tx3
|
|
|
|
let tx4 = build::transaction_with_offset(
|
|
|
|
vec![
|
|
|
|
build::input(150, key_id3.clone()),
|
|
|
|
build::input(90, key_id5.clone()),
|
|
|
|
build::output(220, key_id6.clone()),
|
|
|
|
],
|
|
|
|
&keychain,
|
|
|
|
).unwrap();
|
|
|
|
|
|
|
|
// Now validate the txs against the txhashset (via a readonly extension).
|
|
|
|
// Note: we use a single txhashset extension and we can continue to
|
|
|
|
// apply txs successfully after a failure.
|
|
|
|
let _ = txhashset::extending_readonly(&mut txhashset, |extension| {
|
|
|
|
// Note: we pass in an increasing "height" here so we can rollback
|
|
|
|
// each tx individually as necessary, while maintaining a long lived
|
|
|
|
// txhashset extension.
|
2018-06-18 18:18:38 +03:00
|
|
|
assert!(extension.apply_raw_tx(&tx1).is_ok());
|
|
|
|
assert!(extension.apply_raw_tx(&tx2).is_err());
|
|
|
|
assert!(extension.apply_raw_tx(&tx3).is_ok());
|
|
|
|
assert!(extension.apply_raw_tx(&tx4).is_ok());
|
2018-05-30 23:57:13 +03:00
|
|
|
Ok(())
|
|
|
|
});
|
|
|
|
}
|
2018-08-03 05:16:16 +03:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_unexpected_zip() {
|
|
|
|
let db_root = format!(".grin_txhashset_zip");
|
|
|
|
clean_output_dir(&db_root);
|
|
|
|
let db_env = Arc::new(store::new_env(db_root.clone()));
|
|
|
|
let chain_store = ChainStore::new(db_env).unwrap();
|
|
|
|
let store = Arc::new(chain_store);
|
|
|
|
txhashset::TxHashSet::open(db_root.clone(), store.clone(), None).unwrap();
|
|
|
|
// First check if everything works out of the box
|
|
|
|
assert!(txhashset::zip_read(db_root.clone(), &BlockHeader::default()).is_ok());
|
|
|
|
let zip_path = Path::new(&db_root).join("txhashset_snapshot.zip");
|
|
|
|
let zip_file = File::open(&zip_path).unwrap();
|
|
|
|
assert!(txhashset::zip_write(db_root.clone(), zip_file, &BlockHeader::default()).is_ok());
|
|
|
|
// Remove temp txhashset dir
|
|
|
|
fs::remove_dir_all(Path::new(&db_root).join("txhashset_zip")).unwrap();
|
|
|
|
// Then add strange files in the original txhashset folder
|
|
|
|
write_file(db_root.clone());
|
|
|
|
assert!(txhashset::zip_read(db_root.clone(), &BlockHeader::default()).is_ok());
|
|
|
|
// Check that the temp dir dos not contains the strange files
|
|
|
|
let txhashset_zip_path = Path::new(&db_root).join("txhashset_zip");
|
|
|
|
assert!(txhashset_contains_expected_files("txhashset_zip".to_string(), txhashset_zip_path.clone()));
|
|
|
|
fs::remove_dir_all(Path::new(&db_root).join("txhashset_zip")).unwrap();
|
|
|
|
|
|
|
|
let zip_file = File::open(zip_path).unwrap();
|
|
|
|
assert!(txhashset::zip_write(db_root.clone(), zip_file, &BlockHeader::default()).is_ok());
|
|
|
|
// Check that the txhashset dir dos not contains the strange files
|
|
|
|
let txhashset_path = Path::new(&db_root).join("txhashset");
|
|
|
|
assert!(txhashset_contains_expected_files("txhashset".to_string(), txhashset_path.clone()));
|
|
|
|
fs::remove_dir_all(Path::new(&db_root).join("txhashset")).unwrap();
|
|
|
|
}
|
|
|
|
|
|
|
|
fn write_file (db_root: String) {
|
|
|
|
OpenOptions::new()
|
|
|
|
.create(true)
|
|
|
|
.write(true)
|
|
|
|
.open(Path::new(&db_root).join("txhashset").join("kernel").join("strange0")).unwrap();
|
|
|
|
OpenOptions::new()
|
|
|
|
.create(true)
|
|
|
|
.write(true)
|
|
|
|
.open(Path::new(&db_root).join("txhashset").join("strange1")).unwrap();
|
|
|
|
fs::create_dir(Path::new(&db_root).join("txhashset").join("strange_dir")).unwrap();
|
|
|
|
OpenOptions::new()
|
|
|
|
.create(true)
|
|
|
|
.write(true)
|
|
|
|
.open(Path::new(&db_root).join("txhashset").join("strange_dir").join("strange2")).unwrap();
|
|
|
|
fs::create_dir(Path::new(&db_root).join("txhashset").join("strange_dir").join("strange_subdir")).unwrap();
|
|
|
|
OpenOptions::new()
|
|
|
|
.create(true)
|
|
|
|
.write(true)
|
|
|
|
.open(Path::new(&db_root).join("txhashset").join("strange_dir").join("strange_subdir").join("strange3")).unwrap();
|
|
|
|
}
|
|
|
|
|
|
|
|
fn txhashset_contains_expected_files(dirname: String, path_buf: PathBuf) -> bool {
|
|
|
|
let list_zip_files = file::list_files(path_buf.into_os_string().into_string().unwrap());
|
|
|
|
let zip_files_hashset: HashSet<_> = HashSet::from_iter(list_zip_files.iter().cloned());
|
|
|
|
let expected_files = vec![dirname, "output".to_string(), "rangeproof".to_string(), "kernel".to_string(), "pmmr_hash.bin".to_string(), "pmmr_data.bin".to_string()];
|
|
|
|
let expected_files_hashset = HashSet::from_iter(expected_files.iter().cloned());
|
|
|
|
let intersection: HashSet<_> = zip_files_hashset.difference(&expected_files_hashset).collect();
|
|
|
|
if intersection.is_empty() {
|
|
|
|
true
|
|
|
|
} else {
|
|
|
|
false
|
|
|
|
}
|
|
|
|
}
|