mirror of
https://github.com/vtnerd/monero-lws.git
synced 2025-01-08 19:59:31 +00:00
Do not exit when monerod returns json-rpc error for block fetching (#78)
This commit is contained in:
parent
4704ecc97c
commit
a09b794daa
4 changed files with 69 additions and 25 deletions
|
@ -85,6 +85,8 @@ namespace lws
|
||||||
return "Unspecified error when retrieving exchange rates";
|
return "Unspecified error when retrieving exchange rates";
|
||||||
case error::http_server:
|
case error::http_server:
|
||||||
return "HTTP server failed";
|
return "HTTP server failed";
|
||||||
|
case error::json_rpc:
|
||||||
|
return "Error returned by JSON-RPC server";
|
||||||
case error::exchange_rates_old:
|
case error::exchange_rates_old:
|
||||||
return "Exchange rates are older than cache interval";
|
return "Exchange rates are older than cache interval";
|
||||||
case error::not_enough_mixin:
|
case error::not_enough_mixin:
|
||||||
|
|
|
@ -57,6 +57,7 @@ namespace lws
|
||||||
exchange_rates_fetch, //!< Exchange rates fetching failed
|
exchange_rates_fetch, //!< Exchange rates fetching failed
|
||||||
exchange_rates_old, //!< Exchange rates are older than cache interval
|
exchange_rates_old, //!< Exchange rates are older than cache interval
|
||||||
http_server, //!< HTTP server failure (init or run)
|
http_server, //!< HTTP server failure (init or run)
|
||||||
|
json_rpc, //!< Error returned by JSON-RPC server
|
||||||
not_enough_mixin, //!< Not enough outputs to meet mixin count
|
not_enough_mixin, //!< Not enough outputs to meet mixin count
|
||||||
signal_abort_process, //!< In process ZMQ PUB to abort the process was received
|
signal_abort_process, //!< In process ZMQ PUB to abort the process was received
|
||||||
signal_abort_scan, //!< In process ZMQ PUB to abort the scan was received
|
signal_abort_scan, //!< In process ZMQ PUB to abort the scan was received
|
||||||
|
|
|
@ -26,6 +26,7 @@
|
||||||
// THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
// THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
|
||||||
#include "wire/json.h"
|
#include "wire/json.h"
|
||||||
|
#include "wire/wrapper/variant.h"
|
||||||
|
|
||||||
namespace lws
|
namespace lws
|
||||||
{
|
{
|
||||||
|
@ -65,6 +66,20 @@ namespace rpc
|
||||||
wire::object(dest, WIRE_FIELD_COPY(id), WIRE_FIELD_COPY(jsonrpc), WIRE_FIELD_COPY(method), WIRE_FIELD(params));
|
wire::object(dest, WIRE_FIELD_COPY(id), WIRE_FIELD_COPY(jsonrpc), WIRE_FIELD_COPY(method), WIRE_FIELD(params));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct json_error
|
||||||
|
{
|
||||||
|
json_error()
|
||||||
|
: code(0), message()
|
||||||
|
{}
|
||||||
|
|
||||||
|
std::int32_t code;
|
||||||
|
std::string message;
|
||||||
|
};
|
||||||
|
|
||||||
|
inline void read_bytes(wire::json_reader& source, json_error& self)
|
||||||
|
{
|
||||||
|
wire::object(source, WIRE_FIELD(code), WIRE_FIELD(message));
|
||||||
|
}
|
||||||
|
|
||||||
//! \tparam R implements the READ concept
|
//! \tparam R implements the READ concept
|
||||||
template<typename R>
|
template<typename R>
|
||||||
|
@ -73,13 +88,18 @@ namespace rpc
|
||||||
json_response() = delete;
|
json_response() = delete;
|
||||||
|
|
||||||
unsigned id;
|
unsigned id;
|
||||||
R result;
|
boost::variant<json_error, R> state;
|
||||||
};
|
};
|
||||||
|
|
||||||
template<typename R>
|
template<typename R>
|
||||||
inline void read_bytes(wire::json_reader& source, json_response<R>& self)
|
inline void read_bytes(wire::json_reader& source, json_response<R>& self)
|
||||||
{
|
{
|
||||||
wire::object(source, WIRE_FIELD(id), WIRE_FIELD(result));
|
auto state = wire::variant(std::ref(self.state));
|
||||||
|
wire::object(source,
|
||||||
|
WIRE_FIELD(id),
|
||||||
|
WIRE_OPTION("result", R, state),
|
||||||
|
WIRE_OPTION("error", json_error, state)
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -92,5 +112,26 @@ namespace rpc
|
||||||
using request = json_request<typename M::request, M>;
|
using request = json_request<typename M::request, M>;
|
||||||
using response = json_response<typename M::response>;
|
using response = json_response<typename M::response>;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
//! \tparam M must implement the METHOD concept.
|
||||||
|
template<typename M, typename R = typename M::response>
|
||||||
|
inline expect<R> parse_json_response(std::string&& source)
|
||||||
|
{
|
||||||
|
json_response<R> out{};
|
||||||
|
std::error_code error = wire::json::from_bytes(std::move(source), out);
|
||||||
|
if (error)
|
||||||
|
return error;
|
||||||
|
|
||||||
|
json_error const* const rpc_error = boost::get<json_error>(std::addressof(out.state));
|
||||||
|
if (rpc_error)
|
||||||
|
{
|
||||||
|
MERROR("JSON-RPC server sent error code " << rpc_error->code << " with message: " << rpc_error->message);
|
||||||
|
return {error::json_rpc};
|
||||||
|
}
|
||||||
|
|
||||||
|
return {boost::get<R>(std::move(out.state))};
|
||||||
|
}
|
||||||
|
|
||||||
} // rpc
|
} // rpc
|
||||||
} // lws
|
} // lws
|
||||||
|
|
|
@ -362,11 +362,10 @@ namespace lws
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
rpc::json<rpc::get_transaction_pool>::response txpool{};
|
auto txpool = rpc::parse_json_response<rpc::get_transaction_pool>(std::move(*resp));
|
||||||
const std::error_code err = wire::json::from_bytes(std::move(*resp), txpool);
|
if (!txpool)
|
||||||
if (err)
|
MONERO_THROW(txpool.error(), "Failed fetching transaction pool");
|
||||||
MONERO_THROW(err, "Invalid json-rpc");
|
for (auto& tx : txpool->transactions)
|
||||||
for (auto& tx : txpool.result.transactions)
|
|
||||||
txpool_.emplace(get_transaction_prefix_hash(tx.tx), tx.tx_hash);
|
txpool_.emplace(get_transaction_prefix_hash(tx.tx), tx.tx_hash);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -651,26 +650,27 @@ namespace lws
|
||||||
MONERO_THROW(resp.error(), "Failed to retrieve blocks from daemon");
|
MONERO_THROW(resp.error(), "Failed to retrieve blocks from daemon");
|
||||||
}
|
}
|
||||||
|
|
||||||
rpc::json<rpc::get_blocks_fast>::response fetched{};
|
auto fetched = rpc::parse_json_response<rpc::get_blocks_fast>(std::move(*resp));
|
||||||
|
if (!fetched)
|
||||||
{
|
{
|
||||||
const std::error_code error = wire::json::from_bytes(std::move(*resp), fetched);
|
MERROR("Failed to retrieve next blocks: " << fetched.error().message() << ". Resetting state and trying again");
|
||||||
if (error)
|
return;
|
||||||
throw std::system_error{error};
|
|
||||||
}
|
}
|
||||||
if (fetched.result.blocks.empty())
|
|
||||||
|
if (fetched->blocks.empty())
|
||||||
throw std::runtime_error{"Daemon unexpectedly returned zero blocks"};
|
throw std::runtime_error{"Daemon unexpectedly returned zero blocks"};
|
||||||
|
|
||||||
if (fetched.result.start_height != req.start_height)
|
if (fetched->start_height != req.start_height)
|
||||||
{
|
{
|
||||||
MWARNING("Daemon sent wrong blocks, resetting state");
|
MWARNING("Daemon sent wrong blocks, resetting state");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// prep for next blocks retrieval
|
// prep for next blocks retrieval
|
||||||
req.start_height = fetched.result.start_height + fetched.result.blocks.size() - 1;
|
req.start_height = fetched->start_height + fetched->blocks.size() - 1;
|
||||||
block_request = rpc::client::make_message("get_blocks_fast", req);
|
block_request = rpc::client::make_message("get_blocks_fast", req);
|
||||||
|
|
||||||
if (fetched.result.blocks.size() <= 1)
|
if (fetched->blocks.size() <= 1)
|
||||||
{
|
{
|
||||||
// synced to top of chain, wait for next blocks
|
// synced to top of chain, wait for next blocks
|
||||||
for (bool wait_for_block = true; wait_for_block; )
|
for (bool wait_for_block = true; wait_for_block; )
|
||||||
|
@ -712,26 +712,26 @@ namespace lws
|
||||||
if (!send(client, block_request.clone()))
|
if (!send(client, block_request.clone()))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
if (fetched.result.blocks.size() != fetched.result.output_indices.size())
|
if (fetched->blocks.size() != fetched->output_indices.size())
|
||||||
throw std::runtime_error{"Bad daemon response - need same number of blocks and indices"};
|
throw std::runtime_error{"Bad daemon response - need same number of blocks and indices"};
|
||||||
|
|
||||||
blockchain.push_back(cryptonote::get_block_hash(fetched.result.blocks.front().block));
|
blockchain.push_back(cryptonote::get_block_hash(fetched->blocks.front().block));
|
||||||
|
|
||||||
auto blocks = epee::to_span(fetched.result.blocks);
|
auto blocks = epee::to_span(fetched->blocks);
|
||||||
auto indices = epee::to_span(fetched.result.output_indices);
|
auto indices = epee::to_span(fetched->output_indices);
|
||||||
|
|
||||||
if (fetched.result.start_height != 1)
|
if (fetched->start_height != 1)
|
||||||
{
|
{
|
||||||
// skip overlap block
|
// skip overlap block
|
||||||
blocks.remove_prefix(1);
|
blocks.remove_prefix(1);
|
||||||
indices.remove_prefix(1);
|
indices.remove_prefix(1);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
fetched.result.start_height = 0;
|
fetched->start_height = 0;
|
||||||
|
|
||||||
for (auto block_data : boost::combine(blocks, indices))
|
for (auto block_data : boost::combine(blocks, indices))
|
||||||
{
|
{
|
||||||
++(fetched.result.start_height);
|
++(fetched->start_height);
|
||||||
|
|
||||||
cryptonote::block const& block = boost::get<0>(block_data).block;
|
cryptonote::block const& block = boost::get<0>(block_data).block;
|
||||||
auto const& txes = boost::get<0>(block_data).transactions;
|
auto const& txes = boost::get<0>(block_data).transactions;
|
||||||
|
@ -749,7 +749,7 @@ namespace lws
|
||||||
|
|
||||||
scan_transaction(
|
scan_transaction(
|
||||||
epee::to_mut_span(users),
|
epee::to_mut_span(users),
|
||||||
db::block_id(fetched.result.start_height),
|
db::block_id(fetched->start_height),
|
||||||
block.timestamp,
|
block.timestamp,
|
||||||
miner_tx_hash,
|
miner_tx_hash,
|
||||||
block.miner_tx,
|
block.miner_tx,
|
||||||
|
@ -764,7 +764,7 @@ namespace lws
|
||||||
{
|
{
|
||||||
scan_transaction(
|
scan_transaction(
|
||||||
epee::to_mut_span(users),
|
epee::to_mut_span(users),
|
||||||
db::block_id(fetched.result.start_height),
|
db::block_id(fetched->start_height),
|
||||||
block.timestamp,
|
block.timestamp,
|
||||||
boost::get<0>(tx_data),
|
boost::get<0>(tx_data),
|
||||||
boost::get<1>(tx_data),
|
boost::get<1>(tx_data),
|
||||||
|
@ -798,7 +798,7 @@ namespace lws
|
||||||
}
|
}
|
||||||
|
|
||||||
for (account& user : users)
|
for (account& user : users)
|
||||||
user.updated(db::block_id(fetched.result.start_height));
|
user.updated(db::block_id(fetched->start_height));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
catch (std::exception const& e)
|
catch (std::exception const& e)
|
||||||
|
|
Loading…
Reference in a new issue