Skip to content

Commit

Permalink
create fix amendment for adding sfPreviousTxnID/`sfPreviousTxnLgrSe…
Browse files Browse the repository at this point in the history
…q` everywhere
  • Loading branch information
mvadari committed Nov 2, 2023
1 parent 056255e commit 2712056
Show file tree
Hide file tree
Showing 12 changed files with 139 additions and 34 deletions.
5 changes: 4 additions & 1 deletion src/ripple/ledger/detail/ApplyStateTable.h
Original file line number Diff line number Diff line change
Expand Up @@ -128,7 +128,10 @@ class ApplyStateTable
using Mods = hash_map<key_type, std::shared_ptr<SLE>>;

static void
threadItem(TxMeta& meta, std::shared_ptr<SLE> const& to);
threadItem(
TxMeta& meta,
std::shared_ptr<SLE> const& to,
ReadView const& view);

std::shared_ptr<SLE>
getForMod(
Expand Down
24 changes: 19 additions & 5 deletions src/ripple/ledger/impl/ApplyStateTable.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -21,8 +21,10 @@
#include <ripple/json/to_string.h>
#include <ripple/ledger/detail/ApplyStateTable.h>
#include <ripple/protocol/Feature.h>
#include <ripple/protocol/LedgerFormats.h>
#include <ripple/protocol/st.h>
#include <cassert>
#include <set>

namespace ripple {
namespace detail {
Expand Down Expand Up @@ -191,7 +193,7 @@ ApplyStateTable::apply(

if (curNode->isThreadedType()) // thread transaction to node
// item modified
threadItem(meta, curNode);
threadItem(meta, curNode, to);

STObject prevs(sfPreviousFields);
for (auto const& obj : *origNode)
Expand Down Expand Up @@ -225,7 +227,7 @@ ApplyStateTable::apply(
threadOwners(to, meta, curNode, newMod, j);

if (curNode->isThreadedType()) // always thread to self
threadItem(meta, curNode);
threadItem(meta, curNode, to);

STObject news(sfNewFields);
for (auto const& obj : *curNode)
Expand Down Expand Up @@ -520,12 +522,24 @@ ApplyStateTable::destroyXRP(XRPAmount const& fee)

// Insert this transaction to the SLE's threading list
void
ApplyStateTable::threadItem(TxMeta& meta, std::shared_ptr<SLE> const& sle)
ApplyStateTable::threadItem(
TxMeta& meta,
std::shared_ptr<SLE> const& sle,
ReadView const& view)
{
key_type prevTxID;
LedgerIndex prevLgrID;

if (!sle->thread(meta.getTxID(), meta.getLgrSeq(), prevTxID, prevLgrID))
static std::set<LedgerEntryType> newPreviousTxnIDTypes = {
ltDIR_NODE, ltAMENDMENTS, ltFEE_SETTINGS, ltNEGATIVE_UNL, ltAMM};
bool const includePrevTxnID = view.rules().enabled(fixPreviousTxnID) ||
!newPreviousTxnIDTypes.count(sle->getType());
if (!sle->thread(
meta.getTxID(),
meta.getLgrSeq(),
prevTxID,
prevLgrID,
includePrevTxnID))
return;

if (!prevTxID.isZero())
Expand Down Expand Up @@ -610,7 +624,7 @@ ApplyStateTable::threadTx(
JLOG(j.warn()) << "Threading to non-existent account: " << toBase58(to);
return;
}
threadItem(meta, sle);
threadItem(meta, sle, base);
}

void
Expand Down
3 changes: 2 additions & 1 deletion src/ripple/protocol/Feature.h
Original file line number Diff line number Diff line change
Expand Up @@ -74,7 +74,7 @@ namespace detail {
// Feature.cpp. Because it's only used to reserve storage, and determine how
// large to make the FeatureBitset, it MAY be larger. It MUST NOT be less than
// the actual number of amendments. A LogicError on startup will verify this.
static constexpr std::size_t numFeatures = 65;
static constexpr std::size_t numFeatures = 66;

/** Amendments that this server supports and the default voting behavior.
Whether they are enabled depends on the Rules defined in the validated
Expand Down Expand Up @@ -352,6 +352,7 @@ extern uint256 const featureXChainBridge;
extern uint256 const fixDisallowIncomingV1;
extern uint256 const featureDID;
extern uint256 const fixFillOrKill;
extern uint256 const fixPreviousTxnID;

} // namespace ripple

Expand Down
3 changes: 2 additions & 1 deletion src/ripple/protocol/STLedgerEntry.h
Original file line number Diff line number Diff line change
Expand Up @@ -74,7 +74,8 @@ class STLedgerEntry final : public STObject, public CountedObject<STLedgerEntry>
uint256 const& txID,
std::uint32_t ledgerSeq,
uint256& prevTxID,
std::uint32_t& prevLedgerID);
std::uint32_t& prevLedgerID,
bool const includePrevTxnID);

private:
/* Make STObject comply with the template for this SLE type
Expand Down
3 changes: 2 additions & 1 deletion src/ripple/protocol/impl/Feature.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -458,7 +458,8 @@ REGISTER_FEATURE(AMM, Supported::yes, VoteBehavior::De
REGISTER_FEATURE(XChainBridge, Supported::yes, VoteBehavior::DefaultNo);
REGISTER_FIX (fixDisallowIncomingV1, Supported::yes, VoteBehavior::DefaultNo);
REGISTER_FEATURE(DID, Supported::yes, VoteBehavior::DefaultNo);
REGISTER_FIX(fixFillOrKill, Supported::yes, VoteBehavior::DefaultNo);
REGISTER_FIX (fixFillOrKill, Supported::yes, VoteBehavior::DefaultNo);
REGISTER_FIX (fixPreviousTxnID, Supported::yes, VoteBehavior::DefaultNo);

// The following amendments are obsolete, but must remain supported
// because they could potentially get enabled.
Expand Down
34 changes: 22 additions & 12 deletions src/ripple/protocol/impl/LedgerFormats.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -74,6 +74,8 @@ LedgerFormats::LedgerFormats()
{sfIndexNext, soeOPTIONAL},
{sfIndexPrevious, soeOPTIONAL},
{sfNFTokenID, soeOPTIONAL},
{sfPreviousTxnID, soeOPTIONAL},
{sfPreviousTxnLgrSeq, soeOPTIONAL},
},
commonFields);

Expand Down Expand Up @@ -142,21 +144,25 @@ LedgerFormats::LedgerFormats()
{
{sfAmendments, soeOPTIONAL}, // Enabled
{sfMajorities, soeOPTIONAL},
{sfPreviousTxnID, soeOPTIONAL},
{sfPreviousTxnLgrSeq, soeOPTIONAL},
},
commonFields);

add(jss::FeeSettings,
ltFEE_SETTINGS,
{
// Old version uses raw numbers
{sfBaseFee, soeOPTIONAL},
{sfReferenceFeeUnits, soeOPTIONAL},
{sfReserveBase, soeOPTIONAL},
{sfReserveIncrement, soeOPTIONAL},
{sfBaseFee, soeOPTIONAL},
{sfReferenceFeeUnits, soeOPTIONAL},
{sfReserveBase, soeOPTIONAL},
{sfReserveIncrement, soeOPTIONAL},
// New version uses Amounts
{sfBaseFeeDrops, soeOPTIONAL},
{sfReserveBaseDrops, soeOPTIONAL},
{sfReserveIncrementDrops, soeOPTIONAL},
{sfPreviousTxnID, soeOPTIONAL},
{sfPreviousTxnLgrSeq, soeOPTIONAL},
},
commonFields);

Expand Down Expand Up @@ -240,6 +246,8 @@ LedgerFormats::LedgerFormats()
{sfDisabledValidators, soeOPTIONAL},
{sfValidatorToDisable, soeOPTIONAL},
{sfValidatorToReEnable, soeOPTIONAL},
{sfPreviousTxnID, soeOPTIONAL},
{sfPreviousTxnLgrSeq, soeOPTIONAL},
},
commonFields);

Expand Down Expand Up @@ -272,14 +280,16 @@ LedgerFormats::LedgerFormats()
add(jss::AMM,
ltAMM,
{
{sfAccount, soeREQUIRED},
{sfTradingFee, soeDEFAULT},
{sfVoteSlots, soeOPTIONAL},
{sfAuctionSlot, soeOPTIONAL},
{sfLPTokenBalance, soeREQUIRED},
{sfAsset, soeREQUIRED},
{sfAsset2, soeREQUIRED},
{sfOwnerNode, soeREQUIRED},
{sfAccount, soeREQUIRED},
{sfTradingFee, soeDEFAULT},
{sfVoteSlots, soeOPTIONAL},
{sfAuctionSlot, soeOPTIONAL},
{sfLPTokenBalance, soeREQUIRED},
{sfAsset, soeREQUIRED},
{sfAsset2, soeREQUIRED},
{sfOwnerNode, soeREQUIRED},
{sfPreviousTxnID, soeOPTIONAL},
{sfPreviousTxnLgrSeq, soeOPTIONAL},
},
commonFields);

Expand Down
10 changes: 7 additions & 3 deletions src/ripple/protocol/impl/STLedgerEntry.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -134,7 +134,8 @@ STLedgerEntry::thread(
uint256 const& txID,
std::uint32_t ledgerSeq,
uint256& prevTxID,
std::uint32_t& prevLedgerID)
std::uint32_t& prevLedgerID,
bool const includePrevTxnID)
{
uint256 oldPrevTxID = getFieldH256(sfPreviousTxnID);

Expand All @@ -149,8 +150,11 @@ STLedgerEntry::thread(

prevTxID = oldPrevTxID;
prevLedgerID = getFieldU32(sfPreviousTxnLgrSeq);
setFieldH256(sfPreviousTxnID, txID);
setFieldU32(sfPreviousTxnLgrSeq, ledgerSeq);
if (includePrevTxnID)
{
setFieldH256(sfPreviousTxnID, txID);
setFieldU32(sfPreviousTxnLgrSeq, ledgerSeq);
}
return true;
}

Expand Down
1 change: 1 addition & 0 deletions src/ripple/protocol/jss.h
Original file line number Diff line number Diff line change
Expand Up @@ -502,6 +502,7 @@ JSS(node_writes_duration_us); // out: GetCounts
JSS(node_write_retries); // out: GetCounts
JSS(node_writes_delayed); // out::GetCounts
JSS(nth); // out: RPC server_definitions
JSS(nunl); // in: AccountObjects
JSS(obligations); // out: GatewayBalances
JSS(offer); // in: LedgerEntry
JSS(offers); // out: NetworkOPs, AccountOffers, Subscribe
Expand Down
15 changes: 8 additions & 7 deletions src/ripple/rpc/impl/RPCHelpers.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -988,29 +988,30 @@ chooseLedgerEntryType(Json::Value const& params)
std::pair<RPC::Status, LedgerEntryType> result{RPC::Status::OK, ltANY};
if (params.isMember(jss::type))
{
static constexpr std::array<std::pair<char const*, LedgerEntryType>, 20>
static constexpr std::array<std::pair<char const*, LedgerEntryType>, 21>
types{
{{jss::account, ltACCOUNT_ROOT},
{jss::amendments, ltAMENDMENTS},
{jss::amm, ltAMM},
{jss::bridge, ltBRIDGE},
{jss::check, ltCHECK},
{jss::deposit_preauth, ltDEPOSIT_PREAUTH},
{jss::did, ltDID},
{jss::directory, ltDIR_NODE},
{jss::escrow, ltESCROW},
{jss::fee, ltFEE_SETTINGS},
{jss::hashes, ltLEDGER_HASHES},
{jss::nunl, ltNEGATIVE_UNL},
{jss::nft_offer, ltNFTOKEN_OFFER},
{jss::nft_page, ltNFTOKEN_PAGE},
{jss::offer, ltOFFER},
{jss::payment_channel, ltPAYCHAN},
{jss::signer_list, ltSIGNER_LIST},
{jss::state, ltRIPPLE_STATE},
{jss::ticket, ltTICKET},
{jss::nft_offer, ltNFTOKEN_OFFER},
{jss::nft_page, ltNFTOKEN_PAGE},
{jss::amm, ltAMM},
{jss::bridge, ltBRIDGE},
{jss::xchain_owned_claim_id, ltXCHAIN_OWNED_CLAIM_ID},
{jss::xchain_owned_create_account_claim_id,
ltXCHAIN_OWNED_CREATE_ACCOUNT_CLAIM_ID},
{jss::did, ltDID}}};
ltXCHAIN_OWNED_CREATE_ACCOUNT_CLAIM_ID}}};

auto const& p = params[jss::type];
if (!p.isString())
Expand Down
2 changes: 1 addition & 1 deletion src/test/app/AMMExtended_test.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -3271,7 +3271,7 @@ struct AMMExtended_test : public jtx::AMMTest
if (!BEAST_EXPECT(checkArraySize(affected, 4u)))
return;
auto ff =
affected[2u][sfModifiedNode.fieldName][sfFinalFields.fieldName];
affected[1u][sfModifiedNode.fieldName][sfFinalFields.fieldName];
BEAST_EXPECT(
ff[sfHighLimit.fieldName] ==
bob["USD"](100).value().getJson(JsonOptions::none));
Expand Down
10 changes: 8 additions & 2 deletions src/test/app/TxQ_test.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -2803,6 +2803,12 @@ class TxQ1_test : public beast::unit_test::suite
{
// This test focuses on which gaps in queued transactions are
// allowed to be filled even when the account's queue is full.

// NOTE: This test is fragile and dependent on ordering of
// transactions, which is affected by the closed/validated
// ledger hash. This test may need to be edited if changes
// are made that impact the ledger hash.
// TODO: future-proof this test.
using namespace jtx;
testcase("full queue gap handling");

Expand Down Expand Up @@ -2943,9 +2949,9 @@ class TxQ1_test : public beast::unit_test::suite
// may not reduce to 8.
env.close();
checkMetrics(__LINE__, env, 9, 50, 6, 5, 256);
BEAST_EXPECT(env.seq(alice) == aliceSeq + 13);
BEAST_EXPECT(env.seq(alice) == aliceSeq + 15);

// Close ledger 7. That should remove 7 more of alice's transactions.
// Close ledger 7. That should remove 4 more of alice's transactions.
env.close();
checkMetrics(__LINE__, env, 2, 60, 7, 6, 256);
BEAST_EXPECT(env.seq(alice) == aliceSeq + 19);
Expand Down
63 changes: 63 additions & 0 deletions src/test/ledger/Directory_test.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -396,13 +396,76 @@ struct Directory_test : public beast::unit_test::suite
}
}

void
testPreviousTxnID()
{
testcase("fixPreviousTxnID");
using namespace jtx;

auto const gw = Account{"gateway"};
auto const alice = Account{"alice"};
auto const USD = gw["USD"];

auto ledger_data = [](Env& env) {
Json::Value params;
params[jss::type] = jss::directory;
params[jss::ledger_index] = "validated";
return env.rpc(
"json", "ledger_data", to_string(params))[jss::result];
};

{
// fixPreviousTxnID is disabled.
Env env(*this, supported_amendments() - fixPreviousTxnID);
env.fund(XRP(10000), alice, gw);
env.close();
env.trust(USD(1000), alice);
env(pay(gw, alice, USD(1000)));
env.close();

{
auto const jrr = ledger_data(env);
BEAST_EXPECTS(
checkArraySize(jrr[jss::state], 2), jrr.toStyledString());
auto const directory = jrr[jss::state][0u];
BEAST_EXPECT(
directory["LedgerEntryType"] ==
jss::DirectoryNode); // sanity check
BEAST_EXPECT(!directory.isMember("PreviousTxnID"));
BEAST_EXPECT(!directory.isMember("PreviousTxnLgrSeq"));
}

// Now enable the amendment so the directory node is updated.
env.enableFeature(fixPreviousTxnID);
env.close();

// Make sure the `PreviousTxnID` and `PreviousTxnLgrSeq` fields now
// exist
env(offer(alice, XRP(1), USD(1)));
env.close();

{
auto const jrr = ledger_data(env);
BEAST_EXPECTS(
checkArraySize(jrr[jss::state], 3), jrr.toStyledString());
auto const directory = jrr[jss::state][0u];
BEAST_EXPECT(
directory["LedgerEntryType"] ==
jss::DirectoryNode); // sanity check
BEAST_EXPECT(directory.isMember("PreviousTxnID"));
BEAST_EXPECT(directory.isMember("PreviousTxnLgrSeq"));
}
}
}

void
run() override
{
testDirectoryOrdering();
testDirIsEmpty();
testRipd1353();
testEmptyChain();
testPreviousTxnID();
}
};

Expand Down

0 comments on commit 2712056

Please sign in to comment.