mirror of
https://github.com/bitcoin/bitcoin.git
synced 2025-01-26 03:03:22 -03:00
Merge bitcoin/bitcoin#29209: [26.x] more backports
11f3a7e6ba
Use hardened runtime on macOS release builds. (Mark Friedenbach)ac1b9a51db
[test] import descriptor wallet with reorged parent + IsFromMe child in mempool (glozow)ecb8ebc660
[test] rescan legacy wallet with reorged parent + IsFromMe child in mempool (Gloria Zhao)438ac2947d
snapshots: don't core dump when running -checkblockindex after `loadtxoutset` (Mark Friedenbach)7ec34554af
[log] mempool loading (glozow)fe0f8fe8aa
net: create I2P sessions with both ECIES-X25519 and ElGamal encryption (Jon Atack)fc62271015
[refactor] Add helper for iterating through mempool entries (stickies-v) Pull request description: Backports for 26.x. Includes: -453b481
from #28391 - #29179 - #29200 - #29227 - #28791 - #29127 ACKs for top commit: stickies-v: ACK11f3a7e6ba
Tree-SHA512: 20ef871ec768f2328056d83f958e595b36ae9b8baa8a6e8b0e1f02c3df4b16965a8e05dcb4323afbcc9ecc4bdde10931232512022c39ee7e12baf9795bf25bf1
This commit is contained in:
commit
74df372750
12 changed files with 169 additions and 21 deletions
|
@ -24,7 +24,7 @@ fi
|
|||
rm -rf ${TEMPDIR}
|
||||
mkdir -p ${TEMPDIR}
|
||||
|
||||
${SIGNAPPLE} sign -f --detach "${TEMPDIR}/${OUTROOT}" "$@" "${BUNDLE}"
|
||||
${SIGNAPPLE} sign -f --detach "${TEMPDIR}/${OUTROOT}" "$@" "${BUNDLE}" --hardened-runtime
|
||||
|
||||
tar -C "${TEMPDIR}" -czf "${OUT}" .
|
||||
rm -rf "${TEMPDIR}"
|
||||
|
|
|
@ -427,7 +427,7 @@ void Session::CreateIfNotCreatedAlready()
|
|||
const Reply& reply = SendRequestAndGetReply(
|
||||
*sock,
|
||||
strprintf("SESSION CREATE STYLE=STREAM ID=%s DESTINATION=TRANSIENT SIGNATURE_TYPE=7 "
|
||||
"inbound.quantity=1 outbound.quantity=1",
|
||||
"i2cp.leaseSetEncType=4,0 inbound.quantity=1 outbound.quantity=1",
|
||||
session_id));
|
||||
|
||||
m_private_key = DecodeI2PBase64(reply.Get("DESTINATION"));
|
||||
|
@ -445,7 +445,7 @@ void Session::CreateIfNotCreatedAlready()
|
|||
|
||||
SendRequestAndGetReply(*sock,
|
||||
strprintf("SESSION CREATE STYLE=STREAM ID=%s DESTINATION=%s "
|
||||
"inbound.quantity=3 outbound.quantity=3",
|
||||
"i2cp.leaseSetEncType=4,0 inbound.quantity=3 outbound.quantity=3",
|
||||
session_id,
|
||||
private_key_b64));
|
||||
}
|
||||
|
|
|
@ -176,4 +176,6 @@ public:
|
|||
mutable Epoch::Marker m_epoch_marker; //!< epoch when last touched, useful for graph algorithms
|
||||
};
|
||||
|
||||
using CTxMemPoolEntryRef = CTxMemPoolEntry::CTxMemPoolEntryRef;
|
||||
|
||||
#endif // BITCOIN_KERNEL_MEMPOOL_ENTRY_H
|
||||
|
|
|
@ -60,10 +60,20 @@ bool LoadMempool(CTxMemPool& pool, const fs::path& load_path, Chainstate& active
|
|||
if (version != MEMPOOL_DUMP_VERSION) {
|
||||
return false;
|
||||
}
|
||||
uint64_t num;
|
||||
file >> num;
|
||||
while (num) {
|
||||
--num;
|
||||
uint64_t total_txns_to_load;
|
||||
file >> total_txns_to_load;
|
||||
uint64_t txns_tried = 0;
|
||||
LogPrintf("Loading %u mempool transactions from disk...\n", total_txns_to_load);
|
||||
int next_tenth_to_report = 0;
|
||||
while (txns_tried < total_txns_to_load) {
|
||||
const int percentage_done(100.0 * txns_tried / total_txns_to_load);
|
||||
if (next_tenth_to_report < percentage_done / 10) {
|
||||
LogPrintf("Progress loading mempool transactions from disk: %d%% (tried %u, %u remaining)\n",
|
||||
percentage_done, txns_tried, total_txns_to_load - txns_tried);
|
||||
next_tenth_to_report = percentage_done / 10;
|
||||
}
|
||||
++txns_tried;
|
||||
|
||||
CTransactionRef tx;
|
||||
int64_t nTime;
|
||||
int64_t nFeeDelta;
|
||||
|
|
|
@ -806,7 +806,7 @@ public:
|
|||
{
|
||||
if (!m_node.mempool) return;
|
||||
LOCK2(::cs_main, m_node.mempool->cs);
|
||||
for (const CTxMemPoolEntry& entry : m_node.mempool->mapTx) {
|
||||
for (const CTxMemPoolEntry& entry : m_node.mempool->entryAll()) {
|
||||
notifications.transactionAddedToMempool(entry.GetSharedTx());
|
||||
}
|
||||
}
|
||||
|
|
|
@ -344,14 +344,13 @@ UniValue MempoolToJSON(const CTxMemPool& pool, bool verbose, bool include_mempoo
|
|||
}
|
||||
LOCK(pool.cs);
|
||||
UniValue o(UniValue::VOBJ);
|
||||
for (const CTxMemPoolEntry& e : pool.mapTx) {
|
||||
const uint256& hash = e.GetTx().GetHash();
|
||||
for (const CTxMemPoolEntry& e : pool.entryAll()) {
|
||||
UniValue info(UniValue::VOBJ);
|
||||
entryToJSON(pool, info, e);
|
||||
// Mempool has unique entries so there is no advantage in using
|
||||
// UniValue::pushKV, which checks if the key already exists in O(N).
|
||||
// UniValue::pushKVEnd is used instead which currently is O(1).
|
||||
o.pushKVEnd(hash.ToString(), info);
|
||||
o.pushKVEnd(e.GetTx().GetHash().ToString(), info);
|
||||
}
|
||||
return o;
|
||||
} else {
|
||||
|
|
|
@ -836,6 +836,18 @@ static TxMempoolInfo GetInfo(CTxMemPool::indexed_transaction_set::const_iterator
|
|||
return TxMempoolInfo{it->GetSharedTx(), it->GetTime(), it->GetFee(), it->GetTxSize(), it->GetModifiedFee() - it->GetFee()};
|
||||
}
|
||||
|
||||
std::vector<CTxMemPoolEntryRef> CTxMemPool::entryAll() const
|
||||
{
|
||||
AssertLockHeld(cs);
|
||||
|
||||
std::vector<CTxMemPoolEntryRef> ret;
|
||||
ret.reserve(mapTx.size());
|
||||
for (const auto& it : GetSortedDepthAndScore()) {
|
||||
ret.emplace_back(*it);
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
||||
std::vector<TxMempoolInfo> CTxMemPool::infoAll() const
|
||||
{
|
||||
LOCK(cs);
|
||||
|
|
|
@ -695,6 +695,7 @@ public:
|
|||
/** Returns info for a transaction if its entry_sequence < last_sequence */
|
||||
TxMempoolInfo info_for_relay(const GenTxid& gtxid, uint64_t last_sequence) const;
|
||||
|
||||
std::vector<CTxMemPoolEntryRef> entryAll() const EXCLUSIVE_LOCKS_REQUIRED(cs);
|
||||
std::vector<TxMempoolInfo> infoAll() const;
|
||||
|
||||
size_t DynamicMemoryUsage() const;
|
||||
|
|
|
@ -4850,7 +4850,9 @@ void ChainstateManager::CheckBlockIndex()
|
|||
// For testing, allow transaction counts to be completely unset.
|
||||
|| (pindex->nChainTx == 0 && pindex->nTx == 0)
|
||||
// For testing, allow this nChainTx to be unset if previous is also unset.
|
||||
|| (pindex->nChainTx == 0 && prev_chain_tx == 0 && pindex->pprev));
|
||||
|| (pindex->nChainTx == 0 && prev_chain_tx == 0 && pindex->pprev)
|
||||
// Transaction counts prior to snapshot are unknown.
|
||||
|| pindex->IsAssumedValid());
|
||||
|
||||
if (pindexFirstAssumeValid == nullptr && pindex->nStatus & BLOCK_ASSUMED_VALID) pindexFirstAssumeValid = pindex;
|
||||
if (pindexFirstInvalid == nullptr && pindex->nStatus & BLOCK_FAILED_VALID) pindexFirstInvalid = pindex;
|
||||
|
|
|
@ -330,6 +330,7 @@ BASE_SCRIPTS = [
|
|||
'wallet_create_tx.py --descriptors',
|
||||
'wallet_inactive_hdchains.py --legacy-wallet',
|
||||
'wallet_spend_unconfirmed.py',
|
||||
'wallet_rescan_unconfirmed.py --descriptors',
|
||||
'p2p_fingerprint.py',
|
||||
'feature_uacomment.py',
|
||||
'feature_init.py',
|
||||
|
|
|
@ -20,7 +20,10 @@ happened previously.
|
|||
"""
|
||||
|
||||
from test_framework.test_framework import BitcoinTestFramework
|
||||
from test_framework.address import AddressType
|
||||
from test_framework.address import (
|
||||
AddressType,
|
||||
ADDRESS_BCRT1_UNSPENDABLE,
|
||||
)
|
||||
from test_framework.util import (
|
||||
assert_equal,
|
||||
set_node_times,
|
||||
|
@ -109,7 +112,7 @@ class Variant(collections.namedtuple("Variant", "call data address_type rescan p
|
|||
|
||||
address, = [ad for ad in addresses if txid in ad["txids"]]
|
||||
assert_equal(address["address"], self.address["address"])
|
||||
assert_equal(address["amount"], self.expected_balance)
|
||||
assert_equal(address["amount"], self.amount_received)
|
||||
assert_equal(address["confirmations"], confirmations)
|
||||
# Verify the transaction is correctly marked watchonly depending on
|
||||
# whether the transaction pays to an imported public key or
|
||||
|
@ -223,11 +226,11 @@ class ImportRescanTest(BitcoinTestFramework):
|
|||
variant.node = self.nodes[2 + IMPORT_NODES.index(ImportNode(variant.prune, expect_rescan))]
|
||||
variant.do_import(variant.timestamp)
|
||||
if expect_rescan:
|
||||
variant.expected_balance = variant.initial_amount
|
||||
variant.amount_received = variant.initial_amount
|
||||
variant.expected_txs = 1
|
||||
variant.check(variant.initial_txid, variant.initial_amount, variant.confirmation_height)
|
||||
else:
|
||||
variant.expected_balance = 0
|
||||
variant.amount_received = 0
|
||||
variant.expected_txs = 0
|
||||
variant.check()
|
||||
|
||||
|
@ -247,7 +250,7 @@ class ImportRescanTest(BitcoinTestFramework):
|
|||
# Check the latest results from getbalance and listtransactions.
|
||||
for variant in IMPORT_VARIANTS:
|
||||
self.log.info('Run check for variant {}'.format(variant))
|
||||
variant.expected_balance += variant.sent_amount
|
||||
variant.amount_received += variant.sent_amount
|
||||
variant.expected_txs += 1
|
||||
variant.check(variant.sent_txid, variant.sent_amount, variant.confirmation_height)
|
||||
|
||||
|
@ -267,13 +270,44 @@ class ImportRescanTest(BitcoinTestFramework):
|
|||
address_type=variant.address_type.value,
|
||||
))
|
||||
variant.key = self.nodes[1].dumpprivkey(variant.address["address"])
|
||||
variant.initial_amount = get_rand_amount()
|
||||
variant.initial_amount = get_rand_amount() * 2
|
||||
variant.initial_txid = self.nodes[0].sendtoaddress(variant.address["address"], variant.initial_amount)
|
||||
variant.confirmation_height = 0
|
||||
variant.timestamp = timestamp
|
||||
|
||||
# Mine a block so these parents are confirmed
|
||||
assert_equal(len(self.nodes[0].getrawmempool()), len(mempool_variants))
|
||||
self.sync_mempools()
|
||||
block_to_disconnect = self.generate(self.nodes[0], 1)[0]
|
||||
assert_equal(len(self.nodes[0].getrawmempool()), 0)
|
||||
|
||||
# For each variant, create an unconfirmed child transaction from initial_txid, sending all
|
||||
# the funds to an unspendable address. Importantly, no change output is created so the
|
||||
# transaction can't be recognized using its outputs. The wallet rescan needs to know the
|
||||
# inputs of the transaction to detect it, so the parent must be processed before the child.
|
||||
# An equivalent test for descriptors exists in wallet_rescan_unconfirmed.py.
|
||||
unspent_txid_map = {txin["txid"] : txin for txin in self.nodes[1].listunspent()}
|
||||
for variant in mempool_variants:
|
||||
# Send full amount, subtracting fee from outputs, to ensure no change is created.
|
||||
child = self.nodes[1].send(
|
||||
add_to_wallet=False,
|
||||
inputs=[unspent_txid_map[variant.initial_txid]],
|
||||
outputs=[{ADDRESS_BCRT1_UNSPENDABLE : variant.initial_amount}],
|
||||
subtract_fee_from_outputs=[0]
|
||||
)
|
||||
variant.child_txid = child["txid"]
|
||||
variant.amount_received = 0
|
||||
self.nodes[0].sendrawtransaction(child["hex"])
|
||||
|
||||
# Mempools should contain the child transactions for each variant.
|
||||
assert_equal(len(self.nodes[0].getrawmempool()), len(mempool_variants))
|
||||
self.sync_mempools()
|
||||
|
||||
# Mock a reorg so the parent transactions are added back to the mempool
|
||||
for node in self.nodes:
|
||||
node.invalidateblock(block_to_disconnect)
|
||||
# Mempools should now contain the parent and child for each variant.
|
||||
assert_equal(len(node.getrawmempool()), 2 * len(mempool_variants))
|
||||
|
||||
# For each variation of wallet key import, invoke the import RPC and
|
||||
# check the results from getbalance and listtransactions.
|
||||
|
@ -283,11 +317,15 @@ class ImportRescanTest(BitcoinTestFramework):
|
|||
variant.node = self.nodes[2 + IMPORT_NODES.index(ImportNode(variant.prune, expect_rescan))]
|
||||
variant.do_import(variant.timestamp)
|
||||
if expect_rescan:
|
||||
variant.expected_balance = variant.initial_amount
|
||||
# Ensure both transactions were rescanned. This would raise a JSONRPCError if the
|
||||
# transactions were not identified as belonging to the wallet.
|
||||
assert_equal(variant.node.gettransaction(variant.initial_txid)['confirmations'], 0)
|
||||
assert_equal(variant.node.gettransaction(variant.child_txid)['confirmations'], 0)
|
||||
variant.amount_received = variant.initial_amount
|
||||
variant.expected_txs = 1
|
||||
variant.check(variant.initial_txid, variant.initial_amount)
|
||||
variant.check(variant.initial_txid, variant.initial_amount, 0)
|
||||
else:
|
||||
variant.expected_balance = 0
|
||||
variant.amount_received = 0
|
||||
variant.expected_txs = 0
|
||||
variant.check()
|
||||
|
||||
|
|
83
test/functional/wallet_rescan_unconfirmed.py
Executable file
83
test/functional/wallet_rescan_unconfirmed.py
Executable file
|
@ -0,0 +1,83 @@
|
|||
#!/usr/bin/env python3
|
||||
# Copyright (c) 2024 The Bitcoin Core developers
|
||||
# Distributed under the MIT software license, see the accompanying
|
||||
# file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
||||
"""Test that descriptor wallets rescan mempool transactions properly when importing."""
|
||||
|
||||
from test_framework.address import (
|
||||
address_to_scriptpubkey,
|
||||
ADDRESS_BCRT1_UNSPENDABLE,
|
||||
)
|
||||
from test_framework.messages import COIN
|
||||
from test_framework.test_framework import BitcoinTestFramework
|
||||
from test_framework.util import assert_equal
|
||||
from test_framework.wallet import MiniWallet
|
||||
from test_framework.wallet_util import test_address
|
||||
|
||||
|
||||
class WalletRescanUnconfirmed(BitcoinTestFramework):
|
||||
def add_options(self, parser):
|
||||
self.add_wallet_options(parser, legacy=False)
|
||||
|
||||
def set_test_params(self):
|
||||
self.num_nodes = 1
|
||||
|
||||
def skip_test_if_missing_module(self):
|
||||
self.skip_if_no_wallet()
|
||||
self.skip_if_no_sqlite()
|
||||
|
||||
def run_test(self):
|
||||
self.log.info("Create wallets and mine initial chain")
|
||||
node = self.nodes[0]
|
||||
tester_wallet = MiniWallet(node)
|
||||
|
||||
node.createwallet(wallet_name='w0', disable_private_keys=False)
|
||||
w0 = node.get_wallet_rpc('w0')
|
||||
|
||||
self.log.info("Create a parent tx and mine it in a block that will later be disconnected")
|
||||
parent_address = w0.getnewaddress()
|
||||
tx_parent_to_reorg = tester_wallet.send_to(
|
||||
from_node=node,
|
||||
scriptPubKey=address_to_scriptpubkey(parent_address),
|
||||
amount=COIN,
|
||||
)
|
||||
assert tx_parent_to_reorg["txid"] in node.getrawmempool()
|
||||
block_to_reorg = self.generate(tester_wallet, 1)[0]
|
||||
assert_equal(len(node.getrawmempool()), 0)
|
||||
node.syncwithvalidationinterfacequeue()
|
||||
assert_equal(w0.gettransaction(tx_parent_to_reorg["txid"])["confirmations"], 1)
|
||||
|
||||
# Create an unconfirmed child transaction from the parent tx, sending all
|
||||
# the funds to an unspendable address. Importantly, no change output is created so the
|
||||
# transaction can't be recognized using its outputs. The wallet rescan needs to know the
|
||||
# inputs of the transaction to detect it, so the parent must be processed before the child.
|
||||
w0_utxos = w0.listunspent()
|
||||
|
||||
self.log.info("Create a child tx and wait for it to propagate to all mempools")
|
||||
# The only UTXO available to spend is tx_parent_to_reorg.
|
||||
assert_equal(len(w0_utxos), 1)
|
||||
assert_equal(w0_utxos[0]["txid"], tx_parent_to_reorg["txid"])
|
||||
tx_child_unconfirmed_sweep = w0.sendall([ADDRESS_BCRT1_UNSPENDABLE])
|
||||
assert tx_child_unconfirmed_sweep["txid"] in node.getrawmempool()
|
||||
node.syncwithvalidationinterfacequeue()
|
||||
|
||||
self.log.info("Mock a reorg, causing parent to re-enter mempools after its child")
|
||||
node.invalidateblock(block_to_reorg)
|
||||
assert tx_parent_to_reorg["txid"] in node.getrawmempool()
|
||||
|
||||
self.log.info("Import descriptor wallet on another node")
|
||||
descriptors_to_import = [{"desc": w0.getaddressinfo(parent_address)['parent_desc'], "timestamp": 0, "label": "w0 import"}]
|
||||
|
||||
node.createwallet(wallet_name="w1", disable_private_keys=True)
|
||||
w1 = node.get_wallet_rpc("w1")
|
||||
w1.importdescriptors(descriptors_to_import)
|
||||
|
||||
self.log.info("Check that the importing node has properly rescanned mempool transactions")
|
||||
# Check that parent address is correctly determined as ismine
|
||||
test_address(w1, parent_address, solvable=True, ismine=True)
|
||||
# This would raise a JSONRPCError if the transactions were not identified as belonging to the wallet.
|
||||
assert_equal(w1.gettransaction(tx_parent_to_reorg["txid"])["confirmations"], 0)
|
||||
assert_equal(w1.gettransaction(tx_child_unconfirmed_sweep["txid"])["confirmations"], 0)
|
||||
|
||||
if __name__ == '__main__':
|
||||
WalletRescanUnconfirmed().main()
|
Loading…
Add table
Reference in a new issue