2016-12-31 15:01:21 -03:00
|
|
|
// Copyright (c) 2012-2016 The Bitcoin Core developers
|
2015-09-07 19:22:23 -03:00
|
|
|
// Distributed under the MIT software license, see the accompanying
|
|
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
|
2015-10-22 22:33:06 -03:00
|
|
|
#include "dbwrapper.h"
|
2015-09-07 19:22:23 -03:00
|
|
|
#include "uint256.h"
|
|
|
|
#include "random.h"
|
|
|
|
#include "test/test_bitcoin.h"
|
|
|
|
|
|
|
|
#include <boost/test/unit_test.hpp>
|
2016-12-05 04:03:53 -03:00
|
|
|
|
2015-09-07 19:22:23 -03:00
|
|
|
// Test if a string consists entirely of null characters
|
2016-12-05 04:03:53 -03:00
|
|
|
bool is_null_key(const std::vector<unsigned char>& key) {
|
2015-09-07 19:22:23 -03:00
|
|
|
bool isnull = true;
|
|
|
|
|
|
|
|
for (unsigned int i = 0; i < key.size(); i++)
|
|
|
|
isnull &= (key[i] == '\x00');
|
|
|
|
|
|
|
|
return isnull;
|
|
|
|
}
|
|
|
|
|
2015-10-22 22:02:20 -03:00
|
|
|
BOOST_FIXTURE_TEST_SUITE(dbwrapper_tests, BasicTestingSetup)
|
2015-09-07 19:22:23 -03:00
|
|
|
|
2015-10-22 22:02:20 -03:00
|
|
|
BOOST_AUTO_TEST_CASE(dbwrapper)
|
2015-09-07 19:22:23 -03:00
|
|
|
{
|
|
|
|
// Perform tests both obfuscated and non-obfuscated.
|
|
|
|
for (int i = 0; i < 2; i++) {
|
|
|
|
bool obfuscate = (bool)i;
|
2017-03-01 13:05:50 -03:00
|
|
|
fs::path ph = fs::temp_directory_path() / fs::unique_path();
|
2015-10-22 22:02:20 -03:00
|
|
|
CDBWrapper dbw(ph, (1 << 20), true, false, obfuscate);
|
2015-09-07 19:22:23 -03:00
|
|
|
char key = 'k';
|
2017-06-07 15:03:17 -04:00
|
|
|
uint256 in = InsecureRand256();
|
2015-09-07 19:22:23 -03:00
|
|
|
uint256 res;
|
|
|
|
|
|
|
|
// Ensure that we're doing real obfuscation when obfuscate=true
|
2016-04-20 06:46:01 -03:00
|
|
|
BOOST_CHECK(obfuscate != is_null_key(dbwrapper_private::GetObfuscateKey(dbw)));
|
2015-09-07 19:22:23 -03:00
|
|
|
|
|
|
|
BOOST_CHECK(dbw.Write(key, in));
|
|
|
|
BOOST_CHECK(dbw.Read(key, res));
|
|
|
|
BOOST_CHECK_EQUAL(res.ToString(), in.ToString());
|
|
|
|
}
|
|
|
|
}
|
2015-10-08 05:22:50 -03:00
|
|
|
|
|
|
|
// Test batch operations
|
2015-10-22 22:02:20 -03:00
|
|
|
BOOST_AUTO_TEST_CASE(dbwrapper_batch)
|
2015-10-08 05:22:50 -03:00
|
|
|
{
|
|
|
|
// Perform tests both obfuscated and non-obfuscated.
|
|
|
|
for (int i = 0; i < 2; i++) {
|
|
|
|
bool obfuscate = (bool)i;
|
2017-03-01 13:05:50 -03:00
|
|
|
fs::path ph = fs::temp_directory_path() / fs::unique_path();
|
2015-10-22 22:02:20 -03:00
|
|
|
CDBWrapper dbw(ph, (1 << 20), true, false, obfuscate);
|
2015-10-08 05:22:50 -03:00
|
|
|
|
|
|
|
char key = 'i';
|
2017-06-07 15:03:17 -04:00
|
|
|
uint256 in = InsecureRand256();
|
2015-10-08 05:22:50 -03:00
|
|
|
char key2 = 'j';
|
2017-06-07 15:03:17 -04:00
|
|
|
uint256 in2 = InsecureRand256();
|
2015-10-08 05:22:50 -03:00
|
|
|
char key3 = 'k';
|
2017-06-07 15:03:17 -04:00
|
|
|
uint256 in3 = InsecureRand256();
|
2015-10-08 05:22:50 -03:00
|
|
|
|
|
|
|
uint256 res;
|
2016-04-20 06:46:01 -03:00
|
|
|
CDBBatch batch(dbw);
|
2015-10-08 05:22:50 -03:00
|
|
|
|
|
|
|
batch.Write(key, in);
|
|
|
|
batch.Write(key2, in2);
|
|
|
|
batch.Write(key3, in3);
|
|
|
|
|
|
|
|
// Remove key3 before it's even been written
|
|
|
|
batch.Erase(key3);
|
|
|
|
|
|
|
|
dbw.WriteBatch(batch);
|
|
|
|
|
|
|
|
BOOST_CHECK(dbw.Read(key, res));
|
|
|
|
BOOST_CHECK_EQUAL(res.ToString(), in.ToString());
|
|
|
|
BOOST_CHECK(dbw.Read(key2, res));
|
|
|
|
BOOST_CHECK_EQUAL(res.ToString(), in2.ToString());
|
|
|
|
|
2017-01-13 12:05:16 -03:00
|
|
|
// key3 should've never been written
|
2015-10-08 05:22:50 -03:00
|
|
|
BOOST_CHECK(dbw.Read(key3, res) == false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-10-22 22:02:20 -03:00
|
|
|
BOOST_AUTO_TEST_CASE(dbwrapper_iterator)
|
2015-10-08 05:22:50 -03:00
|
|
|
{
|
|
|
|
// Perform tests both obfuscated and non-obfuscated.
|
|
|
|
for (int i = 0; i < 2; i++) {
|
|
|
|
bool obfuscate = (bool)i;
|
2017-03-01 13:05:50 -03:00
|
|
|
fs::path ph = fs::temp_directory_path() / fs::unique_path();
|
2015-10-22 22:02:20 -03:00
|
|
|
CDBWrapper dbw(ph, (1 << 20), true, false, obfuscate);
|
2015-10-08 05:22:50 -03:00
|
|
|
|
|
|
|
// The two keys are intentionally chosen for ordering
|
|
|
|
char key = 'j';
|
2017-06-07 15:03:17 -04:00
|
|
|
uint256 in = InsecureRand256();
|
2015-10-08 05:22:50 -03:00
|
|
|
BOOST_CHECK(dbw.Write(key, in));
|
|
|
|
char key2 = 'k';
|
2017-06-07 15:03:17 -04:00
|
|
|
uint256 in2 = InsecureRand256();
|
2015-10-08 05:22:50 -03:00
|
|
|
BOOST_CHECK(dbw.Write(key2, in2));
|
|
|
|
|
2017-07-07 03:54:42 -04:00
|
|
|
std::unique_ptr<CDBIterator> it(const_cast<CDBWrapper&>(dbw).NewIterator());
|
2015-10-08 05:22:50 -03:00
|
|
|
|
|
|
|
// Be sure to seek past the obfuscation key (if it exists)
|
|
|
|
it->Seek(key);
|
|
|
|
|
|
|
|
char key_res;
|
|
|
|
uint256 val_res;
|
|
|
|
|
|
|
|
it->GetKey(key_res);
|
|
|
|
it->GetValue(val_res);
|
|
|
|
BOOST_CHECK_EQUAL(key_res, key);
|
|
|
|
BOOST_CHECK_EQUAL(val_res.ToString(), in.ToString());
|
|
|
|
|
|
|
|
it->Next();
|
|
|
|
|
|
|
|
it->GetKey(key_res);
|
|
|
|
it->GetValue(val_res);
|
|
|
|
BOOST_CHECK_EQUAL(key_res, key2);
|
|
|
|
BOOST_CHECK_EQUAL(val_res.ToString(), in2.ToString());
|
|
|
|
|
|
|
|
it->Next();
|
|
|
|
BOOST_CHECK_EQUAL(it->Valid(), false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-09-07 19:22:23 -03:00
|
|
|
// Test that we do not obfuscation if there is existing data.
|
|
|
|
BOOST_AUTO_TEST_CASE(existing_data_no_obfuscate)
|
|
|
|
{
|
2017-03-01 13:05:50 -03:00
|
|
|
// We're going to share this fs::path between two wrappers
|
|
|
|
fs::path ph = fs::temp_directory_path() / fs::unique_path();
|
2015-09-07 19:22:23 -03:00
|
|
|
create_directories(ph);
|
|
|
|
|
|
|
|
// Set up a non-obfuscated wrapper to write some initial data.
|
2015-10-22 22:02:20 -03:00
|
|
|
CDBWrapper* dbw = new CDBWrapper(ph, (1 << 10), false, false, false);
|
2015-09-07 19:22:23 -03:00
|
|
|
char key = 'k';
|
2017-06-07 15:03:17 -04:00
|
|
|
uint256 in = InsecureRand256();
|
2015-09-07 19:22:23 -03:00
|
|
|
uint256 res;
|
|
|
|
|
|
|
|
BOOST_CHECK(dbw->Write(key, in));
|
|
|
|
BOOST_CHECK(dbw->Read(key, res));
|
|
|
|
BOOST_CHECK_EQUAL(res.ToString(), in.ToString());
|
|
|
|
|
|
|
|
// Call the destructor to free leveldb LOCK
|
|
|
|
delete dbw;
|
2017-03-18 08:40:58 -03:00
|
|
|
dbw = nullptr;
|
2015-09-07 19:22:23 -03:00
|
|
|
|
|
|
|
// Now, set up another wrapper that wants to obfuscate the same directory
|
2015-10-22 22:02:20 -03:00
|
|
|
CDBWrapper odbw(ph, (1 << 10), false, false, true);
|
2015-09-07 19:22:23 -03:00
|
|
|
|
|
|
|
// Check that the key/val we wrote with unobfuscated wrapper exists and
|
|
|
|
// is readable.
|
|
|
|
uint256 res2;
|
|
|
|
BOOST_CHECK(odbw.Read(key, res2));
|
|
|
|
BOOST_CHECK_EQUAL(res2.ToString(), in.ToString());
|
|
|
|
|
|
|
|
BOOST_CHECK(!odbw.IsEmpty()); // There should be existing data
|
2016-04-20 06:46:01 -03:00
|
|
|
BOOST_CHECK(is_null_key(dbwrapper_private::GetObfuscateKey(odbw))); // The key should be an empty string
|
2015-09-07 19:22:23 -03:00
|
|
|
|
2017-06-07 15:03:17 -04:00
|
|
|
uint256 in2 = InsecureRand256();
|
2015-09-07 19:22:23 -03:00
|
|
|
uint256 res3;
|
|
|
|
|
|
|
|
// Check that we can write successfully
|
|
|
|
BOOST_CHECK(odbw.Write(key, in2));
|
|
|
|
BOOST_CHECK(odbw.Read(key, res3));
|
|
|
|
BOOST_CHECK_EQUAL(res3.ToString(), in2.ToString());
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure that we start obfuscating during a reindex.
|
|
|
|
BOOST_AUTO_TEST_CASE(existing_data_reindex)
|
|
|
|
{
|
2017-03-01 13:05:50 -03:00
|
|
|
// We're going to share this fs::path between two wrappers
|
|
|
|
fs::path ph = fs::temp_directory_path() / fs::unique_path();
|
2015-09-07 19:22:23 -03:00
|
|
|
create_directories(ph);
|
|
|
|
|
|
|
|
// Set up a non-obfuscated wrapper to write some initial data.
|
2015-10-22 22:02:20 -03:00
|
|
|
CDBWrapper* dbw = new CDBWrapper(ph, (1 << 10), false, false, false);
|
2015-09-07 19:22:23 -03:00
|
|
|
char key = 'k';
|
2017-06-07 15:03:17 -04:00
|
|
|
uint256 in = InsecureRand256();
|
2015-09-07 19:22:23 -03:00
|
|
|
uint256 res;
|
|
|
|
|
|
|
|
BOOST_CHECK(dbw->Write(key, in));
|
|
|
|
BOOST_CHECK(dbw->Read(key, res));
|
|
|
|
BOOST_CHECK_EQUAL(res.ToString(), in.ToString());
|
|
|
|
|
|
|
|
// Call the destructor to free leveldb LOCK
|
|
|
|
delete dbw;
|
2017-03-18 08:40:58 -03:00
|
|
|
dbw = nullptr;
|
2015-09-07 19:22:23 -03:00
|
|
|
|
|
|
|
// Simulate a -reindex by wiping the existing data store
|
2015-10-22 22:02:20 -03:00
|
|
|
CDBWrapper odbw(ph, (1 << 10), false, true, true);
|
2015-09-07 19:22:23 -03:00
|
|
|
|
|
|
|
// Check that the key/val we wrote with unobfuscated wrapper doesn't exist
|
|
|
|
uint256 res2;
|
|
|
|
BOOST_CHECK(!odbw.Read(key, res2));
|
2016-04-20 06:46:01 -03:00
|
|
|
BOOST_CHECK(!is_null_key(dbwrapper_private::GetObfuscateKey(odbw)));
|
2015-09-07 19:22:23 -03:00
|
|
|
|
2017-06-07 15:03:17 -04:00
|
|
|
uint256 in2 = InsecureRand256();
|
2015-09-07 19:22:23 -03:00
|
|
|
uint256 res3;
|
|
|
|
|
|
|
|
// Check that we can write successfully
|
|
|
|
BOOST_CHECK(odbw.Write(key, in2));
|
|
|
|
BOOST_CHECK(odbw.Read(key, res3));
|
|
|
|
BOOST_CHECK_EQUAL(res3.ToString(), in2.ToString());
|
|
|
|
}
|
2016-04-27 06:07:43 -03:00
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(iterator_ordering)
|
|
|
|
{
|
2017-03-01 13:05:50 -03:00
|
|
|
fs::path ph = fs::temp_directory_path() / fs::unique_path();
|
2016-04-27 06:07:43 -03:00
|
|
|
CDBWrapper dbw(ph, (1 << 20), true, false, false);
|
|
|
|
for (int x=0x00; x<256; ++x) {
|
|
|
|
uint8_t key = x;
|
|
|
|
uint32_t value = x*x;
|
|
|
|
BOOST_CHECK(dbw.Write(key, value));
|
|
|
|
}
|
|
|
|
|
2017-07-07 03:54:42 -04:00
|
|
|
std::unique_ptr<CDBIterator> it(const_cast<CDBWrapper&>(dbw).NewIterator());
|
2016-04-27 06:07:43 -03:00
|
|
|
for (int c=0; c<2; ++c) {
|
|
|
|
int seek_start;
|
|
|
|
if (c == 0)
|
|
|
|
seek_start = 0x00;
|
|
|
|
else
|
|
|
|
seek_start = 0x80;
|
|
|
|
it->Seek((uint8_t)seek_start);
|
|
|
|
for (int x=seek_start; x<256; ++x) {
|
|
|
|
uint8_t key;
|
|
|
|
uint32_t value;
|
|
|
|
BOOST_CHECK(it->Valid());
|
|
|
|
if (!it->Valid()) // Avoid spurious errors about invalid iterator's key and value in case of failure
|
|
|
|
break;
|
|
|
|
BOOST_CHECK(it->GetKey(key));
|
|
|
|
BOOST_CHECK(it->GetValue(value));
|
|
|
|
BOOST_CHECK_EQUAL(key, x);
|
|
|
|
BOOST_CHECK_EQUAL(value, x*x);
|
|
|
|
it->Next();
|
|
|
|
}
|
|
|
|
BOOST_CHECK(!it->Valid());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-02 21:52:31 -03:00
|
|
|
struct StringContentsSerializer {
|
|
|
|
// Used to make two serialized objects the same while letting them have a different lengths
|
|
|
|
// This is a terrible idea
|
2016-12-05 04:03:53 -03:00
|
|
|
std::string str;
|
2016-05-02 21:52:31 -03:00
|
|
|
StringContentsSerializer() {}
|
2016-12-05 04:03:53 -03:00
|
|
|
StringContentsSerializer(const std::string& inp) : str(inp) {}
|
2016-05-02 21:52:31 -03:00
|
|
|
|
2016-12-05 04:03:53 -03:00
|
|
|
StringContentsSerializer& operator+=(const std::string& s) {
|
2016-05-02 21:52:31 -03:00
|
|
|
str += s;
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
StringContentsSerializer& operator+=(const StringContentsSerializer& s) { return *this += s.str; }
|
|
|
|
|
|
|
|
ADD_SERIALIZE_METHODS;
|
|
|
|
|
|
|
|
template <typename Stream, typename Operation>
|
2016-10-28 20:29:17 -03:00
|
|
|
inline void SerializationOp(Stream& s, Operation ser_action) {
|
2016-05-02 21:52:31 -03:00
|
|
|
if (ser_action.ForRead()) {
|
|
|
|
str.clear();
|
|
|
|
char c = 0;
|
|
|
|
while (true) {
|
|
|
|
try {
|
|
|
|
READWRITE(c);
|
|
|
|
str.push_back(c);
|
|
|
|
} catch (const std::ios_base::failure& e) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
for (size_t i = 0; i < str.size(); i++)
|
|
|
|
READWRITE(str[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(iterator_string_ordering)
|
|
|
|
{
|
|
|
|
char buf[10];
|
|
|
|
|
2017-03-01 13:05:50 -03:00
|
|
|
fs::path ph = fs::temp_directory_path() / fs::unique_path();
|
2016-05-02 21:52:31 -03:00
|
|
|
CDBWrapper dbw(ph, (1 << 20), true, false, false);
|
|
|
|
for (int x=0x00; x<10; ++x) {
|
|
|
|
for (int y = 0; y < 10; y++) {
|
2017-02-26 17:08:26 -03:00
|
|
|
snprintf(buf, sizeof(buf), "%d", x);
|
2016-05-02 21:52:31 -03:00
|
|
|
StringContentsSerializer key(buf);
|
|
|
|
for (int z = 0; z < y; z++)
|
|
|
|
key += key;
|
|
|
|
uint32_t value = x*x;
|
|
|
|
BOOST_CHECK(dbw.Write(key, value));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-07 03:54:42 -04:00
|
|
|
std::unique_ptr<CDBIterator> it(const_cast<CDBWrapper&>(dbw).NewIterator());
|
2016-05-02 21:52:31 -03:00
|
|
|
for (int c=0; c<2; ++c) {
|
|
|
|
int seek_start;
|
|
|
|
if (c == 0)
|
|
|
|
seek_start = 0;
|
|
|
|
else
|
|
|
|
seek_start = 5;
|
2017-02-26 17:08:26 -03:00
|
|
|
snprintf(buf, sizeof(buf), "%d", seek_start);
|
2016-05-02 21:52:31 -03:00
|
|
|
StringContentsSerializer seek_key(buf);
|
|
|
|
it->Seek(seek_key);
|
|
|
|
for (int x=seek_start; x<10; ++x) {
|
|
|
|
for (int y = 0; y < 10; y++) {
|
2017-02-26 17:08:26 -03:00
|
|
|
snprintf(buf, sizeof(buf), "%d", x);
|
2016-12-05 04:03:53 -03:00
|
|
|
std::string exp_key(buf);
|
2016-05-02 21:52:31 -03:00
|
|
|
for (int z = 0; z < y; z++)
|
|
|
|
exp_key += exp_key;
|
|
|
|
StringContentsSerializer key;
|
|
|
|
uint32_t value;
|
|
|
|
BOOST_CHECK(it->Valid());
|
|
|
|
if (!it->Valid()) // Avoid spurious errors about invalid iterator's key and value in case of failure
|
|
|
|
break;
|
|
|
|
BOOST_CHECK(it->GetKey(key));
|
|
|
|
BOOST_CHECK(it->GetValue(value));
|
|
|
|
BOOST_CHECK_EQUAL(key.str, exp_key);
|
|
|
|
BOOST_CHECK_EQUAL(value, x*x);
|
|
|
|
it->Next();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
BOOST_CHECK(!it->Valid());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2015-09-07 19:22:23 -03:00
|
|
|
BOOST_AUTO_TEST_SUITE_END()
|