wallet: Add ListWalletDir utility function #14291

pull promag wants to merge 5 commits into bitcoin:master from promag:2018-09-list-wallet-dir changing 8 files +129 −3
  1. promag commented at 11:28 pm on September 21, 2018: member

    ListWalletDir returns all available wallets in the current wallet directory.

    Based on MeshCollider work in pull #11485.

  2. promag commented at 11:33 pm on September 21, 2018: member

    After #11485 (comment) decided to push this.

    Consider the following:

     0$ find wallets
     1wallets
     2wallets/wallet.dat
     3wallets/db.log
     4wallets/x1
     5wallets/x1/wallet.dat
     6wallets/x1/db.log
     7wallets/x1/.walletlock
     8wallets/x2
     9wallets/x2/wallet.dat
    10wallets/x2/db.log
    11wallets/x2/.walletlock
    12wallets/.walletlock
    13wallets/w
    14wallets/w/x3
    15wallets/w/x3/wallet.dat
    16wallets/w/x3/db.log
    17wallets/w/x3/.walletlock
    

    ListWalletDir() gives

    0wallet.dat
    1x1
    2x2
    3w/x3
    
  3. promag force-pushed on Sep 21, 2018
  4. promag force-pushed on Sep 21, 2018
  5. fanquake added the label Wallet on Sep 22, 2018
  6. DrahtBot commented at 3:59 am on September 22, 2018: member
    • #13100 (gui: Add dynamic wallets support by promag)

    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.

  7. in src/interfaces/node.cpp:234 in e84e729fd0 outdated
    228+        std::vector<std::string> paths;
    229+        for (auto& path : ListWalletDir()) {
    230+            paths.push_back(path.string());
    231+        }
    232+        return paths;
    233+    }
    


    ken2812221 commented at 7:18 am on September 22, 2018:
    How about moving these functions to interfaces/wallet.cpp. I think they don’t belong here.

    ryanofsky commented at 10:17 am on September 22, 2018:

    How about moving these functions to interfaces/wallet.cpp. I think they don’t belong here.

    This is a reasonable place. These methods fit in with the existing getWallets and handleLoadWallet methods. The Node interface is somewhat monolithic and we could consider breaking it up in various ways. But the Wallet interface would be the wrong place to add these methods. Wallet methods operate on individual wallets, not collections of wallets. And these methods need to be available before any Wallet object is constructed.

  8. in src/wallet/walletutil.cpp:31 in e84e729fd0 outdated
    24@@ -25,3 +25,32 @@ fs::path GetWalletDir()
    25 
    26     return path;
    27 }
    28+
    29+std::vector<fs::path> ListWalletDir()
    30+{
    31+    const unsigned char bdb_magic[4] = {0x62, 0x31, 0x05, 0x00}; // Berkeley DB Btree magic bytes
    


    ryanofsky commented at 8:05 am on September 22, 2018:
    Should cite the source of this information (https://github.com/file/file/blob/5824af38469ec1ca9ac3ffd251e7afe9dc11e227/magic/Magdir/database#L74-L75) to give credit upstream and make this code easier to understand / debug / update in the future.

    ryanofsky commented at 9:44 am on September 22, 2018:
    Since this is declared as bytes, the memcmp below will only look for little-endian btree files, even on big endian platforms. It seems like it would make more sense to look for little endian files on little-endian platforms, and big endian files on big endian platforms. Changing this to uint32_t bdb_magic = 0x00053162 and passing &bdb_magic to memcmp should accomplish this.
  9. in src/wallet/walletutil.cpp:52 in e84e729fd0 outdated
    47+        if (memcmp(file_bytes, bdb_magic, sizeof(bdb_magic)) != 0) continue;
    48+
    49+        // avoid duplicates due to the recursive iteration
    50+        if (!dbs.insert(db).second) continue;
    51+
    52+        paths.push_back(fs::relative(path, wallet_dir));
    


    ryanofsky commented at 9:51 am on September 22, 2018:

    There are a number of problems with this implementation:

    1. Because this opens files recursively, it will return btree paths not in the top directory which CWallet::Verify will reject: https://github.com/bitcoin/bitcoin/blob/920c090f63f4990bf0f3b3d1a6d3d8a8bcd14ba0/src/wallet/wallet.cpp#L3824-L3829
    2. Because is_regular_file will return true for symlinks pointing to files, this will return symlink paths which CWallet::Verify will reject
    3. Because recursive_directory_iterator order is unspecified, on systems where it does postorder instead of preorder traversal, this will return <wallet>/wallet.dat instead of <wallet> paths, which CWallet::Verify will reject.
    4. Less serious, but this returns “wallet.dat” instead of "" as the path for the top level wallet. This is inconsistent with top level wallet path returned in current listwallets and getwalletinfo rpcs in the default bitcoin configuration.
    5. This is less efficient than it needs to be. Instead of only opening files called wallet.dat in subdirectories, it will read logs and other random files which bitcoin would never open as wallets. It also reads wallet.dat files in subdirectories twice and then uses a std::set to filter out the duplicates after they are read, which is unnecessary.

    Following alternate implementation is untested, but I think would avoid these problems:

     0std::vector<fs::path> wallets;
     1for (auto it = fs::recursive_directory_iterator(wallet_dir); it != end(it); ++it) {
     2    if (it->status().type() == fs::directory_file && IsBerkeleyBtree(it->path() / "wallet.dat")) {
     3        // Found a directory which contains wallet.dat btree file, add it as a wallet.
     4        wallets.emplace_back(it->path());
     5    } else if (it.level() == 0 && it->symlink_status().type() == fs::regular_file && IsBerkeleyBtree(it->path())) {
     6        if (it->path() == "wallet.dat") {
     7            // Found top-level wallet.dat btree file, add top level directory ""
     8            // as a wallet.
     9            wallets.emplace_back();
    10        } else {
    11            // Found top-level btree file not called wallet.dat. Current bitcoin
    12            // software will never create these files but will allow them to be
    13            // opened in a shared database environment for backwards compatibility.
    14            // Add it to the list of available wallets.
    15            wallets.emplace_back(it->path());
    16        }
    17    }
    18}
    19return wallets;
    
  10. promag commented at 1:11 pm on September 22, 2018: member
    Thanks @ryanofsky, will update accordingly.
  11. in src/wallet/walletutil.h:17 in e84e729fd0 outdated
    10@@ -11,4 +11,7 @@
    11 //! Get the path of the wallet directory.
    12 fs::path GetWalletDir();
    13 
    14+//! Get wallets in wallet directory.
    15+std::vector<fs::path> ListWalletDir();
    16+
    


    practicalswift commented at 1:34 pm on September 22, 2018:
    Nit: Add some trivial test for ListWalletDir to make it technically in use.

    promag commented at 11:02 am on September 25, 2018:
    There is listwaletdir RPC, which has some tests.
  12. laanwj renamed this:
    wallet: Add ListWalletDir utility
    wallet: Add ListWalletDir utility function
    on Sep 23, 2018
  13. promag force-pushed on Sep 24, 2018
  14. promag commented at 5:41 pm on September 24, 2018: member
    @ryanofsky updated to used your suggested code with some minor details fixed. The result is still relative paths to -walletdir. @practicalswift will add test.
  15. promag force-pushed on Sep 24, 2018
  16. in src/dummywallet.cpp:39 in 20db275b25 outdated
    33@@ -34,6 +34,16 @@ void DummyWalletInit::AddWalletOptions() const
    34 
    35 const WalletInitInterface& g_wallet_init_interface = DummyWalletInit();
    36 
    37+fs::path GetWalletDir()
    38+{
    39+    throw std::logic_error("Wallet function called in non-wallet build.");
    


    Sjors commented at 11:56 am on September 25, 2018:
    Nit: throwNotWalletBuild()?
  17. in src/interfaces/node.h:180 in 20db275b25 outdated
    172@@ -173,6 +173,12 @@ class Node
    173     //! Get unspent outputs associated with a transaction.
    174     virtual bool getUnspentOutput(const COutPoint& output, Coin& coin) = 0;
    175 
    176+    //! Return default wallet directory.
    177+    virtual std::string getWalletDir() = 0;
    178+
    179+    //! Return available wallets in wallet directory.
    180+    virtual std::vector<std::string> listWalletDir() = 0;
    


    Sjors commented at 12:01 pm on September 25, 2018:
    I find this function name confusing. How about listWallets()? A future version of this function could take a directory argument and the name would still make sense.

    promag commented at 1:17 pm on September 25, 2018:
    I like the pairing between getWalletDir, listWalletDir and -walletdir, but I don’t have a strong preference. Let’s see what others say.

    ryanofsky commented at 6:40 am on September 26, 2018:
    I think I agree with promag. listWalletDir describes more specifically what the function does than listWallets and it fits in with the other names cited. I don’t imagine us adding a gui feature that would call for the gui asking the node to list wallets in a directory other than -walletdir, but even if we did and this method gained an optional argument, listWalletDir would still seem like a sensible name.
  18. in src/wallet/rpcwallet.cpp:2453 in 20db275b25 outdated
    2449@@ -2450,6 +2450,32 @@ static UniValue getwalletinfo(const JSONRPCRequest& request)
    2450     return obj;
    2451 }
    2452 
    2453+static UniValue listwalletdir(const JSONRPCRequest& request)
    


    Sjors commented at 12:12 pm on September 25, 2018:

    Ditto as above, but unfortunately it’s too late to rename listwallets to listloadedwallets. Overloading listwallets doesn’t seem like a good idea.

    What about lswallets, listwalletfiles or findwallets?


    promag commented at 1:23 pm on September 25, 2018:
    Agree in not overloading listwallets. There are more alternatives in #11485 discussion thread.

    ryanofsky commented at 7:08 am on September 26, 2018:
    I don’t see a problem with listwalletdir. It describes the thing this RPC does, which is to list the contents of the -walletdir directory. (I do think it would have been clearer if that directory would have been called -walletsdir instead of -walletdir, but surprising to me as a native english speaker, other people seemed to think -walletsdir was clumsy and not grammatical in #12221).
  19. in src/interfaces/node.cpp:233 in 20db275b25 outdated
    228+    {
    229+        std::vector<std::string> paths;
    230+        for (auto& path : ListWalletDir()) {
    231+            paths.push_back(path.string());
    232+        }
    233+        return paths;
    


    Sjors commented at 12:17 pm on September 25, 2018:
    Sort alphabetically?

    promag commented at 1:15 pm on September 25, 2018:
    I think that’s something for the UI to handle. It can sort by other attribute, like last modified, transaction count, balance, etc.

    Sjors commented at 1:34 pm on September 25, 2018:
    That’s fine, as long as the RPC interface does it, that’s also a User Interface :-)

    promag commented at 1:38 pm on September 25, 2018:
    Should we sort listwallets result too?

    Sjors commented at 2:33 pm on September 25, 2018:
    That could be a nice improvement. Not necessarily in this PR, but perhaps there could be a common function to render the list of wallets in RPC, to make sure they’re consistent.

    promag commented at 11:49 am on September 26, 2018:

    Should it ignore case? Should have an option to support numeric/date prefixes?

    From the cli point of view, you can do stuff like bitcoin-cli -regtest listwalletdir | jq -r '.wallets | map(.name) | join("\n")' | sort ....


    Sjors commented at 12:39 pm on September 26, 2018:

    Don’t most alphabetical sort functions ignore case (unless it’s a tie) and put numbers before letters? I’m a fan of jq, but seems overkill for a simple list. It doesn’t seem terribly complicated in C++11 even with UTF-8 (1st answer, 3rd example):

    0std::wstring a[] = {L"Шла", L"Саша", L"по", L"шоссе", L"и", L"сосала", L"сушку"};
    1std::sort(std::begin(a), std::end(a), std::locale("en_US.utf8"));
    2std::wstring_convert<std::codecvt_utf8<wchar_t>> cvt;
    3for(auto& s: a) std::cout << cvt.to_bytes(s) << ' ';
    

    promag commented at 11:02 pm on September 26, 2018:
    If you don’t mind I prefer to follow up that discussion and possible implementation in both RPC. In this PR I’m just following listwallets ordering and for now having the functionality is enough.
  20. Sjors commented at 12:32 pm on September 25, 2018: member

    Concept ACK. Tested 20db275 on macOS (needs testing on Windows).

    My wallet.dat ended up as "" in the list.

    Relying on BDB magic bytes to figure out if something is a wallet seems potentially brittle. Can you add BDB4 and BDB5 wallet payloads to the test suite? In addition the test node could dynamically create a wallet, and then try to find it. That way at least some developer will see that test fail on their machine if they compile with a future BDB version that’s somehow different.

  21. in src/wallet/walletutil.cpp:51 in 90adb2cd61 outdated
    46+{
    47+    const fs::path wallet_dir = GetWalletDir();
    48+    std::vector<fs::path> paths;
    49+
    50+    for (auto it = fs::recursive_directory_iterator(wallet_dir); it != end(it); ++it) {
    51+        std::cout << it->path().string() << '-' << it.level() << std::endl;
    


    ryanofsky commented at 6:28 am on September 26, 2018:

    In commit “wallet: Add ListWalletDir utility” (90adb2cd6149af1869e624a2060ec8c4a543cf08)

    Probably delete this line, it looks leftover from debugging.


    promag commented at 7:35 am on September 26, 2018:
    Ops
  22. in src/wallet/rpcwallet.cpp:2461 in 963eb27e85 outdated
    2456+        throw std::runtime_error(
    2457+            "listwalletdir\n"
    2458+            "Returns a list of wallets in the wallet directory.\n"
    2459+            "\nResult:\n"
    2460+            "[                         (json array of strings)\n"
    2461+            "  \"walletname\"            (string) the wallet name\n"
    


    ryanofsky commented at 7:05 am on September 26, 2018:

    In commit “rpc: Add listwalletdir RPC” (963eb27e8598e2746966ee4056f177ed332f0279)

    I would consider avoiding a mistake we’ve made previously with other api’s (listwallets, listaccounts) where we limited extensibility by returning lists of strings instead of list of json objects. Specifically, I think it’d be better to return

    0[ {"wallet": ""}, {"wallet": "foo"}, {"wallet": "bar"} ]
    

    instead of:

    0[ "", "foo", "bar" ]
    

    Because there’s more metadata we could be returning about these wallets even without opening them, such as last modified times from the filesystem, status strings indicating whether wallets are presently loaded and locked, and path types indicating whether wallet paths refer to directories, symlinks, or legacy btree files.


    promag commented at 8:36 am on September 26, 2018:

    Absolutely, my initial response was like:

    0{
    1    "wallets": [
    2        { "name": "" }
    3    ]
    4}
    

    which permits adding, for instance, walletdir, count, offset..

  23. ryanofsky commented at 7:25 am on September 26, 2018: member
    utACK 20db275b250189cfd2200df8e5535aa92afa594a
  24. in src/wallet/walletutil.cpp:42 in 5e3a14d427 outdated
    38@@ -39,7 +39,7 @@ static bool IsBerkeleyBtree(const fs::path& path)
    39     uint32_t data = 0;
    40     file.read((char*) &data, sizeof(data)); // Read 4 bytes of file to compare against magic
    41 
    42-    return memcmp(&data, &bdb_magic, sizeof(bdb_magic)) == 0;
    


    promag commented at 11:22 am on September 26, 2018:
    @ryanofsky does this looks good to you?
  25. promag force-pushed on Sep 26, 2018
  26. promag commented at 11:06 pm on September 26, 2018: member
    Squashed and added release notes.
  27. promag commented at 8:35 pm on September 27, 2018: member

    the test node could dynamically create a wallet, and then try to find it. That way at least some developer will see that test fail on their machine if they compile with a future BDB version that’s somehow different. @Sjors does the test change looks good to you?

  28. DrahtBot commented at 5:52 am on September 28, 2018: member
    Coverage Change (pull 14291) Reference (master)
    Lines -0.0043 % 87.0361 %
    Functions -0.0186 % 84.1130 %
    Branches -0.0052 % 51.5451 %
  29. Sjors commented at 10:55 am on September 28, 2018: member

    @promag it covers the second part of my suggestion yes: the wallets are generated by the test framework (using whatever version of BDB is on the dev machine), and then listed.

    Can you add BDB4 and BDB5 wallet payloads to the test suite?

    This isn’t explicitly tested, but because some/most Travis machines use bdb4, it’s covered regardless. We’re not officially supporting BDB>4 wallets, so that’s not important.

    tACK fc40216e1b0a3e0462b4119f6e4380ac368ed828 for macOS, modulo wallet.dat being shown as "" (though that can be fixed later).

  30. promag commented at 11:22 am on September 28, 2018: member

    modulo wallet.dat being shown as "" (though that can be fixed later).

    That’s the name of the default wallet, so it doesn’t needs fix. Try:

    0bitcoind -regtest &
    1bitcoin-cli -regtest unloadwallet ""
    

    tACK fc40216 for macOS

    Thanks @Sjors 👍

  31. Sjors commented at 12:01 pm on September 28, 2018: member
    @promag that’s weird, but unrelated to this change. When you do listwallets it shows up as wallet.dat. And can you load it with loadwallet ""?
  32. promag commented at 12:10 pm on September 28, 2018: member

    @Sjors here I have:

    0bitcoind -regtest
    1bitcoin-cli -regtest  listwallets
    2[
    3  ""
    4]
    
  33. Sjors commented at 12:34 pm on September 28, 2018: member
    Try adding a second wallet to bitcoin.conf, then it shows up as "wallet.dat". But in that case you can unload with unloadwallet "wallet.dat". In case you don’t have wallets in bitcoin.conf, there’s never a need to load wallet.dat, so then it doesn’t matter that it’s called "".
  34. laanwj added this to the "Blockers" column in a project

  35. in src/wallet/walletutil.cpp:43 in c3acd2b690 outdated
    35+    std::ifstream file(path.string(), std::ios::binary);
    36+    if (!file.is_open()) return false;
    37+
    38+    file.seekg(12, std::ios::beg); // Magic bytes start at offset 12
    39+    uint32_t data = 0;
    40+    file.read((char*) &data, sizeof(data)); // Read 4 bytes of file to compare against magic
    


    etscrivner commented at 5:06 pm on October 7, 2018:
    Any concerns about the endianness of data as compared to bdb_magic here?

    ryanofsky commented at 1:10 pm on October 16, 2018:

    Any concerns about the endianness of data as compared to bdb_magic here?

    Could add a comment here, but this checks for databases in the native format. On big endian systems, the magic bytes are 00 05 31 62, and on little endian systems they are 62 31 05 00.


    laanwj commented at 7:04 am on October 18, 2018:

    agree with regard to adding comment; at least I too had no idea that BerkeleyDB files generated on big-endian systems are incompatible!

    edit: another theory on IRC is that BerkeleyDB creates the file in native endian but accepts both (does byteswaps if needed)! if this is true, you need to check for both orderings here

    here’s Oracle’s magic file for id’ing BDB files; https://docs.oracle.com/cd/E17275_01/html/programmer_reference/magic.txt tl;dr; check both ways around

  36. etscrivner commented at 10:48 pm on October 7, 2018: contributor
    ACK fc40216e1b0a3e0462b4119f6e4380ac368ed828
  37. achow101 commented at 1:02 am on October 9, 2018: member
    utACK fc40216e1b0a3e0462b4119f6e4380ac368ed828
  38. ryanofsky approved
  39. ryanofsky commented at 1:55 am on October 10, 2018: member
    utACK fc40216e1b0a3e0462b4119f6e4380ac368ed828. Changes since last review are adding release notes, replacing memcmp, and returning objects instead of strings
  40. in src/wallet/walletutil.cpp:61 in fc40216e1b outdated
    49+
    50+    for (auto it = fs::recursive_directory_iterator(wallet_dir); it != end(it); ++it) {
    51+        if (it->status().type() == fs::directory_file && IsBerkeleyBtree(it->path() / "wallet.dat")) {
    52+            // Found a directory which contains wallet.dat btree file, add it as a wallet.
    53+            paths.emplace_back(fs::relative(it->path(), wallet_dir));
    54+        } else if (it.level() == 0 && it->symlink_status().type() == fs::regular_file && IsBerkeleyBtree(it->path())) {
    


    laanwj commented at 10:38 am on October 16, 2018:

    so if i get this right: in the legacy case where walletdir == datadir, this will scan every file in the data directory recursively, and check it for BDB version magic? this includes LevelDB database files, block files, debug log files, authcookie files, lock files … can see tons of ways in which this can go wrong or this is somehow avoided?

    if not, I’d strongly suggest it should just fail in that case—if the user wants to list non-default wallets, let them create a wallets directory


    promag commented at 10:55 am on October 16, 2018:

    No, it only recursively scans directories to try to open wallet.dat inside and then check magic bytes.

    It checks magic bytes on all files in the walletdir (datadir in the legacy case) — not recursively.


    ryanofsky commented at 1:17 pm on October 16, 2018:
    It still might be good to prevent this from opening too many files if there happen to be a lot of stray files in the root directory, or a lot of directories in the tree. Maybe add && (++count) before && IsBerkeleyBtree checks, and if (count > 1000) throw std::runtime_error("Can't list wallets, too many files in walletdir") and the end of the loop as a sanity check.

    promag commented at 1:42 pm on October 16, 2018:
    @ryanofsky an alternative solution is to scan only if walletdir != datadir, WDYT?

    ryanofsky commented at 4:24 pm on October 16, 2018:

    @ryanofsky an alternative solution is to scan only if walletdir != datadir, WDYT?

    That seems like a worse solution. It doesn’t provide protection if there’s a looping directory structure or a lot of junk files in the wallet directory. It’s less user friendly if users with old datadirs have to manually move database files around in order for this one rpc method to work. It also seems like it would be more complicated to test and explain.

    I think my suggestion is better because it would just work in the all the normal cases, and provide a straightforward error message in the unusual case laanwj is concerned about.


    promag commented at 4:55 pm on October 16, 2018:

    It doesn’t provide protection if there’s a looping directory structure

    From the docs:

    By default, recursive_directory_iterator does not follow directory symlinks

    IIUC considering the above looping would’t happen.


    ryanofsky commented at 5:11 pm on October 16, 2018:

    By default, recursive_directory_iterator does not follow directory symlinks

    I know, I was thinking about more about FUSE mounts and such. Anyway, if you think it complicates code too much to deal with unexpected filesystem stuff and provide a nice error message, I don’t personally think you need to add any error handling here. I was just trying to suggest a way to deal with laanwj’s concern.

    But since you asked WDYT, I think special casing datadir=walletdir is a bad user experience and confusing and ugly and hard to explain.

  41. promag force-pushed on Oct 16, 2018
  42. in src/wallet/walletutil.cpp:36 in 47a229e920 outdated
    31+    // Berkeley DB Btree magic bytes, from:
    32+    //  https://github.com/file/file/blob/5824af38469ec1ca9ac3ffd251e7afe9dc11e227/magic/Magdir/database#L74-L75
    33+    const uint32_t bdb_magic = 0x00053162;
    34+
    35+    if (!fs::exists(path)) return false;
    36+    if (fs::file_size(path) < 1024) return false;
    


    promag commented at 5:16 pm on October 16, 2018:
    @ryanofsky @laanwj added these 2 checks to avoid opening unnecessary files.

    ryanofsky commented at 5:57 pm on October 16, 2018:

    I’m not sure what purpose these checks serve. Are they supposed to be an optimization? Is the second check supposed to guard against opening lock files? I would suggest adding a comment here or dropping these.

    The first check actually seems like the opposite of an optimization, since file_size/ifstream calls below inherently have to check that the file exists, so it seems like this just checks the same condition twice.


    promag commented at 6:19 pm on October 16, 2018:
    The first check fs::exists(path) was there to prevent exceptions from the 2nd check fs::file_size(path). Fixed and added comment.
  43. ryanofsky commented at 6:00 pm on October 16, 2018: member
    utACK 47a229e920187dd90af7bba52cc0bfe4baab735c. Only change since last review is adding more IsBerkeleyBtree checks.
  44. promag force-pushed on Oct 16, 2018
  45. ryanofsky commented at 7:10 pm on October 16, 2018: member
    utACK 743a3a9b20768519a11f54834d27fd7585a0670a. Just some cleanup around the file size check since the last review.
  46. promag force-pushed on Oct 17, 2018
  47. promag force-pushed on Oct 17, 2018
  48. promag commented at 11:24 pm on October 17, 2018: member
    Thanks, rebased to remove fixup.
  49. in src/wallet/walletutil.cpp:40 in f27a767846 outdated
    35+    // A Berkeley DB Btree file has at least 4K.
    36+    // This check also prevents opening lock files.
    37+    boost::system::error_code ec;
    38+    if (fs::file_size(path, ec) < 4096) return false;
    39+
    40+    std::ifstream file(path.string(), std::ios::binary);
    


    laanwj commented at 8:38 am on October 18, 2018:
    switch to fs::ifstream post-#13878
  50. laanwj commented at 8:46 am on October 18, 2018: member
    utACK after nits
  51. promag force-pushed on Oct 18, 2018
  52. in src/wallet/walletutil.cpp:49 in b791ef12c3 outdated
    44+
    45+    // Berkeley DB Btree magic bytes, from:
    46+    //  https://github.com/file/file/blob/5824af38469ec1ca9ac3ffd251e7afe9dc11e227/magic/Magdir/database#L74-L75
    47+    //  - big endian systems - 00 05 31 62
    48+    //  - little endian systems - 62 31 05 00
    49+    return data == 0x00053162 || data == 0x62310500;
    


    laanwj commented at 1:57 pm on October 18, 2018:
    LGTM now
  53. ryanofsky commented at 6:02 pm on October 18, 2018: member
    utACK b791ef12c3605c185432e391e1571853b07a7441. Changes since last review were adding minimum file size check, switching to fs::ifstream, and no longer omitting big-endian btree files. It seems that berkeley supports loading databases with any byte order, with native byte order just being slightly more efficient: https://docs.oracle.com/cd/E17276_01/html/programmer_reference/general_am_conf.html#am_conf_byteorder
  54. jonasschnelli removed this from the "Blockers" column in a project

  55. meshcollider commented at 10:21 pm on October 18, 2018: contributor
    re-utACK after squash
  56. wallet: Add ListWalletDir utility
    ListWalletDir returns all available wallets in the current wallet directory.
    
    Based on MeshCollider work in pull #11485.
    fc4db35bfd
  57. interfaces: Add getWalletDir and listWalletDir to Node d1b03b8e5f
  58. rpc: Add listwalletdir RPC cc3377360c
  59. qa: Add tests for listwalletdir RPC 0cb3cad166
  60. docs: Add release notes for listwalletdir RPC d56a068935
  61. promag force-pushed on Oct 18, 2018
  62. meshcollider commented at 10:38 pm on October 18, 2018: contributor
  63. laanwj merged this on Oct 18, 2018
  64. laanwj closed this on Oct 18, 2018

  65. laanwj referenced this in commit 8eb2cd1dda on Oct 18, 2018
  66. promag deleted the branch on Oct 18, 2018
  67. in src/wallet/walletutil.cpp:71 in d56a068935
    66+            } else {
    67+                // Found top-level btree file not called wallet.dat. Current bitcoin
    68+                // software will never create these files but will allow them to be
    69+                // opened in a shared database environment for backwards compatibility.
    70+                // Add it to the list of available wallets.
    71+                paths.emplace_back(fs::relative(it->path(), wallet_dir));
    


    MarcoFalke commented at 10:57 am on October 20, 2018:
    0wallet/walletutil.cpp:71:36: error: ‘relative’ is not a member of ‘fs’
    1                 paths.emplace_back(fs::relative(it->path(), wallet_dir));
    2                                    ^
    

    See: #14528 (travis: Compile once on xenial)

  68. in test/functional/wallet_multiwallet.py:73 in d56a068935
    69@@ -68,6 +70,8 @@ def wallet_file(name):
    70         wallet_names = ['w1', 'w2', 'w3', 'w', 'sub/w5', os.path.join(self.options.tmpdir, 'extern/w6'), 'w7_symlink', 'w8', '']
    71         extra_args = ['-wallet={}'.format(n) for n in wallet_names]
    72         self.start_node(0, extra_args)
    73+        assert_equal(set(map(lambda w: w['name'], self.nodes[0].listwalletdir()['wallets'])), set(['', 'w3', 'w2', 'sub/w5', 'w7', 'w7', 'w1', 'w8', 'w']))
    


    promag commented at 11:32 pm on October 20, 2018:
    This is wrong, fs::relative resolves symlinks but we allow symlinks to be specified as wallet names. I guess listwalletdir should list both w7 and w7_symlink but in runtime only one could be loaded (actually it’s the same wallet but with different names). @ryanofsky @laanwj should I fix this in #14531 or fix in a separate PR?
  69. kallewoof commented at 3:22 am on October 22, 2018: member

    On Debian Jessie PowerPC (Big endian) system (endianness probably unrelated):

     0wallet/walletutil.cpp: In function ‘std::vector<boost::filesystem::path> ListWalletDir()’:
     1wallet/walletutil.cpp:57:78: error: ‘end’ was not declared in this scope
     2     for (auto it = fs::recursive_directory_iterator(wallet_dir); it != end(it); ++it) {
     3                                                                              ^
     4wallet/walletutil.cpp:57:78: note: suggested alternative:
     5In file included from /usr/include/c++/4.9/bits/basic_string.h:42:0,
     6                 from /usr/include/c++/4.9/string:52,
     7                 from ./fs.h:9,
     8                 from ./wallet/walletutil.h:8,
     9                 from wallet/walletutil.cpp:5:
    10/usr/include/c++/4.9/initializer_list:99:5: note:   ‘std::end’
    11     end(initializer_list<_Tp> __ils) noexcept
    12     ^
    13wallet/walletutil.cpp:60:32: error: ‘relative’ is not a member of ‘fs’
    14             paths.emplace_back(fs::relative(it->path(), wallet_dir));
    15                                ^
    16wallet/walletutil.cpp:71:36: error: ‘relative’ is not a member of ‘fs’
    17                 paths.emplace_back(fs::relative(it->path(), wallet_dir));
    18                                    ^
    19Makefile:6919: recipe for target 'wallet/libbitcoin_wallet_a-walletutil.o' failed
    20make[2]: *** [wallet/libbitcoin_wallet_a-walletutil.o] Error 1
    21make[2]: Leaving directory '/root/workspace/bitcoin/src'
    22Makefile:10219: recipe for target 'all-recursive' failed
    23make[1]: *** [all-recursive] Error 1
    24make[1]: Leaving directory '/root/workspace/bitcoin/src'
    25Makefile:697: recipe for target 'all-recursive' failed
    26make: *** [all-recursive] Error 1
    
  70. deadalnix referenced this in commit 50a1fbc83f on Mar 21, 2020
  71. ftrader referenced this in commit 07762066d8 on May 19, 2020
  72. jonspock referenced this in commit 97cad4786b on Aug 14, 2020
  73. jonspock referenced this in commit 330a4229a2 on Aug 27, 2020
  74. jonspock referenced this in commit d662d8d933 on Aug 28, 2020
  75. jonspock referenced this in commit b9cbe54b4b on Sep 4, 2020
  76. jonspock referenced this in commit 306f6a3f38 on Sep 5, 2020
  77. jonspock referenced this in commit d048106696 on Sep 6, 2020
  78. jonspock referenced this in commit 8013aaa5d2 on Sep 13, 2020
  79. jonspock referenced this in commit 7e55805814 on Sep 13, 2020
  80. jonspock referenced this in commit 78cf0d3be0 on Sep 15, 2020
  81. jonspock referenced this in commit 17bccc5d77 on Sep 16, 2020
  82. jonspock referenced this in commit 1df518d0c0 on Sep 16, 2020
  83. jonspock referenced this in commit 650a3f5ca5 on Sep 17, 2020
  84. jonspock referenced this in commit feffe3011e on Sep 18, 2020
  85. jonspock referenced this in commit ff1102c091 on Sep 21, 2020
  86. jonspock referenced this in commit b7e2b5cb77 on Sep 21, 2020
  87. jonspock referenced this in commit 096faf1f58 on Sep 23, 2020
  88. jonspock referenced this in commit 39a3b610a6 on Sep 24, 2020
  89. proteanx referenced this in commit fc65ecb721 on Sep 25, 2020
  90. 5tefan referenced this in commit 31d09b36e1 on Aug 13, 2021
  91. 5tefan referenced this in commit f9498d0051 on Aug 14, 2021
  92. PastaPastaPasta referenced this in commit f370f41210 on Aug 16, 2021
  93. MarcoFalke locked this on Sep 8, 2021

github-metadata-mirror

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 03:12 UTC

This site is hosted by @0xB10C
More mirrored repositories can be found on mirror.b10c.me