Bitcoin Core 31.99.0
P2P Digital Currency
bech32.cpp
Go to the documentation of this file.
1// Copyright (c) 2019-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
5#include <bech32.h>
6#include <test/fuzz/fuzz.h>
8#include <util/strencodings.h>
9
10#include <cassert>
11#include <cstdint>
12#include <string>
13#include <vector>
14
15FUZZ_TARGET(bech32_random_decode)
16{
17 auto limit = bech32::CharLimit::BECH32;
18 FuzzedDataProvider fdp(buffer.data(), buffer.size());
19 auto random_string = fdp.ConsumeRandomLengthString(limit + 1);
20 auto decoded = bech32::Decode(random_string, limit);
21
22 if (decoded.hrp.empty()) {
23 assert(decoded.encoding == bech32::Encoding::INVALID);
24 assert(decoded.data.empty());
25 } else {
26 assert(decoded.encoding != bech32::Encoding::INVALID);
27 auto reencoded = bech32::Encode(decoded.encoding, decoded.hrp, decoded.data);
29 }
30}
31
32// https://github.com/bitcoin/bips/blob/master/bip-0173.mediawiki and https://github.com/bitcoin/bips/blob/master/bip-0350.mediawiki
34{
35 std::string hrp;
36 size_t length = fdp.ConsumeIntegralInRange<size_t>(1, 83);
37 for (size_t i = 0; i < length; ++i) {
38 // Generate lowercase ASCII characters in ([33-126] - ['A'-'Z']) range
39 char c = fdp.ConsumeBool()
40 ? fdp.ConsumeIntegralInRange<char>(33, 'A' - 1)
41 : fdp.ConsumeIntegralInRange<char>('Z' + 1, 126);
42 hrp += c;
43 }
44 return hrp;
45}
46
47FUZZ_TARGET(bech32_roundtrip)
48{
49 FuzzedDataProvider fdp(buffer.data(), buffer.size());
50 auto hrp = GenerateRandomHRP(fdp);
51
52 auto input_chars = fdp.ConsumeBytes<unsigned char>(fdp.ConsumeIntegralInRange<size_t>(0, 82));
53 std::vector<uint8_t> converted_input;
54 ConvertBits<8, 5, true>([&](auto c) { converted_input.push_back(c); }, input_chars.begin(), input_chars.end());
55
56 auto size = converted_input.size() + hrp.length() + std::string({bech32::SEPARATOR}).size() + bech32::CHECKSUM_SIZE;
57 if (size <= bech32::CharLimit::BECH32) {
59 auto encoded = bech32::Encode(encoding, hrp, converted_input);
60 assert(!encoded.empty());
61
62 const auto decoded = bech32::Decode(encoded);
63 assert(decoded.encoding == encoding);
64 assert(decoded.hrp == hrp);
65 assert(decoded.data == converted_input);
66 }
67 }
68}
std::string ConsumeRandomLengthString(size_t max_length)
std::vector< T > ConsumeBytes(size_t num_bytes)
T ConsumeIntegralInRange(T min, T max)
@ INVALID
Failed decoding.
@ BECH32
Bech32 encoding as defined in BIP173.
@ BECH32M
Bech32m encoding as defined in BIP350.
DecodeResult Decode(const std::string &str, CharLimit limit)
Decode a Bech32 or Bech32m string.
Definition: bech32.cpp:374
std::string Encode(Encoding encoding, const std::string &hrp, const data &values)
Encode a Bech32 or Bech32m string.
Definition: bech32.cpp:358
static constexpr size_t CHECKSUM_SIZE
Definition: bech32.h:24
static constexpr char SEPARATOR
Definition: bech32.h:25
std::string GenerateRandomHRP(FuzzedDataProvider &fdp)
Definition: bech32.cpp:33
FUZZ_TARGET(bech32_random_decode)
Definition: bech32.cpp:15
std::string random_string(uint32_t length)
Definition: test_kernel.cpp:30
bool CaseInsensitiveEqual(std::string_view s1, std::string_view s2)
Locale-independent, ASCII-only comparator.
assert(!tx.IsCoinBase())