20 #include <ripple/app/paths/AccountCurrencies.h>
21 #include <ripple/basics/contract.h>
22 #include <ripple/beast/unit_test.h>
23 #include <ripple/core/JobQueue.h>
24 #include <ripple/json/json_reader.h>
25 #include <ripple/json/to_string.h>
26 #include <ripple/protocol/STParsedJSON.h>
27 #include <ripple/protocol/TxFlags.h>
28 #include <ripple/protocol/jss.h>
29 #include <ripple/resource/Fees.h>
30 #include <ripple/rpc/Context.h>
31 #include <ripple/rpc/RPCHandler.h>
32 #include <ripple/rpc/impl/RPCHelpers.h>
33 #include <ripple/rpc/impl/Tuning.h>
38 #include <test/jtx/envconfig.h>
50 jv[jss::command] =
"ripple_path_find";
51 jv[jss::source_account] =
toBase58(src);
65 jv[jss::destination_account] = d;
68 j[jss::currency] =
"USD";
69 j[jss::value] =
"0.01";
87 cfg->PATH_SEARCH_OLD = 7;
89 cfg->PATH_SEARCH_MAX = 10;
105 template <
class Rep,
class Period>
135 auto& app = env.
app();
144 app.getLedgerMaster(),
154 params[jss::command] =
"ripple_path_find";
155 params[jss::source_account] =
toBase58(src);
156 params[jss::destination_account] =
toBase58(dst);
157 params[jss::destination_amount] =
165 j[jss::currency] =
to_string(saSrcCurrency.value());
171 app.getJobQueue().postCoro(
172 jtCLIENT,
"RPC-Client", [&](
auto const& coro) {
173 context.params = std::move(params);
179 using namespace std::chrono_literals;
181 BEAST_EXPECT(!result.
isMember(jss::error));
195 env, src, dst, saDstAmount, saSendMax, saSrcCurrency);
196 BEAST_EXPECT(!result.
isMember(jss::error));
199 if (result.
isMember(jss::destination_amount))
204 if (result.
isMember(jss::alternatives))
206 auto const& alts = result[jss::alternatives];
209 auto const&
path = alts[0u];
211 if (
path.isMember(jss::source_amount))
214 if (
path.isMember(jss::destination_amount))
218 if (
path.isMember(jss::paths_computed))
221 p[
"Paths"] =
path[jss::paths_computed];
234 testcase(
"source currency limits");
235 using namespace std::chrono_literals;
238 auto const gw =
Account(
"gateway");
239 env.
fund(
XRP(10000),
"alice",
"bob", gw);
240 env.
trust(gw[
"USD"](100),
"alice",
"bob");
243 auto& app = env.
app();
252 app.getLedgerMaster(),
263 app.getJobQueue().postCoro(
264 jtCLIENT,
"RPC-Client", [&](
auto const& coro) {
265 context.params =
rpf(
272 BEAST_EXPECT(!result.
isMember(jss::error));
275 app.getJobQueue().postCoro(
276 jtCLIENT,
"RPC-Client", [&](
auto const& coro) {
286 BEAST_EXPECT(result.
isMember(jss::error));
291 app.getJobQueue().postCoro(
292 jtCLIENT,
"RPC-Client", [&](
auto const& coro) {
299 BEAST_EXPECT(!result.
isMember(jss::error));
303 app.getJobQueue().postCoro(
304 jtCLIENT,
"RPC-Client", [&](
auto const& coro) {
311 BEAST_EXPECT(result.
isMember(jss::error));
317 testcase(
"no direct path no intermediary no alternatives");
320 env.
fund(
XRP(10000),
"alice",
"bob");
324 BEAST_EXPECT(std::get<0>(result).empty());
330 testcase(
"direct path no intermediary");
333 env.
fund(
XRP(10000),
"alice",
"bob");
340 BEAST_EXPECT(st.
empty());
347 testcase(
"payment auto path find");
350 auto const gw =
Account(
"gateway");
351 auto const USD = gw[
"USD"];
352 env.
fund(
XRP(10000),
"alice",
"bob", gw);
353 env.
trust(USD(600),
"alice");
354 env.
trust(USD(700),
"bob");
355 env(pay(gw,
"alice", USD(70)));
356 env(pay(
"alice",
"bob", USD(24)));
366 testcase(
"path find");
369 auto const gw =
Account(
"gateway");
370 auto const USD = gw[
"USD"];
371 env.
fund(
XRP(10000),
"alice",
"bob", gw);
372 env.
trust(USD(600),
"alice");
373 env.
trust(USD(700),
"bob");
374 env(pay(gw,
"alice", USD(70)));
375 env(pay(gw,
"bob", USD(50)));
389 testcase(
"XRP to XRP");
391 env.
fund(
XRP(10000),
"alice",
"bob");
394 BEAST_EXPECT(std::get<0>(result).empty());
400 testcase(
"path find consume all");
405 env.
fund(
XRP(10000),
"alice",
"bob",
"carol",
"dan",
"edward");
416 env,
"alice",
"edward",
Account(
"edward")[
"USD"](-1));
424 auto const gw =
Account(
"gateway");
425 auto const USD = gw[
"USD"];
426 env.
fund(
XRP(10000),
"alice",
"bob",
"carol", gw);
427 env.
trust(USD(100),
"bob",
"carol");
428 env(pay(gw,
"carol", USD(100)));
429 env(offer(
"carol",
XRP(100), USD(100)));
440 BEAST_EXPECT(st.
empty());
447 BEAST_EXPECT(sa ==
XRP(100));
455 testcase(
"alternative path consume both");
458 auto const gw =
Account(
"gateway");
459 auto const USD = gw[
"USD"];
460 auto const gw2 =
Account(
"gateway2");
461 auto const gw2_USD = gw2[
"USD"];
462 env.
fund(
XRP(10000),
"alice",
"bob", gw, gw2);
463 env.
trust(USD(600),
"alice");
464 env.
trust(gw2_USD(800),
"alice");
465 env.
trust(USD(700),
"bob");
466 env.
trust(gw2_USD(900),
"bob");
467 env(pay(gw,
"alice", USD(70)));
468 env(pay(gw2,
"alice", gw2_USD(70)));
469 env(pay(
"alice",
"bob",
Account(
"bob")[
"USD"](140)),
484 testcase(
"alternative paths consume best transfer");
487 auto const gw =
Account(
"gateway");
488 auto const USD = gw[
"USD"];
489 auto const gw2 =
Account(
"gateway2");
490 auto const gw2_USD = gw2[
"USD"];
491 env.
fund(
XRP(10000),
"alice",
"bob", gw, gw2);
493 env.
trust(USD(600),
"alice");
494 env.
trust(gw2_USD(800),
"alice");
495 env.
trust(USD(700),
"bob");
496 env.
trust(gw2_USD(900),
"bob");
497 env(pay(gw,
"alice", USD(70)));
498 env(pay(gw2,
"alice", gw2_USD(70)));
499 env(pay(
"alice",
"bob", USD(70)));
513 testcase(
"alternative paths - consume best transfer first");
516 auto const gw =
Account(
"gateway");
517 auto const USD = gw[
"USD"];
518 auto const gw2 =
Account(
"gateway2");
519 auto const gw2_USD = gw2[
"USD"];
520 env.
fund(
XRP(10000),
"alice",
"bob", gw, gw2);
522 env.
trust(USD(600),
"alice");
523 env.
trust(gw2_USD(800),
"alice");
524 env.
trust(USD(700),
"bob");
525 env.
trust(gw2_USD(900),
"bob");
526 env(pay(gw,
"alice", USD(70)));
527 env(pay(gw2,
"alice", gw2_USD(70)));
528 env(pay(
"alice",
"bob",
Account(
"bob")[
"USD"](77)),
544 testcase(
"alternative paths - limit returned paths to best quality");
547 auto const gw =
Account(
"gateway");
548 auto const USD = gw[
"USD"];
549 auto const gw2 =
Account(
"gateway2");
550 auto const gw2_USD = gw2[
"USD"];
551 env.
fund(
XRP(10000),
"alice",
"bob",
"carol",
"dan", gw, gw2);
552 env(
rate(
"carol", 1.1));
553 env.
trust(
Account(
"carol")[
"USD"](800),
"alice",
"bob");
555 env.
trust(USD(800),
"alice",
"bob");
556 env.
trust(gw2_USD(800),
"alice",
"bob");
559 env(pay(gw2,
"alice", gw2_USD(100)));
560 env(pay(
"carol",
"alice",
Account(
"carol")[
"USD"](100)));
561 env(pay(gw,
"alice", USD(100)));
579 testcase(
"path negative: Issue #5");
582 env.
fund(
XRP(10000),
"alice",
"bob",
"carol",
"dan");
583 env.
trust(
Account(
"bob")[
"USD"](100),
"alice",
"carol",
"dan");
586 env(pay(
"bob",
"carol",
Account(
"bob")[
"USD"](75)));
592 BEAST_EXPECT(std::get<0>(result).empty());
597 BEAST_EXPECT(std::get<0>(result).empty());
617 testcase(
"path negative: ripple-client issue #23: smaller");
620 env.
fund(
XRP(10000),
"alice",
"bob",
"carol",
"dan");
625 env(pay(
"alice",
"bob",
Account(
"bob")[
"USD"](55)),
636 testcase(
"path negative: ripple-client issue #23: larger");
639 env.
fund(
XRP(10000),
"alice",
"bob",
"carol",
"dan",
"edward");
645 env(pay(
"alice",
"bob",
Account(
"bob")[
"USD"](50)),
663 testcase(
"via gateway");
666 auto const gw =
Account(
"gateway");
667 auto const AUD = gw[
"AUD"];
668 env.
fund(
XRP(10000),
"alice",
"bob",
"carol", gw);
670 env.
trust(AUD(100),
"bob",
"carol");
671 env(pay(gw,
"carol", AUD(50)));
672 env(offer(
"carol",
XRP(50), AUD(50)));
679 BEAST_EXPECT(std::get<0>(result).empty());
685 testcase(
"path find");
688 env.
fund(
XRP(10000),
"alice",
"bob",
"carol");
703 testcase(
"quality set and test");
706 env.
fund(
XRP(10000),
"alice",
"bob");
707 env(trust(
"bob",
Account(
"alice")[
"USD"](1000)),
716 "issuer" : "rrrrrrrrrrrrrrrrrrrrBZbvji",
722 "issuer" : "rPMh7Pi9ct699iZUTWaytJUoHcJ7cgyziK",
726 "HighQualityIn" : 2000,
727 "HighQualityOut" : 1400000000,
728 "LedgerEntryType" : "RippleState",
731 "issuer" : "rG1QQv2nh2gr7RCZ1P8YYcBUKCCN633jCn",
742 for (
auto it = jv.
begin(); it != jv.
end(); ++it)
743 BEAST_EXPECT(*it == jv_l[it.memberName()]);
749 testcase(
"trust normal clear");
752 env.
fund(
XRP(10000),
"alice",
"bob");
761 "issuer" : "rrrrrrrrrrrrrrrrrrrrBZbvji",
767 "issuer" : "rPMh7Pi9ct699iZUTWaytJUoHcJ7cgyziK",
771 "LedgerEntryType" : "RippleState",
774 "issuer" : "rG1QQv2nh2gr7RCZ1P8YYcBUKCCN633jCn",
786 BEAST_EXPECT(*it == jv_l[it.memberName()]);
799 testcase(
"trust auto clear");
802 env.
fund(
XRP(10000),
"alice",
"bob");
804 env(pay(
"bob",
"alice",
Account(
"bob")[
"USD"](50)));
813 "issuer" : "rrrrrrrrrrrrrrrrrrrrBZbvji",
820 "issuer" : "rPMh7Pi9ct699iZUTWaytJUoHcJ7cgyziK",
824 "LedgerEntryType" : "RippleState",
828 "issuer" : "rG1QQv2nh2gr7RCZ1P8YYcBUKCCN633jCn",
839 for (
auto it = jv.
begin(); it != jv.
end(); ++it)
840 BEAST_EXPECT(*it == jv_l[it.memberName()]);
842 env(pay(
"alice",
"bob",
Account(
"alice")[
"USD"](50)));
852 testcase(
"Path Find: XRP -> XRP and XRP -> IOU");
863 env.fund(
XRP(100000), A1);
864 env.fund(
XRP(10000), A2);
865 env.fund(
XRP(1000), A3, G1, G2, G3, M1);
868 env.trust(G1[
"XYZ"](5000), A1);
869 env.trust(G3[
"ABC"](5000), A1);
870 env.trust(G2[
"XYZ"](5000), A2);
871 env.trust(G3[
"ABC"](5000), A2);
872 env.trust(A2[
"ABC"](1000), A3);
873 env.trust(G1[
"XYZ"](100000), M1);
874 env.trust(G2[
"XYZ"](100000), M1);
875 env.trust(G3[
"ABC"](100000), M1);
878 env(pay(G1, A1, G1[
"XYZ"](3500)));
879 env(pay(G3, A1, G3[
"ABC"](1200)));
880 env(pay(G2, M1, G2[
"XYZ"](25000)));
881 env(pay(G3, M1, G3[
"ABC"](25000)));
884 env(offer(M1, G1[
"XYZ"](1000), G2[
"XYZ"](1000)));
885 env(offer(M1,
XRP(10000), G3[
"ABC"](1000)));
891 auto const& send_amt =
XRP(10);
894 BEAST_EXPECT(
equal(da, send_amt));
895 BEAST_EXPECT(st.
empty());
901 auto const& send_amt =
XRP(200);
904 BEAST_EXPECT(
equal(da, send_amt));
905 BEAST_EXPECT(st.
empty());
909 auto const& send_amt = G3[
"ABC"](10);
912 BEAST_EXPECT(
equal(da, send_amt));
918 auto const& send_amt = A2[
"ABC"](1);
921 BEAST_EXPECT(
equal(da, send_amt));
927 auto const& send_amt = A3[
"ABC"](1);
930 BEAST_EXPECT(
equal(da, send_amt));
931 BEAST_EXPECT(
equal(sa, XRP(10)));
932 BEAST_EXPECT(same(st, stpath(IPE(G3[
"ABC"]), G3, A2)));
939 testcase(
"Path Find: non-XRP -> XRP");
947 env.fund(
XRP(1000), A1, A2, G3);
948 env.fund(
XRP(11000), M1);
951 env.trust(G3[
"ABC"](1000), A1, A2);
952 env.trust(G3[
"ABC"](100000), M1);
955 env(
pay(G3, A1, G3[
"ABC"](1000)));
956 env(
pay(G3, A2, G3[
"ABC"](1000)));
957 env(
pay(G3, M1, G3[
"ABC"](1200)));
960 env(
offer(M1, G3[
"ABC"](1000),
XRP(10000)));
965 auto const& send_amt =
XRP(10);
967 find_paths(env, A1, A2, send_amt, std::nullopt, A2[
"ABC"].currency);
968 BEAST_EXPECT(
equal(da, send_amt));
969 BEAST_EXPECT(
equal(sa, A1[
"ABC"](1)));
976 testcase(
"Path Find: Bitstamp and SnapSwap, liquidity with no offers");
981 Account G1BS{
"G1BS"};
982 Account G2SW{
"G2SW"};
985 env.fund(
XRP(1000), G1BS, G2SW, A1, A2);
986 env.fund(
XRP(11000), M1);
989 env.trust(G1BS[
"HKD"](2000), A1);
990 env.trust(G2SW[
"HKD"](2000), A2);
991 env.trust(G1BS[
"HKD"](100000), M1);
992 env.trust(G2SW[
"HKD"](100000), M1);
995 env(pay(G1BS, A1, G1BS[
"HKD"](1000)));
996 env(pay(G2SW, A2, G2SW[
"HKD"](1000)));
1000 env(pay(G1BS, M1, G1BS[
"HKD"](1200)));
1001 env(pay(G2SW, M1, G2SW[
"HKD"](5000)));
1008 auto const& send_amt = A2[
"HKD"](10);
1010 env, A1, A2, send_amt, std::nullopt, A2[
"HKD"].currency);
1011 BEAST_EXPECT(
equal(da, send_amt));
1012 BEAST_EXPECT(
equal(sa, A1[
"HKD"](10)));
1013 BEAST_EXPECT(
same(st,
stpath(G1BS, M1, G2SW)));
1017 auto const& send_amt = A1[
"HKD"](10);
1019 env, A2, A1, send_amt, std::nullopt, A1[
"HKD"].currency);
1020 BEAST_EXPECT(
equal(da, send_amt));
1021 BEAST_EXPECT(
equal(sa, A2[
"HKD"](10)));
1022 BEAST_EXPECT(
same(st,
stpath(G2SW, M1, G1BS)));
1026 auto const& send_amt = A2[
"HKD"](10);
1028 env, G1BS, A2, send_amt, std::nullopt, A1[
"HKD"].currency);
1029 BEAST_EXPECT(
equal(da, send_amt));
1030 BEAST_EXPECT(
equal(sa, G1BS[
"HKD"](10)));
1031 BEAST_EXPECT(same(st, stpath(M1, G2SW)));
1035 auto const& send_amt = M1[
"HKD"](10);
1037 env, M1, G1BS, send_amt, std::nullopt, A1[
"HKD"].currency);
1038 BEAST_EXPECT(
equal(da, send_amt));
1039 BEAST_EXPECT(
equal(sa, M1[
"HKD"](10)));
1040 BEAST_EXPECT(st.
empty());
1044 auto const& send_amt = A1[
"HKD"](10);
1046 env, G2SW, A1, send_amt, std::nullopt, A1[
"HKD"].currency);
1047 BEAST_EXPECT(
equal(da, send_amt));
1048 BEAST_EXPECT(
equal(sa, G2SW[
"HKD"](10)));
1056 testcase(
"Path Find: non-XRP -> non-XRP, same currency");
1057 using namespace jtx;
1070 env.fund(
XRP(1000), A1, A2, A3, G1, G2, G3, G4);
1071 env.fund(
XRP(10000), A4);
1072 env.fund(
XRP(11000), M1, M2);
1075 env.trust(G1[
"HKD"](2000), A1);
1076 env.trust(G2[
"HKD"](2000), A2);
1077 env.trust(G1[
"HKD"](2000), A3);
1078 env.trust(G1[
"HKD"](100000), M1);
1079 env.trust(G2[
"HKD"](100000), M1);
1080 env.trust(G1[
"HKD"](100000), M2);
1081 env.trust(G2[
"HKD"](100000), M2);
1084 env(
pay(G1, A1, G1[
"HKD"](1000)));
1085 env(
pay(G2, A2, G2[
"HKD"](1000)));
1086 env(
pay(G1, A3, G1[
"HKD"](1000)));
1087 env(
pay(G1, M1, G1[
"HKD"](1200)));
1088 env(
pay(G2, M1, G2[
"HKD"](5000)));
1089 env(
pay(G1, M2, G1[
"HKD"](1200)));
1090 env(
pay(G2, M2, G2[
"HKD"](5000)));
1093 env(
offer(M1, G1[
"HKD"](1000), G2[
"HKD"](1000)));
1094 env(
offer(M2,
XRP(10000), G2[
"HKD"](1000)));
1095 env(
offer(M2, G1[
"HKD"](1000),
XRP(10000)));
1103 auto const& send_amt = G1[
"HKD"](10);
1105 env, A1, G1, send_amt, std::nullopt, G1[
"HKD"].currency);
1106 BEAST_EXPECT(st.empty());
1107 BEAST_EXPECT(
equal(da, send_amt));
1108 BEAST_EXPECT(
equal(sa, A1[
"HKD"](10)));
1114 auto const& send_amt = A1[
"HKD"](10);
1116 env, A1, G1, send_amt, std::nullopt, G1[
"HKD"].currency);
1117 BEAST_EXPECT(st.empty());
1119 BEAST_EXPECT(
equal(sa, A1[
"HKD"](10)));
1125 auto const& send_amt = A3[
"HKD"](10);
1127 env, A1, A3, send_amt, std::nullopt, G1[
"HKD"].currency);
1128 BEAST_EXPECT(
equal(da, send_amt));
1129 BEAST_EXPECT(
equal(sa, A1[
"HKD"](10)));
1136 auto const& send_amt = G2[
"HKD"](10);
1138 env, G1, G2, send_amt, std::nullopt, G1[
"HKD"].currency);
1139 BEAST_EXPECT(
equal(da, send_amt));
1140 BEAST_EXPECT(
equal(sa, G1[
"HKD"](10)));
1143 stpath(IPE(G2[
"HKD"])),
1146 stpath(IPE(
xrpIssue()), IPE(G2[
"HKD"]))));
1152 auto const& send_amt = G2[
"HKD"](10);
1154 env, A1, G2, send_amt, std::nullopt, G1[
"HKD"].currency);
1155 BEAST_EXPECT(
equal(da, send_amt));
1156 BEAST_EXPECT(
equal(sa, A1[
"HKD"](10)));
1161 stpath(G1, IPE(G2[
"HKD"])),
1168 auto const& send_amt = A2[
"HKD"](10);
1170 env, A1, A2, send_amt, std::nullopt, G1[
"HKD"].currency);
1171 BEAST_EXPECT(
equal(da, send_amt));
1172 BEAST_EXPECT(
equal(sa, A1[
"HKD"](10)));
1185 testcase(
"Path Find: non-XRP -> non-XRP, same currency)");
1186 using namespace jtx;
1195 env.fund(
XRP(11000), M1);
1196 env.fund(
XRP(1000), A1, A2, A3, G1, G2);
1199 env.trust(G1[
"HKD"](2000), A1);
1200 env.trust(G2[
"HKD"](2000), A2);
1201 env.trust(A2[
"HKD"](2000), A3);
1202 env.trust(G1[
"HKD"](100000), M1);
1203 env.trust(G2[
"HKD"](100000), M1);
1206 env(
pay(G1, A1, G1[
"HKD"](1000)));
1207 env(
pay(G2, A2, G2[
"HKD"](1000)));
1208 env(
pay(G1, M1, G1[
"HKD"](5000)));
1209 env(
pay(G2, M1, G2[
"HKD"](5000)));
1212 env(
offer(M1, G1[
"HKD"](1000), G2[
"HKD"](1000)));
1216 auto const& send_amt = A2[
"HKD"](10);
1220 find_paths(env, G1, A2, send_amt, std::nullopt, G1[
"HKD"].currency);
1222 BEAST_EXPECT(
equal(sa, G1[
"HKD"](10)));
1229 testcase(
"Receive max");
1230 using namespace jtx;
1231 auto const alice =
Account(
"alice");
1232 auto const bob =
Account(
"bob");
1233 auto const charlie =
Account(
"charlie");
1234 auto const gw =
Account(
"gw");
1235 auto const USD = gw[
"USD"];
1239 env.
fund(
XRP(10000), alice, bob, charlie, gw);
1241 env.
trust(USD(100), alice, bob, charlie);
1243 env(pay(gw, charlie, USD(10)));
1245 env(offer(charlie,
XRP(10), USD(10)));
1249 BEAST_EXPECT(sa ==
XRP(10));
1250 BEAST_EXPECT(
equal(da, USD(10)));
1251 if (BEAST_EXPECT(st.size() == 1 && st[0].size() == 1))
1253 auto const& pathElem = st[0][0];
1255 pathElem.isOffer() && pathElem.getIssuerID() == gw.id() &&
1256 pathElem.getCurrency() == USD.currency);
1262 env.fund(XRP(10000), alice, bob, charlie, gw);
1264 env.trust(USD(100), alice, bob, charlie);
1266 env(pay(gw, alice, USD(10)));
1268 env(offer(charlie, USD(10), XRP(10)));
1271 find_paths(env, alice, bob, drops(-1), USD(100).value());
1272 BEAST_EXPECT(sa == USD(10));
1273 BEAST_EXPECT(
equal(da, XRP(10)));
1274 if (BEAST_EXPECT(st.size() == 1 && st[0].size() == 1))
1276 auto const& pathElem = st[0][0];
1278 pathElem.isOffer() &&
1288 using namespace jtx;
1293 auto const alice =
Account(
"alice");
1294 auto const bob =
Account(
"bob");
1295 auto const george =
Account(
"george");
1296 auto const USD = george[
"USD"];
1323 env(pay(george, alice, USD(70)));
1328 BEAST_EXPECT(
equal(da, bob[
"USD"](5)));
1332 BEAST_EXPECT(st.size() == 1);
1334 BEAST_EXPECT(
equal(sa, alice[
"USD"](5)));
1338 BEAST_EXPECT(st.size() == 0);
1342 test(
"ripple -> ripple",
true,
true,
true);
1343 test(
"ripple -> no ripple",
true,
false,
true);
1344 test(
"no ripple -> ripple",
false,
true,
true);
1345 test(
"no ripple -> no ripple",
false,
false,
false);