mirror of
https://github.com/XRPLF/rippled.git
synced 2025-12-06 17:27:55 +00:00
Amendment: - MPTokensV1 New Transactions: - MPTokenIssuanceCreate - MPTokenIssuanceDestroy - MPTokenIssuanceSet - MPTokenAuthorize Modified Transactions: - Payment - Clawback New Objects: - MPTokenIssuance - MPToken API updates: - ledger_entry - account_objects - ledger_data Other: - Add += and -= operators to ValueProxy Read full spec: https://github.com/XRPLF/XRPL-Standards/tree/master/XLS-0033d-multi-purpose-tokens --------- Co-authored-by: Shawn Xie <shawnxie920@gmail.com> Co-authored-by: Howard Hinnant <howard.hinnant@gmail.com> Co-authored-by: Ed Hennis <ed@ripple.com> Co-authored-by: John Freeman <jfreeman08@gmail.com>
143 lines
4.3 KiB
C++
143 lines
4.3 KiB
C++
//------------------------------------------------------------------------------
|
|
/*
|
|
This file is part of rippled: https://github.com/ripple/rippled
|
|
Copyright (c) 2024 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 <xrpld/app/tx/detail/MPTokenIssuanceCreate.h>
|
|
#include <xrpld/ledger/View.h>
|
|
#include <xrpl/protocol/Feature.h>
|
|
#include <xrpl/protocol/TxFlags.h>
|
|
#include <xrpl/protocol/st.h>
|
|
|
|
namespace ripple {
|
|
|
|
NotTEC
|
|
MPTokenIssuanceCreate::preflight(PreflightContext const& ctx)
|
|
{
|
|
if (!ctx.rules.enabled(featureMPTokensV1))
|
|
return temDISABLED;
|
|
|
|
if (auto const ret = preflight1(ctx); !isTesSuccess(ret))
|
|
return ret;
|
|
|
|
if (ctx.tx.getFlags() & tfMPTokenIssuanceCreateMask)
|
|
return temINVALID_FLAG;
|
|
|
|
if (auto const fee = ctx.tx[~sfTransferFee])
|
|
{
|
|
if (fee > maxTransferFee)
|
|
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 metadata = ctx.tx[~sfMPTokenMetadata])
|
|
{
|
|
if (metadata->length() == 0 ||
|
|
metadata->length() > maxMPTokenMetadataLength)
|
|
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 > maxMPTokenAmount)
|
|
return temMALFORMED;
|
|
}
|
|
return preflight2(ctx);
|
|
}
|
|
|
|
TER
|
|
MPTokenIssuanceCreate::create(
|
|
ApplyView& view,
|
|
beast::Journal journal,
|
|
MPTCreateArgs const& args)
|
|
{
|
|
auto const acct = view.peek(keylet::account(args.account));
|
|
if (!acct)
|
|
return tecINTERNAL;
|
|
|
|
if (args.priorBalance <
|
|
view.fees().accountReserve((*acct)[sfOwnerCount] + 1))
|
|
return tecINSUFFICIENT_RESERVE;
|
|
|
|
auto const mptIssuanceKeylet =
|
|
keylet::mptIssuance(args.sequence, args.account);
|
|
|
|
// create the MPTokenIssuance
|
|
{
|
|
auto const ownerNode = view.dirInsert(
|
|
keylet::ownerDir(args.account),
|
|
mptIssuanceKeylet,
|
|
describeOwnerDir(args.account));
|
|
|
|
if (!ownerNode)
|
|
return tecDIR_FULL;
|
|
|
|
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;
|
|
|
|
view.insert(mptIssuance);
|
|
}
|
|
|
|
// Update owner count.
|
|
adjustOwnerCount(view, acct, 1, journal);
|
|
|
|
return tesSUCCESS;
|
|
}
|
|
|
|
TER
|
|
MPTokenIssuanceCreate::doApply()
|
|
{
|
|
auto const& tx = ctx_.tx;
|
|
return create(
|
|
ctx_.view(),
|
|
ctx_.journal,
|
|
{.priorBalance = mPriorBalance,
|
|
.account = account_,
|
|
.sequence = tx.getSeqProxy().value(),
|
|
.flags = tx.getFlags(),
|
|
.maxAmount = tx[~sfMaximumAmount],
|
|
.assetScale = tx[~sfAssetScale],
|
|
.transferFee = tx[~sfTransferFee],
|
|
.metadata = tx[~sfMPTokenMetadata]});
|
|
}
|
|
|
|
} // namespace ripple
|