1#include <xrpl/tx/transactors/token/Clawback.h>
3#include <xrpl/beast/utility/Zero.h>
4#include <xrpl/core/ServiceRegistry.h>
5#include <xrpl/ledger/helpers/AccountRootHelpers.h>
6#include <xrpl/ledger/helpers/TokenHelpers.h>
7#include <xrpl/protocol/AccountID.h>
8#include <xrpl/protocol/Concepts.h>
9#include <xrpl/protocol/Feature.h>
10#include <xrpl/protocol/Indexes.h>
11#include <xrpl/protocol/Issue.h>
12#include <xrpl/protocol/LedgerFormats.h>
13#include <xrpl/protocol/MPTAmount.h>
14#include <xrpl/protocol/MPTIssue.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/STTx.h>
20#include <xrpl/protocol/TER.h>
21#include <xrpl/protocol/XRPAmount.h>
22#include <xrpl/tx/ApplyContext.h>
23#include <xrpl/tx/Transactor.h>
30template <Val
idIssueType T>
47 if (issuer == holder ||
isXRP(clawAmount) || clawAmount <= beast::kZero)
60 auto const mptHolder = ctx.
tx[~sfHolder];
61 auto const clawAmount = ctx.
tx[sfAmount];
67 if (ctx.
tx[sfAccount] == *mptHolder)
70 if (clawAmount.mpt() >
MPTAmount{kMaxMpTokenAmount} || clawAmount <= beast::kZero)
81 ctx.
tx[sfAmount].asset().value());
88template <Val
idIssueType T>
101 SLE const& sleIssuer,
108 if (!sleIssuer.
isFlag(lsfAllowTrustLineClawback) || sleIssuer.
isFlag(lsfNoFreeze))
111 auto const sleRippleState =
116 STAmount const balance = (*sleRippleState)[sfBalance];
119 if (balance > beast::kZero && issuer < holder)
123 if (balance < beast::kZero && issuer > holder)
141 ctx.
j) <= beast::kZero)
151 SLE const& sleIssuer,
157 auto const sleIssuance = ctx.
view.
read(issuanceKey);
161 if (!sleIssuance->isFlag(lsfMPTCanClawback))
164 if (sleIssuance->getAccountID(sfIssuer) != issuer)
176 ctx.
j) <= beast::kZero)
186 auto const clawAmount = ctx.
tx[sfAmount];
191 if (!sleIssuer || !sleHolder)
200 if (sleHolder->isFieldPresent(sfAMMID))
206 [&]<
typename T>(T
const&) {
209 ctx.
tx[sfAmount].asset().value());
212template <Val
idIssueType T>
225 clawAmount.
get<
Issue>().account = issuer;
226 if (holder == issuer)
247 auto clawAmount = ctx.
tx[sfAmount];
263 std::min(spendableAmount, clawAmount),
273 ctx_.tx[sfAmount].asset().value());
A generic endpoint for log messages.
State information when applying a tx.
beast::Journal const journal
static NotTEC preflight(PreflightContext const &ctx)
void visitInvariantEntry(bool isDelete, SLE::const_ref before, SLE::const_ref after) override
Inspect a single ledger entry modified by this transaction.
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.
A currency issued by an account.
AccountID const & getIssuer() const
constexpr MPTID const & getMptID() const
virtual Rules const & rules() const =0
Returns the tx processing rules.
virtual bool exists(Keylet const &k) const =0
Determine if a state item exists.
virtual SLE::const_pointer read(Keylet const &k) const =0
Return the state item associated with a key.
bool enabled(uint256 const &feature) const
Returns true if a feature is enabled.
constexpr TIss const & get() const
AccountID const & getIssuer() const
std::shared_ptr< STLedgerEntry const > const & const_ref
bool isFlag(std::uint32_t) const
bool isFieldPresent(SField const &field) const
Keylet mptokenIssuance(std::uint32_t seq, AccountID const &issuer) noexcept
Keylet mptoken(MPTID const &issuanceID, AccountID const &holder) noexcept
Keylet account(AccountID const &id) noexcept
AccountID root.
Keylet trustLine(AccountID const &id0, AccountID const &id1, Currency const ¤cy) noexcept
The index of a trust line for a given currency.
Use hash_* containers for keys that do not need a cryptographically secure hashing algorithm.
bool isXRP(AccountID const &c)
static TER applyHelper(ApplyContext &ctx)
static NotTEC preflightHelper(PreflightContext const &ctx)
TERSubset< CanCvtToNotTEC > NotTEC
TER preclaimHelper< Issue >(PreclaimContext const &ctx, SLE const &sleIssuer, STAmount const &clawAmount)
TER directSendNoFee(ApplyView &view, AccountID const &uSenderID, AccountID const &uReceiverID, STAmount const &saAmount, bool bCheckIssuer, beast::Journal j)
Calls static directSendNoFeeIOU if saAmount represents Issue.
NotTEC preflightHelper< MPTIssue >(PreflightContext const &ctx)
BaseUInt< 160, detail::AccountIDTag > AccountID
A 160-bit unsigned that uniquely identifies an account.
NotTEC preflightHelper< Issue >(PreflightContext const &ctx)
bool isTesSuccess(TER x) noexcept
TERSubset< CanCvtToTER > TER
TER preclaimHelper< MPTIssue >(PreclaimContext const &ctx, SLE const &sleIssuer, STAmount const &clawAmount)
TER applyHelper< MPTIssue >(ApplyContext &ctx)
bool isPseudoAccount(SLE::const_pointer sleAcct, std::set< SField const * > const &pseudoFieldFilter={})
Returns true if and only if sleAcct is a pseudo-account or specific pseudo-accounts in pseudoFieldFil...
TER applyHelper< Issue >(ApplyContext &ctx)
STAmount accountHolds(ReadView const &view, AccountID const &account, Currency const ¤cy, AccountID const &issuer, FreezeHandling zeroIfFrozen, beast::Journal j, SpendableHandling includeFullBalance=SpendableHandling::SimpleBalance)
static TER preclaimHelper(PreclaimContext const &ctx, SLE const &sleIssuer, STAmount const &clawAmount)
State information when determining if a tx is likely to claim a fee.
State information when preflighting a tx.