20 #include <ripple/beast/unit_test.h>
21 #include <ripple/protocol/ErrorCodes.h>
22 #include <ripple/protocol/jss.h>
25 #include <boost/container/flat_set.hpp>
39 boost::container::flat_set<std::string>
created;
40 boost::container::flat_set<std::string>
deleted;
41 boost::container::flat_set<std::string>
modified;
51 auto buildSet = [](
auto&& init) {
52 boost::container::flat_set<std::string> r;
53 r.reserve(init.size());
69 BEAST_EXPECT(
txNode[jss::validated].asBool() ==
true);
75 boost::container::flat_set<std::string> createdNodes;
76 boost::container::flat_set<std::string> deletedNodes;
77 boost::container::flat_set<std::string> modifiedNodes;
99 "Unexpected or unlabeled node type in metadata.",
104 BEAST_EXPECT(createdNodes == sane.
created);
105 BEAST_EXPECT(deletedNodes == sane.
deleted);
106 BEAST_EXPECT(modifiedNodes == sane.
modified);
112 using namespace test::jtx;
123 return j.isMember(jss::result) &&
124 (j[jss::result][jss::status] ==
"success") &&
125 (j[jss::result][jss::transactions].size() == 2) &&
126 (j[jss::result][jss::transactions][0u][jss::tx]
127 [jss::TransactionType] == jss::AccountSet) &&
128 (j[jss::result][jss::transactions][1u][jss::tx]
129 [jss::TransactionType] == jss::Payment);
133 return j.isMember(jss::result) &&
134 (j[jss::result][jss::status] ==
"success") &&
135 (j[jss::result][jss::transactions].size() == 0);
140 j[jss::result].
isMember(jss::error) &&
145 jParms[jss::api_version] = apiVersion;
153 jParms[jss::account] =
"0xDEADBEEF";
159 jParms[jss::account] = A1.human();
161 hasTxs(env.
rpc(
"json",
"account_tx",
to_string(jParms))));
166 p[jss::ledger_index_min] = -1;
167 p[jss::ledger_index_max] = -1;
171 p[jss::ledger_index_min] = 0;
172 p[jss::ledger_index_max] = 100;
176 p[jss::ledger_index_min] = 1;
177 p[jss::ledger_index_max] = 2;
181 p[jss::ledger_index_min] = 2;
182 p[jss::ledger_index_max] = 1;
193 p[jss::ledger_index_min] = -1;
197 p[jss::ledger_index_min] = 1;
201 p[jss::ledger_index_min] = env.
current()->info().seq;
212 p[jss::ledger_index_max] = -1;
216 p[jss::ledger_index_max] = env.
current()->info().seq;
220 p[jss::ledger_index_max] = 3;
224 p[jss::ledger_index_max] = env.
closed()->info().seq;
228 p[jss::ledger_index_max] = env.
closed()->info().seq - 1;
237 p[jss::ledger_index] = env.
closed()->info().seq;
241 p[jss::ledger_index] = env.
closed()->info().seq - 1;
245 p[jss::ledger_index] = env.
current()->info().seq;
250 p[jss::ledger_index] = env.
current()->info().seq + 1;
264 p[jss::ledger_hash] =
275 jParms[jss::account] =
"0xDEADBEEF";
276 jParms[jss::account] = A1.human();
279 p[jss::ledger_index_max] = -1;
280 p[jss::ledger_index_min] = -1;
281 p[jss::ledger_index] = -1;
291 p[jss::ledger_index_max] = 100;
292 p[jss::ledger_index_min] = 0;
297 p[jss::ledger_index_max] = -1;
298 p[jss::ledger_index_min] = -1;
302 p[jss::ledger_index_min] = 2;
303 p[jss::ledger_index_max] = 1;
312 p[jss::ledger_index_max] = env.
current()->info().seq;
325 using namespace test::jtx;
326 using namespace std::chrono_literals;
332 auto const USD{gw[
"USD"]};
334 env.
fund(
XRP(1000000), alice, gw);
341 env(pay(alice, gw,
XRP(100)));
348 env(trust(alice, USD(200)),
sig(alie));
350 env(offer(alice, USD(50),
XRP(150)),
sig(alie));
357 env(signers(alice, 1, {{
"bogie", 1}, {
"demon", 1}}),
sig(alie));
362 auto escrow = [](
Account const& account,
366 escro[jss::TransactionType] = jss::EscrowCreate;
368 escro[jss::Account] = account.human();
369 escro[jss::Destination] = to.human();
378 nextTime.time_since_epoch().count();
381 env(escrowWithFinish,
sig(alie));
385 nextTime.time_since_epoch().count();
387 nextTime.time_since_epoch().count() + 1;
390 env(escrowWithCancel,
sig(alie));
395 escrowFinish[jss::TransactionType] = jss::EscrowFinish;
397 escrowFinish[jss::Account] = alice.human();
400 env(escrowFinish,
sig(alie));
404 escrowCancel[jss::TransactionType] = jss::EscrowCancel;
406 escrowCancel[jss::Account] = alice.human();
409 env(escrowCancel,
sig(alie));
418 payChanCreate[jss::TransactionType] = jss::PaymentChannelCreate;
420 payChanCreate[jss::Account] = alice.human();
421 payChanCreate[jss::Destination] = gw.human();
422 payChanCreate[jss::Amount] =
427 env(payChanCreate,
sig(alie));
435 payChanFund[jss::TransactionType] = jss::PaymentChannelFund;
437 payChanFund[jss::Account] = alice.human();
439 payChanFund[jss::Amount] =
441 env(payChanFund,
sig(alie));
446 payChanClaim[jss::TransactionType] = jss::PaymentChannelClaim;
447 payChanClaim[jss::Flags] =
tfClose;
448 payChanClaim[jss::Account] = gw.human();
459 env(check::create(alice, gw,
XRP(300)),
sig(alie));
462 env(check::create(gw, alice,
XRP(200)));
465 env(check::cash(alice, gwCheckId,
XRP(200)),
sig(alie));
466 env(check::cancel(alice, aliceCheckId),
sig(alie));
472 env(ticket::create(alice, 1),
sig(alie));
481 params[jss::account] = alice.human();
482 params[jss::ledger_index_min] = -1;
483 params[jss::ledger_index_max] = -1;
488 BEAST_EXPECT(result[jss::result][jss::status] ==
"success");
489 BEAST_EXPECT(result[jss::result][jss::transactions].isArray());
491 Json::Value const& txs{result[jss::result][jss::transactions]};
498 {0, jss::DepositPreauth, {jss::DepositPreauth}, {jss::Ticket}, {jss::AccountRoot, jss::DirectoryNode}},
499 {1, jss::TicketCreate, {jss::Ticket}, {}, {jss::AccountRoot, jss::DirectoryNode}},
500 {2, jss::CheckCancel, {}, {jss::Check}, {jss::AccountRoot, jss::AccountRoot, jss::DirectoryNode, jss::DirectoryNode}},
501 {3, jss::CheckCash, {}, {jss::Check}, {jss::AccountRoot, jss::AccountRoot, jss::DirectoryNode, jss::DirectoryNode}},
502 {4, jss::CheckCreate, {jss::Check}, {}, {jss::AccountRoot, jss::AccountRoot, jss::DirectoryNode, jss::DirectoryNode}},
503 {5, jss::CheckCreate, {jss::Check}, {}, {jss::AccountRoot, jss::AccountRoot, jss::DirectoryNode, jss::DirectoryNode}},
504 {6, jss::PaymentChannelClaim, {}, {jss::PayChannel}, {jss::AccountRoot, jss::AccountRoot, jss::DirectoryNode, jss::DirectoryNode}},
505 {7, jss::PaymentChannelFund, {}, {}, {jss::AccountRoot, jss::PayChannel}},
506 {8, jss::PaymentChannelCreate, {jss::PayChannel}, {}, {jss::AccountRoot, jss::AccountRoot, jss::DirectoryNode, jss::DirectoryNode}},
507 {9, jss::EscrowCancel, {}, {jss::Escrow}, {jss::AccountRoot, jss::DirectoryNode}},
508 {10, jss::EscrowFinish, {}, {jss::Escrow}, {jss::AccountRoot, jss::DirectoryNode}},
509 {11, jss::EscrowCreate, {jss::Escrow}, {}, {jss::AccountRoot, jss::DirectoryNode}},
510 {12, jss::EscrowCreate, {jss::Escrow}, {}, {jss::AccountRoot, jss::DirectoryNode}},
511 {13, jss::SignerListSet, {jss::SignerList}, {}, {jss::AccountRoot, jss::DirectoryNode}},
512 {14, jss::OfferCancel, {}, {jss::Offer, jss::DirectoryNode}, {jss::AccountRoot, jss::DirectoryNode}},
513 {15, jss::OfferCreate, {jss::Offer, jss::DirectoryNode}, {}, {jss::AccountRoot, jss::DirectoryNode}},
514 {16, jss::TrustSet, {jss::RippleState, jss::DirectoryNode, jss::DirectoryNode}, {}, {jss::AccountRoot, jss::AccountRoot}},
515 {17, jss::SetRegularKey, {}, {}, {jss::AccountRoot}},
516 {18, jss::Payment, {}, {}, {jss::AccountRoot, jss::AccountRoot}},
517 {19, jss::AccountSet, {}, {}, {jss::AccountRoot}},
518 {20, jss::AccountSet, {}, {}, {jss::AccountRoot}},
519 {21, jss::Payment, {jss::AccountRoot}, {}, {jss::AccountRoot}},
524 std::size(sanity) == result[jss::result][jss::transactions].size());
526 for (
unsigned int index{0}; index <
std::size(sanity); ++index)
538 using namespace test::jtx;
539 using namespace std::chrono_literals;
545 env.
fund(
XRP(10000), alice, becky);
550 BEAST_EXPECT(env.
closed()->exists(beckyAcctKey));
562 auto const beckyPreDelBalance{env.
balance(becky)};
564 auto const acctDelFee{
drops(env.
current()->fees().increment)};
569 BEAST_EXPECT(!env.
closed()->exists(beckyAcctKey));
581 { 0, jss::Payment, {}, {}, {jss::AccountRoot, jss::AccountRoot}},
582 { 1, jss::Payment, {jss::AccountRoot}, {}, {jss::AccountRoot}},
583 { 2, jss::AccountDelete, {}, {jss::AccountRoot}, {jss::AccountRoot}},
584 { 3, jss::AccountSet, {}, {}, {jss::AccountRoot}},
585 { 4, jss::AccountSet, {}, {}, {jss::AccountRoot}},
586 { 5, jss::Payment, {jss::AccountRoot}, {}, {jss::AccountRoot}}
594 params[jss::account] = becky.human();
595 params[jss::ledger_index_min] = -1;
596 params[jss::ledger_index_max] = -1;
601 BEAST_EXPECT(result[jss::result][jss::status] ==
"success");
602 BEAST_EXPECT(result[jss::result][jss::transactions].isArray());
605 constexpr
unsigned int beckyDeletedOffest = 2;
608 result[jss::result][jss::transactions].size() +
611 Json::Value const& txs{result[jss::result][jss::transactions]};
613 for (
unsigned int index = beckyDeletedOffest;
617 checkSanity(txs[index - beckyDeletedOffest], sanity[index]);
630 env(pay(alice, becky,
XRP(45)));
634 BEAST_EXPECT(env.
closed()->exists(beckyAcctKey));
638 env(pay(becky, alice,
XRP(20)));
644 params[jss::account] = becky.human();
645 params[jss::ledger_index_min] = -1;
646 params[jss::ledger_index_max] = -1;
651 BEAST_EXPECT(result[jss::result][jss::status] ==
"success");
652 BEAST_EXPECT(result[jss::result][jss::transactions].isArray());
655 std::size(sanity) == result[jss::result][jss::transactions].size());
657 Json::Value const& txs{result[jss::result][jss::transactions]};
659 for (
unsigned int index = 0; index <
std::size(sanity); ++index)