forked from bitcoin/bitcoin
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
kernel: Add pure kernel bitcoin-chainstate
This showcases a re-implementation of bitcoin-chainstate only using the kernel C++ API header.
- Loading branch information
1 parent
ae24d2f
commit f25616b
Showing
4 changed files
with
215 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,200 @@ | ||
#include <kernel/bitcoinkernel_wrapper.h> | ||
|
||
#include <cassert> | ||
#include <filesystem> | ||
#include <iostream> | ||
#include <optional> | ||
#include <string> | ||
#include <string_view> | ||
#include <sstream> | ||
|
||
std::vector<unsigned char> hex_string_to_char_vec(const std::string& hex) | ||
{ | ||
std::vector<unsigned char> bytes; | ||
|
||
for (size_t i{0}; i < hex.length(); i += 2) { | ||
std::string byteString{hex.substr(i, 2)}; | ||
unsigned char byte = (char)std::strtol(byteString.c_str(), nullptr, 16); | ||
bytes.push_back(byte); | ||
} | ||
|
||
return bytes; | ||
} | ||
|
||
class KernelLog | ||
{ | ||
public: | ||
void LogMessage(std::string_view message) | ||
{ | ||
std::cout << "kernel: " << message; | ||
} | ||
}; | ||
|
||
class TestValidationInterface : public ValidationInterface<TestValidationInterface> | ||
{ | ||
public: | ||
TestValidationInterface() : ValidationInterface() {} | ||
|
||
std::optional<std::string> m_expected_valid_block = std::nullopt; | ||
|
||
void BlockChecked(const UnownedBlock block, const BlockValidationState state) override | ||
{ | ||
auto mode{state.ValidationMode()}; | ||
switch (mode) { | ||
case kernel_ValidationMode::kernel_VALIDATION_STATE_VALID: { | ||
std::cout << "Valid block" << std::endl; | ||
return; | ||
} | ||
case kernel_ValidationMode::kernel_VALIDATION_STATE_INVALID: { | ||
std::cout << "Invalid block: "; | ||
auto result{state.BlockValidationResult()}; | ||
switch (result) { | ||
case kernel_BlockValidationResult::kernel_BLOCK_RESULT_UNSET: | ||
std::cout << "initial value. Block has not yet been rejected" << std::endl; | ||
break; | ||
case kernel_BlockValidationResult::kernel_BLOCK_HEADER_LOW_WORK: | ||
std::cout << "the block header may be on a too-little-work chain" << std::endl; | ||
break; | ||
case kernel_BlockValidationResult::kernel_BLOCK_CONSENSUS: | ||
std::cout << "invalid by consensus rules (excluding any below reasons)" << std::endl; | ||
break; | ||
case kernel_BlockValidationResult::kernel_BLOCK_CACHED_INVALID: | ||
std::cout << "this block was cached as being invalid and we didn't store the reason why" << std::endl; | ||
break; | ||
case kernel_BlockValidationResult::kernel_BLOCK_INVALID_HEADER: | ||
std::cout << "invalid proof of work or time too old" << std::endl; | ||
break; | ||
case kernel_BlockValidationResult::kernel_BLOCK_MUTATED: | ||
std::cout << "the block's data didn't match the data committed to by the PoW" << std::endl; | ||
break; | ||
case kernel_BlockValidationResult::kernel_BLOCK_MISSING_PREV: | ||
std::cout << "We don't have the previous block the checked one is built on" << std::endl; | ||
break; | ||
case kernel_BlockValidationResult::kernel_BLOCK_INVALID_PREV: | ||
std::cout << "A block this one builds on is invalid" << std::endl; | ||
break; | ||
case kernel_BlockValidationResult::kernel_BLOCK_TIME_FUTURE: | ||
std::cout << "block timestamp was > 2 hours in the future (or our clock is bad)" << std::endl; | ||
break; | ||
case kernel_BlockValidationResult::kernel_BLOCK_CHECKPOINT: | ||
std::cout << "the block failed to meet one of our checkpoints" << std::endl; | ||
break; | ||
} | ||
return; | ||
} | ||
case kernel_ValidationMode::kernel_VALIDATION_STATE_ERROR: { | ||
std::cout << "Internal error" << std::endl; | ||
return; | ||
} | ||
} | ||
} | ||
}; | ||
|
||
class TestKernelNotifications : public KernelNotifications<TestKernelNotifications> | ||
{ | ||
public: | ||
void BlockTipHandler(kernel_SynchronizationState state, const kernel_BlockIndex* index) override | ||
{ | ||
std::cout << "Block tip changed" << std::endl; | ||
} | ||
|
||
void ProgressHandler(std::string_view title, int progress_percent, bool resume_possible) override | ||
{ | ||
std::cout << "Made progress: " << title << " " << progress_percent << "%" << std::endl; | ||
} | ||
|
||
void WarningSetHandler(kernel_Warning warning, std::string_view message) override | ||
{ | ||
std::cout << message << std::endl; | ||
} | ||
|
||
void WarningUnsetHandler(kernel_Warning warning) override | ||
{ | ||
std::cout << "Warning unset: " << warning << std::endl; | ||
} | ||
|
||
void FlushErrorHandler(std::string_view error) override | ||
{ | ||
std::cout << error << std::endl; | ||
} | ||
|
||
void FatalErrorHandler(std::string_view error) override | ||
{ | ||
std::cout << error << std::endl; | ||
} | ||
}; | ||
|
||
int main(int argc, char* argv[]) | ||
{ | ||
// SETUP: Argument parsing and handling | ||
if (argc != 2) { | ||
std::cerr | ||
<< "Usage: " << argv[0] << " DATADIR" << std::endl | ||
<< "Display DATADIR information, and process hex-encoded blocks on standard input." << std::endl | ||
<< std::endl | ||
<< "IMPORTANT: THIS EXECUTABLE IS EXPERIMENTAL, FOR TESTING ONLY, AND EXPECTED TO" << std::endl | ||
<< " BREAK IN FUTURE VERSIONS. DO NOT USE ON YOUR ACTUAL DATADIR." << std::endl; | ||
return 1; | ||
} | ||
std::filesystem::path abs_datadir{std::filesystem::absolute(argv[1])}; | ||
std::filesystem::create_directories(abs_datadir); | ||
|
||
kernel_LoggingOptions logging_options = { | ||
.log_timestamps = true, | ||
.log_time_micros = false, | ||
.log_threadnames = false, | ||
.log_sourcelocations = false, | ||
.always_print_category_levels = true, | ||
}; | ||
|
||
Logger logger{std::make_unique<KernelLog>(KernelLog{}), logging_options}; | ||
|
||
ContextOptions options{}; | ||
ChainParams params{kernel_ChainType::kernel_CHAIN_TYPE_REGTEST}; | ||
options.SetChainParams(params); | ||
|
||
TestKernelNotifications notifications{}; | ||
options.SetNotifications(notifications); | ||
TestValidationInterface validation_interface{}; | ||
options.SetValidationInterface(validation_interface); | ||
|
||
Context context{options}; | ||
assert(context); | ||
|
||
ChainstateManagerOptions chainman_opts{context, abs_datadir}; | ||
assert(chainman_opts); | ||
chainman_opts.SetWorkerThreads(4); | ||
BlockManagerOptions blockman_opts{context, abs_datadir / "blocks"}; | ||
assert(blockman_opts); | ||
ChainstateLoadOptions chainstate_load_opts{}; | ||
|
||
auto chainman{std::make_unique<ChainMan>(context, chainman_opts, blockman_opts, chainstate_load_opts)}; | ||
assert(chainman); | ||
|
||
std::cout << "Enter the block you want to validate on the next line:" << std::endl; | ||
|
||
for (std::string line; std::getline(std::cin, line);) { | ||
if (line.empty()) { | ||
std::cerr << "Empty line found, try again:" << std::endl; | ||
continue; | ||
} | ||
|
||
auto raw_block{hex_string_to_char_vec(line)}; | ||
auto block = Block{raw_block}; | ||
if (!block) { | ||
std::cout << "Failed to parse entered block, try again:" << std::endl; | ||
continue; | ||
} | ||
|
||
bool new_block = false; | ||
bool accepted = chainman->ProcessBlock(block, &new_block); | ||
if (accepted) { | ||
std::cout << "Validated block successfully." << std::endl; | ||
} else { | ||
std::cout << "Block was not accepted" << std::endl; | ||
} | ||
if (!new_block) { | ||
std::cout << "Block is a duplicate" << std::endl; | ||
} | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters