Bitcoin Core 31.99.0
P2P Digital Currency
test_kernel.cpp
Go to the documentation of this file.
1// Copyright (c) 2024-present The Bitcoin Core developers
2// Distributed under the MIT software license, see the accompanying
3// file COPYING or http://www.opensource.org/licenses/mit-license.php.
4
7#include <util/fs.h>
8
9#define BOOST_TEST_MODULE Bitcoin Kernel Test Suite
10#include <boost/test/included/unit_test.hpp>
11
13#include <test/util/common.h>
14
15#include <charconv>
16#include <cstdint>
17#include <cstdlib>
18#include <iostream>
19#include <memory>
20#include <optional>
21#include <random>
22#include <ranges>
23#include <span>
24#include <string>
25#include <string_view>
26#include <vector>
27
28using namespace btck;
29
30std::string random_string(uint32_t length)
31{
32 const std::string chars = "0123456789"
33 "abcdefghijklmnopqrstuvwxyz"
34 "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
35
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);
39
40 std::string random;
41 random.reserve(length);
42 for (uint32_t i = 0; i < length; i++) {
43 random += chars[distribution(dre)];
44 }
45 return random;
46}
47
48std::vector<std::byte> hex_string_to_byte_vec(std::string_view hex)
49{
50 std::vector<std::byte> bytes;
51 bytes.reserve(hex.length() / 2);
52
53 for (size_t i{0}; i < hex.length(); i += 2) {
54 uint8_t byte_value;
55 auto [ptr, ec] = std::from_chars(hex.data() + i, hex.data() + i + 2, byte_value, 16);
56
57 if (ec != std::errc{} || ptr != hex.data() + i + 2) {
58 throw std::invalid_argument("Invalid hex character");
59 }
60 bytes.push_back(static_cast<std::byte>(byte_value));
61 }
62 return bytes;
63}
64
65std::string byte_span_to_hex_string_reversed(std::span<const std::byte> bytes)
66{
67 std::ostringstream oss;
68
69 // Iterate in reverse order
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));
73 }
74
75 return oss.str();
76}
77
78constexpr auto VERIFY_ALL_PRE_SEGWIT{ScriptVerificationFlags::P2SH | ScriptVerificationFlags::DERSIG |
79 ScriptVerificationFlags::NULLDUMMY | ScriptVerificationFlags::CHECKLOCKTIMEVERIFY |
80 ScriptVerificationFlags::CHECKSEQUENCEVERIFY};
81constexpr auto VERIFY_ALL_PRE_TAPROOT{VERIFY_ALL_PRE_SEGWIT | ScriptVerificationFlags::WITNESS};
82
83void check_equal(std::span<const std::byte> _actual, std::span<const std::byte> _expected, bool equal = true)
84{
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());
90}
91
93{
94public:
95 void LogMessage(std::string_view message)
96 {
97 std::cout << "kernel: " << message;
98 }
99};
100
102 fs::path m_directory;
103 TestDirectory(std::string directory_name)
104 : m_directory{fs::path{fs::temp_directory_path()} / fs::u8path(directory_name + "_🌽_" + random_string(16))}
105 {
106 fs::create_directories(m_directory);
107 }
108
110 {
111 fs::remove_all(m_directory);
112 }
113};
114
116{
117public:
118 void HeaderTipHandler(SynchronizationState state, int64_t height, int64_t timestamp, bool presync) override
119 {
120 BOOST_CHECK_GT(timestamp, 0);
121 }
122
123 void WarningSetHandler(Warning warning, std::string_view message) override
124 {
125 std::cout << "Kernel warning is set: " << message << std::endl;
126 }
127
128 void WarningUnsetHandler(Warning warning) override
129 {
130 std::cout << "Kernel warning was unset." << std::endl;
131 }
132
133 void FlushErrorHandler(std::string_view error) override
134 {
135 std::cout << error << std::endl;
136 }
137
138 void FatalErrorHandler(std::string_view error) override
139 {
140 std::cout << error << std::endl;
141 }
142};
143
145{
146public:
147 std::optional<std::vector<std::byte>> m_expected_valid_block = std::nullopt;
148
149 void BlockChecked(Block block, BlockValidationStateView state) override
150 {
151 if (m_expected_valid_block.has_value()) {
152 auto ser_block{block.ToBytes()};
153 check_equal(m_expected_valid_block.value(), ser_block);
154 }
155
156 auto mode{state.GetValidationMode()};
157 switch (mode) {
158 case ValidationMode::VALID: {
159 std::cout << "Valid block" << std::endl;
160 return;
161 }
163 std::cout << "Invalid block: ";
164 auto result{state.GetBlockValidationResult()};
165 switch (result) {
166 case BlockValidationResult::UNSET:
167 std::cout << "initial value. Block has not yet been rejected" << std::endl;
168 break;
169 case BlockValidationResult::HEADER_LOW_WORK:
170 std::cout << "the block header may be on a too-little-work chain" << std::endl;
171 break;
172 case BlockValidationResult::CONSENSUS:
173 std::cout << "invalid by consensus rules (excluding any below reasons)" << std::endl;
174 break;
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;
177 break;
178 case BlockValidationResult::INVALID_HEADER:
179 std::cout << "invalid proof of work or time too old" << std::endl;
180 break;
181 case BlockValidationResult::MUTATED:
182 std::cout << "the block's data didn't match the data committed to by the PoW" << std::endl;
183 break;
184 case BlockValidationResult::MISSING_PREV:
185 std::cout << "We don't have the previous block the checked one is built on" << std::endl;
186 break;
187 case BlockValidationResult::INVALID_PREV:
188 std::cout << "A block this one builds on is invalid" << std::endl;
189 break;
190 case BlockValidationResult::TIME_FUTURE:
191 std::cout << "block timestamp was > 2 hours in the future (or our clock is bad)" << std::endl;
192 break;
193 }
194 return;
195 }
196 case ValidationMode::INTERNAL_ERROR: {
197 std::cout << "Internal error" << std::endl;
198 return;
199 }
200 }
201 }
202
203 void BlockConnected(Block block, BlockTreeEntry entry) override
204 {
205 std::cout << "Block connected." << std::endl;
206 }
207
208 void PowValidBlock(BlockTreeEntry entry, Block block) override
209 {
210 std::cout << "Block passed pow verification" << std::endl;
211 }
212
213 void BlockDisconnected(Block block, BlockTreeEntry entry) override
214 {
215 std::cout << "Block disconnected." << std::endl;
216 }
217};
218
220 const ScriptPubkey& spent_script_pubkey,
221 const Transaction& spending_tx,
222 const PrecomputedTransactionData* precomputed_txdata,
223 int64_t amount,
224 unsigned int input_index,
225 bool taproot)
226{
227 auto status = ScriptVerifyStatus::OK;
228
229 if (taproot) {
230 BOOST_CHECK(spent_script_pubkey.Verify(
231 amount,
232 spending_tx,
233 precomputed_txdata,
234 input_index,
236 status));
237 BOOST_CHECK(status == ScriptVerifyStatus::OK);
238 } else {
239 BOOST_CHECK(!spent_script_pubkey.Verify(
240 amount,
241 spending_tx,
242 precomputed_txdata,
243 input_index,
245 status));
246 BOOST_CHECK(status == ScriptVerifyStatus::ERROR_SPENT_OUTPUTS_REQUIRED);
247 }
248
249 BOOST_CHECK(spent_script_pubkey.Verify(
250 amount,
251 spending_tx,
252 precomputed_txdata,
253 input_index,
255 status));
256 BOOST_CHECK(status == ScriptVerifyStatus::OK);
257
258 BOOST_CHECK(spent_script_pubkey.Verify(
259 0,
260 spending_tx,
261 precomputed_txdata,
262 input_index,
264 status));
265 BOOST_CHECK(status == ScriptVerifyStatus::OK);
266}
267
268template <typename T>
269concept HasToBytes = requires(T t) {
270 { t.ToBytes() } -> std::convertible_to<std::span<const std::byte>>;
271};
272
273template <typename T>
274void CheckHandle(T object, T distinct_object)
275{
276 BOOST_CHECK(object.get() != nullptr);
277 BOOST_CHECK(distinct_object.get() != nullptr);
278 BOOST_CHECK(object.get() != distinct_object.get());
279
280 if constexpr (HasToBytes<T>) {
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));
284 }
285
286 // Copy constructor
287 T object2(distinct_object);
288 BOOST_CHECK_NE(distinct_object.get(), object2.get());
289 if constexpr (HasToBytes<T>) {
290 check_equal(distinct_object.ToBytes(), object2.ToBytes());
291 }
292
293 // Copy assignment
294 T object3{distinct_object};
295 object2 = object3;
296 BOOST_CHECK_NE(object3.get(), object2.get());
297 if constexpr (HasToBytes<T>) {
298 check_equal(object3.ToBytes(), object2.ToBytes());
299 }
300
301 // Move constructor
302 auto* original_ptr = object2.get();
303 T object4{std::move(object2)};
304 BOOST_CHECK_EQUAL(object4.get(), original_ptr);
305 BOOST_CHECK_EQUAL(object2.get(), nullptr); // NOLINT(bugprone-use-after-move)
306 if constexpr (HasToBytes<T>) {
307 check_equal(object4.ToBytes(), object3.ToBytes());
308 }
309
310 // Move assignment
311 original_ptr = object4.get();
312 object2 = std::move(object4);
313 BOOST_CHECK_EQUAL(object2.get(), original_ptr);
314 BOOST_CHECK_EQUAL(object4.get(), nullptr); // NOLINT(bugprone-use-after-move)
315 if constexpr (HasToBytes<T>) {
316 check_equal(object2.ToBytes(), object3.ToBytes());
317 }
318
319 // Self move-assignment must not destroy the held resource.
320 // Use a reference to avoid -Wself-move warnings.
321 original_ptr = object2.get();
322 auto& object2_ref = object2;
323 object2 = std::move(object2_ref);
324 BOOST_CHECK_EQUAL(object2.get(), original_ptr);
325 if constexpr (HasToBytes<T>) {
326 check_equal(object2.ToBytes(), object3.ToBytes());
327 }
328}
329
330template <typename RangeType>
331 requires std::ranges::random_access_range<RangeType>
332void CheckRange(const RangeType& range, size_t expected_size)
333{
334 using value_type = std::ranges::range_value_t<RangeType>;
335
336 BOOST_CHECK_EQUAL(range.size(), expected_size);
337 BOOST_REQUIRE(range.size() > 0); // Some checks below assume a non-empty range
338 BOOST_REQUIRE(!range.empty());
339
340 BOOST_CHECK(range.begin() != range.end());
341 BOOST_CHECK_EQUAL(std::distance(range.begin(), range.end()), static_cast<std::ptrdiff_t>(expected_size));
342 BOOST_CHECK(range.cbegin() == range.begin());
343 BOOST_CHECK(range.cend() == range.end());
344
345 for (size_t i = 0; i < range.size(); ++i) {
346 BOOST_CHECK_EQUAL(range[i].get(), (*(range.begin() + i)).get());
347 }
348
349 BOOST_CHECK_THROW(range.at(expected_size), std::out_of_range);
350
351 BOOST_CHECK_EQUAL(range.front().get(), range[0].get());
352 BOOST_CHECK_EQUAL(range.back().get(), range[expected_size - 1].get());
353
354 auto it = range.begin();
355 auto it_copy = it;
356 ++it;
357 BOOST_CHECK(it != it_copy);
358 --it;
359 BOOST_CHECK(it == it_copy);
360 it = range.begin();
361 auto old_it = it++;
362 BOOST_CHECK(old_it == range.begin());
363 BOOST_CHECK(it == range.begin() + 1);
364 old_it = it--;
365 BOOST_CHECK(old_it == range.begin() + 1);
366 BOOST_CHECK(it == range.begin());
367
368 it = range.begin();
369 it += 2;
370 BOOST_CHECK(it == range.begin() + 2);
371 it -= 2;
372 BOOST_CHECK(it == range.begin());
373
374 BOOST_CHECK(range.begin() < range.end());
375 BOOST_CHECK(range.begin() <= range.end());
376 BOOST_CHECK(range.end() > range.begin());
377 BOOST_CHECK(range.end() >= range.begin());
378 BOOST_CHECK(range.begin() == range.begin());
379
380 BOOST_CHECK_EQUAL(range.begin()[0].get(), range[0].get());
381
382 size_t count = 0;
383 for (auto rit = range.end(); rit != range.begin();) {
384 --rit;
385 ++count;
386 }
387 BOOST_CHECK_EQUAL(count, expected_size);
388
389 std::vector<value_type> collected;
390 for (const auto& elem : range) {
391 collected.push_back(elem);
392 }
393 BOOST_CHECK_EQUAL(collected.size(), expected_size);
394
395 BOOST_CHECK_EQUAL(std::ranges::size(range), expected_size);
396
397 it = range.begin();
398 auto it2 = 1 + it;
399 BOOST_CHECK(it2 == it + 1);
400}
401
402BOOST_AUTO_TEST_CASE(btck_transaction_tests)
403{
404 auto tx_data{hex_string_to_byte_vec("02000000013f7cebd65c27431a90bba7f796914fe8cc2ddfc3f2cbd6f7e5f2fc854534da95000000006b483045022100de1ac3bcdfb0332207c4a91f3832bd2c2915840165f876ab47c5f8996b971c3602201c6c053d750fadde599e6f5c4e1963df0f01fc0d97815e8157e3d59fe09ca30d012103699b464d1d8bc9e47d4fb1cdaa89a1c5783d68363c4dbc4b524ed3d857148617feffffff02836d3c01000000001976a914fc25d6d5c94003bf5b0c7b640a248e2c637fcfb088ac7ada8202000000001976a914fbed3d9b11183209a57999d54d59f67c019e756c88ac6acb0700")};
405 auto tx{Transaction{tx_data}};
406 auto tx_data_2{hex_string_to_byte_vec("02000000000101904f4ee5c87d20090b642f116e458cd6693292ad9ece23e72f15fb6c05b956210500000000fdffffff02e2010000000000002251200839a723933b56560487ec4d67dda58f09bae518ffa7e148313c5696ac837d9f10060000000000002251205826bcdae7abfb1c468204170eab00d887b61ab143464a4a09e1450bdc59a3340140f26e7af574e647355830772946356c27e7bbc773c5293688890f58983499581be84de40be7311a14e6d6422605df086620e75adae84ff06b75ce5894de5e994a00000000")};
407 auto tx2{Transaction{tx_data_2}};
408 CheckHandle(tx, tx2);
409
410 auto invalid_data = hex_string_to_byte_vec("012300");
411 BOOST_CHECK_THROW(Transaction{invalid_data}, std::runtime_error);
412 auto empty_data = hex_string_to_byte_vec("");
413 BOOST_CHECK_THROW(Transaction{empty_data}, std::runtime_error);
414
415 BOOST_CHECK_EQUAL(tx.CountOutputs(), 2);
416 BOOST_CHECK_EQUAL(tx.CountInputs(), 1);
417 BOOST_CHECK_EQUAL(tx.GetLocktime(), 510826);
418 auto broken_tx_data{std::span<std::byte>{tx_data.begin(), tx_data.begin() + 10}};
419 BOOST_CHECK_THROW(Transaction{broken_tx_data}, std::runtime_error);
420 auto input{tx.GetInput(0)};
421 BOOST_CHECK_EQUAL(input.GetSequence(), 0xfffffffe);
422 auto output{tx.GetOutput(tx.CountOutputs() - 1)};
423 BOOST_CHECK_EQUAL(output.Amount(), 42130042);
424 auto script_pubkey{output.GetScriptPubkey()};
425 {
426 auto tx_new{Transaction{tx_data}};
427 // This is safe, because we now use copy assignment
428 TransactionOutput output = tx_new.GetOutput(tx_new.CountOutputs() - 1);
430
431 TransactionOutputView output2 = tx_new.GetOutput(tx_new.CountOutputs() - 1);
432 BOOST_CHECK_NE(output.get(), output2.get());
433 BOOST_CHECK_EQUAL(output.Amount(), output2.Amount());
434 TransactionOutput output3 = output2;
435 BOOST_CHECK_NE(output3.get(), output2.get());
436 BOOST_CHECK_EQUAL(output3.Amount(), output2.Amount());
437
438 // Non-owned view
439 ScriptPubkeyView script2 = output.GetScriptPubkey();
440 BOOST_CHECK_NE(script.get(), script2.get());
441 check_equal(script.ToBytes(), script2.ToBytes());
442
443 // Non-owned to owned
444 ScriptPubkey script3 = script2;
445 BOOST_CHECK_NE(script3.get(), script2.get());
446 check_equal(script3.ToBytes(), script2.ToBytes());
447 }
448 BOOST_CHECK_EQUAL(output.Amount(), 42130042);
449
450 auto tx_roundtrip{Transaction{tx.ToBytes()}};
451 check_equal(tx_roundtrip.ToBytes(), tx_data);
452
453 // The following code is unsafe, but left here to show limitations of the
454 // API, because we preserve the output view beyond the lifetime of the
455 // transaction. The view type wrapper should make this clear to the user.
456 // auto get_output = [&]() -> TransactionOutputView {
457 // auto tx{Transaction{tx_data}};
458 // return tx.GetOutput(0);
459 // };
460 // auto output_new = get_output();
461 // BOOST_CHECK_EQUAL(output_new.Amount(), 20737411);
462
463 int64_t total_amount{0};
464 for (const auto output : tx.Outputs()) {
465 total_amount += output.Amount();
466 }
467 BOOST_CHECK_EQUAL(total_amount, 62867453);
468
469 auto amount = *(tx.Outputs() | std::ranges::views::filter([](const auto& output) {
470 return output.Amount() == 42130042;
471 }) |
472 std::views::transform([](const auto& output) {
473 return output.Amount();
474 })).begin();
475 BOOST_REQUIRE(amount);
476 BOOST_CHECK_EQUAL(amount, 42130042);
477
478 CheckRange(tx.Outputs(), tx.CountOutputs());
479
480 ScriptPubkey script_pubkey_roundtrip{script_pubkey.ToBytes()};
481 check_equal(script_pubkey_roundtrip.ToBytes(), script_pubkey.ToBytes());
482}
483
484BOOST_AUTO_TEST_CASE(btck_script_pubkey)
485{
486 auto script_data{hex_string_to_byte_vec("76a9144bfbaf6afb76cc5771bc6404810d1cc041a6933988ac")};
487 std::vector<std::byte> script_data_2 = script_data;
488 script_data_2.push_back(std::byte{0x51});
489 ScriptPubkey script{script_data};
490 ScriptPubkey script2{script_data_2};
491 CheckHandle(script, script2);
492
493 std::span<std::byte> empty_data{};
494 ScriptPubkey empty_script{empty_data};
495 CheckHandle(script, empty_script);
496}
497
498BOOST_AUTO_TEST_CASE(btck_transaction_output)
499{
500 ScriptPubkey script{hex_string_to_byte_vec("76a9144bfbaf6afb76cc5771bc6404810d1cc041a6933988ac")};
501 TransactionOutput output{script, 1};
502 TransactionOutput output2{script, 2};
503 CheckHandle(output, output2);
504}
505
506BOOST_AUTO_TEST_CASE(btck_transaction_input)
507{
508 Transaction tx{hex_string_to_byte_vec("020000000248c03e66fd371c7033196ce24298628e59ebefa00363026044e0f35e0325a65d000000006a473044022004893432347f39beaa280e99da595681ddb20fc45010176897e6e055d716dbfa022040a9e46648a5d10c33ef7cee5e6cf4b56bd513eae3ae044f0039824b02d0f44c012102982331a52822fd9b62e9b5d120da1d248558fac3da3a3c51cd7d9c8ad3da760efeffffffb856678c6e4c3c84e39e2ca818807049d6fba274b42af3c6d3f9d4b6513212d2000000006a473044022068bcedc7fe39c9f21ad318df2c2da62c2dc9522a89c28c8420ff9d03d2e6bf7b0220132afd752754e5cb1ea2fd0ed6a38ec666781e34b0e93dc9a08f2457842cf5660121033aeb9c079ea3e08ea03556182ab520ce5c22e6b0cb95cee6435ee17144d860cdfeffffff0260d50b00000000001976a914363cc8d55ea8d0500de728ef6d63804ddddbdc9888ac67040f00000000001976a914c303bdc5064bf9c9a8b507b5496bd0987285707988ac6acb0700")};
509 TransactionInput input_0 = tx.GetInput(0);
510 TransactionInput input_1 = tx.GetInput(1);
511 CheckHandle(input_0, input_1);
512 CheckRange(tx.Inputs(), tx.CountInputs());
513 OutPoint point_0 = input_0.OutPoint();
514 OutPoint point_1 = input_1.OutPoint();
515 CheckHandle(point_0, point_1);
516}
517
518BOOST_AUTO_TEST_CASE(btck_precomputed_txdata) {
519 auto tx_data{hex_string_to_byte_vec("02000000013f7cebd65c27431a90bba7f796914fe8cc2ddfc3f2cbd6f7e5f2fc854534da95000000006b483045022100de1ac3bcdfb0332207c4a91f3832bd2c2915840165f876ab47c5f8996b971c3602201c6c053d750fadde599e6f5c4e1963df0f01fc0d97815e8157e3d59fe09ca30d012103699b464d1d8bc9e47d4fb1cdaa89a1c5783d68363c4dbc4b524ed3d857148617feffffff02836d3c01000000001976a914fc25d6d5c94003bf5b0c7b640a248e2c637fcfb088ac7ada8202000000001976a914fbed3d9b11183209a57999d54d59f67c019e756c88ac6acb0700")};
520 auto tx{Transaction{tx_data}};
521 auto tx_data_2{hex_string_to_byte_vec("02000000000101904f4ee5c87d20090b642f116e458cd6693292ad9ece23e72f15fb6c05b956210500000000fdffffff02e2010000000000002251200839a723933b56560487ec4d67dda58f09bae518ffa7e148313c5696ac837d9f10060000000000002251205826bcdae7abfb1c468204170eab00d887b61ab143464a4a09e1450bdc59a3340140f26e7af574e647355830772946356c27e7bbc773c5293688890f58983499581be84de40be7311a14e6d6422605df086620e75adae84ff06b75ce5894de5e994a00000000")};
522 auto tx2{Transaction{tx_data_2}};
523 auto precomputed_txdata{PrecomputedTransactionData{
524 /*tx_to=*/tx,
525 /*spent_outputs=*/{},
526 }};
527 auto precomputed_txdata_2{PrecomputedTransactionData{
528 /*tx_to=*/tx2,
529 /*spent_outputs=*/{},
530 }};
531 CheckHandle(precomputed_txdata, precomputed_txdata_2);
532}
533
534BOOST_AUTO_TEST_CASE(btck_script_verify_tests)
535{
536 // Legacy transaction aca326a724eda9a461c10a876534ecd5ae7b27f10f26c3862fb996f80ea2d45d
537 auto legacy_spent_script_pubkey{ScriptPubkey{hex_string_to_byte_vec("76a9144bfbaf6afb76cc5771bc6404810d1cc041a6933988ac")}};
538 auto legacy_spending_tx{Transaction{hex_string_to_byte_vec("02000000013f7cebd65c27431a90bba7f796914fe8cc2ddfc3f2cbd6f7e5f2fc854534da95000000006b483045022100de1ac3bcdfb0332207c4a91f3832bd2c2915840165f876ab47c5f8996b971c3602201c6c053d750fadde599e6f5c4e1963df0f01fc0d97815e8157e3d59fe09ca30d012103699b464d1d8bc9e47d4fb1cdaa89a1c5783d68363c4dbc4b524ed3d857148617feffffff02836d3c01000000001976a914fc25d6d5c94003bf5b0c7b640a248e2c637fcfb088ac7ada8202000000001976a914fbed3d9b11183209a57999d54d59f67c019e756c88ac6acb0700")}};
540 /*spent_script_pubkey=*/legacy_spent_script_pubkey,
541 /*spending_tx=*/legacy_spending_tx,
542 /*precomputed_txdata=*/nullptr,
543 /*amount=*/0,
544 /*input_index=*/0,
545 /*taproot=*/false);
546
547 // Legacy transaction aca326a724eda9a461c10a876534ecd5ae7b27f10f26c3862fb996f80ea2d45d with precomputed_txdata
548 auto legacy_precomputed_txdata{PrecomputedTransactionData{
549 /*tx_to=*/legacy_spending_tx,
550 /*spent_outputs=*/{},
551 }};
553 /*spent_script_pubkey=*/legacy_spent_script_pubkey,
554 /*spending_tx=*/legacy_spending_tx,
555 /*precomputed_txdata=*/&legacy_precomputed_txdata,
556 /*amount=*/0,
557 /*input_index=*/0,
558 /*taproot=*/false);
559
560 // Segwit transaction 1a3e89644985fbbb41e0dcfe176739813542b5937003c46a07de1e3ee7a4a7f3
561 auto segwit_spent_script_pubkey{ScriptPubkey{hex_string_to_byte_vec("0020701a8d401c84fb13e6baf169d59684e17abd9fa216c8cc5b9fc63d622ff8c58d")}};
562 auto segwit_spending_tx{Transaction{hex_string_to_byte_vec("010000000001011f97548fbbe7a0db7588a66e18d803d0089315aa7d4cc28360b6ec50ef36718a0100000000ffffffff02df1776000000000017a9146c002a686959067f4866b8fb493ad7970290ab728757d29f0000000000220020701a8d401c84fb13e6baf169d59684e17abd9fa216c8cc5b9fc63d622ff8c58d04004730440220565d170eed95ff95027a69b313758450ba84a01224e1f7f130dda46e94d13f8602207bdd20e307f062594022f12ed5017bbf4a055a06aea91c10110a0e3bb23117fc014730440220647d2dc5b15f60bc37dc42618a370b2a1490293f9e5c8464f53ec4fe1dfe067302203598773895b4b16d37485cbe21b337f4e4b650739880098c592553add7dd4355016952210375e00eb72e29da82b89367947f29ef34afb75e8654f6ea368e0acdfd92976b7c2103a1b26313f430c4b15bb1fdce663207659d8cac749a0e53d70eff01874496feff2103c96d495bfdd5ba4145e3e046fee45e84a8a48ad05bd8dbb395c011a32cf9f88053ae00000000")}};
564 /*spent_script_pubkey=*/segwit_spent_script_pubkey,
565 /*spending_tx=*/segwit_spending_tx,
566 /*precomputed_txdata=*/nullptr,
567 /*amount=*/18393430,
568 /*input_index=*/0,
569 /*taproot=*/false);
570
571 // Segwit transaction 1a3e89644985fbbb41e0dcfe176739813542b5937003c46a07de1e3ee7a4a7f3 with precomputed_txdata
572 auto segwit_precomputed_txdata{PrecomputedTransactionData{
573 /*tx_to=*/segwit_spending_tx,
574 /*spent_outputs=*/{},
575 }};
577 /*spent_script_pubkey=*/segwit_spent_script_pubkey,
578 /*spending_tx=*/segwit_spending_tx,
579 /*precomputed_txdata=*/&segwit_precomputed_txdata,
580 /*amount=*/18393430,
581 /*input_index=*/0,
582 /*taproot=*/false);
583
584 // Taproot transaction 33e794d097969002ee05d336686fc03c9e15a597c1b9827669460fac98799036
585 auto taproot_spent_script_pubkey{ScriptPubkey{hex_string_to_byte_vec("5120339ce7e165e67d93adb3fef88a6d4beed33f01fa876f05a225242b82a631abc0")}};
586 auto taproot_spending_tx{Transaction{hex_string_to_byte_vec("01000000000101d1f1c1f8cdf6759167b90f52c9ad358a369f95284e841d7a2536cef31c0549580100000000fdffffff020000000000000000316a2f49206c696b65205363686e6f7272207369677320616e6420492063616e6e6f74206c69652e204062697462756734329e06010000000000225120a37c3903c8d0db6512e2b40b0dffa05e5a3ab73603ce8c9c4b7771e5412328f90140a60c383f71bac0ec919b1d7dbc3eb72dd56e7aa99583615564f9f99b8ae4e837b758773a5b2e4c51348854c8389f008e05029db7f464a5ff2e01d5e6e626174affd30a00")}};
587 std::vector<TransactionOutput> taproot_spent_outputs;
588 taproot_spent_outputs.emplace_back(taproot_spent_script_pubkey, 88480);
589 auto taproot_precomputed_txdata{PrecomputedTransactionData{
590 /*tx_to=*/taproot_spending_tx,
591 /*spent_outputs=*/taproot_spent_outputs,
592 }};
594 /*spent_script_pubkey=*/taproot_spent_script_pubkey,
595 /*spending_tx=*/taproot_spending_tx,
596 /*precomputed_txdata=*/&taproot_precomputed_txdata,
597 /*amount=*/88480,
598 /*input_index=*/0,
599 /*taproot=*/true);
600
601 // Two-input taproot transaction e8e8320f40c31ed511570e9cdf1d241f8ec9a5cc392e6105240ac8dbea2098de
602 auto taproot2_spent_script_pubkey0{ScriptPubkey{hex_string_to_byte_vec("5120b7da80f57e36930b0515eb09293e25858d13e6b91fee6184943f5a584cb4248e")}};
603 auto taproot2_spent_script_pubkey1{ScriptPubkey{hex_string_to_byte_vec("5120ab78e077d062e7b8acd7063668b4db5355a1b5d5fd2a46a8e98e62e5e63fab77")}};
604 auto taproot2_spending_tx{Transaction{hex_string_to_byte_vec("02000000000102c0f01ead18750892c84b1d4f595149ad38f16847df1fbf490e235b3b78c1f98a0100000000ffffffff456764a19c2682bf5b1567119f06a421849ad1664cf42b5ef95b69d6e2159e9d0000000000ffffffff022202000000000000225120b6c0c2a8ee25a2ae0322ab7f1a06f01746f81f6b90d179c3c2a51a356e6188f1d70e020000000000225120b7da80f57e36930b0515eb09293e25858d13e6b91fee6184943f5a584cb4248e0141933fdc49eb1af1f08ed1e9cf5559259309a8acd25ff1e6999b6955124438aef4fceaa4e6a5f85286631e24837329563595bc3cf4b31e1c687442abb01c4206818101401c9620faf1e8c84187762ad14d04ae3857f59a2f03f1dcbb99290e16dfc572a63b4ea435780a5787af59beb5742fd71cda8a95381517a1ff14b4c67996c4bf8100000000")}};
605 std::vector<TransactionOutput> taproot2_spent_outputs;
606 taproot2_spent_outputs.emplace_back(taproot2_spent_script_pubkey0, 546);
607 taproot2_spent_outputs.emplace_back(taproot2_spent_script_pubkey1, 135125);
608 auto taproot2_precomputed_txdata{PrecomputedTransactionData{
609 /*tx_to=*/taproot2_spending_tx,
610 /*spent_outputs=*/taproot2_spent_outputs,
611 }};
613 /*spent_script_pubkey=*/taproot2_spent_script_pubkey0,
614 /*spending_tx=*/taproot2_spending_tx,
615 /*precomputed_txdata=*/&taproot2_precomputed_txdata,
616 /*amount=*/546,
617 /*input_index=*/0,
618 /*taproot=*/true);
620 /*spent_script_pubkey=*/taproot2_spent_script_pubkey1,
621 /*spending_tx=*/taproot2_spending_tx,
622 /*precomputed_txdata=*/&taproot2_precomputed_txdata,
623 /*amount=*/135125,
624 /*input_index=*/1,
625 /*taproot=*/true);
626}
627
629{
630 btck_LoggingOptions logging_options = {
631 .log_timestamps = true,
632 .log_time_micros = true,
633 .log_threadnames = false,
634 .log_sourcelocations = false,
635 .always_print_category_levels = true,
636 };
637
638 logging_set_options(logging_options);
639 logging_set_level_category(LogCategory::BENCH, LogLevel::TRACE_LEVEL);
643
644 // Check that connecting, connecting another, and then disconnecting and connecting a logger again works.
645 {
646 logging_set_level_category(LogCategory::KERNEL, LogLevel::TRACE_LEVEL);
648 Logger logger{std::make_unique<TestLog>()};
649 Logger logger_2{std::make_unique<TestLog>()};
650 }
651 Logger logger{std::make_unique<TestLog>()};
652}
653
654BOOST_AUTO_TEST_CASE(btck_context_tests)
655{
656 { // test default context
657 Context context{};
658 Context context2{};
659 CheckHandle(context, context2);
660 }
661
662 { // test with context options, but not options set
663 ContextOptions options{};
664 Context context{options};
665 }
666
667 { // test with context options
668 ContextOptions options{};
669 ChainParams params{ChainType::MAINNET};
670 ChainParams regtest_params{ChainType::REGTEST};
671 CheckHandle(params, regtest_params);
672 options.SetChainParams(params);
673 options.SetNotifications(std::make_shared<TestKernelNotifications>());
674 Context context{options};
675 }
676}
677
678BOOST_AUTO_TEST_CASE(btck_block_header_tests)
679{
680 // Block header format: version(4) + prev_hash(32) + merkle_root(32) + timestamp(4) + bits(4) + nonce(4) = 80 bytes
681 BlockHeader header_0{hex_string_to_byte_vec("00e07a26beaaeee2e71d7eb19279545edbaf15de0999983626ec00000000000000000000579cf78b65229bfb93f4a11463af2eaa5ad91780f27f5d147a423bea5f7e4cdf2a47e268b4dd01173a9662ee")};
682 BOOST_CHECK_EQUAL(byte_span_to_hex_string_reversed(header_0.Hash().ToBytes()), "00000000000000000000325c7e14a4ee3b4fcb2343089a839287308a0ddbee4f");
683 BlockHeader header_1{hex_string_to_byte_vec("00c00020e7cb7b4de21d26d55bd384017b8bb9333ac3b2b55bed00000000000000000000d91b4484f801b99f03d36b9d26cfa83420b67f81da12d7e6c1e7f364e743c5ba9946e268b4dd011799c8533d")};
684 CheckHandle(header_0, header_1);
685
686 // Test error handling for invalid data
687 BOOST_CHECK_THROW(BlockHeader{hex_string_to_byte_vec("00")}, std::runtime_error);
688 BOOST_CHECK_THROW(BlockHeader{hex_string_to_byte_vec("")}, std::runtime_error);
689
690 // Test all header field accessors using mainnet block 1
691 auto mainnet_block_1_header = hex_string_to_byte_vec("010000006fe28c0ab6f1b372c1a6a246ae63f74f931e8365e15a089c68d6190000000000982051fd1e4ba744bbbe680e1fee14677ba1a3c3540bf7b1cdb606e857233e0e61bc6649ffff001d01e36299");
692 BlockHeader header{mainnet_block_1_header};
693 BOOST_CHECK_EQUAL(header.Version(), 1);
694 BOOST_CHECK_EQUAL(header.Timestamp(), 1231469665);
695 BOOST_CHECK_EQUAL(header.Bits(), 0x1d00ffff);
696 BOOST_CHECK_EQUAL(header.Nonce(), 2573394689);
697 BOOST_CHECK_EQUAL(byte_span_to_hex_string_reversed(header.Hash().ToBytes()), "00000000839a8e6886ab5951d76f411475428afc90947ee320161bbf18eb6048");
698 auto prev_hash = header.PrevHash();
699 BOOST_CHECK_EQUAL(byte_span_to_hex_string_reversed(prev_hash.ToBytes()), "000000000019d6689c085ae165831e934ff763ae46a2a6c172b3f1b60a8ce26f");
700
701 // Test round-trip serialization of block header
702 auto header_roundtrip{BlockHeader{header.ToBytes()}};
703 check_equal(header_roundtrip.ToBytes(), mainnet_block_1_header);
704
705 auto raw_block = hex_string_to_byte_vec("010000006fe28c0ab6f1b372c1a6a246ae63f74f931e8365e15a089c68d6190000000000982051fd1e4ba744bbbe680e1fee14677ba1a3c3540bf7b1cdb606e857233e0e61bc6649ffff001d01e362990101000000010000000000000000000000000000000000000000000000000000000000000000ffffffff0704ffff001d0104ffffffff0100f2052a0100000043410496b538e853519c726a2c91e61ec11600ae1390813a627c66fb8be7947be63c52da7589379515d4e0a604f8141781e62294721166bf621e73a82cbf2342c858eeac00000000");
706 Block block{raw_block};
707 BlockHeader block_header{block.GetHeader()};
708 BOOST_CHECK_EQUAL(block_header.Version(), 1);
709 BOOST_CHECK_EQUAL(block_header.Timestamp(), 1231469665);
710 BOOST_CHECK_EQUAL(block_header.Bits(), 0x1d00ffff);
711 BOOST_CHECK_EQUAL(block_header.Nonce(), 2573394689);
712 BOOST_CHECK_EQUAL(byte_span_to_hex_string_reversed(block_header.Hash().ToBytes()), "00000000839a8e6886ab5951d76f411475428afc90947ee320161bbf18eb6048");
713
714 // Verify header from block serializes to first 80 bytes of raw block
715 auto block_header_bytes = block_header.ToBytes();
716 BOOST_CHECK_EQUAL(block_header_bytes.size(), 80);
717 check_equal(block_header_bytes, std::span<const std::byte>(raw_block.data(), 80));
718}
719
721{
724 CheckHandle(block, block_100);
726 CheckRange(block_tx.Transactions(), block_tx.CountTransactions());
727 auto invalid_data = hex_string_to_byte_vec("012300");
728 BOOST_CHECK_THROW(Block{invalid_data}, std::runtime_error);
729 auto empty_data = hex_string_to_byte_vec("");
730 BOOST_CHECK_THROW(Block{empty_data}, std::runtime_error);
731}
732
733Context create_context(std::shared_ptr<TestKernelNotifications> notifications, ChainType chain_type, std::shared_ptr<TestValidationInterface> validation_interface = nullptr)
734{
735 ContextOptions options{};
736 ChainParams params{chain_type};
737 options.SetChainParams(params);
738 options.SetNotifications(notifications);
739 if (validation_interface) {
740 options.SetValidationInterface(validation_interface);
741 }
742 auto context{Context{options}};
743 return context;
744}
745
746BOOST_AUTO_TEST_CASE(btck_chainman_tests)
747{
748 Logger logger{std::make_unique<TestLog>()};
749 auto test_directory{TestDirectory{"chainman_test_bitcoin_kernel"}};
750
751 { // test with default context
752 Context context{};
753 ChainstateManagerOptions chainman_opts{context, PathToString(test_directory.m_directory), PathToString(test_directory.m_directory / "blocks")};
754 ChainMan chainman{context, chainman_opts};
755 }
756
757 { // test with default context options
758 ContextOptions options{};
759 Context context{options};
760 ChainstateManagerOptions chainman_opts{context, PathToString(test_directory.m_directory), PathToString(test_directory.m_directory / "blocks")};
761 ChainMan chainman{context, chainman_opts};
762 }
763 { // null or empty data_directory or blocks_directory are not allowed
764 Context context{};
765 auto valid_dir{PathToString(test_directory.m_directory)};
766 std::vector<std::pair<std::string_view, std::string_view>> illegal_cases{
767 {"", valid_dir},
768 {valid_dir, {nullptr, 0}},
769 {"", ""},
770 {{nullptr, 0}, {nullptr, 0}},
771 };
772 for (auto& [data_dir, blocks_dir] : illegal_cases) {
773 BOOST_CHECK_THROW(ChainstateManagerOptions(context, data_dir, blocks_dir),
774 std::runtime_error);
775 };
776 }
777
778 auto notifications{std::make_shared<TestKernelNotifications>()};
779 auto context{create_context(notifications, ChainType::MAINNET)};
780
781 ChainstateManagerOptions chainman_opts{context, PathToString(test_directory.m_directory), PathToString(test_directory.m_directory / "blocks")};
782 chainman_opts.SetWorkerThreads(4);
783 BOOST_CHECK(!chainman_opts.SetWipeDbs(/*wipe_block_tree=*/true, /*wipe_chainstate=*/false));
784 BOOST_CHECK(chainman_opts.SetWipeDbs(/*wipe_block_tree=*/true, /*wipe_chainstate=*/true));
785 BOOST_CHECK(chainman_opts.SetWipeDbs(/*wipe_block_tree=*/false, /*wipe_chainstate=*/true));
786 BOOST_CHECK(chainman_opts.SetWipeDbs(/*wipe_block_tree=*/false, /*wipe_chainstate=*/false));
787 ChainMan chainman{context, chainman_opts};
788}
789
790std::unique_ptr<ChainMan> create_chainman(TestDirectory& test_directory,
791 bool reindex,
792 bool wipe_chainstate,
793 bool block_tree_db_in_memory,
794 bool chainstate_db_in_memory,
795 Context& context)
796{
797 ChainstateManagerOptions chainman_opts{context, PathToString(test_directory.m_directory), PathToString(test_directory.m_directory / "blocks")};
798
799 if (reindex) {
800 chainman_opts.SetWipeDbs(/*wipe_block_tree=*/reindex, /*wipe_chainstate=*/reindex);
801 }
802 if (wipe_chainstate) {
803 chainman_opts.SetWipeDbs(/*wipe_block_tree=*/false, /*wipe_chainstate=*/wipe_chainstate);
804 }
805 if (block_tree_db_in_memory) {
806 chainman_opts.UpdateBlockTreeDbInMemory(block_tree_db_in_memory);
807 }
808 if (chainstate_db_in_memory) {
809 chainman_opts.UpdateChainstateDbInMemory(chainstate_db_in_memory);
810 }
811
812 auto chainman{std::make_unique<ChainMan>(context, chainman_opts)};
813 return chainman;
814}
815
817{
818 auto notifications{std::make_shared<TestKernelNotifications>()};
819 auto context{create_context(notifications, ChainType::MAINNET)};
820 auto chainman{create_chainman(
821 test_directory, /*reindex=*/true, /*wipe_chainstate=*/false,
822 /*block_tree_db_in_memory=*/false, /*chainstate_db_in_memory=*/false, context)};
823
824 std::vector<std::string> import_files;
825 BOOST_CHECK(chainman->ImportBlocks(import_files));
826
827 // Sanity check some block retrievals
828 auto chain{chainman->GetChain()};
829 BOOST_CHECK_THROW(chain.GetByHeight(1000), std::runtime_error);
830 auto genesis_index{chain.Entries().front()};
831 BOOST_CHECK(!genesis_index.GetPrevious());
832 auto genesis_block_raw{chainman->ReadBlock(genesis_index).value().ToBytes()};
833 auto first_index{chain.GetByHeight(0)};
834 auto first_block_raw{chainman->ReadBlock(genesis_index).value().ToBytes()};
835 check_equal(genesis_block_raw, first_block_raw);
836 auto height{first_index.GetHeight()};
837 BOOST_CHECK_EQUAL(height, 0);
838
839 auto next_index{chain.GetByHeight(first_index.GetHeight() + 1)};
840 BOOST_CHECK(chain.Contains(next_index));
841 auto next_block_data{chainman->ReadBlock(next_index).value().ToBytes()};
842 auto tip_index{chain.Entries().back()};
843 auto tip_block_data{chainman->ReadBlock(tip_index).value().ToBytes()};
844 auto second_index{chain.GetByHeight(1)};
845 auto second_block{chainman->ReadBlock(second_index).value()};
846 auto second_block_data{second_block.ToBytes()};
847 auto second_height{second_index.GetHeight()};
848 BOOST_CHECK_EQUAL(second_height, 1);
849 check_equal(next_block_data, tip_block_data);
850 check_equal(next_block_data, second_block_data);
851
852 auto second_hash{second_index.GetHash()};
853 auto another_second_index{chainman->GetBlockTreeEntry(second_hash)};
854 BOOST_CHECK(another_second_index);
855 auto another_second_height{another_second_index->GetHeight()};
856 auto second_block_hash{second_block.GetHash()};
857 check_equal(second_block_hash.ToBytes(), second_hash.ToBytes());
858 BOOST_CHECK_EQUAL(second_height, another_second_height);
859}
860
862{
863 auto notifications{std::make_shared<TestKernelNotifications>()};
864 auto context{create_context(notifications, ChainType::MAINNET)};
865 auto chainman{create_chainman(
866 test_directory, /*reindex=*/false, /*wipe_chainstate=*/true,
867 /*block_tree_db_in_memory=*/false, /*chainstate_db_in_memory=*/false, context)};
868
869 std::vector<std::string> import_files;
870 import_files.push_back(PathToString(test_directory.m_directory / "blocks" / "blk00000.dat"));
871 BOOST_CHECK(chainman->ImportBlocks(import_files));
872}
873
875{
876 auto notifications{std::make_shared<TestKernelNotifications>()};
877 auto validation_interface{std::make_shared<TestValidationInterface>()};
878 auto context{create_context(notifications, ChainType::MAINNET, validation_interface)};
879 auto chainman{create_chainman(
880 test_directory, /*reindex=*/false, /*wipe_chainstate=*/false,
881 /*block_tree_db_in_memory=*/false, /*chainstate_db_in_memory=*/false, context)};
882
883 // mainnet block 1
884 auto raw_block = hex_string_to_byte_vec("010000006fe28c0ab6f1b372c1a6a246ae63f74f931e8365e15a089c68d6190000000000982051fd1e4ba744bbbe680e1fee14677ba1a3c3540bf7b1cdb606e857233e0e61bc6649ffff001d01e362990101000000010000000000000000000000000000000000000000000000000000000000000000ffffffff0704ffff001d0104ffffffff0100f2052a0100000043410496b538e853519c726a2c91e61ec11600ae1390813a627c66fb8be7947be63c52da7589379515d4e0a604f8141781e62294721166bf621e73a82cbf2342c858eeac00000000");
885 Block block{raw_block};
886 BlockHeader header{block.GetHeader()};
887 TransactionView tx{block.GetTransaction(block.CountTransactions() - 1)};
888 BOOST_CHECK_EQUAL(byte_span_to_hex_string_reversed(tx.Txid().ToBytes()), "0e3e2357e806b6cdb1f70b54c3a3a17b6714ee1f0e68bebb44a74b1efd512098");
889 BOOST_CHECK_EQUAL(header.Version(), 1);
890 BOOST_CHECK_EQUAL(header.Timestamp(), 1231469665);
891 BOOST_CHECK_EQUAL(header.Bits(), 0x1d00ffff);
892 BOOST_CHECK_EQUAL(header.Nonce(), 2573394689);
893 BOOST_CHECK_EQUAL(tx.CountInputs(), 1);
894 Transaction tx2 = tx;
896 for (auto transaction : block.Transactions()) {
897 BOOST_CHECK_EQUAL(transaction.CountInputs(), 1);
898 }
899 auto output_counts = *(block.Transactions() | std::views::transform([](const auto& tx) {
900 return tx.CountOutputs();
901 })).begin();
902 BOOST_CHECK_EQUAL(output_counts, 1);
903
904 validation_interface->m_expected_valid_block.emplace(raw_block);
905 auto ser_block{block.ToBytes()};
906 check_equal(ser_block, raw_block);
907 bool new_block = false;
908 BOOST_CHECK(chainman->ProcessBlock(block, &new_block));
909 BOOST_CHECK(new_block);
910
911 validation_interface->m_expected_valid_block = std::nullopt;
912 new_block = false;
914 BOOST_CHECK(!chainman->ProcessBlock(invalid_block, &new_block));
915 BOOST_CHECK(!new_block);
916
917 auto chain{chainman->GetChain()};
918 BOOST_CHECK_EQUAL(chain.Height(), 1);
919 auto tip{chain.Entries().back()};
920 auto read_block{chainman->ReadBlock(tip)};
921 BOOST_REQUIRE(read_block);
922 check_equal(read_block.value().ToBytes(), raw_block);
923
924 // Check that we can read the previous block
925 BlockTreeEntry tip_2{*tip.GetPrevious()};
926 Block read_block_2{*chainman->ReadBlock(tip_2)};
927 BOOST_CHECK_EQUAL(chainman->ReadBlockSpentOutputs(tip_2).Count(), 0);
928 BOOST_CHECK_EQUAL(chainman->ReadBlockSpentOutputs(tip).Count(), 0);
929
930 // It should be an error if we go another block back, since the genesis has no ancestor
931 BOOST_CHECK(!tip_2.GetPrevious());
932
933 // If we try to validate it again, it should be a duplicate
934 BOOST_CHECK(chainman->ProcessBlock(block, &new_block));
935 BOOST_CHECK(!new_block);
936}
937
938BOOST_AUTO_TEST_CASE(btck_check_block_context_free)
939{
940 constexpr size_t MERKLE_ROOT_OFFSET{4 + 32};
941 constexpr size_t NBITS_OFFSET{4 + 32 + 32 + 4};
942 constexpr size_t COINBASE_PREVOUT_N_OFFSET{4 + 32 + 32 + 4 + 4 + 4 + 1 + 4 + 1 + 32};
943
944 // Mainnet block 1
945 auto raw_block = hex_string_to_byte_vec("010000006fe28c0ab6f1b372c1a6a246ae63f74f931e8365e15a089c68d6190000000000982051fd1e4ba744bbbe680e1fee14677ba1a3c3540bf7b1cdb606e857233e0e61bc6649ffff001d01e362990101000000010000000000000000000000000000000000000000000000000000000000000000ffffffff0704ffff001d0104ffffffff0100f2052a0100000043410496b538e853519c726a2c91e61ec11600ae1390813a627c66fb8be7947be63c52da7589379515d4e0a604f8141781e62294721166bf621e73a82cbf2342c858eeac00000000");
946
947 // Context-free block checks still need consensus params for the optional
948 // proof-of-work validation path.
949 ChainParams mainnet_params{ChainType::MAINNET};
950 auto consensus_params = mainnet_params.GetConsensusParams();
951
952 Block block{raw_block};
954
955 BOOST_CHECK(block.Check(consensus_params, BlockCheckFlags::BASE, state));
956 BOOST_CHECK(state.GetValidationMode() == ValidationMode::VALID);
957
958 BOOST_CHECK(block.Check(consensus_params, BlockCheckFlags::ALL, state));
959 BOOST_CHECK(state.GetValidationMode() == ValidationMode::VALID);
960
961 auto bad_merkle_block_data = raw_block;
962 bad_merkle_block_data[MERKLE_ROOT_OFFSET] ^= std::byte{0x01};
963 Block bad_merkle_block{bad_merkle_block_data};
964
965 BOOST_CHECK(!bad_merkle_block.Check(consensus_params, BlockCheckFlags::MERKLE, state));
967 BOOST_CHECK(state.GetBlockValidationResult() == BlockValidationResult::MUTATED);
968
969 BOOST_CHECK(bad_merkle_block.Check(consensus_params, BlockCheckFlags::BASE, state));
970 BOOST_CHECK(state.GetValidationMode() == ValidationMode::VALID);
971
972 auto bad_pow_block_data = raw_block;
973 bad_pow_block_data[NBITS_OFFSET + 3] = std::byte{0x1c};
974 Block bad_pow_block{bad_pow_block_data};
975
976 BOOST_CHECK(!bad_pow_block.Check(consensus_params, BlockCheckFlags::POW, state));
978 BOOST_CHECK(state.GetBlockValidationResult() == BlockValidationResult::INVALID_HEADER);
979
980 BOOST_CHECK(bad_pow_block.Check(consensus_params, BlockCheckFlags::MERKLE, state));
981 BOOST_CHECK(state.GetValidationMode() == ValidationMode::VALID);
982
983 auto bad_base_block_data = raw_block;
984 bad_base_block_data[COINBASE_PREVOUT_N_OFFSET] = std::byte{0x00};
985 Block bad_base_block{bad_base_block_data};
986
987 BOOST_CHECK(!bad_base_block.Check(consensus_params, BlockCheckFlags::BASE, state));
989 BOOST_CHECK(state.GetBlockValidationResult() == BlockValidationResult::CONSENSUS);
990
991 // Test with invalid truncated block data.
992 auto truncated_block_data = hex_string_to_byte_vec("010000006fe28c0ab6f1b372c1a6a246ae63f74f931e8365e15a089c68d6190000000000982051fd1e4ba744bbbe680e1fee14677ba1a3c3540bf7b1cdb606e857233e0e61bc6649ffff001d01e36299");
993 BOOST_CHECK_EXCEPTION(Block{truncated_block_data}, std::runtime_error,
994 HasReason{"failed to instantiate btck object"});
995}
996
997BOOST_AUTO_TEST_CASE(btck_chainman_mainnet_tests)
998{
999 auto test_directory{TestDirectory{"mainnet_test_bitcoin_kernel"}};
1000 chainman_mainnet_validation_test(test_directory);
1001 chainman_reindex_test(test_directory);
1002 chainman_reindex_chainstate_test(test_directory);
1003}
1004
1005BOOST_AUTO_TEST_CASE(btck_block_hash_tests)
1006{
1007 std::array<std::byte, 32> test_hash;
1008 std::array<std::byte, 32> test_hash_2;
1009 for (int i = 0; i < 32; ++i) {
1010 test_hash[i] = static_cast<std::byte>(i);
1011 test_hash_2[i] = static_cast<std::byte>(i + 1);
1012 }
1013 BlockHash block_hash{test_hash};
1014 BlockHash block_hash_2{test_hash_2};
1015 BOOST_CHECK(block_hash != block_hash_2);
1016 BOOST_CHECK(block_hash == block_hash);
1017 CheckHandle(block_hash, block_hash_2);
1018}
1019
1020BOOST_AUTO_TEST_CASE(btck_block_tree_entry_tests)
1021{
1022 auto test_directory{TestDirectory{"block_tree_entry_test_bitcoin_kernel"}};
1023 auto notifications{std::make_shared<TestKernelNotifications>()};
1024 auto context{create_context(notifications, ChainType::REGTEST)};
1025 auto chainman{create_chainman(
1026 test_directory,
1027 /*reindex=*/false,
1028 /*wipe_chainstate=*/false,
1029 /*block_tree_db_in_memory=*/true,
1030 /*chainstate_db_in_memory=*/true,
1031 context)};
1032
1033 // Process a couple of blocks
1034 for (size_t i{0}; i < 3; i++) {
1036 bool new_block{false};
1037 chainman->ProcessBlock(block, &new_block);
1038 BOOST_CHECK(new_block);
1039 }
1040
1041 auto chain{chainman->GetChain()};
1042 auto entry_0{chain.GetByHeight(0)};
1043 auto entry_1{chain.GetByHeight(1)};
1044 auto entry_2{chain.GetByHeight(2)};
1045
1046 // Test inequality
1047 BOOST_CHECK(entry_0 != entry_1);
1048 BOOST_CHECK(entry_1 != entry_2);
1049 BOOST_CHECK(entry_0 != entry_2);
1050
1051 // Test equality with same entry
1052 BOOST_CHECK(entry_0 == chain.GetByHeight(0));
1053 BOOST_CHECK(entry_0 == BlockTreeEntry{entry_0});
1054 BOOST_CHECK(entry_1 == entry_1);
1055
1056 // Test GetPrevious
1057 auto prev{entry_1.GetPrevious()};
1058 BOOST_CHECK(prev.has_value());
1059 BOOST_CHECK(prev.value() == entry_0);
1060
1061 // Test GetAncestor
1062 BOOST_CHECK(entry_2.GetAncestor(2) == entry_2);
1063 BOOST_CHECK(entry_2.GetAncestor(1) == entry_1);
1064 BOOST_CHECK(entry_2.GetAncestor(0) == entry_0);
1065}
1066
1067BOOST_AUTO_TEST_CASE(btck_chainman_in_memory_tests)
1068{
1069 auto in_memory_test_directory{TestDirectory{"in-memory_test_bitcoin_kernel"}};
1070
1071 auto notifications{std::make_shared<TestKernelNotifications>()};
1072 auto context{create_context(notifications, ChainType::REGTEST)};
1073 auto chainman{create_chainman(
1074 in_memory_test_directory, /*reindex=*/false, /*wipe_chainstate=*/false,
1075 /*block_tree_db_in_memory=*/true, /*chainstate_db_in_memory=*/true, context)};
1076
1077 for (auto& raw_block : REGTEST_BLOCK_DATA) {
1078 Block block{hex_string_to_byte_vec(raw_block)};
1079 bool new_block{false};
1080 chainman->ProcessBlock(block, &new_block);
1081 BOOST_CHECK(new_block);
1082 }
1083
1084 BOOST_CHECK(fs::exists(in_memory_test_directory.m_directory / "blocks"));
1085 BOOST_CHECK(!fs::exists(in_memory_test_directory.m_directory / "blocks" / "index"));
1086 BOOST_CHECK(!fs::exists(in_memory_test_directory.m_directory / "chainstate"));
1087
1088 BOOST_CHECK(context.interrupt());
1089}
1090
1091BOOST_AUTO_TEST_CASE(btck_chainman_regtest_tests)
1092{
1093 auto test_directory{TestDirectory{"regtest_test_bitcoin_kernel"}};
1094
1095 auto notifications{std::make_shared<TestKernelNotifications>()};
1096 auto context{create_context(notifications, ChainType::REGTEST)};
1097
1098 {
1099 auto chainman{create_chainman(
1100 test_directory, /*reindex=*/false, /*wipe_chainstate=*/false,
1101 /*block_tree_db_in_memory=*/false, /*chainstate_db_in_memory=*/false, context)};
1102 for (const auto& data : REGTEST_BLOCK_DATA) {
1104 BlockHeader header = block.GetHeader();
1105 BlockValidationState state{};
1106 BOOST_CHECK(state.GetBlockValidationResult() == BlockValidationResult::UNSET);
1107 BOOST_CHECK(chainman->ProcessBlockHeader(header, state));
1108 BOOST_CHECK(state.GetValidationMode() == ValidationMode::VALID);
1109 BlockTreeEntry entry{*chainman->GetBlockTreeEntry(header.Hash())};
1110 BOOST_CHECK(!chainman->GetChain().Contains(entry));
1111 BlockTreeEntry best_entry{chainman->GetBestEntry()};
1112 BlockHash hash{entry.GetHash()};
1113 BOOST_CHECK(hash == best_entry.GetHeader().Hash());
1114 }
1115 }
1116
1117 // Validate 206 regtest blocks in total.
1118 // Stop halfway to check that it is possible to continue validating starting
1119 // from prior state.
1120 const size_t mid{REGTEST_BLOCK_DATA.size() / 2};
1121
1122 {
1123 auto chainman{create_chainman(
1124 test_directory, /*reindex=*/false, /*wipe_chainstate=*/false,
1125 /*block_tree_db_in_memory=*/false, /*chainstate_db_in_memory=*/false, context)};
1126 for (size_t i{0}; i < mid; i++) {
1128 bool new_block{false};
1129 BOOST_CHECK(chainman->ProcessBlock(block, &new_block));
1130 BOOST_CHECK(new_block);
1131 }
1132 }
1133
1134 auto chainman{create_chainman(
1135 test_directory, /*reindex=*/false, /*wipe_chainstate=*/false,
1136 /*block_tree_db_in_memory=*/false, /*chainstate_db_in_memory=*/false, context)};
1137
1138 for (size_t i{mid}; i < REGTEST_BLOCK_DATA.size(); i++) {
1140 bool new_block{false};
1141 BOOST_CHECK(chainman->ProcessBlock(block, &new_block));
1142 BOOST_CHECK(new_block);
1143 }
1144
1145 auto chain = chainman->GetChain();
1146 auto tip = chain.Entries().back();
1147 auto read_block = chainman->ReadBlock(tip).value();
1148 check_equal(read_block.ToBytes(), hex_string_to_byte_vec(REGTEST_BLOCK_DATA[REGTEST_BLOCK_DATA.size() - 1]));
1149
1150 auto tip_2 = tip.GetPrevious().value();
1151 auto read_block_2 = chainman->ReadBlock(tip_2).value();
1152 check_equal(read_block_2.ToBytes(), hex_string_to_byte_vec(REGTEST_BLOCK_DATA[REGTEST_BLOCK_DATA.size() - 2]));
1153
1154 Txid txid = read_block.Transactions()[0].Txid();
1155 Txid txid_2 = read_block_2.Transactions()[0].Txid();
1156 BOOST_CHECK(txid != txid_2);
1157 BOOST_CHECK(txid == txid);
1158 CheckHandle(txid, txid_2);
1159
1160 auto find_transaction = [&chainman](const TxidView& target_txid) -> std::optional<Transaction> {
1161 auto chain = chainman->GetChain();
1162 for (const auto block_tree_entry : chain.Entries()) {
1163 auto block{chainman->ReadBlock(block_tree_entry)};
1164 for (const TransactionView transaction : block->Transactions()) {
1165 if (transaction.Txid() == target_txid) {
1166 return Transaction{transaction};
1167 }
1168 }
1169 }
1170 return std::nullopt;
1171 };
1172
1173 for (const auto block_tree_entry : chain.Entries()) {
1174 auto block{chainman->ReadBlock(block_tree_entry)};
1175 for (const auto transaction : block->Transactions()) {
1176 std::vector<TransactionInput> inputs;
1177 std::vector<TransactionOutput> spent_outputs;
1178 for (const auto input : transaction.Inputs()) {
1179 OutPointView point = input.OutPoint();
1180 if (point.index() == std::numeric_limits<uint32_t>::max()) {
1181 continue;
1182 }
1183 inputs.emplace_back(input);
1184 BOOST_CHECK(point.Txid() != transaction.Txid());
1185 std::optional<Transaction> tx = find_transaction(point.Txid());
1186 BOOST_CHECK(tx.has_value());
1187 BOOST_CHECK(point.Txid() == tx->Txid());
1188 spent_outputs.emplace_back(tx->GetOutput(point.index()));
1189 }
1190 BOOST_CHECK(inputs.size() == spent_outputs.size());
1191 ScriptVerifyStatus status = ScriptVerifyStatus::OK;
1192 const PrecomputedTransactionData precomputed_txdata{transaction, spent_outputs};
1193 for (size_t i{0}; i < inputs.size(); ++i) {
1194 BOOST_CHECK(spent_outputs[i].GetScriptPubkey().Verify(spent_outputs[i].Amount(), transaction, &precomputed_txdata, i, ScriptVerificationFlags::ALL, status));
1195 }
1196 }
1197 }
1198
1199 // Read spent outputs for current tip and its previous block
1200 BlockSpentOutputs block_spent_outputs{chainman->ReadBlockSpentOutputs(tip)};
1201 BlockSpentOutputs block_spent_outputs_prev{chainman->ReadBlockSpentOutputs(*tip.GetPrevious())};
1202 CheckHandle(block_spent_outputs, block_spent_outputs_prev);
1203 CheckRange(block_spent_outputs_prev.TxsSpentOutputs(), block_spent_outputs_prev.Count());
1204 BOOST_CHECK_EQUAL(block_spent_outputs.Count(), 1);
1205
1206 // Get transaction spent outputs from the last transaction in the two blocks
1207 TransactionSpentOutputsView transaction_spent_outputs{block_spent_outputs.GetTxSpentOutputs(block_spent_outputs.Count() - 1)};
1208 TransactionSpentOutputs owned_transaction_spent_outputs{transaction_spent_outputs};
1209 TransactionSpentOutputs owned_transaction_spent_outputs_prev{block_spent_outputs_prev.GetTxSpentOutputs(block_spent_outputs_prev.Count() - 1)};
1210 CheckHandle(owned_transaction_spent_outputs, owned_transaction_spent_outputs_prev);
1211 CheckRange(transaction_spent_outputs.Coins(), transaction_spent_outputs.Count());
1212
1213 // Get the last coin from the transaction spent outputs
1214 CoinView coin{transaction_spent_outputs.GetCoin(transaction_spent_outputs.Count() - 1)};
1215 BOOST_CHECK(!coin.IsCoinbase());
1216 Coin owned_coin{coin};
1217 Coin owned_coin_prev{owned_transaction_spent_outputs_prev.GetCoin(owned_transaction_spent_outputs_prev.Count() - 1)};
1218 CheckHandle(owned_coin, owned_coin_prev);
1219
1220 // Validate coin properties
1221 TransactionOutputView output = coin.GetOutput();
1222 uint32_t coin_height = coin.GetConfirmationHeight();
1223 BOOST_CHECK_EQUAL(coin_height, 205);
1224 BOOST_CHECK_EQUAL(output.Amount(), 100000000);
1225
1226 // Test script pubkey serialization
1227 auto script_pubkey = output.GetScriptPubkey();
1228 auto script_pubkey_bytes{script_pubkey.ToBytes()};
1229 BOOST_CHECK_EQUAL(script_pubkey_bytes.size(), 22);
1230 auto round_trip_script_pubkey{ScriptPubkey(script_pubkey_bytes)};
1231 BOOST_CHECK_EQUAL(round_trip_script_pubkey.ToBytes().size(), 22);
1232
1233 for (const auto tx_spent_outputs : block_spent_outputs.TxsSpentOutputs()) {
1234 for (const auto coins : tx_spent_outputs.Coins()) {
1235 BOOST_CHECK_GT(coins.GetOutput().Amount(), 1);
1236 }
1237 }
1238
1239 CheckRange(chain.Entries(), chain.CountEntries());
1240
1241 for (const BlockTreeEntry entry : chain.Entries()) {
1242 std::optional<Block> block{chainman->ReadBlock(entry)};
1243 if (block) {
1244 for (const TransactionView transaction : block->Transactions()) {
1245 for (const TransactionOutputView output : transaction.Outputs()) {
1246 // skip data carrier outputs
1247 if ((unsigned char)output.GetScriptPubkey().ToBytes()[0] == 0x6a) {
1248 continue;
1249 }
1250 BOOST_CHECK_GT(output.Amount(), 1);
1251 }
1252 }
1253 }
1254 }
1255
1256 int32_t count{0};
1257 for (const auto entry : chain.Entries()) {
1258 BOOST_CHECK_EQUAL(entry.GetHeight(), count);
1259 ++count;
1260 }
1261 BOOST_CHECK_EQUAL(count, chain.CountEntries());
1262
1263
1264 fs::remove(test_directory.m_directory / "blocks" / "blk00000.dat");
1265 BOOST_CHECK(!chainman->ReadBlock(tip_2).has_value());
1266 fs::remove(test_directory.m_directory / "blocks" / "rev00000.dat");
1267 BOOST_CHECK_THROW(chainman->ReadBlockSpentOutputs(tip), std::runtime_error);
1268}
1269
1270// -----------------------------------------------------------------------------
1271// CheckTransaction tests
1272//
1273// Transaction hex below is copied from src/test/data/tx_invalid.json (entries
1274// marked "BADTX") and tx_valid.json. CheckTransaction performs only basic context-free
1275// consensus checks and can only produce two outcomes:
1276// - VALID (ValidationMode::VALID, TxValidationResult::UNSET)
1277// - INVALID (ValidationMode::INVALID, TxValidationResult::CONSENSUS)
1278// Other TxValidationResult values are set by higher-level validation and are
1279// not reachable through btck_transaction_check.
1280// -----------------------------------------------------------------------------
1281BOOST_AUTO_TEST_CASE(btck_transaction_check_tests)
1282{
1283 using namespace btck;
1284
1285 constexpr std::string_view valid_tx_hex{
1286 "01000000010001000000000000000000000000000000000000000000000000000000000000"
1287 "000000006a473044022067288ea50aa799543a536ff9306f8e1cba05b9c6b10951175b92"
1288 "4f96732555ed022026d7b5265f38d21541519e4a1e55044d5b9e17e15cdbaf29ae3792e9"
1289 "9e883e7a012103ba8c8b86dea131c22ab967e6dd99bdae8eff7a1f75a2c35f1f944109e3"
1290 "fe5e22ffffffff010000000000000000015100000000"};
1291 constexpr std::string_view no_outputs_tx_hex{
1292 "01000000010001000000000000000000000000000000000000000000000000000000000000"
1293 "000000006d483045022100f16703104aab4e4088317c862daec83440242411b039d14280e0"
1294 "3dd33b487ab802201318a7be236672c5c56083eb7a5a195bc57a40af7923ff8545016cd3b5"
1295 "71e2a601232103c40e5d339df3f30bf753e7e04450ae4ef76c9e45587d1d993bdc4cd06f06"
1296 "51c7acffffffff0000000000"};
1297
1298 auto expect_valid = [](std::string_view hex) {
1302 BOOST_CHECK(st.GetValidationMode() == ValidationMode::VALID);
1303 BOOST_CHECK(st.GetTxValidationResult() == TxValidationResult::UNSET);
1304 };
1305
1306 auto expect_invalid = [](std::string_view hex) {
1309 BOOST_CHECK(!CheckTransaction(tx, st));
1311 BOOST_CHECK(st.GetTxValidationResult() == TxValidationResult::CONSENSUS);
1312 };
1313
1314 // Valid: simple 1-in 1-out transaction (from tx_valid.json)
1315 expect_valid(valid_tx_hex);
1316
1317 // Valid coinbase with scriptSig size 2 (from tx_valid.json)
1318 expect_valid(
1319 "01000000010000000000000000000000000000000000000000000000000000000000000000"
1320 "ffffffff025151ffffffff010000000000000000015100000000");
1321
1322 // No outputs (BADTX from tx_invalid.json)
1323 expect_invalid(no_outputs_tx_hex);
1324
1325 {
1326 Transaction valid_tx{hex_string_to_byte_vec(valid_tx_hex)};
1327 Transaction invalid_tx{hex_string_to_byte_vec(no_outputs_tx_hex)};
1328 TxValidationState state;
1329
1330 BOOST_CHECK(btck_transaction_check(valid_tx.get(), state.get()) == 1);
1331 BOOST_CHECK(state.GetValidationMode() == ValidationMode::VALID);
1332 BOOST_CHECK(state.GetTxValidationResult() == TxValidationResult::UNSET);
1333
1334 BOOST_CHECK(btck_transaction_check(invalid_tx.get(), state.get()) == 0);
1336 BOOST_CHECK(state.GetTxValidationResult() == TxValidationResult::CONSENSUS);
1337 }
1338
1339 // Negative output (BADTX)
1340 expect_invalid(
1341 "01000000010001000000000000000000000000000000000000000000000000000000000000"
1342 "000000006d4830450220063222cbb128731fc09de0d7323746539166544d6c1df84d867cce"
1343 "a84bcc8903022100bf568e8552844de664cd41648a031554327aa8844af34b4f27397c65b9"
1344 "2c04de0123210243ec37dee0e2e053a9c976f43147e79bc7d9dc606ea51010af1ac80db6b0"
1345 "69e1acffffffff01ffffffffffffffff015100000000");
1346
1347 // MAX_MONEY + 1 output (BADTX)
1348 expect_invalid(
1349 "01000000010001000000000000000000000000000000000000000000000000000000000000"
1350 "000000006e493046022100e1eadba00d9296c743cb6ecc703fd9ddc9b3cd12906176a226ae"
1351 "4c18d6b00796022100a71aef7d2874deff681ba6080f1b278bac7bb99c61b08a85f4311970"
1352 "ffe7f63f012321030c0588dc44d92bdcbf8e72093466766fdc265ead8db64517b0c542275b"
1353 "70fffbacffffffff010140075af0750700015100000000");
1354
1355 // MAX_MONEY output + 1 output: sum exceeds MAX_MONEY (BADTX)
1356 expect_invalid(
1357 "01000000010001000000000000000000000000000000000000000000000000000000000000"
1358 "000000006d483045022027deccc14aa6668e78a8c9da3484fbcd4f9dcc9bb7d1b85146314b"
1359 "21b9ae4d86022100d0b43dece8cfb07348de0ca8bc5b86276fa88f7f2138381128b7c36ab2"
1360 "e42264012321029bb13463ddd5d2cc05da6e84e37536cb9525703cfd8f43afdb414988987a"
1361 "92f6acffffffff020040075af075070001510001000000000000015100000000");
1362
1363 // Duplicate inputs (BADTX)
1364 expect_invalid(
1365 "01000000020001000000000000000000000000000000000000000000000000000000000000"
1366 "000000006c47304402204bb1197053d0d7799bf1b30cd503c44b58d6240cccbdc85b6fe76d"
1367 "087980208f02204beeed78200178ffc6c74237bb74b3f276bbb4098b5605d814304fe128bf"
1368 "1431012321039e8815e15952a7c3fada1905f8cf55419837133bd7756c0ef14fc8dfe50c0d"
1369 "eaacffffffff0001000000000000000000000000000000000000000000000000000000000000"
1370 "000000006c47304402202306489afef52a6f62e90bf750bbcdf40c06f5c6b138286e6b6b8617"
1371 "6bb9341802200dba98486ea68380f47ebb19a7df173b99e6bc9c681d6ccf3bde31465d1f16"
1372 "b3012321039e8815e15952a7c3fada1905f8cf55419837133bd7756c0ef14fc8dfe50c0dea"
1373 "acffffffff010000000000000000015100000000");
1374
1375 // Coinbase with scriptSig size 1: too small (BADTX)
1376 expect_invalid(
1377 "01000000010000000000000000000000000000000000000000000000000000000000000000"
1378 "ffffffff0151ffffffff010000000000000000015100000000");
1379
1380 // Coinbase with scriptSig size 101: too large (BADTX)
1381 expect_invalid(
1382 "01000000010000000000000000000000000000000000000000000000000000000000000000"
1383 "ffffffff6551515151515151515151515151515151515151515151515151515151515151515151"
1384 "515151515151515151515151515151515151515151515151515151515151515151515151515151"
1385 "51515151515151515151515151515151515151515151515151515151ffffffff01000000000000"
1386 "0000015100000000");
1387
1388 // Null prevout in non-coinbase: two inputs, one is null (BADTX)
1389 expect_invalid(
1390 "01000000020000000000000000000000000000000000000000000000000000000000000000"
1391 "ffffffff00ffffffff000100000000000000000000000000000000000000000000000000000000"
1392 "00000000000000ffffffff010000000000000000015100000000");
1393}
int btck_transaction_check(const btck_Transaction *tx, btck_TxValidationState *validation_state)
constexpr std::array< std::string_view, 206 > REGTEST_BLOCK_DATA
Definition: block_data.h:9
BOOST_CHECK_EXCEPTION predicates to check the specific validation error.
Definition: common.h:19
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)
Definition: test_kernel.cpp:95
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::array< std::byte, 80 > ToBytes() const
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)
uint32_t index() const
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
std::vector< std::byte > ToBytes() const
ScriptPubkeyView GetScriptPubkey() const
TxValidationResult GetTxValidationResult() const
ValidationMode GetValidationMode() const
Txid(const TxidView &view)
const CType * get() const
static bool exists(const path &p)
Definition: fs.h:96
#define T(expected, seed, data)
@ ALL
Definition: categories.h:49
@ VALIDATION
Definition: categories.h:37
@ BENCH
Definition: categories.h:20
@ KERNEL
Definition: categories.h:47
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.
Definition: miniscript.h:351
#define BOOST_CHECK_THROW(stmt, excMatch)
Definition: object.cpp:18
#define BOOST_CHECK_EQUAL(v1, v2)
Definition: object.cpp:17
#define BOOST_CHECK(expr)
Definition: object.cpp:16
static bool Verify(const CScript &scriptSig, const CScript &scriptPubKey, bool fStrict, ScriptError &err)
fs::path m_directory
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)
Definition: test_kernel.cpp:65
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)
Definition: test_kernel.cpp:48
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)
Definition: test_kernel.cpp:83
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
Definition: test_kernel.cpp:78
void CheckRange(const RangeType &range, size_t expected_size)
void chainman_mainnet_validation_test(TestDirectory &test_directory)
std::string random_string(uint32_t length)
Definition: test_kernel.cpp:30
constexpr auto VERIFY_ALL_PRE_TAPROOT
Definition: test_kernel.cpp:81
static int count
bool CheckTransaction(const CTransaction &tx, TxValidationState &state)
Definition: tx_check.cpp:11