monero-lws/tests/unit/db/chain.test.cpp
2024-06-06 17:01:01 -04:00

181 lines
6.7 KiB
C++

// Copyright (c) 2023, The Monero Project
// All rights reserved.
//
// Redistribution and use in source and binary forms, with or without modification, are
// permitted provided that the following conditions are met:
//
// 1. Redistributions of source code must retain the above copyright notice, this list of
// conditions and the following disclaimer.
//
// 2. Redistributions in binary form must reproduce the above copyright notice, this list
// of conditions and the following disclaimer in the documentation and/or other
// materials provided with the distribution.
//
// 3. Neither the name of the copyright holder nor the names of its contributors may be
// used to endorse or promote products derived from this software without specific
// prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY
// EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
// MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
// THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
// PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
// STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
// THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
#include "chain.test.h"
#include <cstdint>
#include "checkpoints/checkpoints.h" // monero/src
#include "db/storage.test.h"
#include "error.h"
namespace lws_test
{
void test_chain(lest::env& lest_env, lws::db::storage_reader reader, lws::db::block_id id, epee::span<const crypto::hash> snapshot)
{
EXPECT(1 <= snapshot.size());
std::uint64_t d = std::uint64_t(id);
for (const auto& hash : snapshot)
{
SETUP("Testing Block #: " + std::to_string(d))
{
EXPECT(MONERO_UNWRAP(reader.get_block_hash(lws::db::block_id(d))) == hash);
++d;
}
}
const lws::db::block_info last_block =
MONERO_UNWRAP(reader.get_last_block());
EXPECT(last_block.id == lws::db::block_id(d - 1));
EXPECT(last_block.hash == snapshot[snapshot.size() - 1]);
}
}
LWS_CASE("db::storage::sync_chain")
{
lws::db::account_address account{};
crypto::secret_key view{};
crypto::generate_keys(account.spend_public, view);
crypto::generate_keys(account.view_public, view);
SETUP("Appended Chain")
{
lws::db::test::cleanup_db on_scope_exit{};
lws::db::storage db = lws::db::test::get_fresh_db();
const lws::db::block_info last_block =
MONERO_UNWRAP(MONERO_UNWRAP(db.start_read()).get_last_block());
const auto get_account = [&db, &account] () -> lws::db::account
{
return MONERO_UNWRAP(MONERO_UNWRAP(db.start_read()).get_account(account)).second;
};
const auto get_height = [&db] (const lws::db::block_id height) -> std::vector<lws::db::account_id>
{
return MONERO_UNWRAP(MONERO_UNWRAP(db.start_read()).accounts_at_height(height));
};
const auto scan_height = lws::db::block_id(lmdb::to_native(last_block.id) + 4);
const crypto::hash chain[5] = {
last_block.hash,
crypto::rand<crypto::hash>(),
crypto::rand<crypto::hash>(),
crypto::rand<crypto::hash>(),
crypto::rand<crypto::hash>()
};
EXPECT(db.add_account(account, view));
EXPECT(db.sync_chain(lws::db::block_id(0), chain) == lws::error::bad_blockchain);
EXPECT(db.sync_chain(last_block.id, {chain + 1, 4}) == lws::error::bad_blockchain);
EXPECT(db.sync_chain(last_block.id, chain));
{
const lws::account accounts[1] = {lws::account{get_account(), {}, {}}};
EXPECT(accounts[0].scan_height() == last_block.id);
const auto updated = db.update(last_block.id, chain, accounts, nullptr);
EXPECT(updated);
EXPECT(updated->spend_pubs.empty());
EXPECT(updated->confirm_pubs.empty());
EXPECT(updated->accounts_updated == 1);
EXPECT(get_account().scan_height == scan_height);
const auto height = get_height(scan_height);
EXPECT(height.size() == 1);
EXPECT(height[0] == lws::db::account_id(1));
}
SECTION("Verify Append")
{
lws_test::test_chain(lest_env, MONERO_UNWRAP(db.start_read()), last_block.id, chain);
EXPECT(get_height(lws::db::block_id(0)).empty());
}
SECTION("Fork Chain")
{
const auto fork_height = lws::db::block_id(lmdb::to_native(last_block.id) + 1);
const crypto::hash fchain[5] = {
chain[0],
chain[1],
crypto::rand<crypto::hash>(),
crypto::rand<crypto::hash>(),
crypto::rand<crypto::hash>()
};
EXPECT(db.sync_chain(last_block.id, fchain));
lws_test::test_chain(lest_env, MONERO_UNWRAP(db.start_read()), last_block.id, fchain);
EXPECT(get_account().scan_height == fork_height);
const auto height = get_height(fork_height);
EXPECT(height.size() == 1);
EXPECT(height[0] == lws::db::account_id(1));
}
SECTION("Fork past checkpoint")
{
const auto& checkpoints = lws::db::storage::get_checkpoints();
const auto& points = checkpoints.get_points();
EXPECT(!points.empty());
auto point = points.begin();
const crypto::hash fchain[3] = {
point->second,
crypto::rand<crypto::hash>(),
crypto::rand<crypto::hash>()
};
const auto sync_result = db.sync_chain(lws::db::block_id(point->first), fchain);
EXPECT(sync_result == lws::error::bad_blockchain);
EXPECT(get_account().scan_height == scan_height);
const auto height = get_height(scan_height);
EXPECT(height.size() == 1);
EXPECT(height[0] == lws::db::account_id(1));
}
SECTION("Old Blocks dont rollback height")
{
lws::account accounts[1] {lws::account{get_account(), {}, {}}};
const auto old_block = lws::db::block_id(lmdb::to_native(last_block.id) - 5);
const auto new_block = lws::db::block_id(lmdb::to_native(last_block.id) + 4);
EXPECT(accounts[0].scan_height() == new_block);
const auto updated = db.update(old_block, chain, accounts, nullptr);
EXPECT(updated);
EXPECT(updated->spend_pubs.empty());
EXPECT(updated->confirm_pubs.empty());
EXPECT(updated->accounts_updated == 1);
EXPECT(get_account().scan_height == new_block);
const auto height = get_height(scan_height);
EXPECT(height.size() == 1);
EXPECT(height[0] == lws::db::account_id(1));
accounts[0].updated(old_block);
EXPECT(accounts[0].scan_height() == scan_height);
}
}
}