//------------------------------------------------------------------------------ /* This file is part of rippled: https://github.com/ripple/rippled Copyright (c) 2012, 2013 Ripple Labs Inc. Permission to use, copy, modify, and/or distribute this software for any purpose with or without fee is hereby granted, provided that the above copyright notice and this permission notice appear in all copies. THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL , DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. */ //============================================================================== #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace ripple { template class BookStep : public StepImp> { protected: uint32_t const maxOffersToConsume_; Book book_; AccountID strandSrc_; AccountID strandDst_; // Charge transfer fees when the prev step redeems Step const* const prevStep_ = nullptr; bool const ownerPaysTransferFee_; // Mark as inactive (dry) if too many offers are consumed bool inactive_ = false; /** Number of offers consumed or partially consumed the last time the step ran, including expired and unfunded offers. N.B. This this not the total number offers consumed by this step for the entire payment, it is only the number the last time it ran. Offers may be partially consumed multiple times during a payment. */ std::uint32_t offersUsed_ = 0; beast::Journal const j_; struct Cache { TIn in; TOut out; Cache(TIn const& in_, TOut const& out_) : in(in_), out(out_) { } }; std::optional cache_; static uint32_t getMaxOffersToConsume(StrandContext const& ctx) { if (ctx.view.rules().enabled(fix1515)) return 1000; return 2000; } public: BookStep(StrandContext const& ctx, Issue const& in, Issue const& out) : maxOffersToConsume_(getMaxOffersToConsume(ctx)) , book_(in, out) , strandSrc_(ctx.strandSrc) , strandDst_(ctx.strandDst) , prevStep_(ctx.prevStep) , ownerPaysTransferFee_(ctx.ownerPaysTransferFee) , j_(ctx.j) { } Book const& book() const { return book_; } std::optional cachedIn() const override { if (!cache_) return std::nullopt; return EitherAmount(cache_->in); } std::optional cachedOut() const override { if (!cache_) return std::nullopt; return EitherAmount(cache_->out); } DebtDirection debtDirection(ReadView const& sb, StrandDirection dir) const override { return ownerPaysTransferFee_ ? DebtDirection::issues : DebtDirection::redeems; } std::optional bookStepBook() const override { return book_; } std::pair, DebtDirection> qualityUpperBound(ReadView const& v, DebtDirection prevStepDir) const override; std::uint32_t offersUsed() const override; std::pair revImp( PaymentSandbox& sb, ApplyView& afView, boost::container::flat_set& ofrsToRm, TOut const& out); std::pair fwdImp( PaymentSandbox& sb, ApplyView& afView, boost::container::flat_set& ofrsToRm, TIn const& in); std::pair validFwd(PaymentSandbox& sb, ApplyView& afView, EitherAmount const& in) override; // Check for errors frozen constraints. TER check(StrandContext const& ctx) const; bool inactive() const override { return inactive_; } protected: std::string logStringImpl(char const* name) const { std::ostringstream ostr; ostr << name << ": " << "\ninIss: " << book_.in.account << "\noutIss: " << book_.out.account << "\ninCur: " << book_.in.currency << "\noutCur: " << book_.out.currency; return ostr.str(); } private: friend bool operator==(BookStep const& lhs, BookStep const& rhs) { return lhs.book_ == rhs.book_; } friend bool operator!=(BookStep const& lhs, BookStep const& rhs) { return !(lhs == rhs); } bool equal(Step const& rhs) const override; // Iterate through the offers at the best quality in a book. // Unfunded offers and bad offers are skipped (and returned). // callback is called with the offer SLE, taker pays, taker gets. // If callback returns false, don't process any more offers. // Return the unfunded and bad offers and the number of offers consumed. template std::pair, std::uint32_t> forEachOffer( PaymentSandbox& sb, ApplyView& afView, DebtDirection prevStepDebtDir, Callback& callback) const; void consumeOffer( PaymentSandbox& sb, TOffer& offer, TAmounts const& ofrAmt, TAmounts const& stepAmt, TOut const& ownerGives) const; }; //------------------------------------------------------------------------------ // Flow is used in two different circumstances for transferring funds: // o Payments, and // o Offer crossing. // The rules for handling funds in these two cases are almost, but not // quite, the same. // Payment BookStep template class (not offer crossing). template class BookPaymentStep : public BookStep> { public: explicit BookPaymentStep() = default; using BookStep>::BookStep; using BookStep>::qualityUpperBound; // Never limit self cross quality on a payment. bool limitSelfCrossQuality( AccountID const&, AccountID const&, TOffer const& offer, std::optional&, FlowOfferStream&, bool) const { return false; } // A payment can look at offers of any quality bool checkQualityThreshold(TOffer const& offer) const { return true; } // For a payment ofrInRate is always the same as trIn. std::uint32_t getOfrInRate(Step const*, TOffer const&, std::uint32_t trIn) const { return trIn; } // For a payment ofrOutRate is always the same as trOut. std::uint32_t getOfrOutRate( Step const*, TOffer const&, AccountID const&, std::uint32_t trOut) const { return trOut; } Quality adjustQualityWithFees( ReadView const& v, Quality const& ofrQ, DebtDirection prevStepDir) const { // Charge the offer owner, not the sender // Charge a fee even if the owner is the same as the issuer // (the old code does not charge a fee) // Calculate amount that goes to the taker and the amount charged the // offer owner auto rate = [&](AccountID const& id) { if (isXRP(id) || id == this->strandDst_) return parityRate; return transferRate(v, id); }; auto const trIn = redeems(prevStepDir) ? rate(this->book_.in.account) : parityRate; // Always charge the transfer fee, even if the owner is the issuer auto const trOut = this->ownerPaysTransferFee_ ? rate(this->book_.out.account) : parityRate; Quality const q1{getRate(STAmount(trOut.value), STAmount(trIn.value))}; return composed_quality(q1, ofrQ); } std::string logString() const override { return this->logStringImpl("BookPaymentStep"); } }; // Offer crossing BookStep template class (not a payment). template class BookOfferCrossingStep : public BookStep> { using BookStep>:: qualityUpperBound; private: // Helper function that throws if the optional passed to the constructor // is none. static Quality getQuality(std::optional const& limitQuality) { // It's really a programming error if the quality is missing. assert(limitQuality); if (!limitQuality) Throw(tefINTERNAL, "Offer requires quality."); return *limitQuality; } public: BookOfferCrossingStep( StrandContext const& ctx, Issue const& in, Issue const& out) : BookStep>(ctx, in, out) , defaultPath_(ctx.isDefaultPath) , qualityThreshold_(getQuality(ctx.limitQuality)) { } bool limitSelfCrossQuality( AccountID const& strandSrc, AccountID const& strandDst, TOffer const& offer, std::optional& ofrQ, FlowOfferStream& offers, bool const offerAttempted) const { // This method supports some correct but slightly surprising // behavior in offer crossing. The scenario: // // o alice has already created one or more offers. // o alice creates another offer that can be directly crossed (not // autobridged) by one or more of her previously created offer(s). // // What does the offer crossing do? // // o The offer crossing could go ahead and cross the offers leaving // either one reduced offer (partial crossing) or zero offers // (exact crossing) in the ledger. We don't do this. And, really, // the offer creator probably didn't want us to. // // o We could skip over the self offer in the book and only cross // offers that are not our own. This would make a lot of sense, // but we don't do it. Part of the rationale is that we can only // operate on the tip of the order book. We can't leave an offer // behind -- it would sit on the tip and block access to other // offers. // // o We could delete the self-crossable offer(s) off the tip of the // book and continue with offer crossing. That's what we do. // // To support this scenario offer crossing has a special rule. If: // a. We're offer crossing using default path (no autobridging), and // b. The offer's quality is at least as good as our quality, and // c. We're about to cross one of our own offers, then // d. Delete the old offer from the ledger. if (defaultPath_ && offer.quality() >= qualityThreshold_ && strandSrc == offer.owner() && strandDst == offer.owner()) { // Remove this offer even if no crossing occurs. offers.permRmOffer(offer.key()); // If no offers have been attempted yet then it's okay to move to // a different quality. if (!offerAttempted) ofrQ = std::nullopt; // Return true so the current offer will be deleted. return true; } return false; } // Offer crossing can prune the offers it needs to look at with a // quality threshold. bool checkQualityThreshold(TOffer const& offer) const { return !defaultPath_ || offer.quality() >= qualityThreshold_; } // For offer crossing don't pay the transfer fee if alice is paying alice. // A regular (non-offer-crossing) payment does not apply this rule. std::uint32_t getOfrInRate( Step const* prevStep, TOffer const& offer, std::uint32_t trIn) const { auto const srcAcct = prevStep ? prevStep->directStepSrcAcct() : std::nullopt; return // If offer crossing srcAcct && // && prevStep is DirectI offer.owner() == *srcAcct // && src is offer owner ? QUALITY_ONE : trIn; // then rate = QUALITY_ONE } // See comment on getOfrInRate(). std::uint32_t getOfrOutRate( Step const* prevStep, TOffer const& offer, AccountID const& strandDst, std::uint32_t trOut) const { return // If offer crossing prevStep && prevStep->bookStepBook() && // && prevStep is BookStep offer.owner() == strandDst // && dest is offer owner ? QUALITY_ONE : trOut; // then rate = QUALITY_ONE } Quality adjustQualityWithFees( ReadView const& v, Quality const& ofrQ, DebtDirection prevStepDir) const { // Offer x-ing does not charge a transfer fee when the offer's owner // is the same as the strand dst. It is important that // `qualityUpperBound` is an upper bound on the quality (it is used to // ignore strands whose quality cannot meet a minimum threshold). When // calculating quality assume no fee is charged, or the estimate will no // longer be an upper bound. return ofrQ; } std::string logString() const override { return this->logStringImpl("BookOfferCrossingStep"); } private: bool const defaultPath_; Quality const qualityThreshold_; }; //------------------------------------------------------------------------------ template bool BookStep::equal(Step const& rhs) const { if (auto bs = dynamic_cast const*>(&rhs)) return book_ == bs->book_; return false; } template std::pair, DebtDirection> BookStep::qualityUpperBound( ReadView const& v, DebtDirection prevStepDir) const { auto const dir = this->debtDirection(v, StrandDirection::forward); // This can be simplified (and sped up) if directories are never empty. Sandbox sb(&v, tapNONE); BookTip bt(sb, book_); if (!bt.step(j_)) return {std::nullopt, dir}; Quality const q = static_cast(this)->adjustQualityWithFees( v, bt.quality(), prevStepDir); return {q, dir}; } template std::uint32_t BookStep::offersUsed() const { return offersUsed_; } // Adjust the offer amount and step amount subject to the given input limit template static void limitStepIn( Quality const& ofrQ, TAmounts& ofrAmt, TAmounts& stpAmt, TOut& ownerGives, std::uint32_t transferRateIn, std::uint32_t transferRateOut, TIn const& limit) { if (limit < stpAmt.in) { stpAmt.in = limit; auto const inLmt = mulRatio(stpAmt.in, QUALITY_ONE, transferRateIn, /*roundUp*/ false); ofrAmt = ofrQ.ceil_in(ofrAmt, inLmt); stpAmt.out = ofrAmt.out; ownerGives = mulRatio( ofrAmt.out, transferRateOut, QUALITY_ONE, /*roundUp*/ false); } } // Adjust the offer amount and step amount subject to the given output limit template static void limitStepOut( Quality const& ofrQ, TAmounts& ofrAmt, TAmounts& stpAmt, TOut& ownerGives, std::uint32_t transferRateIn, std::uint32_t transferRateOut, TOut const& limit, Rules const& rules) { if (limit < stpAmt.out) { stpAmt.out = limit; ownerGives = mulRatio( stpAmt.out, transferRateOut, QUALITY_ONE, /*roundUp*/ false); if (rules.enabled(fixReducedOffersV1)) // It turns out that the ceil_out implementation has some slop in // it. ceil_out_strict removes that slop. But removing that slop // affects transaction outcomes, so the change must be made using // an amendment. ofrAmt = ofrQ.ceil_out_strict(ofrAmt, stpAmt.out, /*roundUp*/ true); else ofrAmt = ofrQ.ceil_out(ofrAmt, stpAmt.out); stpAmt.in = mulRatio(ofrAmt.in, transferRateIn, QUALITY_ONE, /*roundUp*/ true); } } template template std::pair, std::uint32_t> BookStep::forEachOffer( PaymentSandbox& sb, ApplyView& afView, DebtDirection prevStepDir, Callback& callback) const { // Charge the offer owner, not the sender // Charge a fee even if the owner is the same as the issuer // (the old code does not charge a fee) // Calculate amount that goes to the taker and the amount charged the offer // owner auto rate = [this, &sb](AccountID const& id) -> std::uint32_t { if (isXRP(id) || id == this->strandDst_) return QUALITY_ONE; return transferRate(sb, id).value; }; std::uint32_t const trIn = redeems(prevStepDir) ? rate(book_.in.account) : QUALITY_ONE; // Always charge the transfer fee, even if the owner is the issuer std::uint32_t const trOut = ownerPaysTransferFee_ ? rate(book_.out.account) : QUALITY_ONE; typename FlowOfferStream::StepCounter counter( maxOffersToConsume_, j_); FlowOfferStream offers( sb, afView, book_, sb.parentCloseTime(), counter, j_); bool const flowCross = afView.rules().enabled(featureFlowCross); bool const fixReduced = afView.rules().enabled(fixReducedOffersV1); bool offerAttempted = false; std::optional ofrQ; while (offers.step()) { auto& offer = offers.tip(); // Note that offer.quality() returns a (non-optional) Quality. So // ofrQ is always safe to use below this point in the loop. if (!ofrQ) ofrQ = offer.quality(); else if (*ofrQ != offer.quality()) break; if (static_cast(this)->limitSelfCrossQuality( strandSrc_, strandDst_, offer, ofrQ, offers, offerAttempted)) continue; // Make sure offer owner has authorization to own IOUs from issuer. // An account can always own XRP or their own IOUs. if (flowCross && (!isXRP(offer.issueIn().currency)) && (offer.owner() != offer.issueIn().account)) { auto const& issuerID = offer.issueIn().account; auto const issuer = afView.read(keylet::account(issuerID)); if (issuer && ((*issuer)[sfFlags] & lsfRequireAuth)) { // Issuer requires authorization. See if offer owner has that. auto const& ownerID = offer.owner(); auto const authFlag = issuerID > ownerID ? lsfHighAuth : lsfLowAuth; auto const line = afView.read( keylet::line(ownerID, issuerID, offer.issueIn().currency)); if (!line || (((*line)[sfFlags] & authFlag) == 0)) { // Offer owner not authorized to hold IOU from issuer. // Remove this offer even if no crossing occurs. offers.permRmOffer(offer.key()); if (!offerAttempted) // Change quality only if no previous offers were tried. ofrQ = std::nullopt; // This continue causes offers.step() to delete the offer. continue; } } } if (!static_cast(this)->checkQualityThreshold(offer)) break; auto const ofrInRate = static_cast(this)->getOfrInRate( prevStep_, offer, trIn); auto const ofrOutRate = static_cast(this)->getOfrOutRate( prevStep_, offer, strandDst_, trOut); auto ofrAmt = offer.amount(); TAmounts stpAmt{ mulRatio(ofrAmt.in, ofrInRate, QUALITY_ONE, /*roundUp*/ true), ofrAmt.out}; // owner pays the transfer fee. auto ownerGives = mulRatio(ofrAmt.out, ofrOutRate, QUALITY_ONE, /*roundUp*/ false); auto const funds = (offer.owner() == offer.issueOut().account) ? ownerGives // Offer owner is issuer; they have unlimited funds : offers.ownerFunds(); if (funds < ownerGives) { // We already know offer.owner()!=offer.issueOut().account ownerGives = funds; stpAmt.out = mulRatio( ownerGives, QUALITY_ONE, ofrOutRate, /*roundUp*/ false); // It turns out we can prevent order book blocking by (strictly) // rounding down the ceil_out() result. This adjustment changes // transaction outcomes, so it must be made under an amendment. if (fixReduced) ofrAmt = ofrQ->ceil_out_strict( ofrAmt, stpAmt.out, /* roundUp */ false); else ofrAmt = ofrQ->ceil_out(ofrAmt, stpAmt.out); stpAmt.in = mulRatio(ofrAmt.in, ofrInRate, QUALITY_ONE, /*roundUp*/ true); } offerAttempted = true; if (!callback(offer, ofrAmt, stpAmt, ownerGives, ofrInRate, ofrOutRate)) break; } return {offers.permToRemove(), counter.count()}; } template void BookStep::consumeOffer( PaymentSandbox& sb, TOffer& offer, TAmounts const& ofrAmt, TAmounts const& stepAmt, TOut const& ownerGives) const { // The offer owner gets the ofrAmt. The difference between ofrAmt and // stepAmt is a transfer fee that goes to book_.in.account { auto const dr = accountSend( sb, book_.in.account, offer.owner(), toSTAmount(ofrAmt.in, book_.in), j_); if (dr != tesSUCCESS) Throw(dr); } // The offer owner pays `ownerGives`. The difference between ownerGives and // stepAmt is a transfer fee that goes to book_.out.account { auto const cr = accountSend( sb, offer.owner(), book_.out.account, toSTAmount(ownerGives, book_.out), j_); if (cr != tesSUCCESS) Throw(cr); } offer.consume(sb, ofrAmt); } template static auto sum(TCollection const& col) { using TResult = std::decay_t; if (col.empty()) return TResult{beast::zero}; return std::accumulate(col.begin() + 1, col.end(), *col.begin()); }; template std::pair BookStep::revImp( PaymentSandbox& sb, ApplyView& afView, boost::container::flat_set& ofrsToRm, TOut const& out) { cache_.reset(); TAmounts result(beast::zero, beast::zero); auto remainingOut = out; boost::container::flat_multiset savedIns; savedIns.reserve(64); boost::container::flat_multiset savedOuts; savedOuts.reserve(64); /* amt fed will be adjusted by owner funds (and may differ from the offer's amounts - tho always <=) Return true to continue to receive offers, false to stop receiving offers. */ auto eachOffer = [&](TOffer& offer, TAmounts const& ofrAmt, TAmounts const& stpAmt, TOut const& ownerGives, std::uint32_t transferRateIn, std::uint32_t transferRateOut) mutable -> bool { if (remainingOut <= beast::zero) return false; if (stpAmt.out <= remainingOut) { savedIns.insert(stpAmt.in); savedOuts.insert(stpAmt.out); result = TAmounts(sum(savedIns), sum(savedOuts)); remainingOut = out - result.out; this->consumeOffer(sb, offer, ofrAmt, stpAmt, ownerGives); // return true b/c even if the payment is satisfied, // we need to consume the offer return true; } else { auto ofrAdjAmt = ofrAmt; auto stpAdjAmt = stpAmt; auto ownerGivesAdj = ownerGives; limitStepOut( offer.quality(), ofrAdjAmt, stpAdjAmt, ownerGivesAdj, transferRateIn, transferRateOut, remainingOut, afView.rules()); remainingOut = beast::zero; savedIns.insert(stpAdjAmt.in); savedOuts.insert(remainingOut); result.in = sum(savedIns); result.out = out; this->consumeOffer(sb, offer, ofrAdjAmt, stpAdjAmt, ownerGivesAdj); // Explicitly check whether the offer is funded. Given that we have // (stpAmt.out > remainingOut), it's natural to assume the offer // will still be funded after consuming remainingOut but that is // not always the case. If the mantissas of two IOU amounts differ // by less than ten, then subtracting them leaves a zero. return offer.fully_consumed(); } }; { auto const prevStepDebtDir = [&] { if (prevStep_) return prevStep_->debtDirection(sb, StrandDirection::reverse); return DebtDirection::issues; }(); auto const r = forEachOffer(sb, afView, prevStepDebtDir, eachOffer); boost::container::flat_set toRm = std::move(std::get<0>(r)); std::uint32_t const offersConsumed = std::get<1>(r); offersUsed_ = offersConsumed; SetUnion(ofrsToRm, toRm); if (offersConsumed >= maxOffersToConsume_) { // Too many iterations, mark this strand as inactive if (!afView.rules().enabled(fix1515)) { // Don't use the liquidity cache_.emplace(beast::zero, beast::zero); return {beast::zero, beast::zero}; } // Use the liquidity, but use this to mark the strand as inactive so // it's not used further inactive_ = true; } } switch (remainingOut.signum()) { case -1: { // something went very wrong JLOG(j_.error()) << "BookStep remainingOut < 0 " << to_string(remainingOut); assert(0); cache_.emplace(beast::zero, beast::zero); return {beast::zero, beast::zero}; } case 0: { // due to normalization, remainingOut can be zero without // result.out == out. Force result.out == out for this case result.out = out; } } cache_.emplace(result.in, result.out); return {result.in, result.out}; } template std::pair BookStep::fwdImp( PaymentSandbox& sb, ApplyView& afView, boost::container::flat_set& ofrsToRm, TIn const& in) { assert(cache_); TAmounts result(beast::zero, beast::zero); auto remainingIn = in; boost::container::flat_multiset savedIns; savedIns.reserve(64); boost::container::flat_multiset savedOuts; savedOuts.reserve(64); // amt fed will be adjusted by owner funds (and may differ from the offer's // amounts - tho always <=) auto eachOffer = [&](TOffer& offer, TAmounts const& ofrAmt, TAmounts const& stpAmt, TOut const& ownerGives, std::uint32_t transferRateIn, std::uint32_t transferRateOut) mutable -> bool { assert(cache_); if (remainingIn <= beast::zero) return false; bool processMore = true; auto ofrAdjAmt = ofrAmt; auto stpAdjAmt = stpAmt; auto ownerGivesAdj = ownerGives; typename boost::container::flat_multiset::const_iterator lastOut; if (stpAmt.in <= remainingIn) { savedIns.insert(stpAmt.in); lastOut = savedOuts.insert(stpAmt.out); result = TAmounts(sum(savedIns), sum(savedOuts)); // consume the offer even if stepAmt.in == remainingIn processMore = true; } else { limitStepIn( offer.quality(), ofrAdjAmt, stpAdjAmt, ownerGivesAdj, transferRateIn, transferRateOut, remainingIn); savedIns.insert(remainingIn); lastOut = savedOuts.insert(stpAdjAmt.out); result.out = sum(savedOuts); result.in = in; processMore = false; } if (result.out > cache_->out && result.in <= cache_->in) { // The step produced more output in the forward pass than the // reverse pass while consuming the same input (or less). If we // compute the input required to produce the cached output // (produced in the reverse step) and the input is equal to // the input consumed in the forward step, then consume the // input provided in the forward step and produce the output // requested from the reverse step. auto const lastOutAmt = *lastOut; savedOuts.erase(lastOut); auto const remainingOut = cache_->out - sum(savedOuts); auto ofrAdjAmtRev = ofrAmt; auto stpAdjAmtRev = stpAmt; auto ownerGivesAdjRev = ownerGives; limitStepOut( offer.quality(), ofrAdjAmtRev, stpAdjAmtRev, ownerGivesAdjRev, transferRateIn, transferRateOut, remainingOut, afView.rules()); if (stpAdjAmtRev.in == remainingIn) { result.in = in; result.out = cache_->out; savedIns.clear(); savedIns.insert(result.in); savedOuts.clear(); savedOuts.insert(result.out); ofrAdjAmt = ofrAdjAmtRev; stpAdjAmt.in = remainingIn; stpAdjAmt.out = remainingOut; ownerGivesAdj = ownerGivesAdjRev; } else { // This is (likely) a problem case, and wil be caught // with later checks savedOuts.insert(lastOutAmt); } } remainingIn = in - result.in; this->consumeOffer(sb, offer, ofrAdjAmt, stpAdjAmt, ownerGivesAdj); // When the mantissas of two iou amounts differ by less than ten, then // subtracting them leaves a result of zero. This can cause the check // for (stpAmt.in > remainingIn) to incorrectly think an offer will be // funded after subtracting remainingIn. return processMore || offer.fully_consumed(); }; { auto const prevStepDebtDir = [&] { if (prevStep_) return prevStep_->debtDirection(sb, StrandDirection::forward); return DebtDirection::issues; }(); auto const r = forEachOffer(sb, afView, prevStepDebtDir, eachOffer); boost::container::flat_set toRm = std::move(std::get<0>(r)); std::uint32_t const offersConsumed = std::get<1>(r); offersUsed_ = offersConsumed; SetUnion(ofrsToRm, toRm); if (offersConsumed >= maxOffersToConsume_) { // Too many iterations, mark this strand as inactive (dry) if (!afView.rules().enabled(fix1515)) { // Don't use the liquidity cache_.emplace(beast::zero, beast::zero); return {beast::zero, beast::zero}; } // Use the liquidity, but use this to mark the strand as inactive so // it's not used further inactive_ = true; } } switch (remainingIn.signum()) { case -1: { // something went very wrong JLOG(j_.error()) << "BookStep remainingIn < 0 " << to_string(remainingIn); assert(0); cache_.emplace(beast::zero, beast::zero); return {beast::zero, beast::zero}; } case 0: { // due to normalization, remainingIn can be zero without // result.in == in. Force result.in == in for this case result.in = in; } } cache_.emplace(result.in, result.out); return {result.in, result.out}; } template std::pair BookStep::validFwd( PaymentSandbox& sb, ApplyView& afView, EitherAmount const& in) { if (!cache_) { JLOG(j_.trace()) << "Expected valid cache in validFwd"; return {false, EitherAmount(TOut(beast::zero))}; } auto const savCache = *cache_; try { boost::container::flat_set dummy; fwdImp(sb, afView, dummy, get(in)); // changes cache } catch (FlowException const&) { return {false, EitherAmount(TOut(beast::zero))}; } if (!(checkNear(savCache.in, cache_->in) && checkNear(savCache.out, cache_->out))) { JLOG(j_.warn()) << "Strand re-execute check failed." << " ExpectedIn: " << to_string(savCache.in) << " CachedIn: " << to_string(cache_->in) << " ExpectedOut: " << to_string(savCache.out) << " CachedOut: " << to_string(cache_->out); return {false, EitherAmount(cache_->out)}; } return {true, EitherAmount(cache_->out)}; } template TER BookStep::check(StrandContext const& ctx) const { if (book_.in == book_.out) { JLOG(j_.debug()) << "BookStep: Book with same in and out issuer " << *this; return temBAD_PATH; } if (!isConsistent(book_.in) || !isConsistent(book_.out)) { JLOG(j_.debug()) << "Book: currency is inconsistent with issuer." << *this; return temBAD_PATH; } // Do not allow two books to output the same issue. This may cause offers on // one step to unfund offers in another step. if (!ctx.seenBookOuts.insert(book_.out).second || ctx.seenDirectIssues[0].count(book_.out)) { JLOG(j_.debug()) << "BookStep: loop detected: " << *this; return temBAD_PATH_LOOP; } if (ctx.seenDirectIssues[1].count(book_.out)) { JLOG(j_.debug()) << "BookStep: loop detected: " << *this; return temBAD_PATH_LOOP; } auto issuerExists = [](ReadView const& view, Issue const& iss) -> bool { return isXRP(iss.account) || view.read(keylet::account(iss.account)); }; if (!issuerExists(ctx.view, book_.in) || !issuerExists(ctx.view, book_.out)) { JLOG(j_.debug()) << "BookStep: deleted issuer detected: " << *this; return tecNO_ISSUER; } if (ctx.prevStep) { if (auto const prev = ctx.prevStep->directStepSrcAcct()) { auto const& view = ctx.view; auto const& cur = book_.in.account; auto sle = view.read(keylet::line(*prev, cur, book_.in.currency)); if (!sle) return terNO_LINE; if ((*sle)[sfFlags] & ((cur > *prev) ? lsfHighNoRipple : lsfLowNoRipple)) return terNO_RIPPLE; } } return tesSUCCESS; } //------------------------------------------------------------------------------ namespace test { // Needed for testing template static bool equalHelper(Step const& step, ripple::Book const& book) { if (auto bs = dynamic_cast const*>(&step)) return book == bs->book(); return false; } bool bookStepEqual(Step const& step, ripple::Book const& book) { bool const inXRP = isXRP(book.in.currency); bool const outXRP = isXRP(book.out.currency); if (inXRP && outXRP) { assert(0); return false; // no such thing as xrp/xrp book step } if (inXRP && !outXRP) return equalHelper< XRPAmount, IOUAmount, BookPaymentStep>(step, book); if (!inXRP && outXRP) return equalHelper< IOUAmount, XRPAmount, BookPaymentStep>(step, book); if (!inXRP && !outXRP) return equalHelper< IOUAmount, IOUAmount, BookPaymentStep>(step, book); return false; } } // namespace test //------------------------------------------------------------------------------ template static std::pair> make_BookStepHelper(StrandContext const& ctx, Issue const& in, Issue const& out) { TER ter = tefINTERNAL; std::unique_ptr r; if (ctx.offerCrossing) { auto offerCrossingStep = std::make_unique>(ctx, in, out); ter = offerCrossingStep->check(ctx); r = std::move(offerCrossingStep); } else // payment { auto paymentStep = std::make_unique>(ctx, in, out); ter = paymentStep->check(ctx); r = std::move(paymentStep); } if (ter != tesSUCCESS) return {ter, nullptr}; return {tesSUCCESS, std::move(r)}; } std::pair> make_BookStepII(StrandContext const& ctx, Issue const& in, Issue const& out) { return make_BookStepHelper(ctx, in, out); } std::pair> make_BookStepIX(StrandContext const& ctx, Issue const& in) { return make_BookStepHelper(ctx, in, xrpIssue()); } std::pair> make_BookStepXI(StrandContext const& ctx, Issue const& out) { return make_BookStepHelper(ctx, xrpIssue(), out); } } // namespace ripple