Bitcoin Core  31.0.0
P2P Digital Currency
dbwrapper_tests.cpp
Go to the documentation of this file.
1 // Copyright (c) 2012-present 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 <dbwrapper.h>
6 #include <test/util/common.h>
7 #include <test/util/random.h>
9 #include <uint256.h>
10 #include <util/string.h>
11 
12 #include <memory>
13 #include <ranges>
14 
15 #include <boost/test/unit_test.hpp>
16 
17 using util::ToString;
18 
20 
22 {
23  // Perform tests both obfuscated and non-obfuscated.
24  for (const bool obfuscate : {false, true}) {
25  constexpr size_t CACHE_SIZE{1_MiB};
26  const fs::path path{m_args.GetDataDirBase() / "dbwrapper"};
27 
28  Obfuscation obfuscation;
29  std::vector<std::pair<uint8_t, uint256>> key_values{};
30 
31  // Write values
32  {
33  CDBWrapper dbw{{.path = path, .cache_bytes = CACHE_SIZE, .wipe_data = true, .obfuscate = obfuscate}};
34  BOOST_CHECK_EQUAL(obfuscate, !dbw.IsEmpty());
35 
36  // Ensure that we're doing real obfuscation when obfuscate=true
37  obfuscation = dbwrapper_private::GetObfuscation(dbw);
39 
40  for (uint8_t k{0}; k < 10; ++k) {
41  uint8_t key{k};
42  uint256 value{m_rng.rand256()};
43  dbw.Write(key, value);
44  key_values.emplace_back(key, value);
45  }
46  }
47 
48  // Verify that the obfuscation key is never obfuscated
49  {
50  CDBWrapper dbw{{.path = path, .cache_bytes = CACHE_SIZE, .obfuscate = false}};
52  }
53 
54  // Read back the values
55  {
56  CDBWrapper dbw{{.path = path, .cache_bytes = CACHE_SIZE, .obfuscate = obfuscate}};
57 
58  // Ensure obfuscation is read back correctly
61 
62  // Verify all written values
63  for (const auto& [key, expected_value] : key_values) {
64  uint256 read_value{};
65  BOOST_CHECK(dbw.Read(key, read_value));
66  BOOST_CHECK_EQUAL(read_value, expected_value);
67  }
68  }
69  }
70 }
71 
72 BOOST_AUTO_TEST_CASE(dbwrapper_basic_data)
73 {
74  // Perform tests both obfuscated and non-obfuscated.
75  for (bool obfuscate : {false, true}) {
76  fs::path ph = m_args.GetDataDirBase() / (obfuscate ? "dbwrapper_1_obfuscate_true" : "dbwrapper_1_obfuscate_false");
77  CDBWrapper dbw({.path = ph, .cache_bytes = 1 << 20, .memory_only = false, .wipe_data = true, .obfuscate = obfuscate});
78 
79  uint256 res;
80  uint32_t res_uint_32;
81  bool res_bool;
82 
83  // Ensure that we're doing real obfuscation when obfuscate=true
85 
86  //Simulate block raw data - "b + block hash"
87  std::string key_block = "b" + m_rng.rand256().ToString();
88 
89  uint256 in_block = m_rng.rand256();
90  dbw.Write(key_block, in_block);
91  BOOST_CHECK(dbw.Read(key_block, res));
92  BOOST_CHECK_EQUAL(res.ToString(), in_block.ToString());
93 
94  //Simulate file raw data - "f + file_number"
95  std::string key_file = strprintf("f%04x", m_rng.rand32());
96 
97  uint256 in_file_info = m_rng.rand256();
98  dbw.Write(key_file, in_file_info);
99  BOOST_CHECK(dbw.Read(key_file, res));
100  BOOST_CHECK_EQUAL(res.ToString(), in_file_info.ToString());
101 
102  //Simulate transaction raw data - "t + transaction hash"
103  std::string key_transaction = "t" + m_rng.rand256().ToString();
104 
105  uint256 in_transaction = m_rng.rand256();
106  dbw.Write(key_transaction, in_transaction);
107  BOOST_CHECK(dbw.Read(key_transaction, res));
108  BOOST_CHECK_EQUAL(res.ToString(), in_transaction.ToString());
109 
110  //Simulate UTXO raw data - "c + transaction hash"
111  std::string key_utxo = "c" + m_rng.rand256().ToString();
112 
113  uint256 in_utxo = m_rng.rand256();
114  dbw.Write(key_utxo, in_utxo);
115  BOOST_CHECK(dbw.Read(key_utxo, res));
116  BOOST_CHECK_EQUAL(res.ToString(), in_utxo.ToString());
117 
118  //Simulate last block file number - "l"
119  uint8_t key_last_blockfile_number{'l'};
120  uint32_t lastblockfilenumber = m_rng.rand32();
121  dbw.Write(key_last_blockfile_number, lastblockfilenumber);
122  BOOST_CHECK(dbw.Read(key_last_blockfile_number, res_uint_32));
123  BOOST_CHECK_EQUAL(lastblockfilenumber, res_uint_32);
124 
125  //Simulate Is Reindexing - "R"
126  uint8_t key_IsReindexing{'R'};
127  bool isInReindexing = m_rng.randbool();
128  dbw.Write(key_IsReindexing, isInReindexing);
129  BOOST_CHECK(dbw.Read(key_IsReindexing, res_bool));
130  BOOST_CHECK_EQUAL(isInReindexing, res_bool);
131 
132  //Simulate last block hash up to which UXTO covers - 'B'
133  uint8_t key_lastblockhash_uxto{'B'};
134  uint256 lastblock_hash = m_rng.rand256();
135  dbw.Write(key_lastblockhash_uxto, lastblock_hash);
136  BOOST_CHECK(dbw.Read(key_lastblockhash_uxto, res));
137  BOOST_CHECK_EQUAL(lastblock_hash, res);
138 
139  //Simulate file raw data - "F + filename_number + filename"
140  std::string file_option_tag = "F";
141  uint8_t filename_length = m_rng.randbits(8);
142  std::string filename = "randomfilename";
143  std::string key_file_option = strprintf("%s%01x%s", file_option_tag, filename_length, filename);
144 
145  bool in_file_bool = m_rng.randbool();
146  dbw.Write(key_file_option, in_file_bool);
147  BOOST_CHECK(dbw.Read(key_file_option, res_bool));
148  BOOST_CHECK_EQUAL(res_bool, in_file_bool);
149  }
150 }
151 
152 // Test batch operations
153 BOOST_AUTO_TEST_CASE(dbwrapper_batch)
154 {
155  // Perform tests both obfuscated and non-obfuscated.
156  for (const bool obfuscate : {false, true}) {
157  fs::path ph = m_args.GetDataDirBase() / (obfuscate ? "dbwrapper_batch_obfuscate_true" : "dbwrapper_batch_obfuscate_false");
158  CDBWrapper dbw({.path = ph, .cache_bytes = 1 << 20, .memory_only = true, .wipe_data = false, .obfuscate = obfuscate});
159 
160  uint8_t key{'i'};
161  uint256 in = m_rng.rand256();
162  uint8_t key2{'j'};
163  uint256 in2 = m_rng.rand256();
164  uint8_t key3{'k'};
165  uint256 in3 = m_rng.rand256();
166 
167  uint256 res;
168  CDBBatch batch(dbw);
169 
170  batch.Write(key, in);
171  batch.Write(key2, in2);
172  batch.Write(key3, in3);
173 
174  // Remove key3 before it's even been written
175  batch.Erase(key3);
176 
177  dbw.WriteBatch(batch);
178 
179  BOOST_CHECK(dbw.Read(key, res));
180  BOOST_CHECK_EQUAL(res.ToString(), in.ToString());
181  BOOST_CHECK(dbw.Read(key2, res));
182  BOOST_CHECK_EQUAL(res.ToString(), in2.ToString());
183 
184  // key3 should've never been written
185  BOOST_CHECK(dbw.Read(key3, res) == false);
186  }
187 }
188 
189 BOOST_AUTO_TEST_CASE(dbwrapper_iterator)
190 {
191  // Perform tests both obfuscated and non-obfuscated.
192  for (const bool obfuscate : {false, true}) {
193  fs::path ph = m_args.GetDataDirBase() / (obfuscate ? "dbwrapper_iterator_obfuscate_true" : "dbwrapper_iterator_obfuscate_false");
194  CDBWrapper dbw({.path = ph, .cache_bytes = 1 << 20, .memory_only = true, .wipe_data = false, .obfuscate = obfuscate});
195 
196  // The two keys are intentionally chosen for ordering
197  uint8_t key{'j'};
198  uint256 in = m_rng.rand256();
199  dbw.Write(key, in);
200  uint8_t key2{'k'};
201  uint256 in2 = m_rng.rand256();
202  dbw.Write(key2, in2);
203 
204  std::unique_ptr<CDBIterator> it(const_cast<CDBWrapper&>(dbw).NewIterator());
205 
206  // Be sure to seek past the obfuscation key (if it exists)
207  it->Seek(key);
208 
209  uint8_t key_res;
210  uint256 val_res;
211 
212  BOOST_REQUIRE(it->GetKey(key_res));
213  BOOST_REQUIRE(it->GetValue(val_res));
214  BOOST_CHECK_EQUAL(key_res, key);
215  BOOST_CHECK_EQUAL(val_res.ToString(), in.ToString());
216 
217  it->Next();
218 
219  BOOST_REQUIRE(it->GetKey(key_res));
220  BOOST_REQUIRE(it->GetValue(val_res));
221  BOOST_CHECK_EQUAL(key_res, key2);
222  BOOST_CHECK_EQUAL(val_res.ToString(), in2.ToString());
223 
224  it->Next();
225  BOOST_CHECK_EQUAL(it->Valid(), false);
226  }
227 }
228 
229 // Test that we do not obfuscation if there is existing data.
230 BOOST_AUTO_TEST_CASE(existing_data_no_obfuscate)
231 {
232  // We're going to share this fs::path between two wrappers
233  fs::path ph = m_args.GetDataDirBase() / "existing_data_no_obfuscate";
234  fs::create_directories(ph);
235 
236  // Set up a non-obfuscated wrapper to write some initial data.
237  std::unique_ptr<CDBWrapper> dbw = std::make_unique<CDBWrapper>(DBParams{.path = ph, .cache_bytes = 1 << 10, .memory_only = false, .wipe_data = false, .obfuscate = false});
238  uint8_t key{'k'};
239  uint256 in = m_rng.rand256();
240  uint256 res;
241 
242  dbw->Write(key, in);
243  BOOST_CHECK(dbw->Read(key, res));
244  BOOST_CHECK_EQUAL(res.ToString(), in.ToString());
245 
246  // Call the destructor to free leveldb LOCK
247  dbw.reset();
248 
249  // Now, set up another wrapper that wants to obfuscate the same directory
250  CDBWrapper odbw({.path = ph, .cache_bytes = 1 << 10, .memory_only = false, .wipe_data = false, .obfuscate = true});
251 
252  // Check that the key/val we wrote with unobfuscated wrapper exists and
253  // is readable.
254  uint256 res2;
255  BOOST_CHECK(odbw.Read(key, res2));
256  BOOST_CHECK_EQUAL(res2.ToString(), in.ToString());
257 
258  BOOST_CHECK(!odbw.IsEmpty());
259  BOOST_CHECK(!dbwrapper_private::GetObfuscation(odbw)); // The key should be an empty string
260 
261  uint256 in2 = m_rng.rand256();
262  uint256 res3;
263 
264  // Check that we can write successfully
265  odbw.Write(key, in2);
266  BOOST_CHECK(odbw.Read(key, res3));
267  BOOST_CHECK_EQUAL(res3.ToString(), in2.ToString());
268 }
269 
270 // Ensure that we start obfuscating during a reindex.
271 BOOST_AUTO_TEST_CASE(existing_data_reindex)
272 {
273  // We're going to share this fs::path between two wrappers
274  fs::path ph = m_args.GetDataDirBase() / "existing_data_reindex";
275  fs::create_directories(ph);
276 
277  // Set up a non-obfuscated wrapper to write some initial data.
278  std::unique_ptr<CDBWrapper> dbw = std::make_unique<CDBWrapper>(DBParams{.path = ph, .cache_bytes = 1 << 10, .memory_only = false, .wipe_data = false, .obfuscate = false});
279  uint8_t key{'k'};
280  uint256 in = m_rng.rand256();
281  uint256 res;
282 
283  dbw->Write(key, in);
284  BOOST_CHECK(dbw->Read(key, res));
285  BOOST_CHECK_EQUAL(res.ToString(), in.ToString());
286 
287  // Call the destructor to free leveldb LOCK
288  dbw.reset();
289 
290  // Simulate a -reindex by wiping the existing data store
291  CDBWrapper odbw({.path = ph, .cache_bytes = 1 << 10, .memory_only = false, .wipe_data = true, .obfuscate = true});
292 
293  // Check that the key/val we wrote with unobfuscated wrapper doesn't exist
294  uint256 res2;
295  BOOST_CHECK(!odbw.Read(key, res2));
297 
298  uint256 in2 = m_rng.rand256();
299  uint256 res3;
300 
301  // Check that we can write successfully
302  odbw.Write(key, in2);
303  BOOST_CHECK(odbw.Read(key, res3));
304  BOOST_CHECK_EQUAL(res3.ToString(), in2.ToString());
305 }
306 
307 BOOST_AUTO_TEST_CASE(iterator_ordering)
308 {
309  fs::path ph = m_args.GetDataDirBase() / "iterator_ordering";
310  CDBWrapper dbw({.path = ph, .cache_bytes = 1 << 20, .memory_only = true, .wipe_data = false, .obfuscate = false});
311  for (int x=0x00; x<256; ++x) {
312  uint8_t key = x;
313  uint32_t value = x*x;
314  if (!(x & 1)) dbw.Write(key, value);
315  }
316 
317  // Check that creating an iterator creates a snapshot
318  std::unique_ptr<CDBIterator> it(const_cast<CDBWrapper&>(dbw).NewIterator());
319 
320  for (unsigned int x=0x00; x<256; ++x) {
321  uint8_t key = x;
322  uint32_t value = x*x;
323  if (x & 1) dbw.Write(key, value);
324  }
325 
326  for (const int seek_start : {0x00, 0x80}) {
327  it->Seek((uint8_t)seek_start);
328  for (unsigned int x=seek_start; x<255; ++x) {
329  uint8_t key;
330  uint32_t value;
331  BOOST_CHECK(it->Valid());
332  if (!it->Valid()) // Avoid spurious errors about invalid iterator's key and value in case of failure
333  break;
334  BOOST_CHECK(it->GetKey(key));
335  if (x & 1) {
336  BOOST_CHECK_EQUAL(key, x + 1);
337  continue;
338  }
339  BOOST_CHECK(it->GetValue(value));
340  BOOST_CHECK_EQUAL(key, x);
341  BOOST_CHECK_EQUAL(value, x*x);
342  it->Next();
343  }
344  BOOST_CHECK(!it->Valid());
345  }
346 }
347 
349  // Used to make two serialized objects the same while letting them have different lengths
350  // This is a terrible idea
351  std::string str;
352  StringContentsSerializer() = default;
353  explicit StringContentsSerializer(const std::string& inp) : str(inp) {}
354 
355  template<typename Stream>
356  void Serialize(Stream& s) const
357  {
358  for (size_t i = 0; i < str.size(); i++) {
359  s << uint8_t(str[i]);
360  }
361  }
362 
363  template<typename Stream>
364  void Unserialize(Stream& s)
365  {
366  str.clear();
367  uint8_t c{0};
368  while (!s.empty()) {
369  s >> c;
370  str.push_back(c);
371  }
372  }
373 };
374 
375 BOOST_AUTO_TEST_CASE(iterator_string_ordering)
376 {
377  fs::path ph = m_args.GetDataDirBase() / "iterator_string_ordering";
378  CDBWrapper dbw({.path = ph, .cache_bytes = 1 << 20, .memory_only = true, .wipe_data = false, .obfuscate = false});
379  for (int x = 0; x < 10; ++x) {
380  for (int y = 0; y < 10; ++y) {
381  std::string key{ToString(x)};
382  for (int z = 0; z < y; ++z)
383  key += key;
384  uint32_t value = x*x;
385  dbw.Write(StringContentsSerializer{key}, value);
386  }
387  }
388 
389  std::unique_ptr<CDBIterator> it(const_cast<CDBWrapper&>(dbw).NewIterator());
390  for (const int seek_start : {0, 5}) {
391  it->Seek(StringContentsSerializer{ToString(seek_start)});
392  for (unsigned int x = seek_start; x < 10; ++x) {
393  for (int y = 0; y < 10; ++y) {
394  std::string exp_key{ToString(x)};
395  for (int z = 0; z < y; ++z)
396  exp_key += exp_key;
398  uint32_t value;
399  BOOST_CHECK(it->Valid());
400  if (!it->Valid()) // Avoid spurious errors about invalid iterator's key and value in case of failure
401  break;
402  BOOST_CHECK(it->GetKey(key));
403  BOOST_CHECK(it->GetValue(value));
404  BOOST_CHECK_EQUAL(key.str, exp_key);
405  BOOST_CHECK_EQUAL(value, x*x);
406  it->Next();
407  }
408  }
409  BOOST_CHECK(!it->Valid());
410  }
411 }
412 
414 {
415  // Attempt to create a database with a UTF8 character in the path.
416  // On Windows this test will fail if the directory is created using
417  // the ANSI CreateDirectoryA call and the code page isn't UTF8.
418  // It will succeed if created with CreateDirectoryW.
419  fs::path ph = m_args.GetDataDirBase() / "test_runner_₿_🏃_20191128_104644";
420  CDBWrapper dbw({.path = ph, .cache_bytes = 1 << 20});
421 
422  fs::path lockPath = ph / "LOCK";
423  BOOST_CHECK(fs::exists(lockPath));
424 }
425 
426 
fs::path path
Location in the filesystem where leveldb data will be stored.
Definition: dbwrapper.h:35
Batch of changes queued to be written to a CDBWrapper.
Definition: dbwrapper.h:71
#define strprintf
Format arguments and return the string or write to given std::ostream (see tinyformat::format doc for...
Definition: tinyformat.h:1172
const Obfuscation & GetObfuscation(const CDBWrapper &w)
Work around circular dependency, as well as for testing in dbwrapper_tests.
Definition: dbwrapper.cpp:388
void Serialize(Stream &s) const
void Erase(const K &key)
Definition: dbwrapper.h:108
Basic testing setup.
Definition: setup_common.h:64
StringContentsSerializer()=default
static bool exists(const path &p)
Definition: fs.h:95
void Write(const K &key, const V &value, bool fSync=false)
Definition: dbwrapper.h:227
BOOST_FIXTURE_TEST_SUITE(cuckoocache_tests, BasicTestingSetup)
Test Suite for CuckooCache.
BOOST_AUTO_TEST_CASE(dbwrapper)
BOOST_AUTO_TEST_SUITE_END()
void Write(const K &key, const V &value)
Definition: dbwrapper.h:96
std::string ToString() const
Definition: uint256.cpp:21
256-bit opaque blob.
Definition: uint256.h:195
#define BOOST_CHECK_EQUAL(v1, v2)
Definition: object.cpp:17
Application-specific storage settings.
Definition: dbwrapper.h:33
Path class wrapper to block calls to the fs::path(std::string) implicit constructor and the fs::path:...
Definition: fs.h:33
StringContentsSerializer(const std::string &inp)
std::string ToString(const T &t)
Locale-independent version of std::to_string.
Definition: string.h:246
#define BOOST_CHECK(expr)
Definition: object.cpp:16