4#include "data/LedgerCacheInterface.hpp"
5#include "data/Types.hpp"
6#include "etl/CorruptionDetector.hpp"
7#include "util/Spawn.hpp"
8#include "util/log/Logger.hpp"
10#include <boost/asio/executor_work_guard.hpp>
11#include <boost/asio/io_context.hpp>
12#include <boost/asio/spawn.hpp>
13#include <boost/json.hpp>
14#include <boost/json/object.hpp>
15#include <boost/utility/result_of.hpp>
16#include <boost/uuid/uuid.hpp>
17#include <xrpl/basics/base_uint.h>
18#include <xrpl/protocol/AccountID.h>
19#include <xrpl/protocol/Fees.h>
20#include <xrpl/protocol/LedgerHeader.h>
28#include <shared_mutex>
46 what()
const throw()
override
48 return "Database read timed out. Please retry the request";
52static constexpr std::size_t kDEFAULT_WAIT_BETWEEN_RETRY = 500;
61template <
typename FnType>
71 LOG(log.
error()) <<
"Database request timed out. Sleeping and retrying ... ";
72 std::this_thread::sleep_for(std::chrono::milliseconds(waitMs));
84template <
typename FnType>
88 boost::asio::io_context ctx;
90 using R =
typename boost::result_of<FnType(boost::asio::yield_context)>::type;
91 if constexpr (!std::is_same_v<R, void>) {
93 util::spawn(ctx, [_ = boost::asio::make_work_guard(ctx), &func, &res](
auto yield) {
100 util::spawn(ctx, [_ = boost::asio::make_work_guard(ctx), &func](
auto yield) {
115template <
typename FnType>
128 mutable std::shared_mutex rngMtx_;
129 std::optional<LedgerRange> range_;
130 std::reference_wrapper<LedgerCacheInterface> cache_;
131 std::optional<etl::CorruptionDetector> corruptionDetector_;
173 corruptionDetector_ = std::move(detector);
183 virtual std::optional<ripple::LedgerHeader>
193 virtual std::optional<ripple::LedgerHeader>
202 virtual std::optional<std::uint32_t>
210 std::optional<LedgerRange>
223 virtual std::vector<ripple::uint256>
225 std::uint32_t number,
226 std::uint32_t pageSize,
228 boost::asio::yield_context yield
256 setRange(uint32_t min, uint32_t max,
bool force =
false);
265 std::optional<ripple::Fees>
266 fetchFees(std::uint32_t seq, boost::asio::yield_context yield)
const;
275 virtual std::optional<TransactionAndMetadata>
285 virtual std::vector<TransactionAndMetadata>
287 std::vector<ripple::uint256>
const& hashes,
288 boost::asio::yield_context yield
303 ripple::AccountID
const& account,
306 std::optional<TransactionsCursor>
const& txnCursor,
307 boost::asio::yield_context yield
317 virtual std::vector<TransactionAndMetadata>
319 std::uint32_t ledgerSequence,
320 boost::asio::yield_context yield
330 virtual std::vector<ripple::uint256>
332 std::uint32_t ledgerSequence,
333 boost::asio::yield_context yield
344 virtual std::optional<NFT>
346 ripple::uint256
const& tokenID,
347 std::uint32_t ledgerSequence,
348 boost::asio::yield_context yield
363 ripple::uint256
const& tokenID,
366 std::optional<TransactionsCursor>
const& cursorIn,
367 boost::asio::yield_context yield
384 ripple::AccountID
const& issuer,
385 std::optional<std::uint32_t>
const& taxon,
386 std::uint32_t ledgerSequence,
388 std::optional<ripple::uint256>
const& cursorIn,
389 boost::asio::yield_context yield
404 ripple::uint192
const& mptID,
405 std::uint32_t
const limit,
406 std::optional<ripple::AccountID>
const& cursorIn,
407 std::uint32_t
const ledgerSequence,
408 boost::asio::yield_context yield
424 ripple::uint256
const& key,
425 std::uint32_t sequence,
426 boost::asio::yield_context yield
439 std::optional<std::uint32_t>
441 ripple::uint256
const& key,
442 std::uint32_t sequence,
443 boost::asio::yield_context yield
460 std::vector<ripple::uint256>
const& keys,
461 std::uint32_t sequence,
462 boost::asio::yield_context yield
473 virtual std::optional<Blob>
475 ripple::uint256
const& key,
476 std::uint32_t sequence,
477 boost::asio::yield_context yield
488 virtual std::optional<std::uint32_t>
490 ripple::uint256
const& key,
491 std::uint32_t sequence,
492 boost::asio::yield_context yield
503 virtual std::vector<Blob>
505 std::vector<ripple::uint256>
const& keys,
506 std::uint32_t sequence,
507 boost::asio::yield_context yield
517 virtual std::vector<LedgerObject>
518 fetchLedgerDiff(std::uint32_t ledgerSequence, boost::asio::yield_context yield)
const = 0;
532 std::optional<ripple::uint256>
const& cursor,
533 std::uint32_t ledgerSequence,
536 boost::asio::yield_context yield
547 std::optional<LedgerObject>
550 std::uint32_t ledgerSequence,
551 boost::asio::yield_context yield
566 std::optional<ripple::uint256>
569 std::uint32_t ledgerSequence,
570 boost::asio::yield_context yield
581 virtual std::optional<ripple::uint256>
584 std::uint32_t ledgerSequence,
585 boost::asio::yield_context yield
599 ripple::uint256
const& book,
600 std::uint32_t ledgerSequence,
602 boost::asio::yield_context yield
612 virtual std::optional<std::string>
614 std::string
const& migratorName,
615 boost::asio::yield_context yield
620 std::expected<std::vector<std::pair<boost::uuids::uuid, std::string>>, std::string>;
639 std::optional<LedgerRange>
648 virtual std::optional<LedgerRange>
656 std::optional<LedgerRange>
666 writeLedger(ripple::LedgerHeader
const& ledgerHeader, std::string&& blob) = 0;
692 std::string&& transaction,
693 std::string&& metadata
744 writeSuccessor(std::string&& key, std::uint32_t seq, std::string&& successor) = 0;
798 virtual boost::json::object
810 doWriteLedgerObject(std::string&& key, std::uint32_t seq, std::string&& blob) = 0;
818 doFinishWrites() = 0;
821 updateRangeImpl(uint32_t newMax);
The interface to the database used by Clio.
Definition BackendInterface.hpp:125
virtual std::vector< TransactionAndMetadata > fetchAllTransactionsInLedger(std::uint32_t ledgerSequence, boost::asio::yield_context yield) const =0
Fetches all transactions from a specific ledger.
virtual void writeNodeMessage(boost::uuids::uuid const &uuid, std::string message)=0
Write a node message. Used by ClusterCommunicationService.
std::optional< ripple::Fees > fetchFees(std::uint32_t seq, boost::asio::yield_context yield) const
Fetch the fees from a specific ledger sequence.
Definition BackendInterface.cpp:349
virtual std::vector< ripple::uint256 > fetchAccountRoots(std::uint32_t number, std::uint32_t pageSize, std::uint32_t seq, boost::asio::yield_context yield) const =0
Fetch the specified number of account root object indexes by page, the accounts need to exist for seq...
virtual bool isTooBusy() const =0
virtual ClioNodesDataFetchResult fetchClioNodesData(boost::asio::yield_context yield) const =0
Fetches the data of all nodes in the cluster.
void forceUpdateRange(uint32_t newMax)
Updates the range of sequences that are stored in the DB without any checks.
Definition BackendInterface.cpp:275
virtual TransactionsAndCursor fetchAccountTransactions(ripple::AccountID const &account, std::uint32_t limit, bool forward, std::optional< TransactionsCursor > const &txnCursor, boost::asio::yield_context yield) const =0
Fetches all transactions for a specific account.
std::expected< std::vector< std::pair< boost::uuids::uuid, std::string > >, std::string > ClioNodesDataFetchResult
Return type for fetchClioNodesData() method.
Definition BackendInterface.hpp:619
virtual void writeNFTs(std::vector< NFTsData > const &data)=0
Writes NFTs to the database.
std::vector< Blob > fetchLedgerObjects(std::vector< ripple::uint256 > const &keys, std::uint32_t sequence, boost::asio::yield_context yield) const
Fetches all ledger objects by their keys.
Definition BackendInterface.cpp:95
virtual std::optional< TransactionAndMetadata > fetchTransaction(ripple::uint256 const &hash, boost::asio::yield_context yield) const =0
Fetches a specific transaction.
void setRange(uint32_t min, uint32_t max, bool force=false)
Sets the range of sequences that are stored in the DB.
Definition BackendInterface.cpp:282
LedgerCacheInterface & cache()
Definition BackendInterface.hpp:160
std::optional< LedgerObject > fetchSuccessorObject(ripple::uint256 key, std::uint32_t ledgerSequence, boost::asio::yield_context yield) const
Fetches the successor object.
Definition BackendInterface.cpp:146
virtual std::vector< Blob > doFetchLedgerObjects(std::vector< ripple::uint256 > const &keys, std::uint32_t sequence, boost::asio::yield_context yield) const =0
The database-specific implementation for fetching ledger objects.
virtual void startWrites() const =0
Starts a write transaction with the DB. No-op for cassandra.
virtual std::vector< ripple::uint256 > fetchAllTransactionHashesInLedger(std::uint32_t ledgerSequence, boost::asio::yield_context yield) const =0
Fetches all transaction hashes from a specific ledger.
std::optional< std::uint32_t > fetchLedgerObjectSeq(ripple::uint256 const &key, std::uint32_t sequence, boost::asio::yield_context yield) const
Fetches a specific ledger object sequence.
Definition BackendInterface.cpp:82
BookOffersPage fetchBookOffers(ripple::uint256 const &book, std::uint32_t ledgerSequence, std::uint32_t limit, boost::asio::yield_context yield) const
Fetches book offers.
Definition BackendInterface.cpp:164
virtual std::optional< LedgerRange > hardFetchLedgerRange(boost::asio::yield_context yield) const =0
Fetches the ledger range from DB.
virtual void writeLedgerObject(std::string &&key, std::uint32_t seq, std::string &&blob)
Writes a new ledger object.
Definition BackendInterface.cpp:46
virtual void writeSuccessor(std::string &&key, std::uint32_t seq, std::string &&successor)=0
Write a new successor.
virtual std::optional< std::string > fetchMigratorStatus(std::string const &migratorName, boost::asio::yield_context yield) const =0
Fetches the status of migrator by name.
virtual void waitForWritesToFinish()=0
Wait for all pending writes to finish.
std::optional< LedgerRange > hardFetchLedgerRange() const
Synchronously fetches the ledger range from DB.
Definition BackendInterface.cpp:243
BackendInterface(LedgerCacheInterface &cache)
Construct a new backend interface instance.
Definition BackendInterface.hpp:139
virtual std::vector< TransactionAndMetadata > fetchTransactions(std::vector< ripple::uint256 > const &hashes, boost::asio::yield_context yield) const =0
Fetches multiple transactions.
virtual void writeAccountTransaction(AccountTransactionsData record)=0
Write a new account transaction.
virtual NFTsAndCursor fetchNFTsByIssuer(ripple::AccountID const &issuer, std::optional< std::uint32_t > const &taxon, std::uint32_t ledgerSequence, std::uint32_t limit, std::optional< ripple::uint256 > const &cursorIn, boost::asio::yield_context yield) const =0
Fetches all NFTs issued by a given address.
virtual std::optional< ripple::LedgerHeader > fetchLedgerBySequence(std::uint32_t sequence, boost::asio::yield_context yield) const =0
Fetches a specific ledger by sequence number.
virtual std::optional< Blob > doFetchLedgerObject(ripple::uint256 const &key, std::uint32_t sequence, boost::asio::yield_context yield) const =0
The database-specific implementation for fetching a ledger object.
std::optional< ripple::uint256 > fetchSuccessorKey(ripple::uint256 key, std::uint32_t ledgerSequence, boost::asio::yield_context yield) const
Fetches the successor key.
Definition BackendInterface.cpp:130
std::optional< LedgerRange > hardFetchLedgerRangeNoThrow() const
Fetches the ledger range from DB retrying until no DatabaseTimeout is thrown.
Definition BackendInterface.cpp:53
virtual void writeNFTTransactions(std::vector< NFTTransactionsData > const &data)=0
Write NFTs transactions.
std::optional< LedgerRange > fetchLedgerRange() const
Fetch the current ledger range.
Definition BackendInterface.cpp:249
virtual void writeTransaction(std::string &&hash, std::uint32_t seq, std::uint32_t date, std::string &&transaction, std::string &&metadata)=0
Writes a new transaction.
bool finishWrites(std::uint32_t ledgerSequence)
Tells database we finished writing all data for a specific ledger.
Definition BackendInterface.cpp:35
virtual boost::json::object stats() const =0
void setCorruptionDetector(etl::CorruptionDetector detector)
Sets the corruption detector.
Definition BackendInterface.hpp:171
virtual TransactionsAndCursor fetchNFTTransactions(ripple::uint256 const &tokenID, std::uint32_t limit, bool forward, std::optional< TransactionsCursor > const &cursorIn, boost::asio::yield_context yield) const =0
Fetches all transactions for a specific NFT.
virtual std::optional< std::uint32_t > fetchLatestLedgerSequence(boost::asio::yield_context yield) const =0
Fetches the latest ledger sequence.
virtual std::optional< NFT > fetchNFT(ripple::uint256 const &tokenID, std::uint32_t ledgerSequence, boost::asio::yield_context yield) const =0
Fetches a specific NFT.
virtual void writeAccountTransactions(std::vector< AccountTransactionsData > data)=0
Write a new set of account transactions.
virtual void writeMigratorStatus(std::string const &migratorName, std::string const &status)=0
Mark the migration status of a migrator as Migrated in the database.
virtual void writeMPTHolders(std::vector< MPTHolderData > const &data)=0
Write accounts that started holding onto a MPT.
virtual std::vector< LedgerObject > fetchLedgerDiff(std::uint32_t ledgerSequence, boost::asio::yield_context yield) const =0
Returns the difference between ledgers.
std::optional< Blob > fetchLedgerObject(ripple::uint256 const &key, std::uint32_t sequence, boost::asio::yield_context yield) const
Fetches a specific ledger object.
Definition BackendInterface.cpp:60
LedgerPage fetchLedgerPage(std::optional< ripple::uint256 > const &cursor, std::uint32_t ledgerSequence, std::uint32_t limit, bool outOfOrder, boost::asio::yield_context yield)
Fetches a page of ledger objects, ordered by key/index.
Definition BackendInterface.cpp:295
virtual std::optional< ripple::uint256 > doFetchSuccessorKey(ripple::uint256 key, std::uint32_t ledgerSequence, boost::asio::yield_context yield) const =0
Database-specific implementation of fetching the successor key.
virtual std::optional< std::uint32_t > doFetchLedgerObjectSeq(ripple::uint256 const &key, std::uint32_t sequence, boost::asio::yield_context yield) const =0
The database-specific implementation for fetching a ledger object sequence.
void updateRange(uint32_t newMax)
Updates the range of sequences that are stored in the DB.
Definition BackendInterface.cpp:256
virtual void writeLedger(ripple::LedgerHeader const &ledgerHeader, std::string &&blob)=0
Writes to a specific ledger.
virtual MPTHoldersAndCursor fetchMPTHolders(ripple::uint192 const &mptID, std::uint32_t const limit, std::optional< ripple::AccountID > const &cursorIn, std::uint32_t const ledgerSequence, boost::asio::yield_context yield) const =0
Fetches all holders' balances for a MPTIssuanceID.
LedgerCacheInterface const & cache() const
Definition BackendInterface.hpp:151
virtual std::optional< ripple::LedgerHeader > fetchLedgerByHash(ripple::uint256 const &hash, boost::asio::yield_context yield) const =0
Fetches a specific ledger by hash.
Represents a database timeout error.
Definition BackendInterface.hpp:40
char const * what() const override
Definition BackendInterface.hpp:46
Cache for an entire ledger.
Definition LedgerCacheInterface.hpp:21
A helper to notify Clio operator about a corruption in the DB.
Definition CorruptionDetector.hpp:16
A simple thread-safe logger for the channel specified in the constructor.
Definition Logger.hpp:77
Pump error(SourceLocationType const &loc=CURRENT_SRC_LOCATION) const
Interface for logging at Severity::ERR severity.
Definition Logger.cpp:498
This namespace implements the data access layer and related components.
Definition AmendmentCenter.cpp:56
auto synchronous(FnType &&func)
Synchronously executes the given function object inside a coroutine.
Definition BackendInterface.hpp:86
auto synchronousAndRetryOnTimeout(FnType &&func)
Synchronously execute the given function object and retry until no DatabaseTimeout is thrown.
Definition BackendInterface.hpp:117
auto retryOnTimeout(FnType func, size_t waitMs=kDEFAULT_WAIT_BETWEEN_RETRY)
A helper function that catches DatabaseTimeout exceptions and retries indefinitely.
Definition BackendInterface.hpp:63
void spawn(Ctx &&ctx, F &&func)
Spawns a coroutine using boost::asio::spawn.
Definition Spawn.hpp:53
Struct used to keep track of what to write to account_transactions/account_tx tables.
Definition DBHelpers.hpp:26
Represents a page of book offer objects.
Definition Types.hpp:41
Represents a page of LedgerObjects.
Definition Types.hpp:33
Represents an array of MPTokens.
Definition Types.hpp:235
Represents a bundle of NFTs with a cursor to the next page.
Definition Types.hpp:227
Represests a bundle of transactions with metadata and a cursor to the next page.
Definition Types.hpp:153