Files
rippled/src/test/nodestore/TestBase.h
2020-10-14 11:17:44 -07:00

232 lines
6.3 KiB
C++

//------------------------------------------------------------------------------
/*
This file is part of rippled: https://github.com/ripple/rippled
Copyright (c) 2012, 2013 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 RIPPLE_NODESTORE_BASE_H_INCLUDED
#define RIPPLE_NODESTORE_BASE_H_INCLUDED
#include <ripple/basics/StringUtilities.h>
#include <ripple/basics/random.h>
#include <ripple/beast/unit_test.h>
#include <ripple/beast/utility/rngfill.h>
#include <ripple/beast/xor_shift_engine.h>
#include <ripple/nodestore/Backend.h>
#include <ripple/nodestore/Database.h>
#include <ripple/nodestore/Types.h>
#include <boost/algorithm/string.hpp>
#include <iomanip>
namespace ripple {
namespace NodeStore {
/** Binary function that satisfies the strict-weak-ordering requirement.
This compares the hashes of both objects and returns true if
the first hash is considered to go before the second.
@see std::sort
*/
struct LessThan
{
bool
operator()(
std::shared_ptr<NodeObject> const& lhs,
std::shared_ptr<NodeObject> const& rhs) const noexcept
{
return lhs->getHash() < rhs->getHash();
}
};
/** Returns `true` if objects are identical. */
inline bool
isSame(
std::shared_ptr<NodeObject> const& lhs,
std::shared_ptr<NodeObject> const& rhs)
{
return (lhs->getType() == rhs->getType()) &&
(lhs->getHash() == rhs->getHash()) &&
(lhs->getData() == rhs->getData());
}
// Some common code for the unit tests
//
class TestBase : public beast::unit_test::suite
{
public:
// Tunable parameters
//
static std::size_t const minPayloadBytes = 1;
static std::size_t const maxPayloadBytes = 2000;
static int const numObjectsToTest = 2000;
public:
// Create a predictable batch of objects
static Batch
createPredictableBatch(int numObjects, std::uint64_t seed)
{
Batch batch;
batch.reserve(numObjects);
beast::xor_shift_engine rng(seed);
for (int i = 0; i < numObjects; ++i)
{
NodeObjectType const type = [&] {
switch (rand_int(rng, 3))
{
case 0:
return hotLEDGER;
case 1:
return hotACCOUNT_NODE;
case 2:
return hotTRANSACTION_NODE;
case 3:
return hotUNKNOWN;
}
// will never happen, but make static analysys tool happy.
return hotUNKNOWN;
}();
uint256 hash;
beast::rngfill(hash.begin(), hash.size(), rng);
Blob blob(rand_int(rng, minPayloadBytes, maxPayloadBytes));
beast::rngfill(blob.data(), blob.size(), rng);
batch.push_back(
NodeObject::createObject(type, std::move(blob), hash));
}
return batch;
}
// Compare two batches for equality
static bool
areBatchesEqual(Batch const& lhs, Batch const& rhs)
{
bool result = true;
if (lhs.size() == rhs.size())
{
for (int i = 0; i < lhs.size(); ++i)
{
if (!isSame(lhs[i], rhs[i]))
{
result = false;
break;
}
}
}
else
{
result = false;
}
return result;
}
// Store a batch in a backend
void
storeBatch(Backend& backend, Batch const& batch)
{
for (int i = 0; i < batch.size(); ++i)
{
backend.store(batch[i]);
}
}
// Get a copy of a batch in a backend
void
fetchCopyOfBatch(Backend& backend, Batch* pCopy, Batch const& batch)
{
pCopy->clear();
pCopy->reserve(batch.size());
for (int i = 0; i < batch.size(); ++i)
{
std::shared_ptr<NodeObject> object;
Status const status =
backend.fetch(batch[i]->getHash().cbegin(), &object);
BEAST_EXPECT(status == ok);
if (status == ok)
{
BEAST_EXPECT(object != nullptr);
pCopy->push_back(object);
}
}
}
void
fetchMissing(Backend& backend, Batch const& batch)
{
for (int i = 0; i < batch.size(); ++i)
{
std::shared_ptr<NodeObject> object;
Status const status =
backend.fetch(batch[i]->getHash().cbegin(), &object);
BEAST_EXPECT(status == notFound);
}
}
// Store all objects in a batch
static void
storeBatch(Database& db, Batch const& batch)
{
for (int i = 0; i < batch.size(); ++i)
{
std::shared_ptr<NodeObject> const object(batch[i]);
Blob data(object->getData());
db.store(
object->getType(),
std::move(data),
object->getHash(),
db.earliestLedgerSeq());
}
}
// Fetch all the hashes in one batch, into another batch.
static void
fetchCopyOfBatch(Database& db, Batch* pCopy, Batch const& batch)
{
pCopy->clear();
pCopy->reserve(batch.size());
for (int i = 0; i < batch.size(); ++i)
{
std::shared_ptr<NodeObject> object =
db.fetchNodeObject(batch[i]->getHash(), 0);
if (object != nullptr)
pCopy->push_back(object);
}
}
};
} // namespace NodeStore
} // namespace ripple
#endif