20#include <xrpld/app/ledger/LedgerMaster.h> 
   21#include <xrpld/app/ledger/OpenLedger.h> 
   22#include <xrpld/app/misc/HashRouter.h> 
   23#include <xrpld/app/misc/Transaction.h> 
   24#include <xrpld/app/misc/TxQ.h> 
   25#include <xrpld/app/tx/apply.h> 
   26#include <xrpld/rpc/Context.h> 
   27#include <xrpld/rpc/DeliveredAmount.h> 
   28#include <xrpld/rpc/GRPCHandlers.h> 
   29#include <xrpld/rpc/MPTokenIssuanceID.h> 
   30#include <xrpld/rpc/detail/TransactionSign.h> 
   32#include <xrpl/protocol/ErrorCodes.h> 
   33#include <xrpl/protocol/NFTSyntheticSerializer.h> 
   34#include <xrpl/protocol/RPCErr.h> 
   35#include <xrpl/protocol/STParsedJSON.h> 
   36#include <xrpl/resource/Fees.h> 
   40static Expected<std::uint32_t, Json::Value>
 
   44    bool const hasTicketSeq = tx_json.
isMember(sfTicketSequence.jsonName);
 
   45    auto const& accountStr = tx_json[jss::Account];
 
   46    if (!accountStr.isString())
 
   53    auto const srcAddressID = parseBase58<AccountID>(accountStr.asString());
 
   54    if (!srcAddressID.has_value())
 
   62    if (!hasTicketSeq && !sle)
 
   65            << 
"Failed to find source account " 
   66            << 
"in current ledger: " << 
toBase58(*srcAddressID);
 
 
   77    if (!sigObject.
isMember(jss::SigningPubKey))
 
   80        sigObject[jss::SigningPubKey] = 
"";
 
   83    if (sigObject.
isMember(jss::Signers))
 
   85        if (!sigObject[jss::Signers].isArray())
 
   88        for (
unsigned index = 0; index < sigObject[jss::Signers].
size();
 
   91            auto& signer = sigObject[jss::Signers][index];
 
   92            if (!signer.isObject() || !signer.isMember(jss::Signer) ||
 
   93                !signer[jss::Signer].isObject())
 
   97            if (!signer[jss::Signer].isMember(jss::SigningPubKey))
 
  100                signer[jss::Signer][jss::SigningPubKey] = 
"";
 
  103            if (!signer[jss::Signer].isMember(jss::TxnSignature))
 
  106                signer[jss::Signer][jss::TxnSignature] = 
"";
 
  108            else if (signer[jss::Signer][jss::TxnSignature] != 
"")
 
  116    if (!sigObject.
isMember(jss::TxnSignature))
 
  119        sigObject[jss::TxnSignature] = 
"";
 
  121    else if (sigObject[jss::TxnSignature] != 
"")
 
 
  144        if (feeOrError.isMember(jss::error))
 
  146        tx_json[jss::Fee] = feeOrError;
 
  152    if (!tx_json.
isMember(jss::Sequence))
 
  157        tx_json[sfSequence.jsonName] = *seq;
 
  160    if (!tx_json.
isMember(jss::NetworkID))
 
  163        if (networkId > 1024)
 
  164            tx_json[jss::NetworkID] = 
to_string(networkId);
 
 
  180                "Can only include one of `tx_blob` and `tx_json`.");
 
  183        auto const tx_blob = params[jss::tx_blob];
 
  184        if (!tx_blob.isString())
 
  189        auto unHexed = 
strUnHex(tx_blob.asString());
 
  190        if (!unHexed || unHexed->empty())
 
  204    else if (params.
isMember(jss::tx_json))
 
  206        tx_json = params[jss::tx_json];
 
  215            "Neither `tx_blob` nor `tx_json` included.");
 
  219    if (!tx_json.
isMember(jss::TransactionType))
 
  224    if (!tx_json.
isMember(jss::Account))
 
 
  241        transaction->getSTransaction(),
 
  245    jvResult[jss::applied] = result.applied;
 
  246    jvResult[jss::ledger_index] = view.
seq();
 
  248    bool const isBinaryOutput = context.
params.
get(jss::binary, 
false).
asBool();
 
  256        jvResult[jss::engine_result] = token;
 
  257        jvResult[jss::engine_result_code] = result.ter;
 
  258        jvResult[jss::engine_result_message] = message;
 
  264        jvResult[jss::engine_result] = 
"unknown";
 
  265        jvResult[jss::engine_result_code] = result.ter;
 
  266        jvResult[jss::engine_result_message] = 
"unknown";
 
  270    if (token == 
"tesSUCCESS")
 
  272        jvResult[jss::engine_result_message] =
 
  273            "The simulated transaction would have been applied.";
 
  280            auto const metaBlob =
 
  281                result.metadata->getAsObject().getSerializer().getData();
 
  290                transaction->getSTransaction(),
 
  293                jvResult, transaction->getSTransaction(), *result.metadata);
 
  296                transaction->getSTransaction(),
 
  304            transaction->getSTransaction()->getSerializer().getData();
 
 
  333    for (
auto const field :
 
  334         {jss::secret, jss::seed, jss::seed_hex, jss::passphrase})
 
  348    if (
auto error = 
autofillTx(tx_json, context))
 
  352    if (!parsed.
object.has_value())
 
  363        jvResult[jss::error] = 
"invalidTransaction";
 
  364        jvResult[jss::error_exception] = e.
what();
 
  368    if (stTx->getTxnType() == ttBATCH)
 
  384        jvResult[jss::error] = 
"internalSimulate";
 
  385        jvResult[jss::error_exception] = e.
what();
 
 
UInt size() const
Number of values in array or object.
 
bool isMember(char const *key) const
Return true if the object has a member named key.
 
Value get(UInt index, Value const &defaultValue) const
If the array contains at least index+1 elements, returns the element value, otherwise returns default...
 
virtual Config & config()=0
 
virtual LoadFeeTrack & getFeeTrack()=0
 
virtual OpenLedger & openLedger()=0
 
virtual beast::Journal journal(std::string const &name)=0
 
std::shared_ptr< OpenView const > current() const
Returns a view to the current open ledger.
 
Writable ledger view that accumulates state and tx changes.
 
LedgerIndex seq() const
Returns the sequence number of the base ledger.
 
Json::Value getJson(JsonOptions=JsonOptions::none) const override
 
Holds the serialized result of parsing an input JSON object.
 
std::optional< STObject > object
The STObject if the parse was successful.
 
Json::Value error
On failure, an appropriate set of error values.
 
constexpr std::uint32_t value() const
 
SeqProxy nextQueuableSeq(std::shared_ptr< SLE const > const &sleAccount) const
Return the next sequence that would go in the TxQ for an account.
 
ApplyResult apply(Application &app, OpenView &view, std::shared_ptr< STTx const > const &tx, ApplyFlags flags, beast::Journal j)
Add a new transaction to the open ledger, hold it in the queue, or reject it.
 
@ objectValue
object value (collection of name/value pairs).
 
Json::Value make_error(error_code_i code)
Returns a new json object that reflects the error code.
 
Json::Value invalid_field_error(std::string const &name)
 
void insertNFTSyntheticInJson(Json::Value &, std::shared_ptr< STTx const > const &, TxMeta const &)
Adds common synthetic fields to transaction-related JSON responses.
 
Json::Value make_param_error(std::string const &message)
Returns a new json object that indicates invalid parameters.
 
void insertMPTokenIssuanceID(Json::Value &response, std::shared_ptr< STTx const > const &transaction, TxMeta const &transactionMeta)
 
std::string invalid_field_message(std::string const &name)
 
Json::Value getCurrentNetworkFee(Role const role, Config const &config, LoadFeeTrack const &feeTrack, TxQ const &txQ, Application const &app, Json::Value const &tx, int mult, int div)
 
Json::Value object_field_error(std::string const &name)
 
void insertDeliveredAmount(Json::Value &meta, ReadView const &, std::shared_ptr< STTx const > const &serializedTx, TxMeta const &)
Add a delivered_amount field to the meta input/output parameter.
 
Json::Value missing_field_error(std::string const &name)
 
Charge const feeMediumBurdenRPC
 
Keylet account(AccountID const &id) noexcept
AccountID root.
 
Use hash_* containers for keys that do not need a cryptographically secure hashing algorithm.
 
std::string toBase58(AccountID const &v)
Convert AccountID to base58 checked string.
 
std::optional< Blob > strUnHex(std::size_t strSize, Iterator begin, Iterator end)
 
static std::optional< Json::Value > autofillTx(Json::Value &tx_json, RPC::JsonContext &context)
 
static Expected< std::uint32_t, Json::Value > getAutofillSequence(Json::Value const &tx_json, RPC::JsonContext &context)
 
Json::Value doSimulate(RPC::JsonContext &)
 
static Json::Value getTxJsonFromParams(Json::Value const ¶ms)
 
Json::Value rpcError(int iError)
 
std::string strHex(FwdIt begin, FwdIt end)
 
std::enable_if_t< std::is_same< T, char >::value||std::is_same< T, unsigned char >::value, Slice > makeSlice(std::array< T, N > const &a)
 
static Json::Value simulateTxn(RPC::JsonContext &context, std::shared_ptr< Transaction > transaction)
 
std::string to_string(base_uint< Bits, Tag > const &a)
 
static std::optional< Json::Value > autofillSignature(Json::Value &sigObject)
 
bool transResultInfo(TER code, std::string &token, std::string &text)
 
Resource::Charge & loadType