LCOV - code coverage report
Current view: top level - src/wallet - dump.cpp (source / functions) Coverage Total Hit
Test: total_coverage.info Lines: 84.4 % 179 151
Test Date: 2025-01-19 05:08:01 Functions: 100.0 % 3 3
Branches: 49.3 % 304 150

             Branch data     Line data    Source code
       1                 :             : // Copyright (c) 2020-2022 The Bitcoin Core developers
       2                 :             : // Distributed under the MIT software license, see the accompanying
       3                 :             : // file COPYING or http://www.opensource.org/licenses/mit-license.php.
       4                 :             : 
       5                 :             : #include <wallet/dump.h>
       6                 :             : 
       7                 :             : #include <common/args.h>
       8                 :             : #include <util/fs.h>
       9                 :             : #include <util/translation.h>
      10                 :             : #include <wallet/wallet.h>
      11                 :             : #include <wallet/walletdb.h>
      12                 :             : 
      13                 :             : #include <algorithm>
      14                 :             : #include <fstream>
      15                 :             : #include <memory>
      16                 :             : #include <string>
      17                 :             : #include <utility>
      18                 :             : #include <vector>
      19                 :             : 
      20                 :             : namespace wallet {
      21                 :             : static const std::string DUMP_MAGIC = "BITCOIN_CORE_WALLET_DUMP";
      22                 :             : uint32_t DUMP_VERSION = 1;
      23                 :             : 
      24                 :           8 : bool DumpWallet(const ArgsManager& args, WalletDatabase& db, bilingual_str& error)
      25                 :             : {
      26                 :             :     // Get the dumpfile
      27   [ +  -  +  - ]:          16 :     std::string dump_filename = args.GetArg("-dumpfile", "");
      28         [ +  + ]:           8 :     if (dump_filename.empty()) {
      29         [ +  - ]:           1 :         error = _("No dump file provided. To use dump, -dumpfile=<filename> must be provided.");
      30                 :           1 :         return false;
      31                 :             :     }
      32                 :             : 
      33         [ +  - ]:           7 :     fs::path path = fs::PathFromString(dump_filename);
      34         [ +  - ]:          14 :     path = fs::absolute(path);
      35   [ +  -  +  + ]:           7 :     if (fs::exists(path)) {
      36   [ +  -  +  - ]:           3 :         error = strprintf(_("File %s already exists. If you are sure this is what you want, move it out of the way first."), fs::PathToString(path));
      37                 :           1 :         return false;
      38                 :             :     }
      39         [ +  - ]:           6 :     std::ofstream dump_file;
      40         [ +  - ]:           6 :     dump_file.open(path);
      41         [ -  + ]:           6 :     if (dump_file.fail()) {
      42   [ #  #  #  # ]:           0 :         error = strprintf(_("Unable to open %s for writing"), fs::PathToString(path));
      43                 :           0 :         return false;
      44                 :             :     }
      45                 :             : 
      46         [ +  - ]:           6 :     HashWriter hasher{};
      47                 :             : 
      48         [ +  - ]:           6 :     std::unique_ptr<DatabaseBatch> batch = db.MakeBatch();
      49                 :             : 
      50                 :           6 :     bool ret = true;
      51         [ +  - ]:           6 :     std::unique_ptr<DatabaseCursor> cursor = batch->GetNewCursor();
      52         [ -  + ]:           6 :     if (!cursor) {
      53         [ #  # ]:           0 :         error = _("Error: Couldn't create cursor into database");
      54                 :           0 :         ret = false;
      55                 :             :     }
      56                 :             : 
      57                 :             :     // Write out a magic string with version
      58         [ +  - ]:           6 :     std::string line = strprintf("%s,%u\n", DUMP_MAGIC, DUMP_VERSION);
      59         [ +  - ]:           6 :     dump_file.write(line.data(), line.size());
      60         [ +  - ]:           6 :     hasher << Span{line};
      61                 :             : 
      62                 :             :     // Write out the file format
      63         [ +  - ]:           6 :     std::string format = db.Format();
      64                 :             :     // BDB files that are opened using BerkeleyRODatabase have it's format as "bdb_ro"
      65                 :             :     // We want to override that format back to "bdb"
      66         [ -  + ]:           6 :     if (format == "bdb_ro") {
      67         [ #  # ]:           0 :         format = "bdb";
      68                 :             :     }
      69         [ +  - ]:           6 :     line = strprintf("%s,%s\n", "format", format);
      70         [ +  - ]:           6 :     dump_file.write(line.data(), line.size());
      71         [ +  - ]:           6 :     hasher << Span{line};
      72                 :             : 
      73         [ +  - ]:           6 :     if (ret) {
      74                 :             : 
      75                 :             :         // Read the records
      76                 :        2392 :         while (true) {
      77                 :        1199 :             DataStream ss_key{};
      78                 :        1199 :             DataStream ss_value{};
      79         [ +  - ]:        1199 :             DatabaseCursor::Status status = cursor->Next(ss_key, ss_value);
      80         [ +  + ]:        1199 :             if (status == DatabaseCursor::Status::DONE) {
      81                 :             :                 ret = true;
      82                 :             :                 break;
      83         [ -  + ]:        1193 :             } else if (status == DatabaseCursor::Status::FAIL) {
      84         [ #  # ]:           0 :                 error = _("Error reading next record from wallet database");
      85                 :           0 :                 ret = false;
      86                 :           0 :                 break;
      87                 :             :             }
      88         [ +  - ]:        1193 :             std::string key_str = HexStr(ss_key);
      89         [ +  - ]:        1193 :             std::string value_str = HexStr(ss_value);
      90         [ +  - ]:        1193 :             line = strprintf("%s,%s\n", key_str, value_str);
      91         [ +  - ]:        1193 :             dump_file.write(line.data(), line.size());
      92         [ +  - ]:        2386 :             hasher << Span{line};
      93                 :        1199 :         }
      94                 :             :     }
      95                 :             : 
      96         [ +  - ]:           6 :     cursor.reset();
      97         [ +  - ]:           6 :     batch.reset();
      98                 :             : 
      99         [ +  - ]:           6 :     if (ret) {
     100                 :             :         // Write the hash
     101   [ +  -  +  -  :           6 :         tfm::format(dump_file, "checksum,%s\n", HexStr(hasher.GetHash()));
                   +  - ]
     102         [ +  - ]:           6 :         dump_file.close();
     103                 :             :     } else {
     104                 :             :         // Remove the dumpfile on failure
     105         [ #  # ]:           0 :         dump_file.close();
     106         [ #  # ]:           0 :         fs::remove(path);
     107                 :             :     }
     108                 :             : 
     109                 :           6 :     return ret;
     110                 :          21 : }
     111                 :             : 
     112                 :             : // The standard wallet deleter function blocks on the validation interface
     113                 :             : // queue, which doesn't exist for the bitcoin-wallet. Define our own
     114                 :             : // deleter here.
     115                 :           7 : static void WalletToolReleaseWallet(CWallet* wallet)
     116                 :             : {
     117                 :           7 :     wallet->WalletLogPrintf("Releasing wallet\n");
     118                 :           7 :     wallet->Close();
     119         [ +  - ]:           7 :     delete wallet;
     120                 :           7 : }
     121                 :             : 
     122                 :          14 : bool CreateFromDump(const ArgsManager& args, const std::string& name, const fs::path& wallet_path, bilingual_str& error, std::vector<bilingual_str>& warnings)
     123                 :             : {
     124                 :             :     // Get the dumpfile
     125   [ +  -  +  - ]:          28 :     std::string dump_filename = args.GetArg("-dumpfile", "");
     126         [ +  + ]:          14 :     if (dump_filename.empty()) {
     127         [ +  - ]:           1 :         error = _("No dump file provided. To use createfromdump, -dumpfile=<filename> must be provided.");
     128                 :           1 :         return false;
     129                 :             :     }
     130                 :             : 
     131         [ +  - ]:          13 :     fs::path dump_path = fs::PathFromString(dump_filename);
     132         [ +  - ]:          26 :     dump_path = fs::absolute(dump_path);
     133   [ +  -  +  + ]:          13 :     if (!fs::exists(dump_path)) {
     134   [ +  -  +  - ]:           3 :         error = strprintf(_("Dump file %s does not exist."), fs::PathToString(dump_path));
     135                 :           1 :         return false;
     136                 :             :     }
     137         [ +  - ]:          12 :     std::ifstream dump_file{dump_path};
     138                 :             : 
     139                 :             :     // Compute the checksum
     140         [ +  - ]:          12 :     HashWriter hasher{};
     141                 :          12 :     uint256 checksum;
     142                 :             : 
     143                 :             :     // Check the magic and version
     144         [ +  - ]:          12 :     std::string magic_key;
     145         [ +  - ]:          12 :     std::getline(dump_file, magic_key, ',');
     146         [ +  - ]:          12 :     std::string version_value;
     147         [ +  - ]:          12 :     std::getline(dump_file, version_value, '\n');
     148         [ +  + ]:          12 :     if (magic_key != DUMP_MAGIC) {
     149         [ +  - ]:           1 :         error = strprintf(_("Error: Dumpfile identifier record is incorrect. Got \"%s\", expected \"%s\"."), magic_key, DUMP_MAGIC);
     150         [ +  - ]:           1 :         dump_file.close();
     151                 :             :         return false;
     152                 :             :     }
     153                 :             :     // Check the version number (value of first record)
     154                 :          11 :     uint32_t ver;
     155   [ +  -  -  + ]:          11 :     if (!ParseUInt32(version_value, &ver)) {
     156         [ #  # ]:           0 :         error =strprintf(_("Error: Unable to parse version %u as a uint32_t"), version_value);
     157         [ #  # ]:           0 :         dump_file.close();
     158                 :             :         return false;
     159                 :             :     }
     160         [ +  + ]:          11 :     if (ver != DUMP_VERSION) {
     161         [ +  - ]:           2 :         error = strprintf(_("Error: Dumpfile version is not supported. This version of bitcoin-wallet only supports version 1 dumpfiles. Got dumpfile with version %s"), version_value);
     162         [ +  - ]:           2 :         dump_file.close();
     163                 :             :         return false;
     164                 :             :     }
     165         [ +  - ]:           9 :     std::string magic_hasher_line = strprintf("%s,%s\n", magic_key, version_value);
     166         [ +  - ]:           9 :     hasher << Span{magic_hasher_line};
     167                 :             : 
     168                 :             :     // Get the stored file format
     169         [ +  - ]:           9 :     std::string format_key;
     170         [ +  - ]:           9 :     std::getline(dump_file, format_key, ',');
     171         [ +  - ]:           9 :     std::string format_value;
     172         [ +  - ]:           9 :     std::getline(dump_file, format_value, '\n');
     173         [ -  + ]:           9 :     if (format_key != "format") {
     174         [ #  # ]:           0 :         error = strprintf(_("Error: Dumpfile format record is incorrect. Got \"%s\", expected \"format\"."), format_key);
     175         [ #  # ]:           0 :         dump_file.close();
     176                 :             :         return false;
     177                 :             :     }
     178                 :             :     // Get the data file format with format_value as the default
     179   [ +  -  +  - ]:           9 :     std::string file_format = args.GetArg("-format", format_value);
     180         [ -  + ]:           9 :     if (file_format.empty()) {
     181         [ #  # ]:           0 :         error = _("No wallet file format provided. To use createfromdump, -format=<format> must be provided.");
     182                 :           0 :         return false;
     183                 :             :     }
     184                 :           9 :     DatabaseFormat data_format;
     185         [ +  - ]:           9 :     if (file_format == "bdb") {
     186                 :             :         data_format = DatabaseFormat::BERKELEY;
     187         [ +  + ]:           9 :     } else if (file_format == "sqlite") {
     188                 :             :         data_format = DatabaseFormat::SQLITE;
     189         [ +  - ]:           1 :     } else if (file_format == "bdb_swap") {
     190                 :             :         data_format = DatabaseFormat::BERKELEY_SWAP;
     191                 :             :     } else {
     192         [ +  - ]:           1 :         error = strprintf(_("Unknown wallet file format \"%s\" provided. Please provide one of \"bdb\" or \"sqlite\"."), file_format);
     193                 :           1 :         return false;
     194                 :             :     }
     195         [ -  + ]:           8 :     if (file_format != format_value) {
     196         [ #  # ]:           0 :         warnings.push_back(strprintf(_("Warning: Dumpfile wallet format \"%s\" does not match command line specified format \"%s\"."), format_value, file_format));
     197                 :             :     }
     198         [ +  - ]:           8 :     std::string format_hasher_line = strprintf("%s,%s\n", format_key, format_value);
     199         [ +  - ]:           8 :     hasher << Span{format_hasher_line};
     200                 :             : 
     201         [ +  - ]:           8 :     DatabaseOptions options;
     202                 :           8 :     DatabaseStatus status;
     203         [ +  - ]:           8 :     ReadDatabaseArgs(args, options);
     204                 :           8 :     options.require_create = true;
     205         [ +  - ]:           8 :     options.require_format = data_format;
     206         [ +  - ]:           8 :     std::unique_ptr<WalletDatabase> database = MakeDatabase(wallet_path, options, status, error);
     207         [ +  + ]:           8 :     if (!database) return false;
     208                 :             : 
     209                 :             :     // dummy chain interface
     210                 :           7 :     bool ret = true;
     211   [ +  -  +  -  :          14 :     std::shared_ptr<CWallet> wallet(new CWallet(/*chain=*/nullptr, name, std::move(database)), WalletToolReleaseWallet);
             +  -  -  - ]
     212                 :           7 :     {
     213         [ +  - ]:           7 :         LOCK(wallet->cs_wallet);
     214         [ +  - ]:           7 :         DBErrors load_wallet_ret = wallet->LoadWallet();
     215         [ -  + ]:           7 :         if (load_wallet_ret != DBErrors::LOAD_OK) {
     216         [ #  # ]:           0 :             error = strprintf(_("Error creating %s"), name);
     217         [ #  # ]:           0 :             return false;
     218                 :             :         }
     219                 :             : 
     220                 :             :         // Get the database handle
     221                 :           7 :         WalletDatabase& db = wallet->GetDatabase();
     222         [ +  - ]:           7 :         std::unique_ptr<DatabaseBatch> batch = db.MakeBatch();
     223         [ +  - ]:           7 :         batch->TxnBegin();
     224                 :             : 
     225                 :             :         // Read the records from the dump file and write them to the database
     226         [ +  + ]:         284 :         while (dump_file.good()) {
     227         [ +  - ]:         283 :             std::string key;
     228         [ +  - ]:         283 :             std::getline(dump_file, key, ',');
     229         [ +  - ]:         283 :             std::string value;
     230         [ +  - ]:         283 :             std::getline(dump_file, value, '\n');
     231                 :             : 
     232         [ +  + ]:         283 :             if (key == "checksum") {
     233         [ +  - ]:           6 :                 std::vector<unsigned char> parsed_checksum = ParseHex(value);
     234         [ +  + ]:           6 :                 if (parsed_checksum.size() != checksum.size()) {
     235   [ +  -  +  - ]:           4 :                     error = Untranslated("Error: Checksum is not the correct size");
     236                 :           2 :                     ret = false;
     237                 :           2 :                     break;
     238                 :             :                 }
     239                 :           4 :                 std::copy(parsed_checksum.begin(), parsed_checksum.end(), checksum.begin());
     240                 :             :                 break;
     241                 :           6 :             }
     242                 :             : 
     243         [ +  - ]:         277 :             std::string line = strprintf("%s,%s\n", key, value);
     244         [ +  - ]:         277 :             hasher << Span{line};
     245                 :             : 
     246   [ +  +  -  + ]:         277 :             if (key.empty() || value.empty()) {
     247                 :           1 :                 continue;
     248                 :             :             }
     249                 :             : 
     250   [ +  -  -  + ]:         276 :             if (!IsHex(key)) {
     251         [ #  # ]:           0 :                 error = strprintf(_("Error: Got key that was not hex: %s"), key);
     252                 :           0 :                 ret = false;
     253                 :           0 :                 break;
     254                 :             :             }
     255   [ +  -  -  + ]:         276 :             if (!IsHex(value)) {
     256         [ #  # ]:           0 :                 error = strprintf(_("Error: Got value that was not hex: %s"), value);
     257                 :           0 :                 ret = false;
     258                 :           0 :                 break;
     259                 :             :             }
     260                 :             : 
     261         [ +  - ]:         276 :             std::vector<unsigned char> k = ParseHex(key);
     262         [ +  - ]:         276 :             std::vector<unsigned char> v = ParseHex(value);
     263   [ +  -  -  + ]:         276 :             if (!batch->Write(Span{k}, Span{v})) {
     264         [ #  # ]:           0 :                 error = strprintf(_("Error: Unable to write record to new wallet"));
     265                 :           0 :                 ret = false;
     266                 :           0 :                 break;
     267                 :             :             }
     268                 :         283 :         }
     269                 :             : 
     270         [ +  + ]:           7 :         if (ret) {
     271         [ +  - ]:           5 :             uint256 comp_checksum = hasher.GetHash();
     272         [ +  + ]:           5 :             if (checksum.IsNull()) {
     273         [ +  - ]:           1 :                 error = _("Error: Missing checksum");
     274                 :           1 :                 ret = false;
     275         [ +  + ]:           4 :             } else if (checksum != comp_checksum) {
     276   [ +  -  +  -  :           2 :                 error = strprintf(_("Error: Dumpfile checksum does not match. Computed %s, expected %s"), HexStr(comp_checksum), HexStr(checksum));
                   +  - ]
     277                 :           1 :                 ret = false;
     278                 :             :             }
     279                 :             :         }
     280                 :             : 
     281                 :           2 :         if (ret) {
     282         [ +  - ]:           3 :             batch->TxnCommit();
     283                 :             :         } else {
     284         [ +  - ]:           4 :             batch->TxnAbort();
     285                 :             :         }
     286                 :             : 
     287         [ +  - ]:           7 :         batch.reset();
     288                 :             : 
     289         [ +  - ]:           7 :         dump_file.close();
     290         [ +  - ]:           7 :     }
     291                 :           7 :     wallet.reset(); // The pointer deleter will close the wallet for us.
     292                 :             : 
     293                 :             :     // Remove the wallet dir if we have a failure
     294         [ +  + ]:           7 :     if (!ret) {
     295         [ +  - ]:           4 :         fs::remove_all(wallet_path);
     296                 :             :     }
     297                 :             : 
     298                 :             :     return ret;
     299                 :          65 : }
     300                 :             : } // namespace wallet
        

Generated by: LCOV version 2.0-1