//------------------------------------------------------------------------------ /* 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 #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace ripple { // Templates so preflight does the right thing with T::ConsequencesFactory. // // This could be done more easily using if constexpr, but Visual Studio // 2017 doesn't handle if constexpr correctly. So once we're no longer // building with Visual Studio 2017 we can consider replacing the four // templates with a single template function that uses if constexpr. // // For Transactor::Normal template < class T, std::enable_if_t = 0> TxConsequences consequences_helper(PreflightContext const& ctx) { return TxConsequences(ctx.tx); }; // For Transactor::Blocker template < class T, std::enable_if_t = 0> TxConsequences consequences_helper(PreflightContext const& ctx) { return TxConsequences(ctx.tx, TxConsequences::blocker); }; // For Transactor::Custom template < class T, std::enable_if_t = 0> TxConsequences consequences_helper(PreflightContext const& ctx) { return T::makeTxConsequences(ctx); }; template std::pair invoke_preflight_helper(PreflightContext const& ctx) { auto const tec = T::preflight(ctx); return { tec, isTesSuccess(tec) ? consequences_helper(ctx) : TxConsequences{tec}}; } static std::pair invoke_preflight(PreflightContext const& ctx) { switch (ctx.tx.getTxnType()) { case ttACCOUNT_DELETE: return invoke_preflight_helper(ctx); case ttACCOUNT_SET: return invoke_preflight_helper(ctx); case ttCHECK_CANCEL: return invoke_preflight_helper(ctx); case ttCHECK_CASH: return invoke_preflight_helper(ctx); case ttCHECK_CREATE: return invoke_preflight_helper(ctx); case ttDEPOSIT_PREAUTH: return invoke_preflight_helper(ctx); case ttOFFER_CANCEL: return invoke_preflight_helper(ctx); case ttOFFER_CREATE: return invoke_preflight_helper(ctx); case ttESCROW_CREATE: return invoke_preflight_helper(ctx); case ttESCROW_FINISH: return invoke_preflight_helper(ctx); case ttESCROW_CANCEL: return invoke_preflight_helper(ctx); case ttPAYCHAN_CLAIM: return invoke_preflight_helper(ctx); case ttPAYCHAN_CREATE: return invoke_preflight_helper(ctx); case ttPAYCHAN_FUND: return invoke_preflight_helper(ctx); case ttPAYMENT: return invoke_preflight_helper(ctx); case ttREGULAR_KEY_SET: return invoke_preflight_helper(ctx); case ttSIGNER_LIST_SET: return invoke_preflight_helper(ctx); case ttTICKET_CREATE: return invoke_preflight_helper(ctx); case ttTRUST_SET: return invoke_preflight_helper(ctx); case ttAMENDMENT: case ttFEE: case ttUNL_MODIFY: return invoke_preflight_helper(ctx); default: assert(false); return {temUNKNOWN, TxConsequences{temUNKNOWN}}; } } /* invoke_preclaim uses name hiding to accomplish compile-time polymorphism of (presumably) static class functions for Transactor and derived classes. */ template static TER invoke_preclaim(PreclaimContext const& ctx) { // If the transactor requires a valid account and the transaction doesn't // list one, preflight will have already a flagged a failure. auto const id = ctx.tx.getAccountID(sfAccount); if (id != beast::zero) { TER result = T::checkSeqProxy(ctx.view, ctx.tx, ctx.j); if (result != tesSUCCESS) return result; result = T::checkPriorTxAndLastLedger(ctx); if (result != tesSUCCESS) return result; result = T::checkFee(ctx, calculateBaseFee(ctx.view, ctx.tx)); if (result != tesSUCCESS) return result; result = T::checkSign(ctx); if (result != tesSUCCESS) return result; } return T::preclaim(ctx); } static TER invoke_preclaim(PreclaimContext const& ctx) { switch (ctx.tx.getTxnType()) { case ttACCOUNT_DELETE: return invoke_preclaim(ctx); case ttACCOUNT_SET: return invoke_preclaim(ctx); case ttCHECK_CANCEL: return invoke_preclaim(ctx); case ttCHECK_CASH: return invoke_preclaim(ctx); case ttCHECK_CREATE: return invoke_preclaim(ctx); case ttDEPOSIT_PREAUTH: return invoke_preclaim(ctx); case ttOFFER_CANCEL: return invoke_preclaim(ctx); case ttOFFER_CREATE: return invoke_preclaim(ctx); case ttESCROW_CREATE: return invoke_preclaim(ctx); case ttESCROW_FINISH: return invoke_preclaim(ctx); case ttESCROW_CANCEL: return invoke_preclaim(ctx); case ttPAYCHAN_CLAIM: return invoke_preclaim(ctx); case ttPAYCHAN_CREATE: return invoke_preclaim(ctx); case ttPAYCHAN_FUND: return invoke_preclaim(ctx); case ttPAYMENT: return invoke_preclaim(ctx); case ttREGULAR_KEY_SET: return invoke_preclaim(ctx); case ttSIGNER_LIST_SET: return invoke_preclaim(ctx); case ttTICKET_CREATE: return invoke_preclaim(ctx); case ttTRUST_SET: return invoke_preclaim(ctx); case ttAMENDMENT: case ttFEE: case ttUNL_MODIFY: return invoke_preclaim(ctx); default: assert(false); return temUNKNOWN; } } static FeeUnit64 invoke_calculateBaseFee(ReadView const& view, STTx const& tx) { switch (tx.getTxnType()) { case ttACCOUNT_DELETE: return DeleteAccount::calculateBaseFee(view, tx); case ttACCOUNT_SET: return SetAccount::calculateBaseFee(view, tx); case ttCHECK_CANCEL: return CancelCheck::calculateBaseFee(view, tx); case ttCHECK_CASH: return CashCheck::calculateBaseFee(view, tx); case ttCHECK_CREATE: return CreateCheck::calculateBaseFee(view, tx); case ttDEPOSIT_PREAUTH: return DepositPreauth::calculateBaseFee(view, tx); case ttOFFER_CANCEL: return CancelOffer::calculateBaseFee(view, tx); case ttOFFER_CREATE: return CreateOffer::calculateBaseFee(view, tx); case ttESCROW_CREATE: return EscrowCreate::calculateBaseFee(view, tx); case ttESCROW_FINISH: return EscrowFinish::calculateBaseFee(view, tx); case ttESCROW_CANCEL: return EscrowCancel::calculateBaseFee(view, tx); case ttPAYCHAN_CLAIM: return PayChanClaim::calculateBaseFee(view, tx); case ttPAYCHAN_CREATE: return PayChanCreate::calculateBaseFee(view, tx); case ttPAYCHAN_FUND: return PayChanFund::calculateBaseFee(view, tx); case ttPAYMENT: return Payment::calculateBaseFee(view, tx); case ttREGULAR_KEY_SET: return SetRegularKey::calculateBaseFee(view, tx); case ttSIGNER_LIST_SET: return SetSignerList::calculateBaseFee(view, tx); case ttTICKET_CREATE: return CreateTicket::calculateBaseFee(view, tx); case ttTRUST_SET: return SetTrust::calculateBaseFee(view, tx); case ttAMENDMENT: case ttFEE: case ttUNL_MODIFY: return Change::calculateBaseFee(view, tx); default: assert(false); return FeeUnit64{0}; } } TxConsequences::TxConsequences(NotTEC pfresult) : isBlocker_(false) , fee_(beast::zero) , potentialSpend_(beast::zero) , seqProx_(SeqProxy::sequence(0)) , sequencesConsumed_(0) { assert(!isTesSuccess(pfresult)); } TxConsequences::TxConsequences(STTx const& tx) : isBlocker_(false) , fee_( tx[sfFee].native() && !tx[sfFee].negative() ? tx[sfFee].xrp() : beast::zero) , potentialSpend_(beast::zero) , seqProx_(tx.getSeqProxy()) , sequencesConsumed_(tx.getSeqProxy().isSeq() ? 1 : 0) { } TxConsequences::TxConsequences(STTx const& tx, Category category) : TxConsequences(tx) { isBlocker_ = (category == blocker); } TxConsequences::TxConsequences(STTx const& tx, XRPAmount potentialSpend) : TxConsequences(tx) { potentialSpend_ = potentialSpend; } TxConsequences::TxConsequences(STTx const& tx, std::uint32_t sequencesConsumed) : TxConsequences(tx) { sequencesConsumed_ = sequencesConsumed; } static std::pair invoke_apply(ApplyContext& ctx) { switch (ctx.tx.getTxnType()) { case ttACCOUNT_DELETE: { DeleteAccount p(ctx); return p(); } case ttACCOUNT_SET: { SetAccount p(ctx); return p(); } case ttCHECK_CANCEL: { CancelCheck p(ctx); return p(); } case ttCHECK_CASH: { CashCheck p(ctx); return p(); } case ttCHECK_CREATE: { CreateCheck p(ctx); return p(); } case ttDEPOSIT_PREAUTH: { DepositPreauth p(ctx); return p(); } case ttOFFER_CANCEL: { CancelOffer p(ctx); return p(); } case ttOFFER_CREATE: { CreateOffer p(ctx); return p(); } case ttESCROW_CREATE: { EscrowCreate p(ctx); return p(); } case ttESCROW_FINISH: { EscrowFinish p(ctx); return p(); } case ttESCROW_CANCEL: { EscrowCancel p(ctx); return p(); } case ttPAYCHAN_CLAIM: { PayChanClaim p(ctx); return p(); } case ttPAYCHAN_CREATE: { PayChanCreate p(ctx); return p(); } case ttPAYCHAN_FUND: { PayChanFund p(ctx); return p(); } case ttPAYMENT: { Payment p(ctx); return p(); } case ttREGULAR_KEY_SET: { SetRegularKey p(ctx); return p(); } case ttSIGNER_LIST_SET: { SetSignerList p(ctx); return p(); } case ttTICKET_CREATE: { CreateTicket p(ctx); return p(); } case ttTRUST_SET: { SetTrust p(ctx); return p(); } case ttAMENDMENT: case ttFEE: case ttUNL_MODIFY: { Change p(ctx); return p(); } default: assert(false); return {temUNKNOWN, false}; } } PreflightResult preflight( Application& app, Rules const& rules, STTx const& tx, ApplyFlags flags, beast::Journal j) { PreflightContext const pfctx(app, tx, rules, flags, j); try { return {pfctx, invoke_preflight(pfctx)}; } catch (std::exception const& e) { JLOG(j.fatal()) << "apply: " << e.what(); return {pfctx, {tefEXCEPTION, TxConsequences{tx}}}; } } PreclaimResult preclaim( PreflightResult const& preflightResult, Application& app, OpenView const& view) { boost::optional ctx; if (preflightResult.rules != view.rules()) { auto secondFlight = preflight( app, view.rules(), preflightResult.tx, preflightResult.flags, preflightResult.j); ctx.emplace( app, view, secondFlight.ter, secondFlight.tx, secondFlight.flags, secondFlight.j); } else { ctx.emplace( app, view, preflightResult.ter, preflightResult.tx, preflightResult.flags, preflightResult.j); } try { if (ctx->preflightResult != tesSUCCESS) return {*ctx, ctx->preflightResult}; return {*ctx, invoke_preclaim(*ctx)}; } catch (std::exception const& e) { JLOG(ctx->j.fatal()) << "apply: " << e.what(); return {*ctx, tefEXCEPTION}; } } FeeUnit64 calculateBaseFee(ReadView const& view, STTx const& tx) { return invoke_calculateBaseFee(view, tx); } XRPAmount calculateDefaultBaseFee(ReadView const& view, STTx const& tx) { return view.fees().toDrops(Transactor::calculateBaseFee(view, tx)); } std::pair doApply(PreclaimResult const& preclaimResult, Application& app, OpenView& view) { if (preclaimResult.view.seq() != view.seq()) { // Logic error from the caller. Don't have enough // info to recover. return {tefEXCEPTION, false}; } try { if (!preclaimResult.likelyToClaimFee) return {preclaimResult.ter, false}; ApplyContext ctx( app, view, preclaimResult.tx, preclaimResult.ter, calculateBaseFee(view, preclaimResult.tx), preclaimResult.flags, preclaimResult.j); return invoke_apply(ctx); } catch (std::exception const& e) { JLOG(preclaimResult.j.fatal()) << "apply: " << e.what(); return {tefEXCEPTION, false}; } } } // namespace ripple