diff --git a/src/rest_server.cpp b/src/rest_server.cpp index 25092a3..c61717c 100644 --- a/src/rest_server.cpp +++ b/src/rest_server.cpp @@ -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) diff --git a/src/rpc/light_wallet.cpp b/src/rpc/light_wallet.cpp index 5994fb1..9f829b8 100644 --- a/src/rpc/light_wallet.cpp +++ b/src/rpc/light_wallet.cpp @@ -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))) ); } diff --git a/src/wire/field.h b/src/wire/field.h index eda5c75..57dc3b7 100644 --- a/src/wire/field.h +++ b/src/wire/field.h @@ -110,7 +110,7 @@ namespace wire static constexpr bool optional_on_empty() noexcept { return is_optional_on_empty::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; } diff --git a/tests/unit/CMakeLists.txt b/tests/unit/CMakeLists.txt index e1463d0..ea99115 100644 --- a/tests/unit/CMakeLists.txt +++ b/tests/unit/CMakeLists.txt @@ -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 diff --git a/tests/unit/rest.test.cpp b/tests/unit/rest.test.cpp new file mode 100644 index 0000000..8e421f4 --- /dev/null +++ b/tests/unit/rest.test.cpp @@ -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 +#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 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 addresses{rest_server}; + server.emplace( + epee::to_span(addresses), + std::vector{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 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() + }; + 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(); + const crypto::public_key pub = crypto::rand(); + const rct::key ringct = crypto::rand(); + const auto extra = + lws::db::extra(lws::db::extra::coinbase_output | lws::db::extra::ringct_output); + const auto payment_id_ = crypto::rand(); + const crypto::key_image image = crypto::rand(); + + 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 hashes{ + last_block.hash, + crypto::rand(), + crypto::rand(), + crypto::rand(), + crypto::rand(), + crypto::rand() + }; + + 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 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() + }; + 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(); + const crypto::public_key pub = crypto::rand(); + const rct::key ringct = crypto::rand(); + const auto extra = + lws::db::extra(lws::db::extra::coinbase_output | lws::db::extra::ringct_output); + const auto payment_id_ = crypto::rand(); + const crypto::hash payment_id = crypto::rand(); + const crypto::key_image image = crypto::rand(); + + 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 hashes{ + last_block.hash, + crypto::rand(), + crypto::rand(), + crypto::rand(), + crypto::rand(), + crypto::rand() + }; + + 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 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]]}]}" + ); + } + } +} + diff --git a/tests/unit/scanner.test.cpp b/tests/unit/scanner.test.cpp index 52ac9c0..e06c9d7 100644 --- a/tests/unit/scanner.test.cpp +++ b/tests/unit/scanner.test.cpp @@ -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 -#include #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 @@ -82,60 +81,6 @@ namespace return cryptonote::rpc::FullMessage::getResponse(message, id); } - void rpc_thread(void* ctx, const std::vector& 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& 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 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); } diff --git a/tests/unit/scanner.test.h b/tests/unit/scanner.test.h new file mode 100644 index 0000000..e7611ec --- /dev/null +++ b/tests/unit/scanner.test.h @@ -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 +#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& reply); +}