mirror of
https://github.com/vtnerd/monero-lws.git
synced 2025-01-19 09:04:32 +00:00
181 lines
6.7 KiB
C++
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);
|
|
}
|
|
}
|
|
}
|
|
|