mirror of
https://github.com/XRPLF/clio.git
synced 2025-11-20 19:56:00 +00:00
@@ -185,12 +185,18 @@ toJson(ripple::STBase const& obj)
|
|||||||
}
|
}
|
||||||
|
|
||||||
std::pair<boost::json::object, boost::json::object>
|
std::pair<boost::json::object, boost::json::object>
|
||||||
toExpandedJson(data::TransactionAndMetadata const& blobs, NFTokenjson nftEnabled, std::optional<uint16_t> networkId)
|
toExpandedJson(
|
||||||
|
data::TransactionAndMetadata const& blobs,
|
||||||
|
std::uint32_t const apiVersion,
|
||||||
|
NFTokenjson nftEnabled,
|
||||||
|
std::optional<uint16_t> networkId
|
||||||
|
)
|
||||||
{
|
{
|
||||||
auto [txn, meta] = deserializeTxPlusMeta(blobs, blobs.ledgerSequence);
|
auto [txn, meta] = deserializeTxPlusMeta(blobs, blobs.ledgerSequence);
|
||||||
auto txnJson = toJson(*txn);
|
auto txnJson = toJson(*txn);
|
||||||
auto metaJson = toJson(*meta);
|
auto metaJson = toJson(*meta);
|
||||||
insertDeliveredAmount(metaJson, txn, meta, blobs.date);
|
insertDeliveredAmount(metaJson, txn, meta, blobs.date);
|
||||||
|
insertDeliverMaxAlias(txnJson, apiVersion);
|
||||||
|
|
||||||
if (nftEnabled == NFTokenjson::ENABLE) {
|
if (nftEnabled == NFTokenjson::ENABLE) {
|
||||||
Json::Value nftJson;
|
Json::Value nftJson;
|
||||||
@@ -246,6 +252,17 @@ insertDeliveredAmount(
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
insertDeliverMaxAlias(boost::json::object& txJson, std::uint32_t const apiVersion)
|
||||||
|
{
|
||||||
|
if (txJson.contains(JS(TransactionType)) and txJson.at(JS(TransactionType)).is_string() and
|
||||||
|
txJson.at(JS(TransactionType)).as_string() == JS(Payment) and txJson.contains(JS(Amount))) {
|
||||||
|
txJson[JS(DeliverMax)] = txJson[JS(Amount)];
|
||||||
|
if (apiVersion > 1)
|
||||||
|
txJson.erase(JS(Amount));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
boost::json::object
|
boost::json::object
|
||||||
toJson(ripple::TxMeta const& meta)
|
toJson(ripple::TxMeta const& meta)
|
||||||
{
|
{
|
||||||
@@ -456,7 +473,8 @@ traverseNFTObjects(
|
|||||||
if (!page) {
|
if (!page) {
|
||||||
if (nextPage == beast::zero) { // no nft objects in lastNFTPage
|
if (nextPage == beast::zero) { // no nft objects in lastNFTPage
|
||||||
return AccountCursor{beast::zero, 0};
|
return AccountCursor{beast::zero, 0};
|
||||||
} // marker is in the right range, but still invalid
|
}
|
||||||
|
// marker is in the right range, but still invalid
|
||||||
return Status{RippledError::rpcINVALID_PARAMS, "Invalid marker."};
|
return Status{RippledError::rpcINVALID_PARAMS, "Invalid marker."};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -73,10 +73,20 @@ deserializeTxPlusMeta(data::TransactionAndMetadata const& blobs, std::uint32_t s
|
|||||||
std::pair<boost::json::object, boost::json::object>
|
std::pair<boost::json::object, boost::json::object>
|
||||||
toExpandedJson(
|
toExpandedJson(
|
||||||
data::TransactionAndMetadata const& blobs,
|
data::TransactionAndMetadata const& blobs,
|
||||||
|
std::uint32_t apiVersion,
|
||||||
NFTokenjson nftEnabled = NFTokenjson::DISABLE,
|
NFTokenjson nftEnabled = NFTokenjson::DISABLE,
|
||||||
std::optional<uint16_t> networkId = std::nullopt
|
std::optional<uint16_t> networkId = std::nullopt
|
||||||
);
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Add "DeliverMax" which is the alias of "Amount" for "Payment" transaction to transaction json. Remove the
|
||||||
|
* "Amount" field when version is greater than 1
|
||||||
|
* @param txJson The transaction json object
|
||||||
|
* @param apiVersion The api version
|
||||||
|
*/
|
||||||
|
void
|
||||||
|
insertDeliverMaxAlias(boost::json::object& txJson, std::uint32_t apiVersion);
|
||||||
|
|
||||||
bool
|
bool
|
||||||
insertDeliveredAmount(
|
insertDeliveredAmount(
|
||||||
boost::json::object& metaJson,
|
boost::json::object& metaJson,
|
||||||
|
|||||||
@@ -165,7 +165,7 @@ AccountTxHandler::process(AccountTxHandler::Input input, Context const& ctx) con
|
|||||||
|
|
||||||
boost::json::object obj;
|
boost::json::object obj;
|
||||||
if (!input.binary) {
|
if (!input.binary) {
|
||||||
auto [txn, meta] = toExpandedJson(txnPlusMeta, NFTokenjson::ENABLE);
|
auto [txn, meta] = toExpandedJson(txnPlusMeta, ctx.apiVersion, NFTokenjson::ENABLE);
|
||||||
obj[JS(meta)] = std::move(meta);
|
obj[JS(meta)] = std::move(meta);
|
||||||
obj[JS(tx)] = std::move(txn);
|
obj[JS(tx)] = std::move(txn);
|
||||||
|
|
||||||
|
|||||||
@@ -67,7 +67,7 @@ LedgerHandler::process(LedgerHandler::Input input, Context const& ctx) const
|
|||||||
[&](auto obj) {
|
[&](auto obj) {
|
||||||
boost::json::object entry;
|
boost::json::object entry;
|
||||||
if (!input.binary) {
|
if (!input.binary) {
|
||||||
auto [txn, meta] = toExpandedJson(obj);
|
auto [txn, meta] = toExpandedJson(obj, ctx.apiVersion);
|
||||||
entry = std::move(txn);
|
entry = std::move(txn);
|
||||||
entry[JS(metaData)] = std::move(meta);
|
entry[JS(metaData)] = std::move(meta);
|
||||||
} else {
|
} else {
|
||||||
|
|||||||
@@ -106,7 +106,7 @@ NFTHistoryHandler::process(NFTHistoryHandler::Input input, Context const& ctx) c
|
|||||||
boost::json::object obj;
|
boost::json::object obj;
|
||||||
|
|
||||||
if (!input.binary) {
|
if (!input.binary) {
|
||||||
auto [txn, meta] = toExpandedJson(txnPlusMeta);
|
auto [txn, meta] = toExpandedJson(txnPlusMeta, ctx.apiVersion);
|
||||||
obj[JS(meta)] = std::move(meta);
|
obj[JS(meta)] = std::move(meta);
|
||||||
obj[JS(tx)] = std::move(txn);
|
obj[JS(tx)] = std::move(txn);
|
||||||
obj[JS(tx)].as_object()[JS(ledger_index)] = txnPlusMeta.ledgerSequence;
|
obj[JS(tx)].as_object()[JS(ledger_index)] = txnPlusMeta.ledgerSequence;
|
||||||
|
|||||||
@@ -47,7 +47,7 @@ TransactionEntryHandler::process(TransactionEntryHandler::Input input, Context c
|
|||||||
return Error{Status{RippledError::rpcTXN_NOT_FOUND, "transactionNotFound", "Transaction not found."}};
|
return Error{Status{RippledError::rpcTXN_NOT_FOUND, "transactionNotFound", "Transaction not found."}};
|
||||||
|
|
||||||
auto output = TransactionEntryHandler::Output{};
|
auto output = TransactionEntryHandler::Output{};
|
||||||
auto [txn, meta] = toExpandedJson(*dbRet);
|
auto [txn, meta] = toExpandedJson(*dbRet, ctx.apiVersion);
|
||||||
|
|
||||||
output.tx = std::move(txn);
|
output.tx = std::move(txn);
|
||||||
output.metadata = std::move(meta);
|
output.metadata = std::move(meta);
|
||||||
|
|||||||
@@ -142,7 +142,7 @@ public:
|
|||||||
return Error{Status{RippledError::rpcTXN_NOT_FOUND}};
|
return Error{Status{RippledError::rpcTXN_NOT_FOUND}};
|
||||||
}
|
}
|
||||||
|
|
||||||
auto const [txn, meta] = toExpandedJson(*dbResponse, NFTokenjson::ENABLE, currentNetId);
|
auto const [txn, meta] = toExpandedJson(*dbResponse, ctx.apiVersion, NFTokenjson::ENABLE, currentNetId);
|
||||||
|
|
||||||
if (!input.binary) {
|
if (!input.binary) {
|
||||||
output.tx = txn;
|
output.tx = txn;
|
||||||
|
|||||||
@@ -21,8 +21,11 @@
|
|||||||
#include <util/Fixtures.h>
|
#include <util/Fixtures.h>
|
||||||
#include <util/TestObject.h>
|
#include <util/TestObject.h>
|
||||||
|
|
||||||
|
#include <boost/json.hpp>
|
||||||
#include <fmt/core.h>
|
#include <fmt/core.h>
|
||||||
|
|
||||||
|
#include <array>
|
||||||
|
#include <string>
|
||||||
#include <variant>
|
#include <variant>
|
||||||
|
|
||||||
using namespace rpc;
|
using namespace rpc;
|
||||||
@@ -337,3 +340,81 @@ TEST_F(RPCHelpersTest, DecodeInvalidCTID)
|
|||||||
EXPECT_FALSE(decodeCTID('c'));
|
EXPECT_FALSE(decodeCTID('c'));
|
||||||
EXPECT_FALSE(decodeCTID(true));
|
EXPECT_FALSE(decodeCTID(true));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TEST_F(RPCHelpersTest, DeliverMaxAliasV1)
|
||||||
|
{
|
||||||
|
std::array<std::string, 3> const inputArray = {
|
||||||
|
R"({
|
||||||
|
"TransactionType": "Payment",
|
||||||
|
"Amount": {
|
||||||
|
"test": "test"
|
||||||
|
}
|
||||||
|
})",
|
||||||
|
R"({
|
||||||
|
"TransactionType": "OfferCreate",
|
||||||
|
"Amount": {
|
||||||
|
"test": "test"
|
||||||
|
}
|
||||||
|
})",
|
||||||
|
R"({
|
||||||
|
"TransactionType": "Payment",
|
||||||
|
"Amount1": {
|
||||||
|
"test": "test"
|
||||||
|
}
|
||||||
|
})"};
|
||||||
|
|
||||||
|
std::array<std::string, 3> outputArray = {
|
||||||
|
R"({
|
||||||
|
"TransactionType": "Payment",
|
||||||
|
"Amount": {
|
||||||
|
"test": "test"
|
||||||
|
},
|
||||||
|
"DeliverMax": {
|
||||||
|
"test": "test"
|
||||||
|
}
|
||||||
|
})",
|
||||||
|
R"({
|
||||||
|
"TransactionType": "OfferCreate",
|
||||||
|
"Amount": {
|
||||||
|
"test": "test"
|
||||||
|
}
|
||||||
|
})",
|
||||||
|
R"({
|
||||||
|
"TransactionType": "Payment",
|
||||||
|
"Amount1": {
|
||||||
|
"test": "test"
|
||||||
|
}
|
||||||
|
})"};
|
||||||
|
|
||||||
|
for (size_t i = 0; i < inputArray.size(); i++) {
|
||||||
|
auto req = boost::json::parse(inputArray[i]).as_object();
|
||||||
|
insertDeliverMaxAlias(req, 1);
|
||||||
|
EXPECT_EQ(req, boost::json::parse(outputArray[i]).as_object());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(RPCHelpersTest, DeliverMaxAliasV2)
|
||||||
|
{
|
||||||
|
auto req = boost::json::parse(
|
||||||
|
R"({
|
||||||
|
"TransactionType": "Payment",
|
||||||
|
"Amount": {
|
||||||
|
"test": "test"
|
||||||
|
}
|
||||||
|
})"
|
||||||
|
)
|
||||||
|
.as_object();
|
||||||
|
|
||||||
|
insertDeliverMaxAlias(req, 2);
|
||||||
|
EXPECT_EQ(
|
||||||
|
req,
|
||||||
|
boost::json::parse(
|
||||||
|
R"({
|
||||||
|
"TransactionType": "Payment",
|
||||||
|
"DeliverMax": {
|
||||||
|
"test": "test"
|
||||||
|
}
|
||||||
|
})"
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|||||||
@@ -1714,6 +1714,7 @@ generateTransactionTypeTestValues()
|
|||||||
"tx": {
|
"tx": {
|
||||||
"Account": "rf1BiGeXwwQoi8Z2ueFYTEXSwuJYfV2Jpn",
|
"Account": "rf1BiGeXwwQoi8Z2ueFYTEXSwuJYfV2Jpn",
|
||||||
"Amount": "1",
|
"Amount": "1",
|
||||||
|
"DeliverMax": "1",
|
||||||
"Destination": "rLEsXccBGNR3UPuPu2hUXPjziKC3qKSBun",
|
"Destination": "rLEsXccBGNR3UPuPu2hUXPjziKC3qKSBun",
|
||||||
"Fee": "1",
|
"Fee": "1",
|
||||||
"Sequence": 32,
|
"Sequence": 32,
|
||||||
@@ -1763,7 +1764,7 @@ generateTransactionTypeTestValues()
|
|||||||
},
|
},
|
||||||
"tx": {
|
"tx": {
|
||||||
"Account": "rf1BiGeXwwQoi8Z2ueFYTEXSwuJYfV2Jpn",
|
"Account": "rf1BiGeXwwQoi8Z2ueFYTEXSwuJYfV2Jpn",
|
||||||
"Amount": "1",
|
"DeliverMax": "1",
|
||||||
"Destination": "rLEsXccBGNR3UPuPu2hUXPjziKC3qKSBun",
|
"Destination": "rLEsXccBGNR3UPuPu2hUXPjziKC3qKSBun",
|
||||||
"Fee": "1",
|
"Fee": "1",
|
||||||
"Sequence": 32,
|
"Sequence": 32,
|
||||||
|
|||||||
@@ -468,6 +468,7 @@ TEST_F(RPCLedgerHandlerTest, TransactionsExpandNotBinary)
|
|||||||
{
|
{
|
||||||
"Account":"rf1BiGeXwwQoi8Z2ueFYTEXSwuJYfV2Jpn",
|
"Account":"rf1BiGeXwwQoi8Z2ueFYTEXSwuJYfV2Jpn",
|
||||||
"Amount":"100",
|
"Amount":"100",
|
||||||
|
"DeliverMax":"100",
|
||||||
"Destination":"rLEsXccBGNR3UPuPu2hUXPjziKC3qKSBun",
|
"Destination":"rLEsXccBGNR3UPuPu2hUXPjziKC3qKSBun",
|
||||||
"Fee":"3",
|
"Fee":"3",
|
||||||
"Sequence":30,
|
"Sequence":30,
|
||||||
@@ -695,6 +696,7 @@ TEST_F(RPCLedgerHandlerTest, OwnerFundsEmtpy)
|
|||||||
{
|
{
|
||||||
"Account":"rf1BiGeXwwQoi8Z2ueFYTEXSwuJYfV2Jpn",
|
"Account":"rf1BiGeXwwQoi8Z2ueFYTEXSwuJYfV2Jpn",
|
||||||
"Amount":"100",
|
"Amount":"100",
|
||||||
|
"DeliverMax":"100",
|
||||||
"Destination":"rLEsXccBGNR3UPuPu2hUXPjziKC3qKSBun",
|
"Destination":"rLEsXccBGNR3UPuPu2hUXPjziKC3qKSBun",
|
||||||
"Fee":"3",
|
"Fee":"3",
|
||||||
"Sequence":30,
|
"Sequence":30,
|
||||||
|
|||||||
@@ -288,6 +288,72 @@ TEST_F(RPCTxTest, DefaultParameter_API_v1)
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TEST_F(RPCTxTest, PaymentTx_API_v1)
|
||||||
|
{
|
||||||
|
auto const rawBackendPtr = dynamic_cast<MockBackend*>(mockBackendPtr.get());
|
||||||
|
ASSERT_NE(rawBackendPtr, nullptr);
|
||||||
|
|
||||||
|
TransactionAndMetadata tx;
|
||||||
|
tx.transaction = CreatePaymentTransactionObject(ACCOUNT, ACCOUNT2, 2, 3, 300).getSerializer().peekData();
|
||||||
|
tx.metadata = CreatePaymentTransactionMetaObject(ACCOUNT, ACCOUNT2, 110, 30).getSerializer().peekData();
|
||||||
|
tx.date = 123456;
|
||||||
|
tx.ledgerSequence = 100;
|
||||||
|
|
||||||
|
EXPECT_CALL(*rawBackendPtr, fetchTransaction(ripple::uint256{TXNID}, _)).WillOnce(Return(tx));
|
||||||
|
|
||||||
|
auto const rawETLPtr = dynamic_cast<MockETLService*>(mockETLServicePtr.get());
|
||||||
|
ASSERT_NE(rawETLPtr, nullptr);
|
||||||
|
EXPECT_CALL(*rawETLPtr, getETLState).WillOnce(Return(etl::ETLState{}));
|
||||||
|
|
||||||
|
runSpawn([this](auto yield) {
|
||||||
|
auto const handler = AnyHandler{TestTxHandler{mockBackendPtr, mockETLServicePtr}};
|
||||||
|
auto const req = json::parse(fmt::format(
|
||||||
|
R"({{
|
||||||
|
"command": "tx",
|
||||||
|
"transaction": "{}"
|
||||||
|
}})",
|
||||||
|
TXNID
|
||||||
|
));
|
||||||
|
auto const output = handler.process(req, Context{.yield = yield, .apiVersion = 1u});
|
||||||
|
ASSERT_TRUE(output);
|
||||||
|
EXPECT_TRUE(output->as_object().contains("DeliverMax"));
|
||||||
|
EXPECT_EQ(output->at("Amount"), output->at("DeliverMax"));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(RPCTxTest, PaymentTx_API_v2)
|
||||||
|
{
|
||||||
|
auto const rawBackendPtr = dynamic_cast<MockBackend*>(mockBackendPtr.get());
|
||||||
|
ASSERT_NE(rawBackendPtr, nullptr);
|
||||||
|
|
||||||
|
TransactionAndMetadata tx;
|
||||||
|
tx.transaction = CreatePaymentTransactionObject(ACCOUNT, ACCOUNT2, 2, 3, 300).getSerializer().peekData();
|
||||||
|
tx.metadata = CreatePaymentTransactionMetaObject(ACCOUNT, ACCOUNT2, 110, 30).getSerializer().peekData();
|
||||||
|
tx.date = 123456;
|
||||||
|
tx.ledgerSequence = 100;
|
||||||
|
|
||||||
|
EXPECT_CALL(*rawBackendPtr, fetchTransaction(ripple::uint256{TXNID}, _)).WillOnce(Return(tx));
|
||||||
|
|
||||||
|
auto const rawETLPtr = dynamic_cast<MockETLService*>(mockETLServicePtr.get());
|
||||||
|
ASSERT_NE(rawETLPtr, nullptr);
|
||||||
|
EXPECT_CALL(*rawETLPtr, getETLState).WillOnce(Return(etl::ETLState{}));
|
||||||
|
|
||||||
|
runSpawn([this](auto yield) {
|
||||||
|
auto const handler = AnyHandler{TestTxHandler{mockBackendPtr, mockETLServicePtr}};
|
||||||
|
auto const req = json::parse(fmt::format(
|
||||||
|
R"({{
|
||||||
|
"command": "tx",
|
||||||
|
"transaction": "{}"
|
||||||
|
}})",
|
||||||
|
TXNID
|
||||||
|
));
|
||||||
|
auto const output = handler.process(req, Context{.yield = yield, .apiVersion = 2u});
|
||||||
|
ASSERT_TRUE(output);
|
||||||
|
EXPECT_TRUE(output->as_object().contains("DeliverMax"));
|
||||||
|
EXPECT_FALSE(output->as_object().contains("Amount"));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
TEST_F(RPCTxTest, DefaultParameter_API_v2)
|
TEST_F(RPCTxTest, DefaultParameter_API_v2)
|
||||||
{
|
{
|
||||||
auto const rawBackendPtr = dynamic_cast<MockBackend*>(mockBackendPtr.get());
|
auto const rawBackendPtr = dynamic_cast<MockBackend*>(mockBackendPtr.get());
|
||||||
|
|||||||
Reference in New Issue
Block a user