1#include <xrpld/app/ledger/LedgerMaster.h>
2#include <xrpld/app/main/Application.h>
3#include <xrpld/app/misc/DeliverMax.h>
4#include <xrpld/app/misc/Transaction.h>
5#include <xrpld/app/rdb/backend/SQLiteDatabase.h>
6#include <xrpld/rpc/Context.h>
7#include <xrpld/rpc/DeliveredAmount.h>
8#include <xrpld/rpc/MPTokenIssuanceID.h>
9#include <xrpld/rpc/Role.h>
10#include <xrpld/rpc/Status.h>
11#include <xrpld/rpc/detail/RPCHelpers.h>
12#include <xrpld/rpc/detail/RPCLedgerHelpers.h>
13#include <xrpld/rpc/detail/Tuning.h>
15#include <xrpl/basics/Log.h>
16#include <xrpl/basics/base_uint.h>
17#include <xrpl/basics/chrono.h>
18#include <xrpl/basics/strHex.h>
19#include <xrpl/beast/utility/instrumentation.h>
20#include <xrpl/json/json_value.h>
21#include <xrpl/ledger/ReadView.h>
22#include <xrpl/protocol/AccountID.h>
23#include <xrpl/protocol/ErrorCodes.h>
24#include <xrpl/protocol/LedgerShortcut.h>
25#include <xrpl/protocol/NFTSyntheticSerializer.h>
26#include <xrpl/protocol/RPCErr.h>
27#include <xrpl/protocol/RippleLedgerHash.h>
28#include <xrpl/protocol/jss.h>
29#include <xrpl/rdb/RelationalDatabase.h>
30#include <xrpl/resource/Fees.h>
57 if ((params.
isMember(jss::ledger_index_min) || params.
isMember(jss::ledger_index_max)) &&
61 status.inject(response);
65 if (params.
isMember(jss::ledger_index_min) || params.
isMember(jss::ledger_index_max))
68 params.
isMember(jss::ledger_index_min) && params[jss::ledger_index_min].
asInt() >= 0
69 ? params[jss::ledger_index_min].
asUInt()
72 params.
isMember(jss::ledger_index_max) && params[jss::ledger_index_max].
asInt() >= 0
73 ? params[jss::ledger_index_max].
asUInt()
78 if (params.
isMember(jss::ledger_hash))
80 auto& hashValue = params[jss::ledger_hash];
81 if (!hashValue.isString())
84 status.inject(response);
89 if (!hash.
parseHex(hashValue.asString()))
92 status.inject(response);
97 if (params.
isMember(jss::ledger_index))
100 if (params[jss::ledger_index].isNumeric())
102 ledger = params[jss::ledger_index].
asUInt();
108 if (ledgerStr ==
"current" || ledgerStr.
empty())
112 else if (ledgerStr ==
"closed")
116 else if (ledgerStr ==
"validated")
123 status.inject(response);
162 if ((ls.max > uValidatedMax && ls.max != -1) ||
163 (ls.min < uValidatedMin && ls.min != 0))
168 if (ls.min > uValidatedMin)
172 if (ls.max < uValidatedMax)
176 if (uLedgerMax < uLedgerMin)
186 auto status = getLedger(ledgerView, ls, context);
194 if (!validated || ledgerView->header().seq > uValidatedMax ||
195 ledgerView->header().seq < uValidatedMin)
199 uLedgerMin = uLedgerMax = ledgerView->header().seq;
208 return LedgerRange{.min = uLedgerMin, .max = uLedgerMax};
222 return {result, *stat};
225 result.
ledgerRange = std::get<LedgerRange>(lgrRange);
242 auto [tx, marker] = db.oldestAccountTxPageB(options);
248 auto [tx, marker] = db.newestAccountTxPageB(options);
257 auto [tx, marker] = db.oldestAccountTxPage(options);
263 auto [tx, marker] = db.newestAccountTxPage(options);
270 JLOG(context.
j.
debug()) << __func__ <<
" : finished";
290 response[jss::validated] =
true;
291 response[jss::limit] = result.
limit;
300 XRPL_ASSERT(!args.
binary,
"xrpl::populateJsonResponse : binary is not set");
302 for (
auto const& [txn, txnMeta] : *txnsData)
307 jvObj[jss::validated] =
true;
309 auto const jsonTx = (context.
apiVersion > 1 ? jss::tx_json : jss::tx);
312 jvObj[jsonTx] = txn->getJson(
318 jvObj[jss::hash] =
to_string(txn->getID());
319 jvObj[jss::ledger_index] = txn->getLedger();
320 jvObj[jss::ledger_hash] =
325 jvObj[jss::close_time_iso] =
toStringIso(*closeTime);
332 auto const& sttx = txn->getSTransaction();
337 insertDeliveredAmount(jvObj[jss::meta], context, txn, *txnMeta);
345 "xrpl::populateJsonResponse : missing "
346 "transaction metadata");
354 XRPL_ASSERT(args.
binary,
"xrpl::populateJsonResponse : binary is set");
356 for (
auto const& binaryData : std::get<TxnsDataBinary>(result.
transactions))
360 jvObj[jss::tx_blob] =
strHex(std::get<0>(binaryData));
361 auto const jsonMeta = (context.
apiVersion > 1 ? jss::meta_blob : jss::meta);
362 jvObj[jsonMeta] =
strHex(std::get<1>(binaryData));
363 jvObj[jss::ledger_index] = std::get<2>(binaryData);
364 jvObj[jss::validated] =
true;
371 response[jss::marker][jss::ledger] = result.
marker->ledgerSeq;
372 response[jss::marker][jss::seq] = result.
marker->txnSeq;
376 JLOG(context.
j.
debug()) << __func__ <<
" : finished";
396 auto& params = context.
params;
404 if (context.
apiVersion > 1u && params.isMember(jss::binary) && !params[jss::binary].isBool())
408 if (context.
apiVersion > 1u && params.isMember(jss::forward) && !params[jss::forward].isBool())
416 args.
binary = params.isMember(jss::binary) && params[jss::binary].asBool();
417 args.
forward = params.isMember(jss::forward) && params[jss::forward].asBool();
419 if (!params.isMember(jss::account))
422 if (!params[jss::account].isString())
437 args.
ledger = std::get<std::optional<LedgerSpecifier>>(parseRes);
439 if (params.isMember(jss::marker))
441 auto& token = params[jss::marker];
442 if (!token.isMember(jss::ledger) || !token.isMember(jss::seq) ||
448 "invalid marker. Provide ledger index via ledger field, and "
449 "transaction sequence number via seq field"};
450 status.inject(response);
454 .ledgerSeq = token[jss::ledger].asUInt(), .txnSeq = token[jss::seq].asUInt()};
458 JLOG(context.
j.
debug()) << __func__ <<
" populating response";
Value & append(Value const &value)
Append value to array at the end.
std::string asString() const
Returns the unquoted string value.
bool isMember(char const *key) const
Return true if the object has a member named key.
virtual Config & config()=0
constexpr bool parseHex(std::string_view sv)
Parse a hex string into a base_uint.
bool getValidatedRange(std::uint32_t &minVal, std::uint32_t &maxVal)
uint256 getHashBySeq(std::uint32_t index)
Get a ledger's hash by sequence number using the cache.
std::optional< NetClock::time_point > getCloseTimeBySeq(LedgerIndex ledgerIndex)
bool isValidated(ReadView const &ledger)
std::vector< AccountTx > AccountTxs
std::vector< txnMetaLedgerType > MetaTxsList
std::variant< LedgerRange, LedgerShortcut, LedgerSequence, LedgerHash > LedgerSpecifier
std::tuple< Blob, Blob, std::uint32_t > txnMetaLedgerType
virtual RelationalDatabase & getRelationalDatabase()=0
@ UInt
unsigned integer value
@ Array
array value (ordered list)
@ Object
object value (collection of name/value pairs).
static constexpr LimitRange kAccountTx
Limits for the account_tx command.
void insertMPTokenIssuanceID(json::Value &response, std::shared_ptr< STTx const > const &transaction, TxMeta const &transactionMeta)
void insertDeliverMax(json::Value &txJson, TxType txnType, unsigned int apiVersion)
Copy Amount field to DeliverMax field in transaction output JSON.
void insertNFTSyntheticInJson(json::Value &, std::shared_ptr< STTx const > const &, TxMeta const &)
Adds common synthetic fields to transaction-related JSON responses.
json::Value invalidFieldError(std::string const &name)
std::optional< json::Value > readLimitField(unsigned int &limit, Tuning::LimitRange const &range, JsonContext const &context)
Retrieves the limit value from a JsonContext or sets a default.
json::Value missingFieldError(std::string const &name)
Charge const kFeeMediumBurdenRpc
Use hash_* containers for keys that do not need a cryptographically secure hashing algorithm.
std::variant< std::optional< LedgerSpecifier >, json::Value > parseLedgerArgs(RPC::Context &context, json::Value const ¶ms)
RelationalDatabase::AccountTxs TxnsData
RelationalDatabase::AccountTxResult AccountTxResult
std::optional< AccountID > parseBase58(std::string const &s)
Parse AccountID from checked, base58 string.
json::Value doAccountTx(RPC::JsonContext &context)
std::string strHex(FwdIt begin, FwdIt end)
RelationalDatabase::MetaTxsList TxnsDataBinary
RelationalDatabase::txnMetaLedgerType TxnDataBinary
@ Closed
The most recently closed ledger (may not be validated).
@ Current
The current working ledger (open, not yet closed).
@ Validated
The most recently validated ledger.
std::string to_string(BaseUInt< Bits, Tag > const &a)
std::string toStringIso(date::sys_time< Duration > tp)
json::Value rpcError(ErrorCodeI iError)
std::variant< LedgerRange, RPC::Status > getLedgerRange(RPC::Context &context, std::optional< LedgerSpecifier > const &ledgerSpecifier)
json::Value populateJsonResponse(std::pair< AccountTxResult, RPC::Status > const &res, AccountTxArgs const &args, RPC::JsonContext const &context)
RelationalDatabase::AccountTxArgs AccountTxArgs
std::pair< AccountTxResult, RPC::Status > doAccountTxHelp(RPC::Context &context, AccountTxArgs const &args)
RelationalDatabase::LedgerSpecifier LedgerSpecifier
bool isUnlimited(Role const &role)
ADMIN and IDENTIFIED roles shall have unlimited resources.
unsigned int underlying_t
The context of information needed to call an RPC.
Resource::Charge & loadType
LedgerMaster & ledgerMaster
Status represents the results of an operation that might fail.
static constexpr Code kOK
void inject(json::Value &object) const
Apply the Status to a JsonObject.
ErrorCodeI toErrorCode() const
Returns the Status as an error_code_i.
std::optional< AccountTxMarker > marker
std::optional< LedgerSpecifier > ledger
std::variant< AccountTxs, MetaTxsList > transactions
std::optional< AccountTxMarker > marker