Add a lock to the wallet directory #11904

pull meshcollider wants to merge 5 commits into bitcoin:master from meshcollider:201712_walletdir_lock changing 6 files +64 −44
  1. meshcollider commented at 11:25 pm on December 14, 2017: contributor

    Fixes #11888, needs a 0.16 milestone

    Also adds a test that the lock works.

    #11687 will probably rework this to a per-wallet lock instead of just the walletdir, but this fixes the current issue

  2. jtimon commented at 0:20 am on December 15, 2017: contributor
    Concept ACK
  3. fanquake added the label Wallet on Dec 15, 2017
  4. fanquake added this to the milestone 0.16.0 on Dec 15, 2017
  5. ryanofsky commented at 2:43 am on December 15, 2017: member
    Suggest moving LockWalletDirectory to wallet/db.cpp and calling it from CDBEnv::Open instead of VerifyWallets. That way this will be compatible with #11687. Also would suggest renaming LockWalletDirectory to LockEnvDirectory to be a little more consistent with terminology used in that file (even though the file isn’t totally consistent right now).
  6. meshcollider commented at 7:14 pm on December 16, 2017: contributor
    @ryanofsky something like this? I had to probe the lock before calling open because of the weird database log backup thing when open fails, perhaps there is a cleaner way to do it e.g. returning different errors from open rather than just true/false. But I think this is ok
  7. in src/wallet/db.cpp:72 in 94be65ea6d outdated
    67+            return false;
    68+        }
    69+        if (probe_only) {
    70+            lock.unlock();
    71+        }
    72+    } catch(const boost::interprocess::interprocess_exception& e) {
    


    promag commented at 0:45 am on December 17, 2017:
    Nit, missing space after catch.
  8. in test/functional/multiwallet.py:65 in 94be65ea6d outdated
    57@@ -58,7 +58,12 @@ def run_test(self):
    58         w5 = self.nodes[0].get_wallet_rpc("w5")
    59         w5_info = w5.getwalletinfo()
    60         assert_equal(w5_info['immature_balance'], 50)
    61+        self.stop_node(0)
    62 
    63+        competing_wallet_dir = os.path.join(self.options.tmpdir, 'competing_walletdir')
    64+        os.mkdir(competing_wallet_dir)
    65+        self.start_node(0, ['-walletdir='+competing_wallet_dir])
    


    promag commented at 0:47 am on December 17, 2017:
    Use restart_node, and remove stop_node above?
  9. promag commented at 0:51 am on December 17, 2017: member

    Concept ACK.

    Is it me or there can be a race between probing and the actual locking?

  10. meshcollider commented at 7:42 am on December 17, 2017: contributor
    Removed the probing with a bit of a rework
  11. in src/wallet/db.h:74 in 8020abc808 outdated
    67@@ -68,7 +68,10 @@ class CDBEnv
    68     typedef std::pair<std::vector<unsigned char>, std::vector<unsigned char> > KeyValPair;
    69     bool Salvage(const std::string& strFile, bool fAggressive, std::vector<KeyValPair>& vResult);
    70 
    71-    bool Open(const fs::path& path);
    72+    enum OpenResult { OPEN_OK,
    73+                        ENV_LOCKED,
    74+                        OPEN_FAIL };
    75+    OpenResult Open(const fs::path& path);
    


    promag commented at 9:48 am on December 17, 2017:
    How about bool Open(const fs::path& path, bool* is_locked = 0)?

    meshcollider commented at 10:14 am on December 17, 2017:
    Could do, but IMO doesn’t simplify the code much, if we’re returning something why not use return :)

    promag commented at 10:19 am on December 17, 2017:

    Because you only care of is_locked when Open fails. For me ENV_LOCKED is the reason Open failed.

    Also, you can avoid those != OPEN_OK conditions.

    But sure, consider my suggestion a nit.


    meshcollider commented at 10:23 am on December 17, 2017:
    Ah that’s true, ok :+1:

    ryanofsky commented at 1:35 pm on December 18, 2017:

    https://github.com/bitcoin/bitcoin/pull/11904/files#r157363498

    Seems fine to use either an enum or a bool, whichever you prefer. The real problem is that VerifyEnvironment is the wrong place for the rename/retry code to live. It creates a bizarre API because you can’t reliably open an environment without making two different calls and pointlessly closing and reopening. It’s also a bad division of code because the verify function has to make assumptions about what the open function is doing internally.

    A better way to organize the code would be to have a single bool CDBEnv::Open(fs::path, bool retry=false) function, where setting the retry parameter to true renames the log directory and reopens if the initial open fails. Verifying would then just call Open(path, true) followed by a Close. Anyway this would be nice to clean up, but no need to do it here as part of fixing the lock issue.

  12. ryanofsky commented at 1:48 pm on December 18, 2017: member
    utACK 8020abc808cbf8e0b8db24f92eafbf3262ea9163
  13. in src/wallet/db.cpp:175 in 6ff6b2fba7 outdated
    171+            // try opening it again one more time
    172+            if (!Open(walletDir, false)) {
    173+                // if it still fails, it probably means we can't even create the database env
    174+                return false;
    175+            }
    176+        return false;
    


    promag commented at 0:04 am on December 20, 2017:
    Something wrong here, missing }?

    meshcollider commented at 0:35 am on December 20, 2017:
    Yep oops, bad code move. Fixed.
  14. in src/wallet/db.h:71 in d421c4bb7f outdated
    67@@ -68,7 +68,7 @@ class CDBEnv
    68     typedef std::pair<std::vector<unsigned char>, std::vector<unsigned char> > KeyValPair;
    69     bool Salvage(const std::string& strFile, bool fAggressive, std::vector<KeyValPair>& vResult);
    70 
    71-    bool Open(const fs::path& path);
    72+    bool Open(const fs::path& path, bool retry = 0);
    


    promag commented at 0:38 am on December 20, 2017:
    = false?

    promag commented at 0:40 am on December 20, 2017:

    BTW, how about:

    0private:
    1    bool Open(const fs::path& path, bool retry);
    2
    3public:
    4    bool Open(const fs::path& path) { return Open(path, true); }
    

    this way the public API remains, for instance, https://github.com/bitcoin/bitcoin/pull/11904/files#diff-5cd083d2c34da5f0ac380011f7b725d2R312 goes away.


    meshcollider commented at 0:49 am on December 20, 2017:
    I’d prefer to leave it explicit I think, is there any other benefit to hiding the retry parameter?

    promag commented at 0:53 am on December 20, 2017:
    It’s an internal detail, you should not be able to call Open(path, false), it’s only there for the 2nd attempt.


    promag commented at 10:49 am on December 21, 2017:
    Ok, missed that.
  15. promag commented at 3:23 pm on December 21, 2017: member
    Tested ACK. Needs squash.
  16. in src/wallet/db.cpp:65 in f70ce89806 outdated
    60+    fs::path lock_file_path = env_path / ".lock";
    61+    FILE* file = fsbridge::fopen(lock_file_path, "a"); // empty lock file; created if it doesn't exist.
    62+    if (file) fclose(file);
    63+
    64+    try {
    65+        static boost::interprocess::file_lock lock(lock_file_path.string().c_str());
    


    ryanofsky commented at 4:16 pm on December 21, 2017:
    Didn’t notice this before, and no need for a change now, but just in case #11687 is somehow merged before this PR, static variable here should move to CDBEnv instance variable. (A single static variable can’t guard different database directories if different database directories may be opened during the lifetime of the process.)
  17. in src/wallet/db.cpp:313 in f70ce89806 outdated
    326-            // if it still fails, it probably means we can't even create the database env
    327-            errorStr = strprintf(_("Error initializing wallet database environment %s!"), walletDir);
    328-            return false;
    329-        }
    330+    if (!bitdb.Open(walletDir, true)) {
    331+        errorStr = strprintf(_("Cannot obtain a lock on wallet directory %s. Another instance of bitcoin may be using it."), walletDir);
    


    ryanofsky commented at 4:22 pm on December 21, 2017:
    Might want to keep the previous generic error here (“Error initializing wallet database environment”) since this can happen in other conditions than failing to obtain the lock.
  18. ryanofsky commented at 4:25 pm on December 21, 2017: member
    utACK f70ce89806a7f0ce1755e7d074f6e07753a85694. Only change since last review is CDBEnv::Open retry refactor.
  19. meshcollider commented at 11:46 pm on December 23, 2017: contributor

    Squashed and added a commit to modify the error message as suggested.

    Note travis failure on second commit is just because the last three tests were cancelled due to third commit being pushed before they started

  20. in src/wallet/db.cpp:60 in edf5a3fd56 outdated
    52@@ -52,6 +53,24 @@ void CheckUniqueFileid(const CDBEnv& env, const std::string& filename, Db& db)
    53         }
    54     }
    55 }
    56+
    57+bool LockEnvDirectory(const fs::path& env_path)
    58+{
    59+    // Make sure only a single Bitcoin process is using the wallet directory.
    60+    fs::path lock_file_path = env_path / ".lock";
    


    sipa commented at 2:51 pm on December 25, 2017:
    What if the walletdir is the same as the datadir? Won’t there be a problem if both use the same lockfile? I would consider renaming this to ".walletlock" perhaps (which is also clearer to people who see the file).

    sipa commented at 2:58 pm on December 25, 2017:

    Reading more, I think this is indeed advisable:

    From http://www.boost.org/doc/libs/1_55_0/doc/html/interprocess/synchronization_mechanisms.html#interprocess.synchronization_mechanisms.file_lock:

    • It’s unspecified if a file_lock synchronizes two threads from the same process.
    • It’s unspecified if a process can use two file_lock objects pointing to the same file.

    meshcollider commented at 10:43 pm on December 25, 2017:
    Good point, fixed!
  21. in src/wallet/db.cpp:57 in edf5a3fd56 outdated
    52@@ -52,6 +53,24 @@ void CheckUniqueFileid(const CDBEnv& env, const std::string& filename, Db& db)
    53         }
    54     }
    55 }
    56+
    57+bool LockEnvDirectory(const fs::path& env_path)
    


    sipa commented at 2:53 pm on December 25, 2017:
    A lot in this function seems to be a duplication from LockDataDirectory in init.cpp. What do you think about abstracting them out to a utility function in util.cpp instead?
  22. sipa commented at 7:06 am on December 26, 2017: member
    utACK 005d4cdef7a7a9f98cf202db7542a7fd01e048b8
  23. meshcollider commented at 7:53 am on December 26, 2017: contributor
    Thanks for the feedback @sipa
  24. in src/util.cpp:386 in 4f600cb0ba outdated
    381+    fs::path pathLockFile = directory / lockfile_name;
    382+    FILE* file = fsbridge::fopen(pathLockFile, "a"); // empty lock file; created if it doesn't exist.
    383+    if (file) fclose(file);
    384+
    385+    try {
    386+        static boost::interprocess::file_lock lock(pathLockFile.string().c_str());
    


    ryanofsky commented at 0:32 am on December 27, 2017:

    From https://botbot.me/freenode/bitcoin-core-dev/msg/95065183/

    I’ve been trying to work out why the last commit in #11904 is breaking the locking function, it works fine before that commit is added, but I can’t work out why it doesn’t work with that change. with a brand new walletdir, it creates the .walletlock file, but lock.try_lock() fails and the node crashes on startup, does anyone with more boost familiarity than me know why lock.try_lock fails when I move it

    I haven’t looked closely at changes here yet, but as I was trying to say in #11904 (review), I think the static declaration on this line becomes broken as soon as more than one directory is locked, because this will only create one single file_lock instance no matter how many different directories need to be locked. I think what you really need here is a map or list of locks, not one single lock. Perhaps something along the lines of (untested):

    0static std::map<std::string, boost::interprocess::file_lock> locks;
    1boost::interprocess::file_lock& lock = locks.emplace(pathLockFile.string(), pathLockFile.string().c_str()).first->second;
    

    meshcollider commented at 1:16 am on December 27, 2017:
    Ah, of course. Fixed. Thank you!
  25. in src/util.cpp:379 in abcfc2713d outdated
    375@@ -375,6 +376,27 @@ int LogPrintStr(const std::string &str)
    376     return ret;
    377 }
    378 
    379+bool LockDirectory(const fs::path& directory, const std::string lockfile_name, bool probe_only)
    


    ryanofsky commented at 6:06 pm on January 3, 2018:
    Maybe combine directory and std::string lockfile_name arguments into a single path argument. Having 2 arguments seems more verbose and less flexible.

    meshcollider commented at 8:10 pm on January 3, 2018:
    Indeed, was just for the purpose of printing out the directory in the log message below rather than the filename too. Will modify if/when there’s another change too

    laanwj commented at 11:44 am on January 15, 2018:
    I kind of like these separated out - after all, the function is called LockDirectory and not CreateLockFile. If you merge the directory and lockfile name, I’d say you’d also have to rename the function because it’s not longer specifically about locking directories.

    laanwj commented at 11:45 am on January 15, 2018:
    Nit: const std::string &lockfile_name
  26. in src/util.cpp:395 in abcfc2713d outdated
    390+        }
    391+        if (probe_only) {
    392+            lock.unlock();
    393+        }
    394+    } catch (const boost::interprocess::interprocess_exception& e) {
    395+        return error("Error while attempting to lock directory %s: %s", directory.string(), e.what());
    


    ryanofsky commented at 6:07 pm on January 3, 2018:
    Would be good to include filename so you could see whether wallet or data lock failed.

    meshcollider commented at 8:08 pm on January 3, 2018:
    Could do, but in both cases another more specific error message is printed out when LockDirectory returns, so it’s always on the next log line anyway
  27. ryanofsky commented at 6:12 pm on January 3, 2018: member

    utACK abcfc2713d4e2696b5178c15975291bfd79a68b7. Main change since previous review was adding the more general LockDirectory util function.

    Would also recommend just squashing everything down into a single commit to avoid this being a garden-path pr.

  28. laanwj commented at 11:48 am on January 15, 2018: member
    utACK, needs rebase
  29. Add a lock to the wallet directory e60cb99c58
  30. Add a test for wallet directory locking c9ed4bd58c
  31. Generalise walletdir lock error message for correctness 64226de908
  32. Make .walletlock distinct from .lock 5260a4aca1
  33. Abstract directory locking into util.cpp 2f3bd47d44
  34. meshcollider commented at 6:06 am on January 16, 2018: contributor
    Rebased
  35. laanwj commented at 10:11 am on January 16, 2018: member

    Tested ACK 2f3bd47

    02018-01-16 10:09:49 Cannot obtain a lock on wallet directory /tmp/wallet1. Another instance of bitcoin may be using it.
    
  36. laanwj merged this on Jan 16, 2018
  37. laanwj closed this on Jan 16, 2018

  38. laanwj referenced this in commit 66e3af709d on Jan 16, 2018
  39. meshcollider deleted the branch on Jan 17, 2018
  40. PastaPastaPasta referenced this in commit 5c27c14f91 on Mar 23, 2020
  41. PastaPastaPasta referenced this in commit 3fbe3612b8 on Mar 28, 2020
  42. PastaPastaPasta referenced this in commit 9a6a558709 on Mar 29, 2020
  43. PastaPastaPasta referenced this in commit 58209be717 on Mar 31, 2020
  44. UdjinM6 referenced this in commit cec34a25d6 on Mar 31, 2020
  45. PastaPastaPasta referenced this in commit 9bedb900a9 on Apr 1, 2020
  46. ckti referenced this in commit d1570a7efb on Mar 28, 2021
  47. gades referenced this in commit 4c10e86e9b on Jun 30, 2021
  48. furszy referenced this in commit b04e1f5a90 on Jul 23, 2021
  49. DrahtBot 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-21 12:12 UTC

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