1#include <xrpl/tx/transactors/lending/LoanManage.h>
3#include <xrpl/basics/Log.h>
4#include <xrpl/basics/Number.h>
5#include <xrpl/beast/utility/Zero.h>
6#include <xrpl/core/ServiceRegistry.h>
7#include <xrpl/ledger/ApplyView.h>
8#include <xrpl/ledger/View.h>
9#include <xrpl/ledger/helpers/LendingHelpers.h>
10#include <xrpl/ledger/helpers/TokenHelpers.h>
11#include <xrpl/protocol/Asset.h>
12#include <xrpl/protocol/Feature.h>
13#include <xrpl/protocol/Indexes.h>
14#include <xrpl/protocol/LedgerFormats.h>
15#include <xrpl/protocol/Protocol.h>
16#include <xrpl/protocol/SField.h>
17#include <xrpl/protocol/STAmount.h>
18#include <xrpl/protocol/STLedgerEntry.h>
19#include <xrpl/protocol/STTakesAsset.h>
20#include <xrpl/protocol/STTx.h>
21#include <xrpl/protocol/TER.h>
22#include <xrpl/protocol/TxFlags.h>
23#include <xrpl/protocol/Units.h>
24#include <xrpl/protocol/XRPAmount.h>
25#include <xrpl/tx/Transactor.h>
40 return tfLoanManageMask;
46 if (ctx.
tx[sfLoanID] == beast::kZero)
50 if (
auto const flagField = ctx.
tx[~sfFlags]; flagField && (*flagField != 0u))
53 if ((flags & (flags - 1)) != 0)
55 JLOG(ctx.
j.
warn()) <<
"LoanManage: Only one of tfLoanDefault, tfLoanImpair, or "
56 "tfLoanUnimpair can be set.";
67 auto const& tx = ctx.
tx;
69 auto const account = tx[sfAccount];
70 auto const loanID = tx[sfLoanID];
75 JLOG(ctx.
j.
warn()) <<
"Loan does not exist.";
84 if (loanSle->isFlag(lsfLoanDefault))
86 JLOG(ctx.
j.
warn()) <<
"Loan is in default. A defaulted loan can not be modified.";
89 if (loanSle->isFlag(lsfLoanImpaired) && tx.isFlag(tfLoanImpair))
91 JLOG(ctx.
j.
warn()) <<
"Loan is impaired. A loan can not be impaired twice.";
94 if (!(loanSle->isFlag(lsfLoanImpaired) || loanSle->isFlag(lsfLoanDefault)) &&
95 (tx.isFlag(tfLoanUnimpair)))
97 JLOG(ctx.
j.
warn()) <<
"Loan is unimpaired. Can not be unimpaired again.";
100 if (loanSle->at(sfPaymentRemaining) == 0)
102 JLOG(ctx.
j.
warn()) <<
"Loan is fully paid. A loan can not be modified "
103 "after it is fully paid.";
106 if (tx.isFlag(tfLoanDefault) &&
107 !
hasExpired(ctx.
view, loanSle->at(sfNextPaymentDueDate) + loanSle->at(sfGracePeriod)))
109 JLOG(ctx.
j.
warn()) <<
"A loan can not be defaulted before the next payment due date.";
113 auto const loanBrokerID = loanSle->at(sfLoanBrokerID);
120 if (loanBrokerSle->at(sfOwner) != account)
122 JLOG(ctx.
j.
warn()) <<
"LoanBroker for Loan does not belong to the account. LoanManage "
123 "can only be submitted by the Loan Broker.";
144 return loanSle->at(sfTotalValueOutstanding) - loanSle->at(sfManagementFeeOutstanding);
153 Asset const& vaultAsset,
158 std::int32_t const loanScale = loanSle->at(sfLoanScale);
159 auto brokerDebtTotalProxy = brokerSle->at(sfDebtTotal);
164 TenthBips32 const coverRateMinimum{brokerSle->at(sfCoverRateMinimum)};
165 TenthBips32 const coverRateLiquidation{brokerSle->at(sfCoverRateLiquidation)};
166 auto const defaultCovered = [&]() {
169 auto const minimumCover =
tenthBipsOfValue(brokerDebtTotalProxy.value(), coverRateMinimum);
180 auto const coverAvailable = *brokerSle->at(sfCoverAvailable);
182 return std::min(covered, coverAvailable);
185 auto const vaultDefaultAmount = totalDefaultAmount - defaultCovered;
196 auto vaultTotalProxy = vaultSle->at(sfAssetsTotal);
197 auto vaultAvailableProxy = vaultSle->at(sfAssetsAvailable);
199 if (vaultTotalProxy < vaultDefaultAmount)
202 JLOG(j.
warn()) <<
"Vault total assets is less than the vault default amount";
209 vaultTotalProxy -= vaultDefaultRounded;
212 vaultAvailableProxy += defaultCovered;
213 if (*vaultAvailableProxy > *vaultTotalProxy && !vaultAsset.
integral())
215 auto const difference = vaultAvailableProxy - vaultTotalProxy;
216 JLOG(j.
debug()) <<
"Vault assets available: " << *vaultAvailableProxy <<
"("
217 << vaultAvailableProxy.value().exponent()
218 <<
"), Total: " << *vaultTotalProxy <<
"("
219 << vaultTotalProxy.value().exponent() <<
"), Difference: " << difference
220 <<
"(" << difference.exponent() <<
")";
221 if (vaultAvailableProxy.value().exponent() - difference.exponent() > 13)
225 JLOG(j.
debug()) <<
"Difference between vault assets available and total is "
226 "dust. Set both to the larger value.";
227 vaultTotalProxy = vaultAvailableProxy;
230 if (*vaultAvailableProxy > *vaultTotalProxy)
233 JLOG(j.
fatal()) <<
"Vault assets available must not be greater "
234 "than assets outstanding. Available: "
235 << *vaultAvailableProxy <<
", Total: " << *vaultTotalProxy;
241 if (loanSle->isFlag(lsfLoanImpaired))
243 auto vaultLossUnrealizedProxy = vaultSle->at(sfLossUnrealized);
244 if (vaultLossUnrealizedProxy < totalDefaultAmount)
247 JLOG(j.
warn()) <<
"Vault unrealized loss is less than the default amount";
252 vaultLossUnrealizedProxy, -totalDefaultAmount, vaultAsset, vaultScale);
254 view.update(vaultSle);
263 auto coverAvailableProxy = brokerSle->at(sfCoverAvailable);
264 if (coverAvailableProxy < defaultCovered)
267 JLOG(j.
warn()) <<
"LoanBroker cover available is less than amount covered";
271 coverAvailableProxy -= defaultCovered;
272 view.update(brokerSle);
276 loanSle->setFlag(lsfLoanDefault);
278 loanSle->at(sfTotalValueOutstanding) = 0;
279 loanSle->at(sfPaymentRemaining) = 0;
280 loanSle->at(sfPrincipalOutstanding) = 0;
281 loanSle->at(sfManagementFeeOutstanding) = 0;
284 loanSle->at(sfNextPaymentDueDate) = 0;
285 view.update(loanSle);
291 brokerSle->at(sfAccount),
292 vaultSle->at(sfAccount),
293 STAmount{vaultAsset, defaultCovered},
303 Asset const& vaultAsset,
314 auto vaultLossUnrealizedProxy = vaultSle->at(sfLossUnrealized);
316 if (vaultLossUnrealizedProxy > vaultSle->at(sfAssetsTotal) - vaultSle->at(sfAssetsAvailable))
320 JLOG(j.
warn()) <<
"Vault unrealized loss is too large, and will "
321 "corrupt the vault.";
324 view.update(vaultSle);
327 loanSle->setFlag(lsfLoanImpaired);
328 auto loanNextDueProxy = loanSle->at(sfNextPaymentDueDate);
333 loanNextDueProxy =
view.parentCloseTime().time_since_epoch().count();
335 view.update(loanSle);
345 Asset const& vaultAsset,
354 auto vaultLossUnrealizedProxy = vaultSle->at(sfLossUnrealized);
356 if (vaultLossUnrealizedProxy < lossReversed)
359 JLOG(j.
warn()) <<
"Vault unrealized loss is less than the amount to be cleared";
366 view.update(vaultSle);
369 loanSle->clearFlag(lsfLoanImpaired);
370 auto const paymentInterval = loanSle->at(sfPaymentInterval);
371 auto const normalPaymentDueDate =
372 std::max(loanSle->at(sfPreviousPaymentDueDate), loanSle->at(sfStartDate)) + paymentInterval;
376 loanSle->at(sfNextPaymentDueDate) = normalPaymentDueDate;
381 loanSle->at(sfNextPaymentDueDate) =
382 view.parentCloseTime().time_since_epoch().count() + paymentInterval;
384 view.update(loanSle);
392 auto const& tx =
ctx_.tx;
395 auto const loanID = tx[sfLoanID];
400 auto const brokerID = loanSle->at(sfLoanBrokerID);
408 auto const vaultAsset = vaultSle->at(sfAsset);
410 auto const result = [&]() ->
TER {
413 if (tx.isFlag(tfLoanDefault))
415 if (tx.isFlag(tfLoanImpair))
417 if (tx.isFlag(tfLoanUnimpair))
A generic endpoint for log messages.
Writeable view to a ledger, for applying a transaction.
void visitInvariantEntry(bool isDelete, SLE::const_ref before, SLE::const_ref after) override
Inspect a single ledger entry modified by this transaction.
static TER defaultLoan(ApplyView &view, SLE::ref loanSle, SLE::ref brokerSle, SLE::ref vaultSle, Asset const &vaultAsset, beast::Journal j)
Helper function that might be needed by other transactors.
static TER preclaim(PreclaimContext const &ctx)
bool finalizeInvariants(STTx const &tx, TER result, XRPAmount fee, ReadView const &view, beast::Journal const &j) override
Check transaction-specific post-conditions after all entries have been visited.
static std::uint32_t getFlagsMask(PreflightContext const &ctx)
static TER unimpairLoan(ApplyView &view, SLE::ref loanSle, SLE::ref vaultSle, Asset const &vaultAsset, beast::Journal j)
Helper function that might be needed by other transactors.
static TER impairLoan(ApplyView &view, SLE::ref loanSle, SLE::ref vaultSle, Asset const &vaultAsset, beast::Journal j)
Helper function that might be needed by other transactors.
static NotTEC preflight(PreflightContext const &ctx)
static bool checkExtraFeatures(PreflightContext const &ctx)
Number is a floating point type that can represent a wide range of values.
virtual SLE::const_pointer read(Keylet const &k) const =0
Return the state item associated with a key.
std::shared_ptr< STLedgerEntry > const & ref
std::shared_ptr< STLedgerEntry const > const & const_ref
Keylet loanBroker(AccountID const &owner, std::uint32_t seq) noexcept
Keylet loan(uint256 const &loanBrokerID, std::uint32_t loanSeq) noexcept
Keylet vault(AccountID const &owner, std::uint32_t seq) noexcept
Use hash_* containers for keys that do not need a cryptographically secure hashing algorithm.
constexpr T tenthBipsOfValue(T value, TenthBips< TBips > bips)
bool hasExpired(ReadView const &view, std::optional< std::uint32_t > const &exp)
Determines whether the given expiration time has passed.
int getAssetsTotalScale(SLE::const_ref vaultSle)
void adjustImpreciseNumber(NumberProxy value, Number const &adjustment, Asset const &asset, int vaultScale)
TenthBips< std::uint32_t > TenthBips32
TERSubset< CanCvtToNotTEC > NotTEC
static Number owedToVault(SLE::ref loanSle)
TER accountSend(ApplyView &view, AccountID const &from, AccountID const &to, STAmount const &saAmount, beast::Journal j, WaiveTransferFee waiveFee=WaiveTransferFee::No, AllowMPTOverflow allowOverflow=AllowMPTOverflow::No)
Calls static accountSendIOU if saAmount represents Issue.
void roundToAsset(A const &asset, Number &value)
Round an arbitrary precision Number IN PLACE to the precision of a given Asset.
bool isTesSuccess(TER x) noexcept
TERSubset< CanCvtToTER > TER
void associateAsset(STLedgerEntry &sle, Asset const &asset)
Associate an Asset with all sMD_NeedsAsset fields in a ledger entry.
constexpr FlagValue tfUniversalMask
bool checkLendingProtocolDependencies(Rules const &rules, STTx const &tx)
State information when determining if a tx is likely to claim a fee.
State information when preflighting a tx.