pub mod common;
use self::core::consensus;
use self::core::core::hash::Hashed;
use self::core::core::{HeaderVersion, KernelFeatures, NRDRelativeHeight, TxKernel};
use self::core::global;
use self::core::libtx::aggsig;
use self::keychain::{BlindingFactor, ExtKeychain, Keychain};
use self::pool::types::PoolError;
use crate::common::*;
use mwc_core as core;
use mwc_keychain as keychain;
use mwc_pool as pool;
use mwc_util as util;
use std::sync::Arc;
#[test]
fn test_nrd_kernel_relative_height() -> Result<(), PoolError> {
util::init_test_logger();
global::set_local_chain_type(global::ChainTypes::AutomatedTesting);
global::set_local_accept_fee_base(10);
global::set_local_nrd_enabled(true);
let keychain: ExtKeychain = Keychain::from_random_seed(false).unwrap();
let db_root = "target/.nrd_kernel_relative_height";
clean_output_dir(db_root.into());
let genesis = genesis_block(&keychain);
let chain = Arc::new(init_chain(db_root, genesis));
let mut pool = init_transaction_pool(Arc::new(ChainAdapter {
chain: chain.clone(),
}));
add_some_blocks(&chain, 3, &keychain);
let header_1 = chain.get_header_by_height(1).unwrap();
let initial_tx =
test_transaction_spending_coinbase(&keychain, &header_1, vec![1_000, 2_000, 3_000, 4_000]);
add_block(&chain, &[initial_tx], &keychain);
add_some_blocks(&chain, 8, &keychain);
let header = chain.head_header().unwrap();
assert_eq!(header.height, 4 * consensus::TESTING_HARD_FORK_INTERVAL);
assert_eq!(header.version, HeaderVersion(4));
let (tx1, tx2, tx3) = {
let mut kernel = TxKernel::with_features(KernelFeatures::NoRecentDuplicate {
fee: 600.into(),
relative_height: NRDRelativeHeight::new(2)?,
});
let msg = kernel.msg_to_sign().unwrap();
let excess = BlindingFactor::rand(keychain.secp());
let skey = excess.secret_key(keychain.secp()).unwrap();
kernel.excess = keychain.secp().commit(0, skey).unwrap();
let pubkey = &kernel.excess.to_pubkey(keychain.secp()).unwrap();
kernel.excess_sig =
aggsig::sign_with_blinding(&keychain.secp(), &msg, &excess, Some(&pubkey)).unwrap();
kernel.verify(chain.secp()).unwrap();
let mut kernel2 = kernel.clone();
kernel2.excess_sig =
aggsig::sign_with_blinding(&keychain.secp(), &msg, &excess, Some(&pubkey)).unwrap();
kernel2.verify(chain.secp()).unwrap();
let tx1 = test_transaction_with_kernel(
&keychain,
vec![1_000, 2_000],
vec![2_400],
kernel.clone(),
excess.clone(),
);
let tx2 = test_transaction_with_kernel(
&keychain,
vec![2_400],
vec![1_800],
kernel2.clone(),
excess.clone(),
);
let mut kernel_short = TxKernel::with_features(KernelFeatures::NoRecentDuplicate {
fee: 300.into(),
relative_height: NRDRelativeHeight::new(1)?,
});
let msg_short = kernel_short.msg_to_sign().unwrap();
kernel_short.excess = kernel.excess;
kernel_short.excess_sig =
aggsig::sign_with_blinding(&keychain.secp(), &msg_short, &excess, Some(&pubkey))
.unwrap();
kernel_short.verify(chain.secp()).unwrap();
let tx3 = test_transaction_with_kernel(
&keychain,
vec![1_800],
vec![1_500],
kernel_short.clone(),
excess.clone(),
);
(tx1, tx2, tx3)
};
assert_eq!(
pool.add_to_pool(test_source(), tx1.clone(), true, &header, chain.secp()),
Ok(()),
);
assert_eq!(pool.stempool.size(), 1);
assert_eq!(
pool.add_to_pool(test_source(), tx2.clone(), true, &header, chain.secp()),
Err(PoolError::NRDKernelRelativeHeight)
);
assert_eq!(
pool.add_to_pool(test_source(), tx1.clone(), false, &header, chain.secp()),
Ok(()),
);
assert_eq!(pool.txpool.size(), 1);
assert_eq!(pool.stempool.size(), 0);
assert_eq!(
pool.add_to_pool(test_source(), tx2.clone(), true, &header, chain.secp()),
Err(PoolError::NRDKernelRelativeHeight)
);
assert_eq!(
pool.add_to_pool(test_source(), tx2.clone(), false, &header, chain.secp()),
Err(PoolError::NRDKernelRelativeHeight)
);
assert_eq!(pool.total_size(), 1);
assert_eq!(pool.txpool.size(), 1);
assert_eq!(pool.stempool.size(), 0);
let txs = pool.prepare_mineable_transactions(chain.secp()).unwrap();
assert_eq!(txs.len(), 1);
add_block(&chain, &txs, &keychain);
let header = chain.head_header().unwrap();
let block = chain.get_block(&header.hash()).unwrap();
pool.reconcile_block(&block, chain.secp())?;
assert_eq!(pool.total_size(), 0);
assert_eq!(pool.txpool.size(), 0);
assert_eq!(pool.stempool.size(), 0);
assert_eq!(
pool.add_to_pool(test_source(), tx2.clone(), true, &header, chain.secp()),
Err(PoolError::NRDKernelRelativeHeight)
);
assert_eq!(
pool.add_to_pool(test_source(), tx2.clone(), false, &header, chain.secp()),
Err(PoolError::NRDKernelRelativeHeight)
);
add_block(&chain, &[], &keychain);
let header = chain.head_header().unwrap();
assert_eq!(
pool.add_to_pool(test_source(), tx2.clone(), true, &header, chain.secp()),
Ok(())
);
assert_eq!(pool.total_size(), 0);
assert_eq!(pool.txpool.size(), 0);
assert_eq!(pool.stempool.size(), 1);
assert_eq!(
pool.add_to_pool(test_source(), tx3.clone(), true, &header, chain.secp()),
Err(PoolError::NRDKernelRelativeHeight)
);
assert_eq!(
pool.add_to_pool(test_source(), tx2.clone(), false, &header, chain.secp()),
Ok(())
);
assert_eq!(
pool.add_to_pool(test_source(), tx3.clone(), false, &header, chain.secp()),
Err(PoolError::NRDKernelRelativeHeight)
);
assert_eq!(pool.total_size(), 1);
assert_eq!(pool.txpool.size(), 1);
assert_eq!(pool.stempool.size(), 0);
let txs = pool.prepare_mineable_transactions(chain.secp()).unwrap();
assert_eq!(txs.len(), 1);
add_block(&chain, &txs, &keychain);
let header = chain.head_header().unwrap();
let block = chain.get_block(&header.hash()).unwrap();
pool.reconcile_block(&block, chain.secp())?;
assert_eq!(pool.total_size(), 0);
assert_eq!(pool.txpool.size(), 0);
assert_eq!(pool.stempool.size(), 0);
assert_eq!(
pool.add_to_pool(test_source(), tx3.clone(), true, &header, chain.secp()),
Ok(())
);
assert_eq!(pool.total_size(), 0);
assert_eq!(pool.txpool.size(), 0);
assert_eq!(pool.stempool.size(), 1);
assert_eq!(
pool.add_to_pool(test_source(), tx3.clone(), false, &header, chain.secp()),
Ok(())
);
assert_eq!(pool.total_size(), 1);
assert_eq!(pool.txpool.size(), 1);
assert_eq!(pool.stempool.size(), 0);
clean_output_dir(db_root.into());
Ok(())
}