Fix tx hash calculation bug

This commit is contained in:
Lee *!* Clagett 2023-06-09 16:26:04 -04:00
parent ee7904459c
commit f5cc35f2d0
3 changed files with 69 additions and 5 deletions

View file

@ -42,6 +42,7 @@ namespace
constexpr const std::size_t default_inputs = 2;
constexpr const std::size_t default_outputs = 4;
constexpr const std::size_t default_txextra_size = 2048;
constexpr const std::size_t default_txpool_size = 32;
}
namespace rct
@ -177,6 +178,11 @@ namespace cryptonote
self.transactions.reserve(default_transaction_count);
wire::object(source, WIRE_FIELD(block), WIRE_FIELD(transactions));
}
static void read_bytes(wire::json_reader& source, tx_in_pool& self)
{
wire::object(source, WIRE_FIELD(tx), WIRE_FIELD(tx_hash));
}
} // rpc
} // cryptonote
@ -187,3 +193,8 @@ void lws::rpc::read_bytes(wire::json_reader& source, get_blocks_fast_response& s
wire::object(source, WIRE_FIELD(blocks), WIRE_FIELD(output_indices), WIRE_FIELD(start_height), WIRE_FIELD(current_height));
}
void lws::rpc::read_bytes(wire::json_reader& source, get_transaction_pool_response& self)
{
self.transactions.reserve(default_txpool_size);
wire::object(source, WIRE_FIELD(transactions));
}

View file

@ -46,6 +46,7 @@ namespace cryptonote
namespace rpc
{
struct block_with_transactions;
struct tx_in_pool;
}
}
@ -74,5 +75,21 @@ namespace rpc
using response = get_blocks_fast_response;
};
void read_bytes(wire::json_reader&, get_blocks_fast_response&);
struct get_transaction_pool_request
{
get_transaction_pool_request() = delete;
};
struct get_transaction_pool_response
{
get_transaction_pool_response() = delete;
std::vector<cryptonote::rpc::tx_in_pool> transactions;
};
struct get_transaction_pool
{
using request = get_transaction_pool_request;
using response = get_transaction_pool_response;
};
void read_bytes(wire::json_reader&, get_transaction_pool_response&);
} // rpc
} // lws

View file

@ -52,7 +52,8 @@
#include "misc_log_ex.h" // monero/contrib/epee/include
#include "net/http_client.h"
#include "net/net_parse_helpers.h"
#include "rpc/daemon_messages.h" // monero/src
#include "rpc/daemon_messages.h" // monero/src
#include "rpc/message_data_structs.h" // monero/src
#include "rpc/daemon_zmq.h"
#include "rpc/json.h"
#include "util/source_location.h"
@ -269,9 +270,17 @@ namespace lws
struct send_webhook
{
db::storage const& disk_;
rpc::client& client_;
net::ssl_verification_t verify_mode_;
bool operator()(lws::account& user, const db::output& out) const
std::unordered_map<crypto::hash, crypto::hash> txpool_;
bool operator()(lws::account& user, const db::output& out)
{
/* Upstream monerod does not send all fields for a transaction, so
mempool notifications cannot compute tx_hash correctly (it is not
sent separately, a further blunder). Instead, if there are matching
outputs with webhooks, fetch mempool to compare tx_prefix_hash and
then use corresponding tx_hash. */
const db::webhook_key key{user.id(), db::webhook_type::tx_confirmation};
std::vector<db::webhook_value> hooks{};
{
@ -290,11 +299,37 @@ namespace lws
hooks = std::move(*found);
}
if (!hooks.empty() && txpool_.empty())
{
cryptonote::rpc::GetTransactionPool::Request req{};
if (!send(client_, rpc::client::make_message("get_transaction_pool", req)))
{
MERROR("Unable to compute tx hash for webhook, aborting");
return false;
}
auto resp = client_.get_message(std::chrono::seconds{3});
if (!resp)
{
MERROR("Unable to get txpool: " << resp.error().message());
return false;
}
auto txpool = MONERO_UNWRAP(wire::json::from_bytes<rpc::json<rpc::get_transaction_pool>::response>(std::move(*resp)));
for (auto& tx : txpool.result.transactions)
txpool_.emplace(get_transaction_prefix_hash(tx.tx), tx.tx_hash);
}
std::vector<db::webhook_tx_confirmation> events{};
for (auto& hook : hooks)
{
events.push_back(db::webhook_tx_confirmation{key, std::move(hook), out});
events.back().value.second.confirmations = 0;
const auto hash = txpool_.find(out.tx_prefix_hash);
if (hash != txpool_.end())
events.back().tx_info.link.tx_hash = hash->second;
else
events.pop_back(); //cannot compute tx_hash
}
send_via_http(epee::to_span(events), std::chrono::seconds{5}, verify_mode_);
return true;
@ -479,7 +514,7 @@ namespace lws
scan_transaction_base(users, height, timestamp, tx_hash, tx, out_ids, add_spend{}, add_output{});
}
void scan_transactions(std::string&& txpool_msg, epee::span<lws::account> users, db::storage const& disk, const net::ssl_verification_t verify_mode)
void scan_transactions(std::string&& txpool_msg, epee::span<lws::account> users, db::storage const& disk, rpc::client& client, const net::ssl_verification_t verify_mode)
{
// uint64::max is for txpool
static const std::vector<std::uint64_t> fake_outs{
@ -496,10 +531,11 @@ namespace lws
const auto time =
boost::numeric_cast<std::uint64_t>(std::chrono::system_clock::to_time_t(std::chrono::system_clock::now()));
send_webhook sender{disk, client, verify_mode};
for (const auto& tx : parsed->txes)
{
const crypto::hash hash = cryptonote::get_transaction_hash(tx);
scan_transaction_base(users, db::block_id::txpool, time, hash, tx, fake_outs, null_spend{}, send_webhook{disk, verify_mode});
scan_transaction_base(users, db::block_id::txpool, time, hash, tx, fake_outs, null_spend{}, sender);
}
}
@ -606,7 +642,7 @@ namespace lws
{
if (message->first != rpc::client::topic::txpool)
break; // inner for loop
scan_transactions(std::move(message->second), epee::to_mut_span(users), disk, webhook_verify);
scan_transactions(std::move(message->second), epee::to_mut_span(users), disk, client, webhook_verify);
}
for ( ; message != new_pubs->end(); ++message)