mirror of
https://github.com/XRPLF/rippled.git
synced 2025-12-06 17:27:55 +00:00
Improve transaction relaying logic:
The existing logic involves every server sending every transaction that it receives to all its peers (except the one that it received a transaction from). This commit instead uses a randomized algorithm, where a node will randomly select peers to relay a given transaction to, caching the list of transaction hashes that are not relayed and forwading them to peers once every second. Peers can then determine whether there are transactions that they have not seen and can request them from the node which has them. It is expected that this feature will further reduce the bandwidth needed to operate a server.
This commit is contained in:
committed by
Nik Bougalis
parent
0d17dd8228
commit
ea145d12c7
@@ -289,6 +289,23 @@ public:
|
||||
{
|
||||
return false;
|
||||
}
|
||||
void
|
||||
sendTxQueue() override
|
||||
{
|
||||
}
|
||||
void
|
||||
addTxQueue(const uint256&) override
|
||||
{
|
||||
}
|
||||
void
|
||||
removeTxQueue(const uint256&) override
|
||||
{
|
||||
}
|
||||
bool
|
||||
txReduceRelayEnabled() const override
|
||||
{
|
||||
return false;
|
||||
}
|
||||
|
||||
bool ledgerReplayEnabled_;
|
||||
};
|
||||
@@ -1060,7 +1077,8 @@ struct LedgerReplayer_test : public beast::unit_test::suite
|
||||
{
|
||||
testcase("handshake test");
|
||||
auto handshake = [&](bool client, bool server, bool expecting) -> bool {
|
||||
auto request = ripple::makeRequest(true, false, false, client);
|
||||
auto request =
|
||||
ripple::makeRequest(true, false, client, false, false);
|
||||
http_request_type http_request;
|
||||
http_request.version(request.version());
|
||||
http_request.base() = request.base();
|
||||
|
||||
@@ -488,8 +488,9 @@ public:
|
||||
auto request = ripple::makeRequest(
|
||||
true,
|
||||
env->app().config().COMPRESSION,
|
||||
env->app().config().VP_REDUCE_RELAY_ENABLE,
|
||||
false);
|
||||
false,
|
||||
env->app().config().TX_REDUCE_RELAY_ENABLE,
|
||||
env->app().config().VP_REDUCE_RELAY_ENABLE);
|
||||
http_request_type http_request;
|
||||
http_request.version(request.version());
|
||||
http_request.base() = request.base();
|
||||
|
||||
@@ -159,6 +159,23 @@ public:
|
||||
{
|
||||
return false;
|
||||
}
|
||||
bool
|
||||
txReduceRelayEnabled() const override
|
||||
{
|
||||
return false;
|
||||
}
|
||||
void
|
||||
sendTxQueue() override
|
||||
{
|
||||
}
|
||||
void
|
||||
addTxQueue(const uint256&) override
|
||||
{
|
||||
}
|
||||
void
|
||||
removeTxQueue(const uint256&) override
|
||||
{
|
||||
}
|
||||
};
|
||||
|
||||
/** Manually advanced clock. */
|
||||
@@ -491,7 +508,7 @@ class OverlaySim : public Overlay, public reduce_relay::SquelchHandler
|
||||
public:
|
||||
using id_t = Peer::id_t;
|
||||
using clock_type = ManualClock;
|
||||
OverlaySim(Application& app) : slots_(app, *this), app_(app)
|
||||
OverlaySim(Application& app) : slots_(app.logs(), *this), logs_(app.logs())
|
||||
{
|
||||
}
|
||||
|
||||
@@ -545,7 +562,7 @@ public:
|
||||
Peer::id_t id;
|
||||
if (peersCache_.empty() || !useCache)
|
||||
{
|
||||
peer = std::make_shared<PeerSim>(*this, app_.journal("Squelch"));
|
||||
peer = std::make_shared<PeerSim>(*this, logs_.journal("Squelch"));
|
||||
id = peer->id();
|
||||
}
|
||||
else
|
||||
@@ -665,7 +682,7 @@ private:
|
||||
Peers peers_;
|
||||
Peers peersCache_;
|
||||
reduce_relay::Slots<ManualClock> slots_;
|
||||
Application& app_;
|
||||
Logs& logs_;
|
||||
};
|
||||
|
||||
class Network
|
||||
@@ -1398,7 +1415,8 @@ vp_squelched=1
|
||||
|
||||
auto run = [&](int npeers) {
|
||||
handler.maxDuration_ = 0;
|
||||
reduce_relay::Slots<ManualClock> slots(env_.app(), handler);
|
||||
reduce_relay::Slots<ManualClock> slots(
|
||||
env_.app().logs(), handler);
|
||||
// 1st message from a new peer switches the slot
|
||||
// to counting state and resets the counts of all peers +
|
||||
// MAX_MESSAGE_THRESHOLD + 1 messages to reach the threshold
|
||||
@@ -1494,8 +1512,9 @@ vp_squelched=1
|
||||
auto request = ripple::makeRequest(
|
||||
true,
|
||||
env_.app().config().COMPRESSION,
|
||||
env_.app().config().VP_REDUCE_RELAY_ENABLE,
|
||||
false);
|
||||
false,
|
||||
env_.app().config().TX_REDUCE_RELAY_ENABLE,
|
||||
env_.app().config().VP_REDUCE_RELAY_ENABLE);
|
||||
http_request_type http_request;
|
||||
http_request.version(request.version());
|
||||
http_request.base() = request.base();
|
||||
|
||||
275
src/test/overlay/tx_reduce_relay_test.cpp
Normal file
275
src/test/overlay/tx_reduce_relay_test.cpp
Normal file
@@ -0,0 +1,275 @@
|
||||
//------------------------------------------------------------------------------
|
||||
/*
|
||||
This file is part of rippled: https://github.com/ripple/rippled
|
||||
Copyright 2020 Ripple Labs Inc.
|
||||
|
||||
Permission to use, copy, modify, and/or distribute this software for any
|
||||
purpose with or without fee is hereby granted, provided that the above
|
||||
copyright notice and this permission notice appear in all copies.
|
||||
|
||||
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
||||
WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
||||
MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
||||
ANY SPECIAL , DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
||||
WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
||||
ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
||||
OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
||||
*/
|
||||
//==============================================================================
|
||||
#include <ripple/basics/make_SSLContext.h>
|
||||
#include <ripple/beast/unit_test.h>
|
||||
#include <ripple/overlay/impl/OverlayImpl.h>
|
||||
#include <ripple/overlay/impl/PeerImp.h>
|
||||
#include <ripple/peerfinder/impl/SlotImp.h>
|
||||
#include <test/jtx/Env.h>
|
||||
|
||||
namespace ripple {
|
||||
|
||||
namespace test {
|
||||
|
||||
class tx_reduce_relay_test : public beast::unit_test::suite
|
||||
{
|
||||
public:
|
||||
using socket_type = boost::asio::ip::tcp::socket;
|
||||
using middle_type = boost::beast::tcp_stream;
|
||||
using stream_type = boost::beast::ssl_stream<middle_type>;
|
||||
using shared_context = std::shared_ptr<boost::asio::ssl::context>;
|
||||
|
||||
private:
|
||||
void
|
||||
doTest(const std::string& msg, bool log, std::function<void(bool)> f)
|
||||
{
|
||||
testcase(msg);
|
||||
f(log);
|
||||
}
|
||||
|
||||
void
|
||||
testConfig(bool log)
|
||||
{
|
||||
doTest("Config Test", log, [&](bool log) {
|
||||
auto test = [&](bool enable,
|
||||
bool metrics,
|
||||
std::uint16_t min,
|
||||
std::uint16_t pct,
|
||||
bool success = true) {
|
||||
std::stringstream str("[reduce_relay]");
|
||||
str << "[reduce_relay]\n"
|
||||
<< "tx_enable=" << static_cast<int>(enable) << "\n"
|
||||
<< "tx_metrics=" << static_cast<int>(metrics) << "\n"
|
||||
<< "tx_min_peers=" << min << "\n"
|
||||
<< "tx_relay_percentage=" << pct << "\n";
|
||||
Config c;
|
||||
try
|
||||
{
|
||||
c.loadFromString(str.str());
|
||||
|
||||
BEAST_EXPECT(c.TX_REDUCE_RELAY_ENABLE == enable);
|
||||
BEAST_EXPECT(c.TX_REDUCE_RELAY_METRICS == metrics);
|
||||
BEAST_EXPECT(c.TX_REDUCE_RELAY_MIN_PEERS == min);
|
||||
BEAST_EXPECT(c.TX_RELAY_PERCENTAGE == pct);
|
||||
if (success)
|
||||
pass();
|
||||
else
|
||||
fail();
|
||||
}
|
||||
catch (...)
|
||||
{
|
||||
if (success)
|
||||
fail();
|
||||
else
|
||||
pass();
|
||||
}
|
||||
};
|
||||
|
||||
test(true, true, 20, 25);
|
||||
test(false, false, 20, 25);
|
||||
test(false, false, 20, 0, false);
|
||||
test(false, false, 20, 101, false);
|
||||
test(false, false, 9, 10, false);
|
||||
test(false, false, 10, 9, false);
|
||||
});
|
||||
}
|
||||
|
||||
class PeerTest : public PeerImp
|
||||
{
|
||||
public:
|
||||
PeerTest(
|
||||
Application& app,
|
||||
std::shared_ptr<PeerFinder::Slot> const& slot,
|
||||
http_request_type&& request,
|
||||
PublicKey const& publicKey,
|
||||
ProtocolVersion protocol,
|
||||
Resource::Consumer consumer,
|
||||
std::unique_ptr<tx_reduce_relay_test::stream_type>&& stream_ptr,
|
||||
OverlayImpl& overlay)
|
||||
: PeerImp(
|
||||
app,
|
||||
sid_,
|
||||
slot,
|
||||
std::move(request),
|
||||
publicKey,
|
||||
protocol,
|
||||
consumer,
|
||||
std::move(stream_ptr),
|
||||
overlay)
|
||||
{
|
||||
sid_++;
|
||||
}
|
||||
~PeerTest() = default;
|
||||
|
||||
void
|
||||
run() override
|
||||
{
|
||||
}
|
||||
void
|
||||
send(std::shared_ptr<Message> const&) override
|
||||
{
|
||||
sendTx_++;
|
||||
}
|
||||
void
|
||||
addTxQueue(const uint256& hash) override
|
||||
{
|
||||
queueTx_++;
|
||||
}
|
||||
static void
|
||||
init()
|
||||
{
|
||||
queueTx_ = 0;
|
||||
sendTx_ = 0;
|
||||
sid_ = 0;
|
||||
}
|
||||
inline static std::size_t sid_ = 0;
|
||||
inline static std::uint16_t queueTx_ = 0;
|
||||
inline static std::uint16_t sendTx_ = 0;
|
||||
};
|
||||
|
||||
std::uint16_t lid_{0};
|
||||
std::uint16_t rid_{1};
|
||||
shared_context context_;
|
||||
ProtocolVersion protocolVersion_;
|
||||
boost::beast::multi_buffer read_buf_;
|
||||
|
||||
public:
|
||||
tx_reduce_relay_test()
|
||||
: context_(make_SSLContext("")), protocolVersion_{1, 7}
|
||||
{
|
||||
}
|
||||
|
||||
private:
|
||||
void
|
||||
addPeer(
|
||||
jtx::Env& env,
|
||||
std::vector<std::shared_ptr<PeerTest>>& peers,
|
||||
std::uint16_t& nDisabled)
|
||||
{
|
||||
auto& overlay = dynamic_cast<OverlayImpl&>(env.app().overlay());
|
||||
boost::beast::http::request<boost::beast::http::dynamic_body> request;
|
||||
(nDisabled == 0)
|
||||
? (void)request.insert(
|
||||
"X-Protocol-Ctl",
|
||||
makeFeaturesRequestHeader(false, false, true, false))
|
||||
: (void)nDisabled--;
|
||||
auto stream_ptr = std::make_unique<stream_type>(
|
||||
socket_type(std::forward<boost::asio::io_service&>(
|
||||
env.app().getIOService())),
|
||||
*context_);
|
||||
beast::IP::Endpoint local(
|
||||
beast::IP::Address::from_string("172.1.1." + std::to_string(lid_)));
|
||||
beast::IP::Endpoint remote(
|
||||
beast::IP::Address::from_string("172.1.1." + std::to_string(rid_)));
|
||||
PublicKey key(std::get<0>(randomKeyPair(KeyType::ed25519)));
|
||||
auto consumer = overlay.resourceManager().newInboundEndpoint(remote);
|
||||
auto slot = overlay.peerFinder().new_inbound_slot(local, remote);
|
||||
auto const peer = std::make_shared<PeerTest>(
|
||||
env.app(),
|
||||
slot,
|
||||
std::move(request),
|
||||
key,
|
||||
protocolVersion_,
|
||||
consumer,
|
||||
std::move(stream_ptr),
|
||||
overlay);
|
||||
overlay.add_active(peer);
|
||||
peers.emplace_back(peer); // overlay stores week ptr to PeerImp
|
||||
lid_ += 2;
|
||||
rid_ += 2;
|
||||
assert(lid_ <= 254);
|
||||
}
|
||||
|
||||
void
|
||||
testRelay(
|
||||
std::string const& test,
|
||||
bool txRREnabled,
|
||||
std::uint16_t nPeers,
|
||||
std::uint16_t nDisabled,
|
||||
std::uint16_t minPeers,
|
||||
std::uint16_t relayPercentage,
|
||||
std::uint16_t expectRelay,
|
||||
std::uint16_t expectQueue,
|
||||
std::set<Peer::id_t> const& toSkip = {})
|
||||
{
|
||||
testcase(test);
|
||||
jtx::Env env(*this);
|
||||
std::vector<std::shared_ptr<PeerTest>> peers;
|
||||
env.app().config().TX_REDUCE_RELAY_ENABLE = txRREnabled;
|
||||
env.app().config().TX_REDUCE_RELAY_MIN_PEERS = minPeers;
|
||||
env.app().config().TX_RELAY_PERCENTAGE = relayPercentage;
|
||||
PeerTest::init();
|
||||
lid_ = 0;
|
||||
rid_ = 0;
|
||||
for (int i = 0; i < nPeers; i++)
|
||||
addPeer(env, peers, nDisabled);
|
||||
protocol::TMTransaction m;
|
||||
m.set_rawtransaction("transaction");
|
||||
m.set_deferred(false);
|
||||
m.set_status(protocol::TransactionStatus::tsNEW);
|
||||
env.app().overlay().relay(uint256{0}, m, toSkip);
|
||||
BEAST_EXPECT(
|
||||
PeerTest::sendTx_ == expectRelay &&
|
||||
PeerTest::queueTx_ == expectQueue);
|
||||
}
|
||||
|
||||
void
|
||||
run() override
|
||||
{
|
||||
bool log = false;
|
||||
std::set<Peer::id_t> skip = {0, 1, 2, 3, 4};
|
||||
testConfig(log);
|
||||
// relay to all peers, no hash queue
|
||||
testRelay("feature disabled", false, 10, 0, 10, 25, 10, 0);
|
||||
// relay to nPeers - skip (10-5=5)
|
||||
testRelay("feature disabled & skip", false, 10, 0, 10, 25, 5, 0, skip);
|
||||
// relay to all peers because min is greater than nPeers
|
||||
testRelay("relay all 1", true, 10, 0, 20, 25, 10, 0);
|
||||
// relay to all peers because min + disabled is greater thant nPeers
|
||||
testRelay("relay all 2", true, 20, 15, 10, 25, 20, 0);
|
||||
// relay to minPeers + 25% of nPeers-minPeers (20+0.25*(60-20)=30),
|
||||
// queue the rest (30)
|
||||
testRelay("relay & queue", true, 60, 0, 20, 25, 30, 30);
|
||||
// relay to minPeers + 25% of (nPeers - nPeers) - skip
|
||||
// (20+0.25*(60-20)-5=25), queue the rest, skip counts towards relayed
|
||||
// (60-25-5=30)
|
||||
testRelay("skip", true, 60, 0, 20, 25, 25, 30, skip);
|
||||
// relay to minPeers + disabled + 25% of (nPeers - minPeers - disalbed)
|
||||
// (20+10+0.25*(70-20-10)=40), queue the rest (30)
|
||||
testRelay("disabled", true, 70, 10, 20, 25, 40, 30);
|
||||
// relay to minPeers + disabled-not-in-skip + 25% of (nPeers - minPeers
|
||||
// - disabled) (20+5+0.25*(70-20-10)=35), queue the rest, skip counts
|
||||
// towards relayed (70-35-5=30))
|
||||
testRelay("disabled & skip", true, 70, 10, 20, 25, 35, 30, skip);
|
||||
// relay to minPeers + disabled + 25% of (nPeers - minPeers - disabled)
|
||||
// - skip (10+5+0.25*(15-10-5)-10=5), queue the rest, skip counts
|
||||
// towards relayed (15-5-10=0)
|
||||
skip = {0, 1, 2, 3, 4, 5, 6, 7, 8, 9};
|
||||
testRelay("disabled & skip, no queue", true, 15, 5, 10, 25, 5, 0, skip);
|
||||
// relay to minPeers + disabled + 25% of (nPeers - minPeers - disabled)
|
||||
// - skip (10+2+0.25*(20-10-2)-14=0), queue the rest, skip counts
|
||||
// towards relayed (20-14=6)
|
||||
skip = {0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13};
|
||||
testRelay("disabled & skip, no relay", true, 20, 2, 10, 25, 0, 6, skip);
|
||||
}
|
||||
};
|
||||
|
||||
BEAST_DEFINE_TESTSUITE(tx_reduce_relay, ripple_data, ripple);
|
||||
} // namespace test
|
||||
} // namespace ripple
|
||||
Reference in New Issue
Block a user