achow101
commented at 8:06 pm on October 20, 2020:
member
Adds a unique id for each wallet that is saved in a new “walletid” record. For compatibility, wallets using BDB will use the BDB generated id. All other wallets will have a randomly generated id if an id does not already exist.
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.
in
src/wallet/wallet.cpp:4138
in
545c362e78outdated
I think this is in the wrong place. From a quick look CWallet::LoadWallet seems a better place, like:
0if (wallet_id.IsNull()) {
1if (database->Format() =="dbd") {
2// use BDB ID
3 } else {
4// generate random id
5 }
6// persist
7 }
8return DBErrors::LOAD_OK;
9}
This would allow you to drop EnsureWalletIDWithDB. Only bad thing here is to compare agains “bdb”. Would be better to add Format() and FormatString().
achow101
commented at 2:24 am on October 22, 2020:
I’ve moved all of the logic into EnsureWalletIDWithDB.
in
src/wallet/wallet.h:1290
in
545c362e78outdated
1282@@ -1281,6 +1283,13 @@ class CWallet final : public WalletStorage, public interfaces::Chain::Notificati
12831284 //! Add a descriptor to the wallet, return a ScriptPubKeyMan & associated output type
1285 ScriptPubKeyMan* AddWalletDescriptor(WalletDescriptor& desc, const FlatSigningProvider& signing_provider, const std::string& label);
1286+
1287+ //! Set the wallet ID
1288+ void LoadWalletID(const uint160& id);
1289+ //! Get the wallet iD
1290+ const uint160& GetWalletID();
achow101
commented at 2:30 am on October 22, 2020:
Done
promag
commented at 11:35 pm on October 21, 2020:
member
Concept ACK.
Can’t we just set the id for sqlite wallets on creation? Or you want to handle wallets already created?
Edit: do we want to modify old wallets? CWallet could return BDB Id on the fly without persisting it. Later we would just persist it if the wallet is converted.
achow101
commented at 11:43 pm on October 21, 2020:
member
Can’t we just set the id for sqlite wallets on creation? Or you want to handle wallets already created?
This needs to handle wallets already created.
Edit: do we want to modify old wallets? CWallet could return BDB Id on the fly without persisting it. Later we would just persist it if the wallet is converted.
It should be persisted as a wallet record. The BDB ID is purely because existing PRs use it as a wallet ID. But proper support for a wallet id means it needs to be a record.
Also, a record means that a wallet migrated with createfromdump will have the ID as a record correctly without having that command know about this stuff.
promag
commented at 11:45 pm on October 21, 2020:
member
luke-jr
commented at 12:33 pm on October 22, 2020:
Is there a reason to limit it to 160-bit? Seems like to future-proof, 256-bit ids would be better. And even if we end up only generating 160-bit ids now, it would be nice to load longer ones so we don’t have to break compatibility if we ever generate more…
achow101
commented at 3:48 pm on October 22, 2020:
The current BDB ID is 160 bits long. I don’t think limiting to “just” 160 bits will be a problem. Why would we need to load longer ones?
Probably 160-bit is enough, but I can’t predict the future. :)
in
src/wallet/wallet.cpp:4779
in
234e4678c7outdated
4600+
4601+ // No ID, so add one
4602+ auto bdb_db = dynamic_cast<BerkeleyDatabase*>(database.get());
4603+ if (bdb_db) {
4604+ // For BDB, use the BDB unique ID
4605+ memcpy(wallet_id.data(), bdb_db->env->m_fileids[bdb_db->strFile].value, wallet_id.size());
luke-jr
commented at 12:35 pm on October 22, 2020:
Maybe an assert that wallet_id.size() == sizeof(bdb_db->env->m_fileids[bdb_db->strFile].value)? (or use the latter and assert that it’s <= wallet_id.size())
achow101
commented at 4:46 pm on October 22, 2020:
Done
luke-jr changes_requested
jonasschnelli
commented at 12:42 pm on October 22, 2020:
contributor
utACK234e4678c774aa0021a45810acd9a0f3f5bf3b64
promag
commented at 12:52 pm on October 22, 2020:
member
For compatibility, wallets using BDB will use the BDB generated id
@achow101 compatibility with what?
achow101 force-pushed
on Oct 22, 2020
luke-jr
commented at 5:15 pm on October 22, 2020:
member
@promag With existing wallets. Right now, the only unique identifier they have is the BDB database id.
laanwj added this to the milestone 0.21.0
on Oct 22, 2020
luke-jr approved
luke-jr
commented at 2:30 am on October 24, 2020:
member
While I think a variable-length wallet id would be more future-proof, I don’t know of a use case, and this doesn’t strictly tie our hands in that regard yet, and it’s a strict improvement, so utACK anyway.
hebasto approved
hebasto
commented at 12:08 pm on October 26, 2020:
member
ACK4b7b2532e42f10b8f695d3800bebff88b586550c, tested on Linux Mint 20 (x86_64) both bdb and sqlite wallets.
Verified that WALLETID key is written to a new sqlite wallet database.
ryanofsky
commented at 1:26 pm on October 26, 2020:
contributor
This change is a bad change. It’s a perfect example of dumb, unreasoning, cargo-cult design (there is no design discussion here) copying a poorly thought BDB feature that directly lead to data corruption previously in #11429 and is a trap and footgun for the future. Merging this change now will encourage adding fragile assumptions to code (uniqueness of ids), and adding pointless and confusing restrictions around wallet usage (disallowing basic file operations like copies).
It’s possible to put thought into design designs that affect data format. For example, There are explicit reasons subversion uses unique ids and git doesn’t use them. There are advantages and disadvantages to different approaches.
In this case, if there is any reason to add wallets id later (maybe adding a rename detection heuristic, maybe trying to help out forensics or recovery tools), nothing prevents wallet ids from being added later. It seems the main things gained now by merging this PR would be:
Extra code that no usage or test coverage
Foot gun and expectation of future foot surgery
laanwj removed this from the milestone 0.21.0
on Oct 29, 2020
laanwj
commented at 10:43 am on October 29, 2020:
member
As this is controversial and needs more discussion, I’m removing it from the 0.21 milestone. From discussion on IRC I get the idea this is a “nice to have” for some people so I don’t agree there is any hurry on this.
luke-jr
commented at 12:54 pm on October 29, 2020:
member
This is a regression bugfix and a blocker for sqlite wallets. 0.21 should not be released without it.
ryanofsky
commented at 1:09 pm on October 29, 2020:
contributor
This is a regression bugfix and a blocker for sqlite wallets. 0.21 should not be released without it.
Can you give an example of something that would be broken, or a feature that would be more difficult to implement if this is not included in 0.21? This would be news to me and I don’t know what it is referring to.
A UUID field (or any field that can ignored by previous releases) can be added at any time. Wallet code is intentionally written to ignore unrecognized rows and non-mandatory flags so things like this can be added.
We can disagree about whether features that depend on UUIDs are ideal or if there are better alternatives. But even if they are ideal and perfect in every way, there’s no reason the UUID fields can’t be added at the same time as features which use them.
luke-jr
commented at 1:30 pm on October 29, 2020:
member
Adding a UUID later would mean backups are missing it, and restoring from a backup could break features relying on it. Even if we ignore these risks, the UUID is itself a feature wallets have always had, that is now missing in sqlite wallets.
Adding it back now has literally zero downsides. Absolute worst case, we could just not use it.
Furthermore, not having it now also means features can’t use it until those features are merged into Core, without risk of their wallet format diverging. Considering Core’s history with refusing to be compatible with other software, that isn’t a good risk to take, so I would actually have to disable such features in Knots when a sqlite wallet is used.
ryanofsky
commented at 1:39 pm on October 29, 2020:
contributor
Adding a UUID later would mean backups are missing it, and restoring from a backup could break features relying on it.
This isn’t true. Unrecognized rows are not stripped from backups. The other comments about core and knots also do not seem to make sense. We shouldn’t go back and forth on this endlessly, but I think the more specific and detailed you can be about problems you are concerned about, the more quickly we will be able to see the right thing to do.
luke-jr
commented at 3:01 pm on October 29, 2020:
member
The UUID isn’t simply unrecognised, it’s ENTIRELY MISSING in sqlite wallets right now.
ryanofsky
commented at 3:23 pm on October 29, 2020:
contributor
The UUID isn’t simply unrecognised, it’s ENTIRELY MISSING in sqlite wallets right now.
I know this. You know this. What I don’t know is what specific problem is fixed by merging this PR now that would not be fixed by merging it later (or perhaps by never relying on uuids at all, but I don’t want to broaden the topic if it is not necessary). I’m encouraging you to describe an example in some level of detail so we can actually think about it.
luke-jr
commented at 4:10 pm on October 29, 2020:
member
Merging it later won’t retroactively ensure wallets created by 0.21 and then copied share a common UUID (which is kinda the whole point of a UUID).
Merging it later also means features using the UUID won’t work on sqlite wallets in the meantime.
jonasschnelli added this to the milestone 0.21.0
on Oct 29, 2020
jonasschnelli removed this from the milestone 0.21.0
on Nov 5, 2020
jonasschnelli
commented at 7:12 pm on November 5, 2020:
contributor
Removed the 0.21 milestone again. This seems indeed be too controversial for a last minute 0.21 merge.
sipa
commented at 7:31 pm on November 5, 2020:
member
“It’s a feature of existing wallets, new wallets should have it too” is not a good argument. It’s a new wallet type, it will support whatever features we declare it to have.
Wallet IDs were originally introduced in BDB because of database environments being unable to open the same database multiple times. Restricting the ability to open multiple copies simultaneously is a necessity here, not a feature. There is no point in maintaining it if not required by the database layer.
However, I believe there are reasons why wallet ids are useful, independent of the argument above: Bitcoin Core wallets are intended to have a linear history. While we take lots of effort to make sure using a backup is won’t cause monetary loss, this isn’t perfect and doesn’t apply to all data in the wallet. Adding address labels, which may matter for accounting reasons, in one copy won’t make them show up in another. Creating two transactions from two versions of the same wallet ~simultaneously (or while one hasn’t caught up with the chain) may result in unintentionally creating conflicting transactions. Because of that, I think it makes conceptual sense to have an ID, and use that to warn (not reject) when opening multiple copies.
I don’t think the above is urgent, or necessary, and I think it should be treated as a new feature. It’s trivial to add IDs though, and having it earlier means features depending on it can be introduced later without needing a new backup.
Concept ACK, and I’m ok with having it in 0.21, but adding wallet IDs whenever a feature actually using them is introduced is fine by me as well.
achow101
commented at 7:40 pm on November 5, 2020:
member
Given that descriptor (and thus sqlite) wallets are considered experimental, I think it’s find to miss the id for 0.21. I think it’s reasonable to ask users make a new backup when we no longer consider descriptor wallets to be experimental, and we can introduce the wallet id before that time.
DrahtBot added the label
Needs rebase
on Nov 9, 2020
achow101 force-pushed
on Nov 9, 2020
DrahtBot removed the label
Needs rebase
on Nov 9, 2020
hebasto approved
hebasto
commented at 7:08 pm on November 9, 2020:
member
re-ACK30a02e9d60af542003b333996e658f982ceb2d9a, only rebased since my previous review, verified with git range-diff master 4b7b253 30a02e9.
ryanofsky
commented at 7:46 pm on December 15, 2020:
contributor
Because of that, I think it makes conceptual sense to have an ID, and use that to warn (not reject) when opening multiple copies.
Without knowing who the warning would be for or what it would say, it doesn’t seem clear to me how it would be useful. But assuming it is useful, wouldn’t the warning be more robust if it checked directly for duplicate seeds and descriptors, instead of going indirectly through this PR, adding a new random number to sqlite databases and checking for uniqueness based on that?
I still do stand by my NACK from #20205 (comment), and I don’t think this code should be merged until there is an actual, real life, properly designed feature which depends on it. I think it is possible to build useful heuristics (usage warnings, rename detection) with UUIDs. I also think it’s possible to build misfeatures that make software painful to work with (and even cause data corruption, see #11429) because of bad and fragile assumptions about uniqueness. So adding this dead code to the wallet which complicates the already messy database layer and exposes a new API which is asking to be misused does not seem like a win to me right now. The harms outweigh the benefits currently, because there aren’t benefits currently.
luke-jr
commented at 7:59 pm on December 15, 2020:
member
I’m aware and not ignoring it. If #19463 is close to being merged and it depends on this PR, then this PR should be merged. But this isn’t the case now. #19463 isn’t close to being merged, and I don’t think it makes sense for #19463 to depend on this PR. I’m just objecting to this PR on its merits, and its harms and benefits right now.
DrahtBot added the label
Needs rebase
on May 19, 2021
achow101 force-pushed
on May 19, 2021
DrahtBot removed the label
Needs rebase
on May 19, 2021
DrahtBot added the label
Needs rebase
on Jun 9, 2021
achow101 force-pushed
on Jun 9, 2021
DrahtBot removed the label
Needs rebase
on Jun 9, 2021
hebasto
commented at 7:39 pm on June 11, 2021:
member
0In file included from wallet/wallet.cpp:34:1./wallet/bdb.h:27:10: fatal error: 'db_cxx.h' file not found
2#include<db_cxx.h>3^~~~~~~~~~41 error generated.
DrahtBot added the label
Needs rebase
on Sep 3, 2021
ghost
commented at 11:39 am on September 3, 2021:
none
Concept ACK
Can we add 0/1/2/3 for every ID based on chain the wallet was created in? 0-Mainnet 1-Testnet 2-Regtest 3-Signet
Chains for wallets can already be identified using genesis block and this is used in #14533, however genesis block can be same in few cases as mentioned in #14533 (comment), we can use this ID to differentiate for wallets created in future with 00/01/02/03.
achow101 force-pushed
on Sep 3, 2021
DrahtBot removed the label
Needs rebase
on Sep 3, 2021
DrahtBot added the label
Needs rebase
on Sep 7, 2021
achow101 force-pushed
on Sep 7, 2021
DrahtBot removed the label
Needs rebase
on Sep 7, 2021
DrahtBot added the label
Needs rebase
on Sep 26, 2021
achow101 force-pushed
on Sep 27, 2021
DrahtBot removed the label
Needs rebase
on Sep 27, 2021
maflcko
commented at 10:52 am on December 3, 2021:
member
DrahtBot added the label
Needs rebase
on Jan 11, 2022
achow101 force-pushed
on Jan 11, 2022
DrahtBot removed the label
Needs rebase
on Jan 11, 2022
Sjors
commented at 10:21 am on January 12, 2022:
member
Being able to load and reference a wallet by ID is useful for external tools. E.g. Specter desktop identifies wallets by their path, which would break if a user renames or even moves it.
I would suggest adding a commit to this PR that exposes the wallet ID for getwalletinfo and allows using it with loadwallet and unloadwallet.
promag
commented at 10:23 am on January 12, 2022:
member
I would suggest adding a commit to this PR that exposed the wallet ID for getwalletinfo and allows using it with loadwallet and unloadwallet.
@Sjors how could loadwallet load with the id?
Sjors
commented at 10:25 am on January 12, 2022:
member
@promag it would look through all wallets in the path (like the GUI does when populating menu items). Though it depends on how much overhead there is in reading the ID out of the wallet files; that might be a bit much (we could document that if you have millions of wallet files for some reason, you should just load by file name).
promag
commented at 10:28 am on January 12, 2022:
member
@SjorsIIRC listing wallets doesn’t open the files. Actually wallet files are opened to read some magic bytes and infer the wallet file format.
But scanning all wallet files in the RPC seems fine to me.
ryanofsky
commented at 3:45 pm on January 12, 2022:
contributor
Sjors could you clarify what the broken case is: What steps does the user perform, and what is the unexpected behavior in this case and the expected behavior? It seems like it would be confusing if external software interacting with the bitcoin wallet tried to identify wallets differently than bitcoin wallet itself does. More generally, I think it would be confusing for any software to track wallets by IDs outside of user control, confuse backup wallets with the latest wallet, assume falsely that two wallets can’t have same id, and be broken in various repeated ways from this assumption.
I do think it would be great to add a descriptive text field to wallet files (call it memo or comment or description and set it initially to the wallet name), which could help users identify wallets if they are copying around and renaming them.
I also think it would be great to list active descriptors in getwalletinfo (or just the active descriptor for the preferred output type, or the active master key for legacy wallets), so external software that knows about a particular descriptor or key can identify wallets it is able to work with.
Sjors
commented at 4:05 pm on January 12, 2022:
member
What steps does the user perform, and what is the unexpected behavior in this case and the expected behavior?
User configures a multisig wallet in Specter, which in turn calls createwallet and remembers the wallet by its path (e.g. specter/my_multisig)
User starts Bitcoin Core and Specter at some later date (which calls loadwallet):
a) the wallet file wasn’t moved or renamed: all works fine
b) the wallet file was moved or renamed: Specter fails to load wallet, user has manually edit a config file to point Specter to the right wallet name (e.g. I ran into this because I didn’t like its lower case naming convention)
The above is not end of the world stuff.
confuse backup wallets with the latest wallet, assume falsely that two wallets can’t have same id
When calling loadwallet with an ID, we could either refuse or pick the most recent one if two are found.
external software that knows about a particular descriptor or key can identify wallets it is able to work with
I’m not opposed to this, but I don’t know if external software should be tracking such detailed information. In case of Specter it does though, but all that redundancy is also a potential source of errors.
ryanofsky
commented at 4:31 pm on January 12, 2022:
contributor
What steps does the user perform, and what is the unexpected behavior in this case and the expected behavior?
1. User configures a multisig wallet in Specter, which in turn calls `createwallet` and remembers the wallet by its path (e.g. `specter/my_multisig`)
2. User starts Bitcoin Core and Specter at some later date (which calls `loadwallet`):
a) the wallet file wasn't moved or renamed: all works fine
b) the wallet file was moved or renamed: Specter fails to load wallet, user has manually edit a config file to point Specter to the right wallet name (e.g. I ran into this because I didn't like its lower case naming convention)
The above is not end of the world stuff.
Agreed nothing is end of the world stuff. But I think the proposed change to fix an understandable and already fixable renaming issue would create worse problems. Both our software and the external software would have to have new logic to decide how to respond if a wallet file is copied and two wallets have same id. If a user wants to reimport the same master key in a new wallet file (maybe they accidentally deleted the original, or thought the file had too much old transaction data, or a too many keys in the keypool), there is no way to get the new wallet file to work with external software because it tracking some uncontrollable UUID instead of the just opening the wallet file with the same name.
confuse backup wallets with the latest wallet, assume falsely that two wallets can’t have same id
When calling loadwallet with an ID, we could either refuse or pick the most recent one if two are found.
Right, I’m saying adding this logic makes software more fragile, more complicated, and results in a worse UX.
external software that knows about a particular descriptor or key can identify wallets it is able to work with
I’m not opposed to this, but I don’t know if external software should be tracking such detailed information. In case of Specter it does though, but all that redundancy is also a potential source of errors.
Yes, I’m sure there are situations where that would be the wrong thing to do, and I’m not familiar with the Specter. The case where it could be the right thing to do is if you have a key on a hardware wallet, and the software is managing that key, and it wants to find wallets on a bitcoind node associated with that key. But if the software is literally managing a particular bitcoin core wallet, then it should probably refer to that wallet the same way bitcoin core does.
Sjors
commented at 6:30 pm on January 12, 2022:
member
I agree there’s something to be said for identifying wallets based on the descriptors (descriptor id) rather than the wallet id.
ryanofsky
commented at 6:34 pm on January 12, 2022:
contributor
I agree there’s something to be said for identifying wallets based on the descriptors (descriptor id) rather than the wallet id.
From the description it seems like #23417 could be helpful for this, too, if it brings back the concept of an active master seed / key id.
Sjors
commented at 6:38 pm on January 12, 2022:
member
Not in the case of multisig wallets. The master seed at best refers to 1 of N master keys that make up a wallet. But a single master key may be used in multiple single / multi signature setups. So it doesn’t make for a good unique identifier.
DrahtBot added the label
Needs rebase
on Mar 24, 2022
achow101 force-pushed
on Mar 24, 2022
DrahtBot removed the label
Needs rebase
on Mar 24, 2022
DrahtBot added the label
Needs rebase
on Sep 1, 2022
achow101 force-pushed
on Sep 1, 2022
achow101 force-pushed
on Sep 1, 2022
bitcoin deleted a comment
on Sep 1, 2022
DrahtBot removed the label
Needs rebase
on Sep 1, 2022
wallet: Properly support a wallet id
Adds a unique id for each wallet that is saved in a new "walletid"
record. For compatibility, wallets using BDB will use the BDB generated
id. All other wallets will have a randomly generated id if an id does
not already exist.
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: 2025-01-22 00:12 UTC
This site is hosted by @0xB10C More mirrored repositories can be found on mirror.b10c.me