/ src / test / fuzz / bech32.cpp
bech32.cpp
 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>
 7  #include <test/fuzz/FuzzedDataProvider.h>
 8  #include <test/util/str.h>
 9  #include <util/strencodings.h>
10  
11  #include <cassert>
12  #include <cstdint>
13  #include <string>
14  #include <vector>
15  
16  FUZZ_TARGET(bech32_random_decode)
17  {
18      auto limit = bech32::CharLimit::BECH32;
19      FuzzedDataProvider fdp(buffer.data(), buffer.size());
20      auto random_string = fdp.ConsumeRandomLengthString(limit + 1);
21      auto decoded = bech32::Decode(random_string, limit);
22  
23      if (decoded.hrp.empty()) {
24          assert(decoded.encoding == bech32::Encoding::INVALID);
25          assert(decoded.data.empty());
26      } else {
27          assert(decoded.encoding != bech32::Encoding::INVALID);
28          auto reencoded = bech32::Encode(decoded.encoding, decoded.hrp, decoded.data);
29          assert(CaseInsensitiveEqual(random_string, reencoded));
30      }
31  }
32  
33  // https://github.com/bitcoin/bips/blob/master/bip-0173.mediawiki and https://github.com/bitcoin/bips/blob/master/bip-0350.mediawiki
34  std::string GenerateRandomHRP(FuzzedDataProvider& fdp)
35  {
36      std::string hrp;
37      size_t length = fdp.ConsumeIntegralInRange<size_t>(1, 83);
38      for (size_t i = 0; i < length; ++i) {
39          // Generate lowercase ASCII characters in ([33-126] - ['A'-'Z']) range
40          char c = fdp.ConsumeBool()
41                   ? fdp.ConsumeIntegralInRange<char>(33, 'A' - 1)
42                   : fdp.ConsumeIntegralInRange<char>('Z' + 1, 126);
43          hrp += c;
44      }
45      return hrp;
46  }
47  
48  FUZZ_TARGET(bech32_roundtrip)
49  {
50      FuzzedDataProvider fdp(buffer.data(), buffer.size());
51      auto hrp = GenerateRandomHRP(fdp);
52  
53      auto input_chars = fdp.ConsumeBytes<unsigned char>(fdp.ConsumeIntegralInRange<size_t>(0, 82));
54      std::vector<uint8_t> converted_input;
55      ConvertBits<8, 5, true>([&](auto c) { converted_input.push_back(c); }, input_chars.begin(), input_chars.end());
56  
57      auto size = converted_input.size() + hrp.length() + std::string({bech32::SEPARATOR}).size() + bech32::CHECKSUM_SIZE;
58      if (size <= bech32::CharLimit::BECH32) {
59          for (auto encoding: {bech32::Encoding::BECH32, bech32::Encoding::BECH32M}) {
60              auto encoded = bech32::Encode(encoding, hrp, converted_input);
61              assert(!encoded.empty());
62  
63              const auto decoded = bech32::Decode(encoded);
64              assert(decoded.encoding == encoding);
65              assert(decoded.hrp == hrp);
66              assert(decoded.data == converted_input);
67          }
68      }
69  }