mirror of
https://github.com/XRPLF/rippled.git
synced 2026-06-08 03:07:13 +00:00
This shores up the View interface support for contextual transaction processing by putting params in the View, and provides support for replacing the open ledger with the open MetaView. Transaction metadata is now part of the View interface. Stacked MetaViews correctly apply their transaction metadata to the parent. * Add lastCloseTime to View * Add insertTx to View, implement in MetaView * Add View::txExists for transaction checking * Add Fees to View, cache fees in Ledger and MetaView * Use ViewFlags in View * Use tapENABLE_TESTING flag for features * Use cached Fees in View * Rename to ViewFlags * Move FreezeHandling to View.h, remove ViewAPIBasics.h * Remove BasicView::parent hack * Remove calls to getLedger in Transactors
427 lines
13 KiB
C++
427 lines
13 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 <BeastConfig.h>
|
|
#include <ripple/app/paths/Tuning.h>
|
|
#include <ripple/app/paths/RippleCalc.h>
|
|
#include <ripple/app/paths/cursor/PathCursor.h>
|
|
#include <ripple/basics/Log.h>
|
|
|
|
namespace ripple {
|
|
namespace path {
|
|
|
|
namespace {
|
|
|
|
static
|
|
TER
|
|
deleteOffers (View& view,
|
|
OfferSet const& offers)
|
|
{
|
|
for (auto& e: offers)
|
|
if (TER r = offerDelete (view,
|
|
view.peek(keylet::offer(e))))
|
|
return r;
|
|
return tesSUCCESS;
|
|
}
|
|
|
|
} // namespace
|
|
|
|
RippleCalc::Output RippleCalc::rippleCalculate (
|
|
PaymentView& view,
|
|
|
|
// Compute paths using this ledger entry set. Up to caller to actually
|
|
// apply to ledger.
|
|
|
|
// Issuer:
|
|
// XRP: xrpAccount()
|
|
// non-XRP: uSrcAccountID (for any issuer) or another account with
|
|
// trust node.
|
|
STAmount const& saMaxAmountReq, // --> -1 = no limit.
|
|
|
|
// Issuer:
|
|
// XRP: xrpAccount()
|
|
// non-XRP: uDstAccountID (for any issuer) or another account with
|
|
// trust node.
|
|
STAmount const& saDstAmountReq,
|
|
|
|
AccountID const& uDstAccountID,
|
|
AccountID const& uSrcAccountID,
|
|
|
|
// A set of paths that are included in the transaction that we'll
|
|
// explore for liquidity.
|
|
STPathSet const& spsPaths,
|
|
Input const* const pInputs)
|
|
{
|
|
RippleCalc rc (
|
|
view,
|
|
saMaxAmountReq,
|
|
saDstAmountReq,
|
|
uDstAccountID,
|
|
uSrcAccountID,
|
|
spsPaths);
|
|
if (pInputs != nullptr)
|
|
{
|
|
rc.inputFlags = *pInputs;
|
|
}
|
|
|
|
auto result = rc.rippleCalculate ();
|
|
Output output;
|
|
output.setResult (result);
|
|
output.actualAmountIn = rc.actualAmountIn_;
|
|
output.actualAmountOut = rc.actualAmountOut_;
|
|
output.pathStateList = rc.pathStateList_;
|
|
|
|
return output;
|
|
}
|
|
|
|
bool RippleCalc::addPathState(STPath const& path, TER& resultCode)
|
|
{
|
|
auto pathState = std::make_shared<PathState> (
|
|
view, saDstAmountReq_, saMaxAmountReq_);
|
|
|
|
if (!pathState)
|
|
{
|
|
resultCode = temUNKNOWN;
|
|
return false;
|
|
}
|
|
|
|
pathState->expandPath (
|
|
path,
|
|
uDstAccountID_,
|
|
uSrcAccountID_);
|
|
|
|
if (pathState->status() == tesSUCCESS)
|
|
pathState->checkNoRipple (uDstAccountID_, uSrcAccountID_);
|
|
|
|
if (pathState->status() == tesSUCCESS)
|
|
pathState->checkFreeze ();
|
|
|
|
pathState->setIndex (pathStateList_.size ());
|
|
|
|
WriteLog (lsDEBUG, RippleCalc)
|
|
<< "rippleCalc: Build direct:"
|
|
<< " status: " << transToken (pathState->status());
|
|
|
|
// Return if malformed.
|
|
if (isTemMalformed (pathState->status()))
|
|
{
|
|
resultCode = pathState->status();
|
|
return false;
|
|
}
|
|
|
|
if (pathState->status () == tesSUCCESS)
|
|
{
|
|
resultCode = pathState->status();
|
|
pathStateList_.push_back (pathState);
|
|
}
|
|
else if (pathState->status () != terNO_LINE)
|
|
{
|
|
resultCode = pathState->status();
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
// OPTIMIZE: When calculating path increment, note if increment consumes all
|
|
// liquidity. No need to revisit path in the future if all liquidity is used.
|
|
|
|
// <-- TER: Only returns tepPATH_PARTIAL if partialPaymentAllowed.
|
|
TER RippleCalc::rippleCalculate ()
|
|
{
|
|
WriteLog (lsTRACE, RippleCalc)
|
|
<< "rippleCalc>"
|
|
<< " saMaxAmountReq_:" << saMaxAmountReq_
|
|
<< " saDstAmountReq_:" << saDstAmountReq_;
|
|
|
|
TER resultCode = temUNCERTAIN;
|
|
permanentlyUnfundedOffers_.clear ();
|
|
mumSource_.clear ();
|
|
|
|
// YYY Might do basic checks on src and dst validity as per doPayment.
|
|
|
|
// Incrementally search paths.
|
|
if (inputFlags.defaultPathsAllowed)
|
|
{
|
|
if (!addPathState (STPath(), resultCode))
|
|
return resultCode;
|
|
}
|
|
else if (spsPaths_.empty ())
|
|
{
|
|
WriteLog (lsDEBUG, RippleCalc)
|
|
<< "rippleCalc: Invalid transaction:"
|
|
<< "No paths and direct ripple not allowed.";
|
|
|
|
return temRIPPLE_EMPTY;
|
|
}
|
|
|
|
// Build a default path. Use saDstAmountReq_ and saMaxAmountReq_ to imply
|
|
// nodes.
|
|
// XXX Might also make a XRP bridge by default.
|
|
|
|
WriteLog (lsTRACE, RippleCalc)
|
|
<< "rippleCalc: Paths in set: " << spsPaths_.size ();
|
|
|
|
// Now expand the path state.
|
|
for (auto const& spPath: spsPaths_)
|
|
{
|
|
if (!addPathState (spPath, resultCode))
|
|
return resultCode;
|
|
}
|
|
|
|
if (resultCode != tesSUCCESS)
|
|
return (resultCode == temUNCERTAIN) ? terNO_LINE : resultCode;
|
|
|
|
resultCode = temUNCERTAIN;
|
|
|
|
actualAmountIn_ = saMaxAmountReq_.zeroed();
|
|
actualAmountOut_ = saDstAmountReq_.zeroed();
|
|
|
|
// When processing, we don't want to complicate directory walking with
|
|
// deletion.
|
|
const std::uint64_t uQualityLimit = inputFlags.limitQuality ?
|
|
getRate (saDstAmountReq_, saMaxAmountReq_) : 0;
|
|
|
|
// Offers that became unfunded.
|
|
OfferSet unfundedOffersFromBestPaths;
|
|
|
|
int iPass = 0;
|
|
|
|
while (resultCode == temUNCERTAIN)
|
|
{
|
|
int iBest = -1;
|
|
int iDry = 0;
|
|
|
|
// True, if ever computed multi-quality.
|
|
bool multiQuality = false;
|
|
|
|
// Find the best path.
|
|
for (auto pathState : pathStateList_)
|
|
{
|
|
if (pathState->quality())
|
|
// Only do active paths.
|
|
{
|
|
// If computing the only non-dry path, compute multi-quality.
|
|
multiQuality = ((pathStateList_.size () - iDry) == 1);
|
|
|
|
// Update to current amount processed.
|
|
pathState->reset (actualAmountIn_, actualAmountOut_);
|
|
|
|
// Error if done, output met.
|
|
PathCursor pc(*this, *pathState, multiQuality);
|
|
pc.nextIncrement ();
|
|
|
|
// Compute increment.
|
|
WriteLog (lsDEBUG, RippleCalc)
|
|
<< "rippleCalc: AFTER:"
|
|
<< " mIndex=" << pathState->index()
|
|
<< " uQuality=" << pathState->quality()
|
|
<< " rate=" << amountFromRate (pathState->quality());
|
|
|
|
if (!pathState->quality())
|
|
{
|
|
// Path was dry.
|
|
|
|
++iDry;
|
|
}
|
|
else if (pathState->outPass() == zero)
|
|
{
|
|
// Path is not dry, but moved no funds
|
|
// This should never happen. Consider the path dry
|
|
|
|
WriteLog (lsWARNING, RippleCalc)
|
|
<< "rippelCalc: Non-dry path moves no funds";
|
|
|
|
assert (false);
|
|
|
|
pathState->setQuality (0);
|
|
++iDry;
|
|
}
|
|
else
|
|
{
|
|
CondLog (!pathState->inPass() || !pathState->outPass(),
|
|
lsDEBUG, RippleCalc)
|
|
<< "rippleCalc: better:"
|
|
<< " uQuality="
|
|
<< amountFromRate (pathState->quality())
|
|
<< " inPass()=" << pathState->inPass()
|
|
<< " saOutPass=" << pathState->outPass();
|
|
|
|
assert (pathState->inPass() && pathState->outPass());
|
|
|
|
if ((!inputFlags.limitQuality ||
|
|
pathState->quality() <= uQualityLimit)
|
|
// Quality is not limited or increment has allowed
|
|
// quality.
|
|
&& (iBest < 0
|
|
// Best is not yet set.
|
|
|| PathState::lessPriority (
|
|
*pathStateList_[iBest], *pathState)))
|
|
// Current is better than set.
|
|
{
|
|
WriteLog (lsDEBUG, RippleCalc)
|
|
<< "rippleCalc: better:"
|
|
<< " mIndex=" << pathState->index()
|
|
<< " uQuality=" << pathState->quality()
|
|
<< " rate="
|
|
<< amountFromRate (pathState->quality())
|
|
<< " inPass()=" << pathState->inPass()
|
|
<< " saOutPass=" << pathState->outPass();
|
|
|
|
iBest = pathState->index ();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
++iPass;
|
|
if (ShouldLog (lsDEBUG, RippleCalc))
|
|
{
|
|
WriteLog (lsDEBUG, RippleCalc)
|
|
<< "rippleCalc: Summary:"
|
|
<< " Pass: " << iPass
|
|
<< " Dry: " << iDry
|
|
<< " Paths: " << pathStateList_.size ();
|
|
for (auto pathState: pathStateList_)
|
|
{
|
|
WriteLog (lsDEBUG, RippleCalc)
|
|
<< "rippleCalc: "
|
|
<< "Summary: " << pathState->index()
|
|
<< " rate: "
|
|
<< amountFromRate (pathState->quality())
|
|
<< " quality:" << pathState->quality()
|
|
<< " best: " << (iBest == pathState->index ());
|
|
}
|
|
}
|
|
|
|
if (iBest >= 0)
|
|
{
|
|
// Apply best path.
|
|
auto pathState = pathStateList_[iBest];
|
|
|
|
WriteLog (lsDEBUG, RippleCalc)
|
|
<< "rippleCalc: best:"
|
|
<< " uQuality="
|
|
<< amountFromRate (pathState->quality())
|
|
<< " inPass()=" << pathState->inPass()
|
|
<< " saOutPass=" << pathState->outPass();
|
|
|
|
// Record best pass' offers that became unfunded for deletion on
|
|
// success.
|
|
|
|
unfundedOffersFromBestPaths.insert (
|
|
pathState->unfundedOffers().begin (),
|
|
pathState->unfundedOffers().end ());
|
|
|
|
// Apply best pass' view
|
|
pathState->view().apply(view);
|
|
|
|
actualAmountIn_ += pathState->inPass();
|
|
actualAmountOut_ += pathState->outPass();
|
|
|
|
if (pathState->allLiquidityConsumed() || multiQuality)
|
|
{
|
|
++iDry;
|
|
pathState->setQuality(0);
|
|
}
|
|
|
|
if (actualAmountOut_ == saDstAmountReq_)
|
|
{
|
|
// Done. Delivered requested amount.
|
|
|
|
resultCode = tesSUCCESS;
|
|
}
|
|
else if (actualAmountOut_ > saDstAmountReq_)
|
|
{
|
|
WriteLog (lsFATAL, RippleCalc)
|
|
<< "rippleCalc: TOO MUCH:"
|
|
<< " actualAmountOut_:" << actualAmountOut_
|
|
<< " saDstAmountReq_:" << saDstAmountReq_;
|
|
|
|
return tefEXCEPTION; // TEMPORARY
|
|
assert (false);
|
|
}
|
|
else if (actualAmountIn_ != saMaxAmountReq_ &&
|
|
iDry != pathStateList_.size ())
|
|
{
|
|
// Have not met requested amount or max send, try to do
|
|
// more. Prepare for next pass.
|
|
//
|
|
// Merge best pass' umReverse.
|
|
mumSource_.insert (
|
|
pathState->reverse().begin (), pathState->reverse().end ());
|
|
|
|
if (iPass >= PAYMENT_MAX_LOOPS)
|
|
{
|
|
// This payment is taking too many passes
|
|
|
|
WriteLog (lsERROR, RippleCalc)
|
|
<< "rippleCalc: pass limit";
|
|
|
|
resultCode = telFAILED_PROCESSING;
|
|
}
|
|
|
|
}
|
|
else if (!inputFlags.partialPaymentAllowed)
|
|
{
|
|
// Have sent maximum allowed. Partial payment not allowed.
|
|
|
|
resultCode = tecPATH_PARTIAL;
|
|
}
|
|
else
|
|
{
|
|
// Have sent maximum allowed. Partial payment allowed. Success.
|
|
|
|
resultCode = tesSUCCESS;
|
|
}
|
|
}
|
|
// Not done and ran out of paths.
|
|
else if (!inputFlags.partialPaymentAllowed)
|
|
{
|
|
// Partial payment not allowed.
|
|
resultCode = tecPATH_PARTIAL;
|
|
}
|
|
// Partial payment ok.
|
|
else if (!actualAmountOut_)
|
|
{
|
|
// No payment at all.
|
|
resultCode = tecPATH_DRY;
|
|
}
|
|
else
|
|
{
|
|
// Don't apply any payment increments
|
|
resultCode = tesSUCCESS;
|
|
}
|
|
}
|
|
|
|
if (resultCode == tesSUCCESS)
|
|
{
|
|
resultCode = deleteOffers(view, unfundedOffersFromBestPaths);
|
|
if (resultCode == tesSUCCESS)
|
|
resultCode = deleteOffers(view, permanentlyUnfundedOffers_);
|
|
}
|
|
|
|
// If isOpenLedger, then ledger is not final, can vote no.
|
|
if (resultCode == telFAILED_PROCESSING && !inputFlags.isLedgerOpen)
|
|
return tecFAILED_PROCESSING;
|
|
return resultCode;
|
|
}
|
|
|
|
} // path
|
|
} // ripple
|