// 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_.long_)) + "\"," "\"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_.long_)) + "\"," "\"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]]}]}" ); } } }