libcryfs/test/cpp-utils/data/DataTest.cpp

224 lines
6.3 KiB
C++
Raw Normal View History

2016-02-11 12:53:42 +01:00
#include "cpp-utils/data/DataFixture.h"
#include "cpp-utils/data/Data.h"
#include <gtest/gtest.h>
2015-04-25 02:21:39 +02:00
2016-02-11 12:53:42 +01:00
#include "cpp-utils/tempfile/TempFile.h"
2015-04-25 02:21:39 +02:00
#include <fstream>
using ::testing::Test;
using ::testing::WithParamInterface;
using ::testing::Values;
2015-04-25 02:40:02 +02:00
using cpputils::TempFile;
2015-04-25 02:21:39 +02:00
using std::ifstream;
using std::ofstream;
using std::string;
2015-04-25 02:21:39 +02:00
namespace bf = boost::filesystem;
using namespace cpputils;
class DataTest: public Test {
public:
bool DataIsZeroes(const Data &data) {
for (size_t i = 0; i != data.size(); ++ i) {
if (((char*)data.data())[i] != 0) {
return false;
}
}
return true;
}
};
class DataTestWithSizeParam: public DataTest, public WithParamInterface<size_t> {
public:
2015-04-25 17:45:18 +02:00
Data randomData;
2015-04-25 02:21:39 +02:00
2015-04-25 17:45:18 +02:00
DataTestWithSizeParam(): randomData(DataFixture::generate(GetParam())) {}
2015-04-25 02:21:39 +02:00
2015-04-25 17:45:18 +02:00
static void StoreData(const Data &data, const bf::path &filepath) {
2015-04-25 02:21:39 +02:00
ofstream file(filepath.c_str(), std::ios::binary | std::ios::trunc);
2015-04-25 17:45:18 +02:00
file.write((char*)data.data(), data.size());
2015-04-25 02:21:39 +02:00
}
2015-04-25 17:45:18 +02:00
static void EXPECT_STORED_FILE_DATA_CORRECT(const Data &data, const bf::path &filepath) {
EXPECT_EQ(data.size(), bf::file_size(filepath));
2015-04-25 02:21:39 +02:00
ifstream file(filepath.c_str(), std::ios::binary);
2015-04-25 17:45:18 +02:00
char *read_data = new char[data.size()];
file.read(read_data, data.size());
2015-04-25 02:21:39 +02:00
2015-04-25 17:45:18 +02:00
EXPECT_EQ(0, std::memcmp(data.data(), read_data, data.size()));
2015-04-25 02:21:39 +02:00
delete[] read_data;
}
};
INSTANTIATE_TEST_CASE_P(DataTestWithSizeParam, DataTestWithSizeParam, Values(0, 1, 2, 1024, 4096, 10*1024*1024));
2015-04-25 17:45:18 +02:00
TEST_P(DataTestWithSizeParam, ZeroInitializedDataIsDifferentToRandomData) {
if (GetParam() != 0) {
Data data(GetParam());
data.FillWithZeroes();
EXPECT_NE(randomData, data);
}
}
2015-04-25 02:21:39 +02:00
// Working on a large data area without a crash is a good indicator that we
// are actually working on memory that was validly allocated for us.
TEST_P(DataTestWithSizeParam, WriteAndCheck) {
2015-04-25 17:45:18 +02:00
Data data = randomData.copy();
EXPECT_EQ(randomData, data);
2015-04-25 02:21:39 +02:00
}
TEST_P(DataTestWithSizeParam, Size) {
Data data(GetParam());
EXPECT_EQ(GetParam(), data.size());
}
TEST_P(DataTestWithSizeParam, CheckStoredFile) {
TempFile file;
2015-04-25 17:45:18 +02:00
randomData.StoreToFile(file.path());
2015-04-25 02:21:39 +02:00
2015-04-25 17:45:18 +02:00
EXPECT_STORED_FILE_DATA_CORRECT(randomData, file.path());
2015-04-25 02:21:39 +02:00
}
TEST_P(DataTestWithSizeParam, CheckLoadedData) {
TempFile file;
2015-04-25 17:45:18 +02:00
StoreData(randomData, file.path());
2015-04-25 02:21:39 +02:00
Data data = Data::LoadFromFile(file.path()).value();
2015-04-25 17:45:18 +02:00
EXPECT_EQ(randomData, data);
2015-04-25 02:21:39 +02:00
}
TEST_P(DataTestWithSizeParam, StoreDoesntChangeData) {
2015-04-25 17:45:18 +02:00
Data data = randomData.copy();
2015-04-25 02:21:39 +02:00
TempFile file;
data.StoreToFile(file.path());
2015-04-25 17:45:18 +02:00
EXPECT_EQ(randomData, data);
2015-04-25 02:21:39 +02:00
}
TEST_P(DataTestWithSizeParam, StoreAndLoad) {
TempFile file;
2015-04-25 17:45:18 +02:00
randomData.StoreToFile(file.path());
2015-04-25 02:21:39 +02:00
Data loaded_data = Data::LoadFromFile(file.path()).value();
2015-04-25 17:45:18 +02:00
EXPECT_EQ(randomData, loaded_data);
2015-04-25 02:21:39 +02:00
}
TEST_P(DataTestWithSizeParam, Copy) {
2015-04-25 17:45:18 +02:00
Data copy = randomData.copy();
EXPECT_EQ(randomData, copy);
2015-04-25 02:21:39 +02:00
}
2015-04-26 00:41:29 +02:00
TEST_F(DataTest, ChangingCopyDoesntChangeOriginal) {
Data original = DataFixture::generate(1024);
Data copy = original.copy();
((uint8_t*)copy.data())[0] = ((uint8_t*)copy.data())[0] + 1;
EXPECT_EQ(DataFixture::generate(1024), original);
EXPECT_NE(copy, original);
}
2015-04-25 02:21:39 +02:00
TEST_F(DataTest, InitializeWithZeroes) {
Data data(10*1024);
data.FillWithZeroes();
EXPECT_TRUE(DataIsZeroes(data));
}
TEST_F(DataTest, FillModifiedDataWithZeroes) {
2015-04-25 17:45:18 +02:00
Data data = DataFixture::generate(10*1024);
2015-04-25 02:21:39 +02:00
EXPECT_FALSE(DataIsZeroes(data));
data.FillWithZeroes();
EXPECT_TRUE(DataIsZeroes(data));
}
2015-04-26 00:41:29 +02:00
TEST_F(DataTest, MoveConstructor) {
Data original = DataFixture::generate(1024);
Data copy(std::move(original));
EXPECT_EQ(DataFixture::generate(1024), copy);
EXPECT_EQ(nullptr, original.data()); // NOLINT (intentional use-after-move)
EXPECT_EQ(0u, original.size()); // NOLINT (intentional use-after-move)
2015-04-26 00:41:29 +02:00
}
TEST_F(DataTest, MoveAssignment) {
Data original = DataFixture::generate(1024);
Data copy(0);
copy = std::move(original);
EXPECT_EQ(DataFixture::generate(1024), copy);
EXPECT_EQ(nullptr, original.data()); // NOLINT (intentional use-after-move)
EXPECT_EQ(0u, original.size()); // NOLINT (intentional use-after-move)
2015-04-26 00:41:29 +02:00
}
TEST_F(DataTest, Equality) {
Data data1 = DataFixture::generate(1024);
Data data2 = DataFixture::generate(1024);
EXPECT_TRUE(data1 == data2);
EXPECT_FALSE(data1 != data2);
}
TEST_F(DataTest, Inequality_DifferentSize) {
Data data1 = DataFixture::generate(1024);
Data data2 = DataFixture::generate(1023);
EXPECT_FALSE(data1 == data2);
EXPECT_TRUE(data1 != data2);
}
TEST_F(DataTest, Inequality_DifferentFirstByte) {
Data data1 = DataFixture::generate(1024);
Data data2 = DataFixture::generate(1024);
((uint8_t*)data2.data())[0] = ((uint8_t*)data2.data())[0] + 1;
EXPECT_FALSE(data1 == data2);
EXPECT_TRUE(data1 != data2);
}
TEST_F(DataTest, Inequality_DifferentMiddleByte) {
Data data1 = DataFixture::generate(1024);
Data data2 = DataFixture::generate(1024);
((uint8_t*)data2.data())[500] = ((uint8_t*)data2.data())[500] + 1;
EXPECT_FALSE(data1 == data2);
EXPECT_TRUE(data1 != data2);
}
TEST_F(DataTest, Inequality_DifferentLastByte) {
Data data1 = DataFixture::generate(1024);
Data data2 = DataFixture::generate(1024);
((uint8_t*)data2.data())[1023] = ((uint8_t*)data2.data())[1023] + 1;
EXPECT_FALSE(data1 == data2);
EXPECT_TRUE(data1 != data2);
}
2015-11-25 15:39:52 +01:00
#ifdef __x86_64__
2015-04-25 02:21:39 +02:00
TEST_F(DataTest, LargesizeSize) {
2015-11-29 21:32:12 +01:00
//Needs 64bit for representation. This value isn't in the size param list, because the list is also used for read/write checks.
uint64_t size = 4.5L*1024*1024*1024;
2015-04-25 02:21:39 +02:00
Data data(size);
EXPECT_EQ(size, data.size());
}
2015-11-25 15:39:52 +01:00
#else
#warning This is not a 64bit architecture. Large size data tests are disabled.
#endif
2015-04-25 02:21:39 +02:00
TEST_F(DataTest, LoadingNonexistingFile) {
TempFile file(false); // Pass false to constructor, so the tempfile is not created
EXPECT_FALSE(Data::LoadFromFile(file.path()));
}
class DataTestWithStringParam: public DataTest, public WithParamInterface<string> {};
INSTANTIATE_TEST_CASE_P(DataTestWithStringParam, DataTestWithStringParam, Values("", "2898B4B8A13C0F0278CCE465DB", "6FFEBAD90C0DAA2B79628F0627CE9841"));
TEST_P(DataTestWithStringParam, FromAndToString) {
Data data = Data::FromString(GetParam());
EXPECT_EQ(GetParam(), data.ToString());
}
TEST_P(DataTestWithStringParam, ToAndFromString) {
Data data = Data::FromString(GetParam());
Data data2 = Data::FromString(data.ToString());
EXPECT_EQ(data, data2);
}