achow101
commented at 10:52 PM on July 21, 2025:
member
The wallet uses SQLite as a key-value store even though SQLite is a powerful relational database engine. This causes us numerous headaches due to the need to serialize multiple fields together, and since record read from the database during loading can come in any order.
Notably, for transactions, if we were able to load them in the transaction order assigned by the wallet, PRs like #27865 would be a bit simpler and easier to reason about.
This PR makes it possible for us to do that by storing transactions in a separate table. This table has a column for each field in CWalletTx, which lets us use a SQL statement like SELECT * FROM transactions ORDER BY pos which guarantees us the order in which the transactions are loaded into the wallet.
The eventual goal is to use SQLite as a relational database with tables that store our keys and other metadata that can reference each other so that the wallet doesn't just use the database as a storage mechanism. But that is still a work in progress.
This PR makes use of the last client opened features introduced in #32895 to determine when the old record style needs to be upgraded to the new transactions table. This should give us sufficient upgrade-downgrade-upgrade handling to not lose any funds.
#35018 (wallet, bench: Use Nanobench setup() for wallet benchmarks, and remove DuplicateMockDatabase by achow101)
#34909 (wallet, refactor: modularise wallet by extracting out legacy wallet migration by rkrux)
#34838 (interfaces: typed receive-request APIs by MkDev11)
#34502 (wallet: remove most asserts of WALLET_FLAG_DESCRIPTORS flag by rkrux)
#33033 (wallet, sqlite: Encapsulate SQLite statements in a RAII class by achow101)
#32895 (wallet: Prepare for future upgrades by recording versions of last client to open and decrypt by achow101)
#32763 (wallet: Replace CWalletTx::mapValue and vOrderForm with explicit class members by achow101)
#32489 (wallet: Add exportwatchonlywallet RPC to export a watchonly version of a wallet by achow101)
#27865 (wallet: Track no-longer-spendable TXOs separately by achow101)
#25722 (refactor: Use util::Result class for wallet loading by ryanofsky)
If you consider this pull request important, please also help to review the conflicting pull requests. Ideally, start with the one that should be merged first.
<!--5faf32d7da4f0f540f40219e4f7537a3-->
LLM Linter (✨ experimental)
Possible typos and grammar issues:
InActive(abandoned=false) -> Inactive(abandoned=false) [Comment typo in TxStateDataVisitor: “InActive” doesn’t match the actual type name (TxStateInactive) and can confuse the intended meaning.]
<sup>2026-04-20 20:03:02</sup>
DrahtBot added the label CI failed on Jul 22, 2025
DrahtBot
commented at 12:40 AM on July 22, 2025:
contributor
<!--85328a0da195eb286784d51f73fa0af9-->
🚧 At least one of the CI tasks failed.
<sub>Task lint: https://github.com/bitcoin/bitcoin/runs/46429203587</sub>
<sub>LLM reason (✨ experimental): The CI failure is caused by Python lint errors due to unresolved 'self' references in test files.</sub>
<details><summary>Hints</summary>
Try to run the tests locally, according to the documentation. However, a CI failure may still
happen due to a number of reasons, for example:
Possibly due to a silent merge conflict (the changes in this pull request being
incompatible with the current code in the target branch). If so, make sure to rebase on the latest
commit of the target branch.
A sanitizer issue, which can only be found by compiling with the sanitizer and running the
affected test.
An intermittent issue.
Leave a comment here, if you need help tracking down a confusing failure.
</details>
achow101 force-pushed on Jul 22, 2025
achow101 force-pushed on Jul 22, 2025
DrahtBot removed the label CI failed on Jul 22, 2025
achow101 force-pushed on Jul 23, 2025
w0xlt
commented at 9:15 PM on July 23, 2025:
contributor
Concept ACK
achow101 force-pushed on Jul 24, 2025
DrahtBot added the label Needs rebase on Jul 29, 2025
achow101 force-pushed on Jul 29, 2025
DrahtBot removed the label Needs rebase on Jul 29, 2025
DrahtBot added the label Needs rebase on Aug 8, 2025
achow101 force-pushed on Aug 8, 2025
DrahtBot removed the label Needs rebase on Aug 8, 2025
DrahtBot added the label Needs rebase on Aug 13, 2025
achow101 force-pushed on Aug 14, 2025
DrahtBot removed the label Needs rebase on Aug 14, 2025
DrahtBot added the label Needs rebase on Aug 16, 2025
achow101 force-pushed on Aug 16, 2025
achow101 force-pushed on Aug 19, 2025
DrahtBot removed the label Needs rebase on Aug 19, 2025
DrahtBot added the label Needs rebase on Sep 23, 2025
achow101 force-pushed on Sep 23, 2025
DrahtBot removed the label Needs rebase on Sep 24, 2025
rkrux
commented at 8:14 AM on October 9, 2025:
contributor
Concept ACKa25557a
I have gone through the PR description and have skimmed over the commit messages as of now. At the outset, I find myself agreeing with the intent here. I am not aware of a strong enough reason to keep using SQLite as a KVS for all purposes, using its relational database properties for wallet transactions seem like a good start.
I did notice some serialisation specifics of transaction properties in a couple previous PRs that I didn't fully understand why were required; it could be a lot cleaner and explicit in implementation if using SQLite as an RDBMS can avoid the need for doing those specific serialisations.
DrahtBot added the label Needs rebase on Dec 2, 2025
achow101 force-pushed on Dec 10, 2025
DrahtBot removed the label Needs rebase on Dec 10, 2025
DrahtBot added the label CI failed on Dec 10, 2025
DrahtBot
commented at 8:15 PM on December 10, 2025:
contributor
Try to run the tests locally, according to the documentation. However, a CI failure may still
happen due to a number of reasons, for example:
Possibly due to a silent merge conflict (the changes in this pull request being
incompatible with the current code in the target branch). If so, make sure to rebase on the latest
commit of the target branch.
A sanitizer issue, which can only be found by compiling with the sanitizer and running the
affected test.
An intermittent issue.
Leave a comment here, if you need help tracking down a confusing failure.
</details>
achow101 force-pushed on Dec 10, 2025
DrahtBot removed the label CI failed on Dec 11, 2025
DrahtBot added the label Needs rebase on Dec 17, 2025
achow101 force-pushed on Dec 22, 2025
DrahtBot removed the label Needs rebase on Dec 22, 2025
DrahtBot added the label CI failed on Dec 23, 2025
achow101 force-pushed on Jan 3, 2026
achow101 force-pushed on Jan 3, 2026
DrahtBot removed the label CI failed on Jan 3, 2026
DrahtBot added the label Needs rebase on Jan 19, 2026
achow101 force-pushed on Jan 19, 2026
DrahtBot added the label CI failed on Jan 19, 2026
DrahtBot
commented at 7:49 PM on January 19, 2026:
contributor
<!--85328a0da195eb286784d51f73fa0af9-->
🚧 At least one of the CI tasks failed.
<sub>Task test max 6 ancestor commits: https://github.com/bitcoin/bitcoin/actions/runs/21148471093/job/60819287525</sub>
<sub>LLM reason (✨ experimental): Build failed: the CI cmake/make run exited with non-zero status (gmake: all target) after linking bitcoinkernel, indicating a generic build error.</sub>
<details><summary>Hints</summary>
Try to run the tests locally, according to the documentation. However, a CI failure may still
happen due to a number of reasons, for example:
Possibly due to a silent merge conflict (the changes in this pull request being
incompatible with the current code in the target branch). If so, make sure to rebase on the latest
commit of the target branch.
A sanitizer issue, which can only be found by compiling with the sanitizer and running the
affected test.
An intermittent issue.
Leave a comment here, if you need help tracking down a confusing failure.
</details>
DrahtBot removed the label Needs rebase on Jan 19, 2026
achow101 force-pushed on Jan 19, 2026
achow101 force-pushed on Jan 26, 2026
DrahtBot added the label Needs rebase on Feb 4, 2026
achow101 force-pushed on Feb 4, 2026
DrahtBot removed the label Needs rebase on Feb 4, 2026
DrahtBot added the label Needs rebase on Mar 2, 2026
achow101 force-pushed on Mar 5, 2026
DrahtBot removed the label Needs rebase on Mar 5, 2026
DrahtBot added the label Needs rebase on Mar 11, 2026
achow101 force-pushed on Mar 14, 2026
DrahtBot removed the label Needs rebase on Mar 14, 2026
achow101 force-pushed on Apr 1, 2026
achow101 force-pushed on Apr 2, 2026
achow101 force-pushed on Apr 2, 2026
achow101 force-pushed on Apr 2, 2026
achow101 force-pushed on Apr 2, 2026
DrahtBot added the label Needs rebase on Apr 19, 2026
wallet: Pass replaces_txid to CommitTransaction outside of mapValue
Instead of updating mapValue with the "replaces_txid" value and passing
the updated mapValue to CommitTransaction, pass the replaces_txid
directly to CommitTransaction which updates mapValue as necessary.
This is prepration for removing mapValue.
2d0c77e650
wallet: Pass comment and comment_to to CommitTransaction
Instead of passing these by setting them in mapValue, pass them directly
to CommitTransaction.
This is preparation for removing mapValue.
57184351b6
wallet: Drop mapValue from CommitTransaction
The values previously passed in mapValue are now parameters to
CommitTransaction so there is no need for mapValue to be passed.
6c4610fa4e
wallet: Make CWalletTx "from" and "message" member variables
Instead of storing "from" and "message" inside of mapValue, store these
explicitly as members of CWalletTx.
6abb8be881
wallet: Make CWalletTx "comment" and "to" member variables
Instead of storing "comment" and "to" inside of mapValue, store these
expliclty as members of CWalletTx.
a7004d1d3d
wallet: Make CWalletTx "replaces_txid" and "replaced_by_txid" member variables
Instead of storing "replaces_txid" and "replaced_by_txid" as strings inside of
mapValue, store these expliclty as members of CWalletTx.
92a8c92470
wallet: Drop mapValue from CWalletTx
It doesn't make sense to be storing relevant metadata variables inside
of a string map in CWalletTx. All of the fields have been pulled out
into separate members, so there is no need for mapValue to stick around.
e159561824
wallet: Drop vOrderForm from CommitTransaction
This parameter is only used to pass in the "Messages" from the GUI.
Instead of making it opaque by putting those into vOrderForm, use a
specific dedicated parameter for providing the messages.
6a662b6192
wallet: Replace CWalletTx's vOrderForm with specific fields
vOrderForm contained 2 kinds of strings: BIP 21 messages, and BIP 70
Payment Requests. Instead of having both inside of a single vOrderForm
field that is opaque, split them into separate std::vector<std::string>
to contain this metadata.
c79c1d4845
walletdb: Decouple last client record from CLIENT_VERSION
Instead of tying the last client record with the node version, introduce
a separate wallet client version that will be increased as necessary
when new automatic upgrades are introduced.
9b43751035
wallet: Introduce LastClientFeatures flags and LAST_OPENED_FEATURES record
LastClientFeatures are feature flags indicating what automatic upgrade
features supported by the last client that opened the wallet.
The LAST_OPENED_FEATURES record stores these flags and must be
set to match the exact features supported by the client that opens a
wallet.
213080d619
wallet: Record the supported features of the last client to decrypt a wallet8126575552
wallet: Set last opened and decrypted features during migration
Set these flags to avoid the automatic upgrade after migrating.
e77138990d
sqlite: Add SQLiteStatement RAII class
This class will be used to encapsulate a sqlite3_stmt
f23f1eac8a
sqlite: Make Column template functionf5b150d650
sqlite: Refactor ReadPragmaInteger to use SQLiteStatementf4a7236714
sqlite: Use SQLiteStatement in PRAGMA integrity_check2a30bfbf2a
sqlite: Use SQLiteStatement in check_main_stmteef991f9b8
sqlite: Have SQLiteCursor store SQLiteStatement3818f59a0b
sqlite: Replace remaining sqlite3_stmt usage with SQLiteStatement31fb32f7a9
sqlite: Refactor common writing code from WriteKey and ExecStatement
WriteKey and ExecStatement use the same code for the actual execution of
the statement. This is refactored into a separate function, also called
ExecStatement, and the original ExecStatement renamed to
ExecEraseStatement as it is only used by the erase functions.
b8b2ae0b82
sqlite: Inline BindBlobToStatement and SpanFromBlobac59387274
sqlite: Construct SQLiteStatements when needed
Instead of constructing all SQLiteStatements when SQLiteBatch is
constructed, construct them once when they are needed before each read,
write, or erase operation. Once constructed, the statement will persist
for the lifetime of the SQLiteBatch to be reused across multiple
statements.
cfa3a0e2b1
util: Add span constructor to (W)Txidcfa018407b
sqlite: Make SQLiteStatement::Bind a template function
We will need to bind data types other than blob
d6c061b3d1
sqlite: Make SQLiteDatabase::Column an std::optional
To handle columns containing NULL values, Column needs to return some
value representing NULL, so make it a std::optional.
851234aefc
sqlite: Add additional blob types to SQLiteStatement::Column
Not all blob data types fit in ColumnBlob, so we need additional
template type requirements to match those.
e560f541bd
wallet: Add Un/Serialize to TxState structse6f136b576
wallet: Add new serialization format functions for TxState72c5c53a8a
sqlite: Create a 'transactions' table if it does not exist3e1f1a9ba6
sqlite: Add functions and statements for transactions table39e8ba63b0
sqlite: Iterate the transactions table with SQLiteCursor1d6b94a85d
walletdb: Add functions to modify transactions tableeb7ad1b384
test: Include sql transactions table in DuplicateMockDatabase132b9cc4d7
wallet: Also write to the new transactions tablead91aaf18a
wallet: Perform automatic upgrade to using the transactions table306e9e510a
walletdb: Load from transactions table and use original tx for upgrade
When loading a wallet, always load from the transactions table, except
when loading a legacy wallet for migration.
The original 'tx' records are only used to upgrade to the transactions
table if an upgraded is necessary.
67d182392b
achow101 force-pushed on Apr 20, 2026
DrahtBot removed the label Needs rebase on Apr 20, 2026
DrahtBot removed the label CI failed on Apr 20, 2026
sedited referenced this in commit db98e357d3 on May 2, 2026
DrahtBot
commented at 8:49 AM on May 2, 2026:
contributor
<!--cf906140f33d8803c4a75a2196329ecb-->
🐙 This pull request conflicts with the target branch and needs rebase.
DrahtBot added the label Needs rebase on May 2, 2026
This is a metadata mirror of the GitHub repository
bitcoin/bitcoin.
This site is not affiliated with GitHub.
Content is generated from a GitHub metadata backup.
generated: 2026-05-09 15:12 UTC
This site is hosted by @0xB10C More mirrored repositories can be found on mirror.b10c.me