oxen-core/src/daemon/rpc_command_executor.h
Jason Rhinelander 6d7428b0e1 Add invoke_simple for invoking simple RPC calls
Simple here means no arguments and no return aside from the "status":
"OK" field.  For commands like stop_mining, stop_daemon, save_bc.
2021-11-01 16:58:59 -04:00

254 lines
9.2 KiB
C++

// Copyright (c) 2018-2020, The Loki Project
// Copyright (c) 2014-2019, 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.
//
// Parts of this file are originally copyright (c) 2012-2013 The Cryptonote developers
#pragma once
#include <exception>
#include <optional>
#include "common/common_fwd.h"
#include "common/scoped_message_writer.h"
#include "rpc/core_rpc_server_commands_defs.h"
#include "rpc/http_client.h"
#include "cryptonote_basic/cryptonote_basic.h"
#include "rpc/core_rpc_server.h"
#undef OXEN_DEFAULT_LOG_CATEGORY
#define OXEN_DEFAULT_LOG_CATEGORY "daemon"
namespace daemonize {
class rpc_command_executor final {
private:
std::variant<cryptonote::rpc::http_client, oxenmq::ConnectionID> m_rpc;
oxenmq::OxenMQ* m_omq = nullptr;
public:
/// Executor for HTTP remote connection RPC
rpc_command_executor(
std::string http_url,
const std::optional<tools::login>& user
);
/// Executor for OMQ RPC, either local (inproc) or remote.
rpc_command_executor(oxenmq::OxenMQ& omq, oxenmq::ConnectionID conn);
/// FIXME: remove this!
///
/// Runs some RPC command either via json_rpc or a direct core rpc call.
///
/// @param req the request object (rvalue reference)
/// @param error print this (and, on exception, the exception message) on failure. If empty then
/// nothing is printed on failure.
/// @param check_status_ok whether we require res.status == STATUS_OK to consider the request
/// successful
template <typename RPC, std::enable_if_t<std::is_base_of_v<cryptonote::rpc::RPC_COMMAND, RPC> && cryptonote::rpc::FIXME_has_nested_response_v<RPC>, int> = 0>
bool invoke(typename RPC::request&& req, typename RPC::response& res, const std::string& error, bool check_status_ok = true)
{
try {
if (auto* rpc_client = std::get_if<cryptonote::rpc::http_client>(&m_rpc)) {
res = rpc_client->json_rpc<RPC>(RPC::names()[0], req);
} else {
throw std::runtime_error{"fixme"};
}
if (!check_status_ok || res.status == cryptonote::rpc::STATUS_OK)
return true;
} catch (const std::exception& e) {
if (!error.empty())
tools::fail_msg_writer() << error << ": " << e.what();
return false;
} catch (...) {}
if (!error.empty())
tools::fail_msg_writer() << error;
return false;
}
/// Runs some RPC command either via json_rpc or an internal rpc call. Returns nlohmann::json
/// results on success, throws on failure.
///
/// Note that for a json_rpc request this is the "result" value inside the json_rpc wrapper, not
/// the wrapper itself.
///
/// This is the low-level implementing method for `invoke<SOMERPC>(...)`.
///
/// @param method the method name, typically `SOMERPC::names()[0]`
/// @param public_method true if this is a public rpc request; this is used, in particular, to
/// decide whether "rpc." or "admin." should be prefixed if this goes through OMQ RPC.
/// @param params the "params" field for the request. Can be nullopt to pass no "params".
/// @param check_status_ok whether we require the result to have a "status" key set to STATUS_OK
/// to consider the request successful. Note that this defaults to *false* if this is called
/// directly, unlike the RPC-type-templated version, below.
nlohmann::json invoke(
std::string_view method,
bool public_method,
std::optional<nlohmann::json> params,
bool check_status_ok = false);
/// Runs some RPC command either via json_rpc or an internal rpc call. Returns nlohmann::json
/// results on success, throws on failure.
///
/// @tparam RPC the rpc type class
/// @param params the "params" value to pass to json_rpc, or std::nullopt to omit it
/// @param check_status_ok whether we require the result to have a "status" key set to STATUS_OK
/// to consider the request successful
template <typename RPC, std::enable_if_t<std::is_base_of_v<cryptonote::rpc::RPC_COMMAND, RPC> && !cryptonote::rpc::FIXME_has_nested_response_v<RPC>, int> = 0>
nlohmann::json invoke(std::optional<nlohmann::json> params = std::nullopt, bool check_status_ok = true)
{
return invoke(RPC::names()[0], std::is_base_of_v<cryptonote::rpc::PUBLIC, RPC>, std::move(params), check_status_ok);
}
// Invokes a simple RPC method that doesn't take any arguments and for which we don't care about
// the return value beyond the "status": "OK" field. Returns true (and prints a success message)
// on success, false (with a failure message printed) on failure.
template <typename RPC, std::enable_if_t<std::is_base_of_v<cryptonote::rpc::RPC_COMMAND, RPC> && !cryptonote::rpc::FIXME_has_nested_response_v<RPC>, int> = 0>
bool invoke_simple(std::string_view error_prefix, std::string_view success_msg) {
if (!try_running([this] { return invoke<RPC>(); }, error_prefix))
return false;
tools::success_msg_writer() << success_msg;
return true;
}
bool print_checkpoints(uint64_t start_height, uint64_t end_height, bool print_json);
bool print_sn_state_changes(uint64_t start_height, uint64_t end_height);
bool print_peer_list(bool white = true, bool gray = true, size_t limit = 0, bool pruned_only = false, bool publicrpc_only = false);
bool print_peer_list_stats();
bool save_blockchain();
bool show_difficulty();
bool show_status();
bool print_connections();
bool print_blockchain_info(int64_t start_block_index, uint64_t end_block_index);
bool print_quorum_state(uint64_t start_height, uint64_t end_height);
bool set_log_level(int8_t level);
bool set_log_categories(std::string categories);
bool print_height();
private:
bool print_block(cryptonote::rpc::GET_BLOCK::request&& req, bool include_hdex);
public:
bool print_block_by_hash(const crypto::hash& block_hash, bool include_hex);
bool print_block_by_height(uint64_t height, bool include_hex);
bool print_transaction(const crypto::hash& transaction_hash, bool include_metadata, bool include_hex, bool include_json);
bool is_key_image_spent(const crypto::key_image &ki);
bool print_transaction_pool_long();
bool print_transaction_pool_short();
bool print_transaction_pool_stats();
bool start_mining(const cryptonote::account_public_address& address, uint64_t num_threads, uint32_t num_blocks, cryptonote::network_type nettype);
bool stop_mining();
bool mining_status();
bool stop_daemon();
bool get_limit(bool up = true, bool down = true);
bool set_limit(int64_t limit_down, int64_t limit_up);
bool out_peers(bool set, uint32_t limit);
bool in_peers(bool set, uint32_t limit);
bool print_bans();
bool ban(const std::string &address, time_t seconds, bool clear_ban = false);
bool unban(const std::string &address);
bool banned(const std::string &address);
bool flush_txpool(std::string txid);
bool output_histogram(const std::vector<uint64_t> &amounts, uint64_t min_count, uint64_t max_count);
bool print_coinbase_tx_sum(uint64_t height, uint64_t count);
bool alt_chain_info(const std::string &tip, size_t above, uint64_t last_blocks);
bool print_blockchain_dynamic_stats(uint64_t nblocks);
bool relay_tx(const std::string &txid);
bool sync_info();
bool pop_blocks(uint64_t num_blocks);
bool print_sn_key();
bool print_sn_status(std::vector<std::string> args);
bool print_sr(uint64_t height);
bool prepare_registration(bool force_registration=false);
bool print_sn(const std::vector<std::string> &args, bool self = false);
bool prune_blockchain();
bool check_blockchain_pruning();
bool print_net_stats();
bool set_bootstrap_daemon(
const std::string &address,
const std::string &username,
const std::string &password);
bool flush_cache(bool bad_txs, bool invalid_blocks);
bool version();
bool test_trigger_uptime_proof();
};
} // namespace daemonize