20#include <xrpld/app/ledger/LedgerMaster.h>
21#include <xrpld/app/main/Application.h>
22#include <xrpld/app/misc/DeliverMax.h>
23#include <xrpld/app/misc/Transaction.h>
24#include <xrpld/app/rdb/backend/SQLiteDatabase.h>
25#include <xrpld/ledger/ReadView.h>
26#include <xrpld/rpc/Context.h>
27#include <xrpld/rpc/DeliveredAmount.h>
28#include <xrpld/rpc/MPTokenIssuanceID.h>
29#include <xrpld/rpc/Role.h>
31#include <xrpl/json/json_value.h>
32#include <xrpl/protocol/ErrorCodes.h>
33#include <xrpl/protocol/NFTSyntheticSerializer.h>
34#include <xrpl/protocol/RPCErr.h>
35#include <xrpl/protocol/UintTypes.h>
36#include <xrpl/protocol/jss.h>
37#include <xrpl/resource/Fees.h>
59 if ((params.
isMember(jss::ledger_index_min) ||
60 params.
isMember(jss::ledger_index_max)) &&
61 (params.
isMember(jss::ledger_hash) ||
65 status.inject(response);
69 if (params.
isMember(jss::ledger_index_min) ||
70 params.
isMember(jss::ledger_index_max))
72 uint32_t min = params.
isMember(jss::ledger_index_min) &&
73 params[jss::ledger_index_min].
asInt() >= 0
74 ? params[jss::ledger_index_min].
asUInt()
76 uint32_t max = params.
isMember(jss::ledger_index_max) &&
77 params[jss::ledger_index_max].
asInt() >= 0
78 ? params[jss::ledger_index_max].
asUInt()
83 else if (params.
isMember(jss::ledger_hash))
85 auto& hashValue = params[jss::ledger_hash];
86 if (!hashValue.isString())
89 status.inject(response);
94 if (!hash.
parseHex(hashValue.asString()))
97 status.inject(response);
102 else if (params.
isMember(jss::ledger_index))
105 if (params[jss::ledger_index].isNumeric())
106 ledger = params[jss::ledger_index].
asUInt();
111 if (ledgerStr ==
"current" || ledgerStr.
empty())
112 ledger = LedgerShortcut::CURRENT;
113 else if (ledgerStr ==
"closed")
114 ledger = LedgerShortcut::CLOSED;
115 else if (ledgerStr ==
"validated")
116 ledger = LedgerShortcut::VALIDATED;
121 status.inject(response);
156 if constexpr (std::is_same_v<T, LedgerRange>)
163 if ((ls.max > uValidatedMax && ls.max != -1) ||
164 (ls.min < uValidatedMin && ls.min != 0))
169 if (ls.min > uValidatedMin)
173 if (ls.max < uValidatedMax)
177 if (uLedgerMax < uLedgerMin)
187 auto const status = getLedger(ledgerView, ls, context);
196 if (!validated || ledgerView->info().seq > uValidatedMax ||
197 ledgerView->info().seq < uValidatedMin)
201 uLedgerMin = uLedgerMax = ledgerView->info().seq;
221 if (
auto stat = std::get_if<RPC::Status>(&lgrRange))
224 return {result, *stat};
227 result.
ledgerRange = std::get<LedgerRange>(lgrRange);
243 Throw<std::runtime_error>(
"Failed to get relational database");
249 auto [tx, marker] = db->oldestAccountTxPageB(options);
255 auto [tx, marker] = db->newestAccountTxPageB(options);
264 auto [tx, marker] = db->oldestAccountTxPage(options);
270 auto [tx, marker] = db->newestAccountTxPage(options);
277 JLOG(context.
j.
debug()) << __func__ <<
" : finished";
292 error.inject(response);
297 response[jss::validated] =
true;
298 response[jss::limit] = result.
limit;
305 if (
auto txnsData = std::get_if<TxnsData>(&result.
transactions))
309 "ripple::populateJsonResponse : binary is not set");
311 for (
auto const& [txn, txnMeta] : *txnsData)
316 jvObj[jss::validated] =
true;
319 (context.
apiVersion > 1 ? jss::tx_json : jss::tx);
322 jvObj[json_tx] = txn->getJson(
326 jvObj[jss::hash] =
to_string(txn->getID());
327 jvObj[jss::ledger_index] = txn->getLedger();
328 jvObj[jss::ledger_hash] =
335 jvObj[jss::close_time_iso] =
342 auto const& sttx = txn->getSTransaction();
344 jvObj[json_tx], sttx->getTxnType(), context.
apiVersion);
349 insertDeliveredAmount(
350 jvObj[jss::meta], context, txn, *txnMeta);
353 jvObj[jss::meta], sttx, *txnMeta);
357 "ripple::populateJsonResponse : missing "
358 "transaction medatata");
365 args.
binary,
"ripple::populateJsonResponse : binary is set");
367 for (
auto const& binaryData :
372 jvObj[jss::tx_blob] =
strHex(std::get<0>(binaryData));
373 auto const json_meta =
374 (context.
apiVersion > 1 ? jss::meta_blob : jss::meta);
375 jvObj[json_meta] =
strHex(std::get<1>(binaryData));
376 jvObj[jss::ledger_index] = std::get<2>(binaryData);
377 jvObj[jss::validated] =
true;
384 response[jss::marker][jss::ledger] = result.
marker->ledgerSeq;
385 response[jss::marker][jss::seq] = result.
marker->txnSeq;
389 JLOG(context.
j.
debug()) << __func__ <<
" : finished";
409 auto& params = context.
params;
417 if (context.
apiVersion > 1u && params.isMember(jss::binary) &&
418 !params[jss::binary].isBool())
422 if (context.
apiVersion > 1u && params.isMember(jss::forward) &&
423 !params[jss::forward].isBool())
428 args.
limit = params.isMember(jss::limit) ? params[jss::limit].asUInt() : 0;
429 args.
binary = params.isMember(jss::binary) && params[jss::binary].asBool();
431 params.isMember(jss::forward) && params[jss::forward].asBool();
433 if (!params.isMember(jss::account))
436 if (!params[jss::account].isString())
440 parseBase58<AccountID>(params[jss::account].asString());
447 if (
auto jv = std::get_if<Json::Value>(&parseRes))
453 args.
ledger = std::get<std::optional<LedgerSpecifier>>(parseRes);
456 if (params.isMember(jss::marker))
458 auto& token = params[jss::marker];
459 if (!token.isMember(jss::ledger) || !token.isMember(jss::seq) ||
465 "invalid marker. Provide ledger index via ledger field, and "
466 "transaction sequence number via seq field"};
467 status.inject(response);
470 args.
marker = {token[jss::ledger].asUInt(), token[jss::seq].asUInt()};
474 JLOG(context.
j.
debug()) << __func__ <<
" populating response";
Value & append(const Value &value)
Append value to array at the end.
std::string asString() const
Returns the unquoted string value.
bool isMember(const char *key) const
Return true if the object has a member named key.
virtual Config & config()=0
virtual RelationalDatabase & getRelationalDatabase()=0
std::optional< NetClock::time_point > getCloseTimeBySeq(LedgerIndex ledgerIndex)
bool isValidated(ReadView const &ledger)
uint256 getHashBySeq(std::uint32_t index)
Get a ledger's hash by sequence number using the cache.
bool getValidatedRange(std::uint32_t &minVal, std::uint32_t &maxVal)
std::vector< txnMetaLedgerType > MetaTxsList
std::vector< AccountTx > AccountTxs
RPC::LedgerShortcut LedgerShortcut
std::tuple< Blob, Blob, std::uint32_t > txnMetaLedgerType
std::variant< LedgerRange, LedgerShortcut, LedgerSequence, LedgerHash > LedgerSpecifier
constexpr bool parseHex(std::string_view sv)
Parse a hex string into a base_uint.
@ arrayValue
array value (ordered list)
@ objectValue
object value (collection of name/value pairs).
@ uintValue
unsigned integer value
Json::Value invalid_field_error(std::string const &name)
void insertMPTokenIssuanceID(Json::Value &response, std::shared_ptr< STTx const > const &transaction, TxMeta const &transactionMeta)
void insertDeliverMax(Json::Value &tx_json, TxType txnType, unsigned int apiVersion)
Copy Amount field to DeliverMax field in transaction output JSON.
Json::Value missing_field_error(std::string const &name)
Charge const feeMediumBurdenRPC
Use hash_* containers for keys that do not need a cryptographically secure hashing algorithm.
std::variant< LedgerRange, RPC::Status > getLedgerRange(RPC::Context &context, std::optional< LedgerSpecifier > const &ledgerSpecifier)
void insertNFTSyntheticInJson(Json::Value &, std::shared_ptr< STTx const > const &, TxMeta const &)
Adds common synthetic fields to transaction-related JSON responses.
Json::Value doAccountTxJson(RPC::JsonContext &context)
Json::Value rpcError(int iError)
bool isUnlimited(Role const &role)
ADMIN and IDENTIFIED roles shall have unlimited resources.
std::string strHex(FwdIt begin, FwdIt end)
std::string to_string_iso(date::sys_time< Duration > tp)
Json::Value populateJsonResponse(std::pair< AccountTxResult, RPC::Status > const &res, AccountTxArgs const &args, RPC::JsonContext const &context)
std::string to_string(base_uint< Bits, Tag > const &a)
std::variant< std::optional< LedgerSpecifier >, Json::Value > parseLedgerArgs(RPC::Context &context, Json::Value const ¶ms)
std::pair< AccountTxResult, RPC::Status > doAccountTxHelp(RPC::Context &context, AccountTxArgs const &args)
The context of information needed to call an RPC.
Resource::Charge & loadType
LedgerMaster & ledgerMaster
Status represents the results of an operation that might fail.
std::optional< AccountTxMarker > marker
std::optional< LedgerSpecifier > ledger
std::optional< AccountTxMarker > marker
std::variant< AccountTxs, MetaTxsList > transactions