mirror of
https://github.com/XRPLF/rippled.git
synced 2026-06-04 01:06:48 +00:00
219 lines
6.1 KiB
C++
219 lines
6.1 KiB
C++
#include <xrpl/tx/transactors/token/MPTokenIssuanceCreate.h>
|
|
|
|
#include <xrpl/basics/Expected.h>
|
|
#include <xrpl/beast/utility/Journal.h>
|
|
#include <xrpl/beast/utility/Zero.h>
|
|
#include <xrpl/core/ServiceRegistry.h>
|
|
#include <xrpl/ledger/ApplyView.h>
|
|
#include <xrpl/ledger/ReadView.h>
|
|
#include <xrpl/ledger/helpers/AccountRootHelpers.h>
|
|
#include <xrpl/ledger/helpers/DirectoryHelpers.h>
|
|
#include <xrpl/ledger/helpers/SponsorHelpers.h>
|
|
#include <xrpl/protocol/Feature.h>
|
|
#include <xrpl/protocol/Indexes.h>
|
|
#include <xrpl/protocol/Protocol.h>
|
|
#include <xrpl/protocol/SField.h>
|
|
#include <xrpl/protocol/STLedgerEntry.h>
|
|
#include <xrpl/protocol/STTx.h>
|
|
#include <xrpl/protocol/TER.h>
|
|
#include <xrpl/protocol/TxFlags.h>
|
|
#include <xrpl/protocol/UintTypes.h>
|
|
#include <xrpl/protocol/XRPAmount.h>
|
|
#include <xrpl/tx/Transactor.h>
|
|
|
|
#include <cstdint>
|
|
#include <memory>
|
|
#include <utility>
|
|
|
|
namespace xrpl {
|
|
|
|
bool
|
|
MPTokenIssuanceCreate::checkExtraFeatures(PreflightContext const& ctx)
|
|
{
|
|
if (ctx.tx.isFieldPresent(sfDomainID) &&
|
|
!(ctx.rules.enabled(featurePermissionedDomains) &&
|
|
ctx.rules.enabled(featureSingleAssetVault)))
|
|
return false;
|
|
|
|
if (ctx.tx.isFieldPresent(sfMutableFlags) && !ctx.rules.enabled(featureDynamicMPT))
|
|
return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
std::uint32_t
|
|
MPTokenIssuanceCreate::getFlagsMask(PreflightContext const& ctx)
|
|
{
|
|
// This mask is only compared against sfFlags
|
|
return tfMPTokenIssuanceCreateMask;
|
|
}
|
|
|
|
NotTEC
|
|
MPTokenIssuanceCreate::preflight(PreflightContext const& ctx)
|
|
{
|
|
// If the mutable flags field is included, at least one flag must be
|
|
// specified.
|
|
if (auto const mutableFlags = ctx.tx[~sfMutableFlags]; mutableFlags &&
|
|
((*mutableFlags == 0u) || ((*mutableFlags & tmfMPTokenIssuanceCreateMutableMask) != 0u)))
|
|
return temINVALID_FLAG;
|
|
|
|
if (auto const fee = ctx.tx[~sfTransferFee])
|
|
{
|
|
if (fee > kMAX_TRANSFER_FEE)
|
|
return temBAD_TRANSFER_FEE;
|
|
|
|
// If a non-zero TransferFee is set then the tfTransferable flag
|
|
// must also be set.
|
|
if (fee > 0u && !ctx.tx.isFlag(tfMPTCanTransfer))
|
|
return temMALFORMED;
|
|
}
|
|
|
|
if (auto const domain = ctx.tx[~sfDomainID])
|
|
{
|
|
if (*domain == beast::kZERO)
|
|
return temMALFORMED;
|
|
|
|
// Domain present implies that MPTokenIssuance is not public
|
|
if ((ctx.tx.getFlags() & tfMPTRequireAuth) == 0)
|
|
return temMALFORMED;
|
|
}
|
|
|
|
if (auto const metadata = ctx.tx[~sfMPTokenMetadata])
|
|
{
|
|
if (metadata->empty() || metadata->length() > kMAX_MP_TOKEN_METADATA_LENGTH)
|
|
return temMALFORMED;
|
|
}
|
|
|
|
// Check if maximumAmount is within unsigned 63 bit range
|
|
if (auto const maxAmt = ctx.tx[~sfMaximumAmount])
|
|
{
|
|
if (maxAmt == 0)
|
|
return temMALFORMED;
|
|
|
|
if (maxAmt > kMAX_MP_TOKEN_AMOUNT)
|
|
return temMALFORMED;
|
|
}
|
|
return tesSUCCESS;
|
|
}
|
|
|
|
Expected<MPTID, TER>
|
|
MPTokenIssuanceCreate::create(
|
|
ApplyView& view,
|
|
STTx const& tx,
|
|
beast::Journal journal,
|
|
MPTCreateArgs const& args)
|
|
{
|
|
auto const acct = view.peek(keylet::account(args.account));
|
|
if (!acct)
|
|
return Unexpected(tecINTERNAL); // LCOV_EXCL_LINE
|
|
|
|
SLE::pointer sponsorSle;
|
|
if (!isPseudoAccount(acct))
|
|
{
|
|
auto sle = getTxReserveSponsor(view, tx);
|
|
if (!sle)
|
|
return Unexpected(sle.error());
|
|
sponsorSle = std::move(*sle);
|
|
}
|
|
|
|
if (args.priorBalance)
|
|
{
|
|
if (auto const ret = checkInsufficientReserve(
|
|
view, tx, acct, *(args.priorBalance), sponsorSle, 1, 0, journal);
|
|
!isTesSuccess(ret))
|
|
return Unexpected(ret); // tecINSUFFICIENT_RESERVE
|
|
}
|
|
|
|
auto const mptId = makeMptID(args.sequence, args.account);
|
|
auto const mptIssuanceKeylet = keylet::mptIssuance(mptId);
|
|
|
|
// create the MPTokenIssuance
|
|
{
|
|
auto const ownerNode = view.dirInsert(
|
|
keylet::ownerDir(args.account), mptIssuanceKeylet, describeOwnerDir(args.account));
|
|
|
|
if (!ownerNode)
|
|
return Unexpected(tecDIR_FULL); // LCOV_EXCL_LINE
|
|
|
|
auto mptIssuance = std::make_shared<SLE>(mptIssuanceKeylet);
|
|
(*mptIssuance)[sfFlags] = args.flags & ~tfUniversal;
|
|
(*mptIssuance)[sfIssuer] = args.account;
|
|
(*mptIssuance)[sfOutstandingAmount] = 0;
|
|
(*mptIssuance)[sfOwnerNode] = *ownerNode;
|
|
(*mptIssuance)[sfSequence] = args.sequence;
|
|
|
|
if (args.maxAmount)
|
|
(*mptIssuance)[sfMaximumAmount] = *args.maxAmount;
|
|
|
|
if (args.assetScale)
|
|
(*mptIssuance)[sfAssetScale] = *args.assetScale;
|
|
|
|
if (args.transferFee)
|
|
(*mptIssuance)[sfTransferFee] = *args.transferFee;
|
|
|
|
if (args.metadata)
|
|
(*mptIssuance)[sfMPTokenMetadata] = *args.metadata;
|
|
|
|
if (args.domainId)
|
|
(*mptIssuance)[sfDomainID] = *args.domainId;
|
|
|
|
if (args.mutableFlags)
|
|
(*mptIssuance)[sfMutableFlags] = *args.mutableFlags;
|
|
|
|
addSponsorToLedgerEntry(mptIssuance, sponsorSle);
|
|
|
|
view.insert(mptIssuance);
|
|
}
|
|
|
|
// Update owner count.
|
|
adjustOwnerCount(view, acct, sponsorSle, 1, journal);
|
|
|
|
return mptId;
|
|
}
|
|
|
|
TER
|
|
MPTokenIssuanceCreate::doApply()
|
|
{
|
|
auto const& tx = ctx_.tx;
|
|
auto const result = create(
|
|
view(),
|
|
tx,
|
|
j_,
|
|
{
|
|
.priorBalance = preFeeBalance_,
|
|
.account = account_,
|
|
.sequence = tx.getSeqValue(),
|
|
.flags = tx.getFlags(),
|
|
.maxAmount = tx[~sfMaximumAmount],
|
|
.assetScale = tx[~sfAssetScale],
|
|
.transferFee = tx[~sfTransferFee],
|
|
.metadata = tx[~sfMPTokenMetadata],
|
|
.domainId = tx[~sfDomainID],
|
|
.mutableFlags = tx[~sfMutableFlags],
|
|
});
|
|
return result ? tesSUCCESS : result.error();
|
|
}
|
|
|
|
void
|
|
MPTokenIssuanceCreate::visitInvariantEntry(
|
|
bool,
|
|
std::shared_ptr<SLE const> const&,
|
|
std::shared_ptr<SLE const> const&)
|
|
{
|
|
// No transaction-specific invariants yet (future work).
|
|
}
|
|
|
|
bool
|
|
MPTokenIssuanceCreate::finalizeInvariants(
|
|
STTx const&,
|
|
TER,
|
|
XRPAmount,
|
|
ReadView const&,
|
|
beast::Journal const&)
|
|
{
|
|
// No transaction-specific invariants yet (future work).
|
|
return true;
|
|
}
|
|
|
|
} // namespace xrpl
|