mirror of
https://github.com/XRPLF/rippled.git
synced 2025-11-04 11:15:56 +00:00
This change introduces the `featurePermissionDelegationV1_1` amendment, which is designed to supersede both `featurePermissionDelegation` and `fixDelegateV1_1 amendments, which should be considered deprecated. The `checkPermission` function will now return `terNO_DELEGATE_PERMISSION` when a delegate transaction lacks the necessary permissions.
710 lines
19 KiB
C++
710 lines
19 KiB
C++
//------------------------------------------------------------------------------
|
|
/*
|
|
This file is part of rippled: https://github.com/ripple/rippled
|
|
Copyright (c) 2012 - 2019 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.
|
|
*/
|
|
//==============================================================================
|
|
|
|
#ifndef RIPPLE_PROTOCOL_TER_H_INCLUDED
|
|
#define RIPPLE_PROTOCOL_TER_H_INCLUDED
|
|
|
|
#include <xrpl/basics/safe_cast.h>
|
|
#include <xrpl/json/json_value.h>
|
|
|
|
#include <optional>
|
|
#include <ostream>
|
|
#include <string>
|
|
#include <unordered_map>
|
|
|
|
namespace ripple {
|
|
|
|
// See https://xrpl.org/transaction-results.html
|
|
//
|
|
// "Transaction Engine Result"
|
|
// or Transaction ERror.
|
|
//
|
|
using TERUnderlyingType = int;
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
enum TELcodes : TERUnderlyingType {
|
|
// Note: Range is stable.
|
|
// Exact numbers are used in ripple-binary-codec:
|
|
// https://github.com/XRPLF/xrpl.js/blob/main/packages/ripple-binary-codec/src/enums/definitions.json
|
|
// Use tokens.
|
|
|
|
// -399 .. -300: L Local error (transaction fee inadequate, exceeds local
|
|
// limit) Only valid during non-consensus processing. Implications:
|
|
// - Not forwarded
|
|
// - No fee check
|
|
telLOCAL_ERROR = -399,
|
|
telBAD_DOMAIN,
|
|
telBAD_PATH_COUNT,
|
|
telBAD_PUBLIC_KEY,
|
|
telFAILED_PROCESSING,
|
|
telINSUF_FEE_P,
|
|
telNO_DST_PARTIAL,
|
|
telCAN_NOT_QUEUE,
|
|
telCAN_NOT_QUEUE_BALANCE,
|
|
telCAN_NOT_QUEUE_BLOCKS,
|
|
telCAN_NOT_QUEUE_BLOCKED,
|
|
telCAN_NOT_QUEUE_FEE,
|
|
telCAN_NOT_QUEUE_FULL,
|
|
telWRONG_NETWORK,
|
|
telREQUIRES_NETWORK_ID,
|
|
telNETWORK_ID_MAKES_TX_NON_CANONICAL,
|
|
telENV_RPC_FAILED
|
|
};
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
enum TEMcodes : TERUnderlyingType {
|
|
// Note: Range is stable.
|
|
// Exact numbers are used in ripple-binary-codec:
|
|
// https://github.com/XRPLF/xrpl.js/blob/main/packages/ripple-binary-codec/src/enums/definitions.json
|
|
// Use tokens.
|
|
|
|
// -299 .. -200: M Malformed (bad signature)
|
|
// Causes:
|
|
// - Transaction corrupt.
|
|
// Implications:
|
|
// - Not applied
|
|
// - Not forwarded
|
|
// - Reject
|
|
// - Cannot succeed in any imagined ledger.
|
|
temMALFORMED = -299,
|
|
|
|
temBAD_AMOUNT,
|
|
temBAD_CURRENCY,
|
|
temBAD_EXPIRATION,
|
|
temBAD_FEE,
|
|
temBAD_ISSUER,
|
|
temBAD_LIMIT,
|
|
temBAD_OFFER,
|
|
temBAD_PATH,
|
|
temBAD_PATH_LOOP,
|
|
temBAD_REGKEY,
|
|
temBAD_SEND_XRP_LIMIT,
|
|
temBAD_SEND_XRP_MAX,
|
|
temBAD_SEND_XRP_NO_DIRECT,
|
|
temBAD_SEND_XRP_PARTIAL,
|
|
temBAD_SEND_XRP_PATHS,
|
|
temBAD_SEQUENCE,
|
|
temBAD_SIGNATURE,
|
|
temBAD_SRC_ACCOUNT,
|
|
temBAD_TRANSFER_RATE,
|
|
temDST_IS_SRC,
|
|
temDST_NEEDED,
|
|
temINVALID,
|
|
temINVALID_FLAG,
|
|
temREDUNDANT,
|
|
temRIPPLE_EMPTY,
|
|
temDISABLED,
|
|
temBAD_SIGNER,
|
|
temBAD_QUORUM,
|
|
temBAD_WEIGHT,
|
|
temBAD_TICK_SIZE,
|
|
temINVALID_ACCOUNT_ID,
|
|
temCANNOT_PREAUTH_SELF,
|
|
temINVALID_COUNT,
|
|
|
|
temUNCERTAIN, // An internal intermediate result; should never be returned.
|
|
temUNKNOWN, // An internal intermediate result; should never be returned.
|
|
|
|
temSEQ_AND_TICKET,
|
|
temBAD_NFTOKEN_TRANSFER_FEE,
|
|
|
|
temBAD_AMM_TOKENS,
|
|
|
|
temXCHAIN_EQUAL_DOOR_ACCOUNTS,
|
|
temXCHAIN_BAD_PROOF,
|
|
temXCHAIN_BRIDGE_BAD_ISSUES,
|
|
temXCHAIN_BRIDGE_NONDOOR_OWNER,
|
|
temXCHAIN_BRIDGE_BAD_MIN_ACCOUNT_CREATE_AMOUNT,
|
|
temXCHAIN_BRIDGE_BAD_REWARD_AMOUNT,
|
|
|
|
temEMPTY_DID,
|
|
|
|
temARRAY_EMPTY,
|
|
temARRAY_TOO_LARGE,
|
|
temBAD_TRANSFER_FEE,
|
|
temINVALID_INNER_BATCH,
|
|
};
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
enum TEFcodes : TERUnderlyingType {
|
|
// Note: Range is stable.
|
|
// Exact numbers are used in ripple-binary-codec:
|
|
// https://github.com/XRPLF/xrpl.js/blob/main/packages/ripple-binary-codec/src/enums/definitions.json
|
|
// Use tokens.
|
|
|
|
// -199 .. -100: F
|
|
// Failure (sequence number previously used)
|
|
//
|
|
// Causes:
|
|
// - Transaction cannot succeed because of ledger state.
|
|
// - Unexpected ledger state.
|
|
// - C++ exception.
|
|
//
|
|
// Implications:
|
|
// - Not applied
|
|
// - Not forwarded
|
|
// - Could succeed in an imagined ledger.
|
|
tefFAILURE = -199,
|
|
tefALREADY,
|
|
tefBAD_ADD_AUTH,
|
|
tefBAD_AUTH,
|
|
tefBAD_LEDGER,
|
|
tefCREATED,
|
|
tefEXCEPTION,
|
|
tefINTERNAL,
|
|
tefNO_AUTH_REQUIRED, // Can't set auth if auth is not required.
|
|
tefPAST_SEQ,
|
|
tefWRONG_PRIOR,
|
|
tefMASTER_DISABLED,
|
|
tefMAX_LEDGER,
|
|
tefBAD_SIGNATURE,
|
|
tefBAD_QUORUM,
|
|
tefNOT_MULTI_SIGNING,
|
|
tefBAD_AUTH_MASTER,
|
|
tefINVARIANT_FAILED,
|
|
tefTOO_BIG,
|
|
tefNO_TICKET,
|
|
tefNFTOKEN_IS_NOT_TRANSFERABLE,
|
|
tefINVALID_LEDGER_FIX_TYPE,
|
|
};
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
enum TERcodes : TERUnderlyingType {
|
|
// Note: Range is stable.
|
|
// Exact numbers are used in ripple-binary-codec:
|
|
// https://github.com/XRPLF/xrpl.js/blob/main/packages/ripple-binary-codec/src/enums/definitions.json
|
|
// Use tokens.
|
|
|
|
// -99 .. -1: R Retry
|
|
// sequence too high, no funds for txn fee, originating -account
|
|
// non-existent
|
|
//
|
|
// Cause:
|
|
// Prior application of another, possibly non-existent, transaction could
|
|
// allow this transaction to succeed.
|
|
//
|
|
// Implications:
|
|
// - Not applied
|
|
// - May be forwarded
|
|
// - Results indicating the txn was forwarded: terQUEUED
|
|
// - All others are not forwarded.
|
|
// - Might succeed later
|
|
// - Hold
|
|
// - Makes hole in sequence which jams transactions.
|
|
terRETRY = -99,
|
|
terFUNDS_SPENT, // DEPRECATED.
|
|
terINSUF_FEE_B, // Can't pay fee, therefore don't burden network.
|
|
terNO_ACCOUNT, // Can't pay fee, therefore don't burden network.
|
|
terNO_AUTH, // Not authorized to hold IOUs.
|
|
terNO_LINE, // Internal flag.
|
|
terOWNERS, // Can't succeed with non-zero owner count.
|
|
terPRE_SEQ, // Can't pay fee, no point in forwarding, so don't
|
|
// burden network.
|
|
terLAST, // DEPRECATED.
|
|
terNO_RIPPLE, // Rippling not allowed
|
|
terQUEUED, // Transaction is being held in TxQ until fee drops
|
|
terPRE_TICKET, // Ticket is not yet in ledger but might be on its way
|
|
terNO_AMM, // AMM doesn't exist for the asset pair
|
|
terADDRESS_COLLISION, // Failed to allocate AccountID when trying to
|
|
// create a pseudo-account
|
|
terNO_DELEGATE_PERMISSION, // Delegate does not have permission
|
|
};
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
enum TEScodes : TERUnderlyingType {
|
|
// Note: Exact number must stay stable. This code is stored by value
|
|
// in metadata for historic transactions.
|
|
|
|
// 0: S Success (success)
|
|
// Causes:
|
|
// - Success.
|
|
// Implications:
|
|
// - Applied
|
|
// - Forwarded
|
|
tesSUCCESS = 0
|
|
};
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
enum TECcodes : TERUnderlyingType {
|
|
// Note: Exact numbers must stay stable. These codes are stored by
|
|
// value in metadata for historic transactions.
|
|
|
|
// 100 .. 255 C
|
|
// Claim fee only (ripple transaction with no good paths, pay to
|
|
// non-existent account, no path)
|
|
//
|
|
// Causes:
|
|
// - Success, but does not achieve optimal result.
|
|
// - Invalid transaction or no effect, but claim fee to use the sequence
|
|
// number.
|
|
//
|
|
// Implications:
|
|
// - Applied
|
|
// - Forwarded
|
|
//
|
|
// Only allowed as a return code of appliedTransaction when !tapRETRY.
|
|
// Otherwise, treated as terRETRY.
|
|
//
|
|
// DO NOT CHANGE THESE NUMBERS: They appear in ledger meta data.
|
|
//
|
|
// Note:
|
|
// tecNO_ENTRY is often used interchangeably with tecOBJECT_NOT_FOUND.
|
|
// While there does not seem to be a clear rule which to use when, the
|
|
// following guidance will help to keep errors consistent with the
|
|
// majority of (but not all) transaction types:
|
|
// - tecNO_ENTRY : cannot find the primary ledger object on which the
|
|
// transaction is being attempted
|
|
// - tecOBJECT_NOT_FOUND : cannot find the additional object(s) needed to
|
|
// complete the transaction
|
|
|
|
tecCLAIM = 100,
|
|
tecPATH_PARTIAL = 101,
|
|
tecUNFUNDED_ADD = 102, // Unused legacy code
|
|
tecUNFUNDED_OFFER = 103,
|
|
tecUNFUNDED_PAYMENT = 104,
|
|
tecFAILED_PROCESSING = 105,
|
|
tecDIR_FULL = 121,
|
|
tecINSUF_RESERVE_LINE = 122,
|
|
tecINSUF_RESERVE_OFFER = 123,
|
|
tecNO_DST = 124,
|
|
tecNO_DST_INSUF_XRP = 125,
|
|
tecNO_LINE_INSUF_RESERVE = 126,
|
|
tecNO_LINE_REDUNDANT = 127,
|
|
tecPATH_DRY = 128,
|
|
tecUNFUNDED = 129,
|
|
tecNO_ALTERNATIVE_KEY = 130,
|
|
tecNO_REGULAR_KEY = 131,
|
|
tecOWNERS = 132,
|
|
tecNO_ISSUER = 133,
|
|
tecNO_AUTH = 134,
|
|
tecNO_LINE = 135,
|
|
tecINSUFF_FEE = 136,
|
|
tecFROZEN = 137,
|
|
tecNO_TARGET = 138,
|
|
tecNO_PERMISSION = 139,
|
|
tecNO_ENTRY = 140,
|
|
tecINSUFFICIENT_RESERVE = 141,
|
|
tecNEED_MASTER_KEY = 142,
|
|
tecDST_TAG_NEEDED = 143,
|
|
tecINTERNAL = 144,
|
|
tecOVERSIZE = 145,
|
|
tecCRYPTOCONDITION_ERROR = 146,
|
|
tecINVARIANT_FAILED = 147,
|
|
tecEXPIRED = 148,
|
|
tecDUPLICATE = 149,
|
|
tecKILLED = 150,
|
|
tecHAS_OBLIGATIONS = 151,
|
|
tecTOO_SOON = 152,
|
|
tecHOOK_REJECTED [[maybe_unused]] = 153,
|
|
tecMAX_SEQUENCE_REACHED = 154,
|
|
tecNO_SUITABLE_NFTOKEN_PAGE = 155,
|
|
tecNFTOKEN_BUY_SELL_MISMATCH = 156,
|
|
tecNFTOKEN_OFFER_TYPE_MISMATCH = 157,
|
|
tecCANT_ACCEPT_OWN_NFTOKEN_OFFER = 158,
|
|
tecINSUFFICIENT_FUNDS = 159,
|
|
tecOBJECT_NOT_FOUND = 160,
|
|
tecINSUFFICIENT_PAYMENT = 161,
|
|
tecUNFUNDED_AMM = 162,
|
|
tecAMM_BALANCE = 163,
|
|
tecAMM_FAILED = 164,
|
|
tecAMM_INVALID_TOKENS = 165,
|
|
tecAMM_EMPTY = 166,
|
|
tecAMM_NOT_EMPTY = 167,
|
|
tecAMM_ACCOUNT = 168,
|
|
tecINCOMPLETE = 169,
|
|
tecXCHAIN_BAD_TRANSFER_ISSUE = 170,
|
|
tecXCHAIN_NO_CLAIM_ID = 171,
|
|
tecXCHAIN_BAD_CLAIM_ID = 172,
|
|
tecXCHAIN_CLAIM_NO_QUORUM = 173,
|
|
tecXCHAIN_PROOF_UNKNOWN_KEY = 174,
|
|
tecXCHAIN_CREATE_ACCOUNT_NONXRP_ISSUE = 175,
|
|
tecXCHAIN_WRONG_CHAIN = 176,
|
|
tecXCHAIN_REWARD_MISMATCH = 177,
|
|
tecXCHAIN_NO_SIGNERS_LIST = 178,
|
|
tecXCHAIN_SENDING_ACCOUNT_MISMATCH = 179,
|
|
tecXCHAIN_INSUFF_CREATE_AMOUNT = 180,
|
|
tecXCHAIN_ACCOUNT_CREATE_PAST = 181,
|
|
tecXCHAIN_ACCOUNT_CREATE_TOO_MANY = 182,
|
|
tecXCHAIN_PAYMENT_FAILED = 183,
|
|
tecXCHAIN_SELF_COMMIT = 184,
|
|
tecXCHAIN_BAD_PUBLIC_KEY_ACCOUNT_PAIR = 185,
|
|
tecXCHAIN_CREATE_ACCOUNT_DISABLED = 186,
|
|
tecEMPTY_DID = 187,
|
|
tecINVALID_UPDATE_TIME = 188,
|
|
tecTOKEN_PAIR_NOT_FOUND = 189,
|
|
tecARRAY_EMPTY = 190,
|
|
tecARRAY_TOO_LARGE = 191,
|
|
tecLOCKED = 192,
|
|
tecBAD_CREDENTIALS = 193,
|
|
tecWRONG_ASSET = 194,
|
|
tecLIMIT_EXCEEDED = 195,
|
|
tecPSEUDO_ACCOUNT = 196,
|
|
tecPRECISION_LOSS = 197,
|
|
// DEPRECATED: This error code tecNO_DELEGATE_PERMISSION is reserved for
|
|
// backward compatibility with historical data on non-prod networks, can be
|
|
// reclaimed after those networks reset.
|
|
tecNO_DELEGATE_PERMISSION = 198,
|
|
};
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
// For generic purposes, a free function that returns the value of a TE*codes.
|
|
constexpr TERUnderlyingType
|
|
TERtoInt(TELcodes v)
|
|
{
|
|
return safe_cast<TERUnderlyingType>(v);
|
|
}
|
|
|
|
constexpr TERUnderlyingType
|
|
TERtoInt(TEMcodes v)
|
|
{
|
|
return safe_cast<TERUnderlyingType>(v);
|
|
}
|
|
|
|
constexpr TERUnderlyingType
|
|
TERtoInt(TEFcodes v)
|
|
{
|
|
return safe_cast<TERUnderlyingType>(v);
|
|
}
|
|
|
|
constexpr TERUnderlyingType
|
|
TERtoInt(TERcodes v)
|
|
{
|
|
return safe_cast<TERUnderlyingType>(v);
|
|
}
|
|
|
|
constexpr TERUnderlyingType
|
|
TERtoInt(TEScodes v)
|
|
{
|
|
return safe_cast<TERUnderlyingType>(v);
|
|
}
|
|
|
|
constexpr TERUnderlyingType
|
|
TERtoInt(TECcodes v)
|
|
{
|
|
return safe_cast<TERUnderlyingType>(v);
|
|
}
|
|
|
|
//------------------------------------------------------------------------------
|
|
// Template class that is specific to selected ranges of error codes. The
|
|
// Trait tells std::enable_if which ranges are allowed.
|
|
template <template <typename> class Trait>
|
|
class TERSubset
|
|
{
|
|
TERUnderlyingType code_;
|
|
|
|
public:
|
|
// Constructors
|
|
constexpr TERSubset() : code_(tesSUCCESS)
|
|
{
|
|
}
|
|
constexpr TERSubset(TERSubset const& rhs) = default;
|
|
constexpr TERSubset(TERSubset&& rhs) = default;
|
|
|
|
private:
|
|
constexpr explicit TERSubset(int rhs) : code_(rhs)
|
|
{
|
|
}
|
|
|
|
public:
|
|
static constexpr TERSubset
|
|
fromInt(int from)
|
|
{
|
|
return TERSubset(from);
|
|
}
|
|
|
|
// Trait tells enable_if which types are allowed for construction.
|
|
template <
|
|
typename T,
|
|
typename = std::enable_if_t<
|
|
Trait<std::remove_cv_t<std::remove_reference_t<T>>>::value>>
|
|
constexpr TERSubset(T rhs) : code_(TERtoInt(rhs))
|
|
{
|
|
}
|
|
|
|
// Assignment
|
|
constexpr TERSubset&
|
|
operator=(TERSubset const& rhs) = default;
|
|
constexpr TERSubset&
|
|
operator=(TERSubset&& rhs) = default;
|
|
|
|
// Trait tells enable_if which types are allowed for assignment.
|
|
template <typename T>
|
|
constexpr auto
|
|
operator=(T rhs) -> std::enable_if_t<Trait<T>::value, TERSubset&>
|
|
{
|
|
code_ = TERtoInt(rhs);
|
|
return *this;
|
|
}
|
|
|
|
// Conversion to bool.
|
|
explicit
|
|
operator bool() const
|
|
{
|
|
return code_ != tesSUCCESS;
|
|
}
|
|
|
|
// Conversion to Json::Value allows assignment to Json::Objects
|
|
// without casting.
|
|
operator Json::Value() const
|
|
{
|
|
return Json::Value{code_};
|
|
}
|
|
|
|
// Streaming operator.
|
|
friend std::ostream&
|
|
operator<<(std::ostream& os, TERSubset const& rhs)
|
|
{
|
|
return os << rhs.code_;
|
|
}
|
|
|
|
// Return the underlying value. Not a member so similarly named free
|
|
// functions can do the same work for the enums.
|
|
//
|
|
// It's worth noting that an explicit conversion operator was considered
|
|
// and rejected. Consider this case, taken from Status.h
|
|
//
|
|
// class Status {
|
|
// int code_;
|
|
// public:
|
|
// Status (TER ter)
|
|
// : code_ (ter) {}
|
|
// }
|
|
//
|
|
// This code compiles with no errors or warnings if TER has an explicit
|
|
// (unnamed) conversion to int. To avoid silent conversions like these
|
|
// we provide (only) a named conversion.
|
|
friend constexpr TERUnderlyingType
|
|
TERtoInt(TERSubset v)
|
|
{
|
|
return v.code_;
|
|
}
|
|
};
|
|
|
|
// Comparison operators.
|
|
// Only enabled if both arguments return int if TERtiInt is called with them.
|
|
template <typename L, typename R>
|
|
constexpr auto
|
|
operator==(L const& lhs, R const& rhs)
|
|
-> std::enable_if_t<
|
|
std::is_same<decltype(TERtoInt(lhs)), int>::value &&
|
|
std::is_same<decltype(TERtoInt(rhs)), int>::value,
|
|
bool>
|
|
{
|
|
return TERtoInt(lhs) == TERtoInt(rhs);
|
|
}
|
|
|
|
template <typename L, typename R>
|
|
constexpr auto
|
|
operator!=(L const& lhs, R const& rhs)
|
|
-> std::enable_if_t<
|
|
std::is_same<decltype(TERtoInt(lhs)), int>::value &&
|
|
std::is_same<decltype(TERtoInt(rhs)), int>::value,
|
|
bool>
|
|
{
|
|
return TERtoInt(lhs) != TERtoInt(rhs);
|
|
}
|
|
|
|
template <typename L, typename R>
|
|
constexpr auto
|
|
operator<(L const& lhs, R const& rhs)
|
|
-> std::enable_if_t<
|
|
std::is_same<decltype(TERtoInt(lhs)), int>::value &&
|
|
std::is_same<decltype(TERtoInt(rhs)), int>::value,
|
|
bool>
|
|
{
|
|
return TERtoInt(lhs) < TERtoInt(rhs);
|
|
}
|
|
|
|
template <typename L, typename R>
|
|
constexpr auto
|
|
operator<=(L const& lhs, R const& rhs)
|
|
-> std::enable_if_t<
|
|
std::is_same<decltype(TERtoInt(lhs)), int>::value &&
|
|
std::is_same<decltype(TERtoInt(rhs)), int>::value,
|
|
bool>
|
|
{
|
|
return TERtoInt(lhs) <= TERtoInt(rhs);
|
|
}
|
|
|
|
template <typename L, typename R>
|
|
constexpr auto
|
|
operator>(L const& lhs, R const& rhs)
|
|
-> std::enable_if_t<
|
|
std::is_same<decltype(TERtoInt(lhs)), int>::value &&
|
|
std::is_same<decltype(TERtoInt(rhs)), int>::value,
|
|
bool>
|
|
{
|
|
return TERtoInt(lhs) > TERtoInt(rhs);
|
|
}
|
|
|
|
template <typename L, typename R>
|
|
constexpr auto
|
|
operator>=(L const& lhs, R const& rhs)
|
|
-> std::enable_if_t<
|
|
std::is_same<decltype(TERtoInt(lhs)), int>::value &&
|
|
std::is_same<decltype(TERtoInt(rhs)), int>::value,
|
|
bool>
|
|
{
|
|
return TERtoInt(lhs) >= TERtoInt(rhs);
|
|
}
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
// Use traits to build a TERSubset that can convert from any of the TE*codes
|
|
// enums *except* TECcodes: NotTEC
|
|
|
|
// NOTE: NotTEC is useful for codes returned by preflight in transactors.
|
|
// Preflight checks occur prior to signature checking. If preflight returned
|
|
// a tec code, then a malicious user could submit a transaction with a very
|
|
// large fee and have that fee charged against an account without using that
|
|
// account's valid signature.
|
|
template <typename FROM>
|
|
class CanCvtToNotTEC : public std::false_type
|
|
{
|
|
};
|
|
template <>
|
|
class CanCvtToNotTEC<TELcodes> : public std::true_type
|
|
{
|
|
};
|
|
template <>
|
|
class CanCvtToNotTEC<TEMcodes> : public std::true_type
|
|
{
|
|
};
|
|
template <>
|
|
class CanCvtToNotTEC<TEFcodes> : public std::true_type
|
|
{
|
|
};
|
|
template <>
|
|
class CanCvtToNotTEC<TERcodes> : public std::true_type
|
|
{
|
|
};
|
|
template <>
|
|
class CanCvtToNotTEC<TEScodes> : public std::true_type
|
|
{
|
|
};
|
|
|
|
using NotTEC = TERSubset<CanCvtToNotTEC>;
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
// Use traits to build a TERSubset that can convert from any of the TE*codes
|
|
// enums as well as from NotTEC.
|
|
template <typename FROM>
|
|
class CanCvtToTER : public std::false_type
|
|
{
|
|
};
|
|
template <>
|
|
class CanCvtToTER<TELcodes> : public std::true_type
|
|
{
|
|
};
|
|
template <>
|
|
class CanCvtToTER<TEMcodes> : public std::true_type
|
|
{
|
|
};
|
|
template <>
|
|
class CanCvtToTER<TEFcodes> : public std::true_type
|
|
{
|
|
};
|
|
template <>
|
|
class CanCvtToTER<TERcodes> : public std::true_type
|
|
{
|
|
};
|
|
template <>
|
|
class CanCvtToTER<TEScodes> : public std::true_type
|
|
{
|
|
};
|
|
template <>
|
|
class CanCvtToTER<TECcodes> : public std::true_type
|
|
{
|
|
};
|
|
template <>
|
|
class CanCvtToTER<NotTEC> : public std::true_type
|
|
{
|
|
};
|
|
|
|
// TER allows all of the subsets.
|
|
using TER = TERSubset<CanCvtToTER>;
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
inline bool
|
|
isTelLocal(TER x) noexcept
|
|
{
|
|
return (x >= telLOCAL_ERROR && x < temMALFORMED);
|
|
}
|
|
|
|
inline bool
|
|
isTemMalformed(TER x) noexcept
|
|
{
|
|
return (x >= temMALFORMED && x < tefFAILURE);
|
|
}
|
|
|
|
inline bool
|
|
isTefFailure(TER x) noexcept
|
|
{
|
|
return (x >= tefFAILURE && x < terRETRY);
|
|
}
|
|
|
|
inline bool
|
|
isTerRetry(TER x) noexcept
|
|
{
|
|
return (x >= terRETRY && x < tesSUCCESS);
|
|
}
|
|
|
|
inline bool
|
|
isTesSuccess(TER x) noexcept
|
|
{
|
|
// Makes use of TERSubset::operator bool()
|
|
return !(x);
|
|
}
|
|
|
|
inline bool
|
|
isTecClaim(TER x) noexcept
|
|
{
|
|
return ((x) >= tecCLAIM);
|
|
}
|
|
|
|
std::unordered_map<
|
|
TERUnderlyingType,
|
|
std::pair<char const* const, char const* const>> const&
|
|
transResults();
|
|
|
|
bool
|
|
transResultInfo(TER code, std::string& token, std::string& text);
|
|
|
|
std::string
|
|
transToken(TER code);
|
|
|
|
std::string
|
|
transHuman(TER code);
|
|
|
|
std::optional<TER>
|
|
transCode(std::string const& token);
|
|
|
|
} // namespace ripple
|
|
|
|
#endif
|