-
Notifications
You must be signed in to change notification settings - Fork 1.5k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
[fold] Add unit tests to compare fast version with ref version
- Loading branch information
Showing
2 changed files
with
356 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,355 @@ | ||
//------------------------------------------------------------------------------ | ||
/* | ||
This file is part of rippled: https://github.com/ripple/rippled | ||
Copyright (c) 2022 Ripple Labs Inc. | ||
Permission to use, copy, modify, and/or distribute this software for any | ||
purpose with or without fee is hereby granted, provided that the above | ||
copyright notice and this permission notice appear in all copies. | ||
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES | ||
WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF | ||
MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR | ||
ANY SPECIAL , DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES | ||
WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN | ||
ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF | ||
OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. | ||
*/ | ||
//============================================================================== | ||
|
||
#ifndef _MSC_VER | ||
|
||
#include <ripple/beast/unit_test.h> | ||
#include <ripple/protocol/impl/b58_utils.h> | ||
#include <ripple/protocol/tokens.h> | ||
|
||
#include <boost/multiprecision/cpp_int.hpp> | ||
#include <boost/random.hpp> | ||
|
||
#include <array> | ||
#include <cstddef> | ||
#include <random> | ||
#include <span> | ||
#include <sstream> | ||
|
||
namespace ripple { | ||
namespace test { | ||
namespace { | ||
|
||
[[nodiscard]] inline auto | ||
randEngine() -> std::mt19937& | ||
{ | ||
static std::mt19937 r = [] { | ||
std::random_device rd; | ||
return std::mt19937{rd()}; | ||
}(); | ||
return r; | ||
} | ||
|
||
constexpr int numTokenTypeIndexes = 9; | ||
|
||
[[nodiscard]] inline auto | ||
tokenTypeAndSize(int i) -> std::tuple<ripple::TokenType, std::size_t> | ||
{ | ||
assert(i < numTokenTypeIndexes); | ||
|
||
switch (i) | ||
{ | ||
using enum ripple::TokenType; | ||
case 0: | ||
return {None, 20}; | ||
case 1: | ||
return {NodePublic, 32}; | ||
case 2: | ||
return {NodePublic, 33}; | ||
case 3: | ||
return {NodePrivate, 32}; | ||
case 4: | ||
return {AccountID, 20}; | ||
case 5: | ||
return {AccountPublic, 32}; | ||
case 6: | ||
return {AccountPublic, 33}; | ||
case 7: | ||
return {AccountSecret, 32}; | ||
case 8: | ||
return {FamilySeed, 16}; | ||
default: | ||
assert(0); | ||
return {AccountPublic, 33}; | ||
} | ||
} | ||
|
||
[[nodiscard]] inline auto | ||
randomTokenTypeAndSize() -> std::tuple<ripple::TokenType, std::size_t> | ||
{ | ||
using namespace ripple; | ||
auto& rng = randEngine(); | ||
std::uniform_int_distribution<> d(0, 8); | ||
return tokenTypeAndSize(d(rng)); | ||
} | ||
|
||
// Return the token type and subspan of `d` to use as test data. | ||
[[nodiscard]] inline auto | ||
randomB256TestData(std::span<std::uint8_t> d) | ||
-> std::tuple<ripple::TokenType, std::span<std::uint8_t>> | ||
{ | ||
auto& rng = randEngine(); | ||
std::uniform_int_distribution<std::uint8_t> dist(0, 255); | ||
auto [tokType, tokSize] = randomTokenTypeAndSize(); | ||
std::generate(d.begin(), d.begin() + tokSize, [&] { return dist(rng); }); | ||
return {tokType, d.subspan(0, tokSize)}; | ||
} | ||
|
||
inline auto | ||
printAsChar(std::span<std::uint8_t> a, std::span<std::uint8_t> b) | ||
{ | ||
auto asString = [](std::span<std::uint8_t> s) { | ||
std::string r; | ||
r.resize(s.size()); | ||
std::copy(s.begin(), s.end(), r.begin()); | ||
return r; | ||
}; | ||
auto sa = asString(a); | ||
auto sb = asString(b); | ||
std::cerr << "\n\n" << sa << "\n" << sb << "\n"; | ||
} | ||
|
||
inline auto | ||
printAsInt(std::span<std::uint8_t> a, std::span<std::uint8_t> b) | ||
{ | ||
auto asString = [](std::span<std::uint8_t> s) -> std::string { | ||
std::stringstream sstr; | ||
for (auto i : s) | ||
{ | ||
sstr << std::setw(3) << int(i) << ','; | ||
} | ||
return sstr.str(); | ||
}; | ||
auto sa = asString(a); | ||
auto sb = asString(b); | ||
std::cerr << "\n\n" << sa << "\n" << sb << "\n"; | ||
} | ||
|
||
} // namespace | ||
|
||
namespace multiprecision_utils { | ||
|
||
boost::multiprecision::checked_uint512_t | ||
toBoostMP(std::span<std::uint64_t> in) | ||
{ | ||
boost::multiprecision::checked_uint512_t mbp = 0; | ||
for (auto i = in.rbegin(); i != in.rend(); ++i) | ||
{ | ||
mbp <<= 64; | ||
mbp += *i; | ||
} | ||
return mbp; | ||
} | ||
|
||
std::vector<std::uint64_t> | ||
randomBigInt(std::uint8_t minSize = 1, std::uint8_t maxSize = 5) | ||
{ | ||
auto eng = randEngine(); | ||
std::uniform_int_distribution<std::uint8_t> numCoeffDist(minSize, maxSize); | ||
std::uniform_int_distribution<std::uint64_t> dist; | ||
auto const numCoeff = numCoeffDist(eng); | ||
std::vector<std::uint64_t> coeffs; | ||
coeffs.reserve(numCoeff); | ||
for (int i = 0; i < numCoeff; ++i) | ||
{ | ||
coeffs.push_back(dist(eng)); | ||
} | ||
return coeffs; | ||
} | ||
} // namespace multiprecision_utils | ||
|
||
class base58_test : public beast::unit_test::suite | ||
{ | ||
void | ||
testMultiprecision() | ||
{ | ||
testcase("b58_multiprecision"); | ||
|
||
// Generate a random boost multiprecision | ||
// Import the bytes into | ||
using namespace boost::multiprecision; | ||
|
||
constexpr std::size_t iters = 1000000; | ||
auto eng = randEngine(); | ||
std::uniform_int_distribution<std::uint64_t> dist; | ||
for (int i = 0; i < iters; ++i) | ||
{ | ||
std::uint64_t const d = dist(eng); | ||
if (!d) | ||
continue; | ||
auto bigInt = multiprecision_utils::randomBigInt(); | ||
auto const boostBigInt = multiprecision_utils::toBoostMP( | ||
std::span<std::uint64_t>(bigInt.data(), bigInt.size())); | ||
|
||
auto const refDiv = boostBigInt / d; | ||
auto const refMod = boostBigInt % d; | ||
|
||
auto const mod = b58_fast::detail::inplace_bigint_div_rem( | ||
std::span<uint64_t>(bigInt.data(), bigInt.size()), d); | ||
auto const foundDiv = multiprecision_utils::toBoostMP(bigInt); | ||
BEAST_EXPECT(refMod.convert_to<std::uint64_t>() == mod); | ||
BEAST_EXPECT(foundDiv == refDiv); | ||
} | ||
for (int i = 0; i < iters; ++i) | ||
{ | ||
std::uint64_t const d = dist(eng); | ||
auto bigInt = multiprecision_utils::randomBigInt(/*minSize*/ 2); | ||
if (bigInt[bigInt.size() - 1] == | ||
std::numeric_limits<std::uint64_t>::max()) | ||
{ | ||
bigInt[bigInt.size() - 1] -= 1; // Prevent overflow | ||
} | ||
auto const boostBigInt = multiprecision_utils::toBoostMP( | ||
std::span<std::uint64_t>(bigInt.data(), bigInt.size())); | ||
|
||
auto const refAdd = boostBigInt + d; | ||
|
||
b58_fast::detail::inplace_bigint_add( | ||
std::span<uint64_t>(bigInt.data(), bigInt.size()), d); | ||
auto const foundAdd = multiprecision_utils::toBoostMP(bigInt); | ||
BEAST_EXPECT(refAdd == foundAdd); | ||
} | ||
for (int i = 0; i < iters; ++i) | ||
{ | ||
std::uint64_t const d = dist(eng); | ||
auto bigInt = multiprecision_utils::randomBigInt(/* minSize */ 2); | ||
// inplace mul requires the most significant coeff to be zero to | ||
// hold the result. | ||
bigInt[bigInt.size() - 1] = 0; | ||
auto const boostBigInt = multiprecision_utils::toBoostMP( | ||
std::span<std::uint64_t>(bigInt.data(), bigInt.size())); | ||
|
||
auto const refMul = boostBigInt * d; | ||
|
||
b58_fast::detail::inplace_bigint_mul( | ||
std::span<uint64_t>(bigInt.data(), bigInt.size()), d); | ||
auto const foundMul = multiprecision_utils::toBoostMP(bigInt); | ||
BEAST_EXPECT(refMul == foundMul); | ||
} | ||
} | ||
|
||
void | ||
testFastMatchesRef() | ||
{ | ||
testcase("fast_matches_ref"); | ||
std::array<std::uint8_t, 128> b256DataBuf; | ||
std::array<std::uint8_t, 64> b58ResultBuf[2]; | ||
std::array<std::span<std::uint8_t>, 2> b58Result; | ||
|
||
std::array<std::uint8_t, 64> b256ResultBuf[2]; | ||
std::array<std::span<std::uint8_t>, 2> b256Result; | ||
|
||
auto testIt = [&](ripple::TokenType const tokType, | ||
std::span<std::uint8_t> const& b256Data) { | ||
for (int i = 0; i < 2; ++i) | ||
{ | ||
auto const outBuf = | ||
std::span(b58ResultBuf[i].data(), b58ResultBuf[i].size()); | ||
if (i == 0) | ||
{ | ||
auto const r = ripple::b58_fast::encodeBase58Token( | ||
tokType, b256Data, outBuf); | ||
BEAST_EXPECT(r); | ||
b58Result[i] = r.value(); | ||
} | ||
else | ||
{ | ||
auto const s = ripple::b58_ref::encodeBase58Token( | ||
tokType, b256Data.data(), b256Data.size()); | ||
BEAST_EXPECT(s.size()); | ||
b58Result[i] = outBuf.subspan(0, s.size()); | ||
std::copy(s.begin(), s.end(), b58Result[i].begin()); | ||
} | ||
} | ||
if (BEAST_EXPECT(b58Result[0].size() == b58Result[1].size())) | ||
{ | ||
if (!BEAST_EXPECT( | ||
memcmp( | ||
b58Result[0].data(), | ||
b58Result[1].data(), | ||
b58Result[0].size()) == 0)) | ||
{ | ||
printAsChar(b58Result[0], b58Result[1]); | ||
} | ||
} | ||
|
||
for (int i = 0; i < 2; ++i) | ||
{ | ||
auto const outBuf = | ||
std::span(b256ResultBuf[i].data(), b256ResultBuf[i].size()); | ||
if (i == 0) | ||
{ | ||
std::string const in( | ||
b58Result[i].data(), | ||
b58Result[i].data() + b58Result[i].size()); | ||
auto const r = ripple::b58_fast::decodeBase58Token( | ||
tokType, in, outBuf); | ||
BEAST_EXPECT(r); | ||
b256Result[i] = r.value(); | ||
} | ||
else | ||
{ | ||
std::string const st( | ||
b58Result[i].begin(), b58Result[i].end()); | ||
auto const s = | ||
ripple::b58_ref::decodeBase58Token(st, tokType); | ||
BEAST_EXPECT(s.size()); | ||
b256Result[i] = outBuf.subspan(0, s.size()); | ||
std::copy(s.begin(), s.end(), b256Result[i].begin()); | ||
} | ||
} | ||
|
||
if (BEAST_EXPECT(b256Result[0].size() == b256Result[1].size())) | ||
{ | ||
if (!BEAST_EXPECT( | ||
memcmp( | ||
b256Result[0].data(), | ||
b256Result[1].data(), | ||
b256Result[0].size()) == 0)) | ||
{ | ||
printAsInt(b256Result[0], b256Result[1]); | ||
} | ||
} | ||
}; | ||
|
||
// test every token type with data where every byte is the same and the | ||
// bytes range from 0-255 | ||
for (int i = 0; i < numTokenTypeIndexes; ++i) | ||
{ | ||
auto const [tokType, tokSize] = tokenTypeAndSize(i); | ||
for (int d = 0; d < 255; ++d) | ||
{ | ||
memset(b256DataBuf.data(), d, tokSize); | ||
testIt(tokType, std::span(b256DataBuf.data(), tokSize)); | ||
} | ||
} | ||
|
||
// test with random data | ||
constexpr std::size_t iters = 1000000; | ||
for (int i = 0; i < iters; ++i) | ||
{ | ||
auto const [tokType, b256Data] = randomB256TestData( | ||
std::span(b256DataBuf.data(), b256DataBuf.size())); | ||
testIt(tokType, b256Data); | ||
} | ||
} | ||
|
||
void | ||
run() override | ||
{ | ||
testMultiprecision(); | ||
testFastMatchesRef(); | ||
} | ||
}; | ||
|
||
BEAST_DEFINE_TESTSUITE(base58, ripple_basics, ripple); | ||
|
||
} // namespace test | ||
} // namespace ripple | ||
#endif |