mirror of
https://github.com/vtnerd/monero-lws.git
synced 2025-01-19 00:54:34 +00:00
Add basic REST tests and fix a few bugs (#103)
This commit is contained in:
parent
de7268f9e6
commit
4c2205974e
7 changed files with 633 additions and 66 deletions
|
@ -544,7 +544,7 @@ namespace lws
|
|||
MONERO_CHECK((*tclient)->send(std::move(msg), std::chrono::seconds{10}));
|
||||
}
|
||||
|
||||
if ((req.use_dust && req.use_dust) || !req.dust_threshold)
|
||||
if ((req.use_dust && *req.use_dust) || !req.dust_threshold)
|
||||
req.dust_threshold = rpc::safe_uint64(0);
|
||||
|
||||
if (!req.mixin)
|
||||
|
|
|
@ -262,6 +262,7 @@ namespace lws
|
|||
wire::field("coinbase", is_coinbase),
|
||||
wire::field("mempool", false),
|
||||
wire::field("mixin", self.value().info.spend_meta.mixin_count),
|
||||
wire::field("recipient", self.value().info.recipient),
|
||||
wire::field("spent_outputs", std::cref(self.value().spends))
|
||||
);
|
||||
}
|
||||
|
@ -316,7 +317,7 @@ namespace lws
|
|||
WIRE_FIELD_COPY(per_byte_fee),
|
||||
WIRE_FIELD_COPY(fee_mask),
|
||||
WIRE_FIELD_COPY(amount),
|
||||
wire::field("outputs", wire::array(boost::adaptors::transform(self.outputs, expand)))
|
||||
wire::optional_field("outputs", wire::array(boost::adaptors::transform(self.outputs, expand)))
|
||||
);
|
||||
}
|
||||
|
||||
|
|
|
@ -110,7 +110,7 @@ namespace wire
|
|||
static constexpr bool optional_on_empty() noexcept
|
||||
{ return is_optional_on_empty<value_type>::value; }
|
||||
|
||||
static constexpr bool is_required() noexcept { return Required; }
|
||||
static constexpr bool is_required() noexcept { return Required && !optional_on_empty(); }
|
||||
static constexpr std::size_t count() noexcept { return 1; }
|
||||
static constexpr unsigned id() noexcept { return I; }
|
||||
|
||||
|
|
|
@ -34,7 +34,7 @@ add_subdirectory(db)
|
|||
add_subdirectory(rpc)
|
||||
add_subdirectory(wire)
|
||||
|
||||
add_executable(monero-lws-unit main.cpp scanner.test.cpp)
|
||||
add_executable(monero-lws-unit main.cpp rest.test.cpp scanner.test.cpp)
|
||||
target_link_libraries(monero-lws-unit
|
||||
monero::libraries
|
||||
monero-lws-daemon-common
|
||||
|
|
529
tests/unit/rest.test.cpp
Normal file
529
tests/unit/rest.test.cpp
Normal file
|
@ -0,0 +1,529 @@
|
|||
// Copyright (c) 2024, 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 "framework.test.h"
|
||||
|
||||
#include <optional>
|
||||
#include "db/data.h"
|
||||
#include "db/storage.test.h"
|
||||
#include "db/string.h"
|
||||
#include "error.h"
|
||||
#include "hex.h" // monero/epee/contrib/include
|
||||
#include "net/http_client.h"
|
||||
#include "rest_server.h"
|
||||
#include "scanner.test.h"
|
||||
|
||||
namespace
|
||||
{
|
||||
namespace enet = epee::net_utils;
|
||||
|
||||
constexpr const char fake_daemon[] = "inproc://fake_daemon";
|
||||
constexpr const char rest_server[] = "http://127.0.0.1:10000";
|
||||
constexpr const char admin_server[] = "http://127.0.0.1:10001";
|
||||
|
||||
struct join
|
||||
{
|
||||
boost::thread& thread;
|
||||
~join() { thread.join(); }
|
||||
};
|
||||
|
||||
struct rct_bytes
|
||||
{
|
||||
rct::key commitment;
|
||||
rct::key mask;
|
||||
rct::key amount;
|
||||
};
|
||||
|
||||
std::string invoke(enet::http::http_simple_client& client, const boost::string_ref uri, const boost::string_ref body)
|
||||
{
|
||||
const enet::http::http_response_info* info = nullptr;
|
||||
if (!client.invoke(uri, "POST", body, std::chrono::milliseconds{500}, std::addressof(info), {}))
|
||||
throw std::runtime_error{"HTTP invoke failed"};
|
||||
if (info->m_response_code != 200)
|
||||
throw std::runtime_error{"HTTP invoke not 200, instead " + std::to_string(info->m_response_code)};
|
||||
return std::string{info->m_body};
|
||||
}
|
||||
|
||||
epee::byte_slice get_fee_response()
|
||||
{
|
||||
return epee::byte_slice{
|
||||
std::string{
|
||||
"{\"jsonrpc\":2.0,\"id\":0,\"result\":{"
|
||||
"\"estimated_base_fee\":10000,\"fee_mask\":1000,\"size_scale\":256,\"hard_fork_version\":16"
|
||||
"}}"
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
rct_bytes get_rct_bytes(const crypto::secret_key& user_key, const crypto::public_key& tx_public, const rct::key& ringct_mask, const std::uint64_t amount, const std::uint32_t index)
|
||||
{
|
||||
rct_bytes out{};
|
||||
|
||||
crypto::key_derivation derived;
|
||||
if (!crypto::generate_key_derivation(tx_public, user_key, derived))
|
||||
MONERO_THROW(lws::error::crypto_failure, "generate_key_derivation failed");
|
||||
|
||||
crypto::secret_key scalar;
|
||||
rct::ecdhTuple encrypted{ringct_mask, rct::d2h(amount)};
|
||||
|
||||
crypto::derivation_to_scalar(derived, index, scalar);
|
||||
rct::ecdhEncode(encrypted, rct::sk2rct(scalar), false);
|
||||
|
||||
out.commitment = rct::commit(amount, ringct_mask);
|
||||
out.mask = encrypted.mask;
|
||||
out.amount = encrypted.amount;
|
||||
return out;
|
||||
}
|
||||
}
|
||||
|
||||
LWS_CASE("rest_server")
|
||||
{
|
||||
lws::db::account_address account{};
|
||||
crypto::secret_key view{};
|
||||
crypto::generate_keys(account.spend_public, view);
|
||||
crypto::generate_keys(account.view_public, view);
|
||||
const std::string address = lws::db::address_string(account);
|
||||
const std::string viewkey = epee::to_hex::string(epee::as_byte_span(unwrap(unwrap(view))));
|
||||
|
||||
SETUP("Database and login")
|
||||
{
|
||||
std::optional<lws::rest_server> server;
|
||||
lws::db::test::cleanup_db on_scope_exit{};
|
||||
lws::db::storage db = lws::db::test::get_fresh_db();
|
||||
auto context =
|
||||
lws::rpc::context::make(lws_test::rpc_rendevous, {}, {}, {}, std::chrono::minutes{0}, false);
|
||||
const auto rpc = MONERO_UNWRAP(context.connect());
|
||||
{
|
||||
const lws::rest_server::configuration config{
|
||||
{}, {}, 1, 20, {}, false, true, true
|
||||
};
|
||||
std::vector<std::string> addresses{rest_server};
|
||||
server.emplace(
|
||||
epee::to_span(addresses),
|
||||
std::vector<std::string>{admin_server},
|
||||
db.clone(),
|
||||
MONERO_UNWRAP(rpc.clone()),
|
||||
config
|
||||
);
|
||||
}
|
||||
|
||||
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;
|
||||
};
|
||||
|
||||
enet::http::http_simple_client client{};
|
||||
client.set_server("127.0.0.1", "10000", boost::none);
|
||||
EXPECT(client.connect(std::chrono::milliseconds{500}));
|
||||
|
||||
std::string message =
|
||||
"{\"address\":\"" + address + "\",\"view_key\":\"" + viewkey + "\",\"create_account\":true,\"generated_locally\":true}";
|
||||
std::string response = invoke(client, "/login", message);
|
||||
EXPECT(response == "{\"new_address\":true,\"generated_locally\":true}");
|
||||
|
||||
auto account = get_account();
|
||||
EXPECT(account.id == lws::db::account_id(1));
|
||||
|
||||
SECTION("Empty Account")
|
||||
{
|
||||
const std::string scan_height = std::to_string(std::uint64_t(account.scan_height));
|
||||
const std::string start_height = std::to_string(std::uint64_t(account.start_height));
|
||||
message = "{\"address\":\"" + address + "\",\"view_key\":\"" + viewkey + "\"}";
|
||||
response = invoke(client, "/get_address_info", message);
|
||||
EXPECT(response ==
|
||||
"{\"locked_funds\":\"0\","
|
||||
"\"total_received\":\"0\","
|
||||
"\"total_sent\":\"0\","
|
||||
"\"scanned_height\":" + scan_height + "," +
|
||||
"\"scanned_block_height\":" + scan_height + ","
|
||||
"\"start_height\":" + start_height + ","
|
||||
"\"transaction_height\":" + scan_height + ","
|
||||
"\"blockchain_height\":" + scan_height + "}"
|
||||
);
|
||||
|
||||
response = invoke(client, "/get_address_txs", message);
|
||||
EXPECT(response ==
|
||||
"{\"total_received\":\"0\","
|
||||
"\"scanned_height\":" + scan_height + "," +
|
||||
"\"scanned_block_height\":" + scan_height + ","
|
||||
"\"start_height\":" + start_height + ","
|
||||
"\"transaction_height\":" + scan_height + ","
|
||||
"\"blockchain_height\":" + scan_height + "}"
|
||||
);
|
||||
|
||||
std::vector<epee::byte_slice> messages;
|
||||
messages.emplace_back(get_fee_response());
|
||||
boost::thread server_thread(&lws_test::rpc_thread, context.zmq_context(), std::cref(messages));
|
||||
const join on_scope_exit{server_thread};
|
||||
|
||||
message = "{\"address\":\"" + address + "\",\"view_key\":\"" + viewkey + "\",\"amount\":\"0\"}";
|
||||
response = invoke(client, "/get_unspent_outs", message);
|
||||
EXPECT(response == "{\"per_byte_fee\":39,\"fee_mask\":1000,\"amount\":\"0\"}");
|
||||
}
|
||||
|
||||
SECTION("One Receive, Zero Spends")
|
||||
{
|
||||
const std::string scan_height = std::to_string(std::uint64_t(account.scan_height) + 5);
|
||||
const std::string start_height = std::to_string(std::uint64_t(account.start_height));
|
||||
message = "{\"address\":\"" + address + "\",\"view_key\":\"" + viewkey + "\"}";
|
||||
|
||||
const lws::db::transaction_link link{
|
||||
lws::db::block_id(4000), crypto::rand<crypto::hash>()
|
||||
};
|
||||
const crypto::public_key tx_public = []() {
|
||||
crypto::secret_key secret;
|
||||
crypto::public_key out;
|
||||
crypto::generate_keys(out, secret);
|
||||
return out;
|
||||
}();
|
||||
const crypto::hash tx_prefix = crypto::rand<crypto::hash>();
|
||||
const crypto::public_key pub = crypto::rand<crypto::public_key>();
|
||||
const rct::key ringct = crypto::rand<rct::key>();
|
||||
const auto extra =
|
||||
lws::db::extra(lws::db::extra::coinbase_output | lws::db::extra::ringct_output);
|
||||
const auto payment_id_ = crypto::rand<lws::db::output::payment_id_>();
|
||||
const crypto::key_image image = crypto::rand<crypto::key_image>();
|
||||
|
||||
lws::account real_account{account, {}, {}};
|
||||
real_account.add_out(
|
||||
lws::db::output{
|
||||
link,
|
||||
lws::db::output::spend_meta_{
|
||||
lws::db::output_id{500, 30},
|
||||
std::uint64_t(40000),
|
||||
std::uint32_t(16),
|
||||
std::uint32_t(2),
|
||||
tx_public
|
||||
},
|
||||
std::uint64_t(7000),
|
||||
std::uint64_t(4670),
|
||||
tx_prefix,
|
||||
pub,
|
||||
ringct,
|
||||
{0, 0, 0, 0, 0, 0, 0},
|
||||
lws::db::pack(extra, sizeof(crypto::hash)),
|
||||
payment_id_,
|
||||
std::uint64_t(100),
|
||||
lws::db::address_index{lws::db::major_index(2), lws::db::minor_index(66)}
|
||||
}
|
||||
);
|
||||
|
||||
{
|
||||
std::vector<crypto::hash> hashes{
|
||||
last_block.hash,
|
||||
crypto::rand<crypto::hash>(),
|
||||
crypto::rand<crypto::hash>(),
|
||||
crypto::rand<crypto::hash>(),
|
||||
crypto::rand<crypto::hash>(),
|
||||
crypto::rand<crypto::hash>()
|
||||
};
|
||||
|
||||
EXPECT(db.update(last_block.id, epee::to_span(hashes), {std::addressof(real_account), 1}, {}));
|
||||
}
|
||||
|
||||
response = invoke(client, "/get_address_info", message);
|
||||
EXPECT(response ==
|
||||
"{\"locked_funds\":\"0\","
|
||||
"\"total_received\":\"40000\","
|
||||
"\"total_sent\":\"0\","
|
||||
"\"scanned_height\":" + scan_height + "," +
|
||||
"\"scanned_block_height\":" + scan_height + ","
|
||||
"\"start_height\":" + start_height + ","
|
||||
"\"transaction_height\":" + scan_height + ","
|
||||
"\"blockchain_height\":" + scan_height + "}"
|
||||
);
|
||||
|
||||
response = invoke(client, "/get_address_txs", message);
|
||||
EXPECT(response ==
|
||||
"{\"total_received\":\"40000\","
|
||||
"\"scanned_height\":" + scan_height + "," +
|
||||
"\"scanned_block_height\":" + scan_height + ","
|
||||
"\"start_height\":" + start_height + ","
|
||||
"\"transaction_height\":" + scan_height + ","
|
||||
"\"blockchain_height\":" + scan_height + ","
|
||||
"\"transactions\":["
|
||||
"{\"id\":0,"
|
||||
"\"hash\":\"" + epee::to_hex::string(epee::as_byte_span(link.tx_hash)) + "\","
|
||||
"\"timestamp\":\"1970-01-01T01:56:40Z\","
|
||||
"\"total_received\":\"40000\","
|
||||
"\"total_sent\":\"0\","
|
||||
"\"fee\":\"100\","
|
||||
"\"unlock_time\":4670,"
|
||||
"\"height\":4000,"
|
||||
"\"payment_id\":\"" + epee::to_hex::string(epee::as_byte_span(payment_id_)) + "\","
|
||||
"\"coinbase\":true,"
|
||||
"\"mempool\":false,"
|
||||
"\"mixin\":16,"
|
||||
"\"recipient\":{\"maj_i\":2,\"min_i\":66}}"
|
||||
"]}"
|
||||
);
|
||||
|
||||
std::vector<epee::byte_slice> messages;
|
||||
messages.emplace_back(get_fee_response());
|
||||
boost::thread server_thread(&lws_test::rpc_thread, context.zmq_context(), std::cref(messages));
|
||||
const join on_scope_exit{server_thread};
|
||||
|
||||
const auto ringct_expanded = get_rct_bytes(view, tx_public, ringct, 40000, 2);
|
||||
message = "{\"address\":\"" + address + "\",\"view_key\":\"" + viewkey + "\",\"amount\":\"0\"}";
|
||||
response = invoke(client, "/get_unspent_outs", message);
|
||||
EXPECT(response ==
|
||||
"{\"per_byte_fee\":39,"
|
||||
"\"fee_mask\":1000,"
|
||||
"\"amount\":\"40000\","
|
||||
"\"outputs\":["
|
||||
"{\"amount\":\"40000\","
|
||||
"\"public_key\":\"" + epee::to_hex::string(epee::as_byte_span(pub)) + "\","
|
||||
"\"index\":2,"
|
||||
"\"global_index\":30,"
|
||||
"\"tx_id\":30,"
|
||||
"\"tx_hash\":\"" + epee::to_hex::string(epee::as_byte_span(link.tx_hash)) + "\","
|
||||
"\"tx_prefix_hash\":\"" + epee::to_hex::string(epee::as_byte_span(tx_prefix)) + "\","
|
||||
"\"tx_pub_key\":\"" + epee::to_hex::string(epee::as_byte_span(tx_public)) + "\","
|
||||
"\"timestamp\":\"1970-01-01T01:56:40Z\","
|
||||
"\"height\":4000,"
|
||||
"\"rct\":\"" + epee::to_hex::string(epee::as_byte_span(ringct_expanded)) + "\","
|
||||
"\"recipient\":{\"maj_i\":2,\"min_i\":66}}"
|
||||
"]}"
|
||||
);
|
||||
}
|
||||
|
||||
SECTION("One Receive, One Spend")
|
||||
{
|
||||
const std::string scan_height = std::to_string(std::uint64_t(account.scan_height) + 5);
|
||||
const std::string start_height = std::to_string(std::uint64_t(account.start_height));
|
||||
message = "{\"address\":\"" + address + "\",\"view_key\":\"" + viewkey + "\"}";
|
||||
|
||||
const lws::db::transaction_link link{
|
||||
lws::db::block_id(4000), crypto::rand<crypto::hash>()
|
||||
};
|
||||
const crypto::public_key tx_public = []() {
|
||||
crypto::secret_key secret;
|
||||
crypto::public_key out;
|
||||
crypto::generate_keys(out, secret);
|
||||
return out;
|
||||
}();
|
||||
const crypto::hash tx_prefix = crypto::rand<crypto::hash>();
|
||||
const crypto::public_key pub = crypto::rand<crypto::public_key>();
|
||||
const rct::key ringct = crypto::rand<rct::key>();
|
||||
const auto extra =
|
||||
lws::db::extra(lws::db::extra::coinbase_output | lws::db::extra::ringct_output);
|
||||
const auto payment_id_ = crypto::rand<lws::db::output::payment_id_>();
|
||||
const crypto::hash payment_id = crypto::rand<crypto::hash>();
|
||||
const crypto::key_image image = crypto::rand<crypto::key_image>();
|
||||
|
||||
lws::account real_account{account, {}, {}};
|
||||
real_account.add_out(
|
||||
lws::db::output{
|
||||
link,
|
||||
lws::db::output::spend_meta_{
|
||||
lws::db::output_id{500, 30},
|
||||
std::uint64_t(40000),
|
||||
std::uint32_t(16),
|
||||
std::uint32_t(2),
|
||||
tx_public
|
||||
},
|
||||
std::uint64_t(7000),
|
||||
std::uint64_t(4670),
|
||||
tx_prefix,
|
||||
pub,
|
||||
ringct,
|
||||
{0, 0, 0, 0, 0, 0, 0},
|
||||
lws::db::pack(extra, sizeof(crypto::hash)),
|
||||
payment_id_,
|
||||
std::uint64_t(100),
|
||||
lws::db::address_index{lws::db::major_index(2), lws::db::minor_index(66)}
|
||||
}
|
||||
);
|
||||
real_account.add_spend(
|
||||
lws::db::spend{
|
||||
link,
|
||||
image,
|
||||
lws::db::output_id{500, 30},
|
||||
std::uint64_t(66),
|
||||
std::uint64_t(1500),
|
||||
std::uint32_t(16),
|
||||
{0, 0, 0},
|
||||
32,
|
||||
payment_id,
|
||||
lws::db::address_index{lws::db::major_index(4), lws::db::minor_index(55)}
|
||||
}
|
||||
);
|
||||
|
||||
{
|
||||
std::vector<crypto::hash> hashes{
|
||||
last_block.hash,
|
||||
crypto::rand<crypto::hash>(),
|
||||
crypto::rand<crypto::hash>(),
|
||||
crypto::rand<crypto::hash>(),
|
||||
crypto::rand<crypto::hash>(),
|
||||
crypto::rand<crypto::hash>()
|
||||
};
|
||||
|
||||
EXPECT(db.update(last_block.id, epee::to_span(hashes), {std::addressof(real_account), 1}, {}));
|
||||
}
|
||||
|
||||
response = invoke(client, "/get_address_info", message);
|
||||
EXPECT(response ==
|
||||
"{\"locked_funds\":\"0\","
|
||||
"\"total_received\":\"40000\","
|
||||
"\"total_sent\":\"40000\","
|
||||
"\"scanned_height\":" + scan_height + "," +
|
||||
"\"scanned_block_height\":" + scan_height + ","
|
||||
"\"start_height\":" + start_height + ","
|
||||
"\"transaction_height\":" + scan_height + ","
|
||||
"\"blockchain_height\":" + scan_height + ","
|
||||
"\"spent_outputs\":[{"
|
||||
"\"amount\":\"40000\","
|
||||
"\"key_image\":\"" + epee::to_hex::string(epee::as_byte_span(image)) + "\","
|
||||
"\"tx_pub_key\":\"" + epee::to_hex::string(epee::as_byte_span(tx_public)) + "\","
|
||||
"\"out_index\":2,"
|
||||
"\"mixin\":16,"
|
||||
"\"sender\":{\"maj_i\":4,\"min_i\":55}"
|
||||
"}]}"
|
||||
);
|
||||
|
||||
response = invoke(client, "/get_address_txs", message);
|
||||
EXPECT(response ==
|
||||
"{\"total_received\":\"40000\","
|
||||
"\"scanned_height\":" + scan_height + "," +
|
||||
"\"scanned_block_height\":" + scan_height + ","
|
||||
"\"start_height\":" + start_height + ","
|
||||
"\"transaction_height\":" + scan_height + ","
|
||||
"\"blockchain_height\":" + scan_height + ","
|
||||
"\"transactions\":["
|
||||
"{\"id\":0,"
|
||||
"\"hash\":\"" + epee::to_hex::string(epee::as_byte_span(link.tx_hash)) + "\","
|
||||
"\"timestamp\":\"1970-01-01T01:56:40Z\","
|
||||
"\"total_received\":\"40000\","
|
||||
"\"total_sent\":\"40000\","
|
||||
"\"fee\":\"100\","
|
||||
"\"unlock_time\":4670,"
|
||||
"\"height\":4000,"
|
||||
"\"payment_id\":\"" + epee::to_hex::string(epee::as_byte_span(payment_id_)) + "\","
|
||||
"\"coinbase\":true,"
|
||||
"\"mempool\":false,"
|
||||
"\"mixin\":16,"
|
||||
"\"recipient\":{\"maj_i\":2,\"min_i\":66},"
|
||||
"\"spent_outputs\":[{"
|
||||
"\"amount\":\"40000\","
|
||||
"\"key_image\":\"" + epee::to_hex::string(epee::as_byte_span(image)) + "\","
|
||||
"\"tx_pub_key\":\"" + epee::to_hex::string(epee::as_byte_span(tx_public)) + "\","
|
||||
"\"out_index\":2,"
|
||||
"\"mixin\":16,"
|
||||
"\"sender\":{\"maj_i\":4,\"min_i\":55}"
|
||||
"}]}"
|
||||
"]}"
|
||||
);
|
||||
|
||||
std::vector<epee::byte_slice> messages;
|
||||
messages.emplace_back(get_fee_response());
|
||||
boost::thread server_thread(&lws_test::rpc_thread, context.zmq_context(), std::cref(messages));
|
||||
const join on_scope_exit{server_thread};
|
||||
|
||||
const auto ringct_expanded = get_rct_bytes(view, tx_public, ringct, 40000, 2);
|
||||
message = "{\"address\":\"" + address + "\",\"view_key\":\"" + viewkey + "\",\"amount\":\"0\"}";
|
||||
response = invoke(client, "/get_unspent_outs", message);
|
||||
EXPECT(response ==
|
||||
"{\"per_byte_fee\":39,"
|
||||
"\"fee_mask\":1000,"
|
||||
"\"amount\":\"40000\","
|
||||
"\"outputs\":["
|
||||
"{\"amount\":\"40000\","
|
||||
"\"public_key\":\"" + epee::to_hex::string(epee::as_byte_span(pub)) + "\","
|
||||
"\"index\":2,"
|
||||
"\"global_index\":30,"
|
||||
"\"tx_id\":30,"
|
||||
"\"tx_hash\":\"" + epee::to_hex::string(epee::as_byte_span(link.tx_hash)) + "\","
|
||||
"\"tx_prefix_hash\":\"" + epee::to_hex::string(epee::as_byte_span(tx_prefix)) + "\","
|
||||
"\"tx_pub_key\":\"" + epee::to_hex::string(epee::as_byte_span(tx_public)) + "\","
|
||||
"\"timestamp\":\"1970-01-01T01:56:40Z\","
|
||||
"\"height\":4000,"
|
||||
"\"spend_key_images\":[\"" + epee::to_hex::string(epee::as_byte_span(image)) + "\"],"
|
||||
"\"rct\":\"" + epee::to_hex::string(epee::as_byte_span(ringct_expanded)) + "\","
|
||||
"\"recipient\":{\"maj_i\":2,\"min_i\":66}}"
|
||||
"]}"
|
||||
);
|
||||
}
|
||||
|
||||
SECTION("provision_subaddrs")
|
||||
{
|
||||
const std::string scan_height = std::to_string(std::uint64_t(account.scan_height) + 5);
|
||||
const std::string start_height = std::to_string(std::uint64_t(account.start_height));
|
||||
message = "{\"address\":\"" + address + "\",\"view_key\":\"" + viewkey + "\",\"maj_i\":0,\"min_i\":0,\"n_maj\":2,\"n_min\":5}";
|
||||
|
||||
response = invoke(client, "/provision_subaddrs", message);
|
||||
EXPECT(response ==
|
||||
"{\"new_subaddrs\":["
|
||||
"{\"key\":0,\"value\":[[0,4]]},"
|
||||
"{\"key\":1,\"value\":[[0,4]]}"
|
||||
"],\"all_subaddrs\":["
|
||||
"{\"key\":0,\"value\":[[0,4]]},"
|
||||
"{\"key\":1,\"value\":[[0,4]]}]}"
|
||||
);
|
||||
|
||||
message = "{\"address\":\"" + address + "\",\"view_key\":\"" + viewkey + "\",\"maj_i\":2,\"min_i\":5,\"n_maj\":2,\"n_min\":5}";
|
||||
response = invoke(client, "/provision_subaddrs", message);
|
||||
EXPECT(response ==
|
||||
"{\"new_subaddrs\":["
|
||||
"{\"key\":2,\"value\":[[5,9]]},"
|
||||
"{\"key\":3,\"value\":[[5,9]]}"
|
||||
"],\"all_subaddrs\":["
|
||||
"{\"key\":0,\"value\":[[0,4]]},"
|
||||
"{\"key\":1,\"value\":[[0,4]]},"
|
||||
"{\"key\":2,\"value\":[[5,9]]},"
|
||||
"{\"key\":3,\"value\":[[5,9]]}]}"
|
||||
);
|
||||
}
|
||||
|
||||
SECTION("upsert_subaddrs")
|
||||
{
|
||||
const std::string scan_height = std::to_string(std::uint64_t(account.scan_height) + 5);
|
||||
const std::string start_height = std::to_string(std::uint64_t(account.start_height));
|
||||
message = "{\"address\":\"" + address + "\",\"view_key\":\"" + viewkey + "\",\"subaddrs\":[{\"key\":0,\"value\":[[1,10]]}]}";
|
||||
|
||||
response = invoke(client, "/upsert_subaddrs", message);
|
||||
EXPECT(response ==
|
||||
"{\"new_subaddrs\":["
|
||||
"{\"key\":0,\"value\":[[1,10]]}"
|
||||
"],\"all_subaddrs\":["
|
||||
"{\"key\":0,\"value\":[[1,10]]}]}"
|
||||
);
|
||||
|
||||
message = "{\"address\":\"" + address + "\",\"view_key\":\"" + viewkey + "\",\"subaddrs\":[{\"key\":0,\"value\":[[11,20]]}]}";
|
||||
response = invoke(client, "/upsert_subaddrs", message);
|
||||
EXPECT(response ==
|
||||
"{\"new_subaddrs\":["
|
||||
"{\"key\":0,\"value\":[[11,20]]}"
|
||||
"],\"all_subaddrs\":["
|
||||
"{\"key\":0,\"value\":[[1,20]]}]}"
|
||||
);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
@ -25,10 +25,10 @@
|
|||
// 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 "scanner.test.h"
|
||||
#include "framework.test.h"
|
||||
|
||||
#include <boost/thread.hpp>
|
||||
#include <iostream>
|
||||
|
||||
#include "cryptonote_basic/account.h" // monero/src
|
||||
#include "cryptonote_basic/cryptonote_format_utils.h" // monero/src
|
||||
|
@ -47,7 +47,6 @@
|
|||
|
||||
namespace
|
||||
{
|
||||
constexpr const char rendevous[] = "inproc://fake_daemon";
|
||||
constexpr const std::chrono::seconds message_timeout{3};
|
||||
|
||||
template<typename T>
|
||||
|
@ -82,60 +81,6 @@ namespace
|
|||
return cryptonote::rpc::FullMessage::getResponse(message, id);
|
||||
}
|
||||
|
||||
void rpc_thread(void* ctx, const std::vector<epee::byte_slice>& reply)
|
||||
{
|
||||
struct stop_
|
||||
{
|
||||
~stop_() noexcept { lws::scanner::stop(); };
|
||||
} stop{};
|
||||
|
||||
try
|
||||
{
|
||||
net::zmq::socket server{};
|
||||
server.reset(zmq_socket(ctx, ZMQ_REP));
|
||||
if (!server || zmq_bind(server.get(), rendevous))
|
||||
{
|
||||
std::cout << "Failed to create ZMQ server" << std::endl;
|
||||
return;
|
||||
}
|
||||
|
||||
for (const epee::byte_slice& message : reply)
|
||||
{
|
||||
const auto start = std::chrono::steady_clock::now();
|
||||
for (;;)
|
||||
{
|
||||
const auto request = net::zmq::receive(server.get(), ZMQ_DONTWAIT);
|
||||
if (request)
|
||||
break;
|
||||
|
||||
if (request != net::zmq::make_error_code(EAGAIN))
|
||||
{
|
||||
std::cout << "Failed to retrieve message in fake ZMQ server: " << request.error().message() << std::endl;;
|
||||
return;
|
||||
}
|
||||
|
||||
if (message_timeout <= std::chrono::steady_clock::now() - start)
|
||||
{
|
||||
std::cout << "Timeout in dummy RPC server" << std::endl;
|
||||
return;
|
||||
}
|
||||
boost::this_thread::sleep_for(boost::chrono::milliseconds{10});
|
||||
} // until error or received message
|
||||
|
||||
const auto sent = net::zmq::send(message.clone(), server.get());
|
||||
if (!sent)
|
||||
{
|
||||
std::cout << "Failed to send dummy RPC message: " << sent.error().message() << std::endl;
|
||||
return;
|
||||
}
|
||||
} // foreach message
|
||||
}
|
||||
catch (const std::exception& e)
|
||||
{
|
||||
std::cout << "Unexpected exception in dummy RPC server: " << e.what() << std::endl;
|
||||
}
|
||||
}
|
||||
|
||||
struct join
|
||||
{
|
||||
boost::thread& thread;
|
||||
|
@ -278,6 +223,63 @@ namespace
|
|||
}
|
||||
return out;
|
||||
}
|
||||
} // anonymous
|
||||
|
||||
namespace lws_test
|
||||
{
|
||||
void rpc_thread(void* ctx, const std::vector<epee::byte_slice>& reply)
|
||||
{
|
||||
struct stop_
|
||||
{
|
||||
~stop_() noexcept { lws::scanner::stop(); };
|
||||
} stop{};
|
||||
|
||||
try
|
||||
{
|
||||
net::zmq::socket server{};
|
||||
server.reset(zmq_socket(ctx, ZMQ_REP));
|
||||
if (!server || zmq_bind(server.get(), lws_test::rpc_rendevous))
|
||||
{
|
||||
std::cout << "Failed to create ZMQ server" << std::endl;
|
||||
return;
|
||||
}
|
||||
|
||||
for (const epee::byte_slice& message : reply)
|
||||
{
|
||||
const auto start = std::chrono::steady_clock::now();
|
||||
for (;;)
|
||||
{
|
||||
const auto request = net::zmq::receive(server.get(), ZMQ_DONTWAIT);
|
||||
if (request)
|
||||
break;
|
||||
|
||||
if (request != net::zmq::make_error_code(EAGAIN))
|
||||
{
|
||||
std::cout << "Failed to retrieve message in fake ZMQ server: " << request.error().message() << std::endl;;
|
||||
return;
|
||||
}
|
||||
|
||||
if (message_timeout <= std::chrono::steady_clock::now() - start)
|
||||
{
|
||||
std::cout << "Timeout in dummy RPC server" << std::endl;
|
||||
return;
|
||||
}
|
||||
boost::this_thread::sleep_for(boost::chrono::milliseconds{10});
|
||||
} // until error or received message
|
||||
|
||||
const auto sent = net::zmq::send(message.clone(), server.get());
|
||||
if (!sent)
|
||||
{
|
||||
std::cout << "Failed to send dummy RPC message: " << sent.error().message() << std::endl;
|
||||
return;
|
||||
}
|
||||
} // foreach message
|
||||
}
|
||||
catch (const std::exception& e)
|
||||
{
|
||||
std::cout << "Unexpected exception in dummy RPC server: " << e.what() << std::endl;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
LWS_CASE("lws::scanner::sync and lws::scanner::run")
|
||||
|
@ -321,7 +323,7 @@ LWS_CASE("lws::scanner::sync and lws::scanner::run")
|
|||
{
|
||||
lws::scanner::reset();
|
||||
auto rpc =
|
||||
lws::rpc::context::make(rendevous, {}, {}, {}, std::chrono::minutes{0}, false);
|
||||
lws::rpc::context::make(lws_test::rpc_rendevous, {}, {}, {}, std::chrono::minutes{0}, false);
|
||||
|
||||
|
||||
lws::db::test::cleanup_db on_scope_exit{};
|
||||
|
@ -343,7 +345,7 @@ LWS_CASE("lws::scanner::sync and lws::scanner::run")
|
|||
std::vector<epee::byte_slice> messages{};
|
||||
messages.push_back(to_json_rpc(1));
|
||||
|
||||
boost::thread server_thread(&rpc_thread, rpc.zmq_context(), std::cref(messages));
|
||||
boost::thread server_thread(&lws_test::rpc_thread, rpc.zmq_context(), std::cref(messages));
|
||||
const join on_scope_exit{server_thread};
|
||||
EXPECT(!lws::scanner::sync(db.clone(), MONERO_UNWRAP(rpc.connect())));
|
||||
lws_test::test_chain(lest_env, MONERO_UNWRAP(db.start_read()), last_block.id, hashes);
|
||||
|
@ -375,7 +377,7 @@ LWS_CASE("lws::scanner::sync and lws::scanner::run")
|
|||
|
||||
lws_test::test_chain(lest_env, MONERO_UNWRAP(db.start_read()), last_block.id, {hashes.data(), 1});
|
||||
{
|
||||
boost::thread server_thread(&rpc_thread, rpc.zmq_context(), std::cref(messages));
|
||||
boost::thread server_thread(&lws_test::rpc_thread, rpc.zmq_context(), std::cref(messages));
|
||||
const join on_scope_exit{server_thread};
|
||||
EXPECT(lws::scanner::sync(db.clone(), MONERO_UNWRAP(rpc.connect())));
|
||||
lws_test::test_chain(lest_env, MONERO_UNWRAP(db.start_read()), last_block.id, epee::to_span(hashes));
|
||||
|
@ -398,7 +400,7 @@ LWS_CASE("lws::scanner::sync and lws::scanner::run")
|
|||
message.hashes.resize(1);
|
||||
messages.push_back(daemon_response(message));
|
||||
|
||||
boost::thread server_thread(&rpc_thread, rpc.zmq_context(), std::cref(messages));
|
||||
boost::thread server_thread(&lws_test::rpc_thread, rpc.zmq_context(), std::cref(messages));
|
||||
const join on_scope_exit{server_thread};
|
||||
EXPECT(lws::scanner::sync(db.clone(), MONERO_UNWRAP(rpc.connect())));
|
||||
lws_test::test_chain(lest_env, MONERO_UNWRAP(db.start_read()), last_block.id, epee::to_span(hashes));
|
||||
|
@ -505,7 +507,7 @@ LWS_CASE("lws::scanner::sync and lws::scanner::run")
|
|||
messages.push_back(daemon_response(hmessage));
|
||||
|
||||
{
|
||||
boost::thread server_thread(&rpc_thread, rpc.zmq_context(), std::cref(messages));
|
||||
boost::thread server_thread(&lws_test::rpc_thread, rpc.zmq_context(), std::cref(messages));
|
||||
const join on_scope_exit{server_thread};
|
||||
EXPECT(lws::scanner::sync(db.clone(), MONERO_UNWRAP(rpc.connect())));
|
||||
lws_test::test_chain(lest_env, MONERO_UNWRAP(db.start_read()), last_block.id, epee::to_span(hashes));
|
||||
|
@ -524,7 +526,7 @@ LWS_CASE("lws::scanner::sync and lws::scanner::run")
|
|||
bmessage.output_indices.resize(1);
|
||||
messages.push_back(daemon_response(bmessage));
|
||||
{
|
||||
boost::thread server_thread(&rpc_thread, rpc.zmq_context(), std::cref(messages));
|
||||
boost::thread server_thread(&lws_test::rpc_thread, rpc.zmq_context(), std::cref(messages));
|
||||
const join on_scope_exit{server_thread};
|
||||
lws::scanner::run(db.clone(), std::move(rpc), 1, epee::net_utils::ssl_verification_t::none, true);
|
||||
}
|
||||
|
|
35
tests/unit/scanner.test.h
Normal file
35
tests/unit/scanner.test.h
Normal file
|
@ -0,0 +1,35 @@
|
|||
// Copyright (c) 2024, 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 <vector>
|
||||
#include "byte_slice.h" // monero/contrib/epee/include
|
||||
|
||||
namespace lws_test
|
||||
{
|
||||
constexpr const char rpc_rendevous[] = "inproc://fake_daemon";
|
||||
void rpc_thread(void* ctx, const std::vector<epee::byte_slice>& reply);
|
||||
}
|
Loading…
Reference in a new issue