// Copyright 2020 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 grin_chain as chain; use grin_core as core; use grin_keychain as keychain; use grin_util as util; use self::chain_test_helper::{clean_output_dir, genesis_block, init_chain}; use crate::chain::{Chain, Error, Options}; use crate::core::core::{ Block, BlockHeader, KernelFeatures, NRDRelativeHeight, Transaction, TxKernel, }; use crate::core::libtx::{aggsig, build, reward, ProofBuilder}; use crate::core::{consensus, global, pow}; use crate::keychain::{BlindingFactor, ExtKeychain, ExtKeychainPath, Identifier, Keychain}; use chrono::Duration; fn build_block( chain: &Chain, keychain: &K, key_id: &Identifier, txs: Vec, ) -> Result where K: Keychain, { let prev = chain.head_header()?; build_block_from_prev(&prev, chain, keychain, key_id, txs) } fn build_block_from_prev( prev: &BlockHeader, chain: &Chain, keychain: &K, key_id: &Identifier, txs: Vec, ) -> Result where K: Keychain, { let next_header_info = consensus::next_difficulty(prev.height, chain.difficulty_iter().unwrap()); let fee = txs.iter().map(|x| x.fee(prev.height + 1)).sum(); let reward = reward::output(keychain, &ProofBuilder::new(keychain), key_id, fee, false).unwrap(); let mut block = Block::new(prev, &txs, next_header_info.clone().difficulty, reward)?; block.header.timestamp = prev.timestamp + Duration::seconds(60); block.header.pow.secondary_scaling = next_header_info.secondary_scaling; chain.set_txhashset_roots(&mut block)?; block.header.pow.proof.edge_bits = global::min_edge_bits(); pow::pow_size( &mut block.header, next_header_info.difficulty, global::proofsize(), global::min_edge_bits(), ) .unwrap(); Ok(block) } #[test] fn process_block_nrd_validation() -> Result<(), Error> { global::set_local_chain_type(global::ChainTypes::AutomatedTesting); global::set_local_nrd_enabled(true); util::init_test_logger(); let chain_dir = ".grin.nrd_kernel"; clean_output_dir(chain_dir); let keychain = ExtKeychain::from_random_seed(false).unwrap(); let builder = ProofBuilder::new(&keychain); let genesis = genesis_block(&keychain); let chain = init_chain(chain_dir, genesis.clone()); for n in 1..9 { let key_id = ExtKeychainPath::new(1, n, 0, 0, 0).to_identifier(); let block = build_block(&chain, &keychain, &key_id, vec![])?; chain.process_block(block, Options::NONE)?; } assert_eq!(chain.head()?.height, 8); let mut kernel = TxKernel::with_features(KernelFeatures::NoRecentDuplicate { fee: 20000.into(), relative_height: NRDRelativeHeight::new(2)?, }); // // Construct the message to be signed. let msg = kernel.msg_to_sign().unwrap(); // // Generate a kernel with public excess and associated signature. 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().unwrap(); let key_id1 = ExtKeychainPath::new(1, 1, 0, 0, 0).to_identifier(); let key_id2 = ExtKeychainPath::new(1, 2, 0, 0, 0).to_identifier(); let key_id3 = ExtKeychainPath::new(1, 3, 0, 0, 0).to_identifier(); let tx1 = build::transaction_with_kernel( &[ build::coinbase_input(consensus::REWARD, key_id1.clone()), build::output(consensus::REWARD - 20000, key_id2.clone()), ], kernel.clone(), excess.clone(), &keychain, &builder, ) .unwrap(); let tx2 = build::transaction_with_kernel( &[ build::input(consensus::REWARD - 20000, key_id2.clone()), build::output(consensus::REWARD - 40000, key_id3.clone()), ], kernel.clone(), excess.clone(), &keychain, &builder, ) .unwrap(); let key_id9 = ExtKeychainPath::new(1, 9, 0, 0, 0).to_identifier(); let key_id10 = ExtKeychainPath::new(1, 10, 0, 0, 0).to_identifier(); let key_id11 = ExtKeychainPath::new(1, 11, 0, 0, 0).to_identifier(); // Block containing both tx1 and tx2 is invalid. // Not valid for two duplicate NRD kernels to co-exist in same block. // Jump through some hoops to build an invalid block by disabling the feature flag. // TODO - We need a good way of building invalid stuff in tests. let block_invalid_9 = { global::set_local_nrd_enabled(false); let block = build_block(&chain, &keychain, &key_id9, vec![tx1.clone(), tx2.clone()])?; global::set_local_nrd_enabled(true); block }; assert!(chain.process_block(block_invalid_9, Options::NONE).is_err()); assert_eq!(chain.head()?.height, 8); // Block containing tx1 is valid. let block_valid_9 = build_block(&chain, &keychain, &key_id9, vec![tx1.clone()])?; chain.process_block(block_valid_9, Options::NONE)?; // Block at height 10 is invalid if it contains tx2 due to NRD rule (relative_height=2). // Jump through some hoops to build an invalid block by disabling the feature flag. // TODO - We need a good way of building invalid stuff in tests. let block_invalid_10 = { global::set_local_nrd_enabled(false); let block = build_block(&chain, &keychain, &key_id10, vec![tx2.clone()])?; global::set_local_nrd_enabled(true); block }; assert!(chain .process_block(block_invalid_10, Options::NONE) .is_err()); // Block at height 10 is valid if we do not include tx2. let block_valid_10 = build_block(&chain, &keychain, &key_id10, vec![])?; chain.process_block(block_valid_10, Options::NONE)?; // Block at height 11 is valid with tx2 as NRD rule is met (relative_height=2). let block_valid_11 = build_block(&chain, &keychain, &key_id11, vec![tx2.clone()])?; chain.process_block(block_valid_11, Options::NONE)?; clean_output_dir(chain_dir); Ok(()) } #[test] fn process_block_nrd_validation_relative_height_1() -> Result<(), Error> { global::set_local_chain_type(global::ChainTypes::AutomatedTesting); global::set_local_nrd_enabled(true); util::init_test_logger(); let chain_dir = ".grin.nrd_kernel_relative_height_1"; clean_output_dir(chain_dir); let keychain = ExtKeychain::from_random_seed(false).unwrap(); let builder = ProofBuilder::new(&keychain); let genesis = genesis_block(&keychain); let chain = init_chain(chain_dir, genesis.clone()); for n in 1..9 { let key_id = ExtKeychainPath::new(1, n, 0, 0, 0).to_identifier(); let block = build_block(&chain, &keychain, &key_id, vec![])?; chain.process_block(block, Options::NONE)?; } assert_eq!(chain.head()?.height, 8); let mut kernel = TxKernel::with_features(KernelFeatures::NoRecentDuplicate { fee: 20000.into(), relative_height: NRDRelativeHeight::new(1)?, }); // // Construct the message to be signed. let msg = kernel.msg_to_sign().unwrap(); // // Generate a kernel with public excess and associated signature. 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().unwrap(); let key_id1 = ExtKeychainPath::new(1, 1, 0, 0, 0).to_identifier(); let key_id2 = ExtKeychainPath::new(1, 2, 0, 0, 0).to_identifier(); let key_id3 = ExtKeychainPath::new(1, 3, 0, 0, 0).to_identifier(); let tx1 = build::transaction_with_kernel( &[ build::coinbase_input(consensus::REWARD, key_id1.clone()), build::output(consensus::REWARD - 20000, key_id2.clone()), ], kernel.clone(), excess.clone(), &keychain, &builder, ) .unwrap(); let tx2 = build::transaction_with_kernel( &[ build::input(consensus::REWARD - 20000, key_id2.clone()), build::output(consensus::REWARD - 40000, key_id3.clone()), ], kernel.clone(), excess.clone(), &keychain, &builder, ) .unwrap(); let key_id9 = ExtKeychainPath::new(1, 9, 0, 0, 0).to_identifier(); let key_id10 = ExtKeychainPath::new(1, 10, 0, 0, 0).to_identifier(); // Block containing both tx1 and tx2 is invalid. // Not valid for two duplicate NRD kernels to co-exist in same block. // Jump through some hoops here to build an "invalid" block. // TODO - We need a good way of building invalid stuff for tests. let block_invalid_9 = { global::set_local_nrd_enabled(false); let block = build_block(&chain, &keychain, &key_id9, vec![tx1.clone(), tx2.clone()])?; global::set_local_nrd_enabled(true); block }; assert!(chain.process_block(block_invalid_9, Options::NONE).is_err()); assert_eq!(chain.head()?.height, 8); // Block containing tx1 is valid. let block_valid_9 = build_block(&chain, &keychain, &key_id9, vec![tx1.clone()])?; chain.process_block(block_valid_9, Options::NONE)?; // Block at height 10 is valid with tx2 as NRD rule is met (relative_height=1). let block_valid_10 = build_block(&chain, &keychain, &key_id10, vec![tx2.clone()])?; chain.process_block(block_valid_10, Options::NONE)?; clean_output_dir(chain_dir); Ok(()) } #[test] fn process_block_nrd_validation_fork() -> Result<(), Error> { global::set_local_chain_type(global::ChainTypes::AutomatedTesting); global::set_local_nrd_enabled(true); util::init_test_logger(); let chain_dir = ".grin.nrd_kernel_fork"; clean_output_dir(chain_dir); let keychain = ExtKeychain::from_random_seed(false).unwrap(); let builder = ProofBuilder::new(&keychain); let genesis = genesis_block(&keychain); let chain = init_chain(chain_dir, genesis.clone()); for n in 1..9 { let key_id = ExtKeychainPath::new(1, n, 0, 0, 0).to_identifier(); let block = build_block(&chain, &keychain, &key_id, vec![])?; chain.process_block(block, Options::NONE)?; } let header_8 = chain.head_header()?; assert_eq!(header_8.height, 8); let mut kernel = TxKernel::with_features(KernelFeatures::NoRecentDuplicate { fee: 20000.into(), relative_height: NRDRelativeHeight::new(2)?, }); // // Construct the message to be signed. let msg = kernel.msg_to_sign().unwrap(); // // Generate a kernel with public excess and associated signature. 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().unwrap(); let key_id1 = ExtKeychainPath::new(1, 1, 0, 0, 0).to_identifier(); let key_id2 = ExtKeychainPath::new(1, 2, 0, 0, 0).to_identifier(); let key_id3 = ExtKeychainPath::new(1, 3, 0, 0, 0).to_identifier(); let tx1 = build::transaction_with_kernel( &[ build::coinbase_input(consensus::REWARD, key_id1.clone()), build::output(consensus::REWARD - 20000, key_id2.clone()), ], kernel.clone(), excess.clone(), &keychain, &builder, ) .unwrap(); let tx2 = build::transaction_with_kernel( &[ build::input(consensus::REWARD - 20000, key_id2.clone()), build::output(consensus::REWARD - 40000, key_id3.clone()), ], kernel.clone(), excess.clone(), &keychain, &builder, ) .unwrap(); let key_id9 = ExtKeychainPath::new(1, 9, 0, 0, 0).to_identifier(); let key_id10 = ExtKeychainPath::new(1, 10, 0, 0, 0).to_identifier(); let key_id11 = ExtKeychainPath::new(1, 11, 0, 0, 0).to_identifier(); // Block containing tx1 is valid. let block_valid_9 = build_block_from_prev(&header_8, &chain, &keychain, &key_id9, vec![tx1.clone()])?; chain.process_block(block_valid_9.clone(), Options::NONE)?; // Block at height 10 is valid if we do not include tx2. let block_valid_10 = build_block_from_prev(&block_valid_9.header, &chain, &keychain, &key_id10, vec![])?; chain.process_block(block_valid_10, Options::NONE)?; // Process an alternative "fork" block also at height 9. // The "other" block at height 9 should not affect this one in terms of NRD kernels // as the recent kernel index should be rewound. let block_valid_9b = build_block_from_prev(&header_8, &chain, &keychain, &key_id9, vec![tx1.clone()])?; chain.process_block(block_valid_9b.clone(), Options::NONE)?; // Process an alternative block at height 10 on this same fork. let block_valid_10b = build_block_from_prev(&block_valid_9b.header, &chain, &keychain, &key_id10, vec![])?; chain.process_block(block_valid_10b.clone(), Options::NONE)?; // Block at height 11 is valid with tx2 as NRD rule is met (relative_height=2). let block_valid_11b = build_block_from_prev( &block_valid_10b.header, &chain, &keychain, &key_id11, vec![tx2.clone()], )?; chain.process_block(block_valid_11b, Options::NONE)?; clean_output_dir(chain_dir); Ok(()) }