#include #include #include #include #include #include #include #include namespace xrpl { class Discrepancy_test : public beast::unit_test::suite { // This is a legacy test ported from js/coffee. The ledger // state was originally setup via a saved ledger file and the relevant // entries have since been converted to the equivalent jtx/Env setup. // A payment with path and sendmax is made and the transaction is queried // to verify that the net of balance changes match the fee charged. void testXRPDiscrepancy(FeatureBitset features) { testcase("Discrepancy test : XRP Discrepancy"); using namespace test::jtx; Env env{*this, features}; Account const A1{"A1"}; Account const A2{"A2"}; Account const A3{"A3"}; Account const A4{"A4"}; Account const A5{"A5"}; Account const A6{"A6"}; Account const A7{"A7"}; env.fund(XRP(2000), A1); env.fund(XRP(1000), A2, A6, A7); env.fund(XRP(5000), A3); env.fund(XRP(1000000), A4); env.fund(XRP(600000), A5); env.close(); env(trust(A1, A3["CNY"](200000))); env(pay(A3, A1, A3["CNY"](31))); env.close(); env(trust(A1, A2["JPY"](1000000))); env(pay(A2, A1, A2["JPY"](729117))); env.close(); env(trust(A4, A2["JPY"](10000000))); env(pay(A2, A4, A2["JPY"](470056))); env.close(); env(trust(A5, A3["CNY"](50000))); env(pay(A3, A5, A3["CNY"](8683))); env.close(); env(trust(A6, A3["CNY"](3000))); env(pay(A3, A6, A3["CNY"](293))); env.close(); env(trust(A7, A6["CNY"](50000))); env(pay(A6, A7, A6["CNY"](261))); env.close(); env(offer(A4, XRP(49147), A2["JPY"](34501))); env(offer(A5, A3["CNY"](3150), XRP(80086))); env(offer(A7, XRP(1233), A6["CNY"](25))); env.close(); test::PathSet const payPaths{ test::Path{A2["JPY"], A2}, test::Path{XRP, A2["JPY"], A2}, test::Path{A6, XRP, A2["JPY"], A2}}; env(pay(A1, A1, A2["JPY"](1000)), json(payPaths.json()), txflags(tfPartialPayment), sendmax(A3["CNY"](56))); env.close(); Json::Value jrq2; jrq2[jss::binary] = false; jrq2[jss::transaction] = env.tx()->getJson(JsonOptions::none)[jss::hash]; jrq2[jss::id] = 3; auto jrr = env.rpc("json", "tx", to_string(jrq2))[jss::result]; uint64_t const fee{jrr[jss::Fee].asUInt()}; auto meta = jrr[jss::meta]; uint64_t sumPrev{0}; uint64_t sumFinal{0}; BEAST_EXPECT(meta[sfAffectedNodes.fieldName].size() == 9); for (auto const& an : meta[sfAffectedNodes.fieldName]) { Json::Value node; if (an.isMember(sfCreatedNode.fieldName)) { node = an[sfCreatedNode.fieldName]; } else if (an.isMember(sfModifiedNode.fieldName)) { node = an[sfModifiedNode.fieldName]; } else if (an.isMember(sfDeletedNode.fieldName)) { node = an[sfDeletedNode.fieldName]; } if (node && node[sfLedgerEntryType.fieldName] == jss::AccountRoot) { Json::Value prevFields = node.isMember(sfPreviousFields.fieldName) ? node[sfPreviousFields.fieldName] : node[sfNewFields.fieldName]; Json::Value finalFields = node.isMember(sfFinalFields.fieldName) ? node[sfFinalFields.fieldName] : node[sfNewFields.fieldName]; if (prevFields) { sumPrev += beast::lexicalCastThrow( prevFields[sfBalance.fieldName].asString()); } if (finalFields) { sumFinal += beast::lexicalCastThrow( finalFields[sfBalance.fieldName].asString()); } } } // the difference in balances (final and prev) should be the // fee charged BEAST_EXPECT(sumPrev - sumFinal == fee); } public: void run() override { using namespace test::jtx; auto const sa = testable_amendments(); testXRPDiscrepancy(sa - featurePermissionedDEX); testXRPDiscrepancy(sa); } }; BEAST_DEFINE_TESTSUITE(Discrepancy, app, xrpl); } // namespace xrpl