mirror of
https://github.com/XRPLF/rippled.git
synced 2025-11-25 05:25:55 +00:00
Curtail the occurrence of order books that are blocked by reduced offers with the implementation of the fixReducedOffersV1 amendment. This commit identifies three ways in which offers can be reduced: 1. A new offer can be partially crossed by existing offers, so the new offer is reduced when placed in the ledger. 2. An in-ledger offer can be partially crossed by a new offer in a transaction. So the in-ledger offer is reduced by the new offer. 3. An in-ledger offer may be under-funded. In this case the in-ledger offer is scaled down to match the available funds. Reduced offers can block order books if the effective quality of the reduced offer is worse than the quality of the original offer (from the perspective of the taker). It turns out that, for small values, the quality of the reduced offer can be significantly affected by the rounding mode used during scaling computations. This commit adjusts some rounding modes so that the quality of a reduced offer is always at least as good (from the taker's perspective) as the original offer. The amendment is titled fixReducedOffersV1 because additional ways of producing reduced offers may come to light. Therefore, there may be a future need for a V2 amendment.
1218 lines
39 KiB
C++
1218 lines
39 KiB
C++
//------------------------------------------------------------------------------
|
|
/*
|
|
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 <ripple/app/paths/Credit.h>
|
|
#include <ripple/app/paths/impl/FlatSets.h>
|
|
#include <ripple/app/paths/impl/Steps.h>
|
|
#include <ripple/app/tx/impl/OfferStream.h>
|
|
#include <ripple/basics/IOUAmount.h>
|
|
#include <ripple/basics/Log.h>
|
|
#include <ripple/basics/XRPAmount.h>
|
|
#include <ripple/basics/contract.h>
|
|
#include <ripple/ledger/Directory.h>
|
|
#include <ripple/ledger/PaymentSandbox.h>
|
|
#include <ripple/protocol/Book.h>
|
|
#include <ripple/protocol/Feature.h>
|
|
#include <ripple/protocol/Quality.h>
|
|
|
|
#include <boost/container/flat_set.hpp>
|
|
|
|
#include <numeric>
|
|
#include <sstream>
|
|
|
|
namespace ripple {
|
|
|
|
template <class TIn, class TOut, class TDerived>
|
|
class BookStep : public StepImp<TIn, TOut, BookStep<TIn, TOut, TDerived>>
|
|
{
|
|
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> 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<EitherAmount>
|
|
cachedIn() const override
|
|
{
|
|
if (!cache_)
|
|
return std::nullopt;
|
|
return EitherAmount(cache_->in);
|
|
}
|
|
|
|
std::optional<EitherAmount>
|
|
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<Book>
|
|
bookStepBook() const override
|
|
{
|
|
return book_;
|
|
}
|
|
|
|
std::pair<std::optional<Quality>, DebtDirection>
|
|
qualityUpperBound(ReadView const& v, DebtDirection prevStepDir)
|
|
const override;
|
|
|
|
std::uint32_t
|
|
offersUsed() const override;
|
|
|
|
std::pair<TIn, TOut>
|
|
revImp(
|
|
PaymentSandbox& sb,
|
|
ApplyView& afView,
|
|
boost::container::flat_set<uint256>& ofrsToRm,
|
|
TOut const& out);
|
|
|
|
std::pair<TIn, TOut>
|
|
fwdImp(
|
|
PaymentSandbox& sb,
|
|
ApplyView& afView,
|
|
boost::container::flat_set<uint256>& ofrsToRm,
|
|
TIn const& in);
|
|
|
|
std::pair<bool, EitherAmount>
|
|
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 <class Callback>
|
|
std::pair<boost::container::flat_set<uint256>, std::uint32_t>
|
|
forEachOffer(
|
|
PaymentSandbox& sb,
|
|
ApplyView& afView,
|
|
DebtDirection prevStepDebtDir,
|
|
Callback& callback) const;
|
|
|
|
void
|
|
consumeOffer(
|
|
PaymentSandbox& sb,
|
|
TOffer<TIn, TOut>& offer,
|
|
TAmounts<TIn, TOut> const& ofrAmt,
|
|
TAmounts<TIn, TOut> 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 TIn, class TOut>
|
|
class BookPaymentStep : public BookStep<TIn, TOut, BookPaymentStep<TIn, TOut>>
|
|
{
|
|
public:
|
|
explicit BookPaymentStep() = default;
|
|
|
|
using BookStep<TIn, TOut, BookPaymentStep<TIn, TOut>>::BookStep;
|
|
using BookStep<TIn, TOut, BookPaymentStep<TIn, TOut>>::qualityUpperBound;
|
|
|
|
// Never limit self cross quality on a payment.
|
|
bool
|
|
limitSelfCrossQuality(
|
|
AccountID const&,
|
|
AccountID const&,
|
|
TOffer<TIn, TOut> const& offer,
|
|
std::optional<Quality>&,
|
|
FlowOfferStream<TIn, TOut>&,
|
|
bool) const
|
|
{
|
|
return false;
|
|
}
|
|
|
|
// A payment can look at offers of any quality
|
|
bool
|
|
checkQualityThreshold(TOffer<TIn, TOut> const& offer) const
|
|
{
|
|
return true;
|
|
}
|
|
|
|
// For a payment ofrInRate is always the same as trIn.
|
|
std::uint32_t
|
|
getOfrInRate(Step const*, TOffer<TIn, TOut> 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<TIn, TOut> 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 TIn, class TOut>
|
|
class BookOfferCrossingStep
|
|
: public BookStep<TIn, TOut, BookOfferCrossingStep<TIn, TOut>>
|
|
{
|
|
using BookStep<TIn, TOut, BookOfferCrossingStep<TIn, TOut>>::
|
|
qualityUpperBound;
|
|
|
|
private:
|
|
// Helper function that throws if the optional passed to the constructor
|
|
// is none.
|
|
static Quality
|
|
getQuality(std::optional<Quality> const& limitQuality)
|
|
{
|
|
// It's really a programming error if the quality is missing.
|
|
assert(limitQuality);
|
|
if (!limitQuality)
|
|
Throw<FlowException>(tefINTERNAL, "Offer requires quality.");
|
|
return *limitQuality;
|
|
}
|
|
|
|
public:
|
|
BookOfferCrossingStep(
|
|
StrandContext const& ctx,
|
|
Issue const& in,
|
|
Issue const& out)
|
|
: BookStep<TIn, TOut, BookOfferCrossingStep<TIn, TOut>>(ctx, in, out)
|
|
, defaultPath_(ctx.isDefaultPath)
|
|
, qualityThreshold_(getQuality(ctx.limitQuality))
|
|
{
|
|
}
|
|
|
|
bool
|
|
limitSelfCrossQuality(
|
|
AccountID const& strandSrc,
|
|
AccountID const& strandDst,
|
|
TOffer<TIn, TOut> const& offer,
|
|
std::optional<Quality>& ofrQ,
|
|
FlowOfferStream<TIn, TOut>& 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<TIn, TOut> 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<TIn, TOut> 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<TIn, TOut> 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 <class TIn, class TOut, class TDerived>
|
|
bool
|
|
BookStep<TIn, TOut, TDerived>::equal(Step const& rhs) const
|
|
{
|
|
if (auto bs = dynamic_cast<BookStep<TIn, TOut, TDerived> const*>(&rhs))
|
|
return book_ == bs->book_;
|
|
return false;
|
|
}
|
|
|
|
template <class TIn, class TOut, class TDerived>
|
|
std::pair<std::optional<Quality>, DebtDirection>
|
|
BookStep<TIn, TOut, TDerived>::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<TDerived const*>(this)->adjustQualityWithFees(
|
|
v, bt.quality(), prevStepDir);
|
|
return {q, dir};
|
|
}
|
|
|
|
template <class TIn, class TOut, class TDerived>
|
|
std::uint32_t
|
|
BookStep<TIn, TOut, TDerived>::offersUsed() const
|
|
{
|
|
return offersUsed_;
|
|
}
|
|
|
|
// Adjust the offer amount and step amount subject to the given input limit
|
|
template <class TIn, class TOut>
|
|
static void
|
|
limitStepIn(
|
|
Quality const& ofrQ,
|
|
TAmounts<TIn, TOut>& ofrAmt,
|
|
TAmounts<TIn, TOut>& 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 <class TIn, class TOut>
|
|
static void
|
|
limitStepOut(
|
|
Quality const& ofrQ,
|
|
TAmounts<TIn, TOut>& ofrAmt,
|
|
TAmounts<TIn, TOut>& 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 <class TIn, class TOut, class TDerived>
|
|
template <class Callback>
|
|
std::pair<boost::container::flat_set<uint256>, std::uint32_t>
|
|
BookStep<TIn, TOut, TDerived>::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<TIn, TOut>::StepCounter counter(
|
|
maxOffersToConsume_, j_);
|
|
|
|
FlowOfferStream<TIn, TOut> 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<Quality> 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<TDerived const*>(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<TDerived const*>(this)->checkQualityThreshold(offer))
|
|
break;
|
|
|
|
auto const ofrInRate = static_cast<TDerived const*>(this)->getOfrInRate(
|
|
prevStep_, offer, trIn);
|
|
|
|
auto const ofrOutRate =
|
|
static_cast<TDerived const*>(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 <class TIn, class TOut, class TDerived>
|
|
void
|
|
BookStep<TIn, TOut, TDerived>::consumeOffer(
|
|
PaymentSandbox& sb,
|
|
TOffer<TIn, TOut>& offer,
|
|
TAmounts<TIn, TOut> const& ofrAmt,
|
|
TAmounts<TIn, TOut> 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<FlowException>(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<FlowException>(cr);
|
|
}
|
|
|
|
offer.consume(sb, ofrAmt);
|
|
}
|
|
|
|
template <class TCollection>
|
|
static auto
|
|
sum(TCollection const& col)
|
|
{
|
|
using TResult = std::decay_t<decltype(*col.begin())>;
|
|
if (col.empty())
|
|
return TResult{beast::zero};
|
|
return std::accumulate(col.begin() + 1, col.end(), *col.begin());
|
|
};
|
|
|
|
template <class TIn, class TOut, class TDerived>
|
|
std::pair<TIn, TOut>
|
|
BookStep<TIn, TOut, TDerived>::revImp(
|
|
PaymentSandbox& sb,
|
|
ApplyView& afView,
|
|
boost::container::flat_set<uint256>& ofrsToRm,
|
|
TOut const& out)
|
|
{
|
|
cache_.reset();
|
|
|
|
TAmounts<TIn, TOut> result(beast::zero, beast::zero);
|
|
|
|
auto remainingOut = out;
|
|
|
|
boost::container::flat_multiset<TIn> savedIns;
|
|
savedIns.reserve(64);
|
|
boost::container::flat_multiset<TOut> 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<TIn, TOut>& offer,
|
|
TAmounts<TIn, TOut> const& ofrAmt,
|
|
TAmounts<TIn, TOut> 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<TIn, TOut>(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<uint256> 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 <class TIn, class TOut, class TDerived>
|
|
std::pair<TIn, TOut>
|
|
BookStep<TIn, TOut, TDerived>::fwdImp(
|
|
PaymentSandbox& sb,
|
|
ApplyView& afView,
|
|
boost::container::flat_set<uint256>& ofrsToRm,
|
|
TIn const& in)
|
|
{
|
|
assert(cache_);
|
|
|
|
TAmounts<TIn, TOut> result(beast::zero, beast::zero);
|
|
|
|
auto remainingIn = in;
|
|
|
|
boost::container::flat_multiset<TIn> savedIns;
|
|
savedIns.reserve(64);
|
|
boost::container::flat_multiset<TOut> 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<TIn, TOut>& offer,
|
|
TAmounts<TIn, TOut> const& ofrAmt,
|
|
TAmounts<TIn, TOut> 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<TOut>::const_iterator lastOut;
|
|
if (stpAmt.in <= remainingIn)
|
|
{
|
|
savedIns.insert(stpAmt.in);
|
|
lastOut = savedOuts.insert(stpAmt.out);
|
|
result = TAmounts<TIn, TOut>(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<uint256> 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 <class TIn, class TOut, class TDerived>
|
|
std::pair<bool, EitherAmount>
|
|
BookStep<TIn, TOut, TDerived>::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<uint256> dummy;
|
|
fwdImp(sb, afView, dummy, get<TIn>(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 <class TIn, class TOut, class TDerived>
|
|
TER
|
|
BookStep<TIn, TOut, TDerived>::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 <class TIn, class TOut, class TDerived>
|
|
static bool
|
|
equalHelper(Step const& step, ripple::Book const& book)
|
|
{
|
|
if (auto bs = dynamic_cast<BookStep<TIn, TOut, TDerived> 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<XRPAmount, IOUAmount>>(step, book);
|
|
if (!inXRP && outXRP)
|
|
return equalHelper<
|
|
IOUAmount,
|
|
XRPAmount,
|
|
BookPaymentStep<IOUAmount, XRPAmount>>(step, book);
|
|
if (!inXRP && !outXRP)
|
|
return equalHelper<
|
|
IOUAmount,
|
|
IOUAmount,
|
|
BookPaymentStep<IOUAmount, IOUAmount>>(step, book);
|
|
return false;
|
|
}
|
|
} // namespace test
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
template <class TIn, class TOut>
|
|
static std::pair<TER, std::unique_ptr<Step>>
|
|
make_BookStepHelper(StrandContext const& ctx, Issue const& in, Issue const& out)
|
|
{
|
|
TER ter = tefINTERNAL;
|
|
std::unique_ptr<Step> r;
|
|
if (ctx.offerCrossing)
|
|
{
|
|
auto offerCrossingStep =
|
|
std::make_unique<BookOfferCrossingStep<TIn, TOut>>(ctx, in, out);
|
|
ter = offerCrossingStep->check(ctx);
|
|
r = std::move(offerCrossingStep);
|
|
}
|
|
else // payment
|
|
{
|
|
auto paymentStep =
|
|
std::make_unique<BookPaymentStep<TIn, TOut>>(ctx, in, out);
|
|
ter = paymentStep->check(ctx);
|
|
r = std::move(paymentStep);
|
|
}
|
|
if (ter != tesSUCCESS)
|
|
return {ter, nullptr};
|
|
|
|
return {tesSUCCESS, std::move(r)};
|
|
}
|
|
|
|
std::pair<TER, std::unique_ptr<Step>>
|
|
make_BookStepII(StrandContext const& ctx, Issue const& in, Issue const& out)
|
|
{
|
|
return make_BookStepHelper<IOUAmount, IOUAmount>(ctx, in, out);
|
|
}
|
|
|
|
std::pair<TER, std::unique_ptr<Step>>
|
|
make_BookStepIX(StrandContext const& ctx, Issue const& in)
|
|
{
|
|
return make_BookStepHelper<IOUAmount, XRPAmount>(ctx, in, xrpIssue());
|
|
}
|
|
|
|
std::pair<TER, std::unique_ptr<Step>>
|
|
make_BookStepXI(StrandContext const& ctx, Issue const& out)
|
|
{
|
|
return make_BookStepHelper<XRPAmount, IOUAmount>(ctx, xrpIssue(), out);
|
|
}
|
|
|
|
} // namespace ripple
|