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
|