9#define BOOST_TEST_MODULE Bitcoin Kernel Test Suite
10#include <boost/test/included/unit_test.hpp>
32 const std::string chars =
"0123456789"
33 "abcdefghijklmnopqrstuvwxyz"
34 "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
36 static std::random_device rd;
37 static std::default_random_engine dre{rd()};
38 static std::uniform_int_distribution<> distribution(0, chars.size() - 1);
41 random.reserve(length);
42 for (uint32_t i = 0; i < length; i++) {
43 random += chars[distribution(dre)];
50 std::vector<std::byte> bytes;
51 bytes.reserve(hex.length() / 2);
53 for (
size_t i{0}; i < hex.length(); i += 2) {
55 auto [ptr, ec] = std::from_chars(hex.data() + i, hex.data() + i + 2, byte_value, 16);
57 if (ec != std::errc{} || ptr != hex.data() + i + 2) {
58 throw std::invalid_argument(
"Invalid hex character");
60 bytes.push_back(
static_cast<std::byte
>(byte_value));
67 std::ostringstream oss;
70 for (
auto it = bytes.rbegin(); it != bytes.rend(); ++it) {
71 oss << std::hex << std::setw(2) << std::setfill(
'0')
72 <<
static_cast<unsigned int>(
static_cast<uint8_t
>(*it));
79 ScriptVerificationFlags::NULLDUMMY | ScriptVerificationFlags::CHECKLOCKTIMEVERIFY |
80 ScriptVerificationFlags::CHECKSEQUENCEVERIFY};
83void check_equal(std::span<const std::byte> _actual, std::span<const std::byte> _expected,
bool equal =
true)
85 std::span<const uint8_t> actual{
reinterpret_cast<const unsigned char*
>(_actual.data()), _actual.size()};
86 std::span<const uint8_t> expected{
reinterpret_cast<const unsigned char*
>(_expected.data()), _expected.size()};
87 BOOST_CHECK_EQUAL_COLLECTIONS(
88 actual.begin(), actual.end(),
89 expected.begin(), expected.end());
97 std::cout <<
"kernel: " << message;
120 BOOST_CHECK_GT(timestamp, 0);
125 std::cout <<
"Kernel warning is set: " << message << std::endl;
130 std::cout <<
"Kernel warning was unset." << std::endl;
135 std::cout << error << std::endl;
140 std::cout << error << std::endl;
152 auto ser_block{block.
ToBytes()};
158 case ValidationMode::VALID: {
159 std::cout <<
"Valid block" << std::endl;
163 std::cout <<
"Invalid block: ";
166 case BlockValidationResult::UNSET:
167 std::cout <<
"initial value. Block has not yet been rejected" << std::endl;
169 case BlockValidationResult::HEADER_LOW_WORK:
170 std::cout <<
"the block header may be on a too-little-work chain" << std::endl;
172 case BlockValidationResult::CONSENSUS:
173 std::cout <<
"invalid by consensus rules (excluding any below reasons)" << std::endl;
175 case BlockValidationResult::CACHED_INVALID:
176 std::cout <<
"this block was cached as being invalid and we didn't store the reason why" << std::endl;
178 case BlockValidationResult::INVALID_HEADER:
179 std::cout <<
"invalid proof of work or time too old" << std::endl;
181 case BlockValidationResult::MUTATED:
182 std::cout <<
"the block's data didn't match the data committed to by the PoW" << std::endl;
184 case BlockValidationResult::MISSING_PREV:
185 std::cout <<
"We don't have the previous block the checked one is built on" << std::endl;
187 case BlockValidationResult::INVALID_PREV:
188 std::cout <<
"A block this one builds on is invalid" << std::endl;
190 case BlockValidationResult::TIME_FUTURE:
191 std::cout <<
"block timestamp was > 2 hours in the future (or our clock is bad)" << std::endl;
196 case ValidationMode::INTERNAL_ERROR: {
197 std::cout <<
"Internal error" << std::endl;
205 std::cout <<
"Block connected." << std::endl;
210 std::cout <<
"Block passed pow verification" << std::endl;
215 std::cout <<
"Block disconnected." << std::endl;
224 unsigned int input_index,
227 auto status = ScriptVerifyStatus::OK;
246 BOOST_CHECK(status == ScriptVerifyStatus::ERROR_SPENT_OUTPUTS_REQUIRED);
270 {
t.ToBytes() } -> std::convertible_to<std::span<const std::byte>>;
278 BOOST_CHECK(
object.get() != distinct_object.get());
281 const auto object_bytes =
object.ToBytes();
282 const auto distinct_bytes = distinct_object.ToBytes();
283 BOOST_CHECK(!std::ranges::equal(object_bytes, distinct_bytes));
287 T object2(distinct_object);
288 BOOST_CHECK_NE(distinct_object.get(), object2.get());
290 check_equal(distinct_object.ToBytes(), object2.ToBytes());
294 T object3{distinct_object};
296 BOOST_CHECK_NE(object3.get(), object2.get());
302 auto* original_ptr = object2.get();
303 T object4{std::move(object2)};
311 original_ptr = object4.get();
312 object2 = std::move(object4);
320template <
typename RangeType>
321 requires std::ranges::random_access_range<RangeType>
324 using value_type = std::ranges::range_value_t<RangeType>;
327 BOOST_REQUIRE(range.size() > 0);
328 BOOST_REQUIRE(!range.empty());
331 BOOST_CHECK_EQUAL(std::distance(range.begin(), range.end()),
static_cast<std::ptrdiff_t
>(expected_size));
335 for (
size_t i = 0; i < range.size(); ++i) {
344 auto it = range.begin();
373 for (
auto rit = range.end(); rit != range.begin();) {
379 std::vector<value_type> collected;
380 for (
const auto& elem : range) {
381 collected.push_back(elem);
394 auto tx_data{
hex_string_to_byte_vec(
"02000000013f7cebd65c27431a90bba7f796914fe8cc2ddfc3f2cbd6f7e5f2fc854534da95000000006b483045022100de1ac3bcdfb0332207c4a91f3832bd2c2915840165f876ab47c5f8996b971c3602201c6c053d750fadde599e6f5c4e1963df0f01fc0d97815e8157e3d59fe09ca30d012103699b464d1d8bc9e47d4fb1cdaa89a1c5783d68363c4dbc4b524ed3d857148617feffffff02836d3c01000000001976a914fc25d6d5c94003bf5b0c7b640a248e2c637fcfb088ac7ada8202000000001976a914fbed3d9b11183209a57999d54d59f67c019e756c88ac6acb0700")};
396 auto tx_data_2{
hex_string_to_byte_vec(
"02000000000101904f4ee5c87d20090b642f116e458cd6693292ad9ece23e72f15fb6c05b956210500000000fdffffff02e2010000000000002251200839a723933b56560487ec4d67dda58f09bae518ffa7e148313c5696ac837d9f10060000000000002251205826bcdae7abfb1c468204170eab00d887b61ab143464a4a09e1450bdc59a3340140f26e7af574e647355830772946356c27e7bbc773c5293688890f58983499581be84de40be7311a14e6d6422605df086620e75adae84ff06b75ce5894de5e994a00000000")};
408 auto broken_tx_data{std::span<std::byte>{tx_data.begin(), tx_data.begin() + 10}};
410 auto input{tx.GetInput(0)};
412 auto output{tx.GetOutput(tx.CountOutputs() - 1)};
414 auto script_pubkey{output.GetScriptPubkey()};
422 BOOST_CHECK_NE(output.
get(), output2.
get());
425 BOOST_CHECK_NE(output3.
get(), output2.
get());
430 BOOST_CHECK_NE(
script.get(), script2.
get());
435 BOOST_CHECK_NE(script3.
get(), script2.
get());
453 int64_t total_amount{0};
454 for (
const auto output : tx.Outputs()) {
455 total_amount += output.Amount();
459 auto amount = *(tx.Outputs() | std::ranges::views::filter([](
const auto& output) {
460 return output.Amount() == 42130042;
462 std::views::transform([](
const auto& output) {
463 return output.Amount();
465 BOOST_REQUIRE(amount);
471 check_equal(script_pubkey_roundtrip.ToBytes(), script_pubkey.ToBytes());
477 std::vector<std::byte> script_data_2 = script_data;
478 script_data_2.push_back(std::byte{0x51});
483 std::span<std::byte> empty_data{};
498 Transaction tx{
hex_string_to_byte_vec(
"020000000248c03e66fd371c7033196ce24298628e59ebefa00363026044e0f35e0325a65d000000006a473044022004893432347f39beaa280e99da595681ddb20fc45010176897e6e055d716dbfa022040a9e46648a5d10c33ef7cee5e6cf4b56bd513eae3ae044f0039824b02d0f44c012102982331a52822fd9b62e9b5d120da1d248558fac3da3a3c51cd7d9c8ad3da760efeffffffb856678c6e4c3c84e39e2ca818807049d6fba274b42af3c6d3f9d4b6513212d2000000006a473044022068bcedc7fe39c9f21ad318df2c2da62c2dc9522a89c28c8420ff9d03d2e6bf7b0220132afd752754e5cb1ea2fd0ed6a38ec666781e34b0e93dc9a08f2457842cf5660121033aeb9c079ea3e08ea03556182ab520ce5c22e6b0cb95cee6435ee17144d860cdfeffffff0260d50b00000000001976a914363cc8d55ea8d0500de728ef6d63804ddddbdc9888ac67040f00000000001976a914c303bdc5064bf9c9a8b507b5496bd0987285707988ac6acb0700")};
509 auto tx_data{
hex_string_to_byte_vec(
"02000000013f7cebd65c27431a90bba7f796914fe8cc2ddfc3f2cbd6f7e5f2fc854534da95000000006b483045022100de1ac3bcdfb0332207c4a91f3832bd2c2915840165f876ab47c5f8996b971c3602201c6c053d750fadde599e6f5c4e1963df0f01fc0d97815e8157e3d59fe09ca30d012103699b464d1d8bc9e47d4fb1cdaa89a1c5783d68363c4dbc4b524ed3d857148617feffffff02836d3c01000000001976a914fc25d6d5c94003bf5b0c7b640a248e2c637fcfb088ac7ada8202000000001976a914fbed3d9b11183209a57999d54d59f67c019e756c88ac6acb0700")};
511 auto tx_data_2{
hex_string_to_byte_vec(
"02000000000101904f4ee5c87d20090b642f116e458cd6693292ad9ece23e72f15fb6c05b956210500000000fdffffff02e2010000000000002251200839a723933b56560487ec4d67dda58f09bae518ffa7e148313c5696ac837d9f10060000000000002251205826bcdae7abfb1c468204170eab00d887b61ab143464a4a09e1450bdc59a3340140f26e7af574e647355830772946356c27e7bbc773c5293688890f58983499581be84de40be7311a14e6d6422605df086620e75adae84ff06b75ce5894de5e994a00000000")};
521 CheckHandle(precomputed_txdata, precomputed_txdata_2);
528 auto legacy_spending_tx{
Transaction{
hex_string_to_byte_vec(
"02000000013f7cebd65c27431a90bba7f796914fe8cc2ddfc3f2cbd6f7e5f2fc854534da95000000006b483045022100de1ac3bcdfb0332207c4a91f3832bd2c2915840165f876ab47c5f8996b971c3602201c6c053d750fadde599e6f5c4e1963df0f01fc0d97815e8157e3d59fe09ca30d012103699b464d1d8bc9e47d4fb1cdaa89a1c5783d68363c4dbc4b524ed3d857148617feffffff02836d3c01000000001976a914fc25d6d5c94003bf5b0c7b640a248e2c637fcfb088ac7ada8202000000001976a914fbed3d9b11183209a57999d54d59f67c019e756c88ac6acb0700")}};
530 legacy_spent_script_pubkey,
543 legacy_spent_script_pubkey,
545 &legacy_precomputed_txdata,
552 auto segwit_spending_tx{
Transaction{
hex_string_to_byte_vec(
"010000000001011f97548fbbe7a0db7588a66e18d803d0089315aa7d4cc28360b6ec50ef36718a0100000000ffffffff02df1776000000000017a9146c002a686959067f4866b8fb493ad7970290ab728757d29f0000000000220020701a8d401c84fb13e6baf169d59684e17abd9fa216c8cc5b9fc63d622ff8c58d04004730440220565d170eed95ff95027a69b313758450ba84a01224e1f7f130dda46e94d13f8602207bdd20e307f062594022f12ed5017bbf4a055a06aea91c10110a0e3bb23117fc014730440220647d2dc5b15f60bc37dc42618a370b2a1490293f9e5c8464f53ec4fe1dfe067302203598773895b4b16d37485cbe21b337f4e4b650739880098c592553add7dd4355016952210375e00eb72e29da82b89367947f29ef34afb75e8654f6ea368e0acdfd92976b7c2103a1b26313f430c4b15bb1fdce663207659d8cac749a0e53d70eff01874496feff2103c96d495bfdd5ba4145e3e046fee45e84a8a48ad05bd8dbb395c011a32cf9f88053ae00000000")}};
554 segwit_spent_script_pubkey,
567 segwit_spent_script_pubkey,
569 &segwit_precomputed_txdata,
576 auto taproot_spending_tx{
Transaction{
hex_string_to_byte_vec(
"01000000000101d1f1c1f8cdf6759167b90f52c9ad358a369f95284e841d7a2536cef31c0549580100000000fdffffff020000000000000000316a2f49206c696b65205363686e6f7272207369677320616e6420492063616e6e6f74206c69652e204062697462756734329e06010000000000225120a37c3903c8d0db6512e2b40b0dffa05e5a3ab73603ce8c9c4b7771e5412328f90140a60c383f71bac0ec919b1d7dbc3eb72dd56e7aa99583615564f9f99b8ae4e837b758773a5b2e4c51348854c8389f008e05029db7f464a5ff2e01d5e6e626174affd30a00")}};
577 std::vector<TransactionOutput> taproot_spent_outputs;
578 taproot_spent_outputs.emplace_back(taproot_spent_script_pubkey, 88480);
581 taproot_spent_outputs,
584 taproot_spent_script_pubkey,
586 &taproot_precomputed_txdata,
594 auto taproot2_spending_tx{
Transaction{
hex_string_to_byte_vec(
"02000000000102c0f01ead18750892c84b1d4f595149ad38f16847df1fbf490e235b3b78c1f98a0100000000ffffffff456764a19c2682bf5b1567119f06a421849ad1664cf42b5ef95b69d6e2159e9d0000000000ffffffff022202000000000000225120b6c0c2a8ee25a2ae0322ab7f1a06f01746f81f6b90d179c3c2a51a356e6188f1d70e020000000000225120b7da80f57e36930b0515eb09293e25858d13e6b91fee6184943f5a584cb4248e0141933fdc49eb1af1f08ed1e9cf5559259309a8acd25ff1e6999b6955124438aef4fceaa4e6a5f85286631e24837329563595bc3cf4b31e1c687442abb01c4206818101401c9620faf1e8c84187762ad14d04ae3857f59a2f03f1dcbb99290e16dfc572a63b4ea435780a5787af59beb5742fd71cda8a95381517a1ff14b4c67996c4bf8100000000")}};
595 std::vector<TransactionOutput> taproot2_spent_outputs;
596 taproot2_spent_outputs.emplace_back(taproot2_spent_script_pubkey0, 546);
597 taproot2_spent_outputs.emplace_back(taproot2_spent_script_pubkey1, 135125);
599 taproot2_spending_tx,
600 taproot2_spent_outputs,
603 taproot2_spent_script_pubkey0,
604 taproot2_spending_tx,
605 &taproot2_precomputed_txdata,
610 taproot2_spent_script_pubkey1,
611 taproot2_spending_tx,
612 &taproot2_precomputed_txdata,
622 .log_time_micros =
true,
623 .log_threadnames =
false,
624 .log_sourcelocations =
false,
625 .always_print_category_levels =
true,
638 Logger logger{std::make_unique<TestLog>()};
639 Logger logger_2{std::make_unique<TestLog>()};
641 Logger logger{std::make_unique<TestLog>()};
662 options.SetChainParams(params);
663 options.SetNotifications(std::make_shared<TestKernelNotifications>());
671 BlockHeader header_0{
hex_string_to_byte_vec(
"00e07a26beaaeee2e71d7eb19279545edbaf15de0999983626ec00000000000000000000579cf78b65229bfb93f4a11463af2eaa5ad91780f27f5d147a423bea5f7e4cdf2a47e268b4dd01173a9662ee")};
673 BlockHeader header_1{
hex_string_to_byte_vec(
"00c00020e7cb7b4de21d26d55bd384017b8bb9333ac3b2b55bed00000000000000000000d91b4484f801b99f03d36b9d26cfa83420b67f81da12d7e6c1e7f364e743c5ba9946e268b4dd011799c8533d")};
681 auto mainnet_block_1_header =
hex_string_to_byte_vec(
"010000006fe28c0ab6f1b372c1a6a246ae63f74f931e8365e15a089c68d6190000000000982051fd1e4ba744bbbe680e1fee14677ba1a3c3540bf7b1cdb606e857233e0e61bc6649ffff001d01e36299");
688 auto prev_hash = header.PrevHash();
693 check_equal(header_roundtrip.ToBytes(), mainnet_block_1_header);
695 auto raw_block =
hex_string_to_byte_vec(
"010000006fe28c0ab6f1b372c1a6a246ae63f74f931e8365e15a089c68d6190000000000982051fd1e4ba744bbbe680e1fee14677ba1a3c3540bf7b1cdb606e857233e0e61bc6649ffff001d01e362990101000000010000000000000000000000000000000000000000000000000000000000000000ffffffff0704ffff001d0104ffffffff0100f2052a0100000043410496b538e853519c726a2c91e61ec11600ae1390813a627c66fb8be7947be63c52da7589379515d4e0a604f8141781e62294721166bf621e73a82cbf2342c858eeac00000000");
696 Block block{raw_block};
705 auto block_header_bytes = block_header.ToBytes();
707 check_equal(block_header_bytes, std::span<const std::byte>(raw_block.data(), 80));
716 CheckRange(block_tx.Transactions(), block_tx.CountTransactions());
723Context create_context(std::shared_ptr<TestKernelNotifications> notifications,
ChainType chain_type, std::shared_ptr<TestValidationInterface> validation_interface =
nullptr)
727 options.SetChainParams(params);
728 options.SetNotifications(notifications);
729 if (validation_interface) {
730 options.SetValidationInterface(validation_interface);
732 auto context{
Context{options}};
738 Logger logger{std::make_unique<TestLog>()};
739 auto test_directory{
TestDirectory{
"chainman_test_bitcoin_kernel"}};
743 ChainstateManagerOptions chainman_opts{context, PathToString(test_directory.m_directory), PathToString(test_directory.m_directory /
"blocks")};
744 ChainMan chainman{context, chainman_opts};
750 ChainstateManagerOptions chainman_opts{context, PathToString(test_directory.m_directory), PathToString(test_directory.m_directory /
"blocks")};
751 ChainMan chainman{context, chainman_opts};
755 auto valid_dir{PathToString(test_directory.m_directory)};
756 std::vector<std::pair<std::string_view, std::string_view>> illegal_cases{
758 {valid_dir, {
nullptr, 0}},
760 {{
nullptr, 0}, {
nullptr, 0}},
762 for (
auto& [data_dir, blocks_dir] : illegal_cases) {
768 auto notifications{std::make_shared<TestKernelNotifications>()};
771 ChainstateManagerOptions chainman_opts{context, PathToString(test_directory.m_directory), PathToString(test_directory.m_directory /
"blocks")};
773 BOOST_CHECK(!chainman_opts.SetWipeDbs(
true,
false));
775 BOOST_CHECK(chainman_opts.SetWipeDbs(
false,
true));
776 BOOST_CHECK(chainman_opts.SetWipeDbs(
false,
false));
777 ChainMan chainman{context, chainman_opts};
782 bool wipe_chainstate,
783 bool block_tree_db_in_memory,
784 bool chainstate_db_in_memory,
790 chainman_opts.SetWipeDbs(reindex, reindex);
792 if (wipe_chainstate) {
793 chainman_opts.SetWipeDbs(
false, wipe_chainstate);
795 if (block_tree_db_in_memory) {
796 chainman_opts.UpdateBlockTreeDbInMemory(block_tree_db_in_memory);
798 if (chainstate_db_in_memory) {
799 chainman_opts.UpdateChainstateDbInMemory(chainstate_db_in_memory);
802 auto chainman{std::make_unique<ChainMan>(context, chainman_opts)};
808 auto notifications{std::make_shared<TestKernelNotifications>()};
811 test_directory,
true,
false,
812 false,
false, context)};
814 std::vector<std::string> import_files;
818 auto chain{chainman->GetChain()};
820 auto genesis_index{chain.Entries().front()};
822 auto genesis_block_raw{chainman->ReadBlock(genesis_index).value().ToBytes()};
823 auto first_index{chain.GetByHeight(0)};
824 auto first_block_raw{chainman->ReadBlock(genesis_index).value().ToBytes()};
826 auto height{first_index.GetHeight()};
829 auto next_index{chain.GetByHeight(first_index.GetHeight() + 1)};
831 auto next_block_data{chainman->ReadBlock(next_index).value().ToBytes()};
832 auto tip_index{chain.Entries().back()};
833 auto tip_block_data{chainman->ReadBlock(tip_index).value().ToBytes()};
834 auto second_index{chain.GetByHeight(1)};
835 auto second_block{chainman->ReadBlock(second_index).value()};
836 auto second_block_data{second_block.ToBytes()};
837 auto second_height{second_index.GetHeight()};
842 auto second_hash{second_index.GetHash()};
843 auto another_second_index{chainman->GetBlockTreeEntry(second_hash)};
845 auto another_second_height{another_second_index->GetHeight()};
846 auto second_block_hash{second_block.GetHash()};
847 check_equal(second_block_hash.ToBytes(), second_hash.ToBytes());
853 auto notifications{std::make_shared<TestKernelNotifications>()};
856 test_directory,
false,
true,
857 false,
false, context)};
859 std::vector<std::string> import_files;
860 import_files.push_back(PathToString(test_directory.
m_directory /
"blocks" /
"blk00000.dat"));
866 auto notifications{std::make_shared<TestKernelNotifications>()};
867 auto validation_interface{std::make_shared<TestValidationInterface>()};
868 auto context{
create_context(notifications, ChainType::MAINNET, validation_interface)};
870 test_directory,
false,
false,
871 false,
false, context)};
874 auto raw_block =
hex_string_to_byte_vec(
"010000006fe28c0ab6f1b372c1a6a246ae63f74f931e8365e15a089c68d6190000000000982051fd1e4ba744bbbe680e1fee14677ba1a3c3540bf7b1cdb606e857233e0e61bc6649ffff001d01e362990101000000010000000000000000000000000000000000000000000000000000000000000000ffffffff0704ffff001d0104ffffffff0100f2052a0100000043410496b538e853519c726a2c91e61ec11600ae1390813a627c66fb8be7947be63c52da7589379515d4e0a604f8141781e62294721166bf621e73a82cbf2342c858eeac00000000");
875 Block block{raw_block};
877 TransactionView tx{block.GetTransaction(block.CountTransactions() - 1)};
886 for (
auto transaction : block.Transactions()) {
889 auto output_counts = *(block.Transactions() | std::views::transform([](
const auto& tx) {
890 return tx.CountOutputs();
894 validation_interface->m_expected_valid_block.emplace(raw_block);
895 auto ser_block{block.ToBytes()};
897 bool new_block =
false;
898 BOOST_CHECK(chainman->ProcessBlock(block, &new_block));
901 validation_interface->m_expected_valid_block = std::nullopt;
904 BOOST_CHECK(!chainman->ProcessBlock(invalid_block, &new_block));
907 auto chain{chainman->GetChain()};
909 auto tip{chain.Entries().back()};
910 auto read_block{chainman->ReadBlock(tip)};
911 BOOST_REQUIRE(read_block);
912 check_equal(read_block.value().ToBytes(), raw_block);
916 Block read_block_2{*chainman->ReadBlock(tip_2)};
924 BOOST_CHECK(chainman->ProcessBlock(block, &new_block));
930 constexpr size_t MERKLE_ROOT_OFFSET{4 + 32};
931 constexpr size_t NBITS_OFFSET{4 + 32 + 32 + 4};
932 constexpr size_t COINBASE_PREVOUT_N_OFFSET{4 + 32 + 32 + 4 + 4 + 4 + 1 + 4 + 1 + 32};
935 auto raw_block =
hex_string_to_byte_vec(
"010000006fe28c0ab6f1b372c1a6a246ae63f74f931e8365e15a089c68d6190000000000982051fd1e4ba744bbbe680e1fee14677ba1a3c3540bf7b1cdb606e857233e0e61bc6649ffff001d01e362990101000000010000000000000000000000000000000000000000000000000000000000000000ffffffff0704ffff001d0104ffffffff0100f2052a0100000043410496b538e853519c726a2c91e61ec11600ae1390813a627c66fb8be7947be63c52da7589379515d4e0a604f8141781e62294721166bf621e73a82cbf2342c858eeac00000000");
942 Block block{raw_block};
945 BOOST_CHECK(block.Check(consensus_params, BlockCheckFlags::BASE, state));
951 auto bad_merkle_block_data = raw_block;
952 bad_merkle_block_data[MERKLE_ROOT_OFFSET] ^= std::byte{0x01};
953 Block bad_merkle_block{bad_merkle_block_data};
955 BOOST_CHECK(!bad_merkle_block.Check(consensus_params, BlockCheckFlags::MERKLE, state));
959 BOOST_CHECK(bad_merkle_block.Check(consensus_params, BlockCheckFlags::BASE, state));
962 auto bad_pow_block_data = raw_block;
963 bad_pow_block_data[NBITS_OFFSET + 3] = std::byte{0x1c};
964 Block bad_pow_block{bad_pow_block_data};
966 BOOST_CHECK(!bad_pow_block.Check(consensus_params, BlockCheckFlags::POW, state));
970 BOOST_CHECK(bad_pow_block.Check(consensus_params, BlockCheckFlags::MERKLE, state));
973 auto bad_base_block_data = raw_block;
974 bad_base_block_data[COINBASE_PREVOUT_N_OFFSET] = std::byte{0x00};
975 Block bad_base_block{bad_base_block_data};
977 BOOST_CHECK(!bad_base_block.Check(consensus_params, BlockCheckFlags::BASE, state));
982 auto truncated_block_data =
hex_string_to_byte_vec(
"010000006fe28c0ab6f1b372c1a6a246ae63f74f931e8365e15a089c68d6190000000000982051fd1e4ba744bbbe680e1fee14677ba1a3c3540bf7b1cdb606e857233e0e61bc6649ffff001d01e36299");
983 BOOST_CHECK_EXCEPTION(
Block{truncated_block_data}, std::runtime_error,
984 HasReason{
"failed to instantiate btck object"});
989 auto test_directory{
TestDirectory{
"mainnet_test_bitcoin_kernel"}};
997 std::array<std::byte, 32> test_hash;
998 std::array<std::byte, 32> test_hash_2;
999 for (
int i = 0; i < 32; ++i) {
1000 test_hash[i] =
static_cast<std::byte
>(i);
1001 test_hash_2[i] =
static_cast<std::byte
>(i + 1);
1012 auto test_directory{
TestDirectory{
"block_tree_entry_test_bitcoin_kernel"}};
1013 auto notifications{std::make_shared<TestKernelNotifications>()};
1024 for (
size_t i{0}; i < 3; i++) {
1026 bool new_block{
false};
1027 chainman->ProcessBlock(block, &new_block);
1031 auto chain{chainman->GetChain()};
1032 auto entry_0{chain.GetByHeight(0)};
1033 auto entry_1{chain.GetByHeight(1)};
1034 auto entry_2{chain.GetByHeight(2)};
1047 auto prev{entry_1.GetPrevious()};
1054 auto in_memory_test_directory{
TestDirectory{
"in-memory_test_bitcoin_kernel"}};
1056 auto notifications{std::make_shared<TestKernelNotifications>()};
1059 in_memory_test_directory,
false,
false,
1060 true,
true, context)};
1064 bool new_block{
false};
1065 chainman->ProcessBlock(block, &new_block);
1078 auto test_directory{
TestDirectory{
"regtest_test_bitcoin_kernel"}};
1080 auto notifications{std::make_shared<TestKernelNotifications>()};
1085 test_directory,
false,
false,
1086 false,
false, context)};
1091 BOOST_CHECK(state.GetBlockValidationResult() == BlockValidationResult::UNSET);
1092 BOOST_CHECK(chainman->ProcessBlockHeader(header, state));
1093 BOOST_CHECK(state.GetValidationMode() == ValidationMode::VALID);
1095 BOOST_CHECK(!chainman->GetChain().Contains(entry));
1098 BOOST_CHECK(hash == best_entry.GetHeader().Hash());
1109 test_directory,
false,
false,
1110 false,
false, context)};
1111 for (
size_t i{0}; i < mid; i++) {
1113 bool new_block{
false};
1114 BOOST_CHECK(chainman->ProcessBlock(block, &new_block));
1120 test_directory,
false,
false,
1121 false,
false, context)};
1125 bool new_block{
false};
1126 BOOST_CHECK(chainman->ProcessBlock(block, &new_block));
1130 auto chain = chainman->GetChain();
1131 auto tip = chain.Entries().back();
1132 auto read_block = chainman->ReadBlock(tip).value();
1135 auto tip_2 = tip.GetPrevious().value();
1136 auto read_block_2 = chainman->ReadBlock(tip_2).value();
1139 Txid txid = read_block.Transactions()[0].
Txid();
1140 Txid txid_2 = read_block_2.Transactions()[0].
Txid();
1145 auto find_transaction = [&chainman](
const TxidView& target_txid) -> std::optional<Transaction> {
1146 auto chain = chainman->GetChain();
1147 for (
const auto block_tree_entry : chain.Entries()) {
1148 auto block{chainman->ReadBlock(block_tree_entry)};
1150 if (transaction.Txid() == target_txid) {
1155 return std::nullopt;
1158 for (
const auto block_tree_entry : chain.Entries()) {
1159 auto block{chainman->ReadBlock(block_tree_entry)};
1160 for (
const auto transaction : block->Transactions()) {
1161 std::vector<TransactionInput> inputs;
1162 std::vector<TransactionOutput> spent_outputs;
1163 for (
const auto input : transaction.Inputs()) {
1165 if (point.
index() == std::numeric_limits<uint32_t>::max()) {
1168 inputs.emplace_back(input);
1170 std::optional<Transaction> tx = find_transaction(point.
Txid());
1173 spent_outputs.emplace_back(tx->GetOutput(point.
index()));
1175 BOOST_CHECK(inputs.size() == spent_outputs.size());
1178 for (
size_t i{0}; i < inputs.size(); ++i) {
1186 BlockSpentOutputs block_spent_outputs_prev{chainman->ReadBlockSpentOutputs(*tip.GetPrevious())};
1187 CheckHandle(block_spent_outputs, block_spent_outputs_prev);
1188 CheckRange(block_spent_outputs_prev.TxsSpentOutputs(), block_spent_outputs_prev.Count());
1194 TransactionSpentOutputs owned_transaction_spent_outputs_prev{block_spent_outputs_prev.GetTxSpentOutputs(block_spent_outputs_prev.Count() - 1)};
1195 CheckHandle(owned_transaction_spent_outputs, owned_transaction_spent_outputs_prev);
1196 CheckRange(transaction_spent_outputs.Coins(), transaction_spent_outputs.Count());
1199 CoinView coin{transaction_spent_outputs.GetCoin(transaction_spent_outputs.Count() - 1)};
1201 Coin owned_coin{coin};
1202 Coin owned_coin_prev{owned_transaction_spent_outputs_prev.GetCoin(owned_transaction_spent_outputs_prev.Count() - 1)};
1207 uint32_t coin_height = coin.GetConfirmationHeight();
1213 auto script_pubkey_bytes{script_pubkey.
ToBytes()};
1215 auto round_trip_script_pubkey{
ScriptPubkey(script_pubkey_bytes)};
1218 for (
const auto tx_spent_outputs : block_spent_outputs.TxsSpentOutputs()) {
1219 for (
const auto coins : tx_spent_outputs.Coins()) {
1220 BOOST_CHECK_GT(coins.GetOutput().Amount(), 1);
1224 CheckRange(chain.Entries(), chain.CountEntries());
1227 std::optional<Block> block{chainman->ReadBlock(entry)};
1235 BOOST_CHECK_GT(output.
Amount(), 1);
1242 for (
const auto entry : chain.Entries()) {
1249 fs::remove(test_directory.m_directory /
"blocks" /
"blk00000.dat");
1250 BOOST_CHECK(!chainman->ReadBlock(tip_2).has_value());
1251 fs::remove(test_directory.m_directory /
"blocks" /
"rev00000.dat");
constexpr std::array< std::string_view, 206 > REGTEST_BLOCK_DATA
BOOST_CHECK_EXCEPTION predicates to check the specific validation error.
void HeaderTipHandler(SynchronizationState state, int64_t height, int64_t timestamp, bool presync) override
void FlushErrorHandler(std::string_view error) override
void FatalErrorHandler(std::string_view error) override
void WarningUnsetHandler(Warning warning) override
void WarningSetHandler(Warning warning, std::string_view message) override
void LogMessage(std::string_view message)
void PowValidBlock(BlockTreeEntry entry, Block block) override
void BlockDisconnected(Block block, BlockTreeEntry entry) override
std::optional< std::string > m_expected_valid_block
void BlockChecked(Block block, BlockValidationStateView state) override
void BlockConnected(Block block, BlockTreeEntry entry) override
std::vector< std::byte > ToBytes() const
std::optional< BlockTreeEntry > GetPrevious() const
ValidationMode GetValidationMode() const
BlockValidationResult GetBlockValidationResult() const
ConsensusParamsView GetConsensusParams() const
void SetWorkerThreads(int worker_threads)
std::vector< std::byte > ToBytes() const
bool Verify(int64_t amount, const Transaction &tx_to, const PrecomputedTransactionData *precomputed_txdata, unsigned int input_index, ScriptVerificationFlags flags, ScriptVerifyStatus &status) const
size_t CountInputs() const
std::vector< std::byte > ToBytes() const
ScriptPubkeyView GetScriptPubkey() const
Txid(const TxidView &view)
const CType * get() const
static bool exists(const path &p)
#define T(expected, seed, data)
void logging_set_options(const btck_LoggingOptions &logging_options)
void logging_set_level_category(LogCategory category, LogLevel level)
void logging_enable_category(LogCategory category)
void logging_disable_category(LogCategory category)
static const auto INVALID
A stack representing the lack of any (dis)satisfactions.
#define BOOST_CHECK_THROW(stmt, excMatch)
#define BOOST_CHECK_EQUAL(v1, v2)
#define BOOST_CHECK(expr)
static bool Verify(const CScript &scriptSig, const CScript &scriptPubKey, bool fStrict, ScriptError &err)
TestDirectory(std::string directory_name)
Options controlling the format of log messages.
int log_timestamps
Prepend a timestamp to log messages.
std::string byte_span_to_hex_string_reversed(std::span< const std::byte > bytes)
void chainman_reindex_chainstate_test(TestDirectory &test_directory)
Context create_context(std::shared_ptr< TestKernelNotifications > notifications, ChainType chain_type, std::shared_ptr< TestValidationInterface > validation_interface=nullptr)
void run_verify_test(const ScriptPubkey &spent_script_pubkey, const Transaction &spending_tx, const PrecomputedTransactionData *precomputed_txdata, int64_t amount, unsigned int input_index, bool taproot)
std::vector< std::byte > hex_string_to_byte_vec(std::string_view hex)
void chainman_reindex_test(TestDirectory &test_directory)
BOOST_AUTO_TEST_CASE(btck_transaction_tests)
void check_equal(std::span< const std::byte > _actual, std::span< const std::byte > _expected, bool equal=true)
void CheckHandle(T object, T distinct_object)
std::unique_ptr< ChainMan > create_chainman(TestDirectory &test_directory, bool reindex, bool wipe_chainstate, bool block_tree_db_in_memory, bool chainstate_db_in_memory, Context &context)
constexpr auto VERIFY_ALL_PRE_SEGWIT
void CheckRange(const RangeType &range, size_t expected_size)
void chainman_mainnet_validation_test(TestDirectory &test_directory)
std::string random_string(uint32_t length)
constexpr auto VERIFY_ALL_PRE_TAPROOT